97 lines
5.9 KiB
Markdown
97 lines
5.9 KiB
Markdown
# 实施计划:DWD 第一阶段重构
|
||
|
||
## 概述
|
||
|
||
按照 confirmed_changes.md 中 2.1-2.4 的顺序,对 `DwdLoadTask` 进行低风险重构。改动集中在 `dwd_load_task.py`,辅以删除 `base_dwd_task.py` 和更新两个外部引用。每个步骤递增构建,确保无悬空代码。
|
||
|
||
## 任务
|
||
|
||
- [x] 1. 统一窗口模式,去掉水位线(需求 1)
|
||
- [x] 1.1 修改 `_merge_fact_increment()` 签名:`window_start` 和 `window_end` 改为必填参数(去掉 `| None` 和默认值 `None`);删除方法体内的 watermark 分支(`elif order_col:` 块及 `watermark = None` 赋值),统一使用 `WHERE fetched_at >= %s AND fetched_at < %s`
|
||
- 文件:`tasks/dwd/dwd_load_task.py`,方法 `_merge_fact_increment`
|
||
- _Requirements: 1.2, 1.4_
|
||
- [x] 1.2 修改 `load()` 中事实表分支:删除 `use_window` 变量及条件判断,始终传 `window_start=context.window_start, window_end=context.window_end`
|
||
- 文件:`tasks/dwd/dwd_load_task.py`,方法 `load`
|
||
- _Requirements: 1.1_
|
||
- [x] 1.3 删除 `_get_fact_watermark()` 方法(约 30 行)
|
||
- 文件:`tasks/dwd/dwd_load_task.py`
|
||
- _Requirements: 1.3_
|
||
- [x] 1.4 编写属性测试:事实表增量 SQL 始终使用窗口范围条件
|
||
- **Property 1: 事实表增量 SQL 始终使用窗口范围条件**
|
||
- **Validates: Requirements 1.1, 1.2**
|
||
- 使用 `hypothesis` 生成随机 `window_start`/`window_end`,通过 `FakeCursor` 捕获 SQL,验证 WHERE 子句包含 `>= %s AND < %s`,不包含单边水位线条件
|
||
- 文件:`tests/test_dwd_phase1_properties.py`
|
||
|
||
- [x] 2. 删除回补机制(需求 2)
|
||
- [x] 2.1 删除 `_merge_fact_increment()` 末尾的回补调用块:移除对 `_insert_missing_by_pk()` 的调用及 `missing_inserted` 相关代码
|
||
- 文件:`tasks/dwd/dwd_load_task.py`,方法 `_merge_fact_increment`
|
||
- _Requirements: 2.3_
|
||
- [x] 2.2 删除 `_insert_missing_by_pk()` 方法(约 100 行)
|
||
- 文件:`tasks/dwd/dwd_load_task.py`
|
||
- _Requirements: 2.1_
|
||
- [x] 2.3 删除 `FACT_MISSING_FILL_TABLES` 常量
|
||
- 文件:`tasks/dwd/dwd_load_task.py`
|
||
- _Requirements: 2.2_
|
||
- [x] 2.4 编写属性测试:事实表增量不执行回补 SQL
|
||
- **Property 2: 事实表增量不执行回补 SQL**
|
||
- **Validates: Requirements 2.3**
|
||
- 通过 `FakeCursor` 捕获所有执行的 SQL,验证无 `LEFT JOIN ... IS NULL` 模式
|
||
- 文件:`tests/test_dwd_phase1_properties.py`
|
||
|
||
- [x] 3. 检查点 - 确保窗口统一和回补删除后测试通过
|
||
- 运行 `cd apps/etl/pipelines/feiqiu && pytest tests/unit` 和 `cd C:\NeoZQYY && pytest tests/ -v`,确保所有测试通过,如有问题请询问用户。
|
||
|
||
- [x] 4. 清理死代码和未使用常量(需求 3)
|
||
- [x] 4.1 删除 `_pick_order_column()` 方法和 `FACT_ORDER_CANDIDATES` 常量
|
||
- 文件:`tasks/dwd/dwd_load_task.py`
|
||
- _Requirements: 3.2, 3.3_
|
||
- [x] 4.2 更新外部引用:在 `debug_dwd.py` 和 `integrity_checker.py` 中将 `DwdLoadTask.FACT_ORDER_CANDIDATES` 替换为模块级常量 `_TIME_COLUMN_CANDIDATES`(内联相同列表)
|
||
- 文件:`scripts/debug/debug_dwd.py`、`quality/integrity_checker.py`
|
||
- _Requirements: 3.9_
|
||
- [x] 4.3 删除逐行 SCD2 方法:`_upsert_scd2_row()`、`_close_current_dim()`、`_insert_dim_row()`
|
||
- 文件:`tasks/dwd/dwd_load_task.py`
|
||
- _Requirements: 3.4, 3.5, 3.6_
|
||
- [x] 4.4 删除 `_merge_dim_type1_upsert()` 方法;简化 `_merge_dim()` 直接调用 `_merge_dim_scd2()`
|
||
- 文件:`tasks/dwd/dwd_load_task.py`
|
||
- _Requirements: 3.7, 3.8_
|
||
- [x] 4.5 删除 `base_dwd_task.py` 文件
|
||
- 文件:`tasks/dwd/base_dwd_task.py`
|
||
- _Requirements: 3.1_
|
||
- [x] 4.6 编写属性测试:维度表始终走 SCD2 路径
|
||
- **Property 4: 维度表始终走 SCD2 路径**
|
||
- **Validates: Requirements 3.8, 5.1**
|
||
- 通过 mock `_merge_dim_scd2` 验证 `_merge_dim()` 始终委托给它
|
||
- 文件:`tests/test_dwd_phase1_properties.py`
|
||
|
||
- [x] 5. 修复 `_build_column_mapping()` 参数 bug(需求 4)
|
||
- [x] 5.1 修改 `_build_column_mapping()` 签名:添加 `cur` 和 `ods_table` 参数;更新所有调用点(`_merge_dim_type1_upsert` 已删除,剩余调用点为 `_merge_dim_scd2`)传入正确参数
|
||
- 文件:`tasks/dwd/dwd_load_task.py`
|
||
- _Requirements: 4.1, 4.2_
|
||
- [x] 5.2 编写单元测试:验证 `_build_column_mapping()` 在 `fetched_at` 缺失时正确使用 `ods_table` 和 `cur` 参数记录日志
|
||
- 文件:`apps/etl/pipelines/feiqiu/tests/unit/test_dwd_phase1_refactor.py`
|
||
- _Requirements: 4.1_
|
||
|
||
- [x] 6. 编写回归单元测试(需求 5)
|
||
- [x] 6.1 编写单元测试:验证死代码已清理(`hasattr` 检查所有已删除的方法和常量)
|
||
- 文件:`apps/etl/pipelines/feiqiu/tests/unit/test_dwd_phase1_refactor.py`
|
||
- _Requirements: 1.3, 2.1, 2.2, 3.1-3.7_
|
||
- [x] 6.2 编写单元测试:验证外部模块导入正常(`debug_dwd.py`、`integrity_checker.py` 无 ImportError)
|
||
- 文件:`apps/etl/pipelines/feiqiu/tests/unit/test_dwd_phase1_refactor.py`
|
||
- _Requirements: 3.9_
|
||
- [x] 6.3 编写属性测试:事实表主增量 SQL 结构等价性
|
||
- **Property 3: 事实表主增量 SQL 结构等价性**
|
||
- **Validates: Requirements 5.2, 5.5**
|
||
- 使用 `hypothesis` 生成随机列集合和窗口参数,通过 `FakeCursor` 捕获 SQL,验证 INSERT INTO ... ON CONFLICT 结构与预期一致
|
||
- 文件:`tests/test_dwd_phase1_properties.py`
|
||
|
||
- [x] 7. 最终检查点 - 确保所有测试通过
|
||
- 运行 `cd apps/etl/pipelines/feiqiu && pytest tests/unit` 和 `cd C:\NeoZQYY && pytest tests/ -v`,确保所有测试通过,如有问题请询问用户。
|
||
|
||
## 备注
|
||
|
||
- 标记 `*` 的子任务为可选测试任务,可跳过以加速 MVP
|
||
- 每个任务引用了具体的需求编号,确保可追溯
|
||
- 检查点确保增量验证
|
||
- 属性测试验证通用正确性属性,单元测试验证具体示例和边界情况
|
||
- 本次重构涉及 `tasks/` 高风险路径,完成后需运行 `/audit`
|