@narrative-os/engine 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/agents/canonValidator.d.ts +9 -0
- package/dist/agents/canonValidator.js +51 -0
- package/dist/agents/chapterPlanner.d.ts +50 -0
- package/dist/agents/chapterPlanner.js +250 -0
- package/dist/agents/completeness.d.ts +7 -0
- package/dist/agents/completeness.js +51 -0
- package/dist/agents/memoryExtractor.d.ts +12 -0
- package/dist/agents/memoryExtractor.js +82 -0
- package/dist/agents/stateUpdater.d.ts +30 -0
- package/dist/agents/stateUpdater.js +150 -0
- package/dist/agents/storyDirector.d.ts +40 -0
- package/dist/agents/storyDirector.js +213 -0
- package/dist/agents/summarizer.d.ts +8 -0
- package/dist/agents/summarizer.js +56 -0
- package/dist/agents/tensionController.d.ts +68 -0
- package/dist/agents/tensionController.js +197 -0
- package/dist/agents/writer.d.ts +12 -0
- package/dist/agents/writer.js +148 -0
- package/dist/constraints/constraintGraph.d.ts +117 -0
- package/dist/constraints/constraintGraph.js +381 -0
- package/dist/constraints/validator.d.ts +58 -0
- package/dist/constraints/validator.js +236 -0
- package/dist/index.d.ts +25 -0
- package/dist/index.js +115 -0
- package/dist/llm/client.d.ts +14 -0
- package/dist/llm/client.js +108 -0
- package/dist/memory/canonStore.d.ts +20 -0
- package/dist/memory/canonStore.js +110 -0
- package/dist/memory/memoryRetriever.d.ts +28 -0
- package/dist/memory/memoryRetriever.js +126 -0
- package/dist/memory/stateUpdater.d.ts +49 -0
- package/dist/memory/stateUpdater.js +315 -0
- package/dist/memory/vectorStore.d.ts +41 -0
- package/dist/memory/vectorStore.js +166 -0
- package/dist/pipeline/generateChapter.d.ts +17 -0
- package/dist/pipeline/generateChapter.js +75 -0
- package/dist/story/bible.d.ts +4 -0
- package/dist/story/bible.js +53 -0
- package/dist/story/state.d.ts +3 -0
- package/dist/story/state.js +27 -0
- package/dist/story/structuredState.d.ts +39 -0
- package/dist/story/structuredState.js +159 -0
- package/dist/test/canon.test.d.ts +1 -0
- package/dist/test/canon.test.js +104 -0
- package/dist/test/chapter-planner.test.d.ts +1 -0
- package/dist/test/chapter-planner.test.js +171 -0
- package/dist/test/constraints.test.d.ts +1 -0
- package/dist/test/constraints.test.js +210 -0
- package/dist/test/simple.test.d.ts +1 -0
- package/dist/test/simple.test.js +51 -0
- package/dist/test/state-updater.test.d.ts +1 -0
- package/dist/test/state-updater.test.js +200 -0
- package/dist/test/story-director.test.d.ts +1 -0
- package/dist/test/story-director.test.js +142 -0
- package/dist/test/structured-state.test.d.ts +1 -0
- package/dist/test/structured-state.test.js +144 -0
- package/dist/test/tension-controller.test.d.ts +1 -0
- package/dist/test/tension-controller.test.js +116 -0
- package/dist/test/vector-memory.test.d.ts +1 -0
- package/dist/test/vector-memory.test.js +153 -0
- package/dist/test/world-simulation.test.d.ts +1 -0
- package/dist/test/world-simulation.test.js +152 -0
- package/dist/types/index.d.ts +79 -0
- package/dist/types/index.js +3 -0
- package/dist/world/characterAgent.d.ts +73 -0
- package/dist/world/characterAgent.js +232 -0
- package/dist/world/eventResolver.d.ts +52 -0
- package/dist/world/eventResolver.js +205 -0
- package/dist/world/worldState.d.ts +93 -0
- package/dist/world/worldState.js +258 -0
- package/package.json +43 -0
|
@@ -0,0 +1,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 {};
|