@agentscope-ai/agentscope 0.0.5 → 0.0.6

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 (44) hide show
  1. package/dist/agent/index.d.mts +4 -4
  2. package/dist/agent/index.d.ts +4 -4
  3. package/dist/agent/index.js +34 -7
  4. package/dist/agent/index.js.map +1 -1
  5. package/dist/agent/index.mjs +34 -7
  6. package/dist/agent/index.mjs.map +1 -1
  7. package/dist/{base-qmU135_k.d.ts → base-13VLaOvY.d.ts} +2 -2
  8. package/dist/{base-BI5s2ksj.d.mts → base-Bc3GkNS7.d.mts} +1 -1
  9. package/dist/{base-DHtZCg94.d.ts → base-CJkm56kB.d.ts} +1 -1
  10. package/dist/{base-CFDeoJRe.d.ts → base-Dfizi3RB.d.ts} +1 -1
  11. package/dist/{base-BDyDUIhj.d.mts → base-L72wZVx8.d.mts} +2 -2
  12. package/dist/{base-BB9eTlit.d.mts → base-Ps8E0j1_.d.mts} +1 -1
  13. package/dist/event/index.d.mts +2 -2
  14. package/dist/event/index.d.ts +2 -2
  15. package/dist/event/index.js.map +1 -1
  16. package/dist/event/index.mjs.map +1 -1
  17. package/dist/formatter/index.d.mts +2 -2
  18. package/dist/formatter/index.d.ts +2 -2
  19. package/dist/formatter/index.js +16 -0
  20. package/dist/formatter/index.js.map +1 -1
  21. package/dist/formatter/index.mjs +16 -0
  22. package/dist/formatter/index.mjs.map +1 -1
  23. package/dist/message/index.d.mts +1 -1
  24. package/dist/message/index.d.ts +1 -1
  25. package/dist/message/index.js +50 -14
  26. package/dist/message/index.js.map +1 -1
  27. package/dist/message/index.mjs +50 -14
  28. package/dist/message/index.mjs.map +1 -1
  29. package/dist/{message-D-LObC06.d.mts → message-COpNEf0G.d.mts} +10 -7
  30. package/dist/{message-DU0_qm3u.d.ts → message-DbCMy5tM.d.ts} +10 -7
  31. package/dist/model/index.d.mts +4 -4
  32. package/dist/model/index.d.ts +4 -4
  33. package/dist/model/index.js +16 -0
  34. package/dist/model/index.js.map +1 -1
  35. package/dist/model/index.mjs +16 -0
  36. package/dist/model/index.mjs.map +1 -1
  37. package/dist/storage/index.d.mts +2 -2
  38. package/dist/storage/index.d.ts +2 -2
  39. package/package.json +1 -1
  40. package/src/agent/agent.ts +20 -7
  41. package/src/event/index.ts +2 -2
  42. package/src/message/append-event.test.ts +17 -11
  43. package/src/message/message.test.ts +2 -2
  44. package/src/message/message.ts +77 -14
@@ -1,8 +1,8 @@
1
1
  import { z } from 'zod';
2
2
  import { J as JSONSerializableObject, T as ToolSchema, a as ToolChoice } from './index-CAxQAkiP.js';
3
3
  import { T as TextBlock, b as ToolCallBlock, a as ThinkingBlock, D as DataBlock } from './block-BqWf-Qcb.js';
4
- import { F as FormatterBase } from './base-CFDeoJRe.js';
5
- import { M as Msg } from './message-DU0_qm3u.js';
4
+ import { F as FormatterBase } from './base-Dfizi3RB.js';
5
+ import { M as Msg } from './message-DbCMy5tM.js';
6
6
 
7
7
  /**
8
8
  * The usage structure for chat models.
@@ -1,4 +1,4 @@
1
- import { M as Msg } from './message-D-LObC06.mjs';
1
+ import { M as Msg } from './message-COpNEf0G.mjs';
2
2
  import { T as TextBlock, D as DataBlock } from './block-BqWf-Qcb.mjs';
3
3
 
4
4
  /**
@@ -1,4 +1,4 @@
1
- import { M as Msg } from './message-DU0_qm3u.js';
1
+ import { M as Msg } from './message-DbCMy5tM.js';
2
2
 
3
3
  /**
4
4
  * The complete agent state including both conversation context and metadata.
@@ -1,4 +1,4 @@
1
- import { M as Msg } from './message-DU0_qm3u.js';
1
+ import { M as Msg } from './message-DbCMy5tM.js';
2
2
  import { T as TextBlock, D as DataBlock } from './block-BqWf-Qcb.js';
3
3
 
4
4
  /**
@@ -1,8 +1,8 @@
1
1
  import { z } from 'zod';
2
2
  import { J as JSONSerializableObject, T as ToolSchema, a as ToolChoice } from './index-CAxQAkiP.mjs';
3
3
  import { T as TextBlock, b as ToolCallBlock, a as ThinkingBlock, D as DataBlock } from './block-BqWf-Qcb.mjs';
4
- import { F as FormatterBase } from './base-BI5s2ksj.mjs';
5
- import { M as Msg } from './message-D-LObC06.mjs';
4
+ import { F as FormatterBase } from './base-Bc3GkNS7.mjs';
5
+ import { M as Msg } from './message-COpNEf0G.mjs';
6
6
 
7
7
  /**
8
8
  * The usage structure for chat models.
@@ -1,4 +1,4 @@
1
- import { M as Msg } from './message-D-LObC06.mjs';
1
+ import { M as Msg } from './message-COpNEf0G.mjs';
2
2
 
3
3
  /**
4
4
  * The complete agent state including both conversation context and metadata.
@@ -131,14 +131,14 @@ interface ToolResultTextDeltaEvent extends EventBase {
131
131
  type: EventType.TOOL_RESULT_TEXT_DELTA;
132
132
  reply_id: string;
133
133
  tool_call_id: string;
134
- block_id: string;
135
134
  delta: string;
136
135
  }
137
136
  interface ToolResultDataDeltaEvent extends EventBase {
138
137
  type: EventType.TOOL_RESULT_DATA_DELTA;
139
138
  reply_id: string;
140
139
  tool_call_id: string;
141
- block_id: string;
140
+ /** Auto-generated in {@link appendEvent} when not provided. */
141
+ block_id?: string;
142
142
  media_type: string;
143
143
  data?: string;
144
144
  url?: string;
@@ -131,14 +131,14 @@ interface ToolResultTextDeltaEvent extends EventBase {
131
131
  type: EventType.TOOL_RESULT_TEXT_DELTA;
132
132
  reply_id: string;
133
133
  tool_call_id: string;
134
- block_id: string;
135
134
  delta: string;
136
135
  }
