Files
xyzw_web_helper/MD说明文件夹/问题修复-重试时保留成功Token进度v3.12.7.md
2025-10-17 20:56:50 +08:00

13 KiB
Raw Blame History

问题修复 - 重试时保留成功Token进度 v3.12.7

版本: v3.12.7
日期: 2025-10-08
类型: 问题修复

问题描述

用户反馈:

"在失败重试阶段执行进度并未能显示失败token的进度"

现象

  • 点击"重试失败"按钮后
  • 执行进度界面显示所有Token都是"等待中"状态
  • 看不到之前成功的Token的进度
  • 看不到正在重试的Token的执行状态
  • 统计信息显示成功0、失败0、跳过0

预期行为

  • 之前成功的Token应该显示为"已完成"状态
  • 正在重试的Token应该显示为"执行中"状态
  • 统计信息应该保留之前的成功数量

问题分析

根本原因

startBatchExecution 函数中,无论是全新开始还是重试,都会完全重置 taskProgress

// 🆕 如果不是继续执行重置taskProgress
if (!continueFromSaved) {
  taskProgress.value = {}  // ❌ 完全清空包括成功的token
}

问题流程

第1次执行批量任务
- 100个Token
- 成功97个
- 失败3个
- taskProgress 包含所有100个Token的进度

用户点击"重试失败"
- 调用 retryFailedTasks()
- 筛选出3个失败的Token
- 调用 startBatchExecution(failedTokenIds, tasks, true)
  ↓
在 startBatchExecution 中:
- isRetry = true
- continueFromSaved = false
- 执行taskProgress.value = {}  ← 清空所有进度
  ↓
结果:
- 97个成功Token的进度信息丢失 ❌
- UI上看不到任何历史进度 ❌
- 统计数据重置为0 ❌

数据流对比

期望的数据流

初始状态(执行完成后):
taskProgress = {
  'token1': { status: 'completed', ... },  ← 保留
  'token2': { status: 'completed', ... },  ← 保留
  'token3': { status: 'failed', ... },     ← 删除并重新初始化
  'token4': { status: 'completed', ... },  ← 保留
  ...
}

重试后:
taskProgress = {
  'token1': { status: 'completed', ... },  ← 保留
  'token2': { status: 'completed', ... },  ← 保留
  'token3': { status: 'executing', ... },  ← 新初始化
  'token4': { status: 'completed', ... },  ← 保留
  ...
}

实际的数据流(修复前)

初始状态(执行完成后):
taskProgress = {
  'token1': { status: 'completed', ... },
  'token2': { status: 'completed', ... },
  'token3': { status: 'failed', ... },
  'token4': { status: 'completed', ... },
  ...
}

重试后:
taskProgress = {
  'token3': { status: 'executing', ... },  ← 只有这个
}

解决方案

修改1重试时保留成功Token的进度

只删除要重试的Token的进度保留其他Token的进度

修改前

// 🆕 如果不是继续执行重置taskProgress
if (!continueFromSaved) {
  taskProgress.value = {}
}

修改后

// 🆕 如果不是继续执行重置taskProgress
if (!continueFromSaved) {
  // 如果是重试模式保留成功的token进度只重置失败的token
  if (isRetry) {
    // 保留已有的进度数据只清空要重试的token
    targetTokens.forEach(tokenId => {
      if (taskProgress.value[tokenId]) {
        delete taskProgress.value[tokenId]
      }
    })
  } else {
    // 全新开始,清空所有进度
    taskProgress.value = {}
  }
}

修改2重试时保留统计数据

保持原有的成功数量,只重置失败计数:

修改前

// 🆕 初始化统计(如果不是继续执行)
if (!continueFromSaved) {
  executionStats.value = {
    total: targetTokens.length,
    success: 0,
    failed: 0,
    skipped: 0,
    startTime: Date.now(),
    endTime: null
  }
}

修改后

// 🆕 初始化统计(如果不是继续执行)
if (!continueFromSaved) {
  if (isRetry) {
    // 重试模式保持原有的total和success重置失败和跳过计数
    executionStats.value = {
      total: executionStats.value.total,
      success: executionStats.value.success,
      failed: 0, // 重置失败计数,重新统计
      skipped: executionStats.value.skipped,
      startTime: Date.now(),
      endTime: null
    }
  } else {
    // 全新开始:重置所有统计
    executionStats.value = {
      total: targetTokens.length,
      success: 0,
      failed: 0,
      skipped: 0,
      startTime: Date.now(),
      endTime: null
    }
  }
}

