@ynhcj/xiaoyi-channel 0.0.123-beta → 0.0.125-beta

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (93) hide show
  1. package/dist/src/channel.js +2 -17
  2. package/dist/src/monitor.js +10 -1
  3. package/dist/src/provider.js +9 -5
  4. package/dist/src/tools/calendar-tool.d.ts +2 -1
  5. package/dist/src/tools/calendar-tool.js +112 -116
  6. package/dist/src/tools/call-device-tool.d.ts +2 -1
  7. package/dist/src/tools/call-device-tool.js +126 -103
  8. package/dist/src/tools/call-phone-tool.d.ts +2 -1
  9. package/dist/src/tools/call-phone-tool.js +109 -113
  10. package/dist/src/tools/create-alarm-tool.d.ts +2 -1
  11. package/dist/src/tools/create-alarm-tool.js +227 -231
  12. package/dist/src/tools/create-all-tools.d.ts +16 -0
  13. package/dist/src/tools/create-all-tools.js +50 -0
  14. package/dist/src/tools/delete-alarm-tool.d.ts +2 -1
  15. package/dist/src/tools/delete-alarm-tool.js +131 -135
  16. package/dist/src/tools/get-alarm-tool-schema.d.ts +2 -1
  17. package/dist/src/tools/get-alarm-tool-schema.js +16 -10
  18. package/dist/src/tools/get-calendar-tool-schema.d.ts +2 -1
  19. package/dist/src/tools/get-calendar-tool-schema.js +12 -8
  20. package/dist/src/tools/get-collection-tool-schema.d.ts +2 -1
  21. package/dist/src/tools/get-collection-tool-schema.js +11 -9
  22. package/dist/src/tools/get-contact-tool-schema.d.ts +2 -1
  23. package/dist/src/tools/get-contact-tool-schema.js +16 -10
  24. package/dist/src/tools/get-device-file-tool-schema.d.ts +2 -1
  25. package/dist/src/tools/get-device-file-tool-schema.js +13 -9
  26. package/dist/src/tools/get-email-tool-schema.d.ts +2 -1
  27. package/dist/src/tools/get-email-tool-schema.js +11 -8
  28. package/dist/src/tools/get-note-tool-schema.d.ts +2 -1
  29. package/dist/src/tools/get-note-tool-schema.js +14 -9
  30. package/dist/src/tools/get-photo-tool-schema.d.ts +2 -1
  31. package/dist/src/tools/get-photo-tool-schema.js +12 -9
  32. package/dist/src/tools/image-reading-tool.d.ts +2 -1
  33. package/dist/src/tools/image-reading-tool.js +86 -90
  34. package/dist/src/tools/location-tool.d.ts +2 -1
  35. package/dist/src/tools/location-tool.js +87 -91
  36. package/dist/src/tools/login-token-tool.d.ts +2 -1
  37. package/dist/src/tools/login-token-tool.js +113 -116
  38. package/dist/src/tools/modify-alarm-tool.d.ts +2 -1
  39. package/dist/src/tools/modify-alarm-tool.js +232 -236
  40. package/dist/src/tools/modify-note-tool.d.ts +2 -1
  41. package/dist/src/tools/modify-note-tool.js +104 -108
  42. package/dist/src/tools/note-tool.d.ts +2 -1
  43. package/dist/src/tools/note-tool.js +103 -107
  44. package/dist/src/tools/query-app-message-tool.d.ts +2 -1
  45. package/dist/src/tools/query-app-message-tool.js +108 -111
  46. package/dist/src/tools/query-memory-data-tool.d.ts +2 -1
  47. package/dist/src/tools/query-memory-data-tool.js +109 -112
  48. package/dist/src/tools/query-todo-task-tool.d.ts +2 -1
  49. package/dist/src/tools/query-todo-task-tool.js +103 -106
  50. package/dist/src/tools/save-file-to-phone-tool.d.ts +2 -1
  51. package/dist/src/tools/save-file-to-phone-tool.js +127 -131
  52. package/dist/src/tools/save-media-to-gallery-tool.d.ts +2 -1
  53. package/dist/src/tools/save-media-to-gallery-tool.js +134 -138
  54. package/dist/src/tools/save-self-evolution-skill-tool.d.ts +2 -1
  55. package/dist/src/tools/save-self-evolution-skill-tool.js +194 -196
  56. package/dist/src/tools/search-alarm-tool.d.ts +2 -1
  57. package/dist/src/tools/search-alarm-tool.js +171 -175
  58. package/dist/src/tools/search-calendar-tool.d.ts +2 -1
  59. package/dist/src/tools/search-calendar-tool.js +145 -149
  60. package/dist/src/tools/search-contact-tool.d.ts +2 -1
  61. package/dist/src/tools/search-contact-tool.js +98 -102
  62. package/dist/src/tools/search-email-tool.d.ts +2 -1
  63. package/dist/src/tools/search-email-tool.js +107 -111
  64. package/dist/src/tools/search-file-tool.d.ts +2 -1
  65. package/dist/src/tools/search-file-tool.js +99 -103
  66. package/dist/src/tools/search-message-tool.d.ts +2 -1
  67. package/dist/src/tools/search-message-tool.js +100 -104
  68. package/dist/src/tools/search-note-tool.d.ts +2 -1
  69. package/dist/src/tools/search-note-tool.js +95 -99
  70. package/dist/src/tools/search-photo-gallery-tool.d.ts +2 -1
  71. package/dist/src/tools/search-photo-gallery-tool.js +34 -38
  72. package/dist/src/tools/send-email-tool.d.ts +2 -1
  73. package/dist/src/tools/send-email-tool.js +105 -108
  74. package/dist/src/tools/send-file-to-user-tool.d.ts +2 -1
  75. package/dist/src/tools/send-file-to-user-tool.js +153 -155
  76. package/dist/src/tools/send-message-tool.d.ts +2 -1
  77. package/dist/src/tools/send-message-tool.js +119 -123
  78. package/dist/src/tools/session-manager.d.ts +18 -1
  79. package/dist/src/tools/session-manager.js +109 -11
  80. package/dist/src/tools/upload-file-tool.d.ts +2 -1
  81. package/dist/src/tools/upload-file-tool.js +78 -82
  82. package/dist/src/tools/upload-photo-tool.d.ts +2 -1
  83. package/dist/src/tools/upload-photo-tool.js +69 -73
  84. package/dist/src/tools/xiaoyi-add-collection-tool.d.ts +2 -1
  85. package/dist/src/tools/xiaoyi-add-collection-tool.js +143 -147
  86. package/dist/src/tools/xiaoyi-collection-tool.d.ts +2 -1
  87. package/dist/src/tools/xiaoyi-collection-tool.js +111 -115
  88. package/dist/src/tools/xiaoyi-delete-collection-tool.d.ts +2 -1
  89. package/dist/src/tools/xiaoyi-delete-collection-tool.js +124 -128
  90. package/dist/src/tools/xiaoyi-gui-tool.d.ts +2 -1
  91. package/dist/src/tools/xiaoyi-gui-tool.js +84 -88
  92. package/dist/src/websocket.js +29 -28
  93. package/package.json +1 -1
