@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,75 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.generateChapter = generateChapter;
4
+ const writer_js_1 = require("../agents/writer.js");
5
+ const completeness_js_1 = require("../agents/completeness.js");
6
+ const summarizer_js_1 = require("../agents/summarizer.js");
7
+ const canonValidator_js_1 = require("../agents/canonValidator.js");
8
+ const memoryExtractor_js_1 = require("../agents/memoryExtractor.js");
9
+ const memoryRetriever_js_1 = require("../memory/memoryRetriever.js");
10
+ async function generateChapter(context, options = {}) {
11
+ const { bible, state, chapterNumber } = context;
12
+ const { canon, vectorStore, validateCanon = true, maxContinuationAttempts = 3, retrieveMemories = true } = options;
13
+ console.log(`Generating Chapter ${chapterNumber}...`);
14
+ // Initialize memory retriever if vector store provided
15
+ let memoryRetriever;
16
+ if (vectorStore && retrieveMemories) {
17
+ await vectorStore.initialize();
18
+ memoryRetriever = (0, memoryRetriever_js_1.createMemoryRetriever)(vectorStore);
19
+ }
20
+ let output = await writer_js_1.writer.write(context, canon, memoryRetriever);
21
+ let attempts = 0;
22
+ while (attempts < maxContinuationAttempts) {
23
+ const check = await completeness_js_1.completenessChecker.check(output.content);
24
+ if (check.isComplete) {
25
+ break;
26
+ }
27
+ console.log(` Chapter incomplete, continuing... (attempt ${attempts + 1})`);
28
+ output.content = await writer_js_1.writer.continue(output.content, context);
29
+ output.wordCount = output.content.split(/\s+/).length;
30
+ attempts++;
31
+ }
32
+ let violations = [];
33
+ if (validateCanon && canon) {
34
+ console.log(' Validating canon...');
35
+ const validation = await canonValidator_js_1.canonValidator.validate(output.content, canon);
36
+ violations = validation.violations;
37
+ if (violations.length > 0) {
38
+ console.log(` ⚠️ Canon violations detected: ${violations.length}`);
39
+ }
40
+ }
41
+ const summary = await summarizer_js_1.summarizer.summarize(output.content, chapterNumber);
42
+ const chapter = {
43
+ id: generateId(),
44
+ storyId: bible.id,
45
+ number: chapterNumber,
46
+ title: output.title,
47
+ content: output.content,
48
+ summary: summary.summary,
49
+ wordCount: output.wordCount,
50
+ generatedAt: new Date(),
51
+ };
52
+ // Extract and store memories
53
+ let memoriesExtracted = 0;
54
+ if (vectorStore) {
55
+ console.log(' Extracting memories...');
56
+ const extracted = await memoryExtractor_js_1.memoryExtractor.extract(chapter, bible);
57
+ for (const memory of extracted) {
58
+ await vectorStore.addMemory({
59
+ storyId: bible.id,
60
+ chapterNumber,
61
+ content: memory.content,
62
+ category: memory.category,
63
+ timestamp: new Date(),
64
+ });
65
+ memoriesExtracted++;
66
+ }
67
+ console.log(` Stored ${memoriesExtracted} memories`);
68
+ }
69
+ console.log(` Generated: ${output.wordCount} words`);
70
+ return { chapter, summary, violations, memoriesExtracted };
71
+ }
72
+ function generateId() {
73
+ return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;
74
+ }
75
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"generateChapter.js","sourceRoot":"","sources":["../../src/pipeline/generateChapter.ts"],"names":[],"mappings":"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","sourcesContent":["import { writer } from '../agents/writer.js';\r\nimport { completenessChecker } from '../agents/completeness.js';\r\nimport { summarizer } from '../agents/summarizer.js';\r\nimport { canonValidator } from '../agents/canonValidator.js';\r\nimport { memoryExtractor } from '../agents/memoryExtractor.js';\r\nimport type { GenerationContext, Chapter, ChapterSummary } from '../types/index.js';\r\nimport type { CanonStore } from '../memory/canonStore.js';\r\nimport type { VectorStore } from '../memory/vectorStore.js';\r\nimport { createMemoryRetriever, MemoryRetriever } from '../memory/memoryRetriever.js';\r\n\r\nexport interface GenerateChapterResult {\r\n  chapter: Chapter;\r\n  summary: ChapterSummary;\r\n  violations: string[];\r\n  memoriesExtracted: number;\r\n}\r\n\r\nexport interface GenerateChapterOptions {\r\n  canon?: CanonStore;\r\n  vectorStore?: VectorStore;\r\n  validateCanon?: boolean;\r\n  maxContinuationAttempts?: number;\r\n  retrieveMemories?: boolean;\r\n}\r\n\r\nexport async function generateChapter(\r\n  context: GenerationContext,\r\n  options: GenerateChapterOptions = {}\r\n): Promise<GenerateChapterResult> {\r\n  const { bible, state, chapterNumber } = context;\r\n  const { canon, vectorStore, validateCanon = true, maxContinuationAttempts = 3, retrieveMemories = true } = options;\r\n\r\n  console.log(`Generating Chapter ${chapterNumber}...`);\r\n\r\n  // Initialize memory retriever if vector store provided\r\n  let memoryRetriever: MemoryRetriever | undefined;\r\n  if (vectorStore && retrieveMemories) {\r\n    await vectorStore.initialize();\r\n    memoryRetriever = createMemoryRetriever(vectorStore);\r\n  }\r\n\r\n  let output = await writer.write(context, canon, memoryRetriever);\r\n  let attempts = 0;\r\n\r\n  while (attempts < maxContinuationAttempts) {\r\n    const check = await completenessChecker.check(output.content);\r\n    \r\n    if (check.isComplete) {\r\n      break;\r\n    }\r\n\r\n    console.log(`  Chapter incomplete, continuing... (attempt ${attempts + 1})`);\r\n    output.content = await writer.continue(output.content, context);\r\n    output.wordCount = output.content.split(/\\s+/).length;\r\n    attempts++;\r\n  }\r\n\r\n  let violations: string[] = [];\r\n  if (validateCanon && canon) {\r\n    console.log('  Validating canon...');\r\n    const validation = await canonValidator.validate(output.content, canon);\r\n    violations = validation.violations;\r\n    if (violations.length > 0) {\r\n      console.log(`  ⚠️  Canon violations detected: ${violations.length}`);\r\n    }\r\n  }\r\n\r\n  const summary = await summarizer.summarize(output.content, chapterNumber);\r\n\r\n  const chapter: Chapter = {\r\n    id: generateId(),\r\n    storyId: bible.id,\r\n    number: chapterNumber,\r\n    title: output.title,\r\n    content: output.content,\r\n    summary: summary.summary,\r\n    wordCount: output.wordCount,\r\n    generatedAt: new Date(),\r\n  };\r\n\r\n  // Extract and store memories\r\n  let memoriesExtracted = 0;\r\n  if (vectorStore) {\r\n    console.log('  Extracting memories...');\r\n    const extracted = await memoryExtractor.extract(chapter, bible);\r\n    \r\n    for (const memory of extracted) {\r\n      await vectorStore.addMemory({\r\n        storyId: bible.id,\r\n        chapterNumber,\r\n        content: memory.content,\r\n        category: memory.category,\r\n        timestamp: new Date(),\r\n      });\r\n      memoriesExtracted++;\r\n    }\r\n    console.log(`  Stored ${memoriesExtracted} memories`);\r\n  }\r\n\r\n  console.log(`  Generated: ${output.wordCount} words`);\r\n\r\n  return { chapter, summary, violations, memoriesExtracted };\r\n}\r\n\r\nfunction generateId(): string {\r\n  return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;\r\n}\r\n"]}
@@ -0,0 +1,4 @@
1
+ import type { StoryBible, CharacterProfile } from '../types/index.js';
2
+ export declare function createStoryBible(title: string, theme: string, genre: string, setting: string, tone: string, premise: string, targetChapters?: number): StoryBible;
3
+ export declare function addCharacter(bible: StoryBible, name: string, role: CharacterProfile['role'], personality: string[], goals: string[]): StoryBible;
4
+ export declare function addPlotThread(bible: StoryBible, name: string, description: string): StoryBible;
@@ -0,0 +1,53 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.createStoryBible = createStoryBible;
4
+ exports.addCharacter = addCharacter;
5
+ exports.addPlotThread = addPlotThread;
6
+ function createStoryBible(title, theme, genre, setting, tone, premise, targetChapters = 10) {
7
+ return {
8
+ id: generateId(),
9
+ title,
10
+ theme,
11
+ genre,
12
+ setting,
13
+ tone,
14
+ targetChapters,
15
+ premise,
16
+ characters: [],
17
+ plotThreads: [],
18
+ createdAt: new Date(),
19
+ updatedAt: new Date(),
20
+ };
21
+ }
22
+ function addCharacter(bible, name, role, personality, goals) {
23
+ const character = {
24
+ id: generateId(),
25
+ name,
26
+ role,
27
+ personality,
28
+ goals,
29
+ };
30
+ return {
31
+ ...bible,
32
+ characters: [...bible.characters, character],
33
+ updatedAt: new Date(),
34
+ };
35
+ }
36
+ function addPlotThread(bible, name, description) {
37
+ const thread = {
38
+ id: generateId(),
39
+ name,
40
+ description,
41
+ status: 'dormant',
42
+ tension: 0.1,
43
+ };
44
+ return {
45
+ ...bible,
46
+ plotThreads: [...bible.plotThreads, thread],
47
+ updatedAt: new Date(),
48
+ };
49
+ }
50
+ function generateId() {
51
+ return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;
52
+ }
53
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmlibGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc3RvcnkvYmlibGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFFQSw0Q0F1QkM7QUFFRCxvQ0FvQkM7QUFFRCxzQ0FrQkM7QUFqRUQsU0FBZ0IsZ0JBQWdCLENBQzlCLEtBQWEsRUFDYixLQUFhLEVBQ2IsS0FBYSxFQUNiLE9BQWUsRUFDZixJQUFZLEVBQ1osT0FBZSxFQUNmLGlCQUF5QixFQUFFO0lBRTNCLE9BQU87UUFDTCxFQUFFLEVBQUUsVUFBVSxFQUFFO1FBQ2hCLEtBQUs7UUFDTCxLQUFLO1FBQ0wsS0FBSztRQUNMLE9BQU87UUFDUCxJQUFJO1FBQ0osY0FBYztRQUNkLE9BQU87UUFDUCxVQUFVLEVBQUUsRUFBRTtRQUNkLFdBQVcsRUFBRSxFQUFFO1FBQ2YsU0FBUyxFQUFFLElBQUksSUFBSSxFQUFFO1FBQ3JCLFNBQVMsRUFBRSxJQUFJLElBQUksRUFBRTtLQUN0QixDQUFDO0FBQ0osQ0FBQztBQUVELFNBQWdCLFlBQVksQ0FDMUIsS0FBaUIsRUFDakIsSUFBWSxFQUNaLElBQThCLEVBQzlCLFdBQXFCLEVBQ3JCLEtBQWU7SUFFZixNQUFNLFNBQVMsR0FBcUI7UUFDbEMsRUFBRSxFQUFFLFVBQVUsRUFBRTtRQUNoQixJQUFJO1FBQ0osSUFBSTtRQUNKLFdBQVc7UUFDWCxLQUFLO0tBQ04sQ0FBQztJQUVGLE9BQU87UUFDTCxHQUFHLEtBQUs7UUFDUixVQUFVLEVBQUUsQ0FBQyxHQUFHLEtBQUssQ0FBQyxVQUFVLEVBQUUsU0FBUyxDQUFDO1FBQzVDLFNBQVMsRUFBRSxJQUFJLElBQUksRUFBRTtLQUN0QixDQUFDO0FBQ0osQ0FBQztBQUVELFNBQWdCLGFBQWEsQ0FDM0IsS0FBaUIsRUFDakIsSUFBWSxFQUNaLFdBQW1CO0lBRW5CLE1BQU0sTUFBTSxHQUFlO1FBQ3pCLEVBQUUsRUFBRSxVQUFVLEVBQUU7UUFDaEIsSUFBSTtRQUNKLFdBQVc7UUFDWCxNQUFNLEVBQUUsU0FBUztRQUNqQixPQUFPLEVBQUUsR0FBRztLQUNiLENBQUM7SUFFRixPQUFPO1FBQ0wsR0FBRyxLQUFLO1FBQ1IsV0FBVyxFQUFFLENBQUMsR0FBRyxLQUFLLENBQUMsV0FBVyxFQUFFLE1BQU0sQ0FBQztRQUMzQyxTQUFTLEVBQUUsSUFBSSxJQUFJLEVBQUU7S0FDdEIsQ0FBQztBQUNKLENBQUM7QUFFRCxTQUFTLFVBQVU7SUFDakIsT0FBTyxHQUFHLElBQUksQ0FBQyxHQUFHLEVBQUUsSUFBSSxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEVBQUUsQ0FBQztBQUN2RSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBTdG9yeUJpYmxlLCBDaGFyYWN0ZXJQcm9maWxlLCBQbG90VGhyZWFkIH0gZnJvbSAnLi4vdHlwZXMvaW5kZXguanMnO1xyXG5cclxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZVN0b3J5QmlibGUoXHJcbiAgdGl0bGU6IHN0cmluZyxcclxuICB0aGVtZTogc3RyaW5nLFxyXG4gIGdlbnJlOiBzdHJpbmcsXHJcbiAgc2V0dGluZzogc3RyaW5nLFxyXG4gIHRvbmU6IHN0cmluZyxcclxuICBwcmVtaXNlOiBzdHJpbmcsXHJcbiAgdGFyZ2V0Q2hhcHRlcnM6IG51bWJlciA9IDEwXHJcbik6IFN0b3J5QmlibGUge1xyXG4gIHJldHVybiB7XHJcbiAgICBpZDogZ2VuZXJhdGVJZCgpLFxyXG4gICAgdGl0bGUsXHJcbiAgICB0aGVtZSxcclxuICAgIGdlbnJlLFxyXG4gICAgc2V0dGluZyxcclxuICAgIHRvbmUsXHJcbiAgICB0YXJnZXRDaGFwdGVycyxcclxuICAgIHByZW1pc2UsXHJcbiAgICBjaGFyYWN0ZXJzOiBbXSxcclxuICAgIHBsb3RUaHJlYWRzOiBbXSxcclxuICAgIGNyZWF0ZWRBdDogbmV3IERhdGUoKSxcclxuICAgIHVwZGF0ZWRBdDogbmV3IERhdGUoKSxcclxuICB9O1xyXG59XHJcblxyXG5leHBvcnQgZnVuY3Rpb24gYWRkQ2hhcmFjdGVyKFxyXG4gIGJpYmxlOiBTdG9yeUJpYmxlLFxyXG4gIG5hbWU6IHN0cmluZyxcclxuICByb2xlOiBDaGFyYWN0ZXJQcm9maWxlWydyb2xlJ10sXHJcbiAgcGVyc29uYWxpdHk6IHN0cmluZ1tdLFxyXG4gIGdvYWxzOiBzdHJpbmdbXVxyXG4pOiBTdG9yeUJpYmxlIHtcclxuICBjb25zdCBjaGFyYWN0ZXI6IENoYXJhY3RlclByb2ZpbGUgPSB7XHJcbiAgICBpZDogZ2VuZXJhdGVJZCgpLFxyXG4gICAgbmFtZSxcclxuICAgIHJvbGUsXHJcbiAgICBwZXJzb25hbGl0eSxcclxuICAgIGdvYWxzLFxyXG4gIH07XHJcbiAgXHJcbiAgcmV0dXJuIHtcclxuICAgIC4uLmJpYmxlLFxyXG4gICAgY2hhcmFjdGVyczogWy4uLmJpYmxlLmNoYXJhY3RlcnMsIGNoYXJhY3Rlcl0sXHJcbiAgICB1cGRhdGVkQXQ6IG5ldyBEYXRlKCksXHJcbiAgfTtcclxufVxyXG5cclxuZXhwb3J0IGZ1bmN0aW9uIGFkZFBsb3RUaHJlYWQoXHJcbiAgYmlibGU6IFN0b3J5QmlibGUsXHJcbiAgbmFtZTogc3RyaW5nLFxyXG4gIGRlc2NyaXB0aW9uOiBzdHJpbmdcclxuKTogU3RvcnlCaWJsZSB7XHJcbiAgY29uc3QgdGhyZWFkOiBQbG90VGhyZWFkID0ge1xyXG4gICAgaWQ6IGdlbmVyYXRlSWQoKSxcclxuICAgIG5hbWUsXHJcbiAgICBkZXNjcmlwdGlvbixcclxuICAgIHN0YXR1czogJ2Rvcm1hbnQnLFxyXG4gICAgdGVuc2lvbjogMC4xLFxyXG4gIH07XHJcbiAgXHJcbiAgcmV0dXJuIHtcclxuICAgIC4uLmJpYmxlLFxyXG4gICAgcGxvdFRocmVhZHM6IFsuLi5iaWJsZS5wbG90VGhyZWFkcywgdGhyZWFkXSxcclxuICAgIHVwZGF0ZWRBdDogbmV3IERhdGUoKSxcclxuICB9O1xyXG59XHJcblxyXG5mdW5jdGlvbiBnZW5lcmF0ZUlkKCk6IHN0cmluZyB7XHJcbiAgcmV0dXJuIGAke0RhdGUubm93KCl9LSR7TWF0aC5yYW5kb20oKS50b1N0cmluZygzNikuc3Vic3RyaW5nKDIsIDkpfWA7XHJcbn1cclxuIl19
@@ -0,0 +1,3 @@
1
+ import type { StoryState, ChapterSummary } from '../types/index.js';
2
+ export declare function createStoryState(storyId: string, totalChapters: number): StoryState;
3
+ export declare function updateStoryState(state: StoryState, summary: ChapterSummary): StoryState;
@@ -0,0 +1,27 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.createStoryState = createStoryState;
4
+ exports.updateStoryState = updateStoryState;
5
+ function createStoryState(storyId, totalChapters) {
6
+ return {
7
+ storyId,
8
+ currentChapter: 0,
9
+ totalChapters,
10
+ currentTension: 0.1,
11
+ activePlotThreads: [],
12
+ chapterSummaries: [],
13
+ };
14
+ }
15
+ function updateStoryState(state, summary) {
16
+ return {
17
+ ...state,
18
+ currentChapter: summary.chapterNumber,
19
+ chapterSummaries: [...state.chapterSummaries, summary],
20
+ currentTension: calculateTension(summary.chapterNumber, state.totalChapters),
21
+ };
22
+ }
23
+ function calculateTension(currentChapter, totalChapters) {
24
+ const progress = currentChapter / totalChapters;
25
+ return 4 * progress * (1 - progress);
26
+ }
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc3Rvcnkvc3RhdGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFFQSw0Q0FTQztBQUVELDRDQVVDO0FBckJELFNBQWdCLGdCQUFnQixDQUFDLE9BQWUsRUFBRSxhQUFxQjtJQUNyRSxPQUFPO1FBQ0wsT0FBTztRQUNQLGNBQWMsRUFBRSxDQUFDO1FBQ2pCLGFBQWE7UUFDYixjQUFjLEVBQUUsR0FBRztRQUNuQixpQkFBaUIsRUFBRSxFQUFFO1FBQ3JCLGdCQUFnQixFQUFFLEVBQUU7S0FDckIsQ0FBQztBQUNKLENBQUM7QUFFRCxTQUFnQixnQkFBZ0IsQ0FDOUIsS0FBaUIsRUFDakIsT0FBdUI7SUFFdkIsT0FBTztRQUNMLEdBQUcsS0FBSztRQUNSLGNBQWMsRUFBRSxPQUFPLENBQUMsYUFBYTtRQUNyQyxnQkFBZ0IsRUFBRSxDQUFDLEdBQUcsS0FBSyxDQUFDLGdCQUFnQixFQUFFLE9BQU8sQ0FBQztRQUN0RCxjQUFjLEVBQUUsZ0JBQWdCLENBQUMsT0FBTyxDQUFDLGFBQWEsRUFBRSxLQUFLLENBQUMsYUFBYSxDQUFDO0tBQzdFLENBQUM7QUFDSixDQUFDO0FBRUQsU0FBUyxnQkFBZ0IsQ0FBQyxjQUFzQixFQUFFLGFBQXFCO0lBQ3JFLE1BQU0sUUFBUSxHQUFHLGNBQWMsR0FBRyxhQUFhLENBQUM7SUFDaEQsT0FBTyxDQUFDLEdBQUcsUUFBUSxHQUFHLENBQUMsQ0FBQyxHQUFHLFFBQVEsQ0FBQyxDQUFDO0FBQ3ZDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFN0b3J5U3RhdGUsIENoYXB0ZXJTdW1tYXJ5IH0gZnJvbSAnLi4vdHlwZXMvaW5kZXguanMnO1xyXG5cclxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZVN0b3J5U3RhdGUoc3RvcnlJZDogc3RyaW5nLCB0b3RhbENoYXB0ZXJzOiBudW1iZXIpOiBTdG9yeVN0YXRlIHtcclxuICByZXR1cm4ge1xyXG4gICAgc3RvcnlJZCxcclxuICAgIGN1cnJlbnRDaGFwdGVyOiAwLFxyXG4gICAgdG90YWxDaGFwdGVycyxcclxuICAgIGN1cnJlbnRUZW5zaW9uOiAwLjEsXHJcbiAgICBhY3RpdmVQbG90VGhyZWFkczogW10sXHJcbiAgICBjaGFwdGVyU3VtbWFyaWVzOiBbXSxcclxuICB9O1xyXG59XHJcblxyXG5leHBvcnQgZnVuY3Rpb24gdXBkYXRlU3RvcnlTdGF0ZShcclxuICBzdGF0ZTogU3RvcnlTdGF0ZSxcclxuICBzdW1tYXJ5OiBDaGFwdGVyU3VtbWFyeVxyXG4pOiBTdG9yeVN0YXRlIHtcclxuICByZXR1cm4ge1xyXG4gICAgLi4uc3RhdGUsXHJcbiAgICBjdXJyZW50Q2hhcHRlcjogc3VtbWFyeS5jaGFwdGVyTnVtYmVyLFxyXG4gICAgY2hhcHRlclN1bW1hcmllczogWy4uLnN0YXRlLmNoYXB0ZXJTdW1tYXJpZXMsIHN1bW1hcnldLFxyXG4gICAgY3VycmVudFRlbnNpb246IGNhbGN1bGF0ZVRlbnNpb24oc3VtbWFyeS5jaGFwdGVyTnVtYmVyLCBzdGF0ZS50b3RhbENoYXB0ZXJzKSxcclxuICB9O1xyXG59XHJcblxyXG5mdW5jdGlvbiBjYWxjdWxhdGVUZW5zaW9uKGN1cnJlbnRDaGFwdGVyOiBudW1iZXIsIHRvdGFsQ2hhcHRlcnM6IG51bWJlcik6IG51bWJlciB7XHJcbiAgY29uc3QgcHJvZ3Jlc3MgPSBjdXJyZW50Q2hhcHRlciAvIHRvdGFsQ2hhcHRlcnM7XHJcbiAgcmV0dXJuIDQgKiBwcm9ncmVzcyAqICgxIC0gcHJvZ3Jlc3MpO1xyXG59XHJcbiJdfQ==
@@ -0,0 +1,39 @@
1
+ import type { StoryBible } from '../types/index.js';
2
+ export interface CharacterState {
3
+ name: string;
4
+ emotionalState: string;
5
+ location: string;
6
+ relationships: Record<string, string>;
7
+ goals: string[];
8
+ knowledge: string[];
9
+ development: string[];
10
+ }
11
+ export interface PlotThreadState {
12
+ id: string;
13
+ name: string;
14
+ status: 'dormant' | 'active' | 'escalating' | 'resolved';
15
+ tension: number;
16
+ lastChapter: number;
17
+ involvedCharacters: string[];
18
+ summary: string;
19
+ }
20
+ export interface StoryStructuredState {
21
+ storyId: string;
22
+ chapter: number;
23
+ tension: number;
24
+ characters: Record<string, CharacterState>;
25
+ plotThreads: Record<string, PlotThreadState>;
26
+ unresolvedQuestions: string[];
27
+ recentEvents: string[];
28
+ }
29
+ export declare function createStructuredState(storyId: string): StoryStructuredState;
30
+ export declare function initializeCharactersFromBible(state: StoryStructuredState, bible: StoryBible): StoryStructuredState;
31
+ export declare function initializePlotThreadsFromBible(state: StoryStructuredState, bible: StoryBible): StoryStructuredState;
32
+ export declare function updateCharacterState(state: StoryStructuredState, characterName: string, updates: Partial<CharacterState>): StoryStructuredState;
33
+ export declare function updatePlotThread(state: StoryStructuredState, threadId: string, updates: Partial<PlotThreadState>, currentChapter: number): StoryStructuredState;
34
+ export declare function addUnresolvedQuestion(state: StoryStructuredState, question: string): StoryStructuredState;
35
+ export declare function resolveQuestion(state: StoryStructuredState, question: string): StoryStructuredState;
36
+ export declare function addRecentEvent(state: StoryStructuredState, event: string): StoryStructuredState;
37
+ export declare function calculateTargetTension(currentChapter: number, totalChapters: number): number;
38
+ export declare function updateStoryTension(state: StoryStructuredState, currentChapter: number, totalChapters: number): StoryStructuredState;
39
+ export declare function formatStructuredStateForPrompt(state: StoryStructuredState): string;
@@ -0,0 +1,159 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.createStructuredState = createStructuredState;
4
+ exports.initializeCharactersFromBible = initializeCharactersFromBible;
5
+ exports.initializePlotThreadsFromBible = initializePlotThreadsFromBible;
6
+ exports.updateCharacterState = updateCharacterState;
7
+ exports.updatePlotThread = updatePlotThread;
8
+ exports.addUnresolvedQuestion = addUnresolvedQuestion;
9
+ exports.resolveQuestion = resolveQuestion;
10
+ exports.addRecentEvent = addRecentEvent;
11
+ exports.calculateTargetTension = calculateTargetTension;
12
+ exports.updateStoryTension = updateStoryTension;
13
+ exports.formatStructuredStateForPrompt = formatStructuredStateForPrompt;
14
+ function createStructuredState(storyId) {
15
+ return {
16
+ storyId,
17
+ chapter: 0,
18
+ tension: 0.1, // Start with low tension
19
+ characters: {},
20
+ plotThreads: {},
21
+ unresolvedQuestions: [],
22
+ recentEvents: [],
23
+ };
24
+ }
25
+ function initializeCharactersFromBible(state, bible) {
26
+ const newState = { ...state };
27
+ for (const char of bible.characters) {
28
+ newState.characters[char.name] = {
29
+ name: char.name,
30
+ emotionalState: 'neutral',
31
+ location: bible.setting,
32
+ relationships: {},
33
+ goals: [...char.goals],
34
+ knowledge: [],
35
+ development: [],
36
+ };
37
+ }
38
+ return newState;
39
+ }
40
+ function initializePlotThreadsFromBible(state, bible) {
41
+ const newState = { ...state };
42
+ for (const thread of bible.plotThreads) {
43
+ newState.plotThreads[thread.id] = {
44
+ id: thread.id,
45
+ name: thread.name,
46
+ status: thread.status,
47
+ tension: thread.tension,
48
+ lastChapter: 0,
49
+ involvedCharacters: [],
50
+ summary: thread.description,
51
+ };
52
+ }
53
+ return newState;
54
+ }
55
+ function updateCharacterState(state, characterName, updates) {
56
+ const newState = { ...state };
57
+ if (newState.characters[characterName]) {
58
+ newState.characters[characterName] = {
59
+ ...newState.characters[characterName],
60
+ ...updates,
61
+ };
62
+ }
63
+ return newState;
64
+ }
65
+ function updatePlotThread(state, threadId, updates, currentChapter) {
66
+ const newState = { ...state };
67
+ if (newState.plotThreads[threadId]) {
68
+ newState.plotThreads[threadId] = {
69
+ ...newState.plotThreads[threadId],
70
+ ...updates,
71
+ lastChapter: currentChapter,
72
+ };
73
+ }
74
+ return newState;
75
+ }
76
+ function addUnresolvedQuestion(state, question) {
77
+ return {
78
+ ...state,
79
+ unresolvedQuestions: [...state.unresolvedQuestions, question],
80
+ };
81
+ }
82
+ function resolveQuestion(state, question) {
83
+ return {
84
+ ...state,
85
+ unresolvedQuestions: state.unresolvedQuestions.filter(q => q !== question),
86
+ };
87
+ }
88
+ function addRecentEvent(state, event) {
89
+ // Keep only last 10 events
90
+ const recentEvents = [...state.recentEvents, event].slice(-10);
91
+ return {
92
+ ...state,
93
+ recentEvents,
94
+ };
95
+ }
96
+ function calculateTargetTension(currentChapter, totalChapters) {
97
+ // Parabolic tension curve: low at start, high in middle, resolution at end
98
+ const progress = currentChapter / totalChapters;
99
+ return 4 * progress * (1 - progress);
100
+ }
101
+ function updateStoryTension(state, currentChapter, totalChapters) {
102
+ const targetTension = calculateTargetTension(currentChapter, totalChapters);
103
+ // Blend current tension toward target (smooth transition)
104
+ const newTension = state.tension * 0.7 + targetTension * 0.3;
105
+ return {
106
+ ...state,
107
+ chapter: currentChapter,
108
+ tension: Math.round(newTension * 100) / 100, // Round to 2 decimals
109
+ };
110
+ }
111
+ function formatStructuredStateForPrompt(state) {
112
+ const lines = ['## Current Story State'];
113
+ // Overall tension
114
+ const tensionPercent = Math.round(state.tension * 100);
115
+ lines.push(`\n**Story Tension:** ${tensionPercent}%`);
116
+ // Active plot threads
117
+ const activeThreads = Object.values(state.plotThreads)
118
+ .filter(t => t.status !== 'resolved' && t.status !== 'dormant');
119
+ if (activeThreads.length > 0) {
120
+ lines.push('\n### Active Plot Threads');
121
+ for (const thread of activeThreads) {
122
+ const threadTension = Math.round(thread.tension * 100);
123
+ lines.push(`- **${thread.name}** (${thread.status}, tension: ${threadTension}%)`);
124
+ lines.push(` - ${thread.summary}`);
125
+ }
126
+ }
127
+ // Character states
128
+ const characters = Object.values(state.characters);
129
+ if (characters.length > 0) {
130
+ lines.push('\n### Character States');
131
+ for (const char of characters) {
132
+ lines.push(`- **${char.name}**: ${char.emotionalState}`);
133
+ if (char.location)
134
+ lines.push(` - Location: ${char.location}`);
135
+ if (Object.keys(char.relationships).length > 0) {
136
+ const rels = Object.entries(char.relationships)
137
+ .map(([name, status]) => `${name} (${status})`)
138
+ .join(', ');
139
+ lines.push(` - Relationships: ${rels}`);
140
+ }
141
+ }
142
+ }
143
+ // Unresolved questions
144
+ if (state.unresolvedQuestions.length > 0) {
145
+ lines.push('\n### Unresolved Questions');
146
+ for (const q of state.unresolvedQuestions) {
147
+ lines.push(`- ${q}`);
148
+ }
149
+ }
150
+ // Recent events
151
+ if (state.recentEvents.length > 0) {
152
+ lines.push('\n### Recent Events');
153
+ for (const event of state.recentEvents.slice(-5)) {
154
+ lines.push(`- ${event}`);
155
+ }
156
+ }
157
+ return lines.join('\n');
158
+ }
159
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"structuredState.js","sourceRoot":"","sources":["../../src/story/structuredState.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Chapter, ChapterSummary, StoryBible } from '../types/index.js';\r\n\r\nexport interface CharacterState {\r\n  name: string;\r\n  emotionalState: string;\r\n  location: string;\r\n  relationships: Record<string, string>; // character name -> relationship status\r\n  goals: string[];\r\n  knowledge: string[]; // What this character knows\r\n  development: string[]; // Character arc changes\r\n}\r\n\r\nexport interface PlotThreadState {\r\n  id: string;\r\n  name: string;\r\n  status: 'dormant' | 'active' | 'escalating' | 'resolved';\r\n  tension: number; // 0.0 to 1.0\r\n  lastChapter: number;\r\n  involvedCharacters: string[];\r\n  summary: string;\r\n}\r\n\r\nexport interface StoryStructuredState {\r\n  storyId: string;\r\n  chapter: number;\r\n  tension: number; // Overall story tension 0.0 to 1.0\r\n  characters: Record<string, CharacterState>;\r\n  plotThreads: Record<string, PlotThreadState>;\r\n  unresolvedQuestions: string[];\r\n  recentEvents: string[];\r\n}\r\n\r\nexport function createStructuredState(storyId: string): StoryStructuredState {\r\n  return {\r\n    storyId,\r\n    chapter: 0,\r\n    tension: 0.1, // Start with low tension\r\n    characters: {},\r\n    plotThreads: {},\r\n    unresolvedQuestions: [],\r\n    recentEvents: [],\r\n  };\r\n}\r\n\r\nexport function initializeCharactersFromBible(\r\n  state: StoryStructuredState,\r\n  bible: StoryBible\r\n): StoryStructuredState {\r\n  const newState = { ...state };\r\n  \r\n  for (const char of bible.characters) {\r\n    newState.characters[char.name] = {\r\n      name: char.name,\r\n      emotionalState: 'neutral',\r\n      location: bible.setting,\r\n      relationships: {},\r\n      goals: [...char.goals],\r\n      knowledge: [],\r\n      development: [],\r\n    };\r\n  }\r\n  \r\n  return newState;\r\n}\r\n\r\nexport function initializePlotThreadsFromBible(\r\n  state: StoryStructuredState,\r\n  bible: StoryBible\r\n): StoryStructuredState {\r\n  const newState = { ...state };\r\n  \r\n  for (const thread of bible.plotThreads) {\r\n    newState.plotThreads[thread.id] = {\r\n      id: thread.id,\r\n      name: thread.name,\r\n      status: thread.status,\r\n      tension: thread.tension,\r\n      lastChapter: 0,\r\n      involvedCharacters: [],\r\n      summary: thread.description,\r\n    };\r\n  }\r\n  \r\n  return newState;\r\n}\r\n\r\nexport function updateCharacterState(\r\n  state: StoryStructuredState,\r\n  characterName: string,\r\n  updates: Partial<CharacterState>\r\n): StoryStructuredState {\r\n  const newState = { ...state };\r\n  \r\n  if (newState.characters[characterName]) {\r\n    newState.characters[characterName] = {\r\n      ...newState.characters[characterName],\r\n      ...updates,\r\n    };\r\n  }\r\n  \r\n  return newState;\r\n}\r\n\r\nexport function updatePlotThread(\r\n  state: StoryStructuredState,\r\n  threadId: string,\r\n  updates: Partial<PlotThreadState>,\r\n  currentChapter: number\r\n): StoryStructuredState {\r\n  const newState = { ...state };\r\n  \r\n  if (newState.plotThreads[threadId]) {\r\n    newState.plotThreads[threadId] = {\r\n      ...newState.plotThreads[threadId],\r\n      ...updates,\r\n      lastChapter: currentChapter,\r\n    };\r\n  }\r\n  \r\n  return newState;\r\n}\r\n\r\nexport function addUnresolvedQuestion(\r\n  state: StoryStructuredState,\r\n  question: string\r\n): StoryStructuredState {\r\n  return {\r\n    ...state,\r\n    unresolvedQuestions: [...state.unresolvedQuestions, question],\r\n  };\r\n}\r\n\r\nexport function resolveQuestion(\r\n  state: StoryStructuredState,\r\n  question: string\r\n): StoryStructuredState {\r\n  return {\r\n    ...state,\r\n    unresolvedQuestions: state.unresolvedQuestions.filter(q => q !== question),\r\n  };\r\n}\r\n\r\nexport function addRecentEvent(\r\n  state: StoryStructuredState,\r\n  event: string\r\n): StoryStructuredState {\r\n  // Keep only last 10 events\r\n  const recentEvents = [...state.recentEvents, event].slice(-10);\r\n  return {\r\n    ...state,\r\n    recentEvents,\r\n  };\r\n}\r\n\r\nexport function calculateTargetTension(\r\n  currentChapter: number,\r\n  totalChapters: number\r\n): number {\r\n  // Parabolic tension curve: low at start, high in middle, resolution at end\r\n  const progress = currentChapter / totalChapters;\r\n  return 4 * progress * (1 - progress);\r\n}\r\n\r\nexport function updateStoryTension(\r\n  state: StoryStructuredState,\r\n  currentChapter: number,\r\n  totalChapters: number\r\n): StoryStructuredState {\r\n  const targetTension = calculateTargetTension(currentChapter, totalChapters);\r\n  \r\n  // Blend current tension toward target (smooth transition)\r\n  const newTension = state.tension * 0.7 + targetTension * 0.3;\r\n  \r\n  return {\r\n    ...state,\r\n    chapter: currentChapter,\r\n    tension: Math.round(newTension * 100) / 100, // Round to 2 decimals\r\n  };\r\n}\r\n\r\nexport function formatStructuredStateForPrompt(state: StoryStructuredState): string {\r\n  const lines: string[] = ['## Current Story State'];\r\n  \r\n  // Overall tension\r\n  const tensionPercent = Math.round(state.tension * 100);\r\n  lines.push(`\\n**Story Tension:** ${tensionPercent}%`);\r\n  \r\n  // Active plot threads\r\n  const activeThreads = Object.values(state.plotThreads)\r\n    .filter(t => t.status !== 'resolved' && t.status !== 'dormant');\r\n  \r\n  if (activeThreads.length > 0) {\r\n    lines.push('\\n### Active Plot Threads');\r\n    for (const thread of activeThreads) {\r\n      const threadTension = Math.round(thread.tension * 100);\r\n      lines.push(`- **${thread.name}** (${thread.status}, tension: ${threadTension}%)`);\r\n      lines.push(`  - ${thread.summary}`);\r\n    }\r\n  }\r\n  \r\n  // Character states\r\n  const characters = Object.values(state.characters);\r\n  if (characters.length > 0) {\r\n    lines.push('\\n### Character States');\r\n    for (const char of characters) {\r\n      lines.push(`- **${char.name}**: ${char.emotionalState}`);\r\n      if (char.location) lines.push(`  - Location: ${char.location}`);\r\n      if (Object.keys(char.relationships).length > 0) {\r\n        const rels = Object.entries(char.relationships)\r\n          .map(([name, status]) => `${name} (${status})`)\r\n          .join(', ');\r\n        lines.push(`  - Relationships: ${rels}`);\r\n      }\r\n    }\r\n  }\r\n  \r\n  // Unresolved questions\r\n  if (state.unresolvedQuestions.length > 0) {\r\n    lines.push('\\n### Unresolved Questions');\r\n    for (const q of state.unresolvedQuestions) {\r\n      lines.push(`- ${q}`);\r\n    }\r\n  }\r\n  \r\n  // Recent events\r\n  if (state.recentEvents.length > 0) {\r\n    lines.push('\\n### Recent Events');\r\n    for (const event of state.recentEvents.slice(-5)) {\r\n      lines.push(`- ${event}`);\r\n    }\r\n  }\r\n  \r\n  return lines.join('\\n');\r\n}\r\n"]}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,104 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const fs_1 = require("fs");
4
+ const path_1 = require("path");
5
+ const os_1 = require("os");
6
+ // Load config BEFORE importing engine
7
+ const configPath = (0, path_1.join)((0, os_1.homedir)(), '.narrative-os', 'config.json');
8
+ if ((0, fs_1.existsSync)(configPath)) {
9
+ const config = JSON.parse((0, fs_1.readFileSync)(configPath, 'utf-8'));
10
+ process.env.LLM_PROVIDER = config.provider;
11
+ process.env.LLM_MODEL = 'deepseek-chat';
12
+ if (config.provider === 'openai') {
13
+ process.env.OPENAI_API_KEY = config.apiKey;
14
+ }
15
+ else if (config.provider === 'deepseek') {
16
+ process.env.DEEPSEEK_API_KEY = config.apiKey;
17
+ }
18
+ console.log(`Loaded config: ${config.provider} / ${config.model}`);
19
+ }
20
+ const index_js_1 = require("../index.js");
21
+ async function testCanonMemory() {
22
+ console.log('Testing Canon Memory System (Phase 2)...\n');
23
+ // Test 1: Canon Store Creation
24
+ console.log('Test 1: Canon Store Creation');
25
+ const canon = (0, index_js_1.createCanonStore)('test-story-123');
26
+ console.log('✅ Canon store created');
27
+ // Test 2: Adding Facts
28
+ console.log('\nTest 2: Adding Facts');
29
+ let canonWithFacts = (0, index_js_1.addFact)(canon, { category: 'character', subject: '李白', attribute: '职业', value: '诗人', chapterEstablished: 1 });
30
+ canonWithFacts = (0, index_js_1.addFact)(canonWithFacts, { category: 'character', subject: '李白', attribute: '朝代', value: '唐朝', chapterEstablished: 1 });
31
+ canonWithFacts = (0, index_js_1.addFact)(canonWithFacts, { category: 'world', subject: '长安', attribute: '地位', value: '唐朝都城', chapterEstablished: 1 });
32
+ canonWithFacts = (0, index_js_1.addFact)(canonWithFacts, { category: 'plot', subject: '主线', attribute: '目标', value: '寻找人生真谛', chapterEstablished: 1 });
33
+ console.log(`✅ Added ${canonWithFacts.facts.length} facts`);
34
+ // Use the updated canon for subsequent tests
35
+ Object.assign(canon, canonWithFacts);
36
+ // Test 3: Format Canon for Prompt
37
+ console.log('\nTest 3: Format Canon for Prompt');
38
+ const formatted = (0, index_js_1.formatCanonForPrompt)(canon);
39
+ console.log('Formatted canon:');
40
+ console.log(formatted);
41
+ console.log('✅ Canon formatted successfully');
42
+ // Test 4: Extract Canon from Bible
43
+ console.log('\nTest 4: Extract Canon from StoryBible');
44
+ const bible = (0, index_js_1.createStoryBible)('测试故事', '测试主题', '测试类型', '测试背景', '测试基调', '测试前提', 5);
45
+ const bibleWithChar = (0, index_js_1.addCharacter)(bible, '杜甫', 'protagonist', ['忧国忧民', '诗圣'], ['记录时代', '关怀百姓']);
46
+ const bibleWithPlot = (0, index_js_1.addPlotThread)(bibleWithChar, '安史之乱', '唐朝由盛转衰的转折点');
47
+ const extractedCanon = (0, index_js_1.extractCanonFromBible)(bibleWithPlot);
48
+ console.log(`✅ Extracted ${extractedCanon.facts.length} facts from bible`);
49
+ console.log('Extracted facts:');
50
+ extractedCanon.facts.forEach(f => {
51
+ console.log(` - ${f.category}: ${f.subject} / ${f.attribute} = ${f.value}`);
52
+ });
53
+ // Test 5: Canon Validation
54
+ console.log('\nTest 5: Canon Validation');
55
+ const testContent = `
56
+ 杜甫是唐朝著名的诗人,被称为"诗圣"。
57
+ 他生活在安史之乱时期,目睹了唐朝的衰落。
58
+ 杜甫非常关心百姓疾苦,写下了许多反映社会现实的诗篇。
59
+ `;
60
+ const validation = await index_js_1.canonValidator.validate(testContent, extractedCanon);
61
+ console.log(`✅ Validation complete`);
62
+ console.log(`Violations found: ${validation.violations.length}`);
63
+ if (validation.violations.length > 0) {
64
+ console.log('Violations:');
65
+ validation.violations.forEach(v => console.log(` - ${v}`));
66
+ }
67
+ // Test 6: Canon Serialization
68
+ console.log('\nTest 6: Canon Serialization');
69
+ const serialized = JSON.stringify(canon, null, 2);
70
+ const deserialized = JSON.parse(serialized);
71
+ console.log(`✅ Serialization round-trip successful`);
72
+ console.log(`Original facts: ${canon.facts.length}, Deserialized: ${deserialized.facts.length}`);
73
+ console.log('\n✅ All Canon Memory tests passed!');
74
+ }
75
+ // Test 7: Detect Contradiction
76
+ async function testContradictionDetection() {
77
+ console.log('\n--- Testing Contradiction Detection ---\n');
78
+ let canon = (0, index_js_1.createCanonStore)('contradiction-test');
79
+ canon = (0, index_js_1.addFact)(canon, { category: 'character', subject: '李白', attribute: '状态', value: '活着', chapterEstablished: 1 });
80
+ canon = (0, index_js_1.addFact)(canon, { category: 'world', subject: '唐朝', attribute: '时期', value: '盛世', chapterEstablished: 1 });
81
+ // Content that contradicts canon
82
+ const contradictingContent = `
83
+ 李白已经去世多年,他的诗歌流传千古。
84
+ 唐朝正处于战乱之中,百姓流离失所。
85
+ `;
86
+ console.log('Canon facts:');
87
+ canon.facts.forEach(f => console.log(` - ${f.subject}: ${f.attribute} = ${f.value}`));
88
+ console.log('\nContent to validate:');
89
+ console.log(contradictingContent);
90
+ const validation = await index_js_1.canonValidator.validate(contradictingContent, canon);
91
+ console.log(`\nViolations detected: ${validation.violations.length}`);
92
+ validation.violations.forEach(v => console.log(` ⚠️ ${v}`));
93
+ }
94
+ testCanonMemory()
95
+ .then(() => testContradictionDetection())
96
+ .then(() => {
97
+ console.log('\n🎉 Phase 2 (Canon Memory) tests complete!');
98
+ process.exit(0);
99
+ })
100
+ .catch((error) => {
101
+ console.error('\n❌ Test failed:', error);
102
+ process.exit(1);
103
+ });
104
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"canon.test.js","sourceRoot":"","sources":["../../src/test/canon.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { readFileSync, existsSync } from 'fs';\r\nimport { join } from 'path';\r\nimport { homedir } from 'os';\r\n\r\n// Load config BEFORE importing engine\r\nconst configPath = join(homedir(), '.narrative-os', 'config.json');\r\nif (existsSync(configPath)) {\r\n  const config = JSON.parse(readFileSync(configPath, 'utf-8'));\r\n  process.env.LLM_PROVIDER = config.provider;\r\n  process.env.LLM_MODEL = 'deepseek-chat';\r\n  if (config.provider === 'openai') {\r\n    process.env.OPENAI_API_KEY = config.apiKey;\r\n  } else if (config.provider === 'deepseek') {\r\n    process.env.DEEPSEEK_API_KEY = config.apiKey;\r\n  }\r\n  console.log(`Loaded config: ${config.provider} / ${config.model}`);\r\n}\r\n\r\nimport {\r\n  createCanonStore,\r\n  addFact,\r\n  formatCanonForPrompt,\r\n  extractCanonFromBible,\r\n  createStoryBible,\r\n  addCharacter,\r\n  addPlotThread,\r\n  canonValidator,\r\n} from '../index.js';\r\nimport type { CanonStore, StoryBible } from '../index.js';\r\n\r\nasync function testCanonMemory() {\r\n  console.log('Testing Canon Memory System (Phase 2)...\\n');\r\n\r\n  // Test 1: Canon Store Creation\r\n  console.log('Test 1: Canon Store Creation');\r\n  const canon = createCanonStore('test-story-123');\r\n  console.log('✅ Canon store created');\r\n\r\n  // Test 2: Adding Facts\r\n  console.log('\\nTest 2: Adding Facts');\r\n  let canonWithFacts = addFact(canon, { category: 'character', subject: '李白', attribute: '职业', value: '诗人', chapterEstablished: 1 });\r\n  canonWithFacts = addFact(canonWithFacts, { category: 'character', subject: '李白', attribute: '朝代', value: '唐朝', chapterEstablished: 1 });\r\n  canonWithFacts = addFact(canonWithFacts, { category: 'world', subject: '长安', attribute: '地位', value: '唐朝都城', chapterEstablished: 1 });\r\n  canonWithFacts = addFact(canonWithFacts, { category: 'plot', subject: '主线', attribute: '目标', value: '寻找人生真谛', chapterEstablished: 1 });\r\n  console.log(`✅ Added ${canonWithFacts.facts.length} facts`);\r\n  \r\n  // Use the updated canon for subsequent tests\r\n  Object.assign(canon, canonWithFacts);\r\n\r\n  // Test 3: Format Canon for Prompt\r\n  console.log('\\nTest 3: Format Canon for Prompt');\r\n  const formatted = formatCanonForPrompt(canon);\r\n  console.log('Formatted canon:');\r\n  console.log(formatted);\r\n  console.log('✅ Canon formatted successfully');\r\n\r\n  // Test 4: Extract Canon from Bible\r\n  console.log('\\nTest 4: Extract Canon from StoryBible');\r\n  const bible = createStoryBible(\r\n    '测试故事',\r\n    '测试主题',\r\n    '测试类型',\r\n    '测试背景',\r\n    '测试基调',\r\n    '测试前提',\r\n    5\r\n  );\r\n\r\n  const bibleWithChar = addCharacter(\r\n    bible,\r\n    '杜甫',\r\n    'protagonist',\r\n    ['忧国忧民', '诗圣'],\r\n    ['记录时代', '关怀百姓']\r\n  );\r\n\r\n  const bibleWithPlot = addPlotThread(\r\n    bibleWithChar,\r\n    '安史之乱',\r\n    '唐朝由盛转衰的转折点'\r\n  );\r\n\r\n  const extractedCanon = extractCanonFromBible(bibleWithPlot);\r\n  console.log(`✅ Extracted ${extractedCanon.facts.length} facts from bible`);\r\n  console.log('Extracted facts:');\r\n  extractedCanon.facts.forEach(f => {\r\n    console.log(`  - ${f.category}: ${f.subject} / ${f.attribute} = ${f.value}`);\r\n  });\r\n\r\n  // Test 5: Canon Validation\r\n  console.log('\\nTest 5: Canon Validation');\r\n  const testContent = `\r\n杜甫是唐朝著名的诗人，被称为\"诗圣\"。\r\n他生活在安史之乱时期，目睹了唐朝的衰落。\r\n杜甫非常关心百姓疾苦，写下了许多反映社会现实的诗篇。\r\n`;\r\n\r\n  const validation = await canonValidator.validate(testContent, extractedCanon);\r\n  console.log(`✅ Validation complete`);\r\n  console.log(`Violations found: ${validation.violations.length}`);\r\n  if (validation.violations.length > 0) {\r\n    console.log('Violations:');\r\n    validation.violations.forEach(v => console.log(`  - ${v}`));\r\n  }\r\n\r\n  // Test 6: Canon Serialization\r\n  console.log('\\nTest 6: Canon Serialization');\r\n  const serialized = JSON.stringify(canon, null, 2);\r\n  const deserialized: CanonStore = JSON.parse(serialized);\r\n  console.log(`✅ Serialization round-trip successful`);\r\n  console.log(`Original facts: ${canon.facts.length}, Deserialized: ${deserialized.facts.length}`);\r\n\r\n  console.log('\\n✅ All Canon Memory tests passed!');\r\n}\r\n\r\n// Test 7: Detect Contradiction\r\nasync function testContradictionDetection() {\r\n  console.log('\\n--- Testing Contradiction Detection ---\\n');\r\n\r\n  let canon = createCanonStore('contradiction-test');\r\n  canon = addFact(canon, { category: 'character', subject: '李白', attribute: '状态', value: '活着', chapterEstablished: 1 });\r\n  canon = addFact(canon, { category: 'world', subject: '唐朝', attribute: '时期', value: '盛世', chapterEstablished: 1 });\r\n\r\n  // Content that contradicts canon\r\n  const contradictingContent = `\r\n李白已经去世多年，他的诗歌流传千古。\r\n唐朝正处于战乱之中，百姓流离失所。\r\n`;\r\n\r\n  console.log('Canon facts:');\r\n  canon.facts.forEach(f => console.log(`  - ${f.subject}: ${f.attribute} = ${f.value}`));\r\n\r\n  console.log('\\nContent to validate:');\r\n  console.log(contradictingContent);\r\n\r\n  const validation = await canonValidator.validate(contradictingContent, canon);\r\n  console.log(`\\nViolations detected: ${validation.violations.length}`);\r\n  validation.violations.forEach(v => console.log(`  ⚠️  ${v}`));\r\n}\r\n\r\ntestCanonMemory()\r\n  .then(() => testContradictionDetection())\r\n  .then(() => {\r\n    console.log('\\n🎉 Phase 2 (Canon Memory) tests complete!');\r\n    process.exit(0);\r\n  })\r\n  .catch((error) => {\r\n    console.error('\\n❌ Test failed:', error);\r\n    process.exit(1);\r\n  });\r\n"]}
@@ -0,0 +1 @@
1
+ export {};