@renxqoo/renx-code 0.0.4 → 0.0.6

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 (210) hide show
  1. package/README.md +82 -51
  2. package/bin/renx.cjs +16 -0
  3. package/package.json +2 -45
  4. package/src/agent/runtime/runtime.context-usage.test.ts +4 -5
  5. package/src/agent/runtime/runtime.error-handling.test.ts +4 -5
  6. package/src/agent/runtime/runtime.test.ts +7 -4
  7. package/src/agent/runtime/runtime.ts +3 -9
  8. package/src/agent/runtime/runtime.usage-forwarding.test.ts +4 -5
  9. package/src/agent/runtime/source-modules.test.ts +16 -35
  10. package/src/agent/runtime/source-modules.ts +17 -0
  11. package/vendor/agent-root/src/agent/ENTERPRISE_ACCEPTANCE_CHECKLIST.md +95 -0
  12. package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.html +1345 -0
  13. package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.md +1353 -0
  14. package/vendor/agent-root/src/agent/ERROR_CONTRACT.md +60 -0
  15. package/vendor/agent-root/src/agent/TEST_COVERAGE_ANALYSIS.md +278 -0
  16. package/vendor/agent-root/src/agent/__test__/error-contract.test.ts +72 -0
  17. package/vendor/agent-root/src/agent/__test__/types.test.ts +137 -0
  18. package/vendor/agent-root/src/agent/agent/__test__/abort-runtime.test.ts +83 -0
  19. package/vendor/agent-root/src/agent/agent/__test__/callback-safety.test.ts +34 -0
  20. package/vendor/agent-root/src/agent/agent/__test__/compaction.test.ts +323 -0
  21. package/vendor/agent-root/src/agent/agent/__test__/concurrency.test.ts +290 -0
  22. package/vendor/agent-root/src/agent/agent/__test__/error-normalizer.test.ts +377 -0
  23. package/vendor/agent-root/src/agent/agent/__test__/error.test.ts +212 -0
  24. package/vendor/agent-root/src/agent/agent/__test__/fault-injection.test.ts +295 -0
  25. package/vendor/agent-root/src/agent/agent/__test__/index.test.ts +3607 -0
  26. package/vendor/agent-root/src/agent/agent/__test__/logger.test.ts +35 -0
  27. package/vendor/agent-root/src/agent/agent/__test__/message-utils.test.ts +517 -0
  28. package/vendor/agent-root/src/agent/agent/__test__/telemetry.test.ts +97 -0
  29. package/vendor/agent-root/src/agent/agent/__test__/timeout-budget.test.ts +479 -0
  30. package/vendor/agent-root/src/agent/agent/__test__/tool-call-merge.test.ts +80 -0
  31. package/vendor/agent-root/src/agent/agent/__test__/tool-execution-ledger.test.ts +76 -0
  32. package/vendor/agent-root/src/agent/agent/__test__/write-buffer.test.ts +173 -0
  33. package/vendor/agent-root/src/agent/agent/__test__/write-file-session.test.ts +109 -0
  34. package/vendor/agent-root/src/agent/agent/abort-runtime.ts +71 -0
  35. package/vendor/agent-root/src/agent/agent/callback-safety.ts +33 -0
  36. package/vendor/agent-root/src/agent/agent/compaction.ts +291 -0
  37. package/vendor/agent-root/src/agent/agent/concurrency.ts +103 -0
  38. package/vendor/agent-root/src/agent/agent/error-normalizer.ts +190 -0
  39. package/vendor/agent-root/src/agent/agent/error.ts +198 -0
  40. package/vendor/agent-root/src/agent/agent/index.ts +1772 -0
  41. package/vendor/agent-root/src/agent/agent/logger.ts +65 -0
  42. package/vendor/agent-root/src/agent/agent/message-utils.ts +101 -0
  43. package/vendor/agent-root/src/agent/agent/stream-events.ts +61 -0
  44. package/vendor/agent-root/src/agent/agent/telemetry.ts +123 -0
  45. package/vendor/agent-root/src/agent/agent/timeout-budget.ts +227 -0
  46. package/vendor/agent-root/src/agent/agent/tool-call-merge.ts +111 -0
  47. package/vendor/agent-root/src/agent/agent/tool-execution-ledger.ts +164 -0
  48. package/vendor/agent-root/src/agent/agent/write-buffer.ts +188 -0
  49. package/vendor/agent-root/src/agent/agent/write-file-session.ts +238 -0
  50. package/vendor/agent-root/src/agent/app/__test__/agent-app-service.test.ts +1053 -0
  51. package/vendor/agent-root/src/agent/app/__test__/minimal-agent-application.test.ts +158 -0
  52. package/vendor/agent-root/src/agent/app/__test__/sqlite-agent-app-store.test.ts +437 -0
  53. package/vendor/agent-root/src/agent/app/agent-app-service.ts +748 -0
  54. package/vendor/agent-root/src/agent/app/contracts.ts +109 -0
  55. package/vendor/agent-root/src/agent/app/index.ts +5 -0
  56. package/vendor/agent-root/src/agent/app/minimal-agent-application.ts +151 -0
  57. package/vendor/agent-root/src/agent/app/ports.ts +72 -0
  58. package/vendor/agent-root/src/agent/app/sqlite-agent-app-store.ts +1182 -0
  59. package/vendor/agent-root/src/agent/app/sqlite-client.ts +177 -0
  60. package/vendor/agent-root/src/agent/docs/cli-app-layer/00-README.md +36 -0
  61. package/vendor/agent-root/src/agent/docs/cli-app-layer/01-scope-and-goals.md +33 -0
  62. package/vendor/agent-root/src/agent/docs/cli-app-layer/02-architecture-overview.md +40 -0
  63. package/vendor/agent-root/src/agent/docs/cli-app-layer/03-domain-model-and-contracts.md +91 -0
  64. package/vendor/agent-root/src/agent/docs/cli-app-layer/04-ports-and-interfaces.md +116 -0
  65. package/vendor/agent-root/src/agent/docs/cli-app-layer/05-run-orchestration-and-state-machine.md +52 -0
  66. package/vendor/agent-root/src/agent/docs/cli-app-layer/06-cli-commands-and-ux.md +53 -0
  67. package/vendor/agent-root/src/agent/docs/cli-app-layer/07-storage-design-local.md +52 -0
  68. package/vendor/agent-root/src/agent/docs/cli-app-layer/08-error-and-observability.md +40 -0
  69. package/vendor/agent-root/src/agent/docs/cli-app-layer/09-security-and-policy-boundary.md +19 -0
  70. package/vendor/agent-root/src/agent/docs/cli-app-layer/10-test-plan-and-acceptance.md +28 -0
  71. package/vendor/agent-root/src/agent/docs/cli-app-layer/11-implementation-phases.md +26 -0
  72. package/vendor/agent-root/src/agent/docs/cli-app-layer/12-open-questions-and-risks.md +30 -0
  73. package/vendor/agent-root/src/agent/docs/cli-app-layer/13-sqlite-schema-fields-and-rationale.md +567 -0
  74. package/vendor/agent-root/src/agent/docs/cli-app-layer/14-project-flow-mermaid.md +583 -0
  75. package/vendor/agent-root/src/agent/docs/cli-app-layer/15-openclaw-style-project-blueprint.md +972 -0
  76. package/vendor/agent-root/src/agent/error-contract.ts +154 -0
  77. package/vendor/agent-root/src/agent/prompts/system.ts +246 -0
  78. package/vendor/agent-root/src/agent/prompts/system1.ts +208 -0
  79. package/vendor/agent-root/src/agent/storage/__test__/file-history-store.test.ts +98 -0
  80. package/vendor/agent-root/src/agent/storage/file-history-store.ts +313 -0
  81. package/vendor/agent-root/src/agent/storage/file-storage-config.ts +94 -0
  82. package/vendor/agent-root/src/agent/storage/file-system.ts +31 -0
  83. package/vendor/agent-root/src/agent/storage/file-write-service.ts +21 -0
  84. package/vendor/agent-root/src/agent/tool/__test__/base-tool.test.ts +413 -0
  85. package/vendor/agent-root/src/agent/tool/__test__/bash-policy.test.ts +356 -0
  86. package/vendor/agent-root/src/agent/tool/__test__/bash.mocked-coverage.test.ts +375 -0
  87. package/vendor/agent-root/src/agent/tool/__test__/bash.test.ts +372 -0
  88. package/vendor/agent-root/src/agent/tool/__test__/error.test.ts +108 -0
  89. package/vendor/agent-root/src/agent/tool/__test__/file-edit-tool.test.ts +258 -0
  90. package/vendor/agent-root/src/agent/tool/__test__/file-history-tools.test.ts +121 -0
  91. package/vendor/agent-root/src/agent/tool/__test__/file-read-tool.test.ts +210 -0
  92. package/vendor/agent-root/src/agent/tool/__test__/glob.test.ts +139 -0
  93. package/vendor/agent-root/src/agent/tool/__test__/grep.mocked-coverage.test.ts +456 -0
  94. package/vendor/agent-root/src/agent/tool/__test__/grep.test.ts +192 -0
  95. package/vendor/agent-root/src/agent/tool/__test__/lsp.test.ts +300 -0
  96. package/vendor/agent-root/src/agent/tool/__test__/outside-workspace-confirmation.test.ts +214 -0
  97. package/vendor/agent-root/src/agent/tool/__test__/path-security.test.ts +336 -0
  98. package/vendor/agent-root/src/agent/tool/__test__/skill-loader.test.ts +494 -0
  99. package/vendor/agent-root/src/agent/tool/__test__/skill-parser.test.ts +543 -0
  100. package/vendor/agent-root/src/agent/tool/__test__/skill-tool.test.ts +172 -0
  101. package/vendor/agent-root/src/agent/tool/__test__/task-concurrency-and-version.test.ts +116 -0
  102. package/vendor/agent-root/src/agent/tool/__test__/task-create-get-list-update.test.ts +267 -0
  103. package/vendor/agent-root/src/agent/tool/__test__/task-create.test.ts +519 -0
  104. package/vendor/agent-root/src/agent/tool/__test__/task-errors.test.ts +225 -0
  105. package/vendor/agent-root/src/agent/tool/__test__/task-output-blocking.test.ts +223 -0
  106. package/vendor/agent-root/src/agent/tool/__test__/task-output.test.ts +184 -0
  107. package/vendor/agent-root/src/agent/tool/__test__/task-parent-abort.test.ts +287 -0
  108. package/vendor/agent-root/src/agent/tool/__test__/task-real-runner-adapter.test.ts +190 -0
  109. package/vendor/agent-root/src/agent/tool/__test__/task-run-lifecycle.test.ts +352 -0
  110. package/vendor/agent-root/src/agent/tool/__test__/task-store-runner-branches.test.ts +395 -0
  111. package/vendor/agent-root/src/agent/tool/__test__/task-store.test.ts +391 -0
  112. package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config-integration.test.ts +176 -0
  113. package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config.test.ts +68 -0
  114. package/vendor/agent-root/src/agent/tool/__test__/task-tools-core-edges.test.ts +630 -0
  115. package/vendor/agent-root/src/agent/tool/__test__/task-tools-runtime-edges.test.ts +732 -0
  116. package/vendor/agent-root/src/agent/tool/__test__/task-types.test.ts +494 -0
  117. package/vendor/agent-root/src/agent/tool/__test__/task-utils-branches.test.ts +175 -0
  118. package/vendor/agent-root/src/agent/tool/__test__/tool-manager.test.ts +505 -0
  119. package/vendor/agent-root/src/agent/tool/__test__/types.test.ts +55 -0
  120. package/vendor/agent-root/src/agent/tool/__test__/web-fetch.test.ts +244 -0
  121. package/vendor/agent-root/src/agent/tool/__test__/web-search.test.ts +290 -0
  122. package/vendor/agent-root/src/agent/tool/__test__/write-file.test.ts +368 -0
  123. package/vendor/agent-root/src/agent/tool/base-tool.ts +345 -0
  124. package/vendor/agent-root/src/agent/tool/bash-policy.ts +636 -0
  125. package/vendor/agent-root/src/agent/tool/bash.ts +688 -0
  126. package/vendor/agent-root/src/agent/tool/error.ts +131 -0
  127. package/vendor/agent-root/src/agent/tool/file-edit-tool.ts +264 -0
  128. package/vendor/agent-root/src/agent/tool/file-history-list.ts +103 -0
  129. package/vendor/agent-root/src/agent/tool/file-history-restore.ts +149 -0
  130. package/vendor/agent-root/src/agent/tool/file-read-tool.ts +211 -0
  131. package/vendor/agent-root/src/agent/tool/glob.ts +171 -0
  132. package/vendor/agent-root/src/agent/tool/grep.ts +496 -0
  133. package/vendor/agent-root/src/agent/tool/lsp.ts +481 -0
  134. package/vendor/agent-root/src/agent/tool/path-security.ts +117 -0
  135. package/vendor/agent-root/src/agent/tool/search/common.ts +153 -0
  136. package/vendor/agent-root/src/agent/tool/skill/index.ts +13 -0
  137. package/vendor/agent-root/src/agent/tool/skill/loader.ts +229 -0
  138. package/vendor/agent-root/src/agent/tool/skill/parser.ts +124 -0
  139. package/vendor/agent-root/src/agent/tool/skill/types.ts +27 -0
  140. package/vendor/agent-root/src/agent/tool/skill-tool.ts +143 -0
  141. package/vendor/agent-root/src/agent/tool/task-create.ts +186 -0
  142. package/vendor/agent-root/src/agent/tool/task-errors.ts +42 -0
  143. package/vendor/agent-root/src/agent/tool/task-get.ts +116 -0
  144. package/vendor/agent-root/src/agent/tool/task-graph.ts +78 -0
  145. package/vendor/agent-root/src/agent/tool/task-list.ts +141 -0
  146. package/vendor/agent-root/src/agent/tool/task-mock-runner-adapter.ts +232 -0
  147. package/vendor/agent-root/src/agent/tool/task-output.ts +223 -0
  148. package/vendor/agent-root/src/agent/tool/task-parent-abort.ts +115 -0
  149. package/vendor/agent-root/src/agent/tool/task-real-runner-adapter.ts +336 -0
  150. package/vendor/agent-root/src/agent/tool/task-runner-adapter.ts +55 -0
  151. package/vendor/agent-root/src/agent/tool/task-stop.ts +187 -0
  152. package/vendor/agent-root/src/agent/tool/task-store.ts +217 -0
  153. package/vendor/agent-root/src/agent/tool/task-subagent-config.ts +149 -0
  154. package/vendor/agent-root/src/agent/tool/task-types.ts +264 -0
  155. package/vendor/agent-root/src/agent/tool/task-update.ts +315 -0
  156. package/vendor/agent-root/src/agent/tool/task.ts +209 -0
  157. package/vendor/agent-root/src/agent/tool/tool-manager.ts +362 -0
  158. package/vendor/agent-root/src/agent/tool/tool-prompts.ts +242 -0
  159. package/vendor/agent-root/src/agent/tool/types.ts +116 -0
  160. package/vendor/agent-root/src/agent/tool/web-fetch.ts +227 -0
  161. package/vendor/agent-root/src/agent/tool/web-search.ts +208 -0
  162. package/vendor/agent-root/src/agent/tool/write-file.ts +497 -0
  163. package/vendor/agent-root/src/agent/types.ts +232 -0
  164. package/vendor/agent-root/src/agent/utils/__tests__/index.test.ts +18 -0
  165. package/vendor/agent-root/src/agent/utils/__tests__/message-utils.test.ts +610 -0
  166. package/vendor/agent-root/src/agent/utils/__tests__/message.test.ts +223 -0
  167. package/vendor/agent-root/src/agent/utils/__tests__/token.test.ts +42 -0
  168. package/vendor/agent-root/src/agent/utils/index.ts +16 -0
  169. package/vendor/agent-root/src/agent/utils/message.ts +171 -0
  170. package/vendor/agent-root/src/agent/utils/token.ts +28 -0
  171. package/vendor/agent-root/src/config/__tests__/load-config-to-env.test.ts +129 -0
  172. package/vendor/agent-root/src/config/__tests__/loader.test.ts +247 -0
  173. package/vendor/agent-root/src/config/__tests__/runtime.test.ts +88 -0
  174. package/vendor/agent-root/src/config/index.ts +54 -0
  175. package/vendor/agent-root/src/config/loader.ts +431 -0
  176. package/vendor/agent-root/src/config/paths.ts +30 -0
  177. package/vendor/agent-root/src/config/runtime.ts +163 -0
  178. package/vendor/agent-root/src/config/types.ts +70 -0
  179. package/vendor/agent-root/src/logger/index.ts +57 -0
  180. package/vendor/agent-root/src/logger/logger.ts +819 -0
  181. package/vendor/agent-root/src/logger/types.ts +150 -0
  182. package/vendor/agent-root/src/providers/__tests__/errors.test.ts +441 -0
  183. package/vendor/agent-root/src/providers/__tests__/index.test.ts +16 -0
  184. package/vendor/agent-root/src/providers/__tests__/openai-compatible.options.test.ts +318 -0
  185. package/vendor/agent-root/src/providers/__tests__/openai-compatible.test.ts +600 -0
  186. package/vendor/agent-root/src/providers/__tests__/registry.test.ts +449 -0
  187. package/vendor/agent-root/src/providers/__tests__/responses-adapter.test.ts +298 -0
  188. package/vendor/agent-root/src/providers/adapters/__tests__/anthropic.test.ts +354 -0
  189. package/vendor/agent-root/src/providers/adapters/__tests__/kimi.test.ts +58 -0
  190. package/vendor/agent-root/src/providers/adapters/__tests__/standard.test.ts +261 -0
  191. package/vendor/agent-root/src/providers/adapters/anthropic.ts +572 -0
  192. package/vendor/agent-root/src/providers/adapters/base.ts +131 -0
  193. package/vendor/agent-root/src/providers/adapters/kimi.ts +48 -0
  194. package/vendor/agent-root/src/providers/adapters/responses.ts +732 -0
  195. package/vendor/agent-root/src/providers/adapters/standard.ts +120 -0
  196. package/vendor/agent-root/src/providers/http/__tests__/client.timeout.test.ts +313 -0
  197. package/vendor/agent-root/src/providers/http/client.ts +289 -0
  198. package/vendor/agent-root/src/providers/http/stream-parser.ts +109 -0
  199. package/vendor/agent-root/src/providers/index.ts +76 -0
  200. package/vendor/agent-root/src/providers/kimi-headers.ts +177 -0
  201. package/vendor/agent-root/src/providers/openai-compatible.ts +387 -0
  202. package/vendor/agent-root/src/providers/registry/model-config.ts +230 -0
  203. package/vendor/agent-root/src/providers/registry/provider-factory.ts +123 -0
  204. package/vendor/agent-root/src/providers/registry.ts +135 -0
  205. package/vendor/agent-root/src/providers/types/api.ts +284 -0
  206. package/vendor/agent-root/src/providers/types/config.ts +58 -0
  207. package/vendor/agent-root/src/providers/types/errors.ts +323 -0
  208. package/vendor/agent-root/src/providers/types/index.ts +72 -0
  209. package/vendor/agent-root/src/providers/types/provider.ts +45 -0
  210. package/vendor/agent-root/src/providers/types/registry.ts +88 -0
