@narrative-os/engine 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/agents/canonValidator.d.ts +9 -0
- package/dist/agents/canonValidator.js +51 -0
- package/dist/agents/chapterPlanner.d.ts +50 -0
- package/dist/agents/chapterPlanner.js +250 -0
- package/dist/agents/completeness.d.ts +7 -0
- package/dist/agents/completeness.js +51 -0
- package/dist/agents/memoryExtractor.d.ts +12 -0
- package/dist/agents/memoryExtractor.js +82 -0
- package/dist/agents/stateUpdater.d.ts +30 -0
- package/dist/agents/stateUpdater.js +150 -0
- package/dist/agents/storyDirector.d.ts +40 -0
- package/dist/agents/storyDirector.js +213 -0
- package/dist/agents/summarizer.d.ts +8 -0
- package/dist/agents/summarizer.js +56 -0
- package/dist/agents/tensionController.d.ts +68 -0
- package/dist/agents/tensionController.js +197 -0
- package/dist/agents/writer.d.ts +12 -0
- package/dist/agents/writer.js +148 -0
- package/dist/constraints/constraintGraph.d.ts +117 -0
- package/dist/constraints/constraintGraph.js +381 -0
- package/dist/constraints/validator.d.ts +58 -0
- package/dist/constraints/validator.js +236 -0
- package/dist/index.d.ts +25 -0
- package/dist/index.js +115 -0
- package/dist/llm/client.d.ts +14 -0
- package/dist/llm/client.js +108 -0
- package/dist/memory/canonStore.d.ts +20 -0
- package/dist/memory/canonStore.js +110 -0
- package/dist/memory/memoryRetriever.d.ts +28 -0
- package/dist/memory/memoryRetriever.js +126 -0
- package/dist/memory/stateUpdater.d.ts +49 -0
- package/dist/memory/stateUpdater.js +315 -0
- package/dist/memory/vectorStore.d.ts +41 -0
- package/dist/memory/vectorStore.js +166 -0
- package/dist/pipeline/generateChapter.d.ts +17 -0
- package/dist/pipeline/generateChapter.js +75 -0
- package/dist/story/bible.d.ts +4 -0
- package/dist/story/bible.js +53 -0
- package/dist/story/state.d.ts +3 -0
- package/dist/story/state.js +27 -0
- package/dist/story/structuredState.d.ts +39 -0
- package/dist/story/structuredState.js +159 -0
- package/dist/test/canon.test.d.ts +1 -0
- package/dist/test/canon.test.js +104 -0
- package/dist/test/chapter-planner.test.d.ts +1 -0
- package/dist/test/chapter-planner.test.js +171 -0
- package/dist/test/constraints.test.d.ts +1 -0
- package/dist/test/constraints.test.js +210 -0
- package/dist/test/simple.test.d.ts +1 -0
- package/dist/test/simple.test.js +51 -0
- package/dist/test/state-updater.test.d.ts +1 -0
- package/dist/test/state-updater.test.js +200 -0
- package/dist/test/story-director.test.d.ts +1 -0
- package/dist/test/story-director.test.js +142 -0
- package/dist/test/structured-state.test.d.ts +1 -0
- package/dist/test/structured-state.test.js +144 -0
- package/dist/test/tension-controller.test.d.ts +1 -0
- package/dist/test/tension-controller.test.js +116 -0
- package/dist/test/vector-memory.test.d.ts +1 -0
- package/dist/test/vector-memory.test.js +153 -0
- package/dist/test/world-simulation.test.d.ts +1 -0
- package/dist/test/world-simulation.test.js +152 -0
- package/dist/types/index.d.ts +79 -0
- package/dist/types/index.js +3 -0
- package/dist/world/characterAgent.d.ts +73 -0
- package/dist/world/characterAgent.js +232 -0
- package/dist/world/eventResolver.d.ts +52 -0
- package/dist/world/eventResolver.js +205 -0
- package/dist/world/worldState.d.ts +93 -0
- package/dist/world/worldState.js +258 -0
- package/package.json +43 -0
|
@@ -0,0 +1,315 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.stateUpdaterPipeline = exports.StateUpdaterPipeline = void 0;
|
|
4
|
+
const client_js_1 = require("../llm/client.js");
|
|
5
|
+
const STATE_EXTRACTION_PROMPT = `You are a narrative state extractor. Analyze this chapter and extract all state changes.
|
|
6
|
+
|
|
7
|
+
## Story Bible
|
|
8
|
+
|
|
9
|
+
**Title:** {{title}}
|
|
10
|
+
**Genre:** {{genre}}
|
|
11
|
+
|
|
12
|
+
## Chapter {{chapterNumber}}: {{chapterTitle}}
|
|
13
|
+
|
|
14
|
+
{{chapterContent}}
|
|
15
|
+
|
|
16
|
+
## Current Character States
|
|
17
|
+
|
|
18
|
+
{{characters}}
|
|
19
|
+
|
|
20
|
+
## Current Plot Threads
|
|
21
|
+
|
|
22
|
+
{{plotThreads}}
|
|
23
|
+
|
|
24
|
+
## Extraction Task
|
|
25
|
+
|
|
26
|
+
Extract the following:
|
|
27
|
+
|
|
28
|
+
1. **Character Changes**: How do characters change? (emotion, location, knowledge, relationships, goals)
|
|
29
|
+
2. **Plot Thread Changes**: How do plot threads progress? (status, tension, summary updates)
|
|
30
|
+
3. **New Facts**: Any new canon facts established?
|
|
31
|
+
4. **World Changes**: Any changes to the world setting?
|
|
32
|
+
|
|
33
|
+
Output JSON:
|
|
34
|
+
{
|
|
35
|
+
"characterChanges": [
|
|
36
|
+
{
|
|
37
|
+
"name": "character name",
|
|
38
|
+
"emotionalState": "new emotional state or null",
|
|
39
|
+
"location": "new location or null",
|
|
40
|
+
"newKnowledge": ["facts learned"],
|
|
41
|
+
"relationshipChanges": [{"with": "other character", "newStatus": "relationship"}],
|
|
42
|
+
"newGoal": "new goal or null"
|
|
43
|
+
}
|
|
44
|
+
],
|
|
45
|
+
"plotThreadChanges": [
|
|
46
|
+
{
|
|
47
|
+
"id": "thread id",
|
|
48
|
+
"status": "new status or null",
|
|
49
|
+
"tensionDelta": 0.1,
|
|
50
|
+
"newSummary": "updated summary or null"
|
|
51
|
+
}
|
|
52
|
+
],
|
|
53
|
+
"newFacts": [
|
|
54
|
+
{
|
|
55
|
+
"category": "character|world|plot",
|
|
56
|
+
"subject": "subject",
|
|
57
|
+
"attribute": "attribute",
|
|
58
|
+
"value": "value"
|
|
59
|
+
}
|
|
60
|
+
],
|
|
61
|
+
"worldChanges": ["any changes to the world"]
|
|
62
|
+
}`;
|
|
63
|
+
class StateUpdaterPipeline {
|
|
64
|
+
/**
|
|
65
|
+
* Run the complete post-chapter update pipeline
|
|
66
|
+
*/
|
|
67
|
+
async update(context) {
|
|
68
|
+
const { chapter, bible, currentState, canon, vectorStore, constraintGraph } = context;
|
|
69
|
+
const changes = [];
|
|
70
|
+
let memoriesAdded = 0;
|
|
71
|
+
let canonFactsAdded = 0;
|
|
72
|
+
// Step 1: Extract state changes using LLM
|
|
73
|
+
const extraction = await this.extractChanges(chapter, bible, currentState);
|
|
74
|
+
// Step 2: Update structured state
|
|
75
|
+
let newState = { ...currentState };
|
|
76
|
+
newState.chapter = chapter.number;
|
|
77
|
+
// Apply character changes
|
|
78
|
+
for (const change of extraction.characterChanges) {
|
|
79
|
+
if (newState.characters[change.name]) {
|
|
80
|
+
const char = newState.characters[change.name];
|
|
81
|
+
if (change.emotionalState) {
|
|
82
|
+
char.emotionalState = change.emotionalState;
|
|
83
|
+
changes.push({
|
|
84
|
+
type: 'character',
|
|
85
|
+
description: `${change.name} emotional state → ${change.emotionalState}`,
|
|
86
|
+
chapter: chapter.number,
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
if (change.location) {
|
|
90
|
+
const oldLocation = char.location;
|
|
91
|
+
char.location = change.location;
|
|
92
|
+
changes.push({
|
|
93
|
+
type: 'character',
|
|
94
|
+
description: `${change.name} moved: ${oldLocation} → ${change.location}`,
|
|
95
|
+
chapter: chapter.number,
|
|
96
|
+
});
|
|
97
|
+
// Update constraint graph
|
|
98
|
+
constraintGraph.updateCharacterLocation(change.name, change.location, chapter.number);
|
|
99
|
+
}
|
|
100
|
+
if (change.newKnowledge) {
|
|
101
|
+
char.knowledge = [...char.knowledge, ...change.newKnowledge];
|
|
102
|
+
// Add to constraint graph
|
|
103
|
+
for (const knowledge of change.newKnowledge) {
|
|
104
|
+
const factId = `fact-${this.sanitizeId(knowledge)}`;
|
|
105
|
+
if (!constraintGraph.getNode(factId)) {
|
|
106
|
+
constraintGraph.addNode({
|
|
107
|
+
id: factId,
|
|
108
|
+
type: 'fact',
|
|
109
|
+
label: knowledge,
|
|
110
|
+
properties: {},
|
|
111
|
+
chapterEstablished: chapter.number,
|
|
112
|
+
});
|
|
113
|
+
}
|
|
114
|
+
constraintGraph.addEdge({
|
|
115
|
+
id: `edge-${change.name}-knows-${factId}`,
|
|
116
|
+
from: `char-${change.name}`,
|
|
117
|
+
to: factId,
|
|
118
|
+
type: 'knows',
|
|
119
|
+
properties: { since: chapter.number },
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
if (change.relationshipChanges) {
|
|
124
|
+
for (const rel of change.relationshipChanges) {
|
|
125
|
+
char.relationships[rel.with] = rel.newStatus;
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
if (change.newGoal) {
|
|
129
|
+
char.goals = [...char.goals, change.newGoal];
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
// Apply plot thread changes
|
|
134
|
+
for (const change of extraction.plotThreadChanges) {
|
|
135
|
+
if (newState.plotThreads[change.id]) {
|
|
136
|
+
const thread = newState.plotThreads[change.id];
|
|
137
|
+
if (change.status) {
|
|
138
|
+
thread.status = change.status;
|
|
139
|
+
}
|
|
140
|
+
if (change.tensionDelta) {
|
|
141
|
+
thread.tension = Math.max(0, Math.min(1, thread.tension + change.tensionDelta));
|
|
142
|
+
}
|
|
143
|
+
if (change.newSummary) {
|
|
144
|
+
thread.summary = change.newSummary;
|
|
145
|
+
}
|
|
146
|
+
thread.lastChapter = chapter.number;
|
|
147
|
+
changes.push({
|
|
148
|
+
type: 'plot',
|
|
149
|
+
description: `${thread.name} updated: ${change.status || 'progress'}`,
|
|
150
|
+
chapter: chapter.number,
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
// Step 3: Add new canon facts
|
|
155
|
+
for (const fact of extraction.newFacts) {
|
|
156
|
+
// In real implementation, would add to canon store
|
|
157
|
+
canonFactsAdded++;
|
|
158
|
+
changes.push({
|
|
159
|
+
type: 'canon',
|
|
160
|
+
description: `Canon: ${fact.subject} ${fact.attribute} = ${fact.value}`,
|
|
161
|
+
chapter: chapter.number,
|
|
162
|
+
});
|
|
163
|
+
}
|
|
164
|
+
// Step 4: Extract and add narrative memories
|
|
165
|
+
const memories = await this.extractMemories(chapter, bible);
|
|
166
|
+
for (const memory of memories) {
|
|
167
|
+
await vectorStore.addMemory({
|
|
168
|
+
storyId: chapter.storyId,
|
|
169
|
+
chapterNumber: chapter.number,
|
|
170
|
+
content: memory.content,
|
|
171
|
+
category: memory.category,
|
|
172
|
+
timestamp: new Date(),
|
|
173
|
+
});
|
|
174
|
+
memoriesAdded++;
|
|
175
|
+
}
|
|
176
|
+
changes.push({
|
|
177
|
+
type: 'memory',
|
|
178
|
+
description: `${memoriesAdded} narrative memories extracted`,
|
|
179
|
+
chapter: chapter.number,
|
|
180
|
+
});
|
|
181
|
+
// Step 5: Add event to constraint graph
|
|
182
|
+
constraintGraph.addEvent(`ch${chapter.number}`, chapter.summary, Object.keys(newState.characters), chapter.number);
|
|
183
|
+
// Step 6: Update recent events
|
|
184
|
+
newState.recentEvents = [...newState.recentEvents, chapter.summary].slice(-10);
|
|
185
|
+
return {
|
|
186
|
+
structuredState: newState,
|
|
187
|
+
memoriesAdded,
|
|
188
|
+
canonFactsAdded,
|
|
189
|
+
graphUpdated: true,
|
|
190
|
+
changes,
|
|
191
|
+
};
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Extract state changes from chapter
|
|
195
|
+
*/
|
|
196
|
+
async extractChanges(chapter, bible, state) {
|
|
197
|
+
const characters = Object.values(state.characters)
|
|
198
|
+
.map(c => `- ${c.name}: ${c.emotionalState}, at ${c.location}`)
|
|
199
|
+
.join('\n');
|
|
200
|
+
const plotThreads = Object.values(state.plotThreads)
|
|
201
|
+
.map(t => `- ${t.name} (${t.status}, ${Math.round(t.tension * 100)}% tension)`)
|
|
202
|
+
.join('\n');
|
|
203
|
+
const prompt = STATE_EXTRACTION_PROMPT
|
|
204
|
+
.replace('{{title}}', bible.title)
|
|
205
|
+
.replace('{{genre}}', bible.genre)
|
|
206
|
+
.replace('{{chapterNumber}}', chapter.number.toString())
|
|
207
|
+
.replace('{{chapterTitle}}', chapter.title)
|
|
208
|
+
.replace('{{chapterContent}}', chapter.content.substring(0, 5000))
|
|
209
|
+
.replace('{{characters}}', characters)
|
|
210
|
+
.replace('{{plotThreads}}', plotThreads);
|
|
211
|
+
const result = await (0, client_js_1.getLLM)().completeJSON(prompt, {
|
|
212
|
+
temperature: 0.3,
|
|
213
|
+
maxTokens: 2000,
|
|
214
|
+
});
|
|
215
|
+
return result;
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* Extract narrative memories from chapter
|
|
219
|
+
*/
|
|
220
|
+
async extractMemories(chapter, bible) {
|
|
221
|
+
// Simplified memory extraction
|
|
222
|
+
// In real implementation, would use MemoryExtractor agent
|
|
223
|
+
const memories = [];
|
|
224
|
+
// Extract key events from summary
|
|
225
|
+
if (chapter.summary) {
|
|
226
|
+
memories.push({
|
|
227
|
+
content: chapter.summary,
|
|
228
|
+
category: 'event',
|
|
229
|
+
});
|
|
230
|
+
}
|
|
231
|
+
// Add chapter title as memory
|
|
232
|
+
memories.push({
|
|
233
|
+
content: `Chapter ${chapter.number}: ${chapter.title}`,
|
|
234
|
+
category: 'plot',
|
|
235
|
+
});
|
|
236
|
+
return memories;
|
|
237
|
+
}
|
|
238
|
+
/**
|
|
239
|
+
* Quick update without LLM (for testing)
|
|
240
|
+
*/
|
|
241
|
+
async quickUpdate(context) {
|
|
242
|
+
const { chapter, currentState, vectorStore, constraintGraph } = context;
|
|
243
|
+
const changes = [];
|
|
244
|
+
let newState = { ...currentState };
|
|
245
|
+
newState.chapter = chapter.number;
|
|
246
|
+
// Add basic memories
|
|
247
|
+
await vectorStore.addMemory({
|
|
248
|
+
storyId: chapter.storyId,
|
|
249
|
+
chapterNumber: chapter.number,
|
|
250
|
+
content: chapter.summary,
|
|
251
|
+
category: 'event',
|
|
252
|
+
timestamp: new Date(),
|
|
253
|
+
});
|
|
254
|
+
await vectorStore.addMemory({
|
|
255
|
+
storyId: chapter.storyId,
|
|
256
|
+
chapterNumber: chapter.number,
|
|
257
|
+
content: `Chapter ${chapter.number}: ${chapter.title}`,
|
|
258
|
+
category: 'plot',
|
|
259
|
+
timestamp: new Date(),
|
|
260
|
+
});
|
|
261
|
+
changes.push({
|
|
262
|
+
type: 'memory',
|
|
263
|
+
description: `2 memories added (quick mode)`,
|
|
264
|
+
chapter: chapter.number,
|
|
265
|
+
});
|
|
266
|
+
// Update recent events
|
|
267
|
+
newState.recentEvents = [...newState.recentEvents, chapter.summary].slice(-10);
|
|
268
|
+
// Add to constraint graph
|
|
269
|
+
constraintGraph.addEvent(`ch${chapter.number}`, chapter.summary, Object.keys(newState.characters), chapter.number);
|
|
270
|
+
return {
|
|
271
|
+
structuredState: newState,
|
|
272
|
+
memoriesAdded: 2,
|
|
273
|
+
canonFactsAdded: 0,
|
|
274
|
+
graphUpdated: true,
|
|
275
|
+
changes,
|
|
276
|
+
};
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* Format update result
|
|
280
|
+
*/
|
|
281
|
+
formatResult(result) {
|
|
282
|
+
const lines = ['## State Update Result'];
|
|
283
|
+
lines.push(`\n**Memories Added:** ${result.memoriesAdded}`);
|
|
284
|
+
lines.push(`**Canon Facts Added:** ${result.canonFactsAdded}`);
|
|
285
|
+
lines.push(`**Graph Updated:** ${result.graphUpdated ? '✅' : '❌'}`);
|
|
286
|
+
if (result.changes.length > 0) {
|
|
287
|
+
lines.push('\n**Changes:**');
|
|
288
|
+
const byType = {
|
|
289
|
+
character: [],
|
|
290
|
+
plot: [],
|
|
291
|
+
world: [],
|
|
292
|
+
canon: [],
|
|
293
|
+
memory: [],
|
|
294
|
+
};
|
|
295
|
+
for (const change of result.changes) {
|
|
296
|
+
byType[change.type].push(change);
|
|
297
|
+
}
|
|
298
|
+
for (const [type, changes] of Object.entries(byType)) {
|
|
299
|
+
if (changes.length > 0) {
|
|
300
|
+
lines.push(`\n*${type.charAt(0).toUpperCase() + type.slice(1)}:*`);
|
|
301
|
+
for (const change of changes) {
|
|
302
|
+
lines.push(` - ${change.description}`);
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
return lines.join('\n');
|
|
308
|
+
}
|
|
309
|
+
sanitizeId(str) {
|
|
310
|
+
return str.replace(/[^a-zA-Z0-9]/g, '_').substring(0, 50);
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
exports.StateUpdaterPipeline = StateUpdaterPipeline;
|
|
314
|
+
exports.stateUpdaterPipeline = new StateUpdaterPipeline();
|
|
315
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stateUpdater.js","sourceRoot":"","sources":["../../src/memory/stateUpdater.ts"],"names":[],"mappings":";;;AAAA,gDAA0C;AA8B1C,MAAM,uBAAuB,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAyD9B,CAAC;AAEH,MAAa,oBAAoB;IAC/B;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,OAAsB;QACjC,MAAM,EAAE,OAAO,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,WAAW,EAAE,eAAe,EAAE,GAAG,OAAO,CAAC;QAEtF,MAAM,OAAO,GAAkB,EAAE,CAAC;QAClC,IAAI,aAAa,GAAG,CAAC,CAAC;QACtB,IAAI,eAAe,GAAG,CAAC,CAAC;QAExB,0CAA0C;QAC1C,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,KAAK,EAAE,YAAY,CAAC,CAAC;QAE3E,kCAAkC;QAClC,IAAI,QAAQ,GAAG,EAAE,GAAG,YAAY,EAAE,CAAC;QACnC,QAAQ,CAAC,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC;QAElC,0BAA0B;QAC1B,KAAK,MAAM,MAAM,IAAI,UAAU,CAAC,gBAAgB,EAAE,CAAC;YACjD,IAAI,QAAQ,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;gBACrC,MAAM,IAAI,GAAG,QAAQ,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAE9C,IAAI,MAAM,CAAC,cAAc,EAAE,CAAC;oBAC1B,IAAI,CAAC,cAAc,GAAG,MAAM,CAAC,cAAc,CAAC;oBAC5C,OAAO,CAAC,IAAI,CAAC;wBACX,IAAI,EAAE,WAAW;wBACjB,WAAW,EAAE,GAAG,MAAM,CAAC,IAAI,sBAAsB,MAAM,CAAC,cAAc,EAAE;wBACxE,OAAO,EAAE,OAAO,CAAC,MAAM;qBACxB,CAAC,CAAC;gBACL,CAAC;gBAED,IAAI,MAAM,CAAC,QAAQ,EAAE,CAAC;oBACpB,MAAM,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC;oBAClC,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC;oBAChC,OAAO,CAAC,IAAI,CAAC;wBACX,IAAI,EAAE,WAAW;wBACjB,WAAW,EAAE,GAAG,MAAM,CAAC,IAAI,WAAW,WAAW,MAAM,MAAM,CAAC,QAAQ,EAAE;wBACxE,OAAO,EAAE,OAAO,CAAC,MAAM;qBACxB,CAAC,CAAC;oBAEH,0BAA0B;oBAC1B,eAAe,CAAC,uBAAuB,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,QAAQ,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;gBACxF,CAAC;gBAED,IAAI,MAAM,CAAC,YAAY,EAAE,CAAC;oBACxB,IAAI,CAAC,SAAS,GAAG,CAAC,GAAG,IAAI,CAAC,SAAS,EAAE,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;oBAE7D,0BAA0B;oBAC1B,KAAK,MAAM,SAAS,IAAI,MAAM,CAAC,YAAY,EAAE,CAAC;wBAC5C,MAAM,MAAM,GAAG,QAAQ,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;wBACpD,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;4BACrC,eAAe,CAAC,OAAO,CAAC;gCACtB,EAAE,EAAE,MAAM;gCACV,IAAI,EAAE,MAAM;gCACZ,KAAK,EAAE,SAAS;gCAChB,UAAU,EAAE,EAAE;gCACd,kBAAkB,EAAE,OAAO,CAAC,MAAM;6BACnC,CAAC,CAAC;wBACL,CAAC;wBAED,eAAe,CAAC,OAAO,CAAC;4BACtB,EAAE,EAAE,QAAQ,MAAM,CAAC,IAAI,UAAU,MAAM,EAAE;4BACzC,IAAI,EAAE,QAAQ,MAAM,CAAC,IAAI,EAAE;4BAC3B,EAAE,EAAE,MAAM;4BACV,IAAI,EAAE,OAAO;4BACb,UAAU,EAAE,EAAE,KAAK,EAAE,OAAO,CAAC,MAAM,EAAE;yBACtC,CAAC,CAAC;oBACL,CAAC;gBACH,CAAC;gBAED,IAAI,MAAM,CAAC,mBAAmB,EAAE,CAAC;oBAC/B,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,mBAAmB,EAAE,CAAC;wBAC7C,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,GAAG,CAAC,SAAS,CAAC;oBAC/C,CAAC;gBACH,CAAC;gBAED,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;oBACnB,IAAI,CAAC,KAAK,GAAG,CAAC,GAAG,IAAI,CAAC,KAAK,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC;gBAC/C,CAAC;YACH,CAAC;QACH,CAAC;QAED,4BAA4B;QAC5B,KAAK,MAAM,MAAM,IAAI,UAAU,CAAC,iBAAiB,EAAE,CAAC;YAClD,IAAI,QAAQ,CAAC,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC;gBACpC,MAAM,MAAM,GAAG,QAAQ,CAAC,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;gBAE/C,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;oBAClB,MAAM,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC;gBAChC,CAAC;gBAED,IAAI,MAAM,CAAC,YAAY,EAAE,CAAC;oBACxB,MAAM,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,MAAM,CAAC,OAAO,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC;gBAClF,CAAC;gBAED,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;oBACtB,MAAM,CAAC,OAAO,GAAG,MAAM,CAAC,UAAU,CAAC;gBACrC,CAAC;gBAED,MAAM,CAAC,WAAW,GAAG,OAAO,CAAC,MAAM,CAAC;gBAEpC,OAAO,CAAC,IAAI,CAAC;oBACX,IAAI,EAAE,MAAM;oBACZ,WAAW,EAAE,GAAG,MAAM,CAAC,IAAI,aAAa,MAAM,CAAC,MAAM,IAAI,UAAU,EAAE;oBACrE,OAAO,EAAE,OAAO,CAAC,MAAM;iBACxB,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,8BAA8B;QAC9B,KAAK,MAAM,IAAI,IAAI,UAAU,CAAC,QAAQ,EAAE,CAAC;YACvC,mDAAmD;YACnD,eAAe,EAAE,CAAC;YAClB,OAAO,CAAC,IAAI,CAAC;gBACX,IAAI,EAAE,OAAO;gBACb,WAAW,EAAE,UAAU,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,SAAS,MAAM,IAAI,CAAC,KAAK,EAAE;gBACvE,OAAO,EAAE,OAAO,CAAC,MAAM;aACxB,CAAC,CAAC;QACL,CAAC;QAED,6CAA6C;QAC7C,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAC5D,KAAK,MAAM,MAAM,IAAI,QAAQ,EAAE,CAAC;YAC9B,MAAM,WAAW,CAAC,SAAS,CAAC;gBAC1B,OAAO,EAAE,OAAO,CAAC,OAAO;gBACxB,aAAa,EAAE,OAAO,CAAC,MAAM;gBAC7B,OAAO,EAAE,MAAM,CAAC,OAAO;gBACvB,QAAQ,EAAE,MAAM,CAAC,QAAQ;gBACzB,SAAS,EAAE,IAAI,IAAI,EAAE;aACtB,CAAC,CAAC;YACH,aAAa,EAAE,CAAC;QAClB,CAAC;QAED,OAAO,CAAC,IAAI,CAAC;YACX,IAAI,EAAE,QAAQ;YACd,WAAW,EAAE,GAAG,aAAa,+BAA+B;YAC5D,OAAO,EAAE,OAAO,CAAC,MAAM;SACxB,CAAC,CAAC;QAEH,wCAAwC;QACxC,eAAe,CAAC,QAAQ,CACtB,KAAK,OAAO,CAAC,MAAM,EAAE,EACrB,OAAO,CAAC,OAAO,EACf,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,EAChC,OAAO,CAAC,MAAM,CACf,CAAC;QAEF,+BAA+B;QAC/B,QAAQ,CAAC,YAAY,GAAG,CAAC,GAAG,QAAQ,CAAC,YAAY,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC;QAE/E,OAAO;YACL,eAAe,EAAE,QAAQ;YACzB,aAAa;YACb,eAAe;YACf,YAAY,EAAE,IAAI;YAClB,OAAO;SACR,CAAC;IACJ,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,cAAc,CAC1B,OAAgB,EAChB,KAAiB,EACjB,KAA2B;QAwB3B,MAAM,UAAU,GAAG,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC;aAC/C,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,IAAI,KAAK,CAAC,CAAC,cAAc,QAAQ,CAAC,CAAC,QAAQ,EAAE,CAAC;aAC9D,IAAI,CAAC,IAAI,CAAC,CAAC;QAEd,MAAM,WAAW,GAAG,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC;aACjD,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,IAAI,KAAK,CAAC,CAAC,MAAM,KAAK,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,GAAG,GAAG,CAAC,YAAY,CAAC;aAC9E,IAAI,CAAC,IAAI,CAAC,CAAC;QAEd,MAAM,MAAM,GAAG,uBAAuB;aACnC,OAAO,CAAC,WAAW,EAAE,KAAK,CAAC,KAAK,CAAC;aACjC,OAAO,CAAC,WAAW,EAAE,KAAK,CAAC,KAAK,CAAC;aACjC,OAAO,CAAC,mBAAmB,EAAE,OAAO,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;aACvD,OAAO,CAAC,kBAAkB,EAAE,OAAO,CAAC,KAAK,CAAC;aAC1C,OAAO,CAAC,oBAAoB,EAAE,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC;aACjE,OAAO,CAAC,gBAAgB,EAAE,UAAU,CAAC;aACrC,OAAO,CAAC,iBAAiB,EAAE,WAAW,CAAC,CAAC;QAE3C,MAAM,MAAM,GAAG,MAAM,IAAA,kBAAM,GAAE,CAAC,YAAY,CAKvC,MAAM,EAAE;YACT,WAAW,EAAE,GAAG;YAChB,SAAS,EAAE,IAAI;SAChB,CAAC,CAAC;QAEH,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,eAAe,CAC3B,OAAgB,EAChB,KAAiB;QAEjB,+BAA+B;QAC/B,0DAA0D;QAC1D,MAAM,QAAQ,GAAsE,EAAE,CAAC;QAEvF,kCAAkC;QAClC,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpB,QAAQ,CAAC,IAAI,CAAC;gBACZ,OAAO,EAAE,OAAO,CAAC,OAAO;gBACxB,QAAQ,EAAE,OAAO;aAClB,CAAC,CAAC;QACL,CAAC;QAED,8BAA8B;QAC9B,QAAQ,CAAC,IAAI,CAAC;YACZ,OAAO,EAAE,WAAW,OAAO,CAAC,MAAM,KAAK,OAAO,CAAC,KAAK,EAAE;YACtD,QAAQ,EAAE,MAAM;SACjB,CAAC,CAAC;QAEH,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW,CAAC,OAAsB;QACtC,MAAM,EAAE,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,eAAe,EAAE,GAAG,OAAO,CAAC;QAExE,MAAM,OAAO,GAAkB,EAAE,CAAC;QAClC,IAAI,QAAQ,GAAG,EAAE,GAAG,YAAY,EAAE,CAAC;QACnC,QAAQ,CAAC,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC;QAElC,qBAAqB;QACrB,MAAM,WAAW,CAAC,SAAS,CAAC;YAC1B,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,aAAa,EAAE,OAAO,CAAC,MAAM;YAC7B,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,QAAQ,EAAE,OAAO;YACjB,SAAS,EAAE,IAAI,IAAI,EAAE;SACtB,CAAC,CAAC;QAEH,MAAM,WAAW,CAAC,SAAS,CAAC;YAC1B,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,aAAa,EAAE,OAAO,CAAC,MAAM;YAC7B,OAAO,EAAE,WAAW,OAAO,CAAC,MAAM,KAAK,OAAO,CAAC,KAAK,EAAE;YACtD,QAAQ,EAAE,MAAM;YAChB,SAAS,EAAE,IAAI,IAAI,EAAE;SACtB,CAAC,CAAC;QAEH,OAAO,CAAC,IAAI,CAAC;YACX,IAAI,EAAE,QAAQ;YACd,WAAW,EAAE,+BAA+B;YAC5C,OAAO,EAAE,OAAO,CAAC,MAAM;SACxB,CAAC,CAAC;QAEH,uBAAuB;QACvB,QAAQ,CAAC,YAAY,GAAG,CAAC,GAAG,QAAQ,CAAC,YAAY,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC;QAE/E,0BAA0B;QAC1B,eAAe,CAAC,QAAQ,CACtB,KAAK,OAAO,CAAC,MAAM,EAAE,EACrB,OAAO,CAAC,OAAO,EACf,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,EAChC,OAAO,CAAC,MAAM,CACf,CAAC;QAEF,OAAO;YACL,eAAe,EAAE,QAAQ;YACzB,aAAa,EAAE,CAAC;YAChB,eAAe,EAAE,CAAC;YAClB,YAAY,EAAE,IAAI;YAClB,OAAO;SACR,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,YAAY,CAAC,MAAyB;QACpC,MAAM,KAAK,GAAa,CAAC,wBAAwB,CAAC,CAAC;QAEnD,KAAK,CAAC,IAAI,CAAC,yBAAyB,MAAM,CAAC,aAAa,EAAE,CAAC,CAAC;QAC5D,KAAK,CAAC,IAAI,CAAC,0BAA0B,MAAM,CAAC,eAAe,EAAE,CAAC,CAAC;QAC/D,KAAK,CAAC,IAAI,CAAC,sBAAsB,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;QAEpE,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC9B,KAAK,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YAE7B,MAAM,MAAM,GAAkC;gBAC5C,SAAS,EAAE,EAAE;gBACb,IAAI,EAAE,EAAE;gBACR,KAAK,EAAE,EAAE;gBACT,KAAK,EAAE,EAAE;gBACT,MAAM,EAAE,EAAE;aACX,CAAC;YAEF,KAAK,MAAM,MAAM,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;gBACpC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YACnC,CAAC;YAED,KAAK,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;gBACrD,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACvB,KAAK,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;oBACnE,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;wBAC7B,KAAK,CAAC,IAAI,CAAC,OAAO,MAAM,CAAC,WAAW,EAAE,CAAC,CAAC;oBAC1C,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAEO,UAAU,CAAC,GAAW;QAC5B,OAAO,GAAG,CAAC,OAAO,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IAC5D,CAAC;CACF;AAtVD,oDAsVC;AAEY,QAAA,oBAAoB,GAAG,IAAI,oBAAoB,EAAE,CAAC","sourcesContent":["import { getLLM } from '../llm/client.js';\r\nimport type { Chapter, StoryBible } from '../types/index.js';\r\nimport type { StoryStructuredState, CharacterState, PlotThreadState } from '../story/structuredState.js';\r\nimport type { VectorStore, NarrativeMemory } from './vectorStore.js';\r\nimport type { CanonStore } from './canonStore.js';\r\nimport { ConstraintGraph } from '../constraints/constraintGraph.js';\r\n\r\nexport interface StateUpdateResult {\r\n  structuredState: StoryStructuredState;\r\n  memoriesAdded: number;\r\n  canonFactsAdded: number;\r\n  graphUpdated: boolean;\r\n  changes: StateChange[];\r\n}\r\n\r\nexport interface StateChange {\r\n  type: 'character' | 'plot' | 'world' | 'canon' | 'memory';\r\n  description: string;\r\n  chapter: number;\r\n}\r\n\r\nexport interface UpdateContext {\r\n  chapter: Chapter;\r\n  bible: StoryBible;\r\n  currentState: StoryStructuredState;\r\n  canon: CanonStore;\r\n  vectorStore: VectorStore;\r\n  constraintGraph: ConstraintGraph;\r\n}\r\n\r\nconst STATE_EXTRACTION_PROMPT = `You are a narrative state extractor. Analyze this chapter and extract all state changes.\r\n\r\n## Story Bible\r\n\r\n**Title:** {{title}}\r\n**Genre:** {{genre}}\r\n\r\n## Chapter {{chapterNumber}}: {{chapterTitle}}\r\n\r\n{{chapterContent}}\r\n\r\n## Current Character States\r\n\r\n{{characters}}\r\n\r\n## Current Plot Threads\r\n\r\n{{plotThreads}}\r\n\r\n## Extraction Task\r\n\r\nExtract the following:\r\n\r\n1. **Character Changes**: How do characters change? (emotion, location, knowledge, relationships, goals)\r\n2. **Plot Thread Changes**: How do plot threads progress? (status, tension, summary updates)\r\n3. **New Facts**: Any new canon facts established?\r\n4. **World Changes**: Any changes to the world setting?\r\n\r\nOutput JSON:\r\n{\r\n  \"characterChanges\": [\r\n    {\r\n      \"name\": \"character name\",\r\n      \"emotionalState\": \"new emotional state or null\",\r\n      \"location\": \"new location or null\",\r\n      \"newKnowledge\": [\"facts learned\"],\r\n      \"relationshipChanges\": [{\"with\": \"other character\", \"newStatus\": \"relationship\"}],\r\n      \"newGoal\": \"new goal or null\"\r\n    }\r\n  ],\r\n  \"plotThreadChanges\": [\r\n    {\r\n      \"id\": \"thread id\",\r\n      \"status\": \"new status or null\",\r\n      \"tensionDelta\": 0.1,\r\n      \"newSummary\": \"updated summary or null\"\r\n    }\r\n  ],\r\n  \"newFacts\": [\r\n    {\r\n      \"category\": \"character|world|plot\",\r\n      \"subject\": \"subject\",\r\n      \"attribute\": \"attribute\",\r\n      \"value\": \"value\"\r\n    }\r\n  ],\r\n  \"worldChanges\": [\"any changes to the world\"]\r\n}`;\r\n\r\nexport class StateUpdaterPipeline {\r\n  /**\r\n   * Run the complete post-chapter update pipeline\r\n   */\r\n  async update(context: UpdateContext): Promise<StateUpdateResult> {\r\n    const { chapter, bible, currentState, canon, vectorStore, constraintGraph } = context;\r\n    \r\n    const changes: StateChange[] = [];\r\n    let memoriesAdded = 0;\r\n    let canonFactsAdded = 0;\r\n    \r\n    // Step 1: Extract state changes using LLM\r\n    const extraction = await this.extractChanges(chapter, bible, currentState);\r\n    \r\n    // Step 2: Update structured state\r\n    let newState = { ...currentState };\r\n    newState.chapter = chapter.number;\r\n    \r\n    // Apply character changes\r\n    for (const change of extraction.characterChanges) {\r\n      if (newState.characters[change.name]) {\r\n        const char = newState.characters[change.name];\r\n        \r\n        if (change.emotionalState) {\r\n          char.emotionalState = change.emotionalState;\r\n          changes.push({\r\n            type: 'character',\r\n            description: `${change.name} emotional state → ${change.emotionalState}`,\r\n            chapter: chapter.number,\r\n          });\r\n        }\r\n        \r\n        if (change.location) {\r\n          const oldLocation = char.location;\r\n          char.location = change.location;\r\n          changes.push({\r\n            type: 'character',\r\n            description: `${change.name} moved: ${oldLocation} → ${change.location}`,\r\n            chapter: chapter.number,\r\n          });\r\n          \r\n          // Update constraint graph\r\n          constraintGraph.updateCharacterLocation(change.name, change.location, chapter.number);\r\n        }\r\n        \r\n        if (change.newKnowledge) {\r\n          char.knowledge = [...char.knowledge, ...change.newKnowledge];\r\n          \r\n          // Add to constraint graph\r\n          for (const knowledge of change.newKnowledge) {\r\n            const factId = `fact-${this.sanitizeId(knowledge)}`;\r\n            if (!constraintGraph.getNode(factId)) {\r\n              constraintGraph.addNode({\r\n                id: factId,\r\n                type: 'fact',\r\n                label: knowledge,\r\n                properties: {},\r\n                chapterEstablished: chapter.number,\r\n              });\r\n            }\r\n            \r\n            constraintGraph.addEdge({\r\n              id: `edge-${change.name}-knows-${factId}`,\r\n              from: `char-${change.name}`,\r\n              to: factId,\r\n              type: 'knows',\r\n              properties: { since: chapter.number },\r\n            });\r\n          }\r\n        }\r\n        \r\n        if (change.relationshipChanges) {\r\n          for (const rel of change.relationshipChanges) {\r\n            char.relationships[rel.with] = rel.newStatus;\r\n          }\r\n        }\r\n        \r\n        if (change.newGoal) {\r\n          char.goals = [...char.goals, change.newGoal];\r\n        }\r\n      }\r\n    }\r\n    \r\n    // Apply plot thread changes\r\n    for (const change of extraction.plotThreadChanges) {\r\n      if (newState.plotThreads[change.id]) {\r\n        const thread = newState.plotThreads[change.id];\r\n        \r\n        if (change.status) {\r\n          thread.status = change.status;\r\n        }\r\n        \r\n        if (change.tensionDelta) {\r\n          thread.tension = Math.max(0, Math.min(1, thread.tension + change.tensionDelta));\r\n        }\r\n        \r\n        if (change.newSummary) {\r\n          thread.summary = change.newSummary;\r\n        }\r\n        \r\n        thread.lastChapter = chapter.number;\r\n        \r\n        changes.push({\r\n          type: 'plot',\r\n          description: `${thread.name} updated: ${change.status || 'progress'}`,\r\n          chapter: chapter.number,\r\n        });\r\n      }\r\n    }\r\n    \r\n    // Step 3: Add new canon facts\r\n    for (const fact of extraction.newFacts) {\r\n      // In real implementation, would add to canon store\r\n      canonFactsAdded++;\r\n      changes.push({\r\n        type: 'canon',\r\n        description: `Canon: ${fact.subject} ${fact.attribute} = ${fact.value}`,\r\n        chapter: chapter.number,\r\n      });\r\n    }\r\n    \r\n    // Step 4: Extract and add narrative memories\r\n    const memories = await this.extractMemories(chapter, bible);\r\n    for (const memory of memories) {\r\n      await vectorStore.addMemory({\r\n        storyId: chapter.storyId,\r\n        chapterNumber: chapter.number,\r\n        content: memory.content,\r\n        category: memory.category,\r\n        timestamp: new Date(),\r\n      });\r\n      memoriesAdded++;\r\n    }\r\n    \r\n    changes.push({\r\n      type: 'memory',\r\n      description: `${memoriesAdded} narrative memories extracted`,\r\n      chapter: chapter.number,\r\n    });\r\n    \r\n    // Step 5: Add event to constraint graph\r\n    constraintGraph.addEvent(\r\n      `ch${chapter.number}`,\r\n      chapter.summary,\r\n      Object.keys(newState.characters),\r\n      chapter.number\r\n    );\r\n    \r\n    // Step 6: Update recent events\r\n    newState.recentEvents = [...newState.recentEvents, chapter.summary].slice(-10);\r\n    \r\n    return {\r\n      structuredState: newState,\r\n      memoriesAdded,\r\n      canonFactsAdded,\r\n      graphUpdated: true,\r\n      changes,\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Extract state changes from chapter\r\n   */\r\n  private async extractChanges(\r\n    chapter: Chapter,\r\n    bible: StoryBible,\r\n    state: StoryStructuredState\r\n  ): Promise<{\r\n    characterChanges: Array<{\r\n      name: string;\r\n      emotionalState?: string;\r\n      location?: string;\r\n      newKnowledge?: string[];\r\n      relationshipChanges?: Array<{ with: string; newStatus: string }>;\r\n      newGoal?: string;\r\n    }>;\r\n    plotThreadChanges: Array<{\r\n      id: string;\r\n      status?: 'dormant' | 'active' | 'escalating' | 'resolved';\r\n      tensionDelta?: number;\r\n      newSummary?: string;\r\n    }>;\r\n    newFacts: Array<{\r\n      category: 'character' | 'world' | 'plot';\r\n      subject: string;\r\n      attribute: string;\r\n      value: string;\r\n    }>;\r\n    worldChanges: string[];\r\n  }> {\r\n    const characters = Object.values(state.characters)\r\n      .map(c => `- ${c.name}: ${c.emotionalState}, at ${c.location}`)\r\n      .join('\\n');\r\n    \r\n    const plotThreads = Object.values(state.plotThreads)\r\n      .map(t => `- ${t.name} (${t.status}, ${Math.round(t.tension * 100)}% tension)`)\r\n      .join('\\n');\r\n    \r\n    const prompt = STATE_EXTRACTION_PROMPT\r\n      .replace('{{title}}', bible.title)\r\n      .replace('{{genre}}', bible.genre)\r\n      .replace('{{chapterNumber}}', chapter.number.toString())\r\n      .replace('{{chapterTitle}}', chapter.title)\r\n      .replace('{{chapterContent}}', chapter.content.substring(0, 5000))\r\n      .replace('{{characters}}', characters)\r\n      .replace('{{plotThreads}}', plotThreads);\r\n    \r\n    const result = await getLLM().completeJSON<{\r\n      characterChanges: any[];\r\n      plotThreadChanges: any[];\r\n      newFacts: any[];\r\n      worldChanges: string[];\r\n    }>(prompt, {\r\n      temperature: 0.3,\r\n      maxTokens: 2000,\r\n    });\r\n    \r\n    return result;\r\n  }\r\n  \r\n  /**\r\n   * Extract narrative memories from chapter\r\n   */\r\n  private async extractMemories(\r\n    chapter: Chapter,\r\n    bible: StoryBible\r\n  ): Promise<Array<{ content: string; category: NarrativeMemory['category'] }>> {\r\n    // Simplified memory extraction\r\n    // In real implementation, would use MemoryExtractor agent\r\n    const memories: Array<{ content: string; category: NarrativeMemory['category'] }> = [];\r\n    \r\n    // Extract key events from summary\r\n    if (chapter.summary) {\r\n      memories.push({\r\n        content: chapter.summary,\r\n        category: 'event',\r\n      });\r\n    }\r\n    \r\n    // Add chapter title as memory\r\n    memories.push({\r\n      content: `Chapter ${chapter.number}: ${chapter.title}`,\r\n      category: 'plot',\r\n    });\r\n    \r\n    return memories;\r\n  }\r\n  \r\n  /**\r\n   * Quick update without LLM (for testing)\r\n   */\r\n  async quickUpdate(context: UpdateContext): Promise<StateUpdateResult> {\r\n    const { chapter, currentState, vectorStore, constraintGraph } = context;\r\n    \r\n    const changes: StateChange[] = [];\r\n    let newState = { ...currentState };\r\n    newState.chapter = chapter.number;\r\n    \r\n    // Add basic memories\r\n    await vectorStore.addMemory({\r\n      storyId: chapter.storyId,\r\n      chapterNumber: chapter.number,\r\n      content: chapter.summary,\r\n      category: 'event',\r\n      timestamp: new Date(),\r\n    });\r\n    \r\n    await vectorStore.addMemory({\r\n      storyId: chapter.storyId,\r\n      chapterNumber: chapter.number,\r\n      content: `Chapter ${chapter.number}: ${chapter.title}`,\r\n      category: 'plot',\r\n      timestamp: new Date(),\r\n    });\r\n    \r\n    changes.push({\r\n      type: 'memory',\r\n      description: `2 memories added (quick mode)`,\r\n      chapter: chapter.number,\r\n    });\r\n    \r\n    // Update recent events\r\n    newState.recentEvents = [...newState.recentEvents, chapter.summary].slice(-10);\r\n    \r\n    // Add to constraint graph\r\n    constraintGraph.addEvent(\r\n      `ch${chapter.number}`,\r\n      chapter.summary,\r\n      Object.keys(newState.characters),\r\n      chapter.number\r\n    );\r\n    \r\n    return {\r\n      structuredState: newState,\r\n      memoriesAdded: 2,\r\n      canonFactsAdded: 0,\r\n      graphUpdated: true,\r\n      changes,\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Format update result\r\n   */\r\n  formatResult(result: StateUpdateResult): string {\r\n    const lines: string[] = ['## State Update Result'];\r\n    \r\n    lines.push(`\\n**Memories Added:** ${result.memoriesAdded}`);\r\n    lines.push(`**Canon Facts Added:** ${result.canonFactsAdded}`);\r\n    lines.push(`**Graph Updated:** ${result.graphUpdated ? '✅' : '❌'}`);\r\n    \r\n    if (result.changes.length > 0) {\r\n      lines.push('\\n**Changes:**');\r\n      \r\n      const byType: Record<string, StateChange[]> = {\r\n        character: [],\r\n        plot: [],\r\n        world: [],\r\n        canon: [],\r\n        memory: [],\r\n      };\r\n      \r\n      for (const change of result.changes) {\r\n        byType[change.type].push(change);\r\n      }\r\n      \r\n      for (const [type, changes] of Object.entries(byType)) {\r\n        if (changes.length > 0) {\r\n          lines.push(`\\n*${type.charAt(0).toUpperCase() + type.slice(1)}:*`);\r\n          for (const change of changes) {\r\n            lines.push(`  - ${change.description}`);\r\n          }\r\n        }\r\n      }\r\n    }\r\n    \r\n    return lines.join('\\n');\r\n  }\r\n  \r\n  private sanitizeId(str: string): string {\r\n    return str.replace(/[^a-zA-Z0-9]/g, '_').substring(0, 50);\r\n  }\r\n}\r\n\r\nexport const stateUpdaterPipeline = new StateUpdaterPipeline();\r\n"]}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
export interface NarrativeMemory {
|
|
2
|
+
id: number;
|
|
3
|
+
storyId: string;
|
|
4
|
+
chapterNumber: number;
|
|
5
|
+
content: string;
|
|
6
|
+
category: 'event' | 'character' | 'world' | 'plot';
|
|
7
|
+
timestamp: Date;
|
|
8
|
+
embedding?: number[];
|
|
9
|
+
}
|
|
10
|
+
export interface MemorySearchResult {
|
|
11
|
+
memory: NarrativeMemory;
|
|
12
|
+
score: number;
|
|
13
|
+
}
|
|
14
|
+
export declare class VectorStore {
|
|
15
|
+
private index;
|
|
16
|
+
private memories;
|
|
17
|
+
private dimension;
|
|
18
|
+
private storyId;
|
|
19
|
+
private nextId;
|
|
20
|
+
constructor(storyId: string);
|
|
21
|
+
initialize(maxElements?: number): Promise<void>;
|
|
22
|
+
/**
|
|
23
|
+
* Resize the index to accommodate more memories
|
|
24
|
+
*/
|
|
25
|
+
resizeIndex(newMaxElements: number): void;
|
|
26
|
+
/**
|
|
27
|
+
* Ensure capacity for upcoming memories
|
|
28
|
+
*/
|
|
29
|
+
ensureCapacity(additionalMemories: number): void;
|
|
30
|
+
addMemory(memory: Omit<NarrativeMemory, 'id' | 'embedding'>): Promise<NarrativeMemory>;
|
|
31
|
+
searchSimilar(query: string, k?: number): Promise<MemorySearchResult[]>;
|
|
32
|
+
searchByCategory(query: string, category: NarrativeMemory['category'], k?: number): Promise<MemorySearchResult[]>;
|
|
33
|
+
getMemoriesByChapter(chapterNumber: number): NarrativeMemory[];
|
|
34
|
+
getAllMemories(): NarrativeMemory[];
|
|
35
|
+
private generateEmbedding;
|
|
36
|
+
private generateMockEmbedding;
|
|
37
|
+
serialize(): string;
|
|
38
|
+
load(data: string): Promise<void>;
|
|
39
|
+
}
|
|
40
|
+
export declare function getVectorStore(storyId: string): VectorStore;
|
|
41
|
+
export declare function clearVectorStore(storyId: string): void;
|
|
@@ -0,0 +1,166 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.VectorStore = void 0;
|
|
4
|
+
exports.getVectorStore = getVectorStore;
|
|
5
|
+
exports.clearVectorStore = clearVectorStore;
|
|
6
|
+
const client_js_1 = require("../llm/client.js");
|
|
7
|
+
const hnswlib_node_1 = require("hnswlib-node");
|
|
8
|
+
class VectorStore {
|
|
9
|
+
index = null;
|
|
10
|
+
memories = new Map();
|
|
11
|
+
dimension = 1536; // text-embedding-3-small dimension
|
|
12
|
+
storyId;
|
|
13
|
+
nextId = 0;
|
|
14
|
+
constructor(storyId) {
|
|
15
|
+
this.storyId = storyId;
|
|
16
|
+
}
|
|
17
|
+
async initialize(maxElements = 10000) {
|
|
18
|
+
this.index = new hnswlib_node_1.HierarchicalNSW('cosine', this.dimension);
|
|
19
|
+
this.index.initIndex(maxElements, 16, 200);
|
|
20
|
+
this.nextId = 0;
|
|
21
|
+
this.memories.clear();
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Resize the index to accommodate more memories
|
|
25
|
+
*/
|
|
26
|
+
resizeIndex(newMaxElements) {
|
|
27
|
+
if (!this.index) {
|
|
28
|
+
throw new Error('VectorStore not initialized. Call initialize() first.');
|
|
29
|
+
}
|
|
30
|
+
if (newMaxElements <= this.memories.size) {
|
|
31
|
+
return; // No need to resize
|
|
32
|
+
}
|
|
33
|
+
this.index.resizeIndex(newMaxElements);
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Ensure capacity for upcoming memories
|
|
37
|
+
*/
|
|
38
|
+
ensureCapacity(additionalMemories) {
|
|
39
|
+
const requiredCapacity = this.memories.size + additionalMemories;
|
|
40
|
+
const currentCapacity = this.index?.getMaxElements() || 0;
|
|
41
|
+
if (requiredCapacity > currentCapacity) {
|
|
42
|
+
// Add 50% more capacity or use required, whichever is larger
|
|
43
|
+
const newCapacity = Math.max(Math.floor(currentCapacity * 1.5), requiredCapacity);
|
|
44
|
+
this.resizeIndex(newCapacity);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
async addMemory(memory) {
|
|
48
|
+
if (!this.index) {
|
|
49
|
+
throw new Error('VectorStore not initialized. Call initialize() first.');
|
|
50
|
+
}
|
|
51
|
+
// Auto-resize if we're near capacity (add 50% more)
|
|
52
|
+
const currentCapacity = this.index.getMaxElements();
|
|
53
|
+
if (this.memories.size >= currentCapacity - 1) {
|
|
54
|
+
this.resizeIndex(Math.floor(currentCapacity * 1.5));
|
|
55
|
+
}
|
|
56
|
+
const id = this.nextId++;
|
|
57
|
+
// Generate embedding using OpenAI
|
|
58
|
+
const embedding = await this.generateEmbedding(memory.content);
|
|
59
|
+
const fullMemory = {
|
|
60
|
+
...memory,
|
|
61
|
+
id,
|
|
62
|
+
embedding,
|
|
63
|
+
};
|
|
64
|
+
// Add to HNSW index
|
|
65
|
+
this.index.addPoint(embedding, id);
|
|
66
|
+
this.memories.set(id, fullMemory);
|
|
67
|
+
return fullMemory;
|
|
68
|
+
}
|
|
69
|
+
async searchSimilar(query, k = 5) {
|
|
70
|
+
if (!this.index) {
|
|
71
|
+
throw new Error('VectorStore not initialized. Call initialize() first.');
|
|
72
|
+
}
|
|
73
|
+
const queryEmbedding = await this.generateEmbedding(query);
|
|
74
|
+
const results = this.index.searchKnn(queryEmbedding, k);
|
|
75
|
+
return results.neighbors.map((id, i) => ({
|
|
76
|
+
memory: this.memories.get(id),
|
|
77
|
+
score: results.distances[i],
|
|
78
|
+
}));
|
|
79
|
+
}
|
|
80
|
+
async searchByCategory(query, category, k = 5) {
|
|
81
|
+
const results = await this.searchSimilar(query, k * 2);
|
|
82
|
+
return results
|
|
83
|
+
.filter(r => r.memory.category === category)
|
|
84
|
+
.slice(0, k);
|
|
85
|
+
}
|
|
86
|
+
getMemoriesByChapter(chapterNumber) {
|
|
87
|
+
return Array.from(this.memories.values())
|
|
88
|
+
.filter(m => m.chapterNumber === chapterNumber);
|
|
89
|
+
}
|
|
90
|
+
getAllMemories() {
|
|
91
|
+
return Array.from(this.memories.values());
|
|
92
|
+
}
|
|
93
|
+
async generateEmbedding(text) {
|
|
94
|
+
// Check if we should use mock embeddings (for testing without OpenAI)
|
|
95
|
+
if (process.env.USE_MOCK_EMBEDDINGS === 'true') {
|
|
96
|
+
return this.generateMockEmbedding(text);
|
|
97
|
+
}
|
|
98
|
+
const llm = (0, client_js_1.getLLM)();
|
|
99
|
+
const openai = llm.provider;
|
|
100
|
+
// Access the underlying OpenAI client for embeddings
|
|
101
|
+
const client = openai.client;
|
|
102
|
+
try {
|
|
103
|
+
const response = await client.embeddings.create({
|
|
104
|
+
model: 'text-embedding-3-small',
|
|
105
|
+
input: text,
|
|
106
|
+
});
|
|
107
|
+
return response.data[0].embedding;
|
|
108
|
+
}
|
|
109
|
+
catch (error) {
|
|
110
|
+
// Fall back to mock embeddings if the API fails (e.g., DeepSeek doesn't support embeddings)
|
|
111
|
+
console.log(' [VectorStore] Using mock embeddings (API unavailable)');
|
|
112
|
+
return this.generateMockEmbedding(text);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
generateMockEmbedding(text) {
|
|
116
|
+
// Generate a deterministic mock embedding based on text content
|
|
117
|
+
// This is NOT for production - just for testing without OpenAI API
|
|
118
|
+
const embedding = [];
|
|
119
|
+
let seed = 0;
|
|
120
|
+
for (let i = 0; i < text.length; i++) {
|
|
121
|
+
seed += text.charCodeAt(i);
|
|
122
|
+
}
|
|
123
|
+
for (let i = 0; i < this.dimension; i++) {
|
|
124
|
+
// Simple pseudo-random based on seed
|
|
125
|
+
seed = (seed * 9301 + 49297) % 233280;
|
|
126
|
+
embedding.push((seed / 233280) * 2 - 1); // Range: -1 to 1
|
|
127
|
+
}
|
|
128
|
+
// Normalize the vector
|
|
129
|
+
const norm = Math.sqrt(embedding.reduce((sum, val) => sum + val * val, 0));
|
|
130
|
+
return embedding.map(val => val / norm);
|
|
131
|
+
}
|
|
132
|
+
// Serialize for persistence
|
|
133
|
+
serialize() {
|
|
134
|
+
return JSON.stringify({
|
|
135
|
+
storyId: this.storyId,
|
|
136
|
+
nextId: this.nextId,
|
|
137
|
+
memories: Array.from(this.memories.entries()),
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
// Deserialize from saved state
|
|
141
|
+
async load(data) {
|
|
142
|
+
const parsed = JSON.parse(data);
|
|
143
|
+
this.memories = new Map(parsed.memories);
|
|
144
|
+
this.nextId = parsed.nextId || this.memories.size;
|
|
145
|
+
// Rebuild HNSW index
|
|
146
|
+
await this.initialize();
|
|
147
|
+
for (const [id, memory] of this.memories) {
|
|
148
|
+
if (memory.embedding) {
|
|
149
|
+
this.index.addPoint(memory.embedding, id);
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
exports.VectorStore = VectorStore;
|
|
155
|
+
// Factory for story-specific stores
|
|
156
|
+
const stores = new Map();
|
|
157
|
+
function getVectorStore(storyId) {
|
|
158
|
+
if (!stores.has(storyId)) {
|
|
159
|
+
stores.set(storyId, new VectorStore(storyId));
|
|
160
|
+
}
|
|
161
|
+
return stores.get(storyId);
|
|
162
|
+
}
|
|
163
|
+
function clearVectorStore(storyId) {
|
|
164
|
+
stores.delete(storyId);
|
|
165
|
+
}
|
|
166
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vectorStore.js","sourceRoot":"","sources":["../../src/memory/vectorStore.ts"],"names":[],"mappings":"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","sourcesContent":["import { getLLM } from '../llm/client.js';\r\nimport { HierarchicalNSW } from 'hnswlib-node';\r\n\r\nexport interface NarrativeMemory {\r\n  id: number;\r\n  storyId: string;\r\n  chapterNumber: number;\r\n  content: string;\r\n  category: 'event' | 'character' | 'world' | 'plot';\r\n  timestamp: Date;\r\n  embedding?: number[];\r\n}\r\n\r\nexport interface MemorySearchResult {\r\n  memory: NarrativeMemory;\r\n  score: number;\r\n}\r\n\r\nexport class VectorStore {\r\n  private index: HierarchicalNSW | null = null;\r\n  private memories: Map<number, NarrativeMemory> = new Map();\r\n  private dimension: number = 1536; // text-embedding-3-small dimension\r\n  private storyId: string;\r\n  private nextId: number = 0;\r\n\r\n  constructor(storyId: string) {\r\n    this.storyId = storyId;\r\n  }\r\n\r\n  async initialize(maxElements: number = 10000): Promise<void> {\r\n    this.index = new HierarchicalNSW('cosine', this.dimension);\r\n    this.index.initIndex(maxElements, 16, 200);\r\n    this.nextId = 0;\r\n    this.memories.clear();\r\n  }\r\n\r\n  /**\r\n   * Resize the index to accommodate more memories\r\n   */\r\n  resizeIndex(newMaxElements: number): void {\r\n    if (!this.index) {\r\n      throw new Error('VectorStore not initialized. Call initialize() first.');\r\n    }\r\n    \r\n    if (newMaxElements <= this.memories.size) {\r\n      return; // No need to resize\r\n    }\r\n    \r\n    this.index.resizeIndex(newMaxElements);\r\n  }\r\n\r\n  /**\r\n   * Ensure capacity for upcoming memories\r\n   */\r\n  ensureCapacity(additionalMemories: number): void {\r\n    const requiredCapacity = this.memories.size + additionalMemories;\r\n    const currentCapacity = this.index?.getMaxElements() || 0;\r\n    \r\n    if (requiredCapacity > currentCapacity) {\r\n      // Add 50% more capacity or use required, whichever is larger\r\n      const newCapacity = Math.max(Math.floor(currentCapacity * 1.5), requiredCapacity);\r\n      this.resizeIndex(newCapacity);\r\n    }\r\n  }\r\n\r\n  async addMemory(memory: Omit<NarrativeMemory, 'id' | 'embedding'>): Promise<NarrativeMemory> {\r\n    if (!this.index) {\r\n      throw new Error('VectorStore not initialized. Call initialize() first.');\r\n    }\r\n\r\n    // Auto-resize if we're near capacity (add 50% more)\r\n    const currentCapacity = this.index.getMaxElements();\r\n    if (this.memories.size >= currentCapacity - 1) {\r\n      this.resizeIndex(Math.floor(currentCapacity * 1.5));\r\n    }\r\n\r\n    const id = this.nextId++;\r\n    \r\n    // Generate embedding using OpenAI\r\n    const embedding = await this.generateEmbedding(memory.content);\r\n    \r\n    const fullMemory: NarrativeMemory = {\r\n      ...memory,\r\n      id,\r\n      embedding,\r\n    };\r\n\r\n    // Add to HNSW index\r\n    this.index.addPoint(embedding, id);\r\n    this.memories.set(id, fullMemory);\r\n\r\n    return fullMemory;\r\n  }\r\n\r\n  async searchSimilar(query: string, k: number = 5): Promise<MemorySearchResult[]> {\r\n    if (!this.index) {\r\n      throw new Error('VectorStore not initialized. Call initialize() first.');\r\n    }\r\n\r\n    const queryEmbedding = await this.generateEmbedding(query);\r\n    const results = this.index.searchKnn(queryEmbedding, k);\r\n\r\n    return results.neighbors.map((id: number, i: number) => ({\r\n      memory: this.memories.get(id)!,\r\n      score: results.distances[i],\r\n    }));\r\n  }\r\n\r\n  async searchByCategory(query: string, category: NarrativeMemory['category'], k: number = 5): Promise<MemorySearchResult[]> {\r\n    const results = await this.searchSimilar(query, k * 2);\r\n    return results\r\n      .filter(r => r.memory.category === category)\r\n      .slice(0, k);\r\n  }\r\n\r\n  getMemoriesByChapter(chapterNumber: number): NarrativeMemory[] {\r\n    return Array.from(this.memories.values())\r\n      .filter(m => m.chapterNumber === chapterNumber);\r\n  }\r\n\r\n  getAllMemories(): NarrativeMemory[] {\r\n    return Array.from(this.memories.values());\r\n  }\r\n\r\n  private async generateEmbedding(text: string): Promise<number[]> {\r\n    // Check if we should use mock embeddings (for testing without OpenAI)\r\n    if (process.env.USE_MOCK_EMBEDDINGS === 'true') {\r\n      return this.generateMockEmbedding(text);\r\n    }\r\n\r\n    const llm = getLLM();\r\n    const openai = (llm as any).provider;\r\n    \r\n    // Access the underlying OpenAI client for embeddings\r\n    const client = (openai as any).client;\r\n    \r\n    try {\r\n      const response = await client.embeddings.create({\r\n        model: 'text-embedding-3-small',\r\n        input: text,\r\n      });\r\n      return response.data[0].embedding;\r\n    } catch (error) {\r\n      // Fall back to mock embeddings if the API fails (e.g., DeepSeek doesn't support embeddings)\r\n      console.log('  [VectorStore] Using mock embeddings (API unavailable)');\r\n      return this.generateMockEmbedding(text);\r\n    }\r\n  }\r\n\r\n  private generateMockEmbedding(text: string): number[] {\r\n    // Generate a deterministic mock embedding based on text content\r\n    // This is NOT for production - just for testing without OpenAI API\r\n    const embedding: number[] = [];\r\n    let seed = 0;\r\n    for (let i = 0; i < text.length; i++) {\r\n      seed += text.charCodeAt(i);\r\n    }\r\n    \r\n    for (let i = 0; i < this.dimension; i++) {\r\n      // Simple pseudo-random based on seed\r\n      seed = (seed * 9301 + 49297) % 233280;\r\n      embedding.push((seed / 233280) * 2 - 1); // Range: -1 to 1\r\n    }\r\n    \r\n    // Normalize the vector\r\n    const norm = Math.sqrt(embedding.reduce((sum, val) => sum + val * val, 0));\r\n    return embedding.map(val => val / norm);\r\n  }\r\n\r\n  // Serialize for persistence\r\n  serialize(): string {\r\n    return JSON.stringify({\r\n      storyId: this.storyId,\r\n      nextId: this.nextId,\r\n      memories: Array.from(this.memories.entries()),\r\n    });\r\n  }\r\n\r\n  // Deserialize from saved state\r\n  async load(data: string): Promise<void> {\r\n    const parsed = JSON.parse(data);\r\n    this.memories = new Map(parsed.memories);\r\n    this.nextId = parsed.nextId || this.memories.size;\r\n    \r\n    // Rebuild HNSW index\r\n    await this.initialize();\r\n    for (const [id, memory] of this.memories) {\r\n      if (memory.embedding) {\r\n        this.index!.addPoint(memory.embedding, id);\r\n      }\r\n    }\r\n  }\r\n}\r\n\r\n// Factory for story-specific stores\r\nconst stores: Map<string, VectorStore> = new Map();\r\n\r\nexport function getVectorStore(storyId: string): VectorStore {\r\n  if (!stores.has(storyId)) {\r\n    stores.set(storyId, new VectorStore(storyId));\r\n  }\r\n  return stores.get(storyId)!;\r\n}\r\n\r\nexport function clearVectorStore(storyId: string): void {\r\n  stores.delete(storyId);\r\n}\r\n"]}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import type { GenerationContext, Chapter, ChapterSummary } from '../types/index.js';
|
|
2
|
+
import type { CanonStore } from '../memory/canonStore.js';
|
|
3
|
+
import type { VectorStore } from '../memory/vectorStore.js';
|
|
4
|
+
export interface GenerateChapterResult {
|
|
5
|
+
chapter: Chapter;
|
|
6
|
+
summary: ChapterSummary;
|
|
7
|
+
violations: string[];
|
|
8
|
+
memoriesExtracted: number;
|
|
9
|
+
}
|
|
10
|
+
export interface GenerateChapterOptions {
|
|
11
|
+
canon?: CanonStore;
|
|
12
|
+
vectorStore?: VectorStore;
|
|
13
|
+
validateCanon?: boolean;
|
|
14
|
+
maxContinuationAttempts?: number;
|
|
15
|
+
retrieveMemories?: boolean;
|
|
16
|
+
}
|
|
17
|
+
export declare function generateChapter(context: GenerationContext, options?: GenerateChapterOptions): Promise<GenerateChapterResult>;
|