包含多个会话的累积代码变更: - backend: AI 聊天服务、触发器调度、认证增强、WebSocket、调度器最小间隔 - admin-web: ETL 状态页、任务管理、调度配置、登录优化 - miniprogram: 看板页面、聊天集成、UI 组件、导航更新 - etl: DWS 新任务(finance_area_daily/board_cache)、连接器增强 - tenant-admin: 项目初始化 - db: 19 个迁移脚本(etl_feiqiu 11 + zqyy_app 8) - packages/shared: 枚举和工具函数更新 - tools: 数据库工具、报表生成、健康检查 - docs: PRD/架构/部署/合约文档更新 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
13 KiB
前后端联调规范手册
最后更新:2026-04-01 适用范围:小程序(
apps/miniprogram/)与 FastAPI 后端(apps/backend/)的联调 约束力:高度建议参考。特殊需求可偏离但需注明原因。 速查索引:.kiro/steering/frontend-backend-integration.md(fileMatch 自动加载)
一、数据契约
1.1 ResponseWrapper
后端 ResponseWrapperMiddleware 自动包装所有 2xx JSON 响应为 {"code": 0, "data": <原始body>}。脚本/测试调用 API 时必须从 resp.json()["data"] 取实际数据,不能直接用 resp.json()(踩坑记录 2026-03-28)。
1.2 字段命名与 CamelCase 转换
- 后端统一
snake_case,PydanticCamelModel自动转camelCase输出 - 前端用 camelCase 读取,兼容写法:
rec.memberId ?? rec.member_id ?? 0 - 踩坑:
me.avatar应为me.avatarUrl(avatar_url转换后)
1.3 条件显示放后端
字段需要根据业务条件决定显示/隐藏时,后端满足条件返回值、不满足返回 null。前端只判 wx:if="{{field}}",禁止做数值比较。
组件 property 收到 null 不走默认值,传入前必须清洗:r.durationRaw ?? 0、r.drinks ?? ''。
1.4 Schema 与 Service 同步规则
Pydantic response schema 必须与 service 返回字段严格一致,否则 500。排查 500 优先检查:
- 数据库迁移是否已执行
- Schema 字段是否与 service 一致
- 环境变量是否缺失
- 鉴权依赖是否匹配
踩坑记录汇总(Pydantic 系列)
| 问题 | 日期 | 表现 | 规则 |
|---|---|---|---|
| 静默丢弃未声明字段 | 2026-03-28 | 新增字段(desc、discount_total)前端收不到 |
新增后端返回字段时必须同步更新 Schema |
list[dict] 不转换内部 key |
2026-03-28 | 前端拿到 snake_case key | 用 CamelModel 子类替代 dict,或 service 层手动递归转换 |
| Optional 与 None 不同步 | 2026-03-29 | service 返回 None 但 Schema 非 Optional → 500 | 修改 service 返回 None 时必须同步 Schema 为 `XxxModel |
| response_model 类型不匹配 | 2026-03-29 | Schema 期望 list[CoachSkillItem] 但返回 list[str] → 500 |
修改 service 返回字段时必须同步检查 Schema 类型 |
| 嵌套 CamelModel 字段名不匹配 | 2026-03-29 | dict key 与 Schema 字段名不一致 → 500 | _build_* 方法必须逐字段对照 Schema |
| 含数字字段名转换 | 2026-03-29 | consumption_60d → consumption60D(大写 D) |
前端用 d.consumption60D ?? d.consumption60d 兼容 |
1.5 接口返回结构
// 列表:{ "records": [...], "total_count": 42, "has_more": true }
// 概览:{ "coach_name": "...", "income_items": [...], "this_month_records": [...] }
收入明细始终返回所有项(即使为 0),前端不做过滤。
二、展示映射
2.1 头像颜色
后端只传 member_id + avatar_char,前端 nameToAvatarColor(String(memberId)) 计算颜色 key。散客(member_id ≤ 0)→ 'default'(灰色),avatar_char 返回 "?"。
2.2 课程标签
后端返回数据库原始 skill_name(中文),WXSS 不支持中文类名,前端映射:
const COURSE_TAG_MAP: Record<string, string> = {
'陪打': 'basic', '基础课': 'basic',
'包厢': 'room', '包厢课': 'room',
'超休': 'incentive', '激励课': 'incentive', '打赏课': 'incentive',
}
2.3 角色标签
后端返回英文 code(coach/staff/head_coach/manager),前端 ROLE_LABELS 映射中文。助教拼接等级:${coachLevel}助教。
2.4 助教姓名显示规则(2026-03-28)
所有助教必须使用昵称/花名(nickname),不得显示真实姓名(real_name)。SQL 中统一用 COALESCE(da.nickname, da.real_name, ''),nickname 优先。
2.5 助教离职过滤(2026-03-29)
所有查询助教的 SQL 必须加 da.leave_status = 0 过滤离职助教。LEFT JOIN 场景用 (da.leave_status IS NULL OR da.leave_status = 0) 兼容无匹配行。dim_assistant 中 leave_status=1 为离职(占 74%),不过滤会展示大量已离职助教。
2.6 预估规则(全小程序统一)
当月且当前日期 ≤ 5号时显示"预估"标签。业务含义:每月 1-5 日为工资审核期。
2.7 WXSS 类名拼接
后端值用于 class="xxx-{{value}}" 时必须是合法 CSS 标识符(纯英文),禁止中文/hex/特殊字符。
2.8 前端筛选枚举同步(踩坑记录 2026-03-28)
前端下拉/筛选组件的 value 必须与后端 Enum 值完全一致(大小写、命名)。修改后端枚举时,必须全局搜索所有使用该枚举的前端页面同步更新。
2.9 WXS 格式化规范
- WXS 数值方法防护:
format.wxs中所有调用.toFixed()的函数必须先parseFloat转数字 - TS 与 WXS 格式化互斥(2026-03-27):WXML 用
{{fmt.money(field)}}时,TS 层setData必须传原始数字(?? 0),禁止formatMoney()预格式化 - 后端 service 层同理(2026-03-29):前端用 WXS 格式化的字段,后端必须返回原始数字(float/int),禁止预格式化为字符串。Schema 类型需同步从
str改为float/int - 环比显示统一规范(2026-03-28):所有环比文本必须使用
fmt.compareText(compare, isDown)+fmt.compareClass(compare, isDown, size)WXS 函数 - WXS 零值语义(2026-03-27):
value === 0不一定是空值。days(0)表示"今天到店"是有效值。新增 WXS 函数时必须区分"零值有业务含义"和"零值等同空值"
三、SQL 查询规范
3.1 ETL 连接复用(必须)
同一 service 方法内创建一个 etl_conn 传给所有 fdw_queries.*。
3.2 多页面复用组件时 SQL 统一
后端为同一组件提供数据的查询必须同口径(共享 JOIN/聚合),各页面只负责过滤和分页。
3.3 一对多明细用 LATERAL 避免行膨胀
LEFT JOIN LATERAL (
SELECT string_agg(gs.ledger_name || '×' || gs.total_count, '、') AS drinks
FROM (...GROUP BY ledger_name) gs
) gs_agg ON true
3.4 助教收入口径
sl.ledger_amount:行级毛收入,不是到手- 到手:
hours × (base_course_price - base_deduction) sh.assistant_pd_money + cx_money:整单级汇总,禁止 JOIN 到行级
3.5 快照值 vs 流量值(2026-03-27)
- 流量值(充值/消费/收入):可以 SUM
- 快照值(卡余额
cash_card_balance/gift_card_balance/total_card_balance):日末快照,多天聚合取最后一天的值,禁止 SUM
3.6 dim_member_card_account 多卡膨胀(2026-03-29)
同一 tenant_member_id 在 scd2_is_current=1 下可能有多条记录(多张卡),必须先 GROUP BY tenant_member_id + SUM(balance) 聚合后再 JOIN。
3.7 台费原价推算(2026-03-29)
台费正价 = table_charge_money + adjust_amount。adjust_amount > 0 时显示原价删除线。
3.8 环比同期对比规则(2026-03-28)
当前周期 end_date cap 到今天,上期取同类周期的同期天数。已完成周期取完整周期对比再上一个完整周期。
3.9 优惠拆分恒等式(2026-03-28)
discount_total = discount_groupbuy + discount_manual + discount_other + discount_vip + discount_gift_card + discount_rounding。拆分展示必须包含全部 6 个子字段。
3.10 团购金额双口径(2026-03-28)
DWS groupbuy_pay_amount 是团购交易金额,daily_revenue_report.py 用 sale_price × 0.75 估算回款。财务看板用 DWS 口径,经营报告用估算回款口径。
3.11 助教财务三列口径(2026-03-28)
pay/share/hourly 全部从 DWS dws_assistant_salary_calc 计算。禁止从 DWD ledger_amount 取客户支付再 JOIN DWS 等级(同一助教同月可有多等级记录导致行膨胀)。
3.12 客源储值口径(纠正 2026-03-29)
客源储值 = 该助教关联客户的卡余额合计,不是充值提成。数据源:v_dws_member_assistant_relation_index(session_count > 0)+ v_dim_member_card_account(先 GROUP BY 聚合)。
3.13 DATE_TRUNC 返回 timestamp 不是 date(2026-03-29)
用作 dict key 时必须先 .date() 转换。
3.14 关系指数表字段名(2026-03-29)
dws_member_assistant_relation_index 的字段是 session_count(非 service_count)、total_duration_minutes(非 total_hours)、ml_allocated_amount(非 total_income)。
3.15 dim_table 主键列名(2026-03-29)
主键是 table_id(不是 site_table_id)。JOIN 时用 ON s.site_table_id = dt.table_id。
四、数据库操作陷阱
4.1 psycopg2 Windows GBK 编码(2026-03-29)
psycopg2.connect() 用关键字参数时,libpq 拼接系统 locale 信息触发 UnicodeDecodeError。解决:用显式 DSN 字符串 + client_encoding=UTF8 + os.environ.setdefault("PGCLIENTENCODING", "UTF8")。
4.2 软删除 + ON CONFLICT
ON CONFLICT DO UPDATE SET is_removed=false,禁止 DO NOTHING。
4.3 ON CONFLICT 精确匹配 partial unique index
列列表和 WHERE 条件必须与索引定义完全对应。
4.4 跨库过滤分页
先构建排除列表,SQL 层统一过滤,禁止内存过滤修正 total。
4.5 共享连接子流程
需事务隔离时用独立连接,避免异常被外层 except 吞掉。
4.6 FDW 查询禁止 N+1 串行(2026-03-29)
必须改为 WHERE assistant_id = ANY(%s) 批量查询 + Python 层 dict 映射。
4.7 FDW 查询必须传 etl_conn(2026-03-29)
所有 _build_* 子函数都应接收并传递 etl_conn,否则 _fdw_context 在业务库连接上设置 RLS 导致空结果。
五、前端交互规范
5.1 页面加载态(2026-03-29)
统一用 wx.showLoading / wx.hideLoading 原生遮罩,禁止自定义加载组件。
5.2 跨页面跳转
URL 参数必须包含可查询 ID(memberId/taskId),不能只传展示字段。
5.3 dataset 同步
TS e.currentTarget.dataset.xxx 必须与 WXML data-xxx 属性同步。
5.4 自定义组件静默不渲染(2026-03-29)
未在页面 JSON usingComponents 中注册的组件不报错但完全不渲染。
5.5 多状态列表页
后端按 status 过滤,前端需并行请求所有需要展示的状态并合并。
5.6 微信头像
chooseAvatar 组件 + wx.uploadFile 上传服务器,临时路径会过期。
5.7 Vite 代理
跨前缀接口必须在 vite.config.ts proxy 中添加规则。
5.8 权限守卫
fallback 目标页必须对当前用户无条件可访问,避免跳转死循环。
5.9 前后端权限一致性
后端 /api/xcx/me 返回 permissions[],前端根据权限码动态控制可见性。新增权限码只需更新数据库 auth.role_permissions。
5.10 custom-tab-bar 异步刷新
更新 globalData.visibleTabs 后必须主动调用 tabBar._refreshTabs()。看板 boardTabs 刷新必须放在 checkPageAccess().then() 回调中。
5.11 禁止页面滚动(2026-03-28)
wx.setPageScrollEnabled 不存在。用 <page-meta page-style="overflow:hidden" /> 控制。
5.12 登录与 /me 接口字段差异(2026-03-28)
/api/xcx/login 返回 WxLoginResponse(不含 permissions),/api/xcx/me 返回 UserStatusResponse(含 permissions)。登录后需权限码时必须额外请求 /me。
5.13 分页接口 API 函数必须返回完整响应(2026-03-29)
必须返回完整 data(含 items/total/page/pageSize),禁止只返回 data.items。
5.14 懒加载分页追加必须去重 + 双条件判底(2026-03-29)
① 追加数据按 id 去重(Set 过滤);② hasMore 用双条件:items.length >= pageSize && merged.length < total。
六、事件驱动触发器
biz.trigger_jobs中trigger_condition='event'的任务必须有明确的fire_event(event_name)发射端- 任务引擎全流程由 ETL 任务
DWS_TASK_ENGINE编排,通过 HTTP 调用后端POST /api/internal/run-job task_expiry_check的 interval 触发器保留(每小时),DWS_TASK_ENGINE是补充而非替代recall_detector直接生成回访任务(CHANGE 2026-03-31)- 任务生成器使用客户级别升级/转移(CHANGE 2026-03-31):基于
t_v / ideal_intervalratio 决定分配范围 - 任务统计写入(CHANGE 2026-03-31):
task_generator.run()末尾调用_update_task_stats()
七、ETL 调用后端 API 注意事项
- ETL 的
AppConfig不设置os.environ,ETL 任务中用os.environ.get()前必须先load_dotenv - 后端
ResponseWrapperMiddleware包装响应,ETL 调用时需从resp.json()["data"]取数据
八、列表与分页
- 客户列表:默认 5 条,展开最多 20 条
- 服务记录:按日期分组(DateGroup),默认 2 组,可展开
- 新客:本月有服务但之前无历史
- 常客:本月服务 ≥ 2 次,展示近 90 天聚合