@renxqoo/renx-code 0.0.4 → 0.0.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (209) hide show
  1. package/bin/renx.cjs +16 -0
  2. package/package.json +2 -45
  3. package/src/agent/runtime/runtime.context-usage.test.ts +4 -5
  4. package/src/agent/runtime/runtime.error-handling.test.ts +4 -5
  5. package/src/agent/runtime/runtime.test.ts +7 -4
  6. package/src/agent/runtime/runtime.ts +3 -9
  7. package/src/agent/runtime/runtime.usage-forwarding.test.ts +4 -5
  8. package/src/agent/runtime/source-modules.test.ts +16 -35
  9. package/src/agent/runtime/source-modules.ts +17 -0
  10. package/vendor/agent-root/src/agent/ENTERPRISE_ACCEPTANCE_CHECKLIST.md +95 -0
  11. package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.html +1345 -0
  12. package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.md +1353 -0
  13. package/vendor/agent-root/src/agent/ERROR_CONTRACT.md +60 -0
  14. package/vendor/agent-root/src/agent/TEST_COVERAGE_ANALYSIS.md +278 -0
  15. package/vendor/agent-root/src/agent/__test__/error-contract.test.ts +72 -0
  16. package/vendor/agent-root/src/agent/__test__/types.test.ts +137 -0
  17. package/vendor/agent-root/src/agent/agent/__test__/abort-runtime.test.ts +83 -0
  18. package/vendor/agent-root/src/agent/agent/__test__/callback-safety.test.ts +34 -0
  19. package/vendor/agent-root/src/agent/agent/__test__/compaction.test.ts +323 -0
  20. package/vendor/agent-root/src/agent/agent/__test__/concurrency.test.ts +290 -0
  21. package/vendor/agent-root/src/agent/agent/__test__/error-normalizer.test.ts +377 -0
  22. package/vendor/agent-root/src/agent/agent/__test__/error.test.ts +212 -0
  23. package/vendor/agent-root/src/agent/agent/__test__/fault-injection.test.ts +295 -0
  24. package/vendor/agent-root/src/agent/agent/__test__/index.test.ts +3607 -0
  25. package/vendor/agent-root/src/agent/agent/__test__/logger.test.ts +35 -0
  26. package/vendor/agent-root/src/agent/agent/__test__/message-utils.test.ts +517 -0
  27. package/vendor/agent-root/src/agent/agent/__test__/telemetry.test.ts +97 -0
  28. package/vendor/agent-root/src/agent/agent/__test__/timeout-budget.test.ts +479 -0
  29. package/vendor/agent-root/src/agent/agent/__test__/tool-call-merge.test.ts +80 -0
  30. package/vendor/agent-root/src/agent/agent/__test__/tool-execution-ledger.test.ts +76 -0
  31. package/vendor/agent-root/src/agent/agent/__test__/write-buffer.test.ts +173 -0
  32. package/vendor/agent-root/src/agent/agent/__test__/write-file-session.test.ts +109 -0
  33. package/vendor/agent-root/src/agent/agent/abort-runtime.ts +71 -0
  34. package/vendor/agent-root/src/agent/agent/callback-safety.ts +33 -0
  35. package/vendor/agent-root/src/agent/agent/compaction.ts +291 -0
  36. package/vendor/agent-root/src/agent/agent/concurrency.ts +103 -0
  37. package/vendor/agent-root/src/agent/agent/error-normalizer.ts +190 -0
  38. package/vendor/agent-root/src/agent/agent/error.ts +198 -0
  39. package/vendor/agent-root/src/agent/agent/index.ts +1772 -0
  40. package/vendor/agent-root/src/agent/agent/logger.ts +65 -0
  41. package/vendor/agent-root/src/agent/agent/message-utils.ts +101 -0
  42. package/vendor/agent-root/src/agent/agent/stream-events.ts +61 -0
  43. package/vendor/agent-root/src/agent/agent/telemetry.ts +123 -0
  44. package/vendor/agent-root/src/agent/agent/timeout-budget.ts +227 -0
  45. package/vendor/agent-root/src/agent/agent/tool-call-merge.ts +111 -0
  46. package/vendor/agent-root/src/agent/agent/tool-execution-ledger.ts +164 -0
  47. package/vendor/agent-root/src/agent/agent/write-buffer.ts +188 -0
  48. package/vendor/agent-root/src/agent/agent/write-file-session.ts +238 -0
  49. package/vendor/agent-root/src/agent/app/__test__/agent-app-service.test.ts +1053 -0
  50. package/vendor/agent-root/src/agent/app/__test__/minimal-agent-application.test.ts +158 -0
  51. package/vendor/agent-root/src/agent/app/__test__/sqlite-agent-app-store.test.ts +437 -0
  52. package/vendor/agent-root/src/agent/app/agent-app-service.ts +748 -0
  53. package/vendor/agent-root/src/agent/app/contracts.ts +109 -0
  54. package/vendor/agent-root/src/agent/app/index.ts +5 -0
  55. package/vendor/agent-root/src/agent/app/minimal-agent-application.ts +151 -0
  56. package/vendor/agent-root/src/agent/app/ports.ts +72 -0
  57. package/vendor/agent-root/src/agent/app/sqlite-agent-app-store.ts +1182 -0
  58. package/vendor/agent-root/src/agent/app/sqlite-client.ts +177 -0
  59. package/vendor/agent-root/src/agent/docs/cli-app-layer/00-README.md +36 -0
  60. package/vendor/agent-root/src/agent/docs/cli-app-layer/01-scope-and-goals.md +33 -0
  61. package/vendor/agent-root/src/agent/docs/cli-app-layer/02-architecture-overview.md +40 -0
  62. package/vendor/agent-root/src/agent/docs/cli-app-layer/03-domain-model-and-contracts.md +91 -0
  63. package/vendor/agent-root/src/agent/docs/cli-app-layer/04-ports-and-interfaces.md +116 -0
  64. package/vendor/agent-root/src/agent/docs/cli-app-layer/05-run-orchestration-and-state-machine.md +52 -0
  65. package/vendor/agent-root/src/agent/docs/cli-app-layer/06-cli-commands-and-ux.md +53 -0
  66. package/vendor/agent-root/src/agent/docs/cli-app-layer/07-storage-design-local.md +52 -0
  67. package/vendor/agent-root/src/agent/docs/cli-app-layer/08-error-and-observability.md +40 -0
  68. package/vendor/agent-root/src/agent/docs/cli-app-layer/09-security-and-policy-boundary.md +19 -0
  69. package/vendor/agent-root/src/agent/docs/cli-app-layer/10-test-plan-and-acceptance.md +28 -0
  70. package/vendor/agent-root/src/agent/docs/cli-app-layer/11-implementation-phases.md +26 -0
  71. package/vendor/agent-root/src/agent/docs/cli-app-layer/12-open-questions-and-risks.md +30 -0
  72. package/vendor/agent-root/src/agent/docs/cli-app-layer/13-sqlite-schema-fields-and-rationale.md +567 -0
  73. package/vendor/agent-root/src/agent/docs/cli-app-layer/14-project-flow-mermaid.md +583 -0
  74. package/vendor/agent-root/src/agent/docs/cli-app-layer/15-openclaw-style-project-blueprint.md +972 -0
  75. package/vendor/agent-root/src/agent/error-contract.ts +154 -0
  76. package/vendor/agent-root/src/agent/prompts/system.ts +246 -0
  77. package/vendor/agent-root/src/agent/prompts/system1.ts +208 -0
  78. package/vendor/agent-root/src/agent/storage/__test__/file-history-store.test.ts +98 -0
  79. package/vendor/agent-root/src/agent/storage/file-history-store.ts +313 -0
  80. package/vendor/agent-root/src/agent/storage/file-storage-config.ts +94 -0
  81. package/vendor/agent-root/src/agent/storage/file-system.ts +31 -0
  82. package/vendor/agent-root/src/agent/storage/file-write-service.ts +21 -0
  83. package/vendor/agent-root/src/agent/tool/__test__/base-tool.test.ts +413 -0
  84. package/vendor/agent-root/src/agent/tool/__test__/bash-policy.test.ts +356 -0
  85. package/vendor/agent-root/src/agent/tool/__test__/bash.mocked-coverage.test.ts +375 -0
  86. package/vendor/agent-root/src/agent/tool/__test__/bash.test.ts +372 -0
  87. package/vendor/agent-root/src/agent/tool/__test__/error.test.ts +108 -0
  88. package/vendor/agent-root/src/agent/tool/__test__/file-edit-tool.test.ts +258 -0
  89. package/vendor/agent-root/src/agent/tool/__test__/file-history-tools.test.ts +121 -0
  90. package/vendor/agent-root/src/agent/tool/__test__/file-read-tool.test.ts +210 -0
  91. package/vendor/agent-root/src/agent/tool/__test__/glob.test.ts +139 -0
  92. package/vendor/agent-root/src/agent/tool/__test__/grep.mocked-coverage.test.ts +456 -0
  93. package/vendor/agent-root/src/agent/tool/__test__/grep.test.ts +192 -0
  94. package/vendor/agent-root/src/agent/tool/__test__/lsp.test.ts +300 -0
  95. package/vendor/agent-root/src/agent/tool/__test__/outside-workspace-confirmation.test.ts +214 -0
  96. package/vendor/agent-root/src/agent/tool/__test__/path-security.test.ts +336 -0
  97. package/vendor/agent-root/src/agent/tool/__test__/skill-loader.test.ts +494 -0
  98. package/vendor/agent-root/src/agent/tool/__test__/skill-parser.test.ts +543 -0
  99. package/vendor/agent-root/src/agent/tool/__test__/skill-tool.test.ts +172 -0
  100. package/vendor/agent-root/src/agent/tool/__test__/task-concurrency-and-version.test.ts +116 -0
  101. package/vendor/agent-root/src/agent/tool/__test__/task-create-get-list-update.test.ts +267 -0
  102. package/vendor/agent-root/src/agent/tool/__test__/task-create.test.ts +519 -0
  103. package/vendor/agent-root/src/agent/tool/__test__/task-errors.test.ts +225 -0
  104. package/vendor/agent-root/src/agent/tool/__test__/task-output-blocking.test.ts +223 -0
  105. package/vendor/agent-root/src/agent/tool/__test__/task-output.test.ts +184 -0
  106. package/vendor/agent-root/src/agent/tool/__test__/task-parent-abort.test.ts +287 -0
  107. package/vendor/agent-root/src/agent/tool/__test__/task-real-runner-adapter.test.ts +190 -0
  108. package/vendor/agent-root/src/agent/tool/__test__/task-run-lifecycle.test.ts +352 -0
  109. package/vendor/agent-root/src/agent/tool/__test__/task-store-runner-branches.test.ts +395 -0
  110. package/vendor/agent-root/src/agent/tool/__test__/task-store.test.ts +391 -0
  111. package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config-integration.test.ts +176 -0
  112. package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config.test.ts +68 -0
  113. package/vendor/agent-root/src/agent/tool/__test__/task-tools-core-edges.test.ts +630 -0
  114. package/vendor/agent-root/src/agent/tool/__test__/task-tools-runtime-edges.test.ts +732 -0
  115. package/vendor/agent-root/src/agent/tool/__test__/task-types.test.ts +494 -0
  116. package/vendor/agent-root/src/agent/tool/__test__/task-utils-branches.test.ts +175 -0
  117. package/vendor/agent-root/src/agent/tool/__test__/tool-manager.test.ts +505 -0
  118. package/vendor/agent-root/src/agent/tool/__test__/types.test.ts +55 -0
  119. package/vendor/agent-root/src/agent/tool/__test__/web-fetch.test.ts +244 -0
  120. package/vendor/agent-root/src/agent/tool/__test__/web-search.test.ts +290 -0
  121. package/vendor/agent-root/src/agent/tool/__test__/write-file.test.ts +368 -0
  122. package/vendor/agent-root/src/agent/tool/base-tool.ts +345 -0
  123. package/vendor/agent-root/src/agent/tool/bash-policy.ts +636 -0
  124. package/vendor/agent-root/src/agent/tool/bash.ts +688 -0
  125. package/vendor/agent-root/src/agent/tool/error.ts +131 -0
  126. package/vendor/agent-root/src/agent/tool/file-edit-tool.ts +264 -0
  127. package/vendor/agent-root/src/agent/tool/file-history-list.ts +103 -0
  128. package/vendor/agent-root/src/agent/tool/file-history-restore.ts +149 -0
  129. package/vendor/agent-root/src/agent/tool/file-read-tool.ts +211 -0
  130. package/vendor/agent-root/src/agent/tool/glob.ts +171 -0
  131. package/vendor/agent-root/src/agent/tool/grep.ts +496 -0
  132. package/vendor/agent-root/src/agent/tool/lsp.ts +481 -0
  133. package/vendor/agent-root/src/agent/tool/path-security.ts +117 -0
  134. package/vendor/agent-root/src/agent/tool/search/common.ts +153 -0
  135. package/vendor/agent-root/src/agent/tool/skill/index.ts +13 -0
  136. package/vendor/agent-root/src/agent/tool/skill/loader.ts +229 -0
  137. package/vendor/agent-root/src/agent/tool/skill/parser.ts +124 -0
  138. package/vendor/agent-root/src/agent/tool/skill/types.ts +27 -0
  139. package/vendor/agent-root/src/agent/tool/skill-tool.ts +143 -0
  140. package/vendor/agent-root/src/agent/tool/task-create.ts +186 -0
  141. package/vendor/agent-root/src/agent/tool/task-errors.ts +42 -0
  142. package/vendor/agent-root/src/agent/tool/task-get.ts +116 -0
  143. package/vendor/agent-root/src/agent/tool/task-graph.ts +78 -0
  144. package/vendor/agent-root/src/agent/tool/task-list.ts +141 -0
  145. package/vendor/agent-root/src/agent/tool/task-mock-runner-adapter.ts +232 -0
  146. package/vendor/agent-root/src/agent/tool/task-output.ts +223 -0
  147. package/vendor/agent-root/src/agent/tool/task-parent-abort.ts +115 -0
  148. package/vendor/agent-root/src/agent/tool/task-real-runner-adapter.ts +336 -0
  149. package/vendor/agent-root/src/agent/tool/task-runner-adapter.ts +55 -0
  150. package/vendor/agent-root/src/agent/tool/task-stop.ts +187 -0
  151. package/vendor/agent-root/src/agent/tool/task-store.ts +217 -0
  152. package/vendor/agent-root/src/agent/tool/task-subagent-config.ts +149 -0
  153. package/vendor/agent-root/src/agent/tool/task-types.ts +264 -0
  154. package/vendor/agent-root/src/agent/tool/task-update.ts +315 -0
  155. package/vendor/agent-root/src/agent/tool/task.ts +209 -0
  156. package/vendor/agent-root/src/agent/tool/tool-manager.ts +362 -0
  157. package/vendor/agent-root/src/agent/tool/tool-prompts.ts +242 -0
  158. package/vendor/agent-root/src/agent/tool/types.ts +116 -0
  159. package/vendor/agent-root/src/agent/tool/web-fetch.ts +227 -0
  160. package/vendor/agent-root/src/agent/tool/web-search.ts +208 -0
  161. package/vendor/agent-root/src/agent/tool/write-file.ts +497 -0
  162. package/vendor/agent-root/src/agent/types.ts +232 -0
  163. package/vendor/agent-root/src/agent/utils/__tests__/index.test.ts +18 -0
  164. package/vendor/agent-root/src/agent/utils/__tests__/message-utils.test.ts +610 -0
  165. package/vendor/agent-root/src/agent/utils/__tests__/message.test.ts +223 -0
  166. package/vendor/agent-root/src/agent/utils/__tests__/token.test.ts +42 -0
  167. package/vendor/agent-root/src/agent/utils/index.ts +16 -0
  168. package/vendor/agent-root/src/agent/utils/message.ts +171 -0
  169. package/vendor/agent-root/src/agent/utils/token.ts +28 -0
  170. package/vendor/agent-root/src/config/__tests__/load-config-to-env.test.ts +129 -0
  171. package/vendor/agent-root/src/config/__tests__/loader.test.ts +247 -0
  172. package/vendor/agent-root/src/config/__tests__/runtime.test.ts +88 -0
  173. package/vendor/agent-root/src/config/index.ts +54 -0
  174. package/vendor/agent-root/src/config/loader.ts +431 -0
  175. package/vendor/agent-root/src/config/paths.ts +30 -0
  176. package/vendor/agent-root/src/config/runtime.ts +163 -0
  177. package/vendor/agent-root/src/config/types.ts +70 -0
  178. package/vendor/agent-root/src/logger/index.ts +57 -0
  179. package/vendor/agent-root/src/logger/logger.ts +819 -0
  180. package/vendor/agent-root/src/logger/types.ts +150 -0
  181. package/vendor/agent-root/src/providers/__tests__/errors.test.ts +441 -0
  182. package/vendor/agent-root/src/providers/__tests__/index.test.ts +16 -0
  183. package/vendor/agent-root/src/providers/__tests__/openai-compatible.options.test.ts +318 -0
  184. package/vendor/agent-root/src/providers/__tests__/openai-compatible.test.ts +600 -0
  185. package/vendor/agent-root/src/providers/__tests__/registry.test.ts +449 -0
  186. package/vendor/agent-root/src/providers/__tests__/responses-adapter.test.ts +298 -0
  187. package/vendor/agent-root/src/providers/adapters/__tests__/anthropic.test.ts +354 -0
  188. package/vendor/agent-root/src/providers/adapters/__tests__/kimi.test.ts +58 -0
  189. package/vendor/agent-root/src/providers/adapters/__tests__/standard.test.ts +261 -0
  190. package/vendor/agent-root/src/providers/adapters/anthropic.ts +572 -0
  191. package/vendor/agent-root/src/providers/adapters/base.ts +131 -0
  192. package/vendor/agent-root/src/providers/adapters/kimi.ts +48 -0
  193. package/vendor/agent-root/src/providers/adapters/responses.ts +732 -0
  194. package/vendor/agent-root/src/providers/adapters/standard.ts +120 -0
  195. package/vendor/agent-root/src/providers/http/__tests__/client.timeout.test.ts +313 -0
  196. package/vendor/agent-root/src/providers/http/client.ts +289 -0
  197. package/vendor/agent-root/src/providers/http/stream-parser.ts +109 -0
  198. package/vendor/agent-root/src/providers/index.ts +76 -0
  199. package/vendor/agent-root/src/providers/kimi-headers.ts +177 -0
  200. package/vendor/agent-root/src/providers/openai-compatible.ts +387 -0
  201. package/vendor/agent-root/src/providers/registry/model-config.ts +230 -0
  202. package/vendor/agent-root/src/providers/registry/provider-factory.ts +123 -0
  203. package/vendor/agent-root/src/providers/registry.ts +135 -0
  204. package/vendor/agent-root/src/providers/types/api.ts +284 -0
  205. package/vendor/agent-root/src/providers/types/config.ts +58 -0
  206. package/vendor/agent-root/src/providers/types/errors.ts +323 -0
  207. package/vendor/agent-root/src/providers/types/index.ts +72 -0
  208. package/vendor/agent-root/src/providers/types/provider.ts +45 -0
  209. package/vendor/agent-root/src/providers/types/registry.ts +88 -0
