@ynhcj/xiaoyi-channel 0.0.75-beta → 0.0.75-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.d.ts +6 -9
- package/dist/index.js +29 -23
- package/dist/src/bot.js +27 -3
- package/dist/src/channel.js +11 -23
- package/dist/src/cspl/call-api.js +14 -11
- package/dist/src/cspl/config.js +3 -3
- package/dist/src/cspl/constants.d.ts +2 -0
- package/dist/src/cspl/constants.js +12 -0
- package/dist/src/cspl/utils.js +4 -2
- package/dist/src/file-download.js +3 -6
- package/dist/src/file-upload.js +52 -5
- package/dist/src/login-token-handler.d.ts +8 -0
- package/dist/src/login-token-handler.js +60 -0
- package/dist/src/message-queue.d.ts +17 -0
- package/dist/src/message-queue.js +51 -0
- package/dist/src/monitor.js +54 -3
- package/dist/src/outbound.js +2 -7
- package/dist/src/provider.d.ts +2 -1
- package/dist/src/provider.js +486 -33
- package/dist/src/reply-dispatcher.js +6 -0
- package/dist/src/runtime.d.ts +3 -11
- package/dist/src/runtime.js +6 -18
- package/dist/src/self-evolution-handler.d.ts +7 -0
- package/dist/src/self-evolution-handler.js +140 -0
- package/dist/src/self-evolution-keyword.d.ts +9 -0
- package/dist/src/self-evolution-keyword.js +147 -0
- package/dist/src/self-evolution-tool-result-nudge.d.ts +3 -0
- package/dist/src/self-evolution-tool-result-nudge.js +96 -0
- package/dist/src/skill-retriever/config.d.ts +4 -0
- package/dist/src/skill-retriever/config.js +23 -0
- package/dist/src/skill-retriever/hooks.d.ts +22 -0
- package/dist/src/skill-retriever/hooks.js +82 -0
- package/dist/src/skill-retriever/tool-search.d.ts +16 -0
- package/dist/src/skill-retriever/tool-search.js +172 -0
- package/dist/src/skill-retriever/types.d.ts +36 -0
- package/dist/src/skill-retriever/types.js +1 -0
- package/dist/src/task-manager.d.ts +4 -0
- package/dist/src/task-manager.js +6 -0
- package/dist/src/tools/call-device-tool.d.ts +5 -0
- package/dist/src/tools/call-device-tool.js +130 -0
- package/dist/src/tools/create-alarm-tool.js +5 -16
- package/dist/src/tools/delete-alarm-tool.js +1 -4
- package/dist/src/tools/device-tool-map.js +5 -4
- package/dist/src/tools/find-pc-devices-tool.d.ts +5 -0
- package/dist/src/tools/find-pc-devices-tool.js +98 -0
- package/dist/src/tools/get-alarm-tool-schema.d.ts +16 -0
- package/dist/src/tools/get-alarm-tool-schema.js +11 -0
- package/dist/src/tools/get-calendar-tool-schema.d.ts +16 -0
- package/dist/src/tools/get-calendar-tool-schema.js +9 -0
- package/dist/src/tools/get-collection-tool-schema.d.ts +16 -0
- package/dist/src/tools/get-collection-tool-schema.js +10 -0
- package/dist/src/tools/get-contact-tool-schema.d.ts +16 -0
- package/dist/src/tools/get-contact-tool-schema.js +11 -0
- package/dist/src/tools/get-device-file-tool-schema.d.ts +16 -0
- package/dist/src/tools/get-device-file-tool-schema.js +10 -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/get-note-tool-schema.d.ts +16 -0
- package/dist/src/tools/get-note-tool-schema.js +10 -0
- package/dist/src/tools/get-photo-tool-schema.d.ts +16 -0
- package/dist/src/tools/get-photo-tool-schema.js +10 -0
- package/dist/src/tools/image-reading-tool.js +4 -7
- package/dist/src/tools/login-token-tool.d.ts +5 -0
- package/dist/src/tools/login-token-tool.js +136 -0
- package/dist/src/tools/modify-alarm-tool.js +10 -23
- 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/dist/src/tools/save-file-to-phone-tool.d.ts +5 -0
- package/dist/src/tools/save-file-to-phone-tool.js +166 -0
- package/dist/src/tools/save-media-to-gallery-tool.js +3 -7
- package/dist/src/tools/save-self-evolution-skill-tool.d.ts +1 -0
- package/dist/src/tools/save-self-evolution-skill-tool.js +412 -0
- package/dist/src/tools/schema-tool-factory.d.ts +27 -0
- package/dist/src/tools/schema-tool-factory.js +32 -0
- package/dist/src/tools/search-alarm-tool.js +6 -13
- package/dist/src/tools/search-calendar-tool.js +2 -0
- package/dist/src/tools/search-email-tool.d.ts +5 -0
- package/dist/src/tools/search-email-tool.js +137 -0
- package/dist/src/tools/search-file-tool.js +4 -4
- package/dist/src/tools/search-message-tool.js +1 -0
- package/dist/src/tools/search-photo-gallery-tool.js +2 -2
- package/dist/src/tools/send-email-tool.d.ts +4 -0
- package/dist/src/tools/send-email-tool.js +134 -0
- package/dist/src/tools/send-file-to-user-tool.js +3 -5
- package/dist/src/tools/session-manager.js +2 -0
- package/dist/src/tools/upload-file-tool.js +4 -4
- package/dist/src/tools/upload-photo-tool.js +2 -2
- package/dist/src/tools/xiaoyi-add-collection-tool.js +23 -4
- package/dist/src/tools/xiaoyi-collection-tool.js +2 -1
- package/dist/src/tools/xiaoyi-delete-collection-tool.js +1 -1
- package/dist/src/utils/runtime-manager.js +24 -2
- package/dist/src/utils/self-evolution-manager.d.ts +10 -0
- package/dist/src/utils/self-evolution-manager.js +68 -0
- package/dist/src/utils/tool-call-nudge-manager.d.ts +16 -0
- package/dist/src/utils/tool-call-nudge-manager.js +47 -0
- package/dist/src/websocket.d.ts +3 -0
- package/dist/src/websocket.js +69 -0
- package/openclaw.plugin.json +21 -0
- package/package.json +3 -3
|
@@ -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
|
+
};
|
|
@@ -0,0 +1,166 @@
|
|
|
1
|
+
import { getXYWebSocketManager } from "../client.js";
|
|
2
|
+
import { sendCommand } from "../formatter.js";
|
|
3
|
+
import { getCurrentSessionContext } from "./session-manager.js";
|
|
4
|
+
import { XYFileUploadService } from "../file-upload.js";
|
|
5
|
+
/**
|
|
6
|
+
* Duck-typed ToolInputError: openclaw 按 .name 字段匹配,不用 instanceof。
|
|
7
|
+
* 抛出此错误会让 openclaw 返回 HTTP 400 而非 500,
|
|
8
|
+
* LLM 会将其识别为参数错误而非瞬时故障,不会触发重试。
|
|
9
|
+
*/
|
|
10
|
+
class ToolInputError extends Error {
|
|
11
|
+
status = 400;
|
|
12
|
+
constructor(message) {
|
|
13
|
+
super(message);
|
|
14
|
+
this.name = "ToolInputError";
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* XY save file to phone tool - saves files to user's device file manager.
|
|
19
|
+
* Supports local file paths (auto-uploaded to get public URL) and public URLs.
|
|
20
|
+
*/
|
|
21
|
+
export const saveFileToPhoneTool = {
|
|
22
|
+
name: "save_file_to_file_manager",
|
|
23
|
+
label: "Save File to Phone",
|
|
24
|
+
description: `将文件保存到用户设备的文件管理器中,通常用户表述为'帮我保存到文管','保存到文件管理'。
|
|
25
|
+
|
|
26
|
+
注意:
|
|
27
|
+
a. 操作超时时间为60秒,请勿重复调用此工具
|
|
28
|
+
b. 如果遇到各类调用失败场景,不可以重试,直接返回错误。
|
|
29
|
+
c. 调用工具前需认真检查调用参数是否满足工具要求
|
|
30
|
+
|
|
31
|
+
回复约束:如果工具返回没有授权或者其他报错,只需要完整描述没有授权或者其他报错内容即可,不需要主动给用户提供解决方案,例如告诉用户如何授权,如何解决报错等都是不需要的,请严格遵守。
|
|
32
|
+
`,
|
|
33
|
+
parameters: {
|
|
34
|
+
type: "object",
|
|
35
|
+
properties: {
|
|
36
|
+
fileName: {
|
|
37
|
+
type: "string",
|
|
38
|
+
description: "必填,文件名称。",
|
|
39
|
+
},
|
|
40
|
+
url: {
|
|
41
|
+
type: "string",
|
|
42
|
+
description: "必填,支持本地路径或者公网url路径。如果是本地路径会先上传获取公网url。",
|
|
43
|
+
},
|
|
44
|
+
suffix: {
|
|
45
|
+
type: "string",
|
|
46
|
+
description: "必填,文件后缀,例如 ppt、doc、pdf 等。",
|
|
47
|
+
},
|
|
48
|
+
},
|
|
49
|
+
required: ["fileName", "url", "suffix"],
|
|
50
|
+
},
|
|
51
|
+
async execute(toolCallId, params) {
|
|
52
|
+
// Validate parameters
|
|
53
|
+
const { fileName, url, suffix } = params;
|
|
54
|
+
if (!url || typeof url !== "string") {
|
|
55
|
+
throw new ToolInputError("缺少必填参数: url");
|
|
56
|
+
}
|
|
57
|
+
if (!fileName || typeof fileName !== "string") {
|
|
58
|
+
throw new ToolInputError("缺少必填参数: fileName");
|
|
59
|
+
}
|
|
60
|
+
if (!suffix || typeof suffix !== "string") {
|
|
61
|
+
throw new ToolInputError("缺少必填参数: suffix");
|
|
62
|
+
}
|
|
63
|
+
// Get session context
|
|
64
|
+
const sessionContext = getCurrentSessionContext();
|
|
65
|
+
if (!sessionContext) {
|
|
66
|
+
throw new Error("No active XY session found. SaveFileToFileManager tool can only be used during an active conversation.");
|
|
67
|
+
}
|
|
68
|
+
const { config, sessionId, taskId, messageId } = sessionContext;
|
|
69
|
+
// Get WebSocket manager
|
|
70
|
+
const wsManager = getXYWebSocketManager(config);
|
|
71
|
+
// Determine the URL: if it's a local path, upload first to get public URL
|
|
72
|
+
let publicUrl = url;
|
|
73
|
+
if (!url.startsWith("http://") && !url.startsWith("https://")) {
|
|
74
|
+
// Local file path - upload to get public URL
|
|
75
|
+
const uploadService = new XYFileUploadService(config.fileUploadUrl, config.apiKey, config.uid);
|
|
76
|
+
publicUrl = await uploadService.uploadFileAndGetUrl(url);
|
|
77
|
+
if (!publicUrl) {
|
|
78
|
+
throw new Error("本地文件上传失败,无法获取公网URL");
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
// Build intentParam
|
|
82
|
+
const intentParam = {
|
|
83
|
+
fileName: fileName,
|
|
84
|
+
url: publicUrl,
|
|
85
|
+
suffix: suffix,
|
|
86
|
+
};
|
|
87
|
+
// Build SaveFileToFileManager command
|
|
88
|
+
const command = {
|
|
89
|
+
header: {
|
|
90
|
+
namespace: "Common",
|
|
91
|
+
name: "Action",
|
|
92
|
+
},
|
|
93
|
+
payload: {
|
|
94
|
+
cardParam: {},
|
|
95
|
+
executeParam: {
|
|
96
|
+
executeMode: "background",
|
|
97
|
+
intentName: "SaveFileToFileManager",
|
|
98
|
+
bundleName: "com.huawei.hmos.vassistant",
|
|
99
|
+
dimension: "",
|
|
100
|
+
needUnlock: true,
|
|
101
|
+
actionResponse: true,
|
|
102
|
+
appType: "OHOS_APP",
|
|
103
|
+
timeOut: 5,
|
|
104
|
+
timeout: 55000,
|
|
105
|
+
intentParam,
|
|
106
|
+
permissionId: ["ohos.permission.WRITE_IMAGEVIDEO"],
|
|
107
|
+
achieveType: "INTENT",
|
|
108
|
+
},
|
|
109
|
+
responses: [
|
|
110
|
+
{
|
|
111
|
+
resultCode: "",
|
|
112
|
+
displayText: "",
|
|
113
|
+
ttsText: "",
|
|
114
|
+
},
|
|
115
|
+
],
|
|
116
|
+
needUploadResult: true,
|
|
117
|
+
noHalfPage: false,
|
|
118
|
+
pageControlRelated: false,
|
|
119
|
+
},
|
|
120
|
+
};
|
|
121
|
+
// Send command and wait for response (60 second timeout)
|
|
122
|
+
return new Promise((resolve, reject) => {
|
|
123
|
+
const timeout = setTimeout(() => {
|
|
124
|
+
wsManager.off("data-event", handler);
|
|
125
|
+
reject(new Error("保存文件到手机超时(60秒)"));
|
|
126
|
+
}, 60000);
|
|
127
|
+
// Listen for data events from WebSocket
|
|
128
|
+
const handler = (event) => {
|
|
129
|
+
if (event.intentName === "SaveFileToFileManager") {
|
|
130
|
+
clearTimeout(timeout);
|
|
131
|
+
wsManager.off("data-event", handler);
|
|
132
|
+
if (event.status === "success" && event.outputs) {
|
|
133
|
+
resolve({
|
|
134
|
+
content: [
|
|
135
|
+
{
|
|
136
|
+
type: "text",
|
|
137
|
+
text: JSON.stringify(event.outputs),
|
|
138
|
+
}
|
|
139
|
+
]
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
else {
|
|
143
|
+
reject(new Error(`保存文件到手机失败: ${event.status}`));
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
};
|
|
147
|
+
// Register event handler
|
|
148
|
+
wsManager.on("data-event", handler);
|
|
149
|
+
// Send the command
|
|
150
|
+
sendCommand({
|
|
151
|
+
config,
|
|
152
|
+
sessionId,
|
|
153
|
+
taskId,
|
|
154
|
+
messageId,
|
|
155
|
+
command,
|
|
156
|
+
})
|
|
157
|
+
.then(() => {
|
|
158
|
+
})
|
|
159
|
+
.catch((error) => {
|
|
160
|
+
clearTimeout(timeout);
|
|
161
|
+
wsManager.off("data-event", handler);
|
|
162
|
+
reject(error);
|
|
163
|
+
});
|
|
164
|
+
});
|
|
165
|
+
},
|
|
166
|
+
};
|
|
@@ -19,13 +19,9 @@ class ToolInputError extends Error {
|
|
|
19
19
|
* Supports local file paths (auto-uploaded to get public URL) and public URLs.
|
|
20
20
|
*/
|
|
21
21
|
export const saveMediaToGalleryTool = {
|
|
22
|
-
name: "
|
|
22
|
+
name: "save_media_to_gallery",
|
|
23
23
|
label: "Save Media to Gallery",
|
|
24
|
-
description:
|
|
25
|
-
工具参数说明:
|
|
26
|
-
a. mediaType:非必填,string类型,不传端侧默认为pic。支持传 pic(图片) 或 video(视频)。
|
|
27
|
-
b. fileName:非必填,string类型,文件名称,不传手机侧默认生成随机uuid。
|
|
28
|
-
c. url:必填,string类型,支持本地路径或者公网url路径。如果是本地路径,会先上传获取公网url再保存到图库。
|
|
24
|
+
description: `将图片文件或者视频文件保存到设备图库。
|
|
29
25
|
|
|
30
26
|
注意:
|
|
31
27
|
a. 操作超时时间为60秒,请勿重复调用此工具
|
|
@@ -47,7 +43,7 @@ export const saveMediaToGalleryTool = {
|
|
|
47
43
|
},
|
|
48
44
|
url: {
|
|
49
45
|
type: "string",
|
|
50
|
-
description: "必填,支持本地路径或者公网url
|
|
46
|
+
description: "必填,支持本地路径或者公网url路径。如果是本地路径,会先上传获取公网url再保存到图库",
|
|
51
47
|
},
|
|
52
48
|
},
|
|
53
49
|
required: ["url"],
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare const saveSelfEvolutionSkillTool: any;
|