refactor: complete module-driven architecture migration

This commit completes the migration to a fully module-driven architecture:

## Models Migration
- Moved all domain models from models/database/ to their respective modules:
  - tenancy: User, Admin, Vendor, Company, Platform, VendorDomain, etc.
  - cms: MediaFile, VendorTheme
  - messaging: Email, VendorEmailSettings, VendorEmailTemplate
  - core: AdminMenuConfig
- models/database/ now only contains Base and TimestampMixin (infrastructure)

## Schemas Migration
- Moved all domain schemas from models/schema/ to their respective modules:
  - tenancy: company, vendor, admin, team, vendor_domain
  - cms: media, image, vendor_theme
  - messaging: email
- models/schema/ now only contains base.py and auth.py (infrastructure)

## Routes Migration
- Moved admin routes from app/api/v1/admin/ to modules:
  - menu_config.py -> core module
  - modules.py -> tenancy module
  - module_config.py -> tenancy module
- app/api/v1/admin/ now only aggregates auto-discovered module routes

## Menu System
- Implemented module-driven menu system with MenuDiscoveryService
- Extended FrontendType enum: PLATFORM, ADMIN, VENDOR, STOREFRONT
- Added MenuItemDefinition and MenuSectionDefinition dataclasses
- Each module now defines its own menu items in definition.py
- MenuService integrates with MenuDiscoveryService for template rendering

## Documentation
- Updated docs/architecture/models-structure.md
- Updated docs/architecture/menu-management.md
- Updated architecture validation rules for new exceptions

## Architecture Validation
- Updated MOD-019 rule to allow base.py in models/schema/
- Created core module exceptions.py and schemas/ directory
- All validation errors resolved (only warnings remain)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
2026-02-01 21:02:56 +01:00
parent 09d7d282c6
commit d7a0ff8818
307 changed files with 5536 additions and 3826 deletions

View File

