@ynhcj/xiaoyi-channel 0.0.168-beta → 0.0.168-next

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 (75) hide show
  1. package/dist/index.js +141 -1
  2. package/dist/src/bot.js +62 -27
  3. package/dist/src/channel.js +9 -9
  4. package/dist/src/cron-command.d.ts +2 -0
  5. package/dist/src/cron-command.js +14 -8
  6. package/dist/src/cron-query-handler.js +45 -8
  7. package/dist/src/cspl/call_api.js +2 -2
  8. package/dist/src/cspl/sentinel_hook.js +2 -2
  9. package/dist/src/cspl/upload_file.js +2 -2
  10. package/dist/src/file-upload.d.ts +5 -0
  11. package/dist/src/file-upload.js +102 -0
  12. package/dist/src/formatter.d.ts +29 -0
  13. package/dist/src/formatter.js +106 -19
  14. package/dist/src/monitor.js +2 -9
  15. package/dist/src/parser.d.ts +6 -0
  16. package/dist/src/parser.js +23 -13
  17. package/dist/src/provider.js +73 -62
  18. package/dist/src/reply-dispatcher.js +57 -60
  19. package/dist/src/task-manager.d.ts +5 -6
  20. package/dist/src/task-manager.js +5 -9
  21. package/dist/src/tools/agent-as-skill-tool.js +4 -4
  22. package/dist/src/tools/calendar-tool.js +4 -4
  23. package/dist/src/tools/call-device-tool.js +5 -6
  24. package/dist/src/tools/call-phone-tool.js +4 -4
  25. package/dist/src/tools/check-plugin-privilege-tool.js +8 -6
  26. package/dist/src/tools/create-alarm-tool.js +4 -4
  27. package/dist/src/tools/create-all-tools.js +2 -0
  28. package/dist/src/tools/delete-alarm-tool.js +4 -4
  29. package/dist/src/tools/device-tool-map.d.ts +1 -1
  30. package/dist/src/tools/device-tool-map.js +8 -1
  31. package/dist/src/tools/discover-cross-devices-tool.js +7 -8
  32. package/dist/src/tools/display-a2ui-card-tool.js +5 -6
  33. package/dist/src/tools/image-reading-tool.js +3 -1
  34. package/dist/src/tools/location-tool.js +4 -4
  35. package/dist/src/tools/login-token-tool.js +6 -7
  36. package/dist/src/tools/modify-alarm-tool.js +21 -4
  37. package/dist/src/tools/modify-note-tool.js +4 -4
  38. package/dist/src/tools/note-tool.js +4 -4
  39. package/dist/src/tools/query-app-message-tool.js +4 -4
  40. package/dist/src/tools/query-memory-data-tool.js +4 -4
  41. package/dist/src/tools/query-todo-task-tool.js +4 -4
  42. package/dist/src/tools/save-file-to-phone-tool.js +4 -4
  43. package/dist/src/tools/save-media-to-gallery-tool.js +4 -4
  44. package/dist/src/tools/save-self-evolution-skill-tool.js +3 -1
  45. package/dist/src/tools/search-alarm-tool.js +4 -4
  46. package/dist/src/tools/search-calendar-tool.js +4 -4
  47. package/dist/src/tools/search-contact-tool.js +4 -4
  48. package/dist/src/tools/search-email-tool.js +4 -4
  49. package/dist/src/tools/search-file-tool.js +4 -4
  50. package/dist/src/tools/search-message-tool.js +4 -4
  51. package/dist/src/tools/search-note-tool.js +4 -4
  52. package/dist/src/tools/search-photo-gallery-tool.js +4 -4
  53. package/dist/src/tools/send-cross-device-task-tool.js +91 -25
  54. package/dist/src/tools/send-email-tool.js +4 -4
  55. package/dist/src/tools/send-file-to-user-tool.d.ts +1 -1
  56. package/dist/src/tools/send-file-to-user-tool.js +15 -19
  57. package/dist/src/tools/send-html-card-tool.d.ts +7 -0
  58. package/dist/src/tools/send-html-card-tool.js +113 -0
  59. package/dist/src/tools/send-message-tool.js +4 -4
  60. package/dist/src/tools/session-manager.d.ts +20 -52
  61. package/dist/src/tools/session-manager.js +97 -239
  62. package/dist/src/tools/upload-file-tool.js +4 -4
  63. package/dist/src/tools/upload-photo-tool.js +4 -4
  64. package/dist/src/tools/xiaoyi-add-collection-tool.js +4 -4
  65. package/dist/src/tools/xiaoyi-collection-tool.js +4 -4
  66. package/dist/src/tools/xiaoyi-delete-collection-tool.js +4 -4
  67. package/dist/src/tools/xiaoyi-gui-tool.js +6 -7
  68. package/dist/src/types.d.ts +6 -6
  69. package/dist/src/utils/config-manager.d.ts +3 -2
  70. package/dist/src/utils/config-manager.js +22 -2
  71. package/dist/src/utils/cron-push-map.d.ts +26 -0
  72. package/dist/src/utils/cron-push-map.js +131 -0
  73. package/dist/src/utils/logger.js +3 -14
  74. package/dist/src/websocket.js +11 -13
  75. package/package.json +1 -1
