@narrative-os/engine 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/agents/canonValidator.d.ts +9 -0
- package/dist/agents/canonValidator.js +51 -0
- package/dist/agents/chapterPlanner.d.ts +50 -0
- package/dist/agents/chapterPlanner.js +250 -0
- package/dist/agents/completeness.d.ts +7 -0
- package/dist/agents/completeness.js +51 -0
- package/dist/agents/memoryExtractor.d.ts +12 -0
- package/dist/agents/memoryExtractor.js +82 -0
- package/dist/agents/stateUpdater.d.ts +30 -0
- package/dist/agents/stateUpdater.js +150 -0
- package/dist/agents/storyDirector.d.ts +40 -0
- package/dist/agents/storyDirector.js +213 -0
- package/dist/agents/summarizer.d.ts +8 -0
- package/dist/agents/summarizer.js +56 -0
- package/dist/agents/tensionController.d.ts +68 -0
- package/dist/agents/tensionController.js +197 -0
- package/dist/agents/writer.d.ts +12 -0
- package/dist/agents/writer.js +148 -0
- package/dist/constraints/constraintGraph.d.ts +117 -0
- package/dist/constraints/constraintGraph.js +381 -0
- package/dist/constraints/validator.d.ts +58 -0
- package/dist/constraints/validator.js +236 -0
- package/dist/index.d.ts +25 -0
- package/dist/index.js +115 -0
- package/dist/llm/client.d.ts +14 -0
- package/dist/llm/client.js +108 -0
- package/dist/memory/canonStore.d.ts +20 -0
- package/dist/memory/canonStore.js +110 -0
- package/dist/memory/memoryRetriever.d.ts +28 -0
- package/dist/memory/memoryRetriever.js +126 -0
- package/dist/memory/stateUpdater.d.ts +49 -0
- package/dist/memory/stateUpdater.js +315 -0
- package/dist/memory/vectorStore.d.ts +41 -0
- package/dist/memory/vectorStore.js +166 -0
- package/dist/pipeline/generateChapter.d.ts +17 -0
- package/dist/pipeline/generateChapter.js +75 -0
- package/dist/story/bible.d.ts +4 -0
- package/dist/story/bible.js +53 -0
- package/dist/story/state.d.ts +3 -0
- package/dist/story/state.js +27 -0
- package/dist/story/structuredState.d.ts +39 -0
- package/dist/story/structuredState.js +159 -0
- package/dist/test/canon.test.d.ts +1 -0
- package/dist/test/canon.test.js +104 -0
- package/dist/test/chapter-planner.test.d.ts +1 -0
- package/dist/test/chapter-planner.test.js +171 -0
- package/dist/test/constraints.test.d.ts +1 -0
- package/dist/test/constraints.test.js +210 -0
- package/dist/test/simple.test.d.ts +1 -0
- package/dist/test/simple.test.js +51 -0
- package/dist/test/state-updater.test.d.ts +1 -0
- package/dist/test/state-updater.test.js +200 -0
- package/dist/test/story-director.test.d.ts +1 -0
- package/dist/test/story-director.test.js +142 -0
- package/dist/test/structured-state.test.d.ts +1 -0
- package/dist/test/structured-state.test.js +144 -0
- package/dist/test/tension-controller.test.d.ts +1 -0
- package/dist/test/tension-controller.test.js +116 -0
- package/dist/test/vector-memory.test.d.ts +1 -0
- package/dist/test/vector-memory.test.js +153 -0
- package/dist/test/world-simulation.test.d.ts +1 -0
- package/dist/test/world-simulation.test.js +152 -0
- package/dist/types/index.d.ts +79 -0
- package/dist/types/index.js +3 -0
- package/dist/world/characterAgent.d.ts +73 -0
- package/dist/world/characterAgent.js +232 -0
- package/dist/world/eventResolver.d.ts +52 -0
- package/dist/world/eventResolver.js +205 -0
- package/dist/world/worldState.d.ts +93 -0
- package/dist/world/worldState.js +258 -0
- package/package.json +43 -0
|
@@ -0,0 +1,200 @@
|
|
|
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 (to initialize LLM correctly)
|
|
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
|
+
// Use deepseek-chat (reasoner may not be available for all API keys)
|
|
12
|
+
process.env.LLM_MODEL = 'deepseek-chat';
|
|
13
|
+
if (config.provider === 'openai') {
|
|
14
|
+
process.env.OPENAI_API_KEY = config.apiKey;
|
|
15
|
+
}
|
|
16
|
+
else if (config.provider === 'deepseek') {
|
|
17
|
+
process.env.DEEPSEEK_API_KEY = config.apiKey;
|
|
18
|
+
}
|
|
19
|
+
console.log(`Loaded config: ${config.provider} / ${config.model}`);
|
|
20
|
+
}
|
|
21
|
+
const index_js_1 = require("../index.js");
|
|
22
|
+
console.log('Testing Memory + Graph Updates Pipeline (Phase 10)...\n');
|
|
23
|
+
// Setup test data
|
|
24
|
+
const bible = (0, index_js_1.createStoryBible)('侦探故事', '真相与正义', 'mystery', '1920年代上海', '悬疑紧张', '侦探调查连环杀人案,揭开真相', 10);
|
|
25
|
+
let structuredState = (0, index_js_1.createStructuredState)('test-story');
|
|
26
|
+
structuredState = (0, index_js_1.initializeCharactersFromBible)(structuredState, (0, index_js_1.addCharacter)((0, index_js_1.addCharacter)(bible, '陈侦探', 'protagonist', ['聪明', '冷静', '执着'], ['找出凶手', '维护正义']), '凶手', 'antagonist', ['狡猾', '残忍', '伪装'], ['逃避追捕', '继续犯罪']));
|
|
27
|
+
// Set up character states
|
|
28
|
+
structuredState.characters['陈侦探'].emotionalState = '专注';
|
|
29
|
+
structuredState.characters['陈侦探'].location = '上海滩';
|
|
30
|
+
structuredState.characters['陈侦探'].knowledge = ['凶手使用特殊刀具'];
|
|
31
|
+
structuredState.characters['凶手'].emotionalState = '警惕';
|
|
32
|
+
structuredState.characters['凶手'].location = '法租界';
|
|
33
|
+
// Set up plot threads
|
|
34
|
+
structuredState.plotThreads['main-mystery'] = {
|
|
35
|
+
id: 'main-mystery',
|
|
36
|
+
name: '连环杀人案调查',
|
|
37
|
+
status: 'active',
|
|
38
|
+
tension: 0.6,
|
|
39
|
+
summary: '陈侦探正在调查连环杀人案',
|
|
40
|
+
lastChapter: 1,
|
|
41
|
+
involvedCharacters: ['陈侦探'],
|
|
42
|
+
};
|
|
43
|
+
// Test 1: Initialize pipeline components
|
|
44
|
+
console.log('Test 1: Initialize Pipeline Components');
|
|
45
|
+
const constraintGraph = (0, index_js_1.createConstraintGraph)();
|
|
46
|
+
constraintGraph.addLocation('上海滩', '繁华码头区', 1);
|
|
47
|
+
constraintGraph.addLocation('法租界', '外国人居住区', 1);
|
|
48
|
+
constraintGraph.addCharacter(structuredState.characters['陈侦探'], 1);
|
|
49
|
+
constraintGraph.addCharacter(structuredState.characters['凶手'], 1);
|
|
50
|
+
const vectorStore = new index_js_1.VectorStore('test-story');
|
|
51
|
+
const canon = (0, index_js_1.createCanonStore)('test-story');
|
|
52
|
+
console.log(` Constraint graph: ${constraintGraph.getStats().nodes} nodes`);
|
|
53
|
+
console.log(` Vector store: ready`);
|
|
54
|
+
console.log(` Canon store: ready`);
|
|
55
|
+
console.log('✅ Pipeline components initialized');
|
|
56
|
+
// Test 2: Create test chapter
|
|
57
|
+
console.log('\nTest 2: Create Test Chapter');
|
|
58
|
+
const testChapter = {
|
|
59
|
+
id: 'ch-2',
|
|
60
|
+
storyId: 'test-story',
|
|
61
|
+
number: 2,
|
|
62
|
+
title: '法租界追踪',
|
|
63
|
+
content: '陈侦探在法租界发现了凶手的踪迹。他跟踪凶手来到一条小巷,发现了关键证据。陈侦探感到兴奋,因为他离真相更近了一步。',
|
|
64
|
+
wordCount: 800,
|
|
65
|
+
summary: '陈侦探在法租界追踪凶手并发现关键证据',
|
|
66
|
+
generatedAt: new Date(),
|
|
67
|
+
};
|
|
68
|
+
console.log(` Chapter ${testChapter.number}: ${testChapter.title}`);
|
|
69
|
+
console.log('✅ Test chapter created');
|
|
70
|
+
// Test 3-5: Run async tests
|
|
71
|
+
async function runAsyncTests() {
|
|
72
|
+
await vectorStore.initialize();
|
|
73
|
+
console.log(' Vector store initialized');
|
|
74
|
+
// Test 3: Quick update (no LLM)
|
|
75
|
+
console.log('\nTest 3: Quick Update (No LLM)');
|
|
76
|
+
const pipeline = new index_js_1.StateUpdaterPipeline();
|
|
77
|
+
const quickResult = await pipeline.quickUpdate({
|
|
78
|
+
chapter: testChapter,
|
|
79
|
+
bible,
|
|
80
|
+
currentState: structuredState,
|
|
81
|
+
canon,
|
|
82
|
+
vectorStore,
|
|
83
|
+
constraintGraph,
|
|
84
|
+
});
|
|
85
|
+
console.log(` Memories added: ${quickResult.memoriesAdded}`);
|
|
86
|
+
console.log(` Graph updated: ${quickResult.graphUpdated}`);
|
|
87
|
+
console.log(` Changes: ${quickResult.changes.length}`);
|
|
88
|
+
for (const change of quickResult.changes) {
|
|
89
|
+
console.log(` - [${change.type}] ${change.description}`);
|
|
90
|
+
}
|
|
91
|
+
console.log('✅ Quick update complete');
|
|
92
|
+
// Test 4: Verify vector store updated
|
|
93
|
+
console.log('\nTest 4: Verify Vector Store Updated');
|
|
94
|
+
const allMemories = vectorStore.getAllMemories();
|
|
95
|
+
console.log(` Total memories: ${allMemories.length}`);
|
|
96
|
+
for (const mem of allMemories) {
|
|
97
|
+
console.log(` - [${mem.category}] ${mem.content.substring(0, 40)}...`);
|
|
98
|
+
}
|
|
99
|
+
console.log('✅ Vector store verified');
|
|
100
|
+
// Test 5: Verify constraint graph updated
|
|
101
|
+
console.log('\nTest 5: Verify Constraint Graph Updated');
|
|
102
|
+
const graphStats = constraintGraph.getStats();
|
|
103
|
+
console.log(` Nodes: ${graphStats.nodes} (including events)`);
|
|
104
|
+
console.log(` Edges: ${graphStats.edges}`);
|
|
105
|
+
const eventNodes = Array.from(constraintGraph.nodes.values())
|
|
106
|
+
.filter((n) => n.type === 'event');
|
|
107
|
+
console.log(` Events: ${eventNodes.length}`);
|
|
108
|
+
console.log('✅ Constraint graph verified');
|
|
109
|
+
return { pipeline, quickResult };
|
|
110
|
+
}
|
|
111
|
+
// Run async tests
|
|
112
|
+
runAsyncTests().then(async ({ pipeline, quickResult }) => {
|
|
113
|
+
// Test 6: Verify structured state updated
|
|
114
|
+
console.log('\nTest 6: Verify Structured State Updated');
|
|
115
|
+
console.log(` Chapter updated to: ${quickResult.structuredState.chapter}`);
|
|
116
|
+
console.log(` Recent events: ${quickResult.structuredState.recentEvents.length}`);
|
|
117
|
+
for (const event of quickResult.structuredState.recentEvents) {
|
|
118
|
+
console.log(` - ${event.substring(0, 40)}...`);
|
|
119
|
+
}
|
|
120
|
+
console.log('✅ Structured state verified');
|
|
121
|
+
// Test 7: Format result
|
|
122
|
+
console.log('\nTest 7: Format Update Result');
|
|
123
|
+
const formatted = pipeline.formatResult(quickResult);
|
|
124
|
+
console.log('--- Formatted Result ---');
|
|
125
|
+
console.log(formatted);
|
|
126
|
+
console.log('✅ Result formatted');
|
|
127
|
+
// Test 8: Full update with LLM
|
|
128
|
+
console.log('\nTest 8: Full Update with LLM');
|
|
129
|
+
try {
|
|
130
|
+
const llmResult = await pipeline.update({
|
|
131
|
+
chapter: testChapter,
|
|
132
|
+
bible,
|
|
133
|
+
currentState: structuredState,
|
|
134
|
+
canon,
|
|
135
|
+
vectorStore,
|
|
136
|
+
constraintGraph,
|
|
137
|
+
});
|
|
138
|
+
console.log(` LLM Memories added: ${llmResult.memoriesAdded}`);
|
|
139
|
+
console.log(` LLM Canon facts: ${llmResult.canonFactsAdded}`);
|
|
140
|
+
console.log(` LLM Changes: ${llmResult.changes.length}`);
|
|
141
|
+
for (const change of llmResult.changes.slice(0, 5)) {
|
|
142
|
+
console.log(` - [${change.type}] ${change.description.substring(0, 50)}...`);
|
|
143
|
+
}
|
|
144
|
+
console.log('✅ LLM update complete');
|
|
145
|
+
}
|
|
146
|
+
catch (error) {
|
|
147
|
+
console.log(' ⚠️ LLM update skipped (API unavailable)');
|
|
148
|
+
}
|
|
149
|
+
// Test 9: Multiple chapters simulation
|
|
150
|
+
console.log('\nTest 9: Multiple Chapters Simulation');
|
|
151
|
+
const chapters = [
|
|
152
|
+
{
|
|
153
|
+
id: 'ch-3',
|
|
154
|
+
storyId: 'test-story',
|
|
155
|
+
number: 3,
|
|
156
|
+
title: '证据分析',
|
|
157
|
+
content: '陈侦探回到办公室分析证据。',
|
|
158
|
+
wordCount: 600,
|
|
159
|
+
summary: '陈侦探分析收集到的证据',
|
|
160
|
+
generatedAt: new Date(),
|
|
161
|
+
},
|
|
162
|
+
{
|
|
163
|
+
id: 'ch-4',
|
|
164
|
+
storyId: 'test-story',
|
|
165
|
+
number: 4,
|
|
166
|
+
title: '真相大白',
|
|
167
|
+
content: '陈侦探终于揭开了凶手的真面目。',
|
|
168
|
+
wordCount: 900,
|
|
169
|
+
summary: '陈侦探揭开真相',
|
|
170
|
+
generatedAt: new Date(),
|
|
171
|
+
},
|
|
172
|
+
];
|
|
173
|
+
let currentState = structuredState;
|
|
174
|
+
for (const chapter of chapters) {
|
|
175
|
+
const result = await pipeline.quickUpdate({
|
|
176
|
+
chapter,
|
|
177
|
+
bible,
|
|
178
|
+
currentState,
|
|
179
|
+
canon,
|
|
180
|
+
vectorStore,
|
|
181
|
+
constraintGraph,
|
|
182
|
+
});
|
|
183
|
+
currentState = result.structuredState;
|
|
184
|
+
console.log(` Ch ${chapter.number}: +${result.memoriesAdded} memories`);
|
|
185
|
+
}
|
|
186
|
+
console.log(` Total memories after 3 chapters: ${vectorStore.getAllMemories().length}`);
|
|
187
|
+
console.log('✅ Multi-chapter simulation complete');
|
|
188
|
+
// Test 10: Search memories
|
|
189
|
+
console.log('\nTest 10: Search Memories');
|
|
190
|
+
const searchResults = await vectorStore.searchSimilar('凶手', 3);
|
|
191
|
+
console.log(` Search '凶手': ${searchResults.length} results`);
|
|
192
|
+
for (const result of searchResults) {
|
|
193
|
+
console.log(` - ${result.memory.content.substring(0, 40)}... (score: ${result.score.toFixed(3)})`);
|
|
194
|
+
}
|
|
195
|
+
console.log('✅ Memory search working');
|
|
196
|
+
console.log('\n✅ All State Updater Pipeline tests passed!');
|
|
197
|
+
console.log('\n🎉 Phase 10 (Memory + Graph Updates) tests complete!');
|
|
198
|
+
console.log('\n🎊 ALL 10 PHASES COMPLETE! Narrative OS core is ready!');
|
|
199
|
+
});
|
|
200
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"state-updater.test.js","sourceRoot":"","sources":["../../src/test/state-updater.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 (to initialize LLM correctly)\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  // Use deepseek-chat (reasoner may not be available for all API keys)\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  createConstraintGraph,\r\n  VectorStore,\r\n  createCanonStore,\r\n  createStructuredState,\r\n  initializeCharactersFromBible,\r\n  createStoryBible,\r\n  addCharacter,\r\n  StateUpdaterPipeline,\r\n  type Chapter,\r\n} from '../index.js';\r\n\r\nconsole.log('Testing Memory + Graph Updates Pipeline (Phase 10)...\\n');\r\n\r\n// Setup test data\r\nconst bible = createStoryBible(\r\n  '侦探故事',\r\n  '真相与正义',\r\n  'mystery',\r\n  '1920年代上海',\r\n  '悬疑紧张',\r\n  '侦探调查连环杀人案，揭开真相',\r\n  10\r\n);\r\n\r\nlet structuredState = createStructuredState('test-story');\r\nstructuredState = initializeCharactersFromBible(\r\n  structuredState,\r\n  addCharacter(\r\n    addCharacter(bible, '陈侦探', 'protagonist', ['聪明', '冷静', '执着'], ['找出凶手', '维护正义']),\r\n    '凶手',\r\n    'antagonist',\r\n    ['狡猾', '残忍', '伪装'],\r\n    ['逃避追捕', '继续犯罪']\r\n  )\r\n);\r\n\r\n// Set up character states\r\nstructuredState.characters['陈侦探'].emotionalState = '专注';\r\nstructuredState.characters['陈侦探'].location = '上海滩';\r\nstructuredState.characters['陈侦探'].knowledge = ['凶手使用特殊刀具'];\r\n\r\nstructuredState.characters['凶手'].emotionalState = '警惕';\r\nstructuredState.characters['凶手'].location = '法租界';\r\n\r\n// Set up plot threads\r\nstructuredState.plotThreads['main-mystery'] = {\r\n  id: 'main-mystery',\r\n  name: '连环杀人案调查',\r\n  status: 'active',\r\n  tension: 0.6,\r\n  summary: '陈侦探正在调查连环杀人案',\r\n  lastChapter: 1,\r\n  involvedCharacters: ['陈侦探'],\r\n};\r\n\r\n// Test 1: Initialize pipeline components\r\nconsole.log('Test 1: Initialize Pipeline Components');\r\nconst constraintGraph = createConstraintGraph();\r\nconstraintGraph.addLocation('上海滩', '繁华码头区', 1);\r\nconstraintGraph.addLocation('法租界', '外国人居住区', 1);\r\nconstraintGraph.addCharacter(structuredState.characters['陈侦探'], 1);\r\nconstraintGraph.addCharacter(structuredState.characters['凶手'], 1);\r\n\r\nconst vectorStore = new VectorStore('test-story');\r\nconst canon = createCanonStore('test-story');\r\n\r\nconsole.log(`  Constraint graph: ${constraintGraph.getStats().nodes} nodes`);\r\nconsole.log(`  Vector store: ready`);\r\nconsole.log(`  Canon store: ready`);\r\nconsole.log('✅ Pipeline components initialized');\r\n\r\n// Test 2: Create test chapter\r\nconsole.log('\\nTest 2: Create Test Chapter');\r\nconst testChapter: Chapter = {\r\n  id: 'ch-2',\r\n  storyId: 'test-story',\r\n  number: 2,\r\n  title: '法租界追踪',\r\n  content: '陈侦探在法租界发现了凶手的踪迹。他跟踪凶手来到一条小巷，发现了关键证据。陈侦探感到兴奋，因为他离真相更近了一步。',\r\n  wordCount: 800,\r\n  summary: '陈侦探在法租界追踪凶手并发现关键证据',\r\n  generatedAt: new Date(),\r\n};\r\nconsole.log(`  Chapter ${testChapter.number}: ${testChapter.title}`);\r\nconsole.log('✅ Test chapter created');\r\n\r\n// Test 3-5: Run async tests\r\nasync function runAsyncTests() {\r\n  await vectorStore.initialize();\r\n  console.log('  Vector store initialized');\r\n  \r\n  // Test 3: Quick update (no LLM)\r\n  console.log('\\nTest 3: Quick Update (No LLM)');\r\n  const pipeline = new StateUpdaterPipeline();\r\n  const quickResult = await pipeline.quickUpdate({\r\n    chapter: testChapter,\r\n    bible,\r\n    currentState: structuredState,\r\n    canon,\r\n    vectorStore,\r\n    constraintGraph,\r\n  });\r\n  \r\n  console.log(`  Memories added: ${quickResult.memoriesAdded}`);\r\n  console.log(`  Graph updated: ${quickResult.graphUpdated}`);\r\n  console.log(`  Changes: ${quickResult.changes.length}`);\r\n  for (const change of quickResult.changes) {\r\n    console.log(`    - [${change.type}] ${change.description}`);\r\n  }\r\n  console.log('✅ Quick update complete');\r\n  \r\n  // Test 4: Verify vector store updated\r\n  console.log('\\nTest 4: Verify Vector Store Updated');\r\n  const allMemories = vectorStore.getAllMemories();\r\n  console.log(`  Total memories: ${allMemories.length}`);\r\n  for (const mem of allMemories) {\r\n    console.log(`    - [${mem.category}] ${mem.content.substring(0, 40)}...`);\r\n  }\r\n  console.log('✅ Vector store verified');\r\n  \r\n  // Test 5: Verify constraint graph updated\r\n  console.log('\\nTest 5: Verify Constraint Graph Updated');\r\n  const graphStats = constraintGraph.getStats();\r\n  console.log(`  Nodes: ${graphStats.nodes} (including events)`);\r\n  console.log(`  Edges: ${graphStats.edges}`);\r\n  const eventNodes = Array.from((constraintGraph as any).nodes.values())\r\n    .filter((n: any) => n.type === 'event');\r\n  console.log(`  Events: ${eventNodes.length}`);\r\n  console.log('✅ Constraint graph verified');\r\n  \r\n  return { pipeline, quickResult };\r\n}\r\n\r\n// Run async tests\r\nrunAsyncTests().then(async ({ pipeline, quickResult }) => {\r\n  // Test 6: Verify structured state updated\r\n  console.log('\\nTest 6: Verify Structured State Updated');\r\n  console.log(`  Chapter updated to: ${quickResult.structuredState.chapter}`);\r\n  console.log(`  Recent events: ${quickResult.structuredState.recentEvents.length}`);\r\n  for (const event of quickResult.structuredState.recentEvents) {\r\n    console.log(`    - ${event.substring(0, 40)}...`);\r\n  }\r\n  console.log('✅ Structured state verified');\r\n  \r\n  // Test 7: Format result\r\n  console.log('\\nTest 7: Format Update Result');\r\n  const formatted = pipeline.formatResult(quickResult);\r\n  console.log('--- Formatted Result ---');\r\n  console.log(formatted);\r\n  console.log('✅ Result formatted');\r\n  \r\n  // Test 8: Full update with LLM\r\n  console.log('\\nTest 8: Full Update with LLM');\r\n  \r\n  try {\r\n    const llmResult = await pipeline.update({\r\n      chapter: testChapter,\r\n      bible,\r\n      currentState: structuredState,\r\n      canon,\r\n      vectorStore,\r\n      constraintGraph,\r\n    });\r\n    \r\n    console.log(`  LLM Memories added: ${llmResult.memoriesAdded}`);\r\n    console.log(`  LLM Canon facts: ${llmResult.canonFactsAdded}`);\r\n    console.log(`  LLM Changes: ${llmResult.changes.length}`);\r\n    for (const change of llmResult.changes.slice(0, 5)) {\r\n      console.log(`    - [${change.type}] ${change.description.substring(0, 50)}...`);\r\n    }\r\n    console.log('✅ LLM update complete');\r\n  } catch (error) {\r\n    console.log('  ⚠️ LLM update skipped (API unavailable)');\r\n  }\r\n  \r\n  // Test 9: Multiple chapters simulation\r\n  console.log('\\nTest 9: Multiple Chapters Simulation');\r\n  const chapters: Chapter[] = [\r\n    {\r\n      id: 'ch-3',\r\n      storyId: 'test-story',\r\n      number: 3,\r\n      title: '证据分析',\r\n      content: '陈侦探回到办公室分析证据。',\r\n      wordCount: 600,\r\n      summary: '陈侦探分析收集到的证据',\r\n      generatedAt: new Date(),\r\n    },\r\n    {\r\n      id: 'ch-4',\r\n      storyId: 'test-story',\r\n      number: 4,\r\n      title: '真相大白',\r\n      content: '陈侦探终于揭开了凶手的真面目。',\r\n      wordCount: 900,\r\n      summary: '陈侦探揭开真相',\r\n      generatedAt: new Date(),\r\n    },\r\n  ];\r\n  \r\n  let currentState = structuredState;\r\n  for (const chapter of chapters) {\r\n    const result = await pipeline.quickUpdate({\r\n      chapter,\r\n      bible,\r\n      currentState,\r\n      canon,\r\n      vectorStore,\r\n      constraintGraph,\r\n    });\r\n    currentState = result.structuredState;\r\n    console.log(`  Ch ${chapter.number}: +${result.memoriesAdded} memories`);\r\n  }\r\n  \r\n  console.log(`  Total memories after 3 chapters: ${vectorStore.getAllMemories().length}`);\r\n  console.log('✅ Multi-chapter simulation complete');\r\n  \r\n  // Test 10: Search memories\r\n  console.log('\\nTest 10: Search Memories');\r\n  const searchResults = await vectorStore.searchSimilar('凶手', 3);\r\n  console.log(`  Search '凶手': ${searchResults.length} results`);\r\n  for (const result of searchResults) {\r\n    console.log(`    - ${result.memory.content.substring(0, 40)}... (score: ${result.score.toFixed(3)})`);\r\n  }\r\n  console.log('✅ Memory search working');\r\n  \r\n  console.log('\\n✅ All State Updater Pipeline tests passed!');\r\n  console.log('\\n🎉 Phase 10 (Memory + Graph Updates) tests complete!');\r\n  console.log('\\n🎊 ALL 10 PHASES COMPLETE! Narrative OS core is ready!');\r\n});\r\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,142 @@
|
|
|
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} / deepseek-chat`);
|
|
19
|
+
}
|
|
20
|
+
const index_js_1 = require("../index.js");
|
|
21
|
+
console.log('Testing Story Director Agent (Phase 6)...\n');
|
|
22
|
+
// Setup test story
|
|
23
|
+
let bible = (0, index_js_1.createStoryBible)('李白传奇', '诗人寻找人生真谛', 'historical-fiction', '唐朝长安及各地', '豪放浪漫', '诗人李白离开长安,游历天下,寻找人生真谛', 10);
|
|
24
|
+
bible = (0, index_js_1.addCharacter)(bible, '李白', 'protagonist', ['豪放', '浪漫', '不羁'], ['寻找人生真谛', '成为伟大诗人']);
|
|
25
|
+
bible = (0, index_js_1.addCharacter)(bible, '杜甫', 'supporting', ['沉郁', '忧国忧民'], ['记录时代变迁']);
|
|
26
|
+
bible = (0, index_js_1.addCharacter)(bible, '孟浩然', 'supporting', ['隐逸', '淡泊'], ['归隐山林']);
|
|
27
|
+
bible = (0, index_js_1.addPlotThread)(bible, '诗人之旅', '李白从长安出发,游历天下');
|
|
28
|
+
bible = (0, index_js_1.addPlotThread)(bible, '宫廷召唤', '唐玄宗召李白入宫的阴谋');
|
|
29
|
+
bible = (0, index_js_1.addPlotThread)(bible, '友情之路', '李白与杜甫、孟浩然的交往');
|
|
30
|
+
const storyState = (0, index_js_1.createStoryState)('test-story', 10);
|
|
31
|
+
storyState.currentChapter = 3;
|
|
32
|
+
let structuredState = (0, index_js_1.createStructuredState)('test-story');
|
|
33
|
+
structuredState = (0, index_js_1.initializeCharactersFromBible)(structuredState, bible);
|
|
34
|
+
structuredState = (0, index_js_1.initializePlotThreadsFromBible)(structuredState, bible);
|
|
35
|
+
// Activate some plot threads
|
|
36
|
+
const threadIds = Object.keys(structuredState.plotThreads);
|
|
37
|
+
if (threadIds[0]) {
|
|
38
|
+
structuredState = (0, index_js_1.updatePlotThread)(structuredState, threadIds[0], { status: 'active', tension: 0.6 }, 3);
|
|
39
|
+
}
|
|
40
|
+
if (threadIds[1]) {
|
|
41
|
+
structuredState = (0, index_js_1.updatePlotThread)(structuredState, threadIds[1], { status: 'escalating', tension: 0.4 }, 3);
|
|
42
|
+
}
|
|
43
|
+
// Add recent events
|
|
44
|
+
structuredState = (0, index_js_1.addRecentEvent)(structuredState, '李白告别长安');
|
|
45
|
+
structuredState = (0, index_js_1.addRecentEvent)(structuredState, '夜宿黄鹤楼');
|
|
46
|
+
structuredState = (0, index_js_1.addRecentEvent)(structuredState, '遇见孟浩然');
|
|
47
|
+
// Add questions
|
|
48
|
+
structuredState = (0, index_js_1.addUnresolvedQuestion)(structuredState, '李白会接受宫廷召唤吗?');
|
|
49
|
+
structuredState = (0, index_js_1.addUnresolvedQuestion)(structuredState, '杜甫何时与李白相遇?');
|
|
50
|
+
// Create tension guidance
|
|
51
|
+
const tensionAnalysis = (0, index_js_1.analyzeTension)(storyState, structuredState);
|
|
52
|
+
const tensionGuidance = (0, index_js_1.generateTensionGuidance)(tensionAnalysis, storyState);
|
|
53
|
+
// Create previous summaries
|
|
54
|
+
const previousSummaries = [
|
|
55
|
+
{
|
|
56
|
+
chapterNumber: 1,
|
|
57
|
+
summary: '李白离开长安,踏上诗人之旅',
|
|
58
|
+
keyEvents: ['告别长安', '独自上路'],
|
|
59
|
+
characterChanges: { '李白': '从迷茫到坚定' },
|
|
60
|
+
},
|
|
61
|
+
{
|
|
62
|
+
chapterNumber: 2,
|
|
63
|
+
summary: '李白夜宿黄鹤楼,思考人生',
|
|
64
|
+
keyEvents: ['夜宿黄鹤楼', '写诗抒怀'],
|
|
65
|
+
characterChanges: { '李白': '更加坚定诗人之路' },
|
|
66
|
+
},
|
|
67
|
+
{
|
|
68
|
+
chapterNumber: 3,
|
|
69
|
+
summary: '李白遇见孟浩然,结为知己',
|
|
70
|
+
keyEvents: ['遇见孟浩然', '饮酒论诗'],
|
|
71
|
+
characterChanges: { '李白': '找到知音', '孟浩然': '欣赏李白才华' },
|
|
72
|
+
},
|
|
73
|
+
];
|
|
74
|
+
// Test 1: Fallback objectives
|
|
75
|
+
console.log('Test 1: Fallback Objectives (No LLM)');
|
|
76
|
+
const fallbackOutput = index_js_1.storyDirector.generateFallbackObjectives(storyState, structuredState);
|
|
77
|
+
console.log(` Chapter: ${fallbackOutput.chapterNumber}`);
|
|
78
|
+
console.log(` Goal: ${fallbackOutput.overallGoal}`);
|
|
79
|
+
console.log(` Objectives: ${fallbackOutput.objectives.length}`);
|
|
80
|
+
console.log(` Focus: ${fallbackOutput.focusCharacters.join(', ')}`);
|
|
81
|
+
console.log('✅ Fallback objectives generated');
|
|
82
|
+
// Test 2: Format for prompt
|
|
83
|
+
console.log('\nTest 2: Format for Prompt');
|
|
84
|
+
const formatted = index_js_1.storyDirector.formatForPrompt(fallbackOutput);
|
|
85
|
+
console.log('--- Formatted Output ---');
|
|
86
|
+
console.log(formatted.substring(0, 600));
|
|
87
|
+
console.log('...');
|
|
88
|
+
console.log('✅ Formatted for writer prompt');
|
|
89
|
+
// Test 3: Director with LLM
|
|
90
|
+
console.log('\nTest 3: Story Director with LLM');
|
|
91
|
+
async function runDirectorTest() {
|
|
92
|
+
try {
|
|
93
|
+
const directorOutput = await index_js_1.storyDirector.direct({
|
|
94
|
+
bible,
|
|
95
|
+
state: storyState,
|
|
96
|
+
structuredState,
|
|
97
|
+
tensionGuidance,
|
|
98
|
+
previousSummaries,
|
|
99
|
+
});
|
|
100
|
+
console.log('✅ Director generated chapter plan');
|
|
101
|
+
console.log(` Chapter: ${directorOutput.chapterNumber}`);
|
|
102
|
+
console.log(` Goal: ${directorOutput.overallGoal}`);
|
|
103
|
+
console.log(` Tone: ${directorOutput.tone}`);
|
|
104
|
+
console.log(` Objectives: ${directorOutput.objectives.length}`);
|
|
105
|
+
for (const obj of directorOutput.objectives) {
|
|
106
|
+
const emoji = { critical: '🔴', high: '🟠', medium: '🟡', low: '🟢' }[obj.priority];
|
|
107
|
+
console.log(` ${emoji} [${obj.type}] ${obj.description}`);
|
|
108
|
+
}
|
|
109
|
+
console.log(` Focus Characters: ${directorOutput.focusCharacters.join(', ')}`);
|
|
110
|
+
console.log(` Suggested Scenes: ${directorOutput.suggestedScenes.length}`);
|
|
111
|
+
// Test 4: Format LLM output
|
|
112
|
+
console.log('\nTest 4: Format LLM Output for Prompt');
|
|
113
|
+
const llmFormatted = index_js_1.storyDirector.formatForPrompt(directorOutput);
|
|
114
|
+
console.log('--- Formatted LLM Output ---');
|
|
115
|
+
console.log(llmFormatted.substring(0, 800));
|
|
116
|
+
console.log('...');
|
|
117
|
+
console.log('✅ LLM output formatted');
|
|
118
|
+
}
|
|
119
|
+
catch (error) {
|
|
120
|
+
console.log('⚠️ Director LLM test failed:', error instanceof Error ? error.message : String(error));
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
runDirectorTest().then(() => {
|
|
124
|
+
// Test 5: Different story phases
|
|
125
|
+
console.log('\nTest 5: Different Story Phases');
|
|
126
|
+
// Early phase
|
|
127
|
+
storyState.currentChapter = 1;
|
|
128
|
+
const earlyFallback = index_js_1.storyDirector.generateFallbackObjectives(storyState, structuredState);
|
|
129
|
+
console.log(` Early (Ch 1): ${earlyFallback.objectives.length} objectives, tone: ${earlyFallback.tone}`);
|
|
130
|
+
// Middle phase
|
|
131
|
+
storyState.currentChapter = 5;
|
|
132
|
+
const middleFallback = index_js_1.storyDirector.generateFallbackObjectives(storyState, structuredState);
|
|
133
|
+
console.log(` Middle (Ch 5): ${middleFallback.objectives.length} objectives, tone: ${middleFallback.tone}`);
|
|
134
|
+
// Late phase
|
|
135
|
+
storyState.currentChapter = 9;
|
|
136
|
+
const lateFallback = index_js_1.storyDirector.generateFallbackObjectives(storyState, structuredState);
|
|
137
|
+
console.log(` Late (Ch 9): ${lateFallback.objectives.length} objectives, tone: ${lateFallback.tone}`);
|
|
138
|
+
console.log('✅ Objectives adapt to story phase');
|
|
139
|
+
console.log('\n✅ All Story Director tests passed!');
|
|
140
|
+
console.log('\n🎉 Phase 6 (Story Director Agent) tests complete!');
|
|
141
|
+
});
|
|
142
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"story-director.test.js","sourceRoot":"","sources":["../../src/test/story-director.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} / deepseek-chat`);\r\n}\r\n\r\nimport {\r\n  storyDirector,\r\n  createStoryBible,\r\n  addCharacter,\r\n  addPlotThread,\r\n  createStoryState,\r\n  createStructuredState,\r\n  initializeCharactersFromBible,\r\n  initializePlotThreadsFromBible,\r\n  updatePlotThread,\r\n  addRecentEvent,\r\n  addUnresolvedQuestion,\r\n  generateTensionGuidance,\r\n  analyzeTension,\r\n  type ChapterSummary,\r\n} from '../index.js';\r\n\r\nconsole.log('Testing Story Director Agent (Phase 6)...\\n');\r\n\r\n// Setup test story\r\nlet bible = createStoryBible(\r\n  '李白传奇',\r\n  '诗人寻找人生真谛',\r\n  'historical-fiction',\r\n  '唐朝长安及各地',\r\n  '豪放浪漫',\r\n  '诗人李白离开长安，游历天下，寻找人生真谛',\r\n  10\r\n);\r\n\r\nbible = addCharacter(bible, '李白', 'protagonist', ['豪放', '浪漫', '不羁'], ['寻找人生真谛', '成为伟大诗人']);\r\nbible = addCharacter(bible, '杜甫', 'supporting', ['沉郁', '忧国忧民'], ['记录时代变迁']);\r\nbible = addCharacter(bible, '孟浩然', 'supporting', ['隐逸', '淡泊'], ['归隐山林']);\r\n\r\nbible = addPlotThread(bible, '诗人之旅', '李白从长安出发，游历天下');\r\nbible = addPlotThread(bible, '宫廷召唤', '唐玄宗召李白入宫的阴谋');\r\nbible = addPlotThread(bible, '友情之路', '李白与杜甫、孟浩然的交往');\r\n\r\nconst storyState = createStoryState('test-story', 10);\r\nstoryState.currentChapter = 3;\r\n\r\nlet structuredState = createStructuredState('test-story');\r\nstructuredState = initializeCharactersFromBible(structuredState, bible);\r\nstructuredState = initializePlotThreadsFromBible(structuredState, bible);\r\n\r\n// Activate some plot threads\r\nconst threadIds = Object.keys(structuredState.plotThreads);\r\nif (threadIds[0]) {\r\n  structuredState = updatePlotThread(structuredState, threadIds[0], { status: 'active', tension: 0.6 }, 3);\r\n}\r\nif (threadIds[1]) {\r\n  structuredState = updatePlotThread(structuredState, threadIds[1], { status: 'escalating', tension: 0.4 }, 3);\r\n}\r\n\r\n// Add recent events\r\nstructuredState = addRecentEvent(structuredState, '李白告别长安');\r\nstructuredState = addRecentEvent(structuredState, '夜宿黄鹤楼');\r\nstructuredState = addRecentEvent(structuredState, '遇见孟浩然');\r\n\r\n// Add questions\r\nstructuredState = addUnresolvedQuestion(structuredState, '李白会接受宫廷召唤吗？');\r\nstructuredState = addUnresolvedQuestion(structuredState, '杜甫何时与李白相遇？');\r\n\r\n// Create tension guidance\r\nconst tensionAnalysis = analyzeTension(storyState, structuredState);\r\nconst tensionGuidance = generateTensionGuidance(tensionAnalysis, storyState);\r\n\r\n// Create previous summaries\r\nconst previousSummaries: ChapterSummary[] = [\r\n  {\r\n    chapterNumber: 1,\r\n    summary: '李白离开长安，踏上诗人之旅',\r\n    keyEvents: ['告别长安', '独自上路'],\r\n    characterChanges: { '李白': '从迷茫到坚定' },\r\n  },\r\n  {\r\n    chapterNumber: 2,\r\n    summary: '李白夜宿黄鹤楼，思考人生',\r\n    keyEvents: ['夜宿黄鹤楼', '写诗抒怀'],\r\n    characterChanges: { '李白': '更加坚定诗人之路' },\r\n  },\r\n  {\r\n    chapterNumber: 3,\r\n    summary: '李白遇见孟浩然，结为知己',\r\n    keyEvents: ['遇见孟浩然', '饮酒论诗'],\r\n    characterChanges: { '李白': '找到知音', '孟浩然': '欣赏李白才华' },\r\n  },\r\n];\r\n\r\n// Test 1: Fallback objectives\r\nconsole.log('Test 1: Fallback Objectives (No LLM)');\r\nconst fallbackOutput = storyDirector.generateFallbackObjectives(storyState, structuredState);\r\nconsole.log(`  Chapter: ${fallbackOutput.chapterNumber}`);\r\nconsole.log(`  Goal: ${fallbackOutput.overallGoal}`);\r\nconsole.log(`  Objectives: ${fallbackOutput.objectives.length}`);\r\nconsole.log(`  Focus: ${fallbackOutput.focusCharacters.join(', ')}`);\r\nconsole.log('✅ Fallback objectives generated');\r\n\r\n// Test 2: Format for prompt\r\nconsole.log('\\nTest 2: Format for Prompt');\r\nconst formatted = storyDirector.formatForPrompt(fallbackOutput);\r\nconsole.log('--- Formatted Output ---');\r\nconsole.log(formatted.substring(0, 600));\r\nconsole.log('...');\r\nconsole.log('✅ Formatted for writer prompt');\r\n\r\n// Test 3: Director with LLM\r\nconsole.log('\\nTest 3: Story Director with LLM');\r\n\r\nasync function runDirectorTest() {\r\n  try {\r\n    const directorOutput = await storyDirector.direct({\r\n      bible,\r\n      state: storyState,\r\n      structuredState,\r\n      tensionGuidance,\r\n      previousSummaries,\r\n    });\r\n    \r\n    console.log('✅ Director generated chapter plan');\r\n    console.log(`  Chapter: ${directorOutput.chapterNumber}`);\r\n    console.log(`  Goal: ${directorOutput.overallGoal}`);\r\n    console.log(`  Tone: ${directorOutput.tone}`);\r\n    console.log(`  Objectives: ${directorOutput.objectives.length}`);\r\n    \r\n    for (const obj of directorOutput.objectives) {\r\n      const emoji = { critical: '🔴', high: '🟠', medium: '🟡', low: '🟢' }[obj.priority];\r\n      console.log(`    ${emoji} [${obj.type}] ${obj.description}`);\r\n    }\r\n    \r\n    console.log(`  Focus Characters: ${directorOutput.focusCharacters.join(', ')}`);\r\n    console.log(`  Suggested Scenes: ${directorOutput.suggestedScenes.length}`);\r\n    \r\n    // Test 4: Format LLM output\r\n    console.log('\\nTest 4: Format LLM Output for Prompt');\r\n    const llmFormatted = storyDirector.formatForPrompt(directorOutput);\r\n    console.log('--- Formatted LLM Output ---');\r\n    console.log(llmFormatted.substring(0, 800));\r\n    console.log('...');\r\n    console.log('✅ LLM output formatted');\r\n    \r\n  } catch (error) {\r\n    console.log('⚠️ Director LLM test failed:', error instanceof Error ? error.message : String(error));\r\n  }\r\n}\r\n\r\nrunDirectorTest().then(() => {\r\n  // Test 5: Different story phases\r\n  console.log('\\nTest 5: Different Story Phases');\r\n\r\n  // Early phase\r\n  storyState.currentChapter = 1;\r\n  const earlyFallback = storyDirector.generateFallbackObjectives(storyState, structuredState);\r\n  console.log(`  Early (Ch 1): ${earlyFallback.objectives.length} objectives, tone: ${earlyFallback.tone}`);\r\n\r\n  // Middle phase\r\n  storyState.currentChapter = 5;\r\n  const middleFallback = storyDirector.generateFallbackObjectives(storyState, structuredState);\r\n  console.log(`  Middle (Ch 5): ${middleFallback.objectives.length} objectives, tone: ${middleFallback.tone}`);\r\n\r\n  // Late phase\r\n  storyState.currentChapter = 9;\r\n  const lateFallback = storyDirector.generateFallbackObjectives(storyState, structuredState);\r\n  console.log(`  Late (Ch 9): ${lateFallback.objectives.length} objectives, tone: ${lateFallback.tone}`);\r\n  console.log('✅ Objectives adapt to story phase');\r\n\r\n  console.log('\\n✅ All Story Director tests passed!');\r\n  console.log('\\n🎉 Phase 6 (Story Director Agent) tests complete!');\r\n});\r\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,144 @@
|
|
|
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
|
+
// Use deepseek-chat for JSON tasks (reasoner model has different output format)
|
|
12
|
+
process.env.LLM_MODEL = 'deepseek-chat';
|
|
13
|
+
if (config.provider === 'openai') {
|
|
14
|
+
process.env.OPENAI_API_KEY = config.apiKey;
|
|
15
|
+
}
|
|
16
|
+
else if (config.provider === 'deepseek') {
|
|
17
|
+
process.env.DEEPSEEK_API_KEY = config.apiKey;
|
|
18
|
+
}
|
|
19
|
+
console.log(`Loaded config: ${config.provider} / deepseek-chat (using chat for JSON tasks)`);
|
|
20
|
+
}
|
|
21
|
+
const index_js_1 = require("../index.js");
|
|
22
|
+
console.log('Testing Structured Story State (Phase 4)...\n');
|
|
23
|
+
// Test 1: Create structured state
|
|
24
|
+
console.log('Test 1: Create Structured State');
|
|
25
|
+
const state = (0, index_js_1.createStructuredState)('test-story');
|
|
26
|
+
console.log('✅ Structured state created');
|
|
27
|
+
console.log(` - Chapter: ${state.chapter}`);
|
|
28
|
+
console.log(` - Tension: ${state.tension}`);
|
|
29
|
+
// Test 2: Create story bible and initialize
|
|
30
|
+
console.log('\nTest 2: Initialize from StoryBible');
|
|
31
|
+
let bible = (0, index_js_1.createStoryBible)('李白传奇', '诗人寻找人生真谛', 'historical-fiction', '唐朝长安及各地', '豪放浪漫', '诗人李白离开长安,游历天下,寻找人生真谛', 10);
|
|
32
|
+
bible = (0, index_js_1.addCharacter)(bible, '李白', 'protagonist', ['豪放', '浪漫', '不羁'], ['寻找人生真谛', '成为伟大诗人']);
|
|
33
|
+
bible = (0, index_js_1.addCharacter)(bible, '杜甫', 'supporting', ['沉郁', '忧国忧民'], ['记录时代变迁']);
|
|
34
|
+
bible = (0, index_js_1.addPlotThread)(bible, '诗人之旅', '李白从长安出发,游历天下,寻找人生真谛');
|
|
35
|
+
bible = (0, index_js_1.addPlotThread)(bible, '宫廷阴谋', '朝堂上的权力斗争');
|
|
36
|
+
let structuredState = (0, index_js_1.initializeCharactersFromBible)(state, bible);
|
|
37
|
+
structuredState = (0, index_js_1.initializePlotThreadsFromBible)(structuredState, bible);
|
|
38
|
+
console.log('✅ Initialized from bible');
|
|
39
|
+
console.log(` - Characters: ${Object.keys(structuredState.characters).length}`);
|
|
40
|
+
console.log(` - Plot threads: ${Object.keys(structuredState.plotThreads).length}`);
|
|
41
|
+
// Test 3: Update character state
|
|
42
|
+
console.log('\nTest 3: Update Character State');
|
|
43
|
+
structuredState = (0, index_js_1.updateCharacterState)(structuredState, '李白', {
|
|
44
|
+
emotionalState: 'melancholic',
|
|
45
|
+
location: '黄鹤楼',
|
|
46
|
+
goals: ['寻找人生真谛', '成为伟大诗人', '与友人告别'],
|
|
47
|
+
});
|
|
48
|
+
console.log('✅ Character state updated');
|
|
49
|
+
console.log(` - 李白 emotion: ${structuredState.characters['李白'].emotionalState}`);
|
|
50
|
+
console.log(` - 李白 location: ${structuredState.characters['李白'].location}`);
|
|
51
|
+
// Test 4: Update plot thread
|
|
52
|
+
console.log('\nTest 4: Update Plot Thread');
|
|
53
|
+
const threadIds = Object.keys(structuredState.plotThreads);
|
|
54
|
+
const firstThreadId = threadIds[0];
|
|
55
|
+
structuredState = (0, index_js_1.updatePlotThread)(structuredState, firstThreadId, {
|
|
56
|
+
status: 'escalating',
|
|
57
|
+
tension: 0.5,
|
|
58
|
+
}, 3);
|
|
59
|
+
console.log('✅ Plot thread updated');
|
|
60
|
+
console.log(` - Thread status: ${structuredState.plotThreads[firstThreadId].status}`);
|
|
61
|
+
console.log(` - Thread tension: ${structuredState.plotThreads[firstThreadId].tension}`);
|
|
62
|
+
// Test 5: Questions
|
|
63
|
+
console.log('\nTest 5: Unresolved Questions');
|
|
64
|
+
structuredState = (0, index_js_1.addUnresolvedQuestion)(structuredState, '李白会去哪里?');
|
|
65
|
+
structuredState = (0, index_js_1.addUnresolvedQuestion)(structuredState, '杜甫何时出现?');
|
|
66
|
+
console.log('✅ Added questions');
|
|
67
|
+
console.log(` - Questions: ${structuredState.unresolvedQuestions.length}`);
|
|
68
|
+
structuredState = (0, index_js_1.resolveQuestion)(structuredState, '李白会去哪里?');
|
|
69
|
+
console.log('✅ Resolved question');
|
|
70
|
+
console.log(` - Remaining: ${structuredState.unresolvedQuestions.length}`);
|
|
71
|
+
// Test 6: Recent events
|
|
72
|
+
console.log('\nTest 6: Recent Events');
|
|
73
|
+
structuredState = (0, index_js_1.addRecentEvent)(structuredState, '李白告别长安');
|
|
74
|
+
structuredState = (0, index_js_1.addRecentEvent)(structuredState, '夜宿黄鹤楼');
|
|
75
|
+
structuredState = (0, index_js_1.addRecentEvent)(structuredState, '遇见孟浩然');
|
|
76
|
+
console.log('✅ Added recent events');
|
|
77
|
+
console.log(` - Events: ${structuredState.recentEvents.length}`);
|
|
78
|
+
// Test 7: Tension calculation
|
|
79
|
+
console.log('\nTest 7: Tension Calculation');
|
|
80
|
+
const targetTension = (0, index_js_1.calculateTargetTension)(5, 10);
|
|
81
|
+
console.log(`✅ Target tension at chapter 5/10: ${targetTension}`);
|
|
82
|
+
structuredState = (0, index_js_1.updateStoryTension)(structuredState, 5, 10);
|
|
83
|
+
console.log(`✅ Updated story tension: ${structuredState.tension}`);
|
|
84
|
+
// Test 8: Format for prompt
|
|
85
|
+
console.log('\nTest 8: Format for Prompt');
|
|
86
|
+
const formatted = (0, index_js_1.formatStructuredStateForPrompt)(structuredState);
|
|
87
|
+
console.log('✅ Formatted state for prompt');
|
|
88
|
+
console.log('\n--- Formatted Output Preview ---');
|
|
89
|
+
console.log(formatted.substring(0, 500) + '...');
|
|
90
|
+
// Test 9: State Updater Agent
|
|
91
|
+
console.log('\nTest 9: State Updater Agent');
|
|
92
|
+
const testChapter = {
|
|
93
|
+
id: 'test-chapter-3',
|
|
94
|
+
storyId: 'test-story',
|
|
95
|
+
number: 3,
|
|
96
|
+
title: '黄鹤楼送别',
|
|
97
|
+
content: `李白站在黄鹤楼上,望着滚滚东去的长江水。暮色四合,江面上泛起层层涟漪。
|
|
98
|
+
|
|
99
|
+
"孟兄,此去经年,不知何时再见。"李白举杯,声音中带着几分惆怅。
|
|
100
|
+
|
|
101
|
+
孟浩然拍了拍他的肩膀:"太白兄,天下无不散之筵席。你我有缘,自会重逢。"
|
|
102
|
+
|
|
103
|
+
李白将杯中酒一饮而尽。他望着远方,心中涌起无限感慨。长安的繁华已成过往,前路漫漫,不知归处。
|
|
104
|
+
|
|
105
|
+
"我李白一生,求的是真性情,写的是真文章。"他喃喃自语,"这天下,这江山,都将成为我的诗篇。"
|
|
106
|
+
|
|
107
|
+
夜色渐深,孟浩然的船已远去。李白独立楼头,江风吹动他的衣袂。他知道,从此刻起,他真正踏上了属于自己的旅程。
|
|
108
|
+
|
|
109
|
+
远处传来渔歌,悠扬婉转。李白提笔,在楼壁上写下:故人西辞黄鹤楼,烟花三月下扬州。孤帆远影碧空尽,唯见长江天际流。`,
|
|
110
|
+
wordCount: 350,
|
|
111
|
+
summary: '李白在黄鹤楼与孟浩然告别,写下著名的送别诗,正式踏上诗人之旅',
|
|
112
|
+
generatedAt: new Date(),
|
|
113
|
+
};
|
|
114
|
+
// Run async test
|
|
115
|
+
async function runStateUpdaterTest() {
|
|
116
|
+
try {
|
|
117
|
+
const updates = await index_js_1.stateUpdater.extractStateChanges(testChapter, bible, structuredState);
|
|
118
|
+
console.log('✅ Extracted state changes');
|
|
119
|
+
console.log(` - Character updates: ${updates.characterUpdates.length}`);
|
|
120
|
+
console.log(` - Plot thread updates: ${updates.plotThreadUpdates.length}`);
|
|
121
|
+
console.log(` - New questions: ${updates.newQuestions.length}`);
|
|
122
|
+
console.log(` - Recent events: ${updates.recentEvents.length}`);
|
|
123
|
+
// Apply updates
|
|
124
|
+
const updatedState = index_js_1.stateUpdater.applyUpdates(structuredState, updates, 3);
|
|
125
|
+
console.log('✅ Applied state updates');
|
|
126
|
+
console.log(` - 李白 emotion: ${updatedState.characters['李白'].emotionalState}`);
|
|
127
|
+
console.log(` - Recent events count: ${updatedState.recentEvents.length}`);
|
|
128
|
+
}
|
|
129
|
+
catch (error) {
|
|
130
|
+
console.log('⚠️ State updater test failed:', error instanceof Error ? error.message : String(error));
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
runStateUpdaterTest().then(() => {
|
|
134
|
+
// Test 10: Serialization
|
|
135
|
+
console.log('\nTest 10: Serialization');
|
|
136
|
+
const serialized = JSON.stringify(structuredState);
|
|
137
|
+
const deserialized = JSON.parse(serialized);
|
|
138
|
+
console.log('✅ Serialization round-trip successful');
|
|
139
|
+
console.log(` - Original characters: ${Object.keys(structuredState.characters).length}`);
|
|
140
|
+
console.log(` - Deserialized characters: ${Object.keys(deserialized.characters).length}`);
|
|
141
|
+
console.log('\n✅ All Structured State tests passed!');
|
|
142
|
+
console.log('\n🎉 Phase 4 (Structured Story State) tests complete!');
|
|
143
|
+
});
|
|
144
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"structured-state.test.js","sourceRoot":"","sources":["../../src/test/structured-state.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  // Use deepseek-chat for JSON tasks (reasoner model has different output format)\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} / deepseek-chat (using chat for JSON tasks)`);\r\n}\r\n\r\nimport {\r\n  createStoryBible,\r\n  addCharacter,\r\n  addPlotThread,\r\n  createStructuredState,\r\n  initializeCharactersFromBible,\r\n  initializePlotThreadsFromBible,\r\n  updateCharacterState,\r\n  updatePlotThread,\r\n  addUnresolvedQuestion,\r\n  resolveQuestion,\r\n  addRecentEvent,\r\n  calculateTargetTension,\r\n  updateStoryTension,\r\n  formatStructuredStateForPrompt,\r\n  stateUpdater,\r\n} from '../index.js';\r\n\r\nconsole.log('Testing Structured Story State (Phase 4)...\\n');\r\n\r\n// Test 1: Create structured state\r\nconsole.log('Test 1: Create Structured State');\r\nconst state = createStructuredState('test-story');\r\nconsole.log('✅ Structured state created');\r\nconsole.log(`  - Chapter: ${state.chapter}`);\r\nconsole.log(`  - Tension: ${state.tension}`);\r\n\r\n// Test 2: Create story bible and initialize\r\nconsole.log('\\nTest 2: Initialize from StoryBible');\r\nlet bible = createStoryBible(\r\n  '李白传奇',\r\n  '诗人寻找人生真谛',\r\n  'historical-fiction',\r\n  '唐朝长安及各地',\r\n  '豪放浪漫',\r\n  '诗人李白离开长安，游历天下，寻找人生真谛',\r\n  10\r\n);\r\nbible = addCharacter(\r\n  bible,\r\n  '李白',\r\n  'protagonist',\r\n  ['豪放', '浪漫', '不羁'],\r\n  ['寻找人生真谛', '成为伟大诗人']\r\n);\r\nbible = addCharacter(\r\n  bible,\r\n  '杜甫',\r\n  'supporting',\r\n  ['沉郁', '忧国忧民'],\r\n  ['记录时代变迁']\r\n);\r\nbible = addPlotThread(\r\n  bible,\r\n  '诗人之旅',\r\n  '李白从长安出发，游历天下，寻找人生真谛'\r\n);\r\nbible = addPlotThread(\r\n  bible,\r\n  '宫廷阴谋',\r\n  '朝堂上的权力斗争'\r\n);\r\n\r\nlet structuredState = initializeCharactersFromBible(state, bible);\r\nstructuredState = initializePlotThreadsFromBible(structuredState, bible);\r\nconsole.log('✅ Initialized from bible');\r\nconsole.log(`  - Characters: ${Object.keys(structuredState.characters).length}`);\r\nconsole.log(`  - Plot threads: ${Object.keys(structuredState.plotThreads).length}`);\r\n\r\n// Test 3: Update character state\r\nconsole.log('\\nTest 3: Update Character State');\r\nstructuredState = updateCharacterState(structuredState, '李白', {\r\n  emotionalState: 'melancholic',\r\n  location: '黄鹤楼',\r\n  goals: ['寻找人生真谛', '成为伟大诗人', '与友人告别'],\r\n});\r\nconsole.log('✅ Character state updated');\r\nconsole.log(`  - 李白 emotion: ${structuredState.characters['李白'].emotionalState}`);\r\nconsole.log(`  - 李白 location: ${structuredState.characters['李白'].location}`);\r\n\r\n// Test 4: Update plot thread\r\nconsole.log('\\nTest 4: Update Plot Thread');\r\nconst threadIds = Object.keys(structuredState.plotThreads);\r\nconst firstThreadId = threadIds[0];\r\nstructuredState = updatePlotThread(structuredState, firstThreadId, {\r\n  status: 'escalating',\r\n  tension: 0.5,\r\n}, 3);\r\nconsole.log('✅ Plot thread updated');\r\nconsole.log(`  - Thread status: ${structuredState.plotThreads[firstThreadId].status}`);\r\nconsole.log(`  - Thread tension: ${structuredState.plotThreads[firstThreadId].tension}`);\r\n\r\n// Test 5: Questions\r\nconsole.log('\\nTest 5: Unresolved Questions');\r\nstructuredState = addUnresolvedQuestion(structuredState, '李白会去哪里？');\r\nstructuredState = addUnresolvedQuestion(structuredState, '杜甫何时出现？');\r\nconsole.log('✅ Added questions');\r\nconsole.log(`  - Questions: ${structuredState.unresolvedQuestions.length}`);\r\n\r\nstructuredState = resolveQuestion(structuredState, '李白会去哪里？');\r\nconsole.log('✅ Resolved question');\r\nconsole.log(`  - Remaining: ${structuredState.unresolvedQuestions.length}`);\r\n\r\n// Test 6: Recent events\r\nconsole.log('\\nTest 6: Recent Events');\r\nstructuredState = addRecentEvent(structuredState, '李白告别长安');\r\nstructuredState = addRecentEvent(structuredState, '夜宿黄鹤楼');\r\nstructuredState = addRecentEvent(structuredState, '遇见孟浩然');\r\nconsole.log('✅ Added recent events');\r\nconsole.log(`  - Events: ${structuredState.recentEvents.length}`);\r\n\r\n// Test 7: Tension calculation\r\nconsole.log('\\nTest 7: Tension Calculation');\r\nconst targetTension = calculateTargetTension(5, 10);\r\nconsole.log(`✅ Target tension at chapter 5/10: ${targetTension}`);\r\n\r\nstructuredState = updateStoryTension(structuredState, 5, 10);\r\nconsole.log(`✅ Updated story tension: ${structuredState.tension}`);\r\n\r\n// Test 8: Format for prompt\r\nconsole.log('\\nTest 8: Format for Prompt');\r\nconst formatted = formatStructuredStateForPrompt(structuredState);\r\nconsole.log('✅ Formatted state for prompt');\r\nconsole.log('\\n--- Formatted Output Preview ---');\r\nconsole.log(formatted.substring(0, 500) + '...');\r\n\r\n// Test 9: State Updater Agent\r\nconsole.log('\\nTest 9: State Updater Agent');\r\nconst testChapter = {\r\n  id: 'test-chapter-3',\r\n  storyId: 'test-story',\r\n  number: 3,\r\n  title: '黄鹤楼送别',\r\n  content: `李白站在黄鹤楼上，望着滚滚东去的长江水。暮色四合，江面上泛起层层涟漪。\r\n\r\n\"孟兄，此去经年，不知何时再见。\"李白举杯，声音中带着几分惆怅。\r\n\r\n孟浩然拍了拍他的肩膀：\"太白兄，天下无不散之筵席。你我有缘，自会重逢。\"\r\n\r\n李白将杯中酒一饮而尽。他望着远方，心中涌起无限感慨。长安的繁华已成过往，前路漫漫，不知归处。\r\n\r\n\"我李白一生，求的是真性情，写的是真文章。\"他喃喃自语，\"这天下，这江山，都将成为我的诗篇。\"\r\n\r\n夜色渐深，孟浩然的船已远去。李白独立楼头，江风吹动他的衣袂。他知道，从此刻起，他真正踏上了属于自己的旅程。\r\n\r\n远处传来渔歌，悠扬婉转。李白提笔，在楼壁上写下：故人西辞黄鹤楼，烟花三月下扬州。孤帆远影碧空尽，唯见长江天际流。`,\r\n  wordCount: 350,\r\n  summary: '李白在黄鹤楼与孟浩然告别，写下著名的送别诗，正式踏上诗人之旅',\r\n  generatedAt: new Date(),\r\n};\r\n\r\n// Run async test\r\nasync function runStateUpdaterTest() {\r\n  try {\r\n    const updates = await stateUpdater.extractStateChanges(testChapter, bible, structuredState);\r\n    console.log('✅ Extracted state changes');\r\n    console.log(`  - Character updates: ${updates.characterUpdates.length}`);\r\n    console.log(`  - Plot thread updates: ${updates.plotThreadUpdates.length}`);\r\n    console.log(`  - New questions: ${updates.newQuestions.length}`);\r\n    console.log(`  - Recent events: ${updates.recentEvents.length}`);\r\n    \r\n    // Apply updates\r\n    const updatedState = stateUpdater.applyUpdates(structuredState, updates, 3);\r\n    console.log('✅ Applied state updates');\r\n    console.log(`  - 李白 emotion: ${updatedState.characters['李白'].emotionalState}`);\r\n    console.log(`  - Recent events count: ${updatedState.recentEvents.length}`);\r\n  } catch (error) {\r\n    console.log('⚠️ State updater test failed:', error instanceof Error ? error.message : String(error));\r\n  }\r\n}\r\n\r\nrunStateUpdaterTest().then(() => {\r\n  // Test 10: Serialization\r\n  console.log('\\nTest 10: Serialization');\r\n  const serialized = JSON.stringify(structuredState);\r\n  const deserialized = JSON.parse(serialized);\r\n  console.log('✅ Serialization round-trip successful');\r\n  console.log(`  - Original characters: ${Object.keys(structuredState.characters).length}`);\r\n  console.log(`  - Deserialized characters: ${Object.keys(deserialized.characters).length}`);\r\n\r\n  console.log('\\n✅ All Structured State tests passed!');\r\n  console.log('\\n🎉 Phase 4 (Structured Story State) tests complete!');\r\n});\r\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|