@ynhcj/xiaoyi-channel 0.0.90-beta → 0.0.92-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.
- package/dist/src/channel.js +5 -3
- package/dist/src/provider.js +96 -10
- package/dist/src/tools/call-device-tool.js +4 -0
- package/dist/src/tools/get-email-tool-schema.d.ts +16 -0
- package/dist/src/tools/get-email-tool-schema.js +9 -0
- package/dist/src/tools/query-app-message-tool.d.ts +4 -0
- package/dist/src/tools/query-app-message-tool.js +138 -0
- package/dist/src/tools/query-memory-data-tool.d.ts +4 -0
- package/dist/src/tools/query-memory-data-tool.js +154 -0
- package/dist/src/tools/query-todo-task-tool.d.ts +4 -0
- package/dist/src/tools/query-todo-task-tool.js +133 -0
- package/package.json +1 -1
package/dist/src/channel.js
CHANGED
|
@@ -7,8 +7,7 @@ import { sendFileToUserTool } from "./tools/send-file-to-user-tool.js";
|
|
|
7
7
|
import { viewPushResultTool } from "./tools/view-push-result-tool.js";
|
|
8
8
|
import { imageReadingTool } from "./tools/image-reading-tool.js";
|
|
9
9
|
import { timestampToUtc8Tool } from "./tools/timestamp-to-utc8-tool.js";
|
|
10
|
-
import {
|
|
11
|
-
import { searchEmailTool } from "./tools/search-email-tool.js";
|
|
10
|
+
import { getEmailToolSchemaTool } from "./tools/get-email-tool-schema.js";
|
|
12
11
|
import { callDeviceTool } from "./tools/call-device-tool.js";
|
|
13
12
|
import { getNoteToolSchemaTool } from "./tools/get-note-tool-schema.js";
|
|
14
13
|
import { getCalendarToolSchemaTool } from "./tools/get-calendar-tool-schema.js";
|
|
@@ -17,6 +16,9 @@ import { getPhotoToolSchemaTool } from "./tools/get-photo-tool-schema.js";
|
|
|
17
16
|
import { getDeviceFileToolSchemaTool } from "./tools/get-device-file-tool-schema.js";
|
|
18
17
|
import { getAlarmToolSchemaTool } from "./tools/get-alarm-tool-schema.js";
|
|
19
18
|
import { getCollectionToolSchemaTool } from "./tools/get-collection-tool-schema.js";
|
|
19
|
+
import { queryAppMessageTool } from "./tools/query-app-message-tool.js";
|
|
20
|
+
import { queryMemoryDataTool } from "./tools/query-memory-data-tool.js";
|
|
21
|
+
import { queryTodoTaskTool } from "./tools/query-todo-task-tool.js";
|
|
20
22
|
import { filterToolsByDevice } from "./tools/device-tool-map.js";
|
|
21
23
|
import { getCurrentSessionContext } from "./tools/session-manager.js";
|
|
22
24
|
import { logger } from "./utils/logger.js";
|
|
@@ -59,7 +61,7 @@ export const xyPlugin = {
|
|
|
59
61
|
},
|
|
60
62
|
outbound: xyOutbound,
|
|
61
63
|
agentTools: () => {
|
|
62
|
-
const allTools = [locationTool, callDeviceTool, getNoteToolSchemaTool, getCalendarToolSchemaTool, getContactToolSchemaTool, getPhotoToolSchemaTool, xiaoyiGuiTool, getDeviceFileToolSchemaTool, getAlarmToolSchemaTool, getCollectionToolSchemaTool, sendFileToUserTool, viewPushResultTool, imageReadingTool, timestampToUtc8Tool,
|
|
64
|
+
const allTools = [locationTool, callDeviceTool, getNoteToolSchemaTool, getCalendarToolSchemaTool, getContactToolSchemaTool, getPhotoToolSchemaTool, xiaoyiGuiTool, getDeviceFileToolSchemaTool, getAlarmToolSchemaTool, getCollectionToolSchemaTool, sendFileToUserTool, viewPushResultTool, imageReadingTool, timestampToUtc8Tool, getEmailToolSchemaTool, queryAppMessageTool, queryMemoryDataTool, queryTodoTaskTool];
|
|
63
65
|
const ctx = getCurrentSessionContext();
|
|
64
66
|
const filtered = filterToolsByDevice(allTools, ctx?.deviceType);
|
|
65
67
|
logger.log(`[DEVICE-FILTER] deviceType=${ctx?.deviceType ?? "(none)"}, tools: ${allTools.length} → ${filtered.length} (${filtered.map(t => t.name).join(", ")})`);
|
package/dist/src/provider.js
CHANGED
|
@@ -9,6 +9,60 @@
|
|
|
9
9
|
// models.providers.xiaoyiprovider.models = [...]
|
|
10
10
|
import { createHash } from "crypto";
|
|
11
11
|
import { getCurrentSessionContext } from "./tools/session-manager.js";
|
|
12
|
+
// ── Retry config ──────────────────────────────────────────────
|
|
13
|
+
const RETRY_DELAYS_MS = [10_000, 20_000, 40_000, 60_000];
|
|
14
|
+
const MAX_RETRY_ATTEMPTS = 8;
|
|
15
|
+
/** Check if an errorMessage indicates a retryable provider error by type. */
|
|
16
|
+
function isRetryableProviderError(message) {
|
|
17
|
+
if (!message)
|
|
18
|
+
return false;
|
|
19
|
+
const lower = message.toLowerCase();
|
|
20
|
+
if (lower.includes("server_error"))
|
|
21
|
+
return true;
|
|
22
|
+
if (lower.includes("rate_limit_error"))
|
|
23
|
+
return true;
|
|
24
|
+
return false;
|
|
25
|
+
}
|
|
26
|
+
/** Compute retry delay in ms for the given 1-based attempt. */
|
|
27
|
+
function getRetryDelayMs(attempt) {
|
|
28
|
+
// attempt 1→10s, 2→20s, 3→40s, 4+→60s
|
|
29
|
+
if (attempt <= RETRY_DELAYS_MS.length)
|
|
30
|
+
return RETRY_DELAYS_MS[attempt - 1];
|
|
31
|
+
return RETRY_DELAYS_MS[RETRY_DELAYS_MS.length - 1]; // 60s cap
|
|
32
|
+
}
|
|
33
|
+
function sleep(ms) {
|
|
34
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Build a minimal EventStream-compatible object that replays a single
|
|
38
|
+
* done/error event. This avoids importing @mariozechner/pi-ai at runtime
|
|
39
|
+
* (the package is not available in the extension sandbox).
|
|
40
|
+
*/
|
|
41
|
+
function buildReplayStream(result) {
|
|
42
|
+
let settled = false;
|
|
43
|
+
const queued = [
|
|
44
|
+
result.stopReason === "error"
|
|
45
|
+
? { type: "error", reason: "error", error: result }
|
|
46
|
+
: { type: "done", reason: result.stopReason, message: result },
|
|
47
|
+
];
|
|
48
|
+
return {
|
|
49
|
+
result: () => Promise.resolve(result),
|
|
50
|
+
push: () => { },
|
|
51
|
+
end: () => { },
|
|
52
|
+
[Symbol.asyncIterator]: () => {
|
|
53
|
+
return {
|
|
54
|
+
next: async () => {
|
|
55
|
+
if (settled || queued.length === 0) {
|
|
56
|
+
settled = true;
|
|
57
|
+
return { value: undefined, done: true };
|
|
58
|
+
}
|
|
59
|
+
settled = true;
|
|
60
|
+
return { value: queued.shift(), done: false };
|
|
61
|
+
},
|
|
62
|
+
};
|
|
63
|
+
},
|
|
64
|
+
};
|
|
65
|
+
}
|
|
12
66
|
/**
|
|
13
67
|
* Dynamic header keys injected via extraParams and forwarded to the HTTP request.
|
|
14
68
|
* Correspond to the three fields written to .xiaoyiruntime:
|
|
@@ -73,10 +127,12 @@ export const xiaoyiProvider = {
|
|
|
73
127
|
},
|
|
74
128
|
/**
|
|
75
129
|
* Wrap the stream function to inject dynamic headers into every
|
|
76
|
-
* HTTP request to the model provider
|
|
130
|
+
* HTTP request to the model provider, and retry on retryable errors
|
|
131
|
+
* (server_error / rate_limit_error) with backoff: 10s, 20s, 40s, 60s (cap).
|
|
77
132
|
*
|
|
78
|
-
*
|
|
79
|
-
*
|
|
133
|
+
* The retry loop awaits stream.result() to detect errors before deciding
|
|
134
|
+
* whether to retry. This keeps the agent loop waiting (no timeout risk
|
|
135
|
+
* since the default agent timeout is 48 hours).
|
|
80
136
|
*/
|
|
81
137
|
wrapStreamFn: (ctx) => {
|
|
82
138
|
const underlying = ctx.streamFn;
|
|
@@ -135,21 +191,51 @@ export const xiaoyiProvider = {
|
|
|
135
191
|
if (sessionCtx?.deviceType) {
|
|
136
192
|
const rawDevice = sessionCtx.deviceType;
|
|
137
193
|
const displayDevice = (rawDevice === "2in1") ? "鸿蒙PC" : rawDevice;
|
|
138
|
-
const deviceSection = `\n\n## Current User Device Context\nThe current user is using the following device: ${displayDevice}\nYou need to be aware of the user
|
|
194
|
+
const deviceSection = `\n\n## Current User Device Context\nThe current user is using the following device: ${displayDevice}\nYou need to be aware of the user's current device and provide guidance accordingly. If the response involves device-related tools or actions, you must tailor the reply based on the user's current device, using device-specific references such as "saved to the Notes/Calendar on your {deviceType}.\n"`;
|
|
139
195
|
context.systemPrompt = (context.systemPrompt ?? "") + deviceSection;
|
|
140
196
|
}
|
|
141
|
-
|
|
197
|
+
// ── Retry loop ─────────────────────────────────────────
|
|
198
|
+
for (let attempt = 0; attempt < MAX_RETRY_ATTEMPTS; attempt++) {
|
|
199
|
+
const stream = await underlying(model, context, {
|
|
200
|
+
...options,
|
|
201
|
+
headers: {
|
|
202
|
+
...options?.headers,
|
|
203
|
+
...dynamicHeaders,
|
|
204
|
+
},
|
|
205
|
+
});
|
|
206
|
+
// Wait for the stream to settle (done or error) to inspect the result.
|
|
207
|
+
// stream.result() resolves to the final AssistantMessage (even on error).
|
|
208
|
+
const result = await stream.result();
|
|
209
|
+
// Check if this is a retryable error
|
|
210
|
+
if (result.stopReason === "error" && isRetryableProviderError(result.errorMessage)) {
|
|
211
|
+
const delayMs = getRetryDelayMs(attempt);
|
|
212
|
+
console.log(`[xiaoyiprovider] retryable error (attempt ${attempt + 1}/${MAX_RETRY_ATTEMPTS}): ` +
|
|
213
|
+
`${result.errorMessage} — retrying in ${delayMs}ms`);
|
|
214
|
+
await sleep(delayMs);
|
|
215
|
+
continue;
|
|
216
|
+
}
|
|
217
|
+
// Success or non-retryable error — log and return
|
|
218
|
+
if (result.stopReason === "error") {
|
|
219
|
+
console.log(`[xiaoyiprovider] non-retryable error: ${result.errorMessage}`);
|
|
220
|
+
}
|
|
221
|
+
else {
|
|
222
|
+
console.log(`[xiaoyiprovider] stream completed, usage: input=${result.usage?.input} output=${result.usage?.output}`);
|
|
223
|
+
}
|
|
224
|
+
// The original stream has already been consumed by result().
|
|
225
|
+
// Build a replay stream that delivers the final result.
|
|
226
|
+
return buildReplayStream(result);
|
|
227
|
+
}
|
|
228
|
+
// All retries exhausted — return the last attempt's real error via a new stream
|
|
229
|
+
console.log(`[xiaoyiprovider] all ${MAX_RETRY_ATTEMPTS} retries exhausted, surfacing last error`);
|
|
230
|
+
const lastStream = await underlying(model, context, {
|
|
142
231
|
...options,
|
|
143
232
|
headers: {
|
|
144
233
|
...options?.headers,
|
|
145
234
|
...dynamicHeaders,
|
|
146
235
|
},
|
|
147
236
|
});
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
console.log(`[xiaoyiprovider] stream completed, usage: input=${result.usage?.input} output=${result.usage?.output}`);
|
|
151
|
-
}, (err) => console.log(`[xiaoyiprovider] stream error: ${JSON.stringify(err)}`));
|
|
152
|
-
return stream;
|
|
237
|
+
const lastResult = await lastStream.result();
|
|
238
|
+
return buildReplayStream(lastResult);
|
|
153
239
|
};
|
|
154
240
|
},
|
|
155
241
|
};
|
|
@@ -20,6 +20,8 @@ import { saveMediaToGalleryTool } from "./save-media-to-gallery-tool.js";
|
|
|
20
20
|
import { searchFileTool } from "./search-file-tool.js";
|
|
21
21
|
import { uploadFileTool } from "./upload-file-tool.js";
|
|
22
22
|
import { saveFileToPhoneTool } from "./save-file-to-phone-tool.js";
|
|
23
|
+
import { sendEmailTool } from "./send-email-tool.js";
|
|
24
|
+
import { searchEmailTool } from "./search-email-tool.js";
|
|
23
25
|
import { sendStatusUpdate } from "../formatter.js";
|
|
24
26
|
import { getCurrentSessionContext } from "./session-manager.js";
|
|
25
27
|
import { getCurrentTaskId, getCurrentMessageId } from "../task-manager.js";
|
|
@@ -49,6 +51,8 @@ const deviceToolRegistry = new Map([
|
|
|
49
51
|
[searchFileTool.name, searchFileTool],
|
|
50
52
|
[uploadFileTool.name, uploadFileTool],
|
|
51
53
|
[saveFileToPhoneTool.name, saveFileToPhoneTool],
|
|
54
|
+
[sendEmailTool.name, sendEmailTool],
|
|
55
|
+
[searchEmailTool.name, searchEmailTool],
|
|
52
56
|
]);
|
|
53
57
|
/**
|
|
54
58
|
* call_device_tool - 通用端工具调度器。
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
export declare const getEmailToolSchemaTool: {
|
|
2
|
+
name: string;
|
|
3
|
+
label: string;
|
|
4
|
+
description: string;
|
|
5
|
+
parameters: {
|
|
6
|
+
type: "object";
|
|
7
|
+
properties: {};
|
|
8
|
+
required: string[];
|
|
9
|
+
};
|
|
10
|
+
execute(_toolCallId: string, _params: any): Promise<{
|
|
11
|
+
content: {
|
|
12
|
+
type: "text";
|
|
13
|
+
text: string;
|
|
14
|
+
}[];
|
|
15
|
+
}>;
|
|
16
|
+
};
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { createSchemaTool } from "./schema-tool-factory.js";
|
|
2
|
+
import { sendEmailTool } from "./send-email-tool.js";
|
|
3
|
+
import { searchEmailTool } from "./search-email-tool.js";
|
|
4
|
+
export const getEmailToolSchemaTool = createSchemaTool({
|
|
5
|
+
name: "get_email_tool_schema",
|
|
6
|
+
label: "Get Email Tool Schema",
|
|
7
|
+
description: "获取可在用户设备上发送邮件、检索邮件的相关端工具列表。",
|
|
8
|
+
tools: [sendEmailTool, searchEmailTool],
|
|
9
|
+
});
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
// QueryAppMessage tool implementation
|
|
2
|
+
import { getXYWebSocketManager } from "../client.js";
|
|
3
|
+
import { sendCommand } from "../formatter.js";
|
|
4
|
+
import { getCurrentSessionContext } from "./session-manager.js";
|
|
5
|
+
class ToolInputError extends Error {
|
|
6
|
+
status = 400;
|
|
7
|
+
constructor(message) {
|
|
8
|
+
super(message);
|
|
9
|
+
this.name = "ToolInputError";
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* 查询指定时间范围内的设备通知消息。
|
|
14
|
+
*/
|
|
15
|
+
export const queryAppMessageTool = {
|
|
16
|
+
name: "query_app_message",
|
|
17
|
+
label: "Query App Message",
|
|
18
|
+
description: `获取指定时间范围内的设备通知消息。适用于需要查询历史通知、按应用筛选通知、或仅查看未读通知的场景。支持按时间范围、应用包名、已读/未读状态进行过滤。
|
|
19
|
+
注意:
|
|
20
|
+
a. 操作超时时间为60秒,请勿重复调用此工具
|
|
21
|
+
b. 如果遇到各类调用失败场景,最多只能重试一次,不可以重复调用多次。
|
|
22
|
+
c. 调用工具前需认真检查调用参数是否满足工具要求
|
|
23
|
+
|
|
24
|
+
回复约束:如果工具返回没有授权或者其他报错,只需要完整描述没有授权或者其他报错内容即可,不需要主动给用户提供解决方案,例如告诉用户如何授权,如何解决报错等都是不需要的,请严格遵守。`,
|
|
25
|
+
parameters: {
|
|
26
|
+
type: "object",
|
|
27
|
+
properties: {
|
|
28
|
+
startTime: {
|
|
29
|
+
type: "string",
|
|
30
|
+
description: "查询通知的起始时间(ISO 8601 字符串)。若endTime为空,则默认值为24小时前。",
|
|
31
|
+
},
|
|
32
|
+
endTime: {
|
|
33
|
+
type: "string",
|
|
34
|
+
description: "查询通知的结束时间(ISO 8601 字符串)。默认值为当前时间。",
|
|
35
|
+
},
|
|
36
|
+
packageName: {
|
|
37
|
+
type: "string",
|
|
38
|
+
description: "按应用名称过滤通知(例如「微信」「小红书」)。默认值为所有应用。",
|
|
39
|
+
},
|
|
40
|
+
state: {
|
|
41
|
+
type: "integer",
|
|
42
|
+
description: "通知的已读/未读状态。0 = 全部,1 = 仅未读。默认值为 0。",
|
|
43
|
+
},
|
|
44
|
+
},
|
|
45
|
+
required: [],
|
|
46
|
+
},
|
|
47
|
+
async execute(_toolCallId, params) {
|
|
48
|
+
const sessionContext = getCurrentSessionContext();
|
|
49
|
+
if (!sessionContext) {
|
|
50
|
+
throw new Error("No active XY session found.");
|
|
51
|
+
}
|
|
52
|
+
const { config, sessionId, taskId, messageId } = sessionContext;
|
|
53
|
+
const wsManager = getXYWebSocketManager(config);
|
|
54
|
+
const intentParam = {};
|
|
55
|
+
if (params.startTime !== undefined)
|
|
56
|
+
intentParam.startTime = params.startTime;
|
|
57
|
+
if (params.endTime !== undefined)
|
|
58
|
+
intentParam.endTime = params.endTime;
|
|
59
|
+
if (params.packageName !== undefined)
|
|
60
|
+
intentParam.packageName = params.packageName;
|
|
61
|
+
if (params.state !== undefined) {
|
|
62
|
+
if (params.state !== 0 && params.state !== 1) {
|
|
63
|
+
throw new ToolInputError("state 参数只能为 0(全部)或 1(仅未读)");
|
|
64
|
+
}
|
|
65
|
+
intentParam.state = params.state;
|
|
66
|
+
}
|
|
67
|
+
const command = {
|
|
68
|
+
header: {
|
|
69
|
+
namespace: "Common",
|
|
70
|
+
name: "Action",
|
|
71
|
+
},
|
|
72
|
+
payload: {
|
|
73
|
+
cardParam: {},
|
|
74
|
+
executeParam: {
|
|
75
|
+
executeMode: "background",
|
|
76
|
+
intentName: "QueryAppMessage",
|
|
77
|
+
bundleName: "com.huawei.hmos.vassistant",
|
|
78
|
+
needUnlock: true,
|
|
79
|
+
actionResponse: true,
|
|
80
|
+
appType: "OHOS_APP",
|
|
81
|
+
timeOut: 5,
|
|
82
|
+
intentParam,
|
|
83
|
+
permissionId: [],
|
|
84
|
+
achieveType: "INTENT",
|
|
85
|
+
},
|
|
86
|
+
responses: [
|
|
87
|
+
{
|
|
88
|
+
resultCode: "",
|
|
89
|
+
displayText: "",
|
|
90
|
+
ttsText: "",
|
|
91
|
+
},
|
|
92
|
+
],
|
|
93
|
+
needUploadResult: true,
|
|
94
|
+
noHalfPage: false,
|
|
95
|
+
pageControlRelated: false,
|
|
96
|
+
},
|
|
97
|
+
};
|
|
98
|
+
return new Promise((resolve, reject) => {
|
|
99
|
+
const timeout = setTimeout(() => {
|
|
100
|
+
wsManager.off("data-event", handler);
|
|
101
|
+
reject(new Error("查询通知消息超时(60秒)"));
|
|
102
|
+
}, 60000);
|
|
103
|
+
const handler = (event) => {
|
|
104
|
+
if (event.intentName === "QueryAppMessage") {
|
|
105
|
+
clearTimeout(timeout);
|
|
106
|
+
wsManager.off("data-event", handler);
|
|
107
|
+
if (event.status === "success" && event.outputs) {
|
|
108
|
+
resolve({
|
|
109
|
+
content: [
|
|
110
|
+
{
|
|
111
|
+
type: "text",
|
|
112
|
+
text: JSON.stringify(event.outputs),
|
|
113
|
+
},
|
|
114
|
+
],
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
else {
|
|
118
|
+
reject(new Error(`查询通知消息失败: ${event.status}`));
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
};
|
|
122
|
+
wsManager.on("data-event", handler);
|
|
123
|
+
sendCommand({
|
|
124
|
+
config,
|
|
125
|
+
sessionId,
|
|
126
|
+
taskId,
|
|
127
|
+
messageId,
|
|
128
|
+
command,
|
|
129
|
+
})
|
|
130
|
+
.then(() => { })
|
|
131
|
+
.catch((error) => {
|
|
132
|
+
clearTimeout(timeout);
|
|
133
|
+
wsManager.off("data-event", handler);
|
|
134
|
+
reject(error);
|
|
135
|
+
});
|
|
136
|
+
});
|
|
137
|
+
},
|
|
138
|
+
};
|
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
// QueryMemoryData tool implementation
|
|
2
|
+
import { getXYWebSocketManager } from "../client.js";
|
|
3
|
+
import { sendCommand } from "../formatter.js";
|
|
4
|
+
import { getCurrentSessionContext } from "./session-manager.js";
|
|
5
|
+
class ToolInputError extends Error {
|
|
6
|
+
status = 400;
|
|
7
|
+
constructor(message) {
|
|
8
|
+
super(message);
|
|
9
|
+
this.name = "ToolInputError";
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
const VALID_CATEGORIES = ["ImportantDay", "Address", "Card", "ServiceOrder", "Event"];
|
|
13
|
+
const VALID_SUB_CATEGORIES = {
|
|
14
|
+
ImportantDay: ["Birthday", "Anniversary", "RepaymentDate", "ExamDate", "SalaryDate", "BigDay"],
|
|
15
|
+
Card: [
|
|
16
|
+
"IDCard", "Passport", "DrivingLicense", "VehicleLicense", "EEPtoHKMO",
|
|
17
|
+
"EEPtoTW", "Invoice", "BusinessCard", "VehicleInspectionCertificate",
|
|
18
|
+
"SocialSecurityCard", "BankCard",
|
|
19
|
+
],
|
|
20
|
+
ServiceOrder: ["FilmTicket", "HotelOrder", "TrainTicket", "AirTicket"],
|
|
21
|
+
Event: [
|
|
22
|
+
"Delicacy", "Work", "FamilyActivities", "Travel", "Training",
|
|
23
|
+
"Health", "Life", "Entertainment", "Calendar",
|
|
24
|
+
],
|
|
25
|
+
};
|
|
26
|
+
/**
|
|
27
|
+
* 查询存储在设备本地的结构化记忆数据。
|
|
28
|
+
*/
|
|
29
|
+
export const queryMemoryDataTool = {
|
|
30
|
+
name: "query_memory_data",
|
|
31
|
+
label: "Query Memory Data",
|
|
32
|
+
description: `查询存储在设备本地的结构化记忆数据。适用于获取特定类别的个人信息,如重要日子、证件卡证、服务订单或日程事件。支持按分类、子分类进行过滤。
|
|
33
|
+
注意:
|
|
34
|
+
a. 操作超时时间为60秒,请勿重复调用此工具
|
|
35
|
+
b. 如果遇到各类调用失败场景,最多只能重试一次,不可以重复调用多次。
|
|
36
|
+
c. 调用工具前需认真检查调用参数是否满足工具要求
|
|
37
|
+
|
|
38
|
+
回复约束:如果工具返回没有授权或者其他报错,只需要完整描述没有授权或者其他报错内容即可,不需要主动给用户提供解决方案,例如告诉用户如何授权,如何解决报错等都是不需要的,请严格遵守。`,
|
|
39
|
+
parameters: {
|
|
40
|
+
type: "object",
|
|
41
|
+
properties: {
|
|
42
|
+
category: {
|
|
43
|
+
type: "string",
|
|
44
|
+
description: '按数据大类进行过滤。可选值为 ""、"ImportantDay"、"Address"、"Card"、"ServiceOrder"、"Event"。默认值为 ""。',
|
|
45
|
+
},
|
|
46
|
+
subCategory: {
|
|
47
|
+
description: '在指定 category 下的子类别过滤,默认值为""。支持字符串或字符串数组。',
|
|
48
|
+
oneOf: [
|
|
49
|
+
{ type: "string" },
|
|
50
|
+
{ type: "array", items: { type: "string" } },
|
|
51
|
+
],
|
|
52
|
+
},
|
|
53
|
+
},
|
|
54
|
+
required: [],
|
|
55
|
+
},
|
|
56
|
+
async execute(_toolCallId, params) {
|
|
57
|
+
const { category, subCategory } = params;
|
|
58
|
+
// Validate category
|
|
59
|
+
if (category && !VALID_CATEGORIES.includes(category)) {
|
|
60
|
+
throw new ToolInputError(`category 参数无效,可选值为:${VALID_CATEGORIES.join("、")}`);
|
|
61
|
+
}
|
|
62
|
+
// Validate subCategory when category is specified
|
|
63
|
+
if (category && subCategory && VALID_SUB_CATEGORIES[category]) {
|
|
64
|
+
const validValues = VALID_SUB_CATEGORIES[category];
|
|
65
|
+
const subValues = Array.isArray(subCategory) ? subCategory : [subCategory];
|
|
66
|
+
for (const sv of subValues) {
|
|
67
|
+
if (sv && !validValues.includes(sv)) {
|
|
68
|
+
throw new ToolInputError(`category 为 "${category}" 时,subCategory 可选值为:${validValues.join("、")}`);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
const sessionContext = getCurrentSessionContext();
|
|
73
|
+
if (!sessionContext) {
|
|
74
|
+
throw new Error("No active XY session found.");
|
|
75
|
+
}
|
|
76
|
+
const { config, sessionId, taskId, messageId } = sessionContext;
|
|
77
|
+
const wsManager = getXYWebSocketManager(config);
|
|
78
|
+
const intentParam = {};
|
|
79
|
+
if (category)
|
|
80
|
+
intentParam.category = category;
|
|
81
|
+
if (subCategory !== undefined)
|
|
82
|
+
intentParam.subCategory = subCategory;
|
|
83
|
+
const command = {
|
|
84
|
+
header: {
|
|
85
|
+
namespace: "Common",
|
|
86
|
+
name: "Action",
|
|
87
|
+
},
|
|
88
|
+
payload: {
|
|
89
|
+
cardParam: {},
|
|
90
|
+
executeParam: {
|
|
91
|
+
executeMode: "background",
|
|
92
|
+
intentName: "QueryMemoryData",
|
|
93
|
+
bundleName: "com.huawei.hmos.vassistant",
|
|
94
|
+
needUnlock: true,
|
|
95
|
+
actionResponse: true,
|
|
96
|
+
appType: "OHOS_APP",
|
|
97
|
+
timeOut: 5,
|
|
98
|
+
intentParam,
|
|
99
|
+
permissionId: [],
|
|
100
|
+
achieveType: "INTENT",
|
|
101
|
+
},
|
|
102
|
+
responses: [
|
|
103
|
+
{
|
|
104
|
+
resultCode: "",
|
|
105
|
+
displayText: "",
|
|
106
|
+
ttsText: "",
|
|
107
|
+
},
|
|
108
|
+
],
|
|
109
|
+
needUploadResult: true,
|
|
110
|
+
noHalfPage: false,
|
|
111
|
+
pageControlRelated: false,
|
|
112
|
+
},
|
|
113
|
+
};
|
|
114
|
+
return new Promise((resolve, reject) => {
|
|
115
|
+
const timeout = setTimeout(() => {
|
|
116
|
+
wsManager.off("data-event", handler);
|
|
117
|
+
reject(new Error("查询记忆数据超时(60秒)"));
|
|
118
|
+
}, 60000);
|
|
119
|
+
const handler = (event) => {
|
|
120
|
+
if (event.intentName === "QueryMemoryData") {
|
|
121
|
+
clearTimeout(timeout);
|
|
122
|
+
wsManager.off("data-event", handler);
|
|
123
|
+
if (event.status === "success" && event.outputs) {
|
|
124
|
+
resolve({
|
|
125
|
+
content: [
|
|
126
|
+
{
|
|
127
|
+
type: "text",
|
|
128
|
+
text: JSON.stringify(event.outputs),
|
|
129
|
+
},
|
|
130
|
+
],
|
|
131
|
+
});
|
|
132
|
+
}
|
|
133
|
+
else {
|
|
134
|
+
reject(new Error(`查询记忆数据失败: ${event.status}`));
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
};
|
|
138
|
+
wsManager.on("data-event", handler);
|
|
139
|
+
sendCommand({
|
|
140
|
+
config,
|
|
141
|
+
sessionId,
|
|
142
|
+
taskId,
|
|
143
|
+
messageId,
|
|
144
|
+
command,
|
|
145
|
+
})
|
|
146
|
+
.then(() => { })
|
|
147
|
+
.catch((error) => {
|
|
148
|
+
clearTimeout(timeout);
|
|
149
|
+
wsManager.off("data-event", handler);
|
|
150
|
+
reject(error);
|
|
151
|
+
});
|
|
152
|
+
});
|
|
153
|
+
},
|
|
154
|
+
};
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
// QueryTodoTask tool implementation
|
|
2
|
+
import { getXYWebSocketManager } from "../client.js";
|
|
3
|
+
import { sendCommand } from "../formatter.js";
|
|
4
|
+
import { getCurrentSessionContext } from "./session-manager.js";
|
|
5
|
+
class ToolInputError extends Error {
|
|
6
|
+
status = 400;
|
|
7
|
+
constructor(message) {
|
|
8
|
+
super(message);
|
|
9
|
+
this.name = "ToolInputError";
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* 获取指定时间范围内的全局待办任务列表。
|
|
14
|
+
*/
|
|
15
|
+
export const queryTodoTaskTool = {
|
|
16
|
+
name: "query_todo_task",
|
|
17
|
+
label: "Query Todo Task",
|
|
18
|
+
description: `获取指定时间范围内的全局待办任务列表。适用于需要查询历史任务、按完成状态筛选、或仅查看待处理任务的场景。支持按时间范围、任务状态进行过滤。
|
|
19
|
+
注意:
|
|
20
|
+
a. 操作超时时间为60秒,请勿重复调用此工具
|
|
21
|
+
b. 如果遇到各类调用失败场景,最多只能重试一次,不可以重复调用多次。
|
|
22
|
+
c. 调用工具前需认真检查调用参数是否满足工具要求
|
|
23
|
+
d. 当只传入 startTime 时,返回该时间点之后的所有任务;当只传入 endTime 时,返回该时间点之前的所有任务;两者都不传则返回所有时间段的任务。
|
|
24
|
+
|
|
25
|
+
回复约束:如果工具返回没有授权或者其他报错,只需要完整描述没有授权或者其他报错内容即可,不需要主动给用户提供解决方案,例如告诉用户如何授权,如何解决报错等都是不需要的,请严格遵守。`,
|
|
26
|
+
parameters: {
|
|
27
|
+
type: "object",
|
|
28
|
+
properties: {
|
|
29
|
+
startTime: {
|
|
30
|
+
type: "string",
|
|
31
|
+
description: "查询创建时间大于此值的任务(ISO 8601 字符串,如 2024-01-01T00:00:00Z)。",
|
|
32
|
+
},
|
|
33
|
+
endTime: {
|
|
34
|
+
type: "string",
|
|
35
|
+
description: "查询创建时间小于此值的任务(ISO 8601 字符串,如 2024-01-31T23:59:59Z)。",
|
|
36
|
+
},
|
|
37
|
+
status: {
|
|
38
|
+
type: "string",
|
|
39
|
+
description: '任务完成状态过滤。可选值为 "all"、"completed"、"pending"。默认为 "all"。',
|
|
40
|
+
},
|
|
41
|
+
},
|
|
42
|
+
required: [],
|
|
43
|
+
},
|
|
44
|
+
async execute(_toolCallId, params) {
|
|
45
|
+
const { status } = params;
|
|
46
|
+
if (status && !["all", "completed", "pending"].includes(status)) {
|
|
47
|
+
throw new ToolInputError('status 参数只能为 "all"、"completed" 或 "pending"');
|
|
48
|
+
}
|
|
49
|
+
const sessionContext = getCurrentSessionContext();
|
|
50
|
+
if (!sessionContext) {
|
|
51
|
+
throw new Error("No active XY session found.");
|
|
52
|
+
}
|
|
53
|
+
const { config, sessionId, taskId, messageId } = sessionContext;
|
|
54
|
+
const wsManager = getXYWebSocketManager(config);
|
|
55
|
+
const intentParam = {};
|
|
56
|
+
if (params.startTime !== undefined)
|
|
57
|
+
intentParam.startTime = params.startTime;
|
|
58
|
+
if (params.endTime !== undefined)
|
|
59
|
+
intentParam.endTime = params.endTime;
|
|
60
|
+
if (status !== undefined)
|
|
61
|
+
intentParam.status = status;
|
|
62
|
+
const command = {
|
|
63
|
+
header: {
|
|
64
|
+
namespace: "Common",
|
|
65
|
+
name: "Action",
|
|
66
|
+
},
|
|
67
|
+
payload: {
|
|
68
|
+
cardParam: {},
|
|
69
|
+
executeParam: {
|
|
70
|
+
executeMode: "background",
|
|
71
|
+
intentName: "QueryTodoTask",
|
|
72
|
+
bundleName: "com.huawei.hmos.vassistant",
|
|
73
|
+
needUnlock: true,
|
|
74
|
+
actionResponse: true,
|
|
75
|
+
appType: "OHOS_APP",
|
|
76
|
+
timeOut: 5,
|
|
77
|
+
intentParam,
|
|
78
|
+
permissionId: [],
|
|
79
|
+
achieveType: "INTENT",
|
|
80
|
+
},
|
|
81
|
+
responses: [
|
|
82
|
+
{
|
|
83
|
+
resultCode: "",
|
|
84
|
+
displayText: "",
|
|
85
|
+
ttsText: "",
|
|
86
|
+
},
|
|
87
|
+
],
|
|
88
|
+
needUploadResult: true,
|
|
89
|
+
noHalfPage: false,
|
|
90
|
+
pageControlRelated: false,
|
|
91
|
+
},
|
|
92
|
+
};
|
|
93
|
+
return new Promise((resolve, reject) => {
|
|
94
|
+
const timeout = setTimeout(() => {
|
|
95
|
+
wsManager.off("data-event", handler);
|
|
96
|
+
reject(new Error("查询待办任务超时(60秒)"));
|
|
97
|
+
}, 60000);
|
|
98
|
+
const handler = (event) => {
|
|
99
|
+
if (event.intentName === "QueryTodoTask") {
|
|
100
|
+
clearTimeout(timeout);
|
|
101
|
+
wsManager.off("data-event", handler);
|
|
102
|
+
if (event.status === "success" && event.outputs) {
|
|
103
|
+
resolve({
|
|
104
|
+
content: [
|
|
105
|
+
{
|
|
106
|
+
type: "text",
|
|
107
|
+
text: JSON.stringify(event.outputs),
|
|
108
|
+
},
|
|
109
|
+
],
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
else {
|
|
113
|
+
reject(new Error(`查询待办任务失败: ${event.status}`));
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
};
|
|
117
|
+
wsManager.on("data-event", handler);
|
|
118
|
+
sendCommand({
|
|
119
|
+
config,
|
|
120
|
+
sessionId,
|
|
121
|
+
taskId,
|
|
122
|
+
messageId,
|
|
123
|
+
command,
|
|
124
|
+
})
|
|
125
|
+
.then(() => { })
|
|
126
|
+
.catch((error) => {
|
|
127
|
+
clearTimeout(timeout);
|
|
128
|
+
wsManager.off("data-event", handler);
|
|
129
|
+
reject(error);
|
|
130
|
+
});
|
|
131
|
+
});
|
|
132
|
+
},
|
|
133
|
+
};
|