package/dist/index.js CHANGED
@@ -3,7 +3,10 @@ import { xiaoyiProvider } from "./src/provider.js";
3
3
  import { xyPlugin } from "./src/channel.js";
4
4
  import registerSentinelHook from "./src/cspl/sentinel_hook.js";
5
5
  import { setXYRuntime } from "./src/runtime.js";
6
- import { markCronToolCall, clearCronToolCall } from "./src/tools/session-manager.js";
6
+ import { markCronToolCall, clearCronToolCall, getCurrentSessionContext } from "./src/tools/session-manager.js";
7
+ import { configManager } from "./src/utils/config-manager.js";
8
+ import { setJobPushId } from "./src/utils/cron-push-map.js";
9
+ import { getAllPushIds } from "./src/utils/pushid-manager.js";
7
10
  import { registerSelfEvolutionToolResultNudge } from "./src/self-evolution-tool-result-nudge.js";
8
11
  import { createBeforePromptBuildHandler } from "./src/skill-retriever/hooks.js";
9
12
  import { normalizeToolRetrieverConfig } from "./src/skill-retriever/config.js";
@@ -25,8 +28,145 @@ function registerCronDetectionHook(api) {
25
28
  if (event.toolCallId) {
26
29
  clearCronToolCall(event.toolCallId);
27
30
  }
31
+ // 捕获对话创建的 cron job:agent 调 cron(add) 后,从 result 拿 jobId,
32
+ // 配合当前会话的 pushId,写入 jobId↔pushId 映射,供 fire 时反查设备。
33
+ await captureCronAddMapping(event, ctx).catch((err) => {
34
+ // 捕获失败不影响工具结果
35
+ console.error("[xy] captureCronAddMapping failed:", err);
36
+ });
28
37
  });
29
38
  }
