dominds 0.8.3 → 0.8.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (111) hide show
  1. package/dist/dialog.js +4 -3
  2. package/dist/docs/keep-going.zh.md +162 -0
  3. package/dist/docs/showing-by-doing.md +208 -0
  4. package/dist/docs/showing-by-doing.zh.md +177 -0
  5. package/dist/llm/defaults.yaml +24 -0
  6. package/dist/llm/driver-v2/core.js +18 -1
  7. package/dist/llm/driver-v2/round.js +1 -0
  8. package/dist/minds/promptdocs.js +263 -0
  9. package/dist/persistence.js +23 -2
  10. package/dist/server/websocket-handler.js +19 -1
  11. package/dist/showing-by-doing.js +34 -31
  12. package/dist/snippets/README.en.md +3 -0
  13. package/dist/static/assets/{_baseUniq-BQQDkYjW.js → _baseUniq-eftsUNcO.js} +2 -2
  14. package/dist/static/assets/{_baseUniq-BQQDkYjW.js.map → _baseUniq-eftsUNcO.js.map} +1 -1
  15. package/dist/static/assets/{arc-D9VW2OIF.js → arc-BoKor7ZE.js} +2 -2
  16. package/dist/static/assets/{arc-D9VW2OIF.js.map → arc-BoKor7ZE.js.map} +1 -1
  17. package/dist/static/assets/{architectureDiagram-VXUJARFQ-DeorRGuF.js → architectureDiagram-VXUJARFQ-DJhWme8U.js} +6 -6
  18. package/dist/static/assets/{architectureDiagram-VXUJARFQ-DeorRGuF.js.map → architectureDiagram-VXUJARFQ-DJhWme8U.js.map} +1 -1
  19. package/dist/static/assets/{blockDiagram-VD42YOAC-CnOA6OCa.js → blockDiagram-VD42YOAC-BMLWzKNK.js} +7 -7
  20. package/dist/static/assets/{blockDiagram-VD42YOAC-CnOA6OCa.js.map → blockDiagram-VD42YOAC-BMLWzKNK.js.map} +1 -1
  21. package/dist/static/assets/{c4Diagram-YG6GDRKO-CjpO0nXo.js → c4Diagram-YG6GDRKO-p2EvHX08.js} +3 -3
  22. package/dist/static/assets/{c4Diagram-YG6GDRKO-CjpO0nXo.js.map → c4Diagram-YG6GDRKO-p2EvHX08.js.map} +1 -1
  23. package/dist/static/assets/{channel-B6t8yjZQ.js → channel-MYAkzdaE.js} +2 -2
  24. package/dist/static/assets/{channel-B6t8yjZQ.js.map → channel-MYAkzdaE.js.map} +1 -1
  25. package/dist/static/assets/{chunk-4BX2VUAB-oA-Uv0N5.js → chunk-4BX2VUAB-C3GhLt2Q.js} +2 -2
  26. package/dist/static/assets/{chunk-4BX2VUAB-oA-Uv0N5.js.map → chunk-4BX2VUAB-C3GhLt2Q.js.map} +1 -1
  27. package/dist/static/assets/{chunk-55IACEB6-hWcXX4Dk.js → chunk-55IACEB6-jht8YofA.js} +2 -2
  28. package/dist/static/assets/{chunk-55IACEB6-hWcXX4Dk.js.map → chunk-55IACEB6-jht8YofA.js.map} +1 -1
  29. package/dist/static/assets/{chunk-B4BG7PRW-vCNRSkGX.js → chunk-B4BG7PRW-CA_-frQ5.js} +5 -5
  30. package/dist/static/assets/{chunk-B4BG7PRW-vCNRSkGX.js.map → chunk-B4BG7PRW-CA_-frQ5.js.map} +1 -1
  31. package/dist/static/assets/{chunk-DI55MBZ5-CMlImJ8W.js → chunk-DI55MBZ5-jM-We68d.js} +4 -4
  32. package/dist/static/assets/{chunk-DI55MBZ5-CMlImJ8W.js.map → chunk-DI55MBZ5-jM-We68d.js.map} +1 -1
  33. package/dist/static/assets/{chunk-FMBD7UC4-C9159kMi.js → chunk-FMBD7UC4-FaTVrJ7l.js} +2 -2
  34. package/dist/static/assets/{chunk-FMBD7UC4-C9159kMi.js.map → chunk-FMBD7UC4-FaTVrJ7l.js.map} +1 -1
  35. package/dist/static/assets/{chunk-QN33PNHL-BrWHoFwa.js → chunk-QN33PNHL-DY14DHwx.js} +2 -2
  36. package/dist/static/assets/{chunk-QN33PNHL-BrWHoFwa.js.map → chunk-QN33PNHL-DY14DHwx.js.map} +1 -1
  37. package/dist/static/assets/{chunk-QZHKN3VN-CgW37qBX.js → chunk-QZHKN3VN--MzSOzG9.js} +2 -2
  38. package/dist/static/assets/{chunk-QZHKN3VN-CgW37qBX.js.map → chunk-QZHKN3VN--MzSOzG9.js.map} +1 -1
  39. package/dist/static/assets/{chunk-TZMSLE5B-CJtwjbgx.js → chunk-TZMSLE5B-CePGjqpY.js} +2 -2
  40. package/dist/static/assets/{chunk-TZMSLE5B-CJtwjbgx.js.map → chunk-TZMSLE5B-CePGjqpY.js.map} +1 -1
  41. package/dist/static/assets/{classDiagram-2ON5EDUG-Dpe8z-s9.js → classDiagram-2ON5EDUG-i5wSGfrt.js} +6 -6
  42. package/dist/static/assets/{classDiagram-2ON5EDUG-Dpe8z-s9.js.map → classDiagram-2ON5EDUG-i5wSGfrt.js.map} +1 -1
  43. package/dist/static/assets/{classDiagram-v2-WZHVMYZB-Dpe8z-s9.js → classDiagram-v2-WZHVMYZB-i5wSGfrt.js} +6 -6
  44. package/dist/static/assets/{classDiagram-v2-WZHVMYZB-Dpe8z-s9.js.map → classDiagram-v2-WZHVMYZB-i5wSGfrt.js.map} +1 -1
  45. package/dist/static/assets/{clone-6qHEU-TP.js → clone-C8ajhCTN.js} +2 -2
  46. package/dist/static/assets/{clone-6qHEU-TP.js.map → clone-C8ajhCTN.js.map} +1 -1
  47. package/dist/static/assets/{cose-bilkent-S5V4N54A-DHXnyLOv.js → cose-bilkent-S5V4N54A-DpE-Lvec.js} +2 -2
  48. package/dist/static/assets/{cose-bilkent-S5V4N54A-DHXnyLOv.js.map → cose-bilkent-S5V4N54A-DpE-Lvec.js.map} +1 -1
  49. package/dist/static/assets/{dagre-6UL2VRFP-DN9OGfii.js → dagre-6UL2VRFP-3b1MFQKy.js} +7 -7
  50. package/dist/static/assets/{dagre-6UL2VRFP-DN9OGfii.js.map → dagre-6UL2VRFP-3b1MFQKy.js.map} +1 -1
  51. package/dist/static/assets/{diagram-PSM6KHXK-R6JaPZML.js → diagram-PSM6KHXK-C4TCSshj.js} +7 -7
  52. package/dist/static/assets/{diagram-PSM6KHXK-R6JaPZML.js.map → diagram-PSM6KHXK-C4TCSshj.js.map} +1 -1
  53. package/dist/static/assets/{diagram-QEK2KX5R-hOtG__bJ.js → diagram-QEK2KX5R-DkG2ybbr.js} +6 -6
  54. package/dist/static/assets/{diagram-QEK2KX5R-hOtG__bJ.js.map → diagram-QEK2KX5R-DkG2ybbr.js.map} +1 -1
  55. package/dist/static/assets/{diagram-S2PKOQOG-BH3MkMHA.js → diagram-S2PKOQOG-B0050uDB.js} +6 -6
  56. package/dist/static/assets/{diagram-S2PKOQOG-BH3MkMHA.js.map → diagram-S2PKOQOG-B0050uDB.js.map} +1 -1
  57. package/dist/static/assets/{erDiagram-Q2GNP2WA-hwCF5S5C.js → erDiagram-Q2GNP2WA-CiQrjdUM.js} +5 -5
  58. package/dist/static/assets/{erDiagram-Q2GNP2WA-hwCF5S5C.js.map → erDiagram-Q2GNP2WA-CiQrjdUM.js.map} +1 -1
  59. package/dist/static/assets/{flowDiagram-NV44I4VS-DSHxoCSZ.js → flowDiagram-NV44I4VS-B85IVzMr.js} +6 -6
  60. package/dist/static/assets/{flowDiagram-NV44I4VS-DSHxoCSZ.js.map → flowDiagram-NV44I4VS-B85IVzMr.js.map} +1 -1
  61. package/dist/static/assets/{ganttDiagram-JELNMOA3-DaJrRw85.js → ganttDiagram-JELNMOA3-DRL3UcLB.js} +3 -3
  62. package/dist/static/assets/{ganttDiagram-JELNMOA3-DaJrRw85.js.map → ganttDiagram-JELNMOA3-DRL3UcLB.js.map} +1 -1
  63. package/dist/static/assets/{gitGraphDiagram-NY62KEGX-65AIqn8B.js → gitGraphDiagram-NY62KEGX--BTdkTj8.js} +7 -7
  64. package/dist/static/assets/{gitGraphDiagram-NY62KEGX-65AIqn8B.js.map → gitGraphDiagram-NY62KEGX--BTdkTj8.js.map} +1 -1
  65. package/dist/static/assets/{graph-txx-dN-Y.js → graph-Cq48BmHF.js} +3 -3
  66. package/dist/static/assets/{graph-txx-dN-Y.js.map → graph-Cq48BmHF.js.map} +1 -1
  67. package/dist/static/assets/{index-l25foR5f.js → index-BwRb9mqE.js} +251 -45
  68. package/dist/static/assets/index-BwRb9mqE.js.map +1 -0
  69. package/dist/static/assets/{infoDiagram-WHAUD3N6-DWsEnxF_.js → infoDiagram-WHAUD3N6-D5LM2I5v.js} +5 -5
  70. package/dist/static/assets/{infoDiagram-WHAUD3N6-DWsEnxF_.js.map → infoDiagram-WHAUD3N6-D5LM2I5v.js.map} +1 -1
  71. package/dist/static/assets/{journeyDiagram-XKPGCS4Q-CmT5AYT3.js → journeyDiagram-XKPGCS4Q-DYbSmtgI.js} +5 -5
  72. package/dist/static/assets/{journeyDiagram-XKPGCS4Q-CmT5AYT3.js.map → journeyDiagram-XKPGCS4Q-DYbSmtgI.js.map} +1 -1
  73. package/dist/static/assets/{kanban-definition-3W4ZIXB7-DT2lqVgc.js → kanban-definition-3W4ZIXB7-CMqglPaJ.js} +3 -3
  74. package/dist/static/assets/{kanban-definition-3W4ZIXB7-DT2lqVgc.js.map → kanban-definition-3W4ZIXB7-CMqglPaJ.js.map} +1 -1
  75. package/dist/static/assets/{layout-S8zTlR0m.js → layout-CoiKguN7.js} +5 -5
  76. package/dist/static/assets/{layout-S8zTlR0m.js.map → layout-CoiKguN7.js.map} +1 -1
  77. package/dist/static/assets/{linear-BXU_vKT1.js → linear-XPX88yLs.js} +2 -2
  78. package/dist/static/assets/{linear-BXU_vKT1.js.map → linear-XPX88yLs.js.map} +1 -1
  79. package/dist/static/assets/{min-Du3h591V.js → min-BwciICs1.js} +3 -3
  80. package/dist/static/assets/{min-Du3h591V.js.map → min-BwciICs1.js.map} +1 -1
  81. package/dist/static/assets/{mindmap-definition-VGOIOE7T-Dg0kj52o.js → mindmap-definition-VGOIOE7T-D04eBe-e.js} +4 -4
  82. package/dist/static/assets/{mindmap-definition-VGOIOE7T-Dg0kj52o.js.map → mindmap-definition-VGOIOE7T-D04eBe-e.js.map} +1 -1
  83. package/dist/static/assets/{pieDiagram-ADFJNKIX-CbUv_9y2.js → pieDiagram-ADFJNKIX-CTPmE-2N.js} +7 -7
  84. package/dist/static/assets/{pieDiagram-ADFJNKIX-CbUv_9y2.js.map → pieDiagram-ADFJNKIX-CTPmE-2N.js.map} +1 -1
  85. package/dist/static/assets/{quadrantDiagram-AYHSOK5B-olHVqYDq.js → quadrantDiagram-AYHSOK5B-CS7Z6xSp.js} +3 -3
  86. package/dist/static/assets/{quadrantDiagram-AYHSOK5B-olHVqYDq.js.map → quadrantDiagram-AYHSOK5B-CS7Z6xSp.js.map} +1 -1
  87. package/dist/static/assets/{requirementDiagram-UZGBJVZJ-BrAoW4wZ.js → requirementDiagram-UZGBJVZJ-D060g5Vz.js} +4 -4
  88. package/dist/static/assets/{requirementDiagram-UZGBJVZJ-BrAoW4wZ.js.map → requirementDiagram-UZGBJVZJ-D060g5Vz.js.map} +1 -1
  89. package/dist/static/assets/{sankeyDiagram-TZEHDZUN-DkPJIMoK.js → sankeyDiagram-TZEHDZUN-DdnhO4xC.js} +2 -2
  90. package/dist/static/assets/{sankeyDiagram-TZEHDZUN-DkPJIMoK.js.map → sankeyDiagram-TZEHDZUN-DdnhO4xC.js.map} +1 -1
  91. package/dist/static/assets/{sequenceDiagram-WL72ISMW-B2DQ5HcH.js → sequenceDiagram-WL72ISMW-KVf0smEC.js} +4 -4
  92. package/dist/static/assets/{sequenceDiagram-WL72ISMW-B2DQ5HcH.js.map → sequenceDiagram-WL72ISMW-KVf0smEC.js.map} +1 -1
  93. package/dist/static/assets/{stateDiagram-FKZM4ZOC-C-dZxUTQ.js → stateDiagram-FKZM4ZOC-bjqKG0l2.js} +9 -9
  94. package/dist/static/assets/{stateDiagram-FKZM4ZOC-C-dZxUTQ.js.map → stateDiagram-FKZM4ZOC-bjqKG0l2.js.map} +1 -1
  95. package/dist/static/assets/{stateDiagram-v2-4FDKWEC3-CE6XVPBT.js → stateDiagram-v2-4FDKWEC3-DwnLWnFf.js} +5 -5
  96. package/dist/static/assets/{stateDiagram-v2-4FDKWEC3-CE6XVPBT.js.map → stateDiagram-v2-4FDKWEC3-DwnLWnFf.js.map} +1 -1
  97. package/dist/static/assets/{timeline-definition-IT6M3QCI-BK1ikH27.js → timeline-definition-IT6M3QCI-DRn-DVo3.js} +3 -3
  98. package/dist/static/assets/{timeline-definition-IT6M3QCI-BK1ikH27.js.map → timeline-definition-IT6M3QCI-DRn-DVo3.js.map} +1 -1
  99. package/dist/static/assets/{treemap-KMMF4GRG-5DBeofAb.js → treemap-KMMF4GRG-u-TJXRTX.js} +4 -4
  100. package/dist/static/assets/{treemap-KMMF4GRG-5DBeofAb.js.map → treemap-KMMF4GRG-u-TJXRTX.js.map} +1 -1
  101. package/dist/static/assets/{xychartDiagram-PRI3JC2R-O27QygbN.js → xychartDiagram-PRI3JC2R-D95qTcTg.js} +3 -3
  102. package/dist/static/assets/{xychartDiagram-PRI3JC2R-O27QygbN.js.map → xychartDiagram-PRI3JC2R-D95qTcTg.js.map} +1 -1
  103. package/dist/static/index.html +1 -1
  104. package/dist/tools/context-health.js +7 -7
  105. package/dist/tools/pending-tellask-reminder.js +2 -1
  106. package/dist/utils/task-doc.js +16 -16
  107. package/package.json +1 -1
  108. package/dist/minds/builtin/cmdr/persona.md +0 -3
  109. package/dist/minds/builtin/dijiang/knowledge.md +0 -287
  110. package/dist/minds/builtin/dijiang/persona.md +0 -7
  111. package/dist/static/assets/index-l25foR5f.js.map +0 -1
