@narrative-os/engine 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (71) hide show
  1. package/dist/agents/canonValidator.d.ts +9 -0
  2. package/dist/agents/canonValidator.js +51 -0
  3. package/dist/agents/chapterPlanner.d.ts +50 -0
  4. package/dist/agents/chapterPlanner.js +250 -0
  5. package/dist/agents/completeness.d.ts +7 -0
  6. package/dist/agents/completeness.js +51 -0
  7. package/dist/agents/memoryExtractor.d.ts +12 -0
  8. package/dist/agents/memoryExtractor.js +82 -0
  9. package/dist/agents/stateUpdater.d.ts +30 -0
  10. package/dist/agents/stateUpdater.js +150 -0
  11. package/dist/agents/storyDirector.d.ts +40 -0
  12. package/dist/agents/storyDirector.js +213 -0
  13. package/dist/agents/summarizer.d.ts +8 -0
  14. package/dist/agents/summarizer.js +56 -0
  15. package/dist/agents/tensionController.d.ts +68 -0
  16. package/dist/agents/tensionController.js +197 -0
  17. package/dist/agents/writer.d.ts +12 -0
  18. package/dist/agents/writer.js +148 -0
  19. package/dist/constraints/constraintGraph.d.ts +117 -0
  20. package/dist/constraints/constraintGraph.js +381 -0
  21. package/dist/constraints/validator.d.ts +58 -0
  22. package/dist/constraints/validator.js +236 -0
  23. package/dist/index.d.ts +25 -0
  24. package/dist/index.js +115 -0
  25. package/dist/llm/client.d.ts +14 -0
  26. package/dist/llm/client.js +108 -0
  27. package/dist/memory/canonStore.d.ts +20 -0
  28. package/dist/memory/canonStore.js +110 -0
  29. package/dist/memory/memoryRetriever.d.ts +28 -0
  30. package/dist/memory/memoryRetriever.js +126 -0
  31. package/dist/memory/stateUpdater.d.ts +49 -0
  32. package/dist/memory/stateUpdater.js +315 -0
  33. package/dist/memory/vectorStore.d.ts +41 -0
  34. package/dist/memory/vectorStore.js +166 -0
  35. package/dist/pipeline/generateChapter.d.ts +17 -0
  36. package/dist/pipeline/generateChapter.js +75 -0
  37. package/dist/story/bible.d.ts +4 -0
  38. package/dist/story/bible.js +53 -0
  39. package/dist/story/state.d.ts +3 -0
  40. package/dist/story/state.js +27 -0
  41. package/dist/story/structuredState.d.ts +39 -0
  42. package/dist/story/structuredState.js +159 -0
  43. package/dist/test/canon.test.d.ts +1 -0
  44. package/dist/test/canon.test.js +104 -0
  45. package/dist/test/chapter-planner.test.d.ts +1 -0
  46. package/dist/test/chapter-planner.test.js +171 -0
  47. package/dist/test/constraints.test.d.ts +1 -0
  48. package/dist/test/constraints.test.js +210 -0
  49. package/dist/test/simple.test.d.ts +1 -0
  50. package/dist/test/simple.test.js +51 -0
  51. package/dist/test/state-updater.test.d.ts +1 -0
  52. package/dist/test/state-updater.test.js +200 -0
  53. package/dist/test/story-director.test.d.ts +1 -0
  54. package/dist/test/story-director.test.js +142 -0
  55. package/dist/test/structured-state.test.d.ts +1 -0
  56. package/dist/test/structured-state.test.js +144 -0
  57. package/dist/test/tension-controller.test.d.ts +1 -0
  58. package/dist/test/tension-controller.test.js +116 -0
  59. package/dist/test/vector-memory.test.d.ts +1 -0
  60. package/dist/test/vector-memory.test.js +153 -0
  61. package/dist/test/world-simulation.test.d.ts +1 -0
  62. package/dist/test/world-simulation.test.js +152 -0
  63. package/dist/types/index.d.ts +79 -0
  64. package/dist/types/index.js +3 -0
  65. package/dist/world/characterAgent.d.ts +73 -0
  66. package/dist/world/characterAgent.js +232 -0
  67. package/dist/world/eventResolver.d.ts +52 -0
  68. package/dist/world/eventResolver.js +205 -0
  69. package/dist/world/worldState.d.ts +93 -0
  70. package/dist/world/worldState.js +258 -0
  71. package/package.json +43 -0
@@ -0,0 +1,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;