klaus-agent 0.3.1 → 0.4.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.
@@ -0,0 +1,241 @@
1
+ // Task graph — dependency-aware DAG with background execution and auto-unlock
2
+ import { readFileSync, writeFileSync, mkdirSync, existsSync, renameSync } from "fs";
3
+ import { join } from "path";
4
+ import { generateId } from "../utils/id.js";
5
+ export class TaskGraph {
6
+ _tasks = new Map();
7
+ _config;
8
+ _completedQueue = [];
9
+ _backgroundAborts = new Map();
10
+ constructor(config = {}) {
11
+ this._config = config;
12
+ if (config.persistDir) {
13
+ mkdirSync(config.persistDir, { recursive: true });
14
+ this._loadFromDisk();
15
+ }
16
+ }
17
+ get(id) {
18
+ const task = this._tasks.get(id);
19
+ return task ? { ...task } : undefined;
20
+ }
21
+ listAll() {
22
+ return [...this._tasks.values()];
23
+ }
24
+ /** Tasks that are pending with no unfinished blockers. */
25
+ listReady() {
26
+ return this.listAll().filter((t) => t.status === "pending" && t.blockedBy.length === 0);
27
+ }
28
+ /** Tasks waiting on unfinished blockers. */
29
+ listBlocked() {
30
+ return this.listAll().filter((t) => t.status === "pending" && t.blockedBy.length > 0);
31
+ }
32
+ create(subject, description = "") {
33
+ const max = this._config.maxTasks ?? 100;
34
+ if (this._tasks.size >= max) {
35
+ throw new Error(`Task limit reached: ${max}.`);
36
+ }
37
+ const node = {
38
+ id: generateId(),
39
+ subject,
40
+ description,
41
+ status: "pending",
42
+ blockedBy: [],
43
+ blocks: [],
44
+ owner: "",
45
+ createdAt: Date.now(),
46
+ updatedAt: Date.now(),
47
+ };
48
+ this._tasks.set(node.id, node);
49
+ this._persist();
50
+ return { ...node };
51
+ }
52
+ /** Add a dependency: `taskId` is blocked by `blockedById`. */
53
+ addDependency(taskId, blockedById) {
54
+ const task = this._require(taskId);
55
+ const blocker = this._require(blockedById);
56
+ if (task.blockedBy.includes(blockedById))
57
+ return;
58
+ // Cycle detection: if blocker is (transitively) blocked by task, adding this edge creates a cycle
59
+ if (this._isTransitivelyBlockedBy(blockedById, taskId)) {
60
+ throw new Error(`Adding dependency ${blockedById} → ${taskId} would create a cycle.`);
61
+ }
62
+ task.blockedBy.push(blockedById);
63
+ blocker.blocks.push(taskId);
64
+ task.updatedAt = Date.now();
65
+ blocker.updatedAt = Date.now();
66
+ this._persist();
67
+ }
68
+ update(taskId, fields) {
69
+ const task = this._require(taskId);
70
+ if (fields.status !== undefined && fields.status !== task.status) {
71
+ if (fields.status === "in_progress" && task.blockedBy.length > 0) {
72
+ throw new Error(`Task ${taskId} is blocked by: ${task.blockedBy.join(", ")}`);
73
+ }
74
+ task.status = fields.status;
75
+ if (fields.status === "completed" || fields.status === "failed") {
76
+ task.result = fields.result ?? task.result;
77
+ const unblocked = this._clearDependency(taskId);
78
+ this._completedQueue.push({
79
+ taskId,
80
+ subject: task.subject,
81
+ result: task.result ?? "",
82
+ status: fields.status,
83
+ unblockedTasks: unblocked,
84
+ });
85
+ }
86
+ }
87
+ if (fields.owner !== undefined)
88
+ task.owner = fields.owner;
89
+ // result is already set inside the completion branch above; only apply here for non-completion updates
90
+ if (fields.result !== undefined && task.status !== "completed" && task.status !== "failed") {
91
+ task.result = fields.result;
92
+ }
93
+ task.updatedAt = Date.now();
94
+ this._persist();
95
+ return { ...task };
96
+ }
97
+ /**
98
+ * Run an async function in the background for a task.
99
+ * Auto-updates task status to in_progress/completed/failed.
100
+ */
101
+ runBackground(taskId, fn) {
102
+ const task = this._require(taskId);
103
+ if (task.blockedBy.length > 0) {
104
+ throw new Error(`Task ${taskId} is blocked by: ${task.blockedBy.join(", ")}`);
105
+ }
106
+ const ac = new AbortController();
107
+ const bgId = generateId();
108
+ task.status = "in_progress";
109
+ task.backgroundId = bgId;
110
+ task.updatedAt = Date.now();
111
+ this._backgroundAborts.set(bgId, ac);
112
+ this._persist();
113
+ fn(ac.signal).then((result) => {
114
+ this._backgroundAborts.delete(bgId);
115
+ try {
116
+ this.update(taskId, { status: "completed", result });
117
+ }
118
+ catch { /* task may have been removed */ }
119
+ }, (err) => {
120
+ this._backgroundAborts.delete(bgId);
121
+ try {
122
+ this.update(taskId, {
123
+ status: "failed",
124
+ result: err instanceof Error ? err.message : String(err),
125
+ });
126
+ }
127
+ catch { /* task may have been removed */ }
128
+ });
129
+ }
130
+ abortBackground(taskId) {
131
+ const task = this._tasks.get(taskId);
132
+ if (!task?.backgroundId)
133
+ return false;
134
+ const ac = this._backgroundAborts.get(task.backgroundId);
135
+ if (!ac)
136
+ return false;
137
+ ac.abort();
138
+ return true;
139
+ }
140
+ drainCompleted() {
141
+ const results = [...this._completedQueue];
142
+ this._completedQueue = [];
143
+ return results;
144
+ }
145
+ render() {
146
+ const tasks = this.listAll();
147
+ if (tasks.length === 0)
148
+ return "No tasks.";
149
+ const lines = tasks.map((t) => {
150
+ const icon = t.status === "completed" ? "[x]"
151
+ : t.status === "failed" ? "[!]"
152
+ : t.status === "in_progress" ? "[>]"
153
+ : t.blockedBy.length > 0 ? "[~]"
154
+ : "[ ]";
155
+ const deps = t.blockedBy.length > 0 ? ` (blocked by: ${t.blockedBy.join(", ")})` : "";
156
+ return `${icon} ${t.id}: ${t.subject}${deps}`;
157
+ });
158
+ const done = tasks.filter((t) => t.status === "completed").length;
159
+ const ready = tasks.filter((t) => t.status === "pending" && t.blockedBy.length === 0).length;
160
+ return `Tasks: ${done}/${tasks.length} done, ${ready} ready\n${lines.join("\n")}`;
161
+ }
162
+ dispose() {
163
+ for (const ac of this._backgroundAborts.values()) {
164
+ ac.abort();
165
+ }
166
+ this._backgroundAborts.clear();
167
+ }
168
+ _require(id) {
169
+ const task = this._tasks.get(id);
170
+ if (!task)
171
+ throw new Error(`Task not found: ${id}`);
172
+ return task;
173
+ }
174
+ /** Remove completedId from all tasks' blockedBy. Returns IDs of newly unblocked tasks. */
175
+ _clearDependency(completedId) {
176
+ const completed = this._tasks.get(completedId);
177
+ if (!completed)
178
+ return [];
179
+ const unblocked = [];
180
+ for (const dependentId of completed.blocks) {
181
+ const task = this._tasks.get(dependentId);
182
+ if (!task)
183
+ continue;
184
+ const idx = task.blockedBy.indexOf(completedId);
185
+ if (idx !== -1) {
186
+ task.blockedBy.splice(idx, 1);
187
+ if (task.blockedBy.length === 0 && task.status === "pending") {
188
+ unblocked.push(task.id);
189
+ }
190
+ }
191
+ }
192
+ return unblocked;
193
+ }
194
+ /** Check if `taskId` is transitively blocked by `targetId`. */
195
+ _isTransitivelyBlockedBy(taskId, targetId) {
196
+ const visited = new Set();
197
+ const stack = [taskId];
198
+ while (stack.length > 0) {
199
+ const current = stack.pop();
200
+ if (current === targetId)
201
+ return true;
202
+ if (visited.has(current))
203
+ continue;
204
+ visited.add(current);
205
+ const node = this._tasks.get(current);
206
+ if (node)
207
+ stack.push(...node.blockedBy);
208
+ }
209
+ return false;
210
+ }
211
+ _persist() {
212
+ if (!this._config.persistDir)
213
+ return;
214
+ const data = JSON.stringify([...this._tasks.values()], null, 2);
215
+ const target = join(this._config.persistDir, "tasks.json");
216
+ const tmp = target + ".tmp";
217
+ writeFileSync(tmp, data, "utf-8");
218
+ renameSync(tmp, target);
219
+ }
220
+ _loadFromDisk() {
221
+ if (!this._config.persistDir)
222
+ return;
223
+ const filePath = join(this._config.persistDir, "tasks.json");
224
+ if (!existsSync(filePath))
225
+ return;
226
+ try {
227
+ const raw = JSON.parse(readFileSync(filePath, "utf-8"));
228
+ if (!Array.isArray(raw))
229
+ return;
230
+ for (const entry of raw) {
231
+ if (entry && typeof entry === "object" && typeof entry.id === "string" && typeof entry.subject === "string") {
232
+ this._tasks.set(entry.id, entry);
233
+ }
234
+ }
235
+ }
236
+ catch {
237
+ // Corrupted file — start fresh
238
+ }
239
+ }
240
+ }
241
+ //# sourceMappingURL=task-graph.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"task-graph.js","sourceRoot":"","sources":["../../src/task-graph/task-graph.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { AgentTool } from "../tools/types.js";
2
+ import type { TaskGraph } from "./task-graph.js";
3
+ export declare function createTaskGraphTools(graph: TaskGraph): AgentTool[];
@@ -0,0 +1,106 @@
1
+ // Task graph tools — CRUD + dependency management + background execution
2
+ import { Type } from "@sinclair/typebox";
3
+ import { TASK_GRAPH_TOOL_NAMES } from "./types.js";
4
+ export function createTaskGraphTools(graph) {
5
+ return [
6
+ {
7
+ name: TASK_GRAPH_TOOL_NAMES.create,
8
+ label: "Create Task",
9
+ description: "Create a new task in the task graph. Tasks start as pending. " +
10
+ "Use task_depend to set up dependency ordering.",
11
+ parameters: Type.Object({
12
+ subject: Type.String({ description: "Short title for the task." }),
13
+ description: Type.Optional(Type.String({ description: "Detailed description." })),
14
+ }),
15
+ async execute(_id, params) {
16
+ const task = graph.create(params.subject, params.description);
17
+ return text(`Created task ${task.id}: ${task.subject}\n\n${graph.render()}`);
18
+ },
19
+ },
20
+ {
21
+ name: TASK_GRAPH_TOOL_NAMES.depend,
22
+ label: "Add Task Dependency",
23
+ description: "Add a dependency: task_id cannot start until blocked_by_id completes. " +
24
+ "Rejects if this would create a cycle.",
25
+ parameters: Type.Object({
26
+ task_id: Type.String({ description: "Task that is blocked." }),
27
+ blocked_by_id: Type.String({ description: "Task that must complete first." }),
28
+ }),
29
+ async execute(_id, params) {
30
+ graph.addDependency(params.task_id, params.blocked_by_id);
31
+ return text(`Dependency added: ${params.task_id} blocked by ${params.blocked_by_id}\n\n${graph.render()}`);
32
+ },
33
+ },
34
+ {
35
+ name: TASK_GRAPH_TOOL_NAMES.update,
36
+ label: "Update Task",
37
+ description: "Update a task's status, owner, or result. " +
38
+ "Setting status to 'completed' auto-unblocks dependent tasks. " +
39
+ "Cannot start a task that has unfinished blockers.",
40
+ parameters: Type.Object({
41
+ task_id: Type.String({ description: "Task ID." }),
42
+ status: Type.Optional(Type.Union([Type.Literal("pending"), Type.Literal("in_progress"), Type.Literal("completed"), Type.Literal("failed")], { description: "New status." })),
43
+ owner: Type.Optional(Type.String({ description: "Assign to an agent or user." })),
44
+ result: Type.Optional(Type.String({ description: "Result summary." })),
45
+ }),
46
+ async execute(_id, params) {
47
+ const task = graph.update(params.task_id, {
48
+ status: params.status,
49
+ owner: params.owner,
50
+ result: params.result,
51
+ });
52
+ return text(`Updated task ${task.id}\n\n${graph.render()}`);
53
+ },
54
+ },
55
+ {
56
+ name: TASK_GRAPH_TOOL_NAMES.list,
57
+ label: "List Tasks",
58
+ description: "List all tasks with their status, dependencies, and progress.",
59
+ parameters: Type.Object({
60
+ filter: Type.Optional(Type.Union([Type.Literal("all"), Type.Literal("ready"), Type.Literal("blocked"), Type.Literal("in_progress"), Type.Literal("completed")], { description: "Filter tasks by category. Default: all." })),
61
+ }),
62
+ async execute(_id, params) {
63
+ let tasks;
64
+ switch (params.filter) {
65
+ case "ready":
66
+ tasks = graph.listReady();
67
+ break;
68
+ case "blocked":
69
+ tasks = graph.listBlocked();
70
+ break;
71
+ case "in_progress":
72
+ tasks = graph.listAll().filter((t) => t.status === "in_progress");
73
+ break;
74
+ case "completed":
75
+ tasks = graph.listAll().filter((t) => t.status === "completed");
76
+ break;
77
+ default: tasks = graph.listAll();
78
+ }
79
+ if (tasks.length === 0)
80
+ return text(`No tasks matching filter: ${params.filter ?? "all"}`);
81
+ if (!params.filter || params.filter === "all")
82
+ return text(graph.render());
83
+ const lines = tasks.map((t) => `${t.id}: ${t.subject} [${t.status}]`);
84
+ return text(`${params.filter}: ${tasks.length} task(s)\n${lines.join("\n")}`);
85
+ },
86
+ },
87
+ {
88
+ name: TASK_GRAPH_TOOL_NAMES.get,
89
+ label: "Get Task",
90
+ description: "Get detailed information about a specific task.",
91
+ parameters: Type.Object({
92
+ task_id: Type.String({ description: "Task ID." }),
93
+ }),
94
+ async execute(_id, params) {
95
+ const task = graph.get(params.task_id);
96
+ if (!task)
97
+ return text(`Task not found: ${params.task_id}`);
98
+ return text(JSON.stringify(task, null, 2));
99
+ },
100
+ },
101
+ ];
102
+ }
103
+ function text(t) {
104
+ return { content: [{ type: "text", text: t }] };
105
+ }
106
+ //# sourceMappingURL=tools.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tools.js","sourceRoot":"","sources":["../../src/task-graph/tools.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,44 @@
1
+ export declare const TASK_GRAPH_TOOL_NAMES: {
2
+ readonly create: "task_create";
3
+ readonly depend: "task_depend";
4
+ readonly update: "task_update";
5
+ readonly list: "task_list";
6
+ readonly get: "task_get";
7
+ };
8
+ export type TaskStatus = "pending" | "in_progress" | "completed" | "failed";
9
+ export interface TaskNode {
10
+ id: string;
11
+ subject: string;
12
+ description: string;
13
+ status: TaskStatus;
14
+ /** IDs of tasks that must complete before this one can start. */
15
+ blockedBy: string[];
16
+ /** IDs of tasks that this task blocks (reverse edges, maintained automatically). */
17
+ blocks: string[];
18
+ /** Agent or user assigned to this task. */
19
+ owner: string;
20
+ /** Result summary after completion/failure. */
21
+ result?: string;
22
+ /** Background execution handle ID, if running in background. */
23
+ backgroundId?: string;
24
+ createdAt: number;
25
+ updatedAt: number;
26
+ }
27
+ export interface TaskGraphConfig {
28
+ /** Directory for persisting task graph to disk. If omitted, in-memory only. */
29
+ persistDir?: string;
30
+ /** Maximum number of tasks. Default: 100. */
31
+ maxTasks?: number;
32
+ /**
33
+ * Auto-inject completed background task results before each LLM call.
34
+ * Default: true.
35
+ */
36
+ autoInjectResults?: boolean;
37
+ }
38
+ export interface CompletedTaskResult {
39
+ taskId: string;
40
+ subject: string;
41
+ result: string;
42
+ status: "completed" | "failed";
43
+ unblockedTasks: string[];
44
+ }
@@ -0,0 +1,9 @@
1
+ // Task graph types — dependency-aware task DAG with background execution
2
+ export const TASK_GRAPH_TOOL_NAMES = {
3
+ create: "task_create",
4
+ depend: "task_depend",
5
+ update: "task_update",
6
+ list: "task_list",
7
+ get: "task_get",
8
+ };
9
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/task-graph/types.ts"],"names":[],"mappings":"AAAA,yEAAyE;AAEzE,MAAM,CAAC,MAAM,qBAAqB,GAAG;IACnC,MAAM,EAAE,aAAa;IACrB,MAAM,EAAE,aAAa;IACrB,MAAM,EAAE,aAAa;IACrB,IAAI,EAAE,WAAW;IACjB,GAAG,EAAE,UAAU;CACP,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "klaus-agent",
3
- "version": "0.3.1",
3
+ "version": "0.4.0",
4
4
  "description": "Universal agent framework",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",
