geo/backend/alembic/versions/488d0bd5ab01_initial_migrat...

128 lines
6.6 KiB
Python

"""Initial migration
Revision ID: 488d0bd5ab01
Revises:
Create Date: 2026-04-22 18:06:46.629263
"""
from typing import Sequence, Union
from alembic import op
import sqlalchemy as sa
from sqlalchemy.dialects import postgresql
# revision identifiers, used by Alembic.
revision: str = '488d0bd5ab01'
down_revision: Union[str, Sequence[str], None] = None
branch_labels: Union[str, Sequence[str], None] = None
depends_on: Union[str, Sequence[str], None] = None
def upgrade() -> None:
"""Upgrade schema."""
# users table
op.create_table(
'users',
sa.Column('id', postgresql.UUID(as_uuid=True), server_default=sa.text('gen_random_uuid()'), nullable=False),
sa.Column('email', sa.String(255), nullable=False),
sa.Column('password_hash', sa.String(255), nullable=False),
sa.Column('name', sa.String(100), nullable=True),
sa.Column('plan', sa.String(20), server_default='free', nullable=False),
sa.Column('max_queries', sa.Integer(), server_default='5', nullable=False),
sa.Column('is_active', sa.Boolean(), server_default='true', nullable=False),
sa.Column('created_at', sa.DateTime(timezone=True), server_default=sa.text('NOW()'), nullable=False),
sa.Column('updated_at', sa.DateTime(timezone=True), server_default=sa.text('NOW()'), nullable=False),
sa.PrimaryKeyConstraint('id'),
sa.UniqueConstraint('email')
)
# queries table
op.create_table(
'queries',
sa.Column('id', postgresql.UUID(as_uuid=True), server_default=sa.text('gen_random_uuid()'), nullable=False),
sa.Column('user_id', postgresql.UUID(as_uuid=True), nullable=False),
sa.Column('keyword', sa.String(200), nullable=False),
sa.Column('target_brand', sa.String(100), nullable=False),
sa.Column('brand_aliases', postgresql.JSONB(), server_default='[]', nullable=False),
sa.Column('platforms', postgresql.JSONB(), server_default='["wenxin", "kimi"]', nullable=False),
sa.Column('frequency', sa.String(20), server_default='weekly', nullable=False),
sa.Column('status', sa.String(20), server_default='active', nullable=False),
sa.Column('last_queried_at', sa.DateTime(timezone=True), nullable=True),
sa.Column('next_query_at', sa.DateTime(timezone=True), nullable=True),
sa.Column('created_at', sa.DateTime(timezone=True), server_default=sa.text('NOW()'), nullable=False),
sa.Column('updated_at', sa.DateTime(timezone=True), server_default=sa.text('NOW()'), nullable=False),
sa.PrimaryKeyConstraint('id'),
sa.ForeignKeyConstraint(['user_id'], ['users.id'], ondelete='CASCADE')
)
op.create_index('idx_queries_user_id', 'queries', ['user_id'])
op.create_index('idx_queries_status', 'queries', ['status'])
op.create_index('idx_queries_next_query_at', 'queries', ['next_query_at'])
# citation_records table
op.create_table(
'citation_records',
sa.Column('id', postgresql.UUID(as_uuid=True), server_default=sa.text('gen_random_uuid()'), nullable=False),
sa.Column('query_id', postgresql.UUID(as_uuid=True), nullable=False),
sa.Column('platform', sa.String(50), nullable=False),
sa.Column('cited', sa.Boolean(), server_default='false', nullable=False),
sa.Column('citation_position', sa.Integer(), nullable=True),
sa.Column('citation_text', sa.Text(), nullable=True),
sa.Column('competitor_brands', postgresql.JSONB(), server_default='[]', nullable=False),
sa.Column('raw_response', sa.Text(), nullable=True),
sa.Column('queried_at', sa.DateTime(timezone=True), server_default=sa.text('NOW()'), nullable=False),
sa.PrimaryKeyConstraint('id'),
sa.ForeignKeyConstraint(['query_id'], ['queries.id'], ondelete='CASCADE')
)
op.create_index('idx_citation_records_query_id', 'citation_records', ['query_id'])
op.create_index('idx_citation_records_queried_at', 'citation_records', ['queried_at'])
op.create_index('idx_citation_records_platform', 'citation_records', ['platform'])
# query_tasks table
op.create_table(
'query_tasks',
sa.Column('id', postgresql.UUID(as_uuid=True), server_default=sa.text('gen_random_uuid()'), nullable=False),
sa.Column('query_id', postgresql.UUID(as_uuid=True), nullable=False),
sa.Column('platform', sa.String(50), nullable=False),
sa.Column('status', sa.String(20), server_default='pending', nullable=False),
sa.Column('error_message', sa.Text(), nullable=True),
sa.Column('scheduled_at', sa.DateTime(timezone=True), server_default=sa.text('NOW()'), nullable=False),
sa.Column('started_at', sa.DateTime(timezone=True), nullable=True),
sa.Column('completed_at', sa.DateTime(timezone=True), nullable=True),
sa.PrimaryKeyConstraint('id'),
sa.ForeignKeyConstraint(['query_id'], ['queries.id'], ondelete='CASCADE')
)
op.create_index('idx_query_tasks_status', 'query_tasks', ['status'])
# subscriptions table
op.create_table(
'subscriptions',
sa.Column('id', postgresql.UUID(as_uuid=True), server_default=sa.text('gen_random_uuid()'), nullable=False),
sa.Column('user_id', postgresql.UUID(as_uuid=True), nullable=False),
sa.Column('plan', sa.String(20), nullable=False),
sa.Column('status', sa.String(20), server_default='active', nullable=False),
sa.Column('start_date', sa.Date(), nullable=False),
sa.Column('end_date', sa.Date(), nullable=False),
sa.Column('amount', sa.Numeric(10, 2), nullable=True),
sa.Column('payment_method', sa.String(50), nullable=True),
sa.Column('payment_id', sa.String(255), nullable=True),
sa.Column('created_at', sa.DateTime(timezone=True), server_default=sa.text('NOW()'), nullable=False),
sa.PrimaryKeyConstraint('id'),
sa.ForeignKeyConstraint(['user_id'], ['users.id'], ondelete='CASCADE')
)
def downgrade() -> None:
"""Downgrade schema."""
op.drop_table('subscriptions')
op.drop_index('idx_query_tasks_status', table_name='query_tasks')
op.drop_table('query_tasks')
op.drop_index('idx_citation_records_platform', table_name='citation_records')
op.drop_index('idx_citation_records_queried_at', table_name='citation_records')
op.drop_index('idx_citation_records_query_id', table_name='citation_records')
op.drop_table('citation_records')
op.drop_index('idx_queries_next_query_at', table_name='queries')
op.drop_index('idx_queries_status', table_name='queries')
op.drop_index('idx_queries_user_id', table_name='queries')
op.drop_table('queries')
op.drop_table('users')