Your Name 4caafdb50f feat: 添加后端核心模块
用户认证:
- User 模型(支持邮箱/手机号登录)
- 双 Token JWT 认证(accessToken + refreshToken)
- 注册/登录/刷新 Token API

组织模型:
- Brand(品牌方)、Agency(代理商)、Creator(达人)
- 多对多关系:品牌方↔代理商、代理商↔达人

项目与任务:
- Project 模型(品牌方发布)
- Task 模型(完整审核流程追踪)
- Brief 模型(解析后的结构化内容)

文件上传:
- 阿里云 OSS 直传签名服务
- 支持分片上传,最大 500MB

数据库迁移:
- 003_user_org_project_task.py

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-02-09 13:47:36 +08:00

75 lines
2.3 KiB
Python

"""
项目模型
"""
from typing import TYPE_CHECKING, Optional
from datetime import datetime
from sqlalchemy import String, Text, ForeignKey, DateTime, Table, Column, Boolean
from sqlalchemy.orm import Mapped, mapped_column, relationship
from app.models.base import Base, TimestampMixin
if TYPE_CHECKING:
from app.models.organization import Brand, Agency
from app.models.task import Task
from app.models.brief import Brief
# 项目-代理商 关联表(一个项目可以分配给多个代理商)
project_agency_association = Table(
"project_agency",
Base.metadata,
Column("project_id", String(64), ForeignKey("projects.id", ondelete="CASCADE"), primary_key=True),
Column("agency_id", String(64), ForeignKey("agencies.id", ondelete="CASCADE"), primary_key=True),
Column("created_at", DateTime(timezone=True), default=datetime.utcnow),
Column("is_active", Boolean, default=True),
)
class Project(Base, TimestampMixin):
"""项目表"""
__tablename__ = "projects"
id: Mapped[str] = mapped_column(String(64), primary_key=True)
brand_id: Mapped[str] = mapped_column(
String(64),
ForeignKey("brands.id", ondelete="CASCADE"),
nullable=False,
index=True,
)
# 项目信息
name: Mapped[str] = mapped_column(String(255), nullable=False)
description: Mapped[Optional[str]] = mapped_column(Text, nullable=True)
# 时间
start_date: Mapped[Optional[datetime]] = mapped_column(DateTime(timezone=True), nullable=True)
deadline: Mapped[Optional[datetime]] = mapped_column(DateTime(timezone=True), nullable=True)
# 状态
status: Mapped[str] = mapped_column(
String(20),
default="active", # active, completed, archived
nullable=False,
index=True,
)
# 关联
brand: Mapped["Brand"] = relationship("Brand", back_populates="projects")
agencies: Mapped[list["Agency"]] = relationship(
"Agency",
secondary=project_agency_association,
backref="projects",
)
tasks: Mapped[list["Task"]] = relationship(
"Task",
back_populates="project",
)
brief: Mapped[Optional["Brief"]] = relationship(
"Brief",
back_populates="project",
uselist=False,
)
def __repr__(self) -> str:
return f"<Project(id={self.id}, name={self.name})>"