package/src/core/agent.ts CHANGED
@@ -22,6 +22,7 @@ import type { SkillSource } from "../skills/types.js";
22
22
  import type { MCPServerConfig, MCPClient } from "../tools/mcp-adapter.js";
23
23
  import type { TaskFactory } from "../background/types.js";
24
24
  import type { PlanningConfig } from "../planning/types.js";
25
+ import type { TaskGraphConfig } from "../task-graph/types.js";
25
26
  import { SessionManager } from "../session/session-manager.js";
26
27
  import { CheckpointManager } from "../checkpoint/checkpoint-manager.js";
27
28
  import { InjectionManager } from "../injection/injection-manager.js";
@@ -39,6 +40,9 @@ import { createBackgroundTaskTools } from "../background/tools.js";
39
40
  import { PlanningManager } from "../planning/planning-manager.js";
40
41
  import { createPlanningTools } from "../planning/tools.js";
41
42
  import { PlanningNagProvider } from "../planning/nag-injection.js";
43
+ import { TaskGraph } from "../task-graph/task-graph.js";
44
+ import { createTaskGraphTools } from "../task-graph/tools.js";
45
+ import { TaskResultInjectionProvider } from "../task-graph/result-injection.js";
42
46
  import { runAgentLoop } from "./agent-loop.js";
