fix(loyalty): resolve critical production readiness issues
Some checks failed
Some checks failed
- Add pessimistic locking (SELECT FOR UPDATE) on card write operations to prevent race conditions in stamp_service and points_service - Replace 16 console.log/error/warn calls with LogConfig.createLogger() in 3 storefront JS files (dashboard, history, enroll) - Delete all stale lu.json locale files across 8 modules (lb is the correct ISO 639-1 code for Luxembourgish) - Update architecture rules and docs to reference lb.json not lu.json - Add production-readiness.md report for loyalty module Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -141,7 +141,7 @@ module_rules:
|
|||||||
en.json
|
en.json
|
||||||
de.json
|
de.json
|
||||||
fr.json
|
fr.json
|
||||||
lu.json
|
lb.json
|
||||||
|
|
||||||
Translation keys are namespaced as {module}.key_name
|
Translation keys are namespaced as {module}.key_name
|
||||||
pattern:
|
pattern:
|
||||||
@@ -269,14 +269,14 @@ module_rules:
|
|||||||
Module locales/ directory should have translation files for
|
Module locales/ directory should have translation files for
|
||||||
all supported languages to ensure consistent i18n.
|
all supported languages to ensure consistent i18n.
|
||||||
|
|
||||||
Supported languages: en, de, fr, lu
|
Supported languages: en, de, fr, lb
|
||||||
|
|
||||||
Structure:
|
Structure:
|
||||||
app/modules/<code>/locales/
|
app/modules/<code>/locales/
|
||||||
├── en.json
|
├── en.json
|
||||||
├── de.json
|
├── de.json
|
||||||
├── fr.json
|
├── fr.json
|
||||||
└── lu.json
|
└── lb.json
|
||||||
|
|
||||||
Missing translations will fall back to English, but it's
|
Missing translations will fall back to English, but it's
|
||||||
better to have all languages covered.
|
better to have all languages covered.
|
||||||
@@ -286,7 +286,7 @@ module_rules:
|
|||||||
- "en.json"
|
- "en.json"
|
||||||
- "de.json"
|
- "de.json"
|
||||||
- "fr.json"
|
- "fr.json"
|
||||||
- "lu.json"
|
- "lb.json"
|
||||||
|
|
||||||
- id: "MOD-007"
|
- id: "MOD-007"
|
||||||
name: "Module definition must match directory structure"
|
name: "Module definition must match directory structure"
|
||||||
|
|||||||
@@ -1,17 +0,0 @@
|
|||||||
{
|
|
||||||
"analytics": {
|
|
||||||
"page_title": "Analysen",
|
|
||||||
"dashboard_title": "Analyse-Dashboard",
|
|
||||||
"dashboard_subtitle": "Kuckt Är Buttek Leeschtungsmetriken an Abléck",
|
|
||||||
"period_7d": "Lescht 7 Deeg",
|
|
||||||
"period_30d": "Lescht 30 Deeg",
|
|
||||||
"period_90d": "Lescht 90 Deeg",
|
|
||||||
"period_1y": "Lescht Joer",
|
|
||||||
"imports_count": "Importer",
|
|
||||||
"products_added": "Produkter bäigesat",
|
|
||||||
"inventory_locations": "Lagerplazen",
|
|
||||||
"data_since": "Donnéeë vun",
|
|
||||||
"loading": "Analysen ginn gelueden...",
|
|
||||||
"error_loading": "Analysedonnéeën konnten net geluede ginn"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
{}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
{}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
{}
|
|
||||||
134
app/modules/loyalty/docs/production-readiness.md
Normal file
134
app/modules/loyalty/docs/production-readiness.md
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
# Loyalty Module — Production Readiness Report
|
||||||
|
|
||||||
|
**Date:** 2026-03-13
|
||||||
|
**Tests:** 213 passing | **Languages:** en, fr, de, lb (complete)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Overall Assessment: READY WITH RESERVATIONS
|
||||||
|
|
||||||
|
The module has strong fundamentals — clean architecture, comprehensive tests, proper exception handling, full i18n (4 languages), wallet integration, and anti-fraud controls. All critical issues have been resolved.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## CRITICAL — Must Fix Before Launch
|
||||||
|
|
||||||
|
### ~~1. Race Conditions on Card Operations~~ — RESOLVED
|
||||||
|
Added `get_card_for_update()` method to `card_service.py` using `SELECT ... FOR UPDATE`. All 7 write methods in `stamp_service.py` (3) and `points_service.py` (4) now re-fetch the card with a row-level lock before modifying fields.
|
||||||
|
|
||||||
|
### ~~2. Stale Locale File `lu.json`~~ — RESOLVED
|
||||||
|
All `lu.json` files deleted across the codebase. `lb` is the canonical Luxembourgish code (ISO 639-1). Architecture rules and docs updated.
|
||||||
|
|
||||||
|
### ~~3. Console.log Statements in Storefront JS~~ — RESOLVED
|
||||||
|
All 16 `console.log/error/warn` calls replaced with `LogConfig.createLogger()` in 3 storefront JS files (dashboard, history, enroll).
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## HIGH — Should Fix Before Launch
|
||||||
|
|
||||||
|
### 4. Rate Limiting on Public Endpoints
|
||||||
|
**Status:** Rate limiter middleware exists (`middleware/rate_limiter.py`) but is NOT applied to loyalty endpoints.
|
||||||
|
|
||||||
|
**Vulnerable endpoints:**
|
||||||
|
- `POST /loyalty/enroll` — public, no auth required
|
||||||
|
- `POST /pins/{pin_id}/verify` — PIN lockout helps but doesn't prevent request flooding
|
||||||
|
|
||||||
|
**Fix:** Apply `@rate_limit()` decorator to public loyalty routes.
|
||||||
|
|
||||||
|
**Effort:** 1 hour
|
||||||
|
|
||||||
|
### 5. T&C Strategy Unresolved
|
||||||
|
**Location:** `storefront/enroll.html` line 140 — TODO comment noting current approach (small text field on program model) won't scale for full legal T&C.
|
||||||
|
|
||||||
|
**Options:**
|
||||||
|
- (A) Leverage CMS module to host T&C pages
|
||||||
|
- (B) Create dedicated T&C page within loyalty module
|
||||||
|
- (C) Accept current approach for MVP, plan post-launch
|
||||||
|
|
||||||
|
**Effort:** Depends on option chosen
|
||||||
|
|
||||||
|
### 6. Accessibility (ARIA Labels)
|
||||||
|
**Status:** Zero `aria-label` attributes across all loyalty templates. Icon-only buttons, modals, and form inputs lack screen reader support.
|
||||||
|
|
||||||
|
**Priority areas:**
|
||||||
|
- PIN entry modal (terminal) — security-critical, needs aria-modal
|
||||||
|
- Icon-only action buttons (view, edit, delete in tables)
|
||||||
|
- Modal dialogs (delete confirmation, terms, barcode)
|
||||||
|
|
||||||
|
**Effort:** 2-3 hours
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## MEDIUM — Address Post-Launch
|
||||||
|
|
||||||
|
### 7. Point Expiration Task Performance
|
||||||
|
`point_expiration.py` processes cards one-by-one. With large card volumes, this could cause long-running DB locks.
|
||||||
|
|
||||||
|
**Fix:** Batch processing with `LIMIT` and chunked commits.
|
||||||
|
|
||||||
|
### 8. Wallet Sync Retry Logic
|
||||||
|
Wallet sync task doesn't retry failed individual cards. A transient API error skips the card until next hourly run.
|
||||||
|
|
||||||
|
**Fix:** Add retry with exponential backoff per card.
|
||||||
|
|
||||||
|
### 9. Audit Logging Enhancement
|
||||||
|
Current: LoyaltyTransaction captures operations but not all operations create transaction records (e.g., program config changes, PIN management).
|
||||||
|
|
||||||
|
**Fix:** Add structured audit logging for admin/management operations.
|
||||||
|
|
||||||
|
### 10. Test Coverage Metrics
|
||||||
|
Tests pass (213) but no coverage measurement. Unknown blind spots.
|
||||||
|
|
||||||
|
**Fix:** Add `pytest-cov` and target >80% line coverage.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## LOW / Nice-to-Have
|
||||||
|
|
||||||
|
### 11. Field-Level Validation Errors
|
||||||
|
Current: Generic toast notifications for form errors. No field-specific error highlighting.
|
||||||
|
|
||||||
|
### 12. Session Storage in Enroll-Success
|
||||||
|
`enroll-success.html` stores wallet URLs in sessionStorage. Fragile if user navigates away before page loads.
|
||||||
|
|
||||||
|
### 13. Tier System (Future)
|
||||||
|
Model has `tier_config` field (Bronze/Silver/Gold) but no business logic implementation. Properly marked as future.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## What's Already Solid
|
||||||
|
|
||||||
|
| Area | Status | Details |
|
||||||
|
|------|--------|---------|
|
||||||
|
| Architecture | Excellent | Clean service layer, proper separation of concerns |
|
||||||
|
| Exception Handling | Excellent | 30+ custom exceptions, no bare except clauses |
|
||||||
|
| Authentication | Strong | All routes protected, role-based access |
|
||||||
|
| PIN Security | Strong | bcrypt hashing, lockout mechanism, timing-safe comparison |
|
||||||
|
| Wallet Security | Strong | JWT/PKCS#7 signing, env-based secrets, token validation |
|
||||||
|
| SQL Injection | Safe | SQLAlchemy ORM throughout, no raw SQL |
|
||||||
|
| Input Validation | Good | Pydantic schemas on all endpoints |
|
||||||
|
| Database Design | Excellent | Proper indexes, FKs, cascades, composite indexes |
|
||||||
|
| Tests | Good | 213 tests, unit + integration coverage |
|
||||||
|
| i18n | Complete | 4 languages (en/fr/de/lb), ~300 keys each |
|
||||||
|
| Dark Mode | Excellent | Consistent dark: variants on all elements |
|
||||||
|
| Responsive Design | Excellent | Mobile-first with proper breakpoints |
|
||||||
|
| Empty States | Excellent | All list views have proper empty states |
|
||||||
|
| Loading States | Excellent | Spinners and loading indicators everywhere |
|
||||||
|
| Confirmation Dialogs | Good | All destructive actions have modals |
|
||||||
|
| Pagination | Excellent | Full pagination on all list views |
|
||||||
|
| Documentation | Good | 5 docs (business logic, data model, user journeys, UI design, analysis) |
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Pre-Launch Checklist
|
||||||
|
|
||||||
|
- [x] ~~Fix race conditions with `with_for_update()` on card operations~~ — DONE
|
||||||
|
- [x] ~~Resolve `lu.json` vs `lb.json` locale situation~~ — DONE
|
||||||
|
- [x] ~~Replace console.log with LogConfig in 3 storefront JS files~~ — DONE
|
||||||
|
- [ ] Apply rate limiting to public enrollment endpoint
|
||||||
|
- [ ] Decide on T&C strategy (or accept current for MVP)
|
||||||
|
- [ ] Add basic ARIA labels to modals and icon-only buttons
|
||||||
|
- [ ] Manual QA: Walk through all 8 user journeys in `docs/user-journeys.md`
|
||||||
|
- [ ] Verify Google Wallet configuration in production environment
|
||||||
|
- [ ] Configure Apple Wallet (if needed for launch) or gracefully disable
|
||||||
|
- [ ] Review `LOYALTY_*` environment variables in production config
|
||||||
@@ -1,72 +0,0 @@
|
|||||||
{
|
|
||||||
"loyalty": {
|
|
||||||
"module": {
|
|
||||||
"name": "Treieprogrammer",
|
|
||||||
"description": "Stempel- a punktebaséiert Treieprogrammer mat Wallet-Integratioun"
|
|
||||||
},
|
|
||||||
"program": {
|
|
||||||
"title": "Treieprogramm",
|
|
||||||
"create": "Programm erstellen",
|
|
||||||
"edit": "Programm beaarbechten",
|
|
||||||
"activate": "Aktivéieren",
|
|
||||||
"deactivate": "Deaktivéieren",
|
|
||||||
"type": {
|
|
||||||
"stamps": "Stempelen",
|
|
||||||
"points": "Punkten",
|
|
||||||
"hybrid": "Hybrid"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"card": {
|
|
||||||
"title": "Treiekaart",
|
|
||||||
"number": "Kaartennummer",
|
|
||||||
"qr_code": "QR-Code",
|
|
||||||
"enroll": "Client umellen",
|
|
||||||
"deactivate": "Kaart deaktivéieren"
|
|
||||||
},
|
|
||||||
"stamp": {
|
|
||||||
"title": "Stempelen",
|
|
||||||
"add": "Stempel dobäisetzen",
|
|
||||||
"redeem": "Belounung aléisen",
|
|
||||||
"count": "{current} vun {target}",
|
|
||||||
"until_reward": "Nach {count} bis zur Belounung"
|
|
||||||
},
|
|
||||||
"points": {
|
|
||||||
"title": "Punkten",
|
|
||||||
"earn": "Punkten sammelen",
|
|
||||||
"redeem": "Punkten aléisen",
|
|
||||||
"balance": "{count} Punkten",
|
|
||||||
"per_euro": "{points} Punkten pro Euro"
|
|
||||||
},
|
|
||||||
"pin": {
|
|
||||||
"title": "Mataarbechter-PINen",
|
|
||||||
"create": "PIN erstellen",
|
|
||||||
"edit": "PIN beaarbechten",
|
|
||||||
"unlock": "PIN entspären",
|
|
||||||
"locked": "PIN gespaart bis {time}"
|
|
||||||
},
|
|
||||||
"wallet": {
|
|
||||||
"google": "Bäi Google Wallet bäisetzen",
|
|
||||||
"apple": "Bäi Apple Wallet bäisetzen"
|
|
||||||
},
|
|
||||||
"stats": {
|
|
||||||
"title": "Statistiken",
|
|
||||||
"total_cards": "Total Kaarten",
|
|
||||||
"active_cards": "Aktiv Kaarten",
|
|
||||||
"stamps_issued": "Ausgestallte Stempelen",
|
|
||||||
"rewards_redeemed": "Agelëst Belounungen"
|
|
||||||
},
|
|
||||||
"errors": {
|
|
||||||
"program_not_found": "Treieprogramm net fonnt",
|
|
||||||
"program_inactive": "Treieprogramm ass net aktiv",
|
|
||||||
"card_not_found": "Treiekaart net fonnt",
|
|
||||||
"card_inactive": "Treiekaart ass net aktiv",
|
|
||||||
"cooldown": "Waart w.e.g. {minutes} Minutten virum nächste Stempel",
|
|
||||||
"daily_limit": "Dageslimit vun {limit} Stempelen erreecht",
|
|
||||||
"insufficient_stamps": "Brauch {required} Stempelen, hutt {current}",
|
|
||||||
"insufficient_points": "Brauch {required} Punkten, hutt {current}",
|
|
||||||
"pin_required": "Mataarbechter-PIN erfuerdert",
|
|
||||||
"pin_invalid": "Ongültege PIN",
|
|
||||||
"pin_locked": "PIN gespaart wéinst ze vill Feelverséich"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -52,6 +52,19 @@ class CardService:
|
|||||||
.first()
|
.first()
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def get_card_for_update(self, db: Session, card_id: int) -> LoyaltyCard | None:
|
||||||
|
"""Get a loyalty card by ID with a row-level lock (SELECT ... FOR UPDATE).
|
||||||
|
|
||||||
|
Note: Does not use joinedload to avoid LEFT OUTER JOIN which is
|
||||||
|
incompatible with FOR UPDATE in PostgreSQL.
|
||||||
|
"""
|
||||||
|
return (
|
||||||
|
db.query(LoyaltyCard)
|
||||||
|
.filter(LoyaltyCard.id == card_id)
|
||||||
|
.with_for_update()
|
||||||
|
.first()
|
||||||
|
)
|
||||||
|
|
||||||
def get_card_by_qr_code(self, db: Session, qr_code: str) -> LoyaltyCard | None:
|
def get_card_by_qr_code(self, db: Session, qr_code: str) -> LoyaltyCard | None:
|
||||||
"""Get a loyalty card by QR code data."""
|
"""Get a loyalty card by QR code data."""
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -140,6 +140,9 @@ class PointsService:
|
|||||||
"total_points_earned": card.total_points_earned,
|
"total_points_earned": card.total_points_earned,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Re-fetch with row lock to prevent concurrent modification
|
||||||
|
card = card_service.get_card_for_update(db, card.id)
|
||||||
|
|
||||||
# Add points
|
# Add points
|
||||||
now = datetime.now(UTC)
|
now = datetime.now(UTC)
|
||||||
card.points_balance += points_earned
|
card.points_balance += points_earned
|
||||||
@@ -271,6 +274,9 @@ class PointsService:
|
|||||||
raise StaffPinRequiredException()
|
raise StaffPinRequiredException()
|
||||||
verified_pin = pin_service.verify_pin(db, program.id, staff_pin, store_id=store_id)
|
verified_pin = pin_service.verify_pin(db, program.id, staff_pin, store_id=store_id)
|
||||||
|
|
||||||
|
# Re-fetch with row lock to prevent concurrent modification
|
||||||
|
card = card_service.get_card_for_update(db, card.id)
|
||||||
|
|
||||||
# Redeem points
|
# Redeem points
|
||||||
now = datetime.now(UTC)
|
now = datetime.now(UTC)
|
||||||
card.points_balance -= points_required
|
card.points_balance -= points_required
|
||||||
@@ -419,6 +425,9 @@ class PointsService:
|
|||||||
"points_balance": card.points_balance,
|
"points_balance": card.points_balance,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Re-fetch with row lock to prevent concurrent modification
|
||||||
|
card = card_service.get_card_for_update(db, card.id)
|
||||||
|
|
||||||
# Void the points (can reduce balance below what was earned)
|
# Void the points (can reduce balance below what was earned)
|
||||||
now = datetime.now(UTC)
|
now = datetime.now(UTC)
|
||||||
actual_voided = min(points_to_void, card.points_balance)
|
actual_voided = min(points_to_void, card.points_balance)
|
||||||
@@ -503,6 +512,9 @@ class PointsService:
|
|||||||
if program.require_staff_pin and staff_pin and store_id:
|
if program.require_staff_pin and staff_pin and store_id:
|
||||||
verified_pin = pin_service.verify_pin(db, program.id, staff_pin, store_id=store_id)
|
verified_pin = pin_service.verify_pin(db, program.id, staff_pin, store_id=store_id)
|
||||||
|
|
||||||
|
# Re-fetch with row lock to prevent concurrent modification
|
||||||
|
card = card_service.get_card_for_update(db, card.id)
|
||||||
|
|
||||||
# Apply adjustment
|
# Apply adjustment
|
||||||
now = datetime.now(UTC)
|
now = datetime.now(UTC)
|
||||||
card.points_balance += points_delta
|
card.points_balance += points_delta
|
||||||
|
|||||||
@@ -125,6 +125,9 @@ class StampService:
|
|||||||
if stamps_today >= program.max_daily_stamps:
|
if stamps_today >= program.max_daily_stamps:
|
||||||
raise DailyStampLimitException(program.max_daily_stamps, stamps_today)
|
raise DailyStampLimitException(program.max_daily_stamps, stamps_today)
|
||||||
|
|
||||||
|
# Re-fetch with row lock to prevent concurrent modification
|
||||||
|
card = card_service.get_card_for_update(db, card.id)
|
||||||
|
|
||||||
# Add the stamp
|
# Add the stamp
|
||||||
card.stamp_count += 1
|
card.stamp_count += 1
|
||||||
card.total_stamps_earned += 1
|
card.total_stamps_earned += 1
|
||||||
@@ -249,6 +252,9 @@ class StampService:
|
|||||||
raise StaffPinRequiredException()
|
raise StaffPinRequiredException()
|
||||||
verified_pin = pin_service.verify_pin(db, program.id, staff_pin, store_id=store_id)
|
verified_pin = pin_service.verify_pin(db, program.id, staff_pin, store_id=store_id)
|
||||||
|
|
||||||
|
# Re-fetch with row lock to prevent concurrent modification
|
||||||
|
card = card_service.get_card_for_update(db, card.id)
|
||||||
|
|
||||||
# Redeem stamps
|
# Redeem stamps
|
||||||
now = datetime.now(UTC)
|
now = datetime.now(UTC)
|
||||||
stamps_redeemed = program.stamps_target
|
stamps_redeemed = program.stamps_target
|
||||||
@@ -383,6 +389,9 @@ class StampService:
|
|||||||
"stamp_count": card.stamp_count,
|
"stamp_count": card.stamp_count,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Re-fetch with row lock to prevent concurrent modification
|
||||||
|
card = card_service.get_card_for_update(db, card.id)
|
||||||
|
|
||||||
# Void the stamps (can reduce balance below what was earned)
|
# Void the stamps (can reduce balance below what was earned)
|
||||||
now = datetime.now(UTC)
|
now = datetime.now(UTC)
|
||||||
actual_voided = min(stamps_to_void, card.stamp_count)
|
actual_voided = min(stamps_to_void, card.stamp_count)
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
// app/modules/loyalty/static/storefront/js/loyalty-dashboard.js
|
// app/modules/loyalty/static/storefront/js/loyalty-dashboard.js
|
||||||
// Customer loyalty dashboard
|
// Customer loyalty dashboard
|
||||||
|
const loyaltyDashboardLog = window.LogConfig.loggers.loyaltyDashboard || window.LogConfig.createLogger('loyaltyDashboard');
|
||||||
|
|
||||||
function customerLoyaltyDashboard() {
|
function customerLoyaltyDashboard() {
|
||||||
return {
|
return {
|
||||||
@@ -20,7 +21,7 @@ function customerLoyaltyDashboard() {
|
|||||||
showBarcode: false,
|
showBarcode: false,
|
||||||
|
|
||||||
async init() {
|
async init() {
|
||||||
console.log('Customer loyalty dashboard initializing...');
|
loyaltyDashboardLog.info('Customer loyalty dashboard initializing...');
|
||||||
await this.loadData();
|
await this.loadData();
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -32,7 +33,7 @@ function customerLoyaltyDashboard() {
|
|||||||
this.loadTransactions()
|
this.loadTransactions()
|
||||||
]);
|
]);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Failed to load loyalty data:', error);
|
loyaltyDashboardLog.error('Failed to load loyalty data:', error);
|
||||||
} finally {
|
} finally {
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
}
|
}
|
||||||
@@ -47,11 +48,11 @@ function customerLoyaltyDashboard() {
|
|||||||
this.rewards = response.program?.points_rewards || [];
|
this.rewards = response.program?.points_rewards || [];
|
||||||
this.locations = response.locations || [];
|
this.locations = response.locations || [];
|
||||||
this.walletUrls = response.wallet_urls || { google_wallet_url: null, apple_wallet_url: null };
|
this.walletUrls = response.wallet_urls || { google_wallet_url: null, apple_wallet_url: null };
|
||||||
console.log('Loyalty card loaded:', this.card?.card_number);
|
loyaltyDashboardLog.info('Loyalty card loaded:', this.card?.card_number);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error.status === 404) {
|
if (error.status === 404) {
|
||||||
console.log('No loyalty card found');
|
loyaltyDashboardLog.info('No loyalty card found');
|
||||||
this.card = null;
|
this.card = null;
|
||||||
} else {
|
} else {
|
||||||
throw error;
|
throw error;
|
||||||
@@ -66,7 +67,7 @@ function customerLoyaltyDashboard() {
|
|||||||
this.transactions = response.transactions;
|
this.transactions = response.transactions;
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.warn('Failed to load transactions:', error.message);
|
loyaltyDashboardLog.warn('Failed to load transactions:', error.message);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
// app/modules/loyalty/static/storefront/js/loyalty-enroll.js
|
// app/modules/loyalty/static/storefront/js/loyalty-enroll.js
|
||||||
// Self-service loyalty enrollment
|
// Self-service loyalty enrollment
|
||||||
|
const loyaltyEnrollLog = window.LogConfig.loggers.loyaltyEnroll || window.LogConfig.createLogger('loyaltyEnroll');
|
||||||
|
|
||||||
function customerLoyaltyEnroll() {
|
function customerLoyaltyEnroll() {
|
||||||
return {
|
return {
|
||||||
@@ -28,7 +29,7 @@ function customerLoyaltyEnroll() {
|
|||||||
showTerms: false,
|
showTerms: false,
|
||||||
|
|
||||||
async init() {
|
async init() {
|
||||||
console.log('Customer loyalty enroll initializing...');
|
loyaltyEnrollLog.info('Customer loyalty enroll initializing...');
|
||||||
await this.loadProgram();
|
await this.loadProgram();
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -38,14 +39,14 @@ function customerLoyaltyEnroll() {
|
|||||||
const response = await apiClient.get('/storefront/loyalty/program');
|
const response = await apiClient.get('/storefront/loyalty/program');
|
||||||
if (response) {
|
if (response) {
|
||||||
this.program = response;
|
this.program = response;
|
||||||
console.log('Program loaded:', this.program.display_name);
|
loyaltyEnrollLog.info('Program loaded:', this.program.display_name);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error.status === 404) {
|
if (error.status === 404) {
|
||||||
console.log('No loyalty program available');
|
loyaltyEnrollLog.info('No loyalty program available');
|
||||||
this.program = null;
|
this.program = null;
|
||||||
} else {
|
} else {
|
||||||
console.error('Failed to load program:', error);
|
loyaltyEnrollLog.error('Failed to load program:', error);
|
||||||
this.error = I18n.t('loyalty.enrollment.errors.load_failed');
|
this.error = I18n.t('loyalty.enrollment.errors.load_failed');
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
@@ -73,7 +74,7 @@ function customerLoyaltyEnroll() {
|
|||||||
|
|
||||||
if (response) {
|
if (response) {
|
||||||
const cardNumber = response.card?.card_number || response.card_number;
|
const cardNumber = response.card?.card_number || response.card_number;
|
||||||
console.log('Enrollment successful:', cardNumber);
|
loyaltyEnrollLog.info('Enrollment successful:', cardNumber);
|
||||||
|
|
||||||
// Store wallet URLs for the success page (no auth needed)
|
// Store wallet URLs for the success page (no auth needed)
|
||||||
if (response.wallet_urls) {
|
if (response.wallet_urls) {
|
||||||
@@ -87,7 +88,7 @@ function customerLoyaltyEnroll() {
|
|||||||
window.location.href = successUrl;
|
window.location.href = successUrl;
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Enrollment failed:', error);
|
loyaltyEnrollLog.error('Enrollment failed:', error);
|
||||||
if (error.message?.includes('already')) {
|
if (error.message?.includes('already')) {
|
||||||
this.error = I18n.t('loyalty.enrollment.errors.email_exists');
|
this.error = I18n.t('loyalty.enrollment.errors.email_exists');
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
// app/modules/loyalty/static/storefront/js/loyalty-history.js
|
// app/modules/loyalty/static/storefront/js/loyalty-history.js
|
||||||
// Customer loyalty transaction history
|
// Customer loyalty transaction history
|
||||||
|
const loyaltyHistoryLog = window.LogConfig.loggers.loyaltyHistory || window.LogConfig.createLogger('loyaltyHistory');
|
||||||
|
|
||||||
function customerLoyaltyHistory() {
|
function customerLoyaltyHistory() {
|
||||||
return {
|
return {
|
||||||
@@ -21,7 +22,7 @@ function customerLoyaltyHistory() {
|
|||||||
loading: false,
|
loading: false,
|
||||||
|
|
||||||
async init() {
|
async init() {
|
||||||
console.log('Customer loyalty history initializing...');
|
loyaltyHistoryLog.info('Customer loyalty history initializing...');
|
||||||
await this.loadData();
|
await this.loadData();
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -33,7 +34,7 @@ function customerLoyaltyHistory() {
|
|||||||
this.loadTransactions()
|
this.loadTransactions()
|
||||||
]);
|
]);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Failed to load history:', error);
|
loyaltyHistoryLog.error('Failed to load history:', error);
|
||||||
} finally {
|
} finally {
|
||||||
this.loading = false;
|
this.loading = false;
|
||||||
}
|
}
|
||||||
@@ -46,7 +47,7 @@ function customerLoyaltyHistory() {
|
|||||||
this.card = response.card;
|
this.card = response.card;
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.warn('Failed to load card:', error.message);
|
loyaltyHistoryLog.warn('Failed to load card:', error.message);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -61,10 +62,10 @@ function customerLoyaltyHistory() {
|
|||||||
this.transactions = response.transactions || [];
|
this.transactions = response.transactions || [];
|
||||||
this.pagination.total = response.total || 0;
|
this.pagination.total = response.total || 0;
|
||||||
this.pagination.pages = Math.ceil(this.pagination.total / this.pagination.per_page);
|
this.pagination.pages = Math.ceil(this.pagination.total / this.pagination.per_page);
|
||||||
console.log(`Loaded ${this.transactions.length} transactions`);
|
loyaltyHistoryLog.info(`Loaded ${this.transactions.length} transactions`);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Failed to load transactions:', error);
|
loyaltyHistoryLog.error('Failed to load transactions:', error);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
{}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
{}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
{}
|
|
||||||
@@ -163,7 +163,7 @@ app/modules/analytics/
|
|||||||
│ ├── en.json
|
│ ├── en.json
|
||||||
│ ├── de.json
|
│ ├── de.json
|
||||||
│ ├── fr.json
|
│ ├── fr.json
|
||||||
│ └── lu.json
|
│ └── lb.json
|
||||||
├── tasks/ # Auto-discovered by Celery
|
├── tasks/ # Auto-discovered by Celery
|
||||||
│ ├── __init__.py # REQUIRED for Celery discovery
|
│ ├── __init__.py # REQUIRED for Celery discovery
|
||||||
│ └── reports.py
|
│ └── reports.py
|
||||||
|
|||||||
@@ -164,7 +164,7 @@ app/modules/mymodule/
|
|||||||
│ ├── en.json
|
│ ├── en.json
|
||||||
│ ├── de.json
|
│ ├── de.json
|
||||||
│ ├── fr.json
|
│ ├── fr.json
|
||||||
│ └── lu.json
|
│ └── lb.json
|
||||||
│
|
│
|
||||||
├── tasks/ # Celery tasks (auto-discovered)
|
├── tasks/ # Celery tasks (auto-discovered)
|
||||||
│ ├── __init__.py # REQUIRED for discovery
|
│ ├── __init__.py # REQUIRED for discovery
|
||||||
@@ -539,7 +539,7 @@ python scripts/validate/validate_architecture.py
|
|||||||
- [ ] Create `exceptions.py`
|
- [ ] Create `exceptions.py`
|
||||||
- [ ] Create routes with `router` variable
|
- [ ] Create routes with `router` variable
|
||||||
- [ ] Create templates with namespace prefix
|
- [ ] Create templates with namespace prefix
|
||||||
- [ ] Create locales (en, de, fr, lu)
|
- [ ] Create locales (en, de, fr, lb)
|
||||||
- [ ] Create `config.py` if module needs environment settings (optional)
|
- [ ] Create `config.py` if module needs environment settings (optional)
|
||||||
- [ ] Create `migrations/versions/` with `__init__.py` files if module has database tables
|
- [ ] Create `migrations/versions/` with `__init__.py` files if module has database tables
|
||||||
- [ ] Create `docs/index.md` with module overview (see [Module Documentation](module-documentation.md))
|
- [ ] Create `docs/index.md` with module overview (see [Module Documentation](module-documentation.md))
|
||||||
|
|||||||
Reference in New Issue
Block a user