@falai/agent 0.3.12 → 0.3.21

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 (69) hide show
  1. package/README.md +74 -0
  2. package/dist/adapters/MemoryAdapter.d.ts +47 -0
  3. package/dist/adapters/MemoryAdapter.d.ts.map +1 -0
  4. package/dist/adapters/MemoryAdapter.js +178 -0
  5. package/dist/adapters/MemoryAdapter.js.map +1 -0
  6. package/dist/adapters/MongoAdapter.d.ts +97 -0
  7. package/dist/adapters/MongoAdapter.d.ts.map +1 -0
  8. package/dist/adapters/MongoAdapter.js +163 -0
  9. package/dist/adapters/MongoAdapter.js.map +1 -0
  10. package/dist/adapters/PostgreSQLAdapter.d.ts +71 -0
  11. package/dist/adapters/PostgreSQLAdapter.d.ts.map +1 -0
  12. package/dist/adapters/PostgreSQLAdapter.js +256 -0
  13. package/dist/adapters/PostgreSQLAdapter.js.map +1 -0
  14. package/dist/adapters/RedisAdapter.d.ts +71 -0
  15. package/dist/adapters/RedisAdapter.d.ts.map +1 -0
  16. package/dist/adapters/RedisAdapter.js +226 -0
  17. package/dist/adapters/RedisAdapter.js.map +1 -0
  18. package/dist/adapters/SQLiteAdapter.d.ts +69 -0
  19. package/dist/adapters/SQLiteAdapter.d.ts.map +1 -0
  20. package/dist/adapters/SQLiteAdapter.js +307 -0
  21. package/dist/adapters/SQLiteAdapter.js.map +1 -0
  22. package/dist/adapters/index.d.ts +9 -0
  23. package/dist/adapters/index.d.ts.map +1 -1
  24. package/dist/adapters/index.js +5 -0
  25. package/dist/adapters/index.js.map +1 -1
  26. package/dist/cjs/adapters/MemoryAdapter.d.ts +47 -0
  27. package/dist/cjs/adapters/MemoryAdapter.d.ts.map +1 -0
  28. package/dist/cjs/adapters/MemoryAdapter.js +182 -0
  29. package/dist/cjs/adapters/MemoryAdapter.js.map +1 -0
  30. package/dist/cjs/adapters/MongoAdapter.d.ts +97 -0
  31. package/dist/cjs/adapters/MongoAdapter.d.ts.map +1 -0
  32. package/dist/cjs/adapters/MongoAdapter.js +167 -0
  33. package/dist/cjs/adapters/MongoAdapter.js.map +1 -0
  34. package/dist/cjs/adapters/PostgreSQLAdapter.d.ts +71 -0
  35. package/dist/cjs/adapters/PostgreSQLAdapter.d.ts.map +1 -0
  36. package/dist/cjs/adapters/PostgreSQLAdapter.js +260 -0
  37. package/dist/cjs/adapters/PostgreSQLAdapter.js.map +1 -0
  38. package/dist/cjs/adapters/RedisAdapter.d.ts +71 -0
  39. package/dist/cjs/adapters/RedisAdapter.d.ts.map +1 -0
  40. package/dist/cjs/adapters/RedisAdapter.js +230 -0
  41. package/dist/cjs/adapters/RedisAdapter.js.map +1 -0
  42. package/dist/cjs/adapters/SQLiteAdapter.d.ts +69 -0
  43. package/dist/cjs/adapters/SQLiteAdapter.d.ts.map +1 -0
  44. package/dist/cjs/adapters/SQLiteAdapter.js +311 -0
  45. package/dist/cjs/adapters/SQLiteAdapter.js.map +1 -0
  46. package/dist/cjs/adapters/index.d.ts +9 -0
  47. package/dist/cjs/adapters/index.d.ts.map +1 -1
  48. package/dist/cjs/adapters/index.js +11 -1
  49. package/dist/cjs/adapters/index.js.map +1 -1
  50. package/dist/cjs/index.d.ts +9 -0
  51. package/dist/cjs/index.d.ts.map +1 -1
  52. package/dist/cjs/index.js +11 -1
  53. package/dist/cjs/index.js.map +1 -1
  54. package/dist/index.d.ts +9 -0
  55. package/dist/index.d.ts.map +1 -1
  56. package/dist/index.js +5 -0
  57. package/dist/index.js.map +1 -1
  58. package/docs/ADAPTERS.md +151 -0
  59. package/docs/API_REFERENCE.md +448 -0
  60. package/docs/PERSISTENCE.md +176 -6
  61. package/examples/redis-persistence.ts +89 -0
  62. package/package.json +26 -2
  63. package/src/adapters/MemoryAdapter.ts +245 -0
  64. package/src/adapters/MongoAdapter.ts +295 -0
  65. package/src/adapters/PostgreSQLAdapter.ts +417 -0
  66. package/src/adapters/RedisAdapter.ts +365 -0
  67. package/src/adapters/SQLiteAdapter.ts +449 -0
  68. package/src/adapters/index.ts +27 -0
  69. package/src/index.ts +22 -0
