@ainetwork/adk 0.2.1 → 0.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (173) hide show
  1. package/README.md +50 -24
  2. package/dist/cjs/{chunk-GB3ON7UV.cjs → chunk-7PZP7YSA.cjs} +5 -5
  3. package/dist/cjs/{chunk-GB3ON7UV.cjs.map → chunk-7PZP7YSA.cjs.map} +1 -1
  4. package/dist/cjs/{chunk-MIWLK57K.cjs → chunk-A2I7D6JW.cjs} +1 -1
  5. package/dist/cjs/chunk-A2I7D6JW.cjs.map +1 -0
  6. package/dist/cjs/{chunk-6YVTJEMQ.cjs → chunk-BBFYVWE2.cjs} +17 -42
  7. package/dist/cjs/chunk-BBFYVWE2.cjs.map +1 -0
  8. package/dist/cjs/{chunk-LRBRG2DS.cjs → chunk-E35E4LQY.cjs} +4 -4
  9. package/dist/cjs/chunk-E35E4LQY.cjs.map +1 -0
  10. package/dist/cjs/{chunk-45G7LDK7.cjs → chunk-FRWC3FQ4.cjs} +3 -3
  11. package/dist/cjs/{chunk-45G7LDK7.cjs.map → chunk-FRWC3FQ4.cjs.map} +1 -1
  12. package/dist/cjs/{chunk-AIE3SVG7.cjs → chunk-KHD7UAFW.cjs} +22 -24
  13. package/dist/cjs/chunk-KHD7UAFW.cjs.map +1 -0
  14. package/dist/cjs/{chunk-2C33PZTV.cjs → chunk-LMR6B76V.cjs} +6 -6
  15. package/dist/cjs/chunk-LMR6B76V.cjs.map +1 -0
  16. package/dist/cjs/{chunk-55IME4BG.cjs → chunk-OKRBMMK2.cjs} +46 -24
  17. package/dist/cjs/chunk-OKRBMMK2.cjs.map +1 -0
  18. package/dist/cjs/{chunk-DX2JH3FJ.cjs → chunk-QHQLU7BI.cjs} +1 -1
  19. package/dist/cjs/{chunk-DX2JH3FJ.cjs.map → chunk-QHQLU7BI.cjs.map} +1 -1
  20. package/dist/cjs/{chunk-QWASSVCU.cjs → chunk-RIJV7J5Z.cjs} +1 -1
  21. package/dist/cjs/chunk-RIJV7J5Z.cjs.map +1 -0
  22. package/dist/cjs/{chunk-3J3ORQOG.cjs → chunk-VLMHDHXT.cjs} +3 -3
  23. package/dist/cjs/{chunk-3J3ORQOG.cjs.map → chunk-VLMHDHXT.cjs.map} +1 -1
  24. package/dist/cjs/{chunk-EOPW5OPE.cjs → chunk-W5VM75PL.cjs} +7 -7
  25. package/dist/cjs/{chunk-4DXS33VP.cjs.map → chunk-W5VM75PL.cjs.map} +1 -1
  26. package/dist/cjs/{chunk-CU5MOHJU.cjs → chunk-ZPYMLCQZ.cjs} +13 -14
  27. package/dist/cjs/chunk-ZPYMLCQZ.cjs.map +1 -0
  28. package/dist/cjs/controllers/api/threads.api.controller.cjs +2 -2
  29. package/dist/cjs/controllers/index.cjs +7 -7
  30. package/dist/cjs/controllers/index.cjs.map +1 -1
  31. package/dist/cjs/controllers/query.controller.cjs +3 -3
  32. package/dist/cjs/index.cjs +45 -31
  33. package/dist/cjs/index.cjs.map +1 -1
  34. package/dist/cjs/middlewares/auth.middleware.cjs +3 -3
  35. package/dist/cjs/modules/a2a/a2a.module.cjs +2 -2
  36. package/dist/cjs/modules/index.cjs +9 -9
  37. package/dist/cjs/modules/index.cjs.map +1 -1
  38. package/dist/cjs/modules/models/model.module.cjs +2 -2
  39. package/dist/cjs/routes/a2a.routes.cjs +5 -5
  40. package/dist/cjs/routes/api/threads.routes.cjs +4 -4
  41. package/dist/cjs/routes/api.routes.cjs +5 -5
  42. package/dist/cjs/routes/index.cjs +17 -17
  43. package/dist/cjs/routes/index.cjs.map +1 -1
  44. package/dist/cjs/routes/query.routes.cjs +6 -6
  45. package/dist/cjs/services/a2a.service.cjs +2 -2
  46. package/dist/cjs/services/index.cjs +4 -4
  47. package/dist/cjs/services/query-stream.service.cjs +2 -2
  48. package/dist/cjs/services/query.service.cjs +2 -2
  49. package/dist/cjs/types/agent.cjs +2 -2
  50. package/dist/esm/{chunk-7D4SKPIS.js → chunk-4C2TLCWI.js} +2 -2
  51. package/dist/esm/{chunk-PN6NMQBM.js → chunk-5JIXQ6PU.js} +2 -2
  52. package/dist/esm/chunk-5JIXQ6PU.js.map +1 -0
  53. package/dist/esm/{chunk-ILKEX4ZV.js → chunk-7IYLOABV.js} +15 -17
  54. package/dist/esm/chunk-7IYLOABV.js.map +1 -0
  55. package/dist/esm/{chunk-EIHZ4S5V.js → chunk-AJRQEL5N.js} +6 -6
  56. package/dist/esm/{chunk-2L2KJNAM.js → chunk-EUUW7J3S.js} +8 -8
  57. package/dist/esm/chunk-EUUW7J3S.js.map +1 -0
  58. package/dist/esm/{chunk-CWFIK2EK.js → chunk-GDA6Z5AL.js} +9 -10
  59. package/dist/esm/chunk-GDA6Z5AL.js.map +1 -0
  60. package/dist/esm/{chunk-LJVVSUF7.js → chunk-MJEPJ62P.js} +15 -40
  61. package/dist/esm/chunk-MJEPJ62P.js.map +1 -0
  62. package/dist/esm/{chunk-2RW77MJB.js → chunk-PUUZRUE6.js} +47 -25
  63. package/dist/esm/chunk-PUUZRUE6.js.map +1 -0
  64. package/dist/esm/{chunk-SYCFBEUA.js → chunk-QHWDMU43.js} +3 -3
  65. package/dist/esm/{chunk-WLT2B2SZ.js → chunk-RL6SRSSK.js} +1 -1
  66. package/dist/esm/chunk-RL6SRSSK.js.map +1 -0
  67. package/dist/esm/{chunk-VDXWZ4NK.js → chunk-TV2GJAEH.js} +2 -2
  68. package/dist/esm/{chunk-PHKK5NHK.js → chunk-VNQCUGC3.js} +1 -1
  69. package/dist/esm/chunk-VNQCUGC3.js.map +1 -0
  70. package/dist/esm/{chunk-FL3JRQM4.js → chunk-ZBUAT73P.js} +1 -1
  71. package/dist/esm/{chunk-FL3JRQM4.js.map → chunk-ZBUAT73P.js.map} +1 -1
  72. package/dist/esm/controllers/a2a.controller.d.ts +2 -2
  73. package/dist/esm/controllers/api/threads.api.controller.js +1 -1
  74. package/dist/esm/controllers/index.d.ts +3 -3
  75. package/dist/esm/controllers/index.js +9 -9
  76. package/dist/esm/controllers/query.controller.d.ts +2 -2
  77. package/dist/esm/controllers/query.controller.js +2 -2
  78. package/dist/esm/index.d.ts +1 -2
  79. package/dist/esm/index.js +44 -30
  80. package/dist/esm/index.js.map +1 -1
  81. package/dist/esm/middlewares/auth.middleware.js +2 -2
  82. package/dist/esm/modules/a2a/a2a.module.d.ts +2 -9
  83. package/dist/esm/modules/a2a/a2a.module.js +2 -2
  84. package/dist/esm/modules/index.js +6 -6
  85. package/dist/esm/modules/models/model.module.d.ts +17 -0
  86. package/dist/esm/modules/models/model.module.js +1 -1
  87. package/dist/esm/routes/a2a.routes.js +5 -5
  88. package/dist/esm/routes/api/threads.routes.js +3 -3
  89. package/dist/esm/routes/api.routes.js +4 -4
  90. package/dist/esm/routes/index.js +16 -16
  91. package/dist/esm/routes/query.routes.js +6 -6
  92. package/dist/esm/services/a2a.service.d.ts +4 -4
  93. package/dist/esm/services/a2a.service.js +1 -1
  94. package/dist/esm/services/index.js +7 -7
  95. package/dist/esm/services/query-stream.service.d.ts +13 -10
  96. package/dist/esm/services/query-stream.service.js +2 -2
  97. package/dist/esm/services/query.service.d.ts +7 -4
  98. package/dist/esm/services/query.service.js +2 -2
  99. package/dist/esm/types/agent.d.ts +0 -2
  100. package/dist/esm/types/agent.js +1 -1
  101. package/package.json +9 -4
  102. package/dist/cjs/chunk-2C33PZTV.cjs.map +0 -1
  103. package/dist/cjs/chunk-344VLDMS.cjs +0 -43
  104. package/dist/cjs/chunk-344VLDMS.cjs.map +0 -1
  105. package/dist/cjs/chunk-4DXS33VP.cjs +0 -43
  106. package/dist/cjs/chunk-55IME4BG.cjs.map +0 -1
  107. package/dist/cjs/chunk-6YVTJEMQ.cjs.map +0 -1
  108. package/dist/cjs/chunk-7GWYMGWW.cjs +0 -248
  109. package/dist/cjs/chunk-7GWYMGWW.cjs.map +0 -1
  110. package/dist/cjs/chunk-7HUWKIXU.cjs +0 -128
  111. package/dist/cjs/chunk-7HUWKIXU.cjs.map +0 -1
  112. package/dist/cjs/chunk-AIE3SVG7.cjs.map +0 -1
  113. package/dist/cjs/chunk-BBCZU5VZ.cjs +0 -17
  114. package/dist/cjs/chunk-BBCZU5VZ.cjs.map +0 -1
  115. package/dist/cjs/chunk-BOKIE5SM.cjs +0 -37
  116. package/dist/cjs/chunk-BOKIE5SM.cjs.map +0 -1
  117. package/dist/cjs/chunk-CU5MOHJU.cjs.map +0 -1
  118. package/dist/cjs/chunk-CYW6ICAW.cjs +0 -106
  119. package/dist/cjs/chunk-CYW6ICAW.cjs.map +0 -1
  120. package/dist/cjs/chunk-EAK7MP6X.cjs +0 -22
  121. package/dist/cjs/chunk-EAK7MP6X.cjs.map +0 -1
  122. package/dist/cjs/chunk-EOPW5OPE.cjs.map +0 -1
  123. package/dist/cjs/chunk-F4CJCUAK.cjs +0 -22
  124. package/dist/cjs/chunk-F4CJCUAK.cjs.map +0 -1
  125. package/dist/cjs/chunk-FCZQBJTF.cjs +0 -36
  126. package/dist/cjs/chunk-FCZQBJTF.cjs.map +0 -1
  127. package/dist/cjs/chunk-G5KEV27W.cjs +0 -45
  128. package/dist/cjs/chunk-G5KEV27W.cjs.map +0 -1
  129. package/dist/cjs/chunk-H35XHCTA.cjs +0 -22
  130. package/dist/cjs/chunk-H35XHCTA.cjs.map +0 -1
  131. package/dist/cjs/chunk-HT3PACCL.cjs +0 -37
  132. package/dist/cjs/chunk-HT3PACCL.cjs.map +0 -1
  133. package/dist/cjs/chunk-JXKLEQD6.cjs +0 -37
  134. package/dist/cjs/chunk-JXKLEQD6.cjs.map +0 -1
  135. package/dist/cjs/chunk-KE7SL3H6.cjs +0 -251
  136. package/dist/cjs/chunk-KE7SL3H6.cjs.map +0 -1
  137. package/dist/cjs/chunk-KRLUWTLN.cjs +0 -35
  138. package/dist/cjs/chunk-KRLUWTLN.cjs.map +0 -1
  139. package/dist/cjs/chunk-LRBRG2DS.cjs.map +0 -1
  140. package/dist/cjs/chunk-MIWLK57K.cjs.map +0 -1
  141. package/dist/cjs/chunk-MLUUV7UY.cjs +0 -36
  142. package/dist/cjs/chunk-MLUUV7UY.cjs.map +0 -1
  143. package/dist/cjs/chunk-QWASSVCU.cjs.map +0 -1
  144. package/dist/cjs/chunk-RX72NXXR.cjs +0 -43
  145. package/dist/cjs/chunk-RX72NXXR.cjs.map +0 -1
  146. package/dist/cjs/chunk-T2AKW5PA.cjs +0 -45
  147. package/dist/cjs/chunk-T2AKW5PA.cjs.map +0 -1
  148. package/dist/cjs/chunk-UTFWH24D.cjs +0 -22
  149. package/dist/cjs/chunk-UTFWH24D.cjs.map +0 -1
  150. package/dist/cjs/chunk-W52YCYRF.cjs +0 -43
  151. package/dist/cjs/chunk-W52YCYRF.cjs.map +0 -1
  152. package/dist/cjs/chunk-YNBMJ5QX.cjs +0 -129
  153. package/dist/cjs/chunk-YNBMJ5QX.cjs.map +0 -1
  154. package/dist/cjs/chunk-ZZK4ACTV.cjs +0 -289
  155. package/dist/cjs/chunk-ZZK4ACTV.cjs.map +0 -1
  156. package/dist/cjs/chunk-ZZPJ5CPR.cjs +0 -328
  157. package/dist/cjs/chunk-ZZPJ5CPR.cjs.map +0 -1
  158. package/dist/cjs/controllers/api/thread.api.controller.cjs +0 -8
  159. package/dist/cjs/controllers/api/thread.api.controller.cjs.map +0 -1
  160. package/dist/cjs/routes/api/thread.routes.cjs +0 -9
  161. package/dist/cjs/routes/api/thread.routes.cjs.map +0 -1
  162. package/dist/esm/chunk-2L2KJNAM.js.map +0 -1
  163. package/dist/esm/chunk-2RW77MJB.js.map +0 -1
  164. package/dist/esm/chunk-CWFIK2EK.js.map +0 -1
  165. package/dist/esm/chunk-ILKEX4ZV.js.map +0 -1
  166. package/dist/esm/chunk-LJVVSUF7.js.map +0 -1
  167. package/dist/esm/chunk-PHKK5NHK.js.map +0 -1
  168. package/dist/esm/chunk-PN6NMQBM.js.map +0 -1
  169. package/dist/esm/chunk-WLT2B2SZ.js.map +0 -1
  170. /package/dist/esm/{chunk-7D4SKPIS.js.map → chunk-4C2TLCWI.js.map} +0 -0
  171. /package/dist/esm/{chunk-EIHZ4S5V.js.map → chunk-AJRQEL5N.js.map} +0 -0
  172. /package/dist/esm/{chunk-SYCFBEUA.js.map → chunk-QHWDMU43.js.map} +0 -0
  173. /package/dist/esm/{chunk-VDXWZ4NK.js.map → chunk-TV2GJAEH.js.map} +0 -0