@@ -0,0 +1,186 @@
1
+ import { z } from 'zod';
2
+ import { BaseTool, type ToolResult } from './base-tool';
3
+ import { buildTaskFailure, buildTaskSuccess, parsePrefixedError } from './task-errors';
4
+ import { getTaskStore, type TaskStore } from './task-store';
5
+ import {
6
+ DEFAULT_RETRY_CONFIG,
7
+ createTaskId,
8
+ safeJsonClone,
9
+ type RetryConfig,
10
+ type TaskCheckpoint,
11
+ type TaskEntity,
12
+ type TaskTag,
13
+ type TaskPriority,
14
+ } from './task-types';
15
+ import { ensureGraphNode } from './task-graph';
16
+ import { TASK_CREATE_DESCRIPTION } from './tool-prompts';
17
+
18
+ const checkpointSchema = z
19
+ .object({
20
+ id: z.string().min(1).describe('Checkpoint identifier'),
21
+ name: z.string().min(1).describe('Checkpoint display name'),
22
+ completed: z.boolean().optional().describe('Whether checkpoint is already completed'),
23
+ })
24
+ .strict();
25
+
26
+ const retryConfigSchema = z
27
+ .object({
28
+ maxRetries: z.number().int().min(0).describe('Maximum retry attempts'),
29
+ retryDelayMs: z.number().int().min(0).describe('Initial delay between retries in milliseconds'),
30
+ backoffMultiplier: z.number().min(1).describe('Exponential backoff multiplier'),
31
+ retryOn: z.array(z.string().min(1)).describe('Error categories that should trigger retry'),
32
+ })
33
+ .strict();
34
+
35
+ const tagSchema = z
36
+ .object({
37
+ name: z.string().min(1).describe('Tag name'),
38
+ color: z.string().optional().describe('Optional color hint'),
39
+ category: z.string().optional().describe('Optional tag category'),
40
+ })
41
+ .strict();
42
+
43
+ const schema = z
44
+ .object({
45
+ namespace: z.string().min(1).optional().describe('Optional task namespace'),
46
+ subject: z.string().min(3).describe('A brief actionable title in imperative form'),
47
+ description: z
48
+ .string()
49
+ .min(10)
50
+ .describe('Detailed task description with context and acceptance criteria'),
51
+ active_form: z
52
+ .string()
53
+ .min(1)
54
+ .optional()
55
+ .describe('Present continuous form shown while task is in progress'),
56
+ priority: z.enum(['critical', 'high', 'normal', 'low']).optional().describe('Task priority'),
57
+ tags: z.array(tagSchema).optional().describe('Optional task tags'),
58
+ checkpoints: z
59
+ .array(checkpointSchema)
60
+ .optional()
61
+ .describe('Optional checkpoints for progress tracking'),
62
+ retry_config: retryConfigSchema.optional().describe('Optional retry behavior configuration'),
63
+ timeout_ms: z
64
+ .number()
65
+ .int()
66
+ .positive()
67
+ .optional()
68
+ .describe('Optional timeout budget in milliseconds'),
69
+ metadata: z.record(z.string(), z.unknown()).optional().describe('Optional metadata map'),
70
+ created_by: z.string().optional().describe('Optional actor identifier'),
71
+ })
72
+ .strict();
73
+
74
+ type TaskCreateArgs = z.infer<typeof schema>;
75
+
76
+ export interface TaskCreateToolOptions {
77
+ store?: TaskStore;
78
+ defaultNamespace?: string;
79
+ }
80
+
81
+ export class TaskCreateTool extends BaseTool<typeof schema> {
82
+ name = 'task_create';
83
+ description = TASK_CREATE_DESCRIPTION;
84
+ parameters = schema;
85
+
86
+ private readonly store: TaskStore;
87
+ private readonly defaultNamespace?: string;
88
+
89
+ constructor(options: TaskCreateToolOptions = {}) {
90
+ super();
91
+ this.store = options.store || getTaskStore();
92
+ this.defaultNamespace = options.defaultNamespace;
93
+ }
94
+
95
+ override getConcurrencyMode(): 'exclusive' {
96
+ return 'exclusive';
97
+ }
98
+
99
+ override getConcurrencyLockKey(args: TaskCreateArgs): string {
100
+ const namespace = args.namespace || this.defaultNamespace || 'default';
101
+ return `taskns:${namespace}`;
102
+ }
103
+
104
+ async execute(args: TaskCreateArgs): Promise<ToolResult> {
105
+ const namespace = args.namespace || this.defaultNamespace;
106
+
107
+ try {
108
+ const created = await this.store.updateState(namespace, (state) => {
109
+ const now = Date.now();
110
+ const normalizedSubject = args.subject.trim();
111
+
112
+ const duplicate = Object.values(state.tasks).find(
113
+ (task) =>
114
+ task.subject === normalizedSubject &&
115
+ task.status !== 'completed' &&
116
+ task.status !== 'cancelled' &&
117
+ task.status !== 'failed'
118
+ );
119
+ if (duplicate) {
120
+ throw new Error(
121
+ `TASK_DUPLICATE_SUBJECT: duplicate active task subject already exists: ${duplicate.id}`
122
+ );
123
+ }
124
+
125
+ const id = createTaskId();
126
+ const checkpointItems: TaskCheckpoint[] = (args.checkpoints || []).map((checkpoint) => ({
127
+ id: checkpoint.id,
128
+ name: checkpoint.name,
129
+ completed: checkpoint.completed || false,
130
+ }));
131
+ const retryConfig: RetryConfig = args.retry_config
132
+ ? safeJsonClone(args.retry_config as RetryConfig)
133
+ : safeJsonClone(DEFAULT_RETRY_CONFIG);
134
+ const tags: TaskTag[] = (args.tags || []).map((tag) => safeJsonClone(tag));
135
+
136
+ const task: TaskEntity = {
137
+ id,
138
+ subject: normalizedSubject,
139
+ description: args.description.trim(),
140
+ activeForm: args.active_form?.trim() || `${normalizedSubject} in progress`,
141
+ status: 'pending',
142
+ priority: (args.priority || 'normal') as TaskPriority,
143
+ owner: null,
144
+ blockedBy: [],
145
+ blocks: [],
146
+ progress: 0,
147
+ checkpoints: checkpointItems,
148
+ retryConfig,
149
+ retryCount: 0,
150
+ timeoutMs: args.timeout_ms,
151
+ tags,
152
+ metadata: safeJsonClone(args.metadata || {}),
153
+ history: [
154
+ {
155
+ timestamp: now,
156
+ action: 'created',
157
+ actor: args.created_by || null,
158
+ metadata: {
159
+ subject: normalizedSubject,
160
+ },
161
+ },
162
+ ],
163
+ createdAt: now,
164
+ updatedAt: now,
165
+ version: 1,
166
+ };
167
+
168
+ state.tasks[id] = task;
169
+ ensureGraphNode(state.graph, id);
170
+
171
+ return safeJsonClone(task);
172
+ });
173
+
174
+ return buildTaskSuccess({
175
+ namespace: this.store.normalizeNamespace(namespace),
176
+ task: created.result,
177
+ });
178
+ } catch (error) {
179
+ const message = error instanceof Error ? error.message : String(error);
180
+ const parsed = parsePrefixedError(message);
181
+ return buildTaskFailure(parsed.code, parsed.detail);
182
+ }
183
+ }
184
+ }
185
+
186
+ export default TaskCreateTool;
@@ -0,0 +1,42 @@
1
+ import { ToolExecutionError } from './error';
2
+ import type { ToolResult } from './base-tool';
3
+
4
+ export function buildTaskFailure(
5
+ code: string,
6
+ message: string,
7
+ details?: Record<string, unknown>
8
+ ): ToolResult {
9
+ const output = `${code}: ${message}`;
10
+ return {
11
+ success: false,
12
+ output,
13
+ error: new ToolExecutionError(output),
14
+ metadata: {
15
+ error: code,
16
+ message,
17
+ ...(details || {}),
18
+ },
19
+ };
20
+ }
21
+
22
+ export function buildTaskSuccess(payload: Record<string, unknown>): ToolResult {
23
+ return {
24
+ success: true,
25
+ output: JSON.stringify(payload),
26
+ metadata: payload,
27
+ };
28
+ }
29
+
30
+ export function parsePrefixedError(message: string): { code: string; detail: string } {
31
+ const matched = message.match(/^([A-Z][A-Z0-9_]{2,}):\s*(.*)$/);
32
+ if (!matched) {
33
+ return {
34
+ code: 'TASK_OPERATION_FAILED',
35
+ detail: message,
36
+ };
37
+ }
38
+ return {
39
+ code: matched[1],
40
+ detail: matched[2],
41
+ };
42
+ }
@@ -0,0 +1,116 @@
1
+ import { z } from 'zod';
2
+ import { BaseTool, type ToolResult } from './base-tool';
3
+ import { buildTaskFailure, buildTaskSuccess } from './task-errors';
4
+ import { getTaskStore, type TaskStore } from './task-store';
5
+ import { evaluateTaskCanStart, safeJsonClone, type TaskEntity } from './task-types';
6
+ import { TASK_GET_DESCRIPTION } from './tool-prompts';
7
+
8
+ const schema = z
9
+ .object({
10
+ namespace: z.string().min(1).optional().describe('Optional task namespace'),
11
+ task_id: z.string().min(1).describe('Task identifier to retrieve'),
12
+ include_history: z.boolean().optional().describe('Include full history entries when true'),
13
+ })
14
+ .strict();
15
+
16
+ type TaskGetArgs = z.infer<typeof schema>;
17
+
18
+ export interface TaskGetToolOptions {
19
+ store?: TaskStore;
20
+ defaultNamespace?: string;
21
+ }
22
+
23
+ function calculateCheckpointProgress(task: TaskEntity): number {
24
+ if (!task.checkpoints || task.checkpoints.length === 0) {
25
+ return 0;
26
+ }
27
+ const completedCount = task.checkpoints.filter((checkpoint) => checkpoint.completed).length;
28
+ return Math.round((completedCount / task.checkpoints.length) * 100);
29
+ }
30
+
31
+ export class TaskGetTool extends BaseTool<typeof schema> {
32
+ name = 'task_get';
33
+ description = TASK_GET_DESCRIPTION;
34
+ parameters = schema;
35
+
36
+ private readonly store: TaskStore;
37
+ private readonly defaultNamespace?: string;
38
+
39
+ constructor(options: TaskGetToolOptions = {}) {
40
+ super();
41
+ this.store = options.store || getTaskStore();
42
+ this.defaultNamespace = options.defaultNamespace;
43
+ }
44
+
45
+ override getConcurrencyMode(): 'parallel-safe' {
46
+ return 'parallel-safe';
47
+ }
48
+
49
+ override getConcurrencyLockKey(args: TaskGetArgs): string {
50
+ const namespace = args.namespace || this.defaultNamespace || 'default';
51
+ return `taskns:${namespace}:task:${args.task_id}`;
52
+ }
53
+
54
+ async execute(args: TaskGetArgs): Promise<ToolResult> {
55
+ const namespace = args.namespace || this.defaultNamespace;
56
+ const normalizedNamespace = this.store.normalizeNamespace(namespace);
57
+ const state = await this.store.getState(normalizedNamespace);
58
+
59
+ const task = state.tasks[args.task_id];
60
+ if (!task) {
61
+ return buildTaskFailure('TASK_NOT_FOUND', `task not found: ${args.task_id}`, {
62
+ namespace: normalizedNamespace,
63
+ task_id: args.task_id,
64
+ });
65
+ }
66
+
67
+ const blockers = task.blockedBy.map((blockerId) => {
68
+ const blocker = state.tasks[blockerId];
69
+ return blocker
70
+ ? {
71
+ id: blocker.id,
72
+ subject: blocker.subject,
73
+ status: blocker.status,
74
+ }
75
+ : {
76
+ id: blockerId,
77
+ subject: '(missing task)',
78
+ status: 'missing',
79
+ };
80
+ });
81
+
82
+ const blockedTasks = task.blocks.map((blockedTaskId) => {
83
+ const blockedTask = state.tasks[blockedTaskId];
84
+ return blockedTask
85
+ ? {
86
+ id: blockedTask.id,
87
+ subject: blockedTask.subject,
88
+ status: blockedTask.status,
89
+ }
90
+ : {
91
+ id: blockedTaskId,
92
+ subject: '(missing task)',
93
+ status: 'missing',
94
+ };
95
+ });
96
+
97
+ const checkpointProgress = calculateCheckpointProgress(task);
98
+ const canStart = evaluateTaskCanStart(task, state.tasks);
99
+ const detail = {
100
+ ...safeJsonClone(task),
101
+ blockers,
102
+ blocked_tasks: blockedTasks,
103
+ can_start: canStart,
104
+ checkpoint_progress: checkpointProgress,
105
+ effective_progress: Math.max(task.progress, checkpointProgress),
106
+ history: args.include_history === true ? safeJsonClone(task.history) : undefined,
107
+ };
108
+
109
+ return buildTaskSuccess({
110
+ namespace: normalizedNamespace,
111
+ task: detail,
112
+ });
113
+ }
114
+ }
115
+
116
+ export default TaskGetTool;
@@ -0,0 +1,78 @@
1
+ import type { DependencyGraphState } from './task-types';
2
+
3
+ function ensureUniquePush(values: string[], next: string): void {
4
+ if (!values.includes(next)) {
5
+ values.push(next);
6
+ }
7
+ }
8
+
9
+ export function ensureGraphNode(graph: DependencyGraphState, taskId: string): void {
10
+ if (!graph.adjacency[taskId]) {
11
+ graph.adjacency[taskId] = [];
12
+ }
13
+ if (!graph.reverse[taskId]) {
14
+ graph.reverse[taskId] = [];
15
+ }
16
+ }
17
+
18
+ export function addDependencyEdge(
19
+ graph: DependencyGraphState,
20
+ blockerId: string,
21
+ dependentId: string
22
+ ): void {
23
+ ensureGraphNode(graph, blockerId);
24
+ ensureGraphNode(graph, dependentId);
25
+ ensureUniquePush(graph.adjacency[blockerId], dependentId);
26
+ ensureUniquePush(graph.reverse[dependentId], blockerId);
27
+ }
28
+
29
+ export function removeDependencyEdge(
30
+ graph: DependencyGraphState,
31
+ blockerId: string,
32
+ dependentId: string
33
+ ): void {
34
+ ensureGraphNode(graph, blockerId);
35
+ ensureGraphNode(graph, dependentId);
36
+ graph.adjacency[blockerId] = graph.adjacency[blockerId].filter((id) => id !== dependentId);
37
+ graph.reverse[dependentId] = graph.reverse[dependentId].filter((id) => id !== blockerId);
38
+ }
39
+
40
+ export function hasPath(graph: DependencyGraphState, fromId: string, toId: string): boolean {
41
+ if (fromId === toId) {
42
+ return true;
43
+ }
44
+
45
+ const visited = new Set<string>();
46
+ const queue: string[] = [fromId];
47
+
48
+ while (queue.length > 0) {
49
+ const current = queue.shift() as string;
50
+ if (current === toId) {
51
+ return true;
52
+ }
53
+ if (visited.has(current)) {
54
+ continue;
55
+ }
56
+ visited.add(current);
57
+ const nextNodes = graph.adjacency[current] || [];
58
+ for (const next of nextNodes) {
59
+ if (!visited.has(next)) {
60
+ queue.push(next);
61
+ }
62
+ }
63
+ }
64
+
65
+ return false;
66
+ }
67
+
68
+ export function wouldCreateCycle(
69
+ graph: DependencyGraphState,
70
+ blockerId: string,
71
+ dependentId: string
72
+ ): boolean {
73
+ if (blockerId === dependentId) {
74
+ return true;
75
+ }
76
+ // If dependent can already reach blocker, blocker -> dependent creates a cycle.
77
+ return hasPath(graph, dependentId, blockerId);
78
+ }
@@ -0,0 +1,141 @@
1
+ import { z } from 'zod';
2
+ import { BaseTool, type ToolResult } from './base-tool';
3
+ import { buildTaskSuccess } from './task-errors';
4
+ import { getTaskStore, type TaskStore } from './task-store';
5
+ import { safeJsonClone, type TaskEntity, type TaskPriority, type TaskStatus } from './task-types';
6
+ import { TASK_LIST_DESCRIPTION } from './tool-prompts';
7
+
8
+ const schema = z
9
+ .object({
10
+ namespace: z.string().min(1).optional().describe('Optional task namespace'),
11
+ statuses: z
12
+ .array(z.enum(['pending', 'in_progress', 'completed', 'cancelled', 'failed']))
13
+ .optional()
14
+ .describe('Optional status filter'),
15
+ owner: z.string().min(1).optional().describe('Optional owner filter'),
16
+ tag: z.string().min(1).optional().describe('Optional tag-name filter'),
17
+ include_history: z.boolean().optional().describe('Include history for each returned task'),
18
+ })
19
+ .strict();
20
+
21
+ type TaskListArgs = z.infer<typeof schema>;
22
+
23
+ export interface TaskListToolOptions {
24
+ store?: TaskStore;
25
+ defaultNamespace?: string;
26
+ }
27
+
28
+ interface TaskSummary {
29
+ id: string;
30
+ subject: string;
31
+ status: TaskStatus;
32
+ priority: TaskPriority;
33
+ owner: string | null;
34
+ blocked_by: string[];
35
+ blocks: string[];
36
+ progress: number;
37
+ is_blocked: boolean;
38
+ can_be_claimed: boolean;
39
+ created_at: number;
40
+ updated_at: number;
41
+ history?: unknown;
42
+ }
43
+
44
+ function computeSummary(task: TaskEntity, taskMap: Record<string, TaskEntity>): TaskSummary {
45
+ const blockedByCount = task.blockedBy.filter((blockerId) => {
46
+ const blocker = taskMap[blockerId];
47
+ return !blocker || blocker.status !== 'completed';
48
+ }).length;
49
+ return {
50
+ id: task.id,
51
+ subject: task.subject,
52
+ status: task.status,
53
+ priority: task.priority,
54
+ owner: task.owner,
55
+ blocked_by: safeJsonClone(task.blockedBy),
56
+ blocks: safeJsonClone(task.blocks),
57
+ progress: task.progress,
58
+ is_blocked: blockedByCount > 0,
59
+ can_be_claimed: task.status === 'pending' && blockedByCount === 0 && !task.owner,
60
+ created_at: task.createdAt,
61
+ updated_at: task.updatedAt,
62
+ };
63
+ }
64
+
65
+ function rankTask(summary: TaskSummary): number {
66
+ if (summary.can_be_claimed && summary.priority === 'critical') return 0;
67
+ if (summary.status === 'in_progress') return 1;
68
+ if (summary.can_be_claimed && summary.priority === 'high') return 2;
69
+ if (summary.can_be_claimed && summary.priority === 'normal') return 3;
70
+ if (summary.can_be_claimed && summary.priority === 'low') return 4;
71
+ if (summary.is_blocked) return 5;
72
+ if (summary.status === 'completed') return 6;
73
+ if (summary.status === 'cancelled' || summary.status === 'failed') return 7;
74
+ return 8;
75
+ }
76
+
77
+ export class TaskListTool extends BaseTool<typeof schema> {
78
+ name = 'task_list';
79
+ description = TASK_LIST_DESCRIPTION;
80
+ parameters = schema;
81
+
82
+ private readonly store: TaskStore;
83
+ private readonly defaultNamespace?: string;
84
+
85
+ constructor(options: TaskListToolOptions = {}) {
86
+ super();
87
+ this.store = options.store || getTaskStore();
88
+ this.defaultNamespace = options.defaultNamespace;
89
+ }
90
+
91
+ override getConcurrencyMode(): 'parallel-safe' {
92
+ return 'parallel-safe';
93
+ }
94
+
95
+ override getConcurrencyLockKey(args: TaskListArgs): string {
96
+ const namespace = args.namespace || this.defaultNamespace || 'default';
97
+ return `taskns:${namespace}:list`;
98
+ }
99
+
100
+ async execute(args: TaskListArgs): Promise<ToolResult> {
101
+ const namespace = args.namespace || this.defaultNamespace;
102
+ const normalizedNamespace = this.store.normalizeNamespace(namespace);
103
+ const state = await this.store.getState(normalizedNamespace);
104
+ const tasks = Object.values(state.tasks);
105
+
106
+ let filtered = tasks;
107
+ if (args.statuses && args.statuses.length > 0) {
108
+ const statusSet = new Set(args.statuses);
109
+ filtered = filtered.filter((task) => statusSet.has(task.status));
110
+ }
111
+ if (args.owner) {
112
+ filtered = filtered.filter((task) => task.owner === args.owner);
113
+ }
114
+ if (args.tag) {
115
+ filtered = filtered.filter((task) => task.tags.some((tag) => tag.name === args.tag));
116
+ }
117
+
118
+ const summaries = filtered.map((task) => {
119
+ const summary = computeSummary(task, state.tasks);
120
+ if (args.include_history === true) {
121
+ summary.history = safeJsonClone(task.history);
122
+ }
123
+ return summary;
124
+ });
125
+
126
+ summaries.sort((a, b) => {
127
+ const rankDelta = rankTask(a) - rankTask(b);
128
+ if (rankDelta !== 0) return rankDelta;
129
+ if (a.created_at !== b.created_at) return a.created_at - b.created_at;
130
+ return a.id.localeCompare(b.id);
131
+ });
132
+
133
+ return buildTaskSuccess({
134
+ namespace: normalizedNamespace,
135
+ total: summaries.length,
136
+ tasks: summaries,
137
+ });
138
+ }
139
+ }
140
+
141
+ export default TaskListTool;