修改文件

src/stores/batchTaskStore.js

修改位置1: Line 331-345

  • 重试时保留成功Token的进度
  • 只删除要重试的Token的进度

修改位置2: Line 350-377

  • 重试时保留统计数据
  • 只重置失败计数

用户体验改进

修改前

UI显示

执行进度:

┌─────────────┐ ┌─────────────┐ ┌─────────────┐
│ Token1      │ │ Token2      │ │ Token3      │
│ 等待中      │ │ 等待中      │ │ 等待中      │  ← 所有都是等待中
└─────────────┘ └─────────────┘ └─────────────┘
... (其余97个看不到)

统计:
总计3  成功0  失败0  ← 丢失了之前的97个成功

问题

  • 看不到之前成功的97个Token
  • 统计数据显示0让用户困惑
  • 看起来像是重新执行所有任务

修改后

UI显示

执行进度:

✅ 已完成           ✅ 已完成           🔄 执行中
┌─────────────┐ ┌─────────────┐ ┌─────────────┐
│ Token1      │ │ Token2      │ │ Token3      │
│ 已完成      │ │ 已完成      │ │ 执行中      │  ← 状态清晰
│ 成功: 7     │ │ 成功: 7     │ │ 进度: 42%   │
└─────────────┘ └─────────────┘ └─────────────┘
... (包括其余97个已完成的)

统计:
总计100  成功97  失败0  ← 保留了之前的成功数
重试轮数3/5

改进

  • 显示所有Token的状态
  • 成功的Token显示为"已完成"
  • 重试的Token显示为"执行中"
  • 统计数据准确反映实际情况

技术实现

数据结构变化

修复前(重试时)

// 执行完第1次后
taskProgress = {
  'token1': { status: 'completed', progress: 100, ... },  // 97个
  'token2': { status: 'completed', progress: 100, ... },
  'token98': { status: 'failed', error: '...', ... },     // 3个
  'token99': { status: 'failed', error: '...', ... },
  'token100': { status: 'failed', error: '...', ... }
}

// 点击重试后
taskProgress = {}  // ❌ 全部清空

// 重新初始化
taskProgress = {
  'token98': { status: 'pending', progress: 0, ... },
  'token99': { status: 'pending', progress: 0, ... },
  'token100': { status: 'pending', progress: 0, ... }
}
// 丢失了97个成功的token ❌

修复后(重试时)

// 执行完第1次后
taskProgress = {
  'token1': { status: 'completed', progress: 100, ... },  // 97个
  'token2': { status: 'completed', progress: 100, ... },
  'token98': { status: 'failed', error: '...', ... },     // 3个
  'token99': { status: 'failed', error: '...', ... },
  'token100': { status: 'failed', error: '...', ... }
}

// 点击重试后
// 只删除失败的token
delete taskProgress['token98']
delete taskProgress['token99']
delete taskProgress['token100']

taskProgress = {
  'token1': { status: 'completed', progress: 100, ... },  // ✅ 保留
  'token2': { status: 'completed', progress: 100, ... },  // ✅ 保留
  // ... 其余95个成功的也保留
}

// 重新初始化失败的token
taskProgress = {
  'token1': { status: 'completed', progress: 100, ... },  // ✅ 保留
  'token2': { status: 'completed', progress: 100, ... },  // ✅ 保留
  'token98': { status: 'pending', progress: 0, ... },     // ✅ 新初始化
  'token99': { status: 'pending', progress: 0, ... },     // ✅ 新初始化
  'token100': { status: 'pending', progress: 0, ... }     // ✅ 新初始化
}

统计数据处理

修复前

// 第1次执行完成
executionStats = {
  total: 100,
  success: 97,
  failed: 3,
  skipped: 0
}

// 重试时
executionStats = {
  total: 3,        // ❌ 错误应该是100
  success: 0,      // ❌ 丢失应该保留97
  failed: 0,
  skipped: 0
}

修复后

// 第1次执行完成
executionStats = {
  total: 100,
  success: 97,
  failed: 3,
  skipped: 0
}

