@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,9 @@
1
+ import type { CanonStore } from '../memory/canonStore.js';
2
+ export interface CanonValidationResult {
3
+ valid: boolean;
4
+ violations: string[];
5
+ }
6
+ export declare class CanonValidator {
7
+ validate(chapterText: string, canon: CanonStore): Promise<CanonValidationResult>;
8
+ }
9
+ export declare const canonValidator: CanonValidator;
@@ -0,0 +1,51 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.canonValidator = exports.CanonValidator = void 0;
4
+ const client_js_1 = require("../llm/client.js");
5
+ const VALIDATOR_PROMPT = `Check the following chapter against the Story Canon. Identify any contradictions.
6
+
7
+ ## Story Canon
8
+ {{canon}}
9
+
10
+ ## Chapter Text
11
+ {{chapterText}}
12
+
13
+ ## Instructions
14
+ List any contradictions between the chapter and the canon facts. A contradiction occurs when:
15
+ - A character's status/background is different from canon
16
+ - A plot thread status is contradicted
17
+ - World rules are violated
18
+
19
+ Return JSON:
20
+ {
21
+ "valid": true/false,
22
+ "violations": ["description of contradiction 1", "description of contradiction 2"]
23
+ }
24
+
25
+ If no contradictions, return {"valid": true, "violations": []}`;
26
+ class CanonValidator {
27
+ async validate(chapterText, canon) {
28
+ if (canon.facts.length === 0) {
29
+ return { valid: true, violations: [] };
30
+ }
31
+ const { formatCanonForPrompt } = await import('../memory/canonStore.js');
32
+ const canonSection = formatCanonForPrompt(canon);
33
+ const prompt = VALIDATOR_PROMPT
34
+ .replace('{{canon}}', canonSection)
35
+ .replace('{{chapterText}}', chapterText.substring(0, 3000));
36
+ const response = await (0, client_js_1.getLLM)().complete(prompt, {
37
+ temperature: 0.1,
38
+ maxTokens: 500,
39
+ });
40
+ try {
41
+ const result = JSON.parse(response);
42
+ return result;
43
+ }
44
+ catch {
45
+ return { valid: true, violations: [] };
46
+ }
47
+ }
48
+ }
49
+ exports.CanonValidator = CanonValidator;
50
+ exports.canonValidator = new CanonValidator();
51
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2Fub25WYWxpZGF0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYWdlbnRzL2Nhbm9uVmFsaWRhdG9yLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLGdEQUEwQztBQVExQyxNQUFNLGdCQUFnQixHQUFHOzs7Ozs7Ozs7Ozs7Ozs7Ozs7OzsrREFvQnNDLENBQUM7QUFFaEUsTUFBYSxjQUFjO0lBQ3pCLEtBQUssQ0FBQyxRQUFRLENBQUMsV0FBbUIsRUFBRSxLQUFpQjtRQUNuRCxJQUFJLEtBQUssQ0FBQyxLQUFLLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQzdCLE9BQU8sRUFBRSxLQUFLLEVBQUUsSUFBSSxFQUFFLFVBQVUsRUFBRSxFQUFFLEVBQUUsQ0FBQztRQUN6QyxDQUFDO1FBRUQsTUFBTSxFQUFFLG9CQUFvQixFQUFFLEdBQUcsTUFBTSxNQUFNLENBQUMseUJBQXlCLENBQUMsQ0FBQztRQUN6RSxNQUFNLFlBQVksR0FBRyxvQkFBb0IsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUVqRCxNQUFNLE1BQU0sR0FBRyxnQkFBZ0I7YUFDNUIsT0FBTyxDQUFDLFdBQVcsRUFBRSxZQUFZLENBQUM7YUFDbEMsT0FBTyxDQUFDLGlCQUFpQixFQUFFLFdBQVcsQ0FBQyxTQUFTLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxDQUFDLENBQUM7UUFFOUQsTUFBTSxRQUFRLEdBQUcsTUFBTSxJQUFBLGtCQUFNLEdBQUUsQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFO1lBQy9DLFdBQVcsRUFBRSxHQUFHO1lBQ2hCLFNBQVMsRUFBRSxHQUFHO1NBQ2YsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDO1lBQ0gsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQTBCLENBQUM7WUFDN0QsT0FBTyxNQUFNLENBQUM7UUFDaEIsQ0FBQztRQUFDLE1BQU0sQ0FBQztZQUNQLE9BQU8sRUFBRSxLQUFLLEVBQUUsSUFBSSxFQUFFLFVBQVUsRUFBRSxFQUFFLEVBQUUsQ0FBQztRQUN6QyxDQUFDO0lBQ0gsQ0FBQztDQUNGO0FBekJELHdDQXlCQztBQUVZLFFBQUEsY0FBYyxHQUFHLElBQUksY0FBYyxFQUFFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBnZXRMTE0gfSBmcm9tICcuLi9sbG0vY2xpZW50LmpzJztcclxuaW1wb3J0IHR5cGUgeyBDYW5vblN0b3JlIH0gZnJvbSAnLi4vbWVtb3J5L2Nhbm9uU3RvcmUuanMnO1xyXG5cclxuZXhwb3J0IGludGVyZmFjZSBDYW5vblZhbGlkYXRpb25SZXN1bHQge1xyXG4gIHZhbGlkOiBib29sZWFuO1xyXG4gIHZpb2xhdGlvbnM6IHN0cmluZ1tdO1xyXG59XHJcblxyXG5jb25zdCBWQUxJREFUT1JfUFJPTVBUID0gYENoZWNrIHRoZSBmb2xsb3dpbmcgY2hhcHRlciBhZ2FpbnN0IHRoZSBTdG9yeSBDYW5vbi4gSWRlbnRpZnkgYW55IGNvbnRyYWRpY3Rpb25zLlxyXG5cclxuIyMgU3RvcnkgQ2Fub25cclxue3tjYW5vbn19XHJcblxyXG4jIyBDaGFwdGVyIFRleHRcclxue3tjaGFwdGVyVGV4dH19XHJcblxyXG4jIyBJbnN0cnVjdGlvbnNcclxuTGlzdCBhbnkgY29udHJhZGljdGlvbnMgYmV0d2VlbiB0aGUgY2hhcHRlciBhbmQgdGhlIGNhbm9uIGZhY3RzLiBBIGNvbnRyYWRpY3Rpb24gb2NjdXJzIHdoZW46XHJcbi0gQSBjaGFyYWN0ZXIncyBzdGF0dXMvYmFja2dyb3VuZCBpcyBkaWZmZXJlbnQgZnJvbSBjYW5vblxyXG4tIEEgcGxvdCB0aHJlYWQgc3RhdHVzIGlzIGNvbnRyYWRpY3RlZFxyXG4tIFdvcmxkIHJ1bGVzIGFyZSB2aW9sYXRlZFxyXG5cclxuUmV0dXJuIEpTT046XHJcbntcclxuICBcInZhbGlkXCI6IHRydWUvZmFsc2UsXHJcbiAgXCJ2aW9sYXRpb25zXCI6IFtcImRlc2NyaXB0aW9uIG9mIGNvbnRyYWRpY3Rpb24gMVwiLCBcImRlc2NyaXB0aW9uIG9mIGNvbnRyYWRpY3Rpb24gMlwiXVxyXG59XHJcblxyXG5JZiBubyBjb250cmFkaWN0aW9ucywgcmV0dXJuIHtcInZhbGlkXCI6IHRydWUsIFwidmlvbGF0aW9uc1wiOiBbXX1gO1xyXG5cclxuZXhwb3J0IGNsYXNzIENhbm9uVmFsaWRhdG9yIHtcclxuICBhc3luYyB2YWxpZGF0ZShjaGFwdGVyVGV4dDogc3RyaW5nLCBjYW5vbjogQ2Fub25TdG9yZSk6IFByb21pc2U8Q2Fub25WYWxpZGF0aW9uUmVzdWx0PiB7XHJcbiAgICBpZiAoY2Fub24uZmFjdHMubGVuZ3RoID09PSAwKSB7XHJcbiAgICAgIHJldHVybiB7IHZhbGlkOiB0cnVlLCB2aW9sYXRpb25zOiBbXSB9O1xyXG4gICAgfVxyXG5cclxuICAgIGNvbnN0IHsgZm9ybWF0Q2Fub25Gb3JQcm9tcHQgfSA9IGF3YWl0IGltcG9ydCgnLi4vbWVtb3J5L2Nhbm9uU3RvcmUuanMnKTtcclxuICAgIGNvbnN0IGNhbm9uU2VjdGlvbiA9IGZvcm1hdENhbm9uRm9yUHJvbXB0KGNhbm9uKTtcclxuXHJcbiAgICBjb25zdCBwcm9tcHQgPSBWQUxJREFUT1JfUFJPTVBUXHJcbiAgICAgIC5yZXBsYWNlKCd7e2Nhbm9ufX0nLCBjYW5vblNlY3Rpb24pXHJcbiAgICAgIC5yZXBsYWNlKCd7e2NoYXB0ZXJUZXh0fX0nLCBjaGFwdGVyVGV4dC5zdWJzdHJpbmcoMCwgMzAwMCkpO1xyXG5cclxuICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZ2V0TExNKCkuY29tcGxldGUocHJvbXB0LCB7XHJcbiAgICAgIHRlbXBlcmF0dXJlOiAwLjEsXHJcbiAgICAgIG1heFRva2VuczogNTAwLFxyXG4gICAgfSk7XHJcblxyXG4gICAgdHJ5IHtcclxuICAgICAgY29uc3QgcmVzdWx0ID0gSlNPTi5wYXJzZShyZXNwb25zZSkgYXMgQ2Fub25WYWxpZGF0aW9uUmVzdWx0O1xyXG4gICAgICByZXR1cm4gcmVzdWx0O1xyXG4gICAgfSBjYXRjaCB7XHJcbiAgICAgIHJldHVybiB7IHZhbGlkOiB0cnVlLCB2aW9sYXRpb25zOiBbXSB9O1xyXG4gICAgfVxyXG4gIH1cclxufVxyXG5cclxuZXhwb3J0IGNvbnN0IGNhbm9uVmFsaWRhdG9yID0gbmV3IENhbm9uVmFsaWRhdG9yKCk7XHJcbiJdfQ==
@@ -0,0 +1,50 @@
1
+ import type { StoryBible, StoryState } from '../types/index.js';
2
+ import type { StoryStructuredState } from '../story/structuredState.js';
3
+ import type { DirectorOutput, ChapterObjective } from './storyDirector.js';
4
+ export interface Scene {
5
+ id: string;
6
+ sequence: number;
7
+ goal: string;
8
+ description: string;
9
+ tension: number;
10
+ characters: string[];
11
+ setting: string;
12
+ estimatedWords: number;
13
+ }
14
+ export interface ChapterOutline {
15
+ chapterNumber: number;
16
+ overallGoal: string;
17
+ tone: string;
18
+ totalEstimatedWords: number;
19
+ scenes: Scene[];
20
+ transitions: string[];
21
+ notes: string;
22
+ }
23
+ export interface PlannerContext {
24
+ bible: StoryBible;
25
+ state: StoryState;
26
+ structuredState: StoryStructuredState;
27
+ directorOutput: DirectorOutput;
28
+ targetWordCount?: number;
29
+ }
30
+ export declare class ChapterPlanner {
31
+ plan(context: PlannerContext): Promise<ChapterOutline>;
32
+ private buildPrompt;
33
+ /**
34
+ * Format chapter outline for writer prompt
35
+ */
36
+ formatForPrompt(outline: ChapterOutline): string;
37
+ /**
38
+ * Generate fallback outline without LLM
39
+ */
40
+ generateFallbackOutline(directorOutput: DirectorOutput, targetWordCount?: number): ChapterOutline;
41
+ /**
42
+ * Validate that outline meets objectives
43
+ */
44
+ validateOutline(outline: ChapterOutline, objectives: ChapterObjective[]): {
45
+ valid: boolean;
46
+ coverage: number;
47
+ missedObjectives: string[];
48
+ };
49
+ }
50
+ export declare const chapterPlanner: ChapterPlanner;
@@ -0,0 +1,250 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.chapterPlanner = exports.ChapterPlanner = void 0;
4
+ const client_js_1 = require("../llm/client.js");
5
+ const CHAPTER_PLANNER_PROMPT = `You are the Chapter Planner for a narrative AI system. Convert chapter objectives into a detailed scene-by-scene outline.
6
+
7
+ ## Story Context
8
+
9
+ **Title:** {{title}}
10
+ **Genre:** {{genre}}
11
+ **Setting:** {{setting}}
12
+
13
+ ## Chapter Direction
14
+
15
+ **Chapter:** {{chapterNumber}}
16
+ **Overall Goal:** {{overallGoal}}
17
+ **Tone:** {{tone}}
18
+ **Target Word Count:** {{targetWordCount}}
19
+
20
+ ### Objectives (in priority order)
21
+ {{objectives}}
22
+
23
+ ### Focus Characters
24
+ {{focusCharacters}}
25
+
26
+ ### Suggested Scenes
27
+ {{suggestedScenes}}
28
+
29
+ ## Current Story State
30
+
31
+ **Story Tension:** {{storyTension}}%
32
+ **Target Tension:** {{targetTension}}%
33
+
34
+ ### Character States
35
+ {{characters}}
36
+
37
+ ### Active Plot Threads
38
+ {{plotThreads}}
39
+
40
+ ## Your Task
41
+
42
+ Create a detailed scene-by-scene outline for this chapter. Each scene should:
43
+ 1. Have a clear goal that serves the chapter objectives
44
+ 2. Build tension progressively toward the chapter climax
45
+ 3. Include specific characters and setting
46
+ 4. Have an estimated word count (scenes typically 300-800 words)
47
+
48
+ Output JSON with this structure:
49
+
50
+ {
51
+ "chapterNumber": {{chapterNumber}},
52
+ "overallGoal": "{{overallGoal}}",
53
+ "tone": "{{tone}}",
54
+ "totalEstimatedWords": 2500,
55
+ "scenes": [
56
+ {
57
+ "id": "scene-1",
58
+ "sequence": 1,
59
+ "goal": "What this scene accomplishes",
60
+ "description": "Detailed description of what happens",
61
+ "tension": 0.2,
62
+ "characters": ["Character names present"],
63
+ "setting": "Where scene takes place",
64
+ "estimatedWords": 400
65
+ }
66
+ ],
67
+ "transitions": ["How scenes connect"],
68
+ "notes": "Additional guidance for the writer"
69
+ }
70
+
71
+ Guidelines:
72
+ - Create 3-6 scenes per chapter
73
+ - Tension should build progressively (low → medium → high)
74
+ - Each scene should serve at least one objective
75
+ - Opening scene: establish situation
76
+ - Middle scenes: develop conflict, advance objectives
77
+ - Final scene: climax or resolution hook
78
+ - Total word count should match target (default ~2500)`;
79
+ class ChapterPlanner {
80
+ async plan(context) {
81
+ const { bible, state, structuredState, directorOutput, targetWordCount = 2500 } = context;
82
+ const prompt = this.buildPrompt(bible, state, structuredState, directorOutput, targetWordCount);
83
+ const result = await (0, client_js_1.getLLM)().completeJSON(prompt, {
84
+ temperature: 0.4,
85
+ maxTokens: 2500,
86
+ });
87
+ return result;
88
+ }
89
+ buildPrompt(bible, state, structuredState, directorOutput, targetWordCount) {
90
+ const currentTension = Math.round(structuredState.tension * 100);
91
+ const targetTension = Math.round(4 * (state.currentChapter / state.totalChapters) * (1 - state.currentChapter / state.totalChapters) * 100);
92
+ // Format objectives
93
+ const objectives = directorOutput.objectives
94
+ .map(obj => {
95
+ const priorityEmoji = { critical: '🔴', high: '🟠', medium: '🟡', low: '🟢' }[obj.priority];
96
+ return `${priorityEmoji} [${obj.type.toUpperCase()}] ${obj.description}`;
97
+ })
98
+ .join('\n');
99
+ // Format characters
100
+ const characters = Object.values(structuredState.characters)
101
+ .map(c => `- ${c.name}: ${c.emotionalState}, at ${c.location}`)
102
+ .join('\n') || 'No character data.';
103
+ // Format plot threads
104
+ const plotThreads = Object.values(structuredState.plotThreads)
105
+ .filter(t => t.status !== 'resolved')
106
+ .map(t => `- ${t.name} (${t.status}, ${Math.round(t.tension * 100)}% tension)`)
107
+ .join('\n') || 'No active plot threads.';
108
+ // Format suggested scenes
109
+ const suggestedScenes = directorOutput.suggestedScenes
110
+ .map(s => `- ${s}`)
111
+ .join('\n') || 'No suggestions.';
112
+ return CHAPTER_PLANNER_PROMPT
113
+ .replace('{{title}}', bible.title)
114
+ .replace('{{genre}}', bible.genre)
115
+ .replace('{{setting}}', bible.setting)
116
+ .replace(/{{chapterNumber}}/g, directorOutput.chapterNumber.toString())
117
+ .replace('{{overallGoal}}', directorOutput.overallGoal)
118
+ .replace('{{tone}}', directorOutput.tone)
119
+ .replace('{{targetWordCount}}', `${targetWordCount} words`)
120
+ .replace('{{objectives}}', objectives)
121
+ .replace('{{focusCharacters}}', directorOutput.focusCharacters.join(', '))
122
+ .replace('{{suggestedScenes}}', suggestedScenes)
123
+ .replace('{{storyTension}}', currentTension.toString())
124
+ .replace('{{targetTension}}', targetTension.toString())
125
+ .replace('{{characters}}', characters)
126
+ .replace('{{plotThreads}}', plotThreads);
127
+ }
128
+ /**
129
+ * Format chapter outline for writer prompt
130
+ */
131
+ formatForPrompt(outline) {
132
+ const lines = ['## Chapter Outline'];
133
+ lines.push(`\n**Chapter ${outline.chapterNumber}: ${outline.overallGoal}**`);
134
+ lines.push(`**Tone:** ${outline.tone}`);
135
+ lines.push(`**Estimated Length:** ${outline.totalEstimatedWords} words`);
136
+ if (outline.scenes.length > 0) {
137
+ lines.push('\n### Scene Breakdown');
138
+ for (const scene of outline.scenes) {
139
+ const tensionBar = '█'.repeat(Math.round(scene.tension * 10)) + '░'.repeat(10 - Math.round(scene.tension * 10));
140
+ lines.push(`\n**Scene ${scene.sequence}** [${tensionBar}] ${Math.round(scene.tension * 100)}% tension`);
141
+ lines.push(`- **Goal:** ${scene.goal}`);
142
+ lines.push(`- **Setting:** ${scene.setting}`);
143
+ lines.push(`- **Characters:** ${scene.characters.join(', ')}`);
144
+ lines.push(`- **Description:** ${scene.description}`);
145
+ lines.push(`- **Estimated:** ${scene.estimatedWords} words`);
146
+ }
147
+ }
148
+ if (outline.transitions.length > 0) {
149
+ lines.push('\n### Scene Transitions');
150
+ for (let i = 0; i < outline.transitions.length; i++) {
151
+ lines.push(`${i + 1} → ${i + 2}: ${outline.transitions[i]}`);
152
+ }
153
+ }
154
+ if (outline.notes) {
155
+ lines.push(`\n**Planner Notes:** ${outline.notes}`);
156
+ }
157
+ return lines.join('\n');
158
+ }
159
+ /**
160
+ * Generate fallback outline without LLM
161
+ */
162
+ generateFallbackOutline(directorOutput, targetWordCount = 2500) {
163
+ const scenes = [];
164
+ const sceneCount = Math.max(3, Math.min(6, Math.floor(targetWordCount / 500)));
165
+ const wordsPerScene = Math.floor(targetWordCount / sceneCount);
166
+ // Build scenes based on objectives
167
+ const criticalObjectives = directorOutput.objectives.filter(o => o.priority === 'critical');
168
+ const highObjectives = directorOutput.objectives.filter(o => o.priority === 'high');
169
+ // Scene 1: Setup
170
+ scenes.push({
171
+ id: 'scene-1',
172
+ sequence: 1,
173
+ goal: 'Establish current situation and character states',
174
+ description: 'Opening scene that sets up the chapter context',
175
+ tension: 0.2,
176
+ characters: directorOutput.focusCharacters.slice(0, 2),
177
+ setting: 'Current location',
178
+ estimatedWords: wordsPerScene,
179
+ });
180
+ // Middle scenes: Build tension and address objectives
181
+ for (let i = 2; i < sceneCount; i++) {
182
+ const tension = 0.3 + (i / sceneCount) * 0.5; // Build from 30% to 80%
183
+ const objective = criticalObjectives[i - 2] || highObjectives[i - 2];
184
+ scenes.push({
185
+ id: `scene-${i}`,
186
+ sequence: i,
187
+ goal: objective ? `Address: ${objective.description.substring(0, 50)}...` : 'Develop plot and character',
188
+ description: objective
189
+ ? `Scene focusing on ${objective.type} objective related to ${objective.relatedCharacter || 'plot'}`
190
+ : 'Development scene advancing the story',
191
+ tension: Math.round(tension * 100) / 100,
192
+ characters: directorOutput.focusCharacters,
193
+ setting: 'Story location',
194
+ estimatedWords: wordsPerScene,
195
+ });
196
+ }
197
+ // Final scene: Climax/Resolution
198
+ scenes.push({
199
+ id: `scene-${sceneCount}`,
200
+ sequence: sceneCount,
201
+ goal: criticalObjectives[0]?.description || 'Chapter climax or resolution',
202
+ description: 'Climactic scene that resolves or escalates the chapter tension',
203
+ tension: 0.9,
204
+ characters: directorOutput.focusCharacters,
205
+ setting: 'Key location',
206
+ estimatedWords: wordsPerScene,
207
+ });
208
+ // Generate transitions
209
+ const transitions = [];
210
+ for (let i = 0; i < scenes.length - 1; i++) {
211
+ transitions.push(`Natural progression from ${scenes[i].goal} to ${scenes[i + 1].goal}`);
212
+ }
213
+ return {
214
+ chapterNumber: directorOutput.chapterNumber,
215
+ overallGoal: directorOutput.overallGoal,
216
+ tone: directorOutput.tone,
217
+ totalEstimatedWords: scenes.reduce((sum, s) => sum + s.estimatedWords, 0),
218
+ scenes,
219
+ transitions,
220
+ notes: 'Auto-generated outline based on director objectives. Adjust as needed for narrative flow.',
221
+ };
222
+ }
223
+ /**
224
+ * Validate that outline meets objectives
225
+ */
226
+ validateOutline(outline, objectives) {
227
+ const covered = new Set();
228
+ const missed = [];
229
+ // Check each objective against scenes
230
+ for (const obj of objectives) {
231
+ const isCovered = outline.scenes.some(scene => scene.goal.toLowerCase().includes(obj.type) ||
232
+ scene.description.toLowerCase().includes(obj.description.toLowerCase().substring(0, 20)));
233
+ if (isCovered) {
234
+ covered.add(obj.id);
235
+ }
236
+ else if (obj.priority === 'critical' || obj.priority === 'high') {
237
+ missed.push(obj.description);
238
+ }
239
+ }
240
+ const coverage = objectives.length > 0 ? covered.size / objectives.length : 1;
241
+ return {
242
+ valid: missed.length === 0,
243
+ coverage,
244
+ missedObjectives: missed,
245
+ };
246
+ }
247
+ }
248
+ exports.ChapterPlanner = ChapterPlanner;
249
+ exports.chapterPlanner = new ChapterPlanner();
250
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chapterPlanner.js","sourceRoot":"","sources":["../../src/agents/chapterPlanner.ts"],"names":[],"mappings":"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","sourcesContent":["import { getLLM } from '../llm/client.js';\r\nimport type { StoryBible, StoryState } from '../types/index.js';\r\nimport type { StoryStructuredState } from '../story/structuredState.js';\r\nimport type { DirectorOutput, ChapterObjective } from './storyDirector.js';\r\n\r\nexport interface Scene {\r\n  id: string;\r\n  sequence: number;\r\n  goal: string;\r\n  description: string;\r\n  tension: number; // 0.0 to 1.0\r\n  characters: string[];\r\n  setting: string;\r\n  estimatedWords: number;\r\n}\r\n\r\nexport interface ChapterOutline {\r\n  chapterNumber: number;\r\n  overallGoal: string;\r\n  tone: string;\r\n  totalEstimatedWords: number;\r\n  scenes: Scene[];\r\n  transitions: string[];\r\n  notes: string;\r\n}\r\n\r\nexport interface PlannerContext {\r\n  bible: StoryBible;\r\n  state: StoryState;\r\n  structuredState: StoryStructuredState;\r\n  directorOutput: DirectorOutput;\r\n  targetWordCount?: number;\r\n}\r\n\r\nconst CHAPTER_PLANNER_PROMPT = `You are the Chapter Planner for a narrative AI system. Convert chapter objectives into a detailed scene-by-scene outline.\r\n\r\n## Story Context\r\n\r\n**Title:** {{title}}\r\n**Genre:** {{genre}}\r\n**Setting:** {{setting}}\r\n\r\n## Chapter Direction\r\n\r\n**Chapter:** {{chapterNumber}}\r\n**Overall Goal:** {{overallGoal}}\r\n**Tone:** {{tone}}\r\n**Target Word Count:** {{targetWordCount}}\r\n\r\n### Objectives (in priority order)\r\n{{objectives}}\r\n\r\n### Focus Characters\r\n{{focusCharacters}}\r\n\r\n### Suggested Scenes\r\n{{suggestedScenes}}\r\n\r\n## Current Story State\r\n\r\n**Story Tension:** {{storyTension}}%\r\n**Target Tension:** {{targetTension}}%\r\n\r\n### Character States\r\n{{characters}}\r\n\r\n### Active Plot Threads\r\n{{plotThreads}}\r\n\r\n## Your Task\r\n\r\nCreate a detailed scene-by-scene outline for this chapter. Each scene should:\r\n1. Have a clear goal that serves the chapter objectives\r\n2. Build tension progressively toward the chapter climax\r\n3. Include specific characters and setting\r\n4. Have an estimated word count (scenes typically 300-800 words)\r\n\r\nOutput JSON with this structure:\r\n\r\n{\r\n  \"chapterNumber\": {{chapterNumber}},\r\n  \"overallGoal\": \"{{overallGoal}}\",\r\n  \"tone\": \"{{tone}}\",\r\n  \"totalEstimatedWords\": 2500,\r\n  \"scenes\": [\r\n    {\r\n      \"id\": \"scene-1\",\r\n      \"sequence\": 1,\r\n      \"goal\": \"What this scene accomplishes\",\r\n      \"description\": \"Detailed description of what happens\",\r\n      \"tension\": 0.2,\r\n      \"characters\": [\"Character names present\"],\r\n      \"setting\": \"Where scene takes place\",\r\n      \"estimatedWords\": 400\r\n    }\r\n  ],\r\n  \"transitions\": [\"How scenes connect\"],\r\n  \"notes\": \"Additional guidance for the writer\"\r\n}\r\n\r\nGuidelines:\r\n- Create 3-6 scenes per chapter\r\n- Tension should build progressively (low → medium → high)\r\n- Each scene should serve at least one objective\r\n- Opening scene: establish situation\r\n- Middle scenes: develop conflict, advance objectives\r\n- Final scene: climax or resolution hook\r\n- Total word count should match target (default ~2500)`;\r\n\r\nexport class ChapterPlanner {\r\n  async plan(context: PlannerContext): Promise<ChapterOutline> {\r\n    const { bible, state, structuredState, directorOutput, targetWordCount = 2500 } = context;\r\n    \r\n    const prompt = this.buildPrompt(bible, state, structuredState, directorOutput, targetWordCount);\r\n    \r\n    const result = await getLLM().completeJSON<ChapterOutline>(prompt, {\r\n      temperature: 0.4,\r\n      maxTokens: 2500,\r\n    });\r\n    \r\n    return result;\r\n  }\r\n  \r\n  private buildPrompt(\r\n    bible: StoryBible,\r\n    state: StoryState,\r\n    structuredState: StoryStructuredState,\r\n    directorOutput: DirectorOutput,\r\n    targetWordCount: number\r\n  ): string {\r\n    const currentTension = Math.round(structuredState.tension * 100);\r\n    const targetTension = Math.round(\r\n      4 * (state.currentChapter / state.totalChapters) * (1 - state.currentChapter / state.totalChapters) * 100\r\n    );\r\n    \r\n    // Format objectives\r\n    const objectives = directorOutput.objectives\r\n      .map(obj => {\r\n        const priorityEmoji = { critical: '🔴', high: '🟠', medium: '🟡', low: '🟢' }[obj.priority];\r\n        return `${priorityEmoji} [${obj.type.toUpperCase()}] ${obj.description}`;\r\n      })\r\n      .join('\\n');\r\n    \r\n    // Format characters\r\n    const characters = Object.values(structuredState.characters)\r\n      .map(c => `- ${c.name}: ${c.emotionalState}, at ${c.location}`)\r\n      .join('\\n') || 'No character data.';\r\n    \r\n    // Format plot threads\r\n    const plotThreads = Object.values(structuredState.plotThreads)\r\n      .filter(t => t.status !== 'resolved')\r\n      .map(t => `- ${t.name} (${t.status}, ${Math.round(t.tension * 100)}% tension)`)\r\n      .join('\\n') || 'No active plot threads.';\r\n    \r\n    // Format suggested scenes\r\n    const suggestedScenes = directorOutput.suggestedScenes\r\n      .map(s => `- ${s}`)\r\n      .join('\\n') || 'No suggestions.';\r\n    \r\n    return CHAPTER_PLANNER_PROMPT\r\n      .replace('{{title}}', bible.title)\r\n      .replace('{{genre}}', bible.genre)\r\n      .replace('{{setting}}', bible.setting)\r\n      .replace(/{{chapterNumber}}/g, directorOutput.chapterNumber.toString())\r\n      .replace('{{overallGoal}}', directorOutput.overallGoal)\r\n      .replace('{{tone}}', directorOutput.tone)\r\n      .replace('{{targetWordCount}}', `${targetWordCount} words`)\r\n      .replace('{{objectives}}', objectives)\r\n      .replace('{{focusCharacters}}', directorOutput.focusCharacters.join(', '))\r\n      .replace('{{suggestedScenes}}', suggestedScenes)\r\n      .replace('{{storyTension}}', currentTension.toString())\r\n      .replace('{{targetTension}}', targetTension.toString())\r\n      .replace('{{characters}}', characters)\r\n      .replace('{{plotThreads}}', plotThreads);\r\n  }\r\n  \r\n  /**\r\n   * Format chapter outline for writer prompt\r\n   */\r\n  formatForPrompt(outline: ChapterOutline): string {\r\n    const lines: string[] = ['## Chapter Outline'];\r\n    \r\n    lines.push(`\\n**Chapter ${outline.chapterNumber}: ${outline.overallGoal}**`);\r\n    lines.push(`**Tone:** ${outline.tone}`);\r\n    lines.push(`**Estimated Length:** ${outline.totalEstimatedWords} words`);\r\n    \r\n    if (outline.scenes.length > 0) {\r\n      lines.push('\\n### Scene Breakdown');\r\n      \r\n      for (const scene of outline.scenes) {\r\n        const tensionBar = '█'.repeat(Math.round(scene.tension * 10)) + '░'.repeat(10 - Math.round(scene.tension * 10));\r\n        lines.push(`\\n**Scene ${scene.sequence}** [${tensionBar}] ${Math.round(scene.tension * 100)}% tension`);\r\n        lines.push(`- **Goal:** ${scene.goal}`);\r\n        lines.push(`- **Setting:** ${scene.setting}`);\r\n        lines.push(`- **Characters:** ${scene.characters.join(', ')}`);\r\n        lines.push(`- **Description:** ${scene.description}`);\r\n        lines.push(`- **Estimated:** ${scene.estimatedWords} words`);\r\n      }\r\n    }\r\n    \r\n    if (outline.transitions.length > 0) {\r\n      lines.push('\\n### Scene Transitions');\r\n      for (let i = 0; i < outline.transitions.length; i++) {\r\n        lines.push(`${i + 1} → ${i + 2}: ${outline.transitions[i]}`);\r\n      }\r\n    }\r\n    \r\n    if (outline.notes) {\r\n      lines.push(`\\n**Planner Notes:** ${outline.notes}`);\r\n    }\r\n    \r\n    return lines.join('\\n');\r\n  }\r\n  \r\n  /**\r\n   * Generate fallback outline without LLM\r\n   */\r\n  generateFallbackOutline(\r\n    directorOutput: DirectorOutput,\r\n    targetWordCount: number = 2500\r\n  ): ChapterOutline {\r\n    const scenes: Scene[] = [];\r\n    const sceneCount = Math.max(3, Math.min(6, Math.floor(targetWordCount / 500)));\r\n    const wordsPerScene = Math.floor(targetWordCount / sceneCount);\r\n    \r\n    // Build scenes based on objectives\r\n    const criticalObjectives = directorOutput.objectives.filter(o => o.priority === 'critical');\r\n    const highObjectives = directorOutput.objectives.filter(o => o.priority === 'high');\r\n    \r\n    // Scene 1: Setup\r\n    scenes.push({\r\n      id: 'scene-1',\r\n      sequence: 1,\r\n      goal: 'Establish current situation and character states',\r\n      description: 'Opening scene that sets up the chapter context',\r\n      tension: 0.2,\r\n      characters: directorOutput.focusCharacters.slice(0, 2),\r\n      setting: 'Current location',\r\n      estimatedWords: wordsPerScene,\r\n    });\r\n    \r\n    // Middle scenes: Build tension and address objectives\r\n    for (let i = 2; i < sceneCount; i++) {\r\n      const tension = 0.3 + (i / sceneCount) * 0.5; // Build from 30% to 80%\r\n      const objective = criticalObjectives[i - 2] || highObjectives[i - 2];\r\n      \r\n      scenes.push({\r\n        id: `scene-${i}`,\r\n        sequence: i,\r\n        goal: objective ? `Address: ${objective.description.substring(0, 50)}...` : 'Develop plot and character',\r\n        description: objective \r\n          ? `Scene focusing on ${objective.type} objective related to ${objective.relatedCharacter || 'plot'}`\r\n          : 'Development scene advancing the story',\r\n        tension: Math.round(tension * 100) / 100,\r\n        characters: directorOutput.focusCharacters,\r\n        setting: 'Story location',\r\n        estimatedWords: wordsPerScene,\r\n      });\r\n    }\r\n    \r\n    // Final scene: Climax/Resolution\r\n    scenes.push({\r\n      id: `scene-${sceneCount}`,\r\n      sequence: sceneCount,\r\n      goal: criticalObjectives[0]?.description || 'Chapter climax or resolution',\r\n      description: 'Climactic scene that resolves or escalates the chapter tension',\r\n      tension: 0.9,\r\n      characters: directorOutput.focusCharacters,\r\n      setting: 'Key location',\r\n      estimatedWords: wordsPerScene,\r\n    });\r\n    \r\n    // Generate transitions\r\n    const transitions: string[] = [];\r\n    for (let i = 0; i < scenes.length - 1; i++) {\r\n      transitions.push(`Natural progression from ${scenes[i].goal} to ${scenes[i + 1].goal}`);\r\n    }\r\n    \r\n    return {\r\n      chapterNumber: directorOutput.chapterNumber,\r\n      overallGoal: directorOutput.overallGoal,\r\n      tone: directorOutput.tone,\r\n      totalEstimatedWords: scenes.reduce((sum, s) => sum + s.estimatedWords, 0),\r\n      scenes,\r\n      transitions,\r\n      notes: 'Auto-generated outline based on director objectives. Adjust as needed for narrative flow.',\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Validate that outline meets objectives\r\n   */\r\n  validateOutline(outline: ChapterOutline, objectives: ChapterObjective[]): {\r\n    valid: boolean;\r\n    coverage: number;\r\n    missedObjectives: string[];\r\n  } {\r\n    const covered = new Set<string>();\r\n    const missed: string[] = [];\r\n    \r\n    // Check each objective against scenes\r\n    for (const obj of objectives) {\r\n      const isCovered = outline.scenes.some(scene => \r\n        scene.goal.toLowerCase().includes(obj.type) ||\r\n        scene.description.toLowerCase().includes(obj.description.toLowerCase().substring(0, 20))\r\n      );\r\n      \r\n      if (isCovered) {\r\n        covered.add(obj.id);\r\n      } else if (obj.priority === 'critical' || obj.priority === 'high') {\r\n        missed.push(obj.description);\r\n      }\r\n    }\r\n    \r\n    const coverage = objectives.length > 0 ? covered.size / objectives.length : 1;\r\n    \r\n    return {\r\n      valid: missed.length === 0,\r\n      coverage,\r\n      missedObjectives: missed,\r\n    };\r\n  }\r\n}\r\n\r\nexport const chapterPlanner = new ChapterPlanner();\r\n"]}
@@ -0,0 +1,7 @@
1
+ import type { CompletenessResult } from '../types/index.js';
2
+ export declare class CompletenessChecker {
3
+ private promptTemplate;
4
+ constructor();
5
+ check(chapterText: string): Promise<CompletenessResult>;
6
+ }
7
+ export declare const completenessChecker: CompletenessChecker;
@@ -0,0 +1,51 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.completenessChecker = exports.CompletenessChecker = void 0;
4
+ const client_js_1 = require("../llm/client.js");
5
+ const COMPLETENESS_PROMPT = `Determine whether the following chapter text ends at a natural stopping point.
6
+
7
+ A chapter is COMPLETE if:
8
+ - It ends at the end of a scene or chapter break
9
+ - It does not cut off mid-sentence or mid-paragraph
10
+ - It has narrative closure for this segment
11
+
12
+ A chapter is INCOMPLETE if:
13
+ - It cuts off mid-sentence
14
+ - It ends abruptly mid-scene
15
+ - The narrative clearly continues
16
+
17
+ ## Chapter Text
18
+
19
+ {{chapterText}}
20
+
21
+ ## Response
22
+
23
+ Return only one word:
24
+
25
+ COMPLETE
26
+
27
+ or
28
+
29
+ INCOMPLETE`;
30
+ class CompletenessChecker {
31
+ promptTemplate;
32
+ constructor() {
33
+ this.promptTemplate = COMPLETENESS_PROMPT;
34
+ }
35
+ async check(chapterText) {
36
+ const prompt = this.promptTemplate.replace('{{chapterText}}', chapterText);
37
+ const response = await (0, client_js_1.getLLM)().complete(prompt, {
38
+ temperature: 0.1,
39
+ maxTokens: 10,
40
+ });
41
+ const normalized = response.trim().toUpperCase();
42
+ const isComplete = normalized.includes('COMPLETE') && !normalized.includes('INCOMPLETE');
43
+ return {
44
+ isComplete,
45
+ reason: isComplete ? undefined : 'Chapter appears to be cut off mid-content',
46
+ };
47
+ }
48
+ }
49
+ exports.CompletenessChecker = CompletenessChecker;
50
+ exports.completenessChecker = new CompletenessChecker();
51
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tcGxldGVuZXNzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2FnZW50cy9jb21wbGV0ZW5lc3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsZ0RBQTBDO0FBRzFDLE1BQU0sbUJBQW1CLEdBQUc7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztXQXdCakIsQ0FBQztBQUVaLE1BQWEsbUJBQW1CO0lBQ3RCLGNBQWMsQ0FBUztJQUUvQjtRQUNFLElBQUksQ0FBQyxjQUFjLEdBQUcsbUJBQW1CLENBQUM7SUFDNUMsQ0FBQztJQUVELEtBQUssQ0FBQyxLQUFLLENBQUMsV0FBbUI7UUFDN0IsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLGNBQWMsQ0FBQyxPQUFPLENBQUMsaUJBQWlCLEVBQUUsV0FBVyxDQUFDLENBQUM7UUFFM0UsTUFBTSxRQUFRLEdBQUcsTUFBTSxJQUFBLGtCQUFNLEdBQUUsQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFO1lBQy9DLFdBQVcsRUFBRSxHQUFHO1lBQ2hCLFNBQVMsRUFBRSxFQUFFO1NBQ2QsQ0FBQyxDQUFDO1FBRUgsTUFBTSxVQUFVLEdBQUcsUUFBUSxDQUFDLElBQUksRUFBRSxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQ2pELE1BQU0sVUFBVSxHQUFHLFVBQVUsQ0FBQyxRQUFRLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsUUFBUSxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBRXpGLE9BQU87WUFDTCxVQUFVO1lBQ1YsTUFBTSxFQUFFLFVBQVUsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQywyQ0FBMkM7U0FDN0UsQ0FBQztJQUNKLENBQUM7Q0FDRjtBQXZCRCxrREF1QkM7QUFFWSxRQUFBLG1CQUFtQixHQUFHLElBQUksbUJBQW1CLEVBQUUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGdldExMTSB9IGZyb20gJy4uL2xsbS9jbGllbnQuanMnO1xyXG5pbXBvcnQgdHlwZSB7IENvbXBsZXRlbmVzc1Jlc3VsdCB9IGZyb20gJy4uL3R5cGVzL2luZGV4LmpzJztcclxuXHJcbmNvbnN0IENPTVBMRVRFTkVTU19QUk9NUFQgPSBgRGV0ZXJtaW5lIHdoZXRoZXIgdGhlIGZvbGxvd2luZyBjaGFwdGVyIHRleHQgZW5kcyBhdCBhIG5hdHVyYWwgc3RvcHBpbmcgcG9pbnQuXHJcblxyXG5BIGNoYXB0ZXIgaXMgQ09NUExFVEUgaWY6XHJcbi0gSXQgZW5kcyBhdCB0aGUgZW5kIG9mIGEgc2NlbmUgb3IgY2hhcHRlciBicmVha1xyXG4tIEl0IGRvZXMgbm90IGN1dCBvZmYgbWlkLXNlbnRlbmNlIG9yIG1pZC1wYXJhZ3JhcGhcclxuLSBJdCBoYXMgbmFycmF0aXZlIGNsb3N1cmUgZm9yIHRoaXMgc2VnbWVudFxyXG5cclxuQSBjaGFwdGVyIGlzIElOQ09NUExFVEUgaWY6XHJcbi0gSXQgY3V0cyBvZmYgbWlkLXNlbnRlbmNlXHJcbi0gSXQgZW5kcyBhYnJ1cHRseSBtaWQtc2NlbmVcclxuLSBUaGUgbmFycmF0aXZlIGNsZWFybHkgY29udGludWVzXHJcblxyXG4jIyBDaGFwdGVyIFRleHRcclxuXHJcbnt7Y2hhcHRlclRleHR9fVxyXG5cclxuIyMgUmVzcG9uc2VcclxuXHJcblJldHVybiBvbmx5IG9uZSB3b3JkOlxyXG5cclxuQ09NUExFVEVcclxuXHJcbm9yXHJcblxyXG5JTkNPTVBMRVRFYDtcclxuXHJcbmV4cG9ydCBjbGFzcyBDb21wbGV0ZW5lc3NDaGVja2VyIHtcclxuICBwcml2YXRlIHByb21wdFRlbXBsYXRlOiBzdHJpbmc7XHJcblxyXG4gIGNvbnN0cnVjdG9yKCkge1xyXG4gICAgdGhpcy5wcm9tcHRUZW1wbGF0ZSA9IENPTVBMRVRFTkVTU19QUk9NUFQ7XHJcbiAgfVxyXG5cclxuICBhc3luYyBjaGVjayhjaGFwdGVyVGV4dDogc3RyaW5nKTogUHJvbWlzZTxDb21wbGV0ZW5lc3NSZXN1bHQ+IHtcclxuICAgIGNvbnN0IHByb21wdCA9IHRoaXMucHJvbXB0VGVtcGxhdGUucmVwbGFjZSgne3tjaGFwdGVyVGV4dH19JywgY2hhcHRlclRleHQpO1xyXG5cclxuICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZ2V0TExNKCkuY29tcGxldGUocHJvbXB0LCB7XHJcbiAgICAgIHRlbXBlcmF0dXJlOiAwLjEsXHJcbiAgICAgIG1heFRva2VuczogMTAsXHJcbiAgICB9KTtcclxuXHJcbiAgICBjb25zdCBub3JtYWxpemVkID0gcmVzcG9uc2UudHJpbSgpLnRvVXBwZXJDYXNlKCk7XHJcbiAgICBjb25zdCBpc0NvbXBsZXRlID0gbm9ybWFsaXplZC5pbmNsdWRlcygnQ09NUExFVEUnKSAmJiAhbm9ybWFsaXplZC5pbmNsdWRlcygnSU5DT01QTEVURScpO1xyXG5cclxuICAgIHJldHVybiB7XHJcbiAgICAgIGlzQ29tcGxldGUsXHJcbiAgICAgIHJlYXNvbjogaXNDb21wbGV0ZSA/IHVuZGVmaW5lZCA6ICdDaGFwdGVyIGFwcGVhcnMgdG8gYmUgY3V0IG9mZiBtaWQtY29udGVudCcsXHJcbiAgICB9O1xyXG4gIH1cclxufVxyXG5cclxuZXhwb3J0IGNvbnN0IGNvbXBsZXRlbmVzc0NoZWNrZXIgPSBuZXcgQ29tcGxldGVuZXNzQ2hlY2tlcigpO1xyXG4iXX0=
@@ -0,0 +1,12 @@
1
+ import type { Chapter, StoryBible } from '../types/index.js';
2
+ import type { NarrativeMemory } from '../memory/vectorStore.js';
3
+ interface ExtractedMemory {
4
+ content: string;
5
+ category: NarrativeMemory['category'];
6
+ }
7
+ export declare class MemoryExtractor {
8
+ extract(chapter: Chapter, bible: StoryBible): Promise<ExtractedMemory[]>;
9
+ extractFromSummary(chapterNumber: number, summary: string, bible: StoryBible): Promise<ExtractedMemory[]>;
10
+ }
11
+ export declare const memoryExtractor: MemoryExtractor;
12
+ export {};
@@ -0,0 +1,82 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.memoryExtractor = exports.MemoryExtractor = void 0;
4
+ const client_js_1 = require("../llm/client.js");
5
+ const EXTRACTION_PROMPT = `You are a narrative memory extractor. Your job is to analyze a chapter and extract important facts that should be remembered for future chapters.
6
+
7
+ ## Story Bible
8
+
9
+ **Title:** {{title}}
10
+ **Genre:** {{genre}}
11
+ **Setting:** {{setting}}
12
+
13
+ ## Chapter to Analyze
14
+
15
+ **Chapter {{chapterNumber}}: {{chapterTitle}}**
16
+
17
+ {{chapterContent}}
18
+
19
+ ## Extraction Task
20
+
21
+ Extract 5-10 important narrative memories from this chapter. Focus on:
22
+
23
+ 1. **Events** - Significant things that happened (plot points, discoveries, battles, meetings)
24
+ 2. **Character** - Character development, new traits revealed, relationships changed
25
+ 3. **World** - New world details revealed (locations, cultures, magic systems, technology)
26
+ 4. **Plot** - Plot thread developments, mysteries introduced, foreshadowing
27
+
28
+ For each memory:
29
+ - Write a clear, standalone sentence that captures the fact
30
+ - Categorize it appropriately
31
+ - Be specific enough that it would be useful for maintaining continuity
32
+
33
+ Respond with JSON only:
34
+ {
35
+ "memories": [
36
+ {"content": "Alice discovered the ancient map hidden in her grandmother's attic", "category": "event"},
37
+ {"content": "Bob is secretly afraid of water due to a childhood drowning incident", "category": "character"},
38
+ {"content": "The city of Eldoria has a strict curfew enforced by mechanical guards", "category": "world"},
39
+ {"content": "The prophecy mentions three keys that must be found before the eclipse", "category": "plot"}
40
+ ]
41
+ }`;
42
+ class MemoryExtractor {
43
+ async extract(chapter, bible) {
44
+ const prompt = EXTRACTION_PROMPT
45
+ .replace('{{title}}', bible.title)
46
+ .replace('{{genre}}', bible.genre)
47
+ .replace('{{setting}}', bible.setting)
48
+ .replace('{{chapterNumber}}', chapter.number.toString())
49
+ .replace('{{chapterTitle}}', chapter.title)
50
+ .replace('{{chapterContent}}', chapter.content.substring(0, 8000)); // Limit content length
51
+ const result = await (0, client_js_1.getLLM)().completeJSON(prompt, {
52
+ temperature: 0.3,
53
+ maxTokens: 2000,
54
+ });
55
+ return result.memories || [];
56
+ }
57
+ async extractFromSummary(chapterNumber, summary, bible) {
58
+ const prompt = `You are a narrative memory extractor. Extract important facts from this chapter summary.
59
+
60
+ ## Story
61
+ **Title:** ${bible.title}
62
+ **Genre:** ${bible.genre}
63
+
64
+ ## Chapter ${chapterNumber} Summary
65
+ ${summary}
66
+
67
+ Extract 3-5 key memories (events, character moments, world details, plot developments). Respond with JSON:
68
+ {
69
+ "memories": [
70
+ {"content": "description of what happened", "category": "event|character|world|plot"}
71
+ ]
72
+ }`;
73
+ const result = await (0, client_js_1.getLLM)().completeJSON(prompt, {
74
+ temperature: 0.3,
75
+ maxTokens: 1000,
76
+ });
77
+ return result.memories || [];
78
+ }
79
+ }
80
+ exports.MemoryExtractor = MemoryExtractor;
81
+ exports.memoryExtractor = new MemoryExtractor();
82
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVtb3J5RXh0cmFjdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2FnZW50cy9tZW1vcnlFeHRyYWN0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsZ0RBQTBDO0FBYTFDLE1BQU0saUJBQWlCLEdBQUc7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztFQW9DeEIsQ0FBQztBQUVILE1BQWEsZUFBZTtJQUMxQixLQUFLLENBQUMsT0FBTyxDQUFDLE9BQWdCLEVBQUUsS0FBaUI7UUFDL0MsTUFBTSxNQUFNLEdBQUcsaUJBQWlCO2FBQzdCLE9BQU8sQ0FBQyxXQUFXLEVBQUUsS0FBSyxDQUFDLEtBQUssQ0FBQzthQUNqQyxPQUFPLENBQUMsV0FBVyxFQUFFLEtBQUssQ0FBQyxLQUFLLENBQUM7YUFDakMsT0FBTyxDQUFDLGFBQWEsRUFBRSxLQUFLLENBQUMsT0FBTyxDQUFDO2FBQ3JDLE9BQU8sQ0FBQyxtQkFBbUIsRUFBRSxPQUFPLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDO2FBQ3ZELE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxPQUFPLENBQUMsS0FBSyxDQUFDO2FBQzFDLE9BQU8sQ0FBQyxvQkFBb0IsRUFBRSxPQUFPLENBQUMsT0FBTyxDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDLHVCQUF1QjtRQUU3RixNQUFNLE1BQU0sR0FBRyxNQUFNLElBQUEsa0JBQU0sR0FBRSxDQUFDLFlBQVksQ0FBbUIsTUFBTSxFQUFFO1lBQ25FLFdBQVcsRUFBRSxHQUFHO1lBQ2hCLFNBQVMsRUFBRSxJQUFJO1NBQ2hCLENBQUMsQ0FBQztRQUVILE9BQU8sTUFBTSxDQUFDLFFBQVEsSUFBSSxFQUFFLENBQUM7SUFDL0IsQ0FBQztJQUVELEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxhQUFxQixFQUFFLE9BQWUsRUFBRSxLQUFpQjtRQUNoRixNQUFNLE1BQU0sR0FBRzs7O2FBR04sS0FBSyxDQUFDLEtBQUs7YUFDWCxLQUFLLENBQUMsS0FBSzs7YUFFWCxhQUFhO0VBQ3hCLE9BQU87Ozs7Ozs7RUFPUCxDQUFDO1FBRUMsTUFBTSxNQUFNLEdBQUcsTUFBTSxJQUFBLGtCQUFNLEdBQUUsQ0FBQyxZQUFZLENBQW1CLE1BQU0sRUFBRTtZQUNuRSxXQUFXLEVBQUUsR0FBRztZQUNoQixTQUFTLEVBQUUsSUFBSTtTQUNoQixDQUFDLENBQUM7UUFFSCxPQUFPLE1BQU0sQ0FBQyxRQUFRLElBQUksRUFBRSxDQUFDO0lBQy9CLENBQUM7Q0FDRjtBQTFDRCwwQ0EwQ0M7QUFFWSxRQUFBLGVBQWUsR0FBRyxJQUFJLGVBQWUsRUFBRSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZ2V0TExNIH0gZnJvbSAnLi4vbGxtL2NsaWVudC5qcyc7XHJcbmltcG9ydCB0eXBlIHsgQ2hhcHRlciwgU3RvcnlCaWJsZSB9IGZyb20gJy4uL3R5cGVzL2luZGV4LmpzJztcclxuaW1wb3J0IHR5cGUgeyBOYXJyYXRpdmVNZW1vcnkgfSBmcm9tICcuLi9tZW1vcnkvdmVjdG9yU3RvcmUuanMnO1xyXG5cclxuaW50ZXJmYWNlIEV4dHJhY3RlZE1lbW9yeSB7XHJcbiAgY29udGVudDogc3RyaW5nO1xyXG4gIGNhdGVnb3J5OiBOYXJyYXRpdmVNZW1vcnlbJ2NhdGVnb3J5J107XHJcbn1cclxuXHJcbmludGVyZmFjZSBFeHRyYWN0aW9uT3V0cHV0IHtcclxuICBtZW1vcmllczogRXh0cmFjdGVkTWVtb3J5W107XHJcbn1cclxuXHJcbmNvbnN0IEVYVFJBQ1RJT05fUFJPTVBUID0gYFlvdSBhcmUgYSBuYXJyYXRpdmUgbWVtb3J5IGV4dHJhY3Rvci4gWW91ciBqb2IgaXMgdG8gYW5hbHl6ZSBhIGNoYXB0ZXIgYW5kIGV4dHJhY3QgaW1wb3J0YW50IGZhY3RzIHRoYXQgc2hvdWxkIGJlIHJlbWVtYmVyZWQgZm9yIGZ1dHVyZSBjaGFwdGVycy5cclxuXHJcbiMjIFN0b3J5IEJpYmxlXHJcblxyXG4qKlRpdGxlOioqIHt7dGl0bGV9fVxyXG4qKkdlbnJlOioqIHt7Z2VucmV9fVxyXG4qKlNldHRpbmc6Kioge3tzZXR0aW5nfX1cclxuXHJcbiMjIENoYXB0ZXIgdG8gQW5hbHl6ZVxyXG5cclxuKipDaGFwdGVyIHt7Y2hhcHRlck51bWJlcn19OiB7e2NoYXB0ZXJUaXRsZX19KipcclxuXHJcbnt7Y2hhcHRlckNvbnRlbnR9fVxyXG5cclxuIyMgRXh0cmFjdGlvbiBUYXNrXHJcblxyXG5FeHRyYWN0IDUtMTAgaW1wb3J0YW50IG5hcnJhdGl2ZSBtZW1vcmllcyBmcm9tIHRoaXMgY2hhcHRlci4gRm9jdXMgb246XHJcblxyXG4xLiAqKkV2ZW50cyoqIC0gU2lnbmlmaWNhbnQgdGhpbmdzIHRoYXQgaGFwcGVuZWQgKHBsb3QgcG9pbnRzLCBkaXNjb3ZlcmllcywgYmF0dGxlcywgbWVldGluZ3MpXHJcbjIuICoqQ2hhcmFjdGVyKiogLSBDaGFyYWN0ZXIgZGV2ZWxvcG1lbnQsIG5ldyB0cmFpdHMgcmV2ZWFsZWQsIHJlbGF0aW9uc2hpcHMgY2hhbmdlZFxyXG4zLiAqKldvcmxkKiogLSBOZXcgd29ybGQgZGV0YWlscyByZXZlYWxlZCAobG9jYXRpb25zLCBjdWx0dXJlcywgbWFnaWMgc3lzdGVtcywgdGVjaG5vbG9neSlcclxuNC4gKipQbG90KiogLSBQbG90IHRocmVhZCBkZXZlbG9wbWVudHMsIG15c3RlcmllcyBpbnRyb2R1Y2VkLCBmb3Jlc2hhZG93aW5nXHJcblxyXG5Gb3IgZWFjaCBtZW1vcnk6XHJcbi0gV3JpdGUgYSBjbGVhciwgc3RhbmRhbG9uZSBzZW50ZW5jZSB0aGF0IGNhcHR1cmVzIHRoZSBmYWN0XHJcbi0gQ2F0ZWdvcml6ZSBpdCBhcHByb3ByaWF0ZWx5XHJcbi0gQmUgc3BlY2lmaWMgZW5vdWdoIHRoYXQgaXQgd291bGQgYmUgdXNlZnVsIGZvciBtYWludGFpbmluZyBjb250aW51aXR5XHJcblxyXG5SZXNwb25kIHdpdGggSlNPTiBvbmx5OlxyXG57XHJcbiAgXCJtZW1vcmllc1wiOiBbXHJcbiAgICB7XCJjb250ZW50XCI6IFwiQWxpY2UgZGlzY292ZXJlZCB0aGUgYW5jaWVudCBtYXAgaGlkZGVuIGluIGhlciBncmFuZG1vdGhlcidzIGF0dGljXCIsIFwiY2F0ZWdvcnlcIjogXCJldmVudFwifSxcclxuICAgIHtcImNvbnRlbnRcIjogXCJCb2IgaXMgc2VjcmV0bHkgYWZyYWlkIG9mIHdhdGVyIGR1ZSB0byBhIGNoaWxkaG9vZCBkcm93bmluZyBpbmNpZGVudFwiLCBcImNhdGVnb3J5XCI6IFwiY2hhcmFjdGVyXCJ9LFxyXG4gICAge1wiY29udGVudFwiOiBcIlRoZSBjaXR5IG9mIEVsZG9yaWEgaGFzIGEgc3RyaWN0IGN1cmZldyBlbmZvcmNlZCBieSBtZWNoYW5pY2FsIGd1YXJkc1wiLCBcImNhdGVnb3J5XCI6IFwid29ybGRcIn0sXHJcbiAgICB7XCJjb250ZW50XCI6IFwiVGhlIHByb3BoZWN5IG1lbnRpb25zIHRocmVlIGtleXMgdGhhdCBtdXN0IGJlIGZvdW5kIGJlZm9yZSB0aGUgZWNsaXBzZVwiLCBcImNhdGVnb3J5XCI6IFwicGxvdFwifVxyXG4gIF1cclxufWA7XHJcblxyXG5leHBvcnQgY2xhc3MgTWVtb3J5RXh0cmFjdG9yIHtcclxuICBhc3luYyBleHRyYWN0KGNoYXB0ZXI6IENoYXB0ZXIsIGJpYmxlOiBTdG9yeUJpYmxlKTogUHJvbWlzZTxFeHRyYWN0ZWRNZW1vcnlbXT4ge1xyXG4gICAgY29uc3QgcHJvbXB0ID0gRVhUUkFDVElPTl9QUk9NUFRcclxuICAgICAgLnJlcGxhY2UoJ3t7dGl0bGV9fScsIGJpYmxlLnRpdGxlKVxyXG4gICAgICAucmVwbGFjZSgne3tnZW5yZX19JywgYmlibGUuZ2VucmUpXHJcbiAgICAgIC5yZXBsYWNlKCd7e3NldHRpbmd9fScsIGJpYmxlLnNldHRpbmcpXHJcbiAgICAgIC5yZXBsYWNlKCd7e2NoYXB0ZXJOdW1iZXJ9fScsIGNoYXB0ZXIubnVtYmVyLnRvU3RyaW5nKCkpXHJcbiAgICAgIC5yZXBsYWNlKCd7e2NoYXB0ZXJUaXRsZX19JywgY2hhcHRlci50aXRsZSlcclxuICAgICAgLnJlcGxhY2UoJ3t7Y2hhcHRlckNvbnRlbnR9fScsIGNoYXB0ZXIuY29udGVudC5zdWJzdHJpbmcoMCwgODAwMCkpOyAvLyBMaW1pdCBjb250ZW50IGxlbmd0aFxyXG5cclxuICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IGdldExMTSgpLmNvbXBsZXRlSlNPTjxFeHRyYWN0aW9uT3V0cHV0Pihwcm9tcHQsIHtcclxuICAgICAgdGVtcGVyYXR1cmU6IDAuMyxcclxuICAgICAgbWF4VG9rZW5zOiAyMDAwLFxyXG4gICAgfSk7XHJcblxyXG4gICAgcmV0dXJuIHJlc3VsdC5tZW1vcmllcyB8fCBbXTtcclxuICB9XHJcblxyXG4gIGFzeW5jIGV4dHJhY3RGcm9tU3VtbWFyeShjaGFwdGVyTnVtYmVyOiBudW1iZXIsIHN1bW1hcnk6IHN0cmluZywgYmlibGU6IFN0b3J5QmlibGUpOiBQcm9taXNlPEV4dHJhY3RlZE1lbW9yeVtdPiB7XHJcbiAgICBjb25zdCBwcm9tcHQgPSBgWW91IGFyZSBhIG5hcnJhdGl2ZSBtZW1vcnkgZXh0cmFjdG9yLiBFeHRyYWN0IGltcG9ydGFudCBmYWN0cyBmcm9tIHRoaXMgY2hhcHRlciBzdW1tYXJ5LlxyXG5cclxuIyMgU3RvcnlcclxuKipUaXRsZToqKiAke2JpYmxlLnRpdGxlfVxyXG4qKkdlbnJlOioqICR7YmlibGUuZ2VucmV9XHJcblxyXG4jIyBDaGFwdGVyICR7Y2hhcHRlck51bWJlcn0gU3VtbWFyeVxyXG4ke3N1bW1hcnl9XHJcblxyXG5FeHRyYWN0IDMtNSBrZXkgbWVtb3JpZXMgKGV2ZW50cywgY2hhcmFjdGVyIG1vbWVudHMsIHdvcmxkIGRldGFpbHMsIHBsb3QgZGV2ZWxvcG1lbnRzKS4gUmVzcG9uZCB3aXRoIEpTT046XHJcbntcclxuICBcIm1lbW9yaWVzXCI6IFtcclxuICAgIHtcImNvbnRlbnRcIjogXCJkZXNjcmlwdGlvbiBvZiB3aGF0IGhhcHBlbmVkXCIsIFwiY2F0ZWdvcnlcIjogXCJldmVudHxjaGFyYWN0ZXJ8d29ybGR8cGxvdFwifVxyXG4gIF1cclxufWA7XHJcblxyXG4gICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgZ2V0TExNKCkuY29tcGxldGVKU09OPEV4dHJhY3Rpb25PdXRwdXQ+KHByb21wdCwge1xyXG4gICAgICB0ZW1wZXJhdHVyZTogMC4zLFxyXG4gICAgICBtYXhUb2tlbnM6IDEwMDAsXHJcbiAgICB9KTtcclxuXHJcbiAgICByZXR1cm4gcmVzdWx0Lm1lbW9yaWVzIHx8IFtdO1xyXG4gIH1cclxufVxyXG5cclxuZXhwb3J0IGNvbnN0IG1lbW9yeUV4dHJhY3RvciA9IG5ldyBNZW1vcnlFeHRyYWN0b3IoKTtcclxuIl19
@@ -0,0 +1,30 @@
1
+ import type { Chapter, StoryBible } from '../types/index.js';
2
+ import type { StoryStructuredState } from '../story/structuredState.js';
3
+ interface StateUpdateOutput {
4
+ characterUpdates: Array<{
5
+ name: string;
6
+ emotionalState?: string;
7
+ location?: string;
8
+ newKnowledge?: string[];
9
+ relationshipChanges?: Array<{
10
+ with: string;
11
+ status: string;
12
+ }>;
13
+ development?: string;
14
+ }>;
15
+ plotThreadUpdates: Array<{
16
+ id: string;
17
+ status?: 'dormant' | 'active' | 'escalating' | 'resolved';
18
+ tensionChange?: number;
19
+ summary?: string;
20
+ }>;
21
+ newQuestions: string[];
22
+ resolvedQuestions: string[];
23
+ recentEvents: string[];
24
+ }
25
+ export declare class StateUpdater {
26
+ extractStateChanges(chapter: Chapter, bible: StoryBible, currentState: StoryStructuredState): Promise<StateUpdateOutput>;
27
+ applyUpdates(state: StoryStructuredState, updates: StateUpdateOutput, chapterNumber: number): StoryStructuredState;
28
+ }
29
+ export declare const stateUpdater: StateUpdater;
30
+ export {};