#!/usr/bin/env python3 # -*- coding: utf-8 -*- from datetime import datetime from typing import Annotated from sqlalchemy import BigInteger, DateTime from sqlalchemy.ext.asyncio import AsyncAttrs from sqlalchemy.orm import DeclarativeBase, Mapped, MappedAsDataclass, declared_attr, mapped_column from backend.utils.snowflake import snowflake from backend.utils.timezone import timezone # 通用 Mapped 类型主键, 需手动添加,参考以下使用方式 # MappedBase -> id: Mapped[id_key] # DataClassBase && Base -> id: Mapped[id_key] = mapped_column(init=False) id_key = Annotated[ int, mapped_column( BigInteger, primary_key=True, unique=True, index=True, autoincrement=True, sort_order=-999, comment='主键 ID', ), ] # 雪花算法 Mapped 类型主键,使用方法与 id_key 相同 # 详情:https://fastapi-practices.github.io/fastapi_best_architecture_docs/backend/reference/pk.html snowflake_id_key = Annotated[ int, mapped_column( BigInteger, primary_key=True, unique=True, index=True, default=snowflake.generate, sort_order=-999, comment='雪花算法主键 ID', ), ] # Mixin: 一种面向对象编程概念, 使结构变得更加清晰, `Wiki `__ class UserMixin(MappedAsDataclass): """用户 Mixin 数据类""" created_by: Mapped[int] = mapped_column(sort_order=998, comment='创建者') updated_by: Mapped[int | None] = mapped_column(init=False, default=None, sort_order=998, comment='修改者') class DateTimeMixin(MappedAsDataclass): """日期时间 Mixin 数据类""" created_time: Mapped[datetime] = mapped_column( DateTime(timezone=True), init=False, default_factory=timezone.now, sort_order=999, comment='创建时间' ) updated_time: Mapped[datetime | None] = mapped_column( DateTime(timezone=True), init=False, onupdate=timezone.now, sort_order=999, comment='更新时间' ) class MappedBase(AsyncAttrs, DeclarativeBase): """ 声明式基类, 作为所有基类或数据模型类的父类而存在 `AsyncAttrs `__ `DeclarativeBase `__ `mapped_column() `__ """ @declared_attr.directive def __tablename__(cls) -> str: """生成表名""" return cls.__name__.lower() @declared_attr.directive def __table_args__(cls) -> dict: """表配置""" return {'comment': cls.__doc__ or ''} class DataClassBase(MappedAsDataclass, MappedBase): """ 声明性数据类基类, 带有数据类集成, 允许使用更高级配置, 但你必须注意它的一些特性, 尤其是和 DeclarativeBase 一起使用时 `MappedAsDataclass `__ """ __abstract__ = True class Base(DataClassBase, DateTimeMixin): """ 声明性数据类基类, 带有数据类集成, 并包含 MiXin 数据类基础表结构 """ __abstract__ = True