Files
ccdi/docs/design/2026-03-29-project-import-history-design.md

13 KiB
Raw Blame History

历史项目导入新项目功能设计文档

背景

当前项目管理页已经存在“导入历史项目”前端弹窗壳子,但能力仍停留在演示阶段:

  • 仅支持前端 Mock 历史项目数据
  • 仅支持单选历史项目
  • 没有真实创建新项目与复制流水的后端逻辑
  • 没有接通现有打标、结果表生成、统计刷新链路

与此同时,系统已经具备以下基础能力:

因此本次需要补齐“从一个或多个历史项目复制流水到新项目”的真实业务闭环。

目标

  • 在“导入历史项目”弹窗中支持填写新项目名称、备注、交易日期范围,并多选历史项目
  • 点击提交后先创建新项目,再异步复制历史流水到新项目
  • 若填写时间范围,仅复制 trxDate 命中的流水
  • 若多个来源项目存在重复流水,按现有流水唯一性口径去重后写入新项目
  • 同步复制真正贡献了被导入流水的成功文件上传记录
  • 历史导入完成后自动触发打标、结果表生成和统计刷新
  • 提交成功后关闭弹窗并跳转到新项目详情页,由详情页承接进度展示
  • 历史导入生成的文件记录在新项目中只读展示,不允许删除

非目标

  • 不支持继承来源项目的参数配置
  • 不支持合并多个来源项目的参数规则
  • 不支持导入“进行中”项目
  • 不支持复制失败或进行中的文件上传记录
  • 不支持删除历史导入生成的文件记录
  • 不新增独立的“导入任务管理页面”
  • 不新增额外复杂导入状态体系或独立进度中心

已确认业务口径

  • 历史项目可选范围:仅“已完成”“已归档”项目
  • 新项目参数来源:固定走现有新建项目默认参数逻辑
  • 时间过滤字段:固定按 trxDate
  • 重复流水处理:按现有唯一性口径去重
  • 提交后页面行为:关闭弹窗并跳转到新项目详情页
  • 文件记录复制范围:仅复制 parsed_success 且确实贡献了被复制流水的记录
  • 文件记录展示:保留原文件名,并标记“历史导入来源”
  • 删除规则:历史导入生成的文件记录不允许删除

方案对比

方案一:创建项目后启动独立历史导入异步任务

  • 做法:提交接口先调用现有 createProject,再在事务提交后启动独立历史导入服务,完成流水复制、文件记录复制、自动打标和结果刷新
  • 优点:业务语义清晰,和“上传文件”链路解耦,适合历史复制场景
  • 缺点:需要新增历史导入 DTO、接口和异步编排服务

方案二:将历史导入伪装成“虚拟上传文件”

  • 做法:把来源项目批次伪装为上传任务,强行复用文件上传服务全链路
  • 优点:看似能复用更多现有代码
  • 缺点:历史导入并不是真实上传,会导致批次、错误信息、来源展示语义混乱

方案三:提交接口同步完成流水复制,末尾再触发打标

  • 做法:导入接口直接完成所有复制逻辑,再异步触发打标
  • 优点:表面上实现路径更短
  • 缺点:来源项目多、流水量大时接口耗时过长,和“先创建项目,再后台处理”的需求不符

最终方案

采用方案一。

导入接口只负责同步创建新项目和提交异步历史导入任务。历史导入任务在后台完成以下步骤:

  1. 校验来源项目和导入条件
  2. trxDate 过滤并复制流水到新项目
  3. 按现有唯一性口径去重
  4. 为真正贡献流水的来源批次生成新的文件上传记录
  5. 刷新新项目 targetCount
  6. 自动触发打标、结果表生成和统计刷新

前端提交成功后直接跳转到新项目详情页,复用现有项目状态轮询和上传记录列表。

现状分析

1. 新建项目现状

CcdiProjectServiceImpl#createProject 会先调用 callLsfxPlatform(projectName) 获取流水分析平台项目 ID再将其写入本地项目表的 lsfxProjectId 字段。

