"""core initial - tenancy, platform, admin tables Revision ID: core_001 Revises: Create Date: 2026-02-07 """ from alembic import op import sqlalchemy as sa revision = "core_001" down_revision = None branch_labels = None depends_on = None def upgrade() -> None: # --- platforms --- op.create_table( "platforms", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("code", sa.String(50), unique=True, nullable=False, index=True, comment="Unique platform identifier (e.g., 'oms', 'loyalty', 'sites')"), sa.Column("name", sa.String(100), nullable=False, comment="Display name (e.g., 'Orion OMS')"), sa.Column("description", sa.Text(), nullable=True, comment="Platform description for admin/marketing purposes"), sa.Column("domain", sa.String(255), unique=True, nullable=True, index=True, comment="Production domain (e.g., 'omsflow.lu', 'rewardflow.lu')"), sa.Column("path_prefix", sa.String(50), unique=True, nullable=True, index=True, comment="Development path prefix (e.g., 'oms' for localhost:9999/oms/*)"), sa.Column("logo", sa.String(500), nullable=True, comment="Logo URL for light mode"), sa.Column("logo_dark", sa.String(500), nullable=True, comment="Logo URL for dark mode"), sa.Column("favicon", sa.String(500), nullable=True, comment="Favicon URL"), sa.Column("theme_config", sa.JSON(), nullable=True, server_default="{}", comment="Theme configuration (colors, fonts, etc.)"), sa.Column("default_language", sa.String(5), nullable=False, server_default="fr", comment="Default language code (e.g., 'fr', 'en', 'de')"), sa.Column("supported_languages", sa.JSON(), nullable=False, comment="List of supported language codes"), sa.Column("is_active", sa.Boolean(), nullable=False, server_default="true", comment="Whether the platform is active and accessible"), sa.Column("is_public", sa.Boolean(), nullable=False, server_default="true", comment="Whether the platform is visible in public listings"), sa.Column("settings", sa.JSON(), nullable=True, server_default="{}", comment="Platform-specific settings and feature flags"), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) op.create_index("idx_platform_active", "platforms", ["is_active"]) op.create_index("idx_platform_public", "platforms", ["is_public", "is_active"]) # --- users --- op.create_table( "users", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("email", sa.String(), unique=True, nullable=False, index=True), sa.Column("username", sa.String(), unique=True, nullable=False, index=True), sa.Column("first_name", sa.String(), nullable=True), sa.Column("last_name", sa.String(), nullable=True), sa.Column("hashed_password", sa.String(), nullable=False), sa.Column("role", sa.String(), nullable=False, server_default="store"), sa.Column("is_active", sa.Boolean(), nullable=False, server_default="true"), sa.Column("is_email_verified", sa.Boolean(), nullable=False, server_default="false"), sa.Column("is_super_admin", sa.Boolean(), nullable=False, server_default="false"), sa.Column("preferred_language", sa.String(5), nullable=True), sa.Column("last_login", sa.DateTime(), nullable=True), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- merchants --- op.create_table( "merchants", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("name", sa.String(), nullable=False, index=True), sa.Column("description", sa.Text(), nullable=True), sa.Column("owner_user_id", sa.Integer(), sa.ForeignKey("users.id"), nullable=False), sa.Column("contact_email", sa.String(), nullable=False), sa.Column("contact_phone", sa.String(), nullable=True), sa.Column("website", sa.String(), nullable=True), sa.Column("business_address", sa.Text(), nullable=True), sa.Column("tax_number", sa.String(), nullable=True), sa.Column("is_active", sa.Boolean(), nullable=False, server_default="true"), sa.Column("is_verified", sa.Boolean(), nullable=False, server_default="false"), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- stores --- op.create_table( "stores", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("merchant_id", sa.Integer(), sa.ForeignKey("merchants.id"), nullable=False, index=True), sa.Column("store_code", sa.String(), unique=True, nullable=False, index=True), sa.Column("subdomain", sa.String(100), unique=True, nullable=False, index=True), sa.Column("name", sa.String(), nullable=False), sa.Column("description", sa.Text(), nullable=True), sa.Column("letzshop_csv_url_fr", sa.String(), nullable=True), sa.Column("letzshop_csv_url_en", sa.String(), nullable=True), sa.Column("letzshop_csv_url_de", sa.String(), nullable=True), sa.Column("letzshop_store_id", sa.String(100), unique=True, nullable=True, index=True), sa.Column("letzshop_store_slug", sa.String(200), nullable=True, index=True), sa.Column("letzshop_default_tax_rate", sa.Integer(), nullable=False, server_default="17"), sa.Column("letzshop_boost_sort", sa.String(10), nullable=True, server_default="5.0"), sa.Column("letzshop_delivery_method", sa.String(100), nullable=True, server_default="package_delivery"), sa.Column("letzshop_preorder_days", sa.Integer(), nullable=True, server_default="1"), sa.Column("is_active", sa.Boolean(), nullable=True, server_default="true"), sa.Column("is_verified", sa.Boolean(), nullable=True, server_default="false"), sa.Column("contact_email", sa.String(255), nullable=True), sa.Column("contact_phone", sa.String(50), nullable=True), sa.Column("website", sa.String(255), nullable=True), sa.Column("business_address", sa.Text(), nullable=True), sa.Column("tax_number", sa.String(100), nullable=True), sa.Column("default_language", sa.String(5), nullable=False, server_default="fr"), sa.Column("dashboard_language", sa.String(5), nullable=False, server_default="fr"), sa.Column("storefront_language", sa.String(5), nullable=False, server_default="fr"), sa.Column("storefront_languages", sa.JSON(), nullable=False), sa.Column("storefront_locale", sa.String(10), nullable=True), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- roles --- op.create_table( "roles", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("store_id", sa.Integer(), sa.ForeignKey("stores.id"), nullable=False), sa.Column("name", sa.String(100), nullable=False), sa.Column("permissions", sa.JSON(), nullable=True), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- store_users --- op.create_table( "store_users", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("store_id", sa.Integer(), sa.ForeignKey("stores.id"), nullable=False), sa.Column("user_id", sa.Integer(), sa.ForeignKey("users.id"), nullable=False), sa.Column("user_type", sa.String(), nullable=False, server_default="member"), sa.Column("role_id", sa.Integer(), sa.ForeignKey("roles.id"), nullable=True), sa.Column("invited_by", sa.Integer(), sa.ForeignKey("users.id"), nullable=True), sa.Column("invitation_token", sa.String(), nullable=True, index=True), sa.Column("invitation_sent_at", sa.DateTime(), nullable=True), sa.Column("invitation_accepted_at", sa.DateTime(), nullable=True), sa.Column("is_active", sa.Boolean(), nullable=False, server_default="false"), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- store_domains --- op.create_table( "store_domains", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("store_id", sa.Integer(), sa.ForeignKey("stores.id", ondelete="CASCADE"), nullable=False), sa.Column("domain", sa.String(255), unique=True, nullable=False, index=True), sa.Column("is_primary", sa.Boolean(), nullable=False, server_default="false"), sa.Column("is_active", sa.Boolean(), nullable=False, server_default="true"), sa.Column("ssl_status", sa.String(50), nullable=True, server_default="pending"), sa.Column("ssl_verified_at", sa.DateTime(timezone=True), nullable=True), sa.Column("verification_token", sa.String(100), unique=True, nullable=True), sa.Column("is_verified", sa.Boolean(), nullable=False, server_default="false"), sa.Column("verified_at", sa.DateTime(timezone=True), nullable=True), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.UniqueConstraint("store_id", "domain", name="uq_store_domain"), ) op.create_index("idx_domain_active", "store_domains", ["domain", "is_active"]) op.create_index("idx_store_domain_primary", "store_domains", ["store_id", "is_primary"]) # --- admin_platforms --- op.create_table( "admin_platforms", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("user_id", sa.Integer(), sa.ForeignKey("users.id", ondelete="CASCADE"), nullable=False, index=True, comment="Reference to the admin user"), sa.Column("platform_id", sa.Integer(), sa.ForeignKey("platforms.id", ondelete="CASCADE"), nullable=False, index=True, comment="Reference to the platform"), sa.Column("is_active", sa.Boolean(), nullable=False, server_default="true", comment="Whether the admin assignment is active"), sa.Column("assigned_at", sa.DateTime(timezone=True), server_default=sa.func.now(), nullable=False, comment="When the admin was assigned to this platform"), sa.Column("assigned_by_user_id", sa.Integer(), sa.ForeignKey("users.id", ondelete="SET NULL"), nullable=True, comment="Super admin who made this assignment"), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.UniqueConstraint("user_id", "platform_id", name="uq_admin_platform"), ) op.create_index("idx_admin_platform_active", "admin_platforms", ["user_id", "platform_id", "is_active"]) op.create_index("idx_admin_platform_user_active", "admin_platforms", ["user_id", "is_active"]) # --- platform_modules --- op.create_table( "platform_modules", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("platform_id", sa.Integer(), sa.ForeignKey("platforms.id", ondelete="CASCADE"), nullable=False, comment="Platform this module configuration belongs to"), sa.Column("module_code", sa.String(50), nullable=False, comment="Module code (e.g., 'billing', 'inventory', 'orders')"), sa.Column("is_enabled", sa.Boolean(), nullable=False, server_default="true", comment="Whether this module is currently enabled for the platform"), sa.Column("enabled_at", sa.DateTime(timezone=True), nullable=True, comment="When the module was last enabled"), sa.Column("enabled_by_user_id", sa.Integer(), sa.ForeignKey("users.id", ondelete="SET NULL"), nullable=True, comment="User who enabled the module"), sa.Column("disabled_at", sa.DateTime(timezone=True), nullable=True, comment="When the module was last disabled"), sa.Column("disabled_by_user_id", sa.Integer(), sa.ForeignKey("users.id", ondelete="SET NULL"), nullable=True, comment="User who disabled the module"), sa.Column("config", sa.JSON(), nullable=False, server_default="{}", comment="Module-specific configuration for this platform"), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.UniqueConstraint("platform_id", "module_code", name="uq_platform_module"), ) op.create_index("idx_platform_module_platform_id", "platform_modules", ["platform_id"]) op.create_index("idx_platform_module_code", "platform_modules", ["module_code"]) op.create_index("idx_platform_module_enabled", "platform_modules", ["platform_id", "is_enabled"]) # --- admin_audit_logs --- op.create_table( "admin_audit_logs", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("admin_user_id", sa.Integer(), sa.ForeignKey("users.id"), nullable=False, index=True), sa.Column("action", sa.String(100), nullable=False, index=True), sa.Column("target_type", sa.String(50), nullable=False, index=True), sa.Column("target_id", sa.String(100), nullable=False, index=True), sa.Column("details", sa.JSON(), nullable=True), sa.Column("ip_address", sa.String(45), nullable=True), sa.Column("user_agent", sa.Text(), nullable=True), sa.Column("request_id", sa.String(100), nullable=True), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- admin_settings --- op.create_table( "admin_settings", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("key", sa.String(100), unique=True, nullable=False, index=True), sa.Column("value", sa.Text(), nullable=False), sa.Column("value_type", sa.String(20), nullable=True, server_default="string"), sa.Column("category", sa.String(50), nullable=True, index=True), sa.Column("description", sa.Text(), nullable=True), sa.Column("is_encrypted", sa.Boolean(), nullable=True, server_default="false"), sa.Column("is_public", sa.Boolean(), nullable=True, server_default="false"), sa.Column("last_modified_by_user_id", sa.Integer(), sa.ForeignKey("users.id"), nullable=True), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- platform_alerts --- op.create_table( "platform_alerts", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("alert_type", sa.String(50), nullable=False, index=True), sa.Column("severity", sa.String(20), nullable=False, index=True), sa.Column("title", sa.String(200), nullable=False), sa.Column("description", sa.Text(), nullable=True), sa.Column("affected_stores", sa.JSON(), nullable=True), sa.Column("affected_systems", sa.JSON(), nullable=True), sa.Column("is_resolved", sa.Boolean(), nullable=True, server_default="false", index=True), sa.Column("resolved_at", sa.DateTime(), nullable=True), sa.Column("resolved_by_user_id", sa.Integer(), sa.ForeignKey("users.id"), nullable=True), sa.Column("resolution_notes", sa.Text(), nullable=True), sa.Column("auto_generated", sa.Boolean(), nullable=True, server_default="true"), sa.Column("occurrence_count", sa.Integer(), nullable=True, server_default="1"), sa.Column("first_occurred_at", sa.DateTime(), nullable=False), sa.Column("last_occurred_at", sa.DateTime(), nullable=False), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- admin_sessions --- op.create_table( "admin_sessions", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("admin_user_id", sa.Integer(), sa.ForeignKey("users.id"), nullable=False, index=True), sa.Column("session_token", sa.String(255), unique=True, nullable=False, index=True), sa.Column("ip_address", sa.String(45), nullable=False), sa.Column("user_agent", sa.Text(), nullable=True), sa.Column("login_at", sa.DateTime(), nullable=False, index=True), sa.Column("last_activity_at", sa.DateTime(), nullable=False), sa.Column("logout_at", sa.DateTime(), nullable=True), sa.Column("is_active", sa.Boolean(), nullable=True, server_default="true", index=True), sa.Column("logout_reason", sa.String(50), nullable=True), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- application_logs --- op.create_table( "application_logs", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("timestamp", sa.DateTime(), nullable=False, index=True), sa.Column("level", sa.String(20), nullable=False, index=True), sa.Column("logger_name", sa.String(200), nullable=False, index=True), sa.Column("module", sa.String(200), nullable=True), sa.Column("function_name", sa.String(100), nullable=True), sa.Column("line_number", sa.Integer(), nullable=True), sa.Column("message", sa.Text(), nullable=False), sa.Column("exception_type", sa.String(200), nullable=True), sa.Column("exception_message", sa.Text(), nullable=True), sa.Column("stack_trace", sa.Text(), nullable=True), sa.Column("request_id", sa.String(100), nullable=True, index=True), sa.Column("user_id", sa.Integer(), sa.ForeignKey("users.id"), nullable=True, index=True), sa.Column("store_id", sa.Integer(), sa.ForeignKey("stores.id"), nullable=True, index=True), sa.Column("context", sa.JSON(), nullable=True), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), ) # --- admin_menu_configs --- op.create_table( "admin_menu_configs", sa.Column("id", sa.Integer(), primary_key=True, index=True), sa.Column("frontend_type", sa.Enum("platform", "admin", "store", "storefront", "merchant", name="frontendtype"), nullable=False, index=True, comment="Which frontend this config applies to (admin or store)"), sa.Column("platform_id", sa.Integer(), sa.ForeignKey("platforms.id", ondelete="CASCADE"), nullable=True, index=True, comment="Platform scope - applies to users/stores of this platform"), sa.Column("user_id", sa.Integer(), sa.ForeignKey("users.id", ondelete="CASCADE"), nullable=True, index=True, comment="User scope - applies to this specific super admin (admin frontend only)"), sa.Column("menu_item_id", sa.String(50), nullable=False, index=True, comment="Menu item identifier from registry (e.g., 'products', 'inventory')"), sa.Column("is_visible", sa.Boolean(), nullable=False, server_default="true", comment="Whether this menu item is visible (False = hidden)"), sa.Column("created_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.Column("updated_at", sa.DateTime(), server_default=sa.func.now(), nullable=False), sa.UniqueConstraint("frontend_type", "platform_id", "menu_item_id", name="uq_frontend_platform_menu_config"), sa.UniqueConstraint("frontend_type", "user_id", "menu_item_id", name="uq_frontend_user_menu_config"), sa.CheckConstraint( "(platform_id IS NOT NULL AND user_id IS NULL) OR (platform_id IS NULL AND user_id IS NOT NULL)", name="ck_admin_menu_config_scope", ), sa.CheckConstraint( "(user_id IS NULL) OR (frontend_type = 'admin')", name="ck_user_scope_admin_only", ), ) op.create_index("idx_admin_menu_config_frontend_platform", "admin_menu_configs", ["frontend_type", "platform_id"]) op.create_index("idx_admin_menu_config_frontend_user", "admin_menu_configs", ["frontend_type", "user_id"]) op.create_index("idx_admin_menu_config_platform_visible", "admin_menu_configs", ["platform_id", "is_visible"]) op.create_index("idx_admin_menu_config_user_visible", "admin_menu_configs", ["user_id", "is_visible"]) def downgrade() -> None: op.drop_table("admin_menu_configs") op.drop_table("application_logs") op.drop_table("admin_sessions") op.drop_table("platform_alerts") op.drop_table("admin_settings") op.drop_table("admin_audit_logs") op.drop_table("platform_modules") op.drop_table("admin_platforms") op.drop_table("store_domains") op.drop_table("store_users") op.drop_table("roles") op.drop_table("stores") op.drop_table("merchants") op.drop_table("users") op.drop_table("platforms") sa.Enum(name="frontendtype").drop(op.get_bind(), checkfirst=True)