Files
Autoparts-DB/dashboard/server.py
consultoria-as 744df6b3b8 feat: add SaaS endpoints — auth, inventory, availability, admin users
New endpoints:
- Auth: register, login, refresh, me
- Admin: list users, activate/deactivate
- Inventory: mapping CRUD, file upload (CSV/Excel), history, items list
- Parts: availability across warehouses, aftermarket alternatives
- Routes: login.html, bodega pages
- Fix: admin stats use pg_class estimates for large tables

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-18 22:24:45 +00:00

3865 lines
168 KiB
Python

from flask import Flask, jsonify, request, send_from_directory, redirect, g
from sqlalchemy import create_engine, text
from sqlalchemy.orm import sessionmaker
from sqlalchemy.exc import IntegrityError
import os
import sys
import json as json_module
import re
import base64
import uuid
import urllib.request
from datetime import datetime, timedelta
sys.path.insert(0, os.path.join(os.path.dirname(os.path.abspath(__file__)), '..'))
from config import DB_URL
from auth import hash_password, check_password, create_access_token, create_refresh_token, decode_token, require_auth
app = Flask(__name__, static_folder='.')
engine = create_engine(DB_URL, pool_pre_ping=True, pool_size=5, max_overflow=10)
Session = sessionmaker(bind=engine)
# ============================================================================
# Helper Functions
# ============================================================================
def get_all_brands(detailed=False, region_mask=None):
session = Session()
try:
region_filter = ""
params = {}
if region_mask is not None:
region_filter = " AND (b.region & :rmask) > 0"
params['rmask'] = region_mask
if detailed:
sql = text("""
SELECT b.name_brand AS name,
COUNT(DISTINCT m.name_model) AS model_count,
COUNT(DISTINCT mye.id_mye) AS vehicle_count
FROM brands b
JOIN models m ON m.brand_id = b.id_brand
JOIN model_year_engine mye ON mye.model_id = m.id_model
WHERE 1=1""" + region_filter + """
GROUP BY b.name_brand ORDER BY b.name_brand LIMIT 1000
""")
rows = session.execute(sql, params).mappings().all()
return [{'name': r['name'], 'model_count': r['model_count'],
'vehicle_count': r['vehicle_count']} for r in rows]
else:
sql = text("""
SELECT DISTINCT b.name_brand AS name
FROM brands b
JOIN models m ON m.brand_id = b.id_brand
JOIN model_year_engine mye ON mye.model_id = m.id_model
WHERE 1=1""" + region_filter + """
ORDER BY b.name_brand LIMIT 1000
""")
rows = session.execute(sql, params).mappings().all()
return [r['name'] for r in rows]
finally:
session.close()
def get_all_years():
session = Session()
try:
rows = session.execute(text(
"SELECT DISTINCT year_car AS year FROM years ORDER BY year_car DESC LIMIT 200"
)).mappings().all()
return [r['year'] for r in rows]
finally:
session.close()
def get_all_engines():
session = Session()
try:
rows = session.execute(text(
"SELECT DISTINCT name_engine AS name FROM engines ORDER BY name_engine LIMIT 5000"
)).mappings().all()
return [r['name'] for r in rows]
finally:
session.close()
def get_models_by_brand(brand_name=None):
session = Session()
try:
if brand_name:
sql = text("""
SELECT DISTINCT m.name_model AS name
FROM models m
JOIN brands b ON m.brand_id = b.id_brand
JOIN model_year_engine mye ON mye.model_id = m.id_model
WHERE b.name_brand ILIKE :brand
ORDER BY m.name_model LIMIT 1000
""")
rows = session.execute(sql, {'brand': brand_name}).mappings().all()
else:
sql = text("""
SELECT DISTINCT m.name_model AS name
FROM models m
JOIN model_year_engine mye ON mye.model_id = m.id_model
ORDER BY m.name_model LIMIT 1000
""")
rows = session.execute(sql).mappings().all()
return [r['name'] for r in rows]
finally:
session.close()
def search_vehicles(brand=None, model=None, year=None, engine_name=None, with_parts=False, page=1, per_page=50):
session = Session()
try:
per_page = min(per_page, 100)
offset = (page - 1) * per_page
base_from = """
FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year
JOIN engines e ON mye.engine_id = e.id_engine
LEFT JOIN fuel_type ft ON e.id_fuel = ft.id_fuel
LEFT JOIN drivetrain dt ON mye.id_drivetrain = dt.id_drivetrain
LEFT JOIN transmission tr ON mye.id_transmission = tr.id_transmission
"""
if with_parts:
base_from += " JOIN (SELECT DISTINCT model_year_engine_id FROM vehicle_parts) AS has_parts ON mye.id_mye = has_parts.model_year_engine_id"
where = " WHERE 1=1"
params = {}
if brand:
where += " AND b.name_brand ILIKE :brand"
params['brand'] = brand
if model:
where += " AND m.name_model ILIKE :model"
params['model'] = model
if year:
where += " AND y.year_car = :year"
params['year'] = int(year)
if engine_name:
where += " AND e.name_engine = :engine"
params['engine'] = engine_name
count_sql = text("SELECT COUNT(*) AS total " + base_from + where)
total_count = session.execute(count_sql, params).mappings().first()['total']
data_params = dict(params)
data_params['limit'] = per_page
data_params['offset'] = offset
query = text("""
SELECT b.name_brand AS brand, m.name_model AS model, y.year_car AS year,
e.name_engine AS engine, e.power_hp, e.torque_nm, e.displacement_cc,
e.cylinders, ft.name_fuel AS fuel_type, mye.trim_level,
dt.name_drivetrain AS drivetrain, tr.name_transmission AS transmission
""" + base_from + where + " ORDER BY b.name_brand, m.name_model, y.year_car LIMIT :limit OFFSET :offset")
rows = session.execute(query, data_params).mappings().all()
vehicles = []
for r in rows:
vehicles.append({
'brand': r['brand'], 'model': r['model'], 'year': r['year'],
'engine': r['engine'], 'power_hp': r['power_hp'] or 0,
'torque_nm': r['torque_nm'] or 0, 'displacement_cc': r['displacement_cc'] or 0,
'cylinders': r['cylinders'] or 0, 'fuel_type': r['fuel_type'] or 'unknown',
'trim_level': r['trim_level'] or 'unknown',
'drivetrain': r['drivetrain'] or 'unknown',
'transmission': r['transmission'] or 'unknown'
})
total_pages = (total_count + per_page - 1) // per_page
return {'data': vehicles, 'pagination': {'page': page, 'per_page': per_page,
'total': total_count, 'total_pages': total_pages}}
finally:
session.close()
# ============================================================================
# Static Routes
# ============================================================================
@app.route('/')
def index():
return redirect('/login.html')
@app.route('/admin')
def admin_page():
return send_from_directory('.', 'admin.html')
@app.route('/landing')
def landing_page():
return send_from_directory('.', 'customer-landing.html')
@app.route('/diagramas')
def diagrams_page():
return send_from_directory('.', 'diagrams.html')
@app.route('/index.html')
def index_html():
return send_from_directory('.', 'index.html')
@app.route('/admin.html')
def admin_html():
return send_from_directory('.', 'admin.html')
@app.route('/customer-landing.html')
def customer_landing_html():
return send_from_directory('.', 'customer-landing.html')
@app.route('/diagrams.html')
def diagrams_html():
return send_from_directory('.', 'diagrams.html')
@app.route('/static/<path:path>')
def static_files(path):
return send_from_directory('static', path)
@app.route('/shared.css')
def shared_css():
return send_from_directory('.', 'shared.css')
@app.route('/nav.js')
def nav_js():
return send_from_directory('.', 'nav.js')
@app.route('/dashboard.js')
def dashboard_js():
return send_from_directory('.', 'dashboard.js')
@app.route('/admin.js')
def admin_js():
return send_from_directory('.', 'admin.js')
@app.route('/enhanced-search.js')
def enhanced_search_js():
return send_from_directory('.', 'enhanced-search.js')
# ============================================================================
# Core API Endpoints
# ============================================================================
@app.route('/api/catalog/stats')
def api_catalog_stats():
session = Session()
try:
row = session.execute(text("""
SELECT
(SELECT COUNT(*) FROM brands) AS brands,
(SELECT COUNT(*) FROM models) AS models,
(SELECT COUNT(*) FROM model_year_engine) AS vehicles,
(SELECT COUNT(*) FROM parts) AS parts
""")).mappings().first()
return jsonify({
'brands': row['brands'], 'models': row['models'],
'vehicles': row['vehicles'], 'parts': row['parts']
})
finally:
session.close()
@app.route('/api/brands')
def api_brands():
detailed = request.args.get('detailed', 'false').lower() == 'true'
region = request.args.get('region')
region_mask = int(region) if region else None
return jsonify(get_all_brands(detailed=detailed, region_mask=region_mask))
@app.route('/api/years')
def api_years():
brand = request.args.get('brand')
model = request.args.get('model')
session = Session()
try:
q = """SELECT DISTINCT y.year_car AS year FROM years y
JOIN model_year_engine mye ON y.id_year = mye.year_id
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand WHERE 1=1"""
params = {}
if brand:
q += " AND b.name_brand ILIKE :brand"
params['brand'] = brand
if model:
q += " AND m.name_model ILIKE :model"
params['model'] = model
q += " ORDER BY y.year_car DESC"
rows = session.execute(text(q), params).mappings().all()
return jsonify([r['year'] for r in rows])
finally:
session.close()
@app.route('/api/engines')
def api_engines():
brand = request.args.get('brand')
model = request.args.get('model')
year = request.args.get('year')
session = Session()
try:
q = """SELECT DISTINCT e.name_engine AS name FROM engines e
JOIN model_year_engine mye ON e.id_engine = mye.engine_id
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year WHERE 1=1"""
params = {}
if brand:
q += " AND b.name_brand ILIKE :brand"
params['brand'] = brand
if model:
q += " AND m.name_model ILIKE :model"
params['model'] = model
if year:
q += " AND y.year_car = :year"
params['year'] = int(year)
q += " ORDER BY e.name_engine"
rows = session.execute(text(q), params).mappings().all()
return jsonify([r['name'] for r in rows])
finally:
session.close()
@app.route('/api/models')
def api_models():
brand = request.args.get('brand')
detailed = request.args.get('detailed', 'false').lower() == 'true'
if detailed and brand:
session = Session()
try:
sql = text("""
SELECT m.name_model AS name, MIN(y.year_car) AS year_min,
MAX(y.year_car) AS year_max, COUNT(DISTINCT y.year_car) AS year_count,
COUNT(DISTINCT mye.id_mye) AS vehicle_count,
COUNT(DISTINCT e.name_engine) AS engine_count
FROM models m
JOIN brands b ON m.brand_id = b.id_brand
JOIN model_year_engine mye ON mye.model_id = m.id_model
JOIN years y ON mye.year_id = y.id_year
JOIN engines e ON mye.engine_id = e.id_engine
WHERE b.name_brand ILIKE :brand
GROUP BY m.name_model ORDER BY m.name_model LIMIT 1000
""")
rows = session.execute(sql, {'brand': brand}).mappings().all()
return jsonify([{'name': r['name'], 'year_min': r['year_min'], 'year_max': r['year_max'],
'year_count': r['year_count'], 'vehicle_count': r['vehicle_count'],
'engine_count': r['engine_count']} for r in rows])
finally:
session.close()
return jsonify(get_models_by_brand(brand))
@app.route('/api/vehicles')
def api_vehicles():
brand = request.args.get('brand')
model = request.args.get('model')
year = request.args.get('year')
eng = request.args.get('engine')
page = request.args.get('page', 1, type=int)
per_page = request.args.get('per_page', 50, type=int)
return jsonify(search_vehicles(brand, model, year, eng, page=page, per_page=per_page))
@app.route('/api/model-year-engine')
def api_model_year_engine():
try:
brand = request.args.get('brand')
model = request.args.get('model')
year = request.args.get('year', type=int)
with_parts = request.args.get('with_parts', 'false').lower() == 'true'
page = request.args.get('page', 1, type=int)
per_page = min(request.args.get('per_page', 50, type=int), 100)
offset = (page - 1) * per_page
session = Session()
try:
base_from = """
FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year
JOIN engines e ON mye.engine_id = e.id_engine
LEFT JOIN drivetrain dt ON mye.id_drivetrain = dt.id_drivetrain
LEFT JOIN transmission tr ON mye.id_transmission = tr.id_transmission
"""
if with_parts:
base_from += " JOIN (SELECT DISTINCT model_year_engine_id FROM vehicle_parts) AS has_parts ON mye.id_mye = has_parts.model_year_engine_id"
where = " WHERE 1=1"
params = {}
if brand:
where += " AND b.name_brand ILIKE :brand"
params['brand'] = brand
if model:
where += " AND m.name_model ILIKE :model"
params['model'] = model
if year:
where += " AND y.year_car = :year"
params['year'] = year
total_count = session.execute(text("SELECT COUNT(*) AS total " + base_from + where), params).mappings().first()['total']
data_params = dict(params)
data_params['limit'] = per_page
data_params['offset'] = offset
query = text("""
SELECT mye.id_mye AS id, b.name_brand AS brand, m.name_model AS model,
y.year_car AS year, e.name_engine AS engine, mye.trim_level,
dt.name_drivetrain AS drivetrain, tr.name_transmission AS transmission
""" + base_from + where + " ORDER BY b.name_brand, m.name_model, y.year_car, e.name_engine LIMIT :limit OFFSET :offset")
rows = session.execute(query, data_params).mappings().all()
records = [{'id': r['id'], 'brand': r['brand'], 'model': r['model'], 'year': r['year'],
'engine': r['engine'], 'trim_level': r['trim_level'],
'drivetrain': r['drivetrain'], 'transmission': r['transmission']} for r in rows]
total_pages = (total_count + per_page - 1) // per_page
return jsonify({'data': records, 'pagination': {'page': page, 'per_page': per_page,
'total': total_count, 'total_pages': total_pages}})
finally:
session.close()
except Exception as e:
return jsonify({'error': str(e)}), 500
# ============================================================================
# Parts Catalog Endpoints
# ============================================================================
@app.route('/api/categories')
def api_categories():
session = Session()
try:
rows = session.execute(text("""
SELECT id_part_category AS id, name_part_category AS name, name_es, slug, icon_name, display_order, parent_id
FROM part_categories ORDER BY display_order, name_part_category LIMIT 50
""")).mappings().all()
categories_dict = {}
root_categories = []
for r in rows:
cat = {'id': r['id'], 'name': r['name'], 'name_es': r['name_es'], 'slug': r['slug'],
'icon_name': r['icon_name'], 'display_order': r['display_order'], 'children': []}
categories_dict[r['id']] = cat
if r['parent_id'] is None:
root_categories.append(cat)
for r in rows:
if r['parent_id'] is not None and r['parent_id'] in categories_dict:
categories_dict[r['parent_id']]['children'].append(categories_dict[r['id']])
return jsonify(root_categories)
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/categories/<int:category_id>/groups')
def api_category_groups(category_id):
session = Session()
try:
rows = session.execute(text("""
SELECT id_part_group AS id, name_part_group AS name, name_es, slug, display_order
FROM part_groups WHERE category_id = :cid ORDER BY display_order, name_part_group LIMIT 200
"""), {'cid': category_id}).mappings().all()
return jsonify([{'id': r['id'], 'name': r['name'], 'name_es': r['name_es'],
'slug': r['slug'], 'display_order': r['display_order']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/parts')
def api_parts():
try:
group_id = request.args.get('group_id', type=int)
category_id = request.args.get('category_id', type=int)
search = request.args.get('search')
page = request.args.get('page', 1, type=int)
per_page = min(request.args.get('per_page', 50, type=int), 100)
offset = (page - 1) * per_page
session = Session()
try:
where = " WHERE 1=1"
params = {}
if group_id:
where += " AND p.group_id = :group_id"
params['group_id'] = group_id
if category_id:
where += " AND pg.category_id = :category_id"
params['category_id'] = category_id
if search:
where += " AND (p.name_part ILIKE :search OR p.name_es ILIKE :search OR p.oem_part_number ILIKE :search)"
params['search'] = '%' + search + '%'
base = """
FROM parts p
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
"""
total_count = session.execute(text("SELECT COUNT(*) AS total " + base + where), params).mappings().first()['total']
data_params = dict(params)
data_params['limit'] = per_page
data_params['offset'] = offset
rows = session.execute(text("""
SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
p.group_id, pg.name_part_group AS group_name, pc.name_part_category AS category_name,
p.image_url
""" + base + where + " ORDER BY p.name_part LIMIT :limit OFFSET :offset"), data_params).mappings().all()
parts = [{'id': r['id'], 'oem_part_number': r['oem_part_number'], 'name': r['name'],
'name_es': r['name_es'], 'group_id': r['group_id'], 'group_name': r['group_name'],
'category_name': r['category_name'], 'image_url': r['image_url']} for r in rows]
total_pages = (total_count + per_page - 1) // per_page
return jsonify({'data': parts, 'pagination': {'page': page, 'per_page': per_page,
'total': total_count, 'total_pages': total_pages}})
finally:
session.close()
except Exception as e:
return jsonify({'error': str(e)}), 500
@app.route('/api/parts/<int:part_id>')
def api_part_detail(part_id):
session = Session()
try:
row = session.execute(text("""
SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
p.description, p.description_es, p.group_id, p.image_url,
pg.name_part_group AS group_name, pg.name_es AS group_name_es,
pc.id_part_category AS category_id, pc.name_part_category AS category_name,
pc.name_es AS category_name_es
FROM parts p
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE p.id_part = :pid
"""), {'pid': part_id}).mappings().first()
if row is None:
return jsonify({'error': 'Part not found'}), 404
return jsonify({'id': row['id'], 'oem_part_number': row['oem_part_number'], 'name': row['name'],
'name_es': row['name_es'], 'description': row['description'],
'description_es': row['description_es'], 'group_id': row['group_id'],
'group_name': row['group_name'], 'image_url': row['image_url'],
'group_name_es': row['group_name_es'], 'category_id': row['category_id'],
'category_name': row['category_name'], 'category_name_es': row['category_name_es']})
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/vehicles/<int:mye_id>/categories')
def api_vehicle_categories(mye_id):
session = Session()
try:
rows = session.execute(text("""
SELECT DISTINCT pc.id_part_category AS id, pc.name_part_category AS name,
pc.name_es, pc.slug, pc.icon_name, pc.display_order
FROM part_categories pc
JOIN part_groups pg ON pg.category_id = pc.id_part_category
JOIN parts p ON p.group_id = pg.id_part_group
JOIN vehicle_parts vp ON vp.part_id = p.id_part
WHERE vp.model_year_engine_id = :mye_id
ORDER BY pc.display_order, pc.name_part_category LIMIT 50
"""), {'mye_id': mye_id}).mappings().all()
return jsonify([{'id': r['id'], 'name': r['name'], 'name_es': r['name_es'],
'slug': r['slug'], 'icon_name': r['icon_name'],
'display_order': r['display_order']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/vehicles/<int:mye_id>/groups')
def api_vehicle_groups(mye_id):
session = Session()
try:
category_id = request.args.get('category_id', type=int)
q = """
SELECT DISTINCT pg.id_part_group AS id, pg.name_part_group AS name, pg.name_es,
pg.slug, pg.display_order, COUNT(DISTINCT p.id_part) AS parts_count
FROM part_groups pg
JOIN parts p ON p.group_id = pg.id_part_group
JOIN vehicle_parts vp ON vp.part_id = p.id_part
WHERE vp.model_year_engine_id = :mye_id
"""
params = {'mye_id': mye_id}
if category_id:
q += " AND pg.category_id = :cid"
params['cid'] = category_id
q += " GROUP BY pg.id_part_group, pg.name_part_group, pg.name_es, pg.slug, pg.display_order ORDER BY pg.display_order, pg.name_part_group LIMIT 200"
rows = session.execute(text(q), params).mappings().all()
return jsonify([{'id': r['id'], 'name': r['name'], 'name_es': r['name_es'],
'slug': r['slug'], 'display_order': r['display_order'],
'parts_count': r['parts_count']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/vehicles/<int:mye_id>/parts')
def api_vehicle_parts(mye_id):
try:
category_id = request.args.get('category_id', type=int)
group_id = request.args.get('group_id', type=int)
page = request.args.get('page', 1, type=int)
per_page = min(request.args.get('per_page', 50, type=int), 100)
offset = (page - 1) * per_page
session = Session()
try:
base = """
FROM vehicle_parts vp
JOIN parts p ON vp.part_id = p.id_part
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
LEFT JOIN position_part pp ON vp.id_position_part = pp.id_position_part
WHERE vp.model_year_engine_id = :mye_id
"""
params = {'mye_id': mye_id}
if category_id:
base += " AND pc.id_part_category = :cid"
params['cid'] = category_id
if group_id:
base += " AND pg.id_part_group = :gid"
params['gid'] = group_id
total_count = session.execute(text("SELECT COUNT(*) AS total " + base), params).mappings().first()['total']
data_params = dict(params)
data_params['limit'] = per_page
data_params['offset'] = offset
rows = session.execute(text("""
SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
vp.quantity_required, pp.name_position_part AS position,
pc.name_part_category AS category_name, pg.name_part_group AS group_name,
p.image_url
""" + base + " ORDER BY pc.display_order, pg.display_order, p.name_part LIMIT :limit OFFSET :offset"), data_params).mappings().all()
parts = [{'id': r['id'], 'oem_part_number': r['oem_part_number'], 'name': r['name'],
'name_es': r['name_es'], 'quantity_required': r['quantity_required'],
'position': r['position'], 'category_name': r['category_name'],
'group_name': r['group_name'], 'image_url': r['image_url']} for r in rows]
total_pages = (total_count + per_page - 1) // per_page
return jsonify({'data': parts, 'pagination': {'page': page, 'per_page': per_page,
'total': total_count, 'total_pages': total_pages}})
finally:
session.close()
except Exception as e:
return jsonify({'error': str(e)}), 500
# ============================================================================
# Cross-References and Aftermarket Endpoints
# ============================================================================
@app.route('/api/manufacturers')
def api_manufacturers():
session = Session()
try:
manufacturer_type = request.args.get('type')
quality_tier = request.args.get('quality_tier')
q = """
SELECT mfr.id_manufacture AS id, mfr.name_manufacture AS name,
mt.name_type_manu AS type, qt.name_quality AS quality_tier,
co.name_country AS country, mfr.logo_url, mfr.website
FROM manufacturers mfr
LEFT JOIN manufacture_type mt ON mfr.id_type_manu = mt.id_type_manu
LEFT JOIN quality_tier qt ON mfr.id_quality_tier = qt.id_quality_tier
LEFT JOIN countries co ON mfr.id_country = co.id_country
WHERE 1=1
"""
params = {}
if manufacturer_type:
q += " AND mt.name_type_manu ILIKE :type"
params['type'] = manufacturer_type
if quality_tier:
q += " AND qt.name_quality ILIKE :qt"
params['qt'] = quality_tier
q += " ORDER BY mfr.name_manufacture LIMIT 200"
rows = session.execute(text(q), params).mappings().all()
return jsonify([{'id': r['id'], 'name': r['name'], 'type': r['type'],
'quality_tier': r['quality_tier'], 'country': r['country'],
'logo_url': r['logo_url'], 'website': r['website']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/parts/<int:part_id>/alternatives')
def api_part_alternatives(part_id):
session = Session()
try:
quality_tier = request.args.get('quality_tier')
manufacturer_id = request.args.get('manufacturer_id', type=int)
q = """
SELECT ap.id_aftermarket_parts AS id, ap.part_number, ap.name_aftermarket_parts AS name,
ap.name_es, mfr.name_manufacture AS manufacturer_name, ap.manufacturer_id,
qt.name_quality AS quality_tier, ap.price_usd, ap.warranty_months
FROM aftermarket_parts ap
JOIN manufacturers mfr ON ap.manufacturer_id = mfr.id_manufacture
LEFT JOIN quality_tier qt ON ap.id_quality_tier = qt.id_quality_tier
WHERE ap.oem_part_id = :pid
"""
params = {'pid': part_id}
if quality_tier:
q += " AND qt.name_quality ILIKE :qt"
params['qt'] = quality_tier
if manufacturer_id:
q += " AND ap.manufacturer_id = :mid"
params['mid'] = manufacturer_id
q += " ORDER BY qt.name_quality DESC, ap.price_usd ASC LIMIT 50"
rows = session.execute(text(q), params).mappings().all()
return jsonify([{'id': r['id'], 'part_number': r['part_number'], 'name': r['name'],
'name_es': r['name_es'], 'manufacturer_name': r['manufacturer_name'],
'manufacturer_id': r['manufacturer_id'], 'quality_tier': r['quality_tier'],
'price_usd': r['price_usd'], 'warranty_months': r['warranty_months'],
'in_stock': None} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/parts/<int:part_id>/cross-references')
def api_part_cross_references(part_id):
session = Session()
try:
rows = session.execute(text("""
SELECT pcr.id_part_cross_ref AS id, pcr.cross_reference_number,
rt.name_ref_type AS reference_type, pcr.source_ref AS source, pcr.notes
FROM part_cross_references pcr
LEFT JOIN reference_type rt ON pcr.id_ref_type = rt.id_ref_type
WHERE pcr.part_id = :pid
ORDER BY rt.name_ref_type, pcr.cross_reference_number LIMIT 100
"""), {'pid': part_id}).mappings().all()
return jsonify([{'id': r['id'], 'cross_reference_number': r['cross_reference_number'],
'reference_type': r['reference_type'], 'source': r['source'],
'notes': r['notes']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/search/part-number/<part_number>')
def api_search_part_number(part_number):
session = Session()
try:
results = []
st = '%' + part_number + '%'
for row in session.execute(text("SELECT id_part AS id, oem_part_number, name_part AS name, name_es FROM parts WHERE oem_part_number ILIKE :s"), {'s': st}).mappings().all():
results.append({'id': row['id'], 'oem_part_number': row['oem_part_number'], 'name': row['name'], 'name_es': row['name_es'], 'match_type': 'oem', 'matched_number': row['oem_part_number']})
for row in session.execute(text("SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es, ap.part_number FROM aftermarket_parts ap JOIN parts p ON ap.oem_part_id = p.id_part WHERE ap.part_number ILIKE :s"), {'s': st}).mappings().all():
results.append({'id': row['id'], 'oem_part_number': row['oem_part_number'], 'name': row['name'], 'name_es': row['name_es'], 'match_type': 'aftermarket', 'matched_number': row['part_number']})
for row in session.execute(text("SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es, pcr.cross_reference_number FROM part_cross_references pcr JOIN parts p ON pcr.part_id = p.id_part WHERE pcr.cross_reference_number ILIKE :s"), {'s': st}).mappings().all():
results.append({'id': row['id'], 'oem_part_number': row['oem_part_number'], 'name': row['name'], 'name_es': row['name_es'], 'match_type': 'cross_reference', 'matched_number': row['cross_reference_number']})
return jsonify(results)
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/aftermarket')
def api_aftermarket_parts():
session = Session()
try:
manufacturer_id = request.args.get('manufacturer_id', type=int)
quality_tier = request.args.get('quality_tier')
search = request.args.get('search')
page = request.args.get('page', 1, type=int)
per_page = min(request.args.get('per_page', 50, type=int), 100)
offset = (page - 1) * per_page
where = " WHERE 1=1"
params = {}
if manufacturer_id:
where += " AND ap.manufacturer_id = :mid"
params['mid'] = manufacturer_id
if quality_tier:
where += " AND qt.name_quality ILIKE :qt"
params['qt'] = quality_tier
if search:
where += " AND (ap.name_aftermarket_parts ILIKE :s OR ap.part_number ILIKE :s OR p.oem_part_number ILIKE :s)"
params['s'] = '%' + search + '%'
base = """
FROM aftermarket_parts ap
JOIN parts p ON ap.oem_part_id = p.id_part
JOIN manufacturers m ON ap.manufacturer_id = m.id_manufacture
LEFT JOIN quality_tier qt ON ap.id_quality_tier = qt.id_quality_tier
"""
total_count = session.execute(text("SELECT COUNT(*) AS total " + base + where), params).mappings().first()['total']
data_params = dict(params)
data_params['limit'] = per_page
data_params['offset'] = offset
rows = session.execute(text("""
SELECT ap.id_aftermarket_parts AS id, ap.part_number, ap.name_aftermarket_parts AS name,
p.oem_part_number, m.name_manufacture AS manufacturer_name,
qt.name_quality AS quality_tier, ap.price_usd
""" + base + where + " ORDER BY ap.name_aftermarket_parts LIMIT :limit OFFSET :offset"), data_params).mappings().all()
parts = [{'id': r['id'], 'part_number': r['part_number'], 'name': r['name'],
'oem_part_number': r['oem_part_number'], 'manufacturer_name': r['manufacturer_name'],
'quality_tier': r['quality_tier'], 'price_usd': r['price_usd']} for r in rows]
total_pages = (total_count + per_page - 1) // per_page
return jsonify({'data': parts, 'pagination': {'page': page, 'per_page': per_page,
'total': total_count, 'total_pages': total_pages}})
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ============================================================================
# Diagram Endpoints
# ============================================================================
@app.route('/api/diagrams')
def api_diagrams():
session = Session()
try:
group_id = request.args.get('group_id', type=int)
q = """
SELECT d.id_diagram AS id, d.name_diagram AS name, d.name_es, d.group_id,
pg.name_part_group AS group_name, d.thumbnail_path, d.display_order
FROM diagrams d JOIN part_groups pg ON d.group_id = pg.id_part_group WHERE 1=1
"""
params = {}
if group_id:
q += " AND d.group_id = :gid"
params['gid'] = group_id
q += " ORDER BY d.display_order, d.name_diagram LIMIT 200"
rows = session.execute(text(q), params).mappings().all()
return jsonify([{'id': r['id'], 'name': r['name'], 'name_es': r['name_es'],
'group_id': r['group_id'], 'group_name': r['group_name'],
'thumbnail_path': r['thumbnail_path'], 'display_order': r['display_order']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/diagrams/<int:diagram_id>')
def api_diagram_detail(diagram_id):
session = Session()
try:
row = session.execute(text("""
SELECT d.id_diagram AS id, d.name_diagram AS name, d.name_es, d.group_id,
pg.name_part_group AS group_name, d.image_path
FROM diagrams d JOIN part_groups pg ON d.group_id = pg.id_part_group
WHERE d.id_diagram = :did
"""), {'did': diagram_id}).mappings().first()
if row is None:
return jsonify({'error': 'Diagram not found'}), 404
image_path = row['image_path'] or ''
image_url = '/' + image_path if image_path and not image_path.startswith('/') else image_path
diagram = {'id': row['id'], 'name': row['name'], 'name_es': row['name_es'],
'group_id': row['group_id'], 'group_name': row['group_name'],
'image_path': image_path, 'image_url': image_url,
'svg_content': None, 'width': None, 'height': None, 'hotspots': []}
hotspot_rows = session.execute(text("""
SELECT h.id_dgr_hotspot AS id, h.part_id, h.callout_number,
sh.name_shape AS shape, h.coords,
p.name_part AS part_name, p.oem_part_number AS part_number
FROM diagram_hotspots h
LEFT JOIN parts p ON h.part_id = p.id_part
LEFT JOIN shapes sh ON h.id_shape = sh.id_shape
WHERE h.diagram_id = :did ORDER BY h.callout_number
"""), {'did': diagram_id}).mappings().all()
for hr in hotspot_rows:
diagram['hotspots'].append({'id': hr['id'], 'part_id': hr['part_id'],
'callout_number': hr['callout_number'], 'label': None, 'shape': hr['shape'],
'coords': hr['coords'], 'color': None, 'part_name': hr['part_name'],
'part_number': hr['part_number']})
return jsonify(diagram)
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/diagrams/<int:diagram_id>/hotspots')
def api_diagram_hotspots(diagram_id):
session = Session()
try:
rows = session.execute(text("""
SELECT h.id_dgr_hotspot AS id, h.part_id, h.callout_number,
sh.name_shape AS shape, h.coords,
p.name_part AS part_name, p.oem_part_number AS part_number
FROM diagram_hotspots h
LEFT JOIN parts p ON h.part_id = p.id_part
LEFT JOIN shapes sh ON h.id_shape = sh.id_shape
WHERE h.diagram_id = :did ORDER BY h.callout_number LIMIT 500
"""), {'did': diagram_id}).mappings().all()
return jsonify([{'id': r['id'], 'part_id': r['part_id'], 'callout_number': r['callout_number'],
'label': None, 'shape': r['shape'], 'coords': r['coords'], 'color': None,
'part_name': r['part_name'], 'part_number': r['part_number']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/groups/<int:group_id>/diagrams')
def api_group_diagrams(group_id):
session = Session()
try:
rows = session.execute(text("""
SELECT id_diagram AS id, name_diagram AS name, name_es, thumbnail_path, display_order
FROM diagrams WHERE group_id = :gid ORDER BY display_order, name_diagram LIMIT 100
"""), {'gid': group_id}).mappings().all()
return jsonify([{'id': r['id'], 'name': r['name'], 'name_es': r['name_es'],
'thumbnail_path': r['thumbnail_path'], 'display_order': r['display_order']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/vehicles/<int:mye_id>/diagrams')
def api_vehicle_diagrams(mye_id):
session = Session()
try:
rows = session.execute(text("""
SELECT DISTINCT d.id_diagram AS id, d.name_diagram AS name, d.name_es,
d.group_id, d.image_path, pg.name_part_group AS group_name,
pc.id_part_category AS category_id, pc.name_part_category AS category_name,
d.thumbnail_path, vd.notes
FROM vehicle_diagrams vd
JOIN diagrams d ON vd.diagram_id = d.id_diagram
JOIN part_groups pg ON d.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE vd.model_year_engine_id = :mye_id
ORDER BY pc.display_order, pg.display_order, d.display_order LIMIT 200
"""), {'mye_id': mye_id}).mappings().all()
diagrams = []
for r in rows:
ip = r['image_path'] or ''
iu = '/' + ip if ip and not ip.startswith('/') else ip
diagrams.append({'id': r['id'], 'name': r['name'], 'name_es': r['name_es'],
'group_id': r['group_id'], 'group_name': r['group_name'],
'category_id': r['category_id'], 'category_name': r['category_name'],
'image_url': iu, 'thumbnail_path': r['thumbnail_path'], 'notes': r['notes']})
return jsonify(diagrams)
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/diagrams/<int:diagram_id>/parts')
def api_diagram_parts(diagram_id):
session = Session()
try:
mye_id = request.args.get('mye_id', type=int)
q = """
SELECT DISTINCT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
p.description, pg.id_part_group AS group_id, pg.name_part_group AS group_name,
pg.name_es AS group_name_es
FROM vehicle_diagrams vd
JOIN vehicle_parts vp ON vp.model_year_engine_id = vd.model_year_engine_id
JOIN parts p ON vp.part_id = p.id_part
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE vd.diagram_id = :did AND pc.id_part_category IN (10, 11)
"""
params = {'did': diagram_id}
if mye_id:
q += " AND vd.model_year_engine_id = :mye_id"
params['mye_id'] = mye_id
q += " ORDER BY pg.name_part_group, p.oem_part_number LIMIT 200"
rows = session.execute(text(q), params).mappings().all()
# Batch cross-refs (N+1 fix)
xrefs_map = {}
if rows:
part_ids = list(set(r['id'] for r in rows))
in_params = {}
in_pl = []
for i, pid in enumerate(part_ids):
in_params[f'pid_{i}'] = pid
in_pl.append(f':pid_{i}')
xref_rows = session.execute(text(f"""
SELECT part_id, cross_reference_number, source_ref AS source
FROM part_cross_references WHERE part_id IN ({', '.join(in_pl)})
"""), in_params).mappings().all()
for xr in xref_rows:
xrefs_map.setdefault(xr['part_id'], []).append({'number': xr['cross_reference_number'], 'source': xr['source']})
return jsonify([{'id': r['id'], 'part_number': r['oem_part_number'], 'name': r['name'],
'name_es': r['name_es'], 'description': r['description'],
'group_name': r['group_name'], 'group_name_es': r['group_name_es'],
'cross_references': xrefs_map.get(r['id'], [])} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/diagrams/search')
def api_diagrams_search():
session = Session()
try:
q = request.args.get('q', '').strip()
brand = request.args.get('brand', '').strip()
model = request.args.get('model', '').strip()
if q:
rows = session.execute(text("""
SELECT DISTINCT d.id_diagram AS id, d.name_diagram AS name, d.name_es,
d.image_path, d.source_diagram AS source
FROM diagrams d WHERE d.name_diagram ILIKE :q OR d.name_es ILIKE :q
ORDER BY d.name_diagram LIMIT 50
"""), {'q': '%' + q + '%'}).mappings().all()
elif brand or model:
sql = """
SELECT DISTINCT d.id_diagram AS id, d.name_diagram AS name, d.name_es,
d.image_path, d.source_diagram AS source
FROM diagrams d
JOIN vehicle_diagrams vd ON vd.diagram_id = d.id_diagram
JOIN model_year_engine mye ON vd.model_year_engine_id = mye.id_mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand WHERE 1=1
"""
params = {}
if brand:
sql += " AND b.name_brand ILIKE :brand"
params['brand'] = brand
if model:
sql += " AND m.name_model ILIKE :model"
params['model'] = model
sql += " ORDER BY d.name_diagram LIMIT 50"
rows = session.execute(text(sql), params).mappings().all()
else:
rows = session.execute(text("""
SELECT d.id_diagram AS id, d.name_diagram AS name, d.name_es,
d.image_path, d.source_diagram AS source
FROM diagrams d WHERE d.source_diagram = 'MOOG Catalog'
ORDER BY d.name_diagram LIMIT 50
""")).mappings().all()
return jsonify([{'id': r['id'], 'name': r['name'], 'name_es': r['name_es'],
'image_path': r['image_path'], 'source': r['source']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/hotspots/<int:hotspot_id>')
def api_hotspot_detail(hotspot_id):
session = Session()
try:
row = session.execute(text("""
SELECT h.id_dgr_hotspot AS id, h.diagram_id, h.part_id, h.callout_number,
sh.name_shape AS shape, h.coords
FROM diagram_hotspots h
LEFT JOIN shapes sh ON h.id_shape = sh.id_shape
WHERE h.id_dgr_hotspot = :hid
"""), {'hid': hotspot_id}).mappings().first()
if row is None:
return jsonify({'error': 'Hotspot not found'}), 404
hotspot = {'id': row['id'], 'diagram_id': row['diagram_id'], 'part_id': row['part_id'],
'callout_number': row['callout_number'], 'label': None, 'shape': row['shape'],
'coords': row['coords'], 'color': None, 'part': None}
if row['part_id']:
pr = session.execute(text("""
SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
pg.name_part_group AS group_name, pc.name_part_category AS category_name
FROM parts p JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE p.id_part = :pid
"""), {'pid': row['part_id']}).mappings().first()
if pr:
hotspot['part'] = {'id': pr['id'], 'oem_part_number': pr['oem_part_number'],
'name': pr['name'], 'name_es': pr['name_es'],
'group_name': pr['group_name'], 'category_name': pr['category_name']}
return jsonify(hotspot)
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ============================================================================
# Search and VIN Endpoints
# ============================================================================
def validate_vin(vin):
if not vin or len(vin) != 17:
return False
return bool(re.compile(r'^[A-HJ-NPR-Z0-9]{17}$', re.IGNORECASE).match(vin))
def find_vehicle_in_terms(session, terms):
if len(terms) < 2:
return None, terms
year_terms = []
other_terms = []
for t in terms:
if t.isdigit() and 1980 <= int(t) <= 2030:
year_terms.append(t)
else:
other_terms.append(t)
if not other_terms:
return None, terms
best_match = None
used_terms = []
for num_terms in range(min(3, len(other_terms)), 0, -1):
if best_match:
break
for i in range(len(other_terms) - num_terms + 1):
test_terms = other_terms[i:i + num_terms]
if year_terms:
test_terms = test_terms + year_terms[:1]
where_clauses = []
params = {}
for idx, t in enumerate(test_terms):
tp = f"%{t}%"
where_clauses.append(f"(b.name_brand ILIKE :t{idx}_b OR m.name_model ILIKE :t{idx}_m OR CAST(y.year_car AS TEXT) ILIKE :t{idx}_y)")
params[f't{idx}_b'] = tp
params[f't{idx}_m'] = tp
params[f't{idx}_y'] = tp
where_sql = " AND ".join(where_clauses)
row = session.execute(text(f"""
SELECT mye.id_mye AS id, b.name_brand AS brand, m.name_model AS model,
y.year_car AS year, e.name_engine AS engine
FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year
JOIN engines e ON mye.engine_id = e.id_engine
WHERE {where_sql} ORDER BY y.year_car DESC LIMIT 1
"""), params).mappings().first()
if row:
best_match = {'id': row['id'], 'brand': row['brand'], 'model': row['model'],
'year': row['year'], 'engine': row['engine']}
used_terms = test_terms
break
if best_match:
remaining = []
used_lower = [t.lower() for t in used_terms]
for t in terms:
if t.lower() not in used_lower:
remaining.append(t)
else:
used_lower.remove(t.lower())
return best_match, remaining
return None, terms
@app.route('/api/search')
def api_search():
session = Session()
try:
q = request.args.get('q', '').strip()
search_type = request.args.get('type', 'all')
category_id = request.args.get('category_id', type=int)
limit = request.args.get('limit', 50, type=int)
offset = request.args.get('offset', 0, type=int)
if not q:
return jsonify({'error': 'Search query is required'}), 400
results = {'parts': [], 'vehicles': [], 'vehicle_parts': [], 'matched_vehicle': None, 'total_count': 0}
terms = q.split()
# Combined vehicle + part search
if len(terms) >= 2 and search_type == 'all':
matched_vehicle, remaining_terms = find_vehicle_in_terms(session, terms)
if matched_vehicle and remaining_terms:
results['matched_vehicle'] = matched_vehicle
mv = matched_vehicle
vp_where_clauses = []
vp_params = {'mv_brand': mv['brand'], 'mv_model': mv['model'], 'mv_year': mv['year'], 'limit': limit}
for i, t in enumerate(remaining_terms):
tp = f"%{t}%"
vp_where_clauses.append(f"(p.name_part ILIKE :vp{i}_n OR p.name_es ILIKE :vp{i}_ne OR p.oem_part_number ILIKE :vp{i}_o OR pg.name_part_group ILIKE :vp{i}_g)")
vp_params[f'vp{i}_n'] = tp
vp_params[f'vp{i}_ne'] = tp
vp_params[f'vp{i}_o'] = tp
vp_params[f'vp{i}_g'] = tp
vp_where_sql = " AND ".join(vp_where_clauses)
rows = session.execute(text(f"""
SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
pg.name_part_group AS group_name, pg.id_part_group AS group_id,
pc.name_part_category AS category_name, pc.id_part_category AS category_id,
vp.quantity_required, pp.name_position_part AS position, p.image_url
FROM vehicle_parts vp
JOIN parts p ON vp.part_id = p.id_part
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
LEFT JOIN position_part pp ON vp.id_position_part = pp.id_position_part
WHERE vp.model_year_engine_id IN (
SELECT mye.id_mye FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year
WHERE b.name_brand ILIKE :mv_brand AND m.name_model ILIKE :mv_model AND y.year_car = :mv_year
) AND ({vp_where_sql}) ORDER BY p.name_part LIMIT :limit
"""), vp_params).mappings().all()
for r in rows:
results['vehicle_parts'].append({'id': r['id'], 'oem_part_number': r['oem_part_number'],
'name': r['name'], 'name_es': r['name_es'], 'image_url': r.get('image_url'),
'group_name': r['group_name'], 'group_id': r['group_id'],
'category_name': r['category_name'], 'category_id': r['category_id'],
'quantity': r['quantity_required'], 'position': r['position'], 'match_type': 'vehicle_part'})
if results['vehicle_parts']:
results['total_count'] = len(results['vehicle_parts'])
return jsonify(results)
# Search parts
if search_type in ('parts', 'all') and terms:
where_clauses = []
params = {}
for i, t in enumerate(terms):
tp = f"%{t}%"
where_clauses.append(f"(p.name_part ILIKE :p{i}_n OR p.name_es ILIKE :p{i}_ne OR p.oem_part_number ILIKE :p{i}_o OR pg.name_part_group ILIKE :p{i}_g OR pc.name_part_category ILIKE :p{i}_c)")
params[f'p{i}_n'] = tp
params[f'p{i}_ne'] = tp
params[f'p{i}_o'] = tp
params[f'p{i}_g'] = tp
params[f'p{i}_c'] = tp
where_sql = " AND ".join(where_clauses)
cat_filter = ""
if category_id:
cat_filter = " AND pc.id_part_category = :cat_id"
params['cat_id'] = category_id
params['first_term'] = f"{terms[0]}%"
params['limit'] = limit
params['offset'] = offset
rows = session.execute(text(f"""
SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
pg.name_part_group AS group_name, pc.name_part_category AS category_name,
p.image_url
FROM parts p JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE ({where_sql}){cat_filter}
ORDER BY CASE WHEN p.oem_part_number ILIKE :first_term THEN 1
WHEN p.name_part ILIKE :first_term THEN 2 ELSE 3 END, p.name_part
LIMIT :limit OFFSET :offset
"""), params).mappings().all()
for r in rows:
results['parts'].append({'id': r['id'], 'oem_part_number': r['oem_part_number'],
'name': r['name'], 'name_es': r['name_es'], 'image_url': r['image_url'],
'group_name': r['group_name'], 'category_name': r['category_name'], 'match_type': 'oem'})
# Aftermarket search
if not category_id:
af_params = {'af_limit': limit, 'af_offset': offset}
af_clauses = []
for i, t in enumerate(terms):
af_clauses.append(f"ap.part_number ILIKE :af{i}")
af_params[f'af{i}'] = f"%{t}%"
af_rows = session.execute(text(f"""
SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
pg.name_part_group AS group_name, pc.name_part_category AS category_name,
ap.part_number AS matched_number, p.image_url
FROM aftermarket_parts ap JOIN parts p ON ap.oem_part_id = p.id_part
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE {' AND '.join(af_clauses)} LIMIT :af_limit OFFSET :af_offset
"""), af_params).mappings().all()
for r in af_rows:
if not any(p['id'] == r['id'] for p in results['parts']):
results['parts'].append({'id': r['id'], 'oem_part_number': r['oem_part_number'],
'name': r['name'], 'name_es': r['name_es'], 'image_url': r['image_url'],
'group_name': r['group_name'], 'category_name': r['category_name'],
'matched_number': r['matched_number'], 'match_type': 'aftermarket'})
# Cross-reference search
cr_params = {'cr_limit': limit, 'cr_offset': offset}
cr_clauses = []
for i, t in enumerate(terms):
cr_clauses.append(f"pcr.cross_reference_number ILIKE :cr{i}")
cr_params[f'cr{i}'] = f"%{t}%"
cr_rows = session.execute(text(f"""
SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
pg.name_part_group AS group_name, pc.name_part_category AS category_name,
pcr.cross_reference_number AS matched_number, p.image_url
FROM part_cross_references pcr JOIN parts p ON pcr.part_id = p.id_part
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE {' AND '.join(cr_clauses)} LIMIT :cr_limit OFFSET :cr_offset
"""), cr_params).mappings().all()
for r in cr_rows:
if not any(p['id'] == r['id'] for p in results['parts']):
results['parts'].append({'id': r['id'], 'oem_part_number': r['oem_part_number'],
'name': r['name'], 'name_es': r['name_es'], 'image_url': r['image_url'],
'group_name': r['group_name'], 'category_name': r['category_name'],
'matched_number': r['matched_number'], 'match_type': 'cross_reference'})
# Search vehicles
if search_type in ('vehicles', 'all') and terms:
v_params = {'v_limit': limit, 'v_offset': offset}
v_clauses = []
for i, t in enumerate(terms):
tp = f"%{t}%"
v_clauses.append(f"(b.name_brand ILIKE :v{i}_b OR m.name_model ILIKE :v{i}_m OR CAST(y.year_car AS TEXT) ILIKE :v{i}_y OR e.name_engine ILIKE :v{i}_e)")
v_params[f'v{i}_b'] = tp
v_params[f'v{i}_m'] = tp
v_params[f'v{i}_y'] = tp
v_params[f'v{i}_e'] = tp
v_rows = session.execute(text(f"""
SELECT mye.id_mye AS id, b.name_brand AS brand, m.name_model AS model,
y.year_car AS year, e.name_engine AS engine
FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year JOIN engines e ON mye.engine_id = e.id_engine
WHERE {' AND '.join(v_clauses)}
ORDER BY y.year_car DESC, b.name_brand, m.name_model
LIMIT :v_limit OFFSET :v_offset
"""), v_params).mappings().all()
for r in v_rows:
results['vehicles'].append({'id': r['id'], 'brand': r['brand'], 'model': r['model'],
'year': r['year'], 'engine': r['engine']})
results['total_count'] = len(results['parts']) + len(results['vehicles'])
return jsonify(results)
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/search/parts')
def api_search_parts():
session = Session()
try:
q = request.args.get('q', '').strip()
category_id = request.args.get('category_id', type=int)
group_id = request.args.get('group_id', type=int)
page = request.args.get('page', 1, type=int)
per_page = min(request.args.get('per_page', 50, type=int), 100)
offset = (page - 1) * per_page
if not q:
return jsonify({'error': 'Search query is required'}), 400
filter_clause = ""
params = {'q': q, 'per_page': per_page, 'offset': offset}
if category_id:
filter_clause += " AND pg.category_id = :cid"
params['cid'] = category_id
if group_id:
filter_clause += " AND p.group_id = :gid"
params['gid'] = group_id
total_count = session.execute(text(f"""
SELECT COUNT(*) AS total FROM parts p
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE p.search_vector @@ plainto_tsquery('spanish', :q) {filter_clause}
"""), params).mappings().first()['total']
rows = session.execute(text(f"""
SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es,
p.description, pg.name_part_group AS group_name, pc.name_part_category AS category_name,
ts_rank(p.search_vector, plainto_tsquery('spanish', :q)) AS rank
FROM parts p JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE p.search_vector @@ plainto_tsquery('spanish', :q) {filter_clause}
ORDER BY rank DESC LIMIT :per_page OFFSET :offset
"""), params).mappings().all()
parts = [{'id': r['id'], 'oem_part_number': r['oem_part_number'], 'name': r['name'],
'name_es': r['name_es'], 'description': r['description'],
'group_name': r['group_name'], 'category_name': r['category_name'],
'rank': float(r['rank'])} for r in rows]
total_pages = (total_count + per_page - 1) // per_page
return jsonify({'data': parts, 'pagination': {'page': page, 'per_page': per_page,
'total': total_count, 'total_pages': total_pages}})
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/vin/decode/<vin>')
def api_vin_decode(vin):
session = Session()
try:
vin = vin.upper().strip()
if not validate_vin(vin):
return jsonify({'error': 'Invalid VIN format. VIN must be 17 alphanumeric characters (no I, O, Q).'}), 400
cached_row = session.execute(text("""
SELECT vin, make, model, year, engine_info, body_class, drive_type,
model_year_engine_id, created_at, expires_at
FROM vin_cache WHERE vin = :vin AND expires_at > NOW()
"""), {'vin': vin}).mappings().first()
if cached_row:
engine_info_data = {}
if cached_row['engine_info']:
try:
engine_info_data = json_module.loads(cached_row['engine_info'])
except Exception:
engine_info_data = {'raw': cached_row['engine_info']}
cached_data = {'vin': cached_row['vin'], 'make': cached_row['make'],
'model': cached_row['model'], 'year': cached_row['year'],
'engine_info': engine_info_data, 'body_class': cached_row['body_class'],
'drive_type': cached_row['drive_type'], 'matched_vehicle': None, 'cached': True}
if cached_row['model_year_engine_id']:
mye_row = session.execute(text("""
SELECT mye.id_mye AS id, b.name_brand AS brand, m.name_model AS model,
y.year_car AS year, e.name_engine AS engine
FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year JOIN engines e ON mye.engine_id = e.id_engine
WHERE mye.id_mye = :mye_id
"""), {'mye_id': cached_row['model_year_engine_id']}).mappings().first()
if mye_row:
cached_data['matched_vehicle'] = {'mye_id': mye_row['id'], 'brand': mye_row['brand'],
'model': mye_row['model'], 'year': mye_row['year'], 'engine': mye_row['engine']}
return jsonify(cached_data)
# Call NHTSA API
nhtsa_url = f'https://vpic.nhtsa.dot.gov/api/vehicles/DecodeVin/{vin}?format=json'
try:
req = urllib.request.Request(nhtsa_url, headers={'User-Agent': 'NexusAutoparts/2.0'})
with urllib.request.urlopen(req, timeout=10) as response:
nhtsa_data = json_module.loads(response.read().decode('utf-8'))
except urllib.error.URLError as e:
return jsonify({'error': f'Failed to connect to NHTSA API: {str(e)}'}), 503
except urllib.error.HTTPError as e:
return jsonify({'error': f'NHTSA API error: {e.code}'}), 502
except Exception as e:
return jsonify({'error': f'Error calling NHTSA API: {str(e)}'}), 500
nhtsa_results = {item['Variable']: item['Value'] for item in nhtsa_data.get('Results', [])}
make = nhtsa_results.get('Make', '')
model = nhtsa_results.get('Model', '')
year_str = nhtsa_results.get('ModelYear', '')
year = int(year_str) if year_str and year_str.isdigit() else None
engine_config = nhtsa_results.get('EngineConfiguration', '')
cylinders_str = nhtsa_results.get('EngineCylinders', '')
cylinders = int(cylinders_str) if cylinders_str and cylinders_str.isdigit() else None
displacement_str = nhtsa_results.get('DisplacementL', '')
displacement_l = float(displacement_str) if displacement_str else None
fuel_type = nhtsa_results.get('FuelTypePrimary', '')
body_class = nhtsa_results.get('BodyClass', '')
drive_type = nhtsa_results.get('DriveType', '')
matched_mye_id = None
matched_vehicle = None
if make and model and year:
mye_row = session.execute(text("""
SELECT mye.id_mye AS id, b.name_brand AS brand, m.name_model AS model,
y.year_car AS year, e.name_engine AS engine
FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year JOIN engines e ON mye.engine_id = e.id_engine
WHERE b.name_brand ILIKE :make AND m.name_model ILIKE :model AND y.year_car = :year LIMIT 1
"""), {'make': make, 'model': model, 'year': year}).mappings().first()
if mye_row:
matched_mye_id = mye_row['id']
matched_vehicle = {'mye_id': mye_row['id'], 'brand': mye_row['brand'],
'model': mye_row['model'], 'year': mye_row['year'], 'engine': mye_row['engine']}
expires_at = datetime.now() + timedelta(days=30)
engine_info = json_module.dumps({'configuration': engine_config, 'cylinders': cylinders,
'displacement_l': displacement_l, 'fuel_type': fuel_type})
session.execute(text("""
INSERT INTO vin_cache (vin, decoded_data, make, model, year, engine_info,
body_class, drive_type, model_year_engine_id, expires_at)
VALUES (:vin, :decoded_data::jsonb, :make, :model, :year, :engine_info,
:body_class, :drive_type, :mye_id, :expires_at)
ON CONFLICT (vin) DO UPDATE SET
decoded_data = EXCLUDED.decoded_data, make = EXCLUDED.make, model = EXCLUDED.model,
year = EXCLUDED.year, engine_info = EXCLUDED.engine_info, body_class = EXCLUDED.body_class,
drive_type = EXCLUDED.drive_type, model_year_engine_id = EXCLUDED.model_year_engine_id,
expires_at = EXCLUDED.expires_at
"""), {'vin': vin, 'decoded_data': json_module.dumps(nhtsa_results), 'make': make,
'model': model, 'year': year, 'engine_info': engine_info, 'body_class': body_class,
'drive_type': drive_type, 'mye_id': matched_mye_id, 'expires_at': expires_at})
session.commit()
return jsonify({'vin': vin, 'make': make, 'model': model, 'year': year,
'engine_info': {'configuration': engine_config, 'cylinders': cylinders,
'displacement_l': displacement_l, 'fuel_type': fuel_type},
'body_class': body_class, 'drive_type': drive_type,
'matched_vehicle': matched_vehicle, 'cached': False})
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/vin/<vin>/parts')
def api_vin_parts(vin):
session = Session()
try:
vin = vin.upper().strip()
if not validate_vin(vin):
return jsonify({'error': 'Invalid VIN format. VIN must be 17 alphanumeric characters (no I, O, Q).'}), 400
category_id = request.args.get('category_id', type=int)
cached_row = session.execute(text("SELECT vin, make, model, year, model_year_engine_id FROM vin_cache WHERE vin = :vin"), {'vin': vin}).mappings().first()
if not cached_row:
return jsonify({'error': 'VIN not found in cache. Please decode the VIN first using /api/vin/decode/<vin>'}), 404
mye_id = cached_row['model_year_engine_id']
vehicle_info = {'vin': cached_row['vin'], 'make': cached_row['make'], 'model': cached_row['model'],
'year': cached_row['year'], 'mye_id': mye_id}
if not mye_id:
return jsonify({'vin': vin, 'vehicle_info': vehicle_info, 'categories': [],
'message': 'No matching vehicle configuration found in database. Use /api/vin/<vin>/match to manually link.'})
params = {'mye_id': mye_id}
cat_filter = ""
if category_id:
cat_filter = " AND pc.id_part_category = :cid"
params['cid'] = category_id
rows = session.execute(text(f"""
SELECT pc.id_part_category AS category_id, pc.name_part_category AS category_name,
pc.name_es AS category_name_es, p.id_part AS part_id, p.oem_part_number,
p.name_part AS part_name, p.name_es AS part_name_es,
pg.name_part_group AS group_name, vp.quantity_required,
pp.name_position_part AS position
FROM vehicle_parts vp JOIN parts p ON vp.part_id = p.id_part
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
LEFT JOIN position_part pp ON vp.id_position_part = pp.id_position_part
WHERE vp.model_year_engine_id = :mye_id {cat_filter}
ORDER BY pc.display_order, pg.display_order, p.name_part
"""), params).mappings().all()
categories_dict = {}
for r in rows:
cid = r['category_id']
if cid not in categories_dict:
categories_dict[cid] = {'id': cid, 'name': r['category_name'], 'name_es': r['category_name_es'], 'parts': []}
categories_dict[cid]['parts'].append({'id': r['part_id'], 'oem_part_number': r['oem_part_number'],
'name': r['part_name'], 'name_es': r['part_name_es'], 'group_name': r['group_name'],
'quantity_required': r['quantity_required'], 'position': r['position']})
return jsonify({'vin': vin, 'vehicle_info': vehicle_info, 'categories': list(categories_dict.values())})
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/vin/<vin>/match')
def api_vin_match(vin):
session = Session()
try:
vin = vin.upper().strip()
if not validate_vin(vin):
return jsonify({'error': 'Invalid VIN format. VIN must be 17 alphanumeric characters (no I, O, Q).'}), 400
mye_id = request.args.get('mye_id', type=int)
if not mye_id:
return jsonify({'error': 'mye_id parameter is required'}), 400
mye_row = session.execute(text("SELECT id_mye FROM model_year_engine WHERE id_mye = :mid"), {'mid': mye_id}).mappings().first()
if not mye_row:
return jsonify({'error': f'model_year_engine_id {mye_id} not found'}), 404
vin_row = session.execute(text("SELECT vin FROM vin_cache WHERE vin = :vin"), {'vin': vin}).mappings().first()
if vin_row:
session.execute(text("UPDATE vin_cache SET model_year_engine_id = :mid WHERE vin = :vin"), {'mid': mye_id, 'vin': vin})
else:
expires_at = datetime.now() + timedelta(days=30)
session.execute(text("""
INSERT INTO vin_cache (vin, decoded_data, model_year_engine_id, created_at, expires_at)
VALUES (:vin, '{}'::jsonb, :mid, NOW(), :expires_at)
"""), {'vin': vin, 'mid': mye_id, 'expires_at': expires_at})
session.commit()
return jsonify({'success': True, 'vin': vin, 'mye_id': mye_id})
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ============================================================================
# Admin Endpoints
# ============================================================================
@app.route('/api/admin/stats')
def api_admin_stats():
session = Session()
try:
stats = {}
# Small tables: exact count
for table, key in [('part_categories', 'categories'), ('part_groups', 'groups'),
('manufacturers', 'manufacturers')]:
stats[key] = session.execute(text(f"SELECT COUNT(*) FROM {table}")).scalar()
# Large tables: use pg estimate for speed
for table, key in [('parts', 'parts'), ('aftermarket_parts', 'aftermarket'),
('vehicle_parts', 'fitment')]:
est = session.execute(text(
"SELECT reltuples::bigint FROM pg_class WHERE relname = :t"
), {'t': table}).scalar()
if est and est > 0:
stats[key] = est
else:
stats[key] = session.execute(text(f"SELECT COUNT(*) FROM {table}")).scalar()
return jsonify(stats)
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ---- Categories CRUD ----
@app.route('/api/admin/categories', methods=['POST'])
def api_admin_create_category():
session = Session()
try:
data = request.get_json()
result = session.execute(text("""
INSERT INTO part_categories (name_part_category, name_es, slug, icon_name, display_order, parent_id)
VALUES (:name, :name_es, :slug, :icon_name, :display_order, :parent_id)
RETURNING id_part_category
"""), {'name': data['name'], 'name_es': data.get('name_es'),
'slug': data.get('slug') or data['name'].lower().replace(' ', '-'),
'icon_name': data.get('icon_name'), 'display_order': data.get('display_order', 0),
'parent_id': data.get('parent_id')})
new_id = result.scalar()
session.commit()
return jsonify({'id': new_id, 'message': 'Category created'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/categories/<int:category_id>', methods=['PUT'])
def api_admin_update_category(category_id):
session = Session()
try:
data = request.get_json()
session.execute(text("""
UPDATE part_categories SET name_part_category = :name, name_es = :name_es,
slug = :slug, icon_name = :icon_name, display_order = :display_order
WHERE id_part_category = :id
"""), {'name': data['name'], 'name_es': data.get('name_es'), 'slug': data.get('slug'),
'icon_name': data.get('icon_name'), 'display_order': data.get('display_order', 0),
'id': category_id})
session.commit()
return jsonify({'message': 'Category updated'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/categories/<int:category_id>', methods=['DELETE'])
def api_admin_delete_category(category_id):
session = Session()
try:
session.execute(text("DELETE FROM part_categories WHERE id_part_category = :id"), {'id': category_id})
session.commit()
return jsonify({'message': 'Category deleted'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ---- Groups CRUD ----
@app.route('/api/admin/groups')
def api_admin_list_groups():
session = Session()
try:
category_id = request.args.get('category_id', type=int)
q = """
SELECT pg.id_part_group AS id, pg.name_part_group AS name, pg.name_es,
pg.category_id, pg.display_order, pg.slug,
pc.name_part_category AS category_name
FROM part_groups pg
LEFT JOIN part_categories pc ON pg.category_id = pc.id_part_category WHERE 1=1
"""
params = {}
if category_id:
q += " AND pg.category_id = :cid"
params['cid'] = category_id
q += " ORDER BY pg.display_order, pg.name_part_group"
rows = session.execute(text(q), params).mappings().all()
return jsonify([{'id': r['id'], 'name': r['name'], 'name_es': r['name_es'],
'category_id': r['category_id'], 'category_name': r['category_name'],
'display_order': r['display_order'], 'slug': r['slug']} for r in rows])
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/groups', methods=['POST'])
def api_admin_create_group():
session = Session()
try:
data = request.get_json()
result = session.execute(text("""
INSERT INTO part_groups (category_id, name_part_group, name_es, slug, display_order)
VALUES (:category_id, :name, :name_es, :slug, :display_order)
RETURNING id_part_group
"""), {'category_id': data['category_id'], 'name': data['name'],
'name_es': data.get('name_es'),
'slug': data.get('slug') or data['name'].lower().replace(' ', '-'),
'display_order': data.get('display_order', 0)})
new_id = result.scalar()
session.commit()
return jsonify({'id': new_id, 'message': 'Group created'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/groups/<int:group_id>', methods=['PUT'])
def api_admin_update_group(group_id):
session = Session()
try:
data = request.get_json()
session.execute(text("""
UPDATE part_groups SET category_id = :category_id, name_part_group = :name,
name_es = :name_es, display_order = :display_order
WHERE id_part_group = :id
"""), {'category_id': data['category_id'], 'name': data['name'],
'name_es': data.get('name_es'), 'display_order': data.get('display_order', 0),
'id': group_id})
session.commit()
return jsonify({'message': 'Group updated'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/groups/<int:group_id>', methods=['DELETE'])
def api_admin_delete_group(group_id):
session = Session()
try:
session.execute(text("DELETE FROM part_groups WHERE id_part_group = :id"), {'id': group_id})
session.commit()
return jsonify({'message': 'Group deleted'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ---- Parts CRUD ----
@app.route('/api/admin/parts', methods=['POST'])
def api_admin_create_part():
session = Session()
try:
data = request.get_json()
result = session.execute(text("""
INSERT INTO parts (oem_part_number, name_part, name_es, group_id, description,
description_es, weight_kg, id_material)
VALUES (:oem, :name, :name_es, :group_id, :desc, :desc_es, :weight,
(SELECT id_material FROM materials WHERE name_material = :material))
RETURNING id_part
"""), {'oem': data['oem_part_number'], 'name': data['name'], 'name_es': data.get('name_es'),
'group_id': data['group_id'], 'desc': data.get('description'),
'desc_es': data.get('description_es'), 'weight': data.get('weight_kg'),
'material': data.get('material')})
new_id = result.scalar()
session.commit()
return jsonify({'id': new_id, 'message': 'Part created'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/parts/<int:part_id>', methods=['PUT'])
def api_admin_update_part(part_id):
session = Session()
try:
data = request.get_json()
session.execute(text("""
UPDATE parts SET oem_part_number = :oem, name_part = :name, name_es = :name_es,
group_id = :group_id, description = :desc, description_es = :desc_es,
weight_kg = :weight,
id_material = (SELECT id_material FROM materials WHERE name_material = :material)
WHERE id_part = :id
"""), {'oem': data['oem_part_number'], 'name': data['name'], 'name_es': data.get('name_es'),
'group_id': data['group_id'], 'desc': data.get('description'),
'desc_es': data.get('description_es'), 'weight': data.get('weight_kg'),
'material': data.get('material'), 'id': part_id})
session.commit()
return jsonify({'message': 'Part updated'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/parts/<int:part_id>', methods=['DELETE'])
def api_admin_delete_part(part_id):
session = Session()
try:
session.execute(text("DELETE FROM parts WHERE id_part = :id"), {'id': part_id})
session.commit()
return jsonify({'message': 'Part deleted'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ---- Manufacturers CRUD ----
@app.route('/api/admin/manufacturers', methods=['POST'])
def api_admin_create_manufacturer():
session = Session()
try:
data = request.get_json()
result = session.execute(text("""
INSERT INTO manufacturers (name_manufacture, id_type_manu, id_quality_tier, id_country, website)
VALUES (:name,
(SELECT id_type_manu FROM manufacture_type WHERE name_type_manu = :type),
(SELECT id_quality_tier FROM quality_tier WHERE name_quality = :qt),
(SELECT id_country FROM countries WHERE name_country = :country),
:website)
RETURNING id_manufacture
"""), {'name': data['name'], 'type': data.get('type', 'aftermarket'),
'qt': data.get('quality_tier', 'standard'), 'country': data.get('country'),
'website': data.get('website')})
new_id = result.scalar()
session.commit()
return jsonify({'id': new_id, 'message': 'Manufacturer created'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/manufacturers/<int:manufacturer_id>', methods=['PUT'])
def api_admin_update_manufacturer(manufacturer_id):
session = Session()
try:
data = request.get_json()
session.execute(text("""
UPDATE manufacturers SET name_manufacture = :name,
id_type_manu = (SELECT id_type_manu FROM manufacture_type WHERE name_type_manu = :type),
id_quality_tier = (SELECT id_quality_tier FROM quality_tier WHERE name_quality = :qt),
id_country = (SELECT id_country FROM countries WHERE name_country = :country),
website = :website
WHERE id_manufacture = :id
"""), {'name': data['name'], 'type': data.get('type'), 'qt': data.get('quality_tier'),
'country': data.get('country'), 'website': data.get('website'), 'id': manufacturer_id})
session.commit()
return jsonify({'message': 'Manufacturer updated'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/manufacturers/<int:manufacturer_id>', methods=['DELETE'])
def api_admin_delete_manufacturer(manufacturer_id):
session = Session()
try:
session.execute(text("DELETE FROM manufacturers WHERE id_manufacture = :id"), {'id': manufacturer_id})
session.commit()
return jsonify({'message': 'Manufacturer deleted'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ---- Aftermarket Parts CRUD ----
@app.route('/api/admin/aftermarket', methods=['POST'])
def api_admin_create_aftermarket():
session = Session()
try:
data = request.get_json()
result = session.execute(text("""
INSERT INTO aftermarket_parts (oem_part_id, manufacturer_id, part_number,
name_aftermarket_parts, name_es, id_quality_tier, price_usd, warranty_months)
VALUES (:oem_part_id, :manufacturer_id, :part_number, :name, :name_es,
(SELECT id_quality_tier FROM quality_tier WHERE name_quality = :qt),
:price_usd, :warranty_months)
RETURNING id_aftermarket_parts
"""), {'oem_part_id': data['oem_part_id'], 'manufacturer_id': data['manufacturer_id'],
'part_number': data['part_number'], 'name': data.get('name'), 'name_es': data.get('name_es'),
'qt': data.get('quality_tier', 'standard'), 'price_usd': data.get('price_usd'),
'warranty_months': data.get('warranty_months')})
new_id = result.scalar()
session.commit()
return jsonify({'id': new_id, 'message': 'Aftermarket part created'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/aftermarket/<int:aftermarket_id>', methods=['PUT'])
def api_admin_update_aftermarket(aftermarket_id):
session = Session()
try:
data = request.get_json()
session.execute(text("""
UPDATE aftermarket_parts SET oem_part_id = :oem_part_id, manufacturer_id = :manufacturer_id,
part_number = :part_number, name_aftermarket_parts = :name, name_es = :name_es,
id_quality_tier = (SELECT id_quality_tier FROM quality_tier WHERE name_quality = :qt),
price_usd = :price_usd, warranty_months = :warranty_months
WHERE id_aftermarket_parts = :id
"""), {'oem_part_id': data['oem_part_id'], 'manufacturer_id': data['manufacturer_id'],
'part_number': data['part_number'], 'name': data.get('name'), 'name_es': data.get('name_es'),
'qt': data.get('quality_tier'), 'price_usd': data.get('price_usd'),
'warranty_months': data.get('warranty_months'), 'id': aftermarket_id})
session.commit()
return jsonify({'message': 'Aftermarket part updated'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/aftermarket/<int:aftermarket_id>', methods=['DELETE'])
def api_admin_delete_aftermarket(aftermarket_id):
session = Session()
try:
session.execute(text("DELETE FROM aftermarket_parts WHERE id_aftermarket_parts = :id"), {'id': aftermarket_id})
session.commit()
return jsonify({'message': 'Aftermarket part deleted'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ---- Cross-References CRUD ----
@app.route('/api/admin/crossref')
def api_admin_list_crossref():
session = Session()
try:
page = request.args.get('page', 1, type=int)
per_page = min(request.args.get('per_page', 50, type=int), 100)
offset = (page - 1) * per_page
total_count = session.execute(text("SELECT COUNT(*) FROM part_cross_references")).scalar()
rows = session.execute(text("""
SELECT pcr.id_part_cross_ref AS id, pcr.part_id, pcr.cross_reference_number,
rt.name_ref_type AS reference_type, pcr.source_ref AS source, pcr.notes,
p.oem_part_number, p.name_part AS part_name
FROM part_cross_references pcr
JOIN parts p ON pcr.part_id = p.id_part
LEFT JOIN reference_type rt ON pcr.id_ref_type = rt.id_ref_type
ORDER BY pcr.id_part_cross_ref DESC LIMIT :limit OFFSET :offset
"""), {'limit': per_page, 'offset': offset}).mappings().all()
refs = [{'id': r['id'], 'part_id': r['part_id'], 'cross_reference_number': r['cross_reference_number'],
'reference_type': r['reference_type'], 'source': r['source'], 'notes': r['notes'],
'oem_part_number': r['oem_part_number'], 'part_name': r['part_name']} for r in rows]
total_pages = (total_count + per_page - 1) // per_page
return jsonify({'data': refs, 'pagination': {'page': page, 'per_page': per_page,
'total': total_count, 'total_pages': total_pages}})
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/crossref', methods=['POST'])
def api_admin_create_crossref():
session = Session()
try:
data = request.get_json()
result = session.execute(text("""
INSERT INTO part_cross_references (part_id, cross_reference_number, id_ref_type, source_ref, notes)
VALUES (:part_id, :cross_ref,
(SELECT id_ref_type FROM reference_type WHERE name_ref_type = :ref_type),
:source, :notes)
RETURNING id_part_cross_ref
"""), {'part_id': data['part_id'], 'cross_ref': data['cross_reference_number'],
'ref_type': data['reference_type'], 'source': data.get('source'), 'notes': data.get('notes')})
new_id = result.scalar()
session.commit()
return jsonify({'id': new_id, 'message': 'Cross-reference created'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/crossref/<int:crossref_id>', methods=['PUT'])
def api_admin_update_crossref(crossref_id):
session = Session()
try:
data = request.get_json()
session.execute(text("""
UPDATE part_cross_references SET part_id = :part_id, cross_reference_number = :cross_ref,
id_ref_type = (SELECT id_ref_type FROM reference_type WHERE name_ref_type = :ref_type),
source_ref = :source, notes = :notes
WHERE id_part_cross_ref = :id
"""), {'part_id': data['part_id'], 'cross_ref': data['cross_reference_number'],
'ref_type': data['reference_type'], 'source': data.get('source'),
'notes': data.get('notes'), 'id': crossref_id})
session.commit()
return jsonify({'message': 'Cross-reference updated'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/crossref/<int:crossref_id>', methods=['DELETE'])
def api_admin_delete_crossref(crossref_id):
session = Session()
try:
session.execute(text("DELETE FROM part_cross_references WHERE id_part_cross_ref = :id"), {'id': crossref_id})
session.commit()
return jsonify({'message': 'Cross-reference deleted'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ---- Fitment CRUD ----
@app.route('/api/admin/fitment')
def api_admin_list_fitment():
session = Session()
try:
page = request.args.get('page', 1, type=int)
per_page = min(request.args.get('per_page', 50, type=int), 500)
offset = (page - 1) * per_page
brand = request.args.get('brand')
model = request.args.get('model')
mye_id = request.args.get('mye_id', type=int)
where = " WHERE 1=1"
params = {'limit': per_page, 'offset': offset}
if mye_id:
where += " AND vp.model_year_engine_id = :mye_id"
params['mye_id'] = mye_id
if brand:
where += " AND b.name_brand ILIKE :brand"
params['brand'] = brand
if model:
where += " AND m.name_model ILIKE :model"
params['model'] = model
base = """
FROM vehicle_parts vp
JOIN model_year_engine mye ON vp.model_year_engine_id = mye.id_mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
"""
total_count = session.execute(text("SELECT COUNT(*) " + base + where), params).scalar()
rows = session.execute(text("""
SELECT vp.id_vehicle_part AS id, vp.model_year_engine_id, vp.part_id,
vp.quantity_required, pp.name_position_part AS position, vp.fitment_notes,
b.name_brand AS brand, m.name_model AS model, y.year_car AS year,
e.name_engine AS engine, p.oem_part_number, p.name_part AS part_name
FROM vehicle_parts vp
JOIN model_year_engine mye ON vp.model_year_engine_id = mye.id_mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year
JOIN engines e ON mye.engine_id = e.id_engine
JOIN parts p ON vp.part_id = p.id_part
LEFT JOIN position_part pp ON vp.id_position_part = pp.id_position_part
""" + where + " ORDER BY vp.id_vehicle_part DESC LIMIT :limit OFFSET :offset"), params).mappings().all()
fitments = [dict(r) for r in rows]
total_pages = (total_count + per_page - 1) // per_page
return jsonify({'data': fitments, 'pagination': {'page': page, 'per_page': per_page,
'total': total_count, 'total_pages': total_pages}})
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/fitment', methods=['POST'])
def api_admin_create_fitment():
session = Session()
try:
data = request.get_json()
result = session.execute(text("""
INSERT INTO vehicle_parts (model_year_engine_id, part_id, quantity_required,
id_position_part, fitment_notes)
VALUES (:mye_id, :part_id, :qty,
(SELECT id_position_part FROM position_part WHERE name_position_part = :position),
:notes)
RETURNING id_vehicle_part
"""), {'mye_id': data['model_year_engine_id'], 'part_id': data['part_id'],
'qty': data.get('quantity_required', 1), 'position': data.get('position'),
'notes': data.get('fitment_notes')})
new_id = result.scalar()
session.commit()
return jsonify({'id': new_id, 'message': 'Fitment created'})
except IntegrityError:
session.rollback()
return jsonify({'error': 'Este fitment ya existe'}), 400
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/fitment/<int:fitment_id>', methods=['DELETE'])
def api_admin_delete_fitment(fitment_id):
session = Session()
try:
session.execute(text("DELETE FROM vehicle_parts WHERE id_vehicle_part = :id"), {'id': fitment_id})
session.commit()
return jsonify({'message': 'Fitment deleted'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ============================================================================
# CSV Import
# ============================================================================
@app.route('/api/admin/import/<import_type>', methods=['POST'])
def api_admin_import_csv(import_type):
session = Session()
try:
data = request.get_json()
records = data.get('records', [])
if not records:
return jsonify({'error': 'No records to import'}), 400
imported = 0
errors = []
for i, rec in enumerate(records):
try:
if import_type == 'categories':
session.execute(text("INSERT INTO part_categories (name_part_category, name_es, slug, icon_name, display_order) VALUES (:name, :name_es, :slug, :icon_name, :do)"),
{'name': rec['name'], 'name_es': rec.get('name_es'), 'slug': rec.get('slug') or rec['name'].lower().replace(' ', '-'), 'icon_name': rec.get('icon_name'), 'do': rec.get('display_order', 0)})
elif import_type == 'groups':
session.execute(text("INSERT INTO part_groups (category_id, name_part_group, name_es, display_order) VALUES (:cid, :name, :name_es, :do)"),
{'cid': rec['category_id'], 'name': rec['name'], 'name_es': rec.get('name_es'), 'do': rec.get('display_order', 0)})
elif import_type == 'parts':
session.execute(text("""INSERT INTO parts (oem_part_number, name_part, name_es, group_id, description, description_es, weight_kg, id_material)
VALUES (:oem, :name, :name_es, :gid, :desc, :desc_es, :weight, (SELECT id_material FROM materials WHERE name_material = :material))"""),
{'oem': rec['oem_part_number'], 'name': rec['name'], 'name_es': rec.get('name_es'), 'gid': rec['group_id'], 'desc': rec.get('description'), 'desc_es': rec.get('description_es'), 'weight': rec.get('weight_kg'), 'material': rec.get('material')})
elif import_type == 'manufacturers':
session.execute(text("""INSERT INTO manufacturers (name_manufacture, id_type_manu, id_quality_tier, id_country, website)
VALUES (:name, (SELECT id_type_manu FROM manufacture_type WHERE name_type_manu = :type),
(SELECT id_quality_tier FROM quality_tier WHERE name_quality = :qt),
(SELECT id_country FROM countries WHERE name_country = :country), :website)"""),
{'name': rec['name'], 'type': rec.get('type', 'aftermarket'), 'qt': rec.get('quality_tier', 'standard'), 'country': rec.get('country'), 'website': rec.get('website')})
elif import_type == 'aftermarket':
session.execute(text("""INSERT INTO aftermarket_parts (oem_part_id, manufacturer_id, part_number, name_aftermarket_parts, name_es, id_quality_tier, price_usd, warranty_months)
VALUES (:oem_part_id, :mid, :pn, :name, :name_es,
(SELECT id_quality_tier FROM quality_tier WHERE name_quality = :qt), :price, :warranty)"""),
{'oem_part_id': rec['oem_part_id'], 'mid': rec['manufacturer_id'], 'pn': rec['part_number'], 'name': rec.get('name'), 'name_es': rec.get('name_es'), 'qt': rec.get('quality_tier', 'standard'), 'price': rec.get('price_usd'), 'warranty': rec.get('warranty_months')})
elif import_type == 'crossref':
session.execute(text("""INSERT INTO part_cross_references (part_id, cross_reference_number, id_ref_type, source_ref, notes)
VALUES (:pid, :cross_ref, (SELECT id_ref_type FROM reference_type WHERE name_ref_type = :ref_type), :source, :notes)"""),
{'pid': rec['part_id'], 'cross_ref': rec['cross_reference_number'], 'ref_type': rec.get('reference_type'), 'source': rec.get('source'), 'notes': rec.get('notes')})
elif import_type == 'fitment':
session.execute(text("""INSERT INTO vehicle_parts (model_year_engine_id, part_id, quantity_required, id_position_part, fitment_notes)
VALUES (:mye_id, :pid, :qty, (SELECT id_position_part FROM position_part WHERE name_position_part = :position), :notes)
ON CONFLICT (model_year_engine_id, part_id, id_position_part) DO NOTHING"""),
{'mye_id': rec['model_year_engine_id'], 'pid': rec['part_id'], 'qty': rec.get('quantity_required', 1), 'position': rec.get('position'), 'notes': rec.get('fitment_notes')})
else:
return jsonify({'error': f'Unknown import type: {import_type}'}), 400
imported += 1
except Exception as e:
errors.append(f"Row {i + 1}: {str(e)}")
session.commit()
result = {'imported': imported}
if errors:
result['errors'] = errors[:10]
return jsonify(result)
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ============================================================================
# CSV Export
# ============================================================================
@app.route('/api/admin/export/<export_type>')
def api_admin_export_csv(export_type):
session = Session()
try:
page = request.args.get('page', 1, type=int)
per_page = min(request.args.get('per_page', 1000, type=int), 10000)
offset = (page - 1) * per_page
export_queries = {
'categories': ("SELECT id_part_category AS id, name_part_category AS name, name_es, slug, icon_name, display_order FROM part_categories ORDER BY display_order, name_part_category", "part_categories"),
'groups': ("SELECT id_part_group AS id, category_id, name_part_group AS name, name_es, display_order FROM part_groups ORDER BY category_id, display_order, name_part_group", "part_groups"),
'parts': ("SELECT p.id_part AS id, p.oem_part_number, p.name_part AS name, p.name_es, p.group_id, p.description, p.description_es, p.weight_kg, mat.name_material AS material FROM parts p LEFT JOIN materials mat ON p.id_material = mat.id_material ORDER BY p.id_part", "parts"),
'manufacturers': ("SELECT mfr.id_manufacture AS id, mfr.name_manufacture AS name, mt.name_type_manu AS type, qt.name_quality AS quality_tier, co.name_country AS country, mfr.website FROM manufacturers mfr LEFT JOIN manufacture_type mt ON mfr.id_type_manu = mt.id_type_manu LEFT JOIN quality_tier qt ON mfr.id_quality_tier = qt.id_quality_tier LEFT JOIN countries co ON mfr.id_country = co.id_country ORDER BY mfr.name_manufacture", "manufacturers"),
'aftermarket': ("SELECT ap.id_aftermarket_parts AS id, ap.oem_part_id, ap.manufacturer_id, ap.part_number, ap.name_aftermarket_parts AS name, ap.name_es, qt.name_quality AS quality_tier, ap.price_usd, ap.warranty_months FROM aftermarket_parts ap LEFT JOIN quality_tier qt ON ap.id_quality_tier = qt.id_quality_tier ORDER BY ap.id_aftermarket_parts", "aftermarket_parts"),
'crossref': ("SELECT pcr.id_part_cross_ref AS id, pcr.part_id, pcr.cross_reference_number, rt.name_ref_type AS reference_type, pcr.source_ref AS source, pcr.notes FROM part_cross_references pcr LEFT JOIN reference_type rt ON pcr.id_ref_type = rt.id_ref_type ORDER BY pcr.id_part_cross_ref", "part_cross_references"),
'fitment': ("SELECT vp.id_vehicle_part AS id, vp.model_year_engine_id, vp.part_id, vp.quantity_required, pp.name_position_part AS position, vp.fitment_notes FROM vehicle_parts vp LEFT JOIN position_part pp ON vp.id_position_part = pp.id_position_part ORDER BY vp.id_vehicle_part", "vehicle_parts"),
}
if export_type not in export_queries:
return jsonify({'error': f'Unknown export type: {export_type}'}), 400
base_query, table_name = export_queries[export_type]
total_count = session.execute(text(f"SELECT COUNT(*) FROM {table_name}")).scalar()
rows = session.execute(text(base_query + " LIMIT :limit OFFSET :offset"), {'limit': per_page, 'offset': offset}).mappings().all()
data_list = [dict(r) for r in rows]
total_pages = (total_count + per_page - 1) // per_page
return jsonify({'data': data_list, 'pagination': {'page': page, 'per_page': per_page,
'total': total_count, 'total_pages': total_pages}})
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ============================================================================
# Image Upload
# ============================================================================
@app.route('/api/admin/upload-image', methods=['POST'])
def api_admin_upload_image():
try:
data = request.get_json()
image_data = data.get('image')
if not image_data:
return jsonify({'error': 'No image data provided'}), 400
if ',' in image_data:
header, encoded = image_data.split(',', 1)
ext = 'png'
if 'jpeg' in header or 'jpg' in header:
ext = 'jpg'
elif 'gif' in header:
ext = 'gif'
elif 'webp' in header:
ext = 'webp'
else:
encoded = image_data
ext = 'png'
image_bytes = base64.b64decode(encoded)
filename = f"{uuid.uuid4().hex}.{ext}"
filepath = os.path.join('static', 'parts_images', filename)
os.makedirs(os.path.join('.', 'static', 'parts_images'), exist_ok=True)
with open(filepath, 'wb') as f:
f.write(image_bytes)
return jsonify({'url': f"/static/parts_images/{filename}", 'filename': filename})
except Exception as e:
return jsonify({'error': str(e)}), 500
# ============================================================================
# Diagrams by Category
# ============================================================================
@app.route('/api/vehicles/<int:mye_id>/diagrams/by-category')
def api_vehicle_diagrams_by_category(mye_id):
session = Session()
try:
category_id = request.args.get('category_id', type=int)
params = {'mye_id': mye_id}
cat_filter = ""
if category_id:
cat_filter = " AND pc.id_part_category = :cid"
params['cid'] = category_id
rows = session.execute(text(f"""
SELECT DISTINCT d.id_diagram AS id, d.name_diagram AS name, d.name_es,
d.group_id, d.image_path, d.thumbnail_path,
pg.name_part_group AS group_name, pg.name_es AS group_name_es,
pc.id_part_category AS category_id, pc.name_part_category AS category_name,
pc.name_es AS category_name_es, vd.notes
FROM vehicle_diagrams vd
JOIN diagrams d ON vd.diagram_id = d.id_diagram
JOIN part_groups pg ON d.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE vd.model_year_engine_id = :mye_id {cat_filter}
ORDER BY pc.display_order, pg.display_order, d.display_order, d.name_diagram
"""), params).mappings().all()
categories = {}
for r in rows:
cid = r['category_id']
if cid not in categories:
categories[cid] = {'category_id': cid, 'category_name': r['category_name'],
'category_name_es': r['category_name_es'], 'diagrams': []}
ip = r['image_path'] or ''
iu = '/' + ip if ip and not ip.startswith('/') else ip
categories[cid]['diagrams'].append({'id': r['id'], 'name': r['name'], 'name_es': r['name_es'],
'group_id': r['group_id'], 'group_name': r['group_name'],
'group_name_es': r['group_name_es'], 'image_url': iu,
'thumbnail_path': r['thumbnail_path'], 'notes': r['notes']})
return jsonify(list(categories.values()))
except Exception as e:
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ============================================================================
# Hotspot CRUD
# ============================================================================
@app.route('/api/admin/hotspots', methods=['POST'])
def api_admin_create_hotspot():
session = Session()
try:
data = request.get_json()
if not data:
return jsonify({'error': 'No data provided'}), 400
diagram_id = data.get('diagram_id')
coords = data.get('coords', '')
if not diagram_id or not coords:
return jsonify({'error': 'diagram_id and coords are required'}), 400
result = session.execute(text("""
INSERT INTO diagram_hotspots (diagram_id, part_id, callout_number, id_shape, coords)
VALUES (:did, :pid, :callout,
(SELECT id_shape FROM shapes WHERE name_shape = :shape), :coords)
RETURNING id_dgr_hotspot
"""), {'did': diagram_id, 'pid': data.get('part_id'),
'callout': data.get('callout_number'), 'shape': data.get('shape', 'circle'),
'coords': coords})
hotspot_id = result.scalar()
session.commit()
return jsonify({'id': hotspot_id, 'message': 'Hotspot created'}), 201
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/hotspots/<int:hotspot_id>', methods=['PUT'])
def api_admin_update_hotspot(hotspot_id):
session = Session()
try:
data = request.get_json()
if not data:
return jsonify({'error': 'No data provided'}), 400
exists = session.execute(text("SELECT id_dgr_hotspot FROM diagram_hotspots WHERE id_dgr_hotspot = :id"), {'id': hotspot_id}).mappings().first()
if not exists:
return jsonify({'error': 'Hotspot not found'}), 404
fields = []
params = {'id': hotspot_id}
if 'part_id' in data:
fields.append("part_id = :part_id")
params['part_id'] = data['part_id']
if 'callout_number' in data:
fields.append("callout_number = :callout_number")
params['callout_number'] = data['callout_number']
if 'shape' in data:
fields.append("id_shape = (SELECT id_shape FROM shapes WHERE name_shape = :shape)")
params['shape'] = data['shape']
if 'coords' in data:
fields.append("coords = :coords")
params['coords'] = data['coords']
if not fields:
return jsonify({'error': 'No fields to update'}), 400
session.execute(text(f"UPDATE diagram_hotspots SET {', '.join(fields)} WHERE id_dgr_hotspot = :id"), params)
session.commit()
return jsonify({'message': 'Hotspot updated'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/admin/hotspots/<int:hotspot_id>', methods=['DELETE'])
def api_admin_delete_hotspot(hotspot_id):
session = Session()
try:
exists = session.execute(text("SELECT id_dgr_hotspot FROM diagram_hotspots WHERE id_dgr_hotspot = :id"), {'id': hotspot_id}).mappings().first()
if not exists:
return jsonify({'error': 'Hotspot not found'}), 404
session.execute(text("DELETE FROM diagram_hotspots WHERE id_dgr_hotspot = :id"), {'id': hotspot_id})
session.commit()
return jsonify({'message': 'Hotspot deleted'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ============================================================================
# Captura (Data Entry) Endpoints
# ============================================================================
@app.route('/captura')
def captura_page():
return send_from_directory('.', 'captura.html')
@app.route('/captura.js')
def captura_js():
return send_from_directory('.', 'captura.js')
@app.route('/captura.css')
def captura_css():
return send_from_directory('.', 'captura.css')
@app.route('/api/captura/vehicles/pending')
def api_captura_vehicles_pending():
session = Session()
try:
brand = request.args.get('brand', '')
model = request.args.get('model', '')
page = int(request.args.get('page', 1))
per_page = min(int(request.args.get('per_page', 50)), 100)
offset = (page - 1) * per_page
filters = ["mye.captura_status = 'pending'"]
params = {'limit': per_page, 'offset': offset}
if brand:
filters.append("b.name_brand = :brand")
params['brand'] = brand
if model:
filters.append("m.name_model ILIKE :model")
params['model'] = f'%{model}%'
where = ' AND '.join(filters)
total = session.execute(text(f"""
SELECT COUNT(*) FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
WHERE {where}
"""), params).scalar()
rows = session.execute(text(f"""
SELECT mye.id_mye, b.name_brand AS brand, m.name_model AS model,
y.year_car AS year, e.name_engine AS engine,
mye.trim_level, mye.captura_status
FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year
JOIN engines e ON mye.engine_id = e.id_engine
WHERE {where}
ORDER BY b.name_brand, m.name_model, y.year_car, e.name_engine
LIMIT :limit OFFSET :offset
"""), params).mappings().all()
return jsonify({'data': [dict(r) for r in rows], 'pagination': {
'page': page, 'per_page': per_page, 'total': total,
'total_pages': (total + per_page - 1) // per_page
}})
finally:
session.close()
@app.route('/api/captura/vehicles/in-progress')
def api_captura_vehicles_in_progress():
session = Session()
try:
page = int(request.args.get('page', 1))
per_page = min(int(request.args.get('per_page', 50)), 100)
offset = (page - 1) * per_page
total = session.execute(text(
"SELECT COUNT(*) FROM model_year_engine WHERE captura_status = 'in_progress'"
)).scalar()
rows = session.execute(text("""
SELECT mye.id_mye, b.name_brand AS brand, m.name_model AS model,
y.year_car AS year, e.name_engine AS engine,
mye.trim_level, mye.captura_status,
COUNT(vp.id_vehicle_part) AS parts_count
FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year
JOIN engines e ON mye.engine_id = e.id_engine
LEFT JOIN vehicle_parts vp ON vp.model_year_engine_id = mye.id_mye
WHERE mye.captura_status = 'in_progress'
GROUP BY mye.id_mye, b.name_brand, m.name_model, y.year_car,
e.name_engine, mye.trim_level, mye.captura_status
ORDER BY b.name_brand, m.name_model, y.year_car
LIMIT :limit OFFSET :offset
"""), {'limit': per_page, 'offset': offset}).mappings().all()
return jsonify({'data': [dict(r) for r in rows], 'pagination': {
'page': page, 'per_page': per_page, 'total': total,
'total_pages': (total + per_page - 1) // per_page
}})
finally:
session.close()
@app.route('/api/captura/vehicles/<int:mye_id>/status', methods=['PUT'])
def api_captura_vehicle_status(mye_id):
session = Session()
try:
data = request.get_json()
status = data.get('status')
if status not in ('pending', 'in_progress', 'completed'):
return jsonify({'error': 'Invalid status'}), 400
session.execute(text(
"UPDATE model_year_engine SET captura_status = :status WHERE id_mye = :id"
), {'status': status, 'id': mye_id})
session.commit()
return jsonify({'message': f'Status updated to {status}'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/captura/vehicles/<int:mye_id>/parts')
def api_captura_vehicle_parts(mye_id):
session = Session()
try:
vehicle = session.execute(text("""
SELECT mye.id_mye, b.name_brand AS brand, m.name_model AS model,
y.year_car AS year, e.name_engine AS engine, mye.trim_level
FROM model_year_engine mye
JOIN models m ON mye.model_id = m.id_model
JOIN brands b ON m.brand_id = b.id_brand
JOIN years y ON mye.year_id = y.id_year
JOIN engines e ON mye.engine_id = e.id_engine
WHERE mye.id_mye = :id
"""), {'id': mye_id}).mappings().first()
if not vehicle:
return jsonify({'error': 'Vehicle not found'}), 404
groups = session.execute(text("""
SELECT pc.id_part_category, pc.name_part_category AS category,
pg.id_part_group, pg.name_part_group AS group_name,
pc.display_order AS cat_order, pg.display_order AS grp_order
FROM part_categories pc
JOIN part_groups pg ON pg.category_id = pc.id_part_category
ORDER BY pc.display_order, pg.display_order
""")).mappings().all()
existing = session.execute(text("""
SELECT vp.id_vehicle_part, vp.part_id, p.oem_part_number, p.name_part,
p.name_es, p.group_id, vp.quantity_required,
pp.name_position_part AS position
FROM vehicle_parts vp
JOIN parts p ON vp.part_id = p.id_part
LEFT JOIN position_part pp ON vp.id_position_part = pp.id_position_part
WHERE vp.model_year_engine_id = :id
ORDER BY p.group_id, p.oem_part_number
"""), {'id': mye_id}).mappings().all()
return jsonify({
'vehicle': dict(vehicle),
'groups': [dict(g) for g in groups],
'parts': [dict(e) for e in existing]
})
finally:
session.close()
@app.route('/api/captura/parts/without-aftermarket')
def api_captura_parts_without_aftermarket():
session = Session()
try:
search = request.args.get('search', '')
group_id = request.args.get('group_id', '')
page = int(request.args.get('page', 1))
per_page = min(int(request.args.get('per_page', 50)), 100)
offset = (page - 1) * per_page
filters = ["NOT EXISTS (SELECT 1 FROM aftermarket_parts ap WHERE ap.oem_part_id = p.id_part)"]
params = {'limit': per_page, 'offset': offset}
if search:
filters.append("(p.oem_part_number ILIKE :search OR p.name_part ILIKE :search)")
params['search'] = f'%{search}%'
if group_id:
filters.append("p.group_id = :group_id")
params['group_id'] = int(group_id)
where = ' AND '.join(filters)
total = session.execute(text(f"SELECT COUNT(*) FROM parts p WHERE {where}"), params).scalar()
rows = session.execute(text(f"""
SELECT p.id_part, p.oem_part_number, p.name_part, p.name_es,
pg.name_part_group AS group_name, pc.name_part_category AS category
FROM parts p
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE {where}
ORDER BY pc.display_order, pg.display_order, p.oem_part_number
LIMIT :limit OFFSET :offset
"""), params).mappings().all()
return jsonify({'data': [dict(r) for r in rows], 'pagination': {
'page': page, 'per_page': per_page, 'total': total,
'total_pages': (total + per_page - 1) // per_page
}})
finally:
session.close()
@app.route('/api/captura/parts/without-image')
def api_captura_parts_without_image():
session = Session()
try:
search = request.args.get('search', '')
page = int(request.args.get('page', 1))
per_page = min(int(request.args.get('per_page', 50)), 100)
offset = (page - 1) * per_page
filters = ["(p.image_url IS NULL OR p.image_url = '')"]
params = {'limit': per_page, 'offset': offset}
if search:
filters.append("(p.oem_part_number ILIKE :search OR p.name_part ILIKE :search)")
params['search'] = f'%{search}%'
where = ' AND '.join(filters)
total = session.execute(text(f"SELECT COUNT(*) FROM parts p WHERE {where}"), params).scalar()
rows = session.execute(text(f"""
SELECT p.id_part, p.oem_part_number, p.name_part, p.name_es,
pg.name_part_group AS group_name, pc.name_part_category AS category
FROM parts p
JOIN part_groups pg ON p.group_id = pg.id_part_group
JOIN part_categories pc ON pg.category_id = pc.id_part_category
WHERE {where}
ORDER BY p.oem_part_number
LIMIT :limit OFFSET :offset
"""), params).mappings().all()
return jsonify({'data': [dict(r) for r in rows], 'pagination': {
'page': page, 'per_page': per_page, 'total': total,
'total_pages': (total + per_page - 1) // per_page
}})
finally:
session.close()
@app.route('/api/captura/parts/<int:part_id>/image', methods=['POST'])
def api_captura_upload_part_image(part_id):
session = Session()
try:
if 'image' not in request.files:
return jsonify({'error': 'No image file provided'}), 400
file = request.files['image']
if not file.filename:
return jsonify({'error': 'No file selected'}), 400
allowed = {'jpg', 'jpeg', 'png', 'webp'}
ext = file.filename.rsplit('.', 1)[-1].lower() if '.' in file.filename else ''
if ext not in allowed:
return jsonify({'error': f'Tipo no permitido. Usar: {", ".join(allowed)}'}), 400
file.seek(0, 2)
size = file.tell()
file.seek(0)
if size > 2 * 1024 * 1024:
return jsonify({'error': 'Archivo muy grande (max 2MB)'}), 400
part = session.execute(text(
"SELECT oem_part_number FROM parts WHERE id_part = :id"
), {'id': part_id}).mappings().first()
if not part:
return jsonify({'error': 'Part not found'}), 404
safe_oem = re.sub(r'[^a-zA-Z0-9_-]', '_', part['oem_part_number'])
filename = f"{safe_oem}.{ext}"
filepath = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'static', 'parts', filename)
file.save(filepath)
image_url = f"/static/parts/{filename}"
session.execute(text(
"UPDATE parts SET image_url = :url WHERE id_part = :id"
), {'url': image_url, 'id': part_id})
session.commit()
return jsonify({'message': 'Image uploaded', 'image_url': image_url})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/captura/parts/check-oem')
def api_captura_check_oem():
session = Session()
try:
oem = request.args.get('oem', '')
if not oem:
return jsonify({'exists': False})
row = session.execute(text(
"SELECT id_part, oem_part_number, name_part, name_es, group_id FROM parts WHERE oem_part_number = :oem"
), {'oem': oem}).mappings().first()
if row:
return jsonify({'exists': True, 'part': dict(row)})
return jsonify({'exists': False})
finally:
session.close()
@app.route('/api/captura/manufacturers')
def api_captura_manufacturers():
session = Session()
try:
rows = session.execute(text("""
SELECT m.id_manufacture AS id, m.name_manufacture AS name,
qt.name_quality AS quality
FROM manufacturers m
LEFT JOIN quality_tier qt ON m.id_quality_tier = qt.id_quality_tier
ORDER BY m.name_manufacture
""")).mappings().all()
return jsonify([dict(r) for r in rows])
finally:
session.close()
@app.route('/api/captura/parts/<int:part_id>/aftermarket')
def api_captura_part_aftermarket(part_id):
session = Session()
try:
rows = session.execute(text("""
SELECT ap.id_aftermarket_parts AS id, ap.part_number,
ap.name_aftermarket_parts AS name,
m.name_manufacture AS manufacturer, qt.name_quality AS quality,
ap.price_usd, ap.warranty_months
FROM aftermarket_parts ap
JOIN manufacturers m ON ap.manufacturer_id = m.id_manufacture
LEFT JOIN quality_tier qt ON ap.id_quality_tier = qt.id_quality_tier
WHERE ap.oem_part_id = :id
ORDER BY m.name_manufacture
"""), {'id': part_id}).mappings().all()
return jsonify([dict(r) for r in rows])
finally:
session.close()
# ============================================================================
# POS (Point of Sale) Endpoints
# ============================================================================
@app.route('/pos')
def pos_page():
return send_from_directory('.', 'pos.html')
@app.route('/pos.js')
def pos_js():
return send_from_directory('.', 'pos.js')
@app.route('/pos.css')
def pos_css():
return send_from_directory('.', 'pos.css')
@app.route('/cuentas')
def cuentas_page():
return send_from_directory('.', 'cuentas.html')
@app.route('/cuentas.js')
def cuentas_js():
return send_from_directory('.', 'cuentas.js')
@app.route('/cuentas.css')
def cuentas_css():
return send_from_directory('.', 'cuentas.css')
# ---- Customers ----
@app.route('/api/pos/customers')
def api_pos_customers():
session = Session()
try:
search = request.args.get('search', '')
page = int(request.args.get('page', 1))
per_page = min(int(request.args.get('per_page', 50)), 100)
offset = (page - 1) * per_page
filters = ["active = TRUE"]
params = {'limit': per_page, 'offset': offset}
if search:
filters.append("(name ILIKE :search OR rfc ILIKE :search OR business_name ILIKE :search)")
params['search'] = f'%{search}%'
where = ' AND '.join(filters)
total = session.execute(text(f"SELECT COUNT(*) FROM customers WHERE {where}"), params).scalar()
rows = session.execute(text(f"""
SELECT id_customer, name, rfc, business_name, phone, balance, credit_limit, payment_terms
FROM customers WHERE {where}
ORDER BY name LIMIT :limit OFFSET :offset
"""), params).mappings().all()
return jsonify({'data': [dict(r) for r in rows], 'pagination': {
'page': page, 'per_page': per_page, 'total': total,
'total_pages': (total + per_page - 1) // per_page
}})
finally:
session.close()
@app.route('/api/pos/customers/<int:customer_id>')
def api_pos_customer_detail(customer_id):
session = Session()
try:
row = session.execute(text(
"SELECT * FROM customers WHERE id_customer = :id"
), {'id': customer_id}).mappings().first()
if not row:
return jsonify({'error': 'Cliente no encontrado'}), 404
return jsonify(dict(row))
finally:
session.close()
@app.route('/api/pos/customers', methods=['POST'])
def api_pos_create_customer():
session = Session()
try:
data = request.get_json()
result = session.execute(text("""
INSERT INTO customers (name, rfc, business_name, email, phone, address, credit_limit, payment_terms)
VALUES (:name, :rfc, :business_name, :email, :phone, :address, :credit_limit, :payment_terms)
RETURNING id_customer
"""), {
'name': data['name'], 'rfc': data.get('rfc'),
'business_name': data.get('business_name'),
'email': data.get('email'), 'phone': data.get('phone'),
'address': data.get('address'),
'credit_limit': data.get('credit_limit', 0),
'payment_terms': data.get('payment_terms', 30)
})
new_id = result.scalar()
session.commit()
return jsonify({'id': new_id, 'message': 'Cliente creado'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/pos/customers/<int:customer_id>', methods=['PUT'])
def api_pos_update_customer(customer_id):
session = Session()
try:
data = request.get_json()
session.execute(text("""
UPDATE customers SET name = :name, rfc = :rfc, business_name = :business_name,
email = :email, phone = :phone, address = :address,
credit_limit = :credit_limit, payment_terms = :payment_terms
WHERE id_customer = :id
"""), {
'name': data['name'], 'rfc': data.get('rfc'),
'business_name': data.get('business_name'),
'email': data.get('email'), 'phone': data.get('phone'),
'address': data.get('address'),
'credit_limit': data.get('credit_limit', 0),
'payment_terms': data.get('payment_terms', 30),
'id': customer_id
})
session.commit()
return jsonify({'message': 'Cliente actualizado'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ---- Invoices ----
@app.route('/api/pos/invoices')
def api_pos_invoices():
session = Session()
try:
customer_id = request.args.get('customer_id', '')
status = request.args.get('status', '')
page = int(request.args.get('page', 1))
per_page = min(int(request.args.get('per_page', 50)), 100)
offset = (page - 1) * per_page
filters = ["1=1"]
params = {'limit': per_page, 'offset': offset}
if customer_id:
filters.append("i.customer_id = :customer_id")
params['customer_id'] = int(customer_id)
if status:
filters.append("i.status = :status")
params['status'] = status
where = ' AND '.join(filters)
total = session.execute(text(f"""
SELECT COUNT(*) FROM invoices i WHERE {where}
"""), params).scalar()
rows = session.execute(text(f"""
SELECT i.id_invoice, i.folio, i.date_issued, i.subtotal, i.tax_amount,
i.total, i.amount_paid, i.status, c.name AS customer_name, c.rfc
FROM invoices i
JOIN customers c ON i.customer_id = c.id_customer
WHERE {where}
ORDER BY i.date_issued DESC
LIMIT :limit OFFSET :offset
"""), params).mappings().all()
return jsonify({'data': [dict(r) for r in rows], 'pagination': {
'page': page, 'per_page': per_page, 'total': total,
'total_pages': (total + per_page - 1) // per_page
}})
finally:
session.close()
@app.route('/api/pos/invoices/<int:invoice_id>')
def api_pos_invoice_detail(invoice_id):
session = Session()
try:
inv = session.execute(text("""
SELECT i.*, c.name AS customer_name, c.rfc, c.business_name, c.address
FROM invoices i JOIN customers c ON i.customer_id = c.id_customer
WHERE i.id_invoice = :id
"""), {'id': invoice_id}).mappings().first()
if not inv:
return jsonify({'error': 'Factura no encontrada'}), 404
items = session.execute(text("""
SELECT ii.*, p.oem_part_number, ap.part_number AS aftermarket_number
FROM invoice_items ii
LEFT JOIN parts p ON ii.part_id = p.id_part
LEFT JOIN aftermarket_parts ap ON ii.aftermarket_id = ap.id_aftermarket_parts
WHERE ii.invoice_id = :id
ORDER BY ii.id_invoice_item
"""), {'id': invoice_id}).mappings().all()
return jsonify({'invoice': dict(inv), 'items': [dict(it) for it in items]})
finally:
session.close()
@app.route('/api/pos/invoices', methods=['POST'])
def api_pos_create_invoice():
session = Session()
try:
data = request.get_json()
customer_id = data['customer_id']
items = data['items']
tax_rate = data.get('tax_rate', 0.16)
notes = data.get('notes', '')
if not items:
return jsonify({'error': 'La factura debe tener al menos una linea'}), 400
folio_num = session.execute(text("SELECT nextval('invoice_folio_seq')")).scalar()
folio = f"NX-{folio_num:06d}"
subtotal = sum(it['quantity'] * it['unit_price'] for it in items)
tax_amount = round(subtotal * tax_rate, 2)
total = round(subtotal + tax_amount, 2)
result = session.execute(text("""
INSERT INTO invoices (customer_id, folio, subtotal, tax_rate, tax_amount, total, notes)
VALUES (:customer_id, :folio, :subtotal, :tax_rate, :tax_amount, :total, :notes)
RETURNING id_invoice
"""), {
'customer_id': customer_id, 'folio': folio,
'subtotal': subtotal, 'tax_rate': tax_rate,
'tax_amount': tax_amount, 'total': total, 'notes': notes
})
invoice_id = result.scalar()
for it in items:
line_total = it['quantity'] * it['unit_price']
session.execute(text("""
INSERT INTO invoice_items (invoice_id, part_id, aftermarket_id, description,
quantity, unit_cost, margin_pct, unit_price, line_total)
VALUES (:inv_id, :part_id, :af_id, :desc, :qty, :cost, :margin, :price, :total)
"""), {
'inv_id': invoice_id,
'part_id': it.get('part_id'),
'af_id': it.get('aftermarket_id'),
'desc': it['description'],
'qty': it['quantity'],
'cost': it.get('unit_cost', 0),
'margin': it.get('margin_pct', 30),
'price': it['unit_price'],
'total': line_total
})
session.execute(text(
"UPDATE customers SET balance = balance + :total WHERE id_customer = :id"
), {'total': total, 'id': customer_id})
session.commit()
return jsonify({'id': invoice_id, 'folio': folio, 'total': total, 'message': 'Factura creada'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/pos/invoices/<int:invoice_id>/cancel', methods=['PUT'])
def api_pos_cancel_invoice(invoice_id):
session = Session()
try:
inv = session.execute(text(
"SELECT total, customer_id, status FROM invoices WHERE id_invoice = :id"
), {'id': invoice_id}).mappings().first()
if not inv:
return jsonify({'error': 'Factura no encontrada'}), 404
if inv['status'] == 'cancelled':
return jsonify({'error': 'La factura ya esta cancelada'}), 400
session.execute(text(
"UPDATE invoices SET status = 'cancelled' WHERE id_invoice = :id"
), {'id': invoice_id})
session.execute(text(
"UPDATE customers SET balance = balance - :total WHERE id_customer = :cid"
), {'total': inv['total'], 'cid': inv['customer_id']})
session.commit()
return jsonify({'message': 'Factura cancelada'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
# ---- Payments ----
@app.route('/api/pos/payments', methods=['POST'])
def api_pos_create_payment():
session = Session()
try:
data = request.get_json()
customer_id = data['customer_id']
amount = float(data['amount'])
payment_method = data.get('payment_method', 'efectivo')
reference = data.get('reference')
invoice_id = data.get('invoice_id')
notes = data.get('notes')
if amount <= 0:
return jsonify({'error': 'El monto debe ser mayor a 0'}), 400
result = session.execute(text("""
INSERT INTO payments (customer_id, invoice_id, amount, payment_method, reference, notes)
VALUES (:cid, :inv_id, :amount, :method, :ref, :notes)
RETURNING id_payment
"""), {
'cid': customer_id, 'inv_id': invoice_id,
'amount': amount, 'method': payment_method,
'ref': reference, 'notes': notes
})
payment_id = result.scalar()
session.execute(text(
"UPDATE customers SET balance = balance - :amount WHERE id_customer = :id"
), {'amount': amount, 'id': customer_id})
if invoice_id:
session.execute(text(
"UPDATE invoices SET amount_paid = amount_paid + :amount WHERE id_invoice = :id"
), {'amount': amount, 'id': invoice_id})
session.execute(text("""
UPDATE invoices SET status = CASE
WHEN amount_paid >= total THEN 'paid'
WHEN amount_paid > 0 THEN 'partial'
ELSE 'pending'
END WHERE id_invoice = :id
"""), {'id': invoice_id})
session.commit()
return jsonify({'id': payment_id, 'message': 'Pago registrado'})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/pos/customers/<int:customer_id>/statement')
def api_pos_customer_statement(customer_id):
session = Session()
try:
customer = session.execute(text(
"SELECT * FROM customers WHERE id_customer = :id"
), {'id': customer_id}).mappings().first()
if not customer:
return jsonify({'error': 'Cliente no encontrado'}), 404
invoices = session.execute(text("""
SELECT id_invoice, folio, date_issued, total, amount_paid, status
FROM invoices WHERE customer_id = :id AND status != 'cancelled'
ORDER BY date_issued DESC LIMIT 100
"""), {'id': customer_id}).mappings().all()
payments = session.execute(text("""
SELECT p.id_payment, p.amount, p.payment_method, p.reference,
p.date_payment, p.notes, i.folio AS invoice_folio
FROM payments p
LEFT JOIN invoices i ON p.invoice_id = i.id_invoice
WHERE p.customer_id = :id
ORDER BY p.date_payment DESC LIMIT 100
"""), {'id': customer_id}).mappings().all()
return jsonify({
'customer': dict(customer),
'invoices': [dict(i) for i in invoices],
'payments': [dict(p) for p in payments]
})
finally:
session.close()
@app.route('/api/pos/search-parts')
def api_pos_search_parts():
session = Session()
try:
q = request.args.get('q', '')
if len(q) < 2:
return jsonify([])
results = []
oem = session.execute(text("""
SELECT p.id_part, p.oem_part_number, p.name_part, p.name_es,
p.cost_usd, pg.name_part_group AS group_name,
'oem' AS part_type
FROM parts p
JOIN part_groups pg ON p.group_id = pg.id_part_group
WHERE p.oem_part_number ILIKE :q OR p.name_part ILIKE :q
ORDER BY p.oem_part_number LIMIT 20
"""), {'q': f'%{q}%'}).mappings().all()
results.extend([dict(r) for r in oem])
af = session.execute(text("""
SELECT ap.id_aftermarket_parts AS id_part, ap.part_number AS oem_part_number,
ap.name_aftermarket_parts AS name_part, ap.name_es,
COALESCE(ap.cost_usd, ap.price_usd) AS cost_usd,
m.name_manufacture AS group_name,
'aftermarket' AS part_type
FROM aftermarket_parts ap
JOIN manufacturers m ON ap.manufacturer_id = m.id_manufacture
WHERE ap.part_number ILIKE :q OR ap.name_aftermarket_parts ILIKE :q
ORDER BY ap.part_number LIMIT 20
"""), {'q': f'%{q}%'}).mappings().all()
results.extend([dict(r) for r in af])
return jsonify(results)
finally:
session.close()
# ============================================================================
# Store Dashboard Endpoints
# ============================================================================
@app.route('/demo')
def demo_page():
return send_from_directory('.', 'demo.html')
@app.route('/bodega')
def bodega_page():
return send_from_directory('.', 'bodega.html')
@app.route('/bodega.js')
def bodega_js():
return send_from_directory('.', 'bodega.js')
@app.route('/bodega.css')
def bodega_css():
return send_from_directory('.', 'bodega.css')
@app.route('/login.html')
def login_page():
return send_from_directory('.', 'login.html')
@app.route('/login.js')
def login_js():
return send_from_directory('.', 'login.js')
@app.route('/login.css')
def login_css():
return send_from_directory('.', 'login.css')
@app.route('/tienda')
def tienda_page():
return send_from_directory('.', 'tienda.html')
@app.route('/tienda.js')
def tienda_js():
return send_from_directory('.', 'tienda.js')
@app.route('/tienda.css')
def tienda_css():
return send_from_directory('.', 'tienda.css')
@app.route('/api/tienda/stats')
def api_tienda_stats():
session = Session()
try:
today = "date_issued::date = CURRENT_DATE"
month = "date_issued >= date_trunc('month', CURRENT_DATE)"
sales_today = session.execute(text(f"""
SELECT COALESCE(SUM(total), 0), COUNT(*)
FROM invoices WHERE {today} AND status != 'cancelled'
""")).fetchone()
sales_month = session.execute(text(f"""
SELECT COALESCE(SUM(total), 0), COUNT(*)
FROM invoices WHERE {month} AND status != 'cancelled'
""")).fetchone()
payments_today = session.execute(text(f"""
SELECT COALESCE(SUM(amount), 0), COUNT(*)
FROM payments WHERE date_payment::date = CURRENT_DATE
""")).fetchone()
pending_balance = session.execute(text(
"SELECT COALESCE(SUM(balance), 0) FROM customers WHERE active = TRUE AND balance > 0"
)).scalar()
pending_invoices = session.execute(text(
"SELECT COUNT(*) FROM invoices WHERE status IN ('pending', 'partial')"
)).scalar()
total_customers = session.execute(text(
"SELECT COUNT(*) FROM customers WHERE active = TRUE"
)).scalar()
total_parts = session.execute(text("SELECT COUNT(*) FROM parts")).scalar()
total_aftermarket = session.execute(text("SELECT COUNT(*) FROM aftermarket_parts")).scalar()
recent_invoices = session.execute(text("""
SELECT i.folio, i.total, i.status, i.date_issued, c.name AS customer_name
FROM invoices i JOIN customers c ON i.customer_id = c.id_customer
ORDER BY i.date_issued DESC LIMIT 8
""")).mappings().all()
top_debtors = session.execute(text("""
SELECT id_customer, name, balance, credit_limit
FROM customers WHERE active = TRUE AND balance > 0
ORDER BY balance DESC LIMIT 6
""")).mappings().all()
return jsonify({
'sales_today': {'total': float(sales_today[0]), 'count': int(sales_today[1])},
'sales_month': {'total': float(sales_month[0]), 'count': int(sales_month[1])},
'payments_today': {'total': float(payments_today[0]), 'count': int(payments_today[1])},
'pending_balance': float(pending_balance),
'pending_invoices': pending_invoices,
'total_customers': total_customers,
'total_parts': total_parts,
'total_aftermarket': total_aftermarket,
'recent_invoices': [dict(r) for r in recent_invoices],
'top_debtors': [dict(r) for r in top_debtors]
})
finally:
session.close()
# ============================================================================
# Auth Endpoints
# ============================================================================
@app.route('/api/auth/register', methods=['POST'])
def auth_register():
"""Register a new user (TALLER or BODEGA). Account starts inactive."""
data = request.get_json()
if not data:
return jsonify({'error': 'Invalid JSON'}), 400
required = ['name', 'email', 'password', 'role', 'business_name']
for field in required:
if not data.get(field):
return jsonify({'error': f'Missing required field: {field}'}), 400
role = data['role'].upper()
if role not in ('TALLER', 'BODEGA'):
return jsonify({'error': 'Role must be TALLER or BODEGA'}), 400
role_map = {'TALLER': 3, 'BODEGA': 4}
id_rol = role_map[role]
hashed = hash_password(data['password'])
session = Session()
try:
session.execute(text(
"""INSERT INTO users (name_user, email, pass, id_rol, business_name, phone, address, is_active, created_at)
VALUES (:name, :email, :pass, :id_rol, :biz, :phone, :addr, false, NOW())"""
), {
'name': data['name'],
'email': data['email'],
'pass': hashed,
'id_rol': id_rol,
'biz': data['business_name'],
'phone': data.get('phone', ''),
'addr': data.get('address', '')
})
session.commit()
return jsonify({'message': 'Registration successful. Account pending activation.'}), 201
except IntegrityError:
session.rollback()
return jsonify({'error': 'Email already registered'}), 409
finally:
session.close()
@app.route('/api/auth/login', methods=['POST'])
def auth_login():
"""Authenticate user and return access + refresh tokens."""
data = request.get_json()
if not data or not data.get('email') or not data.get('password'):
return jsonify({'error': 'Email and password are required'}), 400
session = Session()
try:
row = session.execute(text(
"""SELECT u.id_user, u.name_user, u.email, u.pass, u.is_active,
u.business_name, r.name_rol
FROM users u
JOIN roles r ON r.id_rol = u.id_rol
WHERE u.email = :email"""
), {'email': data['email']}).mappings().first()
if not row:
return jsonify({'error': 'Invalid email or password'}), 401
if not check_password(data['password'], row['pass']):
return jsonify({'error': 'Invalid email or password'}), 401
if not row['is_active']:
return jsonify({'error': 'Account is not active. Contact an administrator.'}), 403
# Update last_login
session.execute(text(
"UPDATE users SET last_login = NOW() WHERE id_user = :uid"
), {'uid': row['id_user']})
session.commit()
access_token = create_access_token(row['id_user'], row['name_rol'], row['business_name'])
refresh_token = create_refresh_token(row['id_user'])
return jsonify({
'access_token': access_token,
'refresh_token': refresh_token,
'user': {
'id': row['id_user'],
'name': row['name_user'],
'role': row['name_rol'],
'business_name': row['business_name']
}
})
finally:
session.close()
@app.route('/api/auth/refresh', methods=['POST'])
def auth_refresh():
"""Exchange a valid refresh token for a new access token."""
data = request.get_json()
if not data or not data.get('refresh_token'):
return jsonify({'error': 'refresh_token is required'}), 400
session = Session()
try:
row = session.execute(text(
"""SELECT s.user_id, s.expires_at, u.business_name, r.name_rol
FROM sessions s
JOIN users u ON u.id_user = s.user_id
JOIN roles r ON r.id_rol = u.id_rol
WHERE s.refresh_token = :token"""
), {'token': data['refresh_token']}).mappings().first()
if not row:
return jsonify({'error': 'Invalid refresh token'}), 401
if row['expires_at'] < datetime.utcnow():
# Clean up expired token
session.execute(text(
"DELETE FROM sessions WHERE refresh_token = :token"
), {'token': data['refresh_token']})
session.commit()
return jsonify({'error': 'Refresh token expired'}), 401
access_token = create_access_token(row['user_id'], row['name_rol'], row['business_name'])
return jsonify({'access_token': access_token})
finally:
session.close()
@app.route('/api/auth/me', methods=['GET'])
@require_auth()
def auth_me():
"""Return the current authenticated user's info from the JWT payload."""
return jsonify(g.user)
# ============================================================================
# Task 5: Admin User Management
# ============================================================================
import csv
import io
import math
@app.route('/api/admin/users', methods=['GET'])
@require_auth('ADMIN', 'OWNER')
def admin_list_users():
"""Return list of all users with role info."""
session = Session()
try:
rows = session.execute(text(
"""SELECT u.id_user, u.name_user, u.email, u.business_name,
u.phone, u.is_active, u.created_at, u.last_login,
r.name_rol
FROM users u
JOIN roles r ON r.id_rol = u.id_rol
ORDER BY u.created_at DESC"""
)).mappings().all()
users = []
for r in rows:
users.append({
'id': r['id_user'],
'name': r['name_user'],
'email': r['email'],
'business_name': r['business_name'],
'phone': r['phone'],
'is_active': r['is_active'],
'created_at': r['created_at'].isoformat() if r['created_at'] else None,
'last_login': r['last_login'].isoformat() if r['last_login'] else None,
'role': r['name_rol']
})
return jsonify(users)
finally:
session.close()
@app.route('/api/admin/users/<int:user_id>/activate', methods=['PUT'])
@require_auth('ADMIN', 'OWNER')
def admin_activate_user(user_id):
"""Activate or deactivate a user."""
data = request.get_json()
if data is None or 'is_active' not in data:
return jsonify({'error': 'is_active field is required'}), 400
session = Session()
try:
result = session.execute(text(
"UPDATE users SET is_active = :active WHERE id_user = :uid"
), {'active': bool(data['is_active']), 'uid': user_id})
session.commit()
if result.rowcount == 0:
return jsonify({'error': 'User not found'}), 404
return jsonify({'message': 'User updated', 'is_active': bool(data['is_active'])})
finally:
session.close()
# ============================================================================
# Task 6: Inventory Endpoints (BODEGA)
# ============================================================================
@app.route('/api/inventory/mapping', methods=['GET'])
@require_auth('BODEGA', 'ADMIN')
def inventory_get_mapping():
"""Return column mapping for current user."""
session = Session()
try:
row = session.execute(text(
"SELECT mapping FROM inventory_column_mappings WHERE user_id = :uid"
), {'uid': g.user['user_id']}).mappings().first()
return jsonify({'mapping': row['mapping'] if row else {}})
finally:
session.close()
@app.route('/api/inventory/mapping', methods=['PUT'])
@require_auth('BODEGA', 'ADMIN')
def inventory_put_mapping():
"""Upsert column mapping for current user."""
data = request.get_json()
if not data or 'mapping' not in data:
return jsonify({'error': 'mapping is required'}), 400
mapping = data['mapping']
required_keys = ['part_number', 'price', 'stock']
missing = [k for k in required_keys if k not in mapping or not mapping[k]]
if missing:
return jsonify({'error': f'Missing required mapping keys: {", ".join(missing)}'}), 400
session = Session()
try:
session.execute(text(
"""INSERT INTO inventory_column_mappings (user_id, mapping)
VALUES (:uid, :mapping)
ON CONFLICT (user_id) DO UPDATE SET mapping = :mapping"""
), {'uid': g.user['user_id'], 'mapping': json_module.dumps(mapping)})
session.commit()
return jsonify({'message': 'Mapping saved', 'mapping': mapping})
finally:
session.close()
@app.route('/api/inventory/upload', methods=['POST'])
@require_auth('BODEGA', 'ADMIN')
def inventory_upload():
"""Upload inventory file (CSV or Excel), apply column mapping, upsert into warehouse_inventory."""
if 'file' not in request.files:
return jsonify({'error': 'No file provided'}), 400
file = request.files['file']
if not file.filename:
return jsonify({'error': 'Empty filename'}), 400
session = Session()
try:
# 1. Get mapping
row = session.execute(text(
"SELECT mapping FROM inventory_column_mappings WHERE user_id = :uid"
), {'uid': g.user['user_id']}).mappings().first()
if not row or not row['mapping']:
return jsonify({'error': 'No column mapping configured. Set mapping first.'}), 400
mapping = row['mapping'] if isinstance(row['mapping'], dict) else json_module.loads(row['mapping'])
# 2. Create upload record
upload = session.execute(text(
"""INSERT INTO inventory_uploads (user_id, filename, status)
VALUES (:uid, :fname, 'processing')
RETURNING id_upload"""
), {'uid': g.user['user_id'], 'fname': file.filename}).mappings().first()
session.commit()
upload_id = upload['id_upload']
# 3. Parse file
filename_lower = file.filename.lower()
rows_data = []
if filename_lower.endswith(('.xlsx', '.xls')):
import openpyxl
wb = openpyxl.load_workbook(io.BytesIO(file.read()), read_only=True, data_only=True)
ws = wb.active
headers = None
for row_cells in ws.iter_rows(values_only=True):
if headers is None:
headers = [str(c).strip() if c else '' for c in row_cells]
continue
row_dict = {}
for i, val in enumerate(row_cells):
if i < len(headers):
row_dict[headers[i]] = val
rows_data.append(row_dict)
wb.close()
else:
# CSV
content = file.read().decode('utf-8-sig', errors='replace')
reader = csv.DictReader(io.StringIO(content))
for row_dict in reader:
rows_data.append(row_dict)
# 4. Process rows
imported = 0
errors = 0
error_samples = []
def clean_price(val):
if val is None:
return None
s = str(val).replace('$', '').replace(',', '').strip()
try:
return float(s)
except (ValueError, TypeError):
return None
def clean_stock(val):
if val is None:
return 0
s = str(val).replace(',', '').strip()
try:
return int(float(s))
except (ValueError, TypeError):
return 0
for i, row_dict in enumerate(rows_data):
try:
part_number_col = mapping.get('part_number', '')
price_col = mapping.get('price', '')
stock_col = mapping.get('stock', '')
location_col = mapping.get('location', '')
part_number = str(row_dict.get(part_number_col, '')).strip()
if not part_number:
errors += 1
if len(error_samples) < 10:
error_samples.append({'row': i + 2, 'error': 'Empty part number'})
continue
price = clean_price(row_dict.get(price_col))
stock = clean_stock(row_dict.get(stock_col))
location = str(row_dict.get(location_col, 'Principal')).strip() if location_col else 'Principal'
if not location:
location = 'Principal'
# Find part by OEM part number
part_row = session.execute(text(
"SELECT id_part FROM parts WHERE oem_part_number = :pn LIMIT 1"
), {'pn': part_number}).mappings().first()
# Also try aftermarket_parts.part_number if OEM not found
if not part_row:
am_row = session.execute(text(
"SELECT oem_part_id FROM aftermarket_parts WHERE part_number = :pn LIMIT 1"
), {'pn': part_number}).mappings().first()
if am_row:
part_row = {'id_part': am_row['oem_part_id']}
if not part_row:
errors += 1
if len(error_samples) < 10:
error_samples.append({'row': i + 2, 'error': f'Part not found: {part_number}'})
continue
# UPSERT into warehouse_inventory
session.execute(text(
"""INSERT INTO warehouse_inventory (user_id, part_id, price, stock_quantity, warehouse_location, updated_at)
VALUES (:uid, :pid, :price, :stock, :loc, NOW())
ON CONFLICT (user_id, part_id, warehouse_location)
DO UPDATE SET price = :price, stock_quantity = :stock, updated_at = NOW()"""
), {
'uid': g.user['user_id'],
'pid': part_row['id_part'],
'price': price,
'stock': stock,
'loc': location
})
imported += 1
except Exception as e:
errors += 1
if len(error_samples) < 10:
error_samples.append({'row': i + 2, 'error': str(e)})
# 5. Update upload record
session.execute(text(
"""UPDATE inventory_uploads
SET status = 'completed', rows_total = :total,
rows_imported = :imported, rows_errors = :errors,
error_log = :elog, completed_at = NOW()
WHERE id_upload = :uid"""
), {
'total': len(rows_data),
'imported': imported,
'errors': errors,
'elog': json_module.dumps(error_samples) if error_samples else None,
'uid': upload_id
})
session.commit()
return jsonify({
'message': 'Upload processed',
'upload_id': upload_id,
'imported': imported,
'errors': errors,
'error_samples': error_samples
})
except Exception as e:
session.rollback()
return jsonify({'error': str(e)}), 500
finally:
session.close()
@app.route('/api/inventory/uploads', methods=['GET'])
@require_auth('BODEGA', 'ADMIN')
def inventory_list_uploads():
"""Return last 50 uploads for current user."""
session = Session()
try:
rows = session.execute(text(
"""SELECT id_upload, filename, status, rows_total, rows_imported,
rows_errors, created_at, completed_at
FROM inventory_uploads
WHERE user_id = :uid
ORDER BY created_at DESC
LIMIT 50"""
), {'uid': g.user['user_id']}).mappings().all()
uploads = []
for r in rows:
uploads.append({
'id': r['id_upload'],
'filename': r['filename'],
'status': r['status'],
'rows_total': r['rows_total'],
'rows_imported': r['rows_imported'],
'rows_errors': r['rows_errors'],
'created_at': r['created_at'].isoformat() if r['created_at'] else None,
'completed_at': r['completed_at'].isoformat() if r['completed_at'] else None
})
return jsonify(uploads)
finally:
session.close()
@app.route('/api/inventory/items', methods=['GET'])
@require_auth('BODEGA', 'ADMIN')
def inventory_list_items():
"""Paginated list of warehouse_inventory for current user."""
page = max(1, request.args.get('page', 1, type=int))
per_page = min(200, max(1, request.args.get('per_page', 50, type=int)))
q = request.args.get('q', '').strip()
session = Session()
try:
params = {'uid': g.user['user_id'], 'offset': (page - 1) * per_page, 'limit': per_page}
where_clause = "WHERE wi.user_id = :uid"
if q:
where_clause += " AND (p.oem_part_number ILIKE :q OR p.name_part ILIKE :q)"
params['q'] = f'%{q}%'
count_row = session.execute(text(
f"""SELECT COUNT(*) AS cnt
FROM warehouse_inventory wi
JOIN parts p ON p.id_part = wi.part_id
{where_clause}"""
), params).mappings().first()
total = count_row['cnt']
rows = session.execute(text(
f"""SELECT wi.id_inventory, wi.part_id, p.oem_part_number,
p.name_part, wi.price, wi.stock_quantity,
wi.warehouse_location, wi.updated_at
FROM warehouse_inventory wi
JOIN parts p ON p.id_part = wi.part_id
{where_clause}
ORDER BY wi.updated_at DESC
LIMIT :limit OFFSET :offset"""
), params).mappings().all()
data = []
for r in rows:
data.append({
'id': r['id_inventory'],
'part_id': r['part_id'],
'oem_part_number': r['oem_part_number'],
'name': r['name_part'],
'price': float(r['price']) if r['price'] else None,
'stock': r['stock_quantity'],
'location': r['warehouse_location'],
'updated_at': r['updated_at'].isoformat() if r['updated_at'] else None
})
total_pages = math.ceil(total / per_page) if total > 0 else 1
return jsonify({
'data': data,
'pagination': {
'page': page,
'per_page': per_page,
'total': total,
'total_pages': total_pages
}
})
finally:
session.close()
@app.route('/api/inventory/items', methods=['DELETE'])
@require_auth('BODEGA', 'ADMIN')
def inventory_delete_all():
"""Delete all warehouse_inventory for current user."""
session = Session()
try:
result = session.execute(text(
"DELETE FROM warehouse_inventory WHERE user_id = :uid"
), {'uid': g.user['user_id']})
session.commit()
return jsonify({'message': 'Inventory cleared', 'deleted': result.rowcount})
finally:
session.close()
# ============================================================================
# Task 7: Part Availability & Aftermarket
# ============================================================================
@app.route('/api/parts/<int:part_id>/availability', methods=['GET'])
@require_auth('TALLER', 'ADMIN', 'OWNER')
def part_availability(part_id):
"""Return all bodegas that have this part in stock."""
session = Session()
try:
rows = session.execute(text(
"""SELECT u.business_name, wi.price, wi.stock_quantity,
wi.warehouse_location, wi.updated_at
FROM warehouse_inventory wi
JOIN users u ON u.id_user = wi.user_id
WHERE wi.part_id = :pid
AND wi.stock_quantity > 0
AND u.is_active = true
ORDER BY wi.price ASC"""
), {'pid': part_id}).mappings().all()
data = []
for r in rows:
data.append({
'bodega': r['business_name'],
'price': float(r['price']) if r['price'] else None,
'stock': r['stock_quantity'],
'location': r['warehouse_location'],
'updated_at': r['updated_at'].isoformat() if r['updated_at'] else None
})
return jsonify(data)
finally:
session.close()
@app.route('/api/parts/<int:part_id>/aftermarket', methods=['GET'])
def part_aftermarket(part_id):
"""Return aftermarket alternatives and cross-references for a part (public)."""
session = Session()
try:
# Aftermarket alternatives
rows = session.execute(text(
"""SELECT ap.id_aftermarket_parts, ap.part_number,
ap.name_aftermarket_parts, m.name_manufacture,
qt.name_quality, ap.price_usd
FROM aftermarket_parts ap
JOIN manufacturers m ON m.id_manufacture = ap.manufacturer_id
LEFT JOIN quality_tier qt ON qt.id_quality_tier = ap.id_quality_tier
WHERE ap.oem_part_id = :pid
ORDER BY ap.price_usd ASC NULLS LAST"""
), {'pid': part_id}).mappings().all()
alternatives = []
for r in rows:
alternatives.append({
'id': r['id_aftermarket_parts'],
'part_number': r['part_number'],
'name': r['name_aftermarket_parts'],
'manufacturer': r['name_manufacture'],
'quality_tier': r['name_quality'],
'price': float(r['price_usd']) if r['price_usd'] else None,
'source': 'aftermarket'
})
# Cross-references
xrefs = session.execute(text(
"""SELECT pcr.cross_reference_number, pcr.source_ref, pcr.notes
FROM part_cross_references pcr
WHERE pcr.part_id = :pid
ORDER BY pcr.cross_reference_number"""
), {'pid': part_id}).mappings().all()
cross_refs = []
for x in xrefs:
cross_refs.append({
'cross_reference_number': x['cross_reference_number'],
'source': x['source_ref'],
'notes': x['notes']
})
return jsonify({
'data': alternatives,
'cross_references': cross_refs
})
finally:
session.close()
# ============================================================================
# Main Block
# ============================================================================
if __name__ == '__main__':
print("Starting Nexus Autoparts Dashboard Server...")
print("Visit http://localhost:5000 to access the dashboard locally")
app.run(debug=False, host='0.0.0.0', port=5000)