package/README.md CHANGED
@@ -41,11 +41,11 @@ To see how to use this package in your project, check out our comprehensive exam
41
41
 
42
42
  ### Core Components
43
43
 
44
- - **AINAgent** (`src/app.ts`): Main Express.js server class that orchestrates all modules
45
- - **ModelModule** (`src/modules/models/`): Manages AI model integrations (OpenAI, Gemini)
44
+ - **AINAgent** (`src/index.ts`): Main Express.js server class that orchestrates all modules
45
+ - **ModelModule** (`src/modules/models/`): Manages AI model integrations with streaming support
46
46
  - **MCPModule** (`src/modules/mcp/`): Handles Model Context Protocol connections
47
47
  - **A2AModule** (`src/modules/a2a/`): Manages agent-to-agent communication
48
- - **MemoryModule** (`src/modules/memory/`): Provides threads and conversation history
48
+ - **MemoryModule** (`src/modules/memory/`): Provides threads, intents, and conversation history
49
49
 
50
50
  ### Module System
51
51
 
@@ -68,20 +68,25 @@ Each module can be independently configured and passed to the agent constructor.
68
68
  - Connects to external MCP servers via stdio
69
69
  - Automatic tool discovery and execution
70
70
  - Supports multiple concurrent MCP servers
71
+ - Protocol-specific tool wrapping as `IMCPTool`
71
72
 
72
73
  #### A2A (Agent-to-Agent)
73
74
  - RESTful API for inter-agent communication
74
- - Streaming response support
75
- - Agent discovery via well-known endpoints
76
- - Task delegation and context passing
75
+ - Streaming response support via SSE
76
+ - Agent discovery via well-known endpoints (`.well-known/agent-card.json`)
77
+ - Task delegation with thread context passing
78
+ - Protocol version 0.3.0 support
77
79
 
78
80
  ### Key Features
79
81
 
80
82
  - **Unified Tool Interface**: Protocol-agnostic `IAgentTool` interface
83
+ - **Streaming Support**: Dual implementation for streaming and non-streaming queries
84
+ - **Intent System**: Intent detection and fulfillment with custom prompts
81
85
  - **Service Layer**: Clean separation with controllers and services
82
- - **Type Safety**: Comprehensive TypeScript types throughout
86
+ - **Type Safety**: Comprehensive TypeScript types with strict mode
83
87
  - **Error Handling**: Global error middleware with structured logging
84
- - **Authentication**: Optional auth middleware support
88
+ - **Authentication**: Optional auth middleware via `BaseAuth` interface
89
+ - **Graceful Shutdown**: Proper cleanup of modules and connections
85
90
 
86
91
  ## Development
87
92
 
@@ -96,10 +101,13 @@ yarn dev # Run TypeScript directly with tsx
96
101
 
97
102
  # Code quality
98
103
  yarn biome # Check code with Biome
99
- yarn check:write # Check and auto-fix with Biome
104
+ yarn biome:write # Check and auto-fix with Biome
100
105
 
101
106
  # Testing
102
107
  yarn test # Run Jest tests
