13 KiB
历史项目导入新项目功能设计文档
背景
当前项目管理页已经存在“导入历史项目”前端弹窗壳子,但能力仍停留在演示阶段:
- 仅支持前端 Mock 历史项目数据
- 仅支持单选历史项目
- 没有真实创建新项目与复制流水的后端逻辑
- 没有接通现有打标、结果表生成、统计刷新链路
与此同时,系统已经具备以下基础能力:
CcdiProjectServiceImpl#createProject可创建本地项目,并在创建时调用流水分析平台getToken接口获取并绑定lsfxProjectIdCcdiFileUploadServiceImpl已打通文件上传后自动触发打标的异步链路CcdiBankTagServiceImpl已具备项目状态切换、标签重算、结果表刷新能力detail.vue已具备项目状态轮询能力
因此本次需要补齐“从一个或多个历史项目复制流水到新项目”的真实业务闭环。
目标
- 在“导入历史项目”弹窗中支持填写新项目名称、备注、交易日期范围,并多选历史项目
- 点击提交后先创建新项目,再异步复制历史流水到新项目
- 若填写时间范围,仅复制
trxDate命中的流水 - 若多个来源项目存在重复流水,按现有流水唯一性口径去重后写入新项目
- 同步复制真正贡献了被导入流水的成功文件上传记录
- 历史导入完成后自动触发打标、结果表生成和统计刷新
- 提交成功后关闭弹窗并跳转到新项目详情页,由详情页承接进度展示
- 历史导入生成的文件记录在新项目中只读展示,不允许删除
非目标
- 不支持继承来源项目的参数配置
- 不支持合并多个来源项目的参数规则
- 不支持导入“进行中”项目
- 不支持复制失败或进行中的文件上传记录
- 不支持删除历史导入生成的文件记录
- 不新增独立的“导入任务管理页面”
- 不新增额外复杂导入状态体系或独立进度中心
已确认业务口径
- 历史项目可选范围:仅“已完成”“已归档”项目
- 新项目参数来源:固定走现有新建项目默认参数逻辑
- 时间过滤字段:固定按
trxDate - 重复流水处理:按现有唯一性口径去重
- 提交后页面行为:关闭弹窗并跳转到新项目详情页
- 文件记录复制范围:仅复制
parsed_success且确实贡献了被复制流水的记录 - 文件记录展示:保留原文件名,并标记“历史导入来源”
- 删除规则:历史导入生成的文件记录不允许删除
方案对比
方案一:创建项目后启动独立历史导入异步任务
- 做法:提交接口先调用现有
createProject,再在事务提交后启动独立历史导入服务,完成流水复制、文件记录复制、自动打标和结果刷新 - 优点:业务语义清晰,和“上传文件”链路解耦,适合历史复制场景
- 缺点:需要新增历史导入 DTO、接口和异步编排服务
方案二:将历史导入伪装成“虚拟上传文件”
- 做法:把来源项目批次伪装为上传任务,强行复用文件上传服务全链路
- 优点:看似能复用更多现有代码
- 缺点:历史导入并不是真实上传,会导致批次、错误信息、来源展示语义混乱
方案三:提交接口同步完成流水复制,末尾再触发打标
- 做法:导入接口直接完成所有复制逻辑,再异步触发打标
- 优点:表面上实现路径更短
- 缺点:来源项目多、流水量大时接口耗时过长,和“先创建项目,再后台处理”的需求不符
最终方案
采用方案一。
导入接口只负责同步创建新项目和提交异步历史导入任务。历史导入任务在后台完成以下步骤:
- 校验来源项目和导入条件
- 按
trxDate过滤并复制流水到新项目 - 按现有唯一性口径去重
- 为真正贡献流水的来源批次生成新的文件上传记录
- 刷新新项目
targetCount - 自动触发打标、结果表生成和统计刷新
前端提交成功后直接跳转到新项目详情页,复用现有项目状态轮询和上传记录列表。
现状分析
1. 新建项目现状
CcdiProjectServiceImpl#createProject 会先调用 callLsfxPlatform(projectName) 获取流水分析平台项目 ID,再将其写入本地项目表的 lsfxProjectId 字段。
callLsfxPlatform 当前通过 lsfxAnalysisClient.getToken(request) 调用流水分析平台接口,并校验返回的 projectId 非空。
这意味着历史项目导入时创建新项目,可以直接复用现有创建逻辑,不需要额外新增“平台建项目”调用路径。
2. 文件上传记录现状
ccdi_file_upload_record 当前存储字段包括:
project_idlsfx_project_idlog_idfile_namefile_statusenterprise_namesaccount_nosupload_timeupload_user
CcdiBankStatementMapper.xml 会按 project_id + log_id(batch_id) 将流水和文件记录关联起来,用于详情页展示原始文件名。
因此历史导入不能直接复用来源项目原始 logId,否则会在新项目维度造成批次语义混乱。新项目内必须对导入批次重新分配新的批次号,并同步写入新项目的文件记录与流水数据。
3. 详情页现状
UploadData.vue 已具备:
- 上传记录分页列表
- 项目打标中的只读锁定
- 轮询刷新文件记录与统计
- 删除文件记录入口
detail.vue 已具备项目状态轮询。
因此本次只需要让历史导入记录进入现有上传记录列表,并在行操作层和删除接口层增加“历史导入不可删除”的约束。
详细设计
1. 后端接口设计
1.1 查询可导入历史项目
- URL:
GET /ccdi/project/history - 权限:沿用项目列表查询权限
- 入参:
projectName:可选,按项目名模糊搜索
- 返回:
- 仅返回状态为“已完成”“已归档”的项目
- 返回字段包含项目 ID、项目名称、状态、创建时间、目标人数等弹窗展示所需字段
1.2 提交历史导入
- URL:
POST /ccdi/project/import - 权限:沿用项目新增权限
- 入参建议:
projectNamedescriptionsourceProjectIdsstartDateendDate
- 返回:
- 新项目
projectId - 新项目基础信息
- 新项目
接口执行顺序:
- 校验入参
- 调用现有
createProject - 在事务提交后启动历史导入异步任务
- 立即返回新项目信息给前端
2. 新增导入 DTO 与 VO
新增独立 DTO,避免和普通新建项目 DTO 混用:
CcdiProjectImportHistoryDTOprojectNamedescriptionList<Long> sourceProjectIdsString startDateString endDate
新增历史项目列表 VO:
CcdiProjectHistoryListItemVOprojectIdprojectNameprojectStatuscreateTimetargetCountwarningCount
3. 历史导入异步服务设计
新增独立历史导入服务,职责单一为“将历史项目流水导入到新项目”。
建议命名示例:
ICcdiProjectHistoryImportServiceCcdiProjectHistoryImportServiceImpl
核心步骤:
- 校验新项目存在且未归档
- 校验来源项目均为“已完成 / 已归档”
- 查询来源项目成功文件记录与流水数据
- 按
trxDate过滤流水 - 按现有唯一性口径去重
- 批量写入新项目流水
- 根据实际贡献流水的批次生成新项目文件记录
- 刷新
targetCount - 调用
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_type:UPLOAD/PULL_BANK_INFO/HISTORY_IMPORTsource_project_idsource_project_name
其中:
- 当前已有上传/拉取记录可回填默认
source_type - 历史导入记录固定写入
HISTORY_IMPORT
6. 删除约束设计
6.1 前端删除约束
历史导入记录在上传列表中只读展示:
- 保留文件名
- 展示“历史导入 · 来源项目名”
- 不显示“删除”按钮
6.2 后端删除约束
deleteFileUploadRecord 需增加后端兜底校验:
- 若记录属于
HISTORY_IMPORT,直接拒绝删除 - 返回明确错误信息,例如“历史导入文件不支持删除”
这样可以避免前端绕过行操作约束后误删导入数据。
7. 打标与结果刷新衔接
历史导入完成后,不新增单独导入状态机,直接复用现有打标链路:
- 复制流水成功
- 刷新新项目
targetCount - 调用
bankTagService.submitAutoRebuild(...) - 项目状态进入“打标中”
- 打标完成后刷新结果表、风险统计和总览数据
- 项目状态回到“已完成”
导入失败时:
- 项目保留
- 不进入打标
- 通过上传记录或错误提示体现失败结果
8. 前端弹窗设计
弹窗采用纵向布局:
8.1 新项目配置区
位于弹窗上方,包含:
- 新项目名称
- 备注
- 交易日期范围
8.2 历史项目选择区
位于弹窗下方,包含:
- 项目名称搜索
- 多选历史项目列表
- 固定高度滚动区域
列表只显示“已完成”“已归档”项目。
9. 提交后交互设计
提交成功后:
- 关闭弹窗
- 提示“历史项目导入任务已开始”
- 跳转到新项目详情页
详情页承接方式:
- 复用现有项目状态轮询
- 复用上传记录列表轮询
- 用户在详情页中观察历史导入记录、项目状态和后续打标完成情况
10. 测试关注点
后续实施时需重点验证:
- 多历史项目多选导入
trxDate时间范围过滤- 跨项目重复流水去重
- 历史导入记录来源展示
- 历史导入记录删除拦截
- 导入完成后自动打标与结果表刷新
- 无可导入流水时的失败呈现
风险与约束
- 现有文件记录与流水关联依赖
project_id + log_id(batch_id),批次号重映射必须保证稳定且唯一 - 历史导入记录一旦允许删除,会破坏“复制数据来源可追溯”的约束,因此本次明确禁止删除
- 本次不引入额外任务中心,导入失败的展示能力依赖现有详情页与记录列表,需要实施时保持错误信息可读
结论
本次采用“先创建新项目,再异步导入历史流水并触发现有打标链路”的方案,保持最短路径实现。
该方案满足以下要求:
- 贴合现有项目创建和详情页状态轮询结构
- 不混淆“上传文件”和“复制历史流水”两类业务语义
- 能复用现有打标、结果表和统计刷新能力
- 能明确保证历史导入文件记录不可删除
后续进入实施计划阶段时,需要按仓库约定分别输出前端、后端两份实施计划。