@yushaw/sanqian-sdk 0.1.1 → 0.1.3

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.mts CHANGED
@@ -109,6 +109,8 @@ interface ToolCall {
109
109
  interface ChatResponse {
110
110
  message: ChatMessage;
111
111
  conversationId: string;
112
+ /** Conversation title (truncated from first user message, or LLM-generated after 3rd round) */
113
+ title?: string;
112
114
  usage?: {
113
115
  prompt_tokens: number;
114
116
  completion_tokens: number;
@@ -120,6 +122,8 @@ interface ChatStreamEvent {
120
122
  content?: string;
121
123
  tool_call?: ToolCall;
122
124
  conversationId?: string;
125
+ /** Conversation title (sent with "done" event) */
126
+ title?: string;
123
127
  error?: string;
124
128
  }
125
129
  /** Agent configuration for create/update */
@@ -249,10 +253,11 @@ declare class SanqianSDK {
249
253
  */
250
254
  isConnected(): boolean;
251
255
  /**
252
- * Ensure connected to Sanqian, auto-launching if needed.
256
+ * Wait for connection to be established.
253
257
  * Used internally by chat() and chatStream() for auto-reconnect.
258
+ * Relies on background scheduleReconnect() to do the actual reconnection.
254
259
  */
255
- private ensureConnected;
260
+ private waitForConnection;
256
261
  /**
257
262
  * Update tool list dynamically
258
263
  */
@@ -362,6 +367,10 @@ declare class SanqianSDK {
362
367
  * Remove event listener
363
368
  */
364
369
  off<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this;
370
+ /**
371
+ * Add one-time event listener (auto-removes after first call)
372
+ */
373
+ once<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this;
365
374
  private emit;
366
375
  private generateId;
367
376
  private createTimeout;
package/dist/index.d.ts CHANGED
@@ -109,6 +109,8 @@ interface ToolCall {
109
109
  interface ChatResponse {
110
110
  message: ChatMessage;
111
111
  conversationId: string;
112
+ /** Conversation title (truncated from first user message, or LLM-generated after 3rd round) */
113
+ title?: string;
112
114
  usage?: {
113
115
  prompt_tokens: number;
114
116
  completion_tokens: number;
@@ -120,6 +122,8 @@ interface ChatStreamEvent {
120
122
  content?: string;
121
123
  tool_call?: ToolCall;
122
124
  conversationId?: string;
125
+ /** Conversation title (sent with "done" event) */
126
+ title?: string;
123
127
  error?: string;
124
128
  }
125
129
  /** Agent configuration for create/update */
@@ -249,10 +253,11 @@ declare class SanqianSDK {
249
253
  */
250
254
  isConnected(): boolean;
251
255
  /**
252
- * Ensure connected to Sanqian, auto-launching if needed.
256
+ * Wait for connection to be established.
253
257
  * Used internally by chat() and chatStream() for auto-reconnect.
258
+ * Relies on background scheduleReconnect() to do the actual reconnection.
254
259
  */
255
- private ensureConnected;
260
+ private waitForConnection;
256
261
  /**
257
262
  * Update tool list dynamically
258
263
  */
@@ -362,6 +367,10 @@ declare class SanqianSDK {
362
367
  * Remove event listener
363
368
  */
364
369
  off<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this;
370
+ /**
371
+ * Add one-time event listener (auto-removes after first call)
372
+ */
373
+ once<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this;
365
374
  private emit;
366
375
  private generateId;
367
376
  private createTimeout;
package/dist/index.js CHANGED
@@ -469,7 +469,7 @@ var SanqianSDK = class {
469
469
  }
470
470
  }
471
471
  handleChatStream(message) {
472
- const { id, event, content, tool_call, tool_result, conversation_id, usage, error } = message;
472
+ const { id, event, content, tool_call, tool_result, conversation_id, title, usage, error } = message;
473
473
  if (!id) return;
474
474
  const handler = this.streamHandlers.get(id);
475
475
  if (!handler) {
@@ -502,6 +502,7 @@ var SanqianSDK = class {
502
502
  handler.onDone({
503
503
  message: message.message || { role: "assistant", content: "" },
504
504
  conversationId: conversation_id || "",
505
+ title,
505
506
  usage
506
507
  });
507
508
  break;
@@ -564,11 +565,14 @@ var SanqianSDK = class {
564
565
  }
565
566
  scheduleReconnect() {
566
567
  if (this.reconnectTimer) return;
567
- const delay = Math.min(
568
- 1e3 * Math.pow(2, this.state.reconnectAttempts),
569
- 3e4
568
+ const baseDelay = Math.min(
569
+ 500 * Math.pow(2, this.state.reconnectAttempts),
570
+ 5e3
571
+ // Cap at 5 seconds
570
572
  );
571
- console.log(`[SDK] Scheduling reconnect in ${delay}ms`);
573
+ const jitter = Math.random() * 500;
574
+ const delay = baseDelay + jitter;
575
+ console.log(`[SDK] Scheduling reconnect in ${Math.round(delay)}ms (attempt ${this.state.reconnectAttempts + 1})`);
572
576
  this.reconnectTimer = setTimeout(async () => {
573
577
  this.reconnectTimer = null;
574
578
  this.state.reconnectAttempts++;
@@ -657,43 +661,42 @@ var SanqianSDK = class {
657
661
  return this.state.connected && this.state.registered;
658
662
  }
659
663
  /**
660
- * Ensure connected to Sanqian, auto-launching if needed.
664
+ * Wait for connection to be established.
661
665
  * Used internally by chat() and chatStream() for auto-reconnect.
666
+ * Relies on background scheduleReconnect() to do the actual reconnection.
662
667
  */
663
- async ensureConnected() {
668
+ async waitForConnection(timeout = 3e4) {
664
669
  if (this.isConnected()) {
665
670
  return;
666
671
  }
667
- this.stopReconnect();
668
- const info = this.discovery.read();
669
- if (info) {
670
- console.log("[SDK] Sanqian is running, reconnecting...");
671
- await this.connectWithInfo(info);
672
- return;
673
- }
674
- if (this.config.autoLaunchSanqian) {
675
- console.log("[SDK] Sanqian not running, attempting to launch...");
676
- const launched = this.discovery.launchSanqian(this.config.sanqianPath);
677
- if (!launched) {
678
- throw new Error("Failed to launch Sanqian. Please start it manually.");
672
+ console.log("[SDK] Not connected, waiting for reconnection...");
673
+ if (!this.reconnectTimer) {
674
+ const info = this.discovery.read();
675
+ if (!info && this.config.autoLaunchSanqian) {
676
+ console.log("[SDK] Sanqian not running, attempting to launch...");
677
+ const launched = this.discovery.launchSanqian(this.config.sanqianPath);
678
+ if (launched) {
679
+ this.scheduleReconnect();
680
+ }
681
+ } else if (!info) {
682
+ throw new Error("Sanqian is not running. Please start it manually.");
683
+ } else {
684
+ this.scheduleReconnect();
679
685
  }
680
686
  }
681
687
  return new Promise((resolve, reject) => {
682
- console.log("[SDK] Waiting for Sanqian to start...");
683
- const timeout = setTimeout(() => {
684
- this.discovery.stopWatching();
685
- reject(new Error("Sanqian connection timeout. Please ensure Sanqian is running."));
686
- }, 3e4);
687
- this.discovery.startWatching(async (newInfo) => {
688
- if (newInfo) {
689
- clearTimeout(timeout);
690
- this.discovery.stopWatching();
691
- try {
692
- await this.connectWithInfo(newInfo);
693
- resolve();
694
- } catch (e) {
695
- reject(e);
696
- }
688
+ let resolved = false;
689
+ const timer = setTimeout(() => {
690
+ if (!resolved) {
691
+ resolved = true;
692
+ reject(new Error("Connection timeout. Please ensure Sanqian is running."));
693
+ }
694
+ }, timeout);
695
+ this.once("registered", () => {
696
+ if (!resolved) {
697
+ resolved = true;
698
+ clearTimeout(timer);
699
+ resolve();
697
700
  }
698
701
  });
699
702
  });
@@ -906,7 +909,7 @@ var SanqianSDK = class {
906
909
  async chat(agentId, messages, options) {
907
910
  if (!this.isConnected()) {
908
911
  console.log("[SDK] Not connected, attempting to reconnect...");
909
- await this.ensureConnected();
912
+ await this.waitForConnection();
910
913
  }
911
914
  const msgId = this.generateId();
912
915
  const message = {
@@ -929,6 +932,7 @@ var SanqianSDK = class {
929
932
  return {
930
933
  message: response.message,
931
934
  conversationId: response.conversation_id,
935
+ title: response.title,
932
936
  usage: response.usage
933
937
  };
934
938
  }
@@ -947,7 +951,7 @@ var SanqianSDK = class {
947
951
  async *chatStream(agentId, messages, options) {
948
952
  if (!this.isConnected()) {
949
953
  console.log("[SDK] Not connected, attempting to reconnect...");
950
- await this.ensureConnected();
954
+ await this.waitForConnection();
951
955
  }
952
956
  const msgId = this.generateId();
953
957
  const message = {
@@ -975,7 +979,8 @@ var SanqianSDK = class {
975
979
  onDone: (response) => {
976
980
  eventQueue.push({
977
981
  type: "done",
978
- conversationId: response.conversationId
982
+ conversationId: response.conversationId,
983
+ title: response.title
979
984
  });
980
985
  done = true;
981
986
  resolveNext?.();
@@ -1043,6 +1048,16 @@ var SanqianSDK = class {
1043
1048
  this.eventListeners.get(event)?.delete(listener);
1044
1049
  return this;
1045
1050
  }
1051
+ /**
1052
+ * Add one-time event listener (auto-removes after first call)
1053
+ */
1054
+ once(event, listener) {
1055
+ const onceWrapper = ((...args) => {
1056
+ this.off(event, onceWrapper);
1057
+ listener(...args);
1058
+ });
1059
+ return this.on(event, onceWrapper);
1060
+ }
1046
1061
  emit(event, ...args) {
1047
1062
  const listeners = this.eventListeners.get(event);
1048
1063
  if (listeners) {
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/client.ts","../src/discovery.ts"],"sourcesContent":["/**\n * Sanqian SDK\n *\n * TypeScript SDK for integrating third-party apps with Sanqian AI Assistant.\n *\n * @example\n * ```typescript\n * import { SanqianSDK } from '@yushaw/sanqian-sdk';\n *\n * const sdk = new SanqianSDK({\n * appName: 'my-app',\n * appVersion: '1.0.0',\n * tools: [\n * {\n * name: 'my_tool',\n * description: 'Does something useful',\n * parameters: { type: 'object', properties: {} },\n * handler: async (args) => {\n * return { success: true };\n * }\n * }\n * ]\n * });\n *\n * await sdk.connect();\n * ```\n */\n\nexport { SanqianSDK, Conversation } from \"./client\";\nexport { DiscoveryManager } from \"./discovery\";\n\n// Types\nexport type {\n SDKConfig,\n ToolDefinition,\n JSONSchema,\n JSONSchemaProperty,\n ConnectionInfo,\n ConnectionState,\n ChatRequest,\n ChatMessage,\n ChatResponse,\n ChatStreamEvent,\n ToolCall,\n SDKEvents,\n SDKEventName,\n // Private Agent API\n AgentConfig,\n AgentInfo,\n AgentUpdateConfig,\n ConversationInfo,\n ConversationMessage,\n ConversationDetail,\n // Remote Tools\n RemoteToolDefinition,\n} from \"./types\";\n","/**\n * Sanqian SDK Client\n *\n * Main class for connecting to Sanqian and registering tools.\n */\n\nimport WebSocket from \"ws\";\nimport { DiscoveryManager } from \"./discovery\";\nimport type {\n SDKConfig,\n ConnectionInfo,\n ConnectionState,\n ToolDefinition,\n RegisterMessage,\n RegisterAckMessage,\n ToolCallMessage,\n ToolResultMessage,\n HeartbeatMessage,\n SDKEvents,\n SDKEventName,\n JSONSchema,\n AgentConfig,\n AgentInfo,\n AgentUpdateConfig,\n ConversationInfo,\n ConversationDetail,\n CreateAgentMessage,\n CreateAgentAckMessage,\n UpdateAgentMessage,\n UpdateAgentAckMessage,\n ListAgentsMessage,\n ListAgentsAckMessage,\n DeleteAgentMessage,\n DeleteAgentAckMessage,\n ListConversationsMessage,\n ListConversationsAckMessage,\n GetConversationMessage,\n GetConversationAckMessage,\n DeleteConversationMessage,\n DeleteConversationAckMessage,\n ChatMessage,\n ChatResponse,\n ChatRequestMessage,\n ChatResponseMessage,\n ChatStreamMessage,\n ChatStreamEvent,\n RemoteToolDefinition,\n} from \"./types\";\n\ntype EventListener<T extends SDKEventName> = SDKEvents[T];\n\nexport class SanqianSDK {\n private config: SDKConfig;\n private discovery: DiscoveryManager;\n private ws: WebSocket | null = null;\n private connectionInfo: ConnectionInfo | null = null;\n\n private state: ConnectionState = {\n connected: false,\n registering: false,\n registered: false,\n reconnectAttempts: 0,\n };\n\n // Tool handlers by name\n private toolHandlers: Map<string, (args: unknown) => Promise<unknown>> =\n new Map();\n\n // Pending request futures\n private pendingRequests: Map<\n string,\n {\n resolve: (value: unknown) => void;\n reject: (error: Error) => void;\n }\n > = new Map();\n\n // Timers\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n private reconnectTimer: ReturnType<typeof setTimeout> | null = null;\n\n // Event listeners\n private eventListeners: Map<SDKEventName, Set<EventListener<SDKEventName>>> =\n new Map();\n\n constructor(config: SDKConfig) {\n this.config = {\n reconnectInterval: 5000,\n heartbeatInterval: 30000,\n toolExecutionTimeout: 30000,\n ...config,\n };\n this.discovery = new DiscoveryManager();\n\n // Register tool handlers\n for (const tool of config.tools) {\n this.toolHandlers.set(tool.name, tool.handler);\n }\n }\n\n // ============================================\n // Lifecycle\n // ============================================\n\n /**\n * Connect to Sanqian\n *\n * Reads connection info, establishes WebSocket, and registers app.\n * Returns when registration is complete.\n *\n * If autoLaunchSanqian is enabled and Sanqian is not running,\n * SDK will attempt to start it in hidden/tray mode.\n */\n async connect(): Promise<void> {\n // Read connection info\n const info = this.discovery.read();\n\n if (!info) {\n // Sanqian not running\n // Try to auto-launch if enabled\n if (this.config.autoLaunchSanqian) {\n console.log(\"[SDK] Sanqian not running, attempting to launch...\");\n const launched = this.discovery.launchSanqian(this.config.sanqianPath);\n if (launched) {\n console.log(\"[SDK] Sanqian launch initiated, waiting for startup...\");\n } else {\n console.warn(\"[SDK] Failed to launch Sanqian, will wait for manual start\");\n }\n }\n\n // Start watching and wait for connection\n return new Promise((resolve, reject) => {\n console.log(\"[SDK] Waiting for Sanqian...\");\n\n const timeout = setTimeout(() => {\n this.discovery.stopWatching();\n reject(new Error(\"Sanqian connection timeout\"));\n }, 60000); // 60 second timeout\n\n this.discovery.startWatching(async (newInfo) => {\n if (newInfo) {\n clearTimeout(timeout);\n this.discovery.stopWatching();\n try {\n await this.connectWithInfo(newInfo);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n });\n });\n }\n\n await this.connectWithInfo(info);\n }\n\n /**\n * Connect with known connection info\n */\n private async connectWithInfo(info: ConnectionInfo): Promise<void> {\n this.connectionInfo = info;\n const url = this.discovery.buildWebSocketUrl(info);\n\n return new Promise((resolve, reject) => {\n console.log(`[SDK] Connecting to ${url}`);\n\n this.ws = new WebSocket(url);\n\n const connectTimeout = setTimeout(() => {\n reject(new Error(\"WebSocket connection timeout\"));\n this.ws?.close();\n }, 10000);\n\n this.ws.on(\"open\", async () => {\n clearTimeout(connectTimeout);\n console.log(\"[SDK] WebSocket connected\");\n this.state.connected = true;\n this.state.reconnectAttempts = 0;\n this.emit(\"connected\");\n\n try {\n await this.register();\n resolve();\n } catch (e) {\n reject(e);\n }\n });\n\n this.ws.on(\"close\", (code, reason) => {\n console.log(`[SDK] WebSocket closed: ${code} ${reason.toString()}`);\n this.handleDisconnect(reason.toString());\n });\n\n this.ws.on(\"error\", (error) => {\n console.error(\"[SDK] WebSocket error:\", error);\n this.state.lastError = error;\n this.emit(\"error\", error);\n });\n\n this.ws.on(\"message\", (data) => {\n try {\n const message = JSON.parse(data.toString());\n this.handleMessage(message);\n } catch (e) {\n console.error(\"[SDK] Failed to parse message:\", e);\n }\n });\n });\n }\n\n /**\n * Disconnect from Sanqian\n */\n async disconnect(): Promise<void> {\n this.stopHeartbeat();\n this.stopReconnect();\n this.discovery.stopWatching();\n\n if (this.ws) {\n this.ws.close(1000, \"Client disconnect\");\n this.ws = null;\n }\n\n this.state = {\n connected: false,\n registering: false,\n registered: false,\n reconnectAttempts: 0,\n };\n }\n\n // ============================================\n // Registration\n // ============================================\n\n private async register(): Promise<void> {\n this.state.registering = true;\n\n const msgId = this.generateId();\n\n const message: RegisterMessage = {\n id: msgId,\n type: \"register\",\n app: {\n name: this.config.appName,\n version: this.config.appVersion,\n display_name: this.config.displayName,\n launch_command: this.config.launchCommand,\n },\n tools: this.config.tools.map((t) => ({\n name: `${this.config.appName}:${t.name}`,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n try {\n const response = await this.sendAndWait<RegisterAckMessage>(\n message,\n msgId,\n 10000\n );\n\n if (!response.success) {\n throw new Error(response.error || \"Registration failed\");\n }\n\n this.state.registering = false;\n this.state.registered = true;\n this.startHeartbeat();\n this.emit(\"registered\");\n\n console.log(`[SDK] Registered as '${this.config.appName}'`);\n } catch (e) {\n this.state.registering = false;\n throw e;\n }\n }\n\n // ============================================\n // Message Handling\n // ============================================\n\n private handleMessage(message: { id?: string; type: string; [key: string]: unknown }): void {\n const { id, type } = message;\n\n // Check if this is a response to a pending request\n if (id && this.pendingRequests.has(id)) {\n const pending = this.pendingRequests.get(id)!;\n this.pendingRequests.delete(id);\n pending.resolve(message);\n return;\n }\n\n // Handle server-initiated messages\n switch (type) {\n case \"tool_call\":\n this.handleToolCall(message as unknown as ToolCallMessage);\n break;\n\n case \"heartbeat_ack\":\n // Heartbeat acknowledged, no action needed\n break;\n\n case \"chat_stream\":\n // Handle streaming chat response\n this.handleChatStream(message as unknown as ChatStreamMessage);\n break;\n\n default:\n console.warn(`[SDK] Unknown message type: ${type}`);\n }\n }\n\n private handleChatStream(message: ChatStreamMessage): void {\n const { id, event, content, tool_call, tool_result, conversation_id, usage, error } = message;\n\n if (!id) return;\n\n const handler = this.streamHandlers.get(id);\n if (!handler) {\n console.warn(`[SDK] No stream handler for message ${id}`);\n return;\n }\n\n switch (event) {\n case \"text\":\n handler.onEvent({ type: \"text\", content });\n break;\n\n case \"tool_call\":\n handler.onEvent({ type: \"tool_call\", tool_call });\n break;\n\n case \"tool_result\":\n if (tool_result) {\n handler.onEvent({\n type: \"tool_call\",\n tool_call: {\n id: tool_result.call_id,\n type: \"function\",\n function: {\n name: \"tool_result\",\n arguments: JSON.stringify(tool_result),\n },\n },\n });\n }\n break;\n\n case \"done\":\n handler.onDone({\n message: message.message || { role: \"assistant\", content: \"\" },\n conversationId: conversation_id || \"\",\n usage,\n });\n break;\n\n case \"error\":\n handler.onError(new Error(error || \"Unknown stream error\"));\n break;\n }\n }\n\n private async handleToolCall(message: ToolCallMessage): Promise<void> {\n console.log(`[SDK] handleToolCall received:`, JSON.stringify(message));\n const { id, call_id, name, arguments: args } = message;\n const msgId = id || call_id;\n\n // Extract actual tool name (remove app prefix)\n const toolName = name.includes(\":\") ? name.split(\":\")[1] : name;\n console.log(`[SDK] Looking for handler: '${toolName}', available handlers:`, Array.from(this.toolHandlers.keys()));\n const handler = this.toolHandlers.get(toolName);\n\n // Emit event for debugging/logging\n this.emit(\"tool_call\", { name: toolName, arguments: args });\n\n if (!handler) {\n await this.sendToolResult(msgId, call_id, false, undefined, `Tool '${toolName}' not found`);\n return;\n }\n\n try {\n console.log(`[SDK] Executing tool '${toolName}' with args:`, args);\n // Execute with timeout\n const result = await Promise.race([\n handler(args),\n this.createTimeout(this.config.toolExecutionTimeout!),\n ]);\n\n console.log(`[SDK] Tool '${toolName}' completed successfully`);\n await this.sendToolResult(msgId, call_id, true, result);\n } catch (e) {\n const error = e instanceof Error ? e.message : String(e);\n console.log(`[SDK] Tool '${toolName}' failed:`, error);\n await this.sendToolResult(msgId, call_id, false, undefined, error);\n }\n }\n\n private async sendToolResult(\n id: string,\n callId: string,\n success: boolean,\n result?: unknown,\n error?: string\n ): Promise<void> {\n const message: ToolResultMessage = {\n id,\n type: \"tool_result\",\n call_id: callId,\n success,\n result,\n error,\n };\n\n console.log(`[SDK] Sending tool_result for ${callId}:`, success ? 'success' : `error: ${error}`);\n this.send(message);\n }\n\n // ============================================\n // Connection Management\n // ============================================\n\n private handleDisconnect(reason: string): void {\n this.stopHeartbeat();\n this.state.connected = false;\n this.state.registered = false;\n this.emit(\"disconnected\", reason);\n\n // Reject pending requests\n for (const [, pending] of this.pendingRequests) {\n pending.reject(new Error(\"Disconnected\"));\n }\n this.pendingRequests.clear();\n\n // Schedule reconnect\n this.scheduleReconnect();\n }\n\n private scheduleReconnect(): void {\n if (this.reconnectTimer) return;\n\n // Exponential backoff: 1s, 2s, 4s, 8s, ... up to 30s\n const delay = Math.min(\n 1000 * Math.pow(2, this.state.reconnectAttempts),\n 30000\n );\n\n console.log(`[SDK] Scheduling reconnect in ${delay}ms`);\n\n this.reconnectTimer = setTimeout(async () => {\n this.reconnectTimer = null;\n this.state.reconnectAttempts++;\n\n // Re-read connection info\n const info = this.discovery.read();\n if (info) {\n try {\n await this.connectWithInfo(info);\n } catch (e) {\n console.error(\"[SDK] Reconnect failed:\", e);\n this.scheduleReconnect();\n }\n } else {\n // Sanqian not running, keep trying\n this.scheduleReconnect();\n }\n }, delay);\n }\n\n private stopReconnect(): void {\n if (this.reconnectTimer) {\n clearTimeout(this.reconnectTimer);\n this.reconnectTimer = null;\n }\n }\n\n // ============================================\n // Heartbeat\n // ============================================\n\n private startHeartbeat(): void {\n this.stopHeartbeat();\n\n this.heartbeatTimer = setInterval(() => {\n const message: HeartbeatMessage = {\n type: \"heartbeat\",\n timestamp: new Date().toISOString(),\n };\n this.send(message);\n }, this.config.heartbeatInterval);\n }\n\n private stopHeartbeat(): void {\n if (this.heartbeatTimer) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n }\n\n // ============================================\n // Communication\n // ============================================\n\n private send(message: object): void {\n if (this.ws && this.ws.readyState === WebSocket.OPEN) {\n const data = JSON.stringify(message);\n console.log(`[SDK] WebSocket send:`, data.substring(0, 200));\n this.ws.send(data);\n } else {\n console.error(`[SDK] WebSocket not open, cannot send:`, message);\n }\n }\n\n private sendAndWait<T>(\n message: object,\n id: string,\n timeout: number\n ): Promise<T> {\n return new Promise((resolve, reject) => {\n const timer = setTimeout(() => {\n this.pendingRequests.delete(id);\n reject(new Error(\"Request timeout\"));\n }, timeout);\n\n this.pendingRequests.set(id, {\n resolve: (value) => {\n clearTimeout(timer);\n resolve(value as T);\n },\n reject: (error) => {\n clearTimeout(timer);\n reject(error);\n },\n });\n\n this.send(message);\n });\n }\n\n // ============================================\n // Public API\n // ============================================\n\n /**\n * Get current connection state\n */\n getState(): ConnectionState {\n return { ...this.state };\n }\n\n /**\n * Check if connected and registered\n */\n isConnected(): boolean {\n return this.state.connected && this.state.registered;\n }\n\n /**\n * Ensure connected to Sanqian, auto-launching if needed.\n * Used internally by chat() and chatStream() for auto-reconnect.\n */\n private async ensureConnected(): Promise<void> {\n if (this.isConnected()) {\n return;\n }\n\n // Stop any pending reconnect timer\n this.stopReconnect();\n\n // Try to read connection info first\n const info = this.discovery.read();\n\n if (info) {\n // Sanqian is running, just reconnect\n console.log(\"[SDK] Sanqian is running, reconnecting...\");\n await this.connectWithInfo(info);\n return;\n }\n\n // Sanqian not running, try to launch if enabled\n if (this.config.autoLaunchSanqian) {\n console.log(\"[SDK] Sanqian not running, attempting to launch...\");\n const launched = this.discovery.launchSanqian(this.config.sanqianPath);\n if (!launched) {\n throw new Error(\"Failed to launch Sanqian. Please start it manually.\");\n }\n }\n\n // Wait for Sanqian to start\n return new Promise((resolve, reject) => {\n console.log(\"[SDK] Waiting for Sanqian to start...\");\n\n const timeout = setTimeout(() => {\n this.discovery.stopWatching();\n reject(new Error(\"Sanqian connection timeout. Please ensure Sanqian is running.\"));\n }, 30000); // 30 second timeout for reconnect\n\n this.discovery.startWatching(async (newInfo) => {\n if (newInfo) {\n clearTimeout(timeout);\n this.discovery.stopWatching();\n try {\n await this.connectWithInfo(newInfo);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n });\n });\n }\n\n /**\n * Update tool list dynamically\n */\n async updateTools(tools: ToolDefinition[]): Promise<void> {\n // Update local handlers\n this.toolHandlers.clear();\n for (const tool of tools) {\n this.toolHandlers.set(tool.name, tool.handler);\n }\n\n // Send update to Sanqian\n if (this.isConnected()) {\n const msgId = this.generateId();\n const message = {\n id: msgId,\n type: \"tools_update\",\n tools: tools.map((t) => ({\n name: `${this.config.appName}:${t.name}`,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n await this.sendAndWait(message, msgId, 5000);\n }\n }\n\n // ============================================\n // Private Agent API\n // ============================================\n\n /**\n * Create or update a private agent\n *\n * Private agents are only visible to this SDK app, not in Sanqian UI.\n * If agent with same ID exists, it will be updated.\n *\n * @param config - Agent configuration\n * @returns Full agent info (or agent_id string for backward compatibility)\n */\n async createAgent(config: AgentConfig): Promise<AgentInfo> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: CreateAgentMessage = {\n id: msgId,\n type: \"create_agent\",\n agent: config,\n };\n\n const response = await this.sendAndWait<CreateAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to create agent\");\n }\n\n // Return full agent info if available, fallback to minimal info\n if (response.agent) {\n return response.agent;\n }\n\n // Backward compatibility: construct minimal AgentInfo from agent_id\n return {\n agent_id: response.agent_id!,\n name: config.name,\n description: config.description,\n tools: config.tools || [],\n };\n }\n\n /**\n * List all private agents owned by this app\n */\n async listAgents(): Promise<AgentInfo[]> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: ListAgentsMessage = {\n id: msgId,\n type: \"list_agents\",\n };\n\n const response = await this.sendAndWait<ListAgentsAckMessage>(message, msgId, 10000);\n\n if (response.error) {\n throw new Error(response.error);\n }\n\n return response.agents;\n }\n\n /**\n * Delete a private agent\n */\n async deleteAgent(agentId: string): Promise<void> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: DeleteAgentMessage = {\n id: msgId,\n type: \"delete_agent\",\n agent_id: agentId,\n };\n\n const response = await this.sendAndWait<DeleteAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to delete agent\");\n }\n }\n\n /**\n * Update a private agent\n *\n * Only updates the fields that are provided.\n * @param agentId - Agent ID (short name or full)\n * @param updates - Fields to update\n * @returns Updated agent info\n */\n async updateAgent(\n agentId: string,\n updates: Omit<AgentUpdateConfig, \"agent_id\">\n ): Promise<AgentInfo> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: UpdateAgentMessage = {\n id: msgId,\n type: \"update_agent\",\n agent: {\n agent_id: agentId,\n ...updates,\n },\n };\n\n const response = await this.sendAndWait<UpdateAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success || !response.agent) {\n throw new Error(response.error || \"Failed to update agent\");\n }\n\n return response.agent;\n }\n\n // ============================================\n // Conversation API\n // ============================================\n\n /**\n * List conversations for this app\n */\n async listConversations(options?: {\n agentId?: string;\n limit?: number;\n offset?: number;\n }): Promise<{ conversations: ConversationInfo[]; total: number }> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: ListConversationsMessage = {\n id: msgId,\n type: \"list_conversations\",\n agent_id: options?.agentId,\n limit: options?.limit,\n offset: options?.offset,\n };\n\n const response = await this.sendAndWait<ListConversationsAckMessage>(message, msgId, 10000);\n\n if (response.error) {\n throw new Error(response.error);\n }\n\n return {\n conversations: response.conversations,\n total: response.total,\n };\n }\n\n /**\n * Get conversation details with messages\n */\n async getConversation(\n conversationId: string,\n options?: {\n includeMessages?: boolean;\n messageLimit?: number;\n messageOffset?: number;\n }\n ): Promise<ConversationDetail> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: GetConversationMessage = {\n id: msgId,\n type: \"get_conversation\",\n conversation_id: conversationId,\n include_messages: options?.includeMessages ?? true,\n message_limit: options?.messageLimit,\n message_offset: options?.messageOffset,\n };\n\n const response = await this.sendAndWait<GetConversationAckMessage>(message, msgId, 10000);\n\n if (!response.success || !response.conversation) {\n throw new Error(response.error || \"Failed to get conversation\");\n }\n\n return response.conversation;\n }\n\n /**\n * Delete a conversation\n */\n async deleteConversation(conversationId: string): Promise<void> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: DeleteConversationMessage = {\n id: msgId,\n type: \"delete_conversation\",\n conversation_id: conversationId,\n };\n\n const response = await this.sendAndWait<DeleteConversationAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to delete conversation\");\n }\n }\n\n // ============================================\n // Chat API\n // ============================================\n\n // Pending stream handlers for streaming chat\n private streamHandlers: Map<string, {\n onEvent: (event: ChatStreamEvent) => void;\n onDone: (response: ChatResponse) => void;\n onError: (error: Error) => void;\n }> = new Map();\n\n /**\n * Send a chat message to an agent (non-streaming)\n *\n * Supports two modes:\n * - Stateless (no conversationId): Caller maintains message history\n * - Stateful (with conversationId): Server stores messages in conversations table\n *\n * @param agentId - Agent ID (short name or full, SDK auto-prefixes)\n * @param messages - Messages to send\n * @param options - Optional settings including conversationId and remoteTools\n * @returns Chat response with assistant message and conversation ID\n */\n async chat(\n agentId: string,\n messages: ChatMessage[],\n options?: {\n conversationId?: string;\n remoteTools?: RemoteToolDefinition[];\n }\n ): Promise<ChatResponse> {\n // Auto-reconnect if disconnected\n if (!this.isConnected()) {\n console.log(\"[SDK] Not connected, attempting to reconnect...\");\n await this.ensureConnected();\n }\n\n const msgId = this.generateId();\n const message: ChatRequestMessage = {\n id: msgId,\n type: \"chat\",\n agent_id: agentId,\n messages,\n conversation_id: options?.conversationId,\n stream: false,\n remote_tools: options?.remoteTools?.map((t) => ({\n name: t.name,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n // Longer timeout for chat (agent may take time with complex multi-tool tasks)\n const response = await this.sendAndWait<ChatResponseMessage>(message, msgId, 600000); // 10 minutes\n\n if (!response.success) {\n throw new Error(response.error || \"Chat request failed\");\n }\n\n return {\n message: response.message!,\n conversationId: response.conversation_id!,\n usage: response.usage,\n };\n }\n\n /**\n * Send a chat message to an agent with streaming response\n *\n * Supports two modes:\n * - Stateless (no conversationId): Caller maintains message history\n * - Stateful (with conversationId): Server stores messages in conversations table\n *\n * @param agentId - Agent ID (short name or full, SDK auto-prefixes)\n * @param messages - Messages to send\n * @param options - Optional settings including conversationId and remoteTools\n * @returns AsyncIterable of stream events\n */\n async *chatStream(\n agentId: string,\n messages: ChatMessage[],\n options?: {\n conversationId?: string;\n remoteTools?: RemoteToolDefinition[];\n }\n ): AsyncGenerator<ChatStreamEvent> {\n // Auto-reconnect if disconnected\n if (!this.isConnected()) {\n console.log(\"[SDK] Not connected, attempting to reconnect...\");\n await this.ensureConnected();\n }\n\n const msgId = this.generateId();\n const message: ChatRequestMessage = {\n id: msgId,\n type: \"chat\",\n agent_id: agentId,\n messages,\n conversation_id: options?.conversationId,\n stream: true,\n remote_tools: options?.remoteTools?.map((t) => ({\n name: t.name,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n // Create a queue to yield events\n const eventQueue: ChatStreamEvent[] = [];\n let done = false;\n let error: Error | null = null;\n let resolveNext: (() => void) | null = null;\n\n // Register stream handler\n this.streamHandlers.set(msgId, {\n onEvent: (event) => {\n eventQueue.push(event);\n resolveNext?.();\n },\n onDone: (response) => {\n eventQueue.push({\n type: \"done\",\n conversationId: response.conversationId,\n });\n done = true;\n resolveNext?.();\n },\n onError: (e) => {\n error = e;\n resolveNext?.();\n },\n });\n\n try {\n // Send request\n this.send(message);\n\n // Yield events as they arrive\n while (!done && !error) {\n if (eventQueue.length > 0) {\n yield eventQueue.shift()!;\n } else {\n // Wait for next event\n await new Promise<void>((resolve) => {\n resolveNext = resolve;\n });\n resolveNext = null;\n }\n }\n\n // Yield remaining events\n while (eventQueue.length > 0) {\n yield eventQueue.shift()!;\n }\n\n if (error) {\n throw error;\n }\n } finally {\n this.streamHandlers.delete(msgId);\n }\n }\n\n /**\n * Start a new conversation with an agent\n *\n * Returns a Conversation object for convenient multi-turn chat.\n *\n * @example\n * ```typescript\n * const conv = sdk.startConversation('assistant')\n * const r1 = await conv.send('Hello')\n * const r2 = await conv.send('Follow up question')\n * console.log(conv.id) // conversation ID\n * ```\n */\n startConversation(agentId: string): Conversation {\n return new Conversation(this, agentId);\n }\n\n // ============================================\n // Events\n // ============================================\n\n /**\n * Add event listener\n */\n on<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n if (!this.eventListeners.has(event)) {\n this.eventListeners.set(event, new Set());\n }\n this.eventListeners.get(event)!.add(listener as EventListener<SDKEventName>);\n return this;\n }\n\n /**\n * Remove event listener\n */\n off<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n this.eventListeners.get(event)?.delete(listener as EventListener<SDKEventName>);\n return this;\n }\n\n private emit<T extends SDKEventName>(\n event: T,\n ...args: Parameters<SDKEvents[T]>\n ): void {\n const listeners = this.eventListeners.get(event);\n if (listeners) {\n for (const listener of listeners) {\n try {\n (listener as (...args: unknown[]) => void)(...args);\n } catch (e) {\n console.error(`[SDK] Event listener error for '${event}':`, e);\n }\n }\n }\n }\n\n // ============================================\n // Utilities\n // ============================================\n\n private generateId(): string {\n return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;\n }\n\n private createTimeout(ms: number): Promise<never> {\n return new Promise((_, reject) => {\n setTimeout(() => reject(new Error(\"Timeout\")), ms);\n });\n }\n}\n\n/**\n * Conversation helper for multi-turn chat\n *\n * Automatically manages conversationId for stateful conversations.\n */\nexport class Conversation {\n private sdk: SanqianSDK;\n private agentId: string;\n private _conversationId: string | null = null;\n\n constructor(sdk: SanqianSDK, agentId: string, conversationId?: string) {\n this.sdk = sdk;\n this.agentId = agentId;\n this._conversationId = conversationId || null;\n }\n\n /**\n * Get the conversation ID (available after first message)\n */\n get id(): string | null {\n return this._conversationId;\n }\n\n /**\n * Send a message and get a response\n *\n * First call creates a new conversation, subsequent calls continue it.\n */\n async send(content: string, options?: { remoteTools?: RemoteToolDefinition[] }): Promise<ChatResponse> {\n const response = await this.sdk.chat(\n this.agentId,\n [{ role: \"user\", content }],\n {\n conversationId: this._conversationId || undefined,\n remoteTools: options?.remoteTools,\n }\n );\n\n // Store conversation ID for subsequent calls\n if (response.conversationId && !this._conversationId) {\n this._conversationId = response.conversationId;\n }\n\n return response;\n }\n\n /**\n * Send a message with streaming response\n */\n async *sendStream(\n content: string,\n options?: { remoteTools?: RemoteToolDefinition[] }\n ): AsyncGenerator<ChatStreamEvent> {\n const stream = this.sdk.chatStream(\n this.agentId,\n [{ role: \"user\", content }],\n {\n conversationId: this._conversationId || undefined,\n remoteTools: options?.remoteTools,\n }\n );\n\n for await (const event of stream) {\n // Capture conversation ID from done event\n if (event.type === \"done\" && event.conversationId && !this._conversationId) {\n this._conversationId = event.conversationId;\n }\n yield event;\n }\n }\n\n /**\n * Delete this conversation\n */\n async delete(): Promise<void> {\n if (!this._conversationId) {\n throw new Error(\"No conversation to delete\");\n }\n await this.sdk.deleteConversation(this._conversationId);\n this._conversationId = null;\n }\n\n /**\n * Get conversation details including message history\n */\n async getDetails(options?: { messageLimit?: number }): Promise<ConversationDetail> {\n if (!this._conversationId) {\n throw new Error(\"No conversation to get details for\");\n }\n return this.sdk.getConversation(this._conversationId, {\n includeMessages: true,\n messageLimit: options?.messageLimit,\n });\n }\n}\n","/**\n * Service Discovery Module\n *\n * Reads Sanqian's connection info from ~/.sanqian/runtime/connection.json\n * and monitors file changes for reconnection.\n * Also handles auto-launching Sanqian when needed.\n */\n\nimport { existsSync, readFileSync, watch, type FSWatcher } from \"fs\";\nimport { homedir, platform } from \"os\";\nimport { join } from \"path\";\nimport { spawn } from \"child_process\";\nimport type { ConnectionInfo } from \"./types\";\n\nexport class DiscoveryManager {\n private connectionInfo: ConnectionInfo | null = null;\n private watcher: FSWatcher | null = null;\n private onChange: ((info: ConnectionInfo | null) => void) | null = null;\n private pollInterval: ReturnType<typeof setInterval> | null = null;\n\n /**\n * Get the path to connection.json\n */\n getConnectionFilePath(): string {\n return join(homedir(), \".sanqian\", \"runtime\", \"connection.json\");\n }\n\n /**\n * Read and validate connection info\n *\n * Returns null if:\n * - File doesn't exist\n * - File is invalid JSON\n * - Process is not running\n */\n read(): ConnectionInfo | null {\n const filePath = this.getConnectionFilePath();\n\n if (!existsSync(filePath)) {\n return null;\n }\n\n try {\n const content = readFileSync(filePath, \"utf-8\");\n const info = JSON.parse(content) as ConnectionInfo;\n\n // Validate required fields\n if (!info.port || !info.token || !info.pid) {\n return null;\n }\n\n // Check if Sanqian process is running\n if (!this.isProcessRunning(info.pid)) {\n return null;\n }\n\n this.connectionInfo = info;\n return info;\n } catch {\n return null;\n }\n }\n\n /**\n * Start watching for connection file changes\n */\n startWatching(onChange: (info: ConnectionInfo | null) => void): void {\n this.onChange = onChange;\n\n const dir = join(homedir(), \".sanqian\", \"runtime\");\n\n // Ensure directory exists\n if (!existsSync(dir)) {\n // Directory doesn't exist, poll for it\n this.pollInterval = setInterval(() => {\n if (existsSync(dir)) {\n if (this.pollInterval) {\n clearInterval(this.pollInterval);\n this.pollInterval = null;\n }\n this.setupWatcher(dir);\n }\n }, 2000);\n return;\n }\n\n this.setupWatcher(dir);\n }\n\n private setupWatcher(dir: string): void {\n try {\n this.watcher = watch(dir, (event, filename) => {\n if (filename === \"connection.json\") {\n // Debounce rapid changes\n setTimeout(() => {\n const newInfo = this.read();\n const oldInfoStr = JSON.stringify(this.connectionInfo);\n const newInfoStr = JSON.stringify(newInfo);\n\n // Only trigger if actually changed\n if (oldInfoStr !== newInfoStr) {\n this.connectionInfo = newInfo;\n this.onChange?.(newInfo);\n }\n }, 100);\n }\n });\n } catch (e) {\n console.error(\"Failed to watch connection file:\", e);\n }\n }\n\n /**\n * Stop watching\n */\n stopWatching(): void {\n if (this.pollInterval) {\n clearInterval(this.pollInterval);\n this.pollInterval = null;\n }\n this.watcher?.close();\n this.watcher = null;\n this.onChange = null;\n }\n\n /**\n * Check if a process is running by PID\n */\n private isProcessRunning(pid: number): boolean {\n try {\n // Sending signal 0 doesn't kill the process, just checks if it exists\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n }\n\n /**\n * Get cached connection info (may be stale)\n */\n getCached(): ConnectionInfo | null {\n return this.connectionInfo;\n }\n\n /**\n * Build WebSocket URL from connection info\n */\n buildWebSocketUrl(info: ConnectionInfo): string {\n return `ws://127.0.0.1:${info.port}${info.ws_path}?token=${info.token}`;\n }\n\n /**\n * Build HTTP base URL from connection info\n */\n buildHttpUrl(info: ConnectionInfo): string {\n return `http://127.0.0.1:${info.port}`;\n }\n\n /**\n * Find Sanqian executable path\n * Searches in standard installation locations for each platform\n */\n findSanqianPath(customPath?: string): string | null {\n // If custom path provided, use it\n if (customPath) {\n if (existsSync(customPath)) {\n return customPath;\n }\n console.warn(`[SDK] Custom Sanqian path not found: ${customPath}`);\n return null;\n }\n\n const os = platform();\n const searchPaths: string[] = [];\n\n if (os === \"darwin\") {\n // macOS: Standard app locations + common dev locations\n searchPaths.push(\n // Production: installed app\n \"/Applications/Sanqian.app/Contents/MacOS/Sanqian\",\n join(homedir(), \"Applications/Sanqian.app/Contents/MacOS/Sanqian\"),\n // Development: electron-builder output\n join(homedir(), \"dev/sanqian/dist/mac-arm64/Sanqian.app/Contents/MacOS/Sanqian\"),\n join(homedir(), \"dev/sanqian/dist/mac/Sanqian.app/Contents/MacOS/Sanqian\"),\n );\n } else if (os === \"win32\") {\n // Windows: Standard installation paths + dev locations\n const programFiles = process.env.PROGRAMFILES || \"C:\\\\Program Files\";\n const programFilesX86 =\n process.env[\"PROGRAMFILES(X86)\"] || \"C:\\\\Program Files (x86)\";\n const localAppData =\n process.env.LOCALAPPDATA || join(homedir(), \"AppData\", \"Local\");\n\n searchPaths.push(\n // Production: installed app\n join(programFiles, \"Sanqian\", \"Sanqian.exe\"),\n join(programFilesX86, \"Sanqian\", \"Sanqian.exe\"),\n join(localAppData, \"Programs\", \"Sanqian\", \"Sanqian.exe\"),\n // Development: electron-builder output\n join(homedir(), \"dev\", \"sanqian\", \"dist\", \"win-unpacked\", \"Sanqian.exe\"),\n );\n } else {\n // Linux: Common locations\n searchPaths.push(\n \"/usr/bin/sanqian\",\n \"/usr/local/bin/sanqian\",\n join(homedir(), \".local/bin/sanqian\"),\n \"/opt/Sanqian/sanqian\",\n // Development\n join(homedir(), \"dev/sanqian/dist/linux-unpacked/sanqian\"),\n );\n }\n\n for (const path of searchPaths) {\n if (existsSync(path)) {\n return path;\n }\n }\n\n return null;\n }\n\n /**\n * Launch Sanqian in hidden/tray mode\n * Returns true if launch was initiated successfully\n */\n launchSanqian(customPath?: string): boolean {\n const sanqianPath = this.findSanqianPath(customPath);\n\n if (!sanqianPath) {\n console.error(\"[SDK] Sanqian executable not found\");\n return false;\n }\n\n console.log(`[SDK] Launching Sanqian from: ${sanqianPath}`);\n\n try {\n const os = platform();\n\n if (os === \"darwin\") {\n // macOS: Use 'open' command with --args for hidden start\n // This properly launches the app bundle\n const appPath = sanqianPath.replace(\n \"/Contents/MacOS/Sanqian\",\n \"\",\n );\n spawn(\"open\", [\"-a\", appPath, \"--args\", \"--hidden\"], {\n detached: true,\n stdio: \"ignore\",\n }).unref();\n } else {\n // Windows/Linux: Direct execution with --hidden flag\n spawn(sanqianPath, [\"--hidden\"], {\n detached: true,\n stdio: \"ignore\",\n // On Windows, hide the console window\n ...(os === \"win32\" && {\n windowsHide: true,\n shell: false,\n }),\n }).unref();\n }\n\n return true;\n } catch (e) {\n console.error(\"[SDK] Failed to launch Sanqian:\", e);\n return false;\n }\n }\n\n /**\n * Check if Sanqian is running\n */\n isSanqianRunning(): boolean {\n return this.read() !== null;\n }\n}\n"],"mappings":"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","names":["WebSocket"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/client.ts","../src/discovery.ts"],"sourcesContent":["/**\n * Sanqian SDK\n *\n * TypeScript SDK for integrating third-party apps with Sanqian AI Assistant.\n *\n * @example\n * ```typescript\n * import { SanqianSDK } from '@yushaw/sanqian-sdk';\n *\n * const sdk = new SanqianSDK({\n * appName: 'my-app',\n * appVersion: '1.0.0',\n * tools: [\n * {\n * name: 'my_tool',\n * description: 'Does something useful',\n * parameters: { type: 'object', properties: {} },\n * handler: async (args) => {\n * return { success: true };\n * }\n * }\n * ]\n * });\n *\n * await sdk.connect();\n * ```\n */\n\nexport { SanqianSDK, Conversation } from \"./client\";\nexport { DiscoveryManager } from \"./discovery\";\n\n// Types\nexport type {\n SDKConfig,\n ToolDefinition,\n JSONSchema,\n JSONSchemaProperty,\n ConnectionInfo,\n ConnectionState,\n ChatRequest,\n ChatMessage,\n ChatResponse,\n ChatStreamEvent,\n ToolCall,\n SDKEvents,\n SDKEventName,\n // Private Agent API\n AgentConfig,\n AgentInfo,\n AgentUpdateConfig,\n ConversationInfo,\n ConversationMessage,\n ConversationDetail,\n // Remote Tools\n RemoteToolDefinition,\n} from \"./types\";\n","/**\n * Sanqian SDK Client\n *\n * Main class for connecting to Sanqian and registering tools.\n */\n\nimport WebSocket from \"ws\";\nimport { DiscoveryManager } from \"./discovery\";\nimport type {\n SDKConfig,\n ConnectionInfo,\n ConnectionState,\n ToolDefinition,\n RegisterMessage,\n RegisterAckMessage,\n ToolCallMessage,\n ToolResultMessage,\n HeartbeatMessage,\n SDKEvents,\n SDKEventName,\n JSONSchema,\n AgentConfig,\n AgentInfo,\n AgentUpdateConfig,\n ConversationInfo,\n ConversationDetail,\n CreateAgentMessage,\n CreateAgentAckMessage,\n UpdateAgentMessage,\n UpdateAgentAckMessage,\n ListAgentsMessage,\n ListAgentsAckMessage,\n DeleteAgentMessage,\n DeleteAgentAckMessage,\n ListConversationsMessage,\n ListConversationsAckMessage,\n GetConversationMessage,\n GetConversationAckMessage,\n DeleteConversationMessage,\n DeleteConversationAckMessage,\n ChatMessage,\n ChatResponse,\n ChatRequestMessage,\n ChatResponseMessage,\n ChatStreamMessage,\n ChatStreamEvent,\n RemoteToolDefinition,\n} from \"./types\";\n\ntype EventListener<T extends SDKEventName> = SDKEvents[T];\n\nexport class SanqianSDK {\n private config: SDKConfig;\n private discovery: DiscoveryManager;\n private ws: WebSocket | null = null;\n private connectionInfo: ConnectionInfo | null = null;\n\n private state: ConnectionState = {\n connected: false,\n registering: false,\n registered: false,\n reconnectAttempts: 0,\n };\n\n // Tool handlers by name\n private toolHandlers: Map<string, (args: unknown) => Promise<unknown>> =\n new Map();\n\n // Pending request futures\n private pendingRequests: Map<\n string,\n {\n resolve: (value: unknown) => void;\n reject: (error: Error) => void;\n }\n > = new Map();\n\n // Timers\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n private reconnectTimer: ReturnType<typeof setTimeout> | null = null;\n\n // Event listeners\n private eventListeners: Map<SDKEventName, Set<EventListener<SDKEventName>>> =\n new Map();\n\n constructor(config: SDKConfig) {\n this.config = {\n reconnectInterval: 5000,\n heartbeatInterval: 30000,\n toolExecutionTimeout: 30000,\n ...config,\n };\n this.discovery = new DiscoveryManager();\n\n // Register tool handlers\n for (const tool of config.tools) {\n this.toolHandlers.set(tool.name, tool.handler);\n }\n }\n\n // ============================================\n // Lifecycle\n // ============================================\n\n /**\n * Connect to Sanqian\n *\n * Reads connection info, establishes WebSocket, and registers app.\n * Returns when registration is complete.\n *\n * If autoLaunchSanqian is enabled and Sanqian is not running,\n * SDK will attempt to start it in hidden/tray mode.\n */\n async connect(): Promise<void> {\n // Read connection info\n const info = this.discovery.read();\n\n if (!info) {\n // Sanqian not running\n // Try to auto-launch if enabled\n if (this.config.autoLaunchSanqian) {\n console.log(\"[SDK] Sanqian not running, attempting to launch...\");\n const launched = this.discovery.launchSanqian(this.config.sanqianPath);\n if (launched) {\n console.log(\"[SDK] Sanqian launch initiated, waiting for startup...\");\n } else {\n console.warn(\"[SDK] Failed to launch Sanqian, will wait for manual start\");\n }\n }\n\n // Start watching and wait for connection\n return new Promise((resolve, reject) => {\n console.log(\"[SDK] Waiting for Sanqian...\");\n\n const timeout = setTimeout(() => {\n this.discovery.stopWatching();\n reject(new Error(\"Sanqian connection timeout\"));\n }, 60000); // 60 second timeout\n\n this.discovery.startWatching(async (newInfo) => {\n if (newInfo) {\n clearTimeout(timeout);\n this.discovery.stopWatching();\n try {\n await this.connectWithInfo(newInfo);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n });\n });\n }\n\n await this.connectWithInfo(info);\n }\n\n /**\n * Connect with known connection info\n */\n private async connectWithInfo(info: ConnectionInfo): Promise<void> {\n this.connectionInfo = info;\n const url = this.discovery.buildWebSocketUrl(info);\n\n return new Promise((resolve, reject) => {\n console.log(`[SDK] Connecting to ${url}`);\n\n this.ws = new WebSocket(url);\n\n const connectTimeout = setTimeout(() => {\n reject(new Error(\"WebSocket connection timeout\"));\n this.ws?.close();\n }, 10000);\n\n this.ws.on(\"open\", async () => {\n clearTimeout(connectTimeout);\n console.log(\"[SDK] WebSocket connected\");\n this.state.connected = true;\n this.state.reconnectAttempts = 0;\n this.emit(\"connected\");\n\n try {\n await this.register();\n resolve();\n } catch (e) {\n reject(e);\n }\n });\n\n this.ws.on(\"close\", (code, reason) => {\n console.log(`[SDK] WebSocket closed: ${code} ${reason.toString()}`);\n this.handleDisconnect(reason.toString());\n });\n\n this.ws.on(\"error\", (error) => {\n console.error(\"[SDK] WebSocket error:\", error);\n this.state.lastError = error;\n this.emit(\"error\", error);\n });\n\n this.ws.on(\"message\", (data) => {\n try {\n const message = JSON.parse(data.toString());\n this.handleMessage(message);\n } catch (e) {\n console.error(\"[SDK] Failed to parse message:\", e);\n }\n });\n });\n }\n\n /**\n * Disconnect from Sanqian\n */\n async disconnect(): Promise<void> {\n this.stopHeartbeat();\n this.stopReconnect();\n this.discovery.stopWatching();\n\n if (this.ws) {\n this.ws.close(1000, \"Client disconnect\");\n this.ws = null;\n }\n\n this.state = {\n connected: false,\n registering: false,\n registered: false,\n reconnectAttempts: 0,\n };\n }\n\n // ============================================\n // Registration\n // ============================================\n\n private async register(): Promise<void> {\n this.state.registering = true;\n\n const msgId = this.generateId();\n\n const message: RegisterMessage = {\n id: msgId,\n type: \"register\",\n app: {\n name: this.config.appName,\n version: this.config.appVersion,\n display_name: this.config.displayName,\n launch_command: this.config.launchCommand,\n },\n tools: this.config.tools.map((t) => ({\n name: `${this.config.appName}:${t.name}`,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n try {\n const response = await this.sendAndWait<RegisterAckMessage>(\n message,\n msgId,\n 10000\n );\n\n if (!response.success) {\n throw new Error(response.error || \"Registration failed\");\n }\n\n this.state.registering = false;\n this.state.registered = true;\n this.startHeartbeat();\n this.emit(\"registered\");\n\n console.log(`[SDK] Registered as '${this.config.appName}'`);\n } catch (e) {\n this.state.registering = false;\n throw e;\n }\n }\n\n // ============================================\n // Message Handling\n // ============================================\n\n private handleMessage(message: { id?: string; type: string; [key: string]: unknown }): void {\n const { id, type } = message;\n\n // Check if this is a response to a pending request\n if (id && this.pendingRequests.has(id)) {\n const pending = this.pendingRequests.get(id)!;\n this.pendingRequests.delete(id);\n pending.resolve(message);\n return;\n }\n\n // Handle server-initiated messages\n switch (type) {\n case \"tool_call\":\n this.handleToolCall(message as unknown as ToolCallMessage);\n break;\n\n case \"heartbeat_ack\":\n // Heartbeat acknowledged, no action needed\n break;\n\n case \"chat_stream\":\n // Handle streaming chat response\n this.handleChatStream(message as unknown as ChatStreamMessage);\n break;\n\n default:\n console.warn(`[SDK] Unknown message type: ${type}`);\n }\n }\n\n private handleChatStream(message: ChatStreamMessage): void {\n const { id, event, content, tool_call, tool_result, conversation_id, title, usage, error } = message;\n\n if (!id) return;\n\n const handler = this.streamHandlers.get(id);\n if (!handler) {\n console.warn(`[SDK] No stream handler for message ${id}`);\n return;\n }\n\n switch (event) {\n case \"text\":\n handler.onEvent({ type: \"text\", content });\n break;\n\n case \"tool_call\":\n handler.onEvent({ type: \"tool_call\", tool_call });\n break;\n\n case \"tool_result\":\n if (tool_result) {\n handler.onEvent({\n type: \"tool_call\",\n tool_call: {\n id: tool_result.call_id,\n type: \"function\",\n function: {\n name: \"tool_result\",\n arguments: JSON.stringify(tool_result),\n },\n },\n });\n }\n break;\n\n case \"done\":\n handler.onDone({\n message: message.message || { role: \"assistant\", content: \"\" },\n conversationId: conversation_id || \"\",\n title,\n usage,\n });\n break;\n\n case \"error\":\n handler.onError(new Error(error || \"Unknown stream error\"));\n break;\n }\n }\n\n private async handleToolCall(message: ToolCallMessage): Promise<void> {\n console.log(`[SDK] handleToolCall received:`, JSON.stringify(message));\n const { id, call_id, name, arguments: args } = message;\n const msgId = id || call_id;\n\n // Extract actual tool name (remove app prefix)\n const toolName = name.includes(\":\") ? name.split(\":\")[1] : name;\n console.log(`[SDK] Looking for handler: '${toolName}', available handlers:`, Array.from(this.toolHandlers.keys()));\n const handler = this.toolHandlers.get(toolName);\n\n // Emit event for debugging/logging\n this.emit(\"tool_call\", { name: toolName, arguments: args });\n\n if (!handler) {\n await this.sendToolResult(msgId, call_id, false, undefined, `Tool '${toolName}' not found`);\n return;\n }\n\n try {\n console.log(`[SDK] Executing tool '${toolName}' with args:`, args);\n // Execute with timeout\n const result = await Promise.race([\n handler(args),\n this.createTimeout(this.config.toolExecutionTimeout!),\n ]);\n\n console.log(`[SDK] Tool '${toolName}' completed successfully`);\n await this.sendToolResult(msgId, call_id, true, result);\n } catch (e) {\n const error = e instanceof Error ? e.message : String(e);\n console.log(`[SDK] Tool '${toolName}' failed:`, error);\n await this.sendToolResult(msgId, call_id, false, undefined, error);\n }\n }\n\n private async sendToolResult(\n id: string,\n callId: string,\n success: boolean,\n result?: unknown,\n error?: string\n ): Promise<void> {\n const message: ToolResultMessage = {\n id,\n type: \"tool_result\",\n call_id: callId,\n success,\n result,\n error,\n };\n\n console.log(`[SDK] Sending tool_result for ${callId}:`, success ? 'success' : `error: ${error}`);\n this.send(message);\n }\n\n // ============================================\n // Connection Management\n // ============================================\n\n private handleDisconnect(reason: string): void {\n this.stopHeartbeat();\n this.state.connected = false;\n this.state.registered = false;\n this.emit(\"disconnected\", reason);\n\n // Reject pending requests\n for (const [, pending] of this.pendingRequests) {\n pending.reject(new Error(\"Disconnected\"));\n }\n this.pendingRequests.clear();\n\n // Schedule reconnect\n this.scheduleReconnect();\n }\n\n private scheduleReconnect(): void {\n if (this.reconnectTimer) return;\n\n // Exponential backoff: 500ms, 1s, 2s, 4s, max 5s\n // With jitter to avoid thundering herd\n const baseDelay = Math.min(\n 500 * Math.pow(2, this.state.reconnectAttempts),\n 5000 // Cap at 5 seconds\n );\n const jitter = Math.random() * 500; // 0-500ms random jitter\n const delay = baseDelay + jitter;\n\n console.log(`[SDK] Scheduling reconnect in ${Math.round(delay)}ms (attempt ${this.state.reconnectAttempts + 1})`);\n\n this.reconnectTimer = setTimeout(async () => {\n this.reconnectTimer = null;\n this.state.reconnectAttempts++;\n\n // Re-read connection info\n const info = this.discovery.read();\n if (info) {\n try {\n await this.connectWithInfo(info);\n } catch (e) {\n console.error(\"[SDK] Reconnect failed:\", e);\n this.scheduleReconnect();\n }\n } else {\n // Sanqian not running, keep trying\n this.scheduleReconnect();\n }\n }, delay);\n }\n\n private stopReconnect(): void {\n if (this.reconnectTimer) {\n clearTimeout(this.reconnectTimer);\n this.reconnectTimer = null;\n }\n }\n\n // ============================================\n // Heartbeat\n // ============================================\n\n private startHeartbeat(): void {\n this.stopHeartbeat();\n\n this.heartbeatTimer = setInterval(() => {\n const message: HeartbeatMessage = {\n type: \"heartbeat\",\n timestamp: new Date().toISOString(),\n };\n this.send(message);\n }, this.config.heartbeatInterval);\n }\n\n private stopHeartbeat(): void {\n if (this.heartbeatTimer) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n }\n\n // ============================================\n // Communication\n // ============================================\n\n private send(message: object): void {\n if (this.ws && this.ws.readyState === WebSocket.OPEN) {\n const data = JSON.stringify(message);\n console.log(`[SDK] WebSocket send:`, data.substring(0, 200));\n this.ws.send(data);\n } else {\n console.error(`[SDK] WebSocket not open, cannot send:`, message);\n }\n }\n\n private sendAndWait<T>(\n message: object,\n id: string,\n timeout: number\n ): Promise<T> {\n return new Promise((resolve, reject) => {\n const timer = setTimeout(() => {\n this.pendingRequests.delete(id);\n reject(new Error(\"Request timeout\"));\n }, timeout);\n\n this.pendingRequests.set(id, {\n resolve: (value) => {\n clearTimeout(timer);\n resolve(value as T);\n },\n reject: (error) => {\n clearTimeout(timer);\n reject(error);\n },\n });\n\n this.send(message);\n });\n }\n\n // ============================================\n // Public API\n // ============================================\n\n /**\n * Get current connection state\n */\n getState(): ConnectionState {\n return { ...this.state };\n }\n\n /**\n * Check if connected and registered\n */\n isConnected(): boolean {\n return this.state.connected && this.state.registered;\n }\n\n /**\n * Wait for connection to be established.\n * Used internally by chat() and chatStream() for auto-reconnect.\n * Relies on background scheduleReconnect() to do the actual reconnection.\n */\n private async waitForConnection(timeout: number = 30000): Promise<void> {\n if (this.isConnected()) {\n return;\n }\n\n console.log(\"[SDK] Not connected, waiting for reconnection...\");\n\n // If no reconnect is scheduled and Sanqian is not running, try to launch it\n if (!this.reconnectTimer) {\n const info = this.discovery.read();\n if (!info && this.config.autoLaunchSanqian) {\n console.log(\"[SDK] Sanqian not running, attempting to launch...\");\n const launched = this.discovery.launchSanqian(this.config.sanqianPath);\n if (launched) {\n // Trigger reconnect attempt\n this.scheduleReconnect();\n }\n } else if (!info) {\n // No connection info and can't auto-launch\n throw new Error(\"Sanqian is not running. Please start it manually.\");\n } else {\n // Connection info exists, trigger reconnect\n this.scheduleReconnect();\n }\n }\n\n // Wait for 'registered' event (emitted when connection + registration completes)\n return new Promise((resolve, reject) => {\n let resolved = false;\n\n const timer = setTimeout(() => {\n if (!resolved) {\n resolved = true;\n reject(new Error(\"Connection timeout. Please ensure Sanqian is running.\"));\n }\n }, timeout);\n\n this.once(\"registered\", () => {\n if (!resolved) {\n resolved = true;\n clearTimeout(timer);\n resolve();\n }\n });\n });\n }\n\n /**\n * Update tool list dynamically\n */\n async updateTools(tools: ToolDefinition[]): Promise<void> {\n // Update local handlers\n this.toolHandlers.clear();\n for (const tool of tools) {\n this.toolHandlers.set(tool.name, tool.handler);\n }\n\n // Send update to Sanqian\n if (this.isConnected()) {\n const msgId = this.generateId();\n const message = {\n id: msgId,\n type: \"tools_update\",\n tools: tools.map((t) => ({\n name: `${this.config.appName}:${t.name}`,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n await this.sendAndWait(message, msgId, 5000);\n }\n }\n\n // ============================================\n // Private Agent API\n // ============================================\n\n /**\n * Create or update a private agent\n *\n * Private agents are only visible to this SDK app, not in Sanqian UI.\n * If agent with same ID exists, it will be updated.\n *\n * @param config - Agent configuration\n * @returns Full agent info (or agent_id string for backward compatibility)\n */\n async createAgent(config: AgentConfig): Promise<AgentInfo> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: CreateAgentMessage = {\n id: msgId,\n type: \"create_agent\",\n agent: config,\n };\n\n const response = await this.sendAndWait<CreateAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to create agent\");\n }\n\n // Return full agent info if available, fallback to minimal info\n if (response.agent) {\n return response.agent;\n }\n\n // Backward compatibility: construct minimal AgentInfo from agent_id\n return {\n agent_id: response.agent_id!,\n name: config.name,\n description: config.description,\n tools: config.tools || [],\n };\n }\n\n /**\n * List all private agents owned by this app\n */\n async listAgents(): Promise<AgentInfo[]> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: ListAgentsMessage = {\n id: msgId,\n type: \"list_agents\",\n };\n\n const response = await this.sendAndWait<ListAgentsAckMessage>(message, msgId, 10000);\n\n if (response.error) {\n throw new Error(response.error);\n }\n\n return response.agents;\n }\n\n /**\n * Delete a private agent\n */\n async deleteAgent(agentId: string): Promise<void> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: DeleteAgentMessage = {\n id: msgId,\n type: \"delete_agent\",\n agent_id: agentId,\n };\n\n const response = await this.sendAndWait<DeleteAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to delete agent\");\n }\n }\n\n /**\n * Update a private agent\n *\n * Only updates the fields that are provided.\n * @param agentId - Agent ID (short name or full)\n * @param updates - Fields to update\n * @returns Updated agent info\n */\n async updateAgent(\n agentId: string,\n updates: Omit<AgentUpdateConfig, \"agent_id\">\n ): Promise<AgentInfo> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: UpdateAgentMessage = {\n id: msgId,\n type: \"update_agent\",\n agent: {\n agent_id: agentId,\n ...updates,\n },\n };\n\n const response = await this.sendAndWait<UpdateAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success || !response.agent) {\n throw new Error(response.error || \"Failed to update agent\");\n }\n\n return response.agent;\n }\n\n // ============================================\n // Conversation API\n // ============================================\n\n /**\n * List conversations for this app\n */\n async listConversations(options?: {\n agentId?: string;\n limit?: number;\n offset?: number;\n }): Promise<{ conversations: ConversationInfo[]; total: number }> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: ListConversationsMessage = {\n id: msgId,\n type: \"list_conversations\",\n agent_id: options?.agentId,\n limit: options?.limit,\n offset: options?.offset,\n };\n\n const response = await this.sendAndWait<ListConversationsAckMessage>(message, msgId, 10000);\n\n if (response.error) {\n throw new Error(response.error);\n }\n\n return {\n conversations: response.conversations,\n total: response.total,\n };\n }\n\n /**\n * Get conversation details with messages\n */\n async getConversation(\n conversationId: string,\n options?: {\n includeMessages?: boolean;\n messageLimit?: number;\n messageOffset?: number;\n }\n ): Promise<ConversationDetail> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: GetConversationMessage = {\n id: msgId,\n type: \"get_conversation\",\n conversation_id: conversationId,\n include_messages: options?.includeMessages ?? true,\n message_limit: options?.messageLimit,\n message_offset: options?.messageOffset,\n };\n\n const response = await this.sendAndWait<GetConversationAckMessage>(message, msgId, 10000);\n\n if (!response.success || !response.conversation) {\n throw new Error(response.error || \"Failed to get conversation\");\n }\n\n return response.conversation;\n }\n\n /**\n * Delete a conversation\n */\n async deleteConversation(conversationId: string): Promise<void> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: DeleteConversationMessage = {\n id: msgId,\n type: \"delete_conversation\",\n conversation_id: conversationId,\n };\n\n const response = await this.sendAndWait<DeleteConversationAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to delete conversation\");\n }\n }\n\n // ============================================\n // Chat API\n // ============================================\n\n // Pending stream handlers for streaming chat\n private streamHandlers: Map<string, {\n onEvent: (event: ChatStreamEvent) => void;\n onDone: (response: ChatResponse) => void;\n onError: (error: Error) => void;\n }> = new Map();\n\n /**\n * Send a chat message to an agent (non-streaming)\n *\n * Supports two modes:\n * - Stateless (no conversationId): Caller maintains message history\n * - Stateful (with conversationId): Server stores messages in conversations table\n *\n * @param agentId - Agent ID (short name or full, SDK auto-prefixes)\n * @param messages - Messages to send\n * @param options - Optional settings including conversationId and remoteTools\n * @returns Chat response with assistant message and conversation ID\n */\n async chat(\n agentId: string,\n messages: ChatMessage[],\n options?: {\n conversationId?: string;\n remoteTools?: RemoteToolDefinition[];\n }\n ): Promise<ChatResponse> {\n // Auto-reconnect if disconnected\n if (!this.isConnected()) {\n console.log(\"[SDK] Not connected, attempting to reconnect...\");\n await this.waitForConnection();\n }\n\n const msgId = this.generateId();\n const message: ChatRequestMessage = {\n id: msgId,\n type: \"chat\",\n agent_id: agentId,\n messages,\n conversation_id: options?.conversationId,\n stream: false,\n remote_tools: options?.remoteTools?.map((t) => ({\n name: t.name,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n // Longer timeout for chat (agent may take time with complex multi-tool tasks)\n const response = await this.sendAndWait<ChatResponseMessage>(message, msgId, 600000); // 10 minutes\n\n if (!response.success) {\n throw new Error(response.error || \"Chat request failed\");\n }\n\n return {\n message: response.message!,\n conversationId: response.conversation_id!,\n title: response.title,\n usage: response.usage,\n };\n }\n\n /**\n * Send a chat message to an agent with streaming response\n *\n * Supports two modes:\n * - Stateless (no conversationId): Caller maintains message history\n * - Stateful (with conversationId): Server stores messages in conversations table\n *\n * @param agentId - Agent ID (short name or full, SDK auto-prefixes)\n * @param messages - Messages to send\n * @param options - Optional settings including conversationId and remoteTools\n * @returns AsyncIterable of stream events\n */\n async *chatStream(\n agentId: string,\n messages: ChatMessage[],\n options?: {\n conversationId?: string;\n remoteTools?: RemoteToolDefinition[];\n }\n ): AsyncGenerator<ChatStreamEvent> {\n // Auto-reconnect if disconnected\n if (!this.isConnected()) {\n console.log(\"[SDK] Not connected, attempting to reconnect...\");\n await this.waitForConnection();\n }\n\n const msgId = this.generateId();\n const message: ChatRequestMessage = {\n id: msgId,\n type: \"chat\",\n agent_id: agentId,\n messages,\n conversation_id: options?.conversationId,\n stream: true,\n remote_tools: options?.remoteTools?.map((t) => ({\n name: t.name,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n // Create a queue to yield events\n const eventQueue: ChatStreamEvent[] = [];\n let done = false;\n let error: Error | null = null;\n let resolveNext: (() => void) | null = null;\n\n // Register stream handler\n this.streamHandlers.set(msgId, {\n onEvent: (event) => {\n eventQueue.push(event);\n resolveNext?.();\n },\n onDone: (response) => {\n eventQueue.push({\n type: \"done\",\n conversationId: response.conversationId,\n title: response.title,\n });\n done = true;\n resolveNext?.();\n },\n onError: (e) => {\n error = e;\n resolveNext?.();\n },\n });\n\n try {\n // Send request\n this.send(message);\n\n // Yield events as they arrive\n while (!done && !error) {\n if (eventQueue.length > 0) {\n yield eventQueue.shift()!;\n } else {\n // Wait for next event\n await new Promise<void>((resolve) => {\n resolveNext = resolve;\n });\n resolveNext = null;\n }\n }\n\n // Yield remaining events\n while (eventQueue.length > 0) {\n yield eventQueue.shift()!;\n }\n\n if (error) {\n throw error;\n }\n } finally {\n this.streamHandlers.delete(msgId);\n }\n }\n\n /**\n * Start a new conversation with an agent\n *\n * Returns a Conversation object for convenient multi-turn chat.\n *\n * @example\n * ```typescript\n * const conv = sdk.startConversation('assistant')\n * const r1 = await conv.send('Hello')\n * const r2 = await conv.send('Follow up question')\n * console.log(conv.id) // conversation ID\n * ```\n */\n startConversation(agentId: string): Conversation {\n return new Conversation(this, agentId);\n }\n\n // ============================================\n // Events\n // ============================================\n\n /**\n * Add event listener\n */\n on<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n if (!this.eventListeners.has(event)) {\n this.eventListeners.set(event, new Set());\n }\n this.eventListeners.get(event)!.add(listener as EventListener<SDKEventName>);\n return this;\n }\n\n /**\n * Remove event listener\n */\n off<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n this.eventListeners.get(event)?.delete(listener as EventListener<SDKEventName>);\n return this;\n }\n\n /**\n * Add one-time event listener (auto-removes after first call)\n */\n once<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n const onceWrapper = ((...args: Parameters<SDKEvents[T]>) => {\n this.off(event, onceWrapper as SDKEvents[T]);\n (listener as (...args: unknown[]) => void)(...args);\n }) as SDKEvents[T];\n return this.on(event, onceWrapper);\n }\n\n private emit<T extends SDKEventName>(\n event: T,\n ...args: Parameters<SDKEvents[T]>\n ): void {\n const listeners = this.eventListeners.get(event);\n if (listeners) {\n for (const listener of listeners) {\n try {\n (listener as (...args: unknown[]) => void)(...args);\n } catch (e) {\n console.error(`[SDK] Event listener error for '${event}':`, e);\n }\n }\n }\n }\n\n // ============================================\n // Utilities\n // ============================================\n\n private generateId(): string {\n return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;\n }\n\n private createTimeout(ms: number): Promise<never> {\n return new Promise((_, reject) => {\n setTimeout(() => reject(new Error(\"Timeout\")), ms);\n });\n }\n}\n\n/**\n * Conversation helper for multi-turn chat\n *\n * Automatically manages conversationId for stateful conversations.\n */\nexport class Conversation {\n private sdk: SanqianSDK;\n private agentId: string;\n private _conversationId: string | null = null;\n\n constructor(sdk: SanqianSDK, agentId: string, conversationId?: string) {\n this.sdk = sdk;\n this.agentId = agentId;\n this._conversationId = conversationId || null;\n }\n\n /**\n * Get the conversation ID (available after first message)\n */\n get id(): string | null {\n return this._conversationId;\n }\n\n /**\n * Send a message and get a response\n *\n * First call creates a new conversation, subsequent calls continue it.\n */\n async send(content: string, options?: { remoteTools?: RemoteToolDefinition[] }): Promise<ChatResponse> {\n const response = await this.sdk.chat(\n this.agentId,\n [{ role: \"user\", content }],\n {\n conversationId: this._conversationId || undefined,\n remoteTools: options?.remoteTools,\n }\n );\n\n // Store conversation ID for subsequent calls\n if (response.conversationId && !this._conversationId) {\n this._conversationId = response.conversationId;\n }\n\n return response;\n }\n\n /**\n * Send a message with streaming response\n */\n async *sendStream(\n content: string,\n options?: { remoteTools?: RemoteToolDefinition[] }\n ): AsyncGenerator<ChatStreamEvent> {\n const stream = this.sdk.chatStream(\n this.agentId,\n [{ role: \"user\", content }],\n {\n conversationId: this._conversationId || undefined,\n remoteTools: options?.remoteTools,\n }\n );\n\n for await (const event of stream) {\n // Capture conversation ID from done event\n if (event.type === \"done\" && event.conversationId && !this._conversationId) {\n this._conversationId = event.conversationId;\n }\n yield event;\n }\n }\n\n /**\n * Delete this conversation\n */\n async delete(): Promise<void> {\n if (!this._conversationId) {\n throw new Error(\"No conversation to delete\");\n }\n await this.sdk.deleteConversation(this._conversationId);\n this._conversationId = null;\n }\n\n /**\n * Get conversation details including message history\n */\n async getDetails(options?: { messageLimit?: number }): Promise<ConversationDetail> {\n if (!this._conversationId) {\n throw new Error(\"No conversation to get details for\");\n }\n return this.sdk.getConversation(this._conversationId, {\n includeMessages: true,\n messageLimit: options?.messageLimit,\n });\n }\n}\n","/**\n * Service Discovery Module\n *\n * Reads Sanqian's connection info from ~/.sanqian/runtime/connection.json\n * and monitors file changes for reconnection.\n * Also handles auto-launching Sanqian when needed.\n */\n\nimport { existsSync, readFileSync, watch, type FSWatcher } from \"fs\";\nimport { homedir, platform } from \"os\";\nimport { join } from \"path\";\nimport { spawn } from \"child_process\";\nimport type { ConnectionInfo } from \"./types\";\n\nexport class DiscoveryManager {\n private connectionInfo: ConnectionInfo | null = null;\n private watcher: FSWatcher | null = null;\n private onChange: ((info: ConnectionInfo | null) => void) | null = null;\n private pollInterval: ReturnType<typeof setInterval> | null = null;\n\n /**\n * Get the path to connection.json\n */\n getConnectionFilePath(): string {\n return join(homedir(), \".sanqian\", \"runtime\", \"connection.json\");\n }\n\n /**\n * Read and validate connection info\n *\n * Returns null if:\n * - File doesn't exist\n * - File is invalid JSON\n * - Process is not running\n */\n read(): ConnectionInfo | null {\n const filePath = this.getConnectionFilePath();\n\n if (!existsSync(filePath)) {\n return null;\n }\n\n try {\n const content = readFileSync(filePath, \"utf-8\");\n const info = JSON.parse(content) as ConnectionInfo;\n\n // Validate required fields\n if (!info.port || !info.token || !info.pid) {\n return null;\n }\n\n // Check if Sanqian process is running\n if (!this.isProcessRunning(info.pid)) {\n return null;\n }\n\n this.connectionInfo = info;\n return info;\n } catch {\n return null;\n }\n }\n\n /**\n * Start watching for connection file changes\n */\n startWatching(onChange: (info: ConnectionInfo | null) => void): void {\n this.onChange = onChange;\n\n const dir = join(homedir(), \".sanqian\", \"runtime\");\n\n // Ensure directory exists\n if (!existsSync(dir)) {\n // Directory doesn't exist, poll for it\n this.pollInterval = setInterval(() => {\n if (existsSync(dir)) {\n if (this.pollInterval) {\n clearInterval(this.pollInterval);\n this.pollInterval = null;\n }\n this.setupWatcher(dir);\n }\n }, 2000);\n return;\n }\n\n this.setupWatcher(dir);\n }\n\n private setupWatcher(dir: string): void {\n try {\n this.watcher = watch(dir, (event, filename) => {\n if (filename === \"connection.json\") {\n // Debounce rapid changes\n setTimeout(() => {\n const newInfo = this.read();\n const oldInfoStr = JSON.stringify(this.connectionInfo);\n const newInfoStr = JSON.stringify(newInfo);\n\n // Only trigger if actually changed\n if (oldInfoStr !== newInfoStr) {\n this.connectionInfo = newInfo;\n this.onChange?.(newInfo);\n }\n }, 100);\n }\n });\n } catch (e) {\n console.error(\"Failed to watch connection file:\", e);\n }\n }\n\n /**\n * Stop watching\n */\n stopWatching(): void {\n if (this.pollInterval) {\n clearInterval(this.pollInterval);\n this.pollInterval = null;\n }\n this.watcher?.close();\n this.watcher = null;\n this.onChange = null;\n }\n\n /**\n * Check if a process is running by PID\n */\n private isProcessRunning(pid: number): boolean {\n try {\n // Sending signal 0 doesn't kill the process, just checks if it exists\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n }\n\n /**\n * Get cached connection info (may be stale)\n */\n getCached(): ConnectionInfo | null {\n return this.connectionInfo;\n }\n\n /**\n * Build WebSocket URL from connection info\n */\n buildWebSocketUrl(info: ConnectionInfo): string {\n return `ws://127.0.0.1:${info.port}${info.ws_path}?token=${info.token}`;\n }\n\n /**\n * Build HTTP base URL from connection info\n */\n buildHttpUrl(info: ConnectionInfo): string {\n return `http://127.0.0.1:${info.port}`;\n }\n\n /**\n * Find Sanqian executable path\n * Searches in standard installation locations for each platform\n */\n findSanqianPath(customPath?: string): string | null {\n // If custom path provided, use it\n if (customPath) {\n if (existsSync(customPath)) {\n return customPath;\n }\n console.warn(`[SDK] Custom Sanqian path not found: ${customPath}`);\n return null;\n }\n\n const os = platform();\n const searchPaths: string[] = [];\n\n if (os === \"darwin\") {\n // macOS: Standard app locations + common dev locations\n searchPaths.push(\n // Production: installed app\n \"/Applications/Sanqian.app/Contents/MacOS/Sanqian\",\n join(homedir(), \"Applications/Sanqian.app/Contents/MacOS/Sanqian\"),\n // Development: electron-builder output\n join(homedir(), \"dev/sanqian/dist/mac-arm64/Sanqian.app/Contents/MacOS/Sanqian\"),\n join(homedir(), \"dev/sanqian/dist/mac/Sanqian.app/Contents/MacOS/Sanqian\"),\n );\n } else if (os === \"win32\") {\n // Windows: Standard installation paths + dev locations\n const programFiles = process.env.PROGRAMFILES || \"C:\\\\Program Files\";\n const programFilesX86 =\n process.env[\"PROGRAMFILES(X86)\"] || \"C:\\\\Program Files (x86)\";\n const localAppData =\n process.env.LOCALAPPDATA || join(homedir(), \"AppData\", \"Local\");\n\n searchPaths.push(\n // Production: installed app\n join(programFiles, \"Sanqian\", \"Sanqian.exe\"),\n join(programFilesX86, \"Sanqian\", \"Sanqian.exe\"),\n join(localAppData, \"Programs\", \"Sanqian\", \"Sanqian.exe\"),\n // Development: electron-builder output\n join(homedir(), \"dev\", \"sanqian\", \"dist\", \"win-unpacked\", \"Sanqian.exe\"),\n );\n } else {\n // Linux: Common locations\n searchPaths.push(\n \"/usr/bin/sanqian\",\n \"/usr/local/bin/sanqian\",\n join(homedir(), \".local/bin/sanqian\"),\n \"/opt/Sanqian/sanqian\",\n // Development\n join(homedir(), \"dev/sanqian/dist/linux-unpacked/sanqian\"),\n );\n }\n\n for (const path of searchPaths) {\n if (existsSync(path)) {\n return path;\n }\n }\n\n return null;\n }\n\n /**\n * Launch Sanqian in hidden/tray mode\n * Returns true if launch was initiated successfully\n */\n launchSanqian(customPath?: string): boolean {\n const sanqianPath = this.findSanqianPath(customPath);\n\n if (!sanqianPath) {\n console.error(\"[SDK] Sanqian executable not found\");\n return false;\n }\n\n console.log(`[SDK] Launching Sanqian from: ${sanqianPath}`);\n\n try {\n const os = platform();\n\n if (os === \"darwin\") {\n // macOS: Use 'open' command with --args for hidden start\n // This properly launches the app bundle\n const appPath = sanqianPath.replace(\n \"/Contents/MacOS/Sanqian\",\n \"\",\n );\n spawn(\"open\", [\"-a\", appPath, \"--args\", \"--hidden\"], {\n detached: true,\n stdio: \"ignore\",\n }).unref();\n } else {\n // Windows/Linux: Direct execution with --hidden flag\n spawn(sanqianPath, [\"--hidden\"], {\n detached: true,\n stdio: \"ignore\",\n // On Windows, hide the console window\n ...(os === \"win32\" && {\n windowsHide: true,\n shell: false,\n }),\n }).unref();\n }\n\n return true;\n } catch (e) {\n console.error(\"[SDK] Failed to launch Sanqian:\", e);\n return false;\n }\n }\n\n /**\n * Check if Sanqian is running\n */\n isSanqianRunning(): boolean {\n return this.read() !== null;\n }\n}\n"],"mappings":"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","names":["WebSocket"]}
package/dist/index.mjs CHANGED
@@ -431,7 +431,7 @@ var SanqianSDK = class {
431
431
  }
432
432
  }
433
433
  handleChatStream(message) {
434
- const { id, event, content, tool_call, tool_result, conversation_id, usage, error } = message;
434
+ const { id, event, content, tool_call, tool_result, conversation_id, title, usage, error } = message;
435
435
  if (!id) return;
436
436
  const handler = this.streamHandlers.get(id);
437
437
  if (!handler) {
@@ -464,6 +464,7 @@ var SanqianSDK = class {
464
464
  handler.onDone({
465
465
  message: message.message || { role: "assistant", content: "" },
466
466
  conversationId: conversation_id || "",
467
+ title,
467
468
  usage
468
469
  });
469
470
  break;
@@ -526,11 +527,14 @@ var SanqianSDK = class {
526
527
  }
527
528
  scheduleReconnect() {
528
529
  if (this.reconnectTimer) return;
529
- const delay = Math.min(
530
- 1e3 * Math.pow(2, this.state.reconnectAttempts),
531
- 3e4
530
+ const baseDelay = Math.min(
531
+ 500 * Math.pow(2, this.state.reconnectAttempts),
532
+ 5e3
533
+ // Cap at 5 seconds
532
534
  );
533
- console.log(`[SDK] Scheduling reconnect in ${delay}ms`);
535
+ const jitter = Math.random() * 500;
536
+ const delay = baseDelay + jitter;
537
+ console.log(`[SDK] Scheduling reconnect in ${Math.round(delay)}ms (attempt ${this.state.reconnectAttempts + 1})`);
534
538
  this.reconnectTimer = setTimeout(async () => {
535
539
  this.reconnectTimer = null;
536
540
  this.state.reconnectAttempts++;
@@ -619,43 +623,42 @@ var SanqianSDK = class {
619
623
  return this.state.connected && this.state.registered;
620
624
  }
621
625
  /**
622
- * Ensure connected to Sanqian, auto-launching if needed.
626
+ * Wait for connection to be established.
623
627
  * Used internally by chat() and chatStream() for auto-reconnect.
628
+ * Relies on background scheduleReconnect() to do the actual reconnection.
624
629
  */
625
- async ensureConnected() {
630
+ async waitForConnection(timeout = 3e4) {
626
631
  if (this.isConnected()) {
627
632
  return;
628
633
  }
629
- this.stopReconnect();
630
- const info = this.discovery.read();
631
- if (info) {
632
- console.log("[SDK] Sanqian is running, reconnecting...");
633
- await this.connectWithInfo(info);
634
- return;
635
- }
636
- if (this.config.autoLaunchSanqian) {
637
- console.log("[SDK] Sanqian not running, attempting to launch...");
638
- const launched = this.discovery.launchSanqian(this.config.sanqianPath);
639
- if (!launched) {
640
- throw new Error("Failed to launch Sanqian. Please start it manually.");
634
+ console.log("[SDK] Not connected, waiting for reconnection...");
635
+ if (!this.reconnectTimer) {
636
+ const info = this.discovery.read();
637
+ if (!info && this.config.autoLaunchSanqian) {
638
+ console.log("[SDK] Sanqian not running, attempting to launch...");
639
+ const launched = this.discovery.launchSanqian(this.config.sanqianPath);
640
+ if (launched) {
641
+ this.scheduleReconnect();
642
+ }
643
+ } else if (!info) {
644
+ throw new Error("Sanqian is not running. Please start it manually.");
645
+ } else {
646
+ this.scheduleReconnect();
641
647
  }
642
648
  }
643
649
  return new Promise((resolve, reject) => {
644
- console.log("[SDK] Waiting for Sanqian to start...");
645
- const timeout = setTimeout(() => {
646
- this.discovery.stopWatching();
647
- reject(new Error("Sanqian connection timeout. Please ensure Sanqian is running."));
648
- }, 3e4);
649
- this.discovery.startWatching(async (newInfo) => {
650
- if (newInfo) {
651
- clearTimeout(timeout);
652
- this.discovery.stopWatching();
653
- try {
654
- await this.connectWithInfo(newInfo);
655
- resolve();
656
- } catch (e) {
657
- reject(e);
658
- }
650
+ let resolved = false;
651
+ const timer = setTimeout(() => {
652
+ if (!resolved) {
653
+ resolved = true;
654
+ reject(new Error("Connection timeout. Please ensure Sanqian is running."));
655
+ }
656
+ }, timeout);
657
+ this.once("registered", () => {
658
+ if (!resolved) {
659
+ resolved = true;
660
+ clearTimeout(timer);
661
+ resolve();
659
662
  }
660
663
  });
661
664
  });
@@ -868,7 +871,7 @@ var SanqianSDK = class {
868
871
  async chat(agentId, messages, options) {
869
872
  if (!this.isConnected()) {
870
873
  console.log("[SDK] Not connected, attempting to reconnect...");
871
- await this.ensureConnected();
874
+ await this.waitForConnection();
872
875
  }
873
876
  const msgId = this.generateId();
874
877
  const message = {
@@ -891,6 +894,7 @@ var SanqianSDK = class {
891
894
  return {
892
895
  message: response.message,
893
896
  conversationId: response.conversation_id,
897
+ title: response.title,
894
898
  usage: response.usage
895
899
  };
896
900
  }
@@ -909,7 +913,7 @@ var SanqianSDK = class {
909
913
  async *chatStream(agentId, messages, options) {
910
914
  if (!this.isConnected()) {
911
915
  console.log("[SDK] Not connected, attempting to reconnect...");
912
- await this.ensureConnected();
916
+ await this.waitForConnection();
913
917
  }
914
918
  const msgId = this.generateId();
915
919
  const message = {
@@ -937,7 +941,8 @@ var SanqianSDK = class {
937
941
  onDone: (response) => {
938
942
  eventQueue.push({
939
943
  type: "done",
940
- conversationId: response.conversationId
944
+ conversationId: response.conversationId,
945
+ title: response.title
941
946
  });
942
947
  done = true;
943
948
  resolveNext?.();
@@ -1005,6 +1010,16 @@ var SanqianSDK = class {
1005
1010
  this.eventListeners.get(event)?.delete(listener);
1006
1011
  return this;
1007
1012
  }
1013
+ /**
1014
+ * Add one-time event listener (auto-removes after first call)
1015
+ */
1016
+ once(event, listener) {
1017
+ const onceWrapper = ((...args) => {
1018
+ this.off(event, onceWrapper);
1019
+ listener(...args);
1020
+ });
1021
+ return this.on(event, onceWrapper);
1022
+ }
1008
1023
  emit(event, ...args) {
1009
1024
  const listeners = this.eventListeners.get(event);
1010
1025
  if (listeners) {
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/client.ts","../src/discovery.ts"],"sourcesContent":["/**\n * Sanqian SDK Client\n *\n * Main class for connecting to Sanqian and registering tools.\n */\n\nimport WebSocket from \"ws\";\nimport { DiscoveryManager } from \"./discovery\";\nimport type {\n SDKConfig,\n ConnectionInfo,\n ConnectionState,\n ToolDefinition,\n RegisterMessage,\n RegisterAckMessage,\n ToolCallMessage,\n ToolResultMessage,\n HeartbeatMessage,\n SDKEvents,\n SDKEventName,\n JSONSchema,\n AgentConfig,\n AgentInfo,\n AgentUpdateConfig,\n ConversationInfo,\n ConversationDetail,\n CreateAgentMessage,\n CreateAgentAckMessage,\n UpdateAgentMessage,\n UpdateAgentAckMessage,\n ListAgentsMessage,\n ListAgentsAckMessage,\n DeleteAgentMessage,\n DeleteAgentAckMessage,\n ListConversationsMessage,\n ListConversationsAckMessage,\n GetConversationMessage,\n GetConversationAckMessage,\n DeleteConversationMessage,\n DeleteConversationAckMessage,\n ChatMessage,\n ChatResponse,\n ChatRequestMessage,\n ChatResponseMessage,\n ChatStreamMessage,\n ChatStreamEvent,\n RemoteToolDefinition,\n} from \"./types\";\n\ntype EventListener<T extends SDKEventName> = SDKEvents[T];\n\nexport class SanqianSDK {\n private config: SDKConfig;\n private discovery: DiscoveryManager;\n private ws: WebSocket | null = null;\n private connectionInfo: ConnectionInfo | null = null;\n\n private state: ConnectionState = {\n connected: false,\n registering: false,\n registered: false,\n reconnectAttempts: 0,\n };\n\n // Tool handlers by name\n private toolHandlers: Map<string, (args: unknown) => Promise<unknown>> =\n new Map();\n\n // Pending request futures\n private pendingRequests: Map<\n string,\n {\n resolve: (value: unknown) => void;\n reject: (error: Error) => void;\n }\n > = new Map();\n\n // Timers\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n private reconnectTimer: ReturnType<typeof setTimeout> | null = null;\n\n // Event listeners\n private eventListeners: Map<SDKEventName, Set<EventListener<SDKEventName>>> =\n new Map();\n\n constructor(config: SDKConfig) {\n this.config = {\n reconnectInterval: 5000,\n heartbeatInterval: 30000,\n toolExecutionTimeout: 30000,\n ...config,\n };\n this.discovery = new DiscoveryManager();\n\n // Register tool handlers\n for (const tool of config.tools) {\n this.toolHandlers.set(tool.name, tool.handler);\n }\n }\n\n // ============================================\n // Lifecycle\n // ============================================\n\n /**\n * Connect to Sanqian\n *\n * Reads connection info, establishes WebSocket, and registers app.\n * Returns when registration is complete.\n *\n * If autoLaunchSanqian is enabled and Sanqian is not running,\n * SDK will attempt to start it in hidden/tray mode.\n */\n async connect(): Promise<void> {\n // Read connection info\n const info = this.discovery.read();\n\n if (!info) {\n // Sanqian not running\n // Try to auto-launch if enabled\n if (this.config.autoLaunchSanqian) {\n console.log(\"[SDK] Sanqian not running, attempting to launch...\");\n const launched = this.discovery.launchSanqian(this.config.sanqianPath);\n if (launched) {\n console.log(\"[SDK] Sanqian launch initiated, waiting for startup...\");\n } else {\n console.warn(\"[SDK] Failed to launch Sanqian, will wait for manual start\");\n }\n }\n\n // Start watching and wait for connection\n return new Promise((resolve, reject) => {\n console.log(\"[SDK] Waiting for Sanqian...\");\n\n const timeout = setTimeout(() => {\n this.discovery.stopWatching();\n reject(new Error(\"Sanqian connection timeout\"));\n }, 60000); // 60 second timeout\n\n this.discovery.startWatching(async (newInfo) => {\n if (newInfo) {\n clearTimeout(timeout);\n this.discovery.stopWatching();\n try {\n await this.connectWithInfo(newInfo);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n });\n });\n }\n\n await this.connectWithInfo(info);\n }\n\n /**\n * Connect with known connection info\n */\n private async connectWithInfo(info: ConnectionInfo): Promise<void> {\n this.connectionInfo = info;\n const url = this.discovery.buildWebSocketUrl(info);\n\n return new Promise((resolve, reject) => {\n console.log(`[SDK] Connecting to ${url}`);\n\n this.ws = new WebSocket(url);\n\n const connectTimeout = setTimeout(() => {\n reject(new Error(\"WebSocket connection timeout\"));\n this.ws?.close();\n }, 10000);\n\n this.ws.on(\"open\", async () => {\n clearTimeout(connectTimeout);\n console.log(\"[SDK] WebSocket connected\");\n this.state.connected = true;\n this.state.reconnectAttempts = 0;\n this.emit(\"connected\");\n\n try {\n await this.register();\n resolve();\n } catch (e) {\n reject(e);\n }\n });\n\n this.ws.on(\"close\", (code, reason) => {\n console.log(`[SDK] WebSocket closed: ${code} ${reason.toString()}`);\n this.handleDisconnect(reason.toString());\n });\n\n this.ws.on(\"error\", (error) => {\n console.error(\"[SDK] WebSocket error:\", error);\n this.state.lastError = error;\n this.emit(\"error\", error);\n });\n\n this.ws.on(\"message\", (data) => {\n try {\n const message = JSON.parse(data.toString());\n this.handleMessage(message);\n } catch (e) {\n console.error(\"[SDK] Failed to parse message:\", e);\n }\n });\n });\n }\n\n /**\n * Disconnect from Sanqian\n */\n async disconnect(): Promise<void> {\n this.stopHeartbeat();\n this.stopReconnect();\n this.discovery.stopWatching();\n\n if (this.ws) {\n this.ws.close(1000, \"Client disconnect\");\n this.ws = null;\n }\n\n this.state = {\n connected: false,\n registering: false,\n registered: false,\n reconnectAttempts: 0,\n };\n }\n\n // ============================================\n // Registration\n // ============================================\n\n private async register(): Promise<void> {\n this.state.registering = true;\n\n const msgId = this.generateId();\n\n const message: RegisterMessage = {\n id: msgId,\n type: \"register\",\n app: {\n name: this.config.appName,\n version: this.config.appVersion,\n display_name: this.config.displayName,\n launch_command: this.config.launchCommand,\n },\n tools: this.config.tools.map((t) => ({\n name: `${this.config.appName}:${t.name}`,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n try {\n const response = await this.sendAndWait<RegisterAckMessage>(\n message,\n msgId,\n 10000\n );\n\n if (!response.success) {\n throw new Error(response.error || \"Registration failed\");\n }\n\n this.state.registering = false;\n this.state.registered = true;\n this.startHeartbeat();\n this.emit(\"registered\");\n\n console.log(`[SDK] Registered as '${this.config.appName}'`);\n } catch (e) {\n this.state.registering = false;\n throw e;\n }\n }\n\n // ============================================\n // Message Handling\n // ============================================\n\n private handleMessage(message: { id?: string; type: string; [key: string]: unknown }): void {\n const { id, type } = message;\n\n // Check if this is a response to a pending request\n if (id && this.pendingRequests.has(id)) {\n const pending = this.pendingRequests.get(id)!;\n this.pendingRequests.delete(id);\n pending.resolve(message);\n return;\n }\n\n // Handle server-initiated messages\n switch (type) {\n case \"tool_call\":\n this.handleToolCall(message as unknown as ToolCallMessage);\n break;\n\n case \"heartbeat_ack\":\n // Heartbeat acknowledged, no action needed\n break;\n\n case \"chat_stream\":\n // Handle streaming chat response\n this.handleChatStream(message as unknown as ChatStreamMessage);\n break;\n\n default:\n console.warn(`[SDK] Unknown message type: ${type}`);\n }\n }\n\n private handleChatStream(message: ChatStreamMessage): void {\n const { id, event, content, tool_call, tool_result, conversation_id, usage, error } = message;\n\n if (!id) return;\n\n const handler = this.streamHandlers.get(id);\n if (!handler) {\n console.warn(`[SDK] No stream handler for message ${id}`);\n return;\n }\n\n switch (event) {\n case \"text\":\n handler.onEvent({ type: \"text\", content });\n break;\n\n case \"tool_call\":\n handler.onEvent({ type: \"tool_call\", tool_call });\n break;\n\n case \"tool_result\":\n if (tool_result) {\n handler.onEvent({\n type: \"tool_call\",\n tool_call: {\n id: tool_result.call_id,\n type: \"function\",\n function: {\n name: \"tool_result\",\n arguments: JSON.stringify(tool_result),\n },\n },\n });\n }\n break;\n\n case \"done\":\n handler.onDone({\n message: message.message || { role: \"assistant\", content: \"\" },\n conversationId: conversation_id || \"\",\n usage,\n });\n break;\n\n case \"error\":\n handler.onError(new Error(error || \"Unknown stream error\"));\n break;\n }\n }\n\n private async handleToolCall(message: ToolCallMessage): Promise<void> {\n console.log(`[SDK] handleToolCall received:`, JSON.stringify(message));\n const { id, call_id, name, arguments: args } = message;\n const msgId = id || call_id;\n\n // Extract actual tool name (remove app prefix)\n const toolName = name.includes(\":\") ? name.split(\":\")[1] : name;\n console.log(`[SDK] Looking for handler: '${toolName}', available handlers:`, Array.from(this.toolHandlers.keys()));\n const handler = this.toolHandlers.get(toolName);\n\n // Emit event for debugging/logging\n this.emit(\"tool_call\", { name: toolName, arguments: args });\n\n if (!handler) {\n await this.sendToolResult(msgId, call_id, false, undefined, `Tool '${toolName}' not found`);\n return;\n }\n\n try {\n console.log(`[SDK] Executing tool '${toolName}' with args:`, args);\n // Execute with timeout\n const result = await Promise.race([\n handler(args),\n this.createTimeout(this.config.toolExecutionTimeout!),\n ]);\n\n console.log(`[SDK] Tool '${toolName}' completed successfully`);\n await this.sendToolResult(msgId, call_id, true, result);\n } catch (e) {\n const error = e instanceof Error ? e.message : String(e);\n console.log(`[SDK] Tool '${toolName}' failed:`, error);\n await this.sendToolResult(msgId, call_id, false, undefined, error);\n }\n }\n\n private async sendToolResult(\n id: string,\n callId: string,\n success: boolean,\n result?: unknown,\n error?: string\n ): Promise<void> {\n const message: ToolResultMessage = {\n id,\n type: \"tool_result\",\n call_id: callId,\n success,\n result,\n error,\n };\n\n console.log(`[SDK] Sending tool_result for ${callId}:`, success ? 'success' : `error: ${error}`);\n this.send(message);\n }\n\n // ============================================\n // Connection Management\n // ============================================\n\n private handleDisconnect(reason: string): void {\n this.stopHeartbeat();\n this.state.connected = false;\n this.state.registered = false;\n this.emit(\"disconnected\", reason);\n\n // Reject pending requests\n for (const [, pending] of this.pendingRequests) {\n pending.reject(new Error(\"Disconnected\"));\n }\n this.pendingRequests.clear();\n\n // Schedule reconnect\n this.scheduleReconnect();\n }\n\n private scheduleReconnect(): void {\n if (this.reconnectTimer) return;\n\n // Exponential backoff: 1s, 2s, 4s, 8s, ... up to 30s\n const delay = Math.min(\n 1000 * Math.pow(2, this.state.reconnectAttempts),\n 30000\n );\n\n console.log(`[SDK] Scheduling reconnect in ${delay}ms`);\n\n this.reconnectTimer = setTimeout(async () => {\n this.reconnectTimer = null;\n this.state.reconnectAttempts++;\n\n // Re-read connection info\n const info = this.discovery.read();\n if (info) {\n try {\n await this.connectWithInfo(info);\n } catch (e) {\n console.error(\"[SDK] Reconnect failed:\", e);\n this.scheduleReconnect();\n }\n } else {\n // Sanqian not running, keep trying\n this.scheduleReconnect();\n }\n }, delay);\n }\n\n private stopReconnect(): void {\n if (this.reconnectTimer) {\n clearTimeout(this.reconnectTimer);\n this.reconnectTimer = null;\n }\n }\n\n // ============================================\n // Heartbeat\n // ============================================\n\n private startHeartbeat(): void {\n this.stopHeartbeat();\n\n this.heartbeatTimer = setInterval(() => {\n const message: HeartbeatMessage = {\n type: \"heartbeat\",\n timestamp: new Date().toISOString(),\n };\n this.send(message);\n }, this.config.heartbeatInterval);\n }\n\n private stopHeartbeat(): void {\n if (this.heartbeatTimer) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n }\n\n // ============================================\n // Communication\n // ============================================\n\n private send(message: object): void {\n if (this.ws && this.ws.readyState === WebSocket.OPEN) {\n const data = JSON.stringify(message);\n console.log(`[SDK] WebSocket send:`, data.substring(0, 200));\n this.ws.send(data);\n } else {\n console.error(`[SDK] WebSocket not open, cannot send:`, message);\n }\n }\n\n private sendAndWait<T>(\n message: object,\n id: string,\n timeout: number\n ): Promise<T> {\n return new Promise((resolve, reject) => {\n const timer = setTimeout(() => {\n this.pendingRequests.delete(id);\n reject(new Error(\"Request timeout\"));\n }, timeout);\n\n this.pendingRequests.set(id, {\n resolve: (value) => {\n clearTimeout(timer);\n resolve(value as T);\n },\n reject: (error) => {\n clearTimeout(timer);\n reject(error);\n },\n });\n\n this.send(message);\n });\n }\n\n // ============================================\n // Public API\n // ============================================\n\n /**\n * Get current connection state\n */\n getState(): ConnectionState {\n return { ...this.state };\n }\n\n /**\n * Check if connected and registered\n */\n isConnected(): boolean {\n return this.state.connected && this.state.registered;\n }\n\n /**\n * Ensure connected to Sanqian, auto-launching if needed.\n * Used internally by chat() and chatStream() for auto-reconnect.\n */\n private async ensureConnected(): Promise<void> {\n if (this.isConnected()) {\n return;\n }\n\n // Stop any pending reconnect timer\n this.stopReconnect();\n\n // Try to read connection info first\n const info = this.discovery.read();\n\n if (info) {\n // Sanqian is running, just reconnect\n console.log(\"[SDK] Sanqian is running, reconnecting...\");\n await this.connectWithInfo(info);\n return;\n }\n\n // Sanqian not running, try to launch if enabled\n if (this.config.autoLaunchSanqian) {\n console.log(\"[SDK] Sanqian not running, attempting to launch...\");\n const launched = this.discovery.launchSanqian(this.config.sanqianPath);\n if (!launched) {\n throw new Error(\"Failed to launch Sanqian. Please start it manually.\");\n }\n }\n\n // Wait for Sanqian to start\n return new Promise((resolve, reject) => {\n console.log(\"[SDK] Waiting for Sanqian to start...\");\n\n const timeout = setTimeout(() => {\n this.discovery.stopWatching();\n reject(new Error(\"Sanqian connection timeout. Please ensure Sanqian is running.\"));\n }, 30000); // 30 second timeout for reconnect\n\n this.discovery.startWatching(async (newInfo) => {\n if (newInfo) {\n clearTimeout(timeout);\n this.discovery.stopWatching();\n try {\n await this.connectWithInfo(newInfo);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n });\n });\n }\n\n /**\n * Update tool list dynamically\n */\n async updateTools(tools: ToolDefinition[]): Promise<void> {\n // Update local handlers\n this.toolHandlers.clear();\n for (const tool of tools) {\n this.toolHandlers.set(tool.name, tool.handler);\n }\n\n // Send update to Sanqian\n if (this.isConnected()) {\n const msgId = this.generateId();\n const message = {\n id: msgId,\n type: \"tools_update\",\n tools: tools.map((t) => ({\n name: `${this.config.appName}:${t.name}`,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n await this.sendAndWait(message, msgId, 5000);\n }\n }\n\n // ============================================\n // Private Agent API\n // ============================================\n\n /**\n * Create or update a private agent\n *\n * Private agents are only visible to this SDK app, not in Sanqian UI.\n * If agent with same ID exists, it will be updated.\n *\n * @param config - Agent configuration\n * @returns Full agent info (or agent_id string for backward compatibility)\n */\n async createAgent(config: AgentConfig): Promise<AgentInfo> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: CreateAgentMessage = {\n id: msgId,\n type: \"create_agent\",\n agent: config,\n };\n\n const response = await this.sendAndWait<CreateAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to create agent\");\n }\n\n // Return full agent info if available, fallback to minimal info\n if (response.agent) {\n return response.agent;\n }\n\n // Backward compatibility: construct minimal AgentInfo from agent_id\n return {\n agent_id: response.agent_id!,\n name: config.name,\n description: config.description,\n tools: config.tools || [],\n };\n }\n\n /**\n * List all private agents owned by this app\n */\n async listAgents(): Promise<AgentInfo[]> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: ListAgentsMessage = {\n id: msgId,\n type: \"list_agents\",\n };\n\n const response = await this.sendAndWait<ListAgentsAckMessage>(message, msgId, 10000);\n\n if (response.error) {\n throw new Error(response.error);\n }\n\n return response.agents;\n }\n\n /**\n * Delete a private agent\n */\n async deleteAgent(agentId: string): Promise<void> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: DeleteAgentMessage = {\n id: msgId,\n type: \"delete_agent\",\n agent_id: agentId,\n };\n\n const response = await this.sendAndWait<DeleteAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to delete agent\");\n }\n }\n\n /**\n * Update a private agent\n *\n * Only updates the fields that are provided.\n * @param agentId - Agent ID (short name or full)\n * @param updates - Fields to update\n * @returns Updated agent info\n */\n async updateAgent(\n agentId: string,\n updates: Omit<AgentUpdateConfig, \"agent_id\">\n ): Promise<AgentInfo> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: UpdateAgentMessage = {\n id: msgId,\n type: \"update_agent\",\n agent: {\n agent_id: agentId,\n ...updates,\n },\n };\n\n const response = await this.sendAndWait<UpdateAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success || !response.agent) {\n throw new Error(response.error || \"Failed to update agent\");\n }\n\n return response.agent;\n }\n\n // ============================================\n // Conversation API\n // ============================================\n\n /**\n * List conversations for this app\n */\n async listConversations(options?: {\n agentId?: string;\n limit?: number;\n offset?: number;\n }): Promise<{ conversations: ConversationInfo[]; total: number }> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: ListConversationsMessage = {\n id: msgId,\n type: \"list_conversations\",\n agent_id: options?.agentId,\n limit: options?.limit,\n offset: options?.offset,\n };\n\n const response = await this.sendAndWait<ListConversationsAckMessage>(message, msgId, 10000);\n\n if (response.error) {\n throw new Error(response.error);\n }\n\n return {\n conversations: response.conversations,\n total: response.total,\n };\n }\n\n /**\n * Get conversation details with messages\n */\n async getConversation(\n conversationId: string,\n options?: {\n includeMessages?: boolean;\n messageLimit?: number;\n messageOffset?: number;\n }\n ): Promise<ConversationDetail> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: GetConversationMessage = {\n id: msgId,\n type: \"get_conversation\",\n conversation_id: conversationId,\n include_messages: options?.includeMessages ?? true,\n message_limit: options?.messageLimit,\n message_offset: options?.messageOffset,\n };\n\n const response = await this.sendAndWait<GetConversationAckMessage>(message, msgId, 10000);\n\n if (!response.success || !response.conversation) {\n throw new Error(response.error || \"Failed to get conversation\");\n }\n\n return response.conversation;\n }\n\n /**\n * Delete a conversation\n */\n async deleteConversation(conversationId: string): Promise<void> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: DeleteConversationMessage = {\n id: msgId,\n type: \"delete_conversation\",\n conversation_id: conversationId,\n };\n\n const response = await this.sendAndWait<DeleteConversationAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to delete conversation\");\n }\n }\n\n // ============================================\n // Chat API\n // ============================================\n\n // Pending stream handlers for streaming chat\n private streamHandlers: Map<string, {\n onEvent: (event: ChatStreamEvent) => void;\n onDone: (response: ChatResponse) => void;\n onError: (error: Error) => void;\n }> = new Map();\n\n /**\n * Send a chat message to an agent (non-streaming)\n *\n * Supports two modes:\n * - Stateless (no conversationId): Caller maintains message history\n * - Stateful (with conversationId): Server stores messages in conversations table\n *\n * @param agentId - Agent ID (short name or full, SDK auto-prefixes)\n * @param messages - Messages to send\n * @param options - Optional settings including conversationId and remoteTools\n * @returns Chat response with assistant message and conversation ID\n */\n async chat(\n agentId: string,\n messages: ChatMessage[],\n options?: {\n conversationId?: string;\n remoteTools?: RemoteToolDefinition[];\n }\n ): Promise<ChatResponse> {\n // Auto-reconnect if disconnected\n if (!this.isConnected()) {\n console.log(\"[SDK] Not connected, attempting to reconnect...\");\n await this.ensureConnected();\n }\n\n const msgId = this.generateId();\n const message: ChatRequestMessage = {\n id: msgId,\n type: \"chat\",\n agent_id: agentId,\n messages,\n conversation_id: options?.conversationId,\n stream: false,\n remote_tools: options?.remoteTools?.map((t) => ({\n name: t.name,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n // Longer timeout for chat (agent may take time with complex multi-tool tasks)\n const response = await this.sendAndWait<ChatResponseMessage>(message, msgId, 600000); // 10 minutes\n\n if (!response.success) {\n throw new Error(response.error || \"Chat request failed\");\n }\n\n return {\n message: response.message!,\n conversationId: response.conversation_id!,\n usage: response.usage,\n };\n }\n\n /**\n * Send a chat message to an agent with streaming response\n *\n * Supports two modes:\n * - Stateless (no conversationId): Caller maintains message history\n * - Stateful (with conversationId): Server stores messages in conversations table\n *\n * @param agentId - Agent ID (short name or full, SDK auto-prefixes)\n * @param messages - Messages to send\n * @param options - Optional settings including conversationId and remoteTools\n * @returns AsyncIterable of stream events\n */\n async *chatStream(\n agentId: string,\n messages: ChatMessage[],\n options?: {\n conversationId?: string;\n remoteTools?: RemoteToolDefinition[];\n }\n ): AsyncGenerator<ChatStreamEvent> {\n // Auto-reconnect if disconnected\n if (!this.isConnected()) {\n console.log(\"[SDK] Not connected, attempting to reconnect...\");\n await this.ensureConnected();\n }\n\n const msgId = this.generateId();\n const message: ChatRequestMessage = {\n id: msgId,\n type: \"chat\",\n agent_id: agentId,\n messages,\n conversation_id: options?.conversationId,\n stream: true,\n remote_tools: options?.remoteTools?.map((t) => ({\n name: t.name,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n // Create a queue to yield events\n const eventQueue: ChatStreamEvent[] = [];\n let done = false;\n let error: Error | null = null;\n let resolveNext: (() => void) | null = null;\n\n // Register stream handler\n this.streamHandlers.set(msgId, {\n onEvent: (event) => {\n eventQueue.push(event);\n resolveNext?.();\n },\n onDone: (response) => {\n eventQueue.push({\n type: \"done\",\n conversationId: response.conversationId,\n });\n done = true;\n resolveNext?.();\n },\n onError: (e) => {\n error = e;\n resolveNext?.();\n },\n });\n\n try {\n // Send request\n this.send(message);\n\n // Yield events as they arrive\n while (!done && !error) {\n if (eventQueue.length > 0) {\n yield eventQueue.shift()!;\n } else {\n // Wait for next event\n await new Promise<void>((resolve) => {\n resolveNext = resolve;\n });\n resolveNext = null;\n }\n }\n\n // Yield remaining events\n while (eventQueue.length > 0) {\n yield eventQueue.shift()!;\n }\n\n if (error) {\n throw error;\n }\n } finally {\n this.streamHandlers.delete(msgId);\n }\n }\n\n /**\n * Start a new conversation with an agent\n *\n * Returns a Conversation object for convenient multi-turn chat.\n *\n * @example\n * ```typescript\n * const conv = sdk.startConversation('assistant')\n * const r1 = await conv.send('Hello')\n * const r2 = await conv.send('Follow up question')\n * console.log(conv.id) // conversation ID\n * ```\n */\n startConversation(agentId: string): Conversation {\n return new Conversation(this, agentId);\n }\n\n // ============================================\n // Events\n // ============================================\n\n /**\n * Add event listener\n */\n on<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n if (!this.eventListeners.has(event)) {\n this.eventListeners.set(event, new Set());\n }\n this.eventListeners.get(event)!.add(listener as EventListener<SDKEventName>);\n return this;\n }\n\n /**\n * Remove event listener\n */\n off<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n this.eventListeners.get(event)?.delete(listener as EventListener<SDKEventName>);\n return this;\n }\n\n private emit<T extends SDKEventName>(\n event: T,\n ...args: Parameters<SDKEvents[T]>\n ): void {\n const listeners = this.eventListeners.get(event);\n if (listeners) {\n for (const listener of listeners) {\n try {\n (listener as (...args: unknown[]) => void)(...args);\n } catch (e) {\n console.error(`[SDK] Event listener error for '${event}':`, e);\n }\n }\n }\n }\n\n // ============================================\n // Utilities\n // ============================================\n\n private generateId(): string {\n return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;\n }\n\n private createTimeout(ms: number): Promise<never> {\n return new Promise((_, reject) => {\n setTimeout(() => reject(new Error(\"Timeout\")), ms);\n });\n }\n}\n\n/**\n * Conversation helper for multi-turn chat\n *\n * Automatically manages conversationId for stateful conversations.\n */\nexport class Conversation {\n private sdk: SanqianSDK;\n private agentId: string;\n private _conversationId: string | null = null;\n\n constructor(sdk: SanqianSDK, agentId: string, conversationId?: string) {\n this.sdk = sdk;\n this.agentId = agentId;\n this._conversationId = conversationId || null;\n }\n\n /**\n * Get the conversation ID (available after first message)\n */\n get id(): string | null {\n return this._conversationId;\n }\n\n /**\n * Send a message and get a response\n *\n * First call creates a new conversation, subsequent calls continue it.\n */\n async send(content: string, options?: { remoteTools?: RemoteToolDefinition[] }): Promise<ChatResponse> {\n const response = await this.sdk.chat(\n this.agentId,\n [{ role: \"user\", content }],\n {\n conversationId: this._conversationId || undefined,\n remoteTools: options?.remoteTools,\n }\n );\n\n // Store conversation ID for subsequent calls\n if (response.conversationId && !this._conversationId) {\n this._conversationId = response.conversationId;\n }\n\n return response;\n }\n\n /**\n * Send a message with streaming response\n */\n async *sendStream(\n content: string,\n options?: { remoteTools?: RemoteToolDefinition[] }\n ): AsyncGenerator<ChatStreamEvent> {\n const stream = this.sdk.chatStream(\n this.agentId,\n [{ role: \"user\", content }],\n {\n conversationId: this._conversationId || undefined,\n remoteTools: options?.remoteTools,\n }\n );\n\n for await (const event of stream) {\n // Capture conversation ID from done event\n if (event.type === \"done\" && event.conversationId && !this._conversationId) {\n this._conversationId = event.conversationId;\n }\n yield event;\n }\n }\n\n /**\n * Delete this conversation\n */\n async delete(): Promise<void> {\n if (!this._conversationId) {\n throw new Error(\"No conversation to delete\");\n }\n await this.sdk.deleteConversation(this._conversationId);\n this._conversationId = null;\n }\n\n /**\n * Get conversation details including message history\n */\n async getDetails(options?: { messageLimit?: number }): Promise<ConversationDetail> {\n if (!this._conversationId) {\n throw new Error(\"No conversation to get details for\");\n }\n return this.sdk.getConversation(this._conversationId, {\n includeMessages: true,\n messageLimit: options?.messageLimit,\n });\n }\n}\n","/**\n * Service Discovery Module\n *\n * Reads Sanqian's connection info from ~/.sanqian/runtime/connection.json\n * and monitors file changes for reconnection.\n * Also handles auto-launching Sanqian when needed.\n */\n\nimport { existsSync, readFileSync, watch, type FSWatcher } from \"fs\";\nimport { homedir, platform } from \"os\";\nimport { join } from \"path\";\nimport { spawn } from \"child_process\";\nimport type { ConnectionInfo } from \"./types\";\n\nexport class DiscoveryManager {\n private connectionInfo: ConnectionInfo | null = null;\n private watcher: FSWatcher | null = null;\n private onChange: ((info: ConnectionInfo | null) => void) | null = null;\n private pollInterval: ReturnType<typeof setInterval> | null = null;\n\n /**\n * Get the path to connection.json\n */\n getConnectionFilePath(): string {\n return join(homedir(), \".sanqian\", \"runtime\", \"connection.json\");\n }\n\n /**\n * Read and validate connection info\n *\n * Returns null if:\n * - File doesn't exist\n * - File is invalid JSON\n * - Process is not running\n */\n read(): ConnectionInfo | null {\n const filePath = this.getConnectionFilePath();\n\n if (!existsSync(filePath)) {\n return null;\n }\n\n try {\n const content = readFileSync(filePath, \"utf-8\");\n const info = JSON.parse(content) as ConnectionInfo;\n\n // Validate required fields\n if (!info.port || !info.token || !info.pid) {\n return null;\n }\n\n // Check if Sanqian process is running\n if (!this.isProcessRunning(info.pid)) {\n return null;\n }\n\n this.connectionInfo = info;\n return info;\n } catch {\n return null;\n }\n }\n\n /**\n * Start watching for connection file changes\n */\n startWatching(onChange: (info: ConnectionInfo | null) => void): void {\n this.onChange = onChange;\n\n const dir = join(homedir(), \".sanqian\", \"runtime\");\n\n // Ensure directory exists\n if (!existsSync(dir)) {\n // Directory doesn't exist, poll for it\n this.pollInterval = setInterval(() => {\n if (existsSync(dir)) {\n if (this.pollInterval) {\n clearInterval(this.pollInterval);\n this.pollInterval = null;\n }\n this.setupWatcher(dir);\n }\n }, 2000);\n return;\n }\n\n this.setupWatcher(dir);\n }\n\n private setupWatcher(dir: string): void {\n try {\n this.watcher = watch(dir, (event, filename) => {\n if (filename === \"connection.json\") {\n // Debounce rapid changes\n setTimeout(() => {\n const newInfo = this.read();\n const oldInfoStr = JSON.stringify(this.connectionInfo);\n const newInfoStr = JSON.stringify(newInfo);\n\n // Only trigger if actually changed\n if (oldInfoStr !== newInfoStr) {\n this.connectionInfo = newInfo;\n this.onChange?.(newInfo);\n }\n }, 100);\n }\n });\n } catch (e) {\n console.error(\"Failed to watch connection file:\", e);\n }\n }\n\n /**\n * Stop watching\n */\n stopWatching(): void {\n if (this.pollInterval) {\n clearInterval(this.pollInterval);\n this.pollInterval = null;\n }\n this.watcher?.close();\n this.watcher = null;\n this.onChange = null;\n }\n\n /**\n * Check if a process is running by PID\n */\n private isProcessRunning(pid: number): boolean {\n try {\n // Sending signal 0 doesn't kill the process, just checks if it exists\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n }\n\n /**\n * Get cached connection info (may be stale)\n */\n getCached(): ConnectionInfo | null {\n return this.connectionInfo;\n }\n\n /**\n * Build WebSocket URL from connection info\n */\n buildWebSocketUrl(info: ConnectionInfo): string {\n return `ws://127.0.0.1:${info.port}${info.ws_path}?token=${info.token}`;\n }\n\n /**\n * Build HTTP base URL from connection info\n */\n buildHttpUrl(info: ConnectionInfo): string {\n return `http://127.0.0.1:${info.port}`;\n }\n\n /**\n * Find Sanqian executable path\n * Searches in standard installation locations for each platform\n */\n findSanqianPath(customPath?: string): string | null {\n // If custom path provided, use it\n if (customPath) {\n if (existsSync(customPath)) {\n return customPath;\n }\n console.warn(`[SDK] Custom Sanqian path not found: ${customPath}`);\n return null;\n }\n\n const os = platform();\n const searchPaths: string[] = [];\n\n if (os === \"darwin\") {\n // macOS: Standard app locations + common dev locations\n searchPaths.push(\n // Production: installed app\n \"/Applications/Sanqian.app/Contents/MacOS/Sanqian\",\n join(homedir(), \"Applications/Sanqian.app/Contents/MacOS/Sanqian\"),\n // Development: electron-builder output\n join(homedir(), \"dev/sanqian/dist/mac-arm64/Sanqian.app/Contents/MacOS/Sanqian\"),\n join(homedir(), \"dev/sanqian/dist/mac/Sanqian.app/Contents/MacOS/Sanqian\"),\n );\n } else if (os === \"win32\") {\n // Windows: Standard installation paths + dev locations\n const programFiles = process.env.PROGRAMFILES || \"C:\\\\Program Files\";\n const programFilesX86 =\n process.env[\"PROGRAMFILES(X86)\"] || \"C:\\\\Program Files (x86)\";\n const localAppData =\n process.env.LOCALAPPDATA || join(homedir(), \"AppData\", \"Local\");\n\n searchPaths.push(\n // Production: installed app\n join(programFiles, \"Sanqian\", \"Sanqian.exe\"),\n join(programFilesX86, \"Sanqian\", \"Sanqian.exe\"),\n join(localAppData, \"Programs\", \"Sanqian\", \"Sanqian.exe\"),\n // Development: electron-builder output\n join(homedir(), \"dev\", \"sanqian\", \"dist\", \"win-unpacked\", \"Sanqian.exe\"),\n );\n } else {\n // Linux: Common locations\n searchPaths.push(\n \"/usr/bin/sanqian\",\n \"/usr/local/bin/sanqian\",\n join(homedir(), \".local/bin/sanqian\"),\n \"/opt/Sanqian/sanqian\",\n // Development\n join(homedir(), \"dev/sanqian/dist/linux-unpacked/sanqian\"),\n );\n }\n\n for (const path of searchPaths) {\n if (existsSync(path)) {\n return path;\n }\n }\n\n return null;\n }\n\n /**\n * Launch Sanqian in hidden/tray mode\n * Returns true if launch was initiated successfully\n */\n launchSanqian(customPath?: string): boolean {\n const sanqianPath = this.findSanqianPath(customPath);\n\n if (!sanqianPath) {\n console.error(\"[SDK] Sanqian executable not found\");\n return false;\n }\n\n console.log(`[SDK] Launching Sanqian from: ${sanqianPath}`);\n\n try {\n const os = platform();\n\n if (os === \"darwin\") {\n // macOS: Use 'open' command with --args for hidden start\n // This properly launches the app bundle\n const appPath = sanqianPath.replace(\n \"/Contents/MacOS/Sanqian\",\n \"\",\n );\n spawn(\"open\", [\"-a\", appPath, \"--args\", \"--hidden\"], {\n detached: true,\n stdio: \"ignore\",\n }).unref();\n } else {\n // Windows/Linux: Direct execution with --hidden flag\n spawn(sanqianPath, [\"--hidden\"], {\n detached: true,\n stdio: \"ignore\",\n // On Windows, hide the console window\n ...(os === \"win32\" && {\n windowsHide: true,\n shell: false,\n }),\n }).unref();\n }\n\n return true;\n } catch (e) {\n console.error(\"[SDK] Failed to launch Sanqian:\", e);\n return false;\n }\n }\n\n /**\n * Check if Sanqian is running\n */\n isSanqianRunning(): boolean {\n return this.read() !== null;\n }\n}\n"],"mappings":";AAMA,OAAO,eAAe;;;ACEtB,SAAS,YAAY,cAAc,aAA6B;AAChE,SAAS,SAAS,gBAAgB;AAClC,SAAS,YAAY;AACrB,SAAS,aAAa;AAGf,IAAM,mBAAN,MAAuB;AAAA,EACpB,iBAAwC;AAAA,EACxC,UAA4B;AAAA,EAC5B,WAA2D;AAAA,EAC3D,eAAsD;AAAA;AAAA;AAAA;AAAA,EAK9D,wBAAgC;AAC9B,WAAO,KAAK,QAAQ,GAAG,YAAY,WAAW,iBAAiB;AAAA,EACjE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,OAA8B;AAC5B,UAAM,WAAW,KAAK,sBAAsB;AAE5C,QAAI,CAAC,WAAW,QAAQ,GAAG;AACzB,aAAO;AAAA,IACT;AAEA,QAAI;AACF,YAAM,UAAU,aAAa,UAAU,OAAO;AAC9C,YAAM,OAAO,KAAK,MAAM,OAAO;AAG/B,UAAI,CAAC,KAAK,QAAQ,CAAC,KAAK,SAAS,CAAC,KAAK,KAAK;AAC1C,eAAO;AAAA,MACT;AAGA,UAAI,CAAC,KAAK,iBAAiB,KAAK,GAAG,GAAG;AACpC,eAAO;AAAA,MACT;AAEA,WAAK,iBAAiB;AACtB,aAAO;AAAA,IACT,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,cAAc,UAAuD;AACnE,SAAK,WAAW;AAEhB,UAAM,MAAM,KAAK,QAAQ,GAAG,YAAY,SAAS;AAGjD,QAAI,CAAC,WAAW,GAAG,GAAG;AAEpB,WAAK,eAAe,YAAY,MAAM;AACpC,YAAI,WAAW,GAAG,GAAG;AACnB,cAAI,KAAK,cAAc;AACrB,0BAAc,KAAK,YAAY;AAC/B,iBAAK,eAAe;AAAA,UACtB;AACA,eAAK,aAAa,GAAG;AAAA,QACvB;AAAA,MACF,GAAG,GAAI;AACP;AAAA,IACF;AAEA,SAAK,aAAa,GAAG;AAAA,EACvB;AAAA,EAEQ,aAAa,KAAmB;AACtC,QAAI;AACF,WAAK,UAAU,MAAM,KAAK,CAAC,OAAO,aAAa;AAC7C,YAAI,aAAa,mBAAmB;AAElC,qBAAW,MAAM;AACf,kBAAM,UAAU,KAAK,KAAK;AAC1B,kBAAM,aAAa,KAAK,UAAU,KAAK,cAAc;AACrD,kBAAM,aAAa,KAAK,UAAU,OAAO;AAGzC,gBAAI,eAAe,YAAY;AAC7B,mBAAK,iBAAiB;AACtB,mBAAK,WAAW,OAAO;AAAA,YACzB;AAAA,UACF,GAAG,GAAG;AAAA,QACR;AAAA,MACF,CAAC;AAAA,IACH,SAAS,GAAG;AACV,cAAQ,MAAM,oCAAoC,CAAC;AAAA,IACrD;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,eAAqB;AACnB,QAAI,KAAK,cAAc;AACrB,oBAAc,KAAK,YAAY;AAC/B,WAAK,eAAe;AAAA,IACtB;AACA,SAAK,SAAS,MAAM;AACpB,SAAK,UAAU;AACf,SAAK,WAAW;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA,EAKQ,iBAAiB,KAAsB;AAC7C,QAAI;AAEF,cAAQ,KAAK,KAAK,CAAC;AACnB,aAAO;AAAA,IACT,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,YAAmC;AACjC,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA,EAKA,kBAAkB,MAA8B;AAC9C,WAAO,kBAAkB,KAAK,IAAI,GAAG,KAAK,OAAO,UAAU,KAAK,KAAK;AAAA,EACvE;AAAA;AAAA;AAAA;AAAA,EAKA,aAAa,MAA8B;AACzC,WAAO,oBAAoB,KAAK,IAAI;AAAA,EACtC;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,gBAAgB,YAAoC;AAElD,QAAI,YAAY;AACd,UAAI,WAAW,UAAU,GAAG;AAC1B,eAAO;AAAA,MACT;AACA,cAAQ,KAAK,wCAAwC,UAAU,EAAE;AACjE,aAAO;AAAA,IACT;AAEA,UAAM,KAAK,SAAS;AACpB,UAAM,cAAwB,CAAC;AAE/B,QAAI,OAAO,UAAU;AAEnB,kBAAY;AAAA;AAAA,QAEV;AAAA,QACA,KAAK,QAAQ,GAAG,iDAAiD;AAAA;AAAA,QAEjE,KAAK,QAAQ,GAAG,+DAA+D;AAAA,QAC/E,KAAK,QAAQ,GAAG,yDAAyD;AAAA,MAC3E;AAAA,IACF,WAAW,OAAO,SAAS;AAEzB,YAAM,eAAe,QAAQ,IAAI,gBAAgB;AACjD,YAAM,kBACJ,QAAQ,IAAI,mBAAmB,KAAK;AACtC,YAAM,eACJ,QAAQ,IAAI,gBAAgB,KAAK,QAAQ,GAAG,WAAW,OAAO;AAEhE,kBAAY;AAAA;AAAA,QAEV,KAAK,cAAc,WAAW,aAAa;AAAA,QAC3C,KAAK,iBAAiB,WAAW,aAAa;AAAA,QAC9C,KAAK,cAAc,YAAY,WAAW,aAAa;AAAA;AAAA,QAEvD,KAAK,QAAQ,GAAG,OAAO,WAAW,QAAQ,gBAAgB,aAAa;AAAA,MACzE;AAAA,IACF,OAAO;AAEL,kBAAY;AAAA,QACV;AAAA,QACA;AAAA,QACA,KAAK,QAAQ,GAAG,oBAAoB;AAAA,QACpC;AAAA;AAAA,QAEA,KAAK,QAAQ,GAAG,yCAAyC;AAAA,MAC3D;AAAA,IACF;AAEA,eAAW,QAAQ,aAAa;AAC9B,UAAI,WAAW,IAAI,GAAG;AACpB,eAAO;AAAA,MACT;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,cAAc,YAA8B;AAC1C,UAAM,cAAc,KAAK,gBAAgB,UAAU;AAEnD,QAAI,CAAC,aAAa;AAChB,cAAQ,MAAM,oCAAoC;AAClD,aAAO;AAAA,IACT;AAEA,YAAQ,IAAI,iCAAiC,WAAW,EAAE;AAE1D,QAAI;AACF,YAAM,KAAK,SAAS;AAEpB,UAAI,OAAO,UAAU;AAGnB,cAAM,UAAU,YAAY;AAAA,UAC1B;AAAA,UACA;AAAA,QACF;AACA,cAAM,QAAQ,CAAC,MAAM,SAAS,UAAU,UAAU,GAAG;AAAA,UACnD,UAAU;AAAA,UACV,OAAO;AAAA,QACT,CAAC,EAAE,MAAM;AAAA,MACX,OAAO;AAEL,cAAM,aAAa,CAAC,UAAU,GAAG;AAAA,UAC/B,UAAU;AAAA,UACV,OAAO;AAAA;AAAA,UAEP,GAAI,OAAO,WAAW;AAAA,YACpB,aAAa;AAAA,YACb,OAAO;AAAA,UACT;AAAA,QACF,CAAC,EAAE,MAAM;AAAA,MACX;AAEA,aAAO;AAAA,IACT,SAAS,GAAG;AACV,cAAQ,MAAM,mCAAmC,CAAC;AAClD,aAAO;AAAA,IACT;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,mBAA4B;AAC1B,WAAO,KAAK,KAAK,MAAM;AAAA,EACzB;AACF;;;ADlOO,IAAM,aAAN,MAAiB;AAAA,EACd;AAAA,EACA;AAAA,EACA,KAAuB;AAAA,EACvB,iBAAwC;AAAA,EAExC,QAAyB;AAAA,IAC/B,WAAW;AAAA,IACX,aAAa;AAAA,IACb,YAAY;AAAA,IACZ,mBAAmB;AAAA,EACrB;AAAA;AAAA,EAGQ,eACN,oBAAI,IAAI;AAAA;AAAA,EAGF,kBAMJ,oBAAI,IAAI;AAAA;AAAA,EAGJ,iBAAwD;AAAA,EACxD,iBAAuD;AAAA;AAAA,EAGvD,iBACN,oBAAI,IAAI;AAAA,EAEV,YAAY,QAAmB;AAC7B,SAAK,SAAS;AAAA,MACZ,mBAAmB;AAAA,MACnB,mBAAmB;AAAA,MACnB,sBAAsB;AAAA,MACtB,GAAG;AAAA,IACL;AACA,SAAK,YAAY,IAAI,iBAAiB;AAGtC,eAAW,QAAQ,OAAO,OAAO;AAC/B,WAAK,aAAa,IAAI,KAAK,MAAM,KAAK,OAAO;AAAA,IAC/C;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAeA,MAAM,UAAyB;AAE7B,UAAM,OAAO,KAAK,UAAU,KAAK;AAEjC,QAAI,CAAC,MAAM;AAGT,UAAI,KAAK,OAAO,mBAAmB;AACjC,gBAAQ,IAAI,oDAAoD;AAChE,cAAM,WAAW,KAAK,UAAU,cAAc,KAAK,OAAO,WAAW;AACrE,YAAI,UAAU;AACZ,kBAAQ,IAAI,wDAAwD;AAAA,QACtE,OAAO;AACL,kBAAQ,KAAK,4DAA4D;AAAA,QAC3E;AAAA,MACF;AAGA,aAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,gBAAQ,IAAI,8BAA8B;AAE1C,cAAM,UAAU,WAAW,MAAM;AAC/B,eAAK,UAAU,aAAa;AAC5B,iBAAO,IAAI,MAAM,4BAA4B,CAAC;AAAA,QAChD,GAAG,GAAK;AAER,aAAK,UAAU,cAAc,OAAO,YAAY;AAC9C,cAAI,SAAS;AACX,yBAAa,OAAO;AACpB,iBAAK,UAAU,aAAa;AAC5B,gBAAI;AACF,oBAAM,KAAK,gBAAgB,OAAO;AAClC,sBAAQ;AAAA,YACV,SAAS,GAAG;AACV,qBAAO,CAAC;AAAA,YACV;AAAA,UACF;AAAA,QACF,CAAC;AAAA,MACH,CAAC;AAAA,IACH;AAEA,UAAM,KAAK,gBAAgB,IAAI;AAAA,EACjC;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,gBAAgB,MAAqC;AACjE,SAAK,iBAAiB;AACtB,UAAM,MAAM,KAAK,UAAU,kBAAkB,IAAI;AAEjD,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,cAAQ,IAAI,uBAAuB,GAAG,EAAE;AAExC,WAAK,KAAK,IAAI,UAAU,GAAG;AAE3B,YAAM,iBAAiB,WAAW,MAAM;AACtC,eAAO,IAAI,MAAM,8BAA8B,CAAC;AAChD,aAAK,IAAI,MAAM;AAAA,MACjB,GAAG,GAAK;AAER,WAAK,GAAG,GAAG,QAAQ,YAAY;AAC7B,qBAAa,cAAc;AAC3B,gBAAQ,IAAI,2BAA2B;AACvC,aAAK,MAAM,YAAY;AACvB,aAAK,MAAM,oBAAoB;AAC/B,aAAK,KAAK,WAAW;AAErB,YAAI;AACF,gBAAM,KAAK,SAAS;AACpB,kBAAQ;AAAA,QACV,SAAS,GAAG;AACV,iBAAO,CAAC;AAAA,QACV;AAAA,MACF,CAAC;AAED,WAAK,GAAG,GAAG,SAAS,CAAC,MAAM,WAAW;AACpC,gBAAQ,IAAI,2BAA2B,IAAI,IAAI,OAAO,SAAS,CAAC,EAAE;AAClE,aAAK,iBAAiB,OAAO,SAAS,CAAC;AAAA,MACzC,CAAC;AAED,WAAK,GAAG,GAAG,SAAS,CAAC,UAAU;AAC7B,gBAAQ,MAAM,0BAA0B,KAAK;AAC7C,aAAK,MAAM,YAAY;AACvB,aAAK,KAAK,SAAS,KAAK;AAAA,MAC1B,CAAC;AAED,WAAK,GAAG,GAAG,WAAW,CAAC,SAAS;AAC9B,YAAI;AACF,gBAAM,UAAU,KAAK,MAAM,KAAK,SAAS,CAAC;AAC1C,eAAK,cAAc,OAAO;AAAA,QAC5B,SAAS,GAAG;AACV,kBAAQ,MAAM,kCAAkC,CAAC;AAAA,QACnD;AAAA,MACF,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,aAA4B;AAChC,SAAK,cAAc;AACnB,SAAK,cAAc;AACnB,SAAK,UAAU,aAAa;AAE5B,QAAI,KAAK,IAAI;AACX,WAAK,GAAG,MAAM,KAAM,mBAAmB;AACvC,WAAK,KAAK;AAAA,IACZ;AAEA,SAAK,QAAQ;AAAA,MACX,WAAW;AAAA,MACX,aAAa;AAAA,MACb,YAAY;AAAA,MACZ,mBAAmB;AAAA,IACrB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAMA,MAAc,WAA0B;AACtC,SAAK,MAAM,cAAc;AAEzB,UAAM,QAAQ,KAAK,WAAW;AAE9B,UAAM,UAA2B;AAAA,MAC/B,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,KAAK;AAAA,QACH,MAAM,KAAK,OAAO;AAAA,QAClB,SAAS,KAAK,OAAO;AAAA,QACrB,cAAc,KAAK,OAAO;AAAA,QAC1B,gBAAgB,KAAK,OAAO;AAAA,MAC9B;AAAA,MACA,OAAO,KAAK,OAAO,MAAM,IAAI,CAAC,OAAO;AAAA,QACnC,MAAM,GAAG,KAAK,OAAO,OAAO,IAAI,EAAE,IAAI;AAAA,QACtC,aAAa,EAAE;AAAA,QACf,YAAY,EAAE;AAAA,MAChB,EAAE;AAAA,IACJ;AAEA,QAAI;AACF,YAAM,WAAW,MAAM,KAAK;AAAA,QAC1B;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAEA,UAAI,CAAC,SAAS,SAAS;AACrB,cAAM,IAAI,MAAM,SAAS,SAAS,qBAAqB;AAAA,MACzD;AAEA,WAAK,MAAM,cAAc;AACzB,WAAK,MAAM,aAAa;AACxB,WAAK,eAAe;AACpB,WAAK,KAAK,YAAY;AAEtB,cAAQ,IAAI,wBAAwB,KAAK,OAAO,OAAO,GAAG;AAAA,IAC5D,SAAS,GAAG;AACV,WAAK,MAAM,cAAc;AACzB,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAMQ,cAAc,SAAsE;AAC1F,UAAM,EAAE,IAAI,KAAK,IAAI;AAGrB,QAAI,MAAM,KAAK,gBAAgB,IAAI,EAAE,GAAG;AACtC,YAAM,UAAU,KAAK,gBAAgB,IAAI,EAAE;AAC3C,WAAK,gBAAgB,OAAO,EAAE;AAC9B,cAAQ,QAAQ,OAAO;AACvB;AAAA,IACF;AAGA,YAAQ,MAAM;AAAA,MACZ,KAAK;AACH,aAAK,eAAe,OAAqC;AACzD;AAAA,MAEF,KAAK;AAEH;AAAA,MAEF,KAAK;AAEH,aAAK,iBAAiB,OAAuC;AAC7D;AAAA,MAEF;AACE,gBAAQ,KAAK,+BAA+B,IAAI,EAAE;AAAA,IACtD;AAAA,EACF;AAAA,EAEQ,iBAAiB,SAAkC;AACzD,UAAM,EAAE,IAAI,OAAO,SAAS,WAAW,aAAa,iBAAiB,OAAO,MAAM,IAAI;AAEtF,QAAI,CAAC,GAAI;AAET,UAAM,UAAU,KAAK,eAAe,IAAI,EAAE;AAC1C,QAAI,CAAC,SAAS;AACZ,cAAQ,KAAK,uCAAuC,EAAE,EAAE;AACxD;AAAA,IACF;AAEA,YAAQ,OAAO;AAAA,MACb,KAAK;AACH,gBAAQ,QAAQ,EAAE,MAAM,QAAQ,QAAQ,CAAC;AACzC;AAAA,MAEF,KAAK;AACH,gBAAQ,QAAQ,EAAE,MAAM,aAAa,UAAU,CAAC;AAChD;AAAA,MAEF,KAAK;AACH,YAAI,aAAa;AACf,kBAAQ,QAAQ;AAAA,YACd,MAAM;AAAA,YACN,WAAW;AAAA,cACT,IAAI,YAAY;AAAA,cAChB,MAAM;AAAA,cACN,UAAU;AAAA,gBACR,MAAM;AAAA,gBACN,WAAW,KAAK,UAAU,WAAW;AAAA,cACvC;AAAA,YACF;AAAA,UACF,CAAC;AAAA,QACH;AACA;AAAA,MAEF,KAAK;AACH,gBAAQ,OAAO;AAAA,UACb,SAAS,QAAQ,WAAW,EAAE,MAAM,aAAa,SAAS,GAAG;AAAA,UAC7D,gBAAgB,mBAAmB;AAAA,UACnC;AAAA,QACF,CAAC;AACD;AAAA,MAEF,KAAK;AACH,gBAAQ,QAAQ,IAAI,MAAM,SAAS,sBAAsB,CAAC;AAC1D;AAAA,IACJ;AAAA,EACF;AAAA,EAEA,MAAc,eAAe,SAAyC;AACpE,YAAQ,IAAI,kCAAkC,KAAK,UAAU,OAAO,CAAC;AACrE,UAAM,EAAE,IAAI,SAAS,MAAM,WAAW,KAAK,IAAI;AAC/C,UAAM,QAAQ,MAAM;AAGpB,UAAM,WAAW,KAAK,SAAS,GAAG,IAAI,KAAK,MAAM,GAAG,EAAE,CAAC,IAAI;AAC3D,YAAQ,IAAI,+BAA+B,QAAQ,0BAA0B,MAAM,KAAK,KAAK,aAAa,KAAK,CAAC,CAAC;AACjH,UAAM,UAAU,KAAK,aAAa,IAAI,QAAQ;AAG9C,SAAK,KAAK,aAAa,EAAE,MAAM,UAAU,WAAW,KAAK,CAAC;AAE1D,QAAI,CAAC,SAAS;AACZ,YAAM,KAAK,eAAe,OAAO,SAAS,OAAO,QAAW,SAAS,QAAQ,aAAa;AAC1F;AAAA,IACF;AAEA,QAAI;AACF,cAAQ,IAAI,yBAAyB,QAAQ,gBAAgB,IAAI;AAEjE,YAAM,SAAS,MAAM,QAAQ,KAAK;AAAA,QAChC,QAAQ,IAAI;AAAA,QACZ,KAAK,cAAc,KAAK,OAAO,oBAAqB;AAAA,MACtD,CAAC;AAED,cAAQ,IAAI,eAAe,QAAQ,0BAA0B;AAC7D,YAAM,KAAK,eAAe,OAAO,SAAS,MAAM,MAAM;AAAA,IACxD,SAAS,GAAG;AACV,YAAM,QAAQ,aAAa,QAAQ,EAAE,UAAU,OAAO,CAAC;AACvD,cAAQ,IAAI,eAAe,QAAQ,aAAa,KAAK;AACrD,YAAM,KAAK,eAAe,OAAO,SAAS,OAAO,QAAW,KAAK;AAAA,IACnE;AAAA,EACF;AAAA,EAEA,MAAc,eACZ,IACA,QACA,SACA,QACA,OACe;AACf,UAAM,UAA6B;AAAA,MACjC;AAAA,MACA,MAAM;AAAA,MACN,SAAS;AAAA,MACT;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAEA,YAAQ,IAAI,iCAAiC,MAAM,KAAK,UAAU,YAAY,UAAU,KAAK,EAAE;AAC/F,SAAK,KAAK,OAAO;AAAA,EACnB;AAAA;AAAA;AAAA;AAAA,EAMQ,iBAAiB,QAAsB;AAC7C,SAAK,cAAc;AACnB,SAAK,MAAM,YAAY;AACvB,SAAK,MAAM,aAAa;AACxB,SAAK,KAAK,gBAAgB,MAAM;AAGhC,eAAW,CAAC,EAAE,OAAO,KAAK,KAAK,iBAAiB;AAC9C,cAAQ,OAAO,IAAI,MAAM,cAAc,CAAC;AAAA,IAC1C;AACA,SAAK,gBAAgB,MAAM;AAG3B,SAAK,kBAAkB;AAAA,EACzB;AAAA,EAEQ,oBAA0B;AAChC,QAAI,KAAK,eAAgB;AAGzB,UAAM,QAAQ,KAAK;AAAA,MACjB,MAAO,KAAK,IAAI,GAAG,KAAK,MAAM,iBAAiB;AAAA,MAC/C;AAAA,IACF;AAEA,YAAQ,IAAI,iCAAiC,KAAK,IAAI;AAEtD,SAAK,iBAAiB,WAAW,YAAY;AAC3C,WAAK,iBAAiB;AACtB,WAAK,MAAM;AAGX,YAAM,OAAO,KAAK,UAAU,KAAK;AACjC,UAAI,MAAM;AACR,YAAI;AACF,gBAAM,KAAK,gBAAgB,IAAI;AAAA,QACjC,SAAS,GAAG;AACV,kBAAQ,MAAM,2BAA2B,CAAC;AAC1C,eAAK,kBAAkB;AAAA,QACzB;AAAA,MACF,OAAO;AAEL,aAAK,kBAAkB;AAAA,MACzB;AAAA,IACF,GAAG,KAAK;AAAA,EACV;AAAA,EAEQ,gBAAsB;AAC5B,QAAI,KAAK,gBAAgB;AACvB,mBAAa,KAAK,cAAc;AAChC,WAAK,iBAAiB;AAAA,IACxB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAMQ,iBAAuB;AAC7B,SAAK,cAAc;AAEnB,SAAK,iBAAiB,YAAY,MAAM;AACtC,YAAM,UAA4B;AAAA,QAChC,MAAM;AAAA,QACN,YAAW,oBAAI,KAAK,GAAE,YAAY;AAAA,MACpC;AACA,WAAK,KAAK,OAAO;AAAA,IACnB,GAAG,KAAK,OAAO,iBAAiB;AAAA,EAClC;AAAA,EAEQ,gBAAsB;AAC5B,QAAI,KAAK,gBAAgB;AACvB,oBAAc,KAAK,cAAc;AACjC,WAAK,iBAAiB;AAAA,IACxB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAMQ,KAAK,SAAuB;AAClC,QAAI,KAAK,MAAM,KAAK,GAAG,eAAe,UAAU,MAAM;AACpD,YAAM,OAAO,KAAK,UAAU,OAAO;AACnC,cAAQ,IAAI,yBAAyB,KAAK,UAAU,GAAG,GAAG,CAAC;AAC3D,WAAK,GAAG,KAAK,IAAI;AAAA,IACnB,OAAO;AACL,cAAQ,MAAM,0CAA0C,OAAO;AAAA,IACjE;AAAA,EACF;AAAA,EAEQ,YACN,SACA,IACA,SACY;AACZ,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,YAAM,QAAQ,WAAW,MAAM;AAC7B,aAAK,gBAAgB,OAAO,EAAE;AAC9B,eAAO,IAAI,MAAM,iBAAiB,CAAC;AAAA,MACrC,GAAG,OAAO;AAEV,WAAK,gBAAgB,IAAI,IAAI;AAAA,QAC3B,SAAS,CAAC,UAAU;AAClB,uBAAa,KAAK;AAClB,kBAAQ,KAAU;AAAA,QACpB;AAAA,QACA,QAAQ,CAAC,UAAU;AACjB,uBAAa,KAAK;AAClB,iBAAO,KAAK;AAAA,QACd;AAAA,MACF,CAAC;AAED,WAAK,KAAK,OAAO;AAAA,IACnB,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,WAA4B;AAC1B,WAAO,EAAE,GAAG,KAAK,MAAM;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA,EAKA,cAAuB;AACrB,WAAO,KAAK,MAAM,aAAa,KAAK,MAAM;AAAA,EAC5C;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,MAAc,kBAAiC;AAC7C,QAAI,KAAK,YAAY,GAAG;AACtB;AAAA,IACF;AAGA,SAAK,cAAc;AAGnB,UAAM,OAAO,KAAK,UAAU,KAAK;AAEjC,QAAI,MAAM;AAER,cAAQ,IAAI,2CAA2C;AACvD,YAAM,KAAK,gBAAgB,IAAI;AAC/B;AAAA,IACF;AAGA,QAAI,KAAK,OAAO,mBAAmB;AACjC,cAAQ,IAAI,oDAAoD;AAChE,YAAM,WAAW,KAAK,UAAU,cAAc,KAAK,OAAO,WAAW;AACrE,UAAI,CAAC,UAAU;AACb,cAAM,IAAI,MAAM,qDAAqD;AAAA,MACvE;AAAA,IACF;AAGA,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,cAAQ,IAAI,uCAAuC;AAEnD,YAAM,UAAU,WAAW,MAAM;AAC/B,aAAK,UAAU,aAAa;AAC5B,eAAO,IAAI,MAAM,+DAA+D,CAAC;AAAA,MACnF,GAAG,GAAK;AAER,WAAK,UAAU,cAAc,OAAO,YAAY;AAC9C,YAAI,SAAS;AACX,uBAAa,OAAO;AACpB,eAAK,UAAU,aAAa;AAC5B,cAAI;AACF,kBAAM,KAAK,gBAAgB,OAAO;AAClC,oBAAQ;AAAA,UACV,SAAS,GAAG;AACV,mBAAO,CAAC;AAAA,UACV;AAAA,QACF;AAAA,MACF,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,YAAY,OAAwC;AAExD,SAAK,aAAa,MAAM;AACxB,eAAW,QAAQ,OAAO;AACxB,WAAK,aAAa,IAAI,KAAK,MAAM,KAAK,OAAO;AAAA,IAC/C;AAGA,QAAI,KAAK,YAAY,GAAG;AACtB,YAAM,QAAQ,KAAK,WAAW;AAC9B,YAAM,UAAU;AAAA,QACd,IAAI;AAAA,QACJ,MAAM;AAAA,QACN,OAAO,MAAM,IAAI,CAAC,OAAO;AAAA,UACvB,MAAM,GAAG,KAAK,OAAO,OAAO,IAAI,EAAE,IAAI;AAAA,UACtC,aAAa,EAAE;AAAA,UACf,YAAY,EAAE;AAAA,QAChB,EAAE;AAAA,MACJ;AAEA,YAAM,KAAK,YAAY,SAAS,OAAO,GAAI;AAAA,IAC7C;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAeA,MAAM,YAAY,QAAyC;AACzD,QAAI,CAAC,KAAK,YAAY,GAAG;AACvB,YAAM,IAAI,MAAM,0BAA0B;AAAA,IAC5C;AAEA,UAAM,QAAQ,KAAK,WAAW;AAC9B,UAAM,UAA8B;AAAA,MAClC,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,OAAO;AAAA,IACT;AAEA,UAAM,WAAW,MAAM,KAAK,YAAmC,SAAS,OAAO,GAAK;AAEpF,QAAI,CAAC,SAAS,SAAS;AACrB,YAAM,IAAI,MAAM,SAAS,SAAS,wBAAwB;AAAA,IAC5D;AAGA,QAAI,SAAS,OAAO;AAClB,aAAO,SAAS;AAAA,IAClB;AAGA,WAAO;AAAA,MACL,UAAU,SAAS;AAAA,MACnB,MAAM,OAAO;AAAA,MACb,aAAa,OAAO;AAAA,MACpB,OAAO,OAAO,SAAS,CAAC;AAAA,IAC1B;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,aAAmC;AACvC,QAAI,CAAC,KAAK,YAAY,GAAG;AACvB,YAAM,IAAI,MAAM,0BAA0B;AAAA,IAC5C;AAEA,UAAM,QAAQ,KAAK,WAAW;AAC9B,UAAM,UAA6B;AAAA,MACjC,IAAI;AAAA,MACJ,MAAM;AAAA,IACR;AAEA,UAAM,WAAW,MAAM,KAAK,YAAkC,SAAS,OAAO,GAAK;AAEnF,QAAI,SAAS,OAAO;AAClB,YAAM,IAAI,MAAM,SAAS,KAAK;AAAA,IAChC;AAEA,WAAO,SAAS;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,YAAY,SAAgC;AAChD,QAAI,CAAC,KAAK,YAAY,GAAG;AACvB,YAAM,IAAI,MAAM,0BAA0B;AAAA,IAC5C;AAEA,UAAM,QAAQ,KAAK,WAAW;AAC9B,UAAM,UAA8B;AAAA,MAClC,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU;AAAA,IACZ;AAEA,UAAM,WAAW,MAAM,KAAK,YAAmC,SAAS,OAAO,GAAK;AAEpF,QAAI,CAAC,SAAS,SAAS;AACrB,YAAM,IAAI,MAAM,SAAS,SAAS,wBAAwB;AAAA,IAC5D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,MAAM,YACJ,SACA,SACoB;AACpB,QAAI,CAAC,KAAK,YAAY,GAAG;AACvB,YAAM,IAAI,MAAM,0BAA0B;AAAA,IAC5C;AAEA,UAAM,QAAQ,KAAK,WAAW;AAC9B,UAAM,UAA8B;AAAA,MAClC,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,OAAO;AAAA,QACL,UAAU;AAAA,QACV,GAAG;AAAA,MACL;AAAA,IACF;AAEA,UAAM,WAAW,MAAM,KAAK,YAAmC,SAAS,OAAO,GAAK;AAEpF,QAAI,CAAC,SAAS,WAAW,CAAC,SAAS,OAAO;AACxC,YAAM,IAAI,MAAM,SAAS,SAAS,wBAAwB;AAAA,IAC5D;AAEA,WAAO,SAAS;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAM,kBAAkB,SAI0C;AAChE,QAAI,CAAC,KAAK,YAAY,GAAG;AACvB,YAAM,IAAI,MAAM,0BAA0B;AAAA,IAC5C;AAEA,UAAM,QAAQ,KAAK,WAAW;AAC9B,UAAM,UAAoC;AAAA,MACxC,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU,SAAS;AAAA,MACnB,OAAO,SAAS;AAAA,MAChB,QAAQ,SAAS;AAAA,IACnB;AAEA,UAAM,WAAW,MAAM,KAAK,YAAyC,SAAS,OAAO,GAAK;AAE1F,QAAI,SAAS,OAAO;AAClB,YAAM,IAAI,MAAM,SAAS,KAAK;AAAA,IAChC;AAEA,WAAO;AAAA,MACL,eAAe,SAAS;AAAA,MACxB,OAAO,SAAS;AAAA,IAClB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,gBACJ,gBACA,SAK6B;AAC7B,QAAI,CAAC,KAAK,YAAY,GAAG;AACvB,YAAM,IAAI,MAAM,0BAA0B;AAAA,IAC5C;AAEA,UAAM,QAAQ,KAAK,WAAW;AAC9B,UAAM,UAAkC;AAAA,MACtC,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,iBAAiB;AAAA,MACjB,kBAAkB,SAAS,mBAAmB;AAAA,MAC9C,eAAe,SAAS;AAAA,MACxB,gBAAgB,SAAS;AAAA,IAC3B;AAEA,UAAM,WAAW,MAAM,KAAK,YAAuC,SAAS,OAAO,GAAK;AAExF,QAAI,CAAC,SAAS,WAAW,CAAC,SAAS,cAAc;AAC/C,YAAM,IAAI,MAAM,SAAS,SAAS,4BAA4B;AAAA,IAChE;AAEA,WAAO,SAAS;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,mBAAmB,gBAAuC;AAC9D,QAAI,CAAC,KAAK,YAAY,GAAG;AACvB,YAAM,IAAI,MAAM,0BAA0B;AAAA,IAC5C;AAEA,UAAM,QAAQ,KAAK,WAAW;AAC9B,UAAM,UAAqC;AAAA,MACzC,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,iBAAiB;AAAA,IACnB;AAEA,UAAM,WAAW,MAAM,KAAK,YAA0C,SAAS,OAAO,GAAK;AAE3F,QAAI,CAAC,SAAS,SAAS;AACrB,YAAM,IAAI,MAAM,SAAS,SAAS,+BAA+B;AAAA,IACnE;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAOQ,iBAIH,oBAAI,IAAI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAcb,MAAM,KACJ,SACA,UACA,SAIuB;AAEvB,QAAI,CAAC,KAAK,YAAY,GAAG;AACvB,cAAQ,IAAI,iDAAiD;AAC7D,YAAM,KAAK,gBAAgB;AAAA,IAC7B;AAEA,UAAM,QAAQ,KAAK,WAAW;AAC9B,UAAM,UAA8B;AAAA,MAClC,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU;AAAA,MACV;AAAA,MACA,iBAAiB,SAAS;AAAA,MAC1B,QAAQ;AAAA,MACR,cAAc,SAAS,aAAa,IAAI,CAAC,OAAO;AAAA,QAC9C,MAAM,EAAE;AAAA,QACR,aAAa,EAAE;AAAA,QACf,YAAY,EAAE;AAAA,MAChB,EAAE;AAAA,IACJ;AAGA,UAAM,WAAW,MAAM,KAAK,YAAiC,SAAS,OAAO,GAAM;AAEnF,QAAI,CAAC,SAAS,SAAS;AACrB,YAAM,IAAI,MAAM,SAAS,SAAS,qBAAqB;AAAA,IACzD;AAEA,WAAO;AAAA,MACL,SAAS,SAAS;AAAA,MAClB,gBAAgB,SAAS;AAAA,MACzB,OAAO,SAAS;AAAA,IAClB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAcA,OAAO,WACL,SACA,UACA,SAIiC;AAEjC,QAAI,CAAC,KAAK,YAAY,GAAG;AACvB,cAAQ,IAAI,iDAAiD;AAC7D,YAAM,KAAK,gBAAgB;AAAA,IAC7B;AAEA,UAAM,QAAQ,KAAK,WAAW;AAC9B,UAAM,UAA8B;AAAA,MAClC,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,UAAU;AAAA,MACV;AAAA,MACA,iBAAiB,SAAS;AAAA,MAC1B,QAAQ;AAAA,MACR,cAAc,SAAS,aAAa,IAAI,CAAC,OAAO;AAAA,QAC9C,MAAM,EAAE;AAAA,QACR,aAAa,EAAE;AAAA,QACf,YAAY,EAAE;AAAA,MAChB,EAAE;AAAA,IACJ;AAGA,UAAM,aAAgC,CAAC;AACvC,QAAI,OAAO;AACX,QAAI,QAAsB;AAC1B,QAAI,cAAmC;AAGvC,SAAK,eAAe,IAAI,OAAO;AAAA,MAC7B,SAAS,CAAC,UAAU;AAClB,mBAAW,KAAK,KAAK;AACrB,sBAAc;AAAA,MAChB;AAAA,MACA,QAAQ,CAAC,aAAa;AACpB,mBAAW,KAAK;AAAA,UACd,MAAM;AAAA,UACN,gBAAgB,SAAS;AAAA,QAC3B,CAAC;AACD,eAAO;AACP,sBAAc;AAAA,MAChB;AAAA,MACA,SAAS,CAAC,MAAM;AACd,gBAAQ;AACR,sBAAc;AAAA,MAChB;AAAA,IACF,CAAC;AAED,QAAI;AAEF,WAAK,KAAK,OAAO;AAGjB,aAAO,CAAC,QAAQ,CAAC,OAAO;AACtB,YAAI,WAAW,SAAS,GAAG;AACzB,gBAAM,WAAW,MAAM;AAAA,QACzB,OAAO;AAEL,gBAAM,IAAI,QAAc,CAAC,YAAY;AACnC,0BAAc;AAAA,UAChB,CAAC;AACD,wBAAc;AAAA,QAChB;AAAA,MACF;AAGA,aAAO,WAAW,SAAS,GAAG;AAC5B,cAAM,WAAW,MAAM;AAAA,MACzB;AAEA,UAAI,OAAO;AACT,cAAM;AAAA,MACR;AAAA,IACF,UAAE;AACA,WAAK,eAAe,OAAO,KAAK;AAAA,IAClC;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAeA,kBAAkB,SAA+B;AAC/C,WAAO,IAAI,aAAa,MAAM,OAAO;AAAA,EACvC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,GAA2B,OAAU,UAA8B;AACjE,QAAI,CAAC,KAAK,eAAe,IAAI,KAAK,GAAG;AACnC,WAAK,eAAe,IAAI,OAAO,oBAAI,IAAI,CAAC;AAAA,IAC1C;AACA,SAAK,eAAe,IAAI,KAAK,EAAG,IAAI,QAAuC;AAC3E,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,IAA4B,OAAU,UAA8B;AAClE,SAAK,eAAe,IAAI,KAAK,GAAG,OAAO,QAAuC;AAC9E,WAAO;AAAA,EACT;AAAA,EAEQ,KACN,UACG,MACG;AACN,UAAM,YAAY,KAAK,eAAe,IAAI,KAAK;AAC/C,QAAI,WAAW;AACb,iBAAW,YAAY,WAAW;AAChC,YAAI;AACF,UAAC,SAA0C,GAAG,IAAI;AAAA,QACpD,SAAS,GAAG;AACV,kBAAQ,MAAM,mCAAmC,KAAK,MAAM,CAAC;AAAA,QAC/D;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAMQ,aAAqB;AAC3B,WAAO,GAAG,KAAK,IAAI,CAAC,IAAI,KAAK,OAAO,EAAE,SAAS,EAAE,EAAE,UAAU,GAAG,CAAC,CAAC;AAAA,EACpE;AAAA,EAEQ,cAAc,IAA4B;AAChD,WAAO,IAAI,QAAQ,CAAC,GAAG,WAAW;AAChC,iBAAW,MAAM,OAAO,IAAI,MAAM,SAAS,CAAC,GAAG,EAAE;AAAA,IACnD,CAAC;AAAA,EACH;AACF;AAOO,IAAM,eAAN,MAAmB;AAAA,EAChB;AAAA,EACA;AAAA,EACA,kBAAiC;AAAA,EAEzC,YAAY,KAAiB,SAAiB,gBAAyB;AACrE,SAAK,MAAM;AACX,SAAK,UAAU;AACf,SAAK,kBAAkB,kBAAkB;AAAA,EAC3C;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,KAAoB;AACtB,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAM,KAAK,SAAiB,SAA2E;AACrG,UAAM,WAAW,MAAM,KAAK,IAAI;AAAA,MAC9B,KAAK;AAAA,MACL,CAAC,EAAE,MAAM,QAAQ,QAAQ,CAAC;AAAA,MAC1B;AAAA,QACE,gBAAgB,KAAK,mBAAmB;AAAA,QACxC,aAAa,SAAS;AAAA,MACxB;AAAA,IACF;AAGA,QAAI,SAAS,kBAAkB,CAAC,KAAK,iBAAiB;AACpD,WAAK,kBAAkB,SAAS;AAAA,IAClC;AAEA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,OAAO,WACL,SACA,SACiC;AACjC,UAAM,SAAS,KAAK,IAAI;AAAA,MACtB,KAAK;AAAA,MACL,CAAC,EAAE,MAAM,QAAQ,QAAQ,CAAC;AAAA,MAC1B;AAAA,QACE,gBAAgB,KAAK,mBAAmB;AAAA,QACxC,aAAa,SAAS;AAAA,MACxB;AAAA,IACF;AAEA,qBAAiB,SAAS,QAAQ;AAEhC,UAAI,MAAM,SAAS,UAAU,MAAM,kBAAkB,CAAC,KAAK,iBAAiB;AAC1E,aAAK,kBAAkB,MAAM;AAAA,MAC/B;AACA,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,SAAwB;AAC5B,QAAI,CAAC,KAAK,iBAAiB;AACzB,YAAM,IAAI,MAAM,2BAA2B;AAAA,IAC7C;AACA,UAAM,KAAK,IAAI,mBAAmB,KAAK,eAAe;AACtD,SAAK,kBAAkB;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,WAAW,SAAkE;AACjF,QAAI,CAAC,KAAK,iBAAiB;AACzB,YAAM,IAAI,MAAM,oCAAoC;AAAA,IACtD;AACA,WAAO,KAAK,IAAI,gBAAgB,KAAK,iBAAiB;AAAA,MACpD,iBAAiB;AAAA,MACjB,cAAc,SAAS;AAAA,IACzB,CAAC;AAAA,EACH;AACF;","names":[]}
1
+ {"version":3,"sources":["../src/client.ts","../src/discovery.ts"],"sourcesContent":["/**\n * Sanqian SDK Client\n *\n * Main class for connecting to Sanqian and registering tools.\n */\n\nimport WebSocket from \"ws\";\nimport { DiscoveryManager } from \"./discovery\";\nimport type {\n SDKConfig,\n ConnectionInfo,\n ConnectionState,\n ToolDefinition,\n RegisterMessage,\n RegisterAckMessage,\n ToolCallMessage,\n ToolResultMessage,\n HeartbeatMessage,\n SDKEvents,\n SDKEventName,\n JSONSchema,\n AgentConfig,\n AgentInfo,\n AgentUpdateConfig,\n ConversationInfo,\n ConversationDetail,\n CreateAgentMessage,\n CreateAgentAckMessage,\n UpdateAgentMessage,\n UpdateAgentAckMessage,\n ListAgentsMessage,\n ListAgentsAckMessage,\n DeleteAgentMessage,\n DeleteAgentAckMessage,\n ListConversationsMessage,\n ListConversationsAckMessage,\n GetConversationMessage,\n GetConversationAckMessage,\n DeleteConversationMessage,\n DeleteConversationAckMessage,\n ChatMessage,\n ChatResponse,\n ChatRequestMessage,\n ChatResponseMessage,\n ChatStreamMessage,\n ChatStreamEvent,\n RemoteToolDefinition,\n} from \"./types\";\n\ntype EventListener<T extends SDKEventName> = SDKEvents[T];\n\nexport class SanqianSDK {\n private config: SDKConfig;\n private discovery: DiscoveryManager;\n private ws: WebSocket | null = null;\n private connectionInfo: ConnectionInfo | null = null;\n\n private state: ConnectionState = {\n connected: false,\n registering: false,\n registered: false,\n reconnectAttempts: 0,\n };\n\n // Tool handlers by name\n private toolHandlers: Map<string, (args: unknown) => Promise<unknown>> =\n new Map();\n\n // Pending request futures\n private pendingRequests: Map<\n string,\n {\n resolve: (value: unknown) => void;\n reject: (error: Error) => void;\n }\n > = new Map();\n\n // Timers\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n private reconnectTimer: ReturnType<typeof setTimeout> | null = null;\n\n // Event listeners\n private eventListeners: Map<SDKEventName, Set<EventListener<SDKEventName>>> =\n new Map();\n\n constructor(config: SDKConfig) {\n this.config = {\n reconnectInterval: 5000,\n heartbeatInterval: 30000,\n toolExecutionTimeout: 30000,\n ...config,\n };\n this.discovery = new DiscoveryManager();\n\n // Register tool handlers\n for (const tool of config.tools) {\n this.toolHandlers.set(tool.name, tool.handler);\n }\n }\n\n // ============================================\n // Lifecycle\n // ============================================\n\n /**\n * Connect to Sanqian\n *\n * Reads connection info, establishes WebSocket, and registers app.\n * Returns when registration is complete.\n *\n * If autoLaunchSanqian is enabled and Sanqian is not running,\n * SDK will attempt to start it in hidden/tray mode.\n */\n async connect(): Promise<void> {\n // Read connection info\n const info = this.discovery.read();\n\n if (!info) {\n // Sanqian not running\n // Try to auto-launch if enabled\n if (this.config.autoLaunchSanqian) {\n console.log(\"[SDK] Sanqian not running, attempting to launch...\");\n const launched = this.discovery.launchSanqian(this.config.sanqianPath);\n if (launched) {\n console.log(\"[SDK] Sanqian launch initiated, waiting for startup...\");\n } else {\n console.warn(\"[SDK] Failed to launch Sanqian, will wait for manual start\");\n }\n }\n\n // Start watching and wait for connection\n return new Promise((resolve, reject) => {\n console.log(\"[SDK] Waiting for Sanqian...\");\n\n const timeout = setTimeout(() => {\n this.discovery.stopWatching();\n reject(new Error(\"Sanqian connection timeout\"));\n }, 60000); // 60 second timeout\n\n this.discovery.startWatching(async (newInfo) => {\n if (newInfo) {\n clearTimeout(timeout);\n this.discovery.stopWatching();\n try {\n await this.connectWithInfo(newInfo);\n resolve();\n } catch (e) {\n reject(e);\n }\n }\n });\n });\n }\n\n await this.connectWithInfo(info);\n }\n\n /**\n * Connect with known connection info\n */\n private async connectWithInfo(info: ConnectionInfo): Promise<void> {\n this.connectionInfo = info;\n const url = this.discovery.buildWebSocketUrl(info);\n\n return new Promise((resolve, reject) => {\n console.log(`[SDK] Connecting to ${url}`);\n\n this.ws = new WebSocket(url);\n\n const connectTimeout = setTimeout(() => {\n reject(new Error(\"WebSocket connection timeout\"));\n this.ws?.close();\n }, 10000);\n\n this.ws.on(\"open\", async () => {\n clearTimeout(connectTimeout);\n console.log(\"[SDK] WebSocket connected\");\n this.state.connected = true;\n this.state.reconnectAttempts = 0;\n this.emit(\"connected\");\n\n try {\n await this.register();\n resolve();\n } catch (e) {\n reject(e);\n }\n });\n\n this.ws.on(\"close\", (code, reason) => {\n console.log(`[SDK] WebSocket closed: ${code} ${reason.toString()}`);\n this.handleDisconnect(reason.toString());\n });\n\n this.ws.on(\"error\", (error) => {\n console.error(\"[SDK] WebSocket error:\", error);\n this.state.lastError = error;\n this.emit(\"error\", error);\n });\n\n this.ws.on(\"message\", (data) => {\n try {\n const message = JSON.parse(data.toString());\n this.handleMessage(message);\n } catch (e) {\n console.error(\"[SDK] Failed to parse message:\", e);\n }\n });\n });\n }\n\n /**\n * Disconnect from Sanqian\n */\n async disconnect(): Promise<void> {\n this.stopHeartbeat();\n this.stopReconnect();\n this.discovery.stopWatching();\n\n if (this.ws) {\n this.ws.close(1000, \"Client disconnect\");\n this.ws = null;\n }\n\n this.state = {\n connected: false,\n registering: false,\n registered: false,\n reconnectAttempts: 0,\n };\n }\n\n // ============================================\n // Registration\n // ============================================\n\n private async register(): Promise<void> {\n this.state.registering = true;\n\n const msgId = this.generateId();\n\n const message: RegisterMessage = {\n id: msgId,\n type: \"register\",\n app: {\n name: this.config.appName,\n version: this.config.appVersion,\n display_name: this.config.displayName,\n launch_command: this.config.launchCommand,\n },\n tools: this.config.tools.map((t) => ({\n name: `${this.config.appName}:${t.name}`,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n try {\n const response = await this.sendAndWait<RegisterAckMessage>(\n message,\n msgId,\n 10000\n );\n\n if (!response.success) {\n throw new Error(response.error || \"Registration failed\");\n }\n\n this.state.registering = false;\n this.state.registered = true;\n this.startHeartbeat();\n this.emit(\"registered\");\n\n console.log(`[SDK] Registered as '${this.config.appName}'`);\n } catch (e) {\n this.state.registering = false;\n throw e;\n }\n }\n\n // ============================================\n // Message Handling\n // ============================================\n\n private handleMessage(message: { id?: string; type: string; [key: string]: unknown }): void {\n const { id, type } = message;\n\n // Check if this is a response to a pending request\n if (id && this.pendingRequests.has(id)) {\n const pending = this.pendingRequests.get(id)!;\n this.pendingRequests.delete(id);\n pending.resolve(message);\n return;\n }\n\n // Handle server-initiated messages\n switch (type) {\n case \"tool_call\":\n this.handleToolCall(message as unknown as ToolCallMessage);\n break;\n\n case \"heartbeat_ack\":\n // Heartbeat acknowledged, no action needed\n break;\n\n case \"chat_stream\":\n // Handle streaming chat response\n this.handleChatStream(message as unknown as ChatStreamMessage);\n break;\n\n default:\n console.warn(`[SDK] Unknown message type: ${type}`);\n }\n }\n\n private handleChatStream(message: ChatStreamMessage): void {\n const { id, event, content, tool_call, tool_result, conversation_id, title, usage, error } = message;\n\n if (!id) return;\n\n const handler = this.streamHandlers.get(id);\n if (!handler) {\n console.warn(`[SDK] No stream handler for message ${id}`);\n return;\n }\n\n switch (event) {\n case \"text\":\n handler.onEvent({ type: \"text\", content });\n break;\n\n case \"tool_call\":\n handler.onEvent({ type: \"tool_call\", tool_call });\n break;\n\n case \"tool_result\":\n if (tool_result) {\n handler.onEvent({\n type: \"tool_call\",\n tool_call: {\n id: tool_result.call_id,\n type: \"function\",\n function: {\n name: \"tool_result\",\n arguments: JSON.stringify(tool_result),\n },\n },\n });\n }\n break;\n\n case \"done\":\n handler.onDone({\n message: message.message || { role: \"assistant\", content: \"\" },\n conversationId: conversation_id || \"\",\n title,\n usage,\n });\n break;\n\n case \"error\":\n handler.onError(new Error(error || \"Unknown stream error\"));\n break;\n }\n }\n\n private async handleToolCall(message: ToolCallMessage): Promise<void> {\n console.log(`[SDK] handleToolCall received:`, JSON.stringify(message));\n const { id, call_id, name, arguments: args } = message;\n const msgId = id || call_id;\n\n // Extract actual tool name (remove app prefix)\n const toolName = name.includes(\":\") ? name.split(\":\")[1] : name;\n console.log(`[SDK] Looking for handler: '${toolName}', available handlers:`, Array.from(this.toolHandlers.keys()));\n const handler = this.toolHandlers.get(toolName);\n\n // Emit event for debugging/logging\n this.emit(\"tool_call\", { name: toolName, arguments: args });\n\n if (!handler) {\n await this.sendToolResult(msgId, call_id, false, undefined, `Tool '${toolName}' not found`);\n return;\n }\n\n try {\n console.log(`[SDK] Executing tool '${toolName}' with args:`, args);\n // Execute with timeout\n const result = await Promise.race([\n handler(args),\n this.createTimeout(this.config.toolExecutionTimeout!),\n ]);\n\n console.log(`[SDK] Tool '${toolName}' completed successfully`);\n await this.sendToolResult(msgId, call_id, true, result);\n } catch (e) {\n const error = e instanceof Error ? e.message : String(e);\n console.log(`[SDK] Tool '${toolName}' failed:`, error);\n await this.sendToolResult(msgId, call_id, false, undefined, error);\n }\n }\n\n private async sendToolResult(\n id: string,\n callId: string,\n success: boolean,\n result?: unknown,\n error?: string\n ): Promise<void> {\n const message: ToolResultMessage = {\n id,\n type: \"tool_result\",\n call_id: callId,\n success,\n result,\n error,\n };\n\n console.log(`[SDK] Sending tool_result for ${callId}:`, success ? 'success' : `error: ${error}`);\n this.send(message);\n }\n\n // ============================================\n // Connection Management\n // ============================================\n\n private handleDisconnect(reason: string): void {\n this.stopHeartbeat();\n this.state.connected = false;\n this.state.registered = false;\n this.emit(\"disconnected\", reason);\n\n // Reject pending requests\n for (const [, pending] of this.pendingRequests) {\n pending.reject(new Error(\"Disconnected\"));\n }\n this.pendingRequests.clear();\n\n // Schedule reconnect\n this.scheduleReconnect();\n }\n\n private scheduleReconnect(): void {\n if (this.reconnectTimer) return;\n\n // Exponential backoff: 500ms, 1s, 2s, 4s, max 5s\n // With jitter to avoid thundering herd\n const baseDelay = Math.min(\n 500 * Math.pow(2, this.state.reconnectAttempts),\n 5000 // Cap at 5 seconds\n );\n const jitter = Math.random() * 500; // 0-500ms random jitter\n const delay = baseDelay + jitter;\n\n console.log(`[SDK] Scheduling reconnect in ${Math.round(delay)}ms (attempt ${this.state.reconnectAttempts + 1})`);\n\n this.reconnectTimer = setTimeout(async () => {\n this.reconnectTimer = null;\n this.state.reconnectAttempts++;\n\n // Re-read connection info\n const info = this.discovery.read();\n if (info) {\n try {\n await this.connectWithInfo(info);\n } catch (e) {\n console.error(\"[SDK] Reconnect failed:\", e);\n this.scheduleReconnect();\n }\n } else {\n // Sanqian not running, keep trying\n this.scheduleReconnect();\n }\n }, delay);\n }\n\n private stopReconnect(): void {\n if (this.reconnectTimer) {\n clearTimeout(this.reconnectTimer);\n this.reconnectTimer = null;\n }\n }\n\n // ============================================\n // Heartbeat\n // ============================================\n\n private startHeartbeat(): void {\n this.stopHeartbeat();\n\n this.heartbeatTimer = setInterval(() => {\n const message: HeartbeatMessage = {\n type: \"heartbeat\",\n timestamp: new Date().toISOString(),\n };\n this.send(message);\n }, this.config.heartbeatInterval);\n }\n\n private stopHeartbeat(): void {\n if (this.heartbeatTimer) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n }\n\n // ============================================\n // Communication\n // ============================================\n\n private send(message: object): void {\n if (this.ws && this.ws.readyState === WebSocket.OPEN) {\n const data = JSON.stringify(message);\n console.log(`[SDK] WebSocket send:`, data.substring(0, 200));\n this.ws.send(data);\n } else {\n console.error(`[SDK] WebSocket not open, cannot send:`, message);\n }\n }\n\n private sendAndWait<T>(\n message: object,\n id: string,\n timeout: number\n ): Promise<T> {\n return new Promise((resolve, reject) => {\n const timer = setTimeout(() => {\n this.pendingRequests.delete(id);\n reject(new Error(\"Request timeout\"));\n }, timeout);\n\n this.pendingRequests.set(id, {\n resolve: (value) => {\n clearTimeout(timer);\n resolve(value as T);\n },\n reject: (error) => {\n clearTimeout(timer);\n reject(error);\n },\n });\n\n this.send(message);\n });\n }\n\n // ============================================\n // Public API\n // ============================================\n\n /**\n * Get current connection state\n */\n getState(): ConnectionState {\n return { ...this.state };\n }\n\n /**\n * Check if connected and registered\n */\n isConnected(): boolean {\n return this.state.connected && this.state.registered;\n }\n\n /**\n * Wait for connection to be established.\n * Used internally by chat() and chatStream() for auto-reconnect.\n * Relies on background scheduleReconnect() to do the actual reconnection.\n */\n private async waitForConnection(timeout: number = 30000): Promise<void> {\n if (this.isConnected()) {\n return;\n }\n\n console.log(\"[SDK] Not connected, waiting for reconnection...\");\n\n // If no reconnect is scheduled and Sanqian is not running, try to launch it\n if (!this.reconnectTimer) {\n const info = this.discovery.read();\n if (!info && this.config.autoLaunchSanqian) {\n console.log(\"[SDK] Sanqian not running, attempting to launch...\");\n const launched = this.discovery.launchSanqian(this.config.sanqianPath);\n if (launched) {\n // Trigger reconnect attempt\n this.scheduleReconnect();\n }\n } else if (!info) {\n // No connection info and can't auto-launch\n throw new Error(\"Sanqian is not running. Please start it manually.\");\n } else {\n // Connection info exists, trigger reconnect\n this.scheduleReconnect();\n }\n }\n\n // Wait for 'registered' event (emitted when connection + registration completes)\n return new Promise((resolve, reject) => {\n let resolved = false;\n\n const timer = setTimeout(() => {\n if (!resolved) {\n resolved = true;\n reject(new Error(\"Connection timeout. Please ensure Sanqian is running.\"));\n }\n }, timeout);\n\n this.once(\"registered\", () => {\n if (!resolved) {\n resolved = true;\n clearTimeout(timer);\n resolve();\n }\n });\n });\n }\n\n /**\n * Update tool list dynamically\n */\n async updateTools(tools: ToolDefinition[]): Promise<void> {\n // Update local handlers\n this.toolHandlers.clear();\n for (const tool of tools) {\n this.toolHandlers.set(tool.name, tool.handler);\n }\n\n // Send update to Sanqian\n if (this.isConnected()) {\n const msgId = this.generateId();\n const message = {\n id: msgId,\n type: \"tools_update\",\n tools: tools.map((t) => ({\n name: `${this.config.appName}:${t.name}`,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n await this.sendAndWait(message, msgId, 5000);\n }\n }\n\n // ============================================\n // Private Agent API\n // ============================================\n\n /**\n * Create or update a private agent\n *\n * Private agents are only visible to this SDK app, not in Sanqian UI.\n * If agent with same ID exists, it will be updated.\n *\n * @param config - Agent configuration\n * @returns Full agent info (or agent_id string for backward compatibility)\n */\n async createAgent(config: AgentConfig): Promise<AgentInfo> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: CreateAgentMessage = {\n id: msgId,\n type: \"create_agent\",\n agent: config,\n };\n\n const response = await this.sendAndWait<CreateAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to create agent\");\n }\n\n // Return full agent info if available, fallback to minimal info\n if (response.agent) {\n return response.agent;\n }\n\n // Backward compatibility: construct minimal AgentInfo from agent_id\n return {\n agent_id: response.agent_id!,\n name: config.name,\n description: config.description,\n tools: config.tools || [],\n };\n }\n\n /**\n * List all private agents owned by this app\n */\n async listAgents(): Promise<AgentInfo[]> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: ListAgentsMessage = {\n id: msgId,\n type: \"list_agents\",\n };\n\n const response = await this.sendAndWait<ListAgentsAckMessage>(message, msgId, 10000);\n\n if (response.error) {\n throw new Error(response.error);\n }\n\n return response.agents;\n }\n\n /**\n * Delete a private agent\n */\n async deleteAgent(agentId: string): Promise<void> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: DeleteAgentMessage = {\n id: msgId,\n type: \"delete_agent\",\n agent_id: agentId,\n };\n\n const response = await this.sendAndWait<DeleteAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to delete agent\");\n }\n }\n\n /**\n * Update a private agent\n *\n * Only updates the fields that are provided.\n * @param agentId - Agent ID (short name or full)\n * @param updates - Fields to update\n * @returns Updated agent info\n */\n async updateAgent(\n agentId: string,\n updates: Omit<AgentUpdateConfig, \"agent_id\">\n ): Promise<AgentInfo> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: UpdateAgentMessage = {\n id: msgId,\n type: \"update_agent\",\n agent: {\n agent_id: agentId,\n ...updates,\n },\n };\n\n const response = await this.sendAndWait<UpdateAgentAckMessage>(message, msgId, 10000);\n\n if (!response.success || !response.agent) {\n throw new Error(response.error || \"Failed to update agent\");\n }\n\n return response.agent;\n }\n\n // ============================================\n // Conversation API\n // ============================================\n\n /**\n * List conversations for this app\n */\n async listConversations(options?: {\n agentId?: string;\n limit?: number;\n offset?: number;\n }): Promise<{ conversations: ConversationInfo[]; total: number }> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: ListConversationsMessage = {\n id: msgId,\n type: \"list_conversations\",\n agent_id: options?.agentId,\n limit: options?.limit,\n offset: options?.offset,\n };\n\n const response = await this.sendAndWait<ListConversationsAckMessage>(message, msgId, 10000);\n\n if (response.error) {\n throw new Error(response.error);\n }\n\n return {\n conversations: response.conversations,\n total: response.total,\n };\n }\n\n /**\n * Get conversation details with messages\n */\n async getConversation(\n conversationId: string,\n options?: {\n includeMessages?: boolean;\n messageLimit?: number;\n messageOffset?: number;\n }\n ): Promise<ConversationDetail> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: GetConversationMessage = {\n id: msgId,\n type: \"get_conversation\",\n conversation_id: conversationId,\n include_messages: options?.includeMessages ?? true,\n message_limit: options?.messageLimit,\n message_offset: options?.messageOffset,\n };\n\n const response = await this.sendAndWait<GetConversationAckMessage>(message, msgId, 10000);\n\n if (!response.success || !response.conversation) {\n throw new Error(response.error || \"Failed to get conversation\");\n }\n\n return response.conversation;\n }\n\n /**\n * Delete a conversation\n */\n async deleteConversation(conversationId: string): Promise<void> {\n if (!this.isConnected()) {\n throw new Error(\"Not connected to Sanqian\");\n }\n\n const msgId = this.generateId();\n const message: DeleteConversationMessage = {\n id: msgId,\n type: \"delete_conversation\",\n conversation_id: conversationId,\n };\n\n const response = await this.sendAndWait<DeleteConversationAckMessage>(message, msgId, 10000);\n\n if (!response.success) {\n throw new Error(response.error || \"Failed to delete conversation\");\n }\n }\n\n // ============================================\n // Chat API\n // ============================================\n\n // Pending stream handlers for streaming chat\n private streamHandlers: Map<string, {\n onEvent: (event: ChatStreamEvent) => void;\n onDone: (response: ChatResponse) => void;\n onError: (error: Error) => void;\n }> = new Map();\n\n /**\n * Send a chat message to an agent (non-streaming)\n *\n * Supports two modes:\n * - Stateless (no conversationId): Caller maintains message history\n * - Stateful (with conversationId): Server stores messages in conversations table\n *\n * @param agentId - Agent ID (short name or full, SDK auto-prefixes)\n * @param messages - Messages to send\n * @param options - Optional settings including conversationId and remoteTools\n * @returns Chat response with assistant message and conversation ID\n */\n async chat(\n agentId: string,\n messages: ChatMessage[],\n options?: {\n conversationId?: string;\n remoteTools?: RemoteToolDefinition[];\n }\n ): Promise<ChatResponse> {\n // Auto-reconnect if disconnected\n if (!this.isConnected()) {\n console.log(\"[SDK] Not connected, attempting to reconnect...\");\n await this.waitForConnection();\n }\n\n const msgId = this.generateId();\n const message: ChatRequestMessage = {\n id: msgId,\n type: \"chat\",\n agent_id: agentId,\n messages,\n conversation_id: options?.conversationId,\n stream: false,\n remote_tools: options?.remoteTools?.map((t) => ({\n name: t.name,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n // Longer timeout for chat (agent may take time with complex multi-tool tasks)\n const response = await this.sendAndWait<ChatResponseMessage>(message, msgId, 600000); // 10 minutes\n\n if (!response.success) {\n throw new Error(response.error || \"Chat request failed\");\n }\n\n return {\n message: response.message!,\n conversationId: response.conversation_id!,\n title: response.title,\n usage: response.usage,\n };\n }\n\n /**\n * Send a chat message to an agent with streaming response\n *\n * Supports two modes:\n * - Stateless (no conversationId): Caller maintains message history\n * - Stateful (with conversationId): Server stores messages in conversations table\n *\n * @param agentId - Agent ID (short name or full, SDK auto-prefixes)\n * @param messages - Messages to send\n * @param options - Optional settings including conversationId and remoteTools\n * @returns AsyncIterable of stream events\n */\n async *chatStream(\n agentId: string,\n messages: ChatMessage[],\n options?: {\n conversationId?: string;\n remoteTools?: RemoteToolDefinition[];\n }\n ): AsyncGenerator<ChatStreamEvent> {\n // Auto-reconnect if disconnected\n if (!this.isConnected()) {\n console.log(\"[SDK] Not connected, attempting to reconnect...\");\n await this.waitForConnection();\n }\n\n const msgId = this.generateId();\n const message: ChatRequestMessage = {\n id: msgId,\n type: \"chat\",\n agent_id: agentId,\n messages,\n conversation_id: options?.conversationId,\n stream: true,\n remote_tools: options?.remoteTools?.map((t) => ({\n name: t.name,\n description: t.description,\n parameters: t.parameters,\n })),\n };\n\n // Create a queue to yield events\n const eventQueue: ChatStreamEvent[] = [];\n let done = false;\n let error: Error | null = null;\n let resolveNext: (() => void) | null = null;\n\n // Register stream handler\n this.streamHandlers.set(msgId, {\n onEvent: (event) => {\n eventQueue.push(event);\n resolveNext?.();\n },\n onDone: (response) => {\n eventQueue.push({\n type: \"done\",\n conversationId: response.conversationId,\n title: response.title,\n });\n done = true;\n resolveNext?.();\n },\n onError: (e) => {\n error = e;\n resolveNext?.();\n },\n });\n\n try {\n // Send request\n this.send(message);\n\n // Yield events as they arrive\n while (!done && !error) {\n if (eventQueue.length > 0) {\n yield eventQueue.shift()!;\n } else {\n // Wait for next event\n await new Promise<void>((resolve) => {\n resolveNext = resolve;\n });\n resolveNext = null;\n }\n }\n\n // Yield remaining events\n while (eventQueue.length > 0) {\n yield eventQueue.shift()!;\n }\n\n if (error) {\n throw error;\n }\n } finally {\n this.streamHandlers.delete(msgId);\n }\n }\n\n /**\n * Start a new conversation with an agent\n *\n * Returns a Conversation object for convenient multi-turn chat.\n *\n * @example\n * ```typescript\n * const conv = sdk.startConversation('assistant')\n * const r1 = await conv.send('Hello')\n * const r2 = await conv.send('Follow up question')\n * console.log(conv.id) // conversation ID\n * ```\n */\n startConversation(agentId: string): Conversation {\n return new Conversation(this, agentId);\n }\n\n // ============================================\n // Events\n // ============================================\n\n /**\n * Add event listener\n */\n on<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n if (!this.eventListeners.has(event)) {\n this.eventListeners.set(event, new Set());\n }\n this.eventListeners.get(event)!.add(listener as EventListener<SDKEventName>);\n return this;\n }\n\n /**\n * Remove event listener\n */\n off<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n this.eventListeners.get(event)?.delete(listener as EventListener<SDKEventName>);\n return this;\n }\n\n /**\n * Add one-time event listener (auto-removes after first call)\n */\n once<T extends SDKEventName>(event: T, listener: SDKEvents[T]): this {\n const onceWrapper = ((...args: Parameters<SDKEvents[T]>) => {\n this.off(event, onceWrapper as SDKEvents[T]);\n (listener as (...args: unknown[]) => void)(...args);\n }) as SDKEvents[T];\n return this.on(event, onceWrapper);\n }\n\n private emit<T extends SDKEventName>(\n event: T,\n ...args: Parameters<SDKEvents[T]>\n ): void {\n const listeners = this.eventListeners.get(event);\n if (listeners) {\n for (const listener of listeners) {\n try {\n (listener as (...args: unknown[]) => void)(...args);\n } catch (e) {\n console.error(`[SDK] Event listener error for '${event}':`, e);\n }\n }\n }\n }\n\n // ============================================\n // Utilities\n // ============================================\n\n private generateId(): string {\n return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;\n }\n\n private createTimeout(ms: number): Promise<never> {\n return new Promise((_, reject) => {\n setTimeout(() => reject(new Error(\"Timeout\")), ms);\n });\n }\n}\n\n/**\n * Conversation helper for multi-turn chat\n *\n * Automatically manages conversationId for stateful conversations.\n */\nexport class Conversation {\n private sdk: SanqianSDK;\n private agentId: string;\n private _conversationId: string | null = null;\n\n constructor(sdk: SanqianSDK, agentId: string, conversationId?: string) {\n this.sdk = sdk;\n this.agentId = agentId;\n this._conversationId = conversationId || null;\n }\n\n /**\n * Get the conversation ID (available after first message)\n */\n get id(): string | null {\n return this._conversationId;\n }\n\n /**\n * Send a message and get a response\n *\n * First call creates a new conversation, subsequent calls continue it.\n */\n async send(content: string, options?: { remoteTools?: RemoteToolDefinition[] }): Promise<ChatResponse> {\n const response = await this.sdk.chat(\n this.agentId,\n [{ role: \"user\", content }],\n {\n conversationId: this._conversationId || undefined,\n remoteTools: options?.remoteTools,\n }\n );\n\n // Store conversation ID for subsequent calls\n if (response.conversationId && !this._conversationId) {\n this._conversationId = response.conversationId;\n }\n\n return response;\n }\n\n /**\n * Send a message with streaming response\n */\n async *sendStream(\n content: string,\n options?: { remoteTools?: RemoteToolDefinition[] }\n ): AsyncGenerator<ChatStreamEvent> {\n const stream = this.sdk.chatStream(\n this.agentId,\n [{ role: \"user\", content }],\n {\n conversationId: this._conversationId || undefined,\n remoteTools: options?.remoteTools,\n }\n );\n\n for await (const event of stream) {\n // Capture conversation ID from done event\n if (event.type === \"done\" && event.conversationId && !this._conversationId) {\n this._conversationId = event.conversationId;\n }\n yield event;\n }\n }\n\n /**\n * Delete this conversation\n */\n async delete(): Promise<void> {\n if (!this._conversationId) {\n throw new Error(\"No conversation to delete\");\n }\n await this.sdk.deleteConversation(this._conversationId);\n this._conversationId = null;\n }\n\n /**\n * Get conversation details including message history\n */\n async getDetails(options?: { messageLimit?: number }): Promise<ConversationDetail> {\n if (!this._conversationId) {\n throw new Error(\"No conversation to get details for\");\n }\n return this.sdk.getConversation(this._conversationId, {\n includeMessages: true,\n messageLimit: options?.messageLimit,\n });\n }\n}\n","/**\n * Service Discovery Module\n *\n * Reads Sanqian's connection info from ~/.sanqian/runtime/connection.json\n * and monitors file changes for reconnection.\n * Also handles auto-launching Sanqian when needed.\n */\n\nimport { existsSync, readFileSync, watch, type FSWatcher } from \"fs\";\nimport { homedir, platform } from \"os\";\nimport { join } from \"path\";\nimport { spawn } from \"child_process\";\nimport type { ConnectionInfo } from \"./types\";\n\nexport class DiscoveryManager {\n private connectionInfo: ConnectionInfo | null = null;\n private watcher: FSWatcher | null = null;\n private onChange: ((info: ConnectionInfo | null) => void) | null = null;\n private pollInterval: ReturnType<typeof setInterval> | null = null;\n\n /**\n * Get the path to connection.json\n */\n getConnectionFilePath(): string {\n return join(homedir(), \".sanqian\", \"runtime\", \"connection.json\");\n }\n\n /**\n * Read and validate connection info\n *\n * Returns null if:\n * - File doesn't exist\n * - File is invalid JSON\n * - Process is not running\n */\n read(): ConnectionInfo | null {\n const filePath = this.getConnectionFilePath();\n\n if (!existsSync(filePath)) {\n return null;\n }\n\n try {\n const content = readFileSync(filePath, \"utf-8\");\n const info = JSON.parse(content) as ConnectionInfo;\n\n // Validate required fields\n if (!info.port || !info.token || !info.pid) {\n return null;\n }\n\n // Check if Sanqian process is running\n if (!this.isProcessRunning(info.pid)) {\n return null;\n }\n\n this.connectionInfo = info;\n return info;\n } catch {\n return null;\n }\n }\n\n /**\n * Start watching for connection file changes\n */\n startWatching(onChange: (info: ConnectionInfo | null) => void): void {\n this.onChange = onChange;\n\n const dir = join(homedir(), \".sanqian\", \"runtime\");\n\n // Ensure directory exists\n if (!existsSync(dir)) {\n // Directory doesn't exist, poll for it\n this.pollInterval = setInterval(() => {\n if (existsSync(dir)) {\n if (this.pollInterval) {\n clearInterval(this.pollInterval);\n this.pollInterval = null;\n }\n this.setupWatcher(dir);\n }\n }, 2000);\n return;\n }\n\n this.setupWatcher(dir);\n }\n\n private setupWatcher(dir: string): void {\n try {\n this.watcher = watch(dir, (event, filename) => {\n if (filename === \"connection.json\") {\n // Debounce rapid changes\n setTimeout(() => {\n const newInfo = this.read();\n const oldInfoStr = JSON.stringify(this.connectionInfo);\n const newInfoStr = JSON.stringify(newInfo);\n\n // Only trigger if actually changed\n if (oldInfoStr !== newInfoStr) {\n this.connectionInfo = newInfo;\n this.onChange?.(newInfo);\n }\n }, 100);\n }\n });\n } catch (e) {\n console.error(\"Failed to watch connection file:\", e);\n }\n }\n\n /**\n * Stop watching\n */\n stopWatching(): void {\n if (this.pollInterval) {\n clearInterval(this.pollInterval);\n this.pollInterval = null;\n }\n this.watcher?.close();\n this.watcher = null;\n this.onChange = null;\n }\n\n /**\n * Check if a process is running by PID\n */\n private isProcessRunning(pid: number): boolean {\n try {\n // Sending signal 0 doesn't kill the process, just checks if it exists\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n }\n\n /**\n * Get cached connection info (may be stale)\n */\n getCached(): ConnectionInfo | null {\n return this.connectionInfo;\n }\n\n /**\n * Build WebSocket URL from connection info\n */\n buildWebSocketUrl(info: ConnectionInfo): string {\n return `ws://127.0.0.1:${info.port}${info.ws_path}?token=${info.token}`;\n }\n\n /**\n * Build HTTP base URL from connection info\n */\n buildHttpUrl(info: ConnectionInfo): string {\n return `http://127.0.0.1:${info.port}`;\n }\n\n /**\n * Find Sanqian executable path\n * Searches in standard installation locations for each platform\n */\n findSanqianPath(customPath?: string): string | null {\n // If custom path provided, use it\n if (customPath) {\n if (existsSync(customPath)) {\n return customPath;\n }\n console.warn(`[SDK] Custom Sanqian path not found: ${customPath}`);\n return null;\n }\n\n const os = platform();\n const searchPaths: string[] = [];\n\n if (os === \"darwin\") {\n // macOS: Standard app locations + common dev locations\n searchPaths.push(\n // Production: installed app\n \"/Applications/Sanqian.app/Contents/MacOS/Sanqian\",\n join(homedir(), \"Applications/Sanqian.app/Contents/MacOS/Sanqian\"),\n // Development: electron-builder output\n join(homedir(), \"dev/sanqian/dist/mac-arm64/Sanqian.app/Contents/MacOS/Sanqian\"),\n join(homedir(), \"dev/sanqian/dist/mac/Sanqian.app/Contents/MacOS/Sanqian\"),\n );\n } else if (os === \"win32\") {\n // Windows: Standard installation paths + dev locations\n const programFiles = process.env.PROGRAMFILES || \"C:\\\\Program Files\";\n const programFilesX86 =\n process.env[\"PROGRAMFILES(X86)\"] || \"C:\\\\Program Files (x86)\";\n const localAppData =\n process.env.LOCALAPPDATA || join(homedir(), \"AppData\", \"Local\");\n\n searchPaths.push(\n // Production: installed app\n join(programFiles, \"Sanqian\", \"Sanqian.exe\"),\n join(programFilesX86, \"Sanqian\", \"Sanqian.exe\"),\n join(localAppData, \"Programs\", \"Sanqian\", \"Sanqian.exe\"),\n // Development: electron-builder output\n join(homedir(), \"dev\", \"sanqian\", \"dist\", \"win-unpacked\", \"Sanqian.exe\"),\n );\n } else {\n // Linux: Common locations\n searchPaths.push(\n \"/usr/bin/sanqian\",\n \"/usr/local/bin/sanqian\",\n join(homedir(), \".local/bin/sanqian\"),\n \"/opt/Sanqian/sanqian\",\n // Development\n join(homedir(), \"dev/sanqian/dist/linux-unpacked/sanqian\"),\n );\n }\n\n for (const path of searchPaths) {\n if (existsSync(path)) {\n return path;\n }\n }\n\n return null;\n }\n\n /**\n * Launch Sanqian in hidden/tray mode\n * Returns true if launch was initiated successfully\n */\n launchSanqian(customPath?: string): boolean {\n const sanqianPath = this.findSanqianPath(customPath);\n\n if (!sanqianPath) {\n console.error(\"[SDK] Sanqian executable not found\");\n return false;\n }\n\n console.log(`[SDK] Launching Sanqian from: ${sanqianPath}`);\n\n try {\n const os = platform();\n\n if (os === \"darwin\") {\n // macOS: Use 'open' command with --args for hidden start\n // This properly launches the app bundle\n const appPath = sanqianPath.replace(\n \"/Contents/MacOS/Sanqian\",\n \"\",\n );\n spawn(\"open\", [\"-a\", appPath, \"--args\", \"--hidden\"], {\n detached: true,\n stdio: \"ignore\",\n }).unref();\n } else {\n // Windows/Linux: Direct execution with --hidden flag\n spawn(sanqianPath, [\"--hidden\"], {\n detached: true,\n stdio: \"ignore\",\n // On Windows, hide the console window\n ...(os === \"win32\" && {\n windowsHide: true,\n shell: false,\n }),\n }).unref();\n }\n\n return true;\n } catch (e) {\n console.error(\"[SDK] Failed to launch Sanqian:\", e);\n return false;\n }\n }\n\n /**\n * Check if Sanqian is running\n */\n isSanqianRunning(): boolean {\n return this.read() !== null;\n }\n}\n"],"mappings":"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","names":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@yushaw/sanqian-sdk",
3
- "version": "0.1.1",
3
+ "version": "0.1.3",
4
4
  "description": "Sanqian SDK for third-party app integration",
5
5
  "main": "./dist/index.js",
6
6
  "module": "./dist/index.mjs",