108
+
109
+ # Clean
110
+ yarn clean # Remove build artifacts
103
111
  ```
104
112
 
105
113
  ## Logging System
@@ -124,12 +132,13 @@ modelLogger.error('Model API error');
124
132
 
125
133
  ### Available Loggers
126
134
  - `agent`: Main agent operations
127
- - `intent`: Query processing and intent analysis
135
+ - `intent`: Non-streaming query processing and intent analysis
136
+ - `intentStream`: Streaming query processing
128
137
  - `mcp`: MCP server connections and tool execution
129
138
  - `a2a`: Agent-to-agent communication
130
139
  - `model`: AI model interactions
131
140
  - `server`: HTTP server operations
132
- - `memory`: Threads, Intents, Agent Data management
141
+ - `memory`: Threads, intents, and agent data management
133
142
 
134
143
  ### Log Levels
135
144
  - `error`: Error conditions
@@ -140,22 +149,35 @@ modelLogger.error('Model API error');
140
149
  ## API Endpoints
141
150
 
142
151
  ### Standard Endpoints
143
- - `GET /` - Welcome message
144
- - `POST /query` - Process queries
145
- - Request: `{ message: string, threadId: string }`
146
- - Response: `{ content: string }`
147
- - `POST /query/stream` - Process queries with streaming
148
- - Request: `{ message: string, threadId: string }`
152
+ - `GET /` - Welcome message and health check
153
+ - `POST /query` - Process queries (non-streaming)
154
+ - Request: `{ message: string, threadId?: string, type?: string }`
149
155
  - Response: `{ content: string }`
156
+ - `POST /query/stream` - Process queries with streaming (SSE)
157
+ - Request: `{ message: string, threadId?: string, type?: string }`
158
+ - Response: Server-Sent Events stream with event types:
159
+ - `text_chunk`: Incremental text response
160
+ - `tool_start`: Tool execution started
161
+ - `tool_output`: Tool execution result
162
+ - `thread_id`: Thread metadata
163
+ - `error`: Error message
150
164
 
151
165
  ### Agent Management
166
+ - `GET /api/threads/:userId` - List user threads
167
+ - `POST /api/threads/:userId` - Create new thread
168
+ - `GET /api/threads/:userId/:threadId` - Get thread details
169
+ - `DELETE /api/threads/:userId/:threadId` - Delete thread
170
+ - `GET /api/model` - Get model configuration
171
+ - `POST /api/model` - Set default model
152
172
 
153
173
  ### A2A Server Endpoints (when `manifest.url` is configured)
154
- - `GET /agent-card` - Get agent card information
155
- - `GET /.well-known/agent.json` - Agent discovery endpoint
174
+ - `GET /.well-known/agent.json` - Agent discovery endpoint (A2A ~v0.2.0)
175
+ - `GET /.well-known/agent-card.json` - Agent discovery endpoint (A2A v0.3.0~)
176
+ - Returns `AgentCard` with capabilities and supported modes
156
177
  - `POST /a2a` - A2A communication endpoint
157
- - Supports streaming responses
158
- - Request: `{ message: string, stream?: boolean }`
178
+ - Supports streaming responses via SSE
179
+ - Request: `{ message: string, threadId?: string, stream?: boolean }`
180
+ - Response: JSON or SSE stream based on `stream` parameter
159
181
 
160
182
  ## Build System
161
183
 
@@ -168,13 +190,15 @@ The project supports dual build output:
168
190
 
169
191
  The library includes comprehensive error handling:
170
192
  - Global error middleware for uncaught errors
171
- - Service-specific error logging
193
+ - Custom `AinHttpError` class for HTTP-specific errors
194
+ - Service-specific error logging with structured context
172
195
  - Graceful handling of:
173
196
  - MCP server connection failures
174
197
  - A2A agent communication errors
175
- - Model API failures
198
+ - Model API failures (rate limits, timeouts)
176
199
  - Tool execution errors
177
200
  - Invalid requests
201
+ - Streaming errors with proper cleanup
178
202
 
179
203
  All errors are logged with appropriate context for debugging.
180
204
 
@@ -197,11 +221,13 @@ const agent = new AINAgent(manifest, modules, new MyAuth());
197
221
 
198
222
  ## Contributing
199
223
 
200
- 1. Follow the established code conventions
224
+ 1. Follow the established code conventions (tabs, double quotes)
201
225
  2. Use TypeScript strict mode
202
226
  3. Add appropriate service-specific logging
203
227
  4. Run `yarn biome:write` and `yarn test` before submitting
204
228
  5. Maintain the modular architecture
229
+ 6. Update JSDoc comments when changing function signatures
230
+ 7. Add streaming support when implementing new query handlers
205
231
 
206
232
  ## License
207
233
 
@@ -1,9 +1,9 @@
1
1
  "use strict";Object.defineProperty(exports, "__esModule", {value: true});
2
2
 
3
- var _chunkMIWLK57Kcjs = require('./chunk-MIWLK57K.cjs');
3
+ var _chunkA2I7D6JWcjs = require('./chunk-A2I7D6JW.cjs');
4
4
 
5
5
 
6
- var _chunkDX2JH3FJcjs = require('./chunk-DX2JH3FJ.cjs');
6
+ var _chunkQHQLU7BIcjs = require('./chunk-QHQLU7BI.cjs');
7
7
 
8
8
  // src/routes/api/threads.routes.ts
9
9
 
@@ -12,11 +12,11 @@ var _express = require('express');
12
12
  var _httpstatuscodes = require('http-status-codes');
13
13
  var createThreadApiRouter = (memoryModule) => {
14
14
  const router = _express.Router.call(void 0, );
15
- const threadApiController = new (0, _chunkMIWLK57Kcjs.ThreadApiController)(memoryModule);
15
+ const threadApiController = new (0, _chunkA2I7D6JWcjs.ThreadApiController)(memoryModule);
16
16
  const checkThreadMemory = (req, res, next) => {
17
17
  const threadMemory = memoryModule.getThreadMemory();
18
18
  if (!threadMemory) {
19
- const error = new (0, _chunkDX2JH3FJcjs.AinHttpError)(
19
+ const error = new (0, _chunkQHQLU7BIcjs.AinHttpError)(
20
20
  _httpstatuscodes.StatusCodes.SERVICE_UNAVAILABLE,
21
21
  "Memory module is not initialized"
22
22
  );
@@ -37,4 +37,4 @@ var createThreadApiRouter = (memoryModule) => {
37
37
 
38
38
 
39
39
  exports.createThreadApiRouter = createThreadApiRouter;
40
- //# sourceMappingURL=chunk-GB3ON7UV.cjs.map
40
+ //# sourceMappingURL=chunk-7PZP7YSA.cjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-GB3ON7UV.cjs","../../src/routes/api/threads.routes.ts"],"names":[],"mappings":"AAAA;AACE;AACF,wDAA6B;AAC7B;AACE;AACF,wDAA6B;AAC7B;AACA;ACPA;AAIC;AAAA,kCACM;AACP,oDAA4B;AAKrB,IAAM,sBAAA,EAAwB,CAAC,YAAA,EAAA,GAAuC;AAC5E,EAAA,MAAM,OAAA,EAAS,6BAAA,CAAO;AACtB,EAAA,MAAM,oBAAA,EAAsB,IAAI,0CAAA,CAAoB,YAAY,CAAA;AAEhE,EAAA,MAAM,kBAAA,EAAoB,CACzB,GAAA,EACA,GAAA,EACA,IAAA,EAAA,GACI;AACJ,IAAA,MAAM,aAAA,EAAe,YAAA,CAAa,eAAA,CAAgB,CAAA;AAClD,IAAA,GAAA,CAAI,CAAC,YAAA,EAAc;AAClB,MAAA,MAAM,MAAA,EAAQ,IAAI,mCAAA;AAAA,QACjB,4BAAA,CAAY,mBAAA;AAAA,QACZ;AAAA,MACD,CAAA;AACA,MAAA,MAAM,KAAA;AAAA,IACP;AACA,IAAA,IAAA,CAAK,CAAA;AAAA,EACN,CAAA;AAGA,EAAA,MAAA,CAAO,GAAA,CAAI,GAAA,EAAK,iBAAA,EAAmB,mBAAA,CAAoB,oBAAoB,CAAA;AAC3E,EAAA,MAAA,CAAO,GAAA,CAAI,MAAA,EAAQ,iBAAA,EAAmB,mBAAA,CAAoB,eAAe,CAAA;AACzE,EAAA,MAAA,CAAO,MAAA;AAAA,IACN,MAAA;AAAA,IACA,iBAAA;AAAA,IACA,mBAAA,CAAoB;AAAA,EACrB,CAAA;AAEA,EAAA,OAAO,MAAA;AACR,CAAA;ADNA;AACA;AACE;AACF,sDAAC","file":"/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-GB3ON7UV.cjs","sourcesContent":[null,"import {\n\ttype NextFunction,\n\ttype Request,\n\ttype Response,\n\tRouter,\n} from \"express\";\nimport { StatusCodes } from \"http-status-codes\";\nimport { ThreadApiController } from \"@/controllers/api/threads.api.controller.js\";\nimport type { MemoryModule } from \"@/modules/index.js\";\nimport { AinHttpError } from \"@/types/agent\";\n\nexport const createThreadApiRouter = (memoryModule: MemoryModule): Router => {\n\tconst router = Router();\n\tconst threadApiController = new ThreadApiController(memoryModule);\n\n\tconst checkThreadMemory = (\n\t\treq: Request,\n\t\tres: Response,\n\t\tnext: NextFunction,\n\t) => {\n\t\tconst threadMemory = memoryModule.getThreadMemory();\n\t\tif (!threadMemory) {\n\t\t\tconst error = new AinHttpError(\n\t\t\t\tStatusCodes.SERVICE_UNAVAILABLE,\n\t\t\t\t\"Memory module is not initialized\",\n\t\t\t);\n\t\t\tthrow error;\n\t\t}\n\t\tnext();\n\t};\n\n\t// APIs (prefix: /api/threads)\n\trouter.get(\"/\", checkThreadMemory, threadApiController.handleGetUserThreads);\n\trouter.get(\"/:id\", checkThreadMemory, threadApiController.handleGetThread);\n\trouter.delete(\n\t\t\"/:id\",\n\t\tcheckThreadMemory,\n\t\tthreadApiController.handleDeleteThread,\n\t);\n\n\treturn router;\n};\n"]}
1
+ {"version":3,"sources":["/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-7PZP7YSA.cjs","../../src/routes/api/threads.routes.ts"],"names":[],"mappings":"AAAA;AACE;AACF,wDAA6B;AAC7B;AACE;AACF,wDAA6B;AAC7B;AACA;ACPA;AAIC;AAAA,kCACM;AACP,oDAA4B;AAKrB,IAAM,sBAAA,EAAwB,CAAC,YAAA,EAAA,GAAuC;AAC5E,EAAA,MAAM,OAAA,EAAS,6BAAA,CAAO;AACtB,EAAA,MAAM,oBAAA,EAAsB,IAAI,0CAAA,CAAoB,YAAY,CAAA;AAEhE,EAAA,MAAM,kBAAA,EAAoB,CACzB,GAAA,EACA,GAAA,EACA,IAAA,EAAA,GACI;AACJ,IAAA,MAAM,aAAA,EAAe,YAAA,CAAa,eAAA,CAAgB,CAAA;AAClD,IAAA,GAAA,CAAI,CAAC,YAAA,EAAc;AAClB,MAAA,MAAM,MAAA,EAAQ,IAAI,mCAAA;AAAA,QACjB,4BAAA,CAAY,mBAAA;AAAA,QACZ;AAAA,MACD,CAAA;AACA,MAAA,MAAM,KAAA;AAAA,IACP;AACA,IAAA,IAAA,CAAK,CAAA;AAAA,EACN,CAAA;AAGA,EAAA,MAAA,CAAO,GAAA,CAAI,GAAA,EAAK,iBAAA,EAAmB,mBAAA,CAAoB,oBAAoB,CAAA;AAC3E,EAAA,MAAA,CAAO,GAAA,CAAI,MAAA,EAAQ,iBAAA,EAAmB,mBAAA,CAAoB,eAAe,CAAA;AACzE,EAAA,MAAA,CAAO,MAAA;AAAA,IACN,MAAA;AAAA,IACA,iBAAA;AAAA,IACA,mBAAA,CAAoB;AAAA,EACrB,CAAA;AAEA,EAAA,OAAO,MAAA;AACR,CAAA;ADNA;AACA;AACE;AACF,sDAAC","file":"/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-7PZP7YSA.cjs","sourcesContent":[null,"import {\n\ttype NextFunction,\n\ttype Request,\n\ttype Response,\n\tRouter,\n} from \"express\";\nimport { StatusCodes } from \"http-status-codes\";\nimport { ThreadApiController } from \"@/controllers/api/threads.api.controller.js\";\nimport type { MemoryModule } from \"@/modules/index.js\";\nimport { AinHttpError } from \"@/types/agent\";\n\nexport const createThreadApiRouter = (memoryModule: MemoryModule): Router => {\n\tconst router = Router();\n\tconst threadApiController = new ThreadApiController(memoryModule);\n\n\tconst checkThreadMemory = (\n\t\treq: Request,\n\t\tres: Response,\n\t\tnext: NextFunction,\n\t) => {\n\t\tconst threadMemory = memoryModule.getThreadMemory();\n\t\tif (!threadMemory) {\n\t\t\tconst error = new AinHttpError(\n\t\t\t\tStatusCodes.SERVICE_UNAVAILABLE,\n\t\t\t\t\"Memory module is not initialized\",\n\t\t\t);\n\t\t\tthrow error;\n\t\t}\n\t\tnext();\n\t};\n\n\t// APIs (prefix: /api/threads)\n\trouter.get(\"/\", checkThreadMemory, threadApiController.handleGetUserThreads);\n\trouter.get(\"/:id\", checkThreadMemory, threadApiController.handleGetThread);\n\trouter.delete(\n\t\t\"/:id\",\n\t\tcheckThreadMemory,\n\t\tthreadApiController.handleDeleteThread,\n\t);\n\n\treturn router;\n};\n"]}
@@ -42,4 +42,4 @@ var ThreadApiController = (_class = class {
42
42
 
43
43
 
44
44
  exports.ThreadApiController = ThreadApiController;
45
- //# sourceMappingURL=chunk-MIWLK57K.cjs.map
45
+ //# sourceMappingURL=chunk-A2I7D6JW.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-A2I7D6JW.cjs","../../src/controllers/api/threads.api.controller.ts"],"names":[],"mappings":"AAAA;ACCA,oDAA4B;AAGrB,IAAM,oBAAA,YAAN,MAA0B;AAAA,EACxB;AAAA,EAER,WAAA,CAAY,YAAA,EAA4B;AACvC,IAAA,IAAA,CAAK,aAAA,EAAe,YAAA;AAAA,EACrB;AAAA,iBAEO,gBAAA,EAAkB,MAAA,CACxB,GAAA,EACA,GAAA,EACA,IAAA,EAAA,GACI;AACJ,IAAA,IAAI;AACH,MAAA,MAAM,EAAE,EAAA,EAAI,SAAS,EAAA,EAAI,GAAA,CAAI,MAAA;AAG7B,MAAA,MAAM,OAAA,EAAS,GAAA,CAAI,MAAA,CAAO,OAAA,GAAU,EAAA;AACpC,MAAA,MAAM,aAAA,EAAe,IAAA,CAAK,YAAA,CAAa,eAAA,CAAgB,CAAA;AACvD,MAAA,MAAM,OAAA,EAAS,sBAAM,YAAA,6BAAc,SAAA,mBAAU,MAAA,EAAQ,QAAQ,GAAA;AAC7D,MAAA,GAAA,CAAI,IAAA,CAAK,MAAM,CAAA;AAAA,IAChB,EAAA,MAAA,CAAS,KAAA,EAAO;AACf,MAAA,IAAA,CAAK,KAAK,CAAA;AAAA,IACX;AAAA,EACD,EAAA;AAAA,kBAEO,mBAAA,EAAqB,MAAA,CAC3B,GAAA,EACA,GAAA,EACA,IAAA,EAAA,GACI;AACJ,IAAA,IAAI;AACH,MAAA,MAAM,EAAE,EAAA,EAAI,SAAS,EAAA,EAAI,GAAA,CAAI,MAAA;AAC7B,MAAA,MAAM,OAAA,EAAS,GAAA,CAAI,MAAA,CAAO,OAAA,GAAU,EAAA;AACpC,MAAA,MAAM,aAAA,EAAe,IAAA,CAAK,YAAA,CAAa,eAAA,CAAgB,CAAA;AACvD,MAAA,sBAAM,YAAA,6BAAc,YAAA,qBAAa,MAAA,EAAQ,QAAQ,GAAA;AACjD,MAAA,GAAA,CAAI,MAAA,CAAO,4BAAA,CAAY,EAAE,CAAA,CAAE,IAAA,CAAK,CAAA;AAAA,IACjC,EAAA,MAAA,CAAS,KAAA,EAAO;AACf,MAAA,IAAA,CAAK,KAAK,CAAA;AAAA,IACX;AAAA,EACD,EAAA;AAAA,kBAEO,qBAAA,EAAuB,MAAA,CAC7B,IAAA,EACA,GAAA,EACA,IAAA,EAAA,GACI;AACJ,IAAA,IAAI;AACH,MAAA,MAAM,OAAA,EAAS,GAAA,CAAI,MAAA,CAAO,OAAA,GAAU,EAAA;AACpC,MAAA,MAAM,aAAA,EAAe,IAAA,CAAK,YAAA,CAAa,eAAA,CAAgB,CAAA;AACvD,MAAA,MAAM,QAAA,EAAU,sBAAM,YAAA,+BAAc,WAAA,qBAAY,MAAM,GAAA;AACtD,MAAA,GAAA,CAAI,IAAA,CAAK,OAAO,CAAA;AAAA,IACjB,EAAA,MAAA,CAAS,KAAA,EAAO;AACf,MAAA,IAAA,CAAK,KAAK,CAAA;AAAA,IACX;AAAA,EACD,EAAA;AACD,UAAA;ADnBA;AACA;AACE;AACF,kDAAC","file":"/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-A2I7D6JW.cjs","sourcesContent":[null,"import type { NextFunction, Request, Response } from \"express\";\nimport { StatusCodes } from \"http-status-codes\";\nimport type { MemoryModule } from \"@/modules/index.js\";\n\nexport class ThreadApiController {\n\tprivate memoryModule: MemoryModule;\n\n\tconstructor(memoryModule: MemoryModule) {\n\t\tthis.memoryModule = memoryModule;\n\t}\n\n\tpublic handleGetThread = async (\n\t\treq: Request,\n\t\tres: Response,\n\t\tnext: NextFunction,\n\t) => {\n\t\ttry {\n\t\t\tconst { id: threadId } = req.params as {\n\t\t\t\tid: string;\n\t\t\t};\n\t\t\tconst userId = res.locals.userId || \"\";\n\t\t\tconst threadMemory = this.memoryModule.getThreadMemory();\n\t\t\tconst thread = await threadMemory?.getThread(userId, threadId);\n\t\t\tres.json(thread);\n\t\t} catch (error) {\n\t\t\tnext(error);\n\t\t}\n\t};\n\n\tpublic handleDeleteThread = async (\n\t\treq: Request,\n\t\tres: Response,\n\t\tnext: NextFunction,\n\t) => {\n\t\ttry {\n\t\t\tconst { id: threadId } = req.params;\n\t\t\tconst userId = res.locals.userId || \"\";\n\t\t\tconst threadMemory = this.memoryModule.getThreadMemory();\n\t\t\tawait threadMemory?.deleteThread(userId, threadId);\n\t\t\tres.status(StatusCodes.OK).send();\n\t\t} catch (error) {\n\t\t\tnext(error);\n\t\t}\n\t};\n\n\tpublic handleGetUserThreads = async (\n\t\t_req: Request,\n\t\tres: Response,\n\t\tnext: NextFunction,\n\t) => {\n\t\ttry {\n\t\t\tconst userId = res.locals.userId || \"\";\n\t\t\tconst threadMemory = this.memoryModule.getThreadMemory();\n\t\t\tconst threads = await threadMemory?.listThreads(userId);\n\t\t\tres.json(threads);\n\t\t} catch (error) {\n\t\t\tnext(error);\n\t\t}\n\t};\n}\n"]}
@@ -1,4 +1,4 @@
1
- "use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } var _class;
1
+ "use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } var _class;
2
2
 
3
3
  var _chunkF7I4G6JGcjs = require('./chunk-F7I4G6JG.cjs');
4
4
 
@@ -8,11 +8,11 @@ var _chunk466ED5WNcjs = require('./chunk-466ED5WN.cjs');
8
8
  // src/modules/a2a/a2a.module.ts
9
9
  var _crypto = require('crypto');
10
10
  var _client = require('@a2a-js/sdk/client');
11
- var A2AModule = (_class = class {constructor() { _class.prototype.__init.call(this);_class.prototype.__init2.call(this);_class.prototype.__init3.call(this);_class.prototype.__init4.call(this); }
11
+ var A2AModule = (_class = class {constructor() { _class.prototype.__init.call(this);_class.prototype.__init2.call(this);_class.prototype.__init3.call(this); }
12
12
  /** Map of A2A server URLs to their corresponding tool instances */
13
13
  __init() {this.a2aPeerServers = /* @__PURE__ */ new Map()}
14
14
  /** Map of session IDs to their A2A session state */
15
- __init2() {this.a2aSessions = /* @__PURE__ */ new Map()}
15
+ __init2() {this.a2aTasks = /* @__PURE__ */ new Map()}
16
16
  __init3() {this.agentId = _crypto.randomUUID.call(void 0, )}
17
17
  /* FIXME */
18
18
  /**
@@ -53,22 +53,6 @@ var A2AModule = (_class = class {constructor() { _class.prototype.__init.call(th
53
53
  }
54
54
  return tools;
55
55
  }
56
- /**
57
- * Gets or creates an A2A session for the given session ID.
58
- *
59
- * @param threadId - The session identifier
60
- * @returns A2ASession object with task and context IDs
61
- */
62
- __init4() {this.getA2ASessionWithId = (threadId) => {
63
- const a2aSession = _nullishCoalesce(this.a2aSessions.get(threadId), () => ( {
64
- taskId: void 0,
65
- contextId: void 0
66
- }));
67
- if (!this.a2aSessions.has(threadId)) {
68
- this.a2aSessions.set(threadId, a2aSession);
69
- }
70
- return a2aSession;
71
- }}
72
56
  /**
73
57
  * Constructs a message payload for A2A communication.
74
58
  *
@@ -87,22 +71,18 @@ var A2AModule = (_class = class {constructor() { _class.prototype.__init.call(th
87
71
  // FIXME: it could be 'agent'
88
72
  metadata: {
89
73
  agentId: this.agentId,
90
- type: "CHAT" /* CHAT */,
91
- threadId
74
+ type: "CHAT" /* CHAT */
92
75
  },
93
76
  parts: [
94
77
  {
95
78
  kind: "text",
96
79
  text: query
97
80
  }
98
- ]
81
+ ],
82
+ contextId: threadId
99
83
  };
100
- const a2aSession = this.getA2ASessionWithId(threadId);
101
- if (a2aSession.taskId) {
102
- messagePayload.taskId = a2aSession.taskId;
103
- }
104
- if (a2aSession.contextId) {
105
- messagePayload.contextId = a2aSession.contextId;
84
+ if (this.a2aTasks.has(threadId)) {
85
+ messagePayload.taskId = this.a2aTasks.get(threadId);
106
86
  }
107
87
  return messagePayload;
108
88
  }
@@ -117,20 +97,20 @@ var A2AModule = (_class = class {constructor() { _class.prototype.__init.call(th
117
97
  * @param threadId - The session identifier for context tracking
118
98
  * @returns Promise resolving to array of text responses from the agent
119
99
  */
120
- async useTool(tool, messagePayload, threadId) {
100
+ async useTool(tool, query, threadId) {
121
101
  const finalText = [];
122
102
  const client = tool.client;
103
+ const messagePayload = this.getMessagePayload(query, threadId);
123
104
  const params = {
124
105
  message: messagePayload
125
106
  };
126
- const a2aSession = this.getA2ASessionWithId(threadId);
127
107
  try {
128
108
  const stream = client.sendMessageStream(params);
129
109
  for await (const event of stream) {
130
110
  if (event.kind === "status-update") {
131
111
  const typedEvent = event;
132
112
  if (typedEvent.final && typedEvent.status.state !== "input-required") {
133
- a2aSession.taskId = void 0;
113
+ this.a2aTasks.delete(threadId);
134
114
  }
135
115
  const texts = _optionalChain([typedEvent, 'access', _ => _.status, 'access', _2 => _2.message, 'optionalAccess', _3 => _3.parts, 'access', _4 => _4.filter, 'call', _5 => _5((part) => part.kind === "text"), 'access', _6 => _6.map, 'call', _7 => _7((part) => part.text), 'access', _8 => _8.join, 'call', _9 => _9("\n")]);
136
116
  if (texts) {
@@ -138,19 +118,14 @@ var A2AModule = (_class = class {constructor() { _class.prototype.__init.call(th
138
118
  }
139
119
  } else if (event.kind === "message") {
140
120
  const msg = event;
141
- if (msg.taskId && msg.taskId !== a2aSession.taskId) {
142
- a2aSession.taskId = msg.taskId;
143
- }
144
- if (msg.contextId && msg.contextId !== a2aSession.contextId) {
145
- a2aSession.contextId = msg.contextId;
121
+ const taskId = this.a2aTasks.get(threadId);
122
+ if (msg.taskId && msg.taskId !== taskId) {
123
+ this.a2aTasks.set(threadId, msg.taskId);
146
124
  }
147
125
  } else if (event.kind === "task") {
148
126
  const task = event;
149
- if (task.id !== a2aSession.taskId) {
150
- a2aSession.taskId = task.id;
151
- }
152
- if (task.contextId && task.contextId !== a2aSession.contextId) {
153
- a2aSession.contextId = task.contextId;
127
+ if (task.id !== this.a2aTasks.get(threadId)) {
128
+ this.a2aTasks.set(threadId, task.id);
154
129
  }
155
130
  } else {
156
131
  _chunk466ED5WNcjs.loggers.a2a.warn("Received unknown event structure from stream:", {
@@ -173,4 +148,4 @@ ${finalText.join("\n")}`;
173
148
 
