@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,73 @@
|
|
|
1
|
+
import type { CharacterState } from '../story/structuredState.js';
|
|
2
|
+
export interface CharacterAgent {
|
|
3
|
+
name: string;
|
|
4
|
+
goals: string[];
|
|
5
|
+
currentGoal: string;
|
|
6
|
+
location: string;
|
|
7
|
+
knowledge: string[];
|
|
8
|
+
relationships: Record<string, string>;
|
|
9
|
+
personality: string[];
|
|
10
|
+
emotionalState: string;
|
|
11
|
+
inventory: string[];
|
|
12
|
+
agenda: AgendaItem[];
|
|
13
|
+
}
|
|
14
|
+
export interface AgendaItem {
|
|
15
|
+
id: string;
|
|
16
|
+
action: string;
|
|
17
|
+
priority: number;
|
|
18
|
+
deadline?: number;
|
|
19
|
+
completed: boolean;
|
|
20
|
+
}
|
|
21
|
+
export interface CharacterDecision {
|
|
22
|
+
character: string;
|
|
23
|
+
action: string;
|
|
24
|
+
target?: string;
|
|
25
|
+
reasoning: string;
|
|
26
|
+
consequences: string[];
|
|
27
|
+
}
|
|
28
|
+
export interface CharacterAgentContext {
|
|
29
|
+
character: CharacterAgent;
|
|
30
|
+
otherCharacters: CharacterAgent[];
|
|
31
|
+
worldEvents: string[];
|
|
32
|
+
currentChapter: number;
|
|
33
|
+
storyContext: string;
|
|
34
|
+
}
|
|
35
|
+
export declare class CharacterAgentSystem {
|
|
36
|
+
/**
|
|
37
|
+
* Create a character agent from structured state
|
|
38
|
+
*/
|
|
39
|
+
createAgent(characterState: CharacterState, personality: string[]): CharacterAgent;
|
|
40
|
+
/**
|
|
41
|
+
* Add an agenda item for a character
|
|
42
|
+
*/
|
|
43
|
+
addAgendaItem(agent: CharacterAgent, action: string, priority?: number, deadline?: number): CharacterAgent;
|
|
44
|
+
/**
|
|
45
|
+
* Complete an agenda item
|
|
46
|
+
*/
|
|
47
|
+
completeAgendaItem(agent: CharacterAgent, agendaId: string): CharacterAgent;
|
|
48
|
+
/**
|
|
49
|
+
* Update character knowledge
|
|
50
|
+
*/
|
|
51
|
+
addKnowledge(agent: CharacterAgent, fact: string): CharacterAgent;
|
|
52
|
+
/**
|
|
53
|
+
* Update relationship
|
|
54
|
+
*/
|
|
55
|
+
updateRelationship(agent: CharacterAgent, otherCharacter: string, relationship: string): CharacterAgent;
|
|
56
|
+
/**
|
|
57
|
+
* Move character to new location
|
|
58
|
+
*/
|
|
59
|
+
moveTo(agent: CharacterAgent, location: string): CharacterAgent;
|
|
60
|
+
/**
|
|
61
|
+
* Get character decision using LLM
|
|
62
|
+
*/
|
|
63
|
+
getDecision(context: CharacterAgentContext): Promise<CharacterDecision>;
|
|
64
|
+
/**
|
|
65
|
+
* Get simple decision without LLM (for testing/fallback)
|
|
66
|
+
*/
|
|
67
|
+
getSimpleDecision(context: CharacterAgentContext): CharacterDecision;
|
|
68
|
+
/**
|
|
69
|
+
* Simulate multiple characters making decisions
|
|
70
|
+
*/
|
|
71
|
+
simulateTurn(agents: CharacterAgent[], worldEvents: string[], currentChapter: number, storyContext: string, useLLM?: boolean): Promise<CharacterDecision[]>;
|
|
72
|
+
}
|
|
73
|
+
export declare const characterAgentSystem: CharacterAgentSystem;
|
|
@@ -0,0 +1,232 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.characterAgentSystem = exports.CharacterAgentSystem = void 0;
|
|
4
|
+
const client_js_1 = require("../llm/client.js");
|
|
5
|
+
const CHARACTER_DECISION_PROMPT = `You are simulating a character in a narrative. Decide what this character would do next based on their personality, goals, and current situation.
|
|
6
|
+
|
|
7
|
+
## Character
|
|
8
|
+
|
|
9
|
+
**Name:** {{name}}
|
|
10
|
+
**Personality:** {{personality}}
|
|
11
|
+
**Current Emotional State:** {{emotionalState}}
|
|
12
|
+
**Location:** {{location}}
|
|
13
|
+
|
|
14
|
+
### Goals
|
|
15
|
+
{{goals}}
|
|
16
|
+
|
|
17
|
+
### Current Agenda
|
|
18
|
+
{{agenda}}
|
|
19
|
+
|
|
20
|
+
### Knowledge
|
|
21
|
+
{{knowledge}}
|
|
22
|
+
|
|
23
|
+
### Relationships
|
|
24
|
+
{{relationships}}
|
|
25
|
+
|
|
26
|
+
## Current Situation
|
|
27
|
+
|
|
28
|
+
**Chapter:** {{chapter}}
|
|
29
|
+
**Story Context:** {{storyContext}}
|
|
30
|
+
|
|
31
|
+
### Other Characters Present
|
|
32
|
+
{{otherCharacters}}
|
|
33
|
+
|
|
34
|
+
### Recent World Events
|
|
35
|
+
{{worldEvents}}
|
|
36
|
+
|
|
37
|
+
## Your Task
|
|
38
|
+
|
|
39
|
+
Decide what {{name}} would do next. Consider:
|
|
40
|
+
1. Their personality and emotional state
|
|
41
|
+
2. Their current goals and agenda
|
|
42
|
+
3. Their relationships with others
|
|
43
|
+
4. Recent events that might affect their decision
|
|
44
|
+
5. What would create interesting dramatic possibilities
|
|
45
|
+
|
|
46
|
+
Output JSON:
|
|
47
|
+
{
|
|
48
|
+
"character": "{{name}}",
|
|
49
|
+
"action": "What they decide to do (be specific and actionable)",
|
|
50
|
+
"target": "Who or what they act toward (if anyone)",
|
|
51
|
+
"reasoning": "Why they make this decision based on their character",
|
|
52
|
+
"consequences": ["Possible outcomes or reactions from this action"]
|
|
53
|
+
}`;
|
|
54
|
+
class CharacterAgentSystem {
|
|
55
|
+
/**
|
|
56
|
+
* Create a character agent from structured state
|
|
57
|
+
*/
|
|
58
|
+
createAgent(characterState, personality) {
|
|
59
|
+
return {
|
|
60
|
+
name: characterState.name,
|
|
61
|
+
goals: [...characterState.goals],
|
|
62
|
+
currentGoal: characterState.goals[0] || 'unknown',
|
|
63
|
+
location: characterState.location,
|
|
64
|
+
knowledge: [...characterState.knowledge],
|
|
65
|
+
relationships: { ...characterState.relationships },
|
|
66
|
+
personality: [...personality],
|
|
67
|
+
emotionalState: characterState.emotionalState,
|
|
68
|
+
inventory: [],
|
|
69
|
+
agenda: [],
|
|
70
|
+
};
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Add an agenda item for a character
|
|
74
|
+
*/
|
|
75
|
+
addAgendaItem(agent, action, priority = 5, deadline) {
|
|
76
|
+
return {
|
|
77
|
+
...agent,
|
|
78
|
+
agenda: [
|
|
79
|
+
...agent.agenda,
|
|
80
|
+
{
|
|
81
|
+
id: `agenda-${Date.now()}-${Math.random().toString(36).substring(2, 9)}`,
|
|
82
|
+
action,
|
|
83
|
+
priority,
|
|
84
|
+
deadline,
|
|
85
|
+
completed: false,
|
|
86
|
+
},
|
|
87
|
+
],
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* Complete an agenda item
|
|
92
|
+
*/
|
|
93
|
+
completeAgendaItem(agent, agendaId) {
|
|
94
|
+
return {
|
|
95
|
+
...agent,
|
|
96
|
+
agenda: agent.agenda.map(item => item.id === agendaId ? { ...item, completed: true } : item),
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Update character knowledge
|
|
101
|
+
*/
|
|
102
|
+
addKnowledge(agent, fact) {
|
|
103
|
+
if (agent.knowledge.includes(fact))
|
|
104
|
+
return agent;
|
|
105
|
+
return {
|
|
106
|
+
...agent,
|
|
107
|
+
knowledge: [...agent.knowledge, fact],
|
|
108
|
+
};
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Update relationship
|
|
112
|
+
*/
|
|
113
|
+
updateRelationship(agent, otherCharacter, relationship) {
|
|
114
|
+
return {
|
|
115
|
+
...agent,
|
|
116
|
+
relationships: {
|
|
117
|
+
...agent.relationships,
|
|
118
|
+
[otherCharacter]: relationship,
|
|
119
|
+
},
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Move character to new location
|
|
124
|
+
*/
|
|
125
|
+
moveTo(agent, location) {
|
|
126
|
+
return {
|
|
127
|
+
...agent,
|
|
128
|
+
location,
|
|
129
|
+
};
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Get character decision using LLM
|
|
133
|
+
*/
|
|
134
|
+
async getDecision(context) {
|
|
135
|
+
const { character, otherCharacters, worldEvents, currentChapter, storyContext } = context;
|
|
136
|
+
const prompt = CHARACTER_DECISION_PROMPT
|
|
137
|
+
.replace(/{{name}}/g, character.name)
|
|
138
|
+
.replace('{{personality}}', character.personality.join(', '))
|
|
139
|
+
.replace('{{emotionalState}}', character.emotionalState)
|
|
140
|
+
.replace('{{location}}', character.location)
|
|
141
|
+
.replace('{{goals}}', character.goals.map(g => `- ${g}`).join('\n'))
|
|
142
|
+
.replace('{{agenda}}', character.agenda.filter(a => !a.completed).map(a => `- ${a.action} (priority: ${a.priority})`).join('\n') || 'No active agenda')
|
|
143
|
+
.replace('{{knowledge}}', character.knowledge.map(k => `- ${k}`).join('\n') || 'No special knowledge')
|
|
144
|
+
.replace('{{relationships}}', Object.entries(character.relationships).map(([name, rel]) => `- ${name}: ${rel}`).join('\n') || 'No significant relationships')
|
|
145
|
+
.replace('{{chapter}}', currentChapter.toString())
|
|
146
|
+
.replace('{{storyContext}}', storyContext)
|
|
147
|
+
.replace('{{otherCharacters}}', otherCharacters.map(c => `- ${c.name} (${c.emotionalState}, at ${c.location})`).join('\n') || 'No other characters present')
|
|
148
|
+
.replace('{{worldEvents}}', worldEvents.map(e => `- ${e}`).join('\n') || 'No recent events');
|
|
149
|
+
const result = await (0, client_js_1.getLLM)().completeJSON(prompt, {
|
|
150
|
+
temperature: 0.5,
|
|
151
|
+
maxTokens: 1000,
|
|
152
|
+
});
|
|
153
|
+
return result;
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Get simple decision without LLM (for testing/fallback)
|
|
157
|
+
*/
|
|
158
|
+
getSimpleDecision(context) {
|
|
159
|
+
const { character, otherCharacters } = context;
|
|
160
|
+
// Find incomplete agenda items
|
|
161
|
+
const activeAgenda = character.agenda.filter(a => !a.completed);
|
|
162
|
+
if (activeAgenda.length > 0) {
|
|
163
|
+
// Follow highest priority agenda item
|
|
164
|
+
const topPriority = activeAgenda.sort((a, b) => b.priority - a.priority)[0];
|
|
165
|
+
return {
|
|
166
|
+
character: character.name,
|
|
167
|
+
action: topPriority.action,
|
|
168
|
+
reasoning: `Following their agenda: ${topPriority.action}`,
|
|
169
|
+
consequences: ['Progress toward goal'],
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
// React to other characters if present
|
|
173
|
+
if (otherCharacters.length > 0) {
|
|
174
|
+
const other = otherCharacters[0];
|
|
175
|
+
const relationship = character.relationships[other.name] || 'neutral';
|
|
176
|
+
if (relationship.includes('friend') || relationship.includes('ally')) {
|
|
177
|
+
return {
|
|
178
|
+
character: character.name,
|
|
179
|
+
action: `Approach ${other.name} to talk`,
|
|
180
|
+
target: other.name,
|
|
181
|
+
reasoning: `They are ${relationship} and nearby`,
|
|
182
|
+
consequences: ['Social interaction', 'Information exchange'],
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
if (relationship.includes('enemy') || relationship.includes('hostile')) {
|
|
186
|
+
return {
|
|
187
|
+
character: character.name,
|
|
188
|
+
action: `Keep distance from ${other.name}`,
|
|
189
|
+
target: other.name,
|
|
190
|
+
reasoning: `They are ${relationship}`,
|
|
191
|
+
consequences: ['Avoiding conflict', 'Maintaining safety'],
|
|
192
|
+
};
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
// Default: pursue current goal
|
|
196
|
+
return {
|
|
197
|
+
character: character.name,
|
|
198
|
+
action: `Work toward goal: ${character.currentGoal}`,
|
|
199
|
+
reasoning: 'No immediate distractions, focusing on primary objective',
|
|
200
|
+
consequences: ['Progress toward goal', 'Possible new opportunities'],
|
|
201
|
+
};
|
|
202
|
+
}
|
|
203
|
+
/**
|
|
204
|
+
* Simulate multiple characters making decisions
|
|
205
|
+
*/
|
|
206
|
+
async simulateTurn(agents, worldEvents, currentChapter, storyContext, useLLM = false) {
|
|
207
|
+
const decisions = [];
|
|
208
|
+
for (const agent of agents) {
|
|
209
|
+
const context = {
|
|
210
|
+
character: agent,
|
|
211
|
+
otherCharacters: agents.filter(a => a.name !== agent.name),
|
|
212
|
+
worldEvents,
|
|
213
|
+
currentChapter,
|
|
214
|
+
storyContext,
|
|
215
|
+
};
|
|
216
|
+
try {
|
|
217
|
+
const decision = useLLM
|
|
218
|
+
? await this.getDecision(context)
|
|
219
|
+
: this.getSimpleDecision(context);
|
|
220
|
+
decisions.push(decision);
|
|
221
|
+
}
|
|
222
|
+
catch (error) {
|
|
223
|
+
// Fallback to simple decision on error
|
|
224
|
+
decisions.push(this.getSimpleDecision(context));
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
return decisions;
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
exports.CharacterAgentSystem = CharacterAgentSystem;
|
|
231
|
+
exports.characterAgentSystem = new CharacterAgentSystem();
|
|
232
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"characterAgent.js","sourceRoot":"","sources":["../../src/world/characterAgent.ts"],"names":[],"mappings":"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","sourcesContent":["import { getLLM } from '../llm/client.js';\r\nimport type { CharacterState } from '../story/structuredState.js';\r\n\r\nexport interface CharacterAgent {\r\n  name: string;\r\n  goals: string[];\r\n  currentGoal: string;\r\n  location: string;\r\n  knowledge: string[];\r\n  relationships: Record<string, string>;\r\n  personality: string[];\r\n  emotionalState: string;\r\n  inventory: string[];\r\n  agenda: AgendaItem[];\r\n}\r\n\r\nexport interface AgendaItem {\r\n  id: string;\r\n  action: string;\r\n  priority: number;\r\n  deadline?: number; // chapter number\r\n  completed: boolean;\r\n}\r\n\r\nexport interface CharacterDecision {\r\n  character: string;\r\n  action: string;\r\n  target?: string;\r\n  reasoning: string;\r\n  consequences: string[];\r\n}\r\n\r\nexport interface CharacterAgentContext {\r\n  character: CharacterAgent;\r\n  otherCharacters: CharacterAgent[];\r\n  worldEvents: string[];\r\n  currentChapter: number;\r\n  storyContext: string;\r\n}\r\n\r\nconst CHARACTER_DECISION_PROMPT = `You are simulating a character in a narrative. Decide what this character would do next based on their personality, goals, and current situation.\r\n\r\n## Character\r\n\r\n**Name:** {{name}}\r\n**Personality:** {{personality}}\r\n**Current Emotional State:** {{emotionalState}}\r\n**Location:** {{location}}\r\n\r\n### Goals\r\n{{goals}}\r\n\r\n### Current Agenda\r\n{{agenda}}\r\n\r\n### Knowledge\r\n{{knowledge}}\r\n\r\n### Relationships\r\n{{relationships}}\r\n\r\n## Current Situation\r\n\r\n**Chapter:** {{chapter}}\r\n**Story Context:** {{storyContext}}\r\n\r\n### Other Characters Present\r\n{{otherCharacters}}\r\n\r\n### Recent World Events\r\n{{worldEvents}}\r\n\r\n## Your Task\r\n\r\nDecide what {{name}} would do next. Consider:\r\n1. Their personality and emotional state\r\n2. Their current goals and agenda\r\n3. Their relationships with others\r\n4. Recent events that might affect their decision\r\n5. What would create interesting dramatic possibilities\r\n\r\nOutput JSON:\r\n{\r\n  \"character\": \"{{name}}\",\r\n  \"action\": \"What they decide to do (be specific and actionable)\",\r\n  \"target\": \"Who or what they act toward (if anyone)\",\r\n  \"reasoning\": \"Why they make this decision based on their character\",\r\n  \"consequences\": [\"Possible outcomes or reactions from this action\"]\r\n}`;\r\n\r\nexport class CharacterAgentSystem {\r\n  /**\r\n   * Create a character agent from structured state\r\n   */\r\n  createAgent(characterState: CharacterState, personality: string[]): CharacterAgent {\r\n    return {\r\n      name: characterState.name,\r\n      goals: [...characterState.goals],\r\n      currentGoal: characterState.goals[0] || 'unknown',\r\n      location: characterState.location,\r\n      knowledge: [...characterState.knowledge],\r\n      relationships: { ...characterState.relationships },\r\n      personality: [...personality],\r\n      emotionalState: characterState.emotionalState,\r\n      inventory: [],\r\n      agenda: [],\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Add an agenda item for a character\r\n   */\r\n  addAgendaItem(\r\n    agent: CharacterAgent,\r\n    action: string,\r\n    priority: number = 5,\r\n    deadline?: number\r\n  ): CharacterAgent {\r\n    return {\r\n      ...agent,\r\n      agenda: [\r\n        ...agent.agenda,\r\n        {\r\n          id: `agenda-${Date.now()}-${Math.random().toString(36).substring(2, 9)}`,\r\n          action,\r\n          priority,\r\n          deadline,\r\n          completed: false,\r\n        },\r\n      ],\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Complete an agenda item\r\n   */\r\n  completeAgendaItem(agent: CharacterAgent, agendaId: string): CharacterAgent {\r\n    return {\r\n      ...agent,\r\n      agenda: agent.agenda.map(item =>\r\n        item.id === agendaId ? { ...item, completed: true } : item\r\n      ),\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Update character knowledge\r\n   */\r\n  addKnowledge(agent: CharacterAgent, fact: string): CharacterAgent {\r\n    if (agent.knowledge.includes(fact)) return agent;\r\n    return {\r\n      ...agent,\r\n      knowledge: [...agent.knowledge, fact],\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Update relationship\r\n   */\r\n  updateRelationship(\r\n    agent: CharacterAgent,\r\n    otherCharacter: string,\r\n    relationship: string\r\n  ): CharacterAgent {\r\n    return {\r\n      ...agent,\r\n      relationships: {\r\n        ...agent.relationships,\r\n        [otherCharacter]: relationship,\r\n      },\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Move character to new location\r\n   */\r\n  moveTo(agent: CharacterAgent, location: string): CharacterAgent {\r\n    return {\r\n      ...agent,\r\n      location,\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Get character decision using LLM\r\n   */\r\n  async getDecision(context: CharacterAgentContext): Promise<CharacterDecision> {\r\n    const { character, otherCharacters, worldEvents, currentChapter, storyContext } = context;\r\n    \r\n    const prompt = CHARACTER_DECISION_PROMPT\r\n      .replace(/{{name}}/g, character.name)\r\n      .replace('{{personality}}', character.personality.join(', '))\r\n      .replace('{{emotionalState}}', character.emotionalState)\r\n      .replace('{{location}}', character.location)\r\n      .replace('{{goals}}', character.goals.map(g => `- ${g}`).join('\\n'))\r\n      .replace('{{agenda}}', character.agenda.filter(a => !a.completed).map(a => `- ${a.action} (priority: ${a.priority})`).join('\\n') || 'No active agenda')\r\n      .replace('{{knowledge}}', character.knowledge.map(k => `- ${k}`).join('\\n') || 'No special knowledge')\r\n      .replace('{{relationships}}', Object.entries(character.relationships).map(([name, rel]) => `- ${name}: ${rel}`).join('\\n') || 'No significant relationships')\r\n      .replace('{{chapter}}', currentChapter.toString())\r\n      .replace('{{storyContext}}', storyContext)\r\n      .replace('{{otherCharacters}}', otherCharacters.map(c => `- ${c.name} (${c.emotionalState}, at ${c.location})`).join('\\n') || 'No other characters present')\r\n      .replace('{{worldEvents}}', worldEvents.map(e => `- ${e}`).join('\\n') || 'No recent events');\r\n    \r\n    const result = await getLLM().completeJSON<CharacterDecision>(prompt, {\r\n      temperature: 0.5,\r\n      maxTokens: 1000,\r\n    });\r\n    \r\n    return result;\r\n  }\r\n  \r\n  /**\r\n   * Get simple decision without LLM (for testing/fallback)\r\n   */\r\n  getSimpleDecision(context: CharacterAgentContext): CharacterDecision {\r\n    const { character, otherCharacters } = context;\r\n    \r\n    // Find incomplete agenda items\r\n    const activeAgenda = character.agenda.filter(a => !a.completed);\r\n    \r\n    if (activeAgenda.length > 0) {\r\n      // Follow highest priority agenda item\r\n      const topPriority = activeAgenda.sort((a, b) => b.priority - a.priority)[0];\r\n      return {\r\n        character: character.name,\r\n        action: topPriority.action,\r\n        reasoning: `Following their agenda: ${topPriority.action}`,\r\n        consequences: ['Progress toward goal'],\r\n      };\r\n    }\r\n    \r\n    // React to other characters if present\r\n    if (otherCharacters.length > 0) {\r\n      const other = otherCharacters[0];\r\n      const relationship = character.relationships[other.name] || 'neutral';\r\n      \r\n      if (relationship.includes('friend') || relationship.includes('ally')) {\r\n        return {\r\n          character: character.name,\r\n          action: `Approach ${other.name} to talk`,\r\n          target: other.name,\r\n          reasoning: `They are ${relationship} and nearby`,\r\n          consequences: ['Social interaction', 'Information exchange'],\r\n        };\r\n      }\r\n      \r\n      if (relationship.includes('enemy') || relationship.includes('hostile')) {\r\n        return {\r\n          character: character.name,\r\n          action: `Keep distance from ${other.name}`,\r\n          target: other.name,\r\n          reasoning: `They are ${relationship}`,\r\n          consequences: ['Avoiding conflict', 'Maintaining safety'],\r\n        };\r\n      }\r\n    }\r\n    \r\n    // Default: pursue current goal\r\n    return {\r\n      character: character.name,\r\n      action: `Work toward goal: ${character.currentGoal}`,\r\n      reasoning: 'No immediate distractions, focusing on primary objective',\r\n      consequences: ['Progress toward goal', 'Possible new opportunities'],\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Simulate multiple characters making decisions\r\n   */\r\n  async simulateTurn(\r\n    agents: CharacterAgent[],\r\n    worldEvents: string[],\r\n    currentChapter: number,\r\n    storyContext: string,\r\n    useLLM: boolean = false\r\n  ): Promise<CharacterDecision[]> {\r\n    const decisions: CharacterDecision[] = [];\r\n    \r\n    for (const agent of agents) {\r\n      const context: CharacterAgentContext = {\r\n        character: agent,\r\n        otherCharacters: agents.filter(a => a.name !== agent.name),\r\n        worldEvents,\r\n        currentChapter,\r\n        storyContext,\r\n      };\r\n      \r\n      try {\r\n        const decision = useLLM\r\n          ? await this.getDecision(context)\r\n          : this.getSimpleDecision(context);\r\n        decisions.push(decision);\r\n      } catch (error) {\r\n        // Fallback to simple decision on error\r\n        decisions.push(this.getSimpleDecision(context));\r\n      }\r\n    }\r\n    \r\n    return decisions;\r\n  }\r\n}\r\n\r\nexport const characterAgentSystem = new CharacterAgentSystem();\r\n"]}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import type { CharacterAgent } from './characterAgent.js';
|
|
2
|
+
import type { CharacterDecision } from './characterAgent.js';
|
|
3
|
+
export interface WorldEvent {
|
|
4
|
+
id: string;
|
|
5
|
+
type: 'interaction' | 'conflict' | 'discovery' | 'movement' | 'environmental';
|
|
6
|
+
description: string;
|
|
7
|
+
participants: string[];
|
|
8
|
+
location: string;
|
|
9
|
+
chapter: number;
|
|
10
|
+
resolved: boolean;
|
|
11
|
+
outcome?: string;
|
|
12
|
+
}
|
|
13
|
+
export interface EventResolution {
|
|
14
|
+
event: WorldEvent;
|
|
15
|
+
outcome: string;
|
|
16
|
+
affectedCharacters: string[];
|
|
17
|
+
consequences: string[];
|
|
18
|
+
newEvents?: string[];
|
|
19
|
+
}
|
|
20
|
+
export interface ConflictResolution {
|
|
21
|
+
winner?: string;
|
|
22
|
+
compromise: boolean;
|
|
23
|
+
outcome: string;
|
|
24
|
+
damage: string[];
|
|
25
|
+
}
|
|
26
|
+
export declare class EventResolver {
|
|
27
|
+
/**
|
|
28
|
+
* Resolve character decisions into world events
|
|
29
|
+
*/
|
|
30
|
+
resolveDecisions(decisions: CharacterDecision[], currentChapter: number): WorldEvent[];
|
|
31
|
+
/**
|
|
32
|
+
* Categorize an action into event type
|
|
33
|
+
*/
|
|
34
|
+
private categorizeAction;
|
|
35
|
+
/**
|
|
36
|
+
* Resolve a conflict between characters
|
|
37
|
+
*/
|
|
38
|
+
resolveConflict(event: WorldEvent, participants: CharacterAgent[]): ConflictResolution;
|
|
39
|
+
/**
|
|
40
|
+
* Resolve an event and determine consequences
|
|
41
|
+
*/
|
|
42
|
+
resolveEvent(event: WorldEvent, participants: CharacterAgent[]): EventResolution;
|
|
43
|
+
/**
|
|
44
|
+
* Process all pending events
|
|
45
|
+
*/
|
|
46
|
+
processEvents(events: WorldEvent[], agents: Map<string, CharacterAgent>): EventResolution[];
|
|
47
|
+
/**
|
|
48
|
+
* Generate narrative description of event resolution
|
|
49
|
+
*/
|
|
50
|
+
narrateResolution(resolution: EventResolution): string;
|
|
51
|
+
}
|
|
52
|
+
export declare const eventResolver: EventResolver;
|
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.eventResolver = exports.EventResolver = void 0;
|
|
4
|
+
class EventResolver {
|
|
5
|
+
/**
|
|
6
|
+
* Resolve character decisions into world events
|
|
7
|
+
*/
|
|
8
|
+
resolveDecisions(decisions, currentChapter) {
|
|
9
|
+
const events = [];
|
|
10
|
+
// Group decisions by location
|
|
11
|
+
const locationGroups = new Map();
|
|
12
|
+
for (const decision of decisions) {
|
|
13
|
+
// Note: location would need to be passed in or tracked
|
|
14
|
+
const location = 'current location'; // Simplified
|
|
15
|
+
const existing = locationGroups.get(location) || [];
|
|
16
|
+
existing.push(decision);
|
|
17
|
+
locationGroups.set(location, existing);
|
|
18
|
+
}
|
|
19
|
+
// Check for interactions between characters at same location
|
|
20
|
+
for (const [location, locationDecisions] of locationGroups) {
|
|
21
|
+
if (locationDecisions.length >= 2) {
|
|
22
|
+
// Check for mutual interactions
|
|
23
|
+
for (let i = 0; i < locationDecisions.length; i++) {
|
|
24
|
+
for (let j = i + 1; j < locationDecisions.length; j++) {
|
|
25
|
+
const d1 = locationDecisions[i];
|
|
26
|
+
const d2 = locationDecisions[j];
|
|
27
|
+
// Check if they're interacting with each other
|
|
28
|
+
if (d1.target === d2.character && d2.target === d1.character) {
|
|
29
|
+
events.push({
|
|
30
|
+
id: `event-${Date.now()}-${events.length}`,
|
|
31
|
+
type: 'interaction',
|
|
32
|
+
description: `${d1.character} and ${d2.character} interact`,
|
|
33
|
+
participants: [d1.character, d2.character],
|
|
34
|
+
location,
|
|
35
|
+
chapter: currentChapter,
|
|
36
|
+
resolved: false,
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
// Add individual actions as events
|
|
43
|
+
for (const decision of locationDecisions) {
|
|
44
|
+
if (!events.some(e => e.participants.includes(decision.character))) {
|
|
45
|
+
events.push({
|
|
46
|
+
id: `event-${Date.now()}-${events.length}`,
|
|
47
|
+
type: this.categorizeAction(decision.action),
|
|
48
|
+
description: `${decision.character}: ${decision.action}`,
|
|
49
|
+
participants: [decision.character],
|
|
50
|
+
location,
|
|
51
|
+
chapter: currentChapter,
|
|
52
|
+
resolved: false,
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
return events;
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Categorize an action into event type
|
|
61
|
+
*/
|
|
62
|
+
categorizeAction(action) {
|
|
63
|
+
const actionLower = action.toLowerCase();
|
|
64
|
+
if (actionLower.includes('fight') || actionLower.includes('attack') || actionLower.includes('confront')) {
|
|
65
|
+
return 'conflict';
|
|
66
|
+
}
|
|
67
|
+
if (actionLower.includes('find') || actionLower.includes('discover') || actionLower.includes('learn')) {
|
|
68
|
+
return 'discovery';
|
|
69
|
+
}
|
|
70
|
+
if (actionLower.includes('go') || actionLower.includes('move') || actionLower.includes('travel')) {
|
|
71
|
+
return 'movement';
|
|
72
|
+
}
|
|
73
|
+
if (actionLower.includes('weather') || actionLower.includes('storm') || actionLower.includes('dark')) {
|
|
74
|
+
return 'environmental';
|
|
75
|
+
}
|
|
76
|
+
return 'interaction';
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Resolve a conflict between characters
|
|
80
|
+
*/
|
|
81
|
+
resolveConflict(event, participants) {
|
|
82
|
+
// Simple resolution based on emotional state and personality
|
|
83
|
+
const scores = participants.map(p => {
|
|
84
|
+
let score = 5; // base
|
|
85
|
+
// Emotional state affects capability
|
|
86
|
+
if (p.emotionalState.includes('angry') || p.emotionalState.includes('furious')) {
|
|
87
|
+
score += 2;
|
|
88
|
+
}
|
|
89
|
+
if (p.emotionalState.includes('fear') || p.emotionalState.includes('terrified')) {
|
|
90
|
+
score -= 2;
|
|
91
|
+
}
|
|
92
|
+
if (p.emotionalState.includes('calm') || p.emotionalState.includes('focused')) {
|
|
93
|
+
score += 1;
|
|
94
|
+
}
|
|
95
|
+
// Personality traits
|
|
96
|
+
if (p.personality.some(t => t.includes('strong') || t.includes('brave'))) {
|
|
97
|
+
score += 1;
|
|
98
|
+
}
|
|
99
|
+
if (p.personality.some(t => t.includes('weak') || t.includes('timid'))) {
|
|
100
|
+
score -= 1;
|
|
101
|
+
}
|
|
102
|
+
return { name: p.name, score };
|
|
103
|
+
});
|
|
104
|
+
// Determine outcome
|
|
105
|
+
scores.sort((a, b) => b.score - a.score);
|
|
106
|
+
const winner = scores[0];
|
|
107
|
+
const loser = scores[scores.length - 1];
|
|
108
|
+
const scoreDiff = winner.score - loser.score;
|
|
109
|
+
if (scoreDiff < 2) {
|
|
110
|
+
// Close match - compromise
|
|
111
|
+
return {
|
|
112
|
+
compromise: true,
|
|
113
|
+
outcome: 'Both sides reach a compromise',
|
|
114
|
+
damage: ['Minor injuries', 'Tension remains'],
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
return {
|
|
118
|
+
winner: winner.name,
|
|
119
|
+
compromise: false,
|
|
120
|
+
outcome: `${winner.name} prevails over ${loser.name}`,
|
|
121
|
+
damage: [`${loser.name} is defeated`, 'Physical or emotional consequences'],
|
|
122
|
+
};
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Resolve an event and determine consequences
|
|
126
|
+
*/
|
|
127
|
+
resolveEvent(event, participants) {
|
|
128
|
+
let outcome = '';
|
|
129
|
+
const consequences = [];
|
|
130
|
+
const newEvents = [];
|
|
131
|
+
switch (event.type) {
|
|
132
|
+
case 'conflict':
|
|
133
|
+
const conflictResult = this.resolveConflict(event, participants);
|
|
134
|
+
outcome = conflictResult.outcome;
|
|
135
|
+
consequences.push(...conflictResult.damage);
|
|
136
|
+
if (conflictResult.winner) {
|
|
137
|
+
consequences.push(`${conflictResult.winner} gains advantage`);
|
|
138
|
+
}
|
|
139
|
+
break;
|
|
140
|
+
case 'discovery':
|
|
141
|
+
outcome = `${participants[0]?.name} makes an important discovery`;
|
|
142
|
+
consequences.push('New knowledge gained', 'Future possibilities opened');
|
|
143
|
+
break;
|
|
144
|
+
case 'interaction':
|
|
145
|
+
if (participants.length >= 2) {
|
|
146
|
+
outcome = `${participants.map(p => p.name).join(' and ')} have a meaningful interaction`;
|
|
147
|
+
consequences.push('Relationship development', 'Information exchange');
|
|
148
|
+
}
|
|
149
|
+
else {
|
|
150
|
+
outcome = `${participants[0]?.name} takes action`;
|
|
151
|
+
consequences.push('Progress toward goal');
|
|
152
|
+
}
|
|
153
|
+
break;
|
|
154
|
+
case 'movement':
|
|
155
|
+
outcome = `${participants[0]?.name} changes location`;
|
|
156
|
+
consequences.push('New environment', 'New opportunities or dangers');
|
|
157
|
+
break;
|
|
158
|
+
case 'environmental':
|
|
159
|
+
outcome = 'Environmental event affects the scene';
|
|
160
|
+
consequences.push('All participants must react', 'Situation becomes more complex');
|
|
161
|
+
break;
|
|
162
|
+
}
|
|
163
|
+
return {
|
|
164
|
+
event: { ...event, resolved: true, outcome },
|
|
165
|
+
outcome,
|
|
166
|
+
affectedCharacters: participants.map(p => p.name),
|
|
167
|
+
consequences,
|
|
168
|
+
newEvents,
|
|
169
|
+
};
|
|
170
|
+
}
|
|
171
|
+
/**
|
|
172
|
+
* Process all pending events
|
|
173
|
+
*/
|
|
174
|
+
processEvents(events, agents) {
|
|
175
|
+
const resolutions = [];
|
|
176
|
+
for (const event of events.filter(e => !e.resolved)) {
|
|
177
|
+
const participants = event.participants
|
|
178
|
+
.map(name => agents.get(name))
|
|
179
|
+
.filter((agent) => agent !== undefined);
|
|
180
|
+
if (participants.length > 0) {
|
|
181
|
+
const resolution = this.resolveEvent(event, participants);
|
|
182
|
+
resolutions.push(resolution);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
return resolutions;
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Generate narrative description of event resolution
|
|
189
|
+
*/
|
|
190
|
+
narrateResolution(resolution) {
|
|
191
|
+
const lines = [];
|
|
192
|
+
lines.push(`**${resolution.event.description}**`);
|
|
193
|
+
lines.push(`Outcome: ${resolution.outcome}`);
|
|
194
|
+
if (resolution.consequences.length > 0) {
|
|
195
|
+
lines.push('Consequences:');
|
|
196
|
+
for (const consequence of resolution.consequences) {
|
|
197
|
+
lines.push(` - ${consequence}`);
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
return lines.join('\n');
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
exports.EventResolver = EventResolver;
|
|
204
|
+
exports.eventResolver = new EventResolver();
|
|
205
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"eventResolver.js","sourceRoot":"","sources":["../../src/world/eventResolver.ts"],"names":[],"mappings":"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","sourcesContent":["import type { CharacterAgent } from './characterAgent.js';\r\nimport type { CharacterDecision } from './characterAgent.js';\r\n\r\nexport interface WorldEvent {\r\n  id: string;\r\n  type: 'interaction' | 'conflict' | 'discovery' | 'movement' | 'environmental';\r\n  description: string;\r\n  participants: string[];\r\n  location: string;\r\n  chapter: number;\r\n  resolved: boolean;\r\n  outcome?: string;\r\n}\r\n\r\nexport interface EventResolution {\r\n  event: WorldEvent;\r\n  outcome: string;\r\n  affectedCharacters: string[];\r\n  consequences: string[];\r\n  newEvents?: string[];\r\n}\r\n\r\nexport interface ConflictResolution {\r\n  winner?: string;\r\n  compromise: boolean;\r\n  outcome: string;\r\n  damage: string[];\r\n}\r\n\r\nexport class EventResolver {\r\n  /**\r\n   * Resolve character decisions into world events\r\n   */\r\n  resolveDecisions(\r\n    decisions: CharacterDecision[],\r\n    currentChapter: number\r\n  ): WorldEvent[] {\r\n    const events: WorldEvent[] = [];\r\n    \r\n    // Group decisions by location\r\n    const locationGroups = new Map<string, CharacterDecision[]>();\r\n    for (const decision of decisions) {\r\n      // Note: location would need to be passed in or tracked\r\n      const location = 'current location'; // Simplified\r\n      const existing = locationGroups.get(location) || [];\r\n      existing.push(decision);\r\n      locationGroups.set(location, existing);\r\n    }\r\n    \r\n    // Check for interactions between characters at same location\r\n    for (const [location, locationDecisions] of locationGroups) {\r\n      if (locationDecisions.length >= 2) {\r\n        // Check for mutual interactions\r\n        for (let i = 0; i < locationDecisions.length; i++) {\r\n          for (let j = i + 1; j < locationDecisions.length; j++) {\r\n            const d1 = locationDecisions[i];\r\n            const d2 = locationDecisions[j];\r\n            \r\n            // Check if they're interacting with each other\r\n            if (d1.target === d2.character && d2.target === d1.character) {\r\n              events.push({\r\n                id: `event-${Date.now()}-${events.length}`,\r\n                type: 'interaction',\r\n                description: `${d1.character} and ${d2.character} interact`,\r\n                participants: [d1.character, d2.character],\r\n                location,\r\n                chapter: currentChapter,\r\n                resolved: false,\r\n              });\r\n            }\r\n          }\r\n        }\r\n      }\r\n      \r\n      // Add individual actions as events\r\n      for (const decision of locationDecisions) {\r\n        if (!events.some(e => e.participants.includes(decision.character))) {\r\n          events.push({\r\n            id: `event-${Date.now()}-${events.length}`,\r\n            type: this.categorizeAction(decision.action),\r\n            description: `${decision.character}: ${decision.action}`,\r\n            participants: [decision.character],\r\n            location,\r\n            chapter: currentChapter,\r\n            resolved: false,\r\n          });\r\n        }\r\n      }\r\n    }\r\n    \r\n    return events;\r\n  }\r\n  \r\n  /**\r\n   * Categorize an action into event type\r\n   */\r\n  private categorizeAction(action: string): WorldEvent['type'] {\r\n    const actionLower = action.toLowerCase();\r\n    \r\n    if (actionLower.includes('fight') || actionLower.includes('attack') || actionLower.includes('confront')) {\r\n      return 'conflict';\r\n    }\r\n    if (actionLower.includes('find') || actionLower.includes('discover') || actionLower.includes('learn')) {\r\n      return 'discovery';\r\n    }\r\n    if (actionLower.includes('go') || actionLower.includes('move') || actionLower.includes('travel')) {\r\n      return 'movement';\r\n    }\r\n    if (actionLower.includes('weather') || actionLower.includes('storm') || actionLower.includes('dark')) {\r\n      return 'environmental';\r\n    }\r\n    \r\n    return 'interaction';\r\n  }\r\n  \r\n  /**\r\n   * Resolve a conflict between characters\r\n   */\r\n  resolveConflict(\r\n    event: WorldEvent,\r\n    participants: CharacterAgent[]\r\n  ): ConflictResolution {\r\n    // Simple resolution based on emotional state and personality\r\n    const scores = participants.map(p => {\r\n      let score = 5; // base\r\n      \r\n      // Emotional state affects capability\r\n      if (p.emotionalState.includes('angry') || p.emotionalState.includes('furious')) {\r\n        score += 2;\r\n      }\r\n      if (p.emotionalState.includes('fear') || p.emotionalState.includes('terrified')) {\r\n        score -= 2;\r\n      }\r\n      if (p.emotionalState.includes('calm') || p.emotionalState.includes('focused')) {\r\n        score += 1;\r\n      }\r\n      \r\n      // Personality traits\r\n      if (p.personality.some(t => t.includes('strong') || t.includes('brave'))) {\r\n        score += 1;\r\n      }\r\n      if (p.personality.some(t => t.includes('weak') || t.includes('timid'))) {\r\n        score -= 1;\r\n      }\r\n      \r\n      return { name: p.name, score };\r\n    });\r\n    \r\n    // Determine outcome\r\n    scores.sort((a, b) => b.score - a.score);\r\n    const winner = scores[0];\r\n    const loser = scores[scores.length - 1];\r\n    const scoreDiff = winner.score - loser.score;\r\n    \r\n    if (scoreDiff < 2) {\r\n      // Close match - compromise\r\n      return {\r\n        compromise: true,\r\n        outcome: 'Both sides reach a compromise',\r\n        damage: ['Minor injuries', 'Tension remains'],\r\n      };\r\n    }\r\n    \r\n    return {\r\n      winner: winner.name,\r\n      compromise: false,\r\n      outcome: `${winner.name} prevails over ${loser.name}`,\r\n      damage: [`${loser.name} is defeated`, 'Physical or emotional consequences'],\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Resolve an event and determine consequences\r\n   */\r\n  resolveEvent(\r\n    event: WorldEvent,\r\n    participants: CharacterAgent[]\r\n  ): EventResolution {\r\n    let outcome = '';\r\n    const consequences: string[] = [];\r\n    const newEvents: string[] = [];\r\n    \r\n    switch (event.type) {\r\n      case 'conflict':\r\n        const conflictResult = this.resolveConflict(event, participants);\r\n        outcome = conflictResult.outcome;\r\n        consequences.push(...conflictResult.damage);\r\n        if (conflictResult.winner) {\r\n          consequences.push(`${conflictResult.winner} gains advantage`);\r\n        }\r\n        break;\r\n        \r\n      case 'discovery':\r\n        outcome = `${participants[0]?.name} makes an important discovery`;\r\n        consequences.push('New knowledge gained', 'Future possibilities opened');\r\n        break;\r\n        \r\n      case 'interaction':\r\n        if (participants.length >= 2) {\r\n          outcome = `${participants.map(p => p.name).join(' and ')} have a meaningful interaction`;\r\n          consequences.push('Relationship development', 'Information exchange');\r\n        } else {\r\n          outcome = `${participants[0]?.name} takes action`;\r\n          consequences.push('Progress toward goal');\r\n        }\r\n        break;\r\n        \r\n      case 'movement':\r\n        outcome = `${participants[0]?.name} changes location`;\r\n        consequences.push('New environment', 'New opportunities or dangers');\r\n        break;\r\n        \r\n      case 'environmental':\r\n        outcome = 'Environmental event affects the scene';\r\n        consequences.push('All participants must react', 'Situation becomes more complex');\r\n        break;\r\n    }\r\n    \r\n    return {\r\n      event: { ...event, resolved: true, outcome },\r\n      outcome,\r\n      affectedCharacters: participants.map(p => p.name),\r\n      consequences,\r\n      newEvents,\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Process all pending events\r\n   */\r\n  processEvents(\r\n    events: WorldEvent[],\r\n    agents: Map<string, CharacterAgent>\r\n  ): EventResolution[] {\r\n    const resolutions: EventResolution[] = [];\r\n    \r\n    for (const event of events.filter(e => !e.resolved)) {\r\n      const participants = event.participants\r\n        .map(name => agents.get(name))\r\n        .filter((agent): agent is CharacterAgent => agent !== undefined);\r\n      \r\n      if (participants.length > 0) {\r\n        const resolution = this.resolveEvent(event, participants);\r\n        resolutions.push(resolution);\r\n      }\r\n    }\r\n    \r\n    return resolutions;\r\n  }\r\n  \r\n  /**\r\n   * Generate narrative description of event resolution\r\n   */\r\n  narrateResolution(resolution: EventResolution): string {\r\n    const lines: string[] = [];\r\n    \r\n    lines.push(`**${resolution.event.description}**`);\r\n    lines.push(`Outcome: ${resolution.outcome}`);\r\n    \r\n    if (resolution.consequences.length > 0) {\r\n      lines.push('Consequences:');\r\n      for (const consequence of resolution.consequences) {\r\n        lines.push(`  - ${consequence}`);\r\n      }\r\n    }\r\n    \r\n    return lines.join('\\n');\r\n  }\r\n}\r\n\r\nexport const eventResolver = new EventResolver();\r\n"]}
|