zeitlich 0.2.11 → 0.2.13

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 (46) hide show
  1. package/README.md +313 -126
  2. package/dist/adapters/langchain/index.cjs +270 -0
  3. package/dist/adapters/langchain/index.cjs.map +1 -0
  4. package/dist/adapters/langchain/index.d.cts +132 -0
  5. package/dist/adapters/langchain/index.d.ts +132 -0
  6. package/dist/adapters/langchain/index.js +265 -0
  7. package/dist/adapters/langchain/index.js.map +1 -0
  8. package/dist/index.cjs +89 -209
  9. package/dist/index.cjs.map +1 -1
  10. package/dist/index.d.cts +62 -46
  11. package/dist/index.d.ts +62 -46
  12. package/dist/index.js +88 -208
  13. package/dist/index.js.map +1 -1
  14. package/dist/{workflow-BhjsEQc1.d.cts → model-invoker-y_zlyMqu.d.cts} +45 -482
  15. package/dist/{workflow-BhjsEQc1.d.ts → model-invoker-y_zlyMqu.d.ts} +45 -482
  16. package/dist/thread-manager-qc0g5Rvd.d.cts +39 -0
  17. package/dist/thread-manager-qc0g5Rvd.d.ts +39 -0
  18. package/dist/workflow.cjs +59 -27
  19. package/dist/workflow.cjs.map +1 -1
  20. package/dist/workflow.d.cts +459 -6
  21. package/dist/workflow.d.ts +459 -6
  22. package/dist/workflow.js +60 -29
  23. package/dist/workflow.js.map +1 -1
  24. package/package.json +17 -2
  25. package/src/adapters/langchain/activities.ts +120 -0
  26. package/src/adapters/langchain/index.ts +38 -0
  27. package/src/adapters/langchain/model-invoker.ts +102 -0
  28. package/src/adapters/langchain/thread-manager.ts +142 -0
  29. package/src/index.ts +24 -23
  30. package/src/lib/fs.ts +25 -0
  31. package/src/lib/model-invoker.ts +15 -75
  32. package/src/lib/session.ts +52 -21
  33. package/src/lib/state-manager.ts +23 -5
  34. package/src/lib/thread-id.ts +25 -0
  35. package/src/lib/thread-manager.ts +18 -142
  36. package/src/lib/tool-router.ts +12 -18
  37. package/src/lib/types.ts +26 -10
  38. package/src/lib/workflow-helpers.ts +50 -0
  39. package/src/tools/ask-user-question/handler.ts +25 -1
  40. package/src/tools/bash/handler.ts +13 -0
  41. package/src/tools/subagent/handler.ts +16 -5
  42. package/src/tools/subagent/tool.ts +34 -15
  43. package/src/workflow.ts +26 -7
  44. package/tsup.config.ts +1 -0
  45. package/src/activities.ts +0 -91
  46. package/src/plugin.ts +0 -28
