@ynhcj/xiaoyi-channel 0.0.171-beta → 0.0.171-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.
- package/dist/index.js +141 -1
- package/dist/src/acp-session-binding.d.ts +37 -0
- package/dist/src/acp-session-binding.js +237 -0
- package/dist/src/bot.js +65 -29
- package/dist/src/channel.js +59 -64
- package/dist/src/client.d.ts +5 -0
- package/dist/src/client.js +10 -0
- package/dist/src/cron-command.d.ts +2 -0
- package/dist/src/cron-command.js +14 -8
- package/dist/src/cron-query-handler.js +45 -8
- package/dist/src/cspl/call_api.js +4 -2
- package/dist/src/cspl/sentinel_hook.js +9 -8
- package/dist/src/cspl/upload_file.js +2 -2
- package/dist/src/file-upload.js +1 -0
- package/dist/src/formatter.js +59 -25
- package/dist/src/log-reporter/config-loader.d.ts +11 -0
- package/dist/src/log-reporter/config-loader.js +68 -0
- package/dist/src/log-reporter/cursor-store.d.ts +5 -0
- package/dist/src/log-reporter/cursor-store.js +26 -0
- package/dist/src/log-reporter/index.d.ts +10 -0
- package/dist/src/log-reporter/index.js +77 -0
- package/dist/src/log-reporter/reporter.d.ts +6 -0
- package/dist/src/log-reporter/reporter.js +17 -0
- package/dist/src/log-reporter/scanner.d.ts +6 -0
- package/dist/src/log-reporter/scanner.js +82 -0
- package/dist/src/log-reporter/types.d.ts +59 -0
- package/dist/src/log-reporter/types.js +2 -0
- package/dist/src/log-reporter/uploader.d.ts +6 -0
- package/dist/src/log-reporter/uploader.js +32 -0
- package/dist/src/memory-query-handler.d.ts +1 -0
- package/dist/src/memory-query-handler.js +250 -0
- package/dist/src/monitor.js +29 -9
- package/dist/src/parser.d.ts +12 -0
- package/dist/src/parser.js +39 -13
- package/dist/src/provider.js +88 -64
- package/dist/src/reply-dispatcher.js +58 -76
- package/dist/src/task-manager.d.ts +5 -6
- package/dist/src/task-manager.js +5 -9
- package/dist/src/tools/agent-as-skill-tool.d.ts +45 -2
- package/dist/src/tools/agent-as-skill-tool.js +149 -151
- package/dist/src/tools/calendar-tool.d.ts +24 -2
- package/dist/src/tools/calendar-tool.js +115 -117
- package/dist/src/tools/call-device-tool.d.ts +20 -6
- package/dist/src/tools/call-device-tool.js +116 -138
- package/dist/src/tools/call-phone-tool.d.ts +21 -2
- package/dist/src/tools/call-phone-tool.js +112 -114
- package/dist/src/tools/check-plugin-privilege-tool.d.ts +16 -2
- package/dist/src/tools/check-plugin-privilege-tool.js +141 -143
- package/dist/src/tools/create-alarm-tool.d.ts +39 -2
- package/dist/src/tools/create-alarm-tool.js +229 -231
- package/dist/src/tools/create-all-tools.js +5 -10
- package/dist/src/tools/delete-alarm-tool.d.ts +15 -2
- package/dist/src/tools/delete-alarm-tool.js +134 -136
- package/dist/src/tools/device-tool-map.d.ts +1 -1
- package/dist/src/tools/device-tool-map.js +8 -1
- package/dist/src/tools/discover-cross-devices-tool.d.ts +16 -2
- package/dist/src/tools/discover-cross-devices-tool.js +121 -124
- package/dist/src/tools/display-a2ui-card-tool.d.ts +27 -2
- package/dist/src/tools/display-a2ui-card-tool.js +65 -68
- package/dist/src/tools/get-alarm-tool-schema.d.ts +1 -2
- package/dist/src/tools/get-alarm-tool-schema.js +10 -16
- package/dist/src/tools/get-calendar-tool-schema.d.ts +1 -2
- package/dist/src/tools/get-calendar-tool-schema.js +8 -12
- package/dist/src/tools/get-collection-tool-schema.d.ts +1 -2
- package/dist/src/tools/get-collection-tool-schema.js +9 -11
- package/dist/src/tools/get-contact-tool-schema.d.ts +1 -2
- package/dist/src/tools/get-contact-tool-schema.js +10 -16
- package/dist/src/tools/get-device-file-tool-schema.d.ts +1 -2
- package/dist/src/tools/get-device-file-tool-schema.js +9 -13
- package/dist/src/tools/get-email-tool-schema.d.ts +1 -2
- package/dist/src/tools/get-email-tool-schema.js +8 -11
- package/dist/src/tools/get-note-tool-schema.d.ts +1 -2
- package/dist/src/tools/get-note-tool-schema.js +9 -14
- package/dist/src/tools/get-photo-tool-schema.d.ts +1 -2
- package/dist/src/tools/get-photo-tool-schema.js +9 -12
- package/dist/src/tools/image-reading-tool.d.ts +28 -2
- package/dist/src/tools/image-reading-tool.js +76 -76
- package/dist/src/tools/location-tool.d.ts +11 -2
- package/dist/src/tools/location-tool.js +93 -92
- package/dist/src/tools/login-token-tool.d.ts +20 -2
- package/dist/src/tools/login-token-tool.js +122 -125
- package/dist/src/tools/modify-alarm-tool.d.ts +47 -2
- package/dist/src/tools/modify-alarm-tool.js +252 -237
- package/dist/src/tools/modify-note-tool.d.ts +20 -2
- package/dist/src/tools/modify-note-tool.js +107 -109
- package/dist/src/tools/note-tool.d.ts +20 -2
- package/dist/src/tools/note-tool.js +106 -108
- package/dist/src/tools/query-app-message-tool.d.ts +28 -2
- package/dist/src/tools/query-app-message-tool.js +111 -113
- package/dist/src/tools/query-memory-data-tool.d.ts +28 -2
- package/dist/src/tools/query-memory-data-tool.js +112 -114
- package/dist/src/tools/query-todo-task-tool.d.ts +24 -2
- package/dist/src/tools/query-todo-task-tool.js +106 -108
- package/dist/src/tools/save-file-to-phone-tool.d.ts +24 -2
- package/dist/src/tools/save-file-to-phone-tool.js +130 -132
- package/dist/src/tools/save-media-to-gallery-tool.d.ts +24 -2
- package/dist/src/tools/save-media-to-gallery-tool.js +137 -139
- package/dist/src/tools/save-self-evolution-skill-tool.d.ts +54 -2
- package/dist/src/tools/save-self-evolution-skill-tool.js +194 -194
- package/dist/src/tools/search-alarm-tool.d.ts +34 -2
- package/dist/src/tools/search-alarm-tool.js +174 -176
- package/dist/src/tools/search-calendar-tool.d.ts +24 -2
- package/dist/src/tools/search-calendar-tool.js +148 -150
- package/dist/src/tools/search-contact-tool.d.ts +16 -2
- package/dist/src/tools/search-contact-tool.js +101 -103
- package/dist/src/tools/search-email-tool.d.ts +21 -2
- package/dist/src/tools/search-email-tool.js +110 -112
- package/dist/src/tools/search-file-tool.d.ts +16 -2
- package/dist/src/tools/search-file-tool.js +104 -106
- package/dist/src/tools/search-message-tool.d.ts +16 -2
- package/dist/src/tools/search-message-tool.js +103 -105
- package/dist/src/tools/search-note-tool.d.ts +16 -2
- package/dist/src/tools/search-note-tool.js +98 -100
- package/dist/src/tools/search-photo-gallery-tool.d.ts +21 -2
- package/dist/src/tools/search-photo-gallery-tool.js +35 -37
- package/dist/src/tools/send-cross-device-task-tool.d.ts +35 -2
- package/dist/src/tools/send-cross-device-task-tool.js +214 -150
- package/dist/src/tools/send-email-tool.d.ts +24 -2
- package/dist/src/tools/send-email-tool.js +108 -110
- package/dist/src/tools/send-file-to-user-tool.d.ts +20 -2
- package/dist/src/tools/send-file-to-user-tool.js +172 -178
- package/dist/src/tools/send-html-card-tool.d.ts +20 -2
- package/dist/src/tools/send-html-card-tool.js +87 -89
- package/dist/src/tools/send-message-tool.d.ts +20 -2
- package/dist/src/tools/send-message-tool.js +122 -124
- package/dist/src/tools/session-manager.d.ts +19 -52
- package/dist/src/tools/session-manager.js +95 -239
- package/dist/src/tools/upload-file-tool.d.ts +20 -2
- package/dist/src/tools/upload-file-tool.js +80 -82
- package/dist/src/tools/upload-photo-tool.d.ts +20 -2
- package/dist/src/tools/upload-photo-tool.js +68 -70
- package/dist/src/tools/xiaoyi-add-collection-tool.d.ts +32 -2
- package/dist/src/tools/xiaoyi-add-collection-tool.js +146 -148
- package/dist/src/tools/xiaoyi-collection-tool.d.ts +20 -2
- package/dist/src/tools/xiaoyi-collection-tool.js +114 -116
- package/dist/src/tools/xiaoyi-delete-collection-tool.d.ts +15 -2
- package/dist/src/tools/xiaoyi-delete-collection-tool.js +127 -129
- package/dist/src/tools/xiaoyi-gui-tool.d.ts +16 -2
- package/dist/src/tools/xiaoyi-gui-tool.js +92 -95
- package/dist/src/types.d.ts +6 -6
- package/dist/src/utils/config-manager.d.ts +3 -2
- package/dist/src/utils/config-manager.js +22 -2
- package/dist/src/utils/cron-push-map.d.ts +26 -0
- package/dist/src/utils/cron-push-map.js +131 -0
- package/dist/src/utils/logger.js +3 -14
- package/dist/src/websocket.d.ts +1 -1
- package/dist/src/websocket.js +29 -13
- 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 || {};
|
|
@@ -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,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, extractModelName, extractTriggerData, extractRunCrossTaskContext } from "./parser.js";
|
|
4
|
+
import { parseA2AMessage, extractTextFromParts, extractFileParts, extractPushId, extractDeviceType, extractAppVer, extractDisplayVersion, 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 {
|
|
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";
|
|
@@ -32,7 +33,6 @@ export async function handleXYMessage(params) {
|
|
|
32
33
|
try {
|
|
33
34
|
// Check for special messages BEFORE parsing (these have different param structures)
|
|
34
35
|
const messageMethod = message.method;
|
|
35
|
-
logger.log(`[BOT] Received A2A message: ${JSON.stringify(message)}`);
|
|
36
36
|
// Handle clearContext messages (sessionId at top level, no params)
|
|
37
37
|
if (messageMethod === "clearContext" || messageMethod === "clear_context") {
|
|
38
38
|
const sessionId = message.sessionId ?? message.params?.sessionId;
|
|
@@ -139,6 +139,15 @@ 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
|
+
}
|
|
142
151
|
// Extract modelName if present (used by provider.ts to override model.id)
|
|
143
152
|
const modelName = extractModelName(parsed.parts);
|
|
144
153
|
if (modelName) {
|
|
@@ -160,19 +169,57 @@ export async function handleXYMessage(params) {
|
|
|
160
169
|
},
|
|
161
170
|
});
|
|
162
171
|
log.log(`[BOT] Resolved route, sessionKey=${route.sessionKey}`);
|
|
163
|
-
//
|
|
172
|
+
// ALS only: no registerSession. The sessionContext built below is handed
|
|
173
|
+
// to runWithSessionContext() inside withReplyDispatcher.run, which is the
|
|
174
|
+
// single wrap point for the whole agent turn.
|
|
164
175
|
if (!skipReg) {
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
+
// 🔑 Sync A2A modelName to OpenClaw session store so that session_status
|
|
177
|
+
// reports the correct model. Without this, session_status returns the
|
|
178
|
+
// configured default model instead of the A2A-specified one.
|
|
179
|
+
if (modelName && modelName.trim() !== "" && modelName.toLowerCase() !== "none") {
|
|
180
|
+
try {
|
|
181
|
+
const storePath = resolveStorePath();
|
|
182
|
+
const result = await updateSessionStoreEntry({
|
|
183
|
+
storePath,
|
|
184
|
+
sessionKey: route.sessionKey,
|
|
185
|
+
update: async () => ({
|
|
186
|
+
providerOverride: "xiaoyiprovider",
|
|
187
|
+
modelOverride: modelName,
|
|
188
|
+
modelOverrideSource: "user",
|
|
189
|
+
model: "",
|
|
190
|
+
modelProvider: "",
|
|
191
|
+
contextTokens: 256_000,
|
|
192
|
+
}),
|
|
193
|
+
});
|
|
194
|
+
if (!result) {
|
|
195
|
+
// Session entry doesn't exist yet (first message, xy_channel
|
|
196
|
+
// bypasses the standard turn kernel). Create a minimal entry
|
|
197
|
+
// with the override via updateSessionStore.
|
|
198
|
+
await updateSessionStore(storePath, (store) => {
|
|
199
|
+
if (!store[route.sessionKey]) {
|
|
200
|
+
store[route.sessionKey] = {
|
|
201
|
+
// sessionId must pass validateSessionId regex /^[a-z0-9][a-z0-9._-]{0,127}$/i
|
|
202
|
+
// route.sessionKey like "agent:main:direct:xxx" contains colons which are invalid.
|
|
203
|
+
// Use parsed.sessionId (raw UUID from A2A) which is always safe.
|
|
204
|
+
sessionId: parsed.sessionId,
|
|
205
|
+
updatedAt: Date.now(),
|
|
206
|
+
providerOverride: "xiaoyiprovider",
|
|
207
|
+
modelOverride: modelName,
|
|
208
|
+
modelOverrideSource: "user",
|
|
209
|
+
contextTokens: 256_000,
|
|
210
|
+
};
|
|
211
|
+
}
|
|
212
|
+
});
|
|
213
|
+
log.log(`[BOT] Created session entry with model override: xiaoyiprovider/${modelName}`);
|
|
214
|
+
}
|
|
215
|
+
else {
|
|
216
|
+
log.log(`[BOT] Patched session store model override: xiaoyiprovider/${modelName}`);
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
catch (patchErr) {
|
|
220
|
+
log.error(`[BOT] Failed to patch session model override:`, patchErr);
|
|
221
|
+
}
|
|
222
|
+
}
|
|
176
223
|
// 🔑 发送初始状态更新
|
|
177
224
|
log.log(`[BOT] Sending initial status update`);
|
|
178
225
|
void sendStatusUpdate({
|
|
@@ -333,7 +380,6 @@ export async function handleXYMessage(params) {
|
|
|
333
380
|
}
|
|
334
381
|
streamingSignals.delete(parsed.sessionId);
|
|
335
382
|
decrementTaskIdRef(parsed.sessionId);
|
|
336
|
-
unregisterSession(route.sessionKey);
|
|
337
383
|
log.log(`[BOT] Cleanup completed`);
|
|
338
384
|
},
|
|
339
385
|
run: () => {
|
|
@@ -343,6 +389,7 @@ export async function handleXYMessage(params) {
|
|
|
343
389
|
// signal init complete to release the global dispatch gate
|
|
344
390
|
// for the next session.
|
|
345
391
|
const dispatchPromise = runWithSessionContext(sessionContext, async () => {
|
|
392
|
+
log.log(`[ALS-PROOF] bot entered dispatch scope sessionId=${sessionContext.sessionId} taskId=${sessionContext.taskId} isSteer=false`);
|
|
346
393
|
log.log(`[BOT-DISPATCH] dispatchReplyFromConfig starting, body.length=${ctxPayload.Body?.length ?? 0}`);
|
|
347
394
|
try {
|
|
348
395
|
const result = await core.channel.reply.dispatchReplyFromConfig({
|
|
@@ -375,7 +422,7 @@ export async function handleXYMessage(params) {
|
|
|
375
422
|
errLog.error("Failed to handle XY message:", err);
|
|
376
423
|
runtime.error?.(`xy: Failed to handle message: ${String(err)}`);
|
|
377
424
|
errLog.log(`[BOT] Error occurred, attempting cleanup`);
|
|
378
|
-
// 🔑 错误时也要清理taskId
|
|
425
|
+
// 🔑 错误时也要清理taskId(session 走 ALS,作用域退出自动清理)
|
|
379
426
|
try {
|
|
380
427
|
const params = message.params;
|
|
381
428
|
const sessionId = params?.sessionId;
|
|
@@ -383,18 +430,6 @@ export async function handleXYMessage(params) {
|
|
|
383
430
|
errLog.log(`[BOT] Cleaning up after error`);
|
|
384
431
|
// 清理 taskId
|
|
385
432
|
decrementTaskIdRef(sessionId);
|
|
386
|
-
// 清理 session
|
|
387
|
-
const core = getXYRuntime();
|
|
388
|
-
const route = core.channel.routing.resolveAgentRoute({
|
|
389
|
-
cfg,
|
|
390
|
-
channel: "xiaoyi-channel",
|
|
391
|
-
accountId,
|
|
392
|
-
peer: {
|
|
393
|
-
kind: "direct",
|
|
394
|
-
id: sessionId,
|
|
395
|
-
},
|
|
396
|
-
});
|
|
397
|
-
unregisterSession(route.sessionKey);
|
|
398
433
|
errLog.log(`[BOT] Cleanup completed after error`);
|
|
399
434
|
}
|
|
400
435
|
}
|
|
@@ -579,6 +614,7 @@ async function dispatchSteerWhenReady(params) {
|
|
|
579
614
|
},
|
|
580
615
|
run: () => {
|
|
581
616
|
return runWithSessionContext(sessionContext, async () => {
|
|
617
|
+
log.log(`[ALS-PROOF] bot entered steer dispatch scope sessionId=${sessionContext.sessionId} taskId=${sessionContext.taskId} isSteer=true`);
|
|
582
618
|
const result = await core.channel.reply.dispatchReplyFromConfig({
|
|
583
619
|
ctx: ctxPayload,
|
|
584
620
|
cfg: params.cfg,
|