174
149
 
175
150
  exports.A2AModule = A2AModule;
176
- //# sourceMappingURL=chunk-6YVTJEMQ.cjs.map
151
+ //# sourceMappingURL=chunk-BBFYVWE2.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-BBFYVWE2.cjs","../../src/modules/a2a/a2a.module.ts"],"names":[],"mappings":"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","file":"/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-BBFYVWE2.cjs","sourcesContent":[null,"import { randomUUID } from \"node:crypto\";\nimport type {\n\tAgentCard,\n\tMessage,\n\tMessageSendParams,\n\tTask,\n\tTaskStatusUpdateEvent,\n\tTextPart,\n} from \"@a2a-js/sdk\";\nimport { A2AClient } from \"@a2a-js/sdk/client\";\nimport { ThreadType } from \"@/types/memory.js\";\nimport { loggers } from \"@/utils/logger.js\";\nimport { A2ATool } from \"./a2a.tool.js\";\n\n/**\n * Module for managing Agent-to-Agent (A2A) protocol connections.\n *\n * This module handles connections to other A2A-compatible agents, manages\n * conversation sessions, and provides an interface for inter-agent communication.\n * Supports multi-turn conversations with task and context tracking.\n */\nexport class A2AModule {\n\t/** Map of A2A server URLs to their corresponding tool instances */\n\tprivate a2aPeerServers: Map<string, A2ATool | null> = new Map();\n\t/** Map of session IDs to their A2A session state */\n\tprivate a2aTasks: Map<string, string> = new Map();\n\tprivate agentId: string = randomUUID(); /* FIXME */\n\n\t/**\n\t * Registers a new A2A peer server URL for connection.\n\t *\n\t * @param url - The URL of the A2A-compatible agent to connect to\n\t */\n\tpublic async addA2APeerServer(url: string): Promise<void> {\n\t\tthis.a2aPeerServers.set(url, null);\n\t}\n\n\t/**\n\t * Retrieves tools from all registered A2A peer servers.\n\t *\n\t * Attempts to connect to each registered server, fetch their agent cards,\n\t * and create tool instances. Disables tools for unreachable servers.\n\t *\n\t * @returns Promise resolving to array of available A2A tools\n\t */\n\tpublic async getTools(): Promise<A2ATool[]> {\n\t\tconst tools: A2ATool[] = [];\n\t\tfor (const url of [...this.a2aPeerServers.keys()]) {\n\t\t\tconst tool = this.a2aPeerServers.get(url);\n\t\t\tif (!tool || !tool.enabled) {\n\t\t\t\ttry {\n\t\t\t\t\tconst client = new A2AClient(url);\n\t\t\t\t\tconst card: AgentCard = await client.getAgentCard();\n\t\t\t\t\tconst toolName = card.name.replaceAll(\" \", \"-\");\n\t\t\t\t\tconst a2aTool = new A2ATool(toolName, client, card);\n\n\t\t\t\t\ttools.push(a2aTool);\n\t\t\t\t} catch (_error: any) {\n\t\t\t\t\t// Agent not responded\n\t\t\t\t\tif (tool) {\n\t\t\t\t\t\ttool.disable();\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\ttools.push(tool);\n\t\t\t}\n\t\t}\n\t\treturn tools;\n\t}\n\n\t/**\n\t * Constructs a message payload for A2A communication.\n\t *\n\t * Includes session context (task ID and context ID) if available\n\t * for maintaining conversation continuity.\n\t *\n\t * @param query - The message content to send\n\t * @param threadId - The session identifier\n\t * @returns Formatted Message object for A2A protocol\n\t */\n\tpublic getMessagePayload(query: string, threadId: string): Message {\n\t\tconst messagePayload: Message = {\n\t\t\tmessageId: randomUUID(),\n\t\t\tkind: \"message\",\n\t\t\trole: \"user\", // FIXME: it could be 'agent'\n\t\t\tmetadata: {\n\t\t\t\tagentId: this.agentId,\n\t\t\t\ttype: ThreadType.CHAT,\n\t\t\t},\n\t\t\tparts: [\n\t\t\t\t{\n\t\t\t\t\tkind: \"text\",\n\t\t\t\t\ttext: query,\n\t\t\t\t},\n\t\t\t],\n\t\t\tcontextId: threadId,\n\t\t};\n\n\t\tif (this.a2aTasks.has(threadId)) {\n\t\t\tmessagePayload.taskId = this.a2aTasks.get(threadId);\n\t\t}\n\n\t\treturn messagePayload;\n\t}\n\n\t/**\n\t * Executes an A2A tool by sending a message to the remote agent.\n\t *\n\t * Handles streaming responses, maintains session state, and extracts\n\t * text content from various event types in the response stream.\n\t *\n\t * @param tool - The A2ATool instance to use\n\t * @param messagePayload - The message to send to the agent\n\t * @param threadId - The session identifier for context tracking\n\t * @returns Promise resolving to array of text responses from the agent\n\t */\n\tpublic async useTool(\n\t\ttool: A2ATool,\n\t\tquery: string,\n\t\tthreadId: string,\n\t): Promise<string> {\n\t\tconst finalText: string[] = [];\n\t\tconst client = tool.client;\n\t\tconst messagePayload = this.getMessagePayload(query, threadId);\n\t\tconst params: MessageSendParams = {\n\t\t\tmessage: messagePayload,\n\t\t};\n\n\t\ttry {\n\t\t\tconst stream = client.sendMessageStream(params);\n\t\t\tfor await (const event of stream) {\n\t\t\t\tif (event.kind === \"status-update\") {\n\t\t\t\t\tconst typedEvent = event as TaskStatusUpdateEvent;\n\t\t\t\t\tif (\n\t\t\t\t\t\ttypedEvent.final &&\n\t\t\t\t\t\ttypedEvent.status.state !== \"input-required\"\n\t\t\t\t\t) {\n\t\t\t\t\t\tthis.a2aTasks.delete(threadId);\n\t\t\t\t\t}\n\t\t\t\t\t// TODO: handle 'file', 'data' parts\n\t\t\t\t\tconst texts = typedEvent.status.message?.parts\n\t\t\t\t\t\t.filter((part) => part.kind === \"text\")\n\t\t\t\t\t\t.map((part: TextPart) => part.text)\n\t\t\t\t\t\t.join(\"\\n\");\n\t\t\t\t\tif (texts) {\n\t\t\t\t\t\tfinalText.push(texts);\n\t\t\t\t\t}\n\t\t\t\t} else if (event.kind === \"message\") {\n\t\t\t\t\t// FIXME: handling text in 'message'?\n\t\t\t\t\tconst msg = event as Message;\n\t\t\t\t\tconst taskId = this.a2aTasks.get(threadId);\n\t\t\t\t\tif (msg.taskId && msg.taskId !== taskId) {\n\t\t\t\t\t\tthis.a2aTasks.set(threadId, msg.taskId);\n\t\t\t\t\t}\n\t\t\t\t} else if (event.kind === \"task\") {\n\t\t\t\t\t// establishing the Task ID\n\t\t\t\t\tconst task = event as Task;\n\t\t\t\t\tif (task.id !== this.a2aTasks.get(threadId)) {\n\t\t\t\t\t\tthis.a2aTasks.set(threadId, task.id);\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\tloggers.a2a.warn(\"Received unknown event structure from stream:\", {\n\t\t\t\t\t\tevent,\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tloggers.a2a.error(\"Error communicating with agent:\", { error });\n\t\t\ttool.disable();\n\t\t\tconst toolResult = `[Bot Called A2A Tool ${tool.card.name}]\\n${typeof error === \"string\" ? error : JSON.stringify(error, null, 2)}`;\n\t\t\treturn toolResult;\n\t\t}\n\n\t\treturn `[Bot Called A2A Tool ${tool.card.name}]\\n${finalText.join(\"\\n\")}`;\n\t}\n}\n"]}
@@ -1,6 +1,6 @@
1
1
  "use strict";Object.defineProperty(exports, "__esModule", {value: true});
