@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,381 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ConstraintGraph = void 0;
4
+ exports.createConstraintGraph = createConstraintGraph;
5
+ class ConstraintGraph {
6
+ nodes = new Map();
7
+ edges = new Map();
8
+ adjacencyList = new Map();
9
+ /**
10
+ * Add a node to the graph
11
+ */
12
+ addNode(node) {
13
+ this.nodes.set(node.id, node);
14
+ if (!this.adjacencyList.has(node.id)) {
15
+ this.adjacencyList.set(node.id, new Set());
16
+ }
17
+ }
18
+ /**
19
+ * Add an edge between nodes
20
+ */
21
+ addEdge(edge) {
22
+ // Validate nodes exist
23
+ if (!this.nodes.has(edge.from) || !this.nodes.has(edge.to)) {
24
+ throw new Error(`Cannot add edge: node not found (${edge.from} -> ${edge.to})`);
25
+ }
26
+ this.edges.set(edge.id, edge);
27
+ // Update adjacency list
28
+ const fromSet = this.adjacencyList.get(edge.from);
29
+ if (fromSet) {
30
+ fromSet.add(edge.to);
31
+ }
32
+ }
33
+ /**
34
+ * Get a node by ID
35
+ */
36
+ getNode(id) {
37
+ return this.nodes.get(id);
38
+ }
39
+ /**
40
+ * Get all edges from a node
41
+ */
42
+ getEdgesFrom(nodeId) {
43
+ return Array.from(this.edges.values()).filter(e => e.from === nodeId);
44
+ }
45
+ /**
46
+ * Get all edges to a node
47
+ */
48
+ getEdgesTo(nodeId) {
49
+ return Array.from(this.edges.values()).filter(e => e.to === nodeId);
50
+ }
51
+ /**
52
+ * Get neighbors of a node
53
+ */
54
+ getNeighbors(nodeId) {
55
+ const neighborIds = this.adjacencyList.get(nodeId);
56
+ if (!neighborIds)
57
+ return [];
58
+ return Array.from(neighborIds)
59
+ .map(id => this.nodes.get(id))
60
+ .filter((n) => n !== undefined);
61
+ }
62
+ /**
63
+ * Add character to graph
64
+ */
65
+ addCharacter(character, chapter) {
66
+ const charNode = {
67
+ id: `char-${character.name}`,
68
+ type: 'character',
69
+ label: character.name,
70
+ properties: {
71
+ emotionalState: character.emotionalState,
72
+ location: character.location,
73
+ goals: character.goals,
74
+ },
75
+ chapterEstablished: chapter,
76
+ };
77
+ this.addNode(charNode);
78
+ // Add location edge
79
+ this.addEdge({
80
+ id: `edge-${character.name}-loc`,
81
+ from: charNode.id,
82
+ to: `loc-${character.location}`,
83
+ type: 'located_at',
84
+ properties: { since: chapter },
85
+ });
86
+ // Add knowledge nodes and edges
87
+ for (const knowledge of character.knowledge) {
88
+ const factId = `fact-${this.sanitizeId(knowledge)}`;
89
+ if (!this.nodes.has(factId)) {
90
+ this.addNode({
91
+ id: factId,
92
+ type: 'fact',
93
+ label: knowledge,
94
+ properties: {},
95
+ chapterEstablished: chapter,
96
+ });
97
+ }
98
+ this.addEdge({
99
+ id: `edge-${character.name}-knows-${factId}`,
100
+ from: charNode.id,
101
+ to: factId,
102
+ type: 'knows',
103
+ properties: { since: chapter },
104
+ });
105
+ }
106
+ }
107
+ /**
108
+ * Add location to graph
109
+ */
110
+ addLocation(name, description, chapter) {
111
+ const locNode = {
112
+ id: `loc-${name}`,
113
+ type: 'location',
114
+ label: name,
115
+ properties: { description },
116
+ chapterEstablished: chapter,
117
+ };
118
+ this.addNode(locNode);
119
+ }
120
+ /**
121
+ * Add event to graph
122
+ */
123
+ addEvent(id, description, participants, chapter) {
124
+ const eventNode = {
125
+ id: `event-${id}`,
126
+ type: 'event',
127
+ label: description,
128
+ properties: { participants },
129
+ chapterEstablished: chapter,
130
+ };
131
+ this.addNode(eventNode);
132
+ // Connect participants to event
133
+ for (const participant of participants) {
134
+ const charId = `char-${participant}`;
135
+ if (this.nodes.has(charId)) {
136
+ this.addEdge({
137
+ id: `edge-${participant}-participates-${id}`,
138
+ from: charId,
139
+ to: eventNode.id,
140
+ type: 'participates_in',
141
+ properties: { chapter },
142
+ });
143
+ }
144
+ }
145
+ }
146
+ /**
147
+ * Update character location
148
+ */
149
+ updateCharacterLocation(characterName, newLocation, chapter) {
150
+ const charId = `char-${characterName}`;
151
+ const charNode = this.nodes.get(charId);
152
+ if (charNode) {
153
+ // Update properties
154
+ charNode.properties.location = newLocation;
155
+ // Remove old location edges
156
+ const oldEdges = this.getEdgesFrom(charId).filter(e => e.type === 'located_at');
157
+ for (const edge of oldEdges) {
158
+ this.edges.delete(edge.id);
159
+ const adjSet = this.adjacencyList.get(edge.from);
160
+ if (adjSet) {
161
+ adjSet.delete(edge.to);
162
+ }
163
+ }
164
+ // Add new location edge
165
+ this.addEdge({
166
+ id: `edge-${characterName}-loc-${chapter}`,
167
+ from: charId,
168
+ to: `loc-${newLocation}`,
169
+ type: 'located_at',
170
+ properties: { since: chapter },
171
+ });
172
+ }
173
+ }
174
+ /**
175
+ * Check for constraint violations
176
+ */
177
+ checkConstraints(currentChapter) {
178
+ const violations = [];
179
+ // Check location consistency
180
+ violations.push(...this.checkLocationConsistency(currentChapter));
181
+ // Check knowledge consistency
182
+ violations.push(...this.checkKnowledgeConsistency(currentChapter));
183
+ // Check timeline consistency
184
+ violations.push(...this.checkTimelineConsistency(currentChapter));
185
+ // Check logical consistency
186
+ violations.push(...this.checkLogicalConsistency(currentChapter));
187
+ return violations;
188
+ }
189
+ /**
190
+ * Check location consistency (no teleporting)
191
+ */
192
+ checkLocationConsistency(currentChapter) {
193
+ const violations = [];
194
+ for (const node of this.nodes.values()) {
195
+ if (node.type === 'character') {
196
+ const locationEdges = this.getEdgesFrom(node.id).filter(e => e.type === 'located_at');
197
+ // Sort by chapter
198
+ const sortedEdges = locationEdges.sort((a, b) => (a.properties.since || 0) - (b.properties.since || 0));
199
+ // Check for impossible movements (simplified)
200
+ // In a real implementation, you'd check travel times between locations
201
+ for (let i = 1; i < sortedEdges.length; i++) {
202
+ const prevEdge = sortedEdges[i - 1];
203
+ const currEdge = sortedEdges[i];
204
+ const timeDiff = (currEdge.properties.since || 0) - (prevEdge.properties.since || 0);
205
+ // If moved in same chapter, might be suspicious
206
+ if (timeDiff === 0 && currEdge.properties.since === currentChapter) {
207
+ violations.push({
208
+ type: 'location',
209
+ severity: 'warning',
210
+ description: `${node.label} moved from ${prevEdge.to} to ${currEdge.to} instantly`,
211
+ nodes: [node.id],
212
+ suggestedFix: 'Add travel scene or justify rapid movement',
213
+ });
214
+ }
215
+ }
216
+ }
217
+ }
218
+ return violations;
219
+ }
220
+ /**
221
+ * Check knowledge consistency (no impossible knowledge)
222
+ */
223
+ checkKnowledgeConsistency(currentChapter) {
224
+ const violations = [];
225
+ for (const node of this.nodes.values()) {
226
+ if (node.type === 'character') {
227
+ const knowledgeEdges = this.getEdgesFrom(node.id).filter(e => e.type === 'knows');
228
+ for (const edge of knowledgeEdges) {
229
+ const factNode = this.nodes.get(edge.to);
230
+ if (factNode && factNode.chapterEstablished) {
231
+ const knownSince = edge.properties.since || currentChapter;
232
+ // Check if character knew fact before it was established
233
+ if (knownSince < factNode.chapterEstablished) {
234
+ violations.push({
235
+ type: 'knowledge',
236
+ severity: 'error',
237
+ description: `${node.label} knows "${factNode.label}" before it happened (Ch ${knownSince} vs Ch ${factNode.chapterEstablished})`,
238
+ nodes: [node.id, factNode.id],
239
+ suggestedFix: 'Remove knowledge or adjust timeline',
240
+ });
241
+ }
242
+ }
243
+ }
244
+ }
245
+ }
246
+ return violations;
247
+ }
248
+ /**
249
+ * Check timeline consistency
250
+ */
251
+ checkTimelineConsistency(currentChapter) {
252
+ const violations = [];
253
+ // Check for events that reference future chapters
254
+ for (const node of this.nodes.values()) {
255
+ if (node.type === 'event' && node.chapterEstablished) {
256
+ if (node.chapterEstablished > currentChapter) {
257
+ violations.push({
258
+ type: 'timeline',
259
+ severity: 'error',
260
+ description: `Event "${node.label}" is set in future chapter ${node.chapterEstablished} but we're in chapter ${currentChapter}`,
261
+ nodes: [node.id],
262
+ suggestedFix: 'Adjust event chapter or remove reference',
263
+ });
264
+ }
265
+ }
266
+ }
267
+ return violations;
268
+ }
269
+ /**
270
+ * Check logical consistency
271
+ */
272
+ checkLogicalConsistency(currentChapter) {
273
+ const violations = [];
274
+ // Check for characters participating in events when not present
275
+ for (const node of this.nodes.values()) {
276
+ if (node.type === 'event') {
277
+ const participants = node.properties.participants || [];
278
+ for (const participant of participants) {
279
+ const charId = `char-${participant}`;
280
+ const charNode = this.nodes.get(charId);
281
+ if (charNode) {
282
+ // Check if character was alive/present at event time
283
+ if (charNode.chapterEstablished && node.chapterEstablished) {
284
+ if (charNode.chapterEstablished > node.chapterEstablished) {
285
+ violations.push({
286
+ type: 'logic',
287
+ severity: 'error',
288
+ description: `${participant} participates in event before they were introduced`,
289
+ nodes: [charId, node.id],
290
+ suggestedFix: 'Adjust character introduction or event timing',
291
+ });
292
+ }
293
+ }
294
+ }
295
+ }
296
+ }
297
+ }
298
+ return violations;
299
+ }
300
+ /**
301
+ * Query what a character knows
302
+ */
303
+ getCharacterKnowledge(characterName) {
304
+ const charId = `char-${characterName}`;
305
+ const knowledgeEdges = this.getEdgesFrom(charId).filter(e => e.type === 'knows');
306
+ return knowledgeEdges
307
+ .map(edge => this.nodes.get(edge.to))
308
+ .filter((n) => n !== undefined && n.type === 'fact');
309
+ }
310
+ /**
311
+ * Query where a character is
312
+ */
313
+ getCharacterLocation(characterName) {
314
+ const charId = `char-${characterName}`;
315
+ const locationEdge = this.getEdgesFrom(charId).find(e => e.type === 'located_at');
316
+ if (locationEdge) {
317
+ const locNode = this.nodes.get(locationEdge.to);
318
+ return locNode?.label;
319
+ }
320
+ return undefined;
321
+ }
322
+ /**
323
+ * Serialize graph
324
+ */
325
+ serialize() {
326
+ return JSON.stringify({
327
+ nodes: Array.from(this.nodes.entries()),
328
+ edges: Array.from(this.edges.entries()),
329
+ });
330
+ }
331
+ /**
332
+ * Load graph from serialized data
333
+ */
334
+ load(data) {
335
+ const parsed = JSON.parse(data);
336
+ this.nodes = new Map(parsed.nodes);
337
+ this.edges = new Map(parsed.edges);
338
+ // Rebuild adjacency list
339
+ this.adjacencyList = new Map();
340
+ for (const [id, node] of this.nodes) {
341
+ this.adjacencyList.set(id, new Set());
342
+ }
343
+ for (const edge of this.edges.values()) {
344
+ const set = this.adjacencyList.get(edge.from);
345
+ if (set) {
346
+ set.add(edge.to);
347
+ }
348
+ }
349
+ }
350
+ /**
351
+ * Get graph statistics
352
+ */
353
+ getStats() {
354
+ const byType = {
355
+ character: 0,
356
+ location: 0,
357
+ fact: 0,
358
+ event: 0,
359
+ item: 0,
360
+ };
361
+ for (const node of this.nodes.values()) {
362
+ byType[node.type]++;
363
+ }
364
+ return {
365
+ nodes: this.nodes.size,
366
+ edges: this.edges.size,
367
+ byType,
368
+ };
369
+ }
370
+ /**
371
+ * Sanitize string for use as ID
372
+ */
373
+ sanitizeId(str) {
374
+ return str.replace(/[^a-zA-Z0-9]/g, '_').substring(0, 50);
375
+ }
376
+ }
377
+ exports.ConstraintGraph = ConstraintGraph;
378
+ function createConstraintGraph() {
379
+ return new ConstraintGraph();
380
+ }
381
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"constraintGraph.js","sourceRoot":"","sources":["../../src/constraints/constraintGraph.ts"],"names":[],"mappings":"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","sourcesContent":["import type { CharacterState } from '../story/structuredState.js';\r\n\r\nexport type NodeType = 'character' | 'location' | 'fact' | 'event' | 'item';\r\n\r\nexport interface ConstraintNode {\r\n  id: string;\r\n  type: NodeType;\r\n  label: string;\r\n  properties: Record<string, any>;\r\n  chapterEstablished?: number;\r\n}\r\n\r\nexport interface ConstraintEdge {\r\n  id: string;\r\n  from: string;\r\n  to: string;\r\n  type: string;\r\n  properties: Record<string, any>;\r\n}\r\n\r\nexport interface ConstraintViolation {\r\n  type: 'canon' | 'location' | 'knowledge' | 'timeline' | 'logic';\r\n  severity: 'error' | 'warning';\r\n  description: string;\r\n  nodes: string[];\r\n  suggestedFix?: string;\r\n}\r\n\r\nexport class ConstraintGraph {\r\n  private nodes: Map<string, ConstraintNode> = new Map();\r\n  private edges: Map<string, ConstraintEdge> = new Map();\r\n  private adjacencyList: Map<string, Set<string>> = new Map();\r\n  \r\n  /**\r\n   * Add a node to the graph\r\n   */\r\n  addNode(node: ConstraintNode): void {\r\n    this.nodes.set(node.id, node);\r\n    if (!this.adjacencyList.has(node.id)) {\r\n      this.adjacencyList.set(node.id, new Set());\r\n    }\r\n  }\r\n  \r\n  /**\r\n   * Add an edge between nodes\r\n   */\r\n  addEdge(edge: ConstraintEdge): void {\r\n    // Validate nodes exist\r\n    if (!this.nodes.has(edge.from) || !this.nodes.has(edge.to)) {\r\n      throw new Error(`Cannot add edge: node not found (${edge.from} -> ${edge.to})`);\r\n    }\r\n    \r\n    this.edges.set(edge.id, edge);\r\n    \r\n    // Update adjacency list\r\n    const fromSet = this.adjacencyList.get(edge.from);\r\n    if (fromSet) {\r\n      fromSet.add(edge.to);\r\n    }\r\n  }\r\n  \r\n  /**\r\n   * Get a node by ID\r\n   */\r\n  getNode(id: string): ConstraintNode | undefined {\r\n    return this.nodes.get(id);\r\n  }\r\n  \r\n  /**\r\n   * Get all edges from a node\r\n   */\r\n  getEdgesFrom(nodeId: string): ConstraintEdge[] {\r\n    return Array.from(this.edges.values()).filter(e => e.from === nodeId);\r\n  }\r\n  \r\n  /**\r\n   * Get all edges to a node\r\n   */\r\n  getEdgesTo(nodeId: string): ConstraintEdge[] {\r\n    return Array.from(this.edges.values()).filter(e => e.to === nodeId);\r\n  }\r\n  \r\n  /**\r\n   * Get neighbors of a node\r\n   */\r\n  getNeighbors(nodeId: string): ConstraintNode[] {\r\n    const neighborIds = this.adjacencyList.get(nodeId);\r\n    if (!neighborIds) return [];\r\n    \r\n    return Array.from(neighborIds)\r\n      .map(id => this.nodes.get(id))\r\n      .filter((n): n is ConstraintNode => n !== undefined);\r\n  }\r\n  \r\n  /**\r\n   * Add character to graph\r\n   */\r\n  addCharacter(character: CharacterState, chapter: number): void {\r\n    const charNode: ConstraintNode = {\r\n      id: `char-${character.name}`,\r\n      type: 'character',\r\n      label: character.name,\r\n      properties: {\r\n        emotionalState: character.emotionalState,\r\n        location: character.location,\r\n        goals: character.goals,\r\n      },\r\n      chapterEstablished: chapter,\r\n    };\r\n    \r\n    this.addNode(charNode);\r\n    \r\n    // Add location edge\r\n    this.addEdge({\r\n      id: `edge-${character.name}-loc`,\r\n      from: charNode.id,\r\n      to: `loc-${character.location}`,\r\n      type: 'located_at',\r\n      properties: { since: chapter },\r\n    });\r\n    \r\n    // Add knowledge nodes and edges\r\n    for (const knowledge of character.knowledge) {\r\n      const factId = `fact-${this.sanitizeId(knowledge)}`;\r\n      if (!this.nodes.has(factId)) {\r\n        this.addNode({\r\n          id: factId,\r\n          type: 'fact',\r\n          label: knowledge,\r\n          properties: {},\r\n          chapterEstablished: chapter,\r\n        });\r\n      }\r\n      \r\n      this.addEdge({\r\n        id: `edge-${character.name}-knows-${factId}`,\r\n        from: charNode.id,\r\n        to: factId,\r\n        type: 'knows',\r\n        properties: { since: chapter },\r\n      });\r\n    }\r\n  }\r\n  \r\n  /**\r\n   * Add location to graph\r\n   */\r\n  addLocation(name: string, description: string, chapter: number): void {\r\n    const locNode: ConstraintNode = {\r\n      id: `loc-${name}`,\r\n      type: 'location',\r\n      label: name,\r\n      properties: { description },\r\n      chapterEstablished: chapter,\r\n    };\r\n    \r\n    this.addNode(locNode);\r\n  }\r\n  \r\n  /**\r\n   * Add event to graph\r\n   */\r\n  addEvent(\r\n    id: string,\r\n    description: string,\r\n    participants: string[],\r\n    chapter: number\r\n  ): void {\r\n    const eventNode: ConstraintNode = {\r\n      id: `event-${id}`,\r\n      type: 'event',\r\n      label: description,\r\n      properties: { participants },\r\n      chapterEstablished: chapter,\r\n    };\r\n    \r\n    this.addNode(eventNode);\r\n    \r\n    // Connect participants to event\r\n    for (const participant of participants) {\r\n      const charId = `char-${participant}`;\r\n      if (this.nodes.has(charId)) {\r\n        this.addEdge({\r\n          id: `edge-${participant}-participates-${id}`,\r\n          from: charId,\r\n          to: eventNode.id,\r\n          type: 'participates_in',\r\n          properties: { chapter },\r\n        });\r\n      }\r\n    }\r\n  }\r\n  \r\n  /**\r\n   * Update character location\r\n   */\r\n  updateCharacterLocation(characterName: string, newLocation: string, chapter: number): void {\r\n    const charId = `char-${characterName}`;\r\n    const charNode = this.nodes.get(charId);\r\n    \r\n    if (charNode) {\r\n      // Update properties\r\n      charNode.properties.location = newLocation;\r\n      \r\n      // Remove old location edges\r\n      const oldEdges = this.getEdgesFrom(charId).filter(e => e.type === 'located_at');\r\n      for (const edge of oldEdges) {\r\n        this.edges.delete(edge.id);\r\n        const adjSet = this.adjacencyList.get(edge.from);\r\n        if (adjSet) {\r\n          adjSet.delete(edge.to);\r\n        }\r\n      }\r\n      \r\n      // Add new location edge\r\n      this.addEdge({\r\n        id: `edge-${characterName}-loc-${chapter}`,\r\n        from: charId,\r\n        to: `loc-${newLocation}`,\r\n        type: 'located_at',\r\n        properties: { since: chapter },\r\n      });\r\n    }\r\n  }\r\n  \r\n  /**\r\n   * Check for constraint violations\r\n   */\r\n  checkConstraints(currentChapter: number): ConstraintViolation[] {\r\n    const violations: ConstraintViolation[] = [];\r\n    \r\n    // Check location consistency\r\n    violations.push(...this.checkLocationConsistency(currentChapter));\r\n    \r\n    // Check knowledge consistency\r\n    violations.push(...this.checkKnowledgeConsistency(currentChapter));\r\n    \r\n    // Check timeline consistency\r\n    violations.push(...this.checkTimelineConsistency(currentChapter));\r\n    \r\n    // Check logical consistency\r\n    violations.push(...this.checkLogicalConsistency(currentChapter));\r\n    \r\n    return violations;\r\n  }\r\n  \r\n  /**\r\n   * Check location consistency (no teleporting)\r\n   */\r\n  private checkLocationConsistency(currentChapter: number): ConstraintViolation[] {\r\n    const violations: ConstraintViolation[] = [];\r\n    \r\n    for (const node of this.nodes.values()) {\r\n      if (node.type === 'character') {\r\n        const locationEdges = this.getEdgesFrom(node.id).filter(e => e.type === 'located_at');\r\n        \r\n        // Sort by chapter\r\n        const sortedEdges = locationEdges.sort((a, b) => \r\n          (a.properties.since || 0) - (b.properties.since || 0)\r\n        );\r\n        \r\n        // Check for impossible movements (simplified)\r\n        // In a real implementation, you'd check travel times between locations\r\n        for (let i = 1; i < sortedEdges.length; i++) {\r\n          const prevEdge = sortedEdges[i - 1];\r\n          const currEdge = sortedEdges[i];\r\n          const timeDiff = (currEdge.properties.since || 0) - (prevEdge.properties.since || 0);\r\n          \r\n          // If moved in same chapter, might be suspicious\r\n          if (timeDiff === 0 && currEdge.properties.since === currentChapter) {\r\n            violations.push({\r\n              type: 'location',\r\n              severity: 'warning',\r\n              description: `${node.label} moved from ${prevEdge.to} to ${currEdge.to} instantly`,\r\n              nodes: [node.id],\r\n              suggestedFix: 'Add travel scene or justify rapid movement',\r\n            });\r\n          }\r\n        }\r\n      }\r\n    }\r\n    \r\n    return violations;\r\n  }\r\n  \r\n  /**\r\n   * Check knowledge consistency (no impossible knowledge)\r\n   */\r\n  private checkKnowledgeConsistency(currentChapter: number): ConstraintViolation[] {\r\n    const violations: ConstraintViolation[] = [];\r\n    \r\n    for (const node of this.nodes.values()) {\r\n      if (node.type === 'character') {\r\n        const knowledgeEdges = this.getEdgesFrom(node.id).filter(e => e.type === 'knows');\r\n        \r\n        for (const edge of knowledgeEdges) {\r\n          const factNode = this.nodes.get(edge.to);\r\n          if (factNode && factNode.chapterEstablished) {\r\n            const knownSince = edge.properties.since || currentChapter;\r\n            \r\n            // Check if character knew fact before it was established\r\n            if (knownSince < factNode.chapterEstablished) {\r\n              violations.push({\r\n                type: 'knowledge',\r\n                severity: 'error',\r\n                description: `${node.label} knows \"${factNode.label}\" before it happened (Ch ${knownSince} vs Ch ${factNode.chapterEstablished})`,\r\n                nodes: [node.id, factNode.id],\r\n                suggestedFix: 'Remove knowledge or adjust timeline',\r\n              });\r\n            }\r\n          }\r\n        }\r\n      }\r\n    }\r\n    \r\n    return violations;\r\n  }\r\n  \r\n  /**\r\n   * Check timeline consistency\r\n   */\r\n  private checkTimelineConsistency(currentChapter: number): ConstraintViolation[] {\r\n    const violations: ConstraintViolation[] = [];\r\n    \r\n    // Check for events that reference future chapters\r\n    for (const node of this.nodes.values()) {\r\n      if (node.type === 'event' && node.chapterEstablished) {\r\n        if (node.chapterEstablished > currentChapter) {\r\n          violations.push({\r\n            type: 'timeline',\r\n            severity: 'error',\r\n            description: `Event \"${node.label}\" is set in future chapter ${node.chapterEstablished} but we're in chapter ${currentChapter}`,\r\n            nodes: [node.id],\r\n            suggestedFix: 'Adjust event chapter or remove reference',\r\n          });\r\n        }\r\n      }\r\n    }\r\n    \r\n    return violations;\r\n  }\r\n  \r\n  /**\r\n   * Check logical consistency\r\n   */\r\n  private checkLogicalConsistency(currentChapter: number): ConstraintViolation[] {\r\n    const violations: ConstraintViolation[] = [];\r\n    \r\n    // Check for characters participating in events when not present\r\n    for (const node of this.nodes.values()) {\r\n      if (node.type === 'event') {\r\n        const participants = node.properties.participants || [];\r\n        \r\n        for (const participant of participants) {\r\n          const charId = `char-${participant}`;\r\n          const charNode = this.nodes.get(charId);\r\n          \r\n          if (charNode) {\r\n            // Check if character was alive/present at event time\r\n            if (charNode.chapterEstablished && node.chapterEstablished) {\r\n              if (charNode.chapterEstablished > node.chapterEstablished) {\r\n                violations.push({\r\n                  type: 'logic',\r\n                  severity: 'error',\r\n                  description: `${participant} participates in event before they were introduced`,\r\n                  nodes: [charId, node.id],\r\n                  suggestedFix: 'Adjust character introduction or event timing',\r\n                });\r\n              }\r\n            }\r\n          }\r\n        }\r\n      }\r\n    }\r\n    \r\n    return violations;\r\n  }\r\n  \r\n  /**\r\n   * Query what a character knows\r\n   */\r\n  getCharacterKnowledge(characterName: string): ConstraintNode[] {\r\n    const charId = `char-${characterName}`;\r\n    const knowledgeEdges = this.getEdgesFrom(charId).filter(e => e.type === 'knows');\r\n    \r\n    return knowledgeEdges\r\n      .map(edge => this.nodes.get(edge.to))\r\n      .filter((n): n is ConstraintNode => n !== undefined && n.type === 'fact');\r\n  }\r\n  \r\n  /**\r\n   * Query where a character is\r\n   */\r\n  getCharacterLocation(characterName: string): string | undefined {\r\n    const charId = `char-${characterName}`;\r\n    const locationEdge = this.getEdgesFrom(charId).find(e => e.type === 'located_at');\r\n    \r\n    if (locationEdge) {\r\n      const locNode = this.nodes.get(locationEdge.to);\r\n      return locNode?.label;\r\n    }\r\n    \r\n    return undefined;\r\n  }\r\n  \r\n  /**\r\n   * Serialize graph\r\n   */\r\n  serialize(): string {\r\n    return JSON.stringify({\r\n      nodes: Array.from(this.nodes.entries()),\r\n      edges: Array.from(this.edges.entries()),\r\n    });\r\n  }\r\n  \r\n  /**\r\n   * Load graph from serialized data\r\n   */\r\n  load(data: string): void {\r\n    const parsed = JSON.parse(data);\r\n    this.nodes = new Map(parsed.nodes);\r\n    this.edges = new Map(parsed.edges);\r\n    \r\n    // Rebuild adjacency list\r\n    this.adjacencyList = new Map();\r\n    for (const [id, node] of this.nodes) {\r\n      this.adjacencyList.set(id, new Set());\r\n    }\r\n    for (const edge of this.edges.values()) {\r\n      const set = this.adjacencyList.get(edge.from);\r\n      if (set) {\r\n        set.add(edge.to);\r\n      }\r\n    }\r\n  }\r\n  \r\n  /**\r\n   * Get graph statistics\r\n   */\r\n  getStats(): { nodes: number; edges: number; byType: Record<NodeType, number> } {\r\n    const byType: Record<NodeType, number> = {\r\n      character: 0,\r\n      location: 0,\r\n      fact: 0,\r\n      event: 0,\r\n      item: 0,\r\n    };\r\n    \r\n    for (const node of this.nodes.values()) {\r\n      byType[node.type]++;\r\n    }\r\n    \r\n    return {\r\n      nodes: this.nodes.size,\r\n      edges: this.edges.size,\r\n      byType,\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Sanitize string for use as ID\r\n   */\r\n  private sanitizeId(str: string): string {\r\n    return str.replace(/[^a-zA-Z0-9]/g, '_').substring(0, 50);\r\n  }\r\n}\r\n\r\nexport function createConstraintGraph(): ConstraintGraph {\r\n  return new ConstraintGraph();\r\n}\r\n"]}
@@ -0,0 +1,58 @@
1
+ import type { Chapter, StoryBible } from '../types/index.js';
2
+ import type { StoryStructuredState } from '../story/structuredState.js';
3
+ import type { CanonStore } from '../memory/canonStore.js';
4
+ import { ConstraintGraph, type ConstraintViolation } from './constraintGraph.js';
5
+ export interface ValidationResult {
6
+ valid: boolean;
7
+ violations: ConstraintViolation[];
8
+ summary: string;
9
+ }
10
+ export interface ChapterValidationContext {
11
+ chapter: Chapter;
12
+ bible: StoryBible;
13
+ structuredState: StoryStructuredState;
14
+ canon: CanonStore;
15
+ previousChapters: Chapter[];
16
+ constraintGraph: ConstraintGraph;
17
+ }
18
+ export declare class Validator {
19
+ private constraintGraph;
20
+ constructor(constraintGraph?: ConstraintGraph);
21
+ /**
22
+ * Validate a chapter using both graph-based and LLM-based checks
23
+ */
24
+ validateChapter(context: ChapterValidationContext): Promise<ValidationResult>;
25
+ /**
26
+ * LLM-based validation
27
+ */
28
+ private llmValidate;
29
+ /**
30
+ * Quick validation without LLM (for testing/fallback)
31
+ */
32
+ quickValidate(context: ChapterValidationContext): ValidationResult;
33
+ /**
34
+ * Basic canon check (keyword matching)
35
+ */
36
+ private checkCanonBasic;
37
+ /**
38
+ * Format canon for prompt
39
+ */
40
+ private formatCanon;
41
+ /**
42
+ * Format characters for prompt
43
+ */
44
+ private formatCharacters;
45
+ /**
46
+ * Format plot threads for prompt
47
+ */
48
+ private formatPlotThreads;
49
+ /**
50
+ * Format validation result for display
51
+ */
52
+ formatResult(result: ValidationResult): string;
53
+ /**
54
+ * Get the constraint graph
55
+ */
56
+ getConstraintGraph(): ConstraintGraph;
57
+ }
58
+ export declare const validator: Validator;