// 重试时
executionStats = {
  total: 100,      // ✅ 保留:总数不变
  success: 97,     // ✅ 保留:之前的成功数
  failed: 0,       // ✅ 重置:重新统计重试结果
  skipped: 0       // ✅ 保留:之前的跳过数
}

// 重试完成后假设3个都成功了
executionStats = {
  total: 100,
  success: 100,    // ✅ 97 + 3 = 100
  failed: 0,
  skipped: 0
}

重试场景测试

场景1部分失败后重试全部成功

第1次执行
- 总计100
- 成功97
- 失败3

点击"重试失败 (3个)"
- taskProgress 保留97个成功的
- 删除3个失败的
- 重新初始化3个失败的

重试执行中:
- UI显示100个Token卡片
- 97个显示"已完成"
- 3个显示"执行中"
- 统计总计100成功97失败0

重试完成(全部成功):
- UI显示100个Token卡片
- 100个都显示"已完成"
- 统计总计100成功100失败0 ✅

场景2部分失败后重试仍有失败

第1次执行
- 总计100
- 成功97
- 失败3

第1次重试
- 成功2从失败变成功
- 失败1仍然失败
- 统计总计100成功99失败1

第2次重试
- 成功1最后1个成功
- 失败0
- 统计总计100成功100失败0 ✅

场景3多轮重试

第1次执行
- 成功95失败5

第1次重试自动
- 成功3失败2
- 统计总计100成功98失败2

第2次重试自动
- 成功1失败1
- 统计总计100成功99失败1

第3次重试自动
- 成功0失败1
- 统计总计100成功99失败1
- 达到最大重试次数,停止重试

边界情况处理

情况1全部失败后重试

第1次执行
- 成功0
- 失败100

重试:
- taskProgress 原本就是空的(或只有失败的)
- 删除所有失败的
- 重新初始化所有token
- 行为正常 ✅

情况2全部成功无需重试

第1次执行
- 成功100
- 失败0

点击重试:
- failedTokenIds = []
- 提示"没有失败的任务需要重试"
- 不执行重试
- 行为正常 ✅

情况3连续重试

第1次执行
- 成功97失败3

第1次重试
- 成功2失败1
- taskProgress 保留99个97+2
- 删除1个失败的
- 统计正确 ✅

第2次重试
- 成功1失败0
- taskProgress 保留100个
- 统计正确 ✅

代码逻辑说明

删除失败Token的逻辑

if (isRetry) {
  // 遍历要重试的token失败的token
  targetTokens.forEach(tokenId => {
    // 如果该token在taskProgress中存在删除它
    if (taskProgress.value[tokenId]) {
      delete taskProgress.value[tokenId]
    }
  })
}

为什么使用 delete

  • delete 从对象中完全移除属性
  • 不留痕迹,确保后续初始化时是全新的状态
  • 避免旧数据影响新的执行

为什么检查 if (taskProgress.value[tokenId])

  • 防御性编程,避免删除不存在的属性
  • 虽然理论上失败的token一定存在但加上检查更安全

保留统计数据的逻辑

if (isRetry) {
  executionStats.value = {
    total: executionStats.value.total,      // 保持不变
    success: executionStats.value.success,  // 保持已有的成功数
    failed: 0,                              // 重置为0重新统计
    skipped: executionStats.value.skipped,  // 保持已有的跳过数
    startTime: Date.now(),                  // 更新为当前时间
    endTime: null                           // 清空结束时间
  }
}

为什么重置 failed 为 0

  • 失败的token正在重试
  • 重试后可能成功,也可能仍然失败
  • 需要重新统计重试的结果

为什么保留 success 和 skipped

  • 这些是之前已经确定的结果
  • 不会因为重试而改变

相关版本

  • v3.7.0: 首次添加自动重试功能
  • v3.12.6: 增强重试日志输出
  • v3.12.7: 修复重试时保留成功Token进度本版本

总结

问题

  • 重试时清空所有进度丢失成功Token信息
  • UI显示不完整只能看到重试的Token
  • 统计数据重置,无法反映真实情况

修复

  • 重试时保留成功Token的进度
  • 只删除和重新初始化失败Token
  • 保留统计数据中的成功数和总数
  • 只重置失败计数

效果

  • UI完整显示所有Token状态
  • 成功Token显示为"已完成"
  • 重试Token显示为"执行中"
  • 统计数据准确反映实际情况
  • 用户体验大幅提升

状态: 已修复
版本: v3.12.7