127 lines
6.4 KiB
Python
127 lines
6.4 KiB
Python
|
|
"""add members area tables
|
||
|
|
|
||
|
|
Revision ID: a1b2c3d4e5f6
|
||
|
|
Revises: f25d0f745a17
|
||
|
|
Create Date: 2026-03-31 12:00:00.000000
|
||
|
|
|
||
|
|
"""
|
||
|
|
from typing import Sequence, Union
|
||
|
|
|
||
|
|
from alembic import op
|
||
|
|
import sqlalchemy as sa
|
||
|
|
|
||
|
|
revision: str = 'a1b2c3d4e5f6'
|
||
|
|
down_revision: Union[str, None] = 'f25d0f745a17'
|
||
|
|
branch_labels: Union[str, Sequence[str], None] = None
|
||
|
|
depends_on: Union[str, Sequence[str], None] = None
|
||
|
|
|
||
|
|
|
||
|
|
def upgrade() -> None:
|
||
|
|
op.create_table(
|
||
|
|
'members',
|
||
|
|
sa.Column('email', sa.String(255), nullable=False),
|
||
|
|
sa.Column('hashed_password', sa.String(255), nullable=True),
|
||
|
|
sa.Column('first_name', sa.String(100), nullable=False),
|
||
|
|
sa.Column('last_name', sa.String(100), nullable=False),
|
||
|
|
sa.Column('phone', sa.String(50), nullable=True),
|
||
|
|
sa.Column('address', sa.String(500), nullable=True),
|
||
|
|
sa.Column('emergency_contact', sa.String(255), nullable=True),
|
||
|
|
sa.Column('is_claimed', sa.Boolean(), nullable=False, server_default='false'),
|
||
|
|
sa.Column('is_active', sa.Boolean(), nullable=False, server_default='true'),
|
||
|
|
sa.Column('onboarding_data', sa.JSON(), nullable=True),
|
||
|
|
sa.Column('id', sa.Uuid(), 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'),
|
||
|
|
)
|
||
|
|
op.create_index(op.f('ix_members_email'), 'members', ['email'], unique=True)
|
||
|
|
|
||
|
|
op.create_table(
|
||
|
|
'member_verification_codes',
|
||
|
|
sa.Column('member_id', sa.Uuid(), nullable=False),
|
||
|
|
sa.Column('code_hash', sa.String(255), nullable=False),
|
||
|
|
sa.Column('purpose', sa.String(20), nullable=False),
|
||
|
|
sa.Column('expires_at', sa.DateTime(timezone=True), nullable=False),
|
||
|
|
sa.Column('used_at', sa.DateTime(timezone=True), nullable=True),
|
||
|
|
sa.Column('created_at', sa.DateTime(timezone=True), server_default=sa.text('now()'), nullable=False),
|
||
|
|
sa.Column('id', sa.Uuid(), nullable=False),
|
||
|
|
sa.ForeignKeyConstraint(['member_id'], ['members.id'], ondelete='CASCADE'),
|
||
|
|
sa.PrimaryKeyConstraint('id'),
|
||
|
|
)
|
||
|
|
op.create_index(op.f('ix_member_verification_codes_member_id'), 'member_verification_codes', ['member_id'], unique=False)
|
||
|
|
|
||
|
|
op.create_table(
|
||
|
|
'member_refresh_tokens',
|
||
|
|
sa.Column('member_id', sa.Uuid(), nullable=False),
|
||
|
|
sa.Column('token_hash', sa.String(255), nullable=False),
|
||
|
|
sa.Column('expires_at', sa.DateTime(timezone=True), nullable=False),
|
||
|
|
sa.Column('revoked', sa.Boolean(), nullable=False, server_default='false'),
|
||
|
|
sa.Column('created_at', sa.DateTime(timezone=True), server_default=sa.text('now()'), nullable=False),
|
||
|
|
sa.Column('id', sa.Uuid(), nullable=False),
|
||
|
|
sa.ForeignKeyConstraint(['member_id'], ['members.id'], ondelete='CASCADE'),
|
||
|
|
sa.PrimaryKeyConstraint('id'),
|
||
|
|
)
|
||
|
|
op.create_index(op.f('ix_member_refresh_tokens_member_id'), 'member_refresh_tokens', ['member_id'], unique=False)
|
||
|
|
|
||
|
|
op.create_table(
|
||
|
|
'walks',
|
||
|
|
sa.Column('member_id', sa.Uuid(), nullable=False),
|
||
|
|
sa.Column('walked_at', sa.DateTime(timezone=True), nullable=False),
|
||
|
|
sa.Column('service_type', sa.String(50), nullable=False),
|
||
|
|
sa.Column('duration_minutes', sa.Integer(), nullable=False, server_default='60'),
|
||
|
|
sa.Column('notes', sa.Text(), nullable=True),
|
||
|
|
sa.Column('recorded_by', sa.String(255), nullable=True),
|
||
|
|
sa.Column('id', sa.Uuid(), 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.ForeignKeyConstraint(['member_id'], ['members.id'], ondelete='CASCADE'),
|
||
|
|
sa.PrimaryKeyConstraint('id'),
|
||
|
|
)
|
||
|
|
op.create_index(op.f('ix_walks_member_id'), 'walks', ['member_id'], unique=False)
|
||
|
|
|
||
|
|
op.create_table(
|
||
|
|
'bookings',
|
||
|
|
sa.Column('member_id', sa.Uuid(), nullable=False),
|
||
|
|
sa.Column('service_type', sa.String(50), nullable=False),
|
||
|
|
sa.Column('requested_date', sa.DateTime(timezone=True), nullable=True),
|
||
|
|
sa.Column('status', sa.String(20), nullable=False, server_default='pending'),
|
||
|
|
sa.Column('notes', sa.Text(), nullable=True),
|
||
|
|
sa.Column('admin_notes', sa.Text(), nullable=True),
|
||
|
|
sa.Column('id', sa.Uuid(), 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.ForeignKeyConstraint(['member_id'], ['members.id'], ondelete='CASCADE'),
|
||
|
|
sa.PrimaryKeyConstraint('id'),
|
||
|
|
)
|
||
|
|
op.create_index(op.f('ix_bookings_member_id'), 'bookings', ['member_id'], unique=False)
|
||
|
|
|
||
|
|
op.create_table(
|
||
|
|
'admin_messages',
|
||
|
|
sa.Column('member_id', sa.Uuid(), nullable=False),
|
||
|
|
sa.Column('subject', sa.String(255), nullable=False),
|
||
|
|
sa.Column('body', sa.Text(), nullable=False),
|
||
|
|
sa.Column('read_at', sa.DateTime(timezone=True), nullable=True),
|
||
|
|
sa.Column('sent_by', sa.String(255), nullable=True),
|
||
|
|
sa.Column('id', sa.Uuid(), 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.ForeignKeyConstraint(['member_id'], ['members.id'], ondelete='CASCADE'),
|
||
|
|
sa.PrimaryKeyConstraint('id'),
|
||
|
|
)
|
||
|
|
op.create_index(op.f('ix_admin_messages_member_id'), 'admin_messages', ['member_id'], unique=False)
|
||
|
|
|
||
|
|
|
||
|
|
def downgrade() -> None:
|
||
|
|
op.drop_index(op.f('ix_admin_messages_member_id'), table_name='admin_messages')
|
||
|
|
op.drop_table('admin_messages')
|
||
|
|
op.drop_index(op.f('ix_bookings_member_id'), table_name='bookings')
|
||
|
|
op.drop_table('bookings')
|
||
|
|
op.drop_index(op.f('ix_walks_member_id'), table_name='walks')
|
||
|
|
op.drop_table('walks')
|
||
|
|
op.drop_index(op.f('ix_member_refresh_tokens_member_id'), table_name='member_refresh_tokens')
|
||
|
|
op.drop_table('member_refresh_tokens')
|
||
|
|
op.drop_index(op.f('ix_member_verification_codes_member_id'), table_name='member_verification_codes')
|
||
|
|
op.drop_table('member_verification_codes')
|
||
|
|
op.drop_index(op.f('ix_members_email'), table_name='members')
|
||
|
|
op.drop_table('members')
|