langchain 1.4.5 → 1.4.6-dev-1781485641139
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/agents/ReactAgent.cjs +5 -2
- package/dist/agents/ReactAgent.cjs.map +1 -1
- package/dist/agents/ReactAgent.d.cts +1 -1
- package/dist/agents/ReactAgent.d.cts.map +1 -1
- package/dist/agents/ReactAgent.d.ts +1 -1
- package/dist/agents/ReactAgent.d.ts.map +1 -1
- package/dist/agents/ReactAgent.js +4 -1
- package/dist/agents/ReactAgent.js.map +1 -1
- package/dist/agents/index.cjs +3 -1
- package/dist/agents/index.cjs.map +1 -1
- package/dist/agents/index.d.cts +3 -1
- package/dist/agents/index.d.cts.map +1 -1
- package/dist/agents/index.d.ts +3 -1
- package/dist/agents/index.d.ts.map +1 -1
- package/dist/agents/index.js +3 -1
- package/dist/agents/index.js.map +1 -1
- package/dist/agents/transformers/index.cjs +2 -0
- package/dist/agents/transformers/index.d.cts +3 -0
- package/dist/agents/transformers/index.d.ts +3 -0
- package/dist/agents/transformers/index.js +3 -0
- package/dist/agents/transformers/subagent.cjs +205 -0
- package/dist/agents/transformers/subagent.cjs.map +1 -0
- package/dist/agents/transformers/subagent.d.cts +34 -0
- package/dist/agents/transformers/subagent.d.cts.map +1 -0
- package/dist/agents/transformers/subagent.d.ts +34 -0
- package/dist/agents/transformers/subagent.d.ts.map +1 -0
- package/dist/agents/transformers/subagent.js +204 -0
- package/dist/agents/transformers/subagent.js.map +1 -0
- package/dist/agents/{stream.cjs → transformers/tool-call.cjs} +3 -15
- package/dist/agents/transformers/tool-call.cjs.map +1 -0
- package/dist/agents/transformers/tool-call.d.cts +17 -0
- package/dist/agents/transformers/tool-call.d.cts.map +1 -0
- package/dist/agents/transformers/tool-call.d.ts +17 -0
- package/dist/agents/transformers/tool-call.d.ts.map +1 -0
- package/dist/agents/{stream.js → transformers/tool-call.js} +2 -14
- package/dist/agents/transformers/tool-call.js.map +1 -0
- package/dist/agents/{stream.d.ts → transformers/types.d.cts} +50 -20
- package/dist/agents/transformers/types.d.cts.map +1 -0
- package/dist/agents/{stream.d.cts → transformers/types.d.ts} +50 -20
- package/dist/agents/transformers/types.d.ts.map +1 -0
- package/dist/browser.cjs +6 -3
- package/dist/browser.d.cts +4 -2
- package/dist/browser.d.ts +4 -2
- package/dist/browser.js +4 -2
- package/dist/index.cjs +6 -3
- package/dist/index.d.cts +4 -2
- package/dist/index.d.ts +4 -2
- package/dist/index.js +4 -2
- package/package.json +1 -1
- package/chat_models/universal.cjs +0 -1
- package/chat_models/universal.d.cts +0 -1
- package/chat_models/universal.d.ts +0 -1
- package/chat_models/universal.js +0 -1
- package/dist/agents/stream.cjs.map +0 -1
- package/dist/agents/stream.d.cts.map +0 -1
- package/dist/agents/stream.d.ts.map +0 -1
- package/dist/agents/stream.js.map +0 -1
- package/hub/node.cjs +0 -1
- package/hub/node.d.cts +0 -1
- package/hub/node.d.ts +0 -1
- package/hub/node.js +0 -1
- package/hub.cjs +0 -1
- package/hub.d.cts +0 -1
- package/hub.d.ts +0 -1
- package/hub.js +0 -1
- package/load/serializable.cjs +0 -1
- package/load/serializable.d.cts +0 -1
- package/load/serializable.d.ts +0 -1
- package/load/serializable.js +0 -1
- package/load.cjs +0 -1
- package/load.d.cts +0 -1
- package/load.d.ts +0 -1
- package/load.js +0 -1
- package/storage/encoder_backed.cjs +0 -1
- package/storage/encoder_backed.d.cts +0 -1
- package/storage/encoder_backed.d.ts +0 -1
- package/storage/encoder_backed.js +0 -1
- package/storage/file_system.cjs +0 -1
- package/storage/file_system.d.cts +0 -1
- package/storage/file_system.d.ts +0 -1
- package/storage/file_system.js +0 -1
- package/storage/in_memory.cjs +0 -1
- package/storage/in_memory.d.cts +0 -1
- package/storage/in_memory.d.ts +0 -1
- package/storage/in_memory.js +0 -1
|
@@ -0,0 +1,204 @@
|
|
|
1
|
+
import { createToolCallTransformer } from "./tool-call.js";
|
|
2
|
+
import { StreamChannel, createMessagesTransformer } from "@langchain/langgraph";
|
|
3
|
+
//#region src/agents/transformers/subagent.ts
|
|
4
|
+
function isRecord(value) {
|
|
5
|
+
return typeof value === "object" && value !== null && !Array.isArray(value);
|
|
6
|
+
}
|
|
7
|
+
/** Stable string key for a namespace. */
|
|
8
|
+
function nsKey(ns) {
|
|
9
|
+
return ns.join("\0");
|
|
10
|
+
}
|
|
11
|
+
/** Tests whether `ns` starts with every segment in `prefix`. */
|
|
12
|
+
function hasPrefix(ns, prefix) {
|
|
13
|
+
if (prefix.length > ns.length) return false;
|
|
14
|
+
for (let i = 0; i < prefix.length; i += 1) if (ns[i] !== prefix[i]) return false;
|
|
15
|
+
return true;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Creates a native transformer that surfaces nested named agents on
|
|
19
|
+
* `run.subagents`.
|
|
20
|
+
*
|
|
21
|
+
* It watches `tasks` events to record each namespace's `lc_agent_name` (set by
|
|
22
|
+
* `createAgent({ name })`) and the triggering tool call, then — for any nested
|
|
23
|
+
* run one level below {@link scope} that carries an `lc_agent_name` — emits a
|
|
24
|
+
* typed {@link SubagentRunStream} handle.
|
|
25
|
+
*
|
|
26
|
+
* Each handle is backed by its own per-subagent transformer instances
|
|
27
|
+
* ({@link createMessagesTransformer}, {@link createToolCallTransformer}, and a
|
|
28
|
+
* nested {@link createSubagentTransformer}) scoped to the subagent's namespace.
|
|
29
|
+
* Every event in the subtree is fed straight into those transformers, which
|
|
30
|
+
* self-filter by namespace; the subagent's final `output` is resolved from its
|
|
31
|
+
* last `values` snapshot when its `lifecycle` completes.
|
|
32
|
+
*
|
|
33
|
+
* Marked `__native: true` — the `subagents` projection lands directly on the
|
|
34
|
+
* `GraphRunStream` instance as `run.subagents`.
|
|
35
|
+
*
|
|
36
|
+
* @param scope - Namespace prefix this transformer is scoped to. The root agent
|
|
37
|
+
* uses `[]`; nested handles use their subagent's namespace, so grandchild
|
|
38
|
+
* subagents are discovered recursively.
|
|
39
|
+
*/
|
|
40
|
+
function createSubagentTransformer(scope = []) {
|
|
41
|
+
return () => {
|
|
42
|
+
const subagentsLog = StreamChannel.local();
|
|
43
|
+
/** `lc_agent_name` observed per namespace (first task event wins). */
|
|
44
|
+
const lcByNs = /* @__PURE__ */ new Map();
|
|
45
|
+
/** Triggering task id -> originating LLM `tool_call_id`. */
|
|
46
|
+
const pendingToolCalls = /* @__PURE__ */ new Map();
|
|
47
|
+
/**
|
|
48
|
+
* Namespace key -> the `tool_call_id` of the most recent tool to start
|
|
49
|
+
* executing there. A tool that invokes a subagent emits its `tool-started`
|
|
50
|
+
* at the tools-node namespace (`tools:<task_id>`) where the subagent then
|
|
51
|
+
* roots, so this is the tool call that caused the subagent.
|
|
52
|
+
*/
|
|
53
|
+
const activeToolCallByNs = /* @__PURE__ */ new Map();
|
|
54
|
+
const handles = /* @__PURE__ */ new Map();
|
|
55
|
+
const depth = scope.length;
|
|
56
|
+
function recordIdentity(ns, data) {
|
|
57
|
+
const key = nsKey(ns);
|
|
58
|
+
if (lcByNs.has(key)) return;
|
|
59
|
+
const lc = (isRecord(data) && isRecord(data.metadata) ? data.metadata : void 0)?.lc_agent_name;
|
|
60
|
+
lcByNs.set(key, typeof lc === "string" ? lc : void 0);
|
|
61
|
+
}
|
|
62
|
+
function recordPendingToolCalls(data) {
|
|
63
|
+
if (!isRecord(data)) return;
|
|
64
|
+
const taskId = data.id;
|
|
65
|
+
if (typeof taskId !== "string") return;
|
|
66
|
+
const input = data.input;
|
|
67
|
+
let toolCallId;
|
|
68
|
+
if (isRecord(input) && isRecord(input.tool_call)) {
|
|
69
|
+
const candidate = input.tool_call.id;
|
|
70
|
+
if (typeof candidate === "string") toolCallId = candidate;
|
|
71
|
+
} else if (Array.isArray(input)) {
|
|
72
|
+
for (const toolCall of input) if (isRecord(toolCall) && typeof toolCall.id === "string") {
|
|
73
|
+
toolCallId = toolCall.id;
|
|
74
|
+
break;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
if (toolCallId != null) pendingToolCalls.set(taskId, toolCallId);
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Derive the `toolCall` cause for a named-subagent namespace.
|
|
81
|
+
*
|
|
82
|
+
* Primary signal: the tool whose `tool-started` event fired at the
|
|
83
|
+
* subagent's own namespace (the tools node it roots under). Fallback: the
|
|
84
|
+
* namespace segment's task id (`node:<task_id>`) joined to a tool call
|
|
85
|
+
* harvested from a `tool_call_with_context`-shaped task input, so the
|
|
86
|
+
* derivation stays correct if that shape reaches the stream in the future.
|
|
87
|
+
*/
|
|
88
|
+
function deriveCause(ns) {
|
|
89
|
+
const active = activeToolCallByNs.get(nsKey(ns));
|
|
90
|
+
if (typeof active === "string" && active.length > 0) return {
|
|
91
|
+
type: "toolCall",
|
|
92
|
+
tool_call_id: active
|
|
93
|
+
};
|
|
94
|
+
const segment = ns[ns.length - 1];
|
|
95
|
+
const colon = segment.indexOf(":");
|
|
96
|
+
if (colon === -1) return void 0;
|
|
97
|
+
const triggerCallId = segment.slice(colon + 1);
|
|
98
|
+
if (triggerCallId.length === 0) return void 0;
|
|
99
|
+
const toolCallId = pendingToolCalls.get(triggerCallId);
|
|
100
|
+
if (typeof toolCallId !== "string" || toolCallId.length === 0) return;
|
|
101
|
+
return {
|
|
102
|
+
type: "toolCall",
|
|
103
|
+
tool_call_id: toolCallId
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
function maybeStartSubagent(ns) {
|
|
107
|
+
if (ns.length !== depth + 1 || !hasPrefix(ns, scope)) return;
|
|
108
|
+
const key = nsKey(ns);
|
|
109
|
+
if (handles.has(key)) return;
|
|
110
|
+
const lc = lcByNs.get(key);
|
|
111
|
+
if (typeof lc !== "string" || lc.length === 0) return;
|
|
112
|
+
const messages = createMessagesTransformer(ns);
|
|
113
|
+
const messagesProjection = messages.init();
|
|
114
|
+
const toolCall = createToolCallTransformer(ns)();
|
|
115
|
+
const toolCallProjection = toolCall.init();
|
|
116
|
+
const nested = createSubagentTransformer(ns)();
|
|
117
|
+
const nestedProjection = nested.init();
|
|
118
|
+
let resolveOutput;
|
|
119
|
+
let rejectOutput;
|
|
120
|
+
const output = new Promise((resolve, reject) => {
|
|
121
|
+
resolveOutput = resolve;
|
|
122
|
+
rejectOutput = reject;
|
|
123
|
+
});
|
|
124
|
+
handles.set(key, {
|
|
125
|
+
key,
|
|
126
|
+
path: ns,
|
|
127
|
+
name: lc,
|
|
128
|
+
messages,
|
|
129
|
+
toolCall,
|
|
130
|
+
nested,
|
|
131
|
+
resolveOutput,
|
|
132
|
+
rejectOutput,
|
|
133
|
+
latestValues: void 0,
|
|
134
|
+
done: false
|
|
135
|
+
});
|
|
136
|
+
subagentsLog.push({
|
|
137
|
+
name: lc,
|
|
138
|
+
cause: deriveCause(ns),
|
|
139
|
+
output,
|
|
140
|
+
messages: messagesProjection.messages,
|
|
141
|
+
toolCalls: toolCallProjection.toolCalls,
|
|
142
|
+
subagents: nestedProjection.subagents
|
|
143
|
+
});
|
|
144
|
+
}
|
|
145
|
+
function finishHandle(handle, outcome) {
|
|
146
|
+
if (handle.done) return;
|
|
147
|
+
handle.done = true;
|
|
148
|
+
if (outcome.type === "resolve") handle.resolveOutput(handle.latestValues);
|
|
149
|
+
else handle.rejectOutput(outcome.error);
|
|
150
|
+
handle.messages.finalize?.();
|
|
151
|
+
handle.toolCall.finalize?.();
|
|
152
|
+
handle.nested.finalize?.();
|
|
153
|
+
}
|
|
154
|
+
return {
|
|
155
|
+
__native: true,
|
|
156
|
+
init: () => ({ subagents: subagentsLog }),
|
|
157
|
+
process(event) {
|
|
158
|
+
const ns = event.params.namespace;
|
|
159
|
+
const data = event.params.data;
|
|
160
|
+
const isTaskResult = event.method === "tasks" && isRecord(data) && "result" in data;
|
|
161
|
+
if (event.method === "tools" && isRecord(data) && data.event === "tool-started" && typeof data.tool_call_id === "string" && data.tool_call_id.length > 0) activeToolCallByNs.set(nsKey(ns), data.tool_call_id);
|
|
162
|
+
if (event.method === "tasks" && !isTaskResult) {
|
|
163
|
+
recordIdentity(ns, data);
|
|
164
|
+
recordPendingToolCalls(data);
|
|
165
|
+
maybeStartSubagent(ns);
|
|
166
|
+
}
|
|
167
|
+
for (const handle of handles.values()) {
|
|
168
|
+
if (handle.done) continue;
|
|
169
|
+
if (!hasPrefix(ns, handle.path)) continue;
|
|
170
|
+
handle.messages.process(event);
|
|
171
|
+
handle.toolCall.process(event);
|
|
172
|
+
handle.nested.process(event);
|
|
173
|
+
if (nsKey(ns) === handle.key) {
|
|
174
|
+
if (event.method === "values" && isRecord(data)) handle.latestValues = data;
|
|
175
|
+
else if (event.method === "lifecycle" && isRecord(data)) {
|
|
176
|
+
const status = data.event;
|
|
177
|
+
if (status === "completed" || status === "interrupted") finishHandle(handle, { type: "resolve" });
|
|
178
|
+
else if (status === "failed") finishHandle(handle, {
|
|
179
|
+
type: "reject",
|
|
180
|
+
error: /* @__PURE__ */ new Error(`Subagent ${handle.name} failed`)
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
return true;
|
|
186
|
+
},
|
|
187
|
+
finalize() {
|
|
188
|
+
for (const handle of handles.values()) finishHandle(handle, { type: "resolve" });
|
|
189
|
+
subagentsLog.close();
|
|
190
|
+
},
|
|
191
|
+
fail(err) {
|
|
192
|
+
for (const handle of handles.values()) finishHandle(handle, {
|
|
193
|
+
type: "reject",
|
|
194
|
+
error: err
|
|
195
|
+
});
|
|
196
|
+
subagentsLog.fail(err);
|
|
197
|
+
}
|
|
198
|
+
};
|
|
199
|
+
};
|
|
200
|
+
}
|
|
201
|
+
//#endregion
|
|
202
|
+
export { createSubagentTransformer };
|
|
203
|
+
|
|
204
|
+
//# sourceMappingURL=subagent.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"subagent.js","names":[],"sources":["../../../src/agents/transformers/subagent.ts"],"sourcesContent":["import {\n StreamChannel,\n createMessagesTransformer,\n type NativeStreamTransformer,\n type ProtocolEvent,\n type Namespace,\n type LifecycleCause,\n} from \"@langchain/langgraph\";\n\nimport { createToolCallTransformer } from \"./tool-call.js\";\nimport type { SubagentRunStream } from \"./types.js\";\n\ninterface SubagentProjection {\n subagents: AsyncIterable<SubagentRunStream>;\n}\n\n/** Per-subagent transformer instances, driven manually by the parent. */\ntype MessagesTransformer = ReturnType<typeof createMessagesTransformer>;\ntype ToolCallTransformer = ReturnType<\n ReturnType<typeof createToolCallTransformer>\n>;\ntype NestedSubagentTransformer = ReturnType<\n ReturnType<typeof createSubagentTransformer>\n>;\n\ninterface SubagentHandle {\n readonly key: string;\n readonly path: Namespace;\n readonly name: string;\n readonly messages: MessagesTransformer;\n readonly toolCall: ToolCallTransformer;\n readonly nested: NestedSubagentTransformer;\n readonly resolveOutput: (value: unknown) => void;\n readonly rejectOutput: (error: unknown) => void;\n latestValues: Record<string, unknown> | undefined;\n done: boolean;\n}\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\n/** Stable string key for a namespace. */\nfunction nsKey(ns: Namespace): string {\n return ns.join(\"\\u0000\");\n}\n\n/** Tests whether `ns` starts with every segment in `prefix`. */\nfunction hasPrefix(ns: Namespace, prefix: Namespace): boolean {\n if (prefix.length > ns.length) return false;\n for (let i = 0; i < prefix.length; i += 1) {\n if (ns[i] !== prefix[i]) return false;\n }\n return true;\n}\n\n/**\n * Creates a native transformer that surfaces nested named agents on\n * `run.subagents`.\n *\n * It watches `tasks` events to record each namespace's `lc_agent_name` (set by\n * `createAgent({ name })`) and the triggering tool call, then — for any nested\n * run one level below {@link scope} that carries an `lc_agent_name` — emits a\n * typed {@link SubagentRunStream} handle.\n *\n * Each handle is backed by its own per-subagent transformer instances\n * ({@link createMessagesTransformer}, {@link createToolCallTransformer}, and a\n * nested {@link createSubagentTransformer}) scoped to the subagent's namespace.\n * Every event in the subtree is fed straight into those transformers, which\n * self-filter by namespace; the subagent's final `output` is resolved from its\n * last `values` snapshot when its `lifecycle` completes.\n *\n * Marked `__native: true` — the `subagents` projection lands directly on the\n * `GraphRunStream` instance as `run.subagents`.\n *\n * @param scope - Namespace prefix this transformer is scoped to. The root agent\n * uses `[]`; nested handles use their subagent's namespace, so grandchild\n * subagents are discovered recursively.\n */\nexport function createSubagentTransformer(\n scope: Namespace = []\n): () => NativeStreamTransformer<SubagentProjection> {\n return () => {\n const subagentsLog = StreamChannel.local<SubagentRunStream>();\n /** `lc_agent_name` observed per namespace (first task event wins). */\n const lcByNs = new Map<string, string | undefined>();\n /** Triggering task id -> originating LLM `tool_call_id`. */\n const pendingToolCalls = new Map<string, string>();\n /**\n * Namespace key -> the `tool_call_id` of the most recent tool to start\n * executing there. A tool that invokes a subagent emits its `tool-started`\n * at the tools-node namespace (`tools:<task_id>`) where the subagent then\n * roots, so this is the tool call that caused the subagent.\n */\n const activeToolCallByNs = new Map<string, string>();\n const handles = new Map<string, SubagentHandle>();\n const depth = scope.length;\n\n function recordIdentity(ns: Namespace, data: unknown): void {\n const key = nsKey(ns);\n if (lcByNs.has(key)) return;\n const metadata =\n isRecord(data) && isRecord(data.metadata) ? data.metadata : undefined;\n const lc = metadata?.lc_agent_name;\n lcByNs.set(key, typeof lc === \"string\" ? lc : undefined);\n }\n\n function recordPendingToolCalls(data: unknown): void {\n if (!isRecord(data)) return;\n const taskId = data.id;\n if (typeof taskId !== \"string\") return;\n const input = data.input;\n let toolCallId: string | undefined;\n if (isRecord(input) && isRecord(input.tool_call)) {\n const candidate = input.tool_call.id;\n if (typeof candidate === \"string\") toolCallId = candidate;\n } else if (Array.isArray(input)) {\n for (const toolCall of input) {\n if (isRecord(toolCall) && typeof toolCall.id === \"string\") {\n toolCallId = toolCall.id;\n break;\n }\n }\n }\n if (toolCallId != null) pendingToolCalls.set(taskId, toolCallId);\n }\n\n /**\n * Derive the `toolCall` cause for a named-subagent namespace.\n *\n * Primary signal: the tool whose `tool-started` event fired at the\n * subagent's own namespace (the tools node it roots under). Fallback: the\n * namespace segment's task id (`node:<task_id>`) joined to a tool call\n * harvested from a `tool_call_with_context`-shaped task input, so the\n * derivation stays correct if that shape reaches the stream in the future.\n */\n function deriveCause(ns: Namespace): LifecycleCause | undefined {\n const active = activeToolCallByNs.get(nsKey(ns));\n if (typeof active === \"string\" && active.length > 0) {\n return { type: \"toolCall\", tool_call_id: active } as LifecycleCause;\n }\n const segment = ns[ns.length - 1];\n const colon = segment.indexOf(\":\");\n if (colon === -1) return undefined;\n const triggerCallId = segment.slice(colon + 1);\n if (triggerCallId.length === 0) return undefined;\n const toolCallId = pendingToolCalls.get(triggerCallId);\n if (typeof toolCallId !== \"string\" || toolCallId.length === 0) {\n return undefined;\n }\n return { type: \"toolCall\", tool_call_id: toolCallId } as LifecycleCause;\n }\n\n function maybeStartSubagent(ns: Namespace): void {\n if (ns.length !== depth + 1 || !hasPrefix(ns, scope)) return;\n const key = nsKey(ns);\n if (handles.has(key)) return;\n const lc = lcByNs.get(key);\n // Only surface nested runs carrying an `lc_agent_name`; plain subgraphs\n // (no name) are excluded so `run.subagents` stays agent-only.\n if (typeof lc !== \"string\" || lc.length === 0) return;\n\n // Per-subagent transformers, each scoped to the subagent's namespace so\n // they pick out only the subagent's own model node / tools / nested\n // agents when fed the full event stream.\n const messages = createMessagesTransformer(ns);\n const messagesProjection = messages.init();\n const toolCall = createToolCallTransformer(ns)();\n const toolCallProjection = toolCall.init();\n const nested = createSubagentTransformer(ns)();\n const nestedProjection = nested.init();\n\n let resolveOutput!: (value: unknown) => void;\n let rejectOutput!: (error: unknown) => void;\n const output = new Promise<Record<string, unknown>>((resolve, reject) => {\n resolveOutput = resolve as (value: unknown) => void;\n rejectOutput = reject;\n });\n\n handles.set(key, {\n key,\n path: ns,\n name: lc,\n messages,\n toolCall,\n nested,\n resolveOutput,\n rejectOutput,\n latestValues: undefined,\n done: false,\n });\n\n subagentsLog.push({\n name: lc,\n cause: deriveCause(ns),\n output,\n messages: messagesProjection.messages,\n toolCalls: toolCallProjection.toolCalls,\n subagents: nestedProjection.subagents,\n });\n }\n\n function finishHandle(\n handle: SubagentHandle,\n outcome: { type: \"resolve\" } | { type: \"reject\"; error: unknown }\n ): void {\n if (handle.done) return;\n handle.done = true;\n if (outcome.type === \"resolve\") {\n handle.resolveOutput(handle.latestValues);\n } else {\n handle.rejectOutput(outcome.error);\n }\n handle.messages.finalize?.();\n handle.toolCall.finalize?.();\n handle.nested.finalize?.();\n }\n\n return {\n __native: true as const,\n\n init: () => ({ subagents: subagentsLog }),\n\n process(event: ProtocolEvent): boolean {\n const ns = event.params.namespace;\n const data = event.params.data;\n const isTaskResult =\n event.method === \"tasks\" && isRecord(data) && \"result\" in data;\n\n // Track the tool currently executing at each namespace. A subagent's\n // dispatching tool starts at the same namespace the subagent roots\n // under, so this records the cause before the subagent is discovered.\n if (\n event.method === \"tools\" &&\n isRecord(data) &&\n data.event === \"tool-started\" &&\n typeof data.tool_call_id === \"string\" &&\n data.tool_call_id.length > 0\n ) {\n activeToolCallByNs.set(nsKey(ns), data.tool_call_id);\n }\n\n // A task start: record identity / tool call, then discover a subagent\n // boundary *before* fanning out so the new handle receives its own\n // subtree events (which Pregel emits after the parent-namespace task).\n if (event.method === \"tasks\" && !isTaskResult) {\n recordIdentity(ns, data);\n recordPendingToolCalls(data);\n maybeStartSubagent(ns);\n }\n\n // Fan the event out to every active subagent whose subtree contains it.\n // The per-subagent transformers self-filter by namespace depth.\n for (const handle of handles.values()) {\n if (handle.done) continue;\n if (!hasPrefix(ns, handle.path)) continue;\n\n handle.messages.process(event);\n handle.toolCall.process(event);\n handle.nested.process(event);\n\n // Track the subagent's own (root-level) state and resolve its\n // `output` from the last snapshot when its lifecycle completes.\n if (nsKey(ns) === handle.key) {\n if (event.method === \"values\" && isRecord(data)) {\n handle.latestValues = data;\n } else if (event.method === \"lifecycle\" && isRecord(data)) {\n const status = data.event;\n if (status === \"completed\" || status === \"interrupted\") {\n finishHandle(handle, { type: \"resolve\" });\n } else if (status === \"failed\") {\n finishHandle(handle, {\n type: \"reject\",\n error: new Error(`Subagent ${handle.name} failed`),\n });\n }\n }\n }\n }\n\n return true;\n },\n\n finalize(): void {\n for (const handle of handles.values()) {\n finishHandle(handle, { type: \"resolve\" });\n }\n subagentsLog.close();\n },\n\n fail(err: unknown): void {\n for (const handle of handles.values()) {\n finishHandle(handle, { type: \"reject\", error: err });\n }\n subagentsLog.fail(err);\n },\n };\n };\n}\n"],"mappings":"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"}
|
|
@@ -1,19 +1,7 @@
|
|
|
1
|
-
require("
|
|
1
|
+
require("../../_virtual/_rolldown/runtime.cjs");
|
|
2
2
|
let _langchain_core_messages = require("@langchain/core/messages");
|
|
3
3
|
let _langchain_langgraph = require("@langchain/langgraph");
|
|
4
|
-
//#region src/agents/
|
|
5
|
-
/**
|
|
6
|
-
* Agent-level streaming support (experimental).
|
|
7
|
-
*
|
|
8
|
-
* Provides native stream transformer factories for tool calls and
|
|
9
|
-
* middleware events. When marked `__native: true`, their projections
|
|
10
|
-
* are assigned directly onto the `GraphRunStream` instance by
|
|
11
|
-
* `createGraphRunStream` in langgraph-core — no subclass or wrapper
|
|
12
|
-
* needed.
|
|
13
|
-
*
|
|
14
|
-
* See protocol proposal §15 (In-Process Streaming Interface) and §16
|
|
15
|
-
* (Native Stream Transformers).
|
|
16
|
-
*/
|
|
4
|
+
//#region src/agents/transformers/tool-call.ts
|
|
17
5
|
/**
|
|
18
6
|
* Returns true when `ns` belongs to the agent's own graph — i.e. it
|
|
19
7
|
* starts with `path` and is at most one level deeper (the agent's
|
|
@@ -173,4 +161,4 @@ function createToolCallTransformer(path) {
|
|
|
173
161
|
//#endregion
|
|
174
162
|
exports.createToolCallTransformer = createToolCallTransformer;
|
|
175
163
|
|
|
176
|
-
//# sourceMappingURL=
|
|
164
|
+
//# sourceMappingURL=tool-call.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tool-call.cjs","names":["ToolMessage","StreamChannel"],"sources":["../../../src/agents/transformers/tool-call.ts"],"sourcesContent":["import {\n StreamChannel,\n type NativeStreamTransformer,\n type ProtocolEvent,\n type ToolCallStream,\n type ToolCallStatus,\n type ToolsEventData,\n type Namespace,\n} from \"@langchain/langgraph\";\nimport { ToolMessage } from \"@langchain/core/messages\";\n\ninterface ToolCallProjection {\n toolCalls: AsyncIterable<ToolCallStream>;\n}\n\n/**\n * Returns true when `ns` belongs to the agent's own graph — i.e. it\n * starts with `path` and is at most one level deeper (the agent's\n * internal nodes like `tools`, `model_request`, etc.).\n *\n * Events from subagent subgraphs (two or more levels deeper) are\n * excluded, so `run.toolCalls` / `run.middleware` only show events\n * from the agent itself, not from its subagents.\n */\nfunction isOwnEvent(ns: Namespace, path: Namespace): boolean {\n if (ns.length < path.length || ns.length > path.length + 1) return false;\n for (let i = 0; i < path.length; i += 1) {\n if (ns[i] !== path[i]) return false;\n }\n return true;\n}\n\nfunction isHeadlessToolInterruptError(\n message: string,\n toolCallId: string | undefined\n): boolean {\n try {\n const parsed = JSON.parse(message) as unknown;\n if (!Array.isArray(parsed)) return false;\n return parsed.some((entry) => {\n if (entry == null || typeof entry !== \"object\") return false;\n const value = (entry as { value?: unknown }).value;\n if (value == null || typeof value !== \"object\") return false;\n const payload = value as {\n type?: unknown;\n toolCall?: { id?: unknown };\n };\n return (\n payload.type === \"tool\" &&\n (toolCallId == null ||\n payload.toolCall?.id == null ||\n payload.toolCall.id === toolCallId)\n );\n });\n } catch {\n return false;\n }\n}\n\n/**\n * Detects serialized LangChain `ToolMessage` values that can appear on\n * `tool-finished.output` after crossing a protocol or serialization boundary.\n *\n * @example\n * ```ts\n * {\n * lc: 1,\n * type: \"constructor\",\n * id: [\"langchain_core\", \"messages\", \"ToolMessage\"],\n * kwargs: { content: \"raw tool result\", tool_call_id: \"call_1\" }\n * }\n * ```\n */\nfunction isSerializedToolMessage(\n value: unknown\n): value is { kwargs?: { content?: unknown } } {\n if (value == null || typeof value !== \"object\") return false;\n const record = value as Record<string, unknown>;\n if (record.type !== \"constructor\" || !Array.isArray(record.id)) return false;\n return record.id[record.id.length - 1] === \"ToolMessage\";\n}\n\nfunction normalizeToolOutput(output: unknown): unknown {\n if (ToolMessage.isInstance(output)) {\n return output.content;\n }\n if (isSerializedToolMessage(output)) {\n return output.kwargs?.content;\n }\n return output;\n}\n\n/**\n * Creates a native transformer that correlates `tools` channel events\n * into per-call {@link ToolCallStream} objects.\n *\n * Marked `__native: true` — projection keys land directly on the\n * `GraphRunStream` instance as `run.toolCalls`.\n */\nexport function createToolCallTransformer(\n path: Namespace\n): () => NativeStreamTransformer<ToolCallProjection> {\n return () => {\n const toolCallsLog = StreamChannel.local<ToolCallStream>();\n\n const pendingCalls = new Map<\n string,\n {\n resolveOutput: (v: unknown) => void;\n rejectOutput: (e: unknown) => void;\n resolveStatus: (v: ToolCallStatus) => void;\n resolveError: (v: string | undefined) => void;\n }\n >();\n\n function createToolCallEntry(\n callId: string,\n name: string,\n rawInput: unknown\n ): void {\n if (pendingCalls.has(callId)) return;\n const input =\n typeof rawInput === \"string\" ? JSON.parse(rawInput) : rawInput;\n\n let resolveOutput!: (v: unknown) => void;\n let rejectOutput!: (e: unknown) => void;\n let resolveStatus!: (v: ToolCallStatus) => void;\n let resolveError!: (v: string | undefined) => void;\n\n const output = new Promise<unknown>((res, rej) => {\n resolveOutput = res;\n rejectOutput = rej;\n });\n const status = new Promise<ToolCallStatus>((res) => {\n resolveStatus = res;\n });\n const error = new Promise<string | undefined>((res) => {\n resolveError = res;\n });\n\n pendingCalls.set(callId, {\n resolveOutput,\n rejectOutput,\n resolveStatus,\n resolveError,\n });\n\n toolCallsLog.push({\n name,\n callId,\n input,\n output,\n status,\n error,\n } as ToolCallStream);\n }\n\n return {\n __native: true as const,\n\n init: () => ({\n toolCalls: toolCallsLog,\n }),\n\n process(event: ProtocolEvent): boolean {\n /**\n * Only process events that are at the same depth as the agent's graph.\n */\n if (!isOwnEvent(event.params.namespace, path)) return true;\n\n if (event.method === \"messages\") {\n const data = event.params.data as Record<string, unknown>;\n if (data.event === \"content-block-finish\") {\n const cb = (data.contentBlock ?? data.content_block) as\n | Record<string, unknown>\n | undefined;\n if (cb?.type === \"tool_call\") {\n createToolCallEntry(\n String(cb.id ?? \"\"),\n String(cb.name ?? \"\"),\n cb.args ?? cb.input\n );\n }\n }\n }\n\n if (event.method === \"tools\") {\n const data = event.params.data as ToolsEventData;\n const toolCallId = (data as Record<string, unknown>)\n .tool_call_id as string;\n\n if (data.event === \"tool-started\") {\n createToolCallEntry(\n toolCallId,\n ((data as Record<string, unknown>).tool_name as string) ??\n \"unknown\",\n (data as Record<string, unknown>).input\n );\n }\n\n const pending = toolCallId ? pendingCalls.get(toolCallId) : undefined;\n\n if (pending) {\n if (data.event === \"tool-finished\") {\n pending.resolveOutput(\n normalizeToolOutput((data as Record<string, unknown>).output)\n );\n pending.resolveStatus(\"finished\");\n pending.resolveError(undefined);\n pendingCalls.delete(toolCallId);\n } else if (data.event === \"tool-error\") {\n const message =\n ((data as Record<string, unknown>).message as string) ??\n \"unknown error\";\n if (isHeadlessToolInterruptError(message, toolCallId)) {\n return true;\n }\n pending.rejectOutput(new Error(message));\n pending.resolveStatus(\"error\");\n pending.resolveError(message);\n pendingCalls.delete(toolCallId);\n }\n }\n }\n\n return true;\n },\n\n finalize(): void {\n for (const pending of pendingCalls.values()) {\n pending.resolveStatus(\"finished\");\n pending.resolveError(undefined);\n pending.resolveOutput(undefined);\n }\n pendingCalls.clear();\n toolCallsLog.close();\n },\n\n fail(err: unknown): void {\n for (const pending of pendingCalls.values()) {\n pending.resolveStatus(\"error\");\n pending.resolveError(\n err instanceof Error ? err.message : String(err)\n );\n pending.rejectOutput(err);\n }\n pendingCalls.clear();\n toolCallsLog.fail(err);\n },\n };\n };\n}\n"],"mappings":"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"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { Namespace, NativeStreamTransformer, ToolCallStream } from "@langchain/langgraph";
|
|
2
|
+
|
|
3
|
+
//#region src/agents/transformers/tool-call.d.ts
|
|
4
|
+
interface ToolCallProjection {
|
|
5
|
+
toolCalls: AsyncIterable<ToolCallStream>;
|
|
6
|
+
}
|
|
7
|
+
/**
|
|
8
|
+
* Creates a native transformer that correlates `tools` channel events
|
|
9
|
+
* into per-call {@link ToolCallStream} objects.
|
|
10
|
+
*
|
|
11
|
+
* Marked `__native: true` — projection keys land directly on the
|
|
12
|
+
* `GraphRunStream` instance as `run.toolCalls`.
|
|
13
|
+
*/
|
|
14
|
+
declare function createToolCallTransformer(path: Namespace): () => NativeStreamTransformer<ToolCallProjection>;
|
|
15
|
+
//#endregion
|
|
16
|
+
export { createToolCallTransformer };
|
|
17
|
+
//# sourceMappingURL=tool-call.d.cts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tool-call.d.cts","names":[],"sources":["../../../src/agents/transformers/tool-call.ts"],"mappings":";;;UAWU,kBAAA;EACR,SAAA,EAAW,aAAA,CAAc,cAAA;AAAA;;;;;;;;iBAuFX,yBAAA,CACd,IAAA,EAAM,SAAA,SACC,uBAAA,CAAwB,kBAAA"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { Namespace, NativeStreamTransformer, ToolCallStream } from "@langchain/langgraph";
|
|
2
|
+
|
|
3
|
+
//#region src/agents/transformers/tool-call.d.ts
|
|
4
|
+
interface ToolCallProjection {
|
|
5
|
+
toolCalls: AsyncIterable<ToolCallStream>;
|
|
6
|
+
}
|
|
7
|
+
/**
|
|
8
|
+
* Creates a native transformer that correlates `tools` channel events
|
|
9
|
+
* into per-call {@link ToolCallStream} objects.
|
|
10
|
+
*
|
|
11
|
+
* Marked `__native: true` — projection keys land directly on the
|
|
12
|
+
* `GraphRunStream` instance as `run.toolCalls`.
|
|
13
|
+
*/
|
|
14
|
+
declare function createToolCallTransformer(path: Namespace): () => NativeStreamTransformer<ToolCallProjection>;
|
|
15
|
+
//#endregion
|
|
16
|
+
export { createToolCallTransformer };
|
|
17
|
+
//# sourceMappingURL=tool-call.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tool-call.d.ts","names":[],"sources":["../../../src/agents/transformers/tool-call.ts"],"mappings":";;;UAWU,kBAAA;EACR,SAAA,EAAW,aAAA,CAAc,cAAA;AAAA;;;;;;;;iBAuFX,yBAAA,CACd,IAAA,EAAM,SAAA,SACC,uBAAA,CAAwB,kBAAA"}
|
|
@@ -1,18 +1,6 @@
|
|
|
1
1
|
import { ToolMessage } from "@langchain/core/messages";
|
|
2
2
|
import { StreamChannel } from "@langchain/langgraph";
|
|
3
|
-
//#region src/agents/
|
|
4
|
-
/**
|
|
5
|
-
* Agent-level streaming support (experimental).
|
|
6
|
-
*
|
|
7
|
-
* Provides native stream transformer factories for tool calls and
|
|
8
|
-
* middleware events. When marked `__native: true`, their projections
|
|
9
|
-
* are assigned directly onto the `GraphRunStream` instance by
|
|
10
|
-
* `createGraphRunStream` in langgraph-core — no subclass or wrapper
|
|
11
|
-
* needed.
|
|
12
|
-
*
|
|
13
|
-
* See protocol proposal §15 (In-Process Streaming Interface) and §16
|
|
14
|
-
* (Native Stream Transformers).
|
|
15
|
-
*/
|
|
3
|
+
//#region src/agents/transformers/tool-call.ts
|
|
16
4
|
/**
|
|
17
5
|
* Returns true when `ns` belongs to the agent's own graph — i.e. it
|
|
18
6
|
* starts with `path` and is at most one level deeper (the agent's
|
|
@@ -172,4 +160,4 @@ function createToolCallTransformer(path) {
|
|
|
172
160
|
//#endregion
|
|
173
161
|
export { createToolCallTransformer };
|
|
174
162
|
|
|
175
|
-
//# sourceMappingURL=
|
|
163
|
+
//# sourceMappingURL=tool-call.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tool-call.js","names":[],"sources":["../../../src/agents/transformers/tool-call.ts"],"sourcesContent":["import {\n StreamChannel,\n type NativeStreamTransformer,\n type ProtocolEvent,\n type ToolCallStream,\n type ToolCallStatus,\n type ToolsEventData,\n type Namespace,\n} from \"@langchain/langgraph\";\nimport { ToolMessage } from \"@langchain/core/messages\";\n\ninterface ToolCallProjection {\n toolCalls: AsyncIterable<ToolCallStream>;\n}\n\n/**\n * Returns true when `ns` belongs to the agent's own graph — i.e. it\n * starts with `path` and is at most one level deeper (the agent's\n * internal nodes like `tools`, `model_request`, etc.).\n *\n * Events from subagent subgraphs (two or more levels deeper) are\n * excluded, so `run.toolCalls` / `run.middleware` only show events\n * from the agent itself, not from its subagents.\n */\nfunction isOwnEvent(ns: Namespace, path: Namespace): boolean {\n if (ns.length < path.length || ns.length > path.length + 1) return false;\n for (let i = 0; i < path.length; i += 1) {\n if (ns[i] !== path[i]) return false;\n }\n return true;\n}\n\nfunction isHeadlessToolInterruptError(\n message: string,\n toolCallId: string | undefined\n): boolean {\n try {\n const parsed = JSON.parse(message) as unknown;\n if (!Array.isArray(parsed)) return false;\n return parsed.some((entry) => {\n if (entry == null || typeof entry !== \"object\") return false;\n const value = (entry as { value?: unknown }).value;\n if (value == null || typeof value !== \"object\") return false;\n const payload = value as {\n type?: unknown;\n toolCall?: { id?: unknown };\n };\n return (\n payload.type === \"tool\" &&\n (toolCallId == null ||\n payload.toolCall?.id == null ||\n payload.toolCall.id === toolCallId)\n );\n });\n } catch {\n return false;\n }\n}\n\n/**\n * Detects serialized LangChain `ToolMessage` values that can appear on\n * `tool-finished.output` after crossing a protocol or serialization boundary.\n *\n * @example\n * ```ts\n * {\n * lc: 1,\n * type: \"constructor\",\n * id: [\"langchain_core\", \"messages\", \"ToolMessage\"],\n * kwargs: { content: \"raw tool result\", tool_call_id: \"call_1\" }\n * }\n * ```\n */\nfunction isSerializedToolMessage(\n value: unknown\n): value is { kwargs?: { content?: unknown } } {\n if (value == null || typeof value !== \"object\") return false;\n const record = value as Record<string, unknown>;\n if (record.type !== \"constructor\" || !Array.isArray(record.id)) return false;\n return record.id[record.id.length - 1] === \"ToolMessage\";\n}\n\nfunction normalizeToolOutput(output: unknown): unknown {\n if (ToolMessage.isInstance(output)) {\n return output.content;\n }\n if (isSerializedToolMessage(output)) {\n return output.kwargs?.content;\n }\n return output;\n}\n\n/**\n * Creates a native transformer that correlates `tools` channel events\n * into per-call {@link ToolCallStream} objects.\n *\n * Marked `__native: true` — projection keys land directly on the\n * `GraphRunStream` instance as `run.toolCalls`.\n */\nexport function createToolCallTransformer(\n path: Namespace\n): () => NativeStreamTransformer<ToolCallProjection> {\n return () => {\n const toolCallsLog = StreamChannel.local<ToolCallStream>();\n\n const pendingCalls = new Map<\n string,\n {\n resolveOutput: (v: unknown) => void;\n rejectOutput: (e: unknown) => void;\n resolveStatus: (v: ToolCallStatus) => void;\n resolveError: (v: string | undefined) => void;\n }\n >();\n\n function createToolCallEntry(\n callId: string,\n name: string,\n rawInput: unknown\n ): void {\n if (pendingCalls.has(callId)) return;\n const input =\n typeof rawInput === \"string\" ? JSON.parse(rawInput) : rawInput;\n\n let resolveOutput!: (v: unknown) => void;\n let rejectOutput!: (e: unknown) => void;\n let resolveStatus!: (v: ToolCallStatus) => void;\n let resolveError!: (v: string | undefined) => void;\n\n const output = new Promise<unknown>((res, rej) => {\n resolveOutput = res;\n rejectOutput = rej;\n });\n const status = new Promise<ToolCallStatus>((res) => {\n resolveStatus = res;\n });\n const error = new Promise<string | undefined>((res) => {\n resolveError = res;\n });\n\n pendingCalls.set(callId, {\n resolveOutput,\n rejectOutput,\n resolveStatus,\n resolveError,\n });\n\n toolCallsLog.push({\n name,\n callId,\n input,\n output,\n status,\n error,\n } as ToolCallStream);\n }\n\n return {\n __native: true as const,\n\n init: () => ({\n toolCalls: toolCallsLog,\n }),\n\n process(event: ProtocolEvent): boolean {\n /**\n * Only process events that are at the same depth as the agent's graph.\n */\n if (!isOwnEvent(event.params.namespace, path)) return true;\n\n if (event.method === \"messages\") {\n const data = event.params.data as Record<string, unknown>;\n if (data.event === \"content-block-finish\") {\n const cb = (data.contentBlock ?? data.content_block) as\n | Record<string, unknown>\n | undefined;\n if (cb?.type === \"tool_call\") {\n createToolCallEntry(\n String(cb.id ?? \"\"),\n String(cb.name ?? \"\"),\n cb.args ?? cb.input\n );\n }\n }\n }\n\n if (event.method === \"tools\") {\n const data = event.params.data as ToolsEventData;\n const toolCallId = (data as Record<string, unknown>)\n .tool_call_id as string;\n\n if (data.event === \"tool-started\") {\n createToolCallEntry(\n toolCallId,\n ((data as Record<string, unknown>).tool_name as string) ??\n \"unknown\",\n (data as Record<string, unknown>).input\n );\n }\n\n const pending = toolCallId ? pendingCalls.get(toolCallId) : undefined;\n\n if (pending) {\n if (data.event === \"tool-finished\") {\n pending.resolveOutput(\n normalizeToolOutput((data as Record<string, unknown>).output)\n );\n pending.resolveStatus(\"finished\");\n pending.resolveError(undefined);\n pendingCalls.delete(toolCallId);\n } else if (data.event === \"tool-error\") {\n const message =\n ((data as Record<string, unknown>).message as string) ??\n \"unknown error\";\n if (isHeadlessToolInterruptError(message, toolCallId)) {\n return true;\n }\n pending.rejectOutput(new Error(message));\n pending.resolveStatus(\"error\");\n pending.resolveError(message);\n pendingCalls.delete(toolCallId);\n }\n }\n }\n\n return true;\n },\n\n finalize(): void {\n for (const pending of pendingCalls.values()) {\n pending.resolveStatus(\"finished\");\n pending.resolveError(undefined);\n pending.resolveOutput(undefined);\n }\n pendingCalls.clear();\n toolCallsLog.close();\n },\n\n fail(err: unknown): void {\n for (const pending of pendingCalls.values()) {\n pending.resolveStatus(\"error\");\n pending.resolveError(\n err instanceof Error ? err.message : String(err)\n );\n pending.rejectOutput(err);\n }\n pendingCalls.clear();\n toolCallsLog.fail(err);\n },\n };\n };\n}\n"],"mappings":";;;;;;;;;;;;AAwBA,SAAS,WAAW,IAAe,MAA0B;AAC3D,KAAI,GAAG,SAAS,KAAK,UAAU,GAAG,SAAS,KAAK,SAAS,EAAG,QAAO;AACnE,MAAK,IAAI,IAAI,GAAG,IAAI,KAAK,QAAQ,KAAK,EACpC,KAAI,GAAG,OAAO,KAAK,GAAI,QAAO;AAEhC,QAAO;;AAGT,SAAS,6BACP,SACA,YACS;AACT,KAAI;EACF,MAAM,SAAS,KAAK,MAAM,QAAQ;AAClC,MAAI,CAAC,MAAM,QAAQ,OAAO,CAAE,QAAO;AACnC,SAAO,OAAO,MAAM,UAAU;AAC5B,OAAI,SAAS,QAAQ,OAAO,UAAU,SAAU,QAAO;GACvD,MAAM,QAAS,MAA8B;AAC7C,OAAI,SAAS,QAAQ,OAAO,UAAU,SAAU,QAAO;GACvD,MAAM,UAAU;AAIhB,UACE,QAAQ,SAAS,WAChB,cAAc,QACb,QAAQ,UAAU,MAAM,QACxB,QAAQ,SAAS,OAAO;IAE5B;SACI;AACN,SAAO;;;;;;;;;;;;;;;;;AAkBX,SAAS,wBACP,OAC6C;AAC7C,KAAI,SAAS,QAAQ,OAAO,UAAU,SAAU,QAAO;CACvD,MAAM,SAAS;AACf,KAAI,OAAO,SAAS,iBAAiB,CAAC,MAAM,QAAQ,OAAO,GAAG,CAAE,QAAO;AACvE,QAAO,OAAO,GAAG,OAAO,GAAG,SAAS,OAAO;;AAG7C,SAAS,oBAAoB,QAA0B;AACrD,KAAI,YAAY,WAAW,OAAO,CAChC,QAAO,OAAO;AAEhB,KAAI,wBAAwB,OAAO,CACjC,QAAO,OAAO,QAAQ;AAExB,QAAO;;;;;;;;;AAUT,SAAgB,0BACd,MACmD;AACnD,cAAa;EACX,MAAM,eAAe,cAAc,OAAuB;EAE1D,MAAM,+BAAe,IAAI,KAQtB;EAEH,SAAS,oBACP,QACA,MACA,UACM;AACN,OAAI,aAAa,IAAI,OAAO,CAAE;GAC9B,MAAM,QACJ,OAAO,aAAa,WAAW,KAAK,MAAM,SAAS,GAAG;GAExD,IAAI;GACJ,IAAI;GACJ,IAAI;GACJ,IAAI;GAEJ,MAAM,SAAS,IAAI,SAAkB,KAAK,QAAQ;AAChD,oBAAgB;AAChB,mBAAe;KACf;GACF,MAAM,SAAS,IAAI,SAAyB,QAAQ;AAClD,oBAAgB;KAChB;GACF,MAAM,QAAQ,IAAI,SAA6B,QAAQ;AACrD,mBAAe;KACf;AAEF,gBAAa,IAAI,QAAQ;IACvB;IACA;IACA;IACA;IACD,CAAC;AAEF,gBAAa,KAAK;IAChB;IACA;IACA;IACA;IACA;IACA;IACD,CAAmB;;AAGtB,SAAO;GACL,UAAU;GAEV,aAAa,EACX,WAAW,cACZ;GAED,QAAQ,OAA+B;;;;AAIrC,QAAI,CAAC,WAAW,MAAM,OAAO,WAAW,KAAK,CAAE,QAAO;AAEtD,QAAI,MAAM,WAAW,YAAY;KAC/B,MAAM,OAAO,MAAM,OAAO;AAC1B,SAAI,KAAK,UAAU,wBAAwB;MACzC,MAAM,KAAM,KAAK,gBAAgB,KAAK;AAGtC,UAAI,IAAI,SAAS,YACf,qBACE,OAAO,GAAG,MAAM,GAAG,EACnB,OAAO,GAAG,QAAQ,GAAG,EACrB,GAAG,QAAQ,GAAG,MACf;;;AAKP,QAAI,MAAM,WAAW,SAAS;KAC5B,MAAM,OAAO,MAAM,OAAO;KAC1B,MAAM,aAAc,KACjB;AAEH,SAAI,KAAK,UAAU,eACjB,qBACE,YACE,KAAiC,aACjC,WACD,KAAiC,MACnC;KAGH,MAAM,UAAU,aAAa,aAAa,IAAI,WAAW,GAAG,KAAA;AAE5D,SAAI;UACE,KAAK,UAAU,iBAAiB;AAClC,eAAQ,cACN,oBAAqB,KAAiC,OAAO,CAC9D;AACD,eAAQ,cAAc,WAAW;AACjC,eAAQ,aAAa,KAAA,EAAU;AAC/B,oBAAa,OAAO,WAAW;iBACtB,KAAK,UAAU,cAAc;OACtC,MAAM,UACF,KAAiC,WACnC;AACF,WAAI,6BAA6B,SAAS,WAAW,CACnD,QAAO;AAET,eAAQ,aAAa,IAAI,MAAM,QAAQ,CAAC;AACxC,eAAQ,cAAc,QAAQ;AAC9B,eAAQ,aAAa,QAAQ;AAC7B,oBAAa,OAAO,WAAW;;;;AAKrC,WAAO;;GAGT,WAAiB;AACf,SAAK,MAAM,WAAW,aAAa,QAAQ,EAAE;AAC3C,aAAQ,cAAc,WAAW;AACjC,aAAQ,aAAa,KAAA,EAAU;AAC/B,aAAQ,cAAc,KAAA,EAAU;;AAElC,iBAAa,OAAO;AACpB,iBAAa,OAAO;;GAGtB,KAAK,KAAoB;AACvB,SAAK,MAAM,WAAW,aAAa,QAAQ,EAAE;AAC3C,aAAQ,cAAc,QAAQ;AAC9B,aAAQ,aACN,eAAe,QAAQ,IAAI,UAAU,OAAO,IAAI,CACjD;AACD,aAAQ,aAAa,IAAI;;AAE3B,iBAAa,OAAO;AACpB,iBAAa,KAAK,IAAI;;GAEzB"}
|
|
@@ -1,11 +1,10 @@
|
|
|
1
1
|
import { ClientTool, DynamicStructuredTool, ServerTool, StructuredToolInterface } from "@langchain/core/tools";
|
|
2
|
-
import {
|
|
2
|
+
import { ChatModelStream, GraphRunStream, LifecycleCause, StreamTransformer, ToolCallStream } from "@langchain/langgraph";
|
|
3
3
|
|
|
4
|
-
//#region src/agents/
|
|
4
|
+
//#region src/agents/transformers/types.d.ts
|
|
5
5
|
/**
|
|
6
6
|
* Infers the merged extensions shape from a tuple of stream transformer
|
|
7
|
-
* factories.
|
|
8
|
-
* is not exported from the package's public surface.
|
|
7
|
+
* factories.
|
|
9
8
|
*
|
|
10
9
|
* Given `[() => StreamTransformer<{ a: number }>, () => StreamTransformer<{ b: string }>]`,
|
|
11
10
|
* produces `{ a: number } & { b: string }`.
|
|
@@ -28,6 +27,41 @@ type ToolOutputOf<T> = T extends DynamicStructuredTool<any, any, any, infer Tool
|
|
|
28
27
|
* plain `(ClientTool | ServerTool)[]` without literal name types.
|
|
29
28
|
*/
|
|
30
29
|
type ToolCallStreamUnion<TTools extends readonly (ClientTool | ServerTool)[]> = { [K in keyof TTools]: ToolCallStream<ToolNameOf<TTools[K]>, ToolInputOf<TTools[K]>, ToolOutputOf<TTools[K]>> }[number];
|
|
30
|
+
/**
|
|
31
|
+
* A nested named-agent execution surfaced on `run.subagents`.
|
|
32
|
+
*
|
|
33
|
+
* A subagent is a nested `createAgent` run whose `lc_agent_name` differs from
|
|
34
|
+
* its parent's — e.g. a `createAgent({ name })` invoked inside a tool body. The
|
|
35
|
+
* handle exposes scoped projections (`messages`, `toolCalls`, nested
|
|
36
|
+
* `subagents`) plus the resolved {@link name}, the {@link cause} that triggered
|
|
37
|
+
* it, and the subagent's final {@link output} state.
|
|
38
|
+
*
|
|
39
|
+
* This is a self-contained handle (no subclass of `SubgraphRunStream`): its
|
|
40
|
+
* projections are produced by per-subagent transformer instances that
|
|
41
|
+
* {@link createSubagentTransformer} drives directly.
|
|
42
|
+
*
|
|
43
|
+
* @typeParam TValues - Shape of the subagent's final output state.
|
|
44
|
+
* @typeParam TTools - Tuple of tools the subagent may call, used to type
|
|
45
|
+
* {@link toolCalls}.
|
|
46
|
+
*/
|
|
47
|
+
interface SubagentRunStream<TValues = Record<string, unknown>, TTools extends readonly (ClientTool | ServerTool)[] = readonly (ClientTool | ServerTool)[]> {
|
|
48
|
+
/** The subagent's `lc_agent_name` (set by `createAgent({ name })`). */
|
|
49
|
+
readonly name: string;
|
|
50
|
+
/**
|
|
51
|
+
* The tool call that dispatched this subagent, as
|
|
52
|
+
* `{ type: "toolCall", tool_call_id }`, or `undefined` when the originating
|
|
53
|
+
* tool call could not be recovered (e.g. it was not triggered from a tool).
|
|
54
|
+
*/
|
|
55
|
+
readonly cause: LifecycleCause | undefined;
|
|
56
|
+
/** Resolves with the subagent's final state once it completes. */
|
|
57
|
+
readonly output: Promise<TValues>;
|
|
58
|
+
/** Per-message chat-model token streams scoped to this subagent. */
|
|
59
|
+
readonly messages: AsyncIterable<Omit<ChatModelStream, "then">>;
|
|
60
|
+
/** The subagent's own tool-call streams. */
|
|
61
|
+
readonly toolCalls: AsyncIterable<ToolCallStreamUnion<TTools>>;
|
|
62
|
+
/** Nested subagents this subagent dispatches from its own tools. */
|
|
63
|
+
readonly subagents: AsyncIterable<SubagentRunStream>;
|
|
64
|
+
}
|
|
31
65
|
/**
|
|
32
66
|
* A {@link GraphRunStream} with native agent-level projections assigned
|
|
33
67
|
* directly on the instance by `createGraphRunStream` (via `__native`
|
|
@@ -40,26 +74,22 @@ type ToolCallStreamUnion<TTools extends readonly (ClientTool | ServerTool)[]> =
|
|
|
40
74
|
* @typeParam TValues - Shape of the graph's state values.
|
|
41
75
|
* @typeParam TTools - Tuple of tools registered on the agent, used to type
|
|
42
76
|
* the per-tool `toolCalls` discriminated union.
|
|
43
|
-
* @typeParam TMiddleware - Tuple of middleware registered on the agent, used
|
|
44
|
-
* to type the per-middleware `middleware` event union.
|
|
45
77
|
* @typeParam TExtensions - Shape of `run.extensions` produced by user-supplied
|
|
46
78
|
* stream transformer factories. Derived via
|
|
47
|
-
* `
|
|
79
|
+
* `InferStreamExtensions<TStreamTransformers>`.
|
|
48
80
|
*/
|
|
49
81
|
type AgentRunStream<TValues = Record<string, unknown>, TTools extends readonly (ClientTool | ServerTool)[] = readonly (ClientTool | ServerTool)[], TExtensions extends Record<string, unknown> = Record<string, unknown>> = GraphRunStream<TValues, TExtensions> & {
|
|
50
82
|
/** Tool call streams from the native ToolCallTransformer. */toolCalls: AsyncIterable<ToolCallStreamUnion<TTools>>;
|
|
83
|
+
/**
|
|
84
|
+
* Named subagent runs dispatched from tools, surfaced by the native
|
|
85
|
+
* {@link createSubagentTransformer}. Each yielded {@link SubagentRunStream}
|
|
86
|
+
* is a nested `createAgent` run whose `lc_agent_name` differs from this
|
|
87
|
+
* agent's — for example a `createAgent({ name })` invoked inside a tool body.
|
|
88
|
+
* Unlike `run.subgraphs` (which also yields the agent's own internal nodes),
|
|
89
|
+
* `run.subagents` only yields real named-agent invocations.
|
|
90
|
+
*/
|
|
91
|
+
subagents: AsyncIterable<SubagentRunStream<TValues>>;
|
|
51
92
|
};
|
|
52
|
-
interface ToolCallProjection {
|
|
53
|
-
toolCalls: AsyncIterable<ToolCallStream>;
|
|
54
|
-
}
|
|
55
|
-
/**
|
|
56
|
-
* Creates a native transformer that correlates `tools` channel events
|
|
57
|
-
* into per-call {@link ToolCallStream} objects.
|
|
58
|
-
*
|
|
59
|
-
* Marked `__native: true` — projection keys land directly on the
|
|
60
|
-
* `GraphRunStream` instance as `run.toolCalls`.
|
|
61
|
-
*/
|
|
62
|
-
declare function createToolCallTransformer(path: Namespace): () => NativeStreamTransformer<ToolCallProjection>;
|
|
63
93
|
//#endregion
|
|
64
|
-
export { AgentRunStream, InferStreamExtensions,
|
|
65
|
-
//# sourceMappingURL=
|
|
94
|
+
export { AgentRunStream, InferStreamExtensions, SubagentRunStream, ToolCallStreamUnion };
|
|
95
|
+
//# sourceMappingURL=types.d.cts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.cts","names":[],"sources":["../../../src/agents/transformers/types.ts"],"mappings":";;;;;;AAuBA;;;;;KAAY,qBAAA,WACA,aAAA,OAAoB,iBAAA,UAC5B,CAAA,uBACA,MAAA,kBACA,CAAA,yBACU,iBAAA,iCACgB,aAAA,OAAoB,iBAAA,UAE5C,CAAA,GAAI,qBAAA,CAAsB,IAAA,IAC1B,MAAA;;KAGD,UAAA,MAAgB,CAAA;EAAY,IAAA;AAAA,IAAiC,CAAA;;KAG7D,WAAA,MACH,CAAA,SAAU,qBAAA,gDACN,YAAA,GACA,CAAA,SAAU,uBAAA,iCACR,YAAA;;KAIH,YAAA,MACH,CAAA,SAAU,qBAAA,+CACN,WAAA,GACA,CAAA,SAAU,uBAAA,gCACR,WAAA;;;;;;;;;KAWI,mBAAA,0BACe,UAAA,GAAa,UAAA,qBAE1B,MAAA,GAAS,cAAA,CACnB,UAAA,CAAW,MAAA,CAAO,CAAA,IAClB,WAAA,CAAY,MAAA,CAAO,CAAA,IACnB,YAAA,CAAa,MAAA,CAAO,CAAA;;;;;;;AAnCM;;;;;;;;;;;UAwDb,iBAAA,WACL,MAAA,4CACe,UAAA,GAAa,UAAA,gBAClC,UAAA,GACA,UAAA;EAtDU;EAAA,SA0DL,IAAA;EAzDT;;;;;EAAA,SAgES,KAAA,EAAO,cAAA;EAjED;EAAA,SAoEN,MAAA,EAAQ,OAAA,CAAQ,OAAA;EAnEf;EAAA,SAsED,QAAA,EAAU,aAAA,CAAc,IAAA,CAAK,eAAA;EArElC;EAAA,SAwEK,SAAA,EAAW,aAAA,CAAc,mBAAA,CAAoB,MAAA;EAvExC;EAAA,SA0EL,SAAA,EAAW,aAAA,CAAc,iBAAA;AAAA;;;AAzEhB;;;;;;;;;;;;;;KA4FR,cAAA,WACA,MAAA,4CACe,UAAA,GAAa,UAAA,gBAClC,UAAA,GACA,UAAA,yBAEgB,MAAA,oBAA0B,MAAA,qBAC5C,cAAA,CAAe,OAAA,EAAS,WAAA;EA5FZ,6DA8Fd,SAAA,EAAW,aAAA,CAAc,mBAAA,CAAoB,MAAA;EA7FvC;;;AAWR;;;;;EA4FE,SAAA,EAAW,aAAA,CAAc,iBAAA,CAAkB,OAAA;AAAA"}
|
|
@@ -1,11 +1,10 @@
|
|
|
1
1
|
import { ClientTool, DynamicStructuredTool, ServerTool, StructuredToolInterface } from "@langchain/core/tools";
|
|
2
|
-
import {
|
|
2
|
+
import { ChatModelStream, GraphRunStream, LifecycleCause, StreamTransformer, ToolCallStream } from "@langchain/langgraph";
|
|
3
3
|
|
|
4
|
-
//#region src/agents/
|
|
4
|
+
//#region src/agents/transformers/types.d.ts
|
|
5
5
|
/**
|
|
6
6
|
* Infers the merged extensions shape from a tuple of stream transformer
|
|
7
|
-
* factories.
|
|
8
|
-
* is not exported from the package's public surface.
|
|
7
|
+
* factories.
|
|
9
8
|
*
|
|
10
9
|
* Given `[() => StreamTransformer<{ a: number }>, () => StreamTransformer<{ b: string }>]`,
|
|
11
10
|
* produces `{ a: number } & { b: string }`.
|
|
@@ -28,6 +27,41 @@ type ToolOutputOf<T> = T extends DynamicStructuredTool<any, any, any, infer Tool
|
|
|
28
27
|
* plain `(ClientTool | ServerTool)[]` without literal name types.
|
|
29
28
|
*/
|
|
30
29
|
type ToolCallStreamUnion<TTools extends readonly (ClientTool | ServerTool)[]> = { [K in keyof TTools]: ToolCallStream<ToolNameOf<TTools[K]>, ToolInputOf<TTools[K]>, ToolOutputOf<TTools[K]>> }[number];
|
|
30
|
+
/**
|
|
31
|
+
* A nested named-agent execution surfaced on `run.subagents`.
|
|
32
|
+
*
|
|
33
|
+
* A subagent is a nested `createAgent` run whose `lc_agent_name` differs from
|
|
34
|
+
* its parent's — e.g. a `createAgent({ name })` invoked inside a tool body. The
|
|
35
|
+
* handle exposes scoped projections (`messages`, `toolCalls`, nested
|
|
36
|
+
* `subagents`) plus the resolved {@link name}, the {@link cause} that triggered
|
|
37
|
+
* it, and the subagent's final {@link output} state.
|
|
38
|
+
*
|
|
39
|
+
* This is a self-contained handle (no subclass of `SubgraphRunStream`): its
|
|
40
|
+
* projections are produced by per-subagent transformer instances that
|
|
41
|
+
* {@link createSubagentTransformer} drives directly.
|
|
42
|
+
*
|
|
43
|
+
* @typeParam TValues - Shape of the subagent's final output state.
|
|
44
|
+
* @typeParam TTools - Tuple of tools the subagent may call, used to type
|
|
45
|
+
* {@link toolCalls}.
|
|
46
|
+
*/
|
|
47
|
+
interface SubagentRunStream<TValues = Record<string, unknown>, TTools extends readonly (ClientTool | ServerTool)[] = readonly (ClientTool | ServerTool)[]> {
|
|
48
|
+
/** The subagent's `lc_agent_name` (set by `createAgent({ name })`). */
|
|
49
|
+
readonly name: string;
|
|
50
|
+
/**
|
|
51
|
+
* The tool call that dispatched this subagent, as
|
|
52
|
+
* `{ type: "toolCall", tool_call_id }`, or `undefined` when the originating
|
|
53
|
+
* tool call could not be recovered (e.g. it was not triggered from a tool).
|
|
54
|
+
*/
|
|
55
|
+
readonly cause: LifecycleCause | undefined;
|
|
56
|
+
/** Resolves with the subagent's final state once it completes. */
|
|
57
|
+
readonly output: Promise<TValues>;
|
|
58
|
+
/** Per-message chat-model token streams scoped to this subagent. */
|
|
59
|
+
readonly messages: AsyncIterable<Omit<ChatModelStream, "then">>;
|
|
60
|
+
/** The subagent's own tool-call streams. */
|
|
61
|
+
readonly toolCalls: AsyncIterable<ToolCallStreamUnion<TTools>>;
|
|
62
|
+
/** Nested subagents this subagent dispatches from its own tools. */
|
|
63
|
+
readonly subagents: AsyncIterable<SubagentRunStream>;
|
|
64
|
+
}
|
|
31
65
|
/**
|
|
32
66
|
* A {@link GraphRunStream} with native agent-level projections assigned
|
|
33
67
|
* directly on the instance by `createGraphRunStream` (via `__native`
|
|
@@ -40,26 +74,22 @@ type ToolCallStreamUnion<TTools extends readonly (ClientTool | ServerTool)[]> =
|
|
|
40
74
|
* @typeParam TValues - Shape of the graph's state values.
|
|
41
75
|
* @typeParam TTools - Tuple of tools registered on the agent, used to type
|
|
42
76
|
* the per-tool `toolCalls` discriminated union.
|
|
43
|
-
* @typeParam TMiddleware - Tuple of middleware registered on the agent, used
|
|
44
|
-
* to type the per-middleware `middleware` event union.
|
|
45
77
|
* @typeParam TExtensions - Shape of `run.extensions` produced by user-supplied
|
|
46
78
|
* stream transformer factories. Derived via
|
|
47
|
-
* `
|
|
79
|
+
* `InferStreamExtensions<TStreamTransformers>`.
|
|
48
80
|
*/
|
|
49
81
|
type AgentRunStream<TValues = Record<string, unknown>, TTools extends readonly (ClientTool | ServerTool)[] = readonly (ClientTool | ServerTool)[], TExtensions extends Record<string, unknown> = Record<string, unknown>> = GraphRunStream<TValues, TExtensions> & {
|
|
50
82
|
/** Tool call streams from the native ToolCallTransformer. */toolCalls: AsyncIterable<ToolCallStreamUnion<TTools>>;
|
|
83
|
+
/**
|
|
84
|
+
* Named subagent runs dispatched from tools, surfaced by the native
|
|
85
|
+
* {@link createSubagentTransformer}. Each yielded {@link SubagentRunStream}
|
|
86
|
+
* is a nested `createAgent` run whose `lc_agent_name` differs from this
|
|
87
|
+
* agent's — for example a `createAgent({ name })` invoked inside a tool body.
|
|
88
|
+
* Unlike `run.subgraphs` (which also yields the agent's own internal nodes),
|
|
89
|
+
* `run.subagents` only yields real named-agent invocations.
|
|
90
|
+
*/
|
|
91
|
+
subagents: AsyncIterable<SubagentRunStream<TValues>>;
|
|
51
92
|
};
|
|
52
|
-
interface ToolCallProjection {
|
|
53
|
-
toolCalls: AsyncIterable<ToolCallStream>;
|
|
54
|
-
}
|
|
55
|
-
/**
|
|
56
|
-
* Creates a native transformer that correlates `tools` channel events
|
|
57
|
-
* into per-call {@link ToolCallStream} objects.
|
|
58
|
-
*
|
|
59
|
-
* Marked `__native: true` — projection keys land directly on the
|
|
60
|
-
* `GraphRunStream` instance as `run.toolCalls`.
|
|
61
|
-
*/
|
|
62
|
-
declare function createToolCallTransformer(path: Namespace): () => NativeStreamTransformer<ToolCallProjection>;
|
|
63
93
|
//#endregion
|
|
64
|
-
export { AgentRunStream, InferStreamExtensions,
|
|
65
|
-
//# sourceMappingURL=
|
|
94
|
+
export { AgentRunStream, InferStreamExtensions, SubagentRunStream, ToolCallStreamUnion };
|
|
95
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","names":[],"sources":["../../../src/agents/transformers/types.ts"],"mappings":";;;;;;AAuBA;;;;;KAAY,qBAAA,WACA,aAAA,OAAoB,iBAAA,UAC5B,CAAA,uBACA,MAAA,kBACA,CAAA,yBACU,iBAAA,iCACgB,aAAA,OAAoB,iBAAA,UAE5C,CAAA,GAAI,qBAAA,CAAsB,IAAA,IAC1B,MAAA;;KAGD,UAAA,MAAgB,CAAA;EAAY,IAAA;AAAA,IAAiC,CAAA;;KAG7D,WAAA,MACH,CAAA,SAAU,qBAAA,gDACN,YAAA,GACA,CAAA,SAAU,uBAAA,iCACR,YAAA;;KAIH,YAAA,MACH,CAAA,SAAU,qBAAA,+CACN,WAAA,GACA,CAAA,SAAU,uBAAA,gCACR,WAAA;;;;;;;;;KAWI,mBAAA,0BACe,UAAA,GAAa,UAAA,qBAE1B,MAAA,GAAS,cAAA,CACnB,UAAA,CAAW,MAAA,CAAO,CAAA,IAClB,WAAA,CAAY,MAAA,CAAO,CAAA,IACnB,YAAA,CAAa,MAAA,CAAO,CAAA;;;;;;;AAnCM;;;;;;;;;;;UAwDb,iBAAA,WACL,MAAA,4CACe,UAAA,GAAa,UAAA,gBAClC,UAAA,GACA,UAAA;EAtDU;EAAA,SA0DL,IAAA;EAzDT;;;;;EAAA,SAgES,KAAA,EAAO,cAAA;EAjED;EAAA,SAoEN,MAAA,EAAQ,OAAA,CAAQ,OAAA;EAnEf;EAAA,SAsED,QAAA,EAAU,aAAA,CAAc,IAAA,CAAK,eAAA;EArElC;EAAA,SAwEK,SAAA,EAAW,aAAA,CAAc,mBAAA,CAAoB,MAAA;EAvExC;EAAA,SA0EL,SAAA,EAAW,aAAA,CAAc,iBAAA;AAAA;;;AAzEhB;;;;;;;;;;;;;;KA4FR,cAAA,WACA,MAAA,4CACe,UAAA,GAAa,UAAA,gBAClC,UAAA,GACA,UAAA,yBAEgB,MAAA,oBAA0B,MAAA,qBAC5C,cAAA,CAAe,OAAA,EAAS,WAAA;EA5FZ,6DA8Fd,SAAA,EAAW,aAAA,CAAc,mBAAA,CAAoB,MAAA;EA7FvC;;;AAWR;;;;;EA4FE,SAAA,EAAW,aAAA,CAAc,iBAAA,CAAkB,OAAA;AAAA"}
|
package/dist/browser.cjs
CHANGED
|
@@ -5,7 +5,8 @@ const require_headless = require("./tools/headless.cjs");
|
|
|
5
5
|
const require_errors = require("./agents/errors.cjs");
|
|
6
6
|
const require_responses = require("./agents/responses.cjs");
|
|
7
7
|
const require_utils = require("./agents/middleware/utils.cjs");
|
|
8
|
-
const
|
|
8
|
+
const require_tool_call = require("./agents/transformers/tool-call.cjs");
|
|
9
|
+
const require_subagent = require("./agents/transformers/subagent.cjs");
|
|
9
10
|
const require_types = require("./agents/middleware/types.cjs");
|
|
10
11
|
const require_middleware = require("./agents/middleware.cjs");
|
|
11
12
|
const require_utils$1 = require("./agents/tests/utils.cjs");
|
|
@@ -72,7 +73,8 @@ var browser_exports = /* @__PURE__ */ require_runtime.__exportAll({
|
|
|
72
73
|
countTokensApproximately: () => require_utils.countTokensApproximately,
|
|
73
74
|
createAgent: () => require_index.createAgent,
|
|
74
75
|
createMiddleware: () => require_middleware.createMiddleware,
|
|
75
|
-
|
|
76
|
+
createSubagentTransformer: () => require_subagent.createSubagentTransformer,
|
|
77
|
+
createToolCallTransformer: () => require_tool_call.createToolCallTransformer,
|
|
76
78
|
detectCreditCard: () => require_pii.detectCreditCard,
|
|
77
79
|
detectEmail: () => require_pii.detectEmail,
|
|
78
80
|
detectIP: () => require_pii.detectIP,
|
|
@@ -231,7 +233,8 @@ exports.contextEditingMiddleware = require_contextEditing.contextEditingMiddlewa
|
|
|
231
233
|
exports.countTokensApproximately = require_utils.countTokensApproximately;
|
|
232
234
|
exports.createAgent = require_index.createAgent;
|
|
233
235
|
exports.createMiddleware = require_middleware.createMiddleware;
|
|
234
|
-
exports.
|
|
236
|
+
exports.createSubagentTransformer = require_subagent.createSubagentTransformer;
|
|
237
|
+
exports.createToolCallTransformer = require_tool_call.createToolCallTransformer;
|
|
235
238
|
exports.detectCreditCard = require_pii.detectCreditCard;
|
|
236
239
|
exports.detectEmail = require_pii.detectEmail;
|
|
237
240
|
exports.detectIP = require_pii.detectIP;
|