@@ -1,24 +1,9 @@
1
1
  import { resolveXYConfig, listXYAccountIds, getDefaultXYAccountId } from "./config.js";
2
2
  import { xyConfigSchema } from "./config-schema.js";
3
3
  import { xyOutbound } from "./outbound.js";
4
- import { locationTool } from "./tools/location-tool.js";
5
- import { xiaoyiGuiTool } from "./tools/xiaoyi-gui-tool.js";
6
- import { sendFileToUserTool } from "./tools/send-file-to-user-tool.js";
7
- import { viewPushResultTool } from "./tools/view-push-result-tool.js";
8
- import { imageReadingTool } from "./tools/image-reading-tool.js";
9
- import { timestampToUtc8Tool } from "./tools/timestamp-to-utc8-tool.js";
10
- import { saveSelfEvolutionSkillTool } from "./tools/save-self-evolution-skill-tool.js";
11
- import { callDeviceTool } from "./tools/call-device-tool.js";
12
- import { getNoteToolSchemaTool } from "./tools/get-note-tool-schema.js";
13
- import { getCalendarToolSchemaTool } from "./tools/get-calendar-tool-schema.js";
14
- import { getContactToolSchemaTool } from "./tools/get-contact-tool-schema.js";
15
- import { getPhotoToolSchemaTool } from "./tools/get-photo-tool-schema.js";
16
- import { getDeviceFileToolSchemaTool } from "./tools/get-device-file-tool-schema.js";
17
- import { getAlarmToolSchemaTool } from "./tools/get-alarm-tool-schema.js";
18
- import { getCollectionToolSchemaTool } from "./tools/get-collection-tool-schema.js";
19
- import { loginTokenTool } from "./tools/login-token-tool.js";
20
4
  import { filterToolsByDevice } from "./tools/device-tool-map.js";
21
5
  import { getCurrentSessionContext } from "./tools/session-manager.js";
6
+ import { createAllTools } from "./tools/create-all-tools.js";
22
7
  import { logger } from "./utils/logger.js";