39
+ /** 从 cron add 工具结果中提取 jobId 并写入 pushId 映射。 */
40
+ async function captureCronAddMapping(event, ctx) {
41
+ // 两条创建路径都要捕获:
42
+ // 1) cron agent 工具:toolName==="cron", params.action==="add"
43
+ // 2) exec 跑 CLI:toolName==="exec", params.command 含 "cron add"
44
+ // (agent 实际用的是这条:openclaw cron add --name ... --cron ... --message ...)
45
+ const isCronAddTool = event.toolName === "cron" &&
46
+ (event.params?.action === "add" || event.params?.action === "create");
47
+ const isExecCronAdd = event.toolName === "exec" && isExecCronAddCommand(event.params?.command);
48
+ if (!isCronAddTool && !isExecCronAdd)
49
+ return;
50
+ console.log(`[CRONMAP] after_tool_call path=${event.toolName}, resultType=${typeof event.result}`);
51
+ const jobId = readJobIdFromResult(event.result);
52
+ if (!jobId) {
53
+ console.log(`[CRONMAP] skip: could not extract jobId. preview=${preview(event.result)}`);
54
+ return;
55
+ }
56
+ console.log(`[CRONMAP] extracted jobId=${jobId}`);
57
+ const sessionCtx = getCurrentSessionContext();
58
+ const sessionId = sessionCtx?.sessionId;
59
+ if (!sessionId) {
60
+ console.log(`[CRONMAP] skip: no sessionId in ALS scope (ctxFound=${!!sessionCtx})`);
61
+ return;
62
+ }
63
+ const pushId = await resolvePushId(sessionId);
64
+ if (!pushId) {
65
+ console.log(`[CRONMAP] skip: no pushId available for sessionId=${sessionId} (no session match, no global, no file)`);
66
+ return;
67
+ }
68
+ console.log(`[CRONMAP] writing map: jobId=${jobId}, sessionId=${sessionId}, pushId=${pushId.substring(0, 16)}...`);
69
+ await setJobPushId(jobId, {
70
+ pushId,
71
+ sessionId,
72
+ deviceType: sessionCtx?.deviceType,
73
+ source: event.toolName === "exec" ? "exec-cli" : "conversation",
74
+ });
75
+ console.log(`[CRONMAP] map written OK`);
76
+ }
77
+ /** 回退链取 pushId:当前会话 → 全局兜底 → 本地文件首个(保底)。 */
78
+ async function resolvePushId(sessionId) {
79
+ // 1. 同会话
80
+ const session = configManager.getPushId(sessionId);
81
+ if (session)
82
+ return session;
83
+ // 2. 全局(任何会话注册过的)
84
+ const global = configManager.getPushId();
85
+ if (global)
86
+ return global;
87
+ // 3. 文件兜底
88
+ try {
89
+ const all = await getAllPushIds();
90
+ if (all.length > 0)
91
+ return all[0];
92
+ }
93
+ catch {
94
+ // ignore
95
+ }
96
+ return null;
97
+ }
98
+ /** 判断 exec 命令是否为 cron add(匹配 "openclaw cron add" 或裸 "cron add",排除 list/remove 等)。 */
99
+ function isExecCronAddCommand(command) {
100
+ if (typeof command !== "string")
101
+ return false;
102
+ return /\bcron\s+add\b/.test(command);
103
+ }
104
+ /** 取结果的短预览,用于诊断。 */
105
+ function preview(value) {
106
+ if (value == null)
107
+ return String(value);
108
+ const s = typeof value === "string" ? value : JSON.stringify(value);
109
+ return s.length > 200 ? s.slice(0, 200) + "…" : s;
110
+ }
111
+ /** 防御性地从 cron add 结果中取 job id。
112
+ * 覆盖:裸 job 对象、JSON 字符串、exec 输出文本、
113
+ * {content:[{text}]} / {stdout} / data/result/job 嵌套。 */
114
+ function readJobIdFromResult(result) {
115
+ if (!result)
116
+ return undefined;
117
+ // {content: [{type:"text", text: "..."}]} — exec 工具的输出信封
118
+ if (result && typeof result === "object") {
119
+ const contentArr = result.content;
120
+ if (Array.isArray(contentArr)) {
121
+ for (const item of contentArr) {
122
+ if (item && typeof item === "object") {
123
+ const text = item.text;
124
+ if (typeof text === "string" && text.trim()) {
125
+ const fromContent = readJobIdFromResult(text);
126
+ if (fromContent)
127
+ return fromContent;
128
+ }
129
+ }
130
+ }
131
+ }
132
+ }
133
+ // {stdout} — 备选 exec 输出信封
134
+ if (result && typeof result === "object") {
135
+ const stdout = result.stdout;
136
+ if (typeof stdout === "string" && stdout.trim()) {
137
+ const fromStdout = readJobIdFromResult(stdout);
138
+ if (fromStdout)
139
+ return fromStdout;
140
+ }
141
+ }
142
+ let obj = result;
143
+ if (typeof result === "string") {
144
+ try {
145
+ obj = JSON.parse(result);
146
+ }
147
+ catch {
148
+ // 纯文本:可能含 stderr 前缀行 + JSON。用正则抓 "id":"..."。
149
+ const m = result.match(/"id"\s*:\s*"([^"]+)"/);
150
+ if (m)
151
+ return m[1];
152
+ return undefined;
153
+ }
154
+ }
155
+ if (obj && typeof obj === "object") {
156
+ const id = obj.id;
157
+ if (typeof id === "string" && id.trim())
158
+ return id.trim();
159
+ for (const k of ["data", "result", "job"]) {
160
+ const inner = obj[k];
161
+ if (inner && typeof inner === "object") {
162
+ const innerId = inner.id;
163
+ if (typeof innerId === "string" && innerId.trim())
164
+ return innerId.trim();
165
+ }
166
+ }
167
+ }
168
+ return undefined;
169
+ }
30
170
  function registerFullHooks(api) {
31
171
  // SKILL RETRIEVER HOOK: before_prompt_build hook
32
172
  const pluginConfig = api.pluginConfig || {};
package/dist/src/bot.js CHANGED
@@ -1,11 +1,12 @@
1
+ import { updateSessionStoreEntry, updateSessionStore, resolveStorePath } from "openclaw/plugin-sdk/session-store-runtime";
1
2
  import { getXYRuntime } from "./runtime.js";
2
3
  import { createXYReplyDispatcher } from "./reply-dispatcher.js";
3
- import { parseA2AMessage, extractTextFromParts, extractFileParts, extractPushId, extractDeviceType, extractTriggerData, extractRunCrossTaskContext } from "./parser.js";
4
+ import { parseA2AMessage, extractTextFromParts, extractFileParts, extractPushId, extractDeviceType, extractModelName, extractTriggerData, extractRunCrossTaskContext } from "./parser.js";
4
5
  import { downloadFilesFromParts } from "./file-download.js";
5
6
  import { resolveXYConfig } from "./config.js";
6
7
  import { sendStatusUpdate, sendClearContextResponse, sendTasksCancelResponse, sendA2AResponse } from "./formatter.js";
7
8
  import { appendSelfEvolutionKeywordNudge, shouldNudgeForSelfEvolutionKeyword, } from "./self-evolution-keyword.js";
8
- import { registerSession, unregisterSession, runWithSessionContext } from "./tools/session-manager.js";
9
+ import { runWithSessionContext } from "./tools/session-manager.js";
9
10
  import { configManager } from "./utils/config-manager.js";
10
11
  import { addPushId } from "./utils/pushid-manager.js";
11
12
  import { getPushDataById } from "./utils/pushdata-manager.js";
@@ -138,6 +139,11 @@ export async function handleXYMessage(params) {
138
139
  if (deviceType) {
139
140
  log.log(`[BOT] Extracted deviceType: ${deviceType}`);
140
141
  }
142
+ // Extract modelName if present (used by provider.ts to override model.id)
143
+ const modelName = extractModelName(parsed.parts);
144
+ if (modelName) {
145
+ log.log(`[BOT] Extracted modelName: ${modelName}`);
146
+ }
141
147
  const runCrossTaskContext = extractRunCrossTaskContext(parsed.parts);
142
148
  // Resolve configuration (needed for status updates)
143
149
  const config = resolveXYConfig(cfg);
@@ -154,18 +160,57 @@ export async function handleXYMessage(params) {
154
160
  },
155
161
  });
