@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.
Files changed (71) hide show
  1. package/dist/agents/canonValidator.d.ts +9 -0
  2. package/dist/agents/canonValidator.js +51 -0
  3. package/dist/agents/chapterPlanner.d.ts +50 -0
  4. package/dist/agents/chapterPlanner.js +250 -0
  5. package/dist/agents/completeness.d.ts +7 -0
  6. package/dist/agents/completeness.js +51 -0
  7. package/dist/agents/memoryExtractor.d.ts +12 -0
  8. package/dist/agents/memoryExtractor.js +82 -0
  9. package/dist/agents/stateUpdater.d.ts +30 -0
  10. package/dist/agents/stateUpdater.js +150 -0
  11. package/dist/agents/storyDirector.d.ts +40 -0
  12. package/dist/agents/storyDirector.js +213 -0
  13. package/dist/agents/summarizer.d.ts +8 -0
  14. package/dist/agents/summarizer.js +56 -0
  15. package/dist/agents/tensionController.d.ts +68 -0
  16. package/dist/agents/tensionController.js +197 -0
  17. package/dist/agents/writer.d.ts +12 -0
  18. package/dist/agents/writer.js +148 -0
  19. package/dist/constraints/constraintGraph.d.ts +117 -0
  20. package/dist/constraints/constraintGraph.js +381 -0
  21. package/dist/constraints/validator.d.ts +58 -0
  22. package/dist/constraints/validator.js +236 -0
  23. package/dist/index.d.ts +25 -0
  24. package/dist/index.js +115 -0
  25. package/dist/llm/client.d.ts +14 -0
  26. package/dist/llm/client.js +108 -0
  27. package/dist/memory/canonStore.d.ts +20 -0
  28. package/dist/memory/canonStore.js +110 -0
  29. package/dist/memory/memoryRetriever.d.ts +28 -0
  30. package/dist/memory/memoryRetriever.js +126 -0
  31. package/dist/memory/stateUpdater.d.ts +49 -0
  32. package/dist/memory/stateUpdater.js +315 -0
  33. package/dist/memory/vectorStore.d.ts +41 -0
  34. package/dist/memory/vectorStore.js +166 -0
  35. package/dist/pipeline/generateChapter.d.ts +17 -0
  36. package/dist/pipeline/generateChapter.js +75 -0
  37. package/dist/story/bible.d.ts +4 -0
  38. package/dist/story/bible.js +53 -0
  39. package/dist/story/state.d.ts +3 -0
  40. package/dist/story/state.js +27 -0
  41. package/dist/story/structuredState.d.ts +39 -0
  42. package/dist/story/structuredState.js +159 -0
  43. package/dist/test/canon.test.d.ts +1 -0
  44. package/dist/test/canon.test.js +104 -0
  45. package/dist/test/chapter-planner.test.d.ts +1 -0
  46. package/dist/test/chapter-planner.test.js +171 -0
  47. package/dist/test/constraints.test.d.ts +1 -0
  48. package/dist/test/constraints.test.js +210 -0
  49. package/dist/test/simple.test.d.ts +1 -0
  50. package/dist/test/simple.test.js +51 -0
  51. package/dist/test/state-updater.test.d.ts +1 -0
  52. package/dist/test/state-updater.test.js +200 -0
  53. package/dist/test/story-director.test.d.ts +1 -0
  54. package/dist/test/story-director.test.js +142 -0
  55. package/dist/test/structured-state.test.d.ts +1 -0
  56. package/dist/test/structured-state.test.js +144 -0
  57. package/dist/test/tension-controller.test.d.ts +1 -0
  58. package/dist/test/tension-controller.test.js +116 -0
  59. package/dist/test/vector-memory.test.d.ts +1 -0
  60. package/dist/test/vector-memory.test.js +153 -0
  61. package/dist/test/world-simulation.test.d.ts +1 -0
  62. package/dist/test/world-simulation.test.js +152 -0
  63. package/dist/types/index.d.ts +79 -0
  64. package/dist/types/index.js +3 -0
  65. package/dist/world/characterAgent.d.ts +73 -0
  66. package/dist/world/characterAgent.js +232 -0
  67. package/dist/world/eventResolver.d.ts +52 -0
  68. package/dist/world/eventResolver.js +205 -0
  69. package/dist/world/worldState.d.ts +93 -0
  70. package/dist/world/worldState.js +258 -0
  71. 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":"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","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>;