Files
orion/app/modules/cms/migrations/versions/cms_001_initial.py
Samir Boulahtit c3d26e9aa4 refactor(migrations): squash 75 migrations into 12 per-module initial migrations
The old migration chain was broken (downgrade path through vendor->merchant
rename made rollbacks impossible). This squashes everything into fresh
per-module migrations with zero schema drift, verified by autogenerate.

Changes:
- Replace 75 accumulated migrations with 12 per-module initial migrations
  (core, billing, catalog, marketplace, cms, customers, orders, inventory,
  cart, messaging, loyalty, dev_tools) in a linear chain
- Fix make db-reset to use SQL DROP SCHEMA instead of alembic downgrade base
- Enable migration autodiscovery for all modules (migrations_path in definitions)
- Rewrite alembic/env.py to import all 75 model tables across 13 modules
- Fix AdminNotification import (was incorrectly from tenancy, now from messaging)
- Update squash_migrations.py to handle all module migration directories

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-08 11:51:37 +01:00

108 lines
6.5 KiB
Python

"""cms initial - content pages, store themes, media files
Revision ID: cms_001
Revises: marketplace_001
Create Date: 2026-02-07
"""
from alembic import op
import sqlalchemy as sa
revision = "cms_001"
down_revision = "marketplace_001"
branch_labels = None
depends_on = None
def upgrade() -> None:
# --- content_pages ---
op.create_table(
"content_pages",
sa.Column("id", sa.Integer(), primary_key=True, index=True),
sa.Column("platform_id", sa.Integer(), sa.ForeignKey("platforms.id", ondelete="CASCADE"), nullable=False, index=True, comment="Platform this page belongs to"),
sa.Column("store_id", sa.Integer(), sa.ForeignKey("stores.id", ondelete="CASCADE"), nullable=True, index=True, comment="Store this page belongs to (NULL for platform/default pages)"),
sa.Column("is_platform_page", sa.Boolean(), nullable=False, server_default="false", comment="True = platform marketing page (homepage, pricing); False = store default or override"),
sa.Column("slug", sa.String(100), nullable=False, index=True),
sa.Column("title", sa.String(200), nullable=False),
sa.Column("content", sa.Text(), nullable=False),
sa.Column("content_format", sa.String(20), nullable=True, server_default="html"),
sa.Column("template", sa.String(50), nullable=False, server_default="default"),
sa.Column("sections", sa.JSON(), nullable=True, comment="Structured homepage sections (hero, features, pricing, cta) with i18n"),
sa.Column("meta_description", sa.String(300), nullable=True),
sa.Column("meta_keywords", sa.String(300), nullable=True),
sa.Column("is_published", sa.Boolean(), nullable=False, server_default="false"),
sa.Column("published_at", sa.DateTime(timezone=True), nullable=True),
sa.Column("display_order", sa.Integer(), nullable=False, server_default="0"),
sa.Column("show_in_footer", sa.Boolean(), nullable=False, server_default="true"),
sa.Column("show_in_header", sa.Boolean(), nullable=False, server_default="false"),
sa.Column("show_in_legal", sa.Boolean(), nullable=False, server_default="false"),
sa.Column("created_at", sa.DateTime(timezone=True), server_default=sa.func.now(), nullable=False),
sa.Column("updated_at", sa.DateTime(timezone=True), server_default=sa.func.now(), nullable=False),
sa.Column("created_by", sa.Integer(), sa.ForeignKey("users.id", ondelete="SET NULL"), nullable=True),
sa.Column("updated_by", sa.Integer(), sa.ForeignKey("users.id", ondelete="SET NULL"), nullable=True),
sa.UniqueConstraint("platform_id", "store_id", "slug", name="uq_platform_store_slug"),
)
op.create_index("idx_platform_store_published", "content_pages", ["platform_id", "store_id", "is_published"])
op.create_index("idx_platform_slug_published", "content_pages", ["platform_id", "slug", "is_published"])
op.create_index("idx_platform_page_type", "content_pages", ["platform_id", "is_platform_page"])
# --- store_themes ---
op.create_table(
"store_themes",
sa.Column("id", sa.Integer(), primary_key=True, index=True),
sa.Column("store_id", sa.Integer(), sa.ForeignKey("stores.id", ondelete="CASCADE"), unique=True, nullable=False),
sa.Column("theme_name", sa.String(100), nullable=True, server_default="default"),
sa.Column("is_active", sa.Boolean(), nullable=True, server_default="true"),
sa.Column("colors", sa.JSON(), nullable=True),
sa.Column("font_family_heading", sa.String(100), nullable=True, server_default="Inter, sans-serif"),
sa.Column("font_family_body", sa.String(100), nullable=True, server_default="Inter, sans-serif"),
sa.Column("logo_url", sa.String(500), nullable=True),
sa.Column("logo_dark_url", sa.String(500), nullable=True),
sa.Column("favicon_url", sa.String(500), nullable=True),
sa.Column("banner_url", sa.String(500), nullable=True),
sa.Column("layout_style", sa.String(50), nullable=True, server_default="grid"),
sa.Column("header_style", sa.String(50), nullable=True, server_default="fixed"),
sa.Column("product_card_style", sa.String(50), nullable=True, server_default="modern"),
sa.Column("custom_css", sa.Text(), nullable=True),
sa.Column("social_links", sa.JSON(), nullable=True),
sa.Column("meta_title_template", sa.String(200), nullable=True),
sa.Column("meta_description", sa.Text(), 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),
)
# --- media_files ---
op.create_table(
"media_files",
sa.Column("id", sa.Integer(), primary_key=True, index=True),
sa.Column("store_id", sa.Integer(), sa.ForeignKey("stores.id"), nullable=False),
sa.Column("filename", sa.String(255), nullable=False),
sa.Column("original_filename", sa.String(255), nullable=True),
sa.Column("file_path", sa.String(500), nullable=False),
sa.Column("media_type", sa.String(20), nullable=False),
sa.Column("mime_type", sa.String(100), nullable=True),
sa.Column("file_size", sa.Integer(), nullable=True),
sa.Column("width", sa.Integer(), nullable=True),
sa.Column("height", sa.Integer(), nullable=True),
sa.Column("thumbnail_path", sa.String(500), nullable=True),
sa.Column("alt_text", sa.String(500), nullable=True),
sa.Column("description", sa.Text(), nullable=True),
sa.Column("folder", sa.String(100), nullable=True, server_default="general"),
sa.Column("tags", sa.JSON(), nullable=True),
sa.Column("extra_metadata", sa.JSON(), nullable=True),
sa.Column("is_optimized", sa.Boolean(), nullable=True, server_default="false"),
sa.Column("optimized_size", sa.Integer(), nullable=True),
sa.Column("usage_count", sa.Integer(), nullable=True, server_default="0"),
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_media_store_id", "media_files", ["store_id"])
op.create_index("idx_media_store_folder", "media_files", ["store_id", "folder"])
op.create_index("idx_media_store_type", "media_files", ["store_id", "media_type"])
op.create_index("idx_media_filename", "media_files", ["filename"])
def downgrade() -> None:
op.drop_table("media_files")
op.drop_table("store_themes")
op.drop_table("content_pages")