137
136
  interface ToolResultDataDeltaEvent extends EventBase {
138
137
  type: EventType.TOOL_RESULT_DATA_DELTA;
139
138
  reply_id: string;
140
139
  tool_call_id: string;
141
- block_id: string;
140
+ /** Auto-generated in {@link appendEvent} when not provided. */
141
+ block_id?: string;
142
142
  media_type: string;
143
143
  data?: string;
144
144
  url?: string;
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/event/index.ts"],"sourcesContent":["import { ToolCallBlock, ToolResultBlock } from '../message';\nimport { PermissionRule } from '../permission';\n\nexport enum EventType {\n REPLY_START = 'REPLY_START',\n REPLY_END = 'REPLY_END',\n\n MODEL_CALL_START = 'MODEL_CALL_START',\n MODEL_CALL_END = 'MODEL_CALL_END',\n\n TEXT_BLOCK_START = 'TEXT_BLOCK_START',\n TEXT_BLOCK_DELTA = 'TEXT_BLOCK_DELTA',\n TEXT_BLOCK_END = 'TEXT_BLOCK_END',\n\n DATA_BLOCK_START = 'DATA_BLOCK_START',\n DATA_BLOCK_DELTA = 'DATA_BLOCK_DELTA',\n DATA_BLOCK_END = 'DATA_BLOCK_END',\n\n THINKING_BLOCK_START = 'THINKING_BLOCK_START',\n THINKING_BLOCK_DELTA = 'THINKING_BLOCK_DELTA',\n THINKING_BLOCK_END = 'THINKING_BLOCK_END',\n\n TOOL_CALL_START = 'TOOL_CALL_START',\n TOOL_CALL_DELTA = 'TOOL_CALL_DELTA',\n TOOL_CALL_END = 'TOOL_CALL_END',\n\n TOOL_RESULT_START = 'TOOL_RESULT_START',\n TOOL_RESULT_TEXT_DELTA = 'TOOL_RESULT_TEXT_DELTA',\n TOOL_RESULT_DATA_DELTA = 'TOOL_RESULT_DATA_DELTA',\n TOOL_RESULT_END = 'TOOL_RESULT_END',\n\n EXCEED_MAX_ITERS = 'EXCEED_MAX_ITERS',\n\n REQUIRE_USER_CONFIRM = 'REQUIRE_USER_CONFIRM',\n REQUIRE_EXTERNAL_EXECUTION = 'REQUIRE_EXTERNAL_EXECUTION',\n\n USER_CONFIRM_RESULT = 'USER_CONFIRM_RESULT',\n EXTERNAL_EXECUTION_RESULT = 'EXTERNAL_EXECUTION_RESULT',\n}\n\nexport interface EventBase {\n id: string;\n created_at: string;\n}\n\nexport interface ReplyStartEvent extends EventBase {\n type: EventType.REPLY_START;\n session_id: string;\n reply_id: string;\n name: string;\n role: 'user' | 'assistant' | 'system';\n}\n\nexport interface ReplyEndEvent extends EventBase {\n type: EventType.REPLY_END;\n session_id: string;\n reply_id: string;\n}\n\nexport interface ModelCallStartEvent extends EventBase {\n type: EventType.MODEL_CALL_START;\n reply_id: string;\n model_name: string;\n}\n\nexport interface ModelCallEndEvent extends EventBase {\n type: EventType.MODEL_CALL_END;\n reply_id: string;\n input_tokens: number;\n output_tokens: number;\n}\n\nexport interface TextBlockStartEvent extends EventBase {\n type: EventType.TEXT_BLOCK_START;\n block_id: string;\n reply_id: string;\n}\n\nexport interface TextBlockDeltaEvent extends EventBase {\n type: EventType.TEXT_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface TextBlockEndEvent extends EventBase {\n type: EventType.TEXT_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface DataBlockStartEvent extends EventBase {\n type: EventType.DATA_BLOCK_START;\n reply_id: string;\n block_id: string;\n media_type: string;\n}\n\nexport interface DataBlockDeltaEvent extends EventBase {\n type: EventType.DATA_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n data: string;\n media_type: string;\n}\n\nexport interface DataBlockEndEvent extends EventBase {\n type: EventType.DATA_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ThinkingBlockStartEvent extends EventBase {\n type: EventType.THINKING_BLOCK_START;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ThinkingBlockDeltaEvent extends EventBase {\n type: EventType.THINKING_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface ThinkingBlockEndEvent extends EventBase {\n type: EventType.THINKING_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ToolCallStartEvent extends EventBase {\n type: EventType.TOOL_CALL_START;\n reply_id: string;\n tool_call_id: string;\n tool_call_name: string;\n}\n\nexport interface ToolCallDeltaEvent extends EventBase {\n type: EventType.TOOL_CALL_DELTA;\n reply_id: string;\n tool_call_id: string;\n delta: string;\n}\n\nexport interface ToolCallEndEvent extends EventBase {\n type: EventType.TOOL_CALL_END;\n reply_id: string;\n tool_call_id: string;\n}\n\nexport interface ToolResultStartEvent extends EventBase {\n type: EventType.TOOL_RESULT_START;\n reply_id: string;\n tool_call_id: string;\n tool_call_name: string;\n}\n\nexport interface ToolResultTextDeltaEvent extends EventBase {\n type: EventType.TOOL_RESULT_TEXT_DELTA;\n reply_id: string;\n tool_call_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface ToolResultDataDeltaEvent extends EventBase {\n type: EventType.TOOL_RESULT_DATA_DELTA;\n reply_id: string;\n tool_call_id: string;\n block_id: string;\n media_type: string;\n data?: string;\n url?: string;\n}\n\nexport interface ToolResultEndEvent extends EventBase {\n type: EventType.TOOL_RESULT_END;\n reply_id: string;\n tool_call_id: string;\n state: ToolResultBlock['state'];\n}\n\nexport interface ExceedMaxItersEvent extends EventBase {\n type: EventType.EXCEED_MAX_ITERS;\n reply_id: string;\n name: string;\n}\n\nexport interface RequireUserConfirmEvent extends EventBase {\n type: EventType.REQUIRE_USER_CONFIRM;\n reply_id: string;\n tool_calls: ToolCallBlock[];\n}\n\nexport interface RequireExternalExecutionEvent extends EventBase {\n type: EventType.REQUIRE_EXTERNAL_EXECUTION;\n reply_id: string;\n tool_calls: ToolCallBlock[];\n}\n\nexport interface ConfirmResult {\n confirmed: boolean;\n tool_call: ToolCallBlock;\n rules?: PermissionRule[] | null;\n}\n\nexport interface UserConfirmResultEvent extends EventBase {\n type: EventType.USER_CONFIRM_RESULT;\n reply_id: string;\n confirm_results: ConfirmResult[];\n}\n\nexport interface ExternalExecutionResultEvent extends EventBase {\n type: EventType.EXTERNAL_EXECUTION_RESULT;\n reply_id: string;\n execution_results: ToolResultBlock[];\n}\n\nexport type AgentEvent =\n // The control events for the whole run\n | ReplyStartEvent\n | ReplyEndEvent\n | ExceedMaxItersEvent\n | RequireUserConfirmEvent\n | RequireExternalExecutionEvent\n | ModelCallStartEvent\n | ModelCallEndEvent\n // The data events for different block types\n | TextBlockStartEvent\n | TextBlockDeltaEvent\n | TextBlockEndEvent\n | DataBlockStartEvent\n | DataBlockDeltaEvent\n | DataBlockEndEvent\n | ThinkingBlockStartEvent\n | ThinkingBlockDeltaEvent\n | ThinkingBlockEndEvent\n | ToolCallStartEvent\n | ToolCallDeltaEvent\n | ToolCallEndEvent\n | ToolResultStartEvent\n | ToolResultTextDeltaEvent\n | ToolResultDataDeltaEvent\n | ToolResultEndEvent\n // The events from the external execution or user confirmation\n | UserConfirmResultEvent\n | ExternalExecutionResultEvent;\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAGO,IAAK,YAAL,kBAAKA,eAAL;AACH,EAAAA,WAAA,iBAAc;AACd,EAAAA,WAAA,eAAY;AAEZ,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,wBAAqB;AAErB,EAAAA,WAAA,qBAAkB;AAClB,EAAAA,WAAA,qBAAkB;AAClB,EAAAA,WAAA,mBAAgB;AAEhB,EAAAA,WAAA,uBAAoB;AACpB,EAAAA,WAAA,4BAAyB;AACzB,EAAAA,WAAA,4BAAyB;AACzB,EAAAA,WAAA,qBAAkB;AAElB,EAAAA,WAAA,sBAAmB;AAEnB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,gCAA6B;AAE7B,EAAAA,WAAA,yBAAsB;AACtB,EAAAA,WAAA,+BAA4B;AAlCpB,SAAAA;AAAA,GAAA;","names":["EventType"]}
1
+ {"version":3,"sources":["../../src/event/index.ts"],"sourcesContent":["import { ToolCallBlock, ToolResultBlock } from '../message';\nimport { PermissionRule } from '../permission';\n\nexport enum EventType {\n REPLY_START = 'REPLY_START',\n REPLY_END = 'REPLY_END',\n\n MODEL_CALL_START = 'MODEL_CALL_START',\n MODEL_CALL_END = 'MODEL_CALL_END',\n\n TEXT_BLOCK_START = 'TEXT_BLOCK_START',\n TEXT_BLOCK_DELTA = 'TEXT_BLOCK_DELTA',\n TEXT_BLOCK_END = 'TEXT_BLOCK_END',\n\n DATA_BLOCK_START = 'DATA_BLOCK_START',\n DATA_BLOCK_DELTA = 'DATA_BLOCK_DELTA',\n DATA_BLOCK_END = 'DATA_BLOCK_END',\n\n THINKING_BLOCK_START = 'THINKING_BLOCK_START',\n THINKING_BLOCK_DELTA = 'THINKING_BLOCK_DELTA',\n THINKING_BLOCK_END = 'THINKING_BLOCK_END',\n\n TOOL_CALL_START = 'TOOL_CALL_START',\n TOOL_CALL_DELTA = 'TOOL_CALL_DELTA',\n TOOL_CALL_END = 'TOOL_CALL_END',\n\n TOOL_RESULT_START = 'TOOL_RESULT_START',\n TOOL_RESULT_TEXT_DELTA = 'TOOL_RESULT_TEXT_DELTA',\n TOOL_RESULT_DATA_DELTA = 'TOOL_RESULT_DATA_DELTA',\n TOOL_RESULT_END = 'TOOL_RESULT_END',\n\n EXCEED_MAX_ITERS = 'EXCEED_MAX_ITERS',\n\n REQUIRE_USER_CONFIRM = 'REQUIRE_USER_CONFIRM',\n REQUIRE_EXTERNAL_EXECUTION = 'REQUIRE_EXTERNAL_EXECUTION',\n\n USER_CONFIRM_RESULT = 'USER_CONFIRM_RESULT',\n EXTERNAL_EXECUTION_RESULT = 'EXTERNAL_EXECUTION_RESULT',\n}\n\nexport interface EventBase {\n id: string;\n created_at: string;\n}\n\nexport interface ReplyStartEvent extends EventBase {\n type: EventType.REPLY_START;\n session_id: string;\n reply_id: string;\n name: string;\n role: 'user' | 'assistant' | 'system';\n}\n\nexport interface ReplyEndEvent extends EventBase {\n type: EventType.REPLY_END;\n session_id: string;\n reply_id: string;\n}\n\nexport interface ModelCallStartEvent extends EventBase {\n type: EventType.MODEL_CALL_START;\n reply_id: string;\n model_name: string;\n}\n\nexport interface ModelCallEndEvent extends EventBase {\n type: EventType.MODEL_CALL_END;\n reply_id: string;\n input_tokens: number;\n output_tokens: number;\n}\n\nexport interface TextBlockStartEvent extends EventBase {\n type: EventType.TEXT_BLOCK_START;\n block_id: string;\n reply_id: string;\n}\n\nexport interface TextBlockDeltaEvent extends EventBase {\n type: EventType.TEXT_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface TextBlockEndEvent extends EventBase {\n type: EventType.TEXT_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface DataBlockStartEvent extends EventBase {\n type: EventType.DATA_BLOCK_START;\n reply_id: string;\n block_id: string;\n media_type: string;\n}\n\nexport interface DataBlockDeltaEvent extends EventBase {\n type: EventType.DATA_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n data: string;\n media_type: string;\n}\n\nexport interface DataBlockEndEvent extends EventBase {\n type: EventType.DATA_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ThinkingBlockStartEvent extends EventBase {\n type: EventType.THINKING_BLOCK_START;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ThinkingBlockDeltaEvent extends EventBase {\n type: EventType.THINKING_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface ThinkingBlockEndEvent extends EventBase {\n type: EventType.THINKING_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ToolCallStartEvent extends EventBase {\n type: EventType.TOOL_CALL_START;\n reply_id: string;\n tool_call_id: string;\n tool_call_name: string;\n}\n\nexport interface ToolCallDeltaEvent extends EventBase {\n type: EventType.TOOL_CALL_DELTA;\n reply_id: string;\n tool_call_id: string;\n delta: string;\n}\n\nexport interface ToolCallEndEvent extends EventBase {\n type: EventType.TOOL_CALL_END;\n reply_id: string;\n tool_call_id: string;\n}\n\nexport interface ToolResultStartEvent extends EventBase {\n type: EventType.TOOL_RESULT_START;\n reply_id: string;\n tool_call_id: string;\n tool_call_name: string;\n}\n\nexport interface ToolResultTextDeltaEvent extends EventBase {\n type: EventType.TOOL_RESULT_TEXT_DELTA;\n reply_id: string;\n tool_call_id: string;\n delta: string;\n}\n\nexport interface ToolResultDataDeltaEvent extends EventBase {\n type: EventType.TOOL_RESULT_DATA_DELTA;\n reply_id: string;\n tool_call_id: string;\n /** Auto-generated in {@link appendEvent} when not provided. */\n block_id?: string;\n media_type: string;\n data?: string;\n url?: string;\n}\n\nexport interface ToolResultEndEvent extends EventBase {\n type: EventType.TOOL_RESULT_END;\n reply_id: string;\n tool_call_id: string;\n state: ToolResultBlock['state'];\n}\n\nexport interface ExceedMaxItersEvent extends EventBase {\n type: EventType.EXCEED_MAX_ITERS;\n reply_id: string;\n name: string;\n}\n\nexport interface RequireUserConfirmEvent extends EventBase {\n type: EventType.REQUIRE_USER_CONFIRM;\n reply_id: string;\n tool_calls: ToolCallBlock[];\n}\n\nexport interface RequireExternalExecutionEvent extends EventBase {\n type: EventType.REQUIRE_EXTERNAL_EXECUTION;\n reply_id: string;\n tool_calls: ToolCallBlock[];\n}\n\nexport interface ConfirmResult {\n confirmed: boolean;\n tool_call: ToolCallBlock;\n rules?: PermissionRule[] | null;\n}\n\nexport interface UserConfirmResultEvent extends EventBase {\n type: EventType.USER_CONFIRM_RESULT;\n reply_id: string;\n confirm_results: ConfirmResult[];\n}\n\nexport interface ExternalExecutionResultEvent extends EventBase {\n type: EventType.EXTERNAL_EXECUTION_RESULT;\n reply_id: string;\n execution_results: ToolResultBlock[];\n}\n\nexport type AgentEvent =\n // The control events for the whole run\n | ReplyStartEvent\n | ReplyEndEvent\n | ExceedMaxItersEvent\n | RequireUserConfirmEvent\n | RequireExternalExecutionEvent\n | ModelCallStartEvent\n | ModelCallEndEvent\n // The data events for different block types\n | TextBlockStartEvent\n | TextBlockDeltaEvent\n | TextBlockEndEvent\n | DataBlockStartEvent\n | DataBlockDeltaEvent\n | DataBlockEndEvent\n | ThinkingBlockStartEvent\n | ThinkingBlockDeltaEvent\n | ThinkingBlockEndEvent\n | ToolCallStartEvent\n | ToolCallDeltaEvent\n | ToolCallEndEvent\n | ToolResultStartEvent\n | ToolResultTextDeltaEvent\n | ToolResultDataDeltaEvent\n | ToolResultEndEvent\n // The events from the external execution or user confirmation\n | UserConfirmResultEvent\n | ExternalExecutionResultEvent;\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAGO,IAAK,YAAL,kBAAKA,eAAL;AACH,EAAAA,WAAA,iBAAc;AACd,EAAAA,WAAA,eAAY;AAEZ,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,wBAAqB;AAErB,EAAAA,WAAA,qBAAkB;AAClB,EAAAA,WAAA,qBAAkB;AAClB,EAAAA,WAAA,mBAAgB;AAEhB,EAAAA,WAAA,uBAAoB;AACpB,EAAAA,WAAA,4BAAyB;AACzB,EAAAA,WAAA,4BAAyB;AACzB,EAAAA,WAAA,qBAAkB;AAElB,EAAAA,WAAA,sBAAmB;AAEnB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,gCAA6B;AAE7B,EAAAA,WAAA,yBAAsB;AACtB,EAAAA,WAAA,+BAA4B;AAlCpB,SAAAA;AAAA,GAAA;","names":["EventType"]}
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/event/index.ts"],"sourcesContent":["import { ToolCallBlock, ToolResultBlock } from '../message';\nimport { PermissionRule } from '../permission';\n\nexport enum EventType {\n REPLY_START = 'REPLY_START',\n REPLY_END = 'REPLY_END',\n\n MODEL_CALL_START = 'MODEL_CALL_START',\n MODEL_CALL_END = 'MODEL_CALL_END',\n\n TEXT_BLOCK_START = 'TEXT_BLOCK_START',\n TEXT_BLOCK_DELTA = 'TEXT_BLOCK_DELTA',\n TEXT_BLOCK_END = 'TEXT_BLOCK_END',\n\n DATA_BLOCK_START = 'DATA_BLOCK_START',\n DATA_BLOCK_DELTA = 'DATA_BLOCK_DELTA',\n DATA_BLOCK_END = 'DATA_BLOCK_END',\n\n THINKING_BLOCK_START = 'THINKING_BLOCK_START',\n THINKING_BLOCK_DELTA = 'THINKING_BLOCK_DELTA',\n THINKING_BLOCK_END = 'THINKING_BLOCK_END',\n\n TOOL_CALL_START = 'TOOL_CALL_START',\n TOOL_CALL_DELTA = 'TOOL_CALL_DELTA',\n TOOL_CALL_END = 'TOOL_CALL_END',\n\n TOOL_RESULT_START = 'TOOL_RESULT_START',\n TOOL_RESULT_TEXT_DELTA = 'TOOL_RESULT_TEXT_DELTA',\n TOOL_RESULT_DATA_DELTA = 'TOOL_RESULT_DATA_DELTA',\n TOOL_RESULT_END = 'TOOL_RESULT_END',\n\n EXCEED_MAX_ITERS = 'EXCEED_MAX_ITERS',\n\n REQUIRE_USER_CONFIRM = 'REQUIRE_USER_CONFIRM',\n REQUIRE_EXTERNAL_EXECUTION = 'REQUIRE_EXTERNAL_EXECUTION',\n\n USER_CONFIRM_RESULT = 'USER_CONFIRM_RESULT',\n EXTERNAL_EXECUTION_RESULT = 'EXTERNAL_EXECUTION_RESULT',\n}\n\nexport interface EventBase {\n id: string;\n created_at: string;\n}\n\nexport interface ReplyStartEvent extends EventBase {\n type: EventType.REPLY_START;\n session_id: string;\n reply_id: string;\n name: string;\n role: 'user' | 'assistant' | 'system';\n}\n\nexport interface ReplyEndEvent extends EventBase {\n type: EventType.REPLY_END;\n session_id: string;\n reply_id: string;\n}\n\nexport interface ModelCallStartEvent extends EventBase {\n type: EventType.MODEL_CALL_START;\n reply_id: string;\n model_name: string;\n}\n\nexport interface ModelCallEndEvent extends EventBase {\n type: EventType.MODEL_CALL_END;\n reply_id: string;\n input_tokens: number;\n output_tokens: number;\n}\n\nexport interface TextBlockStartEvent extends EventBase {\n type: EventType.TEXT_BLOCK_START;\n block_id: string;\n reply_id: string;\n}\n\nexport interface TextBlockDeltaEvent extends EventBase {\n type: EventType.TEXT_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface TextBlockEndEvent extends EventBase {\n type: EventType.TEXT_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface DataBlockStartEvent extends EventBase {\n type: EventType.DATA_BLOCK_START;\n reply_id: string;\n block_id: string;\n media_type: string;\n}\n\nexport interface DataBlockDeltaEvent extends EventBase {\n type: EventType.DATA_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n data: string;\n media_type: string;\n}\n\nexport interface DataBlockEndEvent extends EventBase {\n type: EventType.DATA_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ThinkingBlockStartEvent extends EventBase {\n type: EventType.THINKING_BLOCK_START;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ThinkingBlockDeltaEvent extends EventBase {\n type: EventType.THINKING_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface ThinkingBlockEndEvent extends EventBase {\n type: EventType.THINKING_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ToolCallStartEvent extends EventBase {\n type: EventType.TOOL_CALL_START;\n reply_id: string;\n tool_call_id: string;\n tool_call_name: string;\n}\n\nexport interface ToolCallDeltaEvent extends EventBase {\n type: EventType.TOOL_CALL_DELTA;\n reply_id: string;\n tool_call_id: string;\n delta: string;\n}\n\nexport interface ToolCallEndEvent extends EventBase {\n type: EventType.TOOL_CALL_END;\n reply_id: string;\n tool_call_id: string;\n}\n\nexport interface ToolResultStartEvent extends EventBase {\n type: EventType.TOOL_RESULT_START;\n reply_id: string;\n tool_call_id: string;\n tool_call_name: string;\n}\n\nexport interface ToolResultTextDeltaEvent extends EventBase {\n type: EventType.TOOL_RESULT_TEXT_DELTA;\n reply_id: string;\n tool_call_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface ToolResultDataDeltaEvent extends EventBase {\n type: EventType.TOOL_RESULT_DATA_DELTA;\n reply_id: string;\n tool_call_id: string;\n block_id: string;\n media_type: string;\n data?: string;\n url?: string;\n}\n\nexport interface ToolResultEndEvent extends EventBase {\n type: EventType.TOOL_RESULT_END;\n reply_id: string;\n tool_call_id: string;\n state: ToolResultBlock['state'];\n}\n\nexport interface ExceedMaxItersEvent extends EventBase {\n type: EventType.EXCEED_MAX_ITERS;\n reply_id: string;\n name: string;\n}\n\nexport interface RequireUserConfirmEvent extends EventBase {\n type: EventType.REQUIRE_USER_CONFIRM;\n reply_id: string;\n tool_calls: ToolCallBlock[];\n}\n\nexport interface RequireExternalExecutionEvent extends EventBase {\n type: EventType.REQUIRE_EXTERNAL_EXECUTION;\n reply_id: string;\n tool_calls: ToolCallBlock[];\n}\n\nexport interface ConfirmResult {\n confirmed: boolean;\n tool_call: ToolCallBlock;\n rules?: PermissionRule[] | null;\n}\n\nexport interface UserConfirmResultEvent extends EventBase {\n type: EventType.USER_CONFIRM_RESULT;\n reply_id: string;\n confirm_results: ConfirmResult[];\n}\n\nexport interface ExternalExecutionResultEvent extends EventBase {\n type: EventType.EXTERNAL_EXECUTION_RESULT;\n reply_id: string;\n execution_results: ToolResultBlock[];\n}\n\nexport type AgentEvent =\n // The control events for the whole run\n | ReplyStartEvent\n | ReplyEndEvent\n | ExceedMaxItersEvent\n | RequireUserConfirmEvent\n | RequireExternalExecutionEvent\n | ModelCallStartEvent\n | ModelCallEndEvent\n // The data events for different block types\n | TextBlockStartEvent\n | TextBlockDeltaEvent\n | TextBlockEndEvent\n | DataBlockStartEvent\n | DataBlockDeltaEvent\n | DataBlockEndEvent\n | ThinkingBlockStartEvent\n | ThinkingBlockDeltaEvent\n | ThinkingBlockEndEvent\n | ToolCallStartEvent\n | ToolCallDeltaEvent\n | ToolCallEndEvent\n | ToolResultStartEvent\n | ToolResultTextDeltaEvent\n | ToolResultDataDeltaEvent\n | ToolResultEndEvent\n // The events from the external execution or user confirmation\n | UserConfirmResultEvent\n | ExternalExecutionResultEvent;\n"],"mappings":";AAGO,IAAK,YAAL,kBAAKA,eAAL;AACH,EAAAA,WAAA,iBAAc;AACd,EAAAA,WAAA,eAAY;AAEZ,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,wBAAqB;AAErB,EAAAA,WAAA,qBAAkB;AAClB,EAAAA,WAAA,qBAAkB;AAClB,EAAAA,WAAA,mBAAgB;AAEhB,EAAAA,WAAA,uBAAoB;AACpB,EAAAA,WAAA,4BAAyB;AACzB,EAAAA,WAAA,4BAAyB;AACzB,EAAAA,WAAA,qBAAkB;AAElB,EAAAA,WAAA,sBAAmB;AAEnB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,gCAA6B;AAE7B,EAAAA,WAAA,yBAAsB;AACtB,EAAAA,WAAA,+BAA4B;AAlCpB,SAAAA;AAAA,GAAA;","names":["EventType"]}
1
+ {"version":3,"sources":["../../src/event/index.ts"],"sourcesContent":["import { ToolCallBlock, ToolResultBlock } from '../message';\nimport { PermissionRule } from '../permission';\n\nexport enum EventType {\n REPLY_START = 'REPLY_START',\n REPLY_END = 'REPLY_END',\n\n MODEL_CALL_START = 'MODEL_CALL_START',\n MODEL_CALL_END = 'MODEL_CALL_END',\n\n TEXT_BLOCK_START = 'TEXT_BLOCK_START',\n TEXT_BLOCK_DELTA = 'TEXT_BLOCK_DELTA',\n TEXT_BLOCK_END = 'TEXT_BLOCK_END',\n\n DATA_BLOCK_START = 'DATA_BLOCK_START',\n DATA_BLOCK_DELTA = 'DATA_BLOCK_DELTA',\n DATA_BLOCK_END = 'DATA_BLOCK_END',\n\n THINKING_BLOCK_START = 'THINKING_BLOCK_START',\n THINKING_BLOCK_DELTA = 'THINKING_BLOCK_DELTA',\n THINKING_BLOCK_END = 'THINKING_BLOCK_END',\n\n TOOL_CALL_START = 'TOOL_CALL_START',\n TOOL_CALL_DELTA = 'TOOL_CALL_DELTA',\n TOOL_CALL_END = 'TOOL_CALL_END',\n\n TOOL_RESULT_START = 'TOOL_RESULT_START',\n TOOL_RESULT_TEXT_DELTA = 'TOOL_RESULT_TEXT_DELTA',\n TOOL_RESULT_DATA_DELTA = 'TOOL_RESULT_DATA_DELTA',\n TOOL_RESULT_END = 'TOOL_RESULT_END',\n\n EXCEED_MAX_ITERS = 'EXCEED_MAX_ITERS',\n\n REQUIRE_USER_CONFIRM = 'REQUIRE_USER_CONFIRM',\n REQUIRE_EXTERNAL_EXECUTION = 'REQUIRE_EXTERNAL_EXECUTION',\n\n USER_CONFIRM_RESULT = 'USER_CONFIRM_RESULT',\n EXTERNAL_EXECUTION_RESULT = 'EXTERNAL_EXECUTION_RESULT',\n}\n\nexport interface EventBase {\n id: string;\n created_at: string;\n}\n\nexport interface ReplyStartEvent extends EventBase {\n type: EventType.REPLY_START;\n session_id: string;\n reply_id: string;\n name: string;\n role: 'user' | 'assistant' | 'system';\n}\n\nexport interface ReplyEndEvent extends EventBase {\n type: EventType.REPLY_END;\n session_id: string;\n reply_id: string;\n}\n\nexport interface ModelCallStartEvent extends EventBase {\n type: EventType.MODEL_CALL_START;\n reply_id: string;\n model_name: string;\n}\n\nexport interface ModelCallEndEvent extends EventBase {\n type: EventType.MODEL_CALL_END;\n reply_id: string;\n input_tokens: number;\n output_tokens: number;\n}\n\nexport interface TextBlockStartEvent extends EventBase {\n type: EventType.TEXT_BLOCK_START;\n block_id: string;\n reply_id: string;\n}\n\nexport interface TextBlockDeltaEvent extends EventBase {\n type: EventType.TEXT_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface TextBlockEndEvent extends EventBase {\n type: EventType.TEXT_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface DataBlockStartEvent extends EventBase {\n type: EventType.DATA_BLOCK_START;\n reply_id: string;\n block_id: string;\n media_type: string;\n}\n\nexport interface DataBlockDeltaEvent extends EventBase {\n type: EventType.DATA_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n data: string;\n media_type: string;\n}\n\nexport interface DataBlockEndEvent extends EventBase {\n type: EventType.DATA_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ThinkingBlockStartEvent extends EventBase {\n type: EventType.THINKING_BLOCK_START;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ThinkingBlockDeltaEvent extends EventBase {\n type: EventType.THINKING_BLOCK_DELTA;\n reply_id: string;\n block_id: string;\n delta: string;\n}\n\nexport interface ThinkingBlockEndEvent extends EventBase {\n type: EventType.THINKING_BLOCK_END;\n reply_id: string;\n block_id: string;\n}\n\nexport interface ToolCallStartEvent extends EventBase {\n type: EventType.TOOL_CALL_START;\n reply_id: string;\n tool_call_id: string;\n tool_call_name: string;\n}\n\nexport interface ToolCallDeltaEvent extends EventBase {\n type: EventType.TOOL_CALL_DELTA;\n reply_id: string;\n tool_call_id: string;\n delta: string;\n}\n\nexport interface ToolCallEndEvent extends EventBase {\n type: EventType.TOOL_CALL_END;\n reply_id: string;\n tool_call_id: string;\n}\n\nexport interface ToolResultStartEvent extends EventBase {\n type: EventType.TOOL_RESULT_START;\n reply_id: string;\n tool_call_id: string;\n tool_call_name: string;\n}\n\nexport interface ToolResultTextDeltaEvent extends EventBase {\n type: EventType.TOOL_RESULT_TEXT_DELTA;\n reply_id: string;\n tool_call_id: string;\n delta: string;\n}\n\nexport interface ToolResultDataDeltaEvent extends EventBase {\n type: EventType.TOOL_RESULT_DATA_DELTA;\n reply_id: string;\n tool_call_id: string;\n /** Auto-generated in {@link appendEvent} when not provided. */\n block_id?: string;\n media_type: string;\n data?: string;\n url?: string;\n}\n\nexport interface ToolResultEndEvent extends EventBase {\n type: EventType.TOOL_RESULT_END;\n reply_id: string;\n tool_call_id: string;\n state: ToolResultBlock['state'];\n}\n\nexport interface ExceedMaxItersEvent extends EventBase {\n type: EventType.EXCEED_MAX_ITERS;\n reply_id: string;\n name: string;\n}\n\nexport interface RequireUserConfirmEvent extends EventBase {\n type: EventType.REQUIRE_USER_CONFIRM;\n reply_id: string;\n tool_calls: ToolCallBlock[];\n}\n\nexport interface RequireExternalExecutionEvent extends EventBase {\n type: EventType.REQUIRE_EXTERNAL_EXECUTION;\n reply_id: string;\n tool_calls: ToolCallBlock[];\n}\n\nexport interface ConfirmResult {\n confirmed: boolean;\n tool_call: ToolCallBlock;\n rules?: PermissionRule[] | null;\n}\n\nexport interface UserConfirmResultEvent extends EventBase {\n type: EventType.USER_CONFIRM_RESULT;\n reply_id: string;\n confirm_results: ConfirmResult[];\n}\n\nexport interface ExternalExecutionResultEvent extends EventBase {\n type: EventType.EXTERNAL_EXECUTION_RESULT;\n reply_id: string;\n execution_results: ToolResultBlock[];\n}\n\nexport type AgentEvent =\n // The control events for the whole run\n | ReplyStartEvent\n | ReplyEndEvent\n | ExceedMaxItersEvent\n | RequireUserConfirmEvent\n | RequireExternalExecutionEvent\n | ModelCallStartEvent\n | ModelCallEndEvent\n // The data events for different block types\n | TextBlockStartEvent\n | TextBlockDeltaEvent\n | TextBlockEndEvent\n | DataBlockStartEvent\n | DataBlockDeltaEvent\n | DataBlockEndEvent\n | ThinkingBlockStartEvent\n | ThinkingBlockDeltaEvent\n | ThinkingBlockEndEvent\n | ToolCallStartEvent\n | ToolCallDeltaEvent\n | ToolCallEndEvent\n | ToolResultStartEvent\n | ToolResultTextDeltaEvent\n | ToolResultDataDeltaEvent\n | ToolResultEndEvent\n // The events from the external execution or user confirmation\n | UserConfirmResultEvent\n | ExternalExecutionResultEvent;\n"],"mappings":";AAGO,IAAK,YAAL,kBAAKA,eAAL;AACH,EAAAA,WAAA,iBAAc;AACd,EAAAA,WAAA,eAAY;AAEZ,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,sBAAmB;AACnB,EAAAA,WAAA,oBAAiB;AAEjB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,wBAAqB;AAErB,EAAAA,WAAA,qBAAkB;AAClB,EAAAA,WAAA,qBAAkB;AAClB,EAAAA,WAAA,mBAAgB;AAEhB,EAAAA,WAAA,uBAAoB;AACpB,EAAAA,WAAA,4BAAyB;AACzB,EAAAA,WAAA,4BAAyB;AACzB,EAAAA,WAAA,qBAAkB;AAElB,EAAAA,WAAA,sBAAmB;AAEnB,EAAAA,WAAA,0BAAuB;AACvB,EAAAA,WAAA,gCAA6B;AAE7B,EAAAA,WAAA,yBAAsB;AACtB,EAAAA,WAAA,+BAA4B;AAlCpB,SAAAA;AAAA,GAAA;","names":["EventType"]}
@@ -1,5 +1,5 @@
1
- import { F as FormatterBase } from '../base-BI5s2ksj.mjs';
2
- import { M as Msg } from '../message-D-LObC06.mjs';
1
+ import { F as FormatterBase } from '../base-Bc3GkNS7.mjs';
2
+ import { M as Msg } from '../message-COpNEf0G.mjs';
3
3
  import { T as TextBlock, D as DataBlock } from '../block-BqWf-Qcb.mjs';
4
4
  import '../index-CAxQAkiP.mjs';
5
5
  import '../event/index.mjs';
@@ -1,5 +1,5 @@
1
- import { F as FormatterBase } from '../base-CFDeoJRe.js';
2
- import { M as Msg } from '../message-DU0_qm3u.js';
1
+ import { F as FormatterBase } from '../base-Dfizi3RB.js';
2
+ import { M as Msg } from '../message-DbCMy5tM.js';
3
3
  import { T as TextBlock, D as DataBlock } from '../block-BqWf-Qcb.js';
4
4
  import '../index-CAxQAkiP.js';
5
5
  import '../event/index.js';
@@ -29,6 +29,21 @@ __export(formatter_exports, {
29
29
  module.exports = __toCommonJS(formatter_exports);
30
30
 
31
31
  // src/message/message.ts
32
+ function assertContentBlocksForRole(role, content) {
33
+ if (role === "user") {
34
+ for (const block of content) {
35
+ if (block.type !== "text" && block.type !== "data") {
36
+ throw new Error("User message can only contain text blocks or data blocks.");
37
+ }
38
+ }
39
+ } else if (role === "system") {
40
+ for (const block of content) {
41
+ if (block.type !== "text") {
42
+ throw new Error("System message can only contain text blocks.");
43
+ }
44
+ }
45
+ }
46
+ }
32
47
  function createMsg({
33
48
  name,
34
49
  content,
@@ -40,6 +55,7 @@ function createMsg({
40
55
  usage
41
56
  }) {
42
57
  const contentBlocks = typeof content === "string" ? [{ id: crypto.randomUUID(), type: "text", text: content }] : content;
58
+ assertContentBlocksForRole(role, contentBlocks);
43
59
  return { id, name, role, content: contentBlocks, metadata, created_at, finished_at, usage };
44
60
  }
45
61
  function getTextContent(msg, separator = "\n") {
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/formatter/index.ts","../../src/message/message.ts","../../src/formatter/base.ts","../../src/formatter/dashscope-chat-formatter.ts","../../src/formatter/deepseek-chat-formatter.ts","../../src/formatter/ollama-chat-formatter.ts","../../src/formatter/openai-chat-formatter.ts"],"sourcesContent":["export { FormatterBase } from './base';\nexport { DashScopeChatFormatter } from './dashscope-chat-formatter';\nexport { DeepSeekChatFormatter } from './deepseek-chat-formatter';\nexport { OllamaChatFormatter } from './ollama-chat-formatter';\nexport { OpenAIChatFormatter } from './openai-chat-formatter';\n","import { JSONSerializableObject } from '../type';\nimport {\n ContentBlock,\n TextBlock,\n ThinkingBlock,\n ToolResultBlock,\n ToolCallBlock,\n DataBlock,\n Base64Source,\n URLSource,\n} from './block';\nimport { AgentEvent, EventType } from '../event';\n\n/** A chat message exchanged between agents or between an agent and a model. */\nexport interface Msg {\n /** Unique identifier for the message. */\n id: string;\n /** Display name of the message sender. */\n name: string;\n /** Conversation role of the sender. */\n role: 'user' | 'assistant' | 'system';\n /** Message body. */\n content: ContentBlock[];\n /** Arbitrary key-value metadata attached to the message. */\n metadata: Record<string, JSONSerializableObject>;\n /** ISO-8601 creation timestamp. */\n created_at: string;\n /** ISO-8601 finished timestamp. */\n finished_at?: string | null;\n /** Usage information for the message, such as token counts. */\n usage?: {\n inputTokens: number;\n outputTokens: number;\n };\n}\n\n/**\n * Create a new {@link Msg} object, filling in `id` and `created_at` when omitted.\n * A plain string `content` is automatically wrapped in a single {@link TextBlock}.\n * @param root0\n * @param root0.name\n * @param root0.content\n * @param root0.role\n * @param root0.metadata\n * @param root0.id\n * @param root0.created_at\n * @param root0.finished_at\n * @param root0.usage\n * @returns A Msg object.\n */\nexport function createMsg({\n name,\n content,\n role,\n metadata = {},\n id = crypto.randomUUID(),\n created_at = new Date().toISOString(),\n finished_at,\n usage,\n}: Omit<Msg, 'id' | 'created_at' | 'metadata' | 'content'> &\n Partial<Pick<Msg, 'id' | 'created_at' | 'metadata'>> & {\n content: string | ContentBlock[];\n }): Msg {\n const contentBlocks: ContentBlock[] =\n typeof content === 'string'\n ? [{ id: crypto.randomUUID(), type: 'text', text: content } as TextBlock]\n : content;\n return { id, name, role, content: contentBlocks, metadata, created_at, finished_at, usage };\n}\n\n/**\n * Create a user {@link Msg}.\n * @param root0\n * @param root0.name\n * @param root0.content\n * @param root0.metadata\n * @param root0.id\n * @param root0.created_at\n * @returns A Msg object with role 'user'.\n */\nexport function UserMsg({\n name,\n content,\n metadata = {},\n id = crypto.randomUUID(),\n created_at = new Date().toISOString(),\n}: {\n name: string;\n content: string | ContentBlock[];\n metadata?: Record<string, JSONSerializableObject>;\n id?: string;\n created_at?: string;\n}): Msg {\n return createMsg({ name, content, role: 'user', metadata, id, created_at });\n}\n\n/**\n * Create an assistant {@link Msg}.\n * @param root0\n * @param root0.name\n * @param root0.content\n * @param root0.metadata\n * @param root0.id\n * @param root0.created_at\n * @param root0.usage\n * @returns A Msg object with role 'assistant'.\n */\nexport function AssistantMsg({\n name,\n content,\n metadata = {},\n id = crypto.randomUUID(),\n created_at = new Date().toISOString(),\n usage,\n}: {\n name: string;\n content: string | ContentBlock[];\n metadata?: Record<string, JSONSerializableObject>;\n id?: string;\n created_at?: string;\n usage?: Msg['usage'];\n}): Msg {\n return createMsg({ name, content, role: 'assistant', metadata, id, created_at, usage });\n}\n\n/**\n * Create a system {@link Msg}.\n * @param root0\n * @param root0.name\n * @param root0.content\n * @param root0.metadata\n * @param root0.id\n * @param root0.created_at\n * @returns A Msg object with role 'system'.\n */\nexport function SystemMsg({\n name,\n content,\n metadata = {},\n id = crypto.randomUUID(),\n created_at = new Date().toISOString(),\n}: {\n name: string;\n content: string | ContentBlock[];\n metadata?: Record<string, JSONSerializableObject>;\n id?: string;\n created_at?: string;\n}): Msg {\n return createMsg({ name, content, role: 'system', metadata, id, created_at });\n}\n\n/**\n * Extract the plain-text content from a message.\n *\n * When `content` is a string it is returned as-is. When it is an array of\n * content blocks, all {@link TextBlock} texts are joined with `separator`.\n *\n * @param msg - The message to read.\n * @param separator - String inserted between consecutive text blocks. Defaults to `'\\n'`.\n * @returns The concatenated text, or `null` when no text blocks are present.\n */\nexport function getTextContent(msg: Msg, separator: string = '\\n'): string | null {\n const textBlocks = msg.content.filter(block => block.type === 'text');\n if (textBlocks.length === 0) return null;\n return textBlocks.map(block => (block as TextBlock).text).join(separator);\n}\n\n/**\n * Return all content blocks from a message, regardless of type.\n *\n * When `content` is a plain string it is wrapped in a single {@link TextBlock}.\n *\n * @param msg - The message to read.\n * @returns An array of all {@link ContentBlock} objects.\n */\nexport function getContentBlocks(msg: Msg): ContentBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'text'): TextBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'thinking'): ThinkingBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'data'): DataBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'tool_call'): ToolCallBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'tool_result'): ToolResultBlock[];\nexport function getContentBlocks(\n msg: Msg,\n blockType?: 'text' | 'thinking' | 'data' | 'tool_call' | 'tool_result'\n): ContentBlock[] {\n if (!blockType) return msg.content;\n return msg.content.filter(block => block.type === blockType);\n}\n\n/**\n * Find a content block by type and id within a message.\n * @param msg\n * @param blockType\n * @param blockId\n * @returns The matching {@link ContentBlock}, or `undefined` if not found.\n */\nfunction findBlock(msg: Msg, blockType: string, blockId: string): ContentBlock | undefined {\n return msg.content.find(block => block.type === blockType && block.id === blockId);\n}\n\n/**\n * Apply a streaming {@link AgentEvent} to a {@link Msg}, mutating it in place.\n *\n * Only `content` and `finished_at` are ever modified. Events whose\n * `reply_id` does not match `msg.id` are skipped with a warning.\n * @param msg\n * @param event\n * @returns The mutated {@link Msg} object.\n */\nexport function appendEvent(msg: Msg, event: AgentEvent): Msg {\n if (!('reply_id' in event)) return msg;\n if (event.reply_id !== msg.id) {\n console.warn(\n `Event reply_id \"${event.reply_id}\" does not match message id \"${msg.id}\", skipping.`\n );\n return msg;\n }\n\n switch (event.type) {\n case EventType.REPLY_END:\n msg.finished_at = event.created_at;\n break;\n\n case EventType.TEXT_BLOCK_START:\n msg.content.push({ type: 'text', id: event.block_id, text: '' });\n break;\n\n case EventType.TEXT_BLOCK_DELTA: {\n const block = findBlock(msg, 'text', event.block_id);\n if (!block) {\n console.warn(`TextBlock \"${event.block_id}\" not found, skipping.`);\n } else {\n (block as TextBlock).text += event.delta;\n }\n break;\n }\n\n case EventType.TEXT_BLOCK_END:\n break;\n\n case EventType.THINKING_BLOCK_START:\n msg.content.push({ type: 'thinking', id: event.block_id, thinking: '' });\n break;\n\n case EventType.THINKING_BLOCK_DELTA: {\n const block = findBlock(msg, 'thinking', event.block_id);\n if (!block) {\n console.warn(`ThinkingBlock \"${event.block_id}\" not found, skipping.`);\n } else {\n (block as ThinkingBlock).thinking += event.delta;\n }\n break;\n }\n\n case EventType.THINKING_BLOCK_END:\n break;\n\n case EventType.DATA_BLOCK_START:\n msg.content.push({\n type: 'data',\n id: event.block_id,\n source: { type: 'base64', data: '', media_type: event.media_type },\n });\n break;\n\n case EventType.DATA_BLOCK_DELTA: {\n const block = findBlock(msg, 'data', event.block_id);\n if (!block) {\n console.warn(`DataBlock \"${event.block_id}\" not found, skipping.`);\n } else {\n ((block as DataBlock).source as Base64Source).data += event.data;\n }\n break;\n }\n\n case EventType.DATA_BLOCK_END:\n break;\n\n case EventType.TOOL_CALL_START:\n msg.content.push({\n type: 'tool_call',\n id: event.tool_call_id,\n name: event.tool_call_name,\n input: '',\n state: 'pending',\n });\n break;\n\n case EventType.TOOL_CALL_DELTA: {\n const block = findBlock(msg, 'tool_call', event.tool_call_id);\n if (!block) {\n console.warn(`ToolCallBlock \"${event.tool_call_id}\" not found, skipping.`);\n } else {\n (block as ToolCallBlock).input += event.delta;\n }\n break;\n }\n\n case EventType.TOOL_CALL_END:\n break;\n\n case EventType.TOOL_RESULT_START:\n msg.content.push({\n type: 'tool_result',\n id: event.tool_call_id,\n name: event.tool_call_name,\n output: [],\n state: 'running',\n });\n break;\n\n case EventType.TOOL_RESULT_TEXT_DELTA: {\n const block = findBlock(msg, 'tool_result', event.tool_call_id);\n if (!block) {\n console.warn(`ToolResultBlock \"${event.tool_call_id}\" not found, skipping.`);\n } else {\n const trb = block as ToolResultBlock;\n if (typeof trb.output === 'string') {\n trb.output = [{ type: 'text', id: crypto.randomUUID(), text: trb.output }];\n }\n const last = trb.output[trb.output.length - 1];\n if (!last || last.type !== 'text') {\n trb.output.push({\n type: 'text',\n id: event.block_id ?? crypto.randomUUID(),\n text: event.delta,\n });\n } else {\n (last as TextBlock).text += event.delta;\n }\n }\n break;\n }\n\n case EventType.TOOL_RESULT_DATA_DELTA: {\n const block = findBlock(msg, 'tool_result', event.tool_call_id);\n if (!block) {\n console.warn(`ToolResultBlock \"${event.tool_call_id}\" not found, skipping.`);\n } else {\n const trb = block as ToolResultBlock;\n if (typeof trb.output === 'string') {\n trb.output = [{ type: 'text', id: crypto.randomUUID(), text: trb.output }];\n }\n const source: Base64Source | URLSource =\n event.data != null\n ? { type: 'base64', data: event.data, media_type: event.media_type }\n : { type: 'url', url: event.url!, media_type: event.media_type };\n trb.output.push({ type: 'data', id: event.block_id, source });\n }\n break;\n }\n\n case EventType.TOOL_RESULT_END: {\n const block = findBlock(msg, 'tool_result', event.tool_call_id);\n if (!block) {\n console.warn(`ToolResultBlock \"${event.tool_call_id}\" not found, skipping.`);\n } else {\n (block as ToolResultBlock).state = event.state;\n }\n break;\n }\n\n case EventType.MODEL_CALL_END:\n // Accumulated the input and output tokens here.\n if (msg.usage) {\n msg.usage.inputTokens += event.input_tokens;\n msg.usage.outputTokens += event.output_tokens;\n } else {\n msg.usage = {\n inputTokens: event.input_tokens,\n outputTokens: event.output_tokens,\n };\n }\n break;\n\n case EventType.REQUIRE_USER_CONFIRM:\n for (const tc of event.tool_calls) {\n const b = findBlock(msg, 'tool_call', tc.id);\n if (b) {\n (b as ToolCallBlock).state = 'asking';\n if (tc.suggested_rules !== undefined) {\n (b as ToolCallBlock).suggested_rules = tc.suggested_rules;\n }\n }\n }\n break;\n\n case EventType.USER_CONFIRM_RESULT:\n for (const result of event.confirm_results) {\n const b = findBlock(msg, 'tool_call', result.tool_call.id);\n if (b) {\n (b as ToolCallBlock).state = result.confirmed ? 'allowed' : 'finished';\n }\n }\n break;\n\n case EventType.REQUIRE_EXTERNAL_EXECUTION:\n for (const tc of event.tool_calls) {\n const b = findBlock(msg, 'tool_call', tc.id);\n if (b) (b as ToolCallBlock).state = 'submitted';\n }\n break;\n\n case EventType.EXTERNAL_EXECUTION_RESULT:\n for (const result of event.execution_results) {\n msg.content.push(result);\n }\n break;\n }\n\n return msg;\n}\n","import { Msg, TextBlock, DataBlock, createMsg } from '../message';\n\n/**\n * Base class for message formatters.\n */\nexport abstract class FormatterBase {\n /**\n * Format the input message objects into the required format by the API.\n *\n * @param root0\n * @param root0.msgs - An array of message objects to be formatted.\n * @returns A promise that resolves to an array of formatted message objects.\n */\n abstract format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]>;\n\n /**\n * Convert the tool output to string format for the LLM APIs that only accept text input. If\n * `promoteMultimodalToolResult` is true, the multimodal content will be promoted to be a user message with\n * \"<system-info></system-info>\" tags. Otherwise, the multimodal content will be saved to a storage and a URL link\n * will be provided in the text output.\n *\n * @param output - The tool output, which can be a string or an array of content blocks.\n * @param promoteMultimodalToolResult - Whether to promote the multimodal content to the prompt messages.\n * @returns An object containing the text output and an optional promoted message.\n */\n convertToolOutputToString(\n output: string | (TextBlock | DataBlock)[],\n promoteMultimodalToolResult: boolean | { image?: boolean; audio?: boolean; video?: boolean }\n ) {\n if (typeof output === 'string') return { text: output, promotedMsg: null };\n\n let textualOutput = [];\n\n const promotedData: { id: string; block: DataBlock }[] = [];\n\n for (const block of output) {\n switch (block.type) {\n case 'text':\n textualOutput.push(block.text);\n break;\n default:\n const type = block.source.media_type.split('/')[0];\n if (type !== 'image' && type !== 'audio' && type !== 'video') {\n console.log(\n `Unsupported media type '${block.source.media_type}' in tool output. Only image, audio and video are supported.`\n );\n break;\n }\n if (block.source.type === 'url') {\n textualOutput.push(\n `<system-info>One returned ${type} can be found at: ${block.source.url}</system-info>`\n );\n } else {\n // If we should promote the multimodal content to the prompt messages\n const shouldPromote =\n promoteMultimodalToolResult === true ||\n (typeof promoteMultimodalToolResult === 'object' &&\n promoteMultimodalToolResult[type]);\n\n if (shouldPromote) {\n // Create an ID for the multimodal content first, which should less than 10 characters\n const dataID = Math.random().toString(36).substring(2, 10);\n textualOutput.push(\n `<system-info>One returned ${type} is embedded with ID '${dataID}' and will be attached within '<system-info></system-info>' tags later.</system-info>`\n );\n\n // Record the promoted data\n promotedData.push({ id: dataID, block });\n } else {\n // TODO: save locally\n\n // Save to storage and provide URL link\n textualOutput.push(`The returned ${block.type} is stored locally.`);\n }\n }\n }\n }\n\n // Attach prefix and suffix system-info tags if there are promoted blocks\n const promotedBlocks: (TextBlock | DataBlock)[] = [];\n promotedData.forEach(({ id, block }) => {\n const type = block.source.media_type.split('/')[0];\n promotedBlocks.push({\n id: crypto.randomUUID(),\n type: 'text',\n text: `<${type}_data id='${id}'>`,\n });\n promotedBlocks.push(block);\n promotedBlocks.push({\n id: crypto.randomUUID(),\n type: 'text',\n text: `</${type}_data>\\n`,\n });\n });\n\n if (promotedBlocks.length > 0) {\n // The prefix\n const prefix =\n '<system-info>The multimodal contents returned from the tool call are as follows:\\n';\n\n if (promotedBlocks[0].type === 'text') {\n promotedBlocks[0].text = `${prefix}${promotedBlocks[0].text}`;\n } else {\n promotedBlocks.unshift({\n id: crypto.randomUUID(),\n type: 'text',\n text: `${prefix}`,\n });\n }\n\n // The suffix\n const lastBlock = promotedBlocks[promotedBlocks.length - 1];\n if (lastBlock.type === 'text') {\n promotedBlocks[promotedBlocks.length - 1] = {\n id: crypto.randomUUID(),\n type: 'text',\n text: `${lastBlock.text}</system-info>`,\n };\n } else {\n promotedBlocks.push({\n id: crypto.randomUUID(),\n type: 'text',\n text: `</system-info>`,\n });\n }\n }\n\n return {\n text: textualOutput.join('\\n'),\n promotedMsg: createMsg({ name: 'user', content: promotedBlocks, role: 'user' }),\n };\n }\n}\n","import { FormatterBase } from './base';\nimport { Msg, TextBlock, getContentBlocks } from '../message';\nimport { DataBlock } from '../message';\n\ninterface DashScopeFormatterOptions {\n /**\n * Since DashScope API doesn't support multimodal tool outputs, this option indicates whether to\n * promote the multimodal tool results to the prompt messages, so that LLMs can see them.\n * Note you should ensure your model supports the corresponding modalities.\n */\n promoteMultimodalToolResult?:\n | {\n image?: boolean;\n audio?: boolean;\n video?: boolean;\n }\n | boolean;\n}\n\n/**\n *\n */\nexport class DashScopeChatFormatter extends FormatterBase {\n private promoteMultimodalToolResult:\n | { image?: boolean; audio?: boolean; video?: boolean }\n | boolean;\n\n /**\n * Initialize a DashScopeChatFormatter instance.\n *\n * @param promoteMultimodalToolResult - Since DashScope API doesn't support multimodal tool outputs, this option\n * indicates whether to promote the multimodal tool results to the prompt messages, so that LLMs can see them.\n * Note you should ensure your model supports the corresponding modalities.\n * @param promoteMultimodalToolResult.promoteMultimodalToolResult\n */\n constructor({ promoteMultimodalToolResult = false }: DashScopeFormatterOptions = {}) {\n super();\n this.promoteMultimodalToolResult = promoteMultimodalToolResult;\n }\n\n /**\n * Format the input message objects into the required format by DashScope API.\n *\n * @param msgs - An array of Msg instances to be formatted.\n * @param msgs.msgs\n * @returns A promise that resolves to an array of formatted message objects.\n */\n async format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]> {\n const formattedMsgs: Array<Record<string, unknown>> = [];\n let index = 0;\n while (index < msgs.length) {\n const msg = msgs[index];\n const formattedMsg: {\n role: string;\n content: Record<string, unknown>[];\n tool_calls?: {\n id: string;\n type: 'function';\n function: {\n name: string;\n arguments: string;\n };\n }[];\n } = {\n role: msg.role,\n content: [],\n };\n\n // The cached messages that should be pushed after the current message, to keep the order of messages correct.\n const cachedMsgs = [];\n for (const block of getContentBlocks(msg)) {\n switch (block.type) {\n case 'text':\n formattedMsg.content.push(this._formatTextBlock(block));\n break;\n case 'thinking':\n break;\n case 'tool_call':\n if (!formattedMsg.tool_calls) {\n formattedMsg.tool_calls = [];\n }\n formattedMsg.tool_calls.push({\n id: block.id,\n type: 'function',\n function: {\n name: block.name,\n arguments: block.input,\n },\n });\n break;\n case 'tool_result':\n const formattedToolResult = this.convertToolOutputToString(\n block.output,\n this.promoteMultimodalToolResult\n );\n\n cachedMsgs.push({\n role: 'tool',\n tool_call_id: block.id,\n name: block.name,\n content: formattedToolResult.text,\n });\n if (formattedToolResult.promotedMsg) {\n // Insert the promoted message into the array as the next message to be processed\n msgs.splice(index + 1, 0, formattedToolResult.promotedMsg);\n }\n break;\n case 'data':\n formattedMsg.content.push(...this._formatMultimodalBlock(block));\n break;\n }\n }\n if (formattedMsg.content.length > 0 || formattedMsg.tool_calls) {\n formattedMsgs.push(formattedMsg);\n }\n if (cachedMsgs.length > 0) {\n formattedMsgs.push(...cachedMsgs);\n }\n\n // Process next message\n index++;\n }\n return formattedMsgs;\n }\n\n /**\n * Format a text content block into the required format.\n *\n * @param block - The text content block to format.\n * @returns An object representing the formatted text content.\n */\n _formatTextBlock(block: TextBlock) {\n return { text: block.text };\n }\n\n /**\n * Format a multimodal data block into the required format.\n * In DashScope API, the local file paths should be prefixed with \"file://\". URLs are kept unchanged.\n *\n * @param block - The multimodal content block to format.\n * @returns An object representing the formatted multimodal content.\n */\n _formatMultimodalBlock(block: DataBlock) {\n const type = block.source.media_type.split('/')[0];\n\n if (!['image', 'audio', 'video'].includes(type)) {\n console.log(\n `Skip unsupported media type ${block.source.media_type} in DashScopeChatFormatter. Only image, audio and video are supported.`\n );\n return [];\n }\n\n if (block.source.type === 'url') {\n return [{ [type]: block.source.url }];\n }\n\n return [\n {\n [type]: `data:${block.source.media_type};base64,${block.source.data}`,\n },\n ];\n }\n}\n","import { FormatterBase } from './base';\nimport { Msg, getContentBlocks } from '../message';\n\ninterface DeepSeekFormatterOptions {\n /**\n * Most LLM APIs don't support multimodal tool outputs, this option controls whether to\n * promote multimodal tool results to follow-up user messages.\n */\n promoteMultimodalToolResult?:\n | {\n image?: boolean;\n audio?: boolean;\n video?: boolean;\n }\n | boolean;\n}\n\n/**\n * Format AgentScope message objects into DeepSeek Chat Completions message format.\n */\nexport class DeepSeekChatFormatter extends FormatterBase {\n private promoteMultimodalToolResult:\n | { image?: boolean; audio?: boolean; video?: boolean }\n | boolean;\n\n /**\n * Initializes a new instance of the DeepSeekChatFormatter class.\n * @param root0\n * @param root0.promoteMultimodalToolResult\n */\n constructor({ promoteMultimodalToolResult = false }: DeepSeekFormatterOptions = {}) {\n super();\n this.promoteMultimodalToolResult = promoteMultimodalToolResult;\n }\n\n /**\n * Format the input messages into the structure expected by DeepSeek Chat Completions API.\n * @param root0\n * @param root0.msgs\n * @returns An array of formatted message objects ready to be sent to the DeepSeek API.\n */\n async format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]> {\n const formattedMsgs: Array<Record<string, unknown>> = [];\n let index = 0;\n\n while (index < msgs.length) {\n const msg = msgs[index];\n const formattedMsg: {\n role: string;\n name: string;\n content: Record<string, unknown>[] | null;\n tool_calls?: {\n id: string;\n type: 'function';\n function: {\n name: string;\n arguments: string;\n };\n }[];\n } = {\n role: msg.role,\n name: msg.name,\n content: null,\n };\n const content: Record<string, unknown>[] = [];\n\n // Cache tool-result messages to keep the sequence right after current message.\n const cachedMsgs: Record<string, unknown>[] = [];\n for (const block of getContentBlocks(msg)) {\n switch (block.type) {\n case 'text':\n content.push({\n type: 'text',\n text: block.text,\n });\n break;\n case 'thinking':\n break;\n case 'tool_call':\n if (!formattedMsg.tool_calls) {\n formattedMsg.tool_calls = [];\n }\n formattedMsg.tool_calls.push({\n id: block.id,\n type: 'function',\n function: {\n name: block.name,\n arguments: block.input,\n },\n });\n break;\n case 'tool_result':\n const formattedToolResult = this.convertToolOutputToString(\n block.output,\n this.promoteMultimodalToolResult\n );\n cachedMsgs.push({\n role: 'tool',\n tool_call_id: block.id,\n name: block.name,\n content: formattedToolResult.text,\n });\n if (formattedToolResult.promotedMsg?.content.length) {\n msgs.splice(index + 1, 0, formattedToolResult.promotedMsg);\n }\n break;\n case 'data':\n console.warn(\n `DeepSeek models don't support multimodal data for now (2026-03), skip the data block in message content.`\n );\n break;\n }\n }\n\n if (content.length > 0) {\n formattedMsg.content = content;\n }\n if (formattedMsg.content || formattedMsg.tool_calls) {\n formattedMsgs.push(formattedMsg);\n }\n if (cachedMsgs.length > 0) {\n formattedMsgs.push(...cachedMsgs);\n }\n\n index++;\n }\n\n return formattedMsgs;\n }\n}\n","import { FormatterBase } from './base';\nimport { Msg, getContentBlocks, getTextContent } from '../message';\n\n/**\n * Format AgentScope message objects into Ollama Chat message format.\n * Ollama expects simple string content, not the multimodal array format.\n */\nexport class OllamaChatFormatter extends FormatterBase {\n // eslint-disable-next-line jsdoc/require-returns\n /**\n * Format messages for Ollama API\n * @param root0\n * @param root0.msgs\n */\n async format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]> {\n const formattedMsgs: Array<Record<string, unknown>> = [];\n\n for (const msg of msgs) {\n const formattedMsg: {\n role: string;\n content: string;\n tool_calls?: {\n function: {\n name: string;\n arguments: Record<string, unknown>;\n };\n }[];\n } = {\n role: msg.role,\n content: '',\n };\n\n // Extract text content\n const textContent = getTextContent(msg);\n if (textContent) {\n formattedMsg.content = textContent;\n }\n\n // Handle tool calls\n const toolCalls = getContentBlocks(msg, 'tool_call');\n if (toolCalls.length > 0) {\n formattedMsg.tool_calls = toolCalls.map(toolCall => ({\n function: {\n name: toolCall.name,\n arguments: JSON.parse(toolCall.input),\n },\n }));\n }\n\n // Handle tool results\n const toolResults = getContentBlocks(msg, 'tool_result');\n for (const toolResult of toolResults) {\n const resultText = this.convertToolOutputToString(toolResult.output, false);\n formattedMsgs.push({\n role: 'tool',\n content: resultText.text,\n });\n }\n\n if (formattedMsg.content || formattedMsg.tool_calls) {\n formattedMsgs.push(formattedMsg);\n }\n }\n\n return formattedMsgs;\n }\n}\n","import { existsSync } from 'fs';\nimport { readFile } from 'fs/promises';\nimport { extname } from 'path';\nimport { fileURLToPath } from 'url';\n\nimport { FormatterBase } from './base';\nimport { DataBlock, Msg, TextBlock, getContentBlocks } from '../message';\n\ninterface OpenAIFormatterOptions {\n /**\n * Most LLM APIs don't support multimodal tool outputs, this option controls whether to\n * promote multimodal tool results to follow-up user messages.\n */\n promoteMultimodalToolResult?:\n | {\n image?: boolean;\n audio?: boolean;\n video?: boolean;\n }\n | boolean;\n}\n\n/**\n * Format AgentScope message objects into OpenAI Chat Completions message format.\n */\nexport class OpenAIChatFormatter extends FormatterBase {\n private promoteMultimodalToolResult:\n | { image?: boolean; audio?: boolean; video?: boolean }\n | boolean;\n\n /**\n * Initializes a new instance of the OpenAIChatFormatter class.\n * @param root0\n * @param root0.promoteMultimodalToolResult\n */\n constructor({ promoteMultimodalToolResult = false }: OpenAIFormatterOptions = {}) {\n super();\n this.promoteMultimodalToolResult = promoteMultimodalToolResult;\n }\n\n /**\n * Format the input messages into OpenAI Chat Completions message format.\n * @param root0\n * @param root0.msgs\n * @returns An array of formatted messages compatible with OpenAI Chat Completions API.\n */\n async format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]> {\n const formattedMsgs: Array<Record<string, unknown>> = [];\n let index = 0;\n\n while (index < msgs.length) {\n const msg = msgs[index];\n const formattedMsg: {\n role: string;\n name: string;\n content: Record<string, unknown>[] | null;\n tool_calls?: {\n id: string;\n type: 'function';\n function: {\n name: string;\n arguments: string;\n };\n }[];\n } = {\n role: msg.role,\n name: msg.name,\n content: null,\n };\n const content: Record<string, unknown>[] = [];\n\n // Cache tool-result messages to keep the sequence right after current message.\n const cachedMsgs: Record<string, unknown>[] = [];\n for (const block of getContentBlocks(msg)) {\n switch (block.type) {\n case 'text':\n content.push(this._formatTextBlock(block));\n break;\n case 'thinking':\n break;\n case 'tool_call':\n if (!formattedMsg.tool_calls) {\n formattedMsg.tool_calls = [];\n }\n formattedMsg.tool_calls.push({\n id: block.id,\n type: 'function',\n function: {\n name: block.name,\n arguments: block.input,\n },\n });\n break;\n case 'tool_result':\n const formattedToolResult = this.convertToolOutputToString(\n block.output,\n this.promoteMultimodalToolResult\n );\n cachedMsgs.push({\n role: 'tool',\n tool_call_id: block.id,\n name: block.name,\n content: formattedToolResult.text,\n });\n if (formattedToolResult.promotedMsg?.content.length) {\n msgs.splice(index + 1, 0, formattedToolResult.promotedMsg);\n }\n break;\n case 'data':\n content.push(\n ...(await this._formatMultimodalBlock({ block, role: msg.role }))\n );\n break;\n }\n }\n\n if (content.length > 0) {\n formattedMsg.content = content;\n }\n if (formattedMsg.content || formattedMsg.tool_calls) {\n formattedMsgs.push(formattedMsg);\n }\n if (cachedMsgs.length > 0) {\n formattedMsgs.push(...cachedMsgs);\n }\n\n index++;\n }\n\n return formattedMsgs;\n }\n\n /**\n * Format a text block into OpenAI Chat Completions message content format.\n * @param block\n * @returns An object representing the formatted text block.\n */\n _formatTextBlock(block: TextBlock) {\n return {\n type: 'text',\n text: block.text,\n };\n }\n\n /**\n * Format a multimodal data block into OpenAI Chat Completions message content format.\n * @param root0\n * @param root0.block\n * @param root0.role\n * @returns The formatted content blocks\n */\n async _formatMultimodalBlock({\n block,\n role,\n }: {\n block: DataBlock;\n role: Msg['role'];\n }): Promise<Record<string, unknown>[]> {\n const type = block.source.media_type.split('/')[0];\n if (type === 'image') {\n return [\n {\n type: 'image_url',\n image_url: {\n url: await this._toOpenAIImageURL(block),\n },\n },\n ];\n }\n\n if (type === 'audio') {\n // Skip assistant output audio to avoid carrying generated audio back into next request.\n if (role === 'assistant') {\n return [];\n }\n return [\n {\n type: 'input_audio',\n input_audio: await this._toOpenAIAudioData(block),\n },\n ];\n }\n\n console.log(\n `Skip unsupported media type ${block.source.media_type} in OpenAIChatFormatter. Only image and audio are supported.`\n );\n return [];\n }\n\n /**\n * Convert the data block to an OpenAI compatible image URL.\n * @param block\n * @returns A promise that resolves to a string representing the image URL in a format compatible with OpenAI Chat Completions API.\n */\n protected async _toOpenAIImageURL(block: DataBlock): Promise<string> {\n if (block.source.type === 'base64') {\n return `data:${block.source.media_type};base64,${block.source.data}`;\n }\n\n const sourceUrl = block.source.url;\n if (sourceUrl.startsWith('http://') || sourceUrl.startsWith('https://')) {\n return sourceUrl;\n }\n if (sourceUrl.startsWith('data:')) {\n return sourceUrl;\n }\n\n const localPath = this._toLocalPath(sourceUrl);\n if (!localPath || !existsSync(localPath)) {\n throw new Error(`Image path not found: ${sourceUrl}`);\n }\n\n const ext = extname(localPath).toLowerCase();\n const supportedImageExtensions = ['.png', '.jpg', '.jpeg', '.gif', '.webp'];\n if (!supportedImageExtensions.includes(ext)) {\n throw new TypeError(\n `Unsupported image extension: ${ext}. Supported: ${supportedImageExtensions.join(', ')}`\n );\n }\n\n const file = await readFile(localPath);\n const mime = block.source.media_type || `image/${ext.slice(1)}`;\n return `data:${mime};base64,${file.toString('base64')}`;\n }\n\n /**\n * Converts a data block to OpenAI compatible audio data format.\n *\n * @param block - The data block containing audio information.\n * @returns A promise that resolves to an object with audio data and format.\n */\n protected async _toOpenAIAudioData(\n block: DataBlock\n ): Promise<{ data: string; format: 'wav' | 'mp3' }> {\n const supportedMediaTypes = new Map<string, 'wav' | 'mp3'>([\n ['audio/wav', 'wav'],\n ['audio/mp3', 'mp3'],\n ['audio/mpeg', 'mp3'],\n ]);\n\n if (block.source.type === 'base64') {\n const format = supportedMediaTypes.get(block.source.media_type);\n if (!format) {\n throw new TypeError(\n `Unsupported audio media type: ${block.source.media_type}, only audio/wav and audio/mp3 are supported.`\n );\n }\n return { data: block.source.data, format };\n }\n\n const sourceUrl = block.source.url;\n const localPath = this._toLocalPath(sourceUrl);\n let data: string;\n\n if (localPath && existsSync(localPath)) {\n const file = await readFile(localPath);\n data = file.toString('base64');\n } else if (sourceUrl.startsWith('http://') || sourceUrl.startsWith('https://')) {\n const response = await fetch(sourceUrl);\n if (!response.ok) {\n throw new Error(\n `Failed to fetch audio from URL: ${sourceUrl} (${response.status})`\n );\n }\n const arr = await response.arrayBuffer();\n data = Buffer.from(arr).toString('base64');\n } else {\n throw new Error(\n `Unsupported audio source: ${sourceUrl}, it should be a local file path, file URL, or an HTTP URL.`\n );\n }\n\n const ext = extname(localPath || sourceUrl).toLowerCase();\n const extToFormat = new Map<string, 'wav' | 'mp3'>([\n ['.wav', 'wav'],\n ['.mp3', 'mp3'],\n ]);\n const format = extToFormat.get(ext);\n if (!format) {\n throw new TypeError(`Unsupported audio extension: ${ext}, wav and mp3 are supported.`);\n }\n\n return { data, format };\n }\n\n /**\n * Converts a URL or path to a local file path.\n *\n * @param urlOrPath - The URL or path to convert.\n * @returns The local file path, or null if not a local path.\n */\n protected _toLocalPath(urlOrPath: string) {\n if (urlOrPath.startsWith('file://')) {\n return fileURLToPath(urlOrPath);\n }\n if (!urlOrPath.includes('://')) {\n return urlOrPath;\n }\n return null;\n }\n}\n"],"mappings":"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","names":["format"]}
1
+ {"version":3,"sources":["../../src/formatter/index.ts","../../src/message/message.ts","../../src/formatter/base.ts","../../src/formatter/dashscope-chat-formatter.ts","../../src/formatter/deepseek-chat-formatter.ts","../../src/formatter/ollama-chat-formatter.ts","../../src/formatter/openai-chat-formatter.ts"],"sourcesContent":["export { FormatterBase } from './base';\nexport { DashScopeChatFormatter } from './dashscope-chat-formatter';\nexport { DeepSeekChatFormatter } from './deepseek-chat-formatter';\nexport { OllamaChatFormatter } from './ollama-chat-formatter';\nexport { OpenAIChatFormatter } from './openai-chat-formatter';\n","import { JSONSerializableObject } from '../type';\nimport {\n ContentBlock,\n TextBlock,\n ThinkingBlock,\n ToolResultBlock,\n ToolCallBlock,\n DataBlock,\n Base64Source,\n URLSource,\n} from './block';\nimport { AgentEvent, EventType } from '../event';\n\n/** A chat message exchanged between agents or between an agent and a model. */\nexport interface Msg {\n /** Unique identifier for the message. */\n id: string;\n /** Display name of the message sender. */\n name: string;\n /** Conversation role of the sender. */\n role: 'user' | 'assistant' | 'system';\n /** Message body. */\n content: ContentBlock[];\n /** Arbitrary key-value metadata attached to the message. */\n metadata: Record<string, JSONSerializableObject>;\n /** ISO-8601 creation timestamp. */\n created_at: string;\n /** ISO-8601 finished timestamp. */\n finished_at?: string | null;\n /** Usage information for the message, such as token counts. */\n usage?: {\n input_tokens: number;\n output_tokens: number;\n };\n}\n\n/**\n * Create a new {@link Msg} object, filling in `id` and `created_at` when omitted.\n * A plain string `content` is automatically wrapped in a single {@link TextBlock}.\n * @param root0\n * @param root0.name\n * @param root0.content\n * @param root0.role\n * @param root0.metadata\n * @param root0.id\n * @param root0.created_at\n * @param root0.finished_at\n * @param root0.usage\n * @returns A Msg object.\n */\n/**\n * Validate that content blocks are allowed for the given role.\n *\n * Mirrors the Python `_assert_user_content_blocks` / `_assert_system_content_blocks`\n * guards: user messages may only contain text or data blocks; system messages\n * may only contain text blocks; assistant messages accept any block type.\n * @param role\n * @param content\n */\nfunction assertContentBlocksForRole(role: Msg['role'], content: ContentBlock[]): void {\n if (role === 'user') {\n for (const block of content) {\n if (block.type !== 'text' && block.type !== 'data') {\n throw new Error('User message can only contain text blocks or data blocks.');\n }\n }\n } else if (role === 'system') {\n for (const block of content) {\n if (block.type !== 'text') {\n throw new Error('System message can only contain text blocks.');\n }\n }\n }\n}\n\n/**\n * createMsg is a low-level utility for constructing Msg objects with proper defaults and validation.\n * @param root0\n * @param root0.name\n * @param root0.content\n * @param root0.role\n * @param root0.metadata\n * @param root0.id\n * @param root0.created_at\n * @param root0.finished_at\n * @param root0.usage\n * @returns A Msg object with the specified properties, and defaults for any omitted fields.\n */\nexport function createMsg({\n name,\n content,\n role,\n metadata = {},\n id = crypto.randomUUID(),\n created_at = new Date().toISOString(),\n finished_at,\n usage,\n}: Omit<Msg, 'id' | 'created_at' | 'metadata' | 'content'> &\n Partial<Pick<Msg, 'id' | 'created_at' | 'metadata'>> & {\n content: string | ContentBlock[];\n }): Msg {\n const contentBlocks: ContentBlock[] =\n typeof content === 'string'\n ? [{ id: crypto.randomUUID(), type: 'text', text: content } as TextBlock]\n : content;\n assertContentBlocksForRole(role, contentBlocks);\n return { id, name, role, content: contentBlocks, metadata, created_at, finished_at, usage };\n}\n\n/**\n * Create a user {@link Msg}.\n * @param root0\n * @param root0.name\n * @param root0.content\n * @param root0.metadata\n * @param root0.id\n * @param root0.created_at\n * @param root0.finished_at\n * @returns A Msg object with role 'user'.\n */\nexport function UserMsg({\n name,\n content,\n metadata = {},\n id = crypto.randomUUID(),\n created_at = new Date().toISOString(),\n finished_at,\n}: {\n name: string;\n content: string | ContentBlock[];\n metadata?: Record<string, JSONSerializableObject>;\n id?: string;\n created_at?: string;\n finished_at?: string | null;\n}): Msg {\n return createMsg({\n name,\n content,\n role: 'user',\n metadata,\n id,\n created_at,\n finished_at: finished_at ?? created_at,\n });\n}\n\n/**\n * Create an assistant {@link Msg}.\n * @param root0\n * @param root0.name\n * @param root0.content\n * @param root0.metadata\n * @param root0.id\n * @param root0.created_at\n * @param root0.usage\n * @returns A Msg object with role 'assistant'.\n */\nexport function AssistantMsg({\n name,\n content,\n metadata = {},\n id = crypto.randomUUID(),\n created_at = new Date().toISOString(),\n usage,\n}: {\n name: string;\n content: string | ContentBlock[];\n metadata?: Record<string, JSONSerializableObject>;\n id?: string;\n created_at?: string;\n usage?: Msg['usage'];\n}): Msg {\n return createMsg({ name, content, role: 'assistant', metadata, id, created_at, usage });\n}\n\n/**\n * Create a system {@link Msg}.\n * @param root0\n * @param root0.name\n * @param root0.content\n * @param root0.metadata\n * @param root0.id\n * @param root0.created_at\n * @param root0.finished_at\n * @returns A Msg object with role 'system'.\n */\nexport function SystemMsg({\n name,\n content,\n metadata = {},\n id = crypto.randomUUID(),\n created_at = new Date().toISOString(),\n finished_at,\n}: {\n name: string;\n content: string | ContentBlock[];\n metadata?: Record<string, JSONSerializableObject>;\n id?: string;\n created_at?: string;\n finished_at?: string | null;\n}): Msg {\n return createMsg({\n name,\n content,\n role: 'system',\n metadata,\n id,\n created_at,\n finished_at: finished_at ?? created_at,\n });\n}\n\n/**\n * Extract the plain-text content from a message.\n *\n * When `content` is a string it is returned as-is. When it is an array of\n * content blocks, all {@link TextBlock} texts are joined with `separator`.\n *\n * @param msg - The message to read.\n * @param separator - String inserted between consecutive text blocks. Defaults to `'\\n'`.\n * @returns The concatenated text, or `null` when no text blocks are present.\n */\nexport function getTextContent(msg: Msg, separator: string = '\\n'): string | null {\n const textBlocks = msg.content.filter(block => block.type === 'text');\n if (textBlocks.length === 0) return null;\n return textBlocks.map(block => (block as TextBlock).text).join(separator);\n}\n\n/**\n * Return all content blocks from a message, regardless of type.\n *\n * When `content` is a plain string it is wrapped in a single {@link TextBlock}.\n *\n * @param msg - The message to read.\n * @returns An array of all {@link ContentBlock} objects.\n */\nexport function getContentBlocks(msg: Msg): ContentBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'text'): TextBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'thinking'): ThinkingBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'data'): DataBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'tool_call'): ToolCallBlock[];\nexport function getContentBlocks(msg: Msg, blockType: 'tool_result'): ToolResultBlock[];\nexport function getContentBlocks(\n msg: Msg,\n blockType?: 'text' | 'thinking' | 'data' | 'tool_call' | 'tool_result'\n): ContentBlock[] {\n if (!blockType) return msg.content;\n return msg.content.filter(block => block.type === blockType);\n}\n\n/**\n * Find a content block by type and id within a message.\n * @param msg\n * @param blockType\n * @param blockId\n * @returns The matching {@link ContentBlock}, or `undefined` if not found.\n */\nfunction findBlock(msg: Msg, blockType: string, blockId: string): ContentBlock | undefined {\n return msg.content.find(block => block.type === blockType && block.id === blockId);\n}\n\n/**\n * Apply a streaming {@link AgentEvent} to a {@link Msg}, mutating it in place.\n *\n * Only `content` and `finished_at` are ever modified. Events whose\n * `reply_id` does not match `msg.id` are skipped with a warning.\n * @param msg\n * @param event\n * @returns The mutated {@link Msg} object.\n */\nexport function appendEvent(msg: Msg, event: AgentEvent): Msg {\n if (!('reply_id' in event)) return msg;\n if (event.reply_id !== msg.id) {\n console.warn(\n `Event reply_id \"${event.reply_id}\" does not match message id \"${msg.id}\", skipping.`\n );\n return msg;\n }\n\n switch (event.type) {\n case EventType.REPLY_END:\n msg.finished_at = event.created_at;\n break;\n\n case EventType.TEXT_BLOCK_START:\n msg.content.push({ type: 'text', id: event.block_id, text: '' });\n break;\n\n case EventType.TEXT_BLOCK_DELTA: {\n const block = findBlock(msg, 'text', event.block_id);\n if (!block) {\n console.warn(`TextBlock \"${event.block_id}\" not found, skipping.`);\n } else {\n (block as TextBlock).text += event.delta;\n }\n break;\n }\n\n case EventType.TEXT_BLOCK_END:\n break;\n\n case EventType.THINKING_BLOCK_START:\n msg.content.push({ type: 'thinking', id: event.block_id, thinking: '' });\n break;\n\n case EventType.THINKING_BLOCK_DELTA: {\n const block = findBlock(msg, 'thinking', event.block_id);\n if (!block) {\n console.warn(`ThinkingBlock \"${event.block_id}\" not found, skipping.`);\n } else {\n (block as ThinkingBlock).thinking += event.delta;\n }\n break;\n }\n\n case EventType.THINKING_BLOCK_END:\n break;\n\n case EventType.DATA_BLOCK_START:\n msg.content.push({\n type: 'data',\n id: event.block_id,\n source: { type: 'base64', data: '', media_type: event.media_type },\n });\n break;\n\n case EventType.DATA_BLOCK_DELTA: {\n const block = findBlock(msg, 'data', event.block_id);\n if (!block) {\n console.warn(`DataBlock \"${event.block_id}\" not found, skipping.`);\n } else if (event.data) {\n ((block as DataBlock).source as Base64Source).data += event.data;\n }\n break;\n }\n\n case EventType.DATA_BLOCK_END:\n break;\n\n case EventType.TOOL_CALL_START:\n msg.content.push({\n type: 'tool_call',\n id: event.tool_call_id,\n name: event.tool_call_name,\n input: '',\n state: 'pending',\n });\n break;\n\n case EventType.TOOL_CALL_DELTA: {\n const block = findBlock(msg, 'tool_call', event.tool_call_id);\n if (!block) {\n console.warn(`ToolCallBlock \"${event.tool_call_id}\" not found, skipping.`);\n } else {\n (block as ToolCallBlock).input += event.delta;\n }\n break;\n }\n\n case EventType.TOOL_CALL_END:\n break;\n\n case EventType.TOOL_RESULT_START:\n msg.content.push({\n type: 'tool_result',\n id: event.tool_call_id,\n name: event.tool_call_name,\n output: [],\n state: 'running',\n });\n break;\n\n case EventType.TOOL_RESULT_TEXT_DELTA: {\n const block = findBlock(msg, 'tool_result', event.tool_call_id);\n if (!block) {\n console.warn(`ToolResultBlock \"${event.tool_call_id}\" not found, skipping.`);\n } else {\n const trb = block as ToolResultBlock;\n if (typeof trb.output === 'string') {\n trb.output = [{ type: 'text', id: crypto.randomUUID(), text: trb.output }];\n }\n const last = trb.output[trb.output.length - 1];\n if (!last || last.type !== 'text') {\n trb.output.push({\n type: 'text',\n id: crypto.randomUUID(),\n text: event.delta,\n });\n } else {\n (last as TextBlock).text += event.delta;\n }\n }\n break;\n }\n\n case EventType.TOOL_RESULT_DATA_DELTA: {\n const block = findBlock(msg, 'tool_result', event.tool_call_id);\n if (!block) {\n console.warn(`ToolResultBlock \"${event.tool_call_id}\" not found, skipping.`);\n } else {\n const trb = block as ToolResultBlock;\n if (typeof trb.output === 'string') {\n trb.output = [{ type: 'text', id: crypto.randomUUID(), text: trb.output }];\n }\n const source: Base64Source | URLSource =\n event.data != null\n ? { type: 'base64', data: event.data, media_type: event.media_type }\n : { type: 'url', url: event.url!, media_type: event.media_type };\n trb.output.push({\n type: 'data',\n id: event.block_id ?? crypto.randomUUID(),\n source,\n });\n }\n break;\n }\n\n case EventType.TOOL_RESULT_END: {\n const block = findBlock(msg, 'tool_result', event.tool_call_id);\n if (!block) {\n console.warn(`ToolResultBlock \"${event.tool_call_id}\" not found, skipping.`);\n } else {\n (block as ToolResultBlock).state = event.state;\n }\n break;\n }\n\n case EventType.MODEL_CALL_END:\n // Accumulated the input and output tokens here.\n if (msg.usage) {\n msg.usage.input_tokens += event.input_tokens;\n msg.usage.output_tokens += event.output_tokens;\n } else {\n msg.usage = {\n input_tokens: event.input_tokens,\n output_tokens: event.output_tokens,\n };\n }\n break;\n\n case EventType.REQUIRE_USER_CONFIRM:\n for (const tc of event.tool_calls) {\n const b = findBlock(msg, 'tool_call', tc.id);\n if (b) {\n (b as ToolCallBlock).state = 'asking';\n (b as ToolCallBlock).suggested_rules = tc.suggested_rules || [];\n }\n }\n break;\n\n case EventType.USER_CONFIRM_RESULT:\n for (const result of event.confirm_results) {\n const b = findBlock(msg, 'tool_call', result.tool_call.id);\n if (b) {\n (b as ToolCallBlock).state = result.confirmed ? 'allowed' : 'finished';\n }\n }\n break;\n\n case EventType.REQUIRE_EXTERNAL_EXECUTION:\n for (const tc of event.tool_calls) {\n const b = findBlock(msg, 'tool_call', tc.id);\n if (b) (b as ToolCallBlock).state = 'submitted';\n }\n break;\n\n case EventType.EXTERNAL_EXECUTION_RESULT:\n for (const result of event.execution_results) {\n msg.content.push(result);\n }\n break;\n }\n\n return msg;\n}\n","import { Msg, TextBlock, DataBlock, createMsg } from '../message';\n\n/**\n * Base class for message formatters.\n */\nexport abstract class FormatterBase {\n /**\n * Format the input message objects into the required format by the API.\n *\n * @param root0\n * @param root0.msgs - An array of message objects to be formatted.\n * @returns A promise that resolves to an array of formatted message objects.\n */\n abstract format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]>;\n\n /**\n * Convert the tool output to string format for the LLM APIs that only accept text input. If\n * `promoteMultimodalToolResult` is true, the multimodal content will be promoted to be a user message with\n * \"<system-info></system-info>\" tags. Otherwise, the multimodal content will be saved to a storage and a URL link\n * will be provided in the text output.\n *\n * @param output - The tool output, which can be a string or an array of content blocks.\n * @param promoteMultimodalToolResult - Whether to promote the multimodal content to the prompt messages.\n * @returns An object containing the text output and an optional promoted message.\n */\n convertToolOutputToString(\n output: string | (TextBlock | DataBlock)[],\n promoteMultimodalToolResult: boolean | { image?: boolean; audio?: boolean; video?: boolean }\n ) {\n if (typeof output === 'string') return { text: output, promotedMsg: null };\n\n let textualOutput = [];\n\n const promotedData: { id: string; block: DataBlock }[] = [];\n\n for (const block of output) {\n switch (block.type) {\n case 'text':\n textualOutput.push(block.text);\n break;\n default:\n const type = block.source.media_type.split('/')[0];\n if (type !== 'image' && type !== 'audio' && type !== 'video') {\n console.log(\n `Unsupported media type '${block.source.media_type}' in tool output. Only image, audio and video are supported.`\n );\n break;\n }\n if (block.source.type === 'url') {\n textualOutput.push(\n `<system-info>One returned ${type} can be found at: ${block.source.url}</system-info>`\n );\n } else {\n // If we should promote the multimodal content to the prompt messages\n const shouldPromote =\n promoteMultimodalToolResult === true ||\n (typeof promoteMultimodalToolResult === 'object' &&\n promoteMultimodalToolResult[type]);\n\n if (shouldPromote) {\n // Create an ID for the multimodal content first, which should less than 10 characters\n const dataID = Math.random().toString(36).substring(2, 10);\n textualOutput.push(\n `<system-info>One returned ${type} is embedded with ID '${dataID}' and will be attached within '<system-info></system-info>' tags later.</system-info>`\n );\n\n // Record the promoted data\n promotedData.push({ id: dataID, block });\n } else {\n // TODO: save locally\n\n // Save to storage and provide URL link\n textualOutput.push(`The returned ${block.type} is stored locally.`);\n }\n }\n }\n }\n\n // Attach prefix and suffix system-info tags if there are promoted blocks\n const promotedBlocks: (TextBlock | DataBlock)[] = [];\n promotedData.forEach(({ id, block }) => {\n const type = block.source.media_type.split('/')[0];\n promotedBlocks.push({\n id: crypto.randomUUID(),\n type: 'text',\n text: `<${type}_data id='${id}'>`,\n });\n promotedBlocks.push(block);\n promotedBlocks.push({\n id: crypto.randomUUID(),\n type: 'text',\n text: `</${type}_data>\\n`,\n });\n });\n\n if (promotedBlocks.length > 0) {\n // The prefix\n const prefix =\n '<system-info>The multimodal contents returned from the tool call are as follows:\\n';\n\n if (promotedBlocks[0].type === 'text') {\n promotedBlocks[0].text = `${prefix}${promotedBlocks[0].text}`;\n } else {\n promotedBlocks.unshift({\n id: crypto.randomUUID(),\n type: 'text',\n text: `${prefix}`,\n });\n }\n\n // The suffix\n const lastBlock = promotedBlocks[promotedBlocks.length - 1];\n if (lastBlock.type === 'text') {\n promotedBlocks[promotedBlocks.length - 1] = {\n id: crypto.randomUUID(),\n type: 'text',\n text: `${lastBlock.text}</system-info>`,\n };\n } else {\n promotedBlocks.push({\n id: crypto.randomUUID(),\n type: 'text',\n text: `</system-info>`,\n });\n }\n }\n\n return {\n text: textualOutput.join('\\n'),\n promotedMsg: createMsg({ name: 'user', content: promotedBlocks, role: 'user' }),\n };\n }\n}\n","import { FormatterBase } from './base';\nimport { Msg, TextBlock, getContentBlocks } from '../message';\nimport { DataBlock } from '../message';\n\ninterface DashScopeFormatterOptions {\n /**\n * Since DashScope API doesn't support multimodal tool outputs, this option indicates whether to\n * promote the multimodal tool results to the prompt messages, so that LLMs can see them.\n * Note you should ensure your model supports the corresponding modalities.\n */\n promoteMultimodalToolResult?:\n | {\n image?: boolean;\n audio?: boolean;\n video?: boolean;\n }\n | boolean;\n}\n\n/**\n *\n */\nexport class DashScopeChatFormatter extends FormatterBase {\n private promoteMultimodalToolResult:\n | { image?: boolean; audio?: boolean; video?: boolean }\n | boolean;\n\n /**\n * Initialize a DashScopeChatFormatter instance.\n *\n * @param promoteMultimodalToolResult - Since DashScope API doesn't support multimodal tool outputs, this option\n * indicates whether to promote the multimodal tool results to the prompt messages, so that LLMs can see them.\n * Note you should ensure your model supports the corresponding modalities.\n * @param promoteMultimodalToolResult.promoteMultimodalToolResult\n */\n constructor({ promoteMultimodalToolResult = false }: DashScopeFormatterOptions = {}) {\n super();\n this.promoteMultimodalToolResult = promoteMultimodalToolResult;\n }\n\n /**\n * Format the input message objects into the required format by DashScope API.\n *\n * @param msgs - An array of Msg instances to be formatted.\n * @param msgs.msgs\n * @returns A promise that resolves to an array of formatted message objects.\n */\n async format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]> {\n const formattedMsgs: Array<Record<string, unknown>> = [];\n let index = 0;\n while (index < msgs.length) {\n const msg = msgs[index];\n const formattedMsg: {\n role: string;\n content: Record<string, unknown>[];\n tool_calls?: {\n id: string;\n type: 'function';\n function: {\n name: string;\n arguments: string;\n };\n }[];\n } = {\n role: msg.role,\n content: [],\n };\n\n // The cached messages that should be pushed after the current message, to keep the order of messages correct.\n const cachedMsgs = [];\n for (const block of getContentBlocks(msg)) {\n switch (block.type) {\n case 'text':\n formattedMsg.content.push(this._formatTextBlock(block));\n break;\n case 'thinking':\n break;\n case 'tool_call':\n if (!formattedMsg.tool_calls) {\n formattedMsg.tool_calls = [];\n }\n formattedMsg.tool_calls.push({\n id: block.id,\n type: 'function',\n function: {\n name: block.name,\n arguments: block.input,\n },\n });\n break;\n case 'tool_result':\n const formattedToolResult = this.convertToolOutputToString(\n block.output,\n this.promoteMultimodalToolResult\n );\n\n cachedMsgs.push({\n role: 'tool',\n tool_call_id: block.id,\n name: block.name,\n content: formattedToolResult.text,\n });\n if (formattedToolResult.promotedMsg) {\n // Insert the promoted message into the array as the next message to be processed\n msgs.splice(index + 1, 0, formattedToolResult.promotedMsg);\n }\n break;\n case 'data':\n formattedMsg.content.push(...this._formatMultimodalBlock(block));\n break;\n }\n }\n if (formattedMsg.content.length > 0 || formattedMsg.tool_calls) {\n formattedMsgs.push(formattedMsg);\n }\n if (cachedMsgs.length > 0) {\n formattedMsgs.push(...cachedMsgs);\n }\n\n // Process next message\n index++;\n }\n return formattedMsgs;\n }\n\n /**\n * Format a text content block into the required format.\n *\n * @param block - The text content block to format.\n * @returns An object representing the formatted text content.\n */\n _formatTextBlock(block: TextBlock) {\n return { text: block.text };\n }\n\n /**\n * Format a multimodal data block into the required format.\n * In DashScope API, the local file paths should be prefixed with \"file://\". URLs are kept unchanged.\n *\n * @param block - The multimodal content block to format.\n * @returns An object representing the formatted multimodal content.\n */\n _formatMultimodalBlock(block: DataBlock) {\n const type = block.source.media_type.split('/')[0];\n\n if (!['image', 'audio', 'video'].includes(type)) {\n console.log(\n `Skip unsupported media type ${block.source.media_type} in DashScopeChatFormatter. Only image, audio and video are supported.`\n );\n return [];\n }\n\n if (block.source.type === 'url') {\n return [{ [type]: block.source.url }];\n }\n\n return [\n {\n [type]: `data:${block.source.media_type};base64,${block.source.data}`,\n },\n ];\n }\n}\n","import { FormatterBase } from './base';\nimport { Msg, getContentBlocks } from '../message';\n\ninterface DeepSeekFormatterOptions {\n /**\n * Most LLM APIs don't support multimodal tool outputs, this option controls whether to\n * promote multimodal tool results to follow-up user messages.\n */\n promoteMultimodalToolResult?:\n | {\n image?: boolean;\n audio?: boolean;\n video?: boolean;\n }\n | boolean;\n}\n\n/**\n * Format AgentScope message objects into DeepSeek Chat Completions message format.\n */\nexport class DeepSeekChatFormatter extends FormatterBase {\n private promoteMultimodalToolResult:\n | { image?: boolean; audio?: boolean; video?: boolean }\n | boolean;\n\n /**\n * Initializes a new instance of the DeepSeekChatFormatter class.\n * @param root0\n * @param root0.promoteMultimodalToolResult\n */\n constructor({ promoteMultimodalToolResult = false }: DeepSeekFormatterOptions = {}) {\n super();\n this.promoteMultimodalToolResult = promoteMultimodalToolResult;\n }\n\n /**\n * Format the input messages into the structure expected by DeepSeek Chat Completions API.\n * @param root0\n * @param root0.msgs\n * @returns An array of formatted message objects ready to be sent to the DeepSeek API.\n */\n async format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]> {\n const formattedMsgs: Array<Record<string, unknown>> = [];\n let index = 0;\n\n while (index < msgs.length) {\n const msg = msgs[index];\n const formattedMsg: {\n role: string;\n name: string;\n content: Record<string, unknown>[] | null;\n tool_calls?: {\n id: string;\n type: 'function';\n function: {\n name: string;\n arguments: string;\n };\n }[];\n } = {\n role: msg.role,\n name: msg.name,\n content: null,\n };\n const content: Record<string, unknown>[] = [];\n\n // Cache tool-result messages to keep the sequence right after current message.\n const cachedMsgs: Record<string, unknown>[] = [];\n for (const block of getContentBlocks(msg)) {\n switch (block.type) {\n case 'text':\n content.push({\n type: 'text',\n text: block.text,\n });\n break;\n case 'thinking':\n break;\n case 'tool_call':\n if (!formattedMsg.tool_calls) {\n formattedMsg.tool_calls = [];\n }\n formattedMsg.tool_calls.push({\n id: block.id,\n type: 'function',\n function: {\n name: block.name,\n arguments: block.input,\n },\n });\n break;\n case 'tool_result':\n const formattedToolResult = this.convertToolOutputToString(\n block.output,\n this.promoteMultimodalToolResult\n );\n cachedMsgs.push({\n role: 'tool',\n tool_call_id: block.id,\n name: block.name,\n content: formattedToolResult.text,\n });\n if (formattedToolResult.promotedMsg?.content.length) {\n msgs.splice(index + 1, 0, formattedToolResult.promotedMsg);\n }\n break;\n case 'data':\n console.warn(\n `DeepSeek models don't support multimodal data for now (2026-03), skip the data block in message content.`\n );\n break;\n }\n }\n\n if (content.length > 0) {\n formattedMsg.content = content;\n }\n if (formattedMsg.content || formattedMsg.tool_calls) {\n formattedMsgs.push(formattedMsg);\n }\n if (cachedMsgs.length > 0) {\n formattedMsgs.push(...cachedMsgs);\n }\n\n index++;\n }\n\n return formattedMsgs;\n }\n}\n","import { FormatterBase } from './base';\nimport { Msg, getContentBlocks, getTextContent } from '../message';\n\n/**\n * Format AgentScope message objects into Ollama Chat message format.\n * Ollama expects simple string content, not the multimodal array format.\n */\nexport class OllamaChatFormatter extends FormatterBase {\n // eslint-disable-next-line jsdoc/require-returns\n /**\n * Format messages for Ollama API\n * @param root0\n * @param root0.msgs\n */\n async format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]> {\n const formattedMsgs: Array<Record<string, unknown>> = [];\n\n for (const msg of msgs) {\n const formattedMsg: {\n role: string;\n content: string;\n tool_calls?: {\n function: {\n name: string;\n arguments: Record<string, unknown>;\n };\n }[];\n } = {\n role: msg.role,\n content: '',\n };\n\n // Extract text content\n const textContent = getTextContent(msg);\n if (textContent) {\n formattedMsg.content = textContent;\n }\n\n // Handle tool calls\n const toolCalls = getContentBlocks(msg, 'tool_call');\n if (toolCalls.length > 0) {\n formattedMsg.tool_calls = toolCalls.map(toolCall => ({\n function: {\n name: toolCall.name,\n arguments: JSON.parse(toolCall.input),\n },\n }));\n }\n\n // Handle tool results\n const toolResults = getContentBlocks(msg, 'tool_result');\n for (const toolResult of toolResults) {\n const resultText = this.convertToolOutputToString(toolResult.output, false);\n formattedMsgs.push({\n role: 'tool',\n content: resultText.text,\n });\n }\n\n if (formattedMsg.content || formattedMsg.tool_calls) {\n formattedMsgs.push(formattedMsg);\n }\n }\n\n return formattedMsgs;\n }\n}\n","import { existsSync } from 'fs';\nimport { readFile } from 'fs/promises';\nimport { extname } from 'path';\nimport { fileURLToPath } from 'url';\n\nimport { FormatterBase } from './base';\nimport { DataBlock, Msg, TextBlock, getContentBlocks } from '../message';\n\ninterface OpenAIFormatterOptions {\n /**\n * Most LLM APIs don't support multimodal tool outputs, this option controls whether to\n * promote multimodal tool results to follow-up user messages.\n */\n promoteMultimodalToolResult?:\n | {\n image?: boolean;\n audio?: boolean;\n video?: boolean;\n }\n | boolean;\n}\n\n/**\n * Format AgentScope message objects into OpenAI Chat Completions message format.\n */\nexport class OpenAIChatFormatter extends FormatterBase {\n private promoteMultimodalToolResult:\n | { image?: boolean; audio?: boolean; video?: boolean }\n | boolean;\n\n /**\n * Initializes a new instance of the OpenAIChatFormatter class.\n * @param root0\n * @param root0.promoteMultimodalToolResult\n */\n constructor({ promoteMultimodalToolResult = false }: OpenAIFormatterOptions = {}) {\n super();\n this.promoteMultimodalToolResult = promoteMultimodalToolResult;\n }\n\n /**\n * Format the input messages into OpenAI Chat Completions message format.\n * @param root0\n * @param root0.msgs\n * @returns An array of formatted messages compatible with OpenAI Chat Completions API.\n */\n async format({ msgs }: { msgs: Array<Msg> }): Promise<Record<string, unknown>[]> {\n const formattedMsgs: Array<Record<string, unknown>> = [];\n let index = 0;\n\n while (index < msgs.length) {\n const msg = msgs[index];\n const formattedMsg: {\n role: string;\n name: string;\n content: Record<string, unknown>[] | null;\n tool_calls?: {\n id: string;\n type: 'function';\n function: {\n name: string;\n arguments: string;\n };\n }[];\n } = {\n role: msg.role,\n name: msg.name,\n content: null,\n };\n const content: Record<string, unknown>[] = [];\n\n // Cache tool-result messages to keep the sequence right after current message.\n const cachedMsgs: Record<string, unknown>[] = [];\n for (const block of getContentBlocks(msg)) {\n switch (block.type) {\n case 'text':\n content.push(this._formatTextBlock(block));\n break;\n case 'thinking':\n break;\n case 'tool_call':\n if (!formattedMsg.tool_calls) {\n formattedMsg.tool_calls = [];\n }\n formattedMsg.tool_calls.push({\n id: block.id,\n type: 'function',\n function: {\n name: block.name,\n arguments: block.input,\n },\n });\n break;\n case 'tool_result':\n const formattedToolResult = this.convertToolOutputToString(\n block.output,\n this.promoteMultimodalToolResult\n );\n cachedMsgs.push({\n role: 'tool',\n tool_call_id: block.id,\n name: block.name,\n content: formattedToolResult.text,\n });\n if (formattedToolResult.promotedMsg?.content.length) {\n msgs.splice(index + 1, 0, formattedToolResult.promotedMsg);\n }\n break;\n case 'data':\n content.push(\n ...(await this._formatMultimodalBlock({ block, role: msg.role }))\n );\n break;\n }\n }\n\n if (content.length > 0) {\n formattedMsg.content = content;\n }\n if (formattedMsg.content || formattedMsg.tool_calls) {\n formattedMsgs.push(formattedMsg);\n }\n if (cachedMsgs.length > 0) {\n formattedMsgs.push(...cachedMsgs);\n }\n\n index++;\n }\n\n return formattedMsgs;\n }\n\n /**\n * Format a text block into OpenAI Chat Completions message content format.\n * @param block\n * @returns An object representing the formatted text block.\n */\n _formatTextBlock(block: TextBlock) {\n return {\n type: 'text',\n text: block.text,\n };\n }\n\n /**\n * Format a multimodal data block into OpenAI Chat Completions message content format.\n * @param root0\n * @param root0.block\n * @param root0.role\n * @returns The formatted content blocks\n */\n async _formatMultimodalBlock({\n block,\n role,\n }: {\n block: DataBlock;\n role: Msg['role'];\n }): Promise<Record<string, unknown>[]> {\n const type = block.source.media_type.split('/')[0];\n if (type === 'image') {\n return [\n {\n type: 'image_url',\n image_url: {\n url: await this._toOpenAIImageURL(block),\n },\n },\n ];\n }\n\n if (type === 'audio') {\n // Skip assistant output audio to avoid carrying generated audio back into next request.\n if (role === 'assistant') {\n return [];\n }\n return [\n {\n type: 'input_audio',\n input_audio: await this._toOpenAIAudioData(block),\n },\n ];\n }\n\n console.log(\n `Skip unsupported media type ${block.source.media_type} in OpenAIChatFormatter. Only image and audio are supported.`\n );\n return [];\n }\n\n /**\n * Convert the data block to an OpenAI compatible image URL.\n * @param block\n * @returns A promise that resolves to a string representing the image URL in a format compatible with OpenAI Chat Completions API.\n */\n protected async _toOpenAIImageURL(block: DataBlock): Promise<string> {\n if (block.source.type === 'base64') {\n return `data:${block.source.media_type};base64,${block.source.data}`;\n }\n\n const sourceUrl = block.source.url;\n if (sourceUrl.startsWith('http://') || sourceUrl.startsWith('https://')) {\n return sourceUrl;\n }\n if (sourceUrl.startsWith('data:')) {\n return sourceUrl;\n }\n\n const localPath = this._toLocalPath(sourceUrl);\n if (!localPath || !existsSync(localPath)) {\n throw new Error(`Image path not found: ${sourceUrl}`);\n }\n\n const ext = extname(localPath).toLowerCase();\n const supportedImageExtensions = ['.png', '.jpg', '.jpeg', '.gif', '.webp'];\n if (!supportedImageExtensions.includes(ext)) {\n throw new TypeError(\n `Unsupported image extension: ${ext}. Supported: ${supportedImageExtensions.join(', ')}`\n );\n }\n\n const file = await readFile(localPath);\n const mime = block.source.media_type || `image/${ext.slice(1)}`;\n return `data:${mime};base64,${file.toString('base64')}`;\n }\n\n /**\n * Converts a data block to OpenAI compatible audio data format.\n *\n * @param block - The data block containing audio information.\n * @returns A promise that resolves to an object with audio data and format.\n */\n protected async _toOpenAIAudioData(\n block: DataBlock\n ): Promise<{ data: string; format: 'wav' | 'mp3' }> {\n const supportedMediaTypes = new Map<string, 'wav' | 'mp3'>([\n ['audio/wav', 'wav'],\n ['audio/mp3', 'mp3'],\n ['audio/mpeg', 'mp3'],\n ]);\n\n if (block.source.type === 'base64') {\n const format = supportedMediaTypes.get(block.source.media_type);\n if (!format) {\n throw new TypeError(\n `Unsupported audio media type: ${block.source.media_type}, only audio/wav and audio/mp3 are supported.`\n );\n }\n return { data: block.source.data, format };\n }\n\n const sourceUrl = block.source.url;\n const localPath = this._toLocalPath(sourceUrl);\n let data: string;\n\n if (localPath && existsSync(localPath)) {\n const file = await readFile(localPath);\n data = file.toString('base64');\n } else if (sourceUrl.startsWith('http://') || sourceUrl.startsWith('https://')) {\n const response = await fetch(sourceUrl);\n if (!response.ok) {\n throw new Error(\n `Failed to fetch audio from URL: ${sourceUrl} (${response.status})`\n );\n }\n const arr = await response.arrayBuffer();\n data = Buffer.from(arr).toString('base64');\n } else {\n throw new Error(\n `Unsupported audio source: ${sourceUrl}, it should be a local file path, file URL, or an HTTP URL.`\n );\n }\n\n const ext = extname(localPath || sourceUrl).toLowerCase();\n const extToFormat = new Map<string, 'wav' | 'mp3'>([\n ['.wav', 'wav'],\n ['.mp3', 'mp3'],\n ]);\n const format = extToFormat.get(ext);\n if (!format) {\n throw new TypeError(`Unsupported audio extension: ${ext}, wav and mp3 are supported.`);\n }\n\n return { data, format };\n }\n\n /**\n * Converts a URL or path to a local file path.\n *\n * @param urlOrPath - The URL or path to convert.\n * @returns The local file path, or null if not a local path.\n */\n protected _toLocalPath(urlOrPath: string) {\n if (urlOrPath.startsWith('file://')) {\n return fileURLToPath(urlOrPath);\n }\n if (!urlOrPath.includes('://')) {\n return urlOrPath;\n }\n return null;\n }\n}\n"],"mappings":"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","names":["format"]}
@@ -1,4 +1,19 @@
1
1
  // src/message/message.ts
2
+ function assertContentBlocksForRole(role, content) {
3
+ if (role === "user") {
4
+ for (const block of content) {
5
+ if (block.type !== "text" && block.type !== "data") {
6
+ throw new Error("User message can only contain text blocks or data blocks.");
7
+ }
8
+ }
9
+ } else if (role === "system") {
10
+ for (const block of content) {
11
+ if (block.type !== "text") {
12
+ throw new Error("System message can only contain text blocks.");
13
+ }
14
+ }
15
+ }
16
+ }
2
17
  function createMsg({
3
18
  name,
4
19
  content,
@@ -10,6 +25,7 @@ function createMsg({
10
25
  usage
11
26
  }) {
12
27
  const contentBlocks = typeof content === "string" ? [{ id: crypto.randomUUID(), type: "text", text: content }] : content;
28
+ assertContentBlocksForRole(role, contentBlocks);
13
29
  return { id, name, role, content: contentBlocks, metadata, created_at, finished_at, usage };
14
30
  }
15
31
  function getTextContent(msg, separator = "\n") {