@@ -0,0 +1,449 @@
1
+ /**
2
+ * SQLite adapter for persistence
3
+ * Lightweight, file-based database perfect for local development
4
+ */
5
+
6
+ import type {
7
+ SessionRepository,
8
+ MessageRepository,
9
+ SessionData,
10
+ MessageData,
11
+ SessionStatus,
12
+ PersistenceAdapter,
13
+ } from "../types/persistence";
14
+
15
+ /**
16
+ * SQLite database interface - matches better-sqlite3
17
+ */
18
+ export interface SqliteDatabase {
19
+ prepare(sql: string): SqliteStatement;
20
+ exec(sql: string): void;
21
+ close(): void;
22
+ }
23
+
24
+ /**
25
+ * SQLite statement interface
26
+ */
27
+ export interface SqliteStatement {
28
+ run(...params: unknown[]): { changes: number; lastInsertRowid: number };
29
+ get(...params: unknown[]): Record<string, unknown> | undefined;
30
+ all(...params: unknown[]): Array<Record<string, unknown>>;
31
+ }
32
+
33
+ /**
34
+ * Options for SQLite adapter
35
+ */
36
+ export interface SQLiteAdapterOptions {
37
+ /**
38
+ * SQLite database instance (better-sqlite3)
39
+ */
40
+ db: SqliteDatabase;
41
+
42
+ /**
43
+ * Table names (default: "agent_sessions" and "agent_messages")
44
+ */
45
+ tables?: {
46
+ sessions?: string;
47
+ messages?: string;
48
+ };
49
+ }
50
+
51
+ /**
52
+ * SQLite Adapter - Provider-style API for SQLite persistence
53
+ *
54
+ * @example
55
+ * ```typescript
56
+ * import Database from 'better-sqlite3';
57
+ * import { Agent, SQLiteAdapter } from '@falai/agent';
58
+ *
59
+ * const db = new Database('agent.db');
60
+ *
61
+ * const agent = new Agent({
62
+ * name: "My Agent",
63
+ * ai: provider,
64
+ * persistence: {
65
+ * adapter: new SQLiteAdapter({ db }),
66
+ * userId: "user_123",
67
+ * },
68
+ * });
69
+ * ```
70
+ */
71
+ export class SQLiteAdapter implements PersistenceAdapter {
72
+ public readonly sessionRepository: SessionRepository;
73
+ public readonly messageRepository: MessageRepository;
74
+ private db: SqliteDatabase;
75
+
76
+ constructor(options: SQLiteAdapterOptions) {
77
+ this.db = options.db;
78
+
79
+ const sessionTable = options.tables?.sessions || "agent_sessions";
80
+ const messageTable = options.tables?.messages || "agent_messages";
81
+
82
+ this.sessionRepository = new SQLiteSessionRepository(this.db, sessionTable);
83
+ this.messageRepository = new SQLiteMessageRepository(this.db, messageTable);
84
+ }
85
+
86
+ async initialize(): Promise<void> {
87
+ const sessionTable = "agent_sessions";
88
+ const messageTable = "agent_messages";
89
+
90
+ // Create sessions table
91
+ this.db.exec(`
92
+ CREATE TABLE IF NOT EXISTS ${sessionTable} (
93
+ id TEXT PRIMARY KEY,
94
+ user_id TEXT,
95
+ agent_name TEXT,
96
+ status TEXT DEFAULT 'active',
97
+ current_route TEXT,
98
+ current_state TEXT,
99
+ collected_data TEXT,
100
+ message_count INTEGER DEFAULT 0,
101
+ last_message_at TEXT,
102
+ completed_at TEXT,
103
+ created_at TEXT NOT NULL,
104
+ updated_at TEXT NOT NULL
105
+ )
106
+ `);
107
+
108
+ // Create indexes for sessions
109
+ this.db.exec(`
110
+ CREATE INDEX IF NOT EXISTS idx_sessions_user_id ON ${sessionTable}(user_id);
111
+ CREATE INDEX IF NOT EXISTS idx_sessions_status ON ${sessionTable}(status);
112
+ `);
113
+
114
+ // Create messages table
115
+ this.db.exec(`
116
+ CREATE TABLE IF NOT EXISTS ${messageTable} (
117
+ id TEXT PRIMARY KEY,
118
+ session_id TEXT NOT NULL,
119
+ user_id TEXT,
120
+ role TEXT NOT NULL,
121
+ content TEXT NOT NULL,
122
+ route TEXT,
123
+ state TEXT,
124
+ tool_calls TEXT,
125
+ event TEXT,
126
+ created_at TEXT NOT NULL,
127
+ FOREIGN KEY (session_id) REFERENCES ${sessionTable}(id) ON DELETE CASCADE
128
+ )
129
+ `);
130
+
131
+ // Create indexes for messages
132
+ this.db.exec(`
133
+ CREATE INDEX IF NOT EXISTS idx_messages_session_id ON ${messageTable}(session_id);
134
+ CREATE INDEX IF NOT EXISTS idx_messages_user_id ON ${messageTable}(user_id);
135
+ `);
136
+ await Promise.resolve();
137
+ }
138
+
139
+ async disconnect(): Promise<void> {
140
+ this.db.close();
141
+ await Promise.resolve();
142
+ }
143
+ }
144
+
145
+ /**
146
+ * SQLite Session Repository
147
+ */
148
+ class SQLiteSessionRepository implements SessionRepository {
149
+ constructor(private db: SqliteDatabase, private tableName: string) {}
150
+
151
+ create(
152
+ data: Omit<SessionData, "id" | "createdAt" | "updatedAt">
153
+ ): Promise<SessionData> {
154
+ const id = `session_${Date.now()}_${Math.random().toString(36).slice(2)}`;
155
+ const now = new Date();
156
+
157
+ const session: SessionData = {
158
+ ...data,
159
+ id,
160
+ status: data.status || "active",
161
+ messageCount: data.messageCount || 0,
162
+ createdAt: now,
163
+ updatedAt: now,
164
+ };
165
+
166
+ const stmt = this.db.prepare(`
167
+ INSERT INTO ${this.tableName}
168
+ (id, user_id, agent_name, status, collected_data, message_count, created_at, updated_at)
169
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?)
170
+ `);
171
+
172
+ stmt.run(
173
+ id,
174
+ data.userId || null,
175
+ data.agentName || null,
176
+ session.status,
177
+ JSON.stringify(data.collectedData || {}),
178
+ session.messageCount,
179
+ now.toISOString(),
180
+ now.toISOString()
181
+ );
182
+
183
+ return Promise.resolve(session);
184
+ }
185
+
186
+ findById(id: string): Promise<SessionData | null> {
187
+ const stmt = this.db.prepare(
188
+ `SELECT * FROM ${this.tableName} WHERE id = ?`
189
+ );
190
+ const row = stmt.get(id);
191
+ return Promise.resolve(row ? this.deserializeSession(row) : null);
192
+ }
193
+
194
+ findActiveByUserId(userId: string): Promise<SessionData | null> {
195
+ const stmt = this.db.prepare(
196
+ `SELECT * FROM ${this.tableName}
197
+ WHERE user_id = ? AND status = 'active'
198
+ ORDER BY created_at DESC
199
+ LIMIT 1`
200
+ );
201
+ const row = stmt.get(userId);
202
+ return Promise.resolve(row ? this.deserializeSession(row) : null);
203
+ }
204
+
205
+ findByUserId(userId: string, limit = 100): Promise<SessionData[]> {
206
+ const stmt = this.db.prepare(
207
+ `SELECT * FROM ${this.tableName}
208
+ WHERE user_id = ?
209
+ ORDER BY created_at DESC
210
+ LIMIT ?`
211
+ );
212
+ const rows = stmt.all(userId, limit);
213
+ return Promise.resolve(rows.map((row) => this.deserializeSession(row)));
214
+ }
215
+
216
+ async update(
217
+ id: string,
218
+ data: Partial<Omit<SessionData, "id" | "createdAt">>
219
+ ): Promise<SessionData | null> {
220
+ const fields: string[] = [];
221
+ const values: unknown[] = [];
222
+
223
+ if (data.status !== undefined) {
224
+ fields.push("status = ?");
225
+ values.push(data.status);
226
+ }
227
+ if (data.collectedData !== undefined) {
228
+ fields.push("collected_data = ?");
229
+ values.push(JSON.stringify(data.collectedData));
230
+ }
231
+ if (data.currentRoute !== undefined) {
232
+ fields.push("current_route = ?");
233
+ values.push(data.currentRoute);
234
+ }
235
+ if (data.currentState !== undefined) {
236
+ fields.push("current_state = ?");
237
+ values.push(data.currentState);
238
+ }
239
+ if (data.messageCount !== undefined) {
240
+ fields.push("message_count = ?");
241
+ values.push(data.messageCount);
242
+ }
243
+ if (data.lastMessageAt !== undefined) {
244
+ fields.push("last_message_at = ?");
245
+ values.push(data.lastMessageAt.toISOString());
246
+ }
247
+ if (data.completedAt !== undefined) {
248
+ fields.push("completed_at = ?");
249
+ values.push(data.completedAt.toISOString());
250
+ }
251
+
252
+ fields.push("updated_at = ?");
253
+ values.push(new Date().toISOString());
254
+
255
+ values.push(id);
256
+
257
+ const stmt = this.db.prepare(
258
+ `UPDATE ${this.tableName}
259
+ SET ${fields.join(", ")}
260
+ WHERE id = ?`
261
+ );
262
+
263
+ const result = stmt.run(...values);
264
+ if (result.changes === 0) return null;
265
+
266
+ return await this.findById(id);
267
+ }
268
+
269
+ async updateStatus(
270
+ id: string,
271
+ status: SessionStatus,
272
+ completedAt?: Date
273
+ ): Promise<SessionData | null> {
274
+ return await this.update(id, { status, completedAt });
275
+ }
276
+
277
+ async updateCollectedData(
278
+ id: string,
279
+ collectedData: Record<string, unknown>
280
+ ): Promise<SessionData | null> {
281
+ return await this.update(id, { collectedData });
282
+ }
283
+
284
+ async updateRouteState(
285
+ id: string,
286
+ route?: string,
287
+ state?: string
288
+ ): Promise<SessionData | null> {
289
+ return await this.update(id, {
290
+ currentRoute: route,
291
+ currentState: state,
292
+ });
293
+ }
294
+
295
+ async incrementMessageCount(id: string): Promise<SessionData | null> {
296
+ const stmt = this.db.prepare(
297
+ `UPDATE ${this.tableName}
298
+ SET message_count = message_count + 1,
299
+ last_message_at = ?,
300
+ updated_at = ?
301
+ WHERE id = ?`
302
+ );
303
+
304
+ const now = new Date();
305
+ const result = stmt.run(now.toISOString(), now.toISOString(), id);
306
+
307
+ if (result.changes === 0) return null;
308
+ return await this.findById(id);
309
+ }
310
+
311
+ delete(id: string): Promise<boolean> {
312
+ const stmt = this.db.prepare(`DELETE FROM ${this.tableName} WHERE id = ?`);
313
+ const result = stmt.run(id);
314
+ return Promise.resolve(result.changes > 0);
315
+ }
316
+
317
+ private deserializeSession(row: Record<string, unknown>): SessionData {
318
+ return {
319
+ id: row.id as string,
320
+ userId: (row.user_id as string) || undefined,
321
+ agentName: (row.agent_name as string) || undefined,
322
+ status: row.status as SessionStatus,
323
+ currentRoute: (row.current_route as string) || undefined,
324
+ currentState: (row.current_state as string) || undefined,
325
+ collectedData: row.collected_data
326
+ ? (JSON.parse(row.collected_data as string) as Record<string, unknown>)
327
+ : undefined,
328
+ messageCount: (row.message_count as number) || 0,
329
+ lastMessageAt: row.last_message_at
330
+ ? new Date(row.last_message_at as string)
331
+ : undefined,
332
+ completedAt: row.completed_at
333
+ ? new Date(row.completed_at as string)
334
+ : undefined,
335
+ createdAt: new Date(row.created_at as string),
336
+ updatedAt: new Date(row.updated_at as string),
337
+ };
338
+ }
339
+ }
340
+
341
+ /**
342
+ * SQLite Message Repository
343
+ */
344
+ class SQLiteMessageRepository implements MessageRepository {
345
+ constructor(private db: SqliteDatabase, private tableName: string) {}
346
+
347
+ create(data: Omit<MessageData, "id" | "createdAt">): Promise<MessageData> {
348
+ const id = `msg_${Date.now()}_${Math.random().toString(36).slice(2)}`;
349
+ const now = new Date();
350
+
351
+ const message: MessageData = {
352
+ ...data,
353
+ id,
354
+ createdAt: now,
355
+ };
356
+
357
+ const stmt = this.db.prepare(`
358
+ INSERT INTO ${this.tableName}
359
+ (id, session_id, user_id, role, content, route, state, tool_calls, event, created_at)
360
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
361
+ `);
362
+
363
+ stmt.run(
364
+ id,
365
+ data.sessionId,
366
+ data.userId || null,
367
+ data.role,
368
+ data.content,
369
+ data.route || null,
370
+ data.state || null,
371
+ JSON.stringify(data.toolCalls || null),
372
+ JSON.stringify(data.event || null),
373
+ now.toISOString()
374
+ );
375
+
376
+ return Promise.resolve(message);
377
+ }
378
+
379
+ findById(id: string): Promise<MessageData | null> {
380
+ const stmt = this.db.prepare(
381
+ `SELECT * FROM ${this.tableName} WHERE id = ?`
382
+ );
383
+ const row = stmt.get(id);
384
+ return Promise.resolve(row ? this.deserializeMessage(row) : null);
385
+ }
386
+
387
+ findBySessionId(sessionId: string, limit = 1000): Promise<MessageData[]> {
388
+ const stmt = this.db.prepare(
389
+ `SELECT * FROM ${this.tableName}
390
+ WHERE session_id = ?
391
+ ORDER BY created_at ASC
392
+ LIMIT ?`
393
+ );
394
+ const rows = stmt.all(sessionId, limit);
395
+ return Promise.resolve(rows.map((row) => this.deserializeMessage(row)));
396
+ }
397
+
398
+ findByUserId(userId: string, limit = 100): Promise<MessageData[]> {
399
+ const stmt = this.db.prepare(
400
+ `SELECT * FROM ${this.tableName}
401
+ WHERE user_id = ?
402
+ ORDER BY created_at DESC
403
+ LIMIT ?`
404
+ );
405
+ const rows = stmt.all(userId, limit);
406
+ return Promise.resolve(rows.map((row) => this.deserializeMessage(row)));
407
+ }
408
+
409
+ delete(id: string): Promise<boolean> {
410
+ const stmt = this.db.prepare(`DELETE FROM ${this.tableName} WHERE id = ?`);
411
+ const result = stmt.run(id);
412
+ return Promise.resolve(result.changes > 0);
413
+ }
414
+
415
+ deleteBySessionId(sessionId: string): Promise<number> {
416
+ const stmt = this.db.prepare(
417
+ `DELETE FROM ${this.tableName} WHERE session_id = ?`
418
+ );
419
+ const result = stmt.run(sessionId);
420
+ return Promise.resolve(result.changes);
421
+ }
422
+
423
+ deleteByUserId(userId: string): Promise<number> {
424
+ const stmt = this.db.prepare(
425
+ `DELETE FROM ${this.tableName} WHERE user_id = ?`
426
+ );
427
+ const result = stmt.run(userId);
428
+ return Promise.resolve(result.changes);
429
+ }
430
+
431
+ private deserializeMessage(row: Record<string, unknown>): MessageData {
432
+ return {
433
+ id: row.id as string,
434
+ sessionId: row.session_id as string,
435
+ userId: (row.user_id as string) || undefined,
436
+ role: row.role as MessageData["role"],
437
+ content: row.content as string,
438
+ route: (row.route as string) || undefined,
439
+ state: (row.state as string) || undefined,
440
+ toolCalls: row.tool_calls
441
+ ? (JSON.parse(row.tool_calls as string) as MessageData["toolCalls"])
442
+ : undefined,
443
+ event: row.event
444
+ ? (JSON.parse(row.event as string) as MessageData["event"])
445
+ : undefined,
446
+ createdAt: new Date(row.created_at as string),
447
+ };
448
+ }
449
+ }
@@ -8,3 +8,30 @@ export type {
8
8
  FieldMappings,
9
9
  PrismaAdapterOptions,
10
10
  } from "./PrismaAdapter";