callLsfxPlatform 当前通过 lsfxAnalysisClient.getToken(request) 调用流水分析平台接口,并校验返回的 projectId 非空。

这意味着历史项目导入时创建新项目,可以直接复用现有创建逻辑,不需要额外新增“平台建项目”调用路径。

2. 文件上传记录现状

ccdi_file_upload_record 当前存储字段包括:

  • project_id
  • lsfx_project_id
  • log_id
  • file_name
  • file_status
  • enterprise_names
  • account_nos
  • upload_time
  • upload_user

CcdiBankStatementMapper.xml 会按 project_id + log_id(batch_id) 将流水和文件记录关联起来,用于详情页展示原始文件名。

因此历史导入不能直接复用来源项目原始 logId,否则会在新项目维度造成批次语义混乱。新项目内必须对导入批次重新分配新的批次号,并同步写入新项目的文件记录与流水数据。

3. 详情页现状

UploadData.vue 已具备:

  • 上传记录分页列表
  • 项目打标中的只读锁定
  • 轮询刷新文件记录与统计
  • 删除文件记录入口

detail.vue 已具备项目状态轮询。

因此本次只需要让历史导入记录进入现有上传记录列表,并在行操作层和删除接口层增加“历史导入不可删除”的约束。

详细设计

1. 后端接口设计

1.1 查询可导入历史项目

  • URLGET /ccdi/project/history
  • 权限:沿用项目列表查询权限
  • 入参:
    • projectName:可选,按项目名模糊搜索
  • 返回:
    • 仅返回状态为“已完成”“已归档”的项目
    • 返回字段包含项目 ID、项目名称、状态、创建时间、目标人数等弹窗展示所需字段

1.2 提交历史导入

  • URLPOST /ccdi/project/import
  • 权限:沿用项目新增权限
  • 入参建议:
    • projectName
    • description
    • sourceProjectIds
    • startDate
    • endDate
  • 返回:
    • 新项目 projectId
    • 新项目基础信息

接口执行顺序:

  1. 校验入参
  2. 调用现有 createProject
  3. 在事务提交后启动历史导入异步任务
  4. 立即返回新项目信息给前端

2. 新增导入 DTO 与 VO

新增独立 DTO避免和普通新建项目 DTO 混用:

  • CcdiProjectImportHistoryDTO
    • projectName
    • description
    • List<Long> sourceProjectIds
    • String startDate
    • String endDate

新增历史项目列表 VO

  • CcdiProjectHistoryListItemVO
    • projectId
    • projectName
    • projectStatus
    • createTime
    • targetCount
    • warningCount

3. 历史导入异步服务设计

新增独立历史导入服务,职责单一为“将历史项目流水导入到新项目”。

建议命名示例:

  • ICcdiProjectHistoryImportService
  • CcdiProjectHistoryImportServiceImpl

核心步骤:

  1. 校验新项目存在且未归档
  2. 校验来源项目均为“已完成 / 已归档”
  3. 查询来源项目成功文件记录与流水数据
  4. trxDate 过滤流水
  5. 按现有唯一性口径去重
  6. 批量写入新项目流水
  7. 根据实际贡献流水的批次生成新项目文件记录
  8. 刷新 targetCount
  9. 调用 bankTagService.submitAutoRebuild(projectId, triggerType)

4. 流水复制规则

4.1 复制范围

  • 仅处理所选来源项目
  • 若未填写时间范围,则复制全部流水
  • 若填写时间范围,则仅复制 trxDate 位于区间内的流水

4.2 去重规则

  • 沿用当前 ccdi_bank_statement 入库唯一性口径
  • 若多个来源项目存在重复流水,新项目中仅保留一份

4.3 空导入处理

  • 若所有来源项目在过滤后均无可导入流水,则导入任务判定为业务失败
  • 新项目保留,但不触发后续打标
  • 前端详情页可看到失败结果,不回滚项目

