@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,150 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.stateUpdater = exports.StateUpdater = void 0;
|
|
4
|
+
const client_js_1 = require("../llm/client.js");
|
|
5
|
+
const STATE_UPDATE_PROMPT = `You are a narrative state tracker. Analyze the chapter and extract state changes.
|
|
6
|
+
|
|
7
|
+
## Story Bible
|
|
8
|
+
|
|
9
|
+
**Title:** {{title}}
|
|
10
|
+
**Genre:** {{genre}}
|
|
11
|
+
|
|
12
|
+
## Characters
|
|
13
|
+
|
|
14
|
+
{{characters}}
|
|
15
|
+
|
|
16
|
+
## Current Plot Threads
|
|
17
|
+
|
|
18
|
+
{{plotThreads}}
|
|
19
|
+
|
|
20
|
+
## Chapter Content
|
|
21
|
+
|
|
22
|
+
**Chapter {{chapterNumber}}: {{chapterTitle}}**
|
|
23
|
+
|
|
24
|
+
{{chapterContent}}
|
|
25
|
+
|
|
26
|
+
## Current Unresolved Questions
|
|
27
|
+
|
|
28
|
+
{{unresolvedQuestions}}
|
|
29
|
+
|
|
30
|
+
## Task
|
|
31
|
+
|
|
32
|
+
Analyze what changed in this chapter. Output JSON with:
|
|
33
|
+
|
|
34
|
+
1. **characterUpdates**: How characters changed (emotion, location, knowledge, relationships)
|
|
35
|
+
2. **plotThreadUpdates**: How plot threads progressed (status, tension)
|
|
36
|
+
3. **newQuestions**: New mysteries or questions raised
|
|
37
|
+
4. **resolvedQuestions**: Which current questions were answered
|
|
38
|
+
5. **recentEvents**: Key events that happened (2-3 bullet points)
|
|
39
|
+
|
|
40
|
+
Example output:
|
|
41
|
+
{
|
|
42
|
+
"characterUpdates": [
|
|
43
|
+
{
|
|
44
|
+
"name": "Alice",
|
|
45
|
+
"emotionalState": "anxious",
|
|
46
|
+
"location": "the abandoned warehouse",
|
|
47
|
+
"newKnowledge": ["the code is 8472"],
|
|
48
|
+
"relationshipChanges": [{"with": "Bob", "status": "distrustful"}],
|
|
49
|
+
"development": "Alice realizes she can't trust her mentor"
|
|
50
|
+
}
|
|
51
|
+
],
|
|
52
|
+
"plotThreadUpdates": [
|
|
53
|
+
{
|
|
54
|
+
"id": "missing_brother",
|
|
55
|
+
"status": "escalating",
|
|
56
|
+
"tensionChange": 0.1,
|
|
57
|
+
"summary": "Discovered brother was kidnapped by the syndicate"
|
|
58
|
+
}
|
|
59
|
+
],
|
|
60
|
+
"newQuestions": ["Who is the mysterious informant?"],
|
|
61
|
+
"resolvedQuestions": ["Where was the brother taken?"],
|
|
62
|
+
"recentEvents": ["Alice broke into the warehouse", "Found the hidden room"]
|
|
63
|
+
}`;
|
|
64
|
+
class StateUpdater {
|
|
65
|
+
async extractStateChanges(chapter, bible, currentState) {
|
|
66
|
+
const characters = Object.values(currentState.characters)
|
|
67
|
+
.map(c => `- ${c.name}: currently ${c.emotionalState}, at ${c.location}`)
|
|
68
|
+
.join('\n') || 'No characters tracked yet.';
|
|
69
|
+
const plotThreads = Object.values(currentState.plotThreads)
|
|
70
|
+
.map(t => `- ${t.name} (${t.status}, tension: ${Math.round(t.tension * 100)}%): ${t.summary}`)
|
|
71
|
+
.join('\n') || 'No active plot threads.';
|
|
72
|
+
const unresolvedQuestions = currentState.unresolvedQuestions.length > 0
|
|
73
|
+
? currentState.unresolvedQuestions.map(q => `- ${q}`).join('\n')
|
|
74
|
+
: 'None';
|
|
75
|
+
const prompt = STATE_UPDATE_PROMPT
|
|
76
|
+
.replace('{{title}}', bible.title)
|
|
77
|
+
.replace('{{genre}}', bible.genre)
|
|
78
|
+
.replace('{{characters}}', characters)
|
|
79
|
+
.replace('{{plotThreads}}', plotThreads)
|
|
80
|
+
.replace('{{chapterNumber}}', chapter.number.toString())
|
|
81
|
+
.replace('{{chapterTitle}}', chapter.title)
|
|
82
|
+
.replace('{{chapterContent}}', chapter.content.substring(0, 6000))
|
|
83
|
+
.replace('{{unresolvedQuestions}}', unresolvedQuestions);
|
|
84
|
+
const result = await (0, client_js_1.getLLM)().completeJSON(prompt, {
|
|
85
|
+
temperature: 0.3,
|
|
86
|
+
maxTokens: 2000,
|
|
87
|
+
});
|
|
88
|
+
return result;
|
|
89
|
+
}
|
|
90
|
+
applyUpdates(state, updates, chapterNumber) {
|
|
91
|
+
let newState = { ...state };
|
|
92
|
+
// Apply character updates
|
|
93
|
+
for (const update of updates.characterUpdates) {
|
|
94
|
+
if (newState.characters[update.name]) {
|
|
95
|
+
const char = newState.characters[update.name];
|
|
96
|
+
if (update.emotionalState) {
|
|
97
|
+
char.emotionalState = update.emotionalState;
|
|
98
|
+
}
|
|
99
|
+
if (update.location) {
|
|
100
|
+
char.location = update.location;
|
|
101
|
+
}
|
|
102
|
+
if (update.newKnowledge) {
|
|
103
|
+
char.knowledge = [...char.knowledge, ...update.newKnowledge];
|
|
104
|
+
}
|
|
105
|
+
if (update.relationshipChanges) {
|
|
106
|
+
for (const rel of update.relationshipChanges) {
|
|
107
|
+
char.relationships[rel.with] = rel.status;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
if (update.development) {
|
|
111
|
+
char.development = [...char.development, update.development];
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
// Apply plot thread updates
|
|
116
|
+
for (const update of updates.plotThreadUpdates) {
|
|
117
|
+
if (newState.plotThreads[update.id]) {
|
|
118
|
+
const thread = newState.plotThreads[update.id];
|
|
119
|
+
if (update.status) {
|
|
120
|
+
thread.status = update.status;
|
|
121
|
+
}
|
|
122
|
+
if (update.tensionChange !== undefined) {
|
|
123
|
+
thread.tension = Math.max(0, Math.min(1, thread.tension + update.tensionChange));
|
|
124
|
+
}
|
|
125
|
+
if (update.summary) {
|
|
126
|
+
thread.summary = update.summary;
|
|
127
|
+
}
|
|
128
|
+
thread.lastChapter = chapterNumber;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
// Add new questions
|
|
132
|
+
for (const question of updates.newQuestions) {
|
|
133
|
+
if (!newState.unresolvedQuestions.includes(question)) {
|
|
134
|
+
newState.unresolvedQuestions = [...newState.unresolvedQuestions, question];
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
// Remove resolved questions
|
|
138
|
+
for (const question of updates.resolvedQuestions) {
|
|
139
|
+
newState.unresolvedQuestions = newState.unresolvedQuestions.filter(q => q !== question);
|
|
140
|
+
}
|
|
141
|
+
// Add recent events
|
|
142
|
+
for (const event of updates.recentEvents) {
|
|
143
|
+
newState.recentEvents = [...newState.recentEvents, event].slice(-10);
|
|
144
|
+
}
|
|
145
|
+
return newState;
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
exports.StateUpdater = StateUpdater;
|
|
149
|
+
exports.stateUpdater = new StateUpdater();
|
|
150
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stateUpdater.js","sourceRoot":"","sources":["../../src/agents/stateUpdater.ts"],"names":[],"mappings":"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","sourcesContent":["import { getLLM } from '../llm/client.js';\r\nimport type { Chapter, StoryBible } from '../types/index.js';\r\nimport type { StoryStructuredState, CharacterState, PlotThreadState } from '../story/structuredState.js';\r\n\r\ninterface StateUpdateOutput {\r\n  characterUpdates: Array<{\r\n    name: string;\r\n    emotionalState?: string;\r\n    location?: string;\r\n    newKnowledge?: string[];\r\n    relationshipChanges?: Array<{ with: string; status: string }>;\r\n    development?: string;\r\n  }>;\r\n  plotThreadUpdates: Array<{\r\n    id: string;\r\n    status?: 'dormant' | 'active' | 'escalating' | 'resolved';\r\n    tensionChange?: number; // -0.1 to +0.1\r\n    summary?: string;\r\n  }>;\r\n  newQuestions: string[];\r\n  resolvedQuestions: string[];\r\n  recentEvents: string[];\r\n}\r\n\r\nconst STATE_UPDATE_PROMPT = `You are a narrative state tracker. Analyze the chapter and extract state changes.\r\n\r\n## Story Bible\r\n\r\n**Title:** {{title}}\r\n**Genre:** {{genre}}\r\n\r\n## Characters\r\n\r\n{{characters}}\r\n\r\n## Current Plot Threads\r\n\r\n{{plotThreads}}\r\n\r\n## Chapter Content\r\n\r\n**Chapter {{chapterNumber}}: {{chapterTitle}}**\r\n\r\n{{chapterContent}}\r\n\r\n## Current Unresolved Questions\r\n\r\n{{unresolvedQuestions}}\r\n\r\n## Task\r\n\r\nAnalyze what changed in this chapter. Output JSON with:\r\n\r\n1. **characterUpdates**: How characters changed (emotion, location, knowledge, relationships)\r\n2. **plotThreadUpdates**: How plot threads progressed (status, tension)\r\n3. **newQuestions**: New mysteries or questions raised\r\n4. **resolvedQuestions**: Which current questions were answered\r\n5. **recentEvents**: Key events that happened (2-3 bullet points)\r\n\r\nExample output:\r\n{\r\n  \"characterUpdates\": [\r\n    {\r\n      \"name\": \"Alice\",\r\n      \"emotionalState\": \"anxious\",\r\n      \"location\": \"the abandoned warehouse\",\r\n      \"newKnowledge\": [\"the code is 8472\"],\r\n      \"relationshipChanges\": [{\"with\": \"Bob\", \"status\": \"distrustful\"}],\r\n      \"development\": \"Alice realizes she can't trust her mentor\"\r\n    }\r\n  ],\r\n  \"plotThreadUpdates\": [\r\n    {\r\n      \"id\": \"missing_brother\",\r\n      \"status\": \"escalating\",\r\n      \"tensionChange\": 0.1,\r\n      \"summary\": \"Discovered brother was kidnapped by the syndicate\"\r\n    }\r\n  ],\r\n  \"newQuestions\": [\"Who is the mysterious informant?\"],\r\n  \"resolvedQuestions\": [\"Where was the brother taken?\"],\r\n  \"recentEvents\": [\"Alice broke into the warehouse\", \"Found the hidden room\"]\r\n}`;\r\n\r\nexport class StateUpdater {\r\n  async extractStateChanges(\r\n    chapter: Chapter,\r\n    bible: StoryBible,\r\n    currentState: StoryStructuredState\r\n  ): Promise<StateUpdateOutput> {\r\n    const characters = Object.values(currentState.characters)\r\n      .map(c => `- ${c.name}: currently ${c.emotionalState}, at ${c.location}`)\r\n      .join('\\n') || 'No characters tracked yet.';\r\n\r\n    const plotThreads = Object.values(currentState.plotThreads)\r\n      .map(t => `- ${t.name} (${t.status}, tension: ${Math.round(t.tension * 100)}%): ${t.summary}`)\r\n      .join('\\n') || 'No active plot threads.';\r\n\r\n    const unresolvedQuestions = currentState.unresolvedQuestions.length > 0\r\n      ? currentState.unresolvedQuestions.map(q => `- ${q}`).join('\\n')\r\n      : 'None';\r\n\r\n    const prompt = STATE_UPDATE_PROMPT\r\n      .replace('{{title}}', bible.title)\r\n      .replace('{{genre}}', bible.genre)\r\n      .replace('{{characters}}', characters)\r\n      .replace('{{plotThreads}}', plotThreads)\r\n      .replace('{{chapterNumber}}', chapter.number.toString())\r\n      .replace('{{chapterTitle}}', chapter.title)\r\n      .replace('{{chapterContent}}', chapter.content.substring(0, 6000))\r\n      .replace('{{unresolvedQuestions}}', unresolvedQuestions);\r\n\r\n    const result = await getLLM().completeJSON<StateUpdateOutput>(prompt, {\r\n      temperature: 0.3,\r\n      maxTokens: 2000,\r\n    });\r\n\r\n    return result;\r\n  }\r\n\r\n  applyUpdates(\r\n    state: StoryStructuredState,\r\n    updates: StateUpdateOutput,\r\n    chapterNumber: number\r\n  ): StoryStructuredState {\r\n    let newState = { ...state };\r\n\r\n    // Apply character updates\r\n    for (const update of updates.characterUpdates) {\r\n      if (newState.characters[update.name]) {\r\n        const char = newState.characters[update.name];\r\n        \r\n        if (update.emotionalState) {\r\n          char.emotionalState = update.emotionalState;\r\n        }\r\n        if (update.location) {\r\n          char.location = update.location;\r\n        }\r\n        if (update.newKnowledge) {\r\n          char.knowledge = [...char.knowledge, ...update.newKnowledge];\r\n        }\r\n        if (update.relationshipChanges) {\r\n          for (const rel of update.relationshipChanges) {\r\n            char.relationships[rel.with] = rel.status;\r\n          }\r\n        }\r\n        if (update.development) {\r\n          char.development = [...char.development, update.development];\r\n        }\r\n      }\r\n    }\r\n\r\n    // Apply plot thread updates\r\n    for (const update of updates.plotThreadUpdates) {\r\n      if (newState.plotThreads[update.id]) {\r\n        const thread = newState.plotThreads[update.id];\r\n        \r\n        if (update.status) {\r\n          thread.status = update.status;\r\n        }\r\n        if (update.tensionChange !== undefined) {\r\n          thread.tension = Math.max(0, Math.min(1, thread.tension + update.tensionChange));\r\n        }\r\n        if (update.summary) {\r\n          thread.summary = update.summary;\r\n        }\r\n        thread.lastChapter = chapterNumber;\r\n      }\r\n    }\r\n\r\n    // Add new questions\r\n    for (const question of updates.newQuestions) {\r\n      if (!newState.unresolvedQuestions.includes(question)) {\r\n        newState.unresolvedQuestions = [...newState.unresolvedQuestions, question];\r\n      }\r\n    }\r\n\r\n    // Remove resolved questions\r\n    for (const question of updates.resolvedQuestions) {\r\n      newState.unresolvedQuestions = newState.unresolvedQuestions.filter(q => q !== question);\r\n    }\r\n\r\n    // Add recent events\r\n    for (const event of updates.recentEvents) {\r\n      newState.recentEvents = [...newState.recentEvents, event].slice(-10);\r\n    }\r\n\r\n    return newState;\r\n  }\r\n}\r\n\r\nexport const stateUpdater = new StateUpdater();\r\n"]}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import type { StoryBible, StoryState, ChapterSummary } from '../types/index.js';
|
|
2
|
+
import type { StoryStructuredState } from '../story/structuredState.js';
|
|
3
|
+
import type { TensionGuidance } from './tensionController.js';
|
|
4
|
+
export interface ChapterObjective {
|
|
5
|
+
id: string;
|
|
6
|
+
description: string;
|
|
7
|
+
priority: 'critical' | 'high' | 'medium' | 'low';
|
|
8
|
+
type: 'plot' | 'character' | 'world' | 'tension' | 'resolution';
|
|
9
|
+
relatedPlotThreadId?: string;
|
|
10
|
+
relatedCharacter?: string;
|
|
11
|
+
}
|
|
12
|
+
export interface DirectorOutput {
|
|
13
|
+
chapterNumber: number;
|
|
14
|
+
overallGoal: string;
|
|
15
|
+
objectives: ChapterObjective[];
|
|
16
|
+
focusCharacters: string[];
|
|
17
|
+
suggestedScenes: string[];
|
|
18
|
+
tone: string;
|
|
19
|
+
notes: string;
|
|
20
|
+
}
|
|
21
|
+
export interface DirectorContext {
|
|
22
|
+
bible: StoryBible;
|
|
23
|
+
state: StoryState;
|
|
24
|
+
structuredState: StoryStructuredState;
|
|
25
|
+
tensionGuidance: TensionGuidance;
|
|
26
|
+
previousSummaries: ChapterSummary[];
|
|
27
|
+
}
|
|
28
|
+
export declare class StoryDirector {
|
|
29
|
+
direct(context: DirectorContext): Promise<DirectorOutput>;
|
|
30
|
+
private buildPrompt;
|
|
31
|
+
/**
|
|
32
|
+
* Format director output for writer prompt
|
|
33
|
+
*/
|
|
34
|
+
formatForPrompt(output: DirectorOutput): string;
|
|
35
|
+
/**
|
|
36
|
+
* Get quick objectives without LLM call (for testing/fallback)
|
|
37
|
+
*/
|
|
38
|
+
generateFallbackObjectives(state: StoryState, structuredState: StoryStructuredState): DirectorOutput;
|
|
39
|
+
}
|
|
40
|
+
export declare const storyDirector: StoryDirector;
|
|
@@ -0,0 +1,213 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.storyDirector = exports.StoryDirector = void 0;
|
|
4
|
+
const client_js_1 = require("../llm/client.js");
|
|
5
|
+
const STORY_DIRECTOR_PROMPT = `You are the Story Director for a narrative AI system. Your job is to decide what the next chapter should accomplish.
|
|
6
|
+
|
|
7
|
+
## Story Bible
|
|
8
|
+
|
|
9
|
+
**Title:** {{title}}
|
|
10
|
+
**Genre:** {{genre}}
|
|
11
|
+
**Theme:** {{theme}}
|
|
12
|
+
**Premise:** {{premise}}
|
|
13
|
+
|
|
14
|
+
## Current Story State
|
|
15
|
+
|
|
16
|
+
**Chapter:** {{currentChapter}} / {{totalChapters}}
|
|
17
|
+
**Story Tension:** {{storyTension}}%
|
|
18
|
+
**Target Tension:** {{targetTension}}%
|
|
19
|
+
|
|
20
|
+
### Active Plot Threads
|
|
21
|
+
{{plotThreads}}
|
|
22
|
+
|
|
23
|
+
### Character States
|
|
24
|
+
{{characters}}
|
|
25
|
+
|
|
26
|
+
### Unresolved Questions
|
|
27
|
+
{{questions}}
|
|
28
|
+
|
|
29
|
+
### Recent Events
|
|
30
|
+
{{recentEvents}}
|
|
31
|
+
|
|
32
|
+
## Tension Guidance
|
|
33
|
+
|
|
34
|
+
{{tensionGuidance}}
|
|
35
|
+
|
|
36
|
+
## Previous Chapter Summaries
|
|
37
|
+
{{summaries}}
|
|
38
|
+
|
|
39
|
+
## Your Task
|
|
40
|
+
|
|
41
|
+
Based on all the above information, decide what Chapter {{nextChapter}} should accomplish. Consider:
|
|
42
|
+
|
|
43
|
+
1. **Plot Progression**: Which plot threads need advancement?
|
|
44
|
+
2. **Character Development**: Which characters need focus or growth?
|
|
45
|
+
3. **Tension Management**: How should tension change based on target?
|
|
46
|
+
4. **Unresolved Questions**: Which mysteries should be addressed?
|
|
47
|
+
5. **Story Arc**: Where are we in the overall narrative (setup/rising action/climax/resolution)?
|
|
48
|
+
|
|
49
|
+
Output a JSON object with:
|
|
50
|
+
|
|
51
|
+
{
|
|
52
|
+
"chapterNumber": {{nextChapter}},
|
|
53
|
+
"overallGoal": "One-sentence description of what this chapter achieves",
|
|
54
|
+
"objectives": [
|
|
55
|
+
{
|
|
56
|
+
"id": "unique-id",
|
|
57
|
+
"description": "What needs to happen",
|
|
58
|
+
"priority": "critical|high|medium|low",
|
|
59
|
+
"type": "plot|character|world|tension|resolution",
|
|
60
|
+
"relatedPlotThreadId": "thread-id or omit",
|
|
61
|
+
"relatedCharacter": "character name or omit"
|
|
62
|
+
}
|
|
63
|
+
],
|
|
64
|
+
"focusCharacters": ["Character names that should be central"],
|
|
65
|
+
"suggestedScenes": ["Scene ideas that could achieve objectives"],
|
|
66
|
+
"tone": "emotional tone for this chapter",
|
|
67
|
+
"notes": "Additional guidance for the writer"
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
Be specific and actionable. The writer will use your direction to craft the chapter.`;
|
|
71
|
+
class StoryDirector {
|
|
72
|
+
async direct(context) {
|
|
73
|
+
const { bible, state, structuredState, tensionGuidance, previousSummaries } = context;
|
|
74
|
+
const prompt = this.buildPrompt(bible, state, structuredState, tensionGuidance, previousSummaries);
|
|
75
|
+
const result = await (0, client_js_1.getLLM)().completeJSON(prompt, {
|
|
76
|
+
temperature: 0.4,
|
|
77
|
+
maxTokens: 2000,
|
|
78
|
+
});
|
|
79
|
+
return result;
|
|
80
|
+
}
|
|
81
|
+
buildPrompt(bible, state, structuredState, tensionGuidance, summaries) {
|
|
82
|
+
const currentTension = Math.round(structuredState.tension * 100);
|
|
83
|
+
const targetTension = Math.round(tensionGuidance.targetTension * 100);
|
|
84
|
+
// Format plot threads
|
|
85
|
+
const activeThreads = Object.values(structuredState.plotThreads)
|
|
86
|
+
.filter(t => t.status !== 'resolved')
|
|
87
|
+
.map(t => `- **${t.name}** (${t.status}, ${Math.round(t.tension * 100)}% tension): ${t.summary}`)
|
|
88
|
+
.join('\n') || 'No active plot threads.';
|
|
89
|
+
// Format characters
|
|
90
|
+
const characters = Object.values(structuredState.characters)
|
|
91
|
+
.map(c => `- **${c.name}**: ${c.emotionalState}, at ${c.location}`)
|
|
92
|
+
.join('\n') || 'No character data.';
|
|
93
|
+
// Format questions
|
|
94
|
+
const questions = structuredState.unresolvedQuestions.length > 0
|
|
95
|
+
? structuredState.unresolvedQuestions.map(q => `- ${q}`).join('\n')
|
|
96
|
+
: 'None';
|
|
97
|
+
// Format recent events
|
|
98
|
+
const recentEvents = structuredState.recentEvents.length > 0
|
|
99
|
+
? structuredState.recentEvents.slice(-5).map(e => `- ${e}`).join('\n')
|
|
100
|
+
: 'None';
|
|
101
|
+
// Format tension guidance
|
|
102
|
+
const tensionText = `Target: ${targetTension}%
|
|
103
|
+
Guidance: ${tensionGuidance.guidance}
|
|
104
|
+
Scene Types: ${tensionGuidance.sceneTypes.join(', ')}
|
|
105
|
+
Pacing: ${tensionGuidance.pacingNotes}`;
|
|
106
|
+
// Format summaries (last 3)
|
|
107
|
+
const recentSummaries = summaries.slice(-3);
|
|
108
|
+
const summariesText = recentSummaries.length > 0
|
|
109
|
+
? recentSummaries.map(s => `Chapter ${s.chapterNumber}: ${s.summary}`).join('\n')
|
|
110
|
+
: 'No previous chapters.';
|
|
111
|
+
return STORY_DIRECTOR_PROMPT
|
|
112
|
+
.replace('{{title}}', bible.title)
|
|
113
|
+
.replace('{{genre}}', bible.genre)
|
|
114
|
+
.replace('{{theme}}', bible.theme)
|
|
115
|
+
.replace('{{premise}}', bible.premise)
|
|
116
|
+
.replace('{{currentChapter}}', state.currentChapter.toString())
|
|
117
|
+
.replace('{{totalChapters}}', state.totalChapters.toString())
|
|
118
|
+
.replace('{{storyTension}}', currentTension.toString())
|
|
119
|
+
.replace('{{targetTension}}', targetTension.toString())
|
|
120
|
+
.replace('{{plotThreads}}', activeThreads)
|
|
121
|
+
.replace('{{characters}}', characters)
|
|
122
|
+
.replace('{{questions}}', questions)
|
|
123
|
+
.replace('{{recentEvents}}', recentEvents)
|
|
124
|
+
.replace('{{tensionGuidance}}', tensionText)
|
|
125
|
+
.replace('{{summaries}}', summariesText)
|
|
126
|
+
.replace(/{{nextChapter}}/g, (state.currentChapter + 1).toString());
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* Format director output for writer prompt
|
|
130
|
+
*/
|
|
131
|
+
formatForPrompt(output) {
|
|
132
|
+
const lines = ['## Chapter Direction'];
|
|
133
|
+
lines.push(`\n**Chapter ${output.chapterNumber} Goal:** ${output.overallGoal}`);
|
|
134
|
+
lines.push(`\n**Tone:** ${output.tone}`);
|
|
135
|
+
if (output.focusCharacters.length > 0) {
|
|
136
|
+
lines.push(`\n**Focus Characters:** ${output.focusCharacters.join(', ')}`);
|
|
137
|
+
}
|
|
138
|
+
if (output.objectives.length > 0) {
|
|
139
|
+
lines.push('\n**Objectives (in priority order):**');
|
|
140
|
+
const sorted = [...output.objectives].sort((a, b) => {
|
|
141
|
+
const priorityOrder = { critical: 0, high: 1, medium: 2, low: 3 };
|
|
142
|
+
return priorityOrder[a.priority] - priorityOrder[b.priority];
|
|
143
|
+
});
|
|
144
|
+
for (const obj of sorted) {
|
|
145
|
+
const emoji = { critical: '🔴', high: '🟠', medium: '🟡', low: '🟢' }[obj.priority];
|
|
146
|
+
lines.push(`${emoji} **[${obj.type.toUpperCase()}]** ${obj.description}`);
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
if (output.suggestedScenes.length > 0) {
|
|
150
|
+
lines.push('\n**Suggested Scenes:**');
|
|
151
|
+
for (const scene of output.suggestedScenes) {
|
|
152
|
+
lines.push(`- ${scene}`);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
if (output.notes) {
|
|
156
|
+
lines.push(`\n**Director's Notes:** ${output.notes}`);
|
|
157
|
+
}
|
|
158
|
+
return lines.join('\n');
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Get quick objectives without LLM call (for testing/fallback)
|
|
162
|
+
*/
|
|
163
|
+
generateFallbackObjectives(state, structuredState) {
|
|
164
|
+
const nextChapter = state.currentChapter + 1;
|
|
165
|
+
const objectives = [];
|
|
166
|
+
// Add plot thread objectives
|
|
167
|
+
const activeThreads = Object.values(structuredState.plotThreads)
|
|
168
|
+
.filter(t => t.status === 'active' || t.status === 'escalating');
|
|
169
|
+
for (const thread of activeThreads.slice(0, 2)) {
|
|
170
|
+
objectives.push({
|
|
171
|
+
id: `plot-${thread.id}`,
|
|
172
|
+
description: `Advance the "${thread.name}" plot thread`,
|
|
173
|
+
priority: 'high',
|
|
174
|
+
type: 'plot',
|
|
175
|
+
relatedPlotThreadId: thread.id,
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
// Add character objective
|
|
179
|
+
const characters = Object.values(structuredState.characters);
|
|
180
|
+
if (characters.length > 0) {
|
|
181
|
+
const char = characters[nextChapter % characters.length];
|
|
182
|
+
objectives.push({
|
|
183
|
+
id: `char-${char.name}`,
|
|
184
|
+
description: `Develop ${char.name}'s character arc`,
|
|
185
|
+
priority: 'medium',
|
|
186
|
+
type: 'character',
|
|
187
|
+
relatedCharacter: char.name,
|
|
188
|
+
});
|
|
189
|
+
}
|
|
190
|
+
// Add tension objective if needed
|
|
191
|
+
const targetTension = 4 * (nextChapter / state.totalChapters) * (1 - nextChapter / state.totalChapters);
|
|
192
|
+
if (structuredState.tension < targetTension - 0.2) {
|
|
193
|
+
objectives.push({
|
|
194
|
+
id: 'tension-escalate',
|
|
195
|
+
description: 'Escalate tension toward target level',
|
|
196
|
+
priority: 'high',
|
|
197
|
+
type: 'tension',
|
|
198
|
+
});
|
|
199
|
+
}
|
|
200
|
+
return {
|
|
201
|
+
chapterNumber: nextChapter,
|
|
202
|
+
overallGoal: `Advance the story toward chapter ${nextChapter} with focus on active plot threads`,
|
|
203
|
+
objectives,
|
|
204
|
+
focusCharacters: characters.slice(0, 2).map(c => c.name),
|
|
205
|
+
suggestedScenes: ['Opening scene establishing current situation', 'Development of main plot thread', 'Character interaction moment'],
|
|
206
|
+
tone: 'dramatic',
|
|
207
|
+
notes: 'Auto-generated objectives based on current story state',
|
|
208
|
+
};
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
exports.StoryDirector = StoryDirector;
|
|
212
|
+
exports.storyDirector = new StoryDirector();
|
|
213
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storyDirector.js","sourceRoot":"","sources":["../../src/agents/storyDirector.ts"],"names":[],"mappings":"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","sourcesContent":["import { getLLM } from '../llm/client.js';\r\nimport type { StoryBible, StoryState, ChapterSummary } from '../types/index.js';\r\nimport type { StoryStructuredState } from '../story/structuredState.js';\r\nimport type { TensionGuidance } from './tensionController.js';\r\n\r\nexport interface ChapterObjective {\r\n  id: string;\r\n  description: string;\r\n  priority: 'critical' | 'high' | 'medium' | 'low';\r\n  type: 'plot' | 'character' | 'world' | 'tension' | 'resolution';\r\n  relatedPlotThreadId?: string;\r\n  relatedCharacter?: string;\r\n}\r\n\r\nexport interface DirectorOutput {\r\n  chapterNumber: number;\r\n  overallGoal: string;\r\n  objectives: ChapterObjective[];\r\n  focusCharacters: string[];\r\n  suggestedScenes: string[];\r\n  tone: string;\r\n  notes: string;\r\n}\r\n\r\nexport interface DirectorContext {\r\n  bible: StoryBible;\r\n  state: StoryState;\r\n  structuredState: StoryStructuredState;\r\n  tensionGuidance: TensionGuidance;\r\n  previousSummaries: ChapterSummary[];\r\n}\r\n\r\nconst STORY_DIRECTOR_PROMPT = `You are the Story Director for a narrative AI system. Your job is to decide what the next chapter should accomplish.\r\n\r\n## Story Bible\r\n\r\n**Title:** {{title}}\r\n**Genre:** {{genre}}\r\n**Theme:** {{theme}}\r\n**Premise:** {{premise}}\r\n\r\n## Current Story State\r\n\r\n**Chapter:** {{currentChapter}} / {{totalChapters}}\r\n**Story Tension:** {{storyTension}}%\r\n**Target Tension:** {{targetTension}}%\r\n\r\n### Active Plot Threads\r\n{{plotThreads}}\r\n\r\n### Character States\r\n{{characters}}\r\n\r\n### Unresolved Questions\r\n{{questions}}\r\n\r\n### Recent Events\r\n{{recentEvents}}\r\n\r\n## Tension Guidance\r\n\r\n{{tensionGuidance}}\r\n\r\n## Previous Chapter Summaries\r\n{{summaries}}\r\n\r\n## Your Task\r\n\r\nBased on all the above information, decide what Chapter {{nextChapter}} should accomplish. Consider:\r\n\r\n1. **Plot Progression**: Which plot threads need advancement?\r\n2. **Character Development**: Which characters need focus or growth?\r\n3. **Tension Management**: How should tension change based on target?\r\n4. **Unresolved Questions**: Which mysteries should be addressed?\r\n5. **Story Arc**: Where are we in the overall narrative (setup/rising action/climax/resolution)?\r\n\r\nOutput a JSON object with:\r\n\r\n{\r\n  \"chapterNumber\": {{nextChapter}},\r\n  \"overallGoal\": \"One-sentence description of what this chapter achieves\",\r\n  \"objectives\": [\r\n    {\r\n      \"id\": \"unique-id\",\r\n      \"description\": \"What needs to happen\",\r\n      \"priority\": \"critical|high|medium|low\",\r\n      \"type\": \"plot|character|world|tension|resolution\",\r\n      \"relatedPlotThreadId\": \"thread-id or omit\",\r\n      \"relatedCharacter\": \"character name or omit\"\r\n    }\r\n  ],\r\n  \"focusCharacters\": [\"Character names that should be central\"],\r\n  \"suggestedScenes\": [\"Scene ideas that could achieve objectives\"],\r\n  \"tone\": \"emotional tone for this chapter\",\r\n  \"notes\": \"Additional guidance for the writer\"\r\n}\r\n\r\nBe specific and actionable. The writer will use your direction to craft the chapter.`;\r\n\r\nexport class StoryDirector {\r\n  async direct(context: DirectorContext): Promise<DirectorOutput> {\r\n    const { bible, state, structuredState, tensionGuidance, previousSummaries } = context;\r\n    \r\n    const prompt = this.buildPrompt(bible, state, structuredState, tensionGuidance, previousSummaries);\r\n    \r\n    const result = await getLLM().completeJSON<DirectorOutput>(prompt, {\r\n      temperature: 0.4,\r\n      maxTokens: 2000,\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    tensionGuidance: TensionGuidance,\r\n    summaries: ChapterSummary[]\r\n  ): string {\r\n    const currentTension = Math.round(structuredState.tension * 100);\r\n    const targetTension = Math.round(tensionGuidance.targetTension * 100);\r\n    \r\n    // Format plot threads\r\n    const activeThreads = 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): ${t.summary}`)\r\n      .join('\\n') || 'No active plot threads.';\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 questions\r\n    const questions = structuredState.unresolvedQuestions.length > 0\r\n      ? structuredState.unresolvedQuestions.map(q => `- ${q}`).join('\\n')\r\n      : 'None';\r\n    \r\n    // Format recent events\r\n    const recentEvents = structuredState.recentEvents.length > 0\r\n      ? structuredState.recentEvents.slice(-5).map(e => `- ${e}`).join('\\n')\r\n      : 'None';\r\n    \r\n    // Format tension guidance\r\n    const tensionText = `Target: ${targetTension}%\r\nGuidance: ${tensionGuidance.guidance}\r\nScene Types: ${tensionGuidance.sceneTypes.join(', ')}\r\nPacing: ${tensionGuidance.pacingNotes}`;\r\n    \r\n    // Format summaries (last 3)\r\n    const recentSummaries = summaries.slice(-3);\r\n    const summariesText = recentSummaries.length > 0\r\n      ? recentSummaries.map(s => `Chapter ${s.chapterNumber}: ${s.summary}`).join('\\n')\r\n      : 'No previous chapters.';\r\n    \r\n    return STORY_DIRECTOR_PROMPT\r\n      .replace('{{title}}', bible.title)\r\n      .replace('{{genre}}', bible.genre)\r\n      .replace('{{theme}}', bible.theme)\r\n      .replace('{{premise}}', bible.premise)\r\n      .replace('{{currentChapter}}', state.currentChapter.toString())\r\n      .replace('{{totalChapters}}', state.totalChapters.toString())\r\n      .replace('{{storyTension}}', currentTension.toString())\r\n      .replace('{{targetTension}}', targetTension.toString())\r\n      .replace('{{plotThreads}}', activeThreads)\r\n      .replace('{{characters}}', characters)\r\n      .replace('{{questions}}', questions)\r\n      .replace('{{recentEvents}}', recentEvents)\r\n      .replace('{{tensionGuidance}}', tensionText)\r\n      .replace('{{summaries}}', summariesText)\r\n      .replace(/{{nextChapter}}/g, (state.currentChapter + 1).toString());\r\n  }\r\n  \r\n  /**\r\n   * Format director output for writer prompt\r\n   */\r\n  formatForPrompt(output: DirectorOutput): string {\r\n    const lines: string[] = ['## Chapter Direction'];\r\n    \r\n    lines.push(`\\n**Chapter ${output.chapterNumber} Goal:** ${output.overallGoal}`);\r\n    lines.push(`\\n**Tone:** ${output.tone}`);\r\n    \r\n    if (output.focusCharacters.length > 0) {\r\n      lines.push(`\\n**Focus Characters:** ${output.focusCharacters.join(', ')}`);\r\n    }\r\n    \r\n    if (output.objectives.length > 0) {\r\n      lines.push('\\n**Objectives (in priority order):**');\r\n      const sorted = [...output.objectives].sort((a, b) => {\r\n        const priorityOrder = { critical: 0, high: 1, medium: 2, low: 3 };\r\n        return priorityOrder[a.priority] - priorityOrder[b.priority];\r\n      });\r\n      \r\n      for (const obj of sorted) {\r\n        const emoji = { critical: '🔴', high: '🟠', medium: '🟡', low: '🟢' }[obj.priority];\r\n        lines.push(`${emoji} **[${obj.type.toUpperCase()}]** ${obj.description}`);\r\n      }\r\n    }\r\n    \r\n    if (output.suggestedScenes.length > 0) {\r\n      lines.push('\\n**Suggested Scenes:**');\r\n      for (const scene of output.suggestedScenes) {\r\n        lines.push(`- ${scene}`);\r\n      }\r\n    }\r\n    \r\n    if (output.notes) {\r\n      lines.push(`\\n**Director's Notes:** ${output.notes}`);\r\n    }\r\n    \r\n    return lines.join('\\n');\r\n  }\r\n  \r\n  /**\r\n   * Get quick objectives without LLM call (for testing/fallback)\r\n   */\r\n  generateFallbackObjectives(\r\n    state: StoryState,\r\n    structuredState: StoryStructuredState\r\n  ): DirectorOutput {\r\n    const nextChapter = state.currentChapter + 1;\r\n    const objectives: ChapterObjective[] = [];\r\n    \r\n    // Add plot thread objectives\r\n    const activeThreads = Object.values(structuredState.plotThreads)\r\n      .filter(t => t.status === 'active' || t.status === 'escalating');\r\n    \r\n    for (const thread of activeThreads.slice(0, 2)) {\r\n      objectives.push({\r\n        id: `plot-${thread.id}`,\r\n        description: `Advance the \"${thread.name}\" plot thread`,\r\n        priority: 'high',\r\n        type: 'plot',\r\n        relatedPlotThreadId: thread.id,\r\n      });\r\n    }\r\n    \r\n    // Add character objective\r\n    const characters = Object.values(structuredState.characters);\r\n    if (characters.length > 0) {\r\n      const char = characters[nextChapter % characters.length];\r\n      objectives.push({\r\n        id: `char-${char.name}`,\r\n        description: `Develop ${char.name}'s character arc`,\r\n        priority: 'medium',\r\n        type: 'character',\r\n        relatedCharacter: char.name,\r\n      });\r\n    }\r\n    \r\n    // Add tension objective if needed\r\n    const targetTension = 4 * (nextChapter / state.totalChapters) * (1 - nextChapter / state.totalChapters);\r\n    if (structuredState.tension < targetTension - 0.2) {\r\n      objectives.push({\r\n        id: 'tension-escalate',\r\n        description: 'Escalate tension toward target level',\r\n        priority: 'high',\r\n        type: 'tension',\r\n      });\r\n    }\r\n    \r\n    return {\r\n      chapterNumber: nextChapter,\r\n      overallGoal: `Advance the story toward chapter ${nextChapter} with focus on active plot threads`,\r\n      objectives,\r\n      focusCharacters: characters.slice(0, 2).map(c => c.name),\r\n      suggestedScenes: ['Opening scene establishing current situation', 'Development of main plot thread', 'Character interaction moment'],\r\n      tone: 'dramatic',\r\n      notes: 'Auto-generated objectives based on current story state',\r\n    };\r\n  }\r\n}\r\n\r\nexport const storyDirector = new StoryDirector();\r\n"]}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import type { ChapterSummary } from '../types/index.js';
|
|
2
|
+
export declare class ChapterSummarizer {
|
|
3
|
+
private promptTemplate;
|
|
4
|
+
constructor();
|
|
5
|
+
summarize(chapterText: string, chapterNumber: number): Promise<ChapterSummary>;
|
|
6
|
+
private extractKeyEvents;
|
|
7
|
+
}
|
|
8
|
+
export declare const summarizer: ChapterSummarizer;
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.summarizer = exports.ChapterSummarizer = void 0;
|
|
4
|
+
const client_js_1 = require("../llm/client.js");
|
|
5
|
+
const SUMMARIZER_PROMPT = `Summarize the following chapter in under 120 tokens.
|
|
6
|
+
|
|
7
|
+
Focus on:
|
|
8
|
+
- Major events that occurred
|
|
9
|
+
- Plot progress made
|
|
10
|
+
- Important character changes or revelations
|
|
11
|
+
|
|
12
|
+
## Chapter Text
|
|
13
|
+
|
|
14
|
+
{{chapterText}}
|
|
15
|
+
|
|
16
|
+
## Summary`;
|
|
17
|
+
class ChapterSummarizer {
|
|
18
|
+
promptTemplate;
|
|
19
|
+
constructor() {
|
|
20
|
+
this.promptTemplate = SUMMARIZER_PROMPT;
|
|
21
|
+
}
|
|
22
|
+
async summarize(chapterText, chapterNumber) {
|
|
23
|
+
const prompt = this.promptTemplate.replace('{{chapterText}}', chapterText);
|
|
24
|
+
const summary = await (0, client_js_1.getLLM)().complete(prompt, {
|
|
25
|
+
temperature: 0.3,
|
|
26
|
+
maxTokens: 200,
|
|
27
|
+
});
|
|
28
|
+
return {
|
|
29
|
+
chapterNumber,
|
|
30
|
+
summary: summary.trim(),
|
|
31
|
+
keyEvents: this.extractKeyEvents(chapterText),
|
|
32
|
+
characterChanges: {},
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
extractKeyEvents(text) {
|
|
36
|
+
const sentences = text.match(/[^.!?]+[.!?]+/g) || [];
|
|
37
|
+
const events = [];
|
|
38
|
+
for (const sentence of sentences.slice(0, 20)) {
|
|
39
|
+
const lower = sentence.toLowerCase();
|
|
40
|
+
if (lower.includes('discovered') ||
|
|
41
|
+
lower.includes('realized') ||
|
|
42
|
+
lower.includes('decided') ||
|
|
43
|
+
lower.includes('arrived') ||
|
|
44
|
+
lower.includes('found') ||
|
|
45
|
+
lower.includes('learned')) {
|
|
46
|
+
events.push(sentence.trim());
|
|
47
|
+
}
|
|
48
|
+
if (events.length >= 3)
|
|
49
|
+
break;
|
|
50
|
+
}
|
|
51
|
+
return events;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
exports.ChapterSummarizer = ChapterSummarizer;
|
|
55
|
+
exports.summarizer = new ChapterSummarizer();
|
|
56
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3VtbWFyaXplci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9hZ2VudHMvc3VtbWFyaXplci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxnREFBMEM7QUFHMUMsTUFBTSxpQkFBaUIsR0FBRzs7Ozs7Ozs7Ozs7V0FXZixDQUFDO0FBRVosTUFBYSxpQkFBaUI7SUFDcEIsY0FBYyxDQUFTO0lBRS9CO1FBQ0UsSUFBSSxDQUFDLGNBQWMsR0FBRyxpQkFBaUIsQ0FBQztJQUMxQyxDQUFDO0lBRUQsS0FBSyxDQUFDLFNBQVMsQ0FBQyxXQUFtQixFQUFFLGFBQXFCO1FBQ3hELE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxjQUFjLENBQUMsT0FBTyxDQUFDLGlCQUFpQixFQUFFLFdBQVcsQ0FBQyxDQUFDO1FBRTNFLE1BQU0sT0FBTyxHQUFHLE1BQU0sSUFBQSxrQkFBTSxHQUFFLENBQUMsUUFBUSxDQUFDLE1BQU0sRUFBRTtZQUM5QyxXQUFXLEVBQUUsR0FBRztZQUNoQixTQUFTLEVBQUUsR0FBRztTQUNmLENBQUMsQ0FBQztRQUVILE9BQU87WUFDTCxhQUFhO1lBQ2IsT0FBTyxFQUFFLE9BQU8sQ0FBQyxJQUFJLEVBQUU7WUFDdkIsU0FBUyxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxXQUFXLENBQUM7WUFDN0MsZ0JBQWdCLEVBQUUsRUFBRTtTQUNyQixDQUFDO0lBQ0osQ0FBQztJQUVPLGdCQUFnQixDQUFDLElBQVk7UUFDbkMsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUNyRCxNQUFNLE1BQU0sR0FBYSxFQUFFLENBQUM7UUFFNUIsS0FBSyxNQUFNLFFBQVEsSUFBSSxTQUFTLENBQUMsS0FBSyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsRUFBRSxDQUFDO1lBQzlDLE1BQU0sS0FBSyxHQUFHLFFBQVEsQ0FBQyxXQUFXLEVBQUUsQ0FBQztZQUNyQyxJQUNFLEtBQUssQ0FBQyxRQUFRLENBQUMsWUFBWSxDQUFDO2dCQUM1QixLQUFLLENBQUMsUUFBUSxDQUFDLFVBQVUsQ0FBQztnQkFDMUIsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTLENBQUM7Z0JBQ3pCLEtBQUssQ0FBQyxRQUFRLENBQUMsU0FBUyxDQUFDO2dCQUN6QixLQUFLLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQztnQkFDdkIsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTLENBQUMsRUFDekIsQ0FBQztnQkFDRCxNQUFNLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDO1lBQy9CLENBQUM7WUFDRCxJQUFJLE1BQU0sQ0FBQyxNQUFNLElBQUksQ0FBQztnQkFBRSxNQUFNO1FBQ2hDLENBQUM7UUFFRCxPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDO0NBQ0Y7QUE1Q0QsOENBNENDO0FBRVksUUFBQSxVQUFVLEdBQUcsSUFBSSxpQkFBaUIsRUFBRSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZ2V0TExNIH0gZnJvbSAnLi4vbGxtL2NsaWVudC5qcyc7XHJcbmltcG9ydCB0eXBlIHsgQ2hhcHRlclN1bW1hcnkgfSBmcm9tICcuLi90eXBlcy9pbmRleC5qcyc7XHJcblxyXG5jb25zdCBTVU1NQVJJWkVSX1BST01QVCA9IGBTdW1tYXJpemUgdGhlIGZvbGxvd2luZyBjaGFwdGVyIGluIHVuZGVyIDEyMCB0b2tlbnMuXHJcblxyXG5Gb2N1cyBvbjpcclxuLSBNYWpvciBldmVudHMgdGhhdCBvY2N1cnJlZFxyXG4tIFBsb3QgcHJvZ3Jlc3MgbWFkZVxyXG4tIEltcG9ydGFudCBjaGFyYWN0ZXIgY2hhbmdlcyBvciByZXZlbGF0aW9uc1xyXG5cclxuIyMgQ2hhcHRlciBUZXh0XHJcblxyXG57e2NoYXB0ZXJUZXh0fX1cclxuXHJcbiMjIFN1bW1hcnlgO1xyXG5cclxuZXhwb3J0IGNsYXNzIENoYXB0ZXJTdW1tYXJpemVyIHtcclxuICBwcml2YXRlIHByb21wdFRlbXBsYXRlOiBzdHJpbmc7XHJcblxyXG4gIGNvbnN0cnVjdG9yKCkge1xyXG4gICAgdGhpcy5wcm9tcHRUZW1wbGF0ZSA9IFNVTU1BUklaRVJfUFJPTVBUO1xyXG4gIH1cclxuXHJcbiAgYXN5bmMgc3VtbWFyaXplKGNoYXB0ZXJUZXh0OiBzdHJpbmcsIGNoYXB0ZXJOdW1iZXI6IG51bWJlcik6IFByb21pc2U8Q2hhcHRlclN1bW1hcnk+IHtcclxuICAgIGNvbnN0IHByb21wdCA9IHRoaXMucHJvbXB0VGVtcGxhdGUucmVwbGFjZSgne3tjaGFwdGVyVGV4dH19JywgY2hhcHRlclRleHQpO1xyXG5cclxuICAgIGNvbnN0IHN1bW1hcnkgPSBhd2FpdCBnZXRMTE0oKS5jb21wbGV0ZShwcm9tcHQsIHtcclxuICAgICAgdGVtcGVyYXR1cmU6IDAuMyxcclxuICAgICAgbWF4VG9rZW5zOiAyMDAsXHJcbiAgICB9KTtcclxuXHJcbiAgICByZXR1cm4ge1xyXG4gICAgICBjaGFwdGVyTnVtYmVyLFxyXG4gICAgICBzdW1tYXJ5OiBzdW1tYXJ5LnRyaW0oKSxcclxuICAgICAga2V5RXZlbnRzOiB0aGlzLmV4dHJhY3RLZXlFdmVudHMoY2hhcHRlclRleHQpLFxyXG4gICAgICBjaGFyYWN0ZXJDaGFuZ2VzOiB7fSxcclxuICAgIH07XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGV4dHJhY3RLZXlFdmVudHModGV4dDogc3RyaW5nKTogc3RyaW5nW10ge1xyXG4gICAgY29uc3Qgc2VudGVuY2VzID0gdGV4dC5tYXRjaCgvW14uIT9dK1suIT9dKy9nKSB8fCBbXTtcclxuICAgIGNvbnN0IGV2ZW50czogc3RyaW5nW10gPSBbXTtcclxuICAgIFxyXG4gICAgZm9yIChjb25zdCBzZW50ZW5jZSBvZiBzZW50ZW5jZXMuc2xpY2UoMCwgMjApKSB7XHJcbiAgICAgIGNvbnN0IGxvd2VyID0gc2VudGVuY2UudG9Mb3dlckNhc2UoKTtcclxuICAgICAgaWYgKFxyXG4gICAgICAgIGxvd2VyLmluY2x1ZGVzKCdkaXNjb3ZlcmVkJykgfHxcclxuICAgICAgICBsb3dlci5pbmNsdWRlcygncmVhbGl6ZWQnKSB8fFxyXG4gICAgICAgIGxvd2VyLmluY2x1ZGVzKCdkZWNpZGVkJykgfHxcclxuICAgICAgICBsb3dlci5pbmNsdWRlcygnYXJyaXZlZCcpIHx8XHJcbiAgICAgICAgbG93ZXIuaW5jbHVkZXMoJ2ZvdW5kJykgfHxcclxuICAgICAgICBsb3dlci5pbmNsdWRlcygnbGVhcm5lZCcpXHJcbiAgICAgICkge1xyXG4gICAgICAgIGV2ZW50cy5wdXNoKHNlbnRlbmNlLnRyaW0oKSk7XHJcbiAgICAgIH1cclxuICAgICAgaWYgKGV2ZW50cy5sZW5ndGggPj0gMykgYnJlYWs7XHJcbiAgICB9XHJcbiAgICBcclxuICAgIHJldHVybiBldmVudHM7XHJcbiAgfVxyXG59XHJcblxyXG5leHBvcnQgY29uc3Qgc3VtbWFyaXplciA9IG5ldyBDaGFwdGVyU3VtbWFyaXplcigpO1xyXG4iXX0=
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import type { Chapter, StoryState } from '../types/index.js';
|
|
2
|
+
import type { StoryStructuredState } from '../story/structuredState.js';
|
|
3
|
+
export interface TensionAnalysis {
|
|
4
|
+
currentTension: number;
|
|
5
|
+
targetTension: number;
|
|
6
|
+
tensionGap: number;
|
|
7
|
+
recommendedAction: 'escalate' | 'maintain' | 'resolve' | 'climax';
|
|
8
|
+
reasoning: string;
|
|
9
|
+
}
|
|
10
|
+
export interface TensionGuidance {
|
|
11
|
+
targetTension: number;
|
|
12
|
+
guidance: string;
|
|
13
|
+
sceneTypes: string[];
|
|
14
|
+
pacingNotes: string;
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Calculate target tension using parabolic curve
|
|
18
|
+
* Formula: targetTension = 4 * progress * (1 - progress)
|
|
19
|
+
*
|
|
20
|
+
* This creates a natural dramatic arc:
|
|
21
|
+
* - Chapter 0: 0% tension (setup)
|
|
22
|
+
* - Middle chapters: ~100% tension (peak drama)
|
|
23
|
+
* - Final chapter: 0% tension (resolution)
|
|
24
|
+
*/
|
|
25
|
+
export declare function calculateTargetTension(currentChapter: number, totalChapters: number): number;
|
|
26
|
+
/**
|
|
27
|
+
* Calculate tension for next chapter based on ideal arc
|
|
28
|
+
*/
|
|
29
|
+
export declare function calculateNextChapterTension(currentChapter: number, totalChapters: number): number;
|
|
30
|
+
/**
|
|
31
|
+
* Analyze current tension vs target and provide guidance
|
|
32
|
+
*/
|
|
33
|
+
export declare function analyzeTension(storyState: StoryState, structuredState: StoryStructuredState): TensionAnalysis;
|
|
34
|
+
/**
|
|
35
|
+
* Generate tension guidance for the writer
|
|
36
|
+
*/
|
|
37
|
+
export declare function generateTensionGuidance(analysis: TensionAnalysis, storyState: StoryState): TensionGuidance;
|
|
38
|
+
/**
|
|
39
|
+
* Format tension guidance for writer prompt
|
|
40
|
+
*/
|
|
41
|
+
export declare function formatTensionForPrompt(guidance: TensionGuidance): string;
|
|
42
|
+
/**
|
|
43
|
+
* Calculate tension based on chapter content analysis
|
|
44
|
+
* This is a heuristic for estimating tension from text
|
|
45
|
+
*/
|
|
46
|
+
export declare function estimateTensionFromChapter(chapter: Chapter): number;
|
|
47
|
+
/**
|
|
48
|
+
* TensionController class for managing story tension
|
|
49
|
+
*/
|
|
50
|
+
export declare class TensionController {
|
|
51
|
+
/**
|
|
52
|
+
* Analyze current story state and provide tension guidance
|
|
53
|
+
*/
|
|
54
|
+
analyze(storyState: StoryState, structuredState: StoryStructuredState): TensionAnalysis;
|
|
55
|
+
/**
|
|
56
|
+
* Generate guidance for the next chapter
|
|
57
|
+
*/
|
|
58
|
+
generateGuidance(storyState: StoryState, structuredState: StoryStructuredState): TensionGuidance;
|
|
59
|
+
/**
|
|
60
|
+
* Calculate what the tension should be for a specific chapter
|
|
61
|
+
*/
|
|
62
|
+
calculateTarget(chapterNumber: number, totalChapters: number): number;
|
|
63
|
+
/**
|
|
64
|
+
* Estimate tension from chapter content
|
|
65
|
+
*/
|
|
66
|
+
estimateFromContent(chapter: Chapter): number;
|
|
67
|
+
}
|
|
68
|
+
export declare const tensionController: TensionController;
|