@ynhcj/xiaoyi-channel 0.0.170-beta → 0.0.170-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/bot.js +55 -28
- package/dist/src/channel.js +54 -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 +2 -2
- package/dist/src/cspl/sentinel_hook.js +2 -2
- package/dist/src/cspl/upload_file.js +2 -2
- package/dist/src/file-upload.d.ts +5 -0
- package/dist/src/file-upload.js +102 -0
- package/dist/src/formatter.d.ts +29 -0
- package/dist/src/formatter.js +106 -19
- package/dist/src/monitor.js +2 -9
- package/dist/src/parser.d.ts +1 -1
- package/dist/src/parser.js +9 -15
- package/dist/src/provider.js +67 -62
- package/dist/src/reply-dispatcher.js +57 -60
- 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 +2 -0
- 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 +25 -0
- package/dist/src/tools/send-html-card-tool.js +111 -0
- 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 +16 -53
- 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 +11 -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 || {};
|
package/dist/src/bot.js
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
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
4
|
import { parseA2AMessage, extractTextFromParts, extractFileParts, extractPushId, extractDeviceType, extractModelName, extractTriggerData, extractRunCrossTaskContext } from "./parser.js";
|
|
@@ -5,7 +6,7 @@ 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;
|
|
@@ -160,19 +160,57 @@ export async function handleXYMessage(params) {
|
|
|
160
160
|
},
|
|
161
161
|
});
|
|
162
162
|
log.log(`[BOT] Resolved route, sessionKey=${route.sessionKey}`);
|
|
163
|
-
//
|
|
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.
|
|
164
166
|
if (!skipReg) {
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
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
|
+
}
|
|
176
214
|
// 🔑 发送初始状态更新
|
|
177
215
|
log.log(`[BOT] Sending initial status update`);
|
|
178
216
|
void sendStatusUpdate({
|
|
@@ -333,7 +371,6 @@ export async function handleXYMessage(params) {
|
|
|
333
371
|
}
|
|
334
372
|
streamingSignals.delete(parsed.sessionId);
|
|
335
373
|
decrementTaskIdRef(parsed.sessionId);
|
|
336
|
-
unregisterSession(route.sessionKey);
|
|
337
374
|
log.log(`[BOT] Cleanup completed`);
|
|
338
375
|
},
|
|
339
376
|
run: () => {
|
|
@@ -343,6 +380,7 @@ export async function handleXYMessage(params) {
|
|
|
343
380
|
// signal init complete to release the global dispatch gate
|
|
344
381
|
// for the next session.
|
|
345
382
|
const dispatchPromise = runWithSessionContext(sessionContext, async () => {
|
|
383
|
+
log.log(`[ALS-PROOF] bot entered dispatch scope sessionId=${sessionContext.sessionId} taskId=${sessionContext.taskId} isSteer=false`);
|
|
346
384
|
log.log(`[BOT-DISPATCH] dispatchReplyFromConfig starting, body.length=${ctxPayload.Body?.length ?? 0}`);
|
|
347
385
|
try {
|
|
348
386
|
const result = await core.channel.reply.dispatchReplyFromConfig({
|
|
@@ -375,7 +413,7 @@ export async function handleXYMessage(params) {
|
|
|
375
413
|
errLog.error("Failed to handle XY message:", err);
|
|
376
414
|
runtime.error?.(`xy: Failed to handle message: ${String(err)}`);
|
|
377
415
|
errLog.log(`[BOT] Error occurred, attempting cleanup`);
|
|
378
|
-
// 🔑 错误时也要清理taskId
|
|
416
|
+
// 🔑 错误时也要清理taskId(session 走 ALS,作用域退出自动清理)
|
|
379
417
|
try {
|
|
380
418
|
const params = message.params;
|
|
381
419
|
const sessionId = params?.sessionId;
|
|
@@ -383,18 +421,6 @@ export async function handleXYMessage(params) {
|
|
|
383
421
|
errLog.log(`[BOT] Cleaning up after error`);
|
|
384
422
|
// 清理 taskId
|
|
385
423
|
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
424
|
errLog.log(`[BOT] Cleanup completed after error`);
|
|
399
425
|
}
|
|
400
426
|
}
|
|
@@ -579,6 +605,7 @@ async function dispatchSteerWhenReady(params) {
|
|
|
579
605
|
},
|
|
580
606
|
run: () => {
|
|
581
607
|
return runWithSessionContext(sessionContext, async () => {
|
|
608
|
+
log.log(`[ALS-PROOF] bot entered steer dispatch scope sessionId=${sessionContext.sessionId} taskId=${sessionContext.taskId} isSteer=true`);
|
|
582
609
|
const result = await core.channel.reply.dispatchReplyFromConfig({
|
|
583
610
|
ctx: ctxPayload,
|
|
584
611
|
cfg: params.cfg,
|
package/dist/src/channel.js
CHANGED
|
@@ -2,15 +2,55 @@ 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
|
|
6
|
-
import { createAllTools } from "./tools/create-all-tools.js";
|
|
5
|
+
import { getCurrentSessionContext } from "./tools/session-manager.js";
|
|
7
6
|
import { logger } from "./utils/logger.js";
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
7
|
+
// Static tool imports (3.24 pattern)
|
|
8
|
+
import { locationTool } from "./tools/location-tool.js";
|
|
9
|
+
import { xiaoyiGuiTool } from "./tools/xiaoyi-gui-tool.js";
|
|
10
|
+
import { sendFileToUserTool } from "./tools/send-file-to-user-tool.js";
|
|
11
|
+
import { sendHtmlCardTool } from "./tools/send-html-card-tool.js";
|
|
12
|
+
import { viewPushResultTool } from "./tools/view-push-result-tool.js";
|
|
13
|
+
import { imageReadingTool } from "./tools/image-reading-tool.js";
|
|
14
|
+
import { timestampToUtc8Tool } from "./tools/timestamp-to-utc8-tool.js";
|
|
15
|
+
import { saveSelfEvolutionSkillTool } from "./tools/save-self-evolution-skill-tool.js";
|
|
16
|
+
import { callDeviceTool } from "./tools/call-device-tool.js";
|
|
17
|
+
import { getNoteToolSchemaTool } from "./tools/get-note-tool-schema.js";
|
|
18
|
+
import { getCalendarToolSchemaTool } from "./tools/get-calendar-tool-schema.js";
|
|
19
|
+
import { getContactToolSchemaTool } from "./tools/get-contact-tool-schema.js";
|
|
20
|
+
import { getPhotoToolSchemaTool } from "./tools/get-photo-tool-schema.js";
|
|
21
|
+
import { getDeviceFileToolSchemaTool } from "./tools/get-device-file-tool-schema.js";
|
|
22
|
+
import { getAlarmToolSchemaTool } from "./tools/get-alarm-tool-schema.js";
|
|
23
|
+
import { getCollectionToolSchemaTool } from "./tools/get-collection-tool-schema.js";
|
|
24
|
+
import { loginTokenTool } from "./tools/login-token-tool.js";
|
|
25
|
+
import { agentAsSkillTool } from "./tools/agent-as-skill-tool.js";
|
|
26
|
+
import { discoverCrossDevicesTool } from "./tools/discover-cross-devices-tool.js";
|
|
27
|
+
import { sendCrossDeviceTaskTool } from "./tools/send-cross-device-task-tool.js";
|
|
28
|
+
import { displayA2UICardTool } from "./tools/display-a2ui-card-tool.js";
|
|
29
|
+
import { checkPluginPrivilegeTool } from "./tools/check-plugin-privilege-tool.js";
|
|
30
|
+
const ALL_TOOLS = [
|
|
31
|
+
locationTool,
|
|
32
|
+
discoverCrossDevicesTool,
|
|
33
|
+
sendCrossDeviceTaskTool,
|
|
34
|
+
displayA2UICardTool,
|
|
35
|
+
callDeviceTool,
|
|
36
|
+
getNoteToolSchemaTool,
|
|
37
|
+
getCalendarToolSchemaTool,
|
|
38
|
+
getContactToolSchemaTool,
|
|
39
|
+
getPhotoToolSchemaTool,
|
|
40
|
+
xiaoyiGuiTool,
|
|
41
|
+
getDeviceFileToolSchemaTool,
|
|
42
|
+
getAlarmToolSchemaTool,
|
|
43
|
+
getCollectionToolSchemaTool,
|
|
44
|
+
sendFileToUserTool,
|
|
45
|
+
sendHtmlCardTool,
|
|
46
|
+
viewPushResultTool,
|
|
47
|
+
imageReadingTool,
|
|
48
|
+
timestampToUtc8Tool,
|
|
49
|
+
saveSelfEvolutionSkillTool,
|
|
50
|
+
loginTokenTool,
|
|
51
|
+
agentAsSkillTool,
|
|
52
|
+
checkPluginPrivilegeTool,
|
|
53
|
+
];
|
|
14
54
|
/**
|
|
15
55
|
* Xiaoyi Channel Plugin for OpenClaw.
|
|
16
56
|
* Implements Xiaoyi A2A protocol with dual WebSocket connections.
|
|
@@ -32,7 +72,7 @@ export const xyPlugin = {
|
|
|
32
72
|
],
|
|
33
73
|
},
|
|
34
74
|
capabilities: {
|
|
35
|
-
chatTypes: ["direct"],
|
|
75
|
+
chatTypes: ["direct"],
|
|
36
76
|
polls: false,
|
|
37
77
|
threads: false,
|
|
38
78
|
media: true,
|
|
@@ -49,59 +89,11 @@ export const xyPlugin = {
|
|
|
49
89
|
schema: xyConfigSchema,
|
|
50
90
|
},
|
|
51
91
|
outbound: xyOutbound,
|
|
52
|
-
/**
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
* 1. **Normal (WebSocket) session** – `getCurrentSessionContext()` returns
|
|
58
|
-
* a context that was registered by bot.ts during message processing.
|
|
59
|
-
* Tools send commands through the WebSocket and listen for responses.
|
|
60
|
-
*
|
|
61
|
-
* 2. **Cron / scheduled-task session** – openclaw's cron runner calls
|
|
62
|
-
* `agentTools({ cfg })` without an active WebSocket session. When no
|
|
63
|
-
* session context exists but `cfg` is provided, we create a synthetic
|
|
64
|
-
* "cron session" with `isCron: true` and a `cron-`-prefixed sessionId.
|
|
65
|
-
* `sendCommand()` detects this prefix and routes commands through the
|
|
66
|
-
* push channel. Response listening (WebSocket events) works unchanged
|
|
67
|
-
* because the gateway WebSocket connection is always active.
|
|
68
|
-
*/
|
|
69
|
-
agentTools: (params) => {
|
|
70
|
-
let ctx = getCurrentSessionContext();
|
|
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.
|
|
78
|
-
if (!ctx && params?.cfg) {
|
|
79
|
-
try {
|
|
80
|
-
const config = resolveXYConfig(params.cfg);
|
|
81
|
-
const cronId = `${CRON_SESSION_PREFIX}${Date.now()}-${Math.random().toString(36).substring(2, 8)}`;
|
|
82
|
-
ctx = {
|
|
83
|
-
config,
|
|
84
|
-
sessionId: cronId,
|
|
85
|
-
taskId: cronId,
|
|
86
|
-
messageId: cronId,
|
|
87
|
-
agentId: "default",
|
|
88
|
-
isCron: true,
|
|
89
|
-
};
|
|
90
|
-
// Register so getCurrentSessionContext() fallback can find it
|
|
91
|
-
registerSession(`__cron__${cronId}`, ctx);
|
|
92
|
-
logger.log(`[CRON-TOOLS] Created cron session context: ${cronId}`);
|
|
93
|
-
}
|
|
94
|
-
catch (err) {
|
|
95
|
-
logger.error("[CRON-TOOLS] Failed to create cron context:", err);
|
|
96
|
-
}
|
|
97
|
-
}
|
|
98
|
-
if (!ctx) {
|
|
99
|
-
logger.log("[CREATE-ALL-TOOLS] no session context, returning empty tools list");
|
|
100
|
-
return [];
|
|
101
|
-
}
|
|
102
|
-
const allTools = createAllTools(ctx);
|
|
103
|
-
const filtered = filterToolsByDevice(allTools, ctx.deviceType);
|
|
104
|
-
logger.log(`[DEVICE-FILTER] deviceType=${ctx.deviceType ?? "(none)"}, tools: ${allTools.length} → ${filtered.length} (${filtered.map(t => t.name).join(", ")})`);
|
|
92
|
+
/** Static tool list (3.24 pattern). Tools read SessionContext at execute time via ALS. */
|
|
93
|
+
agentTools: () => {
|
|
94
|
+
const ctx = getCurrentSessionContext();
|
|
95
|
+
const filtered = filterToolsByDevice(ALL_TOOLS, ctx?.deviceType);
|
|
96
|
+
logger.log(`[DEVICE-FILTER] deviceType=${ctx?.deviceType ?? "(none)"}, tools: ${ALL_TOOLS.length} → ${filtered.length} (${filtered.map(t => t.name).join(", ")})`);
|
|
105
97
|
return filtered;
|
|
106
98
|
},
|
|
107
99
|
messaging: {
|
|
@@ -113,7 +105,6 @@ export const xyPlugin = {
|
|
|
113
105
|
},
|
|
114
106
|
targetResolver: {
|
|
115
107
|
looksLikeId: (raw) => {
|
|
116
|
-
// 信任所有非空字符串作为有效的 sessionId
|
|
117
108
|
const trimmed = raw.trim();
|
|
118
109
|
return trimmed.length > 0;
|
|
119
110
|
},
|
|
@@ -139,7 +130,6 @@ export const xyPlugin = {
|
|
|
139
130
|
reload: {
|
|
140
131
|
configPrefixes: ["channels.xiaoyi-channel"],
|
|
141
132
|
},
|
|
142
|
-
// Gateway adapter for receiving messages
|
|
143
133
|
gateway: {
|
|
144
134
|
async startAccount(context) {
|
|
145
135
|
const { monitorXYProvider } = await import("./monitor.js");
|
package/dist/src/client.d.ts
CHANGED
|
@@ -1,6 +1,11 @@
|
|
|
1
1
|
import { XYWebSocketManager } from "./websocket.js";
|
|
2
2
|
import type { XYChannelConfig } from "./types.js";
|
|
3
3
|
import type { RuntimeEnv } from "openclaw/plugin-sdk";
|
|
4
|
+
/**
|
|
5
|
+
* Get a cached WebSocket manager without requiring config.
|
|
6
|
+
* Returns the first available manager. Use when ALS has no SessionContext.
|
|
7
|
+
*/
|
|
8
|
+
export declare function getCachedXYWebSocketManager(): XYWebSocketManager;
|
|
4
9
|
/**
|
|
5
10
|
* Get or create a WebSocket manager for the given configuration.
|
|
6
11
|
* Reuses existing managers if config matches.
|
package/dist/src/client.js
CHANGED
|
@@ -13,6 +13,16 @@ if (!_g.__xyWsManagerCache) {
|
|
|
13
13
|
_g.__xyWsManagerCache = new Map();
|
|
14
14
|
}
|
|
15
15
|
const wsManagerCache = _g.__xyWsManagerCache;
|
|
16
|
+
/**
|
|
17
|
+
* Get a cached WebSocket manager without requiring config.
|
|
18
|
+
* Returns the first available manager. Use when ALS has no SessionContext.
|
|
19
|
+
*/
|
|
20
|
+
export function getCachedXYWebSocketManager() {
|
|
21
|
+
if (wsManagerCache.size === 0) {
|
|
22
|
+
throw new Error("No WebSocket manager available in cache");
|
|
23
|
+
}
|
|
24
|
+
return wsManagerCache.values().next().value;
|
|
25
|
+
}
|
|
16
26
|
/**
|
|
17
27
|
* Get or create a WebSocket manager for the given configuration.
|
|
18
28
|
* Reuses existing managers if config matches.
|
|
@@ -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).
|
package/dist/src/cron-command.js
CHANGED
|
@@ -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.
|
|
27
|
+
// 1. 选 pushId:优先用调用方解析出的设备 pushId(多设备路由正确);
|
|
28
|
+
// 未提供时回退到 getAllPushIds()[0](单设备兼容旧行为)。
|
|
28
29
|
let pushId = config.pushId;
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
if (pushIdList.length > 0) {
|
|
32
|
-
pushId = pushIdList[0];
|
|
33
|
-
}
|
|
30
|
+
if (params.pushId) {
|
|
31
|
+
pushId = params.pushId;
|
|
34
32
|
}
|
|
35
|
-
|
|
36
|
-
|
|
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.
|
|
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
|
-
|
|
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
|
-
|
|
79
|
+
log.error(`[CRON-QUERY] RPC call failed for action=${action}:`, err);
|
|
72
80
|
result = { error };
|
|
73
81
|
}
|
|
74
82
|
// Log the result
|
|
75
|
-
|
|
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:
|
|
104
|
+
final: sessionId.toLowerCase().endsWith("cronquery"),
|
|
97
105
|
});
|
|
98
|
-
|
|
106
|
+
log.log(`[CRON-QUERY] Sent response via sendCommand, action=${action}`);
|
|
99
107
|
}
|
|
100
108
|
catch (sendErr) {
|
|
101
|
-
|
|
109
|
+
log.error(`[CRON-QUERY] Failed to send response via sendCommand:`, sendErr);
|
|
102
110
|
}
|
|
103
111
|
}
|
|
104
112
|
else {
|
|
105
|
-
|
|
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 {
|
|
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 ||
|
|
25
|
+
port: urlObj.port || DEFAULT_HTTPS_PORT,
|
|
26
26
|
path: urlObj.pathname,
|
|
27
27
|
method: "POST",
|
|
28
28
|
headers: headers,
|