23
8
  /**
24
9
  * Xiaoyi Channel Plugin for OpenClaw.
@@ -59,8 +44,8 @@ export const xyPlugin = {
59
44
  },
60
45
  outbound: xyOutbound,
61
46
  agentTools: () => {
62
- const allTools = [locationTool, callDeviceTool, getNoteToolSchemaTool, getCalendarToolSchemaTool, getContactToolSchemaTool, getPhotoToolSchemaTool, xiaoyiGuiTool, getDeviceFileToolSchemaTool, getAlarmToolSchemaTool, getCollectionToolSchemaTool, sendFileToUserTool, viewPushResultTool, imageReadingTool, timestampToUtc8Tool, saveSelfEvolutionSkillTool, loginTokenTool];
63
47
  const ctx = getCurrentSessionContext();
48
+ const allTools = createAllTools(ctx);
64
49
  const filtered = filterToolsByDevice(allTools, ctx?.deviceType);
65
50
  logger.log(`[DEVICE-FILTER] deviceType=${ctx?.deviceType ?? "(none)"}, tools: ${allTools.length} → ${filtered.length} (${filtered.map(t => t.name).join(", ")})`);
66
51
  return filtered;
@@ -8,6 +8,7 @@ import { handleTriggerEvent } from "./trigger-handler.js";
8
8
  import { handleSelfEvolutionEvent, handleSelfEvolutionStateGetEvent } from "./self-evolution-handler.js";
9
9
  import { handleLoginTokenEvent } from "./login-token-handler.js";
10
10
  import { cleanupStaleTempFiles } from "./reply-dispatcher.js";
11
+ import { cleanupStaleSessions, getActiveSessionCount, cleanupAllSessions } from "./tools/session-manager.js";
11
12
  /**
12
13
  * Per-session serial queue that ensures messages from the same session are processed
13
14
  * in arrival order while allowing different sessions to run concurrently.
@@ -198,9 +199,11 @@ export async function monitorXYProvider(opts = {}) {
198
199
  wsManager.disconnect();
199
200
  // ✅ Remove manager from cache to prevent reusing dirty state
200
201
  removeXYWebSocketManager(account);
202
+ // Clean up all active sessions
203
+ cleanupAllSessions();
201
204
  loggedServers.clear();
202
205
  activeMessages.clear();
203
- log(`[MONITOR-HANDLER] 🧹 Cleanup complete, cleared active messages`);
206
+ log(`[MONITOR-HANDLER] 🧹 Cleanup complete, cleared active messages and sessions`);
204
207
  // 🔍 Diagnose after cleanup
205
208
  console.log("🔍 [DIAGNOSTICS] Checking WebSocket managers after cleanup...");
206
209
  diagnoseAllManagers();
@@ -264,6 +267,12 @@ export async function monitorXYProvider(opts = {}) {
264
267
  if (cleaned > 0) {
265
268
  console.log(`🧹 [HEALTH CHECK] Auto-cleaned ${cleaned} manager(s) with orphan connections`);
266
269
  }
270
+ // Cleanup stale sessions (older than 10min TTL)
271
+ const cleanedSessions = cleanupStaleSessions();
272
+ const remainingSessions = getActiveSessionCount();
273
+ if (cleanedSessions > 0 || remainingSessions > 0) {
274
+ console.log(`🧹 [HEALTH CHECK] Sessions: cleaned=${cleanedSessions}, active=${remainingSessions}`);
275
+ }
267
276
  // Cleanup stale temp files (older than 24 hours)
268
277
  void cleanupStaleTempFiles();
269
278
  }, 6 * 60 * 60 * 1000); // 6 hours
@@ -230,6 +230,8 @@ const HEADER_SESSION_ID = "x-session-id";
230
230
  const HEADER_INTERACTION_ID = "x-interaction-id";
231
231
  /** Internal key for passing fallback uid prefix from prepareExtraParams to wrapStreamFn. */
232
232
  const FALLBACK_PREFIX_KEY = "_xiaoyi_fallback_prefix";
233
+ /** Internal key for passing deviceType from prepareExtraParams to wrapStreamFn. */
234
+ const DEVICE_TYPE_KEY = "_xiaoyi_device_type";
233
235
  const SELF_EVOLUTION_PROMPT_BEGIN = "<self_evolution_prompt>";
234
236
  const SELF_EVOLUTION_PROMPT_END = "</self_evolution_prompt>";
235
237
  const SELF_EVOLUTION_ENABLED_PROMPT_SECTION = `
@@ -413,6 +415,7 @@ export const xiaoyiProvider = {
413
415
  [HEADER_TRACE_ID]: taskId,
414
416
  [HEADER_SESSION_ID]: sessionId,
415
417
  [HEADER_INTERACTION_ID]: interactionId,
418
+ [DEVICE_TYPE_KEY]: sessionCtx.deviceType ?? "",
416
419
  };
417
420
  }
418
421
  // Fallback: store uid prefix for lazy timestamp generation in wrapStreamFn.
@@ -486,7 +489,9 @@ export const xiaoyiProvider = {
486
489
  if (context.systemPrompt) {
487
490
  console.log(`[xiaoyiprovider] system prompt length: ${context.systemPrompt.length}`);
488
491
  }
489
- const sessionCtx = getCurrentSessionContext();
492
+ // Reuse deviceType from extraParams instead of calling getCurrentSessionContext()
493
+ // again (which may be ambiguous in multi-session or async scenarios).
494
+ const deviceType = ctx.extraParams?.[DEVICE_TYPE_KEY] || undefined;
490
495
  // 在发送给模型前,优化 systemPrompt 结构
491
496
  if (context.systemPrompt) {
492
497
  let sp = context.systemPrompt;
@@ -519,10 +524,9 @@ export const xiaoyiProvider = {
519
524
  const selfEvolutionEnabled = await selfEvolutionManager.isEnabled();
520
525
  logger.log(`[selfEvolution] selfEvolution flag: ${selfEvolutionEnabled}`);
521
526
  context.systemPrompt = applySelfEvolutionPrompt(context.systemPrompt, selfEvolutionEnabled);
522
- // Append device context to systemPrompt
523
- if (sessionCtx?.deviceType) {
524
- const rawDevice = sessionCtx.deviceType;
525
- const displayDevice = (rawDevice === "2in1") ? "鸿蒙PC" : rawDevice;
527
+ // Append device context to systemPrompt (using pre-captured deviceType from prepareExtraParams)
528
+ if (deviceType) {
529
+ const displayDevice = (deviceType === "2in1") ? "鸿蒙PC" : deviceType;
526
530
  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"`;
