@ynhcj/xiaoyi-channel 1.1.24 → 1.1.26

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 (141) hide show
  1. package/dist/index.d.ts +4 -5
  2. package/dist/index.js +102 -84
  3. package/dist/provider-discovery.d.ts +2 -0
  4. package/dist/provider-discovery.js +4 -0
  5. package/dist/src/bot.d.ts +13 -0
  6. package/dist/src/bot.js +345 -136
  7. package/dist/src/channel.js +2 -17
  8. package/dist/src/client.d.ts +1 -5
  9. package/dist/src/client.js +32 -37
  10. package/dist/src/cspl/call-api.d.ts +6 -0
  11. package/dist/src/cspl/call-api.js +37 -16
  12. package/dist/src/cspl/config.d.ts +11 -1
  13. package/dist/src/cspl/config.js +30 -0
  14. package/dist/src/cspl/middleware.d.ts +8 -0
  15. package/dist/src/cspl/middleware.js +90 -0
  16. package/dist/src/cspl/steer-context.d.ts +21 -0
  17. package/dist/src/cspl/steer-context.js +78 -0
  18. package/dist/src/file-download.js +4 -3
  19. package/dist/src/file-upload.js +19 -18
  20. package/dist/src/formatter.js +32 -44
  21. package/dist/src/heartbeat.js +4 -3
  22. package/dist/src/login-token-handler.js +13 -10
  23. package/dist/src/message-queue.js +2 -1
  24. package/dist/src/monitor.js +62 -41
  25. package/dist/src/outbound.js +22 -18
  26. package/dist/src/provider.d.ts +1 -1
  27. package/dist/src/provider.js +147 -71
  28. package/dist/src/push.js +16 -15
  29. package/dist/src/reply-dispatcher.d.ts +3 -1
  30. package/dist/src/reply-dispatcher.js +65 -63
  31. package/dist/src/runtime.d.ts +3 -11
  32. package/dist/src/runtime.js +6 -18
  33. package/dist/src/self-evolution-handler.js +11 -14
  34. package/dist/src/self-evolution-tool-result-nudge.d.ts +3 -0
  35. package/dist/src/self-evolution-tool-result-nudge.js +96 -0
  36. package/dist/src/skill-retriever/hooks.js +7 -16
  37. package/dist/src/skill-retriever/tool-search.js +16 -17
  38. package/dist/src/steer-injector.js +1 -1
  39. package/dist/src/task-manager.d.ts +4 -27
  40. package/dist/src/task-manager.js +19 -79
  41. package/dist/src/tools/calendar-tool.d.ts +2 -1
  42. package/dist/src/tools/calendar-tool.js +116 -116
  43. package/dist/src/tools/call-device-tool.d.ts +2 -1
  44. package/dist/src/tools/call-device-tool.js +126 -103
  45. package/dist/src/tools/call-phone-tool.d.ts +2 -1
  46. package/dist/src/tools/call-phone-tool.js +113 -113
  47. package/dist/src/tools/create-alarm-tool.d.ts +2 -1
  48. package/dist/src/tools/create-alarm-tool.js +231 -231
  49. package/dist/src/tools/create-all-tools.d.ts +16 -0
  50. package/dist/src/tools/create-all-tools.js +50 -0
  51. package/dist/src/tools/delete-alarm-tool.d.ts +2 -1
  52. package/dist/src/tools/delete-alarm-tool.js +135 -135
  53. package/dist/src/tools/get-alarm-tool-schema.d.ts +2 -1
  54. package/dist/src/tools/get-alarm-tool-schema.js +16 -10
  55. package/dist/src/tools/get-calendar-tool-schema.d.ts +2 -1
  56. package/dist/src/tools/get-calendar-tool-schema.js +12 -8
  57. package/dist/src/tools/get-collection-tool-schema.d.ts +2 -1
  58. package/dist/src/tools/get-collection-tool-schema.js +11 -9
  59. package/dist/src/tools/get-contact-tool-schema.d.ts +2 -1
  60. package/dist/src/tools/get-contact-tool-schema.js +16 -10
  61. package/dist/src/tools/get-device-file-tool-schema.d.ts +2 -1
  62. package/dist/src/tools/get-device-file-tool-schema.js +13 -9
  63. package/dist/src/tools/get-email-tool-schema.d.ts +2 -1
  64. package/dist/src/tools/get-email-tool-schema.js +11 -8
  65. package/dist/src/tools/get-note-tool-schema.d.ts +2 -1
  66. package/dist/src/tools/get-note-tool-schema.js +14 -9
  67. package/dist/src/tools/get-photo-tool-schema.d.ts +2 -1
  68. package/dist/src/tools/get-photo-tool-schema.js +12 -9
  69. package/dist/src/tools/image-reading-tool.d.ts +3 -2
  70. package/dist/src/tools/image-reading-tool.js +86 -165
  71. package/dist/src/tools/location-tool.d.ts +2 -1
  72. package/dist/src/tools/location-tool.js +91 -91
  73. package/dist/src/tools/login-token-tool.d.ts +2 -1
  74. package/dist/src/tools/login-token-tool.js +124 -116
  75. package/dist/src/tools/modify-alarm-tool.d.ts +2 -1
  76. package/dist/src/tools/modify-alarm-tool.js +236 -236
  77. package/dist/src/tools/modify-note-tool.d.ts +2 -1
  78. package/dist/src/tools/modify-note-tool.js +108 -108
  79. package/dist/src/tools/note-tool.d.ts +2 -1
  80. package/dist/src/tools/note-tool.js +107 -107
  81. package/dist/src/tools/query-app-message-tool.d.ts +2 -1
  82. package/dist/src/tools/query-app-message-tool.js +112 -111
  83. package/dist/src/tools/query-memory-data-tool.d.ts +2 -1
  84. package/dist/src/tools/query-memory-data-tool.js +113 -112
  85. package/dist/src/tools/query-todo-task-tool.d.ts +2 -1
  86. package/dist/src/tools/query-todo-task-tool.js +107 -106
  87. package/dist/src/tools/save-file-to-phone-tool.d.ts +2 -1
  88. package/dist/src/tools/save-file-to-phone-tool.js +131 -131
  89. package/dist/src/tools/save-media-to-gallery-tool.d.ts +2 -1
  90. package/dist/src/tools/save-media-to-gallery-tool.js +138 -138
  91. package/dist/src/tools/save-self-evolution-skill-tool.d.ts +2 -1
  92. package/dist/src/tools/save-self-evolution-skill-tool.js +194 -196
  93. package/dist/src/tools/search-alarm-tool.d.ts +2 -1
  94. package/dist/src/tools/search-alarm-tool.js +175 -175
  95. package/dist/src/tools/search-calendar-tool.d.ts +2 -1
  96. package/dist/src/tools/search-calendar-tool.js +149 -149
  97. package/dist/src/tools/search-contact-tool.d.ts +2 -1
  98. package/dist/src/tools/search-contact-tool.js +102 -102
  99. package/dist/src/tools/search-email-tool.d.ts +2 -1
  100. package/dist/src/tools/search-email-tool.js +111 -111
  101. package/dist/src/tools/search-file-tool.d.ts +2 -1
  102. package/dist/src/tools/search-file-tool.js +103 -103
  103. package/dist/src/tools/search-message-tool.d.ts +2 -1
  104. package/dist/src/tools/search-message-tool.js +104 -104
  105. package/dist/src/tools/search-note-tool.d.ts +2 -1
  106. package/dist/src/tools/search-note-tool.js +99 -99
  107. package/dist/src/tools/search-photo-gallery-tool.d.ts +2 -1
  108. package/dist/src/tools/search-photo-gallery-tool.js +38 -38
  109. package/dist/src/tools/send-email-tool.d.ts +2 -1
  110. package/dist/src/tools/send-email-tool.js +109 -108
  111. package/dist/src/tools/send-file-to-user-tool.d.ts +2 -1
  112. package/dist/src/tools/send-file-to-user-tool.js +157 -155
  113. package/dist/src/tools/send-message-tool.d.ts +2 -1
  114. package/dist/src/tools/send-message-tool.js +123 -123
  115. package/dist/src/tools/session-helper.d.ts +24 -0
  116. package/dist/src/tools/session-helper.js +45 -0
  117. package/dist/src/tools/session-manager.d.ts +29 -6
  118. package/dist/src/tools/session-manager.js +134 -19
  119. package/dist/src/tools/upload-file-tool.d.ts +2 -1
  120. package/dist/src/tools/upload-file-tool.js +82 -82
  121. package/dist/src/tools/upload-photo-tool.d.ts +2 -1
  122. package/dist/src/tools/upload-photo-tool.js +73 -73
  123. package/dist/src/tools/xiaoyi-add-collection-tool.d.ts +2 -1
  124. package/dist/src/tools/xiaoyi-add-collection-tool.js +147 -147
  125. package/dist/src/tools/xiaoyi-collection-tool.d.ts +2 -1
  126. package/dist/src/tools/xiaoyi-collection-tool.js +115 -115
  127. package/dist/src/tools/xiaoyi-delete-collection-tool.d.ts +2 -1
  128. package/dist/src/tools/xiaoyi-delete-collection-tool.js +128 -128
  129. package/dist/src/tools/xiaoyi-gui-tool.d.ts +2 -1
  130. package/dist/src/tools/xiaoyi-gui-tool.js +89 -88
  131. package/dist/src/trigger-handler.js +8 -9
  132. package/dist/src/utils/logger.js +105 -19
  133. package/dist/src/utils/self-evolution-manager.d.ts +5 -0
  134. package/dist/src/utils/self-evolution-manager.js +45 -23
  135. package/dist/src/utils/throw.d.ts +5 -0
  136. package/dist/src/utils/throw.js +10 -0
  137. package/dist/src/websocket.js +35 -31
  138. package/dist/src/xy-session-store.d.ts +79 -0
  139. package/dist/src/xy-session-store.js +153 -0
  140. package/openclaw.plugin.json +25 -0
  141. package/package.json +7 -6
