v0.1.0: CRM/ERP 系统内测版本 - 安全加固完成

- Docker bridge 网络隔离(8000 端口封死)
- Gunicorn 4 Worker 多进程
- Alembic 数据库迁移基线
- 日志轮转 20m×3
- JWT 密钥 + DB 密码 + CORS 收紧
- 3-2-1 备份链路(NAS + R740-B 冷备)
- 连接池 pool_pre_ping + pool_recycle=3600
This commit is contained in:
hankin
2026-03-16 07:31:37 +00:00
commit 423baff73b
2578 changed files with 824643 additions and 0 deletions
View File
+35
View File
@@ -0,0 +1,35 @@
"""
Action Card 协议 v1 Schema & 回调处理
"""
from __future__ import annotations
from pydantic import BaseModel
class ActionCardField(BaseModel):
label: str
value: str
editable: bool = False
class ActionButton(BaseModel):
key: str # "confirm" | "cancel" | "edit"
label: str # "确认建单" | "取消"
style: str = "primary" # "primary" | "danger" | "default"
class ActionCardPayload(BaseModel):
card_id: str # 前端生成的唯一 ID
card_type: str # "create_order" | "create_customer" | "create_shipping" ...
title: str
summary: str
fields: list[ActionCardField] = []
actions: list[ActionButton] = []
params: dict = {} # 原始工具参数,回调时传回
class ActionCardCallback(BaseModel):
"""前端点击确认/取消后的回调请求"""
card_id: str | None = None
card_type: str
action_key: str # "confirm" | "cancel"
params: dict = {} # 可能被用户在卡片上修改过的参数
+41
View File
@@ -0,0 +1,41 @@
"""
Auth 相关 Pydantic V2 Schema
"""
from __future__ import annotations
import uuid
from pydantic import BaseModel, Field
# ── 登录请求 ──────────────────────────────────────────────
class LoginRequest(BaseModel):
username: str = Field(..., min_length=1, max_length=50, examples=["admin"])
password: str = Field(..., min_length=1, max_length=128, examples=["123456"])
# ── Token 响应 ────────────────────────────────────────────
class TokenResponse(BaseModel):
access_token: str
token_type: str = "bearer"
# ── 当前用户信息(从 JWT 解析 + DB 查表组合而来)──────────
class CurrentUserPayload(BaseModel):
"""注入到 Dependency 中的用户权限上下文"""
user_id: uuid.UUID
username: str
real_name: str | None = None
dept_id: uuid.UUID | None = None
dept_name: str | None = None
role_id: uuid.UUID | None = None
role_name: str | None = None
data_scope: str = "self" # all / dept_and_sub / self
menu_keys: list[str] = Field(default_factory=list)
# ── 修改密码请求 ────────────────────────────────────
class UpdatePasswordRequest(BaseModel):
old_password: str = Field(..., min_length=1, max_length=128)
new_password: str = Field(..., min_length=6, max_length=128, description="新密码至少6位")
+66
View File
@@ -0,0 +1,66 @@
"""
CRM 客户域 Pydantic V2 Schemas
"""
from __future__ import annotations
import uuid
from datetime import datetime
from typing import Any
from pydantic import BaseModel, Field
# ── 创建 ──────────────────────────────────────────────────
class CustomerCreate(BaseModel):
name: str = Field(..., min_length=1, max_length=200, examples=["中石化润滑油公司"])
level: str = Field(default="C", pattern=r"^[ABC]$")
industry: str | None = Field(default=None, max_length=100)
contact: str | None = Field(default=None, max_length=50)
phone: str | None = Field(default=None, max_length=30)
email: str | None = Field(default=None, max_length=100)
address: str | None = None
status: int = Field(default=1, ge=0, le=1)
# ── 更新 ──────────────────────────────────────────────────
class CustomerUpdate(BaseModel):
name: str | None = Field(default=None, min_length=1, max_length=200)
level: str | None = Field(default=None, pattern=r"^[ABC]$")
industry: str | None = Field(default=None, max_length=100)
contact: str | None = Field(default=None, max_length=50)
phone: str | None = Field(default=None, max_length=30)
email: str | None = Field(default=None, max_length=100)
address: str | None = None
status: int | None = Field(default=None, ge=0, le=1)
# ── 响应 ──────────────────────────────────────────────────
class CustomerResponse(BaseModel):
id: uuid.UUID
name: str
level: str
industry: str | None = None
contact: str | None = None
phone: str | None = None
email: str | None = None
address: str | None = None
ai_score: float = 0
ai_persona: dict[str, Any] | None = None
owner_id: uuid.UUID | None = None
owner_name: str | None = None
status: int = 1
is_deleted: bool = False
created_at: datetime
updated_at: datetime
model_config = {"from_attributes": True}
# ── 分页列表 ─────────────────────────────────────────────
class CustomerListResponse(BaseModel):
total: int
items: list[CustomerResponse]
page: int
size: int
+121
View File
@@ -0,0 +1,121 @@
"""
ERP 供应链域 Pydantic V2 Schemas
分类树 / 产品 SKU / 库存流水
"""
from __future__ import annotations
import uuid
from datetime import datetime
from pydantic import BaseModel, Field
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 分类树
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class CategoryNode(BaseModel):
"""递归树节点,适配前端 el-tree"""
id: uuid.UUID
parent_id: uuid.UUID | None = None
name: str
sort_order: int = 0
children: list[CategoryNode] = Field(default_factory=list)
model_config = {"from_attributes": True}
class CategoryCreate(BaseModel):
name: str = Field(..., min_length=1, max_length=100)
parent_id: uuid.UUID | None = None
sort_order: int = Field(default=0)
class CategoryUpdate(BaseModel):
name: str | None = Field(default=None, min_length=1, max_length=100)
parent_id: uuid.UUID | None = None
sort_order: int | None = None
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 产品 SKU
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class SkuCreate(BaseModel):
sku_code: str = Field(..., min_length=1, max_length=50, examples=["SKU-LUB-001"])
name: str = Field(..., min_length=1, max_length=200, examples=["昆仑天润 KR9"])
category_id: uuid.UUID | None = None
spec: str | None = Field(default=None, max_length=100, examples=["200L/桶"])
standard_price: float = Field(default=0, ge=0)
stock_qty: float = Field(default=0, ge=0, description="初始库存(仅创建时可设,后续只能通过流水变更)")
warning_threshold: float = Field(default=0, ge=0)
unit: str = Field(default="", max_length=20)
status: int = Field(default=1, ge=0, le=1)
class SkuUpdate(BaseModel):
"""
⚠️ 架构师红线:不包含 stock_qty 字段!
库存只能通过 InventoryFlow 事务接口变更。
"""
name: str | None = Field(default=None, min_length=1, max_length=200)
category_id: uuid.UUID | None = None
spec: str | None = Field(default=None, max_length=100)
standard_price: float | None = Field(default=None, ge=0)
warning_threshold: float | None = Field(default=None, ge=0)
unit: str | None = Field(default=None, max_length=20)
status: int | None = Field(default=None, ge=0, le=1)
class SkuResponse(BaseModel):
id: uuid.UUID
sku_code: str
name: str
category_id: uuid.UUID | None = None
category_name: str | None = None
spec: str | None = None
standard_price: float = 0
stock_qty: float = 0
warning_threshold: float = 0
unit: str = ""
status: int = 1
created_at: datetime
updated_at: datetime
model_config = {"from_attributes": True}
class SkuListResponse(BaseModel):
total: int
items: list[SkuResponse]
page: int
size: int
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 库存流水
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class InventoryFlowCreate(BaseModel):
"""库存变更请求"""
sku_id: uuid.UUID = Field(..., description="产品 SKU ID")
change_qty: float = Field(..., description="变更数量,正=入库 负=出库")
reason: str = Field(
..., max_length=50,
description="变动原因: purchase/shipment/loss/adjust",
examples=["purchase"],
)
remark: str | None = Field(default=None, description="备注")
class InventoryFlowResponse(BaseModel):
id: uuid.UUID
sku_id: uuid.UUID
sku_code: str | None = None
sku_name: str | None = None
change_qty: float
reason: str
remark: str | None = None
operator_id: uuid.UUID | None = None
operator_name: str | None = None
created_at: datetime
model_config = {"from_attributes": True}
+139
View File
@@ -0,0 +1,139 @@
"""
财务票据域 Pydantic V2 Schemas
票据池 / 报销单(嵌套明细)/ 审批状态变更
"""
from __future__ import annotations
import uuid
from datetime import date, datetime
from typing import Any
from pydantic import BaseModel, Field
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 票据池
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class InvoiceCreate(BaseModel):
file_url: str | None = Field(default=None, max_length=500)
merchant_name: str | None = Field(default=None, max_length=200)
amount: float = Field(default=0, ge=0)
invoice_date: date | None = None
type: str = Field(default="expense", pattern=r"^(expense|customer)$")
ai_extracted_data: dict[str, Any] = Field(
default_factory=dict,
description="OCR/AI 解析的结构化数据 (JSONB)",
examples=[{"merchant": "中国石化", "amount": 580.00, "date": "2026-02-20"}],
)
class InvoiceResponse(BaseModel):
id: uuid.UUID
uploader_id: uuid.UUID | None = None
uploader_name: str | None = None
file_url: str | None = None
merchant_name: str | None = None
amount: float = 0
invoice_date: date | None = None
type: str
ai_extracted_data: dict[str, Any] = Field(default_factory=dict)
is_used: bool = False
created_at: datetime
model_config = {"from_attributes": True}
class InvoiceListResponse(BaseModel):
total: int
items: list[InvoiceResponse]
page: int
size: int
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 报销单创建(嵌套明细)
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class ExpenseDetailCreate(BaseModel):
invoice_id: uuid.UUID = Field(..., description="关联发票 ID")
expense_desc: str | None = Field(default=None, max_length=500)
expense_date: date | None = Field(default=None, description="费用发生日期")
original_type: str | None = Field(
default=None, max_length=50, examples=["fuel", "entertainment", "travel", "office"]
)
offset_type: str | None = Field(default=None, max_length=50)
amount: float = Field(..., gt=0, description="本行报销金额")
class ExpenseCreate(BaseModel):
total_amount: float = Field(..., gt=0)
remark: str | None = None
items: list[ExpenseDetailCreate] = Field(..., min_length=1)
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 报销明细响应
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class ExpenseDetailResponse(BaseModel):
id: uuid.UUID
invoice_id: uuid.UUID | None = None
invoice_merchant: str | None = None
invoice_amount: float | None = None
expense_desc: str | None = None
expense_date: date | None = None
original_type: str | None = None
offset_type: str | None = None
amount: float = 0
model_config = {"from_attributes": True}
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 报销单响应
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class ExpenseResponse(BaseModel):
id: uuid.UUID
system_no: str
applicant_id: uuid.UUID
applicant_name: str | None = None
total_amount: float
status: str
remark: str | None = None
approved_by: uuid.UUID | None = None
approver_name: str | None = None
approved_at: datetime | None = None
details: list[ExpenseDetailResponse] = Field(default_factory=list)
created_at: datetime
updated_at: datetime
model_config = {"from_attributes": True}
class ExpenseBriefResponse(BaseModel):
id: uuid.UUID
system_no: str
applicant_id: uuid.UUID
applicant_name: str | None = None
total_amount: float
status: str
created_at: datetime
model_config = {"from_attributes": True}
class ExpenseListResponse(BaseModel):
total: int
items: list[ExpenseBriefResponse]
page: int
size: int
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 审批状态变更
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class ExpenseStatusUpdate(BaseModel):
action: str = Field(
..., pattern=r"^(withdraw|approve|reject)$",
description="状态变更动作: withdraw/approve/reject",
)
reason: str | None = Field(default=None, description="审批意见或驳回原因")
+111
View File
@@ -0,0 +1,111 @@
"""
ERP 交易域 Pydantic V2 Schemas
订单创建(嵌套明细)/ 详情响应 / 动态定价
"""
from __future__ import annotations
import uuid
from datetime import date, datetime
from pydantic import BaseModel, Field
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# B2B 动态定价
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class PriceCalculateRequest(BaseModel):
customer_id: uuid.UUID
sku_id: uuid.UUID
class PriceCalculateResponse(BaseModel):
sku_id: uuid.UUID
sku_code: str
sku_name: str
unit_price: float
price_source: str # "history" | "standard"
last_order_no: str | None = None
last_order_date: date | None = None
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 订单创建(嵌套明细)
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class OrderItemCreate(BaseModel):
sku_id: uuid.UUID
qty: float = Field(..., gt=0, description="订购数量")
unit_price: float = Field(..., ge=0, description="成交单价")
class OrderCreate(BaseModel):
customer_id: uuid.UUID
remark: str | None = None
order_date: date | None = None
items: list[OrderItemCreate] = Field(..., min_length=1, description="至少一个明细行")
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 订单明细响应
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class OrderItemResponse(BaseModel):
id: uuid.UUID
sku_id: uuid.UUID
sku_code: str | None = None
sku_name: str | None = None
spec: str | None = None
qty: float
unit_price: float
sub_total: float
shipped_qty: float = 0
model_config = {"from_attributes": True}
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 订单主表响应
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class OrderResponse(BaseModel):
id: uuid.UUID
order_no: str
customer_id: uuid.UUID
customer_name: str | None = None
salesperson_id: uuid.UUID | None = None
salesperson_name: str | None = None
total_amount: float
shipping_state: str
payment_state: str
paid_amount: float = 0
remark: str | None = None
order_date: date
items: list[OrderItemResponse] = Field(default_factory=list)
created_at: datetime
updated_at: datetime
model_config = {"from_attributes": True}
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 订单列表(不含 items 明细,减少传输体积)
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class OrderBriefResponse(BaseModel):
id: uuid.UUID
order_no: str
customer_id: uuid.UUID
customer_name: str | None = None
salesperson_name: str | None = None
total_amount: float
shipping_state: str
payment_state: str
paid_amount: float = 0
order_date: date
created_at: datetime
model_config = {"from_attributes": True}
class OrderListResponse(BaseModel):
total: int
items: list[OrderBriefResponse]
page: int
size: int
+81
View File
@@ -0,0 +1,81 @@
"""
V5.0 AI 画像强约束 Pydantic Schema
- CompanyPersona: 企业级画像 (crm_customers.ai_persona)
- BuyerPersona: 联系人级画像 (crm_contacts.ai_buyer_persona)
"""
from __future__ import annotations
from pydantic import BaseModel, Field
# ═══════════════════════════════════════════════════════
# 企业画像 (crm_customers.ai_persona)
# ═══════════════════════════════════════════════════════
class Firmographics(BaseModel):
"""企业属性"""
industry: str | None = None
scale: str | None = None
business_model: str | None = None
class DynamicStatus(BaseModel):
"""时序动态"""
pain_points: list[str] = Field(default_factory=list)
purchase_intent: str | None = None
recent_events: list[str] = Field(default_factory=list)
class CompanyPersona(BaseModel):
"""企业级 AI 画像 Schema"""
firmographics: Firmographics = Field(default_factory=Firmographics)
dynamic_status: DynamicStatus = Field(default_factory=DynamicStatus)
summary: str | None = None
# ═══════════════════════════════════════════════════════
# 联系人画像 (crm_contacts.ai_buyer_persona)
# ═══════════════════════════════════════════════════════
class BuyerRole(BaseModel):
"""决策角色"""
decision_role: str | None = None # 决策者 / 影响者 / 执行者
authority_level: str | None = None # 高 / 中 / 低
class BuyerKPI(BaseModel):
"""核心痛点与目标"""
core_goals: list[str] = Field(default_factory=list)
pain_points: list[str] = Field(default_factory=list)
class BuyerPreference(BaseModel):
"""交互偏好"""
comm_style: str | None = None
meeting_preference: str | None = None
topics_of_interest: list[str] = Field(default_factory=list)
class BuyerPersona(BaseModel):
"""联系人级 AI 画像 Schema"""
role: BuyerRole = Field(default_factory=BuyerRole)
kpi: BuyerKPI = Field(default_factory=BuyerKPI)
preference: BuyerPreference = Field(default_factory=BuyerPreference)
# ═══════════════════════════════════════════════════════
# 双轨画像回写 PayloadDify Workflow 回调用)
# ═══════════════════════════════════════════════════════
class ContactPersonaUpdate(BaseModel):
"""单个联系人画像增量"""
contact_id: str
role: dict | None = None
kpi: dict | None = None
preference: dict | None = None
class PersonaUpdatePayload(BaseModel):
"""双轨画像回写请求体"""
company_updates: dict | None = None
contact_updates: list[ContactPersonaUpdate] | None = None
+22
View File
@@ -0,0 +1,22 @@
"""
统一响应 Schema
"""
from __future__ import annotations
from typing import Any, Generic, TypeVar
from pydantic import BaseModel
T = TypeVar("T")
class ApiResponse(BaseModel, Generic[T]):
code: int = 200
data: T | None = None
message: str = "ok"
def ok(data: Any = None, message: str = "ok") -> dict:
"""快捷返回成功响应"""
return {"code": 200, "data": data, "message": message}
+55
View File
@@ -0,0 +1,55 @@
"""
销项发票 Pydantic V2 Schemas
"""
from __future__ import annotations
import uuid
from datetime import date, datetime
from pydantic import BaseModel, Field
class SalesInvoiceCreate(BaseModel):
issuer: str = Field(..., min_length=1, max_length=200, examples=["XX润滑油有限公司"])
receiver_customer_id: uuid.UUID = Field(..., description="受票方客户 ID")
invoice_number: str = Field(..., min_length=1, max_length=100, examples=["INV-20260312-001"])
amount: float = Field(..., gt=0, description="票面金额")
billing_date: date = Field(..., description="开票日期")
remark: str | None = None
class SalesInvoiceUpdate(BaseModel):
payment_status: str | None = Field(
default=None, pattern=r"^(未回款|部分回款|已结清)$",
description="回款状态",
)
payment_date: date | None = None
payment_amount: float | None = Field(default=None, ge=0, description="已回款金额")
remark: str | None = None
class SalesInvoiceResponse(BaseModel):
id: uuid.UUID
issuer: str
receiver_customer_id: uuid.UUID
customer_name: str | None = None
invoice_number: str
amount: float
billing_date: date
payment_status: str = "未回款"
payment_date: date | None = None
payment_amount: float = 0
remark: str | None = None
created_by: uuid.UUID | None = None
creator_name: str | None = None
created_at: datetime
updated_at: datetime
model_config = {"from_attributes": True}
class SalesInvoiceListResponse(BaseModel):
total: int
items: list[SalesInvoiceResponse]
page: int
size: int
+95
View File
@@ -0,0 +1,95 @@
"""
ERP 物流域 Pydantic V2 Schemas
发货单创建(嵌套明细)/ 列表 / 详情
"""
from __future__ import annotations
import uuid
from datetime import date, datetime
from pydantic import BaseModel, Field
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 发货明细创建
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class ShippingItemCreate(BaseModel):
order_item_id: uuid.UUID = Field(..., description="关联的订单明细行 ID")
sku_id: uuid.UUID = Field(..., description="产品 SKU ID")
shipped_qty: float = Field(..., gt=0, description="本次发货数量,必须 > 0")
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 发货单创建
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class ShippingCreate(BaseModel):
order_id: uuid.UUID = Field(..., description="关联订单 ID")
carrier: str | None = Field(default=None, max_length=100, examples=["德邦物流"])
tracking_no: str | None = Field(default=None, max_length=100, examples=["DB202602270001"])
ship_date: date | None = None
remark: str | None = None
items: list[ShippingItemCreate] = Field(..., min_length=1, description="至少一个发货明细行")
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 发货明细响应
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class ShippingItemResponse(BaseModel):
id: uuid.UUID
order_item_id: uuid.UUID
sku_id: uuid.UUID
sku_code: str | None = None
sku_name: str | None = None
spec: str | None = None
unit: str | None = None
shipped_qty: float
model_config = {"from_attributes": True}
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 发货单响应
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class ShippingResponse(BaseModel):
id: uuid.UUID
shipping_no: str
order_id: uuid.UUID
order_no: str | None = None
customer_name: str | None = None
carrier: str | None = None
tracking_no: str | None = None
status: str
ship_date: date
remark: str | None = None
operator_name: str | None = None
items: list[ShippingItemResponse] = Field(default_factory=list)
created_at: datetime
model_config = {"from_attributes": True}
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 发货列表(Brief,不带明细减少体积)
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class ShippingBriefResponse(BaseModel):
id: uuid.UUID
shipping_no: str
order_id: uuid.UUID
order_no: str | None = None
customer_name: str | None = None
carrier: str | None = None
tracking_no: str | None = None
status: str
ship_date: date
operator_name: str | None = None
created_at: datetime
model_config = {"from_attributes": True}
class ShippingListResponse(BaseModel):
total: int
items: list[ShippingBriefResponse]
page: int
size: int
+114
View File
@@ -0,0 +1,114 @@
"""
系统设置域 Pydantic V2 Schemas
部门树 / 角色(含 JSONB menu_keys/ 用户管理
"""
from __future__ import annotations
import uuid
from datetime import datetime
from pydantic import BaseModel, Field
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 部门树
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class DeptNode(BaseModel):
id: uuid.UUID
parent_id: uuid.UUID | None = None
name: str
sort_order: int = 0
status: int = 1
children: list[DeptNode] = Field(default_factory=list)
model_config = {"from_attributes": True}
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 角色
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class RoleCreate(BaseModel):
role_name: str = Field(..., min_length=1, max_length=50)
data_scope: str = Field(
default="self", pattern=r"^(all|dept_and_sub|self)$"
)
menu_keys: list[str] = Field(
default_factory=list,
description="前端路由 name 数组,JSONB 存储",
examples=[["CustomerList", "OrderList", "ProductList"]],
)
description: str | None = Field(default=None, max_length=255)
status: int = Field(default=1, ge=0, le=1)
class RoleUpdate(BaseModel):
role_name: str | None = Field(default=None, min_length=1, max_length=50)
data_scope: str | None = Field(default=None, pattern=r"^(all|dept_and_sub|self)$")
menu_keys: list[str] | None = None
description: str | None = Field(default=None, max_length=255)
status: int | None = Field(default=None, ge=0, le=1)
class RoleResponse(BaseModel):
id: uuid.UUID
role_name: str
data_scope: str
menu_keys: list[str] = Field(default_factory=list)
description: str | None = None
status: int = 1
created_at: datetime
model_config = {"from_attributes": True}
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
# 用户/员工
# ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
class UserCreate(BaseModel):
username: str = Field(..., min_length=2, max_length=50)
password: str = Field(..., min_length=6, max_length=128, description="明文密码,后端 bcrypt 哈希后存储")
real_name: str | None = Field(default=None, max_length=50)
phone: str | None = Field(default=None, max_length=20)
email: str | None = Field(default=None, max_length=100)
dept_id: uuid.UUID | None = None
role_id: uuid.UUID | None = None
status: int = Field(default=1, ge=0, le=1)
class UserUpdate(BaseModel):
real_name: str | None = Field(default=None, max_length=50)
phone: str | None = Field(default=None, max_length=20)
email: str | None = Field(default=None, max_length=100)
dept_id: uuid.UUID | None = None
role_id: uuid.UUID | None = None
status: int | None = Field(default=None, ge=0, le=1)
class UserResetPassword(BaseModel):
new_password: str = Field(..., min_length=6, max_length=128)
class UserResponse(BaseModel):
id: uuid.UUID
username: str
real_name: str | None = None
phone: str | None = None
email: str | None = None
dept_id: uuid.UUID | None = None
dept_name: str | None = None
role_id: uuid.UUID | None = None
role_name: str | None = None
data_scope: str | None = None
status: int = 1
last_login_at: datetime | None = None
created_at: datetime
model_config = {"from_attributes": True}
class UserListResponse(BaseModel):
total: int
items: list[UserResponse]
page: int
size: int