This commit is contained in:
@@ -0,0 +1,54 @@
|
||||
"""init
|
||||
|
||||
Revision ID: 769f535c9249
|
||||
Revises:
|
||||
Create Date: 2025-08-08 11:20:05.142049+00:00
|
||||
"""
|
||||
|
||||
from alembic import op
|
||||
import sqlalchemy as sa
|
||||
|
||||
# revision identifiers, used by Alembic.
|
||||
revision = '769f535c9249'
|
||||
down_revision = None
|
||||
branch_labels = None
|
||||
depends_on = None
|
||||
|
||||
def upgrade():
|
||||
# ### commands auto generated by Alembic - please adjust! ###
|
||||
op.create_table('photos',
|
||||
sa.Column('id', sa.UUID(), nullable=False),
|
||||
sa.Column('profile_id', sa.UUID(), nullable=False),
|
||||
sa.Column('url', sa.String(length=500), nullable=False),
|
||||
sa.Column('is_main', sa.Boolean(), nullable=False),
|
||||
sa.Column('status', sa.String(length=16), nullable=False),
|
||||
sa.Column('created_at', sa.DateTime(timezone=True), server_default=sa.text('now()'), nullable=False),
|
||||
sa.PrimaryKeyConstraint('id')
|
||||
)
|
||||
op.create_index(op.f('ix_photos_profile_id'), 'photos', ['profile_id'], unique=False)
|
||||
op.create_table('profiles',
|
||||
sa.Column('id', sa.UUID(), nullable=False),
|
||||
sa.Column('user_id', sa.UUID(), nullable=False),
|
||||
sa.Column('gender', sa.String(length=16), nullable=False),
|
||||
sa.Column('birthdate', sa.Date(), nullable=True),
|
||||
sa.Column('city', sa.String(length=120), nullable=True),
|
||||
sa.Column('bio', sa.Text(), nullable=True),
|
||||
sa.Column('languages', postgresql.JSONB(astext_type=sa.Text()), nullable=True),
|
||||
sa.Column('interests', postgresql.JSONB(astext_type=sa.Text()), nullable=True),
|
||||
sa.Column('preferences', postgresql.JSONB(astext_type=sa.Text()), nullable=True),
|
||||
sa.Column('verification_status', sa.String(length=16), 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_profiles_user_id'), 'profiles', ['user_id'], unique=False)
|
||||
# ### end Alembic commands ###
|
||||
|
||||
|
||||
def downgrade():
|
||||
# ### commands auto generated by Alembic - please adjust! ###
|
||||
op.drop_index(op.f('ix_profiles_user_id'), table_name='profiles')
|
||||
op.drop_table('profiles')
|
||||
op.drop_index(op.f('ix_photos_profile_id'), table_name='photos')
|
||||
op.drop_table('photos')
|
||||
# ### end Alembic commands ###
|
||||
@@ -0,0 +1,55 @@
|
||||
"""init
|
||||
|
||||
Revision ID: 769f535c9249
|
||||
Revises:
|
||||
Create Date: 2025-08-08 11:20:05.142049+00:00
|
||||
"""
|
||||
|
||||
from alembic import op
|
||||
import sqlalchemy as sa
|
||||
from sqlalchemy.dialects import postgresql
|
||||
|
||||
# revision identifiers, used by Alembic.
|
||||
revision = '769f535c9249'
|
||||
down_revision = None
|
||||
branch_labels = None
|
||||
depends_on = None
|
||||
|
||||
def upgrade():
|
||||
# ### commands auto generated by Alembic - please adjust! ###
|
||||
op.create_table('photos',
|
||||
sa.Column('id', sa.UUID(), nullable=False),
|
||||
sa.Column('profile_id', sa.UUID(), nullable=False),
|
||||
sa.Column('url', sa.String(length=500), nullable=False),
|
||||
sa.Column('is_main', sa.Boolean(), nullable=False),
|
||||
sa.Column('status', sa.String(length=16), nullable=False),
|
||||
sa.Column('created_at', sa.DateTime(timezone=True), server_default=sa.text('now()'), nullable=False),
|
||||
sa.PrimaryKeyConstraint('id')
|
||||
)
|
||||
op.create_index(op.f('ix_photos_profile_id'), 'photos', ['profile_id'], unique=False)
|
||||
op.create_table('profiles',
|
||||
sa.Column('id', sa.UUID(), nullable=False),
|
||||
sa.Column('user_id', sa.UUID(), nullable=False),
|
||||
sa.Column('gender', sa.String(length=16), nullable=False),
|
||||
sa.Column('birthdate', sa.Date(), nullable=True),
|
||||
sa.Column('city', sa.String(length=120), nullable=True),
|
||||
sa.Column('bio', sa.Text(), nullable=True),
|
||||
sa.Column('languages', postgresql.JSONB(astext_type=sa.Text()), nullable=True),
|
||||
sa.Column('interests', postgresql.JSONB(astext_type=sa.Text()), nullable=True),
|
||||
sa.Column('preferences', postgresql.JSONB(astext_type=sa.Text()), nullable=True),
|
||||
sa.Column('verification_status', sa.String(length=16), 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_profiles_user_id'), 'profiles', ['user_id'], unique=False)
|
||||
# ### end Alembic commands ###
|
||||
|
||||
|
||||
def downgrade():
|
||||
# ### commands auto generated by Alembic - please adjust! ###
|
||||
op.drop_index(op.f('ix_profiles_user_id'), table_name='profiles')
|
||||
op.drop_table('profiles')
|
||||
op.drop_index(op.f('ix_photos_profile_id'), table_name='photos')
|
||||
op.drop_table('photos')
|
||||
# ### end Alembic commands ###
|
||||
@@ -0,0 +1,6 @@
|
||||
#!/usr/bin/env sh
|
||||
set -e
|
||||
# Run migrations (no-op if no revisions yet)
|
||||
alembic -c alembic.ini upgrade head || true
|
||||
# Start app
|
||||
exec uvicorn app.main:app --host 0.0.0.0 --port 8000
|
||||
@@ -0,0 +1,6 @@
|
||||
#!/usr/bin/env sh
|
||||
set -e
|
||||
# Run migrations (no-op if no revisions yet)
|
||||
alembic -c alembic.ini upgrade head || true
|
||||
# Start app
|
||||
exec uvicorn app.main:app --host 0.0.0.0 --port 8000 --log-level debug
|
||||
@@ -0,0 +1,22 @@
|
||||
from __future__ import annotations
|
||||
import uuid
|
||||
from datetime import datetime
|
||||
|
||||
from sqlalchemy import String, Boolean, DateTime, ForeignKey
|
||||
from sqlalchemy.dialects.postgresql import UUID
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
from sqlalchemy.sql import func
|
||||
|
||||
from app.db.session import Base
|
||||
|
||||
class Photo(Base):
|
||||
__tablename__ = "photos"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
profile_id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), index=True, nullable=False)
|
||||
url: Mapped[str] = mapped_column(String(500), nullable=False)
|
||||
is_main: Mapped[bool] = mapped_column(Boolean, default=False)
|
||||
status: Mapped[str] = mapped_column(String(16), default="pending") # pending/approved/rejected
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), nullable=False)
|
||||
|
||||
profile = relationship("Profile", back_populates="photos", primaryjoin="Photo.profile_id==Profile.id", viewonly=True)
|
||||
@@ -0,0 +1,27 @@
|
||||
from __future__ import annotations
|
||||
import uuid
|
||||
from datetime import datetime
|
||||
|
||||
from sqlalchemy import String, Boolean, DateTime, ForeignKey
|
||||
from sqlalchemy.dialects.postgresql import UUID
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
from sqlalchemy.sql import func
|
||||
|
||||
from app.db.session import Base
|
||||
|
||||
class Photo(Base):
|
||||
__tablename__ = "photos"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
profile_id: Mapped[uuid.UUID] = mapped_column(
|
||||
UUID(as_uuid=True),
|
||||
ForeignKey("profiles.id", ondelete="CASCADE"),
|
||||
index=True,
|
||||
nullable=False,
|
||||
)
|
||||
url: Mapped[str] = mapped_column(String(500), nullable=False)
|
||||
is_main: Mapped[bool] = mapped_column(Boolean, default=False)
|
||||
status: Mapped[str] = mapped_column(String(16), default="pending") # pending/approved/rejected
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), nullable=False)
|
||||
|
||||
profile = relationship("Profile", back_populates="photos")
|
||||
@@ -0,0 +1,28 @@
|
||||
from __future__ import annotations
|
||||
import uuid
|
||||
from datetime import date, datetime
|
||||
|
||||
from sqlalchemy import String, Date, DateTime, Text
|
||||
from sqlalchemy.dialects.postgresql import UUID, JSONB
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
from sqlalchemy.sql import func
|
||||
|
||||
from app.db.session import Base
|
||||
|
||||
class Profile(Base):
|
||||
__tablename__ = "profiles"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
user_id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), index=True, nullable=False)
|
||||
gender: Mapped[str] = mapped_column(String(16), nullable=False) # male/female/other
|
||||
birthdate: Mapped[date | None] = mapped_column(Date, default=None)
|
||||
city: Mapped[str | None] = mapped_column(String(120), default=None)
|
||||
bio: Mapped[str | None] = mapped_column(Text, default=None)
|
||||
languages: Mapped[dict | None] = mapped_column(JSONB, default=list) # e.g., ["ru","en"]
|
||||
interests: Mapped[dict | None] = mapped_column(JSONB, default=list)
|
||||
preferences: Mapped[dict | None] = mapped_column(JSONB, default=dict)
|
||||
verification_status: Mapped[str] = mapped_column(String(16), default="unverified")
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), nullable=False)
|
||||
updated_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), onupdate=func.now(), nullable=False)
|
||||
|
||||
photos = relationship("Photo", back_populates="profile", cascade="all, delete-orphan")
|
||||
@@ -0,0 +1,29 @@
|
||||
from __future__ import annotations
|
||||
import uuid
|
||||
from datetime import date, datetime
|
||||
|
||||
from sqlalchemy import String, Date, DateTime, Text
|
||||
from sqlalchemy.dialects.postgresql import UUID, JSONB
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
from sqlalchemy.sql import func
|
||||
from typing import Optional
|
||||
|
||||
from app.db.session import Base
|
||||
|
||||
class Profile(Base):
|
||||
__tablename__ = "profiles"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
user_id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), index=True, nullable=False)
|
||||
gender: Mapped[str] = mapped_column(String(16), nullable=False) # male/female/other
|
||||
birthdate: Mapped[date | None] = mapped_column(Date, default=None)
|
||||
city: Mapped[str | None] = mapped_column(String(120), default=None)
|
||||
bio: Mapped[str | None] = mapped_column(Text, default=None)
|
||||
languages: Mapped[dict | None] = mapped_column(JSONB, default=list) # e.g., ["ru","en"]
|
||||
interests: Mapped[dict | None] = mapped_column(JSONB, default=list)
|
||||
preferences: Mapped[dict | None] = mapped_column(JSONB, default=dict)
|
||||
verification_status: Mapped[str] = mapped_column(String(16), default="unverified")
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), nullable=False)
|
||||
updated_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), onupdate=func.now(), nullable=False)
|
||||
|
||||
photos = relationship("Photo", back_populates="profile", cascade="all, delete-orphan")
|
||||
@@ -0,0 +1,29 @@
|
||||
from __future__ import annotations
|
||||
import uuid
|
||||
from datetime import date, datetime
|
||||
|
||||
from sqlalchemy import String, Date, DateTime, Text
|
||||
from sqlalchemy.dialects.postgresql import UUID, JSONB
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
from sqlalchemy.sql import func
|
||||
from typing import Optional
|
||||
|
||||
from app.db.session import Base
|
||||
|
||||
class Profile(Base):
|
||||
__tablename__ = "profiles"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
user_id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), index=True, nullable=False)
|
||||
gender: Mapped[str] = mapped_column(String(16), nullable=False) # male/female/other
|
||||
birthdate: Mapped[date | None] = mapped_column(Date, default=None)
|
||||
city: Mapped[str | None] = mapped_column(String(120), default=None)
|
||||
bio: Mapped[str | None] = mapped_column(Text, default=None)
|
||||
languages: Mapped[Optional[list[str]]] = mapped_column(JSONB, default=list)
|
||||
interests: Mapped[Optional[list[str]]] = mapped_column(JSONB, default=list)
|
||||
preferences: Mapped[dict | None] = mapped_column(JSONB, default=dict)
|
||||
verification_status: Mapped[str] = mapped_column(String(16), default="unverified")
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), nullable=False)
|
||||
updated_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), onupdate=func.now(), nullable=False)
|
||||
|
||||
photos = relationship("Photo", back_populates="profile", cascade="all, delete-orphan")
|
||||
@@ -0,0 +1,24 @@
|
||||
from __future__ import annotations
|
||||
import uuid
|
||||
from datetime import date, datetime
|
||||
|
||||
from sqlalchemy import String, Date, DateTime, Text
|
||||
from sqlalchemy.dialects.postgresql import UUID, JSONB
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
from sqlalchemy.sql import func
|
||||
from typing import Optional
|
||||
|
||||
from app.db.session import Base
|
||||
|
||||
class Profile(Base):
|
||||
__tablename__ = "profiles"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
user_id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), index=True, nullable=False)
|
||||
gender: Mapped[str] = mapped_column(String(16), nullable=False) # male/female/other
|
||||
birthdate: Mapped[date | None] = mapped_column(Date, default=None)
|
||||
city: Mapped[str | None] = mapped_column(String(120), default=None)
|
||||
bio: Mapped[str | None] = mapped_column(Text, default=None)
|
||||
languages: Mapped[Optional[list[str]]] = mapped_column(JSONB, default=list)
|
||||
interests: Mapped[Optional[list[str]]] = mapped_column(JSONB, default=list)
|
||||
preferences: Mapped[Optional[dict[str, str]]] = mapped_column(JSONB, default=dict)
|
||||
@@ -0,0 +1,29 @@
|
||||
from __future__ import annotations
|
||||
import uuid
|
||||
from datetime import date, datetime
|
||||
|
||||
from sqlalchemy import String, Date, DateTime, Text
|
||||
from sqlalchemy.dialects.postgresql import UUID, JSONB
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
from sqlalchemy.sql import func
|
||||
from typing import Optional
|
||||
|
||||
from app.db.session import Base
|
||||
|
||||
class Profile(Base):
|
||||
__tablename__ = "profiles"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
user_id: Mapped[uuid.UUID] = mapped_column(UUID(as_uuid=True), index=True, nullable=False)
|
||||
gender: Mapped[str] = mapped_column(String(16), nullable=False) # male/female/other
|
||||
birthdate: Mapped[date | None] = mapped_column(Date, default=None)
|
||||
city: Mapped[str | None] = mapped_column(String(120), default=None)
|
||||
bio: Mapped[str | None] = mapped_column(Text, default=None)
|
||||
languages: Mapped[Optional[list[str]]] = mapped_column(JSONB, default=list)
|
||||
interests: Mapped[Optional[list[str]]] = mapped_column(JSONB, default=list)
|
||||
preferences: Mapped[dict | None] = mapped_column(JSONB, default=dict)
|
||||
verification_status: Mapped[str] = mapped_column(String(16), default="unverified")
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), nullable=False)
|
||||
updated_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), onupdate=func.now(), nullable=False)
|
||||
|
||||
photos = relationship("Photo", back_populates="profile", cascade="all, delete-orphan")
|
||||
Reference in New Issue
Block a user