43
47
 
44
48
  export interface AgentConfig {
@@ -64,6 +68,7 @@ export interface AgentConfig {
64
68
  mcp?: { servers: MCPServerConfig[]; clientFactory: (config: MCPServerConfig) => MCPClient };
65
69
  wire?: { bufferSize?: number };
66
70
  backgroundTasks?: { factories?: Record<string, TaskFactory> };
71
+ taskGraph?: TaskGraphConfig;
67
72
  planning?: PlanningConfig;
68
73
  }
69
74
 
@@ -89,6 +94,7 @@ export class Agent {
89
94
  private _wire: Wire;
90
95
  private _backgroundTaskManager: BackgroundTaskManager | undefined;
91
96
  private _planningManager: PlanningManager | undefined;
97
+ private _taskGraph: TaskGraph;
92
98
  private _initialized = false;
93
99
 
94
100
  constructor(config: AgentConfig) {
@@ -156,6 +162,9 @@ export class Agent {
156
162
  if (config.planning) {
157
163
  this._planningManager = new PlanningManager(config.planning);
158
164
  }
165
+
166
+ // Task graph
167
+ this._taskGraph = new TaskGraph(config.taskGraph ?? {});
159
168
  }
160
169
 
161
170
  // --- Public API ---
@@ -253,6 +262,10 @@ export class Agent {
253
262
  return this._planningManager;
254
263
  }
255
264
 
265
+ get taskGraph(): TaskGraph {
266
+ return this._taskGraph;
267
+ }
268
+
256
269
  setSystemPrompt(prompt: string): void {
257
270
  this._state.systemPrompt = prompt;
258
271
  }
@@ -282,6 +295,7 @@ export class Agent {
282
295
  this._followUpQueue = [];
283
296
  await this._mcpAdapter?.dispose();
284
297
  this._backgroundTaskManager?.dispose();
298
+ this._taskGraph.dispose();
285
299
  this._wire.dispose();
286
300
  }
287
301
 
@@ -373,12 +387,21 @@ export class Agent {
373
387
  this._state.tools = [...this._state.tools, ...createPlanningTools(this._planningManager)];
374
388
 
375
389
  // Register nag provider into injection manager (create one if needed)
376
- const nagProvider = new PlanningNagProvider(this._planningManager);
377
- if (this._injectionManager) {
378
- this._injectionManager.addProvider(nagProvider);
379
- } else {
380
- this._injectionManager = new InjectionManager([nagProvider]);
381
- }
390
+ this._addInjectionProvider(new PlanningNagProvider(this._planningManager));
391
+ }
392
+
393
+ // Task graph tools + result auto-injection
394
+ this._state.tools = [...this._state.tools, ...createTaskGraphTools(this._taskGraph)];
395
+ if (this._config.taskGraph?.autoInjectResults !== false) {
396
+ this._addInjectionProvider(new TaskResultInjectionProvider(this._taskGraph));
397
+ }
398
+ }
399
+
400
+ private _addInjectionProvider(provider: DynamicInjectionProvider): void {
401
+ if (this._injectionManager) {
402
+ this._injectionManager.addProvider(provider);
403
+ } else {
404
+ this._injectionManager = new InjectionManager([provider]);
382
405
  }
383
406
  }
384
407
 
package/src/index.ts CHANGED
@@ -16,6 +16,7 @@ import type { SkillSource } from "./skills/types.js";
16
16
  import type { MCPServerConfig, MCPClient } from "./tools/mcp-adapter.js";
17
17
  import type { TaskFactory } from "./background/types.js";
18
18
  import type { PlanningConfig } from "./planning/types.js";
19
+ import type { TaskGraphConfig } from "./task-graph/types.js";
19
20
 
20
21
  export interface CreateAgentConfig {
21
22
  // Required
@@ -43,6 +44,7 @@ export interface CreateAgentConfig {
43
44
  wire?: { bufferSize?: number };
44
45
  backgroundTasks?: { factories?: Record<string, TaskFactory> };
45
46
  planning?: PlanningConfig;
47
+ taskGraph?: TaskGraphConfig;
46
48
 
47
49
  // Advanced: provide your own LLM provider
48
50
  provider?: LLMProvider;
@@ -74,6 +76,7 @@ export function createAgent(config: CreateAgentConfig): Agent {
74
76
  wire: config.wire,
75
77
  backgroundTasks: config.backgroundTasks,
76
78
  planning: config.planning,
79
+ taskGraph: config.taskGraph,
77
80
  });
78
81
  }
79
82
 
@@ -107,6 +110,9 @@ export { createBackgroundTaskTools } from "./background/tools.js";
107
110
  export { PlanningManager } from "./planning/planning-manager.js";
108
111
  export { createPlanningTools } from "./planning/tools.js";
109
112
  export { PlanningNagProvider } from "./planning/nag-injection.js";
113
+ export { TaskGraph } from "./task-graph/task-graph.js";
114
+ export { createTaskGraphTools } from "./task-graph/tools.js";
115
+ export { TaskResultInjectionProvider } from "./task-graph/result-injection.js";
110
116
 
111
117
  // Core types
112
118
  export type {
@@ -243,3 +249,12 @@ export type {
243
249
  } from "./planning/types.js";
244
250
 
245
251
  export { PLANNING_TOOL_NAMES } from "./planning/types.js";
252
+ export { TASK_GRAPH_TOOL_NAMES } from "./task-graph/types.js";
253
+
254
+ // Task graph types
255
+ export type {
256
+ TaskGraphConfig,
257
+ TaskNode,
258
+ TaskStatus,
259
+ CompletedTaskResult,
260
+ } from "./task-graph/types.js";
@@ -0,0 +1,29 @@
1
+ // Auto-inject completed background task results before each LLM call
2
+
3
+ import type { DynamicInjectionProvider, DynamicInjection } from "../injection/types.js";
4
+ import type { AgentMessage } from "../types.js";
5
+ import type { TaskGraph } from "./task-graph.js";
6
+
7
+ export class TaskResultInjectionProvider implements DynamicInjectionProvider {
8
+ constructor(private _graph: TaskGraph) {}
9
+
10
+ async getInjections(_history: AgentMessage[]): Promise<DynamicInjection[]> {
11
+ const completed = this._graph.drainCompleted();
12
+ if (completed.length === 0) return [];
13
+
14
+ const lines = completed.map((c) => {
15
+ const status = c.status === "completed" ? "completed" : "FAILED";
16
+ const unblocked = c.unblockedTasks.length > 0
17
+ ? ` → unblocked: ${c.unblockedTasks.join(", ")}`
18
+ : "";
19
+ return `[task:${c.taskId}] ${c.subject} — ${status}: ${c.result}${unblocked}`;
20
+ });
21
+
22
+ return [
23
+ {
24
+ type: "task-results",
25
+ content: `<background-results>\n${lines.join("\n")}\n</background-results>`,
26
+ },
27
+ ];
28
+ }
29
+ }