@@ -0,0 +1,270 @@
1
+ 'use strict';
2
+
3
+ var messages = require('@langchain/core/messages');
4
+ var crypto = require('crypto');
5
+
6
+ // src/adapters/langchain/thread-manager.ts
7
+
8
+ // node_modules/uuid/dist/esm/stringify.js
9
+ var byteToHex = [];
10
+ for (let i = 0; i < 256; ++i) {
11
+ byteToHex.push((i + 256).toString(16).slice(1));
12
+ }
13
+ function unsafeStringify(arr, offset = 0) {
14
+ return (byteToHex[arr[offset + 0]] + byteToHex[arr[offset + 1]] + byteToHex[arr[offset + 2]] + byteToHex[arr[offset + 3]] + "-" + byteToHex[arr[offset + 4]] + byteToHex[arr[offset + 5]] + "-" + byteToHex[arr[offset + 6]] + byteToHex[arr[offset + 7]] + "-" + byteToHex[arr[offset + 8]] + byteToHex[arr[offset + 9]] + "-" + byteToHex[arr[offset + 10]] + byteToHex[arr[offset + 11]] + byteToHex[arr[offset + 12]] + byteToHex[arr[offset + 13]] + byteToHex[arr[offset + 14]] + byteToHex[arr[offset + 15]]).toLowerCase();
15
+ }
16
+ var rnds8Pool = new Uint8Array(256);
17
+ var poolPtr = rnds8Pool.length;
18
+ function rng() {
19
+ if (poolPtr > rnds8Pool.length - 16) {
20
+ crypto.randomFillSync(rnds8Pool);
21
+ poolPtr = 0;
22
+ }
23
+ return rnds8Pool.slice(poolPtr, poolPtr += 16);
24
+ }
25
+ var native_default = { randomUUID: crypto.randomUUID };
26
+
27
+ // node_modules/uuid/dist/esm/v4.js
28
+ function v4(options, buf, offset) {
29
+ if (native_default.randomUUID && !buf && !options) {
30
+ return native_default.randomUUID();
31
+ }
32
+ options = options || {};
33
+ const rnds = options.random ?? options.rng?.() ?? rng();
34
+ if (rnds.length < 16) {
35
+ throw new Error("Random bytes length must be >= 16");
36
+ }
37
+ rnds[6] = rnds[6] & 15 | 64;
38
+ rnds[8] = rnds[8] & 63 | 128;
39
+ if (buf) {
40
+ offset = offset || 0;
41
+ if (offset < 0 || offset + 16 > buf.length) {
42
+ throw new RangeError(`UUID byte range ${offset}:${offset + 15} is out of buffer bounds`);
43
+ }
44
+ for (let i = 0; i < 16; ++i) {
45
+ buf[offset + i] = rnds[i];
46
+ }
47
+ return buf;
48
+ }
49
+ return unsafeStringify(rnds);
50
+ }
51
+ var v4_default = v4;
52
+
53
+ // src/lib/thread-manager.ts
54
+ var THREAD_TTL_SECONDS = 60 * 60 * 24 * 90;
55
+ var APPEND_IDEMPOTENT_SCRIPT = `
56
+ if redis.call('EXISTS', KEYS[1]) == 1 then
57
+ return 0
58
+ end
59
+ for i = 2, #ARGV do
60
+ redis.call('RPUSH', KEYS[2], ARGV[i])
61
+ end
62
+ redis.call('EXPIRE', KEYS[2], tonumber(ARGV[1]))
63
+ redis.call('SET', KEYS[1], '1', 'EX', tonumber(ARGV[1]))
64
+ return 1
65
+ `;
66
+ function getThreadKey(threadId, key) {
67
+ return `thread:${threadId}:${key}`;
68
+ }
69
+ function createThreadManager(config) {
70
+ const {
71
+ redis,
72
+ threadId,
73
+ key = "messages",
74
+ serialize = (m) => JSON.stringify(m),
75
+ deserialize = (raw) => JSON.parse(raw),
76
+ idOf
77
+ } = config;
78
+ const redisKey = getThreadKey(threadId, key);
79
+ const metaKey = getThreadKey(threadId, `${key}:meta`);
80
+ async function assertThreadExists() {
81
+ const exists = await redis.exists(metaKey);
82
+ if (!exists) {
83
+ throw new Error(`Thread "${threadId}" (key: ${key}) does not exist`);
84
+ }
85
+ }
86
+ return {
87
+ async initialize() {
88
+ await redis.del(redisKey);
89
+ await redis.set(metaKey, "1", "EX", THREAD_TTL_SECONDS);
90
+ },
91
+ async load() {
92
+ await assertThreadExists();
93
+ const data = await redis.lrange(redisKey, 0, -1);
94
+ return data.map(deserialize);
95
+ },
96
+ async append(messages) {
97
+ if (messages.length === 0) return;
98
+ await assertThreadExists();
99
+ if (idOf) {
100
+ const dedupId = messages.map(idOf).join(":");
101
+ const dedupKey = getThreadKey(threadId, `dedup:${dedupId}`);
102
+ await redis.eval(
103
+ APPEND_IDEMPOTENT_SCRIPT,
104
+ 2,
105
+ dedupKey,
106
+ redisKey,
107
+ String(THREAD_TTL_SECONDS),
108
+ ...messages.map(serialize)
109
+ );
110
+ } else {
111
+ await redis.rpush(redisKey, ...messages.map(serialize));
112
+ await redis.expire(redisKey, THREAD_TTL_SECONDS);
113
+ }
114
+ },
115
+ async delete() {
116
+ await redis.del(redisKey, metaKey);
117
+ }
118
+ };
119
+ }
120
+
121
+ // src/adapters/langchain/thread-manager.ts
122
+ function storedMessageId(msg) {
123
+ return msg.data.id ?? "";
124
+ }
125
+ function createLangChainThreadManager(config) {
126
+ const baseConfig = {
127
+ redis: config.redis,
128
+ threadId: config.threadId,
129
+ key: config.key,
130
+ idOf: storedMessageId
131
+ };
132
+ const base = createThreadManager(baseConfig);
133
+ const helpers = {
134
+ createHumanMessage(content) {
135
+ return new messages.HumanMessage({
136
+ id: v4_default(),
137
+ content
138
+ }).toDict();
139
+ },
140
+ createSystemMessage(content) {
141
+ return new messages.SystemMessage({
142
+ id: v4_default(),
143
+ content
144
+ }).toDict();
145
+ },
146
+ createAIMessage(content, kwargs) {
147
+ return new messages.AIMessage({
148
+ id: v4_default(),
149
+ content,
150
+ additional_kwargs: kwargs ? {
151
+ header: kwargs.header,
152
+ options: kwargs.options,
153
+ multiSelect: kwargs.multiSelect
154
+ } : void 0
155
+ }).toDict();
156
+ },
157
+ createToolMessage(content, toolCallId) {
158
+ return new messages.ToolMessage({
159
+ id: v4_default(),
160
+ content,
161
+ tool_call_id: toolCallId
162
+ }).toDict();
163
+ },
164
+ async appendHumanMessage(content) {
165
+ const message = helpers.createHumanMessage(content);
166
+ await base.append([message]);
167
+ },
168
+ async appendToolMessage(content, toolCallId) {
169
+ const message = helpers.createToolMessage(content, toolCallId);
170
+ await base.append([message]);
171
+ },
172
+ async appendAIMessage(content) {
173
+ const message = helpers.createAIMessage(content);
174
+ await base.append([message]);
175
+ },
176
+ async appendSystemMessage(content) {
177
+ const message = helpers.createSystemMessage(content);
178
+ await base.initialize();
179
+ await base.append([message]);
180
+ }
181
+ };
182
+ return Object.assign(base, helpers);
183
+ }
184
+ function createLangChainModelInvoker({
185
+ redis,
186
+ model
187
+ }) {
188
+ return async function invokeLangChainModel2(config) {
189
+ const { threadId, agentName, state, metadata } = config;
190
+ const thread = createLangChainThreadManager({ redis, threadId });
191
+ const runId = v4_default();
192
+ const messages$1 = await thread.load();
193
+ const response = await model.invoke(
194
+ [...messages.mapStoredMessagesToChatMessages(messages$1)],
195
+ {
196
+ runName: agentName,
197
+ runId,
198
+ metadata: { thread_id: threadId, ...metadata },
199
+ tools: state.tools
200
+ }
201
+ );
202
+ await thread.append([response.toDict()]);
203
+ const toolCalls = response.tool_calls ?? [];
204
+ return {
205
+ message: response.toDict(),
206
+ rawToolCalls: toolCalls.map((tc) => ({
207
+ id: tc.id,
208
+ name: tc.name,
209
+ args: tc.args
210
+ })),
211
+ usage: {
212
+ inputTokens: response.usage_metadata?.input_tokens,
213
+ outputTokens: response.usage_metadata?.output_tokens,
214
+ reasonTokens: response.usage_metadata?.output_token_details?.reasoning,
215
+ cachedWriteTokens: response.usage_metadata?.input_token_details?.cache_creation,
216
+ cachedReadTokens: response.usage_metadata?.input_token_details?.cache_read
217
+ }
218
+ };
219
+ };
220
+ }
221
+ async function invokeLangChainModel({
222
+ redis,
223
+ model,
224
+ config
225
+ }) {
226
+ const invoker = createLangChainModelInvoker({ redis, model });
227
+ return invoker(config);
228
+ }
229
+
230
+ // src/adapters/langchain/activities.ts
231
+ function createLangChainAdapter(config) {
232
+ const { redis } = config;
233
+ const threadOps = {
234
+ async initializeThread(threadId) {
235
+ const thread = createLangChainThreadManager({ redis, threadId });
236
+ await thread.initialize();
237
+ },
238
+ async appendHumanMessage(threadId, content) {
239
+ const thread = createLangChainThreadManager({ redis, threadId });
240
+ await thread.appendHumanMessage(content);
241
+ },
242
+ async appendSystemMessage(threadId, content) {
243
+ const thread = createLangChainThreadManager({ redis, threadId });
244
+ await thread.appendSystemMessage(content);
245
+ },
246
+ async appendToolResult(cfg) {
247
+ const { threadId, toolCallId, content } = cfg;
248
+ const thread = createLangChainThreadManager({ redis, threadId });
249
+ await thread.appendToolMessage(content, toolCallId);
250
+ }
251
+ };
252
+ const makeInvoker = (model) => createLangChainModelInvoker({ redis, model });
253
+ const invoker = config.model ? makeInvoker(config.model) : (() => {
254
+ throw new Error(
255
+ "No default model provided to createLangChainAdapter. Either pass `model` in the config or use `createModelInvoker(model)` instead."
256
+ );
257
+ });
258
+ return {
259
+ threadOps,
260
+ invoker,
261
+ createModelInvoker: makeInvoker
262
+ };
263
+ }
264
+
265
+ exports.createLangChainAdapter = createLangChainAdapter;
266
+ exports.createLangChainModelInvoker = createLangChainModelInvoker;
267
+ exports.createLangChainThreadManager = createLangChainThreadManager;
268
+ exports.invokeLangChainModel = invokeLangChainModel;
269
+ //# sourceMappingURL=index.cjs.map
270
+ //# sourceMappingURL=index.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../node_modules/uuid/dist/esm/stringify.js","../../../node_modules/uuid/dist/esm/rng.js","../../../node_modules/uuid/dist/esm/native.js","../../../node_modules/uuid/dist/esm/v4.js","../../../src/lib/thread-manager.ts","../../../src/adapters/langchain/thread-manager.ts","../../../src/adapters/langchain/model-invoker.ts","../../../src/adapters/langchain/activities.ts"],"names":["randomFillSync","randomUUID","HumanMessage","SystemMessage","AIMessage","ToolMessage","invokeLangChainModel","messages","mapStoredMessagesToChatMessages"],"mappings":"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","file":"index.cjs","sourcesContent":["import validate from './validate.js';\nconst byteToHex = [];\nfor (let i = 0; i < 256; ++i) {\n byteToHex.push((i + 0x100).toString(16).slice(1));\n}\nexport function unsafeStringify(arr, offset = 0) {\n return (byteToHex[arr[offset + 0]] +\n byteToHex[arr[offset + 1]] +\n byteToHex[arr[offset + 2]] +\n byteToHex[arr[offset + 3]] +\n '-' +\n byteToHex[arr[offset + 4]] +\n byteToHex[arr[offset + 5]] +\n '-' +\n byteToHex[arr[offset + 6]] +\n byteToHex[arr[offset + 7]] +\n '-' +\n byteToHex[arr[offset + 8]] +\n byteToHex[arr[offset + 9]] +\n '-' +\n byteToHex[arr[offset + 10]] +\n byteToHex[arr[offset + 11]] +\n byteToHex[arr[offset + 12]] +\n byteToHex[arr[offset + 13]] +\n byteToHex[arr[offset + 14]] +\n byteToHex[arr[offset + 15]]).toLowerCase();\n}\nfunction stringify(arr, offset = 0) {\n const uuid = unsafeStringify(arr, offset);\n if (!validate(uuid)) {\n throw TypeError('Stringified UUID is invalid');\n }\n return uuid;\n}\nexport default stringify;\n","import { randomFillSync } from 'crypto';\nconst rnds8Pool = new Uint8Array(256);\nlet poolPtr = rnds8Pool.length;\nexport default function rng() {\n if (poolPtr > rnds8Pool.length - 16) {\n randomFillSync(rnds8Pool);\n poolPtr = 0;\n }\n return rnds8Pool.slice(poolPtr, (poolPtr += 16));\n}\n","import { randomUUID } from 'crypto';\nexport default { randomUUID };\n","import native from './native.js';\nimport rng from './rng.js';\nimport { unsafeStringify } from './stringify.js';\nfunction v4(options, buf, offset) {\n if (native.randomUUID && !buf && !options) {\n return native.randomUUID();\n }\n options = options || {};\n const rnds = options.random ?? options.rng?.() ?? rng();\n if (rnds.length < 16) {\n throw new Error('Random bytes length must be >= 16');\n }\n rnds[6] = (rnds[6] & 0x0f) | 0x40;\n rnds[8] = (rnds[8] & 0x3f) | 0x80;\n if (buf) {\n offset = offset || 0;\n if (offset < 0 || offset + 16 > buf.length) {\n throw new RangeError(`UUID byte range ${offset}:${offset + 15} is out of buffer bounds`);\n }\n for (let i = 0; i < 16; ++i) {\n buf[offset + i] = rnds[i];\n }\n return buf;\n }\n return unsafeStringify(rnds);\n}\nexport default v4;\n","import type Redis from \"ioredis\";\n\nconst THREAD_TTL_SECONDS = 60 * 60 * 24 * 90; // 90 days\n\n/**\n * Lua script for atomic idempotent append.\n * Checks a dedup key; if it exists the message was already appended and we\n * return 0. Otherwise appends all messages to the list, sets TTL on both\n * the list and the dedup key, and returns 1.\n *\n * KEYS[1] = dedup key, KEYS[2] = list key\n * ARGV[1] = TTL seconds, ARGV[2..N] = serialised messages\n */\nconst APPEND_IDEMPOTENT_SCRIPT = `\nif redis.call('EXISTS', KEYS[1]) == 1 then\n return 0\nend\nfor i = 2, #ARGV do\n redis.call('RPUSH', KEYS[2], ARGV[i])\nend\nredis.call('EXPIRE', KEYS[2], tonumber(ARGV[1]))\nredis.call('SET', KEYS[1], '1', 'EX', tonumber(ARGV[1]))\nreturn 1\n`;\n\nfunction getThreadKey(threadId: string, key: string): string {\n return `thread:${threadId}:${key}`;\n}\n\nexport interface ThreadManagerConfig<T> {\n redis: Redis;\n threadId: string;\n /** Thread key, defaults to 'messages' */\n key?: string;\n /** Custom serializer, defaults to JSON.stringify */\n serialize?: (message: T) => string;\n /** Custom deserializer, defaults to JSON.parse */\n deserialize?: (raw: string) => T;\n /**\n * Extract a unique id from a message for idempotent appends.\n * When provided, `append` uses an atomic Lua script to skip duplicate writes.\n */\n idOf?: (message: T) => string;\n}\n\n/** Generic thread manager for any message type */\nexport interface BaseThreadManager<T> {\n /** Initialize an empty thread */\n initialize(): Promise<void>;\n /** Load all messages from the thread */\n load(): Promise<T[]>;\n /**\n * Append messages to the thread.\n * When `idOf` is configured, appends are idempotent — retries with the\n * same message ids are atomically skipped via a Redis Lua script.\n */\n append(messages: T[]): Promise<void>;\n /** Delete the thread */\n delete(): Promise<void>;\n}\n\n/**\n * Creates a generic thread manager for handling conversation state in Redis.\n * Framework-agnostic — works with any serializable message type.\n */\nexport function createThreadManager<T>(\n config: ThreadManagerConfig<T>,\n): BaseThreadManager<T> {\n const {\n redis,\n threadId,\n key = \"messages\",\n serialize = (m: T): string => JSON.stringify(m),\n deserialize = (raw: string): T => JSON.parse(raw) as T,\n idOf,\n } = config;\n const redisKey = getThreadKey(threadId, key);\n const metaKey = getThreadKey(threadId, `${key}:meta`);\n\n async function assertThreadExists(): Promise<void> {\n const exists = await redis.exists(metaKey);\n if (!exists) {\n throw new Error(`Thread \"${threadId}\" (key: ${key}) does not exist`);\n }\n }\n\n return {\n async initialize(): Promise<void> {\n await redis.del(redisKey);\n await redis.set(metaKey, \"1\", \"EX\", THREAD_TTL_SECONDS);\n },\n\n async load(): Promise<T[]> {\n await assertThreadExists();\n const data = await redis.lrange(redisKey, 0, -1);\n return data.map(deserialize);\n },\n\n async append(messages: T[]): Promise<void> {\n if (messages.length === 0) return;\n await assertThreadExists();\n\n if (idOf) {\n const dedupId = messages.map(idOf).join(\":\");\n const dedupKey = getThreadKey(threadId, `dedup:${dedupId}`);\n await redis.eval(\n APPEND_IDEMPOTENT_SCRIPT,\n 2,\n dedupKey,\n redisKey,\n String(THREAD_TTL_SECONDS),\n ...messages.map(serialize),\n );\n } else {\n await redis.rpush(redisKey, ...messages.map(serialize));\n await redis.expire(redisKey, THREAD_TTL_SECONDS);\n }\n },\n\n async delete(): Promise<void> {\n await redis.del(redisKey, metaKey);\n },\n };\n}\n","import type Redis from \"ioredis\";\nimport {\n type $InferMessageContent,\n AIMessage,\n HumanMessage,\n type MessageContent,\n type MessageStructure,\n type StoredMessage,\n SystemMessage,\n ToolMessage,\n} from \"@langchain/core/messages\";\nimport { v4 as uuidv4 } from \"uuid\";\nimport {\n createThreadManager,\n type BaseThreadManager,\n type ThreadManagerConfig,\n} from \"../../lib/thread-manager\";\n\nexport type LangChainToolMessageContent = $InferMessageContent<\n MessageStructure,\n \"tool\"\n>;\n\nexport interface LangChainThreadManagerConfig {\n redis: Redis;\n threadId: string;\n /** Thread key, defaults to 'messages' */\n key?: string;\n}\n\n/** Thread manager with LangChain StoredMessage convenience helpers */\nexport interface LangChainThreadManager extends BaseThreadManager<StoredMessage> {\n createHumanMessage(content: string | MessageContent): StoredMessage;\n createSystemMessage(content: string): StoredMessage;\n createAIMessage(\n content: string | MessageContent,\n kwargs?: { header?: string; options?: string[]; multiSelect?: boolean },\n ): StoredMessage;\n createToolMessage(\n content: LangChainToolMessageContent,\n toolCallId: string,\n ): StoredMessage;\n appendHumanMessage(content: string | MessageContent): Promise<void>;\n appendSystemMessage(content: string): Promise<void>;\n appendToolMessage(\n content: LangChainToolMessageContent,\n toolCallId: string,\n ): Promise<void>;\n appendAIMessage(content: string | MessageContent): Promise<void>;\n}\n\nfunction storedMessageId(msg: StoredMessage): string {\n return msg.data.id ?? \"\";\n}\n\n/**\n * Creates a LangChain-specific thread manager that stores StoredMessage\n * instances in Redis and provides convenience helpers for creating and\n * appending typed LangChain messages.\n */\nexport function createLangChainThreadManager(\n config: LangChainThreadManagerConfig,\n): LangChainThreadManager {\n const baseConfig: ThreadManagerConfig<StoredMessage> = {\n redis: config.redis,\n threadId: config.threadId,\n key: config.key,\n idOf: storedMessageId,\n };\n\n const base = createThreadManager(baseConfig);\n\n const helpers = {\n createHumanMessage(content: string | MessageContent): StoredMessage {\n return new HumanMessage({\n id: uuidv4(),\n content: content as string,\n }).toDict();\n },\n\n createSystemMessage(content: string): StoredMessage {\n return new SystemMessage({\n id: uuidv4(),\n content: content as string,\n }).toDict();\n },\n\n createAIMessage(\n content: string,\n kwargs?: { header?: string; options?: string[]; multiSelect?: boolean },\n ): StoredMessage {\n return new AIMessage({\n id: uuidv4(),\n content,\n additional_kwargs: kwargs\n ? {\n header: kwargs.header,\n options: kwargs.options,\n multiSelect: kwargs.multiSelect,\n }\n : undefined,\n }).toDict();\n },\n\n createToolMessage(\n content: LangChainToolMessageContent,\n toolCallId: string,\n ): StoredMessage {\n return new ToolMessage({\n id: uuidv4(),\n content: content as MessageContent,\n tool_call_id: toolCallId,\n }).toDict();\n },\n\n async appendHumanMessage(content: string | MessageContent): Promise<void> {\n const message = helpers.createHumanMessage(content);\n await base.append([message]);\n },\n\n async appendToolMessage(\n content: LangChainToolMessageContent,\n toolCallId: string,\n ): Promise<void> {\n const message = helpers.createToolMessage(content, toolCallId);\n await base.append([message]);\n },\n\n async appendAIMessage(content: string | MessageContent): Promise<void> {\n const message = helpers.createAIMessage(content as string);\n await base.append([message]);\n },\n\n async appendSystemMessage(content: string): Promise<void> {\n const message = helpers.createSystemMessage(content);\n await base.initialize();\n await base.append([message]);\n },\n };\n\n return Object.assign(base, helpers);\n}\n","import type Redis from \"ioredis\";\nimport type { AgentResponse } from \"../../lib/types\";\nimport type { ModelInvokerConfig } from \"../../lib/model-invoker\";\nimport { mapStoredMessagesToChatMessages } from \"@langchain/core/messages\";\nimport type { StoredMessage } from \"@langchain/core/messages\";\nimport { v4 as uuidv4 } from \"uuid\";\nimport type {\n BaseChatModel,\n BaseChatModelCallOptions,\n BindToolsInput,\n} from \"@langchain/core/language_models/chat_models\";\nimport { createLangChainThreadManager } from \"./thread-manager\";\n\nexport interface LangChainModelInvokerConfig {\n redis: Redis;\n model: BaseChatModel<BaseChatModelCallOptions & { tools?: BindToolsInput }>;\n}\n\n/**\n * Creates a LangChain-based model invoker that satisfies the generic\n * `ModelInvoker<StoredMessage>` contract.\n *\n * Loads the conversation thread from Redis, invokes a LangChain chat model,\n * appends the AI response, and returns a normalised AgentResponse.\n *\n * @example\n * ```typescript\n * import { createLangChainModelInvoker } from 'zeitlich/adapters/langchain';\n * import { createRunAgentActivity } from 'zeitlich';\n * import { ChatAnthropic } from '@langchain/anthropic';\n *\n * const model = new ChatAnthropic({ model: \"claude-sonnet-4-6\" });\n * const invoker = createLangChainModelInvoker({ redis, model });\n *\n * // Wrap with createRunAgentActivity to use as runAgent activity:\n * return { runAgent: createRunAgentActivity(client, invoker) };\n * ```\n */\nexport function createLangChainModelInvoker({\n redis,\n model,\n}: LangChainModelInvokerConfig) {\n return async function invokeLangChainModel(\n config: ModelInvokerConfig\n ): Promise<AgentResponse<StoredMessage>> {\n const { threadId, agentName, state, metadata } = config;\n\n const thread = createLangChainThreadManager({ redis, threadId });\n const runId = uuidv4();\n\n const messages = await thread.load();\n const response = await model.invoke(\n [...mapStoredMessagesToChatMessages(messages)],\n {\n runName: agentName,\n runId,\n metadata: { thread_id: threadId, ...metadata },\n tools: state.tools,\n }\n );\n\n await thread.append([response.toDict()]);\n\n const toolCalls = response.tool_calls ?? [];\n\n return {\n message: response.toDict(),\n rawToolCalls: toolCalls.map((tc) => ({\n id: tc.id,\n name: tc.name,\n args: tc.args,\n })),\n usage: {\n inputTokens: response.usage_metadata?.input_tokens,\n outputTokens: response.usage_metadata?.output_tokens,\n reasonTokens: response.usage_metadata?.output_token_details?.reasoning,\n cachedWriteTokens:\n response.usage_metadata?.input_token_details?.cache_creation,\n cachedReadTokens:\n response.usage_metadata?.input_token_details?.cache_read,\n },\n };\n };\n}\n\n/**\n * Standalone function for one-shot LangChain model invocation.\n * Convenience wrapper around createLangChainModelInvoker for cases where\n * you don't need to reuse the invoker.\n */\nexport async function invokeLangChainModel({\n redis,\n model,\n config,\n}: {\n redis: Redis;\n config: ModelInvokerConfig;\n model: BaseChatModel<BaseChatModelCallOptions & { tools?: BindToolsInput }>;\n}): Promise<AgentResponse<StoredMessage>> {\n const invoker = createLangChainModelInvoker({ redis, model });\n return invoker(config);\n}\n","import type Redis from \"ioredis\";\nimport type { ThreadOps, ToolResultConfig } from \"../../lib/types\";\nimport type { MessageContent } from \"@langchain/core/messages\";\nimport type { ModelInvoker } from \"../../lib/model-invoker\";\nimport type { StoredMessage } from \"@langchain/core/messages\";\nimport type {\n BaseChatModel,\n BaseChatModelCallOptions,\n BindToolsInput,\n} from \"@langchain/core/language_models/chat_models\";\nimport { createLangChainThreadManager } from \"./thread-manager\";\nimport { createLangChainModelInvoker } from \"./model-invoker\";\n\ntype LangChainModel = BaseChatModel<\n BaseChatModelCallOptions & { tools?: BindToolsInput }\n>;\n\nexport interface LangChainAdapterConfig {\n redis: Redis;\n /** Optional default model — if omitted, use `createModelInvoker()` to create invokers later */\n model?: LangChainModel;\n}\n\nexport interface LangChainAdapter {\n /** Thread operations (register these as Temporal activities on the worker) */\n threadOps: ThreadOps;\n /** Model invoker using the default model (only available when `model` was provided) */\n invoker: ModelInvoker<StoredMessage>;\n /** Create an invoker for a specific model (for multi-model setups) */\n createModelInvoker(model: LangChainModel): ModelInvoker<StoredMessage>;\n}\n\n/**\n * Creates a LangChain adapter that bundles thread operations and model\n * invocation using a consistent message format (StoredMessage).\n *\n * The returned `threadOps` should be registered as Temporal activities on\n * the worker. The `invoker` (or invokers created via `createModelInvoker`)\n * should be wrapped with `createRunAgentActivity` for per-agent activities.\n *\n * @example\n * ```typescript\n * import { createLangChainAdapter } from 'zeitlich/adapters/langchain';\n * import { createRunAgentActivity } from 'zeitlich';\n *\n * const adapter = createLangChainAdapter({ redis, model });\n *\n * export function createActivities(client: WorkflowClient) {\n * return {\n * ...adapter.threadOps,\n * runAgent: createRunAgentActivity(client, adapter.invoker),\n * };\n * }\n * ```\n *\n * @example Multi-model setup\n * ```typescript\n * const adapter = createLangChainAdapter({ redis });\n *\n * export function createActivities(client: WorkflowClient) {\n * return {\n * ...adapter.threadOps,\n * runResearchAgent: createRunAgentActivity(client, adapter.createModelInvoker(claude)),\n * runWriterAgent: createRunAgentActivity(client, adapter.createModelInvoker(gpt4)),\n * };\n * }\n * ```\n */\nexport function createLangChainAdapter(\n config: LangChainAdapterConfig\n): LangChainAdapter {\n const { redis } = config;\n\n const threadOps: ThreadOps = {\n async initializeThread(threadId: string): Promise<void> {\n const thread = createLangChainThreadManager({ redis, threadId });\n await thread.initialize();\n },\n\n async appendHumanMessage(\n threadId: string,\n content: string | MessageContent\n ): Promise<void> {\n const thread = createLangChainThreadManager({ redis, threadId });\n await thread.appendHumanMessage(content);\n },\n\n async appendSystemMessage(\n threadId: string,\n content: string\n ): Promise<void> {\n const thread = createLangChainThreadManager({ redis, threadId });\n await thread.appendSystemMessage(content);\n },\n\n async appendToolResult(cfg: ToolResultConfig): Promise<void> {\n const { threadId, toolCallId, content } = cfg;\n const thread = createLangChainThreadManager({ redis, threadId });\n await thread.appendToolMessage(content, toolCallId);\n },\n };\n\n const makeInvoker = (model: LangChainModel): ModelInvoker<StoredMessage> =>\n createLangChainModelInvoker({ redis, model });\n\n const invoker: ModelInvoker<StoredMessage> = config.model\n ? makeInvoker(config.model)\n : ((() => {\n throw new Error(\n \"No default model provided to createLangChainAdapter. \" +\n \"Either pass `model` in the config or use `createModelInvoker(model)` instead.\"\n );\n }) as unknown as ModelInvoker<StoredMessage>);\n\n return {\n threadOps,\n invoker,\n createModelInvoker: makeInvoker,\n };\n}\n"]}
@@ -0,0 +1,132 @@
1
+ import Redis from 'ioredis';
2
+ import { T as ThreadOps, M as ModelInvoker, a as ModelInvokerConfig, A as AgentResponse } from '../../model-invoker-y_zlyMqu.cjs';
3
+ import { StoredMessage, MessageContent, $InferMessageContent, MessageStructure } from '@langchain/core/messages';
4
+ import { BaseChatModel, BaseChatModelCallOptions, BindToolsInput } from '@langchain/core/language_models/chat_models';
5
+ import { B as BaseThreadManager } from '../../thread-manager-qc0g5Rvd.cjs';
6
+ import 'zod';
7
+ import '@temporalio/common';
8
+
9
+ type LangChainModel = BaseChatModel<BaseChatModelCallOptions & {
10
+ tools?: BindToolsInput;
11
+ }>;
12
+ interface LangChainAdapterConfig {
13
+ redis: Redis;
14
+ /** Optional default model — if omitted, use `createModelInvoker()` to create invokers later */
15
+ model?: LangChainModel;
16
+ }
17
+ interface LangChainAdapter {
18
+ /** Thread operations (register these as Temporal activities on the worker) */
19
+ threadOps: ThreadOps;
20
+ /** Model invoker using the default model (only available when `model` was provided) */
21
+ invoker: ModelInvoker<StoredMessage>;
22
+ /** Create an invoker for a specific model (for multi-model setups) */
23
+ createModelInvoker(model: LangChainModel): ModelInvoker<StoredMessage>;
24
+ }
25
+ /**
26
+ * Creates a LangChain adapter that bundles thread operations and model
27
+ * invocation using a consistent message format (StoredMessage).
28
+ *
29
+ * The returned `threadOps` should be registered as Temporal activities on
30
+ * the worker. The `invoker` (or invokers created via `createModelInvoker`)
31
+ * should be wrapped with `createRunAgentActivity` for per-agent activities.
32
+ *
33
+ * @example
34
+ * ```typescript
35
+ * import { createLangChainAdapter } from 'zeitlich/adapters/langchain';
36
+ * import { createRunAgentActivity } from 'zeitlich';
37
+ *
38
+ * const adapter = createLangChainAdapter({ redis, model });
39
+ *
40
+ * export function createActivities(client: WorkflowClient) {
41
+ * return {
42
+ * ...adapter.threadOps,
43
+ * runAgent: createRunAgentActivity(client, adapter.invoker),
44
+ * };
45
+ * }
46
+ * ```
47
+ *
48
+ * @example Multi-model setup
49
+ * ```typescript
50
+ * const adapter = createLangChainAdapter({ redis });
51
+ *
52
+ * export function createActivities(client: WorkflowClient) {
53
+ * return {
54
+ * ...adapter.threadOps,
55
+ * runResearchAgent: createRunAgentActivity(client, adapter.createModelInvoker(claude)),
56
+ * runWriterAgent: createRunAgentActivity(client, adapter.createModelInvoker(gpt4)),
57
+ * };
58
+ * }
59
+ * ```
60
+ */
61
+ declare function createLangChainAdapter(config: LangChainAdapterConfig): LangChainAdapter;
62
+
63
+ type LangChainToolMessageContent = $InferMessageContent<MessageStructure, "tool">;
64
+ interface LangChainThreadManagerConfig {
65
+ redis: Redis;
66
+ threadId: string;
67
+ /** Thread key, defaults to 'messages' */
68
+ key?: string;
69
+ }
70
+ /** Thread manager with LangChain StoredMessage convenience helpers */
71
+ interface LangChainThreadManager extends BaseThreadManager<StoredMessage> {
72
+ createHumanMessage(content: string | MessageContent): StoredMessage;
73
+ createSystemMessage(content: string): StoredMessage;
74
+ createAIMessage(content: string | MessageContent, kwargs?: {
75
+ header?: string;
76
+ options?: string[];
77
+ multiSelect?: boolean;
78
+ }): StoredMessage;
79
+ createToolMessage(content: LangChainToolMessageContent, toolCallId: string): StoredMessage;
80
+ appendHumanMessage(content: string | MessageContent): Promise<void>;
81
+ appendSystemMessage(content: string): Promise<void>;
82
+ appendToolMessage(content: LangChainToolMessageContent, toolCallId: string): Promise<void>;
83
+ appendAIMessage(content: string | MessageContent): Promise<void>;
84
+ }
85
+ /**
86
+ * Creates a LangChain-specific thread manager that stores StoredMessage
87
+ * instances in Redis and provides convenience helpers for creating and
88
+ * appending typed LangChain messages.
89
+ */
90
+ declare function createLangChainThreadManager(config: LangChainThreadManagerConfig): LangChainThreadManager;
91
+
92
+ interface LangChainModelInvokerConfig {
93
+ redis: Redis;
94
+ model: BaseChatModel<BaseChatModelCallOptions & {
95
+ tools?: BindToolsInput;
96
+ }>;
97
+ }
98
+ /**
99
+ * Creates a LangChain-based model invoker that satisfies the generic
100
+ * `ModelInvoker<StoredMessage>` contract.
101
+ *
102
+ * Loads the conversation thread from Redis, invokes a LangChain chat model,
103
+ * appends the AI response, and returns a normalised AgentResponse.
104
+ *
105
+ * @example
106
+ * ```typescript
107
+ * import { createLangChainModelInvoker } from 'zeitlich/adapters/langchain';
108
+ * import { createRunAgentActivity } from 'zeitlich';
109
+ * import { ChatAnthropic } from '@langchain/anthropic';
110
+ *
111
+ * const model = new ChatAnthropic({ model: "claude-sonnet-4-6" });
112
+ * const invoker = createLangChainModelInvoker({ redis, model });
113
+ *
114
+ * // Wrap with createRunAgentActivity to use as runAgent activity:
115
+ * return { runAgent: createRunAgentActivity(client, invoker) };
116
+ * ```
117
+ */
118
+ declare function createLangChainModelInvoker({ redis, model, }: LangChainModelInvokerConfig): (config: ModelInvokerConfig) => Promise<AgentResponse<StoredMessage>>;
119
+ /**
120
+ * Standalone function for one-shot LangChain model invocation.
121
+ * Convenience wrapper around createLangChainModelInvoker for cases where
122
+ * you don't need to reuse the invoker.
123
+ */
124
+ declare function invokeLangChainModel({ redis, model, config, }: {
125
+ redis: Redis;
126
+ config: ModelInvokerConfig;
127
+ model: BaseChatModel<BaseChatModelCallOptions & {
128
+ tools?: BindToolsInput;
129
+ }>;
130
+ }): Promise<AgentResponse<StoredMessage>>;
131
+
132
+ export { type LangChainAdapter, type LangChainAdapterConfig, type LangChainModelInvokerConfig, type LangChainThreadManager, type LangChainThreadManagerConfig, type LangChainToolMessageContent, createLangChainAdapter, createLangChainModelInvoker, createLangChainThreadManager, invokeLangChainModel };
@@ -0,0 +1,132 @@
1
+ import Redis from 'ioredis';
2
+ import { T as ThreadOps, M as ModelInvoker, a as ModelInvokerConfig, A as AgentResponse } from '../../model-invoker-y_zlyMqu.js';
3
+ import { StoredMessage, MessageContent, $InferMessageContent, MessageStructure } from '@langchain/core/messages';
4
+ import { BaseChatModel, BaseChatModelCallOptions, BindToolsInput } from '@langchain/core/language_models/chat_models';
5
+ import { B as BaseThreadManager } from '../../thread-manager-qc0g5Rvd.js';
6
+ import 'zod';
7
+ import '@temporalio/common';
8
+
9
+ type LangChainModel = BaseChatModel<BaseChatModelCallOptions & {
10
+ tools?: BindToolsInput;
11
+ }>;
12
+ interface LangChainAdapterConfig {
13
+ redis: Redis;
14
+ /** Optional default model — if omitted, use `createModelInvoker()` to create invokers later */
15
+ model?: LangChainModel;
16
+ }
17
+ interface LangChainAdapter {
18
+ /** Thread operations (register these as Temporal activities on the worker) */
19
+ threadOps: ThreadOps;
20
+ /** Model invoker using the default model (only available when `model` was provided) */
21
+ invoker: ModelInvoker<StoredMessage>;
22
+ /** Create an invoker for a specific model (for multi-model setups) */
23
+ createModelInvoker(model: LangChainModel): ModelInvoker<StoredMessage>;
24
+ }
25
+ /**
26
+ * Creates a LangChain adapter that bundles thread operations and model
27
+ * invocation using a consistent message format (StoredMessage).
28
+ *
29
+ * The returned `threadOps` should be registered as Temporal activities on
30
+ * the worker. The `invoker` (or invokers created via `createModelInvoker`)
31
+ * should be wrapped with `createRunAgentActivity` for per-agent activities.
32
+ *
33
+ * @example
34
+ * ```typescript
35
+ * import { createLangChainAdapter } from 'zeitlich/adapters/langchain';
36
+ * import { createRunAgentActivity } from 'zeitlich';
37
+ *
38
+ * const adapter = createLangChainAdapter({ redis, model });
39
+ *
40
+ * export function createActivities(client: WorkflowClient) {
41
+ * return {
42
+ * ...adapter.threadOps,
43
+ * runAgent: createRunAgentActivity(client, adapter.invoker),
44
+ * };
45
+ * }
46
+ * ```
47
+ *
48
+ * @example Multi-model setup
49
+ * ```typescript
50
+ * const adapter = createLangChainAdapter({ redis });
51
+ *
52
+ * export function createActivities(client: WorkflowClient) {
53
+ * return {
54
+ * ...adapter.threadOps,
55
+ * runResearchAgent: createRunAgentActivity(client, adapter.createModelInvoker(claude)),
56
+ * runWriterAgent: createRunAgentActivity(client, adapter.createModelInvoker(gpt4)),
57
+ * };
58
+ * }
59
+ * ```
60
+ */
61
+ declare function createLangChainAdapter(config: LangChainAdapterConfig): LangChainAdapter;
62
+
63
+ type LangChainToolMessageContent = $InferMessageContent<MessageStructure, "tool">;
64
+ interface LangChainThreadManagerConfig {
65
+ redis: Redis;
66
+ threadId: string;
67
+ /** Thread key, defaults to 'messages' */
68
+ key?: string;
69
+ }
70
+ /** Thread manager with LangChain StoredMessage convenience helpers */
71
+ interface LangChainThreadManager extends BaseThreadManager<StoredMessage> {
72
+ createHumanMessage(content: string | MessageContent): StoredMessage;
73
+ createSystemMessage(content: string): StoredMessage;
74
+ createAIMessage(content: string | MessageContent, kwargs?: {
75
+ header?: string;
76
+ options?: string[];
77
+ multiSelect?: boolean;
78
+ }): StoredMessage;
79
+ createToolMessage(content: LangChainToolMessageContent, toolCallId: string): StoredMessage;
80
+ appendHumanMessage(content: string | MessageContent): Promise<void>;
81
+ appendSystemMessage(content: string): Promise<void>;
82
+ appendToolMessage(content: LangChainToolMessageContent, toolCallId: string): Promise<void>;
83
+ appendAIMessage(content: string | MessageContent): Promise<void>;
84
+ }
85
+ /**
86
+ * Creates a LangChain-specific thread manager that stores StoredMessage
87
+ * instances in Redis and provides convenience helpers for creating and
88
+ * appending typed LangChain messages.
89
+ */
90
+ declare function createLangChainThreadManager(config: LangChainThreadManagerConfig): LangChainThreadManager;
91
+
92
+ interface LangChainModelInvokerConfig {
93
+ redis: Redis;
94
+ model: BaseChatModel<BaseChatModelCallOptions & {
95
+ tools?: BindToolsInput;
96
+ }>;
97
+ }
98
+ /**
99
+ * Creates a LangChain-based model invoker that satisfies the generic
100
+ * `ModelInvoker<StoredMessage>` contract.
101
+ *
102
+ * Loads the conversation thread from Redis, invokes a LangChain chat model,
103
+ * appends the AI response, and returns a normalised AgentResponse.
104
+ *
105
+ * @example
106
+ * ```typescript
107
+ * import { createLangChainModelInvoker } from 'zeitlich/adapters/langchain';
108
+ * import { createRunAgentActivity } from 'zeitlich';
109
+ * import { ChatAnthropic } from '@langchain/anthropic';
110
+ *
111
+ * const model = new ChatAnthropic({ model: "claude-sonnet-4-6" });
112
+ * const invoker = createLangChainModelInvoker({ redis, model });
113
+ *
114
+ * // Wrap with createRunAgentActivity to use as runAgent activity:
115
+ * return { runAgent: createRunAgentActivity(client, invoker) };
116
+ * ```
117
+ */
118
+ declare function createLangChainModelInvoker({ redis, model, }: LangChainModelInvokerConfig): (config: ModelInvokerConfig) => Promise<AgentResponse<StoredMessage>>;
119
+ /**
120
+ * Standalone function for one-shot LangChain model invocation.
121
+ * Convenience wrapper around createLangChainModelInvoker for cases where
122
+ * you don't need to reuse the invoker.
123
+ */
124
+ declare function invokeLangChainModel({ redis, model, config, }: {
125
+ redis: Redis;
126
+ config: ModelInvokerConfig;
127
+ model: BaseChatModel<BaseChatModelCallOptions & {
128
+ tools?: BindToolsInput;
129
+ }>;
130
+ }): Promise<AgentResponse<StoredMessage>>;
131
+
132
+ export { type LangChainAdapter, type LangChainAdapterConfig, type LangChainModelInvokerConfig, type LangChainThreadManager, type LangChainThreadManagerConfig, type LangChainToolMessageContent, createLangChainAdapter, createLangChainModelInvoker, createLangChainThreadManager, invokeLangChainModel };