@@ -1,122 +1,61 @@
{
"title": "Marktplatz-Integration",
"description": "Letzshop Produkt- und Bestellsynchronisation",
"products": {
"title": "Marktplatz-Produkte",
"subtitle": "Von Marktplätzen importierte Produkte",
"empty": "Keine Produkte gefunden",
"empty_search": "Keine Produkte entsprechen Ihrer Suche",
"import": "Produkte importieren"
},
"import": {
"title": "Produkte importieren",
"subtitle": "Produkte aus Marktplatz-Feeds importieren",
"source_url": "Feed-URL",
"source_url_help": "URL zum Marktplatz-CSV-Feed",
"marketplace": "Marktplatz",
"language": "Sprache",
"language_help": "Sprache für Produktübersetzungen",
"batch_size": "Batch-Größe",
"marketplace": {
"title": "Marktplatz",
"import": "Importieren",
"export": "Exportieren",
"sync": "Synchronisieren",
"source": "Quelle",
"source_url": "Quell-URL",
"import_products": "Produkte importieren",
"start_import": "Import starten",
"cancel": "Abbrechen"
},
"import_jobs": {
"title": "Import-Verlauf",
"subtitle": "Vergangene und aktuelle Import-Jobs",
"empty": "Keine Import-Jobs",
"job_id": "Job-ID",
"marketplace": "Marktplatz",
"vendor": "Verkäufer",
"status": "Status",
"imported": "Importiert",
"updated": "Aktualisiert",
"errors": "Fehler",
"created": "Erstellt",
"completed": "Abgeschlossen",
"statuses": {
"pending": "Ausstehend",
"processing": "In Bearbeitung",
"completed": "Abgeschlossen",
"completed_with_errors": "Mit Fehlern abgeschlossen",
"failed": "Fehlgeschlagen"
}
"importing": "Importiere...",
"import_complete": "Import abgeschlossen",
"import_failed": "Import fehlgeschlagen",
"import_history": "Import-Verlauf",
"job_id": "Auftrags-ID",
"started_at": "Gestartet um",
"completed_at": "Abgeschlossen um",
"duration": "Dauer",
"imported_count": "Importiert",
"error_count": "Fehler",
"total_processed": "Gesamt verarbeitet",
"progress": "Fortschritt",
"no_import_jobs": "Noch keine Imports",
"start_first_import": "Starten Sie Ihren ersten Import mit dem Formular oben"
},
"letzshop": {
"title": "Letzshop-Integration",
"subtitle": "Letzshop-Verbindung und Synchronisation verwalten",
"credentials": {
"title": "API-Anmeldedaten",
"api_key": "API-Schlüssel",
"api_key_help": "Ihr Letzshop API-Schlüssel",
"endpoint": "API-Endpunkt",
"test_mode": "Testmodus",
"test_mode_help": "Wenn aktiviert, werden keine Änderungen bei Letzshop vorgenommen"
},
"sync": {
"title": "Synchronisation",
"auto_sync": "Auto-Sync",
"auto_sync_help": "Bestellungen automatisch von Letzshop synchronisieren",
"interval": "Sync-Intervall",
"interval_help": "Minuten zwischen Synchronisationen",
"last_sync": "Letzte Sync",
"last_status": "Letzter Status",
"sync_now": "Jetzt synchronisieren"
},
"carrier": {
"title": "Versanddiensteinstellungen",
"default_carrier": "Standard-Versanddienstleister",
"greco": "Greco",
"colissimo": "Colissimo",
"xpresslogistics": "XpressLogistics",
"label_url": "Label-URL-Präfix"
},
"historical": {
"title": "Historischer Import",
"subtitle": "Vergangene Bestellungen von Letzshop importieren",
"start_import": "Historischen Import starten",
"phase": "Phase",
"confirmed": "Bestätigte Bestellungen",
"unconfirmed": "Unbestätigte Bestellungen",
"fetching": "Abrufen...",
"processing": "Verarbeiten...",
"page": "Seite",
"fetched": "Abgerufen",
"processed": "Verarbeitet",
"imported": "Importiert",
"updated": "Aktualisiert",
"skipped": "Übersprungen"
},
"vendors": {
"title": "Verkäuferverzeichnis",
"subtitle": "Letzshop-Verkäufer durchsuchen",
"claim": "Beanspruchen",
"claimed": "Beansprucht",
"unclaimed": "Nicht beansprucht",
"last_synced": "Zuletzt synchronisiert"
"connection": "Verbindung",
"credentials": "Zugangsdaten",
"api_key": "API-Schlüssel",
"api_endpoint": "API-Endpunkt",
"auto_sync": "Auto-Sync",
"sync_interval": "Sync-Intervall",
"every_hour": "Jede Stunde",
"every_day": "Jeden Tag",
"test_connection": "Verbindung testen",
"save_credentials": "Zugangsdaten speichern",
"connection_success": "Verbindung erfolgreich",
"connection_failed": "Verbindung fehlgeschlagen",
"last_sync": "Letzte Synchronisation",
"sync_status": "Sync-Status",
"import_orders": "Bestellungen importieren",
"export_products": "Produkte exportieren",
"no_credentials": "Konfigurieren Sie Ihren API-Schlüssel in den Einstellungen",
"carriers": {
"dhl": "DHL",
"ups": "UPS",
"fedex": "FedEx",
"dpd": "DPD",
"gls": "GLS",
"post_luxembourg": "Post Luxemburg",
"other": "Andere"
}
},
"export": {
"title": "Produkte exportieren",
"subtitle": "Produkte im Marktplatz-Format exportieren",
"format": "Format",
"format_csv": "CSV",
"format_xml": "XML",
"download": "Export herunterladen"
},
"messages": {
"import_started": "Import erfolgreich gestartet",
"import_completed": "Import abgeschlossen",
"import_failed": "Import fehlgeschlagen",
"credentials_saved": "Anmeldedaten erfolgreich gespeichert",
"sync_started": "Synchronisation gestartet",
"sync_completed": "Synchronisation abgeschlossen",
"sync_failed": "Synchronisation fehlgeschlagen",
"export_ready": "Export zum Download bereit",
"error_loading": "Fehler beim Laden der Daten"
},
"filters": {
"all_marketplaces": "Alle Marktplätze",
"all_vendors": "Alle Verkäufer",
"search_placeholder": "Produkte suchen..."
"no_error_details_available": "No error details available",
"failed_to_load_error_details": "Failed to load error details",
"copied_to_clipboard": "Copied to clipboard",
"failed_to_copy_to_clipboard": "Failed to copy to clipboard"
}
}

View File

@@ -1,122 +1,61 @@
{
"title": "Intégration Marketplace",
"description": "Synchronisation des produits et commandes Letzshop",
"products": {
"title": "Produits Marketplace",
"subtitle": "Produits importés des marketplaces",
"empty": "Aucun produit trouvé",
"empty_search": "Aucun produit ne correspond à votre recherche",
"import": "Importer des produits"
},
"import": {
"title": "Importer des produits",
"subtitle": "Importer des produits depuis les flux marketplace",
"source_url": "URL du flux",
"source_url_help": "URL du flux CSV marketplace",
"marketplace": "Marketplace",
"language": "Langue",
"language_help": "Langue pour les traductions de produits",
"batch_size": "Taille du lot",
"start_import": "Démarrer l'import",
"cancel": "Annuler"
},
"import_jobs": {
"title": "Historique des imports",
"subtitle": "Imports passés et en cours",
"empty": "Aucun import",
"job_id": "ID du job",
"marketplace": "Marketplace",
"vendor": "Vendeur",
"status": "Statut",
"imported": "Importés",
"updated": "Mis à jour",
"errors": "Erreurs",
"created": "Créé",
"completed": "Terminé",
"statuses": {
"pending": "En attente",
"processing": "En cours",
"completed": "Terminé",
"completed_with_errors": "Terminé avec erreurs",
"failed": "Échoué"
}
"marketplace": {
"title": "Marketplace",
"import": "Importer",
"export": "Exporter",
"sync": "Synchroniser",
"source": "Source",
"source_url": "URL source",
"import_products": "Importer des produits",
"start_import": "Démarrer l'importation",
"importing": "Importation en cours...",
"import_complete": "Importation terminée",
"import_failed": "Échec de l'importation",
"import_history": "Historique des importations",
"job_id": "ID du travail",
"started_at": "Démarré à",
"completed_at": "Terminé à",
"duration": "Durée",
"imported_count": "Importés",
"error_count": "Erreurs",
"total_processed": "Total traité",
"progress": "Progression",
"no_import_jobs": "Aucune importation pour le moment",
"start_first_import": "Lancez votre première importation avec le formulaire ci-dessus"
},
"letzshop": {
"title": "Intégration Letzshop",
"subtitle": "Gérer la connexion et la synchronisation Letzshop",
"credentials": {
"title": "Identifiants API",
"api_key": "Clé API",
"api_key_help": "Votre clé API Letzshop",
"endpoint": "Point d'accès API",
"test_mode": "Mode test",
"test_mode_help": "Lorsqu'activé, aucune modification n'est effectuée sur Letzshop"
},
"sync": {
"title": "Synchronisation",
"auto_sync": "Sync automatique",
"auto_sync_help": "Synchroniser automatiquement les commandes depuis Letzshop",
"interval": "Intervalle de sync",
"interval_help": "Minutes entre les synchronisations",
"last_sync": "Dernière sync",
"last_status": "Dernier statut",
"sync_now": "Synchroniser maintenant"
},
"carrier": {
"title": "Paramètres transporteur",
"default_carrier": "Transporteur par défaut",
"greco": "Greco",
"colissimo": "Colissimo",
"xpresslogistics": "XpressLogistics",
"label_url": "Préfixe URL étiquette"
},
"historical": {
"title": "Import historique",
"subtitle": "Importer les commandes passées depuis Letzshop",
"start_import": "Démarrer l'import historique",
"phase": "Phase",
"confirmed": "Commandes confirmées",
"unconfirmed": "Commandes non confirmées",
"fetching": "Récupération...",
"processing": "Traitement...",
"page": "Page",
"fetched": "Récupérées",
"processed": "Traitées",
"imported": "Importées",
"updated": "Mises à jour",
"skipped": "Ignorées"
},
"vendors": {
"title": "Annuaire des vendeurs",
"subtitle": "Parcourir les vendeurs Letzshop",
"claim": "Revendiquer",
"claimed": "Revendiqué",
"unclaimed": "Non revendiqué",
"last_synced": "Dernière sync"
"connection": "Connexion",
"credentials": "Identifiants",
"api_key": "Clé API",
"api_endpoint": "Point d'accès API",
"auto_sync": "Synchronisation automatique",
"sync_interval": "Intervalle de synchronisation",
"every_hour": "Toutes les heures",
"every_day": "Tous les jours",
"test_connection": "Tester la connexion",
"save_credentials": "Enregistrer les identifiants",
"connection_success": "Connexion réussie",
"connection_failed": "Échec de la connexion",
"last_sync": "Dernière synchronisation",
"sync_status": "Statut de synchronisation",
"import_orders": "Importer les commandes",
"export_products": "Exporter les produits",
"no_credentials": "Configurez votre clé API dans les paramètres pour commencer",
"carriers": {
"dhl": "DHL",
"ups": "UPS",
"fedex": "FedEx",
"dpd": "DPD",
"gls": "GLS",
"post_luxembourg": "Post Luxembourg",
"other": "Autre"
}
},
"export": {
"title": "Exporter les produits",
"subtitle": "Exporter les produits au format marketplace",
"format": "Format",
"format_csv": "CSV",
"format_xml": "XML",
"download": "Télécharger l'export"
},
"messages": {
"import_started": "Import démarré avec succès",
"import_completed": "Import terminé",
"import_failed": "Import échoué",
"credentials_saved": "Identifiants enregistrés avec succès",
"sync_started": "Synchronisation démarrée",
"sync_completed": "Synchronisation terminée",
"sync_failed": "Synchronisation échouée",
"export_ready": "Export prêt au téléchargement",
"error_loading": "Erreur lors du chargement des données"
},
"filters": {
"all_marketplaces": "Tous les marketplaces",
"all_vendors": "Tous les vendeurs",
"search_placeholder": "Rechercher des produits..."
"no_error_details_available": "No error details available",
"failed_to_load_error_details": "Failed to load error details",
"copied_to_clipboard": "Copied to clipboard",
"failed_to_copy_to_clipboard": "Failed to copy to clipboard"
}
}

View File

@@ -1,122 +1,61 @@
{
"title": "Marketplace-Integratioun",
"description": "Letzshop Produkt- a Bestellsynchronisatioun",
"products": {
"title": "Marketplace-Produkter",
"subtitle": "Vun Marketplacen importéiert Produkter",
"empty": "Keng Produkter fonnt",
"empty_search": "Keng Produkter passen zu Ärer Sich",
"import": "Produkter importéieren"
},
"import": {
"title": "Produkter importéieren",
"subtitle": "Produkter aus Marketplace-Feeds importéieren",
"source_url": "Feed-URL",
"source_url_help": "URL zum Marketplace-CSV-Feed",
"marketplace": "Marketplace",
"language": "Sprooch",
"language_help": "Sprooch fir Produktiwwersetzungen",
"batch_size": "Batch-Gréisst",
"marketplace": {
"title": "Marchéplaz",
"import": "Import",
"export": "Export",
"sync": "Synchroniséieren",
"source": "Quell",
"source_url": "Quell URL",
"import_products": "Produkter importéieren",
"start_import": "Import starten",
"cancel": "Ofbriechen"
},
"import_jobs": {
"title": "Import-Verlaf",
"subtitle": "Vergaangen an aktuell Import-Jobs",
"empty": "Keng Import-Jobs",
"job_id": "Job-ID",
"marketplace": "Marketplace",
"vendor": "Verkeefer",
"status": "Status",
"imported": "Importéiert",
"updated": "Aktualiséiert",
"errors": "Feeler",
"created": "Erstallt",
"completed": "Ofgeschloss",
"statuses": {
"pending": "Waarden",
"processing": "Am Gaang",
"completed": "Ofgeschloss",
"completed_with_errors": "Mat Feeler ofgeschloss",
"failed": "Feelgeschloen"
}
"importing": "Importéieren...",
"import_complete": "Import fäerdeg",
"import_failed": "Import feelgeschloen",
"import_history": "Importgeschicht",
"job_id": "Job ID",
"started_at": "Ugefaang um",
"completed_at": "Fäerdeg um",
"duration": "Dauer",
"imported_count": "Importéiert",
"error_count": "Feeler",
"total_processed": "Total veraarbecht",
"progress": "Fortschrëtt",
"no_import_jobs": "Nach keng Import Jobs",
"start_first_import": "Start Ären éischten Import mat der Form uewendriwwer"
},
"letzshop": {
"title": "Letzshop-Integratioun",
"subtitle": "Letzshop-Verbindung a Synchronisatioun verwalten",
"credentials": {
"title": "API-Umeldedaten",
"api_key": "API-Schlëssel",
"api_key_help": "Ären Letzshop API-Schlëssel",
"endpoint": "API-Endpunkt",
"test_mode": "Testmodus",
"test_mode_help": "Wann aktivéiert, ginn keng Ännerungen bei Letzshop gemaach"
},
"sync": {
"title": "Synchronisatioun",
"auto_sync": "Auto-Sync",
"auto_sync_help": "Bestellungen automatesch vun Letzshop synchroniséieren",
"interval": "Sync-Intervall",
"interval_help": "Minutten tëscht Synchronisatiounen",
"last_sync": "Lescht Sync",
"last_status": "Leschte Status",
"sync_now": "Elo synchroniséieren"
},
"carrier": {
"title": "Versandastellungen",
"default_carrier": "Standard-Versanddienstleeschter",
"greco": "Greco",
"colissimo": "Colissimo",
"xpresslogistics": "XpressLogistics",
"label_url": "Label-URL-Präfix"
},
"historical": {
"title": "Historeschen Import",
"subtitle": "Vergaangen Bestellungen vun Letzshop importéieren",
"start_import": "Historeschen Import starten",
"phase": "Phas",
"confirmed": "Bestätegt Bestellungen",
"unconfirmed": "Onbestätegt Bestellungen",
"fetching": "Ofruff...",
"processing": "Veraarbecht...",
"page": "Säit",
"fetched": "Ofgeruff",
"processed": "Veraarbecht",
"imported": "Importéiert",
"updated": "Aktualiséiert",
"skipped": "Iwwersprong"
},
"vendors": {
"title": "Verkeeferverzeechnes",
"subtitle": "Letzshop-Verkeefer duerchsichen",
"claim": "Reklaméieren",
"claimed": "Reklaméiert",
"unclaimed": "Net reklaméiert",
"last_synced": "Lescht synchroniséiert"
"title": "Letzshop Integratioun",
"connection": "Verbindung",
"credentials": "Umeldungsdaten",
"api_key": "API Schlëssel",
"api_endpoint": "API Endpunkt",
"auto_sync": "Automatesch Sync",
"sync_interval": "Sync Intervall",
"every_hour": "All Stonn",
"every_day": "All Dag",
"test_connection": "Verbindung testen",
"save_credentials": "Umeldungsdaten späicheren",
"connection_success": "Verbindung erfollegräich",
"connection_failed": "Verbindung feelgeschloen",
"last_sync": "Läschte Sync",
"sync_status": "Sync Status",
"import_orders": "Bestellungen importéieren",
"export_products": "Produkter exportéieren",
"no_credentials": "Konfiguréiert Ären API Schlëssel an den Astellungen fir unzefänken",
"carriers": {
"dhl": "DHL",
"ups": "UPS",
"fedex": "FedEx",
"dpd": "DPD",
"gls": "GLS",
"post_luxembourg": "Post Lëtzebuerg",
"other": "Anerer"
}
},
"export": {
"title": "Produkter exportéieren",
"subtitle": "Produkter am Marketplace-Format exportéieren",
"format": "Format",
"format_csv": "CSV",
"format_xml": "XML",
"download": "Export eroflueden"
},
"messages": {
"import_started": "Import erfollegräich gestart",
"import_completed": "Import ofgeschloss",
"import_failed": "Import feelgeschloen",
"credentials_saved": "Umeldedaten erfollegräich gespäichert",
"sync_started": "Synchronisatioun gestart",
"sync_completed": "Synchronisatioun ofgeschloss",
"sync_failed": "Synchronisatioun feelgeschloen",
"export_ready": "Export prett zum Eroflueden",
"error_loading": "Feeler beim Lueden vun den Daten"
},
"filters": {
"all_marketplaces": "All Marketplacen",
"all_vendors": "All Verkeefer",
"search_placeholder": "Produkter sichen..."
"no_error_details_available": "No error details available",
"failed_to_load_error_details": "Failed to load error details",
"copied_to_clipboard": "Copied to clipboard",
"failed_to_copy_to_clipboard": "Failed to copy to clipboard"
}
}