@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,110 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.createCanonStore = createCanonStore;
4
+ exports.extractCanonFromBible = extractCanonFromBible;
5
+ exports.addFact = addFact;
6
+ exports.getFactsByCategory = getFactsByCategory;
7
+ exports.getFact = getFact;
8
+ exports.updateFact = updateFact;
9
+ exports.formatCanonForPrompt = formatCanonForPrompt;
10
+ function createCanonStore(storyId) {
11
+ return {
12
+ storyId,
13
+ facts: [],
14
+ };
15
+ }
16
+ function extractCanonFromBible(bible) {
17
+ const store = createCanonStore(bible.id);
18
+ for (const char of bible.characters) {
19
+ addFact(store, {
20
+ category: 'character',
21
+ subject: char.name,
22
+ attribute: 'role',
23
+ value: char.role,
24
+ chapterEstablished: 0,
25
+ });
26
+ if (char.background) {
27
+ addFact(store, {
28
+ category: 'character',
29
+ subject: char.name,
30
+ attribute: 'background',
31
+ value: char.background,
32
+ chapterEstablished: 0,
33
+ });
34
+ }
35
+ }
36
+ for (const thread of bible.plotThreads) {
37
+ addFact(store, {
38
+ category: 'plot',
39
+ subject: thread.name,
40
+ attribute: 'status',
41
+ value: thread.status,
42
+ chapterEstablished: 0,
43
+ });
44
+ }
45
+ return store;
46
+ }
47
+ function addFact(store, fact) {
48
+ const newFact = {
49
+ ...fact,
50
+ id: generateId(),
51
+ };
52
+ return {
53
+ ...store,
54
+ facts: [...store.facts, newFact],
55
+ };
56
+ }
57
+ function getFactsByCategory(store, category) {
58
+ return store.facts.filter(f => f.category === category);
59
+ }
60
+ function getFact(store, subject, attribute) {
61
+ return store.facts.find(f => f.subject === subject && f.attribute === attribute);
62
+ }
63
+ function updateFact(store, subject, attribute, value, chapter) {
64
+ const existingIndex = store.facts.findIndex(f => f.subject === subject && f.attribute === attribute);
65
+ if (existingIndex >= 0) {
66
+ const updatedFacts = [...store.facts];
67
+ updatedFacts[existingIndex] = {
68
+ ...updatedFacts[existingIndex],
69
+ value,
70
+ chapterEstablished: chapter,
71
+ };
72
+ return { ...store, facts: updatedFacts };
73
+ }
74
+ return addFact(store, {
75
+ category: 'plot',
76
+ subject,
77
+ attribute,
78
+ value,
79
+ chapterEstablished: chapter,
80
+ });
81
+ }
82
+ function formatCanonForPrompt(store) {
83
+ const lines = ['## Story Canon (NEVER contradict these facts)'];
84
+ const characters = getFactsByCategory(store, 'character');
85
+ if (characters.length > 0) {
86
+ lines.push('\n### Characters');
87
+ for (const fact of characters) {
88
+ lines.push(`- ${fact.subject}: ${fact.attribute} = ${fact.value}`);
89
+ }
90
+ }
91
+ const world = getFactsByCategory(store, 'world');
92
+ if (world.length > 0) {
93
+ lines.push('\n### World');
94
+ for (const fact of world) {
95
+ lines.push(`- ${fact.subject}: ${fact.attribute} = ${fact.value}`);
96
+ }
97
+ }
98
+ const plot = getFactsByCategory(store, 'plot');
99
+ if (plot.length > 0) {
100
+ lines.push('\n### Plot');
101
+ for (const fact of plot) {
102
+ lines.push(`- ${fact.subject}: ${fact.attribute} = ${fact.value}`);
103
+ }
104
+ }
105
+ return lines.join('\n');
106
+ }
107
+ function generateId() {
108
+ return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;
109
+ }
110
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"canonStore.js","sourceRoot":"","sources":["../../src/memory/canonStore.ts"],"names":[],"mappings":"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","sourcesContent":["import type { StoryBible } from '../types/index.js';\r\n\r\nexport interface CanonFact {\r\n  id: string;\r\n  category: 'character' | 'world' | 'plot' | 'timeline';\r\n  subject: string;\r\n  attribute: string;\r\n  value: string;\r\n  chapterEstablished: number;\r\n}\r\n\r\nexport interface CanonStore {\r\n  storyId: string;\r\n  facts: CanonFact[];\r\n}\r\n\r\nexport function createCanonStore(storyId: string): CanonStore {\r\n  return {\r\n    storyId,\r\n    facts: [],\r\n  };\r\n}\r\n\r\nexport function extractCanonFromBible(bible: StoryBible): CanonStore {\r\n  const store = createCanonStore(bible.id);\r\n\r\n  for (const char of bible.characters) {\r\n    addFact(store, {\r\n      category: 'character',\r\n      subject: char.name,\r\n      attribute: 'role',\r\n      value: char.role,\r\n      chapterEstablished: 0,\r\n    });\r\n\r\n    if (char.background) {\r\n      addFact(store, {\r\n        category: 'character',\r\n        subject: char.name,\r\n        attribute: 'background',\r\n        value: char.background,\r\n        chapterEstablished: 0,\r\n      });\r\n    }\r\n  }\r\n\r\n  for (const thread of bible.plotThreads) {\r\n    addFact(store, {\r\n      category: 'plot',\r\n      subject: thread.name,\r\n      attribute: 'status',\r\n      value: thread.status,\r\n      chapterEstablished: 0,\r\n    });\r\n  }\r\n\r\n  return store;\r\n}\r\n\r\nexport function addFact(store: CanonStore, fact: Omit<CanonFact, 'id'>): CanonStore {\r\n  const newFact: CanonFact = {\r\n    ...fact,\r\n    id: generateId(),\r\n  };\r\n  return {\r\n    ...store,\r\n    facts: [...store.facts, newFact],\r\n  };\r\n}\r\n\r\nexport function getFactsByCategory(store: CanonStore, category: CanonFact['category']): CanonFact[] {\r\n  return store.facts.filter(f => f.category === category);\r\n}\r\n\r\nexport function getFact(store: CanonStore, subject: string, attribute: string): CanonFact | undefined {\r\n  return store.facts.find(f => f.subject === subject && f.attribute === attribute);\r\n}\r\n\r\nexport function updateFact(store: CanonStore, subject: string, attribute: string, value: string, chapter: number): CanonStore {\r\n  const existingIndex = store.facts.findIndex(f => f.subject === subject && f.attribute === attribute);\r\n  \r\n  if (existingIndex >= 0) {\r\n    const updatedFacts = [...store.facts];\r\n    updatedFacts[existingIndex] = {\r\n      ...updatedFacts[existingIndex],\r\n      value,\r\n      chapterEstablished: chapter,\r\n    };\r\n    return { ...store, facts: updatedFacts };\r\n  }\r\n  \r\n  return addFact(store, {\r\n    category: 'plot',\r\n    subject,\r\n    attribute,\r\n    value,\r\n    chapterEstablished: chapter,\r\n  });\r\n}\r\n\r\nexport function formatCanonForPrompt(store: CanonStore): string {\r\n  const lines: string[] = ['## Story Canon (NEVER contradict these facts)'];\r\n  \r\n  const characters = getFactsByCategory(store, 'character');\r\n  if (characters.length > 0) {\r\n    lines.push('\\n### Characters');\r\n    for (const fact of characters) {\r\n      lines.push(`- ${fact.subject}: ${fact.attribute} = ${fact.value}`);\r\n    }\r\n  }\r\n  \r\n  const world = getFactsByCategory(store, 'world');\r\n  if (world.length > 0) {\r\n    lines.push('\\n### World');\r\n    for (const fact of world) {\r\n      lines.push(`- ${fact.subject}: ${fact.attribute} = ${fact.value}`);\r\n    }\r\n  }\r\n  \r\n  const plot = getFactsByCategory(store, 'plot');\r\n  if (plot.length > 0) {\r\n    lines.push('\\n### Plot');\r\n    for (const fact of plot) {\r\n      lines.push(`- ${fact.subject}: ${fact.attribute} = ${fact.value}`);\r\n    }\r\n  }\r\n  \r\n  return lines.join('\\n');\r\n}\r\n\r\nfunction generateId(): string {\r\n  return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;\r\n}\r\n"]}
@@ -0,0 +1,28 @@
1
+ import { VectorStore, NarrativeMemory } from './vectorStore.js';
2
+ import type { StoryBible, StoryState } from '../types/index.js';
3
+ export interface RetrievalContext {
4
+ bible: StoryBible;
5
+ state: StoryState;
6
+ currentChapter: number;
7
+ query?: string;
8
+ }
9
+ export interface RetrievedMemory {
10
+ memory: NarrativeMemory;
11
+ relevance: number;
12
+ reason: string;
13
+ }
14
+ export declare class MemoryRetriever {
15
+ private vectorStore;
16
+ constructor(vectorStore: VectorStore);
17
+ retrieveForChapter(context: RetrievalContext, k?: number): Promise<RetrievedMemory[]>;
18
+ retrieveForCharacter(characterName: string, context: string, k?: number): Promise<RetrievedMemory[]>;
19
+ retrieveForPlotThread(plotThreadId: string, bible: StoryBible, k?: number): Promise<RetrievedMemory[]>;
20
+ retrieveRelevantEvents(query: string, k?: number): Promise<RetrievedMemory[]>;
21
+ formatMemoriesForPrompt(memories: RetrievedMemory[]): string;
22
+ private generateContextualQuery;
23
+ private rerankMemories;
24
+ private inferRelevanceReason;
25
+ private groupByCategory;
26
+ private capitalize;
27
+ }
28
+ export declare function createMemoryRetriever(vectorStore: VectorStore): MemoryRetriever;
@@ -0,0 +1,126 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MemoryRetriever = void 0;
4
+ exports.createMemoryRetriever = createMemoryRetriever;
5
+ class MemoryRetriever {
6
+ vectorStore;
7
+ constructor(vectorStore) {
8
+ this.vectorStore = vectorStore;
9
+ }
10
+ async retrieveForChapter(context, k = 5) {
11
+ const { bible, state, currentChapter, query } = context;
12
+ // Generate a contextual query if none provided
13
+ const searchQuery = query || this.generateContextualQuery(bible, state, currentChapter);
14
+ // Search for similar memories
15
+ const results = await this.vectorStore.searchSimilar(searchQuery, k * 2);
16
+ // Filter out memories from the current chapter (can't recall what hasn't happened yet)
17
+ const pastMemories = results.filter(r => r.memory.chapterNumber < currentChapter);
18
+ // Re-rank and filter for relevance
19
+ const ranked = await this.rerankMemories(searchQuery, pastMemories, k);
20
+ return ranked;
21
+ }
22
+ async retrieveForCharacter(characterName, context, k = 3) {
23
+ const query = `Character ${characterName}: ${context}`;
24
+ const results = await this.vectorStore.searchByCategory(query, 'character', k * 2);
25
+ // Filter for memories actually about this character
26
+ const characterMemories = results.filter(r => r.memory.content.toLowerCase().includes(characterName.toLowerCase()));
27
+ return characterMemories.slice(0, k).map(r => ({
28
+ memory: r.memory,
29
+ relevance: r.score,
30
+ reason: 'Character-relevant memory',
31
+ }));
32
+ }
33
+ async retrieveForPlotThread(plotThreadId, bible, k = 3) {
34
+ const plotThread = bible.plotThreads.find(p => p.id === plotThreadId);
35
+ if (!plotThread)
36
+ return [];
37
+ const query = `Plot thread "${plotThread.name}": ${plotThread.description}`;
38
+ const results = await this.vectorStore.searchByCategory(query, 'plot', k * 2);
39
+ return results.slice(0, k).map(r => ({
40
+ memory: r.memory,
41
+ relevance: r.score,
42
+ reason: `Relevant to plot thread: ${plotThread.name}`,
43
+ }));
44
+ }
45
+ async retrieveRelevantEvents(query, k = 5) {
46
+ const results = await this.vectorStore.searchByCategory(query, 'event', k);
47
+ return results.map(r => ({
48
+ memory: r.memory,
49
+ relevance: r.score,
50
+ reason: 'Relevant past event',
51
+ }));
52
+ }
53
+ formatMemoriesForPrompt(memories) {
54
+ if (memories.length === 0) {
55
+ return '';
56
+ }
57
+ const sections = [];
58
+ // Group by category
59
+ const byCategory = this.groupByCategory(memories);
60
+ for (const [category, items] of Object.entries(byCategory)) {
61
+ if (items.length > 0) {
62
+ sections.push(`### ${this.capitalize(category)} Memories\n${items.map(m => `- ${m.memory.content} (Ch. ${m.memory.chapterNumber})`).join('\n')}`);
63
+ }
64
+ }
65
+ return `## Relevant Past Memories\n\n${sections.join('\n\n')}`;
66
+ }
67
+ generateContextualQuery(bible, state, currentChapter) {
68
+ const progress = currentChapter / bible.targetChapters;
69
+ const activeThreads = bible.plotThreads
70
+ .filter(p => state.activePlotThreads.includes(p.id))
71
+ .map(p => p.name)
72
+ .join(', ');
73
+ return `Chapter ${currentChapter} of ${bible.title}. Genre: ${bible.genre}. ` +
74
+ `Story progress: ${Math.round(progress * 100)}%. ` +
75
+ `Active plot threads: ${activeThreads}. ` +
76
+ `Premise: ${bible.premise.substring(0, 200)}`;
77
+ }
78
+ async rerankMemories(query, candidates, topK) {
79
+ // Simple re-ranking: use the vector similarity score directly
80
+ // In a more sophisticated version, we could use an LLM to judge relevance
81
+ return candidates
82
+ .slice(0, topK)
83
+ .map(r => ({
84
+ memory: r.memory,
85
+ relevance: r.score,
86
+ reason: this.inferRelevanceReason(r.memory),
87
+ }));
88
+ }
89
+ inferRelevanceReason(memory) {
90
+ switch (memory.category) {
91
+ case 'event':
92
+ return 'Relevant past event';
93
+ case 'character':
94
+ return 'Character background or development';
95
+ case 'world':
96
+ return 'World-building detail';
97
+ case 'plot':
98
+ return 'Plot-relevant information';
99
+ default:
100
+ return 'Relevant memory';
101
+ }
102
+ }
103
+ groupByCategory(memories) {
104
+ const grouped = {
105
+ event: [],
106
+ character: [],
107
+ world: [],
108
+ plot: [],
109
+ };
110
+ for (const memory of memories) {
111
+ const cat = memory.memory.category;
112
+ if (!grouped[cat])
113
+ grouped[cat] = [];
114
+ grouped[cat].push(memory);
115
+ }
116
+ return grouped;
117
+ }
118
+ capitalize(str) {
119
+ return str.charAt(0).toUpperCase() + str.slice(1);
120
+ }
121
+ }
122
+ exports.MemoryRetriever = MemoryRetriever;
123
+ function createMemoryRetriever(vectorStore) {
124
+ return new MemoryRetriever(vectorStore);
125
+ }
126
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"memoryRetriever.js","sourceRoot":"","sources":["../../src/memory/memoryRetriever.ts"],"names":[],"mappings":"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","sourcesContent":["import { VectorStore, NarrativeMemory, MemorySearchResult } from './vectorStore.js';\r\nimport { getLLM } from '../llm/client.js';\r\nimport type { StoryBible, StoryState } from '../types/index.js';\r\n\r\nexport interface RetrievalContext {\r\n  bible: StoryBible;\r\n  state: StoryState;\r\n  currentChapter: number;\r\n  query?: string;\r\n}\r\n\r\nexport interface RetrievedMemory {\r\n  memory: NarrativeMemory;\r\n  relevance: number;\r\n  reason: string;\r\n}\r\n\r\nexport class MemoryRetriever {\r\n  private vectorStore: VectorStore;\r\n\r\n  constructor(vectorStore: VectorStore) {\r\n    this.vectorStore = vectorStore;\r\n  }\r\n\r\n  async retrieveForChapter(context: RetrievalContext, k: number = 5): Promise<RetrievedMemory[]> {\r\n    const { bible, state, currentChapter, query } = context;\r\n\r\n    // Generate a contextual query if none provided\r\n    const searchQuery = query || this.generateContextualQuery(bible, state, currentChapter);\r\n\r\n    // Search for similar memories\r\n    const results = await this.vectorStore.searchSimilar(searchQuery, k * 2);\r\n\r\n    // Filter out memories from the current chapter (can't recall what hasn't happened yet)\r\n    const pastMemories = results.filter(r => r.memory.chapterNumber < currentChapter);\r\n\r\n    // Re-rank and filter for relevance\r\n    const ranked = await this.rerankMemories(searchQuery, pastMemories, k);\r\n\r\n    return ranked;\r\n  }\r\n\r\n  async retrieveForCharacter(characterName: string, context: string, k: number = 3): Promise<RetrievedMemory[]> {\r\n    const query = `Character ${characterName}: ${context}`;\r\n    \r\n    const results = await this.vectorStore.searchByCategory(query, 'character', k * 2);\r\n    \r\n    // Filter for memories actually about this character\r\n    const characterMemories = results.filter(r => \r\n      r.memory.content.toLowerCase().includes(characterName.toLowerCase())\r\n    );\r\n\r\n    return characterMemories.slice(0, k).map(r => ({\r\n      memory: r.memory,\r\n      relevance: r.score,\r\n      reason: 'Character-relevant memory',\r\n    }));\r\n  }\r\n\r\n  async retrieveForPlotThread(plotThreadId: string, bible: StoryBible, k: number = 3): Promise<RetrievedMemory[]> {\r\n    const plotThread = bible.plotThreads.find(p => p.id === plotThreadId);\r\n    if (!plotThread) return [];\r\n\r\n    const query = `Plot thread \"${plotThread.name}\": ${plotThread.description}`;\r\n    \r\n    const results = await this.vectorStore.searchByCategory(query, 'plot', k * 2);\r\n\r\n    return results.slice(0, k).map(r => ({\r\n      memory: r.memory,\r\n      relevance: r.score,\r\n      reason: `Relevant to plot thread: ${plotThread.name}`,\r\n    }));\r\n  }\r\n\r\n  async retrieveRelevantEvents(query: string, k: number = 5): Promise<RetrievedMemory[]> {\r\n    const results = await this.vectorStore.searchByCategory(query, 'event', k);\r\n\r\n    return results.map(r => ({\r\n      memory: r.memory,\r\n      relevance: r.score,\r\n      reason: 'Relevant past event',\r\n    }));\r\n  }\r\n\r\n  formatMemoriesForPrompt(memories: RetrievedMemory[]): string {\r\n    if (memories.length === 0) {\r\n      return '';\r\n    }\r\n\r\n    const sections: string[] = [];\r\n    \r\n    // Group by category\r\n    const byCategory = this.groupByCategory(memories);\r\n\r\n    for (const [category, items] of Object.entries(byCategory)) {\r\n      if (items.length > 0) {\r\n        sections.push(`### ${this.capitalize(category)} Memories\\n${items.map(m => `- ${m.memory.content} (Ch. ${m.memory.chapterNumber})`).join('\\n')}`);\r\n      }\r\n    }\r\n\r\n    return `## Relevant Past Memories\\n\\n${sections.join('\\n\\n')}`;\r\n  }\r\n\r\n  private generateContextualQuery(bible: StoryBible, state: StoryState, currentChapter: number): string {\r\n    const progress = currentChapter / bible.targetChapters;\r\n    const activeThreads = bible.plotThreads\r\n      .filter(p => state.activePlotThreads.includes(p.id))\r\n      .map(p => p.name)\r\n      .join(', ');\r\n\r\n    return `Chapter ${currentChapter} of ${bible.title}. Genre: ${bible.genre}. ` +\r\n           `Story progress: ${Math.round(progress * 100)}%. ` +\r\n           `Active plot threads: ${activeThreads}. ` +\r\n           `Premise: ${bible.premise.substring(0, 200)}`;\r\n  }\r\n\r\n  private async rerankMemories(\r\n    query: string, \r\n    candidates: MemorySearchResult[], \r\n    topK: number\r\n  ): Promise<RetrievedMemory[]> {\r\n    // Simple re-ranking: use the vector similarity score directly\r\n    // In a more sophisticated version, we could use an LLM to judge relevance\r\n    \r\n    return candidates\r\n      .slice(0, topK)\r\n      .map(r => ({\r\n        memory: r.memory,\r\n        relevance: r.score,\r\n        reason: this.inferRelevanceReason(r.memory),\r\n      }));\r\n  }\r\n\r\n  private inferRelevanceReason(memory: NarrativeMemory): string {\r\n    switch (memory.category) {\r\n      case 'event':\r\n        return 'Relevant past event';\r\n      case 'character':\r\n        return 'Character background or development';\r\n      case 'world':\r\n        return 'World-building detail';\r\n      case 'plot':\r\n        return 'Plot-relevant information';\r\n      default:\r\n        return 'Relevant memory';\r\n    }\r\n  }\r\n\r\n  private groupByCategory(memories: RetrievedMemory[]): Record<string, RetrievedMemory[]> {\r\n    const grouped: Record<string, RetrievedMemory[]> = {\r\n      event: [],\r\n      character: [],\r\n      world: [],\r\n      plot: [],\r\n    };\r\n\r\n    for (const memory of memories) {\r\n      const cat = memory.memory.category;\r\n      if (!grouped[cat]) grouped[cat] = [];\r\n      grouped[cat].push(memory);\r\n    }\r\n\r\n    return grouped;\r\n  }\r\n\r\n  private capitalize(str: string): string {\r\n    return str.charAt(0).toUpperCase() + str.slice(1);\r\n  }\r\n}\r\n\r\nexport function createMemoryRetriever(vectorStore: VectorStore): MemoryRetriever {\r\n  return new MemoryRetriever(vectorStore);\r\n}\r\n"]}
@@ -0,0 +1,49 @@
1
+ import type { Chapter, StoryBible } from '../types/index.js';
2
+ import type { StoryStructuredState } from '../story/structuredState.js';
3
+ import type { VectorStore } from './vectorStore.js';
4
+ import type { CanonStore } from './canonStore.js';
5
+ import { ConstraintGraph } from '../constraints/constraintGraph.js';
6
+ export interface StateUpdateResult {
7
+ structuredState: StoryStructuredState;
8
+ memoriesAdded: number;
9
+ canonFactsAdded: number;
10
+ graphUpdated: boolean;
11
+ changes: StateChange[];
12
+ }
13
+ export interface StateChange {
14
+ type: 'character' | 'plot' | 'world' | 'canon' | 'memory';
15
+ description: string;
16
+ chapter: number;
17
+ }
18
+ export interface UpdateContext {
19
+ chapter: Chapter;
20
+ bible: StoryBible;
21
+ currentState: StoryStructuredState;
22
+ canon: CanonStore;
23
+ vectorStore: VectorStore;
24
+ constraintGraph: ConstraintGraph;
25
+ }
26
+ export declare class StateUpdaterPipeline {
27
+ /**
28
+ * Run the complete post-chapter update pipeline
29
+ */
30
+ update(context: UpdateContext): Promise<StateUpdateResult>;
31
+ /**
32
+ * Extract state changes from chapter
33
+ */
34
+ private extractChanges;
35
+ /**
36
+ * Extract narrative memories from chapter
37
+ */
38
+ private extractMemories;
39
+ /**
40
+ * Quick update without LLM (for testing)
41
+ */
42
+ quickUpdate(context: UpdateContext): Promise<StateUpdateResult>;
43
+ /**
44
+ * Format update result
45
+ */
46
+ formatResult(result: StateUpdateResult): string;
47
+ private sanitizeId;
48
+ }
49
+ export declare const stateUpdaterPipeline: StateUpdaterPipeline;