@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.
- package/dist/index.d.ts +4 -5
- package/dist/index.js +102 -84
- package/dist/provider-discovery.d.ts +2 -0
- package/dist/provider-discovery.js +4 -0
- package/dist/src/bot.d.ts +13 -0
- package/dist/src/bot.js +345 -136
- package/dist/src/channel.js +2 -17
- package/dist/src/client.d.ts +1 -5
- package/dist/src/client.js +32 -37
- package/dist/src/cspl/call-api.d.ts +6 -0
- package/dist/src/cspl/call-api.js +37 -16
- package/dist/src/cspl/config.d.ts +11 -1
- package/dist/src/cspl/config.js +30 -0
- package/dist/src/cspl/middleware.d.ts +8 -0
- package/dist/src/cspl/middleware.js +90 -0
- package/dist/src/cspl/steer-context.d.ts +21 -0
- package/dist/src/cspl/steer-context.js +78 -0
- package/dist/src/file-download.js +4 -3
- package/dist/src/file-upload.js +19 -18
- package/dist/src/formatter.js +32 -44
- package/dist/src/heartbeat.js +4 -3
- package/dist/src/login-token-handler.js +13 -10
- package/dist/src/message-queue.js +2 -1
- package/dist/src/monitor.js +62 -41
- package/dist/src/outbound.js +22 -18
- package/dist/src/provider.d.ts +1 -1
- package/dist/src/provider.js +147 -71
- package/dist/src/push.js +16 -15
- package/dist/src/reply-dispatcher.d.ts +3 -1
- package/dist/src/reply-dispatcher.js +65 -63
- package/dist/src/runtime.d.ts +3 -11
- package/dist/src/runtime.js +6 -18
- package/dist/src/self-evolution-handler.js +11 -14
- package/dist/src/self-evolution-tool-result-nudge.d.ts +3 -0
- package/dist/src/self-evolution-tool-result-nudge.js +96 -0
- package/dist/src/skill-retriever/hooks.js +7 -16
- package/dist/src/skill-retriever/tool-search.js +16 -17
- package/dist/src/steer-injector.js +1 -1
- package/dist/src/task-manager.d.ts +4 -27
- package/dist/src/task-manager.js +19 -79
- package/dist/src/tools/calendar-tool.d.ts +2 -1
- package/dist/src/tools/calendar-tool.js +116 -116
- package/dist/src/tools/call-device-tool.d.ts +2 -1
- package/dist/src/tools/call-device-tool.js +126 -103
- package/dist/src/tools/call-phone-tool.d.ts +2 -1
- package/dist/src/tools/call-phone-tool.js +113 -113
- package/dist/src/tools/create-alarm-tool.d.ts +2 -1
- package/dist/src/tools/create-alarm-tool.js +231 -231
- package/dist/src/tools/create-all-tools.d.ts +16 -0
- package/dist/src/tools/create-all-tools.js +50 -0
- package/dist/src/tools/delete-alarm-tool.d.ts +2 -1
- package/dist/src/tools/delete-alarm-tool.js +135 -135
- package/dist/src/tools/get-alarm-tool-schema.d.ts +2 -1
- package/dist/src/tools/get-alarm-tool-schema.js +16 -10
- package/dist/src/tools/get-calendar-tool-schema.d.ts +2 -1
- package/dist/src/tools/get-calendar-tool-schema.js +12 -8
- package/dist/src/tools/get-collection-tool-schema.d.ts +2 -1
- package/dist/src/tools/get-collection-tool-schema.js +11 -9
- package/dist/src/tools/get-contact-tool-schema.d.ts +2 -1
- package/dist/src/tools/get-contact-tool-schema.js +16 -10
- package/dist/src/tools/get-device-file-tool-schema.d.ts +2 -1
- package/dist/src/tools/get-device-file-tool-schema.js +13 -9
- package/dist/src/tools/get-email-tool-schema.d.ts +2 -1
- package/dist/src/tools/get-email-tool-schema.js +11 -8
- package/dist/src/tools/get-note-tool-schema.d.ts +2 -1
- package/dist/src/tools/get-note-tool-schema.js +14 -9
- package/dist/src/tools/get-photo-tool-schema.d.ts +2 -1
- package/dist/src/tools/get-photo-tool-schema.js +12 -9
- package/dist/src/tools/image-reading-tool.d.ts +3 -2
- package/dist/src/tools/image-reading-tool.js +86 -165
- package/dist/src/tools/location-tool.d.ts +2 -1
- package/dist/src/tools/location-tool.js +91 -91
- package/dist/src/tools/login-token-tool.d.ts +2 -1
- package/dist/src/tools/login-token-tool.js +124 -116
- package/dist/src/tools/modify-alarm-tool.d.ts +2 -1
- package/dist/src/tools/modify-alarm-tool.js +236 -236
- package/dist/src/tools/modify-note-tool.d.ts +2 -1
- package/dist/src/tools/modify-note-tool.js +108 -108
- package/dist/src/tools/note-tool.d.ts +2 -1
- package/dist/src/tools/note-tool.js +107 -107
- package/dist/src/tools/query-app-message-tool.d.ts +2 -1
- package/dist/src/tools/query-app-message-tool.js +112 -111
- package/dist/src/tools/query-memory-data-tool.d.ts +2 -1
- package/dist/src/tools/query-memory-data-tool.js +113 -112
- package/dist/src/tools/query-todo-task-tool.d.ts +2 -1
- package/dist/src/tools/query-todo-task-tool.js +107 -106
- package/dist/src/tools/save-file-to-phone-tool.d.ts +2 -1
- package/dist/src/tools/save-file-to-phone-tool.js +131 -131
- package/dist/src/tools/save-media-to-gallery-tool.d.ts +2 -1
- package/dist/src/tools/save-media-to-gallery-tool.js +138 -138
- package/dist/src/tools/save-self-evolution-skill-tool.d.ts +2 -1
- package/dist/src/tools/save-self-evolution-skill-tool.js +194 -196
- package/dist/src/tools/search-alarm-tool.d.ts +2 -1
- package/dist/src/tools/search-alarm-tool.js +175 -175
- package/dist/src/tools/search-calendar-tool.d.ts +2 -1
- package/dist/src/tools/search-calendar-tool.js +149 -149
- package/dist/src/tools/search-contact-tool.d.ts +2 -1
- package/dist/src/tools/search-contact-tool.js +102 -102
- package/dist/src/tools/search-email-tool.d.ts +2 -1
- package/dist/src/tools/search-email-tool.js +111 -111
- package/dist/src/tools/search-file-tool.d.ts +2 -1
- package/dist/src/tools/search-file-tool.js +103 -103
- package/dist/src/tools/search-message-tool.d.ts +2 -1
- package/dist/src/tools/search-message-tool.js +104 -104
- package/dist/src/tools/search-note-tool.d.ts +2 -1
- package/dist/src/tools/search-note-tool.js +99 -99
- package/dist/src/tools/search-photo-gallery-tool.d.ts +2 -1
- package/dist/src/tools/search-photo-gallery-tool.js +38 -38
- package/dist/src/tools/send-email-tool.d.ts +2 -1
- package/dist/src/tools/send-email-tool.js +109 -108
- package/dist/src/tools/send-file-to-user-tool.d.ts +2 -1
- package/dist/src/tools/send-file-to-user-tool.js +157 -155
- package/dist/src/tools/send-message-tool.d.ts +2 -1
- package/dist/src/tools/send-message-tool.js +123 -123
- package/dist/src/tools/session-helper.d.ts +24 -0
- package/dist/src/tools/session-helper.js +45 -0
- package/dist/src/tools/session-manager.d.ts +29 -6
- package/dist/src/tools/session-manager.js +134 -19
- package/dist/src/tools/upload-file-tool.d.ts +2 -1
- package/dist/src/tools/upload-file-tool.js +82 -82
- package/dist/src/tools/upload-photo-tool.d.ts +2 -1
- package/dist/src/tools/upload-photo-tool.js +73 -73
- package/dist/src/tools/xiaoyi-add-collection-tool.d.ts +2 -1
- package/dist/src/tools/xiaoyi-add-collection-tool.js +147 -147
- package/dist/src/tools/xiaoyi-collection-tool.d.ts +2 -1
- package/dist/src/tools/xiaoyi-collection-tool.js +115 -115
- package/dist/src/tools/xiaoyi-delete-collection-tool.d.ts +2 -1
- package/dist/src/tools/xiaoyi-delete-collection-tool.js +128 -128
- package/dist/src/tools/xiaoyi-gui-tool.d.ts +2 -1
- package/dist/src/tools/xiaoyi-gui-tool.js +89 -88
- package/dist/src/trigger-handler.js +8 -9
- package/dist/src/utils/logger.js +105 -19
- package/dist/src/utils/self-evolution-manager.d.ts +5 -0
- package/dist/src/utils/self-evolution-manager.js +45 -23
- package/dist/src/utils/throw.d.ts +5 -0
- package/dist/src/utils/throw.js +10 -0
- package/dist/src/websocket.js +35 -31
- package/dist/src/xy-session-store.d.ts +79 -0
- package/dist/src/xy-session-store.js +153 -0
- package/openclaw.plugin.json +25 -0
- 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 {
|
|
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
|
-
|
|
25
|
-
|
|
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
|
-
// 🔑
|
|
104
|
-
const
|
|
105
|
-
const
|
|
106
|
-
if (
|
|
107
|
-
log(`[BOT] 🔄 STEER MODE - Second message detected (will
|
|
108
|
-
log(`[BOT] - Session: ${parsed.sessionId}`);
|
|
109
|
-
log(`[BOT] - New taskId: ${parsed.taskId}
|
|
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
|
-
//
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
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
|
-
|
|
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
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
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
|
-
|
|
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
|
-
|
|
201
|
-
//
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
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.
|
|
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
|
-
// 🔑
|
|
246
|
-
|
|
247
|
-
|
|
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
|
-
|
|
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] -
|
|
278
|
-
// 🔑
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
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
|
-
|
|
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
|
+
}
|