Some checks failed
Storefront homepage & module gating:
- CMS owns storefront GET / (slug="home" with 3-tier resolution)
- Catalog loses GET / (keeps /products only)
- Store root redirect (GET / → /store/dashboard or /store/login)
- Route gating: non-core modules return 404 when disabled for platform
- Seed store default homepages per platform
Widget protocol for customer dashboard:
- StorefrontDashboardCard contract in widgets.py
- Widget aggregator get_storefront_dashboard_cards()
- Orders and Loyalty module widget providers
- Dashboard template renders contributed cards (no module names)
Landing template module-agnostic:
- CTAs driven by storefront_nav (not hardcoded module names)
- Header actions check nav item IDs (not enabled_modules)
- Remove hardcoded "Add Product" sidebar button
- Remove all enabled_modules checks from storefront templates
i18n fixes:
- Title placeholder resolution ({{store_name}}) for store default pages
- Storefront nav label_keys prefixed with module code
- Add storefront.account.* keys to 6 modules (en/fr/de/lb)
- Header/footer CMS pages use get_translated_title(current_language)
- Footer labels use i18n keys instead of hardcoded English
Icon cleanup:
- Standardize on map-pin (remove location-marker alias)
- Replace all location-marker references across templates and docs
Docs:
- Storefront builder vision proposal (6 phases)
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
298 lines
9.6 KiB
Python
298 lines
9.6 KiB
Python
# app/modules/customers/routes/pages/storefront.py
|
|
"""
|
|
Customers Storefront Page Routes (HTML rendering).
|
|
|
|
Storefront (customer shop) pages for customer account:
|
|
- Registration
|
|
- Login
|
|
- Forgot password
|
|
- Reset password
|
|
- Account dashboard
|
|
- Profile management
|
|
- Addresses
|
|
- Settings
|
|
"""
|
|
|
|
import logging
|
|
|
|
from fastapi import APIRouter, Depends, Request
|
|
from fastapi.responses import HTMLResponse, RedirectResponse
|
|
from sqlalchemy.orm import Session
|
|
|
|
from app.api.deps import get_current_customer_from_cookie_or_header, get_db
|
|
from app.modules.core.utils.page_context import get_storefront_context
|
|
from app.modules.customers.models import Customer
|
|
from app.templates_config import templates
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
router = APIRouter()
|
|
|
|
|
|
# ============================================================================
|
|
# CUSTOMER ACCOUNT - PUBLIC ROUTES (No Authentication)
|
|
# ============================================================================
|
|
|
|
|
|
@router.get("/account/register", response_class=HTMLResponse, include_in_schema=False)
|
|
async def shop_register_page(request: Request, db: Session = Depends(get_db)):
|
|
"""
|
|
Render customer registration page.
|
|
No authentication required.
|
|
"""
|
|
logger.debug(
|
|
"[STOREFRONT] shop_register_page REACHED",
|
|
extra={
|
|
"path": request.url.path,
|
|
"store": getattr(request.state, "store", "NOT SET"),
|
|
"context": getattr(request.state, "context_type", "NOT SET"),
|
|
},
|
|
)
|
|
|
|
return templates.TemplateResponse(
|
|
"customers/storefront/register.html", get_storefront_context(request, db=db)
|
|
)
|
|
|
|
|
|
@router.get("/account/login", response_class=HTMLResponse, include_in_schema=False)
|
|
async def shop_login_page(request: Request, db: Session = Depends(get_db)):
|
|
"""
|
|
Render customer login page.
|
|
No authentication required.
|
|
"""
|
|
logger.debug(
|
|
"[STOREFRONT] shop_login_page REACHED",
|
|
extra={
|
|
"path": request.url.path,
|
|
"store": getattr(request.state, "store", "NOT SET"),
|
|
"context": getattr(request.state, "context_type", "NOT SET"),
|
|
},
|
|
)
|
|
|
|
return templates.TemplateResponse(
|
|
"customers/storefront/login.html", get_storefront_context(request, db=db)
|
|
)
|
|
|
|
|
|
@router.get(
|
|
"/account/forgot-password", response_class=HTMLResponse, include_in_schema=False
|
|
)
|
|
async def shop_forgot_password_page(request: Request, db: Session = Depends(get_db)):
|
|
"""
|
|
Render forgot password page.
|
|
Allows customers to reset their password.
|
|
"""
|
|
logger.debug(
|
|
"[STOREFRONT] shop_forgot_password_page REACHED",
|
|
extra={
|
|
"path": request.url.path,
|
|
"store": getattr(request.state, "store", "NOT SET"),
|
|
"context": getattr(request.state, "context_type", "NOT SET"),
|
|
},
|
|
)
|
|
|
|
return templates.TemplateResponse(
|
|
"customers/storefront/forgot-password.html", get_storefront_context(request, db=db)
|
|
)
|
|
|
|
|
|
@router.get(
|
|
"/account/reset-password", response_class=HTMLResponse, include_in_schema=False
|
|
)
|
|
async def shop_reset_password_page(
|
|
request: Request, token: str = None, db: Session = Depends(get_db)
|
|
):
|
|
"""
|
|
Render reset password page.
|
|
User lands here after clicking the reset link in their email.
|
|
Token is passed as query parameter.
|
|
"""
|
|
logger.debug(
|
|
"[STOREFRONT] shop_reset_password_page REACHED",
|
|
extra={
|
|
"path": request.url.path,
|
|
"store": getattr(request.state, "store", "NOT SET"),
|
|
"context": getattr(request.state, "context_type", "NOT SET"),
|
|
"has_token": bool(token),
|
|
},
|
|
)
|
|
|
|
return templates.TemplateResponse(
|
|
"customers/storefront/reset-password.html", get_storefront_context(request, db=db)
|
|
)
|
|
|
|
|
|
# ============================================================================
|
|
# CUSTOMER ACCOUNT - ROOT REDIRECT
|
|
# ============================================================================
|
|
|
|
|
|
@router.get("/account", response_class=RedirectResponse, include_in_schema=False)
|
|
@router.get("/account/", response_class=RedirectResponse, include_in_schema=False)
|
|
async def shop_account_root(request: Request):
|
|
"""
|
|
Redirect /storefront/account or /storefront/account/ to dashboard.
|
|
"""
|
|
logger.debug(
|
|
"[STOREFRONT] shop_account_root REACHED",
|
|
extra={
|
|
"path": request.url.path,
|
|
"store": getattr(request.state, "store", "NOT SET"),
|
|
"context": getattr(request.state, "context_type", "NOT SET"),
|
|
},
|
|
)
|
|
|
|
# Get base_url from context for proper redirect
|
|
store = getattr(request.state, "store", None)
|
|
store_context = getattr(request.state, "store_context", None)
|
|
access_method = (
|
|
store_context.get("detection_method", "unknown")
|
|
if store_context
|
|
else "unknown"
|
|
)
|
|
|
|
base_url = "/"
|
|
if access_method == "path" and store:
|
|
platform = getattr(request.state, "platform", None)
|
|
platform_original_path = getattr(request.state, "platform_original_path", None)
|
|
if platform and platform_original_path and platform_original_path.startswith("/platforms/"):
|
|
base_url = f"/platforms/{platform.code}/storefront/{store.store_code}/"
|
|
else:
|
|
full_prefix = (
|
|
store_context.get("full_prefix", "/storefront/")
|
|
if store_context
|
|
else "/storefront/"
|
|
)
|
|
base_url = f"{full_prefix}{store.store_code}/"
|
|
|
|
return RedirectResponse(url=f"{base_url}account/dashboard", status_code=302)
|
|
|
|
|
|
# ============================================================================
|
|
# CUSTOMER ACCOUNT - AUTHENTICATED ROUTES
|
|
# ============================================================================
|
|
|
|
|
|
@router.get(
|
|
"/account/dashboard", response_class=HTMLResponse, include_in_schema=False
|
|
)
|
|
async def shop_account_dashboard_page(
|
|
request: Request,
|
|
current_customer: Customer = Depends(get_current_customer_from_cookie_or_header),
|
|
db: Session = Depends(get_db),
|
|
):
|
|
"""
|
|
Render customer account dashboard.
|
|
Shows account overview, recent orders, and quick links.
|
|
Requires customer authentication.
|
|
"""
|
|
logger.debug(
|
|
"[STOREFRONT] shop_account_dashboard_page REACHED",
|
|
extra={
|
|
"path": request.url.path,
|
|
"store": getattr(request.state, "store", "NOT SET"),
|
|
"context": getattr(request.state, "context_type", "NOT SET"),
|
|
},
|
|
)
|
|
|
|
# Collect dashboard cards from enabled modules via widget protocol
|
|
from app.modules.core.services.widget_aggregator import widget_aggregator
|
|
|
|
store = getattr(request.state, "store", None)
|
|
platform = getattr(request.state, "platform", None)
|
|
dashboard_cards = []
|
|
if store and platform:
|
|
dashboard_cards = widget_aggregator.get_storefront_dashboard_cards(
|
|
db,
|
|
store_id=store.id,
|
|
customer_id=current_customer.id,
|
|
platform_id=platform.id,
|
|
)
|
|
|
|
return templates.TemplateResponse(
|
|
"customers/storefront/dashboard.html",
|
|
get_storefront_context(
|
|
request, db=db, user=current_customer, dashboard_cards=dashboard_cards
|
|
),
|
|
)
|
|
|
|
|
|
@router.get("/account/profile", response_class=HTMLResponse, include_in_schema=False)
|
|
async def shop_profile_page(
|
|
request: Request,
|
|
current_customer: Customer = Depends(get_current_customer_from_cookie_or_header),
|
|
db: Session = Depends(get_db),
|
|
):
|
|
"""
|
|
Render customer profile page.
|
|
Edit personal information and preferences.
|
|
Requires customer authentication.
|
|
"""
|
|
logger.debug(
|
|
"[STOREFRONT] shop_profile_page REACHED",
|
|
extra={
|
|
"path": request.url.path,
|
|
"store": getattr(request.state, "store", "NOT SET"),
|
|
"context": getattr(request.state, "context_type", "NOT SET"),
|
|
},
|
|
)
|
|
|
|
return templates.TemplateResponse(
|
|
"customers/storefront/profile.html",
|
|
get_storefront_context(request, db=db, user=current_customer),
|
|
)
|
|
|
|
|
|
@router.get(
|
|
"/account/addresses", response_class=HTMLResponse, include_in_schema=False
|
|
)
|
|
async def shop_addresses_page(
|
|
request: Request,
|
|
current_customer: Customer = Depends(get_current_customer_from_cookie_or_header),
|
|
db: Session = Depends(get_db),
|
|
):
|
|
"""
|
|
Render customer addresses management page.
|
|
Manage shipping and billing addresses.
|
|
Requires customer authentication.
|
|
"""
|
|
logger.debug(
|
|
"[STOREFRONT] shop_addresses_page REACHED",
|
|
extra={
|
|
"path": request.url.path,
|
|
"store": getattr(request.state, "store", "NOT SET"),
|
|
"context": getattr(request.state, "context_type", "NOT SET"),
|
|
},
|
|
)
|
|
|
|
return templates.TemplateResponse(
|
|
"customers/storefront/addresses.html",
|
|
get_storefront_context(request, db=db, user=current_customer),
|
|
)
|
|
|
|
|
|
@router.get("/account/settings", response_class=HTMLResponse, include_in_schema=False)
|
|
async def shop_settings_page(
|
|
request: Request,
|
|
current_customer: Customer = Depends(get_current_customer_from_cookie_or_header),
|
|
db: Session = Depends(get_db),
|
|
):
|
|
"""
|
|
Render customer account settings page.
|
|
Configure notifications, privacy, and preferences.
|
|
Requires customer authentication.
|
|
"""
|
|
logger.debug(
|
|
"[STOREFRONT] shop_settings_page REACHED",
|
|
extra={
|
|
"path": request.url.path,
|
|
"store": getattr(request.state, "store", "NOT SET"),
|
|
"context": getattr(request.state, "context_type", "NOT SET"),
|
|
},
|
|
)
|
|
|
|
return templates.TemplateResponse(
|
|
"customers/storefront/settings.html",
|
|
get_storefront_context(request, db=db, user=current_customer),
|
|
)
|