package/dist/dialog.js CHANGED
@@ -519,6 +519,7 @@ class Dialog {
519
519
  msgId: options.msgId,
520
520
  grammar: options.grammar,
521
521
  userLanguageCode: options.userLanguageCode ?? this._lastUserLanguageCode,
522
+ q4hAnswerCallIds: options.q4hAnswerCallIds,
522
523
  };
523
524
  this._updatedAt = (0, time_1.formatUnifiedTimestamp)(new Date());
524
525
  }
@@ -670,8 +671,8 @@ class Dialog {
670
671
  async updateQuestions4Human(questions) {
671
672
  return await this.dlgStore.updateQuestions4Human(this, questions);
672
673
  }
673
- async persistUserMessage(content, msgId, grammar, userLanguageCode) {
674
- return await this.dlgStore.persistUserMessage(this, content, msgId, grammar, userLanguageCode);
674
+ async persistUserMessage(content, msgId, grammar, userLanguageCode, q4hAnswerCallIds) {
675
+ return await this.dlgStore.persistUserMessage(this, content, msgId, grammar, userLanguageCode, q4hAnswerCallIds);
675
676
  }
676
677
  async persistAgentMessage(content, genseq, type, provider_data) {
677
678
  return await this.dlgStore.persistAgentMessage(this, content, genseq, type, provider_data);
@@ -1047,7 +1048,7 @@ class DialogStore {
1047
1048
  /**
1048
1049
  * Persist a user message to storage
1049
1050
  */
1050
- async persistUserMessage(_dialog, _content, _msgId, _grammar, _userLanguageCode) { }
1051
+ async persistUserMessage(_dialog, _content, _msgId, _grammar, _userLanguageCode, _q4hAnswerCallIds) { }
1051
1052
  /**
1052
1053
  * Persist an assistant message to storage
1053
1054
  */
@@ -0,0 +1,162 @@
1
+ # Keep-Going(鞭策)— 设计文档
2
+
3
+ ## 概述
4
+
5
+ Dominds 根对话旨在长期运行。根对话"停止"(变为空闲)通常不是操作员想要的:他们希望智能体持续推进,直到:
6
+
7
+ - 合法地暂停等待人类决策(Q4H),或
8
+ - 合法地暂停等待子对话(tellask/backfill)。
9
+
10
+ 本文档指定了一个运行时机制("keep-going"),仅针对**根/主对话**,防止对话停止:每当驱动程序即将停止时,它会自动发送一个简短的鞭策语(渲染为正常的用户气泡)并继续生成,除非对话处于合法暂停状态(Q4H 或待处理的子对话)。
11
+
12
+ ## 目标
13
+
14
+ - 防止根对话停止,除非处于合法暂停状态(Q4H / 子对话)。
15
+ - 保持行为可预测和有界(无无限循环)。
16
+ - 使鞭策语文本可按工作区(rtws)和语言配置。
17
+ - 提供清晰的用户控制的"禁用"机制。
18
+
19
+ ## 非目标
20
+
21
+ - 自动完成/自动将对话标记为完成。
22
+ - 将此行为应用于子对话(子对话保持范围,应向其调用者报告)。
23
+
24
+ ## 定义
25
+
26
+ - **根/主对话**:`RootDialog`(`dlg.id.rootId === dlg.id.selfId`),主要对话线程。
27
+ - **子对话**:`SubDialog`,为 tellask / 作用域工作创建。
28
+ - **Q4H**:"Questions for Human",通过 `!?@human` 发起,暂停对话进度直到人类响应。
29
+
30
+ ## 预期的"正常"完成路径(推荐)
31
+
32
+ 当智能体需要人类决策来结束时(例如,确认选择或决定是否将对话标记为完成),正确的路径是:
33
+
34
+ 1. 智能体发出 Q4H(`!?@human`)并提供必要的上下文和明确的决策请求。
35
+ 2. WebUI 清楚地呈现 Q4H。
36
+ 3. 人类决定并:
37
+ - 手动将根对话标记为"完成",或
38
+ - 提供请求的信息以便对话继续。
39
+
40
+ 这是"受控收敛"路径。keep-going 机制不应覆盖合法的暂停状态。
41
+
42
+ ## Keep-Going 行为("自动继续"回退)
43
+
44
+ ### 触发条件(必须全部满足)
45
+
46
+ - 对话是**根/主对话**(绝不会是子对话)。
47
+ - 对话**未暂停**:
48
+ - 没有待处理的 Q4H,并且
49
+ - 没有待处理的子对话(等待回填)。
50
+ - 驱动程序即将停止生成循环(即没有工具/函数输出需要另一次迭代)。
51
+
52
+ ### 操作
53
+
54
+ 运行时自动发送一个鞭策语(渲染为正常的用户气泡)并运行另一次生成迭代。
55
+
56
+ ### 有界性
57
+
58
+ 为避免无限循环,keep-going 有一个按对话的预算(每个成员的 `diligence-push-max`),控制对于给定对话在运行时强制 Q4H 暂停之前可以注入多少个自动继续的鞭策语。
59
+
60
+ - 默认值:**3**
61
+ - 如果 `< 1`,则该成员的 keep-going 有效禁用
62
+ - 可通过 `.minds/team.yaml` 中的 `diligence-push-max` 按成员配置
63
+
64
+ ### Q4H 时重置
65
+
66
+ 当对话因待处理的 Q4H(Questions for Human)而暂停时,keep-going 注入计数器会被重置。这确保在人类回答 Q4H 并恢复对话后,对话会获得新的 keep-going 预算。
67
+
68
+ ### 预算耗尽 → 强制 Q4H
69
+
70
+ 当 keep-going 预算耗尽时,运行时会创建一个 Q4H 条目,询问人类是否继续或停止。这将"有界性"转换为合法的暂停点,并避免无限自动继续循环。
71
+
72
+ ### 禁用开关
73
+
74
+ 可以通过以下任一方式按 rtws 禁用 keep-going:
75
+
76
+ - 如果选中的鞭策语文件存在但其内容为空/仅空白,则禁用 keep-going(不注入)。
77
+
78
+ 可以通过以下任一方式按成员禁用 keep-going:
79
+
80
+ - 如果 `diligence-push-max < 1`,则该成员的 keep-going 被禁用(不注入)。
81
+
82
+ ## 鞭策语解析
83
+
84
+ 让 `<rtws>` 为当前运行时工作区(即 `process.cwd()`)。
85
+
86
+ 解析顺序:
87
+
88
+ 1. `<rtws>/.minds/diligence.<work-lang-id>.md`(例如,`diligence.zh.md`)
89
+ 2. `<rtws>/.minds/diligence.md`(语言无关的回退)
90
+ 3. 内置回退文本(硬编码的 i18n;`zh` 是规范的并嵌入在源代码中)
91
+
92
+ 如果上述顺序中第一个存在的文件具有空/空白内容,则**禁用** keep-going。
93
+
94
+ 注意:鞭策语文件中的 YAML frontmatter 会被运行时忽略。如果存在,它被视为非内容元数据并从提示正文中剥离。
95
+
96
+ ### 团队成员上限:`diligence-push-max`
97
+
98
+ 每个团队成员可以选择通过 `.minds/team.yaml` 限制 keep-going:
99
+
100
+ ```yaml
101
+ members:
102
+ alice:
103
+ diligence-push-max: 10
104
+ ```
105
+
106
+ 规则:
107
+
108
+ - 如果缺失,`diligence-push-max` 对于该成员默认为 **3**。
109
+ - 如果 `diligence-push-max < 1`,则该成员的 keep-going 被禁用(不注入),即使鞭策语文件存在。
110
+ - 内置影子成员 `fuxi` 和 `pangu` 默认为 `diligence-push-max: 0`,除非在 team.yaml 中显式覆盖。
111
+
112
+ ## UX 备注
113
+
114
+ - Keep-going 是一个仅运行时的提示,但它应该是**可见的**:鞭策语渲染为正常的用户消息气泡(由运行时自动发送),以便操作员理解为什么会出现额外的迭代。
115
+ - 用户应该观察到智能体在仅工具操作后继续简短的跟进。
116
+ - 当智能体真正需要用户干预时,它应该使用 Q4H。Keep-going 不应试图"假装"完成。
117
+
118
+ ## 实现(后端)
119
+
120
+ ### 位置
121
+
122
+ 在 LLM 驱动程序循环中实现(`dominds/main/llm/driver.ts`),作为迭代后的小检查:
123
+
124
+ 1. 如果 `suspendForHuman` 为 true,则停止(Q4H / 子对话待处理)。
125
+ 2. 如果有任何工具反馈,则正常继续。
126
+ 3. 否则(仅限根),尝试 keep-going 自动继续:
127
+ - 如果禁用 → 正常停止。
128
+ - 如果预算耗尽 → 创建 Q4H 并停止。
129
+ - 否则 → 自动发送鞭策语并继续。
130
+
131
+ ### 消息类型
132
+
133
+ 我们将鞭策语作为自动发送的用户消息注入:
134
+
135
+ - 类型为 `prompting_msg`、角色为 `'user'` 的 `ChatMessage`
136
+
137
+ 这确保了:
138
+
139
+ - 它存在于模型上下文中
140
+ - 它作为人类消息记录持久化
141
+ - 它在聊天时间线中渲染为正常的用户气泡(像任何其他用户消息一样)
142
+
143
+ ## 可观察性
144
+
145
+ 建议的后续步骤(初始实现不需要):
146
+
147
+ - 当触发 keep-going 时添加结构化日志行,包括:
148
+ - 对话 id
149
+ - 语言
150
+ - 使用的鞭策语来源(语言特定/通用/内置/禁用)
151
+ - 为"keep-going 触发"和"keep-going 因空文件禁用"添加可选的指标计数器。
152
+
153
+ ## 测试
154
+
155
+ 回归测试应覆盖:
156
+
157
+ - 根对话:仅工具输出 → 鞭策语注入 → 继续响应
158
+ - 根对话:空助手输出 → 鞭策语注入 → 继续响应
159
+ - 子对话:无鞭策语注入
160
+ - 工作区配置:
161
+ - 当语言特定文件缺失时,`.minds/diligence.md` 被遵守
162
+ - 空的鞭策语文件禁用 keep-going
@@ -0,0 +1,208 @@
1
+ # Showing-by-Doing (做给祂看): Make Tellask Real
2
+
3
+ Chinese version: [中文版](./showing-by-doing.zh.md)
4
+
5
+ ## Summary
6
+
7
+ Dominds has a real, runtime-enforced Tellask mechanism (`!?@...`) and a real, runtime-enforced Fresh Boots Reasoning
8
+ (FBR) mechanism (`!?@self`). Even if system prompts explain these mechanisms in detail, most foundation models were not
9
+ trained in a world where “asking a teammate to run a shell command” is actually possible, so they often treat such text
10
+ as aspirational or hypothetical.
11
+
12
+ **Showing-by-Doing (做给祂看)** is a tiny, highly realistic “first impression” procedure executed at dialog creation
13
+ time. It runs a short, real Tellask + real FBR round-trip so the model gains _felt-sense_ that:
14
+
15
+ - teammate Tellasks are real and will be executed
16
+ - tool outputs are real and will be returned
17
+ - FBR is real and will produce a report back to the caller
18
+
19
+ This makes subsequent use of Tellasks and `!?@self` FBR more reliable across the entire dialog lifecycle.
20
+
21
+ As a consequence, we can **dramatically simplify system prompts**: keep only short, accurate statements of the Tellask/FBR
22
+ contracts, and rely on Showing-by-Doing to establish “this is real and works here”.
23
+
24
+ Related docs:
25
+
26
+ - Tellask runtime: `dominds/docs/dialog-system.md`
27
+ - Terminology (Mainline/Sideline): `dominds/docs/dominds-terminology.md`
28
+ - FBR (`!?@self`): `dominds/docs/fbr.md`
29
+ - Work language vs UI language: `dominds/docs/i18n.md`
30
+
31
+ ---
32
+
33
+ ## Goals
34
+
35
+ - Establish immediate trust that **Tellask is not “just prompt text”**: it is an executable, backend-driven mechanism.
36
+ - Establish immediate trust that **FBR is not “just reasoning”**: it is a concrete sideline dialog that reports back.
37
+ - Keep the procedure safe, small, and deterministic (default command: `uname -a`).
38
+ - Persist and display the interaction so it is credible from multiple angles (backend record + frontend transcript).
39
+
40
+ ## Non-goals
41
+
42
+ - Running arbitrary user-provided commands as part of dialog creation.
43
+ - Collecting sensitive system information (beyond minimal OS/kernel identification).
44
+ - Replacing proper documentation: this is an experiential supplement, not the main spec.
45
+
46
+ ---
47
+
48
+ ## Definitions (user-facing)
49
+
50
+ - **Mainline dialog**: the primary thread where the user and the main agent interact.
51
+ - **Sideline dialog**: a temporary work thread created by Tellask / FBR, reporting results back to the mainline.
52
+ - **Tellask**: a structured request (`!?@<memberId> ...`) from a tellasker to a tellaskee.
53
+ - **Shell specialist**: a teammate designated to run shell commands safely (configured via `shell_specialists`).
54
+ - **FBR**: Fresh Boots Reasoning, implemented as `!?@self` (a tool-less sideline dialog). See `dominds/docs/fbr.md`.
55
+
56
+ ---
57
+
58
+ ## Runtime flow (at dialog creation)
59
+
60
+ Showing-by-Doing runs **before** the first user message is processed (a “prelude”), unless the user opts out.
61
+
62
+ ### 1) Choose the shell execution path
63
+
64
+ 1. If the team config includes at least one `shell_specialists` member, pick one deterministically (e.g. the first).
65
+ 2. Otherwise, let the **Dominds runtime** execute the baseline command directly (not via agent tools).
66
+
67
+ Note: team config validation normally requires `shell_specialists` to be non-empty if any teammate has shell tools, and
68
+ also requires that only listed shell specialists have shell tools. In other words, “no `shell_specialists` configured”
69
+ usually means “shell tools are disabled for all user-facing teammates”.
70
+
71
+ Baseline command:
72
+
73
+ - `uname -a`
74
+
75
+ If `uname -a` fails (e.g. non-POSIX host), the runtime should fall back to a platform-appropriate equivalent, but the
76
+ default “muscle memory” path is intentionally `uname -a` because it is common, low-risk, and quick.
77
+
78
+ ### 2) Real teammate Tellask: ask for `uname -a`
79
+
80
+ If a shell specialist exists, the main agent (tellasker) issues **a real Tellask** to the shell specialist (tellaskee),
81
+ in the server-wide **work language** (see `dominds/docs/i18n.md`).
82
+
83
+ The Tellask body should be short and operational:
84
+
85
+ - ask the shell specialist to run `uname -a`
86
+ - ask them to return the raw output verbatim
87
+ - ask them to include an error + alternative command if `uname` is not available
88
+
89
+ The result must be recorded as a normal, debuggable transcript that clearly shows:
90
+
91
+ - the Tellask headline + body
92
+ - the shell specialist’s command execution
93
+ - the returned output
94
+
95
+ Minimal configuration example:
96
+
97
+ ```yaml
98
+ # .minds/team.yaml
99
+ shell_specialists:
100
+ - cmdr
101
+ ```
102
+
103
+ ### 3) Real FBR: reflect on what the environment implies
104
+
105
+ After obtaining the environment snapshot (from the shell specialist or direct tool run), the main agent issues a real
106
+ `!?@self` Tellask to trigger **FBR**.
107
+
108
+ The FBR Tellask body should include:
109
+
110
+ - the exact `uname -a` output
111
+ - the main agent’s tool availability assumptions (e.g. “in mainline I may have `fs`/`os`/`mem`; in FBR you have no tools”)
112
+ - the question: “What should I be careful about in this environment? Which CLI tools should I prioritize, and why?”
113
+
114
+ The FBR response should return a compact report that the main agent can reuse, typically:
115
+
116
+ - environment notes (OS family, kernel, container/VM hints if any)
117
+ - do/don’t list (pitfalls)
118
+ - a short “preferred commands” shortlist (e.g. `rg`, `sed`, `jq`, `tar`, `ps`, `lsof`, depending on host)
119
+
120
+ ### 4) Summarize into an “Environment Quickstart” note
121
+
122
+ The main agent then writes a short, user-visible summary to the mainline dialog so the knowledge is explicit and
123
+ searchable in the transcript.
124
+
125
+ ---
126
+
127
+ ## Fallback behavior
128
+
129
+ Showing-by-Doing should degrade gracefully:
130
+
131
+ - **No `shell_specialists` configured**: the Dominds runtime runs `uname -a` directly, then do FBR.
132
+ - **Shell specialist fails or times out**: the Dominds runtime runs `uname -a` directly and continue, marking the
133
+ specialist attempt as failed in the transcript.
134
+ - **`uname` not available / command fails**: run FBR with “environment unknown”; the report should explicitly list what
135
+ is missing and suggest what to gather later.
136
+
137
+ ---
138
+
139
+ ## Persistence, caching, and reuse
140
+
141
+ ### 1) Persist as real dialog records (backend + frontend)
142
+
143
+ All Showing-by-Doing steps must be persisted as standard dialog artifacts (messages + events) so they remain credible:
144
+
145
+ - they survive restarts (subject to normal dialog persistence rules)
146
+ - they are debuggable in logs/storage
147
+ - they are rendered in the WebUI transcript (ideally as a collapsible “Prelude” section)
148
+
149
+ ### 2) Process-wide cache (per teammate)
150
+
151
+ To avoid repeating `uname -a` + FBR on every new dialog, the backend process may maintain a **server-scoped cache** keyed
152
+ by (at minimum):
153
+
154
+ - tellasker teammate id (the agent whose dialogs are being created)
155
+ - an environment fingerprint derived from the `uname -a` output (exact string or hashed)
156
+
157
+ Cache payload (recommended):
158
+
159
+ - the full Showing-by-Doing transcript (Tellask + output + FBR + summary)
160
+ - a short “Environment Quickstart” distilled form (for compact context injection)
161
+ - timestamps + a version tag for future schema changes
162
+
163
+ Reuse policy:
164
+
165
+ - If a valid cache entry exists, a new dialog may **reuse** the cached transcript instead of re-running.
166
+ - Reused entries should still be visible in the new dialog’s transcript, clearly labeled as “Reused from cache”.
167
+
168
+ Invalidation (recommended):
169
+
170
+ - invalidate when the environment fingerprint changes
171
+ - allow a manual “Refresh environment snapshot” action for debugging
172
+
173
+ ### 3) Carry across `clear_mind`
174
+
175
+ After each `clear_mind` (entering the next “journey stage”), do **not** rely on reminders for this.
176
+
177
+ Instead, inject a small, stable **course prefix** into the model context at the start of each new course (i.e. the first
178
+ few messages in the historical dialog section). This prefix should be a condensed transcript distilled from the original
179
+ Showing-by-Doing run (shell snapshot + FBR highlights + quickstart), so the agent keeps its “felt-sense” of Tellask/FBR
180
+ and environment constraints without re-running the prelude.
181
+
182
+ ---
183
+
184
+ ## UX requirements (WebUI)
185
+
186
+ ### Display
187
+
188
+ - Render Showing-by-Doing as a realistic transcript the user can inspect.
189
+ - Prefer a collapsible “Prelude” section at the top of the dialog, with clear labels:
190
+ - “Teammate Tellask (shell)”
191
+ - “FBR (`!?@self`)”
192
+ - “Environment Quickstart”
193
+
194
+ ### Opt-out
195
+
196
+ The dialog creation UI should include an explicit opt-out, e.g.:
197
+
198
+ - Checkbox: “Skip Showing-by-Doing (do not run environment prelude)”
199
+
200
+ If opted out, no prelude actions are executed and no prelude transcript is generated.
201
+
202
+ ---
203
+
204
+ ## Safety notes
205
+
206
+ - Default to a single, low-risk command (`uname -a`).
207
+ - Do not run anything that modifies the filesystem or network as part of this prelude.
208
+ - Treat the prelude transcript as user-visible by default; avoid including secrets or personally identifying details.
@@ -0,0 +1,177 @@
1
+ # 做给祂看(Showing-by-Doing):把诉请机制变成“体感”
2
+
3
+ 英文版:[English](./showing-by-doing.md)
4
+
5
+ ## 摘要
6
+
7
+ Dominds 的诉请(Tellask,`!?@...`)与扪心自问(FBR,`!?@self`)并不是“写在系统提示里的约定”,而是**后端真实可驱动、可执行、可回传、可持久化**的机制。
8
+
9
+ 问题在于:即使系统提示把诉请机制写得再细,大多数基础模型也并未在“真的可以随时拉一个队友去跑 shell 命令并把结果带回来”的世界里被训练过,因此容易把这些提示当成“说说而已 / 仅供参考”。这会直接削弱诉请与 FBR 的可用性。
10
+
11
+ 因此引入 **“做给祂看”**:在一个对话创建之初,运行一段极短、极拟真的“真实诉请 + 真实 FBR”往返,让智能体产生明确的体感:
12
+
13
+ - 诉请是真的,会被执行
14
+ - 工具输出是真的,会被带回并写入对话记录
15
+ - `!?@self` FBR 也是真的,会作为支线对话产出报告并回传
16
+
17
+ 体感建立后,系统提示可以**大幅简化**:只保留“短而准确”的契约说明即可,把“这在本环境里真实可用”的可信度交给做给祂看来完成。
18
+
19
+ 相关文档:
20
+
21
+ - 诉请机制与子对话:`dominds/docs/dialog-system.md`
22
+ - 术语约定(主线/支线;诉请方/应答方):`dominds/docs/dominds-terminology.md`
23
+ - FBR(`!?@self`):`dominds/docs/fbr.md`
24
+ - 工作语言 vs UI 语言:`dominds/docs/i18n.md`
25
+
26
+ ---
27
+
28
+ ## 目标
29
+
30
+ - 在对话一开始就建立“诉请/回传/持久化都是真的”的信任。
31
+ - 在对话一开始就跑通一次 `!?@self` 的 FBR 回路。
32
+ - 过程足够小、可控、低风险(默认只执行 `uname -a`)。
33
+ - 记录从多角度都高度拟真:后端存档 + 前端可见的对话转录。
34
+
35
+ ## 非目标
36
+
37
+ - 不在创建对话时执行任意用户输入的命令。
38
+ - 不收集敏感信息(默认只采集最小的 OS/Kernel 识别信息)。
39
+ - 不替代系统提示/设计文档:这是“体感补强”,不是完整规范的主载体。
40
+
41
+ ---
42
+
43
+ ## 概念(用户视角)
44
+
45
+ - **主线对话**:用户与主要智能体交互的那条线。
46
+ - **支线对话**:由诉请 / FBR 触发的临时工作线,产出结果回传主线。
47
+ - **诉请(Tellask)**:以 `!?@<memberId> ...` 向队友/对话发起的结构化请求。
48
+ - **Shell 专家(shell specialist)**:被允许执行 shell 命令并回传结果的队友(由 `shell_specialists` 配置指定)。
49
+ - **FBR(扪心自问)**:以 `!?@self` 触发的“无工具支线推理”,产出报告回传主线。
50
+
51
+ ---
52
+
53
+ ## 运行流程(对话创建时)
54
+
55
+ 做给祂看发生在**第一条用户消息被处理之前**(可视为一个 Prelude / 序幕),除非用户在创建对话时明确选择跳过。
56
+
57
+ ### 1)选择执行路径
58
+
59
+ 1. 如果 team 配置里存在 `shell_specialists`,选择一个 shell 专员(建议确定性选择:列表第一个)。
60
+ 2. 否则,由 **Dominds 运行时**直接执行基线命令(不通过智能体工具)。
61
+
62
+ 补充:team 配置校验一般会要求——只要有队友拥有 shell 工具,就必须配置非空的 `shell_specialists`;同时,非 shell 专员不应持有 shell 工具。因此“没有配置 `shell_specialists`”通常意味着“对用户可见的队友都没有 shell 工具可用”。
63
+
64
+ 基线命令(默认):
65
+
66
+ - `uname -a`
67
+
68
+ `uname -a` 失败时(例如非 POSIX 环境),可以改用平台等价命令;但默认肌肉记忆路径保留 `uname -a`,因为它低风险、快、且信息量刚好够用。
69
+
70
+ ### 2)真实队友诉请:让 shell 专员执行 `uname -a`
71
+
72
+ 若存在 shell 专员,则主线智能体(诉请方)以**工作语言**(server-wide,见 `dominds/docs/i18n.md`)向 shell 专员(应答方)发起真实诉请,要求:
73
+
74
+ - 执行 `uname -a`
75
+ - 原样返回输出
76
+ - 若命令不可用,返回错误信息,并给出替代命令建议
77
+
78
+ 结果必须以正常对话转录的方式落盘并展示,至少清晰体现:
79
+
80
+ - 诉请的 headline + body
81
+ - shell 专员实际执行了命令
82
+ - 返回的原始输出
83
+
84
+ 最小配置示例:
85
+
86
+ ```yaml
87
+ # .minds/team.yaml
88
+ shell_specialists:
89
+ - cmdr
90
+ ```
91
+
92
+ ### 3)真实 FBR:基于环境信息做一次“扪心自问”
93
+
94
+ 拿到环境快照(来自 shell 专员或直接运行)后,主线智能体触发一次 `!?@self` FBR,并在 FBR 的诉请正文里带上:
95
+
96
+ - `uname -a` 的完整输出
97
+ - 主线可用工具的假设(例如“主线可能有 `fs`/`os`/`mem`;但 FBR 你没有任何工具”)
98
+ - 问题:在这个环境里要注意什么?优先用哪些命令行工具,为什么?
99
+
100
+ FBR 产出应尽量短而可复用,典型包括:
101
+
102
+ - 环境判断(OS 家族、内核、是否容器/VM 线索)
103
+ - 注意事项(坑点清单)
104
+ - “优先工具/命令” shortlist(例如 `rg`、`sed`、`jq`、`tar`、`ps`、`lsof` 等,视环境而定)
105
+
106
+ ### 4)把结果收敛成“环境速记”
107
+
108
+ 主线智能体在主线对话里追加一条短总结(Environment Quickstart),让关键信息显性化、可检索。
109
+
110
+ ---
111
+
112
+ ## 降级策略
113
+
114
+ - **未配置 `shell_specialists`**:由 Dominds 运行时直接执行 `uname -a`,然后进入 FBR。
115
+ - **shell 专员失败/超时**:记录失败原因,由 Dominds 运行时直接执行 `uname -a`,继续进入 FBR。
116
+ - **`uname` 不存在/执行失败**:跳过命令执行,直接让 FBR 以“环境未知”输出:缺什么信息、为什么关键、后续如何补齐。
117
+
118
+ ---
119
+
120
+ ## 持久化、缓存与复用
121
+
122
+ ### 1)记录必须“像真的一样”
123
+
124
+ 做给祂看的每一步都必须作为标准对话记录落盘(消息 + 事件),并在 WebUI 中可见:
125
+
126
+ - 可跨重启(遵循常规对话持久化)
127
+ - 可 debug(日志/存储都有迹可循)
128
+ - 前端可查看转录(建议 UI 上做一个可折叠 Prelude 区域)
129
+
130
+ ### 2)仅进程内缓存(按智能体维度)
131
+
132
+ 为了避免每次新对话都重复跑 `uname -a` + FBR,后端进程可以维护一个**进程内缓存**(重启失效),按(至少)主线智能体 id 做 key。
133
+
134
+ 缓存内容建议包含:
135
+
136
+ - 完整转录(诉请 + shell 输出 + FBR + 速记)
137
+ - 一份更短的 distilled “环境速记”(用于注入上下文)
138
+ - 时间戳 + 版本号(未来 schema 变化用)
139
+
140
+ 复用策略:
141
+
142
+ - 若命中缓存,新对话可以直接复用首次的转录与速记,不再重复执行。
143
+ - 复用的记录在新对话里仍应可见,并清楚标记“来自缓存复用”。
144
+
145
+ ### 3)跨 `clear_mind` 携带
146
+
147
+ 每次 `clear_mind` 进入下一程时,不要依赖 reminder 来承载这件事。
148
+
149
+ 改为:在每一程对话的上下文里,注入一段小而稳定的 **course prefix**(可理解为“历史消息区最开头的几条 msg”),内容是从首次做给祂看中提炼出的压缩转录(shell 快照 + FBR 摘要 + 环境速记)。这样智能体能持续保留“体感”和环境约束,而无需重复执行 Prelude,也不必每次把整段长转录重新塞进上下文。
150
+
151
+ ---
152
+
153
+ ## WebUI 需求
154
+
155
+ ### 展示
156
+
157
+ - 做给祂看应以真实对话转录展示,用户可展开查看细节。
158
+ - 建议在对话顶部提供一个可折叠 Prelude 区域,并带上清晰标签:
159
+ - “队友诉请(shell)”
160
+ - “FBR(`!?@self`)”
161
+ - “环境速记(Environment Quickstart)”
162
+
163
+ ### 创建对话时可选择跳过(opt-out)
164
+
165
+ 创建对话框上提供明确开关,例如:
166
+
167
+ - Checkbox:“跳过做给祂看(不执行环境序幕)”
168
+
169
+ 若选择跳过,则不执行任何 Prelude 行为,也不生成 Prelude 转录。
170
+
171
+ ---
172
+
173
+ ## 安全注意事项
174
+
175
+ - 默认只执行 `uname -a`(低风险、只读、快)。
176
+ - Prelude 禁止任何会改写文件/网络/系统状态的操作。
177
+ - 由于 Prelude 默认用户可见,严禁输出密钥/敏感信息。
@@ -121,6 +121,12 @@ providers:
121
121
  max: 1
122
122
  description: Fully supported. Nucleus sampling parameter.
123
123
  models:
124
+ MiniMax-M2.5:
125
+ name: MiniMax M2.5
126
+ context_length: 204800
127
+ input_length: 204800
128
+ output_length: 8192
129
+ context_window: '204K'
124
130
  MiniMax-M2.1:
125
131
  name: MiniMax M2.1
126
132
  context_length: 204800
@@ -159,6 +165,12 @@ providers:
159
165
  max: 1
160
166
  description: Fully supported. Nucleus sampling parameter.
161
167
  models:
168
+ MiniMax-M2.5:
169
+ name: MiniMax M2.5
170
+ context_length: 204800
171
+ input_length: 204800
172
+ output_length: 8192
173
+ context_window: '204K'
162
174
  MiniMax-M2.1:
163
175
  name: MiniMax M2.1
164
176
  context_length: 204800
@@ -197,6 +209,12 @@ providers:
197
209
  max: 1
198
210
  description: Fully supported. Nucleus sampling parameter.
199
211
  models:
212
+ MiniMax-M2.5:
213
+ name: MiniMax M2.5
214
+ context_length: 204800
215
+ input_length: 204800
216
+ output_length: 8192
217
+ context_window: '204K'
200
218
  MiniMax-M2.1:
201
219
  name: MiniMax M2.1
202
220
  context_length: 204800
@@ -235,6 +253,12 @@ providers:
235
253
  max: 1
236
254
  description: Fully supported. Nucleus sampling parameter.
237
255
  models:
256
+ MiniMax-M2.5:
257
+ name: MiniMax M2.5
258
+ context_length: 204800
259
+ input_length: 204800
260
+ output_length: 8192
261
+ context_window: '204K'
238
262
  MiniMax-M2.1:
239
263
  name: MiniMax M2.1
240
264
  context_length: 204800
@@ -58,6 +58,7 @@ function resolveUpNextPrompt(dlg) {
58
58
  msgId: upNext.msgId,
59
59
  grammar: upNext.grammar ?? 'markdown',
60
60
  userLanguageCode: upNext.userLanguageCode,
61
+ q4hAnswerCallIds: upNext.q4hAnswerCallIds,
61
62
  };
62
63
  }
63
64
  function isUserOriginPrompt(prompt) {
@@ -66,6 +67,20 @@ function isUserOriginPrompt(prompt) {
66
67
  }
67
68
  return (prompt.origin ?? 'user') === 'user';
68
69
  }
70
+ function normalizeQ4HAnswerCallIds(raw) {
71
+ if (!raw || raw.length === 0)
72
+ return undefined;
73
+ const seen = new Set();
74
+ const normalized = [];
75
+ for (const value of raw) {
76
+ const callId = value.trim();
77
+ if (callId === '' || seen.has(callId))
78
+ continue;
79
+ seen.add(callId);
80
+ normalized.push(callId);
81
+ }
82
+ return normalized.length > 0 ? normalized : undefined;
83
+ }
69
84
  async function emitUserMarkdown(dlg, content) {
70
85
  if (!content.trim()) {
71
86
  return;
@@ -860,6 +875,7 @@ async function driveDialogStreamCoreV2(dlg, humanPrompt, driveOptions, callbacks
860
875
  const msgId = currentPrompt.msgId;
861
876
  const promptGrammar = currentPrompt.grammar;
862
877
  const persistedUserLanguageCode = currentPrompt.userLanguageCode ?? dlg.getLastUserLanguageCode();
878
+ const q4hAnswerCallIds = normalizeQ4HAnswerCallIds(currentPrompt.q4hAnswerCallIds);
863
879
  const requestedPersistMode = currentPrompt.persistMode ?? 'persist';
864
880
  const persistMode = isDiligencePrompt ? 'persist' : requestedPersistMode;
865
881
  if (persistMode === 'internal') {
@@ -881,7 +897,7 @@ async function driveDialogStreamCoreV2(dlg, humanPrompt, driveOptions, callbacks
881
897
  msgId,
882
898
  grammar: promptGrammar,
883
899
  });
884
- await dlg.persistUserMessage(promptContent, msgId, promptGrammar, persistedUserLanguageCode);
900
+ await dlg.persistUserMessage(promptContent, msgId, promptGrammar, persistedUserLanguageCode, q4hAnswerCallIds);
885
901
  if (currentPrompt.subdialogReplyTarget) {
886
902
  const normalizedCallId = currentPrompt.subdialogReplyTarget.callId.trim();
887
903
  if (normalizedCallId === '') {
@@ -918,6 +934,7 @@ async function driveDialogStreamCoreV2(dlg, humanPrompt, driveOptions, callbacks
918
934
  content: promptContent,
919
935
  grammar: promptGrammar,
920
936
  userLanguageCode: persistedUserLanguageCode,
937
+ q4hAnswerCallIds,
921
938
  });
922
939
  }
923
940
  catch (err) {
@@ -61,6 +61,7 @@ function resolveEffectivePrompt(dialog, humanPrompt) {
61
61
  userLanguageCode: upNext.userLanguageCode === 'zh' || upNext.userLanguageCode === 'en'
62
62
  ? upNext.userLanguageCode
63
63
  : undefined,
64
+ q4hAnswerCallIds: upNext.q4hAnswerCallIds,
64
65
  };
65
66
  }
66
67
  async function executeDriveRound(args) {