5. 文件上传记录复制规则

5.1 可复制记录范围

  • 仅复制 parsed_success
  • 仅复制那些“至少贡献了一条被导入流水”的来源批次
  • 失败记录、进行中记录一律不复制

5.2 批次号重映射

新项目内每个来源批次必须生成新的批次标识,并同步写入:

  • 新项目文件上传记录的 logId
  • 新项目流水表的 batchId

这样可以确保:

  • 与新项目后续真实上传文件的批次不冲突
  • 详情页仍能正确通过 project_id + log_id(batch_id) 关联文件与流水

5.3 来源标识

为支持前端展示“历史导入来源”,建议在 ccdi_file_upload_record 增加最小必要字段:

  • source_typeUPLOAD / PULL_BANK_INFO / HISTORY_IMPORT
  • source_project_id
  • source_project_name

其中:

  • 当前已有上传/拉取记录可回填默认 source_type
  • 历史导入记录固定写入 HISTORY_IMPORT

6. 删除约束设计

6.1 前端删除约束

历史导入记录在上传列表中只读展示:

  • 保留文件名
  • 展示“历史导入 · 来源项目名”
  • 不显示“删除”按钮

6.2 后端删除约束

deleteFileUploadRecord 需增加后端兜底校验:

  • 若记录属于 HISTORY_IMPORT,直接拒绝删除
  • 返回明确错误信息,例如“历史导入文件不支持删除”

这样可以避免前端绕过行操作约束后误删导入数据。

7. 打标与结果刷新衔接

历史导入完成后,不新增单独导入状态机,直接复用现有打标链路:

  1. 复制流水成功
  2. 刷新新项目 targetCount
  3. 调用 bankTagService.submitAutoRebuild(...)
  4. 项目状态进入“打标中”
  5. 打标完成后刷新结果表、风险统计和总览数据
  6. 项目状态回到“已完成”

导入失败时:

  • 项目保留
  • 不进入打标
  • 通过上传记录或错误提示体现失败结果

8. 前端弹窗设计

弹窗采用纵向布局:

8.1 新项目配置区

位于弹窗上方,包含:

  • 新项目名称
  • 备注
  • 交易日期范围

8.2 历史项目选择区

位于弹窗下方,包含:

  • 项目名称搜索
  • 多选历史项目列表
  • 固定高度滚动区域

列表只显示“已完成”“已归档”项目。

9. 提交后交互设计

提交成功后:

  1. 关闭弹窗
  2. 提示“历史项目导入任务已开始”
  3. 跳转到新项目详情页

详情页承接方式:

  • 复用现有项目状态轮询
  • 复用上传记录列表轮询
  • 用户在详情页中观察历史导入记录、项目状态和后续打标完成情况

10. 测试关注点

后续实施时需重点验证:

  • 多历史项目多选导入
  • trxDate 时间范围过滤
  • 跨项目重复流水去重
  • 历史导入记录来源展示
  • 历史导入记录删除拦截
  • 导入完成后自动打标与结果表刷新
  • 无可导入流水时的失败呈现

风险与约束

  • 现有文件记录与流水关联依赖 project_id + log_id(batch_id),批次号重映射必须保证稳定且唯一
  • 历史导入记录一旦允许删除,会破坏“复制数据来源可追溯”的约束,因此本次明确禁止删除
  • 本次不引入额外任务中心,导入失败的展示能力依赖现有详情页与记录列表,需要实施时保持错误信息可读

结论

本次采用“先创建新项目,再异步导入历史流水并触发现有打标链路”的方案,保持最短路径实现。

该方案满足以下要求:

  • 贴合现有项目创建和详情页状态轮询结构
  • 不混淆“上传文件”和“复制历史流水”两类业务语义
  • 能复用现有打标、结果表和统计刷新能力
  • 能明确保证历史导入文件记录不可删除

后续进入实施计划阶段时,需要按仓库约定分别输出前端、后端两份实施计划。