@dtelecom/agents-js 0.1.0 → 0.1.2
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/index.d.mts +3 -69
- package/dist/index.d.ts +3 -69
- package/dist/index.js +1 -10
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +3 -11
- package/dist/index.mjs.map +1 -1
- package/dist/memory/index.d.mts +70 -0
- package/dist/memory/index.d.ts +70 -0
- package/dist/memory/index.js +486 -0
- package/dist/memory/index.js.map +1 -0
- package/dist/memory/index.mjs +12 -0
- package/dist/memory/index.mjs.map +1 -0
- package/dist/providers/index.d.mts +1 -1
- package/dist/providers/index.d.ts +1 -1
- package/dist/{types-Cs5uUoTC.d.mts → types-EvtHMokR.d.mts} +1 -1
- package/dist/{types-Cs5uUoTC.d.ts → types-EvtHMokR.d.ts} +1 -1
- package/package.json +6 -1
|
@@ -0,0 +1,486 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __create = Object.create;
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
+
var __export = (target, all) => {
|
|
9
|
+
for (var name in all)
|
|
10
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
11
|
+
};
|
|
12
|
+
var __copyProps = (to, from, except, desc) => {
|
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
14
|
+
for (let key of __getOwnPropNames(from))
|
|
15
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
16
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
17
|
+
}
|
|
18
|
+
return to;
|
|
19
|
+
};
|
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
21
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
22
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
23
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
24
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
25
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
26
|
+
mod
|
|
27
|
+
));
|
|
28
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
29
|
+
|
|
30
|
+
// src/memory/index.ts
|
|
31
|
+
var memory_exports = {};
|
|
32
|
+
__export(memory_exports, {
|
|
33
|
+
Embedder: () => Embedder,
|
|
34
|
+
MemoryStore: () => MemoryStore,
|
|
35
|
+
RoomMemory: () => RoomMemory
|
|
36
|
+
});
|
|
37
|
+
module.exports = __toCommonJS(memory_exports);
|
|
38
|
+
|
|
39
|
+
// src/utils/logger.ts
|
|
40
|
+
var LEVELS = {
|
|
41
|
+
debug: 0,
|
|
42
|
+
info: 1,
|
|
43
|
+
warn: 2,
|
|
44
|
+
error: 3,
|
|
45
|
+
silent: 4
|
|
46
|
+
};
|
|
47
|
+
function detectLevel() {
|
|
48
|
+
const debug = typeof process !== "undefined" && process.env?.DEBUG;
|
|
49
|
+
if (debug && (debug === "*" || debug.includes("@dtelecom/agents"))) {
|
|
50
|
+
return "debug";
|
|
51
|
+
}
|
|
52
|
+
return "info";
|
|
53
|
+
}
|
|
54
|
+
var globalLevel = detectLevel();
|
|
55
|
+
function timestamp() {
|
|
56
|
+
const d = /* @__PURE__ */ new Date();
|
|
57
|
+
const h = String(d.getHours()).padStart(2, "0");
|
|
58
|
+
const m = String(d.getMinutes()).padStart(2, "0");
|
|
59
|
+
const s = String(d.getSeconds()).padStart(2, "0");
|
|
60
|
+
const ms = String(d.getMilliseconds()).padStart(3, "0");
|
|
61
|
+
return `${h}:${m}:${s}.${ms}`;
|
|
62
|
+
}
|
|
63
|
+
function createLogger(tag) {
|
|
64
|
+
const prefix = `[@dtelecom/agents:${tag}]`;
|
|
65
|
+
return {
|
|
66
|
+
debug(...args) {
|
|
67
|
+
if (LEVELS[globalLevel] <= LEVELS.debug) console.debug(timestamp(), prefix, ...args);
|
|
68
|
+
},
|
|
69
|
+
info(...args) {
|
|
70
|
+
if (LEVELS[globalLevel] <= LEVELS.info) console.info(timestamp(), prefix, ...args);
|
|
71
|
+
},
|
|
72
|
+
warn(...args) {
|
|
73
|
+
if (LEVELS[globalLevel] <= LEVELS.warn) console.warn(timestamp(), prefix, ...args);
|
|
74
|
+
},
|
|
75
|
+
error(...args) {
|
|
76
|
+
if (LEVELS[globalLevel] <= LEVELS.error) console.error(timestamp(), prefix, ...args);
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
// src/memory/embedder.ts
|
|
82
|
+
var log = createLogger("Embedder");
|
|
83
|
+
var MODEL_NAME = "Xenova/all-MiniLM-L6-v2";
|
|
84
|
+
var EMBEDDING_DIM = 384;
|
|
85
|
+
var Embedder = class {
|
|
86
|
+
pipeline = null;
|
|
87
|
+
initPromise = null;
|
|
88
|
+
get dimensions() {
|
|
89
|
+
return EMBEDDING_DIM;
|
|
90
|
+
}
|
|
91
|
+
/** Load the embedding model. Call once at startup. */
|
|
92
|
+
async init() {
|
|
93
|
+
if (this.pipeline) return;
|
|
94
|
+
if (this.initPromise) return this.initPromise;
|
|
95
|
+
this.initPromise = this.loadModel();
|
|
96
|
+
return this.initPromise;
|
|
97
|
+
}
|
|
98
|
+
async loadModel() {
|
|
99
|
+
const start = performance.now();
|
|
100
|
+
log.info(`Loading embedding model "${MODEL_NAME}"...`);
|
|
101
|
+
const { pipeline } = await import("@huggingface/transformers");
|
|
102
|
+
this.pipeline = await pipeline("feature-extraction", MODEL_NAME);
|
|
103
|
+
log.info(`Embedding model loaded in ${(performance.now() - start).toFixed(0)}ms`);
|
|
104
|
+
}
|
|
105
|
+
/** Embed a single text. Returns Float32Array of length 384. */
|
|
106
|
+
async embed(text) {
|
|
107
|
+
await this.init();
|
|
108
|
+
const result = await this.pipeline(text, {
|
|
109
|
+
pooling: "mean",
|
|
110
|
+
normalize: true
|
|
111
|
+
});
|
|
112
|
+
return new Float32Array(result.data);
|
|
113
|
+
}
|
|
114
|
+
/** Cosine similarity between two normalized vectors. Returns value in [-1, 1]. */
|
|
115
|
+
static cosineSimilarity(a, b) {
|
|
116
|
+
let dot = 0;
|
|
117
|
+
for (let i = 0; i < a.length; i++) {
|
|
118
|
+
dot += a[i] * b[i];
|
|
119
|
+
}
|
|
120
|
+
return dot;
|
|
121
|
+
}
|
|
122
|
+
/** Embed multiple texts in one call (more efficient than calling embed() in a loop). */
|
|
123
|
+
async embedBatch(texts) {
|
|
124
|
+
if (texts.length === 0) return [];
|
|
125
|
+
await this.init();
|
|
126
|
+
const results = [];
|
|
127
|
+
for (const text of texts) {
|
|
128
|
+
const result = await this.pipeline(text, {
|
|
129
|
+
pooling: "mean",
|
|
130
|
+
normalize: true
|
|
131
|
+
});
|
|
132
|
+
results.push(new Float32Array(result.data));
|
|
133
|
+
}
|
|
134
|
+
return results;
|
|
135
|
+
}
|
|
136
|
+
};
|
|
137
|
+
|
|
138
|
+
// src/memory/memory-store.ts
|
|
139
|
+
var import_better_sqlite3 = __toESM(require("better-sqlite3"));
|
|
140
|
+
var sqliteVec = __toESM(require("sqlite-vec"));
|
|
141
|
+
var log2 = createLogger("MemoryStore");
|
|
142
|
+
var MemoryStore = class {
|
|
143
|
+
db;
|
|
144
|
+
constructor(dbPath) {
|
|
145
|
+
this.db = new import_better_sqlite3.default(dbPath);
|
|
146
|
+
this.db.pragma("journal_mode = WAL");
|
|
147
|
+
this.db.pragma("synchronous = NORMAL");
|
|
148
|
+
sqliteVec.load(this.db);
|
|
149
|
+
this.createTables();
|
|
150
|
+
log2.info(`Memory store opened: ${dbPath}`);
|
|
151
|
+
}
|
|
152
|
+
createTables() {
|
|
153
|
+
this.db.exec(`
|
|
154
|
+
CREATE TABLE IF NOT EXISTS turns (
|
|
155
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
156
|
+
room TEXT NOT NULL,
|
|
157
|
+
session_id TEXT NOT NULL,
|
|
158
|
+
speaker TEXT NOT NULL,
|
|
159
|
+
text TEXT NOT NULL,
|
|
160
|
+
is_agent BOOLEAN DEFAULT 0,
|
|
161
|
+
created_at INTEGER NOT NULL
|
|
162
|
+
);
|
|
163
|
+
|
|
164
|
+
CREATE TABLE IF NOT EXISTS sessions (
|
|
165
|
+
id TEXT PRIMARY KEY,
|
|
166
|
+
room TEXT NOT NULL,
|
|
167
|
+
started_at INTEGER NOT NULL,
|
|
168
|
+
ended_at INTEGER,
|
|
169
|
+
participants TEXT,
|
|
170
|
+
summary TEXT,
|
|
171
|
+
turn_count INTEGER DEFAULT 0
|
|
172
|
+
);
|
|
173
|
+
|
|
174
|
+
CREATE INDEX IF NOT EXISTS idx_turns_room_session ON turns(room, session_id);
|
|
175
|
+
CREATE INDEX IF NOT EXISTS idx_turns_room_time ON turns(room, created_at);
|
|
176
|
+
CREATE INDEX IF NOT EXISTS idx_sessions_room ON sessions(room);
|
|
177
|
+
`);
|
|
178
|
+
const hasVecTable = this.db.prepare(
|
|
179
|
+
"SELECT name FROM sqlite_master WHERE type='table' AND name=?"
|
|
180
|
+
);
|
|
181
|
+
if (!hasVecTable.get("turn_vectors")) {
|
|
182
|
+
this.db.exec(`
|
|
183
|
+
CREATE VIRTUAL TABLE turn_vectors USING vec0(
|
|
184
|
+
turn_id INTEGER PRIMARY KEY,
|
|
185
|
+
embedding FLOAT[384] distance_metric=cosine
|
|
186
|
+
);
|
|
187
|
+
`);
|
|
188
|
+
}
|
|
189
|
+
if (!hasVecTable.get("session_vectors")) {
|
|
190
|
+
this.db.exec(`
|
|
191
|
+
CREATE VIRTUAL TABLE session_vectors USING vec0(
|
|
192
|
+
session_id TEXT PRIMARY KEY,
|
|
193
|
+
embedding FLOAT[384] distance_metric=cosine
|
|
194
|
+
);
|
|
195
|
+
`);
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
/** Insert a turn and its embedding vector. */
|
|
199
|
+
insertTurn(room, sessionId, speaker, text, isAgent, embedding) {
|
|
200
|
+
const stmt = this.db.prepare(`
|
|
201
|
+
INSERT INTO turns (room, session_id, speaker, text, is_agent, created_at)
|
|
202
|
+
VALUES (?, ?, ?, ?, ?, ?)
|
|
203
|
+
`);
|
|
204
|
+
const info = stmt.run(room, sessionId, speaker, text, isAgent ? 1 : 0, Date.now());
|
|
205
|
+
const turnId = info.lastInsertRowid;
|
|
206
|
+
this.db.prepare(
|
|
207
|
+
"INSERT INTO turn_vectors (turn_id, embedding) VALUES (?, ?)"
|
|
208
|
+
).run(BigInt(turnId), Buffer.from(embedding.buffer));
|
|
209
|
+
return Number(turnId);
|
|
210
|
+
}
|
|
211
|
+
/** Create a new session record. */
|
|
212
|
+
insertSession(id, room) {
|
|
213
|
+
this.db.prepare(`
|
|
214
|
+
INSERT INTO sessions (id, room, started_at)
|
|
215
|
+
VALUES (?, ?, ?)
|
|
216
|
+
`).run(id, room, Date.now());
|
|
217
|
+
}
|
|
218
|
+
/** Update a session with summary and end time. */
|
|
219
|
+
updateSessionSummary(sessionId, summary, turnCount, participants, embedding) {
|
|
220
|
+
this.db.prepare(`
|
|
221
|
+
UPDATE sessions
|
|
222
|
+
SET summary = ?, ended_at = ?, turn_count = ?, participants = ?
|
|
223
|
+
WHERE id = ?
|
|
224
|
+
`).run(summary, Date.now(), turnCount, JSON.stringify(participants), sessionId);
|
|
225
|
+
this.db.prepare(
|
|
226
|
+
"INSERT INTO session_vectors (session_id, embedding) VALUES (?, ?)"
|
|
227
|
+
).run(sessionId, Buffer.from(embedding.buffer));
|
|
228
|
+
}
|
|
229
|
+
/** End a session without summary (e.g., too few turns). */
|
|
230
|
+
endSession(sessionId, turnCount, participants) {
|
|
231
|
+
this.db.prepare(`
|
|
232
|
+
UPDATE sessions
|
|
233
|
+
SET ended_at = ?, turn_count = ?, participants = ?
|
|
234
|
+
WHERE id = ?
|
|
235
|
+
`).run(Date.now(), turnCount, JSON.stringify(participants), sessionId);
|
|
236
|
+
}
|
|
237
|
+
/** KNN search turns by embedding similarity. */
|
|
238
|
+
searchTurns(room, queryEmbedding, limit) {
|
|
239
|
+
const rows = this.db.prepare(`
|
|
240
|
+
SELECT t.speaker, t.text, t.created_at, t.session_id, tv.distance
|
|
241
|
+
FROM turn_vectors tv
|
|
242
|
+
JOIN turns t ON t.id = tv.turn_id
|
|
243
|
+
WHERE t.room = ?
|
|
244
|
+
AND tv.embedding MATCH ?
|
|
245
|
+
AND k = ?
|
|
246
|
+
ORDER BY tv.distance
|
|
247
|
+
`).all(room, Buffer.from(queryEmbedding.buffer), limit * 2);
|
|
248
|
+
return rows.slice(0, limit).map((r) => ({
|
|
249
|
+
speaker: r.speaker,
|
|
250
|
+
text: r.text,
|
|
251
|
+
created_at: r.created_at,
|
|
252
|
+
session_id: r.session_id,
|
|
253
|
+
distance: r.distance
|
|
254
|
+
}));
|
|
255
|
+
}
|
|
256
|
+
/** KNN search session summaries by embedding similarity. */
|
|
257
|
+
searchSessions(room, queryEmbedding, limit) {
|
|
258
|
+
const rows = this.db.prepare(`
|
|
259
|
+
SELECT s.id as session_id, s.summary, s.started_at, sv.distance
|
|
260
|
+
FROM session_vectors sv
|
|
261
|
+
JOIN sessions s ON s.id = sv.session_id
|
|
262
|
+
WHERE s.room = ?
|
|
263
|
+
AND sv.embedding MATCH ?
|
|
264
|
+
AND k = ?
|
|
265
|
+
ORDER BY sv.distance
|
|
266
|
+
`).all(room, Buffer.from(queryEmbedding.buffer), limit * 2);
|
|
267
|
+
return rows.filter((r) => r.summary).slice(0, limit);
|
|
268
|
+
}
|
|
269
|
+
/** Get the last N turns from a specific session. */
|
|
270
|
+
getRecentTurns(room, sessionId, limit) {
|
|
271
|
+
return this.db.prepare(`
|
|
272
|
+
SELECT * FROM turns
|
|
273
|
+
WHERE room = ? AND session_id = ?
|
|
274
|
+
ORDER BY created_at DESC
|
|
275
|
+
LIMIT ?
|
|
276
|
+
`).all(room, sessionId, limit);
|
|
277
|
+
}
|
|
278
|
+
/** Get all turns for a session (for summarization). */
|
|
279
|
+
getSessionTurns(sessionId) {
|
|
280
|
+
return this.db.prepare(`
|
|
281
|
+
SELECT * FROM turns
|
|
282
|
+
WHERE session_id = ?
|
|
283
|
+
ORDER BY created_at ASC
|
|
284
|
+
`).all(sessionId);
|
|
285
|
+
}
|
|
286
|
+
/** Get total turn count for a session. */
|
|
287
|
+
getSessionTurnCount(sessionId) {
|
|
288
|
+
const row = this.db.prepare(
|
|
289
|
+
"SELECT COUNT(*) as count FROM turns WHERE session_id = ?"
|
|
290
|
+
).get(sessionId);
|
|
291
|
+
return row.count;
|
|
292
|
+
}
|
|
293
|
+
/** Close the database. */
|
|
294
|
+
close() {
|
|
295
|
+
this.db.close();
|
|
296
|
+
log2.info("Memory store closed");
|
|
297
|
+
}
|
|
298
|
+
};
|
|
299
|
+
|
|
300
|
+
// src/memory/room-memory.ts
|
|
301
|
+
var import_crypto = require("crypto");
|
|
302
|
+
var log3 = createLogger("RoomMemory");
|
|
303
|
+
var RoomMemory = class {
|
|
304
|
+
store;
|
|
305
|
+
embedder;
|
|
306
|
+
room;
|
|
307
|
+
sessionId = null;
|
|
308
|
+
participants = /* @__PURE__ */ new Set();
|
|
309
|
+
pendingTurns = [];
|
|
310
|
+
flushTimer = null;
|
|
311
|
+
flushIntervalMs;
|
|
312
|
+
flushing = false;
|
|
313
|
+
constructor(config) {
|
|
314
|
+
this.store = new MemoryStore(config.dbPath);
|
|
315
|
+
this.embedder = new Embedder();
|
|
316
|
+
this.room = config.room;
|
|
317
|
+
this.flushIntervalMs = config.flushIntervalMs ?? 5e3;
|
|
318
|
+
}
|
|
319
|
+
/** Get the embedder instance (for reuse in other components). */
|
|
320
|
+
getEmbedder() {
|
|
321
|
+
return this.embedder;
|
|
322
|
+
}
|
|
323
|
+
/** Initialize embedder (loads model). Call once at startup. */
|
|
324
|
+
async init() {
|
|
325
|
+
await this.embedder.init();
|
|
326
|
+
}
|
|
327
|
+
/** Start a new session for this room. */
|
|
328
|
+
startSession() {
|
|
329
|
+
this.sessionId = (0, import_crypto.randomUUID)();
|
|
330
|
+
this.participants.clear();
|
|
331
|
+
this.store.insertSession(this.sessionId, this.room);
|
|
332
|
+
this.flushTimer = setInterval(() => {
|
|
333
|
+
this.flushPending().catch((err) => {
|
|
334
|
+
log3.error("Error flushing pending turns:", err);
|
|
335
|
+
});
|
|
336
|
+
}, this.flushIntervalMs);
|
|
337
|
+
log3.info(`Session started: ${this.sessionId}`);
|
|
338
|
+
return this.sessionId;
|
|
339
|
+
}
|
|
340
|
+
/** Track a participant joining. */
|
|
341
|
+
addParticipant(identity) {
|
|
342
|
+
this.participants.add(identity);
|
|
343
|
+
}
|
|
344
|
+
/**
|
|
345
|
+
* Store a turn to memory. Non-blocking — queues for batch embedding.
|
|
346
|
+
* Call this for EVERY final transcription, even if agent doesn't respond.
|
|
347
|
+
*/
|
|
348
|
+
storeTurn(speaker, text, isAgent) {
|
|
349
|
+
if (!this.sessionId) {
|
|
350
|
+
log3.warn("storeTurn called without active session");
|
|
351
|
+
return;
|
|
352
|
+
}
|
|
353
|
+
this.pendingTurns.push({ speaker, text, isAgent });
|
|
354
|
+
if (this.pendingTurns.length >= 5) {
|
|
355
|
+
this.flushPending().catch((err) => {
|
|
356
|
+
log3.error("Error flushing pending turns:", err);
|
|
357
|
+
});
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
/** Flush pending turns: embed and insert into database. */
|
|
361
|
+
async flushPending() {
|
|
362
|
+
if (this.flushing || this.pendingTurns.length === 0 || !this.sessionId) return;
|
|
363
|
+
this.flushing = true;
|
|
364
|
+
const batch = this.pendingTurns.splice(0);
|
|
365
|
+
const texts = batch.map((t) => `[${t.speaker}]: ${t.text}`);
|
|
366
|
+
try {
|
|
367
|
+
const embeddings = await this.embedder.embedBatch(texts);
|
|
368
|
+
for (let i = 0; i < batch.length; i++) {
|
|
369
|
+
const turn = batch[i];
|
|
370
|
+
this.store.insertTurn(
|
|
371
|
+
this.room,
|
|
372
|
+
this.sessionId,
|
|
373
|
+
turn.speaker,
|
|
374
|
+
turn.text,
|
|
375
|
+
turn.isAgent,
|
|
376
|
+
embeddings[i]
|
|
377
|
+
);
|
|
378
|
+
}
|
|
379
|
+
log3.debug(`Flushed ${batch.length} turns to memory`);
|
|
380
|
+
} catch (err) {
|
|
381
|
+
log3.error("Error embedding/storing turns:", err);
|
|
382
|
+
this.pendingTurns.unshift(...batch);
|
|
383
|
+
} finally {
|
|
384
|
+
this.flushing = false;
|
|
385
|
+
}
|
|
386
|
+
}
|
|
387
|
+
/**
|
|
388
|
+
* Search memory for context relevant to a query.
|
|
389
|
+
* Returns formatted string ready to inject into LLM system prompt.
|
|
390
|
+
*/
|
|
391
|
+
async searchRelevant(query, turnLimit = 5, sessionLimit = 2) {
|
|
392
|
+
const queryEmbedding = await this.embedder.embed(query);
|
|
393
|
+
const turns = this.store.searchTurns(this.room, queryEmbedding, turnLimit);
|
|
394
|
+
const sessions = this.store.searchSessions(this.room, queryEmbedding, sessionLimit);
|
|
395
|
+
if (turns.length === 0 && sessions.length === 0) {
|
|
396
|
+
return "";
|
|
397
|
+
}
|
|
398
|
+
const parts = [];
|
|
399
|
+
if (sessions.length > 0) {
|
|
400
|
+
parts.push("Past session summaries:");
|
|
401
|
+
for (const s of sessions) {
|
|
402
|
+
const date = new Date(s.started_at).toLocaleDateString();
|
|
403
|
+
parts.push(` [${date}]: ${s.summary}`);
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
if (turns.length > 0) {
|
|
407
|
+
parts.push("Relevant past turns:");
|
|
408
|
+
for (const t of turns) {
|
|
409
|
+
const date = new Date(t.created_at).toLocaleDateString();
|
|
410
|
+
const time = new Date(t.created_at).toLocaleTimeString([], { hour: "2-digit", minute: "2-digit" });
|
|
411
|
+
parts.push(` [${date} ${time}, ${t.speaker}]: ${t.text}`);
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
return parts.join("\n");
|
|
415
|
+
}
|
|
416
|
+
/**
|
|
417
|
+
* End the current session. Generates an LLM summary and stores it.
|
|
418
|
+
*/
|
|
419
|
+
async endSession(llm) {
|
|
420
|
+
if (!this.sessionId) return;
|
|
421
|
+
await this.flushPending();
|
|
422
|
+
if (this.flushTimer) {
|
|
423
|
+
clearInterval(this.flushTimer);
|
|
424
|
+
this.flushTimer = null;
|
|
425
|
+
}
|
|
426
|
+
const turnCount = this.store.getSessionTurnCount(this.sessionId);
|
|
427
|
+
const participantList = Array.from(this.participants);
|
|
428
|
+
if (turnCount < 3) {
|
|
429
|
+
this.store.endSession(this.sessionId, turnCount, participantList);
|
|
430
|
+
log3.info(`Session ended (${turnCount} turns, no summary)`);
|
|
431
|
+
this.sessionId = null;
|
|
432
|
+
return;
|
|
433
|
+
}
|
|
434
|
+
try {
|
|
435
|
+
const turns = this.store.getSessionTurns(this.sessionId);
|
|
436
|
+
const transcript = turns.map((t) => `[${t.speaker}]: ${t.text}`).join("\n");
|
|
437
|
+
const messages = [
|
|
438
|
+
{
|
|
439
|
+
role: "system",
|
|
440
|
+
content: "Summarize this tutoring session concisely. Include: topics covered, phrases practiced, mistakes the student made, what they struggled with, and what they did well. Be factual and brief."
|
|
441
|
+
},
|
|
442
|
+
{ role: "user", content: transcript }
|
|
443
|
+
];
|
|
444
|
+
let summary = "";
|
|
445
|
+
for await (const chunk of llm.chat(messages)) {
|
|
446
|
+
if (chunk.type === "token" && chunk.token) {
|
|
447
|
+
summary += chunk.token;
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
if (summary.trim()) {
|
|
451
|
+
const embedding = await this.embedder.embed(summary.trim());
|
|
452
|
+
this.store.updateSessionSummary(
|
|
453
|
+
this.sessionId,
|
|
454
|
+
summary.trim(),
|
|
455
|
+
turnCount,
|
|
456
|
+
participantList,
|
|
457
|
+
embedding
|
|
458
|
+
);
|
|
459
|
+
log3.info(`Session ended with summary (${turnCount} turns, ${participantList.length} participants)`);
|
|
460
|
+
} else {
|
|
461
|
+
this.store.endSession(this.sessionId, turnCount, participantList);
|
|
462
|
+
log3.info(`Session ended (${turnCount} turns, summary was empty)`);
|
|
463
|
+
}
|
|
464
|
+
} catch (err) {
|
|
465
|
+
log3.error("Error generating session summary:", err);
|
|
466
|
+
this.store.endSession(this.sessionId, turnCount, participantList);
|
|
467
|
+
}
|
|
468
|
+
this.sessionId = null;
|
|
469
|
+
}
|
|
470
|
+
/** Close the memory store. Flush pending turns first. */
|
|
471
|
+
async close() {
|
|
472
|
+
if (this.flushTimer) {
|
|
473
|
+
clearInterval(this.flushTimer);
|
|
474
|
+
this.flushTimer = null;
|
|
475
|
+
}
|
|
476
|
+
await this.flushPending();
|
|
477
|
+
this.store.close();
|
|
478
|
+
}
|
|
479
|
+
};
|
|
480
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
481
|
+
0 && (module.exports = {
|
|
482
|
+
Embedder,
|
|
483
|
+
MemoryStore,
|
|
484
|
+
RoomMemory
|
|
485
|
+
});
|
|
486
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/memory/index.ts","../../src/utils/logger.ts","../../src/memory/embedder.ts","../../src/memory/memory-store.ts","../../src/memory/room-memory.ts"],"sourcesContent":["export { Embedder } from './embedder';\nexport { MemoryStore } from './memory-store';\nexport type { TurnRow, SessionRow, SearchResult, SessionSearchResult } from './memory-store';\nexport { RoomMemory } from './room-memory';\nexport type { RoomMemoryConfig } from './room-memory';\n","export type LogLevel = 'debug' | 'info' | 'warn' | 'error' | 'silent';\n\nconst LEVELS: Record<LogLevel, number> = {\n debug: 0,\n info: 1,\n warn: 2,\n error: 3,\n silent: 4,\n};\n\n/** Default to 'debug' if DEBUG env var matches our namespace */\nfunction detectLevel(): LogLevel {\n const debug = typeof process !== 'undefined' && process.env?.DEBUG;\n if (debug && (debug === '*' || debug.includes('@dtelecom/agents'))) {\n return 'debug';\n }\n return 'info';\n}\n\nlet globalLevel: LogLevel = detectLevel();\n\nexport function setLogLevel(level: LogLevel): void {\n globalLevel = level;\n}\n\nexport function getLogLevel(): LogLevel {\n return globalLevel;\n}\n\nexport interface Logger {\n debug(...args: unknown[]): void;\n info(...args: unknown[]): void;\n warn(...args: unknown[]): void;\n error(...args: unknown[]): void;\n}\n\nfunction timestamp(): string {\n const d = new Date();\n const h = String(d.getHours()).padStart(2, '0');\n const m = String(d.getMinutes()).padStart(2, '0');\n const s = String(d.getSeconds()).padStart(2, '0');\n const ms = String(d.getMilliseconds()).padStart(3, '0');\n return `${h}:${m}:${s}.${ms}`;\n}\n\nexport function createLogger(tag: string): Logger {\n const prefix = `[@dtelecom/agents:${tag}]`;\n return {\n debug(...args: unknown[]) {\n if (LEVELS[globalLevel] <= LEVELS.debug) console.debug(timestamp(), prefix, ...args);\n },\n info(...args: unknown[]) {\n if (LEVELS[globalLevel] <= LEVELS.info) console.info(timestamp(), prefix, ...args);\n },\n warn(...args: unknown[]) {\n if (LEVELS[globalLevel] <= LEVELS.warn) console.warn(timestamp(), prefix, ...args);\n },\n error(...args: unknown[]) {\n if (LEVELS[globalLevel] <= LEVELS.error) console.error(timestamp(), prefix, ...args);\n },\n };\n}\n","/**\n * Embedder — local text embedding via @huggingface/transformers.\n *\n * Uses Xenova/all-MiniLM-L6-v2 (384 dimensions, ~22MB model).\n * Runs entirely in-process — no API calls, no cost.\n */\n\nimport { createLogger } from '../utils/logger';\n\nconst log = createLogger('Embedder');\n\nconst MODEL_NAME = 'Xenova/all-MiniLM-L6-v2';\nconst EMBEDDING_DIM = 384;\n\ntype FeatureExtractionPipeline = (\n text: string | string[],\n options?: { pooling?: string; normalize?: boolean },\n) => Promise<{ data: Float32Array }>;\n\nexport class Embedder {\n private pipeline: FeatureExtractionPipeline | null = null;\n private initPromise: Promise<void> | null = null;\n\n get dimensions(): number {\n return EMBEDDING_DIM;\n }\n\n /** Load the embedding model. Call once at startup. */\n async init(): Promise<void> {\n if (this.pipeline) return;\n if (this.initPromise) return this.initPromise;\n\n this.initPromise = this.loadModel();\n return this.initPromise;\n }\n\n private async loadModel(): Promise<void> {\n const start = performance.now();\n log.info(`Loading embedding model \"${MODEL_NAME}\"...`);\n\n const { pipeline } = await import('@huggingface/transformers');\n this.pipeline = (await pipeline('feature-extraction', MODEL_NAME)) as unknown as FeatureExtractionPipeline;\n\n log.info(`Embedding model loaded in ${(performance.now() - start).toFixed(0)}ms`);\n }\n\n /** Embed a single text. Returns Float32Array of length 384. */\n async embed(text: string): Promise<Float32Array> {\n await this.init();\n\n const result = await this.pipeline!(text, {\n pooling: 'mean',\n normalize: true,\n });\n\n return new Float32Array(result.data);\n }\n\n /** Cosine similarity between two normalized vectors. Returns value in [-1, 1]. */\n static cosineSimilarity(a: Float32Array, b: Float32Array): number {\n let dot = 0;\n for (let i = 0; i < a.length; i++) {\n dot += a[i] * b[i];\n }\n return dot;\n }\n\n /** Embed multiple texts in one call (more efficient than calling embed() in a loop). */\n async embedBatch(texts: string[]): Promise<Float32Array[]> {\n if (texts.length === 0) return [];\n await this.init();\n\n const results: Float32Array[] = [];\n // Process one at a time to avoid memory issues with large batches\n for (const text of texts) {\n const result = await this.pipeline!(text, {\n pooling: 'mean',\n normalize: true,\n });\n results.push(new Float32Array(result.data));\n }\n\n return results;\n }\n}\n","/**\n * MemoryStore — SQLite + sqlite-vec database layer for room memory.\n *\n * Single .db file stores:\n * - turns: every spoken turn (full transcript)\n * - sessions: meeting metadata + LLM-generated summaries\n * - turn_vectors: embedding index for semantic turn search\n * - session_vectors: embedding index for session summary search\n */\n\nimport Database from 'better-sqlite3';\nimport * as sqliteVec from 'sqlite-vec';\nimport { createLogger } from '../utils/logger';\n\nconst log = createLogger('MemoryStore');\n\nexport interface TurnRow {\n id: number;\n room: string;\n session_id: string;\n speaker: string;\n text: string;\n is_agent: number;\n created_at: number;\n}\n\nexport interface SessionRow {\n id: string;\n room: string;\n started_at: number;\n ended_at: number | null;\n participants: string | null;\n summary: string | null;\n turn_count: number;\n}\n\nexport interface SearchResult {\n speaker: string;\n text: string;\n created_at: number;\n session_id: string;\n distance: number;\n}\n\nexport interface SessionSearchResult {\n session_id: string;\n summary: string;\n started_at: number;\n distance: number;\n}\n\nexport class MemoryStore {\n private db: Database.Database;\n\n constructor(dbPath: string) {\n this.db = new Database(dbPath);\n this.db.pragma('journal_mode = WAL');\n this.db.pragma('synchronous = NORMAL');\n\n // Load sqlite-vec extension\n sqliteVec.load(this.db);\n\n this.createTables();\n log.info(`Memory store opened: ${dbPath}`);\n }\n\n private createTables(): void {\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS turns (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n room TEXT NOT NULL,\n session_id TEXT NOT NULL,\n speaker TEXT NOT NULL,\n text TEXT NOT NULL,\n is_agent BOOLEAN DEFAULT 0,\n created_at INTEGER NOT NULL\n );\n\n CREATE TABLE IF NOT EXISTS sessions (\n id TEXT PRIMARY KEY,\n room TEXT NOT NULL,\n started_at INTEGER NOT NULL,\n ended_at INTEGER,\n participants TEXT,\n summary TEXT,\n turn_count INTEGER DEFAULT 0\n );\n\n CREATE INDEX IF NOT EXISTS idx_turns_room_session ON turns(room, session_id);\n CREATE INDEX IF NOT EXISTS idx_turns_room_time ON turns(room, created_at);\n CREATE INDEX IF NOT EXISTS idx_sessions_room ON sessions(room);\n `);\n\n // Vector tables — sqlite-vec virtual tables\n // Check if they exist first (CREATE VIRTUAL TABLE doesn't support IF NOT EXISTS)\n const hasVecTable = this.db.prepare(\n \"SELECT name FROM sqlite_master WHERE type='table' AND name=?\",\n );\n\n if (!hasVecTable.get('turn_vectors')) {\n this.db.exec(`\n CREATE VIRTUAL TABLE turn_vectors USING vec0(\n turn_id INTEGER PRIMARY KEY,\n embedding FLOAT[384] distance_metric=cosine\n );\n `);\n }\n\n if (!hasVecTable.get('session_vectors')) {\n this.db.exec(`\n CREATE VIRTUAL TABLE session_vectors USING vec0(\n session_id TEXT PRIMARY KEY,\n embedding FLOAT[384] distance_metric=cosine\n );\n `);\n }\n }\n\n /** Insert a turn and its embedding vector. */\n insertTurn(\n room: string,\n sessionId: string,\n speaker: string,\n text: string,\n isAgent: boolean,\n embedding: Float32Array,\n ): number {\n const stmt = this.db.prepare(`\n INSERT INTO turns (room, session_id, speaker, text, is_agent, created_at)\n VALUES (?, ?, ?, ?, ?, ?)\n `);\n const info = stmt.run(room, sessionId, speaker, text, isAgent ? 1 : 0, Date.now());\n const turnId = info.lastInsertRowid;\n\n // Insert embedding vector — sqlite-vec requires BigInt for integer PKs\n this.db.prepare(\n 'INSERT INTO turn_vectors (turn_id, embedding) VALUES (?, ?)',\n ).run(BigInt(turnId), Buffer.from(embedding.buffer));\n\n return Number(turnId);\n }\n\n /** Create a new session record. */\n insertSession(id: string, room: string): void {\n this.db.prepare(`\n INSERT INTO sessions (id, room, started_at)\n VALUES (?, ?, ?)\n `).run(id, room, Date.now());\n }\n\n /** Update a session with summary and end time. */\n updateSessionSummary(\n sessionId: string,\n summary: string,\n turnCount: number,\n participants: string[],\n embedding: Float32Array,\n ): void {\n this.db.prepare(`\n UPDATE sessions\n SET summary = ?, ended_at = ?, turn_count = ?, participants = ?\n WHERE id = ?\n `).run(summary, Date.now(), turnCount, JSON.stringify(participants), sessionId);\n\n // Insert summary embedding\n this.db.prepare(\n 'INSERT INTO session_vectors (session_id, embedding) VALUES (?, ?)',\n ).run(sessionId, Buffer.from(embedding.buffer));\n }\n\n /** End a session without summary (e.g., too few turns). */\n endSession(sessionId: string, turnCount: number, participants: string[]): void {\n this.db.prepare(`\n UPDATE sessions\n SET ended_at = ?, turn_count = ?, participants = ?\n WHERE id = ?\n `).run(Date.now(), turnCount, JSON.stringify(participants), sessionId);\n }\n\n /** KNN search turns by embedding similarity. */\n searchTurns(room: string, queryEmbedding: Float32Array, limit: number): SearchResult[] {\n const rows = this.db.prepare(`\n SELECT t.speaker, t.text, t.created_at, t.session_id, tv.distance\n FROM turn_vectors tv\n JOIN turns t ON t.id = tv.turn_id\n WHERE t.room = ?\n AND tv.embedding MATCH ?\n AND k = ?\n ORDER BY tv.distance\n `).all(room, Buffer.from(queryEmbedding.buffer), limit * 2) as (TurnRow & { distance: number })[];\n\n // sqlite-vec returns k results from the vector index, then we filter by room\n return rows.slice(0, limit).map((r) => ({\n speaker: r.speaker,\n text: r.text,\n created_at: r.created_at,\n session_id: r.session_id,\n distance: r.distance,\n }));\n }\n\n /** KNN search session summaries by embedding similarity. */\n searchSessions(room: string, queryEmbedding: Float32Array, limit: number): SessionSearchResult[] {\n const rows = this.db.prepare(`\n SELECT s.id as session_id, s.summary, s.started_at, sv.distance\n FROM session_vectors sv\n JOIN sessions s ON s.id = sv.session_id\n WHERE s.room = ?\n AND sv.embedding MATCH ?\n AND k = ?\n ORDER BY sv.distance\n `).all(room, Buffer.from(queryEmbedding.buffer), limit * 2) as SessionSearchResult[];\n\n return rows\n .filter((r) => r.summary)\n .slice(0, limit);\n }\n\n /** Get the last N turns from a specific session. */\n getRecentTurns(room: string, sessionId: string, limit: number): TurnRow[] {\n return this.db.prepare(`\n SELECT * FROM turns\n WHERE room = ? AND session_id = ?\n ORDER BY created_at DESC\n LIMIT ?\n `).all(room, sessionId, limit) as TurnRow[];\n }\n\n /** Get all turns for a session (for summarization). */\n getSessionTurns(sessionId: string): TurnRow[] {\n return this.db.prepare(`\n SELECT * FROM turns\n WHERE session_id = ?\n ORDER BY created_at ASC\n `).all(sessionId) as TurnRow[];\n }\n\n /** Get total turn count for a session. */\n getSessionTurnCount(sessionId: string): number {\n const row = this.db.prepare(\n 'SELECT COUNT(*) as count FROM turns WHERE session_id = ?',\n ).get(sessionId) as { count: number };\n return row.count;\n }\n\n /** Close the database. */\n close(): void {\n this.db.close();\n log.info('Memory store closed');\n }\n}\n","/**\n * RoomMemory — high-level persistent memory for a room.\n *\n * Stores all conversation turns, provides semantic search,\n * and generates session summaries on session end.\n *\n * Uses SQLite + sqlite-vec for storage and local embeddings\n * via @huggingface/transformers. Everything runs in-process,\n * no external services needed.\n */\n\nimport { randomUUID } from 'crypto';\nimport { MemoryStore, type TurnRow, type SearchResult, type SessionSearchResult } from './memory-store';\nimport { Embedder } from './embedder';\nimport type { LLMPlugin, Message } from '../core/types';\nimport { createLogger } from '../utils/logger';\n\nconst log = createLogger('RoomMemory');\n\n/** Pending turn waiting to be embedded and stored. */\ninterface PendingTurn {\n speaker: string;\n text: string;\n isAgent: boolean;\n}\n\nexport interface RoomMemoryConfig {\n /** Path to SQLite database file */\n dbPath: string;\n /** Room name (scopes all data) */\n room: string;\n /** Flush pending turns every N ms (default: 5000) */\n flushIntervalMs?: number;\n}\n\nexport class RoomMemory {\n private readonly store: MemoryStore;\n private readonly embedder: Embedder;\n private readonly room: string;\n private sessionId: string | null = null;\n private participants = new Set<string>();\n private pendingTurns: PendingTurn[] = [];\n private flushTimer: ReturnType<typeof setInterval> | null = null;\n private readonly flushIntervalMs: number;\n private flushing = false;\n\n constructor(config: RoomMemoryConfig) {\n this.store = new MemoryStore(config.dbPath);\n this.embedder = new Embedder();\n this.room = config.room;\n this.flushIntervalMs = config.flushIntervalMs ?? 5000;\n }\n\n /** Get the embedder instance (for reuse in other components). */\n getEmbedder(): Embedder {\n return this.embedder;\n }\n\n /** Initialize embedder (loads model). Call once at startup. */\n async init(): Promise<void> {\n await this.embedder.init();\n }\n\n /** Start a new session for this room. */\n startSession(): string {\n this.sessionId = randomUUID();\n this.participants.clear();\n this.store.insertSession(this.sessionId, this.room);\n\n // Start periodic flush of pending turns\n this.flushTimer = setInterval(() => {\n this.flushPending().catch((err) => {\n log.error('Error flushing pending turns:', err);\n });\n }, this.flushIntervalMs);\n\n log.info(`Session started: ${this.sessionId}`);\n return this.sessionId;\n }\n\n /** Track a participant joining. */\n addParticipant(identity: string): void {\n this.participants.add(identity);\n }\n\n /**\n * Store a turn to memory. Non-blocking — queues for batch embedding.\n * Call this for EVERY final transcription, even if agent doesn't respond.\n */\n storeTurn(speaker: string, text: string, isAgent: boolean): void {\n if (!this.sessionId) {\n log.warn('storeTurn called without active session');\n return;\n }\n\n this.pendingTurns.push({ speaker, text, isAgent });\n\n // Flush immediately if we have 5+ pending turns\n if (this.pendingTurns.length >= 5) {\n this.flushPending().catch((err) => {\n log.error('Error flushing pending turns:', err);\n });\n }\n }\n\n /** Flush pending turns: embed and insert into database. */\n private async flushPending(): Promise<void> {\n if (this.flushing || this.pendingTurns.length === 0 || !this.sessionId) return;\n this.flushing = true;\n\n const batch = this.pendingTurns.splice(0);\n const texts = batch.map((t) => `[${t.speaker}]: ${t.text}`);\n\n try {\n const embeddings = await this.embedder.embedBatch(texts);\n\n for (let i = 0; i < batch.length; i++) {\n const turn = batch[i];\n this.store.insertTurn(\n this.room,\n this.sessionId,\n turn.speaker,\n turn.text,\n turn.isAgent,\n embeddings[i],\n );\n }\n\n log.debug(`Flushed ${batch.length} turns to memory`);\n } catch (err) {\n log.error('Error embedding/storing turns:', err);\n // Put turns back for retry\n this.pendingTurns.unshift(...batch);\n } finally {\n this.flushing = false;\n }\n }\n\n /**\n * Search memory for context relevant to a query.\n * Returns formatted string ready to inject into LLM system prompt.\n */\n async searchRelevant(query: string, turnLimit = 5, sessionLimit = 2): Promise<string> {\n const queryEmbedding = await this.embedder.embed(query);\n\n const turns = this.store.searchTurns(this.room, queryEmbedding, turnLimit);\n const sessions = this.store.searchSessions(this.room, queryEmbedding, sessionLimit);\n\n if (turns.length === 0 && sessions.length === 0) {\n return '';\n }\n\n const parts: string[] = [];\n\n if (sessions.length > 0) {\n parts.push('Past session summaries:');\n for (const s of sessions) {\n const date = new Date(s.started_at).toLocaleDateString();\n parts.push(` [${date}]: ${s.summary}`);\n }\n }\n\n if (turns.length > 0) {\n parts.push('Relevant past turns:');\n for (const t of turns) {\n const date = new Date(t.created_at).toLocaleDateString();\n const time = new Date(t.created_at).toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' });\n parts.push(` [${date} ${time}, ${t.speaker}]: ${t.text}`);\n }\n }\n\n return parts.join('\\n');\n }\n\n /**\n * End the current session. Generates an LLM summary and stores it.\n */\n async endSession(llm: LLMPlugin): Promise<void> {\n if (!this.sessionId) return;\n\n // Flush any remaining pending turns\n await this.flushPending();\n\n // Stop flush timer\n if (this.flushTimer) {\n clearInterval(this.flushTimer);\n this.flushTimer = null;\n }\n\n const turnCount = this.store.getSessionTurnCount(this.sessionId);\n const participantList = Array.from(this.participants);\n\n if (turnCount < 3) {\n // Too few turns for meaningful summary\n this.store.endSession(this.sessionId, turnCount, participantList);\n log.info(`Session ended (${turnCount} turns, no summary)`);\n this.sessionId = null;\n return;\n }\n\n // Generate summary\n try {\n const turns = this.store.getSessionTurns(this.sessionId);\n const transcript = turns\n .map((t) => `[${t.speaker}]: ${t.text}`)\n .join('\\n');\n\n const messages: Message[] = [\n {\n role: 'system',\n content: 'Summarize this tutoring session concisely. Include: topics covered, phrases practiced, mistakes the student made, what they struggled with, and what they did well. Be factual and brief.',\n },\n { role: 'user', content: transcript },\n ];\n\n let summary = '';\n for await (const chunk of llm.chat(messages)) {\n if (chunk.type === 'token' && chunk.token) {\n summary += chunk.token;\n }\n }\n\n if (summary.trim()) {\n const embedding = await this.embedder.embed(summary.trim());\n this.store.updateSessionSummary(\n this.sessionId,\n summary.trim(),\n turnCount,\n participantList,\n embedding,\n );\n log.info(`Session ended with summary (${turnCount} turns, ${participantList.length} participants)`);\n } else {\n this.store.endSession(this.sessionId, turnCount, participantList);\n log.info(`Session ended (${turnCount} turns, summary was empty)`);\n }\n } catch (err) {\n log.error('Error generating session summary:', err);\n this.store.endSession(this.sessionId, turnCount, participantList);\n }\n\n this.sessionId = null;\n }\n\n /** Close the memory store. Flush pending turns first. */\n async close(): Promise<void> {\n if (this.flushTimer) {\n clearInterval(this.flushTimer);\n this.flushTimer = null;\n }\n\n await this.flushPending();\n this.store.close();\n }\n}\n"],"mappings":"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","names":["log","Database","log"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { h as STTPlugin, i as STTStreamOptions, S as STTStream, L as LLMPlugin, M as Message, e as LLMChunk, j as TTSPlugin } from '../types-EvtHMokR.mjs';
|
|
2
2
|
import '@dtelecom/server-sdk-node';
|
|
3
3
|
|
|
4
4
|
/**
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { h as STTPlugin, i as STTStreamOptions, S as STTStream, L as LLMPlugin, M as Message, e as LLMChunk, j as TTSPlugin } from '../types-EvtHMokR.js';
|
|
2
2
|
import '@dtelecom/server-sdk-node';
|
|
3
3
|
|
|
4
4
|
/**
|
|
@@ -256,4 +256,4 @@ interface PipelineEvents {
|
|
|
256
256
|
error: (error: Error) => void;
|
|
257
257
|
}
|
|
258
258
|
|
|
259
|
-
export { type AgentConfig as A, type DataMessageHandler as D, Embedder as E, type LLMPlugin as L, type Message as M, type PipelineOptions as P, type RespondMode as R, type STTStream as S, type TranscriptionResult as T, type AgentStartOptions as a, type AgentState as b, type AgentEvents as c, AudioOutput as d, type LLMChunk as e, type MemoryConfig as f, type PipelineEvents as g,
|
|
259
|
+
export { type AgentConfig as A, type DataMessageHandler as D, Embedder as E, type LLMPlugin as L, type Message as M, type PipelineOptions as P, type RespondMode as R, type STTStream as S, type TranscriptionResult as T, type AgentStartOptions as a, type AgentState as b, type AgentEvents as c, AudioOutput as d, type LLMChunk as e, type MemoryConfig as f, type PipelineEvents as g, type STTPlugin as h, type STTStreamOptions as i, type TTSPlugin as j, RoomMemory as k, type RoomMemoryConfig as l };
|
|
@@ -256,4 +256,4 @@ interface PipelineEvents {
|
|
|
256
256
|
error: (error: Error) => void;
|
|
257
257
|
}
|
|
258
258
|
|
|
259
|
-
export { type AgentConfig as A, type DataMessageHandler as D, Embedder as E, type LLMPlugin as L, type Message as M, type PipelineOptions as P, type RespondMode as R, type STTStream as S, type TranscriptionResult as T, type AgentStartOptions as a, type AgentState as b, type AgentEvents as c, AudioOutput as d, type LLMChunk as e, type MemoryConfig as f, type PipelineEvents as g,
|
|
259
|
+
export { type AgentConfig as A, type DataMessageHandler as D, Embedder as E, type LLMPlugin as L, type Message as M, type PipelineOptions as P, type RespondMode as R, type STTStream as S, type TranscriptionResult as T, type AgentStartOptions as a, type AgentState as b, type AgentEvents as c, AudioOutput as d, type LLMChunk as e, type MemoryConfig as f, type PipelineEvents as g, type STTPlugin as h, type STTStreamOptions as i, type TTSPlugin as j, RoomMemory as k, type RoomMemoryConfig as l };
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dtelecom/agents-js",
|
|
3
|
-
"version": "0.1.
|
|
3
|
+
"version": "0.1.2",
|
|
4
4
|
"description": "AI voice agent framework for dTelecom rooms",
|
|
5
5
|
"main": "./dist/index.js",
|
|
6
6
|
"module": "./dist/index.mjs",
|
|
@@ -15,6 +15,11 @@
|
|
|
15
15
|
"types": "./dist/providers/index.d.ts",
|
|
16
16
|
"import": "./dist/providers/index.mjs",
|
|
17
17
|
"require": "./dist/providers/index.js"
|
|
18
|
+
},
|
|
19
|
+
"./memory": {
|
|
20
|
+
"types": "./dist/memory/index.d.ts",
|
|
21
|
+
"import": "./dist/memory/index.mjs",
|
|
22
|
+
"require": "./dist/memory/index.js"
|
|
18
23
|
}
|
|
19
24
|
},
|
|
20
25
|
"files": [
|