docs: development standards + test plan

This commit is contained in:
知微
2026-07-04 09:57:18 +08:00
parent 5b9f22a643
commit 7f84471aca
+196
View File
@@ -0,0 +1,196 @@
# MoFin 开发规范
> 版本:1.0 | 日期:2026-07-03 | 维护:Sisyphus + Zhiwei
---
## 一、代码结构
### 1.1 无重复代码
- **一个功能一个文件**。禁止出现两份同名/同功能的 `.py` 文件。
- 公共逻辑抽取到 `mo_models.py` / `mo_data.py` / `mofin_db.py`,各脚本只调不写。
- 每次新增功能前,先搜索是否已有类似实现。
### 1.2 文件职责
| 文件 | 职责 | 允许做的事 |
|------|------|-----------|
| `mo_models.py` | 统一数据模型 | `calc_total_assets`, `is_hk_stock`, `to_cny`, `get_hk_rate` |
| `mo_data.py` | 统一读取层 | `read_portfolio`, `read_decisions`, `read_watchlist` — 只读 DB |
| `mofin_db.py` | DB 层 | 建表、写函数、查询函数 — 所有 SQL 集中在这里 |
| `price_monitor.py` | 价格更新 | 从 API 拉价格,写入 holdings + portfolio_summary |
| `strategy_lifecycle.py` | 策略生命周期 | `regenerate_all` + quality gates |
| `scripts/` | 工具/一次性脚本 | 不通过 cron 调用的辅助工具 |
### 1.3 禁止事项
- ❌ 在业务脚本里直接写 SQL(必须通过 `mofin_db.py`
- ❌ 在业务脚本里直接 `json.load(open(...))` 读数据(必须通过 `mo_data.py`
- ❌ 自己实现 `calc_total_assets` / `is_hk_stock`(必须用 `mo_models.py`
- ❌ 自己实现 `read_portfolio` / `read_decisions` / `read_watchlist`(必须用 `mo_data.py`
- ❌ 新增 JSON 数据文件(所有持久化数据必须走 DB)
---
## 二、数据规范
### 2.1 币种
| 场景 | 港股 | A股 |
|------|------|-----|
| 个股 price/cost/stop_loss | HKD | CNY |
| `currency` 字段 | `'HKD'` | `'CNY'` |
| 总资产/总市值 | CNY(汇总时 × 汇率) | CNY |
| 个股 P&L 计算 | 同币种 `(price-cost)/cost` | 同币种 |
| API/报告输出 | 标注 `(HKD)` | 无标注 |
**铁律**:禁止跨币种比较或加减。`calc_total_assets` / `calc_total_mv` 是唯一可以做币种转换的地方。
### 2.2 汇率
- 汇率必须通过 `get_hk_rate()` 获取(实时 API + 缓存)
- 禁止硬编码 `0.87` / `0.866` 等值(仅 `hk_rate.py` 内部兜底例外)
- 汇率 API 不可达时自动使用上次缓存值
### 2.3 数据源
- **唯一写入源**`price_monitor.py` 是唯一的价格写入者
- **DB 优先读取**:所有数据读取走 `mo_data.py` → DB
- **API 兜底**DB 无数据时,`price_monitor` 从外部 API 拉取
- **禁止各脚本各自拉 API** 写数据
---
## 三、DB 规范
### 3.1 表设计
- 新增表必须在 `mofin_db.py``init_all_tables()` 中定义
- 新表必须有对应的 `write_*()` 函数和 `query_*()` 函数
- 涉及币种的列必须有 `CHECK(currency IN ('CNY','HKD'))` 约束
- `code` 列建议加 UNIQUE INDEX
### 3.2 迁移
- 加列用 `ALTER TABLE ADD COLUMN`,幂等(`IF NOT EXISTS` 或 try/except
- 不直接改线上 DB 结构,通过 `mofin_db.py` 的 init 函数执行
- 数据修正脚本放到 `scripts/` 目录
---
## 四、LLM Prompt 规范
### 4.1 强制要求
| # | 规则 | 检查方法 |
|---|------|---------|
| S1 | 不引用 JSON 文件名 | grep `\.json` |
| S2 | 港股价格标注 `(HKD)` | 检查 prompt 模板 |
| S3 | 不指示跨币种直接比较 | 人工审查 |
| S4 | 数据源标注为 DB 表名 | `strategy_evaluations` 不是 `evaluation.json` |
| S5 | 不在 prompt 里硬编码路径 | grep `/home/hmo/` |
### 4.2 新增/修改 Prompt
-`prompt_manager/init_registry.py` 注册新版本
- 旧版本标记 `status="deprecated"`,保留历史记录
- 更新后运行测试脚本验证无 JSON 引用
---
## 五、Cron 规范
### 5.1 调度
- Cron 脚本必须能独立运行(`python3 xxx.py` 不报错)
- 处理异常,不因单个 API 失败而崩溃
- 关键步骤加 try/except + `print(..., file=sys.stderr)` 日志
### 5.2 幂等性
- `price_monitor` 价格不变时不写入 DB
- `regenerate_all` 可重复运行不产生副作用
- 所有写操作支持重复执行
---
## 六、开发流程
### 6.1 日常开发步骤
```
1. 理解需求 → 检查现有代码 → 确定改动范围
2. 本地修改代码
3. 运行 scripts/run_all_tests.py(必须全部通过)
4. 如有新增功能,补充 TEST_PLAN.md 测试用例
5. scp 部署到服务器
6. 在服务器上再次运行 run_all_tests.py
7. git commit + push
8. 如涉及 server.py 修改,重启 Flask
9. 更新 CHANGELOG.md
```
### 6.2 测试要求
**每次代码变更后必须:**
```bash
# 在服务器上运行全面测试
cd /home/hmo/MoFin
python3 scripts/run_all_tests.py
```
**输出必须满足:**
- 0 个 `❌`
- 最多 1-2 个 `⚠️`(已知问题,已记录在案)
**如测试不通过:**
1. 修复问题
2. 重新测试
3. 不通过不提交
### 6.3 文档同步
**哪些文档需要同步更新:**
| 改动内容 | 需更新的文档 |
|---------|------------|
| 新增/修改表 | `docs/DATABASE_ARCHITECTURE.md` |
| 新增/修改 API | `docs/SYSTEM_ARCHITECTURE.md` |
| 架构变更 | `docs/SYSTEM_ARCHITECTURE.md` |
| 数据模型变更 | `docs/portfolio-data-model.md` |
| 任何功能变更 | `CHANGELOG.md`(追加顶部) |
| 新增测试用例 | `docs/TEST_PLAN.md` |
### 6.4 提交规范
- Commit message 格式:`type: 简短描述`
- 类型:`fix:` / `feat:` / `refactor:` / `docs:` / `test:` / `migrate:`
- 一次 commit 一个逻辑变更,不要混搭
---
## 七、快速检查清单
**每次提交前自检:**
```
□ run_all_tests.py 全部通过
□ 无新增 json.load / json.dump 数据文件
□ 港股 currency='HKD'、A股 currency='CNY'
□ 无硬编码汇率
□ 无重复代码
□ CHANGELOG.md 已更新
□ 无残留 print 调试语句
□ git status 确认只改了应改的文件
```
---
## 八、紧急联系人
| 角色 | 负责 |
|------|------|
| 小小莫 | 架构、DB、重构、price_monitor |
| 知微 | 策略、LLM prompt、cron、日常运营 |