@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,197 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.tensionController = exports.TensionController = void 0;
|
|
4
|
+
exports.calculateTargetTension = calculateTargetTension;
|
|
5
|
+
exports.calculateNextChapterTension = calculateNextChapterTension;
|
|
6
|
+
exports.analyzeTension = analyzeTension;
|
|
7
|
+
exports.generateTensionGuidance = generateTensionGuidance;
|
|
8
|
+
exports.formatTensionForPrompt = formatTensionForPrompt;
|
|
9
|
+
exports.estimateTensionFromChapter = estimateTensionFromChapter;
|
|
10
|
+
/**
|
|
11
|
+
* Calculate target tension using parabolic curve
|
|
12
|
+
* Formula: targetTension = 4 * progress * (1 - progress)
|
|
13
|
+
*
|
|
14
|
+
* This creates a natural dramatic arc:
|
|
15
|
+
* - Chapter 0: 0% tension (setup)
|
|
16
|
+
* - Middle chapters: ~100% tension (peak drama)
|
|
17
|
+
* - Final chapter: 0% tension (resolution)
|
|
18
|
+
*/
|
|
19
|
+
function calculateTargetTension(currentChapter, totalChapters) {
|
|
20
|
+
if (totalChapters <= 1)
|
|
21
|
+
return 0.5;
|
|
22
|
+
// Progress from 0 to 1
|
|
23
|
+
const progress = (currentChapter - 1) / (totalChapters - 1);
|
|
24
|
+
// Parabolic curve: 4 * x * (1 - x)
|
|
25
|
+
// Peak at 0.5 (middle of story)
|
|
26
|
+
const targetTension = 4 * progress * (1 - progress);
|
|
27
|
+
// Round to 2 decimal places
|
|
28
|
+
return Math.round(targetTension * 100) / 100;
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Calculate tension for next chapter based on ideal arc
|
|
32
|
+
*/
|
|
33
|
+
function calculateNextChapterTension(currentChapter, totalChapters) {
|
|
34
|
+
return calculateTargetTension(currentChapter + 1, totalChapters);
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Analyze current tension vs target and provide guidance
|
|
38
|
+
*/
|
|
39
|
+
function analyzeTension(storyState, structuredState) {
|
|
40
|
+
const targetTension = calculateTargetTension(storyState.currentChapter, storyState.totalChapters);
|
|
41
|
+
const currentTension = structuredState.tension;
|
|
42
|
+
const tensionGap = targetTension - currentTension;
|
|
43
|
+
let recommendedAction;
|
|
44
|
+
let reasoning;
|
|
45
|
+
if (storyState.currentChapter === storyState.totalChapters) {
|
|
46
|
+
recommendedAction = 'resolve';
|
|
47
|
+
reasoning = 'Final chapter - bring tensions to resolution';
|
|
48
|
+
}
|
|
49
|
+
else if (tensionGap > 0.2) {
|
|
50
|
+
recommendedAction = 'escalate';
|
|
51
|
+
reasoning = `Tension is ${(tensionGap * 100).toFixed(0)}% below target. Escalate conflict.`;
|
|
52
|
+
}
|
|
53
|
+
else if (tensionGap < -0.15) {
|
|
54
|
+
recommendedAction = 'maintain';
|
|
55
|
+
reasoning = `Tension is ${(Math.abs(tensionGap) * 100).toFixed(0)}% above target. Allow breathing room.`;
|
|
56
|
+
}
|
|
57
|
+
else if (targetTension > 0.85) {
|
|
58
|
+
recommendedAction = 'climax';
|
|
59
|
+
reasoning = 'Near peak tension - build toward climax';
|
|
60
|
+
}
|
|
61
|
+
else {
|
|
62
|
+
recommendedAction = 'maintain';
|
|
63
|
+
reasoning = 'Tension is on track with target arc';
|
|
64
|
+
}
|
|
65
|
+
return {
|
|
66
|
+
currentTension,
|
|
67
|
+
targetTension,
|
|
68
|
+
tensionGap,
|
|
69
|
+
recommendedAction,
|
|
70
|
+
reasoning,
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Generate tension guidance for the writer
|
|
75
|
+
*/
|
|
76
|
+
function generateTensionGuidance(analysis, storyState) {
|
|
77
|
+
const { targetTension, recommendedAction } = analysis;
|
|
78
|
+
let guidance;
|
|
79
|
+
let sceneTypes;
|
|
80
|
+
let pacingNotes;
|
|
81
|
+
switch (recommendedAction) {
|
|
82
|
+
case 'escalate':
|
|
83
|
+
guidance = 'Increase dramatic tension. Introduce complications, raise stakes, or deepen conflicts.';
|
|
84
|
+
sceneTypes = ['confrontation', 'discovery', 'setback', 'danger'];
|
|
85
|
+
pacingNotes = 'Faster pace, shorter scenes, more action/dialogue';
|
|
86
|
+
break;
|
|
87
|
+
case 'maintain':
|
|
88
|
+
guidance = 'Maintain current tension level. Balance action with character moments.';
|
|
89
|
+
sceneTypes = ['development', 'interaction', 'preparation', 'reflection'];
|
|
90
|
+
pacingNotes = 'Moderate pace, mix of action and quiet moments';
|
|
91
|
+
break;
|
|
92
|
+
case 'resolve':
|
|
93
|
+
guidance = 'Begin resolving tensions. Answer questions, conclude arcs, provide closure.';
|
|
94
|
+
sceneTypes = ['resolution', 'revelation', 'farewell', 'new beginning'];
|
|
95
|
+
pacingNotes = 'Slower pace, focus on emotional satisfaction';
|
|
96
|
+
break;
|
|
97
|
+
case 'climax':
|
|
98
|
+
guidance = 'Build toward peak tension. Maximum stakes, critical decisions, turning points.';
|
|
99
|
+
sceneTypes = ['climax', 'showdown', 'revelation', 'sacrifice'];
|
|
100
|
+
pacingNotes = 'Fastest pace, continuous escalation, no relief';
|
|
101
|
+
break;
|
|
102
|
+
default:
|
|
103
|
+
guidance = 'Follow the natural flow of the story.';
|
|
104
|
+
sceneTypes = ['mixed'];
|
|
105
|
+
pacingNotes = 'Balanced pacing';
|
|
106
|
+
}
|
|
107
|
+
return {
|
|
108
|
+
targetTension,
|
|
109
|
+
guidance,
|
|
110
|
+
sceneTypes,
|
|
111
|
+
pacingNotes,
|
|
112
|
+
};
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Format tension guidance for writer prompt
|
|
116
|
+
*/
|
|
117
|
+
function formatTensionForPrompt(guidance) {
|
|
118
|
+
const tensionPercent = Math.round(guidance.targetTension * 100);
|
|
119
|
+
return `## Tension Guidance
|
|
120
|
+
|
|
121
|
+
**Target Tension Level:** ${tensionPercent}%
|
|
122
|
+
|
|
123
|
+
**Guidance:** ${guidance.guidance}
|
|
124
|
+
|
|
125
|
+
**Recommended Scene Types:** ${guidance.sceneTypes.join(', ')}
|
|
126
|
+
|
|
127
|
+
**Pacing Notes:** ${guidance.pacingNotes}
|
|
128
|
+
`;
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Calculate tension based on chapter content analysis
|
|
132
|
+
* This is a heuristic for estimating tension from text
|
|
133
|
+
*/
|
|
134
|
+
function estimateTensionFromChapter(chapter) {
|
|
135
|
+
const content = chapter.content.toLowerCase();
|
|
136
|
+
// Tension indicators
|
|
137
|
+
const tensionWords = [
|
|
138
|
+
'fear', 'danger', 'threat', 'attack', 'fight', 'battle',
|
|
139
|
+
'scream', 'cry', 'panic', 'terror', 'horror',
|
|
140
|
+
'chase', 'escape', 'hide', 'run', 'flee',
|
|
141
|
+
'discover', 'reveal', 'secret', 'mystery', 'truth',
|
|
142
|
+
'confront', 'accuse', 'blame', 'angry', 'furious',
|
|
143
|
+
'worry', 'anxious', 'nervous', 'tense', 'stress',
|
|
144
|
+
'urgent', 'emergency', 'crisis', 'critical', 'desperate',
|
|
145
|
+
];
|
|
146
|
+
const calmWords = [
|
|
147
|
+
'peace', 'calm', 'quiet', 'relax', 'rest',
|
|
148
|
+
'happy', 'joy', 'laugh', 'smile', 'comfort',
|
|
149
|
+
'safe', 'secure', 'peaceful', 'tranquil', 'serene',
|
|
150
|
+
];
|
|
151
|
+
let tensionScore = 0.5; // baseline
|
|
152
|
+
// Count tension indicators
|
|
153
|
+
for (const word of tensionWords) {
|
|
154
|
+
const count = (content.match(new RegExp(word, 'g')) || []).length;
|
|
155
|
+
tensionScore += count * 0.02;
|
|
156
|
+
}
|
|
157
|
+
// Subtract calm indicators
|
|
158
|
+
for (const word of calmWords) {
|
|
159
|
+
const count = (content.match(new RegExp(word, 'g')) || []).length;
|
|
160
|
+
tensionScore -= count * 0.015;
|
|
161
|
+
}
|
|
162
|
+
// Clamp between 0 and 1
|
|
163
|
+
return Math.max(0, Math.min(1, Math.round(tensionScore * 100) / 100));
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
* TensionController class for managing story tension
|
|
167
|
+
*/
|
|
168
|
+
class TensionController {
|
|
169
|
+
/**
|
|
170
|
+
* Analyze current story state and provide tension guidance
|
|
171
|
+
*/
|
|
172
|
+
analyze(storyState, structuredState) {
|
|
173
|
+
return analyzeTension(storyState, structuredState);
|
|
174
|
+
}
|
|
175
|
+
/**
|
|
176
|
+
* Generate guidance for the next chapter
|
|
177
|
+
*/
|
|
178
|
+
generateGuidance(storyState, structuredState) {
|
|
179
|
+
const analysis = this.analyze(storyState, structuredState);
|
|
180
|
+
return generateTensionGuidance(analysis, storyState);
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Calculate what the tension should be for a specific chapter
|
|
184
|
+
*/
|
|
185
|
+
calculateTarget(chapterNumber, totalChapters) {
|
|
186
|
+
return calculateTargetTension(chapterNumber, totalChapters);
|
|
187
|
+
}
|
|
188
|
+
/**
|
|
189
|
+
* Estimate tension from chapter content
|
|
190
|
+
*/
|
|
191
|
+
estimateFromContent(chapter) {
|
|
192
|
+
return estimateTensionFromChapter(chapter);
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
exports.TensionController = TensionController;
|
|
196
|
+
exports.tensionController = new TensionController();
|
|
197
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tensionController.js","sourceRoot":"","sources":["../../src/agents/tensionController.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Chapter, StoryBible, StoryState } from '../types/index.js';\r\nimport type { StoryStructuredState } from '../story/structuredState.js';\r\n\r\nexport interface TensionAnalysis {\r\n  currentTension: number;\r\n  targetTension: number;\r\n  tensionGap: number; // positive = need to increase, negative = need to decrease\r\n  recommendedAction: 'escalate' | 'maintain' | 'resolve' | 'climax';\r\n  reasoning: string;\r\n}\r\n\r\nexport interface TensionGuidance {\r\n  targetTension: number;\r\n  guidance: string;\r\n  sceneTypes: string[];\r\n  pacingNotes: string;\r\n}\r\n\r\n/**\r\n * Calculate target tension using parabolic curve\r\n * Formula: targetTension = 4 * progress * (1 - progress)\r\n * \r\n * This creates a natural dramatic arc:\r\n * - Chapter 0: 0% tension (setup)\r\n * - Middle chapters: ~100% tension (peak drama)\r\n * - Final chapter: 0% tension (resolution)\r\n */\r\nexport function calculateTargetTension(\r\n  currentChapter: number,\r\n  totalChapters: number\r\n): number {\r\n  if (totalChapters <= 1) return 0.5;\r\n  \r\n  // Progress from 0 to 1\r\n  const progress = (currentChapter - 1) / (totalChapters - 1);\r\n  \r\n  // Parabolic curve: 4 * x * (1 - x)\r\n  // Peak at 0.5 (middle of story)\r\n  const targetTension = 4 * progress * (1 - progress);\r\n  \r\n  // Round to 2 decimal places\r\n  return Math.round(targetTension * 100) / 100;\r\n}\r\n\r\n/**\r\n * Calculate tension for next chapter based on ideal arc\r\n */\r\nexport function calculateNextChapterTension(\r\n  currentChapter: number,\r\n  totalChapters: number\r\n): number {\r\n  return calculateTargetTension(currentChapter + 1, totalChapters);\r\n}\r\n\r\n/**\r\n * Analyze current tension vs target and provide guidance\r\n */\r\nexport function analyzeTension(\r\n  storyState: StoryState,\r\n  structuredState: StoryStructuredState\r\n): TensionAnalysis {\r\n  const targetTension = calculateTargetTension(\r\n    storyState.currentChapter,\r\n    storyState.totalChapters\r\n  );\r\n  \r\n  const currentTension = structuredState.tension;\r\n  const tensionGap = targetTension - currentTension;\r\n  \r\n  let recommendedAction: TensionAnalysis['recommendedAction'];\r\n  let reasoning: string;\r\n  \r\n  if (storyState.currentChapter === storyState.totalChapters) {\r\n    recommendedAction = 'resolve';\r\n    reasoning = 'Final chapter - bring tensions to resolution';\r\n  } else if (tensionGap > 0.2) {\r\n    recommendedAction = 'escalate';\r\n    reasoning = `Tension is ${(tensionGap * 100).toFixed(0)}% below target. Escalate conflict.`;\r\n  } else if (tensionGap < -0.15) {\r\n    recommendedAction = 'maintain';\r\n    reasoning = `Tension is ${(Math.abs(tensionGap) * 100).toFixed(0)}% above target. Allow breathing room.`;\r\n  } else if (targetTension > 0.85) {\r\n    recommendedAction = 'climax';\r\n    reasoning = 'Near peak tension - build toward climax';\r\n  } else {\r\n    recommendedAction = 'maintain';\r\n    reasoning = 'Tension is on track with target arc';\r\n  }\r\n  \r\n  return {\r\n    currentTension,\r\n    targetTension,\r\n    tensionGap,\r\n    recommendedAction,\r\n    reasoning,\r\n  };\r\n}\r\n\r\n/**\r\n * Generate tension guidance for the writer\r\n */\r\nexport function generateTensionGuidance(\r\n  analysis: TensionAnalysis,\r\n  storyState: StoryState\r\n): TensionGuidance {\r\n  const { targetTension, recommendedAction } = analysis;\r\n  \r\n  let guidance: string;\r\n  let sceneTypes: string[];\r\n  let pacingNotes: string;\r\n  \r\n  switch (recommendedAction) {\r\n    case 'escalate':\r\n      guidance = 'Increase dramatic tension. Introduce complications, raise stakes, or deepen conflicts.';\r\n      sceneTypes = ['confrontation', 'discovery', 'setback', 'danger'];\r\n      pacingNotes = 'Faster pace, shorter scenes, more action/dialogue';\r\n      break;\r\n      \r\n    case 'maintain':\r\n      guidance = 'Maintain current tension level. Balance action with character moments.';\r\n      sceneTypes = ['development', 'interaction', 'preparation', 'reflection'];\r\n      pacingNotes = 'Moderate pace, mix of action and quiet moments';\r\n      break;\r\n      \r\n    case 'resolve':\r\n      guidance = 'Begin resolving tensions. Answer questions, conclude arcs, provide closure.';\r\n      sceneTypes = ['resolution', 'revelation', 'farewell', 'new beginning'];\r\n      pacingNotes = 'Slower pace, focus on emotional satisfaction';\r\n      break;\r\n      \r\n    case 'climax':\r\n      guidance = 'Build toward peak tension. Maximum stakes, critical decisions, turning points.';\r\n      sceneTypes = ['climax', 'showdown', 'revelation', 'sacrifice'];\r\n      pacingNotes = 'Fastest pace, continuous escalation, no relief';\r\n      break;\r\n      \r\n    default:\r\n      guidance = 'Follow the natural flow of the story.';\r\n      sceneTypes = ['mixed'];\r\n      pacingNotes = 'Balanced pacing';\r\n  }\r\n  \r\n  return {\r\n    targetTension,\r\n    guidance,\r\n    sceneTypes,\r\n    pacingNotes,\r\n  };\r\n}\r\n\r\n/**\r\n * Format tension guidance for writer prompt\r\n */\r\nexport function formatTensionForPrompt(guidance: TensionGuidance): string {\r\n  const tensionPercent = Math.round(guidance.targetTension * 100);\r\n  \r\n  return `## Tension Guidance\r\n\r\n**Target Tension Level:** ${tensionPercent}%\r\n\r\n**Guidance:** ${guidance.guidance}\r\n\r\n**Recommended Scene Types:** ${guidance.sceneTypes.join(', ')}\r\n\r\n**Pacing Notes:** ${guidance.pacingNotes}\r\n`;\r\n}\r\n\r\n/**\r\n * Calculate tension based on chapter content analysis\r\n * This is a heuristic for estimating tension from text\r\n */\r\nexport function estimateTensionFromChapter(chapter: Chapter): number {\r\n  const content = chapter.content.toLowerCase();\r\n  \r\n  // Tension indicators\r\n  const tensionWords = [\r\n    'fear', 'danger', 'threat', 'attack', 'fight', 'battle',\r\n    'scream', 'cry', 'panic', 'terror', 'horror',\r\n    'chase', 'escape', 'hide', 'run', 'flee',\r\n    'discover', 'reveal', 'secret', 'mystery', 'truth',\r\n    'confront', 'accuse', 'blame', 'angry', 'furious',\r\n    'worry', 'anxious', 'nervous', 'tense', 'stress',\r\n    'urgent', 'emergency', 'crisis', 'critical', 'desperate',\r\n  ];\r\n  \r\n  const calmWords = [\r\n    'peace', 'calm', 'quiet', 'relax', 'rest',\r\n    'happy', 'joy', 'laugh', 'smile', 'comfort',\r\n    'safe', 'secure', 'peaceful', 'tranquil', 'serene',\r\n  ];\r\n  \r\n  let tensionScore = 0.5; // baseline\r\n  \r\n  // Count tension indicators\r\n  for (const word of tensionWords) {\r\n    const count = (content.match(new RegExp(word, 'g')) || []).length;\r\n    tensionScore += count * 0.02;\r\n  }\r\n  \r\n  // Subtract calm indicators\r\n  for (const word of calmWords) {\r\n    const count = (content.match(new RegExp(word, 'g')) || []).length;\r\n    tensionScore -= count * 0.015;\r\n  }\r\n  \r\n  // Clamp between 0 and 1\r\n  return Math.max(0, Math.min(1, Math.round(tensionScore * 100) / 100));\r\n}\r\n\r\n/**\r\n * TensionController class for managing story tension\r\n */\r\nexport class TensionController {\r\n  /**\r\n   * Analyze current story state and provide tension guidance\r\n   */\r\n  analyze(\r\n    storyState: StoryState,\r\n    structuredState: StoryStructuredState\r\n  ): TensionAnalysis {\r\n    return analyzeTension(storyState, structuredState);\r\n  }\r\n  \r\n  /**\r\n   * Generate guidance for the next chapter\r\n   */\r\n  generateGuidance(\r\n    storyState: StoryState,\r\n    structuredState: StoryStructuredState\r\n  ): TensionGuidance {\r\n    const analysis = this.analyze(storyState, structuredState);\r\n    return generateTensionGuidance(analysis, storyState);\r\n  }\r\n  \r\n  /**\r\n   * Calculate what the tension should be for a specific chapter\r\n   */\r\n  calculateTarget(chapterNumber: number, totalChapters: number): number {\r\n    return calculateTargetTension(chapterNumber, totalChapters);\r\n  }\r\n  \r\n  /**\r\n   * Estimate tension from chapter content\r\n   */\r\n  estimateFromContent(chapter: Chapter): number {\r\n    return estimateTensionFromChapter(chapter);\r\n  }\r\n}\r\n\r\nexport const tensionController = new TensionController();\r\n"]}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import type { GenerationContext, WriterOutput } from '../types/index.js';
|
|
2
|
+
import type { CanonStore } from '../memory/canonStore.js';
|
|
3
|
+
import type { MemoryRetriever } from '../memory/memoryRetriever.js';
|
|
4
|
+
export declare class ChapterWriter {
|
|
5
|
+
private promptTemplate;
|
|
6
|
+
constructor();
|
|
7
|
+
write(context: GenerationContext, canon?: CanonStore, memoryRetriever?: MemoryRetriever): Promise<WriterOutput>;
|
|
8
|
+
continue(existingContent: string, context: GenerationContext): Promise<string>;
|
|
9
|
+
private inferChapterGoal;
|
|
10
|
+
private extractTitle;
|
|
11
|
+
}
|
|
12
|
+
export declare const writer: ChapterWriter;
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.writer = exports.ChapterWriter = void 0;
|
|
4
|
+
const client_js_1 = require("../llm/client.js");
|
|
5
|
+
const canonStore_js_1 = require("../memory/canonStore.js");
|
|
6
|
+
const WRITER_PROMPT = `You are a professional novelist writing immersive narrative prose.
|
|
7
|
+
|
|
8
|
+
Write Chapter {{chapterNumber}} of the novel.
|
|
9
|
+
|
|
10
|
+
## Story Bible
|
|
11
|
+
|
|
12
|
+
**Title:** {{title}}
|
|
13
|
+
**Theme:** {{theme}}
|
|
14
|
+
**Genre:** {{genre}}
|
|
15
|
+
**Setting:** {{setting}}
|
|
16
|
+
**Tone:** {{tone}}
|
|
17
|
+
|
|
18
|
+
**Premise:**
|
|
19
|
+
{{premise}}
|
|
20
|
+
|
|
21
|
+
## Characters
|
|
22
|
+
|
|
23
|
+
{{characters}}
|
|
24
|
+
|
|
25
|
+
## Story Canon
|
|
26
|
+
|
|
27
|
+
{{canon}}
|
|
28
|
+
|
|
29
|
+
## Relevant Memories
|
|
30
|
+
|
|
31
|
+
{{memories}}
|
|
32
|
+
|
|
33
|
+
## Recent Chapter Summaries
|
|
34
|
+
|
|
35
|
+
{{recentSummaries}}
|
|
36
|
+
|
|
37
|
+
## Chapter Goal
|
|
38
|
+
|
|
39
|
+
{{chapterGoal}}
|
|
40
|
+
|
|
41
|
+
## Writing Guidelines
|
|
42
|
+
|
|
43
|
+
- Write in third person limited perspective
|
|
44
|
+
- Show, don't tell
|
|
45
|
+
- Maintain consistent character voices
|
|
46
|
+
- Build toward the chapter goal naturally
|
|
47
|
+
- End at a natural stopping point (not mid-scene)
|
|
48
|
+
- Reference relevant past memories naturally when appropriate
|
|
49
|
+
- Target length: {{targetWordCount}} words
|
|
50
|
+
|
|
51
|
+
Write the full chapter now.`;
|
|
52
|
+
class ChapterWriter {
|
|
53
|
+
promptTemplate;
|
|
54
|
+
constructor() {
|
|
55
|
+
this.promptTemplate = WRITER_PROMPT;
|
|
56
|
+
}
|
|
57
|
+
async write(context, canon, memoryRetriever) {
|
|
58
|
+
const { bible, state, chapterNumber, targetWordCount = 2000 } = context;
|
|
59
|
+
const canonSection = canon ? (0, canonStore_js_1.formatCanonForPrompt)(canon) : '';
|
|
60
|
+
// Retrieve relevant memories
|
|
61
|
+
let memoriesSection = 'No relevant memories yet.';
|
|
62
|
+
if (memoryRetriever && chapterNumber > 1) {
|
|
63
|
+
const memories = await memoryRetriever.retrieveForChapter({
|
|
64
|
+
bible,
|
|
65
|
+
state,
|
|
66
|
+
currentChapter: chapterNumber,
|
|
67
|
+
});
|
|
68
|
+
memoriesSection = memoryRetriever.formatMemoriesForPrompt(memories) || 'No highly relevant memories for this chapter.';
|
|
69
|
+
}
|
|
70
|
+
const recentSummaries = state.chapterSummaries
|
|
71
|
+
.slice(-3)
|
|
72
|
+
.map(s => `Chapter ${s.chapterNumber}: ${s.summary}`)
|
|
73
|
+
.join('\n\n') || 'This is the first chapter.';
|
|
74
|
+
const characters = bible.characters
|
|
75
|
+
.map(c => `- ${c.name} (${c.role}): ${c.personality.join(', ')}. Goals: ${c.goals.join(', ')}`)
|
|
76
|
+
.join('\n');
|
|
77
|
+
const chapterGoal = this.inferChapterGoal(bible, state, chapterNumber);
|
|
78
|
+
const prompt = this.promptTemplate
|
|
79
|
+
.replace('{{chapterNumber}}', chapterNumber.toString())
|
|
80
|
+
.replace('{{title}}', bible.title)
|
|
81
|
+
.replace('{{theme}}', bible.theme)
|
|
82
|
+
.replace('{{genre}}', bible.genre)
|
|
83
|
+
.replace('{{setting}}', bible.setting)
|
|
84
|
+
.replace('{{tone}}', bible.tone)
|
|
85
|
+
.replace('{{premise}}', bible.premise)
|
|
86
|
+
.replace('{{characters}}', characters)
|
|
87
|
+
.replace('{{memories}}', memoriesSection)
|
|
88
|
+
.replace('{{recentSummaries}}', recentSummaries)
|
|
89
|
+
.replace('{{chapterGoal}}', chapterGoal)
|
|
90
|
+
.replace('{{targetWordCount}}', targetWordCount.toString())
|
|
91
|
+
.replace('{{canon}}', canonSection);
|
|
92
|
+
const content = await (0, client_js_1.getLLM)().complete(prompt, {
|
|
93
|
+
temperature: 0.8,
|
|
94
|
+
maxTokens: 4000,
|
|
95
|
+
});
|
|
96
|
+
const title = this.extractTitle(content) || `Chapter ${chapterNumber}`;
|
|
97
|
+
const wordCount = content.split(/\s+/).length;
|
|
98
|
+
return { content, title, wordCount };
|
|
99
|
+
}
|
|
100
|
+
async continue(existingContent, context) {
|
|
101
|
+
const prompt = `Continue the following chapter from where it left off. Do not repeat what has already been written. Write the continuation only.
|
|
102
|
+
|
|
103
|
+
## Story Context
|
|
104
|
+
|
|
105
|
+
${context.bible.premise}
|
|
106
|
+
|
|
107
|
+
## Existing Chapter Content
|
|
108
|
+
|
|
109
|
+
${existingContent}
|
|
110
|
+
|
|
111
|
+
## Continuation
|
|
112
|
+
|
|
113
|
+
Continue naturally from the last sentence:`;
|
|
114
|
+
const continuation = await (0, client_js_1.getLLM)().complete(prompt, {
|
|
115
|
+
temperature: 0.8,
|
|
116
|
+
maxTokens: 2000,
|
|
117
|
+
});
|
|
118
|
+
return existingContent + '\n\n' + continuation;
|
|
119
|
+
}
|
|
120
|
+
inferChapterGoal(bible, state, chapterNumber) {
|
|
121
|
+
const progress = chapterNumber / bible.targetChapters;
|
|
122
|
+
if (progress < 0.2) {
|
|
123
|
+
return 'Establish the setting, introduce the main character, and hint at the central conflict.';
|
|
124
|
+
}
|
|
125
|
+
else if (progress < 0.5) {
|
|
126
|
+
return 'Develop the conflict, introduce complications, and raise stakes.';
|
|
127
|
+
}
|
|
128
|
+
else if (progress < 0.8) {
|
|
129
|
+
return 'Escalate tension, reveal important information, and move toward climax.';
|
|
130
|
+
}
|
|
131
|
+
else {
|
|
132
|
+
return 'Build toward resolution and begin tying up plot threads.';
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
extractTitle(content) {
|
|
136
|
+
const lines = content.split('\n');
|
|
137
|
+
for (const line of lines.slice(0, 10)) {
|
|
138
|
+
const trimmed = line.trim();
|
|
139
|
+
if (trimmed.startsWith('# ') || trimmed.startsWith('Chapter')) {
|
|
140
|
+
return trimmed.replace(/^#\s*/, '').trim();
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
return null;
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
exports.ChapterWriter = ChapterWriter;
|
|
147
|
+
exports.writer = new ChapterWriter();
|
|
148
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"writer.js","sourceRoot":"","sources":["../../src/agents/writer.ts"],"names":[],"mappings":"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","sourcesContent":["import { getLLM } from '../llm/client.js';\r\nimport type { GenerationContext, WriterOutput } from '../types/index.js';\r\nimport type { CanonStore } from '../memory/canonStore.js';\r\nimport { formatCanonForPrompt } from '../memory/canonStore.js';\r\nimport type { MemoryRetriever } from '../memory/memoryRetriever.js';\r\n\r\nconst WRITER_PROMPT = `You are a professional novelist writing immersive narrative prose.\r\n\r\nWrite Chapter {{chapterNumber}} of the novel.\r\n\r\n## Story Bible\r\n\r\n**Title:** {{title}}\r\n**Theme:** {{theme}}\r\n**Genre:** {{genre}}\r\n**Setting:** {{setting}}\r\n**Tone:** {{tone}}\r\n\r\n**Premise:**\r\n{{premise}}\r\n\r\n## Characters\r\n\r\n{{characters}}\r\n\r\n## Story Canon\r\n\r\n{{canon}}\r\n\r\n## Relevant Memories\r\n\r\n{{memories}}\r\n\r\n## Recent Chapter Summaries\r\n\r\n{{recentSummaries}}\r\n\r\n## Chapter Goal\r\n\r\n{{chapterGoal}}\r\n\r\n## Writing Guidelines\r\n\r\n- Write in third person limited perspective\r\n- Show, don't tell\r\n- Maintain consistent character voices\r\n- Build toward the chapter goal naturally\r\n- End at a natural stopping point (not mid-scene)\r\n- Reference relevant past memories naturally when appropriate\r\n- Target length: {{targetWordCount}} words\r\n\r\nWrite the full chapter now.`;\r\n\r\nexport class ChapterWriter {\r\n  private promptTemplate: string;\r\n\r\n  constructor() {\r\n    this.promptTemplate = WRITER_PROMPT;\r\n  }\r\n\r\n  async write(context: GenerationContext, canon?: CanonStore, memoryRetriever?: MemoryRetriever): Promise<WriterOutput> {\r\n    const { bible, state, chapterNumber, targetWordCount = 2000 } = context;\r\n    \r\n    const canonSection = canon ? formatCanonForPrompt(canon) : '';\r\n\r\n    // Retrieve relevant memories\r\n    let memoriesSection = 'No relevant memories yet.';\r\n    if (memoryRetriever && chapterNumber > 1) {\r\n      const memories = await memoryRetriever.retrieveForChapter({\r\n        bible,\r\n        state,\r\n        currentChapter: chapterNumber,\r\n      });\r\n      memoriesSection = memoryRetriever.formatMemoriesForPrompt(memories) || 'No highly relevant memories for this chapter.';\r\n    }\r\n\r\n    const recentSummaries = state.chapterSummaries\r\n      .slice(-3)\r\n      .map(s => `Chapter ${s.chapterNumber}: ${s.summary}`)\r\n      .join('\\n\\n') || 'This is the first chapter.';\r\n\r\n    const characters = bible.characters\r\n      .map(c => `- ${c.name} (${c.role}): ${c.personality.join(', ')}. Goals: ${c.goals.join(', ')}`)\r\n      .join('\\n');\r\n\r\n    const chapterGoal = this.inferChapterGoal(bible, state, chapterNumber);\r\n\r\n    const prompt = this.promptTemplate\r\n      .replace('{{chapterNumber}}', chapterNumber.toString())\r\n      .replace('{{title}}', bible.title)\r\n      .replace('{{theme}}', bible.theme)\r\n      .replace('{{genre}}', bible.genre)\r\n      .replace('{{setting}}', bible.setting)\r\n      .replace('{{tone}}', bible.tone)\r\n      .replace('{{premise}}', bible.premise)\r\n      .replace('{{characters}}', characters)\r\n      .replace('{{memories}}', memoriesSection)\r\n      .replace('{{recentSummaries}}', recentSummaries)\r\n      .replace('{{chapterGoal}}', chapterGoal)\r\n      .replace('{{targetWordCount}}', targetWordCount.toString())\r\n      .replace('{{canon}}', canonSection);\r\n\r\n    const content = await getLLM().complete(prompt, {\r\n      temperature: 0.8,\r\n      maxTokens: 4000,\r\n    });\r\n\r\n    const title = this.extractTitle(content) || `Chapter ${chapterNumber}`;\r\n    const wordCount = content.split(/\\s+/).length;\r\n\r\n    return { content, title, wordCount };\r\n  }\r\n\r\n  async continue(existingContent: string, context: GenerationContext): Promise<string> {\r\n    const prompt = `Continue the following chapter from where it left off. Do not repeat what has already been written. Write the continuation only.\r\n\r\n## Story Context\r\n\r\n${context.bible.premise}\r\n\r\n## Existing Chapter Content\r\n\r\n${existingContent}\r\n\r\n## Continuation\r\n\r\nContinue naturally from the last sentence:`;\r\n\r\n    const continuation = await getLLM().complete(prompt, {\r\n      temperature: 0.8,\r\n      maxTokens: 2000,\r\n    });\r\n\r\n    return existingContent + '\\n\\n' + continuation;\r\n  }\r\n\r\n  private inferChapterGoal(bible: GenerationContext['bible'], state: GenerationContext['state'], chapterNumber: number): string {\r\n    const progress = chapterNumber / bible.targetChapters;\r\n    \r\n    if (progress < 0.2) {\r\n      return 'Establish the setting, introduce the main character, and hint at the central conflict.';\r\n    } else if (progress < 0.5) {\r\n      return 'Develop the conflict, introduce complications, and raise stakes.';\r\n    } else if (progress < 0.8) {\r\n      return 'Escalate tension, reveal important information, and move toward climax.';\r\n    } else {\r\n      return 'Build toward resolution and begin tying up plot threads.';\r\n    }\r\n  }\r\n\r\n  private extractTitle(content: string): string | null {\r\n    const lines = content.split('\\n');\r\n    for (const line of lines.slice(0, 10)) {\r\n      const trimmed = line.trim();\r\n      if (trimmed.startsWith('# ') || trimmed.startsWith('Chapter')) {\r\n        return trimmed.replace(/^#\\s*/, '').trim();\r\n      }\r\n    }\r\n    return null;\r\n  }\r\n}\r\n\r\nexport const writer = new ChapterWriter();\r\n"]}
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
import type { CharacterState } from '../story/structuredState.js';
|
|
2
|
+
export type NodeType = 'character' | 'location' | 'fact' | 'event' | 'item';
|
|
3
|
+
export interface ConstraintNode {
|
|
4
|
+
id: string;
|
|
5
|
+
type: NodeType;
|
|
6
|
+
label: string;
|
|
7
|
+
properties: Record<string, any>;
|
|
8
|
+
chapterEstablished?: number;
|
|
9
|
+
}
|
|
10
|
+
export interface ConstraintEdge {
|
|
11
|
+
id: string;
|
|
12
|
+
from: string;
|
|
13
|
+
to: string;
|
|
14
|
+
type: string;
|
|
15
|
+
properties: Record<string, any>;
|
|
16
|
+
}
|
|
17
|
+
export interface ConstraintViolation {
|
|
18
|
+
type: 'canon' | 'location' | 'knowledge' | 'timeline' | 'logic';
|
|
19
|
+
severity: 'error' | 'warning';
|
|
20
|
+
description: string;
|
|
21
|
+
nodes: string[];
|
|
22
|
+
suggestedFix?: string;
|
|
23
|
+
}
|
|
24
|
+
export declare class ConstraintGraph {
|
|
25
|
+
private nodes;
|
|
26
|
+
private edges;
|
|
27
|
+
private adjacencyList;
|
|
28
|
+
/**
|
|
29
|
+
* Add a node to the graph
|
|
30
|
+
*/
|
|
31
|
+
addNode(node: ConstraintNode): void;
|
|
32
|
+
/**
|
|
33
|
+
* Add an edge between nodes
|
|
34
|
+
*/
|
|
35
|
+
addEdge(edge: ConstraintEdge): void;
|
|
36
|
+
/**
|
|
37
|
+
* Get a node by ID
|
|
38
|
+
*/
|
|
39
|
+
getNode(id: string): ConstraintNode | undefined;
|
|
40
|
+
/**
|
|
41
|
+
* Get all edges from a node
|
|
42
|
+
*/
|
|
43
|
+
getEdgesFrom(nodeId: string): ConstraintEdge[];
|
|
44
|
+
/**
|
|
45
|
+
* Get all edges to a node
|
|
46
|
+
*/
|
|
47
|
+
getEdgesTo(nodeId: string): ConstraintEdge[];
|
|
48
|
+
/**
|
|
49
|
+
* Get neighbors of a node
|
|
50
|
+
*/
|
|
51
|
+
getNeighbors(nodeId: string): ConstraintNode[];
|
|
52
|
+
/**
|
|
53
|
+
* Add character to graph
|
|
54
|
+
*/
|
|
55
|
+
addCharacter(character: CharacterState, chapter: number): void;
|
|
56
|
+
/**
|
|
57
|
+
* Add location to graph
|
|
58
|
+
*/
|
|
59
|
+
addLocation(name: string, description: string, chapter: number): void;
|
|
60
|
+
/**
|
|
61
|
+
* Add event to graph
|
|
62
|
+
*/
|
|
63
|
+
addEvent(id: string, description: string, participants: string[], chapter: number): void;
|
|
64
|
+
/**
|
|
65
|
+
* Update character location
|
|
66
|
+
*/
|
|
67
|
+
updateCharacterLocation(characterName: string, newLocation: string, chapter: number): void;
|
|
68
|
+
/**
|
|
69
|
+
* Check for constraint violations
|
|
70
|
+
*/
|
|
71
|
+
checkConstraints(currentChapter: number): ConstraintViolation[];
|
|
72
|
+
/**
|
|
73
|
+
* Check location consistency (no teleporting)
|
|
74
|
+
*/
|
|
75
|
+
private checkLocationConsistency;
|
|
76
|
+
/**
|
|
77
|
+
* Check knowledge consistency (no impossible knowledge)
|
|
78
|
+
*/
|
|
79
|
+
private checkKnowledgeConsistency;
|
|
80
|
+
/**
|
|
81
|
+
* Check timeline consistency
|
|
82
|
+
*/
|
|
83
|
+
private checkTimelineConsistency;
|
|
84
|
+
/**
|
|
85
|
+
* Check logical consistency
|
|
86
|
+
*/
|
|
87
|
+
private checkLogicalConsistency;
|
|
88
|
+
/**
|
|
89
|
+
* Query what a character knows
|
|
90
|
+
*/
|
|
91
|
+
getCharacterKnowledge(characterName: string): ConstraintNode[];
|
|
92
|
+
/**
|
|
93
|
+
* Query where a character is
|
|
94
|
+
*/
|
|
95
|
+
getCharacterLocation(characterName: string): string | undefined;
|
|
96
|
+
/**
|
|
97
|
+
* Serialize graph
|
|
98
|
+
*/
|
|
99
|
+
serialize(): string;
|
|
100
|
+
/**
|
|
101
|
+
* Load graph from serialized data
|
|
102
|
+
*/
|
|
103
|
+
load(data: string): void;
|
|
104
|
+
/**
|
|
105
|
+
* Get graph statistics
|
|
106
|
+
*/
|
|
107
|
+
getStats(): {
|
|
108
|
+
nodes: number;
|
|
109
|
+
edges: number;
|
|
110
|
+
byType: Record<NodeType, number>;
|
|
111
|
+
};
|
|
112
|
+
/**
|
|
113
|
+
* Sanitize string for use as ID
|
|
114
|
+
*/
|
|
115
|
+
private sanitizeId;
|
|
116
|
+
}
|
|
117
|
+
export declare function createConstraintGraph(): ConstraintGraph;
|