527
531
  context.systemPrompt = (context.systemPrompt ?? "") + deviceSection;
528
532
  }
@@ -1,6 +1,7 @@
1
+ import type { SessionContext } from "./session-manager.js";
1
2
  /**
2
3
  * XY calendar event tool - creates a calendar event on user's device.
3
4
  * Requires title, dtStart (start time), and dtEnd (end time) parameters.
4
5
  * Time format must be: yyyy-mm-dd hh:mm:ss
5
6
  */
6
- export declare const calendarTool: any;
7
+ export declare function createCalendarTool(ctx: SessionContext): any;
@@ -1,134 +1,130 @@
1
1
  import { getXYWebSocketManager } from "../client.js";
2
2
  import { sendCommand } from "../formatter.js";
3
- import { getCurrentSessionContext } from "./session-manager.js";
4
3
  /**
5
4
  * XY calendar event tool - creates a calendar event on user's device.
6
5
  * Requires title, dtStart (start time), and dtEnd (end time) parameters.
7
6
  * Time format must be: yyyy-mm-dd hh:mm:ss
8
7
  */
9
- export const calendarTool = {
10
- name: "create_calendar_event",
11
- label: "Create Calendar Event",
12
- description: `在用户设备上创建日程。需要提供日程标题、开始时间和结束时间。时间格式必须为:yyyy-mm-dd hh:mm:ss(例如:2024-01-15 14:30:00)。注意:该工具执行时间较长(最多60秒),请勿重复调用,超时或失败时最多重试一次。
8
+ export function createCalendarTool(ctx) {
9
+ const { config, sessionId, taskId, messageId } = ctx;
10
+ return {
11
+ name: "create_calendar_event",
12
+ label: "Create Calendar Event",
13
+ description: `在用户设备上创建日程。需要提供日程标题、开始时间和结束时间。时间格式必须为:yyyy-mm-dd hh:mm:ss(例如:2024-01-15 14:30:00)。注意:该工具执行时间较长(最多60秒),请勿重复调用,超时或失败时最多重试一次。
13
14
  注意事项:使用该工具之前需获取当前真实时间
14
15
 
15
16
  回复约束:如果工具返回没有授权或者其他报错,只需要完整描述没有授权或者其他报错内容即可,不需要主动给用户提供解决方案,例如告诉用户如何授权,如何解决报错等都是不需要的,请严格遵守。
16
17
  `,
17
- parameters: {
18
- type: "object",
19
- properties: {
20
- title: {
21
- type: "string",
22
- description: "日程标题/名称",
23
- },
24
- dtStart: {
25
- type: "string",
26
- description: "日程开始时间,格式必须为:yyyy-mm-dd hh:mm:ss(例如:2024-01-15 14:30:00)",
27
- },
28
- dtEnd: {
29
- type: "string",
30
- description: "日程结束时间,格式必须为:yyyy-mm-dd hh:mm:ss(例如:2024-01-15 17:30:00)",
18
+ parameters: {
19
+ type: "object",
20
+ properties: {
21
+ title: {
22
+ type: "string",
23
+ description: "日程标题/名称",
24
+ },
25
+ dtStart: {
26
+ type: "string",
27
+ description: "日程开始时间,格式必须为:yyyy-mm-dd hh:mm:ss(例如:2024-01-15 14:30:00)",
28
+ },
29
+ dtEnd: {
30
+ type: "string",
31
+ description: "日程结束时间,格式必须为:yyyy-mm-dd hh:mm:ss(例如:2024-01-15 17:30:00)",
32
+ },
31
33
  },
34
+ required: ["title", "dtStart", "dtEnd"],
32
35
  },
33
- required: ["title", "dtStart", "dtEnd"],
34
- },
35
- async execute(toolCallId, params) {
36
- // Validate parameters
37
- if (!params.title || !params.dtStart || !params.dtEnd) {
38
- throw new Error("Missing required parameters: title, dtStart, and dtEnd are required");
39
- }
40
- // Convert time strings to millisecond timestamps
41
- const dtStartMs = new Date(params.dtStart).getTime();
42
- const dtEndMs = new Date(params.dtEnd).getTime();
43
- if (isNaN(dtStartMs) || isNaN(dtEndMs)) {
44
- throw new Error("Invalid time format. Required format: yyyy-mm-dd hh:mm:ss (e.g., 2024-01-15 14:30:00)");
45
- }
46
- // Get session context
47
- const sessionContext = getCurrentSessionContext();
48
- if (!sessionContext) {
49
- throw new Error("No active XY session found. Calendar tool can only be used during an active conversation.");
50
- }
51
- const { config, sessionId, taskId, messageId } = sessionContext;
52
- // Get WebSocket manager
53
- const wsManager = getXYWebSocketManager(config);
54
- // Build CreateCalendarEvent command
55
- const command = {
56
- header: {
57
- namespace: "Common",
58
- name: "ActionAndResult",
59
- },
60
- payload: {
61
- cardParam: {},
62
- executeParam: {
63
- executeMode: "background",
64
- intentName: "CreateCalendarEvent",
65
- bundleName: "com.huawei.hmos.calendardata",
66
- dimension: "",
67
- needUnlock: true,
68
- actionResponse: true,
69
- timeOut: 5,
70
- intentParam: {
71
- title: params.title,
72
- dtStart: dtStartMs,
73
- dtEnd: dtEndMs,
74
- },
75
- achieveType: "INTENT",
36
+ async execute(toolCallId, params) {
37
+ // Validate parameters
38
+ if (!params.title || !params.dtStart || !params.dtEnd) {
39
+ throw new Error("Missing required parameters: title, dtStart, and dtEnd are required");
40
+ }
41
+ // Convert time strings to millisecond timestamps
42
+ const dtStartMs = new Date(params.dtStart).getTime();
43
+ const dtEndMs = new Date(params.dtEnd).getTime();
44
+ if (isNaN(dtStartMs) || isNaN(dtEndMs)) {
45
+ throw new Error("Invalid time format. Required format: yyyy-mm-dd hh:mm:ss (e.g., 2024-01-15 14:30:00)");
46
+ }
47
+ // Get WebSocket manager
48
+ const wsManager = getXYWebSocketManager(config);
49
+ // Build CreateCalendarEvent command
50
+ const command = {
51
+ header: {
52
+ namespace: "Common",
53
+ name: "ActionAndResult",
76
54
  },
77
- responses: [
78
- {
79
- resultCode: "",
80
- displayText: "",
81
- ttsText: "",
55
+ payload: {
56
+ cardParam: {},
57
+ executeParam: {
58
+ executeMode: "background",
59
+ intentName: "CreateCalendarEvent",
60
+ bundleName: "com.huawei.hmos.calendardata",
61
+ dimension: "",
62
+ needUnlock: true,
63
+ actionResponse: true,
64
+ timeOut: 5,
65
+ intentParam: {
66
+ title: params.title,
67
+ dtStart: dtStartMs,
68
+ dtEnd: dtEndMs,
69
+ },
70
+ achieveType: "INTENT",
82
71
  },
83
- ],
84
- needUploadResult: true,
85
- noHalfPage: false,
86
- pageControlRelated: false,
87
- },
88
- };
89
- // Send command and wait for response (60 second timeout)
90
- return new Promise((resolve, reject) => {
91
- const timeout = setTimeout(() => {
92
- wsManager.off("data-event", handler);
93
- reject(new Error("创建日程超时(60秒)"));
94
- }, 60000);
95
- // Listen for data events from WebSocket
96
- const handler = (event) => {
97
- if (event.intentName === "CreateCalendarEvent") {
98
- clearTimeout(timeout);
72
+ responses: [
73
+ {
74
+ resultCode: "",
75
+ displayText: "",
76
+ ttsText: "",
77
+ },
78
+ ],
79
+ needUploadResult: true,
80
+ noHalfPage: false,
81
+ pageControlRelated: false,
82
+ },
83
+ };
84
+ // Send command and wait for response (60 second timeout)
85
+ return new Promise((resolve, reject) => {
86
+ const timeout = setTimeout(() => {
99
87
  wsManager.off("data-event", handler);
100
- if (event.status === "success" && event.outputs) {
101
- resolve({
102
- content: [
103
- {
104
- type: "text",
105
- text: JSON.stringify(event.outputs),
106
- },
107
- ],
108
- });
109
- }
110
- else {
111
- reject(new Error(`创建日程失败: ${event.status}`));
88
+ reject(new Error("创建日程超时(60秒)"));
89
+ }, 60000);
90
+ // Listen for data events from WebSocket
91
+ const handler = (event) => {
92
+ if (event.intentName === "CreateCalendarEvent") {
93
+ clearTimeout(timeout);
94
+ wsManager.off("data-event", handler);
95
+ if (event.status === "success" && event.outputs) {
96
+ resolve({
97
+ content: [
98
+ {
99
+ type: "text",
100
+ text: JSON.stringify(event.outputs),
101
+ },
102
+ ],
103
+ });
104
+ }
105
+ else {
106
+ reject(new Error(`创建日程失败: ${event.status}`));
107
+ }
112
108
  }
113
- }
114
- };
115
- // Register event handler
116
- wsManager.on("data-event", handler);
117
- // Send the command
118
- sendCommand({
119
- config,
120
- sessionId,
121
- taskId,
122
- messageId,
123
- command,
124
- })
125
- .then(() => {
126
- })
127
- .catch((error) => {
128
- clearTimeout(timeout);
129
- wsManager.off("data-event", handler);
130
- reject(error);
109
+ };
110
+ // Register event handler
111
+ wsManager.on("data-event", handler);
112
+ // Send the command
113
+ sendCommand({
114
+ config,
115
+ sessionId,
116
+ taskId,
117
+ messageId,
118
+ command,
119
+ })
120
+ .then(() => {
121
+ })
122
+ .catch((error) => {
123
+ clearTimeout(timeout);
124
+ wsManager.off("data-event", handler);
125
+ reject(error);
126
+ });
131
127
  });
132
- });
133
- },
134
- };
128
+ },
129
+ };
130
+ }
@@ -1,5 +1,6 @@
1
+ import type { SessionContext } from "./session-manager.js";
1
2
  /**
2
3
  * call_device_tool - 通用端工具调度器。
3
4
  * LLM 必须先通过 get_xxx_tool_schema 获取具体工具 schema,再用本工具执行。
4
5
  */
5
- export declare const callDeviceTool: any;
6
+ export declare function createCallDeviceTool(ctx: SessionContext): any;
@@ -1,92 +1,115 @@
1
- import { noteTool } from "./note-tool.js";
2
- import { searchNoteTool } from "./search-note-tool.js";
3
- import { modifyNoteTool } from "./modify-note-tool.js";
4
- import { createAlarmTool } from "./create-alarm-tool.js";
5
- import { searchAlarmTool } from "./search-alarm-tool.js";
6
- import { modifyAlarmTool } from "./modify-alarm-tool.js";
7
- import { deleteAlarmTool } from "./delete-alarm-tool.js";
8
- import { searchContactTool } from "./search-contact-tool.js";
9
- import { callPhoneTool } from "./call-phone-tool.js";
10
- import { searchMessageTool } from "./search-message-tool.js";
11
- import { sendMessageTool } from "./send-message-tool.js";
12
- import { xiaoyiAddCollectionTool } from "./xiaoyi-add-collection-tool.js";
13
- import { xiaoyiCollectionTool } from "./xiaoyi-collection-tool.js";
14
- import { xiaoyiDeleteCollectionTool } from "./xiaoyi-delete-collection-tool.js";
15
- import { calendarTool } from "./calendar-tool.js";
16
- import { searchCalendarTool } from "./search-calendar-tool.js";
17
- import { searchPhotoGalleryTool } from "./search-photo-gallery-tool.js";
18
- import { uploadPhotoTool } from "./upload-photo-tool.js";
19
- import { saveMediaToGalleryTool } from "./save-media-to-gallery-tool.js";
20
- import { searchFileTool } from "./search-file-tool.js";
21
- import { uploadFileTool } from "./upload-file-tool.js";
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";
1
+ import { createNoteTool } from "./note-tool.js";
2
+ import { createSearchNoteTool } from "./search-note-tool.js";
3
+ import { createModifyNoteTool } from "./modify-note-tool.js";
4
+ import { makeAlarmTool } from "./create-alarm-tool.js";
5
+ import { createSearchAlarmTool } from "./search-alarm-tool.js";
6
+ import { createModifyAlarmTool } from "./modify-alarm-tool.js";
7
+ import { createDeleteAlarmTool } from "./delete-alarm-tool.js";
8
+ import { createSearchContactTool } from "./search-contact-tool.js";
9
+ import { createCallPhoneTool } from "./call-phone-tool.js";
10
+ import { createSearchMessageTool } from "./search-message-tool.js";
11
+ import { createSendMessageTool } from "./send-message-tool.js";
12
+ import { createXiaoyiAddCollectionTool } from "./xiaoyi-add-collection-tool.js";
13
+ import { createXiaoyiCollectionTool } from "./xiaoyi-collection-tool.js";
14
+ import { createXiaoyiDeleteCollectionTool } from "./xiaoyi-delete-collection-tool.js";
15
+ import { createCalendarTool } from "./calendar-tool.js";
16
+ import { createSearchCalendarTool } from "./search-calendar-tool.js";
17
+ import { createSearchPhotoGalleryTool } from "./search-photo-gallery-tool.js";
18
+ import { createUploadPhotoTool } from "./upload-photo-tool.js";
19
+ import { createSaveMediaToGalleryTool } from "./save-media-to-gallery-tool.js";
20
+ import { createSearchFileTool } from "./search-file-tool.js";
21
+ import { createUploadFileTool } from "./upload-file-tool.js";
22
+ import { createSaveFileToPhoneTool } from "./save-file-to-phone-tool.js";
23
+ import { createSendEmailTool } from "./send-email-tool.js";
24
+ import { createSearchEmailTool } from "./search-email-tool.js";
25
25
  import { sendStatusUpdate } from "../formatter.js";
26
- import { getCurrentSessionContext } from "./session-manager.js";
27
26
  import { getCurrentTaskId, getCurrentMessageId } from "../task-manager.js";
28
- /**
29
- * 端工具注册表 —— 按 name 索引所有可通过 call_device_tool 调度的工具。
30
- */
31
- const deviceToolRegistry = new Map([
32
- [noteTool.name, noteTool],
33
- [searchNoteTool.name, searchNoteTool],
34
- [modifyNoteTool.name, modifyNoteTool],
35
- [createAlarmTool.name, createAlarmTool],
36
- [searchAlarmTool.name, searchAlarmTool],
37
- [modifyAlarmTool.name, modifyAlarmTool],
38
- [deleteAlarmTool.name, deleteAlarmTool],
39
- [searchContactTool.name, searchContactTool],
40
- [callPhoneTool.name, callPhoneTool],
41
- [searchMessageTool.name, searchMessageTool],
42
- [sendMessageTool.name, sendMessageTool],
43
- [xiaoyiAddCollectionTool.name, xiaoyiAddCollectionTool],
44
- [xiaoyiCollectionTool.name, xiaoyiCollectionTool],
45
- [xiaoyiDeleteCollectionTool.name, xiaoyiDeleteCollectionTool],
46
- [calendarTool.name, calendarTool],
47
- [searchCalendarTool.name, searchCalendarTool],
48
- [searchPhotoGalleryTool.name, searchPhotoGalleryTool],
49
- [uploadPhotoTool.name, uploadPhotoTool],
50
- [saveMediaToGalleryTool.name, saveMediaToGalleryTool],
51
- [searchFileTool.name, searchFileTool],
52
- [uploadFileTool.name, uploadFileTool],
53
- [saveFileToPhoneTool.name, saveFileToPhoneTool],
54
- [sendEmailTool.name, sendEmailTool],
55
- [searchEmailTool.name, searchEmailTool],
56
- ]);
57
27
  /**
58
28
  * call_device_tool - 通用端工具调度器。
59
29
  * LLM 必须先通过 get_xxx_tool_schema 获取具体工具 schema,再用本工具执行。
60
30
  */
61
- export const callDeviceTool = {
62
- name: "call_device_tool",
63
- label: "Call Device Tool",
64
- description: "用户设备侧工具调用。必须先调用get_xxx_tool_schema获取了具体的工具schema,才能使用本工具执行对应设备侧工具。",
65
- parameters: {
66
- type: "object",
67
- properties: {
68
- toolName: {
69
- type: "string",
70
- description: "要调用的具体端工具名称,即get_xxx_tool_schema返回的工具的name",
71
- },
72
- arguments: {
73
- type: "object",
74
- description: "工具所需的具体参数JSON键值对",
31
+ export function createCallDeviceTool(ctx) {
32
+ const { config, sessionId, taskId, messageId } = ctx;
33
+ const noteTool = createNoteTool(ctx);
34
+ const modifyNoteTool = createModifyNoteTool(ctx);
35
+ const createAlarmTool = makeAlarmTool(ctx);
36
+ const modifyAlarmTool = createModifyAlarmTool(ctx);
37
+ const deleteAlarmTool = createDeleteAlarmTool(ctx);
38
+ const callPhoneTool = createCallPhoneTool(ctx);
39
+ const calendarTool = createCalendarTool(ctx);
40
+ const searchNoteTool = createSearchNoteTool(ctx);
41
+ const searchMessageTool = createSearchMessageTool(ctx);
42
+ const sendMessageTool = createSendMessageTool(ctx);
43
+ const xiaoyiAddCollectionTool = createXiaoyiAddCollectionTool(ctx);
44
+ const xiaoyiCollectionTool = createXiaoyiCollectionTool(ctx);
45
+ const xiaoyiDeleteCollectionTool = createXiaoyiDeleteCollectionTool(ctx);
46
+ const searchPhotoGalleryTool = createSearchPhotoGalleryTool(ctx);
47
+ const uploadPhotoTool = createUploadPhotoTool(ctx);
48
+ const uploadFileTool = createUploadFileTool(ctx);
49
+ const sendEmailTool = createSendEmailTool(ctx);
50
+ const searchAlarmTool = createSearchAlarmTool(ctx);
51
+ const searchContactTool = createSearchContactTool(ctx);
52
+ const searchCalendarTool = createSearchCalendarTool(ctx);
53
+ const saveMediaToGalleryTool = createSaveMediaToGalleryTool(ctx);
54
+ const searchFileTool = createSearchFileTool(ctx);
55
+ const saveFileToPhoneTool = createSaveFileToPhoneTool(ctx);
56
+ const searchEmailTool = createSearchEmailTool(ctx);
57
+ /**
58
+ * 端工具注册表 —— 按 name 索引所有可通过 call_device_tool 调度的工具。
59
+ */
60
+ const deviceToolRegistry = new Map([
61
+ [noteTool.name, noteTool],
62
+ [searchNoteTool.name, searchNoteTool],
63
+ [modifyNoteTool.name, modifyNoteTool],
64
+ [createAlarmTool.name, createAlarmTool],
65
+ [searchAlarmTool.name, searchAlarmTool],
66
+ [modifyAlarmTool.name, modifyAlarmTool],
67
+ [deleteAlarmTool.name, deleteAlarmTool],
68
+ [searchContactTool.name, searchContactTool],
69
+ [callPhoneTool.name, callPhoneTool],
70
+ [searchMessageTool.name, searchMessageTool],
71
+ [sendMessageTool.name, sendMessageTool],
72
+ [xiaoyiAddCollectionTool.name, xiaoyiAddCollectionTool],
73
+ [xiaoyiCollectionTool.name, xiaoyiCollectionTool],
74
+ [xiaoyiDeleteCollectionTool.name, xiaoyiDeleteCollectionTool],
75
+ [calendarTool.name, calendarTool],
76
+ [searchCalendarTool.name, searchCalendarTool],
77
+ [searchPhotoGalleryTool.name, searchPhotoGalleryTool],
78
+ [uploadPhotoTool.name, uploadPhotoTool],
79
+ [saveMediaToGalleryTool.name, saveMediaToGalleryTool],
80
+ [searchFileTool.name, searchFileTool],
81
+ [uploadFileTool.name, uploadFileTool],
82
+ [saveFileToPhoneTool.name, saveFileToPhoneTool],
83
+ [sendEmailTool.name, sendEmailTool],
84
+ [searchEmailTool.name, searchEmailTool],
85
+ ]);
86
+ return {
87
+ name: "call_device_tool",
88
+ label: "Call Device Tool",
89
+ description: "用户设备侧工具调用。必须先调用get_xxx_tool_schema获取了具体的工具schema,才能使用本工具执行对应设备侧工具。",
90
+ parameters: {
91
+ type: "object",
92
+ properties: {
93
+ toolName: {
94
+ type: "string",
95
+ description: "要调用的具体端工具名称,即get_xxx_tool_schema返回的工具的name",
96
+ },
97
+ arguments: {
98
+ type: "object",
99
+ description: "工具所需的具体参数JSON键值对",
100
+ },
75
101
  },
102
+ required: ["toolName", "arguments"],
76
103
  },
77
- required: ["toolName", "arguments"],
78
- },
79
- async execute(toolCallId, params) {
80
- const { toolName, arguments: toolArgs } = params;
81
- // 向用户端发送具体工具名的状态更新
82
- const ctx = getCurrentSessionContext();
83
- if (ctx) {
84
- const currentTaskId = getCurrentTaskId(ctx.sessionId) ?? ctx.taskId;
85
- const currentMessageId = getCurrentMessageId(ctx.sessionId) ?? ctx.messageId;
104
+ async execute(toolCallId, params) {
105
+ const { toolName, arguments: toolArgs } = params;
106
+ // 向用户端发送具体工具名的状态更新
107
+ const currentTaskId = getCurrentTaskId(sessionId) ?? taskId;
108
+ const currentMessageId = getCurrentMessageId(sessionId) ?? messageId;
86
109
  try {
87
110
  await sendStatusUpdate({
88
- config: ctx.config,
89
- sessionId: ctx.sessionId,
111
+ config,
112
+ sessionId,
90
113
  taskId: currentTaskId,
91
114
  messageId: currentMessageId,
92
115
  text: `正在使用工具: ${toolName}...`,
@@ -96,35 +119,35 @@ export const callDeviceTool = {
96
119
  catch (_) {
97
120
  // 状态更新失败不影响工具执行
98
121
  }
99
- }
100
- const tool = deviceToolRegistry.get(toolName);
101
- if (!tool) {
102
- return {
103
- content: [
104
- {
105
- type: "text",
106
- text: `端工具${toolName}不存在。请确保toolName为get_xxx_tool_schema返回的工具的name。`,
107
- },
108
- ],
109
- };
110
- }
111
- try {
112
- return await tool.execute(toolCallId, toolArgs);
113
- }
114
- catch (error) {
115
- // ToolInputError (.name === "ToolInputError") 或其他参数校验错误
116
- if (error.name === "ToolInputError") {
122
+ const tool = deviceToolRegistry.get(toolName);
123
+ if (!tool) {
117
124
  return {
118
125
  content: [
119
126
  {
120
127
  type: "text",
121
- text: `端工具参数错误:${error.message}。请确保arguments符合get_xxx_tool_schema返回的工具schema。`,
128
+ text: `端工具${toolName}不存在。请确保toolName为get_xxx_tool_schema返回的工具的name。`,
122
129
  },
123
130
  ],
124
131
  };
125
132
  }
126
- // 非参数错误(网络超时等),直接向上抛出
127
- throw error;
128
- }
129
- },
130
- };
133
+ try {
134
+ return await tool.execute(toolCallId, toolArgs);
135
+ }
136
+ catch (error) {
137
+ // ToolInputError (.name === "ToolInputError") 或其他参数校验错误
138
+ if (error.name === "ToolInputError") {
139
+ return {
140
+ content: [
141
+ {
142
+ type: "text",
143
+ text: `端工具参数错误:${error.message}。请确保arguments符合get_xxx_tool_schema返回的工具schema。`,
144
+ },
145
+ ],
146
+ };
147
+ }
148
+ // 非参数错误(网络超时等),直接向上抛出
149
+ throw error;
150
+ }
151
+ },
152
+ };
153
+ }
@@ -1,5 +1,6 @@
1
+ import type { SessionContext } from "./session-manager.js";
1
2
  /**
2
3
  * XY call phone tool - makes a phone call on user's device.
3
4
  * Requires phoneNumber parameter and optional slotId (0 for primary SIM, 1 for secondary SIM).
4
5
  */
5
- export declare const callPhoneTool: any;
6
+ export declare function createCallPhoneTool(ctx: SessionContext): any;