2
2
 
3
- var _chunkDX2JH3FJcjs = require('./chunk-DX2JH3FJ.cjs');
3
+ var _chunkQHQLU7BIcjs = require('./chunk-QHQLU7BI.cjs');
4
4
 
5
5
  // src/middlewares/auth.middleware.ts
6
6
  var _httpstatuscodes = require('http-status-codes');
@@ -17,7 +17,7 @@ var AuthMiddleware = class {
17
17
  res.locals.userId = authRes.userId;
18
18
  next();
19
19
  } else {
20
- const error = new (0, _chunkDX2JH3FJcjs.AinHttpError)(
20
+ const error = new (0, _chunkQHQLU7BIcjs.AinHttpError)(
21
21
  _httpstatuscodes.StatusCodes.UNAUTHORIZED,
22
22
  "Unauthorized"
23
23
  );
@@ -25,7 +25,7 @@ var AuthMiddleware = class {
25
25
  }
26
26
  } catch (e) {
27
27
  if (!e.status) {
28
- const error = new (0, _chunkDX2JH3FJcjs.AinHttpError)(
28
+ const error = new (0, _chunkQHQLU7BIcjs.AinHttpError)(
29
29
  _httpstatuscodes.StatusCodes.INTERNAL_SERVER_ERROR,
30
30
  `Authentication error: ${JSON.stringify(e)}`
31
31
  );
@@ -40,4 +40,4 @@ var AuthMiddleware = class {
40
40
 
41
41
 
42
42
  exports.AuthMiddleware = AuthMiddleware;
43
- //# sourceMappingURL=chunk-LRBRG2DS.cjs.map
43
+ //# sourceMappingURL=chunk-E35E4LQY.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-E35E4LQY.cjs","../../src/middlewares/auth.middleware.ts"],"names":[],"mappings":"AAAA;AACE;AACF,wDAA6B;AAC7B;AACA;ACHA,oDAA4B;AAKrB,IAAM,eAAA,EAAN,MAAqB;AAAA,EACnB;AAAA,EACR,WAAA,CAAY,IAAA,EAAgB;AAC3B,IAAA,IAAA,CAAK,KAAA,EAAO,IAAA;AAAA,EACb;AAAA,EAEO,UAAA,CAAA,EAA6B;AACnC,IAAA,OAAO,MAAA,CAAO,GAAA,EAAc,GAAA,EAAe,IAAA,EAAA,GAAuB;AACjE,MAAA,IAAI;AACH,QAAA,MAAM,QAAA,EAAwB,MAAM,IAAA,CAAK,IAAA,CAAK,YAAA,CAAa,GAAA,EAAK,GAAG,CAAA;AACnE,QAAA,GAAA,CAAI,OAAA,CAAQ,eAAA,EAAiB;AAC5B,UAAA,GAAA,CAAI,MAAA,CAAO,OAAA,EAAS,OAAA,CAAQ,MAAA;AAC5B,UAAA,IAAA,CAAK,CAAA;AAAA,QACN,EAAA,KAAO;AACN,UAAA,MAAM,MAAA,EAAsB,IAAI,mCAAA;AAAA,YAC/B,4BAAA,CAAY,YAAA;AAAA,YACZ;AAAA,UACD,CAAA;AACA,UAAA,MAAM,KAAA;AAAA,QACP;AAAA,MACD,EAAA,MAAA,CAAS,CAAA,EAAQ;AAChB,QAAA,GAAA,CAAI,CAAC,CAAA,CAAE,MAAA,EAAQ;AACd,UAAA,MAAM,MAAA,EAAsB,IAAI,mCAAA;AAAA,YAC/B,4BAAA,CAAY,qBAAA;AAAA,YACZ,CAAA,sBAAA,EAAyB,IAAA,CAAK,SAAA,CAAU,CAAC,CAAC,CAAA;AAAA,UAAA;AAE3C,UAAA;AAAM,QAAA;AAEP,QAAA;AAAM,MAAA;AACP,IAAA;AACD,EAAA;AAEF;ADAA;AACA;AACA;AACA","file":"/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-E35E4LQY.cjs","sourcesContent":[null,"import type { NextFunction, Request, RequestHandler, Response } from \"express\";\nimport { StatusCodes } from \"http-status-codes\";\nimport type { BaseAuth } from \"@/modules/auth/base.auth\";\nimport { AinHttpError } from \"@/types/agent\";\nimport type { AuthResponse } from \"@/types/auth\";\n\nexport class AuthMiddleware {\n\tprivate auth: BaseAuth;\n\tconstructor(auth: BaseAuth) {\n\t\tthis.auth = auth;\n\t}\n\n\tpublic middleware(): RequestHandler {\n\t\treturn async (req: Request, res: Response, next: NextFunction) => {\n\t\t\ttry {\n\t\t\t\tconst authRes: AuthResponse = await this.auth.authenticate(req, res);\n\t\t\t\tif (authRes.isAuthenticated) {\n\t\t\t\t\tres.locals.userId = authRes.userId;\n\t\t\t\t\tnext();\n\t\t\t\t} else {\n\t\t\t\t\tconst error: AinHttpError = new AinHttpError(\n\t\t\t\t\t\tStatusCodes.UNAUTHORIZED,\n\t\t\t\t\t\t\"Unauthorized\",\n\t\t\t\t\t);\n\t\t\t\t\tthrow error;\n\t\t\t\t}\n\t\t\t} catch (e: any) {\n\t\t\t\tif (!e.status) {\n\t\t\t\t\tconst error: AinHttpError = new AinHttpError(\n\t\t\t\t\t\tStatusCodes.INTERNAL_SERVER_ERROR,\n\t\t\t\t\t\t`Authentication error: ${JSON.stringify(e)}`,\n\t\t\t\t\t);\n\t\t\t\t\tthrow error;\n\t\t\t\t}\n\t\t\t\tthrow e;\n\t\t\t}\n\t\t};\n\t}\n}\n"]}
@@ -3,7 +3,7 @@
3
3
  var _chunkX46GJS7Xcjs = require('./chunk-X46GJS7X.cjs');
4
4
 
5
5
 
6
- var _chunkGB3ON7UVcjs = require('./chunk-GB3ON7UV.cjs');
6
+ var _chunk7PZP7YSAcjs = require('./chunk-7PZP7YSA.cjs');
7
7
 
8
8
  // src/routes/api.routes.ts
9
9
  var _express = require('express');
@@ -11,7 +11,7 @@ var createApiRouter = (agent) => {
11
11
  const router = _express.Router.call(void 0, );
12
12
  router.use("/model", _chunkX46GJS7Xcjs.createModelApiRouter.call(void 0, agent.modelModule));
13
13
  if (agent.memoryModule) {
14
- router.use("/threads", _chunkGB3ON7UVcjs.createThreadApiRouter.call(void 0, agent.memoryModule));
14
+ router.use("/threads", _chunk7PZP7YSAcjs.createThreadApiRouter.call(void 0, agent.memoryModule));
15
15
  }
16
16
  return router;
17
17
  };
@@ -19,4 +19,4 @@ var createApiRouter = (agent) => {
19
19
 
20
20
 
21
21
  exports.createApiRouter = createApiRouter;
22
- //# sourceMappingURL=chunk-45G7LDK7.cjs.map
22
+ //# sourceMappingURL=chunk-FRWC3FQ4.cjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-45G7LDK7.cjs","../../src/routes/api.routes.ts"],"names":[],"mappings":"AAAA;AACE;AACF,wDAA6B;AAC7B;AACE;AACF,wDAA6B;AAC7B;AACA;ACPA,kCAAuB;AAMhB,IAAM,gBAAA,EAAkB,CAAC,KAAA,EAAA,GAA4B;AAC3D,EAAA,MAAM,OAAA,EAAS,6BAAA,CAAO;AAEtB,EAAA,MAAA,CAAO,GAAA,CAAI,QAAA,EAAU,oDAAA,KAAqB,CAAM,WAAW,CAAC,CAAA;AAC5D,EAAA,GAAA,CAAI,KAAA,CAAM,YAAA,EAAc;AACvB,IAAA,MAAA,CAAO,GAAA,CAAI,UAAA,EAAY,qDAAA,KAAsB,CAAM,YAAY,CAAC,CAAA;AAAA,EACjE;AAEA,EAAA,OAAO,MAAA;AACR,CAAA;ADEA;AACA;AACE;AACF,0CAAC","file":"/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-45G7LDK7.cjs","sourcesContent":[null,"import { Router } from \"express\";\nimport type { AINAgent } from \"@/index.js\";\n\nimport { createModelApiRouter } from \"./api/model.routes.js\";\nimport { createThreadApiRouter } from \"./api/threads.routes.js\";\n\nexport const createApiRouter = (agent: AINAgent): Router => {\n\tconst router = Router();\n\n\trouter.use(\"/model\", createModelApiRouter(agent.modelModule));\n\tif (agent.memoryModule) {\n\t\trouter.use(\"/threads\", createThreadApiRouter(agent.memoryModule));\n\t}\n\n\treturn router;\n};\n"]}
1
+ {"version":3,"sources":["/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-FRWC3FQ4.cjs","../../src/routes/api.routes.ts"],"names":[],"mappings":"AAAA;AACE;AACF,wDAA6B;AAC7B;AACE;AACF,wDAA6B;AAC7B;AACA;ACPA,kCAAuB;AAMhB,IAAM,gBAAA,EAAkB,CAAC,KAAA,EAAA,GAA4B;AAC3D,EAAA,MAAM,OAAA,EAAS,6BAAA,CAAO;AAEtB,EAAA,MAAA,CAAO,GAAA,CAAI,QAAA,EAAU,oDAAA,KAAqB,CAAM,WAAW,CAAC,CAAA;AAC5D,EAAA,GAAA,CAAI,KAAA,CAAM,YAAA,EAAc;AACvB,IAAA,MAAA,CAAO,GAAA,CAAI,UAAA,EAAY,qDAAA,KAAsB,CAAM,YAAY,CAAC,CAAA;AAAA,EACjE;AAEA,EAAA,OAAO,MAAA;AACR,CAAA;ADEA;AACA;AACE;AACF,0CAAC","file":"/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-FRWC3FQ4.cjs","sourcesContent":[null,"import { Router } from \"express\";\nimport type { AINAgent } from \"@/index.js\";\n\nimport { createModelApiRouter } from \"./api/model.routes.js\";\nimport { createThreadApiRouter } from \"./api/threads.routes.js\";\n\nexport const createApiRouter = (agent: AINAgent): Router => {\n\tconst router = Router();\n\n\trouter.use(\"/model\", createModelApiRouter(agent.modelModule));\n\tif (agent.memoryModule) {\n\t\trouter.use(\"/threads\", createThreadApiRouter(agent.memoryModule));\n\t}\n\n\treturn router;\n};\n"]}
@@ -72,18 +72,20 @@ Please select and answer the most appropriate intent name from the available int
72
72
  return intent;
73
73
  }
74
74
  /**
75
- * Fulfills the detected intent by generating a response.
75
+ * Fulfills the detected intent by generating a streaming response.
76
76
  *
77
77
  * Manages the complete inference loop including:
78
78
  * - Loading prompts and conversation history
79
79
  * - Collecting available tools from modules
80
80
  * - Executing model inference with tool support
81
81
  * - Processing tool calls iteratively until completion
82
+ * - Streaming results as Server-Sent Events
82
83
  *
83
84
  * @param query - The user's input query
84
85
  * @param threadId - Thread identifier for context
85
86
  * @param thread - Previous conversation history
86
- * @returns Object containing process steps and final response
87
+ * @param intent - Optional detected intent with custom prompt
88
+ * @returns AsyncGenerator yielding StreamEvent objects
87
89
  */
88
90
  async *intentFulfilling(query, threadId, thread, intent) {
89
91
  const systemPrompt = `
@@ -138,10 +140,6 @@ ${_optionalChain([intent, 'optionalAccess', _8 => _8.prompt]) || ""}
138
140
  assembledToolCalls
139
141
  });
140
142
  if (assembledToolCalls.length > 0) {
141
- const messagePayload = _optionalChain([this, 'access', _14 => _14.a2aModule, 'optionalAccess', _15 => _15.getMessagePayload, 'call', _16 => _16(
142
- query,
143
- threadId
144
- )]);
145
143
  for (const toolCall of assembledToolCalls) {
146
144
  const toolCallId = _crypto.randomUUID.call(void 0, );
147
145
  const toolName = toolCall.function.name;
@@ -176,8 +174,7 @@ ${_optionalChain([intent, 'optionalAccess', _8 => _8.prompt]) || ""}
176
174
  _chunk466ED5WNcjs.loggers.intent.debug("A2A tool call", { toolName });
177
175
  toolResult = await this.a2aModule.useTool(
178
176
  selectedTool,
179
- // biome-ignore lint/style/noNonNullAssertion: <a2aModule is guaranteed to be defined>
180
- messagePayload,
177
+ query,
181
178
  threadId
182
179
  );
183
180
  } else {
@@ -232,32 +229,33 @@ ${_optionalChain([intent, 'optionalAccess', _8 => _8.prompt]) || ""}
232
229
  }
233
230
  }
234
231
  /**
235
- * Main entry point for processing user queries.
232
+ * Main entry point for processing streaming user queries.
236
233
  *
237
234
  * Handles the complete query lifecycle:
238
- * 1. Loads thread from memory
235
+ * 1. Loads or creates thread from memory
239
236
  * 2. Detects intent from the query
240
- * 3. Fulfills the intent with AI response
241
- * 4. Updates conversation history
237
+ * 3. Fulfills the intent with streaming AI response
238
+ * 4. Updates conversation history in real-time
242
239
  *
243
- * @param type - The type of thread (e.g., chat, workflow)
244
- * @param userId - The user's unique identifier
245
- * @param threadId - Unique thread identifier
240
+ * @param threadMetadata - Metadata containing type, userId, and optional threadId
241
+ * @param threadMetadata.type - The type of thread (e.g., chat, workflow)
242
+ * @param threadMetadata.userId - The user's unique identifier
243
+ * @param threadMetadata.threadId - Optional thread identifier
246
244
  * @param query - The user's input query
247
- * @returns Object containing the AI-generated response
245
+ * @returns AsyncGenerator yielding StreamEvent objects for SSE
248
246
  */
249
247
  async *handleQueryStream(threadMetadata, query) {
250
248
  const { type, userId } = threadMetadata;
251
249
  const queryStartAt = Date.now();
252
- const threadMemory = _optionalChain([this, 'access', _17 => _17.memoryModule, 'optionalAccess', _18 => _18.getThreadMemory, 'call', _19 => _19()]);
250
+ const threadMemory = _optionalChain([this, 'access', _14 => _14.memoryModule, 'optionalAccess', _15 => _15.getThreadMemory, 'call', _16 => _16()]);
253
251
  let threadId = threadMetadata.threadId;
254
252
  let thread;
255
253
  if (threadId) {
256
- thread = await _optionalChain([threadMemory, 'optionalAccess', _20 => _20.getThread, 'call', _21 => _21(userId, threadId)]);
254
+ thread = await _optionalChain([threadMemory, 'optionalAccess', _17 => _17.getThread, 'call', _18 => _18(userId, threadId)]);
257
255
  } else {
258
256
  threadId = _crypto.randomUUID.call(void 0, );
259
257
  const title = await this.generateTitle(query);
260
- const metadata = await _optionalChain([threadMemory, 'optionalAccess', _22 => _22.createThread, 'call', _23 => _23(type, userId, threadId, title)]) || {
258
+ const metadata = await _optionalChain([threadMemory, 'optionalAccess', _19 => _19.createThread, 'call', _20 => _20(type, userId, threadId, title)]) || {
261
259
  type,
262
260
  threadId,
263
261
  title,
@@ -266,7 +264,7 @@ ${_optionalChain([intent, 'optionalAccess', _8 => _8.prompt]) || ""}
266
264
  _chunk466ED5WNcjs.loggers.intentStream.info("Create new thread", { metadata });
267
265
  yield { event: "thread_id", data: metadata };
268
266
  }
269
- await _optionalChain([threadMemory, 'optionalAccess', _24 => _24.addMessagesToThread, 'call', _25 => _25(userId, threadId, [
267
+ await _optionalChain([threadMemory, 'optionalAccess', _21 => _21.addMessagesToThread, 'call', _22 => _22(userId, threadId, [
270
268
  {
271
269
  role: "USER" /* USER */,
272
270
  timestamp: queryStartAt,
@@ -281,7 +279,7 @@ ${_optionalChain([intent, 'optionalAccess', _8 => _8.prompt]) || ""}
281
279
  _chunk466ED5WNcjs.loggers.intentStream.debug("text_chunk", { event });
282
280
  finalResponseText += event.data.delta;
283
281
  } else if (event.event === "tool_start") {
284
- await _optionalChain([threadMemory, 'optionalAccess', _26 => _26.addMessagesToThread, 'call', _27 => _27(userId, threadId, [
282
+ await _optionalChain([threadMemory, 'optionalAccess', _23 => _23.addMessagesToThread, 'call', _24 => _24(userId, threadId, [
285
283
  {
286
284
  role: "MODEL" /* MODEL */,
287
285
  timestamp: Date.now(),
@@ -297,7 +295,7 @@ ${_optionalChain([intent, 'optionalAccess', _8 => _8.prompt]) || ""}
297
295
  }
298
296
  ])]);
299
297
  } else if (event.event === "tool_output") {
300
- await _optionalChain([threadMemory, 'optionalAccess', _28 => _28.addMessagesToThread, 'call', _29 => _29(userId, threadId, [
298
+ await _optionalChain([threadMemory, 'optionalAccess', _25 => _25.addMessagesToThread, 'call', _26 => _26(userId, threadId, [
301
299
  {
302
300
  role: "MODEL" /* MODEL */,
303
301
  timestamp: Date.now(),
@@ -312,7 +310,7 @@ ${_optionalChain([intent, 'optionalAccess', _8 => _8.prompt]) || ""}
312
310
  }
313
311
  yield event;
314
312
  }
315
- await _optionalChain([threadMemory, 'optionalAccess', _30 => _30.addMessagesToThread, 'call', _31 => _31(userId, threadId, [
313
+ await _optionalChain([threadMemory, 'optionalAccess', _27 => _27.addMessagesToThread, 'call', _28 => _28(userId, threadId, [
316
314
  {
317
315
  role: "MODEL" /* MODEL */,
318
316
  timestamp: Date.now(),
@@ -325,4 +323,4 @@ ${_optionalChain([intent, 'optionalAccess', _8 => _8.prompt]) || ""}
325
323
 
326
324
 
327
325
  exports.QueryStreamService = QueryStreamService;
328
- //# sourceMappingURL=chunk-AIE3SVG7.cjs.map
326
+ //# sourceMappingURL=chunk-KHD7UAFW.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-KHD7UAFW.cjs","../../src/services/query-stream.service.ts"],"names":["intent"],"mappings":"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","file":"/Users/shyun/comcom/ain-agent/ain-adk/dist/cjs/chunk-KHD7UAFW.cjs","sourcesContent":[null,"import { randomUUID } from \"node:crypto\";\nimport type {\n\tA2AModule,\n\tMCPModule,\n\tMemoryModule,\n\tModelModule,\n} from \"@/modules/index.js\";\nimport type { AinAgentPrompts } from \"@/types/agent.js\";\nimport {\n\ttype Intent,\n\tMessageRole,\n\ttype ThreadMetadata,\n\ttype ThreadObject,\n\ttype ThreadType,\n} from \"@/types/memory.js\";\nimport type { StreamEvent } from \"@/types/stream\";\nimport {\n\ttype IA2ATool,\n\ttype IAgentTool,\n\ttype IMCPTool,\n\tTOOL_PROTOCOL_TYPE,\n} from \"@/types/tool.js\";\nimport { loggers } from \"@/utils/logger.js\";\n\n/**\n * Service for processing user queries through the agent's AI pipeline.\n *\n * Orchestrates the query processing workflow including intent detection,\n * model inference, tool execution, and response generation. Manages\n * conversation context and coordinates between different modules.\n */\nexport class QueryStreamService {\n\tprivate modelModule: ModelModule;\n\tprivate a2aModule?: A2AModule;\n\tprivate mcpModule?: MCPModule;\n\tprivate memoryModule?: MemoryModule;\n\tprivate prompts?: AinAgentPrompts;\n\n\tconstructor(\n\t\tmodelModule: ModelModule,\n\t\ta2aModule?: A2AModule,\n\t\tmcpModule?: MCPModule,\n\t\tmemoryModule?: MemoryModule,\n\t\tprompts?: AinAgentPrompts,\n\t) {\n\t\tthis.modelModule = modelModule;\n\t\tthis.a2aModule = a2aModule;\n\t\tthis.mcpModule = mcpModule;\n\t\tthis.memoryModule = memoryModule;\n\t\tthis.prompts = prompts;\n\t}\n\n\t/**\n\t * Detects the intent from context.\n\t *\n\t * @param query - The user's input query\n\t * @param thread - The thread history\n\t * @returns The detected intent\n\t */\n\tprivate async intentTriggering(\n\t\tquery: string,\n\t\tthread: ThreadObject | undefined,\n\t): Promise<Intent | undefined> {\n\t\tconst modelInstance = this.modelModule.getModel();\n\t\tconst intentMemory = this.memoryModule?.getIntentMemory();\n\t\tif (!intentMemory) {\n\t\t\treturn undefined;\n\t\t}\n\n\t\t// 인텐트 목록 가져오기\n\t\tconst intents = await intentMemory.listIntents();\n\n\t\tif (intents.length === 0) {\n\t\t\tloggers.intentStream.warn(\"No intent found\");\n\t\t\treturn undefined;\n\t\t}\n\n\t\tconst intentList = intents\n\t\t\t.map((intent) => `- ${intent.name}: ${intent.description}`)\n\t\t\t.join(\"\\n\");\n\n\t\t// Convert session history to a string\n\t\tconst historyMessages = !thread\n\t\t\t? \"\"\n\t\t\t: Object.entries(thread.messages)\n\t\t\t\t\t.sort(([, a], [, b]) => a.timestamp - b.timestamp)\n\t\t\t\t\t.map(([_chatId, chat]) => {\n\t\t\t\t\t\tconst role =\n\t\t\t\t\t\t\tchat.role === \"USER\"\n\t\t\t\t\t\t\t\t? \"User\"\n\t\t\t\t\t\t\t\t: chat.role === \"MODEL\"\n\t\t\t\t\t\t\t\t\t? \"Assistant\"\n\t\t\t\t\t\t\t\t\t: \"System\";\n\t\t\t\t\t\tconst content = Array.isArray(chat.content.parts)\n\t\t\t\t\t\t\t? chat.content.parts.join(\" \")\n\t\t\t\t\t\t\t: String(chat.content.parts);\n\t\t\t\t\t\treturn `${role}: \"\"\"${content}\"\"\"`;\n\t\t\t\t\t})\n\t\t\t\t\t.join(\"\\n\");\n\n\t\tconst systemPrompt = `You are an expert in accurately identifying user intentions.\n\nAvailable intent list:\n${intentList}\n\nPlease select and answer only from the above intent list. \nPlease return only the exact intent name without any additional explanations or text.`;\n\n\t\tconst userMessage = `The following is the conversation history with the user:\n\n${historyMessages}\n\nLast user question: \"${query}\"\n\nBased on the above conversation history, please determine what the intention of the last user question is. \nPlease select and answer the most appropriate intent name from the available intent list.`;\n\n\t\tconst messages = modelInstance.generateMessages({\n\t\t\tquery: userMessage,\n\t\t\tsystemPrompt,\n\t\t});\n\n\t\tconst response = await modelInstance.fetch(messages);\n\t\tif (!response.content) {\n\t\t\tthrow new Error(\"No intent detected\");\n\t\t}\n\t\tconst intentName = response.content.trim();\n\t\tconst intent = await intentMemory.getIntentByName(intentName);\n\t\tif (!intent) {\n\t\t\tthrow new Error(`No intent found: ${intentName}`);\n\t\t}\n\t\treturn intent;\n\t}\n\n\t/**\n\t * Fulfills the detected intent by generating a streaming response.\n\t *\n\t * Manages the complete inference loop including:\n\t * - Loading prompts and conversation history\n\t * - Collecting available tools from modules\n\t * - Executing model inference with tool support\n\t * - Processing tool calls iteratively until completion\n\t * - Streaming results as Server-Sent Events\n\t *\n\t * @param query - The user's input query\n\t * @param threadId - Thread identifier for context\n\t * @param thread - Previous conversation history\n\t * @param intent - Optional detected intent with custom prompt\n\t * @returns AsyncGenerator yielding StreamEvent objects\n\t */\n\tpublic async *intentFulfilling(\n\t\tquery: string,\n\t\tthreadId: string,\n\t\tthread?: ThreadObject,\n\t\tintent?: Intent,\n\t): AsyncGenerator<StreamEvent> {\n\t\tconst systemPrompt = `\nToday is ${new Date().toLocaleDateString()}.\n\n${this.prompts?.agent || \"\"}\n\n${this.prompts?.system || \"\"}\n\n${intent?.prompt || \"\"}\n\t`.trim();\n\n\t\tconst modelInstance = this.modelModule.getModel();\n\t\tconst messages = modelInstance.generateMessages({\n\t\t\tquery,\n\t\t\tthread,\n\t\t\tsystemPrompt: systemPrompt.trim(),\n\t\t});\n\n\t\tconst tools: IAgentTool[] = [];\n\t\tthis.mcpModule && tools.push(...this.mcpModule.getTools());\n\t\tthis.a2aModule && tools.push(...(await this.a2aModule.getTools()));\n\n\t\tconst functions = modelInstance.convertToolsToFunctions(tools);\n\n\t\tconst processList: string[] = [];\n\n\t\twhile (true) {\n\t\t\tconst responseStream = await modelInstance.fetchStreamWithContextMessage(\n\t\t\t\tmessages,\n\t\t\t\tfunctions,\n\t\t\t);\n\n\t\t\tconst assembledToolCalls: {\n\t\t\t\tid: string;\n\t\t\t\ttype: \"function\";\n\t\t\t\tfunction: { name: string; arguments: string };\n\t\t\t}[] = [];\n\n\t\t\tloggers.intentStream.debug(\"messages\", { messages });\n\n\t\t\tfor await (const chunk of responseStream) {\n\t\t\t\tconst delta = chunk.delta;\n\t\t\t\tif (delta?.tool_calls) {\n\t\t\t\t\tfor (const { index, id, function: func } of delta.tool_calls) {\n\t\t\t\t\t\tassembledToolCalls[index] ??= {\n\t\t\t\t\t\t\tid: \"\",\n\t\t\t\t\t\t\ttype: \"function\",\n\t\t\t\t\t\t\tfunction: { name: \"\", arguments: \"\" },\n\t\t\t\t\t\t};\n\n\t\t\t\t\t\tif (id) assembledToolCalls[index].id = id;\n\t\t\t\t\t\tif (func?.name) assembledToolCalls[index].function.name = func.name;\n\t\t\t\t\t\tif (func?.arguments)\n\t\t\t\t\t\t\tassembledToolCalls[index].function.arguments += func.arguments;\n\t\t\t\t\t}\n\t\t\t\t} else if (chunk.delta?.content) {\n\t\t\t\t\tyield {\n\t\t\t\t\t\tevent: \"text_chunk\",\n\t\t\t\t\t\tdata: { delta: chunk.delta.content },\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tloggers.intentStream.debug(\"assembledToolCalls\", {\n\t\t\t\tassembledToolCalls,\n\t\t\t});\n\n\t\t\tif (assembledToolCalls.length > 0) {\n\t\t\t\tfor (const toolCall of assembledToolCalls) {\n\t\t\t\t\tconst toolCallId = randomUUID();\n\t\t\t\t\tconst toolName = toolCall.function.name;\n\t\t\t\t\tconst selectedTool = tools.filter((tool) => tool.id === toolName)[0];\n\n\t\t\t\t\tlet toolResult = \"\";\n\t\t\t\t\tif (\n\t\t\t\t\t\tthis.mcpModule &&\n\t\t\t\t\t\tselectedTool.protocol === TOOL_PROTOCOL_TYPE.MCP\n\t\t\t\t\t) {\n\t\t\t\t\t\tconst toolArgs = JSON.parse(toolCall.function.arguments) as\n\t\t\t\t\t\t\t| { [x: string]: unknown }\n\t\t\t\t\t\t\t| undefined;\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\tevent: \"tool_start\",\n\t\t\t\t\t\t\tdata: {\n\t\t\t\t\t\t\t\ttoolCallId,\n\t\t\t\t\t\t\t\tprotocol: TOOL_PROTOCOL_TYPE.MCP,\n\t\t\t\t\t\t\t\ttoolName,\n\t\t\t\t\t\t\t\ttoolArgs,\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t};\n\t\t\t\t\t\tloggers.intent.debug(\"MCP tool call\", { toolName, toolArgs });\n\t\t\t\t\t\ttoolResult = await this.mcpModule.useTool(\n\t\t\t\t\t\t\tselectedTool as IMCPTool,\n\t\t\t\t\t\t\ttoolArgs,\n\t\t\t\t\t\t);\n\t\t\t\t\t} else if (\n\t\t\t\t\t\tthis.a2aModule &&\n\t\t\t\t\t\tselectedTool.protocol === TOOL_PROTOCOL_TYPE.A2A\n\t\t\t\t\t) {\n\t\t\t\t\t\tyield {\n\t\t\t\t\t\t\tevent: \"tool_start\",\n\t\t\t\t\t\t\tdata: {\n\t\t\t\t\t\t\t\ttoolCallId,\n\t\t\t\t\t\t\t\tprotocol: TOOL_PROTOCOL_TYPE.A2A,\n\t\t\t\t\t\t\t\ttoolName,\n\t\t\t\t\t\t\t\ttoolArgs: null,\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t};\n\t\t\t\t\t\tloggers.intent.debug(\"A2A tool call\", { toolName });\n\t\t\t\t\t\ttoolResult = await this.a2aModule.useTool(\n\t\t\t\t\t\t\tselectedTool as IA2ATool,\n\t\t\t\t\t\t\tquery,\n\t\t\t\t\t\t\tthreadId,\n\t\t\t\t\t\t);\n\t\t\t\t\t} else {\n\t\t\t\t\t\t// Unrecognized tool type. It cannot be happened...\n\t\t\t\t\t\tloggers.intent.warn(\n\t\t\t\t\t\t\t`Unrecognized tool type: ${selectedTool.protocol}`,\n\t\t\t\t\t\t);\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\t\t\t\t\tyield {\n\t\t\t\t\t\tevent: \"tool_output\",\n\t\t\t\t\t\tdata: {\n\t\t\t\t\t\t\ttoolCallId,\n\t\t\t\t\t\t\tprotocol: selectedTool.protocol,\n\t\t\t\t\t\t\ttoolName,\n\t\t\t\t\t\t\tresult: toolResult,\n\t\t\t\t\t\t},\n\t\t\t\t\t};\n\t\t\t\t\tloggers.intent.debug(\"toolResult\", { toolResult });\n\n\t\t\t\t\tprocessList.push(toolResult);\n\t\t\t\t\tmodelInstance.appendMessages(messages, toolResult);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Generates a title for the conversation based on the query.\n\t *\n\t * @param query - The user's input query\n\t * @returns Promise resolving to a generated title\n\t */\n\n\tprivate async generateTitle(query: string): Promise<string> {\n\t\tconst DEFAULT_TITLE = \"New Chat\";\n\t\ttry {\n\t\t\tconst modelInstance = this.modelModule.getModel();\n\t\t\tconst messages = modelInstance.generateMessages({\n\t\t\t\tquery,\n\t\t\t\tsystemPrompt: `You are a helpful assistant that generates titles for conversations.\n Please analyze the user's query and create a concise title that accurately reflects the conversation's core topic.\n The title must be no more than 5 words long.\n Respond with only the title. Do not include any punctuation or extra explanations.`,\n\t\t\t});\n\t\t\tconst response = await modelInstance.fetch(messages);\n\t\t\treturn response.content || DEFAULT_TITLE;\n\t\t} catch (error) {\n\t\t\tloggers.intentStream.error(\"Error generating title\", {\n\t\t\t\terror,\n\t\t\t\tquery,\n\t\t\t});\n\t\t\treturn DEFAULT_TITLE;\n\t\t}\n\t}\n\n\t/**\n\t * Main entry point for processing streaming user queries.\n\t *\n\t * Handles the complete query lifecycle:\n\t * 1. Loads or creates thread from memory\n\t * 2. Detects intent from the query\n\t * 3. Fulfills the intent with streaming AI response\n\t * 4. Updates conversation history in real-time\n\t *\n\t * @param threadMetadata - Metadata containing type, userId, and optional threadId\n\t * @param threadMetadata.type - The type of thread (e.g., chat, workflow)\n\t * @param threadMetadata.userId - The user's unique identifier\n\t * @param threadMetadata.threadId - Optional thread identifier\n\t * @param query - The user's input query\n\t * @returns AsyncGenerator yielding StreamEvent objects for SSE\n\t */\n\tpublic async *handleQueryStream(\n\t\tthreadMetadata: {\n\t\t\ttype: ThreadType;\n\t\t\tuserId: string;\n\t\t\tthreadId?: string;\n\t\t},\n\t\tquery: string,\n\t): AsyncGenerator<StreamEvent> {\n\t\tconst { type, userId } = threadMetadata;\n\t\tconst queryStartAt = Date.now();\n\t\tconst threadMemory = this.memoryModule?.getThreadMemory();\n\n\t\t// 1. Load or create thread\n\t\tlet threadId = threadMetadata.threadId;\n\t\tlet thread: ThreadObject | undefined;\n\t\tif (threadId) {\n\t\t\tthread = await threadMemory?.getThread(userId, threadId);\n\t\t} else {\n\t\t\tthreadId = randomUUID();\n\t\t\tconst title = await this.generateTitle(query);\n\n\t\t\tconst metadata =\n\t\t\t\t(await threadMemory?.createThread(type, userId, threadId, title)) ||\n\t\t\t\t({\n\t\t\t\t\ttype,\n\t\t\t\t\tthreadId,\n\t\t\t\t\ttitle,\n\t\t\t\t\tupdatedAt: Date.now(),\n\t\t\t\t} as ThreadMetadata);\n\t\t\tloggers.intentStream.info(\"Create new thread\", { metadata });\n\t\t\tyield { event: \"thread_id\", data: metadata };\n\t\t}\n\n\t\tawait threadMemory?.addMessagesToThread(userId, threadId, [\n\t\t\t{\n\t\t\t\trole: MessageRole.USER,\n\t\t\t\ttimestamp: queryStartAt,\n\t\t\t\tcontent: { type: \"text\", parts: [query] },\n\t\t\t},\n\t\t]);\n\n\t\t// 2. intent triggering\n\t\tconst intent = await this.intentTriggering(query, thread);\n\n\t\t// 3. intent fulfillment\n\t\tconst stream = this.intentFulfilling(query, threadId, thread, intent);\n\n\t\tlet finalResponseText = \"\";\n\t\tfor await (const event of stream) {\n\t\t\tif (event.event === \"text_chunk\" && event.data.delta) {\n\t\t\t\tloggers.intentStream.debug(\"text_chunk\", { event });\n\t\t\t\tfinalResponseText += event.data.delta;\n\t\t\t} else if (event.event === \"tool_start\") {\n\t\t\t\tawait threadMemory?.addMessagesToThread(userId, threadId, [\n\t\t\t\t\t{\n\t\t\t\t\t\trole: MessageRole.MODEL,\n\t\t\t\t\t\ttimestamp: Date.now(),\n\t\t\t\t\t\tcontent: {\n\t\t\t\t\t\t\ttype: \"text\",\n\t\t\t\t\t\t\tparts: [JSON.stringify(event.data.toolArgs)],\n\t\t\t\t\t\t},\n\t\t\t\t\t\tmetadata: {\n\t\t\t\t\t\t\ttoolCallId: event.data.toolCallId,\n\t\t\t\t\t\t\ttoolName: event.data.toolName,\n\t\t\t\t\t\t\tprotocol: event.data.protocol,\n\t\t\t\t\t\t},\n\t\t\t\t\t},\n\t\t\t\t]);\n\t\t\t} else if (event.event === \"tool_output\") {\n\t\t\t\tawait threadMemory?.addMessagesToThread(userId, threadId, [\n\t\t\t\t\t{\n\t\t\t\t\t\trole: MessageRole.MODEL,\n\t\t\t\t\t\ttimestamp: Date.now(),\n\t\t\t\t\t\tcontent: { type: \"text\", parts: [event.data.result] },\n\t\t\t\t\t\tmetadata: {\n\t\t\t\t\t\t\ttoolCallId: event.data.toolCallId,\n\t\t\t\t\t\t\ttoolName: event.data.toolName,\n\t\t\t\t\t\t\tprotocol: event.data.protocol,\n\t\t\t\t\t\t},\n\t\t\t\t\t},\n\t\t\t\t]);\n\t\t\t}\n\t\t\tyield event;\n\t\t}\n\n\t\tawait threadMemory?.addMessagesToThread(userId, threadId, [\n\t\t\t{\n\t\t\t\trole: MessageRole.MODEL,\n\t\t\t\ttimestamp: Date.now(),\n\t\t\t\tcontent: { type: \"text\", parts: [finalResponseText] },\n\t\t\t},\n\t\t]);\n\t}\n}\n"]}