159 lines
8.3 KiB
Python
159 lines
8.3 KiB
Python
# /opt/docker/dev/service_finder/backend/app/models/identity.py
|
|
import uuid
|
|
import enum
|
|
from datetime import datetime
|
|
from typing import Any, List, Optional
|
|
from sqlalchemy import String, Boolean, DateTime, ForeignKey, JSON, Numeric, text, Integer, BigInteger, UniqueConstraint
|
|
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
|
from sqlalchemy.dialects.postgresql import UUID as PG_UUID, ENUM as PG_ENUM
|
|
from sqlalchemy.sql import func
|
|
|
|
# MB 2.0: Központi aszinkron adatbázis motorból húzzuk be a Base-t
|
|
from app.database import Base
|
|
|
|
class UserRole(str, enum.Enum):
|
|
superadmin = "superadmin"
|
|
admin = "admin"
|
|
region_admin = "region_admin"
|
|
country_admin = "country_admin"
|
|
moderator = "moderator"
|
|
sales_agent = "sales_agent"
|
|
user = "user"
|
|
service_owner = "service_owner"
|
|
fleet_manager = "fleet_manager"
|
|
driver = "driver"
|
|
|
|
class Person(Base):
|
|
"""
|
|
Természetes személy identitása. A DNS szint.
|
|
Minden identitás adat az 'identity' sémába kerül.
|
|
"""
|
|
__tablename__ = "persons"
|
|
__table_args__ = {"schema": "identity"}
|
|
|
|
id: Mapped[int] = mapped_column(BigInteger, primary_key=True, index=True)
|
|
id_uuid: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), default=uuid.uuid4, unique=True, nullable=False)
|
|
|
|
# A lakcím a 'data' sémában marad
|
|
address_id: Mapped[Optional[uuid.UUID]] = mapped_column(PG_UUID(as_uuid=True), ForeignKey("data.addresses.id"))
|
|
|
|
identity_hash: Mapped[Optional[str]] = mapped_column(String(64), unique=True, index=True)
|
|
|
|
last_name: Mapped[str] = mapped_column(String, nullable=False)
|
|
first_name: Mapped[str] = mapped_column(String, nullable=False)
|
|
phone: Mapped[Optional[str]] = mapped_column(String)
|
|
|
|
mothers_last_name: Mapped[Optional[str]] = mapped_column(String)
|
|
mothers_first_name: Mapped[Optional[str]] = mapped_column(String)
|
|
birth_place: Mapped[Optional[str]] = mapped_column(String)
|
|
birth_date: Mapped[Optional[datetime]] = mapped_column(DateTime)
|
|
|
|
identity_docs: Mapped[Any] = mapped_column(JSON, server_default=text("'{}'::jsonb"))
|
|
ice_contact: Mapped[Any] = mapped_column(JSON, server_default=text("'{}'::jsonb"))
|
|
|
|
lifetime_xp: Mapped[int] = mapped_column(BigInteger, server_default=text("0"))
|
|
penalty_points: Mapped[int] = mapped_column(Integer, server_default=text("0"))
|
|
social_reputation: Mapped[float] = mapped_column(Numeric(3, 2), server_default=text("1.00"))
|
|
|
|
is_sales_agent: Mapped[bool] = mapped_column(Boolean, server_default=text("false"))
|
|
is_active: Mapped[bool] = mapped_column(Boolean, default=True, nullable=False)
|
|
is_ghost: Mapped[bool] = mapped_column(Boolean, default=False, nullable=False)
|
|
|
|
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
|
|
updated_at: Mapped[Optional[datetime]] = mapped_column(DateTime(timezone=True), onupdate=func.now())
|
|
|
|
# Kapcsolatok
|
|
users: Mapped[List["User"]] = relationship("User", back_populates="person")
|
|
memberships: Mapped[List["OrganizationMember"]] = relationship("OrganizationMember", back_populates="person")
|
|
|
|
class User(Base):
|
|
""" Login entitás. Bármikor törölhető (GDPR), de Person-höz kötött. """
|
|
__tablename__ = "users"
|
|
__table_args__ = {"schema": "identity"}
|
|
|
|
id: Mapped[int] = mapped_column(Integer, primary_key=True, index=True)
|
|
email: Mapped[str] = mapped_column(String, unique=True, index=True, nullable=False)
|
|
hashed_password: Mapped[Optional[str]] = mapped_column(String)
|
|
|
|
role: Mapped[UserRole] = mapped_column(
|
|
PG_ENUM(UserRole, name="userrole", schema="identity"),
|
|
default=UserRole.user
|
|
)
|
|
|
|
# MB 2.0 JAVÍTÁS: A hivatkozások az identity sémára mutatnak!
|
|
person_id: Mapped[Optional[int]] = mapped_column(BigInteger, ForeignKey("identity.persons.id"))
|
|
|
|
subscription_plan: Mapped[str] = mapped_column(String(30), server_default=text("'FREE'"))
|
|
subscription_expires_at: Mapped[Optional[datetime]] = mapped_column(DateTime(timezone=True))
|
|
is_vip: Mapped[bool] = mapped_column(Boolean, server_default=text("false"))
|
|
|
|
referral_code: Mapped[Optional[str]] = mapped_column(String(20), unique=True)
|
|
|
|
# MB 2.0 JAVÍTÁS: Önhivatkozások az identity sémán belül
|
|
referred_by_id: Mapped[Optional[int]] = mapped_column(Integer, ForeignKey("identity.users.id"))
|
|
current_sales_agent_id: Mapped[Optional[int]] = mapped_column(Integer, ForeignKey("identity.users.id"))
|
|
|
|
is_active: Mapped[bool] = mapped_column(Boolean, default=False)
|
|
is_deleted: Mapped[bool] = mapped_column(Boolean, default=False)
|
|
folder_slug: Mapped[Optional[str]] = mapped_column(String(12), unique=True, index=True)
|
|
|
|
preferred_language: Mapped[str] = mapped_column(String(5), server_default="hu")
|
|
region_code: Mapped[str] = mapped_column(String(5), server_default="HU")
|
|
preferred_currency: Mapped[str] = mapped_column(String(3), server_default="HUF")
|
|
|
|
scope_level: Mapped[str] = mapped_column(String(30), server_default="individual")
|
|
scope_id: Mapped[Optional[str]] = mapped_column(String(50))
|
|
custom_permissions: Mapped[Any] = mapped_column(JSON, server_default=text("'{}'::jsonb"))
|
|
|
|
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
|
|
|
|
# Kapcsolatok
|
|
person: Mapped[Optional["Person"]] = relationship("Person", back_populates="users")
|
|
wallet: Mapped[Optional["Wallet"]] = relationship("Wallet", back_populates="user", uselist=False)
|
|
social_accounts: Mapped[List["SocialAccount"]] = relationship("SocialAccount", back_populates="user", cascade="all, delete-orphan")
|
|
owned_organizations: Mapped[List["Organization"]] = relationship("Organization", back_populates="owner")
|
|
stats: Mapped[Optional["UserStats"]] = relationship("UserStats", back_populates="user", uselist=False, cascade="all, delete-orphan")
|
|
ownership_history: Mapped[List["VehicleOwnership"]] = relationship("VehicleOwnership", back_populates="user")
|
|
|
|
class Wallet(Base):
|
|
__tablename__ = "wallets"
|
|
__table_args__ = {"schema": "identity"}
|
|
|
|
id: Mapped[int] = mapped_column(Integer, primary_key=True, index=True)
|
|
user_id: Mapped[int] = mapped_column(Integer, ForeignKey("identity.users.id"), unique=True)
|
|
|
|
earned_credits: Mapped[float] = mapped_column(Numeric(18, 4), server_default=text("0"))
|
|
purchased_credits: Mapped[float] = mapped_column(Numeric(18, 4), server_default=text("0"))
|
|
service_coins: Mapped[float] = mapped_column(Numeric(18, 4), server_default=text("0"))
|
|
|
|
currency: Mapped[str] = mapped_column(String(3), default="HUF")
|
|
user: Mapped["User"] = relationship("User", back_populates="wallet")
|
|
|
|
class VerificationToken(Base):
|
|
__tablename__ = "verification_tokens"
|
|
__table_args__ = {"schema": "identity"}
|
|
|
|
id: Mapped[int] = mapped_column(Integer, primary_key=True, index=True)
|
|
token: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), default=uuid.uuid4, unique=True, nullable=False)
|
|
user_id: Mapped[int] = mapped_column(Integer, ForeignKey("identity.users.id", ondelete="CASCADE"), nullable=False)
|
|
token_type: Mapped[str] = mapped_column(String(20), nullable=False)
|
|
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
|
|
expires_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), nullable=False)
|
|
is_used: Mapped[bool] = mapped_column(Boolean, default=False)
|
|
|
|
class SocialAccount(Base):
|
|
__tablename__ = "social_accounts"
|
|
__table_args__ = (
|
|
UniqueConstraint('provider', 'social_id', name='uix_social_provider_id'),
|
|
{"schema": "identity"}
|
|
)
|
|
|
|
id: Mapped[int] = mapped_column(Integer, primary_key=True, index=True)
|
|
user_id: Mapped[int] = mapped_column(Integer, ForeignKey("identity.users.id", ondelete="CASCADE"), nullable=False)
|
|
provider: Mapped[str] = mapped_column(String(50), nullable=False)
|
|
social_id: Mapped[str] = mapped_column(String(255), nullable=False, index=True)
|
|
email: Mapped[str] = mapped_column(String(255), nullable=False)
|
|
extra_data: Mapped[Any] = mapped_column(JSON, server_default=text("'{}'::jsonb"))
|
|
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
|
|
|
|
user: Mapped["User"] = relationship("User", back_populates="social_accounts") |