@@ -0,0 +1,519 @@
1
+ import { describe, expect, it, beforeEach, afterEach } from 'vitest';
2
+ import * as fs from 'node:fs/promises';
3
+ import * as os from 'node:os';
4
+ import * as path from 'node:path';
5
+ import { TaskCreateTool } from '../task-create';
6
+ import { TaskStore } from '../task-store';
7
+
8
+ describe('TaskCreateTool', () => {
9
+ let tempDir: string;
10
+ let store: TaskStore;
11
+ let tool: TaskCreateTool;
12
+
13
+ beforeEach(async () => {
14
+ tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'task-create-test-'));
15
+ store = new TaskStore({ baseDir: tempDir });
16
+ tool = new TaskCreateTool({ store });
17
+ });
18
+
19
+ afterEach(async () => {
20
+ await fs.rm(tempDir, { recursive: true, force: true });
21
+ });
22
+
23
+ describe('name and description', () => {
24
+ it('has correct name', () => {
25
+ expect(tool.name).toBe('task_create');
26
+ });
27
+
28
+ it('has description', () => {
29
+ expect(tool.description).toBeTruthy();
30
+ });
31
+ });
32
+
33
+ describe('parameters', () => {
34
+ it('has required parameters', () => {
35
+ const schema = tool.parameters;
36
+ expect(schema).toBeDefined();
37
+ });
38
+ });
39
+
40
+ describe('execute', () => {
41
+ it('creates a task successfully', async () => {
42
+ const result = await tool.execute({
43
+ subject: 'Test Task',
44
+ description: 'This is a test task description with enough length',
45
+ });
46
+
47
+ expect(result.success).toBe(true);
48
+ expect(result.output).toContain('Test Task');
49
+ });
50
+
51
+ it('creates task with all optional fields', async () => {
52
+ const result = await tool.execute({
53
+ subject: 'Test Task',
54
+ description: 'This is a test task description with enough length',
55
+ active_form: 'Testing',
56
+ priority: 'high',
57
+ tags: [{ name: 'test', color: 'red' }],
58
+ checkpoints: [{ id: 'cp1', name: 'Checkpoint 1' }],
59
+ retry_config: {
60
+ maxRetries: 5,
61
+ retryDelayMs: 1000,
62
+ backoffMultiplier: 2,
63
+ retryOn: ['timeout'],
64
+ },
65
+ timeout_ms: 60000,
66
+ metadata: { key: 'value' },
67
+ created_by: 'user_1',
68
+ });
69
+
70
+ expect(result.success).toBe(true);
71
+ });
72
+
73
+ it('creates task with custom namespace', async () => {
74
+ const result = await tool.execute({
75
+ subject: 'Test Task',
76
+ description: 'This is a test task description with enough length',
77
+ namespace: 'custom',
78
+ });
79
+
80
+ expect(result.success).toBe(true);
81
+ });
82
+
83
+ it('rejects duplicate active task subject', async () => {
84
+ // Create first task
85
+ await tool.execute({
86
+ subject: 'Test Task',
87
+ description: 'This is a test task description with enough length',
88
+ });
89
+
90
+ // Try to create duplicate
91
+ const result = await tool.execute({
92
+ subject: 'Test Task',
93
+ description: 'Another test task description with enough length',
94
+ });
95
+
96
+ expect(result.success).toBe(false);
97
+ expect(result.output).toContain('TASK_DUPLICATE_SUBJECT');
98
+ });
99
+
100
+ it('allows duplicate subject after task is completed', async () => {
101
+ // Create first task
102
+ const first = await tool.execute({
103
+ subject: 'Test Task',
104
+ description: 'This is a test task description with enough length',
105
+ });
106
+
107
+ // Get task ID from output
108
+ const firstData = JSON.parse(first.output!);
109
+ const taskId = firstData.task.id;
110
+
111
+ // Mark task as completed
112
+ await store.updateState('default', (state) => {
113
+ state.tasks[taskId].status = 'completed';
114
+ return 'updated';
115
+ });
116
+
117
+ // Create duplicate should succeed
118
+ const result = await tool.execute({
119
+ subject: 'Test Task',
120
+ description: 'Another test task description with enough length',
121
+ });
122
+
123
+ expect(result.success).toBe(true);
124
+ });
125
+
126
+ it('allows duplicate subject after task is cancelled', async () => {
127
+ // Create first task
128
+ const first = await tool.execute({
129
+ subject: 'Test Task',
130
+ description: 'This is a test task description with enough length',
131
+ });
132
+
133
+ const firstData = JSON.parse(first.output!);
134
+ const taskId = firstData.task.id;
135
+
136
+ // Mark task as cancelled
137
+ await store.updateState('default', (state) => {
138
+ state.tasks[taskId].status = 'cancelled';
139
+ return 'updated';
140
+ });
141
+
142
+ // Create duplicate should succeed
143
+ const result = await tool.execute({
144
+ subject: 'Test Task',
145
+ description: 'Another test task description with enough length',
146
+ });
147
+
148
+ expect(result.success).toBe(true);
149
+ });
150
+
151
+ it('allows duplicate subject after task is failed', async () => {
152
+ // Create first task
153
+ const first = await tool.execute({
154
+ subject: 'Test Task',
155
+ description: 'This is a test task description with enough length',
156
+ });
157
+
158
+ const firstData = JSON.parse(first.output!);
159
+ const taskId = firstData.task.id;
160
+
161
+ // Mark task as failed
162
+ await store.updateState('default', (state) => {
163
+ state.tasks[taskId].status = 'failed';
164
+ return 'updated';
165
+ });
166
+
167
+ // Create duplicate should succeed
168
+ const result = await tool.execute({
169
+ subject: 'Test Task',
170
+ description: 'Another test task description with enough length',
171
+ });
172
+
173
+ expect(result.success).toBe(true);
174
+ });
175
+
176
+ it('trims subject and description', async () => {
177
+ const result = await tool.execute({
178
+ subject: ' Test Task ',
179
+ description: ' This is a test task description with enough length ',
180
+ });
181
+
182
+ expect(result.success).toBe(true);
183
+ const data = JSON.parse(result.output!);
184
+ expect(data.task.subject).toBe('Test Task');
185
+ expect(data.task.description).toBe('This is a test task description with enough length');
186
+ });
187
+
188
+ it('uses default active_form when not provided', async () => {
189
+ const result = await tool.execute({
190
+ subject: 'Test Task',
191
+ description: 'This is a test task description with enough length',
192
+ });
193
+
194
+ expect(result.success).toBe(true);
195
+ const data = JSON.parse(result.output!);
196
+ expect(data.task.activeForm).toBe('Test Task in progress');
197
+ });
198
+
199
+ it('uses default priority when not provided', async () => {
200
+ const result = await tool.execute({
201
+ subject: 'Test Task',
202
+ description: 'This is a test task description with enough length',
203
+ });
204
+
205
+ expect(result.success).toBe(true);
206
+ const data = JSON.parse(result.output!);
207
+ expect(data.task.priority).toBe('normal');
208
+ });
209
+
210
+ it('uses default retry config when not provided', async () => {
211
+ const result = await tool.execute({
212
+ subject: 'Test Task',
213
+ description: 'This is a test task description with enough length',
214
+ });
215
+
216
+ expect(result.success).toBe(true);
217
+ const data = JSON.parse(result.output!);
218
+ expect(data.task.retryConfig.maxRetries).toBe(3);
219
+ expect(data.task.retryConfig.retryDelayMs).toBe(5000);
220
+ expect(data.task.retryConfig.backoffMultiplier).toBe(2);
221
+ });
222
+
223
+ it('creates task with checkpoints', async () => {
224
+ const result = await tool.execute({
225
+ subject: 'Test Task',
226
+ description: 'This is a test task description with enough length',
227
+ checkpoints: [
228
+ { id: 'cp1', name: 'Checkpoint 1', completed: false },
229
+ { id: 'cp2', name: 'Checkpoint 2', completed: true },
230
+ ],
231
+ });
232
+
233
+ expect(result.success).toBe(true);
234
+ const data = JSON.parse(result.output!);
235
+ expect(data.task.checkpoints).toHaveLength(2);
236
+ expect(data.task.checkpoints[0].id).toBe('cp1');
237
+ expect(data.task.checkpoints[1].completed).toBe(true);
238
+ });
239
+
240
+ it('creates task with tags', async () => {
241
+ const result = await tool.execute({
242
+ subject: 'Test Task',
243
+ description: 'This is a test task description with enough length',
244
+ tags: [
245
+ { name: 'urgent', color: 'red', category: 'priority' },
246
+ { name: 'backend', category: 'team' },
247
+ ],
248
+ });
249
+
250
+ expect(result.success).toBe(true);
251
+ const data = JSON.parse(result.output!);
252
+ expect(data.task.tags).toHaveLength(2);
253
+ expect(data.task.tags[0].name).toBe('urgent');
254
+ expect(data.task.tags[0].color).toBe('red');
255
+ });
256
+
257
+ it('creates task with metadata', async () => {
258
+ const result = await tool.execute({
259
+ subject: 'Test Task',
260
+ description: 'This is a test task description with enough length',
261
+ metadata: { key1: 'value1', key2: 42, key3: true },
262
+ });
263
+
264
+ expect(result.success).toBe(true);
265
+ const data = JSON.parse(result.output!);
266
+ expect(data.task.metadata.key1).toBe('value1');
267
+ expect(data.task.metadata.key2).toBe(42);
268
+ expect(data.task.metadata.key3).toBe(true);
269
+ });
270
+
271
+ it('creates task with timeout', async () => {
272
+ const result = await tool.execute({
273
+ subject: 'Test Task',
274
+ description: 'This is a test task description with enough length',
275
+ timeout_ms: 60000,
276
+ });
277
+
278
+ expect(result.success).toBe(true);
279
+ const data = JSON.parse(result.output!);
280
+ expect(data.task.timeoutMs).toBe(60000);
281
+ });
282
+
283
+ it('creates task with created_by', async () => {
284
+ const result = await tool.execute({
285
+ subject: 'Test Task',
286
+ description: 'This is a test task description with enough length',
287
+ created_by: 'user_1',
288
+ });
289
+
290
+ expect(result.success).toBe(true);
291
+ const data = JSON.parse(result.output!);
292
+ expect(data.task.history[0].actor).toBe('user_1');
293
+ });
294
+
295
+ it('creates task with default namespace', async () => {
296
+ const tool = new TaskCreateTool({ store, defaultNamespace: 'custom' });
297
+ const result = await tool.execute({
298
+ subject: 'Test Task',
299
+ description: 'This is a test task description with enough length',
300
+ });
301
+
302
+ expect(result.success).toBe(true);
303
+ const data = JSON.parse(result.output!);
304
+ expect(data.namespace).toBe('custom');
305
+ });
306
+
307
+ it('creates task with namespace from args overriding default', async () => {
308
+ const tool = new TaskCreateTool({ store, defaultNamespace: 'default' });
309
+ const result = await tool.execute({
310
+ subject: 'Test Task',
311
+ description: 'This is a test task description with enough length',
312
+ namespace: 'custom',
313
+ });
314
+
315
+ expect(result.success).toBe(true);
316
+ const data = JSON.parse(result.output!);
317
+ expect(data.namespace).toBe('custom');
318
+ });
319
+
320
+ it('handles store errors', async () => {
321
+ // Create a store that throws errors
322
+ const errorStore = {
323
+ normalizeNamespace: () => 'default',
324
+ updateState: async () => {
325
+ throw new Error('Store error');
326
+ },
327
+ } as any;
328
+
329
+ const tool = new TaskCreateTool({ store: errorStore });
330
+ const result = await tool.execute({
331
+ subject: 'Test Task',
332
+ description: 'This is a test task description with enough length',
333
+ });
334
+
335
+ expect(result.success).toBe(false);
336
+ expect(result.output).toContain('TASK_OPERATION_FAILED');
337
+ });
338
+
339
+ it('handles prefixed errors', async () => {
340
+ // Create a store that throws prefixed errors
341
+ const errorStore = {
342
+ normalizeNamespace: () => 'default',
343
+ updateState: async () => {
344
+ throw new Error('TASK_INVALID_NAMESPACE: invalid namespace');
345
+ },
346
+ } as any;
347
+
348
+ const tool = new TaskCreateTool({ store: errorStore });
349
+ const result = await tool.execute({
350
+ subject: 'Test Task',
351
+ description: 'This is a test task description with enough length',
352
+ });
353
+
354
+ expect(result.success).toBe(false);
355
+ expect(result.output).toContain('TASK_INVALID_NAMESPACE');
356
+ });
357
+
358
+ it('generates unique task IDs', async () => {
359
+ const result1 = await tool.execute({
360
+ subject: 'Test Task 1',
361
+ description: 'This is a test task description with enough length',
362
+ });
363
+
364
+ const result2 = await tool.execute({
365
+ subject: 'Test Task 2',
366
+ description: 'This is another test task description with enough length',
367
+ });
368
+
369
+ expect(result1.success).toBe(true);
370
+ expect(result2.success).toBe(true);
371
+
372
+ const data1 = JSON.parse(result1.output!);
373
+ const data2 = JSON.parse(result2.output!);
374
+
375
+ expect(data1.task.id).not.toBe(data2.task.id);
376
+ });
377
+
378
+ it('sets initial task status to pending', async () => {
379
+ const result = await tool.execute({
380
+ subject: 'Test Task',
381
+ description: 'This is a test task description with enough length',
382
+ });
383
+
384
+ expect(result.success).toBe(true);
385
+ const data = JSON.parse(result.output!);
386
+ expect(data.task.status).toBe('pending');
387
+ });
388
+
389
+ it('sets initial progress to 0', async () => {
390
+ const result = await tool.execute({
391
+ subject: 'Test Task',
392
+ description: 'This is a test task description with enough length',
393
+ });
394
+
395
+ expect(result.success).toBe(true);
396
+ const data = JSON.parse(result.output!);
397
+ expect(data.task.progress).toBe(0);
398
+ });
399
+
400
+ it('sets initial retryCount to 0', async () => {
401
+ const result = await tool.execute({
402
+ subject: 'Test Task',
403
+ description: 'This is a test task description with enough length',
404
+ });
405
+
406
+ expect(result.success).toBe(true);
407
+ const data = JSON.parse(result.output!);
408
+ expect(data.task.retryCount).toBe(0);
409
+ });
410
+
411
+ it('sets owner to null', async () => {
412
+ const result = await tool.execute({
413
+ subject: 'Test Task',
414
+ description: 'This is a test task description with enough length',
415
+ });
416
+
417
+ expect(result.success).toBe(true);
418
+ const data = JSON.parse(result.output!);
419
+ expect(data.task.owner).toBeNull();
420
+ });
421
+
422
+ it('sets blockedBy and blocks to empty arrays', async () => {
423
+ const result = await tool.execute({
424
+ subject: 'Test Task',
425
+ description: 'This is a test task description with enough length',
426
+ });
427
+
428
+ expect(result.success).toBe(true);
429
+ const data = JSON.parse(result.output!);
430
+ expect(data.task.blockedBy).toEqual([]);
431
+ expect(data.task.blocks).toEqual([]);
432
+ });
433
+
434
+ it('sets version to 1', async () => {
435
+ const result = await tool.execute({
436
+ subject: 'Test Task',
437
+ description: 'This is a test task description with enough length',
438
+ });
439
+
440
+ expect(result.success).toBe(true);
441
+ const data = JSON.parse(result.output!);
442
+ expect(data.task.version).toBe(1);
443
+ });
444
+
445
+ it('sets createdAt and updatedAt timestamps', async () => {
446
+ const before = Date.now();
447
+ const result = await tool.execute({
448
+ subject: 'Test Task',
449
+ description: 'This is a test task description with enough length',
450
+ });
451
+ const after = Date.now();
452
+
453
+ expect(result.success).toBe(true);
454
+ const data = JSON.parse(result.output!);
455
+ expect(data.task.createdAt).toBeGreaterThanOrEqual(before);
456
+ expect(data.task.createdAt).toBeLessThanOrEqual(after);
457
+ expect(data.task.updatedAt).toBeGreaterThanOrEqual(before);
458
+ expect(data.task.updatedAt).toBeLessThanOrEqual(after);
459
+ });
460
+
461
+ it('creates history entry', async () => {
462
+ const result = await tool.execute({
463
+ subject: 'Test Task',
464
+ description: 'This is a test task description with enough length',
465
+ });
466
+
467
+ expect(result.success).toBe(true);
468
+ const data = JSON.parse(result.output!);
469
+ expect(data.task.history).toHaveLength(1);
470
+ expect(data.task.history[0].action).toBe('created');
471
+ expect(data.task.history[0].metadata.subject).toBe('Test Task');
472
+ });
473
+ });
474
+
475
+ describe('shouldConfirm', () => {
476
+ it('returns false', () => {
477
+ expect(tool.shouldConfirm({ subject: 'Test', description: 'Test description' })).toBe(false);
478
+ });
479
+ });
480
+
481
+ describe('getConfirmDetails', () => {
482
+ it('returns null', () => {
483
+ expect(
484
+ tool.getConfirmDetails({ subject: 'Test', description: 'Test description' })
485
+ ).toBeNull();
486
+ });
487
+ });
488
+
489
+ describe('getConcurrencyMode', () => {
490
+ it('returns exclusive', () => {
491
+ expect(tool.getConcurrencyMode()).toBe('exclusive');
492
+ });
493
+ });
494
+
495
+ describe('getConcurrencyLockKey', () => {
496
+ it('returns lock key based on namespace', () => {
497
+ expect(
498
+ tool.getConcurrencyLockKey({ subject: 'Test', description: 'Test description' } as any)
499
+ ).toBe('taskns:default');
500
+ });
501
+
502
+ it('returns lock key with custom namespace', () => {
503
+ expect(
504
+ tool.getConcurrencyLockKey({
505
+ subject: 'Test',
506
+ description: 'Test description',
507
+ namespace: 'custom',
508
+ } as any)
509
+ ).toBe('taskns:custom');
510
+ });
511
+
512
+ it('returns lock key with default namespace', () => {
513
+ const tool = new TaskCreateTool({ store, defaultNamespace: 'custom' });
514
+ expect(
515
+ tool.getConcurrencyLockKey({ subject: 'Test', description: 'Test description' } as any)
516
+ ).toBe('taskns:custom');
517
+ });
518
+ });
519
+ });