package/dist/src/bot.js CHANGED
@@ -1,5 +1,4 @@
1
1
  import { getXYRuntime } from "./runtime.js";
2
- import { setCachedContext } from "./steer-injector.js";
3
2
  import { createXYReplyDispatcher } from "./reply-dispatcher.js";
4
3
  import { parseA2AMessage, extractTextFromParts, extractFileParts, extractPushId, extractDeviceType, extractTriggerData } from "./parser.js";
5
4
  import { downloadFilesFromParts } from "./file-download.js";
@@ -13,7 +12,9 @@ import { getPushDataById } from "./utils/pushdata-manager.js";
13
12
  import { selfEvolutionManager } from "./utils/self-evolution-manager.js";
14
13
  import { saveRuntimeInfo } from "./utils/runtime-manager.js";
15
14
  import { toolCallNudgeManager } from "./utils/tool-call-nudge-manager.js";
16
- import { registerTaskId, decrementTaskIdRef, lockTaskId, unlockTaskId, hasActiveTask, } from "./task-manager.js";
15
+ import { setCsplSteerContext } from "./cspl/steer-context.js";
16
+ import { registerTaskId, decrementTaskIdRef, hasActiveTask, } from "./task-manager.js";
17
+ import { logger } from "./utils/logger.js";
17
18
  /**
18
19
  * Handle an incoming A2A message.
19
20
  * This is the main entry point for message processing.
@@ -21,10 +22,8 @@ import { registerTaskId, decrementTaskIdRef, lockTaskId, unlockTaskId, hasActive
21
22
  */
