@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.
- package/README.md +74 -0
- package/dist/adapters/MemoryAdapter.d.ts +47 -0
- package/dist/adapters/MemoryAdapter.d.ts.map +1 -0
- package/dist/adapters/MemoryAdapter.js +178 -0
- package/dist/adapters/MemoryAdapter.js.map +1 -0
- package/dist/adapters/MongoAdapter.d.ts +97 -0
- package/dist/adapters/MongoAdapter.d.ts.map +1 -0
- package/dist/adapters/MongoAdapter.js +163 -0
- package/dist/adapters/MongoAdapter.js.map +1 -0
- package/dist/adapters/PostgreSQLAdapter.d.ts +71 -0
- package/dist/adapters/PostgreSQLAdapter.d.ts.map +1 -0
- package/dist/adapters/PostgreSQLAdapter.js +256 -0
- package/dist/adapters/PostgreSQLAdapter.js.map +1 -0
- package/dist/adapters/RedisAdapter.d.ts +71 -0
- package/dist/adapters/RedisAdapter.d.ts.map +1 -0
- package/dist/adapters/RedisAdapter.js +226 -0
- package/dist/adapters/RedisAdapter.js.map +1 -0
- package/dist/adapters/SQLiteAdapter.d.ts +69 -0
- package/dist/adapters/SQLiteAdapter.d.ts.map +1 -0
- package/dist/adapters/SQLiteAdapter.js +307 -0
- package/dist/adapters/SQLiteAdapter.js.map +1 -0
- package/dist/adapters/index.d.ts +9 -0
- package/dist/adapters/index.d.ts.map +1 -1
- package/dist/adapters/index.js +5 -0
- package/dist/adapters/index.js.map +1 -1
- package/dist/cjs/adapters/MemoryAdapter.d.ts +47 -0
- package/dist/cjs/adapters/MemoryAdapter.d.ts.map +1 -0
- package/dist/cjs/adapters/MemoryAdapter.js +182 -0
- package/dist/cjs/adapters/MemoryAdapter.js.map +1 -0
- package/dist/cjs/adapters/MongoAdapter.d.ts +97 -0
- package/dist/cjs/adapters/MongoAdapter.d.ts.map +1 -0
- package/dist/cjs/adapters/MongoAdapter.js +167 -0
- package/dist/cjs/adapters/MongoAdapter.js.map +1 -0
- package/dist/cjs/adapters/PostgreSQLAdapter.d.ts +71 -0
- package/dist/cjs/adapters/PostgreSQLAdapter.d.ts.map +1 -0
- package/dist/cjs/adapters/PostgreSQLAdapter.js +260 -0
- package/dist/cjs/adapters/PostgreSQLAdapter.js.map +1 -0
- package/dist/cjs/adapters/RedisAdapter.d.ts +71 -0
- package/dist/cjs/adapters/RedisAdapter.d.ts.map +1 -0
- package/dist/cjs/adapters/RedisAdapter.js +230 -0
- package/dist/cjs/adapters/RedisAdapter.js.map +1 -0
- package/dist/cjs/adapters/SQLiteAdapter.d.ts +69 -0
- package/dist/cjs/adapters/SQLiteAdapter.d.ts.map +1 -0
- package/dist/cjs/adapters/SQLiteAdapter.js +311 -0
- package/dist/cjs/adapters/SQLiteAdapter.js.map +1 -0
- package/dist/cjs/adapters/index.d.ts +9 -0
- package/dist/cjs/adapters/index.d.ts.map +1 -1
- package/dist/cjs/adapters/index.js +11 -1
- package/dist/cjs/adapters/index.js.map +1 -1
- package/dist/cjs/index.d.ts +9 -0
- package/dist/cjs/index.d.ts.map +1 -1
- package/dist/cjs/index.js +11 -1
- package/dist/cjs/index.js.map +1 -1
- package/dist/index.d.ts +9 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +5 -0
- package/dist/index.js.map +1 -1
- package/docs/ADAPTERS.md +151 -0
- package/docs/API_REFERENCE.md +448 -0
- package/docs/PERSISTENCE.md +176 -6
- package/examples/redis-persistence.ts +89 -0
- package/package.json +26 -2
- package/src/adapters/MemoryAdapter.ts +245 -0
- package/src/adapters/MongoAdapter.ts +295 -0
- package/src/adapters/PostgreSQLAdapter.ts +417 -0
- package/src/adapters/RedisAdapter.ts +365 -0
- package/src/adapters/SQLiteAdapter.ts +449 -0
- package/src/adapters/index.ts +27 -0
- 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
|
+
}
|
package/src/adapters/index.ts
CHANGED
|
@@ -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";
|