11
+
12
+ export { RedisAdapter } from "./RedisAdapter";
13
+ export type { RedisClient, RedisAdapterOptions } from "./RedisAdapter";
14
+
15
+ export { MongoAdapter } from "./MongoAdapter";
16
+ export type {
17
+ MongoClient,
18
+ MongoDatabase,
19
+ MongoCollection,
20
+ MongoAdapterOptions,
21
+ } from "./MongoAdapter";
22
+
23
+ export { PostgreSQLAdapter } from "./PostgreSQLAdapter";
24
+ export type {
25
+ PgClient,
26
+ PgQueryResult,
27
+ PostgreSQLAdapterOptions,
28
+ } from "./PostgreSQLAdapter";
29
+
30
+ export { SQLiteAdapter } from "./SQLiteAdapter";
31
+ export type {
32
+ SqliteDatabase,
33
+ SqliteStatement,
34
+ SQLiteAdapterOptions,
35
+ } from "./SQLiteAdapter";
36
+
37
+ export { MemoryAdapter } from "./MemoryAdapter";
package/src/index.ts CHANGED
@@ -35,6 +35,28 @@ export type {
35
35
  FieldMappings,
36
36
  PrismaAdapterOptions,
37
37
  } from "./adapters/PrismaAdapter";
38
+ export { RedisAdapter } from "./adapters/RedisAdapter";
39
+ export type { RedisClient, RedisAdapterOptions } from "./adapters/RedisAdapter";
40
+ export { MongoAdapter } from "./adapters/MongoAdapter";
41
+ export type {
42
+ MongoClient,
43
+ MongoDatabase,
44
+ MongoCollection,
45
+ MongoAdapterOptions,
46
+ } from "./adapters/MongoAdapter";
47
+ export { PostgreSQLAdapter } from "./adapters/PostgreSQLAdapter";
48
+ export type {
49
+ PgClient,
50
+ PgQueryResult,
51
+ PostgreSQLAdapterOptions,
52
+ } from "./adapters/PostgreSQLAdapter";
53
+ export { SQLiteAdapter } from "./adapters/SQLiteAdapter";
54
+ export type {
55
+ SqliteDatabase,
56
+ SqliteStatement,
57
+ SQLiteAdapterOptions,
58
+ } from "./adapters/SQLiteAdapter";
59
+ export { MemoryAdapter } from "./adapters/MemoryAdapter";
38
60
 
39
61
  // Constants
40
62
  export { END_ROUTE } from "./constants";