156
162
  log.log(`[BOT] Resolved route, sessionKey=${route.sessionKey}`);
157
- // Steer injections skip session registration to avoid refCount leaks
163
+ // ALS only: no registerSession. The sessionContext built below is handed
164
+ // to runWithSessionContext() inside withReplyDispatcher.run, which is the
165
+ // single wrap point for the whole agent turn.
158
166
  if (!skipReg) {
159
- registerSession(route.sessionKey, {
160
- config,
161
- sessionId: parsed.sessionId,
162
- distributionSessionId,
163
- taskId: parsed.taskId,
164
- messageId: parsed.messageId,
165
- agentId: route.accountId,
166
- deviceType,
167
- runCrossTaskContext: runCrossTaskContext ?? undefined,
168
- });
167
+ // 🔑 Sync A2A modelName to OpenClaw session store so that session_status
168
+ // reports the correct model. Without this, session_status returns the
169
+ // configured default model instead of the A2A-specified one.
170
+ if (modelName && modelName.trim() !== "" && modelName.toLowerCase() !== "none") {
171
+ try {
172
+ const storePath = resolveStorePath();
173
+ const result = await updateSessionStoreEntry({
174
+ storePath,
175
+ sessionKey: route.sessionKey,
176
+ update: async () => ({
177
+ providerOverride: "xiaoyiprovider",
178
+ modelOverride: modelName,
179
+ modelOverrideSource: "user",
180
+ model: "",
181
+ modelProvider: "",
182
+ contextTokens: 256_000,
183
+ }),
184
+ });
185
+ if (!result) {
186
+ // Session entry doesn't exist yet (first message, xy_channel
187
+ // bypasses the standard turn kernel). Create a minimal entry
188
+ // with the override via updateSessionStore.
189
+ await updateSessionStore(storePath, (store) => {
190
+ if (!store[route.sessionKey]) {
191
+ store[route.sessionKey] = {
192
+ // sessionId must pass validateSessionId regex /^[a-z0-9][a-z0-9._-]{0,127}$/i
193
+ // route.sessionKey like "agent:main:direct:xxx" contains colons which are invalid.
194
+ // Use parsed.sessionId (raw UUID from A2A) which is always safe.
195
+ sessionId: parsed.sessionId,
196
+ updatedAt: Date.now(),
197
+ providerOverride: "xiaoyiprovider",
198
+ modelOverride: modelName,
199
+ modelOverrideSource: "user",
200
+ contextTokens: 256_000,
201
+ };
202
+ }
203
+ });
204
+ log.log(`[BOT] Created session entry with model override: xiaoyiprovider/${modelName}`);
205
+ }
206
+ else {
207
+ log.log(`[BOT] Patched session store model override: xiaoyiprovider/${modelName}`);
208
+ }
209
+ }
210
+ catch (patchErr) {
211
+ log.error(`[BOT] Failed to patch session model override:`, patchErr);
212
+ }
213
+ }
169
214
  // 🔑 发送初始状态更新
170
215
  log.log(`[BOT] Sending initial status update`);
171
216
  void sendStatusUpdate({
@@ -311,6 +356,7 @@ export async function handleXYMessage(params) {
311
356
  messageId: parsed.messageId,
312
357
  agentId: route.accountId,
313
358
  deviceType,
359
+ modelName,
314
360
  runCrossTaskContext: runCrossTaskContext ?? undefined,
315
361
  };
316
362
  log.log(`[BOT-DISPATCH] withReplyDispatcher starting, sessionKey=${route.sessionKey}`);
@@ -325,7 +371,6 @@ export async function handleXYMessage(params) {
325
371
  }
326
372
  streamingSignals.delete(parsed.sessionId);
327
373
  decrementTaskIdRef(parsed.sessionId);
328
- unregisterSession(route.sessionKey);
329
374
  log.log(`[BOT] Cleanup completed`);
330
375
  },
331
376
  run: () => {
@@ -335,6 +380,7 @@ export async function handleXYMessage(params) {
335
380
  // signal init complete to release the global dispatch gate
336
381
  // for the next session.
337
382
  const dispatchPromise = runWithSessionContext(sessionContext, async () => {
383
+ log.log(`[ALS-PROOF] bot entered dispatch scope sessionId=${sessionContext.sessionId} taskId=${sessionContext.taskId} isSteer=false`);
338
384
  log.log(`[BOT-DISPATCH] dispatchReplyFromConfig starting, body.length=${ctxPayload.Body?.length ?? 0}`);
339
385
  try {
340
386
  const result = await core.channel.reply.dispatchReplyFromConfig({
@@ -367,7 +413,7 @@ export async function handleXYMessage(params) {
367
413
  errLog.error("Failed to handle XY message:", err);
368
414
  runtime.error?.(`xy: Failed to handle message: ${String(err)}`);
369
415
  errLog.log(`[BOT] Error occurred, attempting cleanup`);
370
- // 🔑 错误时也要清理taskIdsession
416
+ // 🔑 错误时也要清理taskIdsession 走 ALS,作用域退出自动清理)
371
417
  try {
372
418
  const params = message.params;
373
419
  const sessionId = params?.sessionId;
@@ -375,18 +421,6 @@ export async function handleXYMessage(params) {
375
421
  errLog.log(`[BOT] Cleaning up after error`);
376
422
  // 清理 taskId
377
423
  decrementTaskIdRef(sessionId);
378
- // 清理 session
379
- const core = getXYRuntime();
380
- const route = core.channel.routing.resolveAgentRoute({
381
- cfg,
382
- channel: "xiaoyi-channel",
383
- accountId,
384
- peer: {
385
- kind: "direct",
386
- id: sessionId,
387
- },
388
- });
389
- unregisterSession(route.sessionKey);
390
424
  errLog.log(`[BOT] Cleanup completed after error`);
391
425
  }
392
426
  }
@@ -571,6 +605,7 @@ async function dispatchSteerWhenReady(params) {
571
605
  },
572
606
  run: () => {
573
607
  return runWithSessionContext(sessionContext, async () => {
608
+ log.log(`[ALS-PROOF] bot entered steer dispatch scope sessionId=${sessionContext.sessionId} taskId=${sessionContext.taskId} isSteer=true`);
574
609
  const result = await core.channel.reply.dispatchReplyFromConfig({
575
610
  ctx: ctxPayload,
576
611
  cfg: params.cfg,
@@ -2,7 +2,7 @@ import { resolveXYConfig, listXYAccountIds, getDefaultXYAccountId } from "./conf
2
2
  import { xyConfigSchema } from "./config-schema.js";
3
3
  import { xyOutbound } from "./outbound.js";
4
4
  import { filterToolsByDevice } from "./tools/device-tool-map.js";
5
- import { getCurrentSessionContext, registerSession } from "./tools/session-manager.js";
5
+ import { getCurrentSessionContext } from "./tools/session-manager.js";
6
6
  import { createAllTools } from "./tools/create-all-tools.js";
7
7
  import { logger } from "./utils/logger.js";
8
8
  /**
@@ -69,12 +69,10 @@ export const xyPlugin = {
69
69
  agentTools: (params) => {
70
70
  let ctx = getCurrentSessionContext();
71
71
  // ── Cron / non-session fallback ──────────────────────────────
72
- // When no active xy WebSocket session exists but the openclaw cfg
73
- // is provided (framework calls agentTools({ cfg })), create a
74
- // synthetic "cron session". This enables cron-triggered agent
75
- // turns and cross-channel tool calls to use xiaoyi tools via the
76
- // push channel. sendCommand() detects the "cron-" sessionId
77
- // prefix and routes commands through push instead of WebSocket.
72
+ // cron 路径不进 ALS: openclaw cron runner 同步调 agentTools({cfg})
73
+ // 返回工具后才在别处跑 turn, xy_channel 没有 wrap 整个 turn 的点。
74
+ // 这里同步构造合成 ctx 给工具闭包捕获, 工具调用走 sendCommand/push,
75
+ // 不依赖 getCurrentSessionContext。所以不注册任何全局状态。
78
76
  if (!ctx && params?.cfg) {
79
77
  try {
80
78
  const config = resolveXYConfig(params.cfg);
@@ -87,14 +85,16 @@ export const xyPlugin = {
87
85
  agentId: "default",
88
86
  isCron: true,
89
87
  };
90
- // Register so getCurrentSessionContext() fallback can find it
91
- registerSession(`__cron__${cronId}`, ctx);
88
+ logger.log(`[ALS-PROOF] agentTools ctx from ALS miss, using synthetic cron ctx sessionId=${cronId} isCron=true`);
92
89
  logger.log(`[CRON-TOOLS] Created cron session context: ${cronId}`);
93
90
  }
94
91
  catch (err) {
95
92
  logger.error("[CRON-TOOLS] Failed to create cron context:", err);
96
93
  }
97
94
  }
95
+ else {
96
+ logger.log(`[ALS-PROOF] agentTools ctx from ALS sessionId=${ctx?.sessionId} taskId=${ctx?.taskId} isCron=${ctx?.isCron === true}`);
97
+ }
98
98
  if (!ctx) {
99
99
  logger.log("[CREATE-ALL-TOOLS] no session context, returning empty tools list");
100
100
  return [];
@@ -2,6 +2,8 @@ import type { XYChannelConfig, A2ACommand } from "./types.js";
2
2
  export interface SendCommandViaPushParams {
3
3
  config: XYChannelConfig;
4
4
  command: A2ACommand;
5
+ /** 指定设备的 pushId(多设备路由)。未传时回退到 getAllPushIds()[0]。 */
6
+ pushId?: string;
5
7
  }
6
8
  /**
7
9
  * Send a tool command through the push channel (for cron-triggered tool calls).
@@ -24,16 +24,22 @@ export async function sendCommandViaPush(params) {
24
24
  command.header?.name ??
25
25
  "Command";
26
26
  logger.log(`[CRON-CMD] Sending command via push, intent=${intentName}`);
27
- // 1. Load push IDs, use first one
27
+ // 1. pushId:优先用调用方解析出的设备 pushId(多设备路由正确);
28
+ // 未提供时回退到 getAllPushIds()[0](单设备兼容旧行为)。
28
29
  let pushId = config.pushId;
29
- try {
30
- const pushIdList = await getAllPushIds();
31
- if (pushIdList.length > 0) {
32
- pushId = pushIdList[0];
33
- }
30
+ if (params.pushId) {
31
+ pushId = params.pushId;
34
32
  }
35
- catch (error) {
36
- logger.error("[CRON-CMD] Failed to load pushIds:", error);
33
+ else {
34
+ try {
35
+ const pushIdList = await getAllPushIds();
36
+ if (pushIdList.length > 0) {
37
+ pushId = pushIdList[0];
38
+ }
39
+ }
40
+ catch (error) {
41
+ logger.error("[CRON-CMD] Failed to load pushIds:", error);
42
+ }
37
43
  }
38
44
  // 2. Build and send push notification with command in directives
39
45
  const pushService = new XYPushService(config);
@@ -7,6 +7,8 @@ import { callGatewayTool } from "openclaw/plugin-sdk/agent-harness-runtime";
7
7
  import * as os from "os";
8
8
  import { sendCommand } from "./formatter.js";
9
9
  import { resolveXYConfig } from "./config.js";
10
+ import { configManager } from "./utils/config-manager.js";
11
+ import { setJobPushId } from "./utils/cron-push-map.js";
10
12
  import { logger } from "./utils/logger.js";
11
13
  import { readFileSync, readdirSync } from "fs";
12
14
  import { join } from "path";
@@ -19,7 +21,8 @@ const GATEWAY_TIMEOUT_MS = 60_000;
19
21
  */
20
22
  export async function handleCronQueryEvent(context, cfg) {
21
23
  const { action, jobId, params, sessionId, taskId, messageId } = context;
22
- logger.log(`[CRON-QUERY] Received event: action=${action}, jobId=${jobId ?? "(none)"}`);
24
+ const log = logger.withContext(sessionId ?? "", taskId ?? "");
25
+ log.log(`[CRON-QUERY] Received event: action=${action}, jobId=${jobId ?? "(none)"}`);
23
26
  let result;
24
27
  let error;
25
28
  try {
@@ -38,6 +41,11 @@ export async function handleCronQueryEvent(context, cfg) {
38
41
  break;
39
42
  case "add":
40
43
  result = await callGatewayTool("cron.add", { timeoutMs: GATEWAY_TIMEOUT_MS }, params ?? {});
44
+ // 捕获 jobId↔pushId:cron-query 路径由 channel 自己建 job,
45
+ // 此处 context 握着 sessionId,configManager 有对应设备 pushId。
46
+ await persistCronPushMap(context.sessionId, result).catch((err) => {
47
+ logger.error(`[CRON-QUERY] Failed to persist cron-push-map:`, err);
48
+ });
41
49
  break;
42
50
  case "update":
43
51
  result = await callGatewayTool("cron.update", { timeoutMs: GATEWAY_TIMEOUT_MS }, {
@@ -62,17 +70,17 @@ export async function handleCronQueryEvent(context, cfg) {
62
70
  break;
63
71
  default:
64
72
  error = `Unknown action: ${context.action}`;
65
- logger.error(`[CRON-QUERY] ${error}`);
73
+ log.error(`[CRON-QUERY] ${error}`);
66
74
  result = { error };
67
75
  }
68
76
  }
69
77
  catch (err) {
70
78
  error = err instanceof Error ? err.message : String(err);
71
- logger.error(`[CRON-QUERY] RPC call failed for action=${action}:`, err);
79
+ log.error(`[CRON-QUERY] RPC call failed for action=${action}:`, err);
72
80
  result = { error };
73
81
  }
74
82
  // Log the result
75
- logger.log(`[CRON-QUERY] RPC result for action=${action}: ${JSON.stringify(result, null, 2)}`);
83
+ log.log(`[CRON-QUERY] RPC result for action=${action}: ${JSON.stringify(result, null, 2)}`);
76
84
  // Send result back via sendCommand as System.CronQuery with payload.ans
77
85
  if (cfg && sessionId && taskId && messageId) {
78
86
  try {
@@ -93,17 +101,46 @@ export async function handleCronQueryEvent(context, cfg) {
93
101
  taskId,
94
102
  messageId,
95
103
  command,
96
- final: true,
104
+ final: sessionId.toLowerCase().endsWith("cronquery"),
97
105
  });
98
- logger.log(`[CRON-QUERY] Sent response via sendCommand, action=${action}`);
106
+ log.log(`[CRON-QUERY] Sent response via sendCommand, action=${action}`);
99
107
  }
100
108
  catch (sendErr) {
101
- logger.error(`[CRON-QUERY] Failed to send response via sendCommand:`, sendErr);
109
+ log.error(`[CRON-QUERY] Failed to send response via sendCommand:`, sendErr);
102
110
  }
103
111
  }
104
112
  else {
105
- logger.warn(`[CRON-QUERY] Missing cfg/sessionId/taskId/messageId, skipping sendCommand`);
113
+ log.warn(`[CRON-QUERY] Missing cfg/sessionId/taskId/messageId, skipping sendCommand`);
114
+ }
115
+ }
116
+ /**
117
+ * 从 cron.add 结果中提取 jobId,配合 sessionId 对应的 pushId 写入映射。
118
+ */
119
+ async function persistCronPushMap(sessionId, result) {
120
+ logger.log(`[CRONMAP] cron-query persist: sessionId=${sessionId ?? "(none)"}, resultType=${typeof result}`);
121
+ if (!sessionId) {
122
+ logger.log(`[CRONMAP] cron-query skip: no sessionId in context`);
123
+ return;
124
+ }
125
+ let jobId;
126
+ if (result && typeof result === "object") {
127
+ const id = result.id;
128
+ if (typeof id === "string" && id.trim())
129
+ jobId = id.trim();
130
+ }
131
+ if (!jobId) {
132
+ const preview = typeof result === "string" ? result.slice(0, 200) : JSON.stringify(result)?.slice(0, 200);
133
+ logger.log(`[CRONMAP] cron-query skip: no jobId in result. preview=${preview ?? "(empty)"}`);
134
+ return;
135
+ }
136
+ const pushId = configManager.getPushId(sessionId);
137
+ if (!pushId) {
138
+ logger.log(`[CRONMAP] cron-query skip: configManager has no pushId for sessionId=${sessionId}`);
139
+ return;
106
140
  }
141
+ logger.log(`[CRONMAP] cron-query writing map: jobId=${jobId}, pushId=${pushId.substring(0, 16)}...`);
142
+ await setJobPushId(jobId, { pushId, sessionId, source: "cron-query" });
143
+ logger.log(`[CRONMAP] cron-query map written OK`);
107
144
  }
108
145
  /**
109
146
  * Read local cron folder directly (bypassing openclaw RPC) and return
@@ -4,7 +4,7 @@
4
4
  import https from 'https';
5
5
  import { URL } from 'url';
6
6
  import { getConfig } from './config.js';
7
- import { DEFAULT_HTTP_PORT, HTTP_STATUS_BAD_REQUEST, API_URL_SUFFIX } from './constants.js';
7
+ import { DEFAULT_HTTPS_PORT, HTTP_STATUS_BAD_REQUEST, API_URL_SUFFIX } from './constants.js';
8
8
  function buildHeadersForCelia(config, sessionId) {
9
9
  if (!config.uid || !config.apiKey || !config.skillId || !config.requestFrom) {
10
10
  throw new Error('[SENTINEL HOOK] Missing required configuration: uid, apiKey, skillId, or requestFrom is not defined');
@@ -22,7 +22,7 @@ function buildRequestOptions(url, headers, timeout) {
22
22
  const urlObj = new URL(url);
23
23
  return {
24
24
  hostname: urlObj.hostname,
25
- port: urlObj.port || DEFAULT_HTTP_PORT,
25
+ port: urlObj.port || DEFAULT_HTTPS_PORT,
26
26
  path: urlObj.pathname,
27
27
  method: "POST",
28
28
  headers: headers,
@@ -6,7 +6,7 @@ import { callApi } from './call_api.js';
6
6
  import { processText, extractResultText, validateAndTruncateText, parseSecurityResult, handleExecToolInput, handleMessageToolInput, handleOtherToolInput } from './utils.js';
7
7
  import { ALLOWED_TOOLS, MAX_TEXT_LENGTH, MAX_TOTAL_LENGTH, MIN_TEXT_LENGTH, STEER_ABORT_MESSAGE, TOOL_OUTPUT_ACTION } from './constants.js';
8
8
  import { logger } from '../utils/logger.js';
9
- import { getSessionContext } from '../tools/session-manager.js';
9
+ import { getCurrentSessionContext } from '../tools/session-manager.js';
10
10
  import { tryInjectSteer } from './steer-context.js';
11
11
  // 主入口模块
12
12
  export default function register(api) {
@@ -78,7 +78,7 @@ export default function register(api) {
78
78
  logger.log(`[SENTINEL HOOK] TOOL_OUTPUT response: status=${result.status}.`);
79
79
  if (result.status === 'REJECT') {
80
80
  logger.warn('[SENTINEL HOOK] REJECT detected, attempting steer injection');
81
- const sessionCtx = ctx.sessionKey ? getSessionContext(ctx.sessionKey) : null;
81
+ const sessionCtx = getCurrentSessionContext();
82
82
  if (sessionCtx?.sessionId && sessionCtx?.taskId) {
83
83
  await tryInjectSteer({
84
84
  sessionId: sessionCtx.sessionId,
@@ -6,7 +6,7 @@ import path from 'path';
6
6
  import https from 'https';
7
7
  import { URL } from 'url';
8
8
  import { getConfig } from './config.js';
9
- import { DEFAULT_HTTP_PORT, DEFAULT_HTTPS_PORT, MAX_TIMES, CONNECT_TIMEOUT, READ_TIMEOUT, EXPIRE_TIME, OSMS_PREPARE_URL, OSMS_COMPLETE_URL, TEMPORARY_MATERIAL_PACKAGE, FILE_OWNER_UID, FILE_OWNER_TEAM_ID } from './constants.js';
9
+ import { DEFAULT_HTTPS_PORT, MAX_TIMES, CONNECT_TIMEOUT, READ_TIMEOUT, EXPIRE_TIME, OSMS_PREPARE_URL, OSMS_COMPLETE_URL, TEMPORARY_MATERIAL_PACKAGE, FILE_OWNER_UID, FILE_OWNER_TEAM_ID } from './constants.js';
10
10
  function buildOsmsHeaders(config, traceId) {
11
11
  return {
12
12
  'content-type': 'application/json',
@@ -22,7 +22,7 @@ function httpRequest(url, method, headers, body, timeout) {
22
22
  const urlObj = new URL(url);
23
23
  const options = {
24
24
  hostname: urlObj.hostname,
25
- port: urlObj.port || DEFAULT_HTTP_PORT,
25
+ port: urlObj.port || DEFAULT_HTTPS_PORT,
26
26
  path: urlObj.pathname + urlObj.search,
27
27
  method: method,
28
28
  headers: headers,
@@ -17,6 +17,11 @@ export declare class XYFileUploadService {
17
17
  * Uses completeAndQuery endpoint to get the file URL directly.
18
18
  */
19
19
  uploadFileAndGetUrl(filePath: string, objectType?: string): Promise<string>;
20
+ /**
21
+ * Upload a file and return a preview-able URL (needPreview=true).
22
+ * Same as uploadFileAndGetUrl but adds needPreview flag to get a directly viewable URL.
23
+ */
24
+ uploadFileAndGetPreviewUrl(filePath: string, objectType?: string): Promise<string>;
20
25
  /**
21
26
  * Upload multiple files and return their file IDs.
22
27
  */