| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- from datetime import datetime
- from typing import TYPE_CHECKING
- from sqlalchemy import Boolean, DateTime, ForeignKey, Integer, String
- from sqlalchemy.orm import Mapped, mapped_column, relationship
- from app.core.base_model import MappedBase, ModelMixin, TenantMixin, UserMixin
- if TYPE_CHECKING:
- from app.api.v1.module_system.dept.model import DeptModel
- from app.api.v1.module_system.position.model import PositionModel
- from app.api.v1.module_system.role.model import RoleModel
- from app.api.v1.module_system.tenant.model import TenantModel
- class UserRolesModel(MappedBase):
- """
- 用户角色关联表
- 定义用户与角色的多对多关系
- """
- __tablename__: str = "sys_user_roles"
- __table_args__: dict[str, str] = {"comment": "用户角色关联表"}
- user_id: Mapped[int] = mapped_column(
- Integer,
- ForeignKey("sys_user.id", ondelete="CASCADE", onupdate="CASCADE"),
- primary_key=True,
- comment="用户ID",
- )
- role_id: Mapped[int] = mapped_column(
- Integer,
- ForeignKey("sys_role.id", ondelete="CASCADE", onupdate="CASCADE"),
- primary_key=True,
- comment="角色ID",
- )
- class UserPositionsModel(MappedBase):
- """
- 用户岗位关联表
- 定义用户与岗位的多对多关系
- """
- __tablename__: str = "sys_user_positions"
- __table_args__: dict[str, str] = {"comment": "用户岗位关联表"}
- user_id: Mapped[int] = mapped_column(
- Integer,
- ForeignKey("sys_user.id", ondelete="CASCADE", onupdate="CASCADE"),
- primary_key=True,
- comment="用户ID",
- )
- position_id: Mapped[int] = mapped_column(
- Integer,
- ForeignKey("sys_position.id", ondelete="CASCADE", onupdate="CASCADE"),
- primary_key=True,
- comment="岗位ID",
- )
- class UserModel(ModelMixin, TenantMixin, UserMixin):
- """
- 用户模型
- """
- __tablename__: str = "sys_user"
- __table_args__: dict[str, str] = {"comment": "用户表"}
- __loader_options__: list[str] = [
- "tenant",
- "dept",
- "roles",
- "positions",
- "created_by",
- "updated_by",
- ]
- username: Mapped[str] = mapped_column(
- String(64), nullable=False, unique=True, comment="用户名/登录账号"
- )
- password: Mapped[str] = mapped_column(String(255), nullable=False, comment="密码哈希")
- name: Mapped[str] = mapped_column(String(32), nullable=False, comment="昵称")
- mobile: Mapped[str | None] = mapped_column(
- String(11), nullable=True, unique=True, comment="手机号"
- )
- email: Mapped[str | None] = mapped_column(
- String(64), nullable=True, unique=True, comment="邮箱"
- )
- gender: Mapped[str | None] = mapped_column(
- String(1), default="2", nullable=True, comment="性别(0:男 1:女 2:未知)"
- )
- avatar: Mapped[str | None] = mapped_column(String(255), nullable=True, comment="头像URL地址")
- is_superuser: Mapped[bool] = mapped_column(
- Boolean, default=False, nullable=False, comment="是否超管"
- )
- last_login: Mapped[datetime | None] = mapped_column(
- DateTime(timezone=True), nullable=True, comment="最后登录时间"
- )
- gitee_login: Mapped[str | None] = mapped_column(String(32), nullable=True, comment="Gitee登录")
- github_login: Mapped[str | None] = mapped_column(
- String(32), nullable=True, comment="Github登录"
- )
- wx_login: Mapped[str | None] = mapped_column(String(32), nullable=True, comment="微信登录")
- qq_login: Mapped[str | None] = mapped_column(String(32), nullable=True, comment="QQ登录")
- dept_id: Mapped[int | None] = mapped_column(
- Integer,
- ForeignKey("sys_dept.id", ondelete="SET NULL", onupdate="CASCADE"),
- nullable=True,
- index=True,
- comment="部门ID",
- )
- tenant: Mapped["TenantModel | None"] = relationship(
- "TenantModel",
- foreign_keys="UserModel.tenant_id",
- lazy="selectin",
- viewonly=True,
- )
- dept: Mapped["DeptModel | None"] = relationship(
- back_populates="users", foreign_keys=[dept_id], lazy="selectin"
- )
- roles: Mapped[list["RoleModel"]] = relationship(
- secondary="sys_user_roles", back_populates="users", lazy="selectin"
- )
- positions: Mapped[list["PositionModel"]] = relationship(
- secondary="sys_user_positions", back_populates="users", lazy="selectin"
- )
- # 覆盖 UserMixin 的关系定义,显式指定 foreign_keys 避免自引用混淆
- created_by: Mapped["UserModel | None"] = relationship(
- "UserModel",
- foreign_keys="UserModel.created_id",
- remote_side="UserModel.id",
- lazy="selectin",
- uselist=False,
- viewonly=True, # 防止级联操作
- )
- updated_by: Mapped["UserModel | None"] = relationship(
- "UserModel",
- foreign_keys="UserModel.updated_id",
- remote_side="UserModel.id",
- lazy="selectin",
- uselist=False,
- viewonly=True, # 防止级联操作
- )
|