22
23
  export async function handleXYMessage(params) {
23
24
  const { cfg, runtime, message, accountId, webSocketSessionId } = params;
24
- const log = runtime?.log ?? console.log;
25
- const error = runtime?.error ?? console.error;
26
- // 每次收到消息时更新缓存,供 steer 注入使用
27
- setCachedContext(cfg, runtime, accountId);
25
+ // Cache context for CSPL steer injection (after_tool_call hook)
26
+ setCsplSteerContext(cfg, runtime);
28
27
  // Get runtime (already validated in monitor.ts, but get reference for use)
29
28
  const core = getXYRuntime();
30
29
  try {
@@ -36,7 +35,7 @@ export async function handleXYMessage(params) {
36
35
  if (!sessionId) {
37
36
  throw new Error("clearContext request missing sessionId in params");
38
37
  }
39
- log(`Clear context request for session ${sessionId}`);
38
+ logger.log(`Clear context request for session ${sessionId}`);
40
39
  const config = resolveXYConfig(cfg);
41
40
  await sendClearContextResponse({
42
41
  config,
@@ -52,7 +51,7 @@ export async function handleXYMessage(params) {
52
51
  if (!sessionId) {
53
52
  throw new Error("tasks/cancel request missing sessionId in params");
54
53
  }
55
- log(`Tasks cancel request for session ${sessionId}, task ${taskId}`);
54
+ logger.log(`Tasks cancel request for session ${sessionId}, task ${taskId}`);
56
55
  const config = resolveXYConfig(cfg);
57
56
  await sendTasksCancelResponse({
58
57
  config,
@@ -68,18 +67,18 @@ export async function handleXYMessage(params) {
68
67
  // 如果消息中包含 Trigger 事件数据,直接返回 pushData 内容,不走正常流程
69
68
  const triggerData = extractTriggerData(parsed.parts);
70
69
  if (triggerData) {
71
- log(`[BOT] 📌 Detected Trigger message with pushDataId: ${triggerData.pushDataId}`);
72
- log(`[BOT] - Session ID: ${parsed.sessionId}`);
73
- log(`[BOT] - Task ID: ${parsed.taskId}`);
70
+ logger.log(`[BOT] 📌 Detected Trigger message with pushDataId: ${triggerData.pushDataId}`);
71
+ logger.log(`[BOT] - Session ID: ${parsed.sessionId}`);
72
+ logger.log(`[BOT] - Task ID: ${parsed.taskId}`);
74
73
  try {
75
74
  // 读取 pushData
76
75
  const pushDataItem = await getPushDataById(triggerData.pushDataId);
77
76
  if (!pushDataItem) {
78
- error(`[BOT] ❌ pushData not found for ID: ${triggerData.pushDataId}`);
77
+ logger.error(`[BOT] ❌ pushData not found for ID: ${triggerData.pushDataId}`);
79
78
  return;
80
79
  }
81
- log(`[BOT] ✅ Found pushData, sending direct response`);
82
- log(`[BOT] - pushDataId: ${pushDataItem.pushDataId}`);
80
+ logger.log(`[BOT] ✅ Found pushData, sending direct response`);
81
+ logger.log(`[BOT] - pushDataId: ${pushDataItem.pushDataId}`);
83
82
  const config = resolveXYConfig(cfg);
84
83
  // 直接发送响应(final=true,不走 openclaw 流程)
85
84
  await sendA2AResponse({
@@ -91,56 +90,52 @@ export async function handleXYMessage(params) {
91
90
  append: false,
92
91
  final: true,
93
92
  });
94
- log(`[BOT] ✅ Trigger response sent successfully, exiting early`);
93
+ logger.log(`[BOT] ✅ Trigger response sent successfully, exiting early`);
95
94
  return; // 提前返回,不继续处理
96
95
  }
97
96
  catch (err) {
98
- error(`[BOT] ❌ Failed to handle Trigger message:`, err);
97
+ logger.error(`[BOT] ❌ Failed to handle Trigger message:`, err);
99
98
  return;
100
99
  }
101
100
  }
102
101
  // ========================================
103
- // 🔑 检测steer模式和是否是第二条消息
104
- const isSteerMode = cfg.messages?.queue?.mode === "steer";
105
- const isSecondMessage = isSteerMode && hasActiveTask(parsed.sessionId);
106
- if (isSecondMessage) {
107
- log(`[BOT] 🔄 STEER MODE - Second message detected (will be follower)`);
108
- log(`[BOT] - Session: ${parsed.sessionId}`);
109
- log(`[BOT] - New taskId: ${parsed.taskId} (will replace current)`);
102
+ // 🔑 注册taskId(检测是否是已有活跃任务的 session)
103
+ const isUpdate = hasActiveTask(parsed.sessionId);
104
+ const skipReg = params.skipRegistration === true;
105
+ if (isUpdate) {
106
+ logger.log(`[BOT] 🔄 STEER MODE - Second message detected (core will handle steer)`);
107
+ logger.log(`[BOT] - Session: ${parsed.sessionId}`);
108
+ logger.log(`[BOT] - New taskId: ${parsed.taskId}`);
110
109
  }
111
- // 🔑 注册taskId(第二条消息会覆盖第一条的taskId
112
- const { isUpdate, refCount } = registerTaskId(parsed.sessionId, parsed.taskId, parsed.messageId, { incrementRef: true } // 增加引用计数
113
- );
114
- // 🔑 如果是第一条消息,锁定taskId防止被过早清理
115
- if (!isUpdate) {
116
- lockTaskId(parsed.sessionId);
117
- log(`[BOT] 🔒 Locked taskId for first message`);
118
- }
119
- // Extract and update push_id if present
120
- const pushId = extractPushId(parsed.parts);
121
- if (pushId) {
122
- log(`[BOT] 📌 Extracted push_id from user message`);
123
- configManager.updatePushId(parsed.sessionId, pushId);
124
- // 持久化 pushId 到本地文件(异步,不阻塞主流程)
125
- addPushId(pushId).catch((err) => {
126
- error(`[BOT] Failed to persist pushId:`, err);
110
+ // Steer injections skip taskId registration to avoid overwriting the active taskId
111
+ if (!skipReg) {
112
+ registerTaskId(parsed.sessionId, parsed.taskId, parsed.messageId);
113
+ // Extract and update push_id if present
114
+ const pushId = extractPushId(parsed.parts);
115
+ if (pushId) {
116
+ logger.log(`[BOT] 📌 Extracted push_id from user message`);
117
+ configManager.updatePushId(parsed.sessionId, pushId);
118
+ // 持久化 pushId 到本地文件(异步,不阻塞主流程)
119
+ addPushId(pushId).catch((err) => {
120
+ logger.error(`[BOT] Failed to persist pushId:`, err);
121
+ });
122
+ }
123
+ else {
124
+ logger.log(`[BOT] ℹ️ No push_id found in message, will use config default`);
125
+ }
126
+ // 保存 runtime 信息到 .xiaoyiruntime 文件(异步,不阻塞主流程)
127
+ saveRuntimeInfo(webSocketSessionId || parsed.sessionId, // SESSION_ID (WebSocket 层级,如果没有则 fallback)
128
+ parsed.sessionId, // CONVERSATION_ID (param 里的 sessionId)
129
+ parsed.taskId // TASK_ID (param.id)
130
+ ).catch((err) => {
131
+ logger.error(`[BOT] Failed to save runtime info:`, err);
127
132
  });
128
133
  }
129
- else {
130
- log(`[BOT] ℹ️ No push_id found in message, will use config default`);
131
- }
132
- // Extract deviceType if present (same level as push_id in systemVariables)
134
+ // Extract deviceType if present (always parse — used in ctxPayload.MessageSid)
133
135
  const deviceType = extractDeviceType(parsed.parts);
134
136
  if (deviceType) {
135
- log(`[BOT] 📱 Extracted deviceType from user message: ${deviceType}`);
137
+ logger.log(`[BOT] 📱 Extracted deviceType from user message: ${deviceType}`);
136
138
  }
137
- // 保存 runtime 信息到 .xiaoyiruntime 文件(异步,不阻塞主流程)
138
- saveRuntimeInfo(webSocketSessionId || parsed.sessionId, // SESSION_ID (WebSocket 层级,如果没有则 fallback)
139
- parsed.sessionId, // CONVERSATION_ID (param 里的 sessionId)
140
- parsed.taskId // TASK_ID (param.id)
141
- ).catch((err) => {
142
- error(`[BOT] Failed to save runtime info:`, err);
143
- });
144
139
  // Resolve configuration (needed for status updates)
145
140
  const config = resolveXYConfig(cfg);
146
141
  // ✅ Resolve agent route (following feishu pattern)
@@ -155,53 +150,95 @@ export async function handleXYMessage(params) {
155
150
  id: parsed.sessionId, // ✅ Use sessionId to share context within the same conversation session
156
151
  },
157
152
  });
158
- log(`xy: resolved route accountId=${route.accountId}, sessionKey=${route.sessionKey}`);
159
- registerSession(route.sessionKey, {
160
- config,
161
- sessionId: parsed.sessionId,
162
- taskId: parsed.taskId,
163
- messageId: parsed.messageId,
164
- agentId: route.accountId,
165
- });
166
- // 🔑 发送初始状态更新(第二条消息也要发,用新taskId)
167
- log(`[STATUS] Sending initial status update for session ${parsed.sessionId}`);
168
- void sendStatusUpdate({
169
- config,
170
- sessionId: parsed.sessionId,
171
- taskId: parsed.taskId,
172
- messageId: parsed.messageId,
173
- text: isSecondMessage ? "新消息已接收,正在处理..." : "任务正在处理中,请稍候~",
174
- state: "working",
175
- }).catch((err) => {
176
- error(`Failed to send initial status update:`, err);
177
- });
153
+ logger.log(`xy: resolved route accountId=${route.accountId}, sessionKey=${route.sessionKey}`);
154
+ // Steer injections skip session registration to avoid refCount leaks
155
+ if (!skipReg) {
156
+ registerSession(route.sessionKey, {
157
+ config,
158
+ sessionId: parsed.sessionId,
159
+ taskId: parsed.taskId,
160
+ messageId: parsed.messageId,
161
+ agentId: route.accountId,
162
+ deviceType,
163
+ });
164
+ // 🔑 发送初始状态更新
165
+ logger.log(`[STATUS] Sending initial status update for session ${parsed.sessionId}`);
166
+ void sendStatusUpdate({
167
+ config,
168
+ sessionId: parsed.sessionId,
169
+ taskId: parsed.taskId,
170
+ messageId: parsed.messageId,
171
+ text: "任务正在处理中,请稍候~",
172
+ state: "working",
173
+ }).catch((err) => {
174
+ logger.error(`Failed to send initial status update:`, err);
175
+ });
176
+ }
178
177
  // Extract text and files from parts
179
178
  const text = extractTextFromParts(parsed.parts);
180
179
  let textForAgent = text || "";
181
- if (route.sessionKey && textForAgent) {
180
+ // Self-evolution keyword nudge — only for real user messages, not steer injections
181
+ if (!skipReg && route.sessionKey && textForAgent) {
182
182
  try {
183
183
  const selfEvolutionEnabled = await selfEvolutionManager.isEnabled();
184
184
  if (selfEvolutionEnabled && shouldNudgeForSelfEvolutionKeyword(textForAgent)) {
185
185
  const shouldNudge = toolCallNudgeManager.tryMarkKeywordNudge(route.sessionKey);
186
- log(`[SELF_EVOLUTION] Keyword check hit during inbound build: sessionKey=${route.sessionKey}, shouldNudge=${shouldNudge}`);
186
+ logger.log(`[SELF_EVOLUTION] Keyword check hit during inbound build: sessionKey=${route.sessionKey}, shouldNudge=${shouldNudge}`);
187
187
  if (shouldNudge) {
188
188
  const augmented = appendSelfEvolutionKeywordNudge(textForAgent);
189
189
  textForAgent = augmented.text;
190
190
  if (augmented.appended) {
191
- log(`[SELF_EVOLUTION] Keyword-triggered inline nudge appended: sessionKey=${route.sessionKey}`);
191
+ logger.log(`[SELF_EVOLUTION] Keyword-triggered inline nudge appended: sessionKey=${route.sessionKey}`);
192
192
  }
193
193
  }
194
194
  }
195
195
  }
196
196
  catch (selfEvolutionError) {
197
- error(`[SELF_EVOLUTION] Failed to append inline keyword nudge: ${String(selfEvolutionError)}`);
197
+ logger.error(`[SELF_EVOLUTION] Failed to append inline keyword nudge: ${String(selfEvolutionError)}`);
198
198
  }
199
199
  }
200
- const fileParts = extractFileParts(parsed.parts);
201
- // Download files to local disk
202
- const downloadedFiles = await downloadFilesFromParts(fileParts);
203
- console.log("Downloaded files:", JSON.stringify(downloadedFiles, null, 2));
204
- const mediaPayload = buildXYMediaPayload(downloadedFiles);
200
+ // 🔑 Steer消息: 跳过旧路径直接进入 streaming-signal 队列
201
+ // /steer 前缀由 dispatchSteerWhenReady 内部添加
202
+ if (isUpdate) {
203
+ // 立即释放 init gate——steer 不走 withReplyDispatcher 的 run()
204
+ // 回调,onInitComplete 永远不会被触发。如果不释放,后续消息
205
+ // 会被 globalDispatchInitGate 永久阻塞。
206
+ params.onInitComplete?.();
207
+ // Steer 也支持文件 —— 提取并下载,附带到 mediaPayload
208
+ const steerFileParts = extractFileParts(parsed.parts);
209
+ const steerDownloadedFiles = await downloadFilesFromParts(steerFileParts);
210
+ const steerMediaPayload = buildXYMediaPayload(steerDownloadedFiles);
211
+ if (steerFileParts.length > 0) {
212
+ logger.log(`[BOT] 📎 Steer message with files: ${steerFileParts.length} file(s)`);
213
+ }
214
+ logger.log(`[BOT] 🔄 Steer message — enqueuing to streaming-signal queue`);
215
+ await enqueueSteer({
216
+ sessionId: parsed.sessionId,
217
+ sessionKey: route.sessionKey,
218
+ steerText: textForAgent, // 原始文本,不带 /steer 前缀
219
+ mediaPayload: steerMediaPayload,
220
+ cfg,
221
+ runtime,
222
+ parsed,
223
+ route,
224
+ deviceType,
225
+ });
226
+ logger.log(`[BOT] ✅ Steer queue completed for session: ${parsed.sessionId}`);
227
+ logger.log(`xy: dispatch complete (session=${parsed.sessionId})`);
228
+ return;
229
+ }
230
+ // ── First message (non-steer) path below ──────────────────────
231
+ // 🔑 立即创建 streaming 信号——必须在文件下载等耗时操作之前,
232
+ // 否则 steer 消息的 dispatchSteerWhenReady 会找不到信号而跳过等待。
233
+ createStreamingSignal(parsed.sessionId);
234
+ // File download — only for real user messages, steer injections have no files
235
+ let mediaPayload = {};
236
+ if (!skipReg) {
237
+ const fileParts = extractFileParts(parsed.parts);
238
+ const downloadedFiles = await downloadFilesFromParts(fileParts);
239
+ logger.log("Downloaded files:", JSON.stringify(downloadedFiles, null, 2));
240
+ mediaPayload = buildXYMediaPayload(downloadedFiles);
241
+ }
205
242
  // Resolve envelope format options (following feishu pattern)
206
243
  const envelopeOptions = core.channel.reply.resolveEnvelopeFormatOptions(cfg);
207
244
  // Build message body with speaker prefix (following feishu pattern)
@@ -233,7 +270,7 @@ export async function handleXYMessage(params) {
233
270
  SenderId: parsed.sessionId,
234
271
  Provider: "xiaoyi-channel",
235
272
  Surface: "xiaoyi-channel",
236
- MessageSid: parsed.messageId,
273
+ MessageSid: `xiaoyi_${parsed.taskId}_${deviceType}`,
237
274
  Timestamp: Date.now(),
238
275
  WasMentioned: false,
239
276
  CommandAuthorized: true,
@@ -242,9 +279,11 @@ export async function handleXYMessage(params) {
242
279
  ReplyToBody: undefined, // A2A protocol doesn't support reply/quote
243
280
  ...mediaPayload,
244
281
  });
245
- // 🔑 创建dispatcher(dispatcher会自动使用动态taskId)
246
- log(`[BOT-DISPATCHER] 🎯 Creating reply dispatcher`);
247
- log(`[BOT-DISPATCHER] - taskId: ${parsed.taskId}`);
282
+ // 🔑 Streaming 信号已在上方创建(在文件下载之前)
283
+ const steerState = { steered: false };
284
+ // 🔑 创建dispatcher
285
+ logger.log(`[BOT-DISPATCHER] 🎯 Creating reply dispatcher`);
286
+ logger.log(`[BOT-DISPATCHER] - taskId: ${parsed.taskId}`);
248
287
  const { dispatcher, replyOptions, markDispatchIdle, startStatusInterval } = createXYReplyDispatcher({
249
288
  cfg,
250
289
  runtime,
@@ -252,13 +291,11 @@ export async function handleXYMessage(params) {
252
291
  taskId: parsed.taskId,
253
292
  messageId: parsed.messageId,
254
293
  accountId: route.accountId,
255
- isSteerFollower: isSecondMessage, // 🔑 标记第二条消息
294
+ steerState,
256
295
  });
257
- // 🔑 只有第一条消息启动状态定时器
258
- // 第二条消息会很快返回,不需要定时器
259
- if (!isSecondMessage) {
296
+ // Steer injections don't need status intervals
297
+ if (!skipReg) {
260
298
  startStatusInterval();
261
- log(`[BOT-DISPATCHER] ✅ Status interval started for first message`);
262
299
  }
263
300
  // Build session context for AsyncLocalStorage
264
301
  const sessionContext = {
@@ -269,69 +306,75 @@ export async function handleXYMessage(params) {
269
306
  agentId: route.accountId,
270
307
  deviceType,
271
308
  };
272
- log(`[BOT-DISPATCH] ⏳ withReplyDispatcher starting, sessionKey=${route.sessionKey}`);
309
+ logger.log(`[BOT-DISPATCH] ⏳ withReplyDispatcher starting, sessionKey=${route.sessionKey}`);
273
310
  await core.channel.reply.withReplyDispatcher({
274
311
  dispatcher,
275
312
  onSettled: () => {
276
- log(`[BOT] 🏁 onSettled called for session: ${route.sessionKey}`);
277
- log(`[BOT] - isSecondMessage: ${isSecondMessage}`);
278
- // 🔑 减少引用计数
279
- decrementTaskIdRef(parsed.sessionId);
280
- // 🔑 如果是第一条消息完成,解锁
281
- if (!isSecondMessage) {
282
- unlockTaskId(parsed.sessionId);
283
- log(`[BOT] 🔓 Unlocked taskId (first message completed)`);
313
+ logger.log(`[BOT] 🏁 onSettled called for session: ${route.sessionKey}`);
314
+ logger.log(`[BOT] - steered: ${steerState.steered}`);
315
+ // 🔑 When steered, skip heavy cleanup — the first message's dispatcher is still running
316
+ if (steerState.steered) {
317
+ logger.log(`[BOT] Steered dispatch settled (skipping cleanup)`);
318
+ return;
284
319
  }
285
- // 减少session引用计数
320
+ streamingSignals.delete(parsed.sessionId);
321
+ decrementTaskIdRef(parsed.sessionId);
286
322
  unregisterSession(route.sessionKey);
287
- log(`[BOT] ✅ Cleanup completed`);
323
+ logger.log(`[BOT] ✅ Cleanup completed`);
324
+ },
325
+ run: () => {
326
+ // 🔐 Use AsyncLocalStorage to provide session context to tools.
327
+ // runWithSessionContext returns after the sync part of dispatch
328
+ // (including agentTools + wrapStreamFn) has executed, so we
329
+ // signal init complete to release the global dispatch gate
330
+ // for the next session.
331
+ const dispatchPromise = runWithSessionContext(sessionContext, async () => {
332
+ logger.log(`[BOT-DISPATCH] ⏳ dispatchReplyFromConfig starting...`);
333
+ logger.log(`[BOT-DISPATCH] - sessionKey: ${ctxPayload.SessionKey}`);
334
+ logger.log(`[BOT-DISPATCH] - provider: ${ctxPayload.Provider}`);
335
+ logger.log(`[BOT-DISPATCH] - surface: ${ctxPayload.Surface}`);
336
+ logger.log(`[BOT-DISPATCH] - from: ${ctxPayload.From}`);
337
+ logger.log(`[BOT-DISPATCH] - body length: ${ctxPayload.Body?.length ?? 0}`);
338
+ try {
339
+ const result = await core.channel.reply.dispatchReplyFromConfig({
340
+ ctx: ctxPayload,
341
+ cfg,
342
+ dispatcher,
343
+ replyOptions,
344
+ });
345
+ logger.log(`[BOT-DISPATCH] ✅ dispatchReplyFromConfig returned`);
346
+ logger.log(`[BOT-DISPATCH] - result: ${JSON.stringify(result)}`);
347
+ return result;
348
+ }
349
+ catch (dispatchErr) {
350
+ logger.error(`[BOT-DISPATCH] ❌ dispatchReplyFromConfig threw`);
351
+ logger.error(`[BOT-DISPATCH] - error name: ${dispatchErr instanceof Error ? dispatchErr.name : "unknown"}`);
352
+ logger.error(`[BOT-DISPATCH] - error message: ${String(dispatchErr)}`);
353
+ logger.error(`[BOT-DISPATCH] - error stack: ${dispatchErr instanceof Error ? dispatchErr.stack?.slice(0, 500) : "N/A"}`);
354
+ throw dispatchErr;
355
+ }
356
+ });
357
+ // Signal init complete — sync part (agentTools, wrapStreamFn) is done
358
+ params.onInitComplete?.();
359
+ return dispatchPromise;
288
360
  },
289
- run: () =>
290
- // 🔐 Use AsyncLocalStorage to provide session context to tools
291
- runWithSessionContext(sessionContext, async () => {
292
- log(`[BOT-DISPATCH] ⏳ dispatchReplyFromConfig starting...`);
293
- log(`[BOT-DISPATCH] - sessionKey: ${ctxPayload.SessionKey}`);
294
- log(`[BOT-DISPATCH] - provider: ${ctxPayload.Provider}`);
295
- log(`[BOT-DISPATCH] - surface: ${ctxPayload.Surface}`);
296
- log(`[BOT-DISPATCH] - from: ${ctxPayload.From}`);
297
- log(`[BOT-DISPATCH] - body length: ${ctxPayload.Body?.length ?? 0}`);
298
- try {
299
- const result = await core.channel.reply.dispatchReplyFromConfig({
300
- ctx: ctxPayload,
301
- cfg,
302
- dispatcher,
303
- replyOptions,
304
- });
305
- log(`[BOT-DISPATCH] ✅ dispatchReplyFromConfig returned`);
306
- log(`[BOT-DISPATCH] - result: ${JSON.stringify(result)}`);
307
- return result;
308
- }
309
- catch (dispatchErr) {
310
- error(`[BOT-DISPATCH] ❌ dispatchReplyFromConfig threw`);
311
- error(`[BOT-DISPATCH] - error name: ${dispatchErr instanceof Error ? dispatchErr.name : "unknown"}`);
312
- error(`[BOT-DISPATCH] - error message: ${String(dispatchErr)}`);
313
- error(`[BOT-DISPATCH] - error stack: ${dispatchErr instanceof Error ? dispatchErr.stack?.slice(0, 500) : "N/A"}`);
314
- throw dispatchErr;
315
- }
316
- }),
317
361
  });
318
- log(`[BOT] ✅ Dispatcher completed for session: ${parsed.sessionId}`);
319
- log(`xy: dispatch complete (session=${parsed.sessionId})`);
362
+ logger.log(`[BOT] ✅ Dispatcher completed for session: ${parsed.sessionId}`);
363
+ logger.log(`xy: dispatch complete (session=${parsed.sessionId})`);
320
364
  }
321
365
  catch (err) {
322
366
  // ✅ Only log error, don't re-throw to prevent gateway restart
323
- error("Failed to handle XY message:", err);
367
+ logger.error("Failed to handle XY message:", err);
324
368
  runtime.error?.(`xy: Failed to handle message: ${String(err)}`);
325
- log(`[BOT] ❌ Error occurred, attempting cleanup...`);
369
+ logger.log(`[BOT] ❌ Error occurred, attempting cleanup...`);
326
370
  // 🔑 错误时也要清理taskId和session
327
371
  try {
328
372
  const params = message.params;
329
373
  const sessionId = params?.sessionId;
330
374
  if (sessionId) {
331
- log(`[BOT] 🧹 Cleaning up after error: ${sessionId}`);
375
+ logger.log(`[BOT] 🧹 Cleaning up after error: ${sessionId}`);
332
376
  // 清理 taskId
333
377
  decrementTaskIdRef(sessionId);
334
- unlockTaskId(sessionId);
335
378
  // 清理 session
336
379
  const core = getXYRuntime();
337
380
  const route = core.channel.routing.resolveAgentRoute({
@@ -344,11 +387,11 @@ export async function handleXYMessage(params) {
344
387
  },
345
388
  });
346
389
  unregisterSession(route.sessionKey);
347
- log(`[BOT] ✅ Cleanup completed after error`);
390
+ logger.log(`[BOT] ✅ Cleanup completed after error`);
348
391
  }
349
392
  }
350
393
  catch (cleanupErr) {
351
- log(`[BOT] ⚠️ Cleanup failed:`, cleanupErr);
394
+ logger.log(`[BOT] ⚠️ Cleanup failed:`, cleanupErr);
352
395
  // Ignore cleanup errors
353
396
  }
354
397
  // ❌ Don't re-throw: message processing error should not affect gateway stability
@@ -371,3 +414,169 @@ function buildXYMediaPayload(mediaList) {
371
414
  MediaTypes: mediaTypes.length > 0 ? mediaTypes : undefined,
372
415
  };
373
416
  }
417
+ // Use globalThis to survive module deduplication — provider.ts may load a
418
+ // different copy of bot.ts, so a plain module-level Map would be two objects.
419
+ const _g = globalThis;
420
+ if (!_g.__xyStreamingSignals)
421
+ _g.__xyStreamingSignals = new Map();
422
+ if (!_g.__xySteerQueues)
423
+ _g.__xySteerQueues = new Map();
424
+ const streamingSignals = _g.__xyStreamingSignals;
425
+ const steerQueues = _g.__xySteerQueues;
426
+ /**
427
+ * 由 provider.ts 在 wrapStreamFn 调用时触发。
428
+ * 这是模型 API 被调用的精确时刻,此时 isStreaming 一定为 true。
429
+ */
430
+ export function notifyModelStreaming(sessionId) {
431
+ const signal = streamingSignals.get(sessionId);
432
+ if (signal) {
433
+ // 不删除 signal——后续 steer 需要靠它判断模型已在 streaming。
434
+ // 清理由第一条消息的 onSettled 兜底。
435
+ signal.notify();
436
+ logger.log(`[STEER-QUEUE] 📡 Model streaming signal fired for session=${sessionId}`);
437
+ }
438
+ }
439
+ function createStreamingSignal(sessionId) {
440
+ let resolve;
441
+ const promise = new Promise(r => { resolve = r; });
442
+ const signal = { promise, notify: resolve };
443
+ streamingSignals.set(sessionId, signal);
444
+ logger.log(`[STEER-QUEUE] 🟢 Streaming signal created for session ${sessionId}`);
445
+ return signal;
446
+ }
447
+ /**
448
+ * 将 steer 消息放入 per-session 串行队列。
449
+ * 等待第一条消息的 streaming 信号(deliver 首次触发),然后 dispatch。
450
+ * 多个 steer 按到达顺序串行处理,无需重试。
451
+ */
452
+ function enqueueSteer(params) {
453
+ const { sessionId } = params;
454
+ // 取出当前队列尾部(或 undefined),然后链上新的 Promise
455
+ const prev = steerQueues.get(sessionId);
456
+ const next = (prev ?? Promise.resolve()).then(() => dispatchSteerWhenReady(params));
457
+ steerQueues.set(sessionId, next);
458
+ // 链条结束后清理
459
+ next.catch((err) => {
460
+ logger.error(`[STEER-QUEUE] ❌ Steer chain failed: ${String(err)}`);
461
+ }).finally(() => {
462
+ if (steerQueues.get(sessionId) === next) {
463
+ steerQueues.delete(sessionId);
464
+ }
465
+ });
466
+ return next;
467
+ }
468
+ async function dispatchSteerWhenReady(params) {
469
+ const { sessionId, sessionKey, steerText } = params;
470
+ // 1. 等待第一条消息开始 streaming
471
+ // signal 可能尚未创建(第一条消息还在文件下载等耗时操作中),
472
+ // 轮询等待直到 signal 出现,最長等待 ~5 秒。
473
+ let signal = streamingSignals.get(sessionId);
474
+ if (!signal) {
475
+ logger.log(`[STEER-QUEUE] ⏳ Signal not yet created, polling for session=${sessionId}`);
476
+ for (let i = 0; i < 50; i++) {
477
+ await new Promise(r => setTimeout(r, 100));
478
+ signal = streamingSignals.get(sessionId);
479
+ if (signal)
480
+ break;
481
+ if (!hasActiveTask(sessionId)) {
482
+ logger.log(`[STEER-QUEUE] ℹ️ First message completed while waiting, skip steer`);
483
+ return;
484
+ }
485
+ }
486
+ }
487
+ if (signal) {
488
+ logger.log(`[STEER-QUEUE] ⏳ Waiting for streaming signal, session=${sessionId}`);
489
+ await signal.promise;
490
+ logger.log(`[STEER-QUEUE] ✅ Streaming signal received, session=${sessionId}`);
491
+ }
492
+ else {
493
+ // 轮询超时且 hasActiveTask 仍为 true——说明第一条消息可能卡在异常路径,
494
+ // 没有创建 signal。此时 dispatch 会与第一条消息的模型调用并发冲突,放弃。
495
+ logger.log(`[STEER-QUEUE] ⚠️ Signal never appeared after polling, skip steer to avoid collision`);
496
+ return;
497
+ }
498
+ // 2. 第一条消息已结束 → 放弃
499
+ if (!hasActiveTask(sessionId)) {
500
+ logger.log(`[STEER-QUEUE] ℹ️ First message completed, skip steer`);
501
+ return;
502
+ }
503
+ // 3. 构建 dispatch 上下文并 dispatch /steer
504
+ const core = getXYRuntime();
505
+ const speaker = sessionId;
506
+ // 如果有文件附件,把路径拼到 steer 文本末尾,让模型通过工具读取
507
+ const mediaPaths = params.mediaPayload?.MediaPaths;
508
+ const fileHint = mediaPaths && mediaPaths.length > 0
509
+ ? `\n【用户上传附件】:${JSON.stringify(mediaPaths)}`
510
+ : "";
511
+ const steerCommand = `/steer ${steerText}${fileHint}`;
512
+ const messageBody = `${speaker}: ${steerCommand}`;
513
+ const envelopeOptions = core.channel.reply.resolveEnvelopeFormatOptions(params.cfg);
514
+ const body = core.channel.reply.formatAgentEnvelope({
515
+ channel: "xiaoyi-channel",
516
+ from: speaker,
517
+ timestamp: new Date(),
518
+ envelope: envelopeOptions,
519
+ body: messageBody,
520
+ });
521
+ const ctxPayload = core.channel.reply.finalizeInboundContext({
522
+ Body: body,
523
+ RawBody: steerCommand,
524
+ CommandBody: steerCommand,
525
+ From: sessionId,
526
+ To: sessionId,
527
+ SessionKey: params.route.sessionKey,
528
+ AccountId: params.route.accountId,
529
+ ChatType: "direct",
530
+ GroupSubject: undefined,
531
+ SenderName: sessionId,
532
+ SenderId: sessionId,
533
+ Provider: "xiaoyi-channel",
534
+ Surface: "xiaoyi-channel",
535
+ MessageSid: `xiaoyi_${params.parsed.taskId}_${params.deviceType}`,
536
+ Timestamp: Date.now(),
537
+ WasMentioned: false,
538
+ CommandAuthorized: true,
539
+ OriginatingChannel: "xiaoyi-channel",
540
+ OriginatingTo: sessionId,
541
+ ReplyToBody: undefined,
542
+ ...params.mediaPayload,
543
+ });
544
+ const steerState = { steered: true };
545
+ const { dispatcher, replyOptions } = createXYReplyDispatcher({
546
+ cfg: params.cfg,
547
+ runtime: params.runtime,
548
+ sessionId,
549
+ taskId: params.parsed.taskId,
550
+ messageId: params.parsed.messageId,
551
+ accountId: params.route.accountId,
552
+ steerState,
553
+ });
554
+ const sessionContext = {
555
+ config: resolveXYConfig(params.cfg),
556
+ sessionId,
557
+ taskId: params.parsed.taskId,
558
+ messageId: params.parsed.messageId,
559
+ agentId: params.route.accountId,
560
+ deviceType: params.deviceType,
561
+ };
562
+ logger.log(`[STEER-QUEUE] 🚀 Dispatching steer for session=${sessionId}`);
563
+ await core.channel.reply.withReplyDispatcher({
564
+ dispatcher,
565
+ onSettled: () => {
566
+ logger.log(`[STEER-QUEUE] 🏁 Steer dispatch settled for session=${sessionId}`);
567
+ },
568
+ run: () => {
569
+ return runWithSessionContext(sessionContext, async () => {
570
+ const result = await core.channel.reply.dispatchReplyFromConfig({
571
+ ctx: ctxPayload,
572
+ cfg: params.cfg,
573
+ dispatcher,
574
+ replyOptions,
575
+ });
576
+ logger.log(`[STEER-QUEUE] dispatch result: ${JSON.stringify(result)}`);
577
+ return result;
578
+ });
579
+ },
580
+ });
581
+ logger.log(`[STEER-QUEUE] ✅ Steer dispatch completed for session=${sessionId}`);
582
+ }