@ynhcj/xiaoyi-channel 0.0.196-beta → 0.0.198-beta

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 (37) hide show
  1. package/dist/index.js +1 -141
  2. package/dist/src/acp-session-binding.d.ts +37 -0
  3. package/dist/src/acp-session-binding.js +237 -0
  4. package/dist/src/bot.js +1 -12
  5. package/dist/src/cron-command.d.ts +0 -2
  6. package/dist/src/cron-command.js +8 -14
  7. package/dist/src/cron-query-handler.js +0 -36
  8. package/dist/src/formatter.js +5 -51
  9. package/dist/src/log-reporter/config-loader.d.ts +11 -0
  10. package/dist/src/log-reporter/config-loader.js +68 -0
  11. package/dist/src/log-reporter/cursor-store.d.ts +5 -0
  12. package/dist/src/log-reporter/cursor-store.js +26 -0
  13. package/dist/src/log-reporter/index.d.ts +10 -0
  14. package/dist/src/log-reporter/index.js +77 -0
  15. package/dist/src/log-reporter/reporter.d.ts +6 -0
  16. package/dist/src/log-reporter/reporter.js +17 -0
  17. package/dist/src/log-reporter/scanner.d.ts +6 -0
  18. package/dist/src/log-reporter/scanner.js +82 -0
  19. package/dist/src/log-reporter/types.d.ts +59 -0
  20. package/dist/src/log-reporter/types.js +2 -0
  21. package/dist/src/log-reporter/uploader.d.ts +6 -0
  22. package/dist/src/log-reporter/uploader.js +32 -0
  23. package/dist/src/memory-query-handler.js +1 -1
  24. package/dist/src/parser.d.ts +0 -12
  25. package/dist/src/parser.js +13 -39
  26. package/dist/src/provider.js +3 -30
  27. package/dist/src/reply-dispatcher.js +11 -26
  28. package/dist/src/tools/create-all-tools.js +5 -10
  29. package/dist/src/tools/send-cross-device-task-tool.js +15 -84
  30. package/dist/src/tools/send-file-to-user-tool.js +11 -9
  31. package/dist/src/tools/session-manager.d.ts +2 -12
  32. package/dist/src/tools/session-manager.js +17 -56
  33. package/dist/src/types.d.ts +6 -6
  34. package/dist/src/utils/config-manager.d.ts +2 -3
  35. package/dist/src/utils/config-manager.js +2 -22
  36. package/dist/src/websocket.js +13 -12
  37. package/package.json +1 -1
package/dist/index.js CHANGED
@@ -3,10 +3,7 @@ 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, getSessionContext } 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";
6
+ import { markCronToolCall, clearCronToolCall } from "./src/tools/session-manager.js";
10
7
  import { registerSelfEvolutionToolResultNudge } from "./src/self-evolution-tool-result-nudge.js";
11
8
  import { createBeforePromptBuildHandler } from "./src/skill-retriever/hooks.js";
12
9
  import { normalizeToolRetrieverConfig } from "./src/skill-retriever/config.js";
@@ -28,145 +25,8 @@ function registerCronDetectionHook(api) {
28
25
  if (event.toolCallId) {
29
26
  clearCronToolCall(event.toolCallId);
30
27
  }
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
- });
37
28
  });
38
29
  }
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 = ctx.sessionKey ? getSessionContext(ctx.sessionKey) : null;
58
- const sessionId = sessionCtx?.sessionId;
59
- if (!sessionId) {
60
- console.log(`[CRONMAP] skip: no sessionId (sessionKey=${ctx.sessionKey}, 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
- }
170
30
  function registerFullHooks(api) {
171
31
  // SKILL RETRIEVER HOOK: before_prompt_build hook
172
32
  const pluginConfig = api.pluginConfig || {};
@@ -0,0 +1,37 @@
1
+ import { type BindingTargetKind } from "openclaw/plugin-sdk/conversation-runtime";
2
+ type XyBindingTargetKind = "subagent" | "acp";
3
+ type XyAcpBindingRecord = {
4
+ accountId: string;
5
+ conversationId: string;
6
+ parentConversationId?: string;
7
+ deliveryTo?: string;
8
+ targetKind: XyBindingTargetKind;
9
+ targetSessionKey: string;
10
+ agentId?: string;
11
+ label?: string;
12
+ boundBy?: string;
13
+ boundAt: number;
14
+ lastActivityAt: number;
15
+ };
16
+ type XyAcpBindingManager = {
17
+ accountId: string;
18
+ getByConversationId: (conversationId: string) => XyAcpBindingRecord | undefined;
19
+ listBySessionKey: (targetSessionKey: string) => XyAcpBindingRecord[];
20
+ bindConversation: (params: {
21
+ conversationId: string;
22
+ parentConversationId?: string;
23
+ targetKind: BindingTargetKind;
24
+ targetSessionKey: string;
25
+ metadata?: Record<string, unknown>;
26
+ }) => XyAcpBindingRecord | null;
27
+ touchConversation: (conversationId: string, at?: number) => XyAcpBindingRecord | null;
28
+ unbindConversation: (conversationId: string) => XyAcpBindingRecord | null;
29
+ unbindBySessionKey: (targetSessionKey: string) => XyAcpBindingRecord[];
30
+ stop: () => void;
31
+ };
32
+ export declare function createXyAcpBindingManager(params: {
33
+ accountId?: string;
34
+ cfg: any;
35
+ }): XyAcpBindingManager;
36
+ export declare function getXyAcpBindingManager(accountId?: string): XyAcpBindingManager | null;
37
+ export {};
@@ -0,0 +1,237 @@
1
+ // ACP Session Binding Adapter for xiaoyi-channel.
2
+ // Follows the feishu thread-bindings.ts pattern.
3
+ //
4
+ // Maps A2A sessionId (stable conversation identifier) to ACP/subagent
5
+ // session keys so that openclaw can bind spawned sessions to the
6
+ // current xiaoyi conversation.
7
+ //
8
+ // Key design: xiaoyi-channel only supports `placement: "current"` —
9
+ // it cannot create child threads (unlike Discord). All spawned sessions
10
+ // are bound to the current A2A conversation identified by sessionId.
11
+ // NOTE: Using `any` for cfg type to avoid version mismatch between
12
+ // local and global openclaw installs (auth.profiles.aws-sdk union).
13
+ import { resolveThreadBindingIdleTimeoutMsForChannel, resolveThreadBindingMaxAgeMsForChannel, resolveThreadBindingConversationIdFromBindingId, registerSessionBindingAdapter, unregisterSessionBindingAdapter, } from "openclaw/plugin-sdk/conversation-runtime";
14
+ import { normalizeAccountId, resolveAgentIdFromSessionKey } from "openclaw/plugin-sdk/routing";
15
+ import { normalizeOptionalString } from "openclaw/plugin-sdk/string-coerce-runtime";
16
+ import { logger } from "./utils/logger.js";
17
+ // ─── Global state (survives module dedup) ─────────────────────
18
+ const XY_ACP_BINDINGS_KEY = Symbol.for("openclaw.xyAcpBindingsState");
19
+ let state;
20
+ function getState() {
21
+ if (!state) {
22
+ const globalStore = globalThis;
23
+ state = globalStore[XY_ACP_BINDINGS_KEY] ?? {
24
+ managersByAccountId: new Map(),
25
+ bindingsByAccountConversation: new Map(),
26
+ };
27
+ globalStore[XY_ACP_BINDINGS_KEY] = state;
28
+ }
29
+ return state;
30
+ }
31
+ function resolveBindingKey(params) {
32
+ return `${params.accountId}:${params.conversationId}`;
33
+ }
34
+ // ─── Kind conversion ──────────────────────────────────────────
35
+ function toSessionBindingTargetKind(raw) {
36
+ return raw === "subagent" ? "subagent" : "session";
37
+ }
38
+ function toXyTargetKind(raw) {
39
+ return raw === "subagent" ? "subagent" : "acp";
40
+ }
41
+ // ─── Record conversion ────────────────────────────────────────
42
+ function toSessionBindingRecord(record, defaults) {
43
+ const idleExpiresAt = defaults.idleTimeoutMs > 0 ? record.lastActivityAt + defaults.idleTimeoutMs : undefined;
44
+ const maxAgeExpiresAt = defaults.maxAgeMs > 0 ? record.boundAt + defaults.maxAgeMs : undefined;
45
+ const expiresAt = idleExpiresAt != null && maxAgeExpiresAt != null
46
+ ? Math.min(idleExpiresAt, maxAgeExpiresAt)
47
+ : (idleExpiresAt ?? maxAgeExpiresAt);
48
+ return {
49
+ bindingId: resolveBindingKey({
50
+ accountId: record.accountId,
51
+ conversationId: record.conversationId,
52
+ }),
53
+ targetSessionKey: record.targetSessionKey,
54
+ targetKind: toSessionBindingTargetKind(record.targetKind),
55
+ conversation: {
56
+ channel: "xiaoyi-channel",
57
+ accountId: record.accountId,
58
+ conversationId: record.conversationId,
59
+ parentConversationId: record.parentConversationId,
60
+ },
61
+ status: "active",
62
+ boundAt: record.boundAt,
63
+ expiresAt,
64
+ metadata: {
65
+ agentId: record.agentId,
66
+ label: record.label,
67
+ boundBy: record.boundBy,
68
+ deliveryTo: record.deliveryTo,
69
+ lastActivityAt: record.lastActivityAt,
70
+ idleTimeoutMs: defaults.idleTimeoutMs,
71
+ maxAgeMs: defaults.maxAgeMs,
72
+ },
73
+ };
74
+ }
75
+ // ─── Manager factory ──────────────────────────────────────────
76
+ export function createXyAcpBindingManager(params) {
77
+ const accountId = normalizeAccountId(params.accountId);
78
+ const existing = getState().managersByAccountId.get(accountId);
79
+ if (existing) {
80
+ return existing;
81
+ }
82
+ const idleTimeoutMs = resolveThreadBindingIdleTimeoutMsForChannel({
83
+ cfg: params.cfg,
84
+ channel: "xiaoyi-channel",
85
+ accountId,
86
+ });
87
+ const maxAgeMs = resolveThreadBindingMaxAgeMsForChannel({
88
+ cfg: params.cfg,
89
+ channel: "xiaoyi-channel",
90
+ accountId,
91
+ });
92
+ const log = logger.withContext("", "");
93
+ const manager = {
94
+ accountId,
95
+ getByConversationId: (conversationId) => getState().bindingsByAccountConversation.get(resolveBindingKey({ accountId, conversationId })),
96
+ listBySessionKey: (targetSessionKey) => [...getState().bindingsByAccountConversation.values()].filter((record) => record.accountId === accountId && record.targetSessionKey === targetSessionKey),
97
+ bindConversation: ({ conversationId, parentConversationId, targetKind, targetSessionKey, metadata, }) => {
98
+ const normalizedConversationId = conversationId.trim();
99
+ const normalizedTargetSessionKey = targetSessionKey.trim();
100
+ if (!normalizedConversationId || !normalizedTargetSessionKey) {
101
+ return null;
102
+ }
103
+ const existingLocal = getState().bindingsByAccountConversation.get(resolveBindingKey({ accountId, conversationId: normalizedConversationId }));
104
+ const now = Date.now();
105
+ const record = {
106
+ accountId,
107
+ conversationId: normalizedConversationId,
108
+ parentConversationId: normalizeOptionalString(parentConversationId) ?? existingLocal?.parentConversationId,
109
+ deliveryTo: typeof metadata?.deliveryTo === "string" && metadata.deliveryTo.trim()
110
+ ? metadata.deliveryTo.trim()
111
+ : existingLocal?.deliveryTo,
112
+ targetKind: toXyTargetKind(targetKind),
113
+ targetSessionKey: normalizedTargetSessionKey,
114
+ agentId: typeof metadata?.agentId === "string" && metadata.agentId.trim()
115
+ ? metadata.agentId.trim()
116
+ : (existingLocal?.agentId ?? resolveAgentIdFromSessionKey(normalizedTargetSessionKey)),
117
+ label: typeof metadata?.label === "string" && metadata.label.trim()
118
+ ? metadata.label.trim()
119
+ : existingLocal?.label,
120
+ boundBy: typeof metadata?.boundBy === "string" && metadata.boundBy.trim()
121
+ ? metadata.boundBy.trim()
122
+ : existingLocal?.boundBy,
123
+ boundAt: now,
124
+ lastActivityAt: now,
125
+ };
126
+ getState().bindingsByAccountConversation.set(resolveBindingKey({ accountId, conversationId: normalizedConversationId }), record);
127
+ log.log(`[XY-ACP-BIND] Bound ${targetKind} session ${normalizedTargetSessionKey.slice(0, 30)} to conversation ${normalizedConversationId.slice(0, 12)}`);
128
+ return record;
129
+ },
130
+ touchConversation: (conversationId, at = Date.now()) => {
131
+ const key = resolveBindingKey({ accountId, conversationId });
132
+ const existingRecord = getState().bindingsByAccountConversation.get(key);
133
+ if (!existingRecord) {
134
+ return null;
135
+ }
136
+ const updated = { ...existingRecord, lastActivityAt: at };
137
+ getState().bindingsByAccountConversation.set(key, updated);
138
+ return updated;
139
+ },
140
+ unbindConversation: (conversationId) => {
141
+ const key = resolveBindingKey({ accountId, conversationId });
142
+ const existingRecord = getState().bindingsByAccountConversation.get(key);
143
+ if (!existingRecord) {
144
+ return null;
145
+ }
146
+ getState().bindingsByAccountConversation.delete(key);
147
+ return existingRecord;
148
+ },
149
+ unbindBySessionKey: (targetSessionKey) => {
150
+ const removed = [];
151
+ for (const record of getState().bindingsByAccountConversation.values()) {
152
+ if (record.accountId !== accountId || record.targetSessionKey !== targetSessionKey) {
153
+ continue;
154
+ }
155
+ getState().bindingsByAccountConversation.delete(resolveBindingKey({ accountId, conversationId: record.conversationId }));
156
+ removed.push(record);
157
+ }
158
+ return removed;
159
+ },
160
+ stop: () => {
161
+ for (const key of getState().bindingsByAccountConversation.keys()) {
162
+ if (key.startsWith(`${accountId}:`)) {
163
+ getState().bindingsByAccountConversation.delete(key);
164
+ }
165
+ }
166
+ getState().managersByAccountId.delete(accountId);
167
+ unregisterSessionBindingAdapter({
168
+ channel: "xiaoyi-channel",
169
+ accountId,
170
+ adapter: sessionBindingAdapter,
171
+ });
172
+ log.log(`[XY-ACP-BIND] Stopped binding manager for account ${accountId}`);
173
+ },
174
+ };
175
+ const sessionBindingAdapter = {
176
+ channel: "xiaoyi-channel",
177
+ accountId,
178
+ capabilities: {
179
+ placements: ["current"],
180
+ },
181
+ bind: async (input) => {
182
+ if (input.conversation.channel !== "xiaoyi-channel" || input.placement === "child") {
183
+ return null;
184
+ }
185
+ const bound = manager.bindConversation({
186
+ conversationId: input.conversation.conversationId,
187
+ parentConversationId: input.conversation.parentConversationId,
188
+ targetKind: input.targetKind,
189
+ targetSessionKey: input.targetSessionKey,
190
+ metadata: input.metadata,
191
+ });
192
+ return bound ? toSessionBindingRecord(bound, { idleTimeoutMs, maxAgeMs }) : null;
193
+ },
194
+ listBySession: (targetSessionKey) => manager
195
+ .listBySessionKey(targetSessionKey)
196
+ .map((entry) => toSessionBindingRecord(entry, { idleTimeoutMs, maxAgeMs })),
197
+ resolveByConversation: (ref) => {
198
+ if (ref.channel !== "xiaoyi-channel") {
199
+ return null;
200
+ }
201
+ const found = manager.getByConversationId(ref.conversationId);
202
+ return found ? toSessionBindingRecord(found, { idleTimeoutMs, maxAgeMs }) : null;
203
+ },
204
+ touch: (bindingId, at) => {
205
+ const conversationId = resolveThreadBindingConversationIdFromBindingId({
206
+ accountId,
207
+ bindingId,
208
+ });
209
+ if (conversationId) {
210
+ manager.touchConversation(conversationId, at);
211
+ }
212
+ },
213
+ unbind: async (input) => {
214
+ if (input.targetSessionKey?.trim()) {
215
+ return manager
216
+ .unbindBySessionKey(input.targetSessionKey.trim())
217
+ .map((entry) => toSessionBindingRecord(entry, { idleTimeoutMs, maxAgeMs }));
218
+ }
219
+ const conversationId = resolveThreadBindingConversationIdFromBindingId({
220
+ accountId,
221
+ bindingId: input.bindingId,
222
+ });
223
+ if (!conversationId) {
224
+ return [];
225
+ }
226
+ const removed = manager.unbindConversation(conversationId);
227
+ return removed ? [toSessionBindingRecord(removed, { idleTimeoutMs, maxAgeMs })] : [];
228
+ },
229
+ };
230
+ registerSessionBindingAdapter(sessionBindingAdapter);
231
+ getState().managersByAccountId.set(accountId, manager);
232
+ log.log(`[XY-ACP-BIND] Created binding manager for account ${accountId} (idleTimeout=${idleTimeoutMs}ms, maxAge=${maxAgeMs}ms)`);
233
+ return manager;
234
+ }
235
+ export function getXyAcpBindingManager(accountId) {
236
+ return getState().managersByAccountId.get(normalizeAccountId(accountId)) ?? null;
237
+ }
package/dist/src/bot.js CHANGED
@@ -1,7 +1,7 @@
1
1
  import { updateSessionStoreEntry, updateSessionStore, resolveStorePath } from "openclaw/plugin-sdk/session-store-runtime";
2
2
  import { getXYRuntime } from "./runtime.js";
3
3
  import { createXYReplyDispatcher } from "./reply-dispatcher.js";
4
- import { parseA2AMessage, extractTextFromParts, extractFileParts, extractPushId, extractDeviceType, extractAppVer, extractDisplayVersion, extractModelName, extractTriggerData, extractRunCrossTaskContext } from "./parser.js";
4
+ import { parseA2AMessage, extractTextFromParts, extractFileParts, extractPushId, extractDeviceType, extractModelName, extractTriggerData, extractRunCrossTaskContext } from "./parser.js";
5
5
  import { downloadFilesFromParts } from "./file-download.js";
6
6
  import { resolveXYConfig } from "./config.js";
7
7
  import { sendStatusUpdate, sendClearContextResponse, sendTasksCancelResponse, sendA2AResponse } from "./formatter.js";
@@ -139,15 +139,6 @@ export async function handleXYMessage(params) {
139
139
  if (deviceType) {
140
140
  log.log(`[BOT] Extracted deviceType: ${deviceType}`);
141
141
  }
142
- // Extract app_ver and display_version if present
143
- const appVer = extractAppVer(parsed.parts);
144
- if (appVer) {
145
- log.log(`[BOT] Extracted app_ver: ${appVer}`);
146
- }
147
- const displayVersion = extractDisplayVersion(parsed.parts);
148
- if (displayVersion) {
149
- log.log(`[BOT] Extracted display_version: ${displayVersion}`);
150
- }
151
142
  // Extract modelName if present (used by provider.ts to override model.id)
152
143
  const modelName = extractModelName(parsed.parts);
153
144
  if (modelName) {
@@ -179,8 +170,6 @@ export async function handleXYMessage(params) {
179
170
  messageId: parsed.messageId,
180
171
  agentId: route.accountId,
181
172
  deviceType,
182
- appVer: appVer ?? undefined,
183
- displayVersion: displayVersion ?? undefined,
184
173
  modelName,
185
174
  runCrossTaskContext: runCrossTaskContext ?? undefined,
186
175
  });
@@ -2,8 +2,6 @@ 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;
7
5
  }
8
6
  /**
9
7
  * Send a tool command through the push channel (for cron-triggered tool calls).
@@ -24,23 +24,17 @@ 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. pushId:优先用调用方解析出的设备 pushId(多设备路由正确);
28
- // 未提供时回退到 getAllPushIds()[0](单设备兼容旧行为)。
27
+ // 1. Load push IDs, use first one
29
28
  let pushId = config.pushId;
30
- if (params.pushId) {
31
- pushId = params.pushId;
32
- }
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);
29
+ try {
30
+ const pushIdList = await getAllPushIds();
31
+ if (pushIdList.length > 0) {
32
+ pushId = pushIdList[0];
42
33
  }
43
34
  }
35
+ catch (error) {
36
+ logger.error("[CRON-CMD] Failed to load pushIds:", error);
37
+ }
44
38
  // 2. Build and send push notification with command in directives
45
39
  const pushService = new XYPushService(config);
46
40
  const sessionId = randomUUID();
@@ -7,8 +7,6 @@ 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";
12
10
  import { logger } from "./utils/logger.js";
13
11
  import { readFileSync, readdirSync } from "fs";
14
12
  import { join } from "path";
@@ -41,11 +39,6 @@ export async function handleCronQueryEvent(context, cfg) {
41
39
  break;
42
40
  case "add":
43
41
  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
- });
49
42
  break;
50
43
  case "update":
51
44
  result = await callGatewayTool("cron.update", { timeoutMs: GATEWAY_TIMEOUT_MS }, {
@@ -113,35 +106,6 @@ export async function handleCronQueryEvent(context, cfg) {
113
106
  log.warn(`[CRON-QUERY] Missing cfg/sessionId/taskId/messageId, skipping sendCommand`);
114
107
  }
115
108
  }
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;
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`);
144
- }
145
109
  /**
146
110
  * Read local cron folder directly (bypassing openclaw RPC) and return
147
111
  * run records from the last 7 days, grouped by date and sorted by time.
@@ -5,10 +5,7 @@ import { logger } from "./utils/logger.js";
5
5
  import { getCurrentTaskId, getCurrentMessageId } from "./task-manager.js";
6
6
  import { redactSensitiveText, containsSensitiveInfo } from "./sensitive-redactor.js";
7
7
  import { rewriteOutboundApprovalText } from "./approval-bridge.js";
8
- import { isCronToolCall, getCurrentCronJobId } from "./tools/session-manager.js";
9
- import { configManager } from "./utils/config-manager.js";
10
- import { getPushIdByJobId } from "./utils/cron-push-map.js";
11
- import { getAllPushIds } from "./utils/pushid-manager.js";
8
+ import { isCronToolCall } from "./tools/session-manager.js";
12
9
  // ─────────────────────────────────────────────────────────────
13
10
  // 敏感信息脱敏辅助函数
14
11
  // ─────────────────────────────────────────────────────────────
@@ -204,45 +201,6 @@ export async function sendStatusUpdate(params) {
204
201
  log.log(`[A2A_STATUS] Sending status-update, text="${redactedText}"`);
205
202
  await wsManager.sendMessage(sessionId, outboundMessage);
206
203
  }
207
- /**
208
- * 解析 cron fire 时应使用的 pushId(多设备路由)。
209
- *
210
- * 查询链(逐级回退):
211
- * 1. 合成 sessionId → jobId → cron-push-map.json → 创建时记录的设备 pushId
212
- * 2. configManager 同进程的 sessionId→pushId(进程未重启时兜底)
213
- * 3. getAllPushIds()[0](单设备兼容旧行为)
214
- * 返回 undefined 表示走兜底(由 sendCommandViaPush 内部处理)。
215
- */
216
- async function resolveCronPushId(sessionId, config) {
217
- // 1. jobId → 持久化映射
218
- const jobId = getCurrentCronJobId(sessionId);
219
- if (jobId) {
220
- const hit = await getPushIdByJobId(jobId);
221
- if (hit?.pushId) {
222
- logger.log(`[CRON-PUSH] Resolved pushId via map, jobId=${jobId}`);
223
- return hit.pushId;
224
- }
225
- }
226
- // 2. 同进程 configManager 兜底
227
- const sessionPushId = configManager.getPushId(sessionId);
228
- if (sessionPushId) {
229
- logger.log(`[CRON-PUSH] Resolved pushId via configManager (fallback)`);
230
- return sessionPushId;
231
- }
232
- // 3. config.pushId / getAllPushIds()[0] 交给 sendCommandViaPush 内部处理
233
- void config;
234
- try {
235
- const all = await getAllPushIds();
236
- if (all.length > 0) {
237
- logger.log(`[CRON-PUSH] Resolved pushId via getAllPushIds[0] (legacy fallback)`);
238
- return all[0];
239
- }
240
- }
241
- catch (error) {
242
- logger.error(`[CRON-PUSH] getAllPushIds failed:`, error);
243
- }
244
- return undefined;
245
- }
246
204
  /**
247
205
  * Send a command as an artifact update (final=false).
248
206
  *
@@ -259,15 +217,11 @@ export async function sendCommand(params) {
259
217
  if (commands.length === 0) {
260
218
  throw new Error("sendCommand requires command or commands.");
261
219
  }
262
- // ── Cron mode: route through push channel ──────────────────────
263
- // Detected via: (a) sessionId "cron-" prefix from synthetic session, OR
264
- // (b) toolCallId marked by before_tool_call hook from openclaw's sessionKey.
220
+ // ── Cron mode: disabled ────────────────────────────────────────
221
+ // sendCommandViaPush is disabled in this version. Cron-triggered
222
+ // tool calls that try to send commands will be rejected.
265
223
  if (sessionId.startsWith("cron-") || isCronToolCall(toolCallId)) {
266
- const { sendCommandViaPush } = await import("./cron-command.js");
267
- // 解析正确设备的 pushId:合成 sessionId → jobId → cron-push-map。
268
- // provider.ts 在 isCron 分支已把 jobId 绑定到该 sessionId。
269
- const pushId = await resolveCronPushId(sessionId, config);
270
- return sendCommandViaPush({ config, command: commands[0], pushId });
224
+ throw new Error("sendCommandViaPush is disabled in this version");
271
225
  }
272
226
  // ── Normal mode: WebSocket ─────────────────────────────────────
273
227
  // Dynamic lookup: use latest taskId/messageId from task-manager (handles steer/interrupt),
@@ -0,0 +1,11 @@
1
+ import type { LogReporterConfig } from "./types.js";
2
+ /**
3
+ * Load and validate the JSON config file.
4
+ * Falls back to defaults for optional fields.
5
+ */
6
+ export declare function loadConfig(configPath: string): LogReporterConfig;
7
+ /**
8
+ * Resolve a path template with date wildcards to actual file paths on disk.
9
+ * Scans the directory and returns all files matching the pattern.
10
+ */
11
+ export declare function resolveLogFiles(templatePath: string): string[];