hzl-cli 1.17.0 → 1.18.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.
Files changed (183) hide show
  1. package/README.md +7 -1
  2. package/dist/commands/add-dep.d.ts +15 -0
  3. package/dist/commands/add-dep.d.ts.map +1 -0
  4. package/dist/commands/add-dep.js +86 -0
  5. package/dist/commands/add-dep.js.map +1 -0
  6. package/dist/commands/add-dep.test.d.ts +2 -0
  7. package/dist/commands/add-dep.test.d.ts.map +1 -0
  8. package/dist/commands/add-dep.test.js +47 -0
  9. package/dist/commands/add-dep.test.js.map +1 -0
  10. package/dist/commands/add.d.ts +23 -0
  11. package/dist/commands/add.d.ts.map +1 -0
  12. package/dist/commands/add.js +65 -0
  13. package/dist/commands/add.js.map +1 -0
  14. package/dist/commands/add.test.d.ts +2 -0
  15. package/dist/commands/add.test.d.ts.map +1 -0
  16. package/dist/commands/add.test.js +60 -0
  17. package/dist/commands/add.test.js.map +1 -0
  18. package/dist/commands/archive.d.ts +16 -0
  19. package/dist/commands/archive.d.ts.map +1 -0
  20. package/dist/commands/archive.js +51 -0
  21. package/dist/commands/archive.js.map +1 -0
  22. package/dist/commands/archive.test.d.ts +2 -0
  23. package/dist/commands/archive.test.d.ts.map +1 -0
  24. package/dist/commands/archive.test.js +48 -0
  25. package/dist/commands/archive.test.js.map +1 -0
  26. package/dist/commands/checkpoint.d.ts +17 -0
  27. package/dist/commands/checkpoint.d.ts.map +1 -0
  28. package/dist/commands/checkpoint.js +64 -0
  29. package/dist/commands/checkpoint.js.map +1 -0
  30. package/dist/commands/checkpoint.test.d.ts +2 -0
  31. package/dist/commands/checkpoint.test.d.ts.map +1 -0
  32. package/dist/commands/checkpoint.test.js +47 -0
  33. package/dist/commands/checkpoint.test.js.map +1 -0
  34. package/dist/commands/claim.d.ts +20 -0
  35. package/dist/commands/claim.d.ts.map +1 -0
  36. package/dist/commands/claim.js +62 -0
  37. package/dist/commands/claim.js.map +1 -0
  38. package/dist/commands/claim.test.d.ts +2 -0
  39. package/dist/commands/claim.test.d.ts.map +1 -0
  40. package/dist/commands/claim.test.js +48 -0
  41. package/dist/commands/claim.test.js.map +1 -0
  42. package/dist/commands/comment.d.ts +16 -0
  43. package/dist/commands/comment.d.ts.map +1 -0
  44. package/dist/commands/comment.js +53 -0
  45. package/dist/commands/comment.js.map +1 -0
  46. package/dist/commands/comment.test.d.ts +2 -0
  47. package/dist/commands/comment.test.d.ts.map +1 -0
  48. package/dist/commands/comment.test.js +46 -0
  49. package/dist/commands/comment.test.js.map +1 -0
  50. package/dist/commands/complete.d.ts +15 -0
  51. package/dist/commands/complete.d.ts.map +1 -0
  52. package/dist/commands/complete.js +47 -0
  53. package/dist/commands/complete.js.map +1 -0
  54. package/dist/commands/complete.test.d.ts +2 -0
  55. package/dist/commands/complete.test.d.ts.map +1 -0
  56. package/dist/commands/complete.test.js +42 -0
  57. package/dist/commands/complete.test.js.map +1 -0
  58. package/dist/commands/history.d.ts +22 -0
  59. package/dist/commands/history.d.ts.map +1 -0
  60. package/dist/commands/history.js +70 -0
  61. package/dist/commands/history.js.map +1 -0
  62. package/dist/commands/history.test.d.ts +2 -0
  63. package/dist/commands/history.test.d.ts.map +1 -0
  64. package/dist/commands/history.test.js +41 -0
  65. package/dist/commands/history.test.js.map +1 -0
  66. package/dist/commands/list.d.ts +26 -0
  67. package/dist/commands/list.d.ts.map +1 -0
  68. package/dist/commands/list.js +85 -0
  69. package/dist/commands/list.js.map +1 -0
  70. package/dist/commands/list.test.d.ts +2 -0
  71. package/dist/commands/list.test.d.ts.map +1 -0
  72. package/dist/commands/list.test.js +55 -0
  73. package/dist/commands/list.test.js.map +1 -0
  74. package/dist/commands/move.d.ts +15 -0
  75. package/dist/commands/move.d.ts.map +1 -0
  76. package/dist/commands/move.js +62 -0
  77. package/dist/commands/move.js.map +1 -0
  78. package/dist/commands/move.test.d.ts +2 -0
  79. package/dist/commands/move.test.d.ts.map +1 -0
  80. package/dist/commands/move.test.js +55 -0
  81. package/dist/commands/move.test.js.map +1 -0
  82. package/dist/commands/next.d.ts +18 -0
  83. package/dist/commands/next.d.ts.map +1 -0
  84. package/dist/commands/next.js +64 -0
  85. package/dist/commands/next.js.map +1 -0
  86. package/dist/commands/next.test.d.ts +2 -0
  87. package/dist/commands/next.test.d.ts.map +1 -0
  88. package/dist/commands/next.test.js +52 -0
  89. package/dist/commands/next.test.js.map +1 -0
  90. package/dist/commands/projects.d.ts +16 -0
  91. package/dist/commands/projects.d.ts.map +1 -0
  92. package/dist/commands/projects.js +64 -0
  93. package/dist/commands/projects.js.map +1 -0
  94. package/dist/commands/projects.test.d.ts +2 -0
  95. package/dist/commands/projects.test.d.ts.map +1 -0
  96. package/dist/commands/projects.test.js +47 -0
  97. package/dist/commands/projects.test.js.map +1 -0
  98. package/dist/commands/release.d.ts +17 -0
  99. package/dist/commands/release.d.ts.map +1 -0
  100. package/dist/commands/release.js +52 -0
  101. package/dist/commands/release.js.map +1 -0
  102. package/dist/commands/release.test.d.ts +2 -0
  103. package/dist/commands/release.test.d.ts.map +1 -0
  104. package/dist/commands/release.test.js +47 -0
  105. package/dist/commands/release.test.js.map +1 -0
  106. package/dist/commands/remove-dep.d.ts +15 -0
  107. package/dist/commands/remove-dep.d.ts.map +1 -0
  108. package/dist/commands/remove-dep.js +55 -0
  109. package/dist/commands/remove-dep.js.map +1 -0
  110. package/dist/commands/remove-dep.test.d.ts +2 -0
  111. package/dist/commands/remove-dep.test.d.ts.map +1 -0
  112. package/dist/commands/remove-dep.test.js +47 -0
  113. package/dist/commands/remove-dep.test.js.map +1 -0
  114. package/dist/commands/rename-project.d.ts +16 -0
  115. package/dist/commands/rename-project.d.ts.map +1 -0
  116. package/dist/commands/rename-project.js +70 -0
  117. package/dist/commands/rename-project.js.map +1 -0
  118. package/dist/commands/rename-project.test.d.ts +2 -0
  119. package/dist/commands/rename-project.test.d.ts.map +1 -0
  120. package/dist/commands/rename-project.test.js +75 -0
  121. package/dist/commands/rename-project.test.js.map +1 -0
  122. package/dist/commands/reopen.d.ts +18 -0
  123. package/dist/commands/reopen.d.ts.map +1 -0
  124. package/dist/commands/reopen.js +56 -0
  125. package/dist/commands/reopen.js.map +1 -0
  126. package/dist/commands/reopen.test.d.ts +2 -0
  127. package/dist/commands/reopen.test.d.ts.map +1 -0
  128. package/dist/commands/reopen.test.js +48 -0
  129. package/dist/commands/reopen.test.js.map +1 -0
  130. package/dist/commands/search.d.ts +23 -0
  131. package/dist/commands/search.d.ts.map +1 -0
  132. package/dist/commands/search.js +89 -0
  133. package/dist/commands/search.js.map +1 -0
  134. package/dist/commands/search.test.d.ts +2 -0
  135. package/dist/commands/search.test.d.ts.map +1 -0
  136. package/dist/commands/search.test.js +41 -0
  137. package/dist/commands/search.test.js.map +1 -0
  138. package/dist/commands/set-status.d.ts +18 -0
  139. package/dist/commands/set-status.d.ts.map +1 -0
  140. package/dist/commands/set-status.js +60 -0
  141. package/dist/commands/set-status.js.map +1 -0
  142. package/dist/commands/set-status.test.d.ts +2 -0
  143. package/dist/commands/set-status.test.d.ts.map +1 -0
  144. package/dist/commands/set-status.test.js +45 -0
  145. package/dist/commands/set-status.test.js.map +1 -0
  146. package/dist/commands/show.d.ts +34 -0
  147. package/dist/commands/show.d.ts.map +1 -0
  148. package/dist/commands/show.js +96 -0
  149. package/dist/commands/show.js.map +1 -0
  150. package/dist/commands/show.test.d.ts +2 -0
  151. package/dist/commands/show.test.d.ts.map +1 -0
  152. package/dist/commands/show.test.js +50 -0
  153. package/dist/commands/show.test.js.map +1 -0
  154. package/dist/commands/steal.d.ts +19 -0
  155. package/dist/commands/steal.d.ts.map +1 -0
  156. package/dist/commands/steal.js +90 -0
  157. package/dist/commands/steal.js.map +1 -0
  158. package/dist/commands/steal.test.d.ts +2 -0
  159. package/dist/commands/steal.test.d.ts.map +1 -0
  160. package/dist/commands/steal.test.js +66 -0
  161. package/dist/commands/steal.test.js.map +1 -0
  162. package/dist/commands/stuck.d.ts +22 -0
  163. package/dist/commands/stuck.d.ts.map +1 -0
  164. package/dist/commands/stuck.js +91 -0
  165. package/dist/commands/stuck.js.map +1 -0
  166. package/dist/commands/stuck.test.d.ts +2 -0
  167. package/dist/commands/stuck.test.d.ts.map +1 -0
  168. package/dist/commands/stuck.test.js +51 -0
  169. package/dist/commands/stuck.test.js.map +1 -0
  170. package/dist/commands/task.d.ts +32 -0
  171. package/dist/commands/task.d.ts.map +1 -0
  172. package/dist/commands/task.js +72 -0
  173. package/dist/commands/task.js.map +1 -0
  174. package/dist/commands/update.d.ts +23 -0
  175. package/dist/commands/update.d.ts.map +1 -0
  176. package/dist/commands/update.js +105 -0
  177. package/dist/commands/update.js.map +1 -0
  178. package/dist/commands/update.test.d.ts +2 -0
  179. package/dist/commands/update.test.d.ts.map +1 -0
  180. package/dist/commands/update.test.js +63 -0
  181. package/dist/commands/update.test.js.map +1 -0
  182. package/dist/types.d.ts +1 -1
  183. package/package.json +3 -3
@@ -0,0 +1,90 @@
1
+ // packages/hzl-cli/src/commands/steal.ts
2
+ import { Command } from 'commander';
3
+ import { resolveDbPath } from '../config.js';
4
+ import { initializeDb, closeDb } from '../db.js';
5
+ import { handleError, CLIError, ExitCode } from '../errors.js';
6
+ export function runSteal(options) {
7
+ const { services, taskId, newOwner, force, ifExpired, json } = options;
8
+ const task = services.taskService.getTaskById(taskId);
9
+ if (!task) {
10
+ throw new CLIError(`Task not found: ${taskId}`, ExitCode.NotFound);
11
+ }
12
+ const previousOwner = task.claimed_by_author;
13
+ // Check if we're allowed to steal (pre-validation)
14
+ if (!force && !ifExpired) {
15
+ throw new CLIError(`Must specify --force or --if-expired to steal a task`, ExitCode.InvalidInput);
16
+ }
17
+ if (!force && ifExpired) {
18
+ // Only steal if lease has expired
19
+ if (task.lease_until) {
20
+ const leaseExpiry = new Date(task.lease_until);
21
+ if (leaseExpiry > new Date()) {
22
+ throw new CLIError(`Task lease has not expired (expires ${task.lease_until})`, ExitCode.InvalidInput);
23
+ }
24
+ }
25
+ else if (task.claimed_by_author) {
26
+ // No lease set, but task is claimed - require force
27
+ throw new CLIError(`Task is claimed but has no lease. Use --force to steal.`, ExitCode.InvalidInput);
28
+ }
29
+ }
30
+ // Steal by using stealTask service method
31
+ // Pass ifExpired when force is not set, force when force is set
32
+ const stealResult = services.taskService.stealTask(taskId, {
33
+ author: newOwner,
34
+ force: force || false,
35
+ ifExpired: ifExpired && !force,
36
+ });
37
+ if (!stealResult.success) {
38
+ throw new CLIError(stealResult.error || 'Failed to steal task', ExitCode.InvalidInput);
39
+ }
40
+ // Get updated task
41
+ const stolenTask = services.taskService.getTaskById(taskId);
42
+ const result = {
43
+ task_id: stolenTask.task_id,
44
+ title: stolenTask.title,
45
+ status: stolenTask.status,
46
+ claimed_by_author: stolenTask.claimed_by_author,
47
+ stolen_from: previousOwner,
48
+ };
49
+ if (json) {
50
+ console.log(JSON.stringify(result));
51
+ }
52
+ else {
53
+ console.log(`✓ Stole task ${stolenTask.task_id}: ${stolenTask.title}`);
54
+ if (previousOwner)
55
+ console.log(` Previous owner: ${previousOwner}`);
56
+ if (newOwner)
57
+ console.log(` New owner: ${newOwner}`);
58
+ }
59
+ return result;
60
+ }
61
+ export function createStealCommand() {
62
+ return new Command('steal')
63
+ .description('Steal a claimed task')
64
+ .argument('<taskId>', 'Task ID')
65
+ .option('--owner <name>', 'New owner name')
66
+ .option('--force', 'Force steal even if lease is active')
67
+ .option('--if-expired', 'Only steal if lease has expired')
68
+ .action(function (taskId, opts) {
69
+ const globalOpts = this.optsWithGlobals();
70
+ const dbPath = resolveDbPath(globalOpts.db);
71
+ const services = initializeDb(dbPath);
72
+ try {
73
+ runSteal({
74
+ services,
75
+ taskId,
76
+ newOwner: opts.owner,
77
+ force: opts.force,
78
+ ifExpired: opts.ifExpired,
79
+ json: globalOpts.json ?? false,
80
+ });
81
+ }
82
+ catch (e) {
83
+ handleError(e, globalOpts.json);
84
+ }
85
+ finally {
86
+ closeDb(services);
87
+ }
88
+ });
89
+ }
90
+ //# sourceMappingURL=steal.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"steal.js","sourceRoot":"","sources":["../../src/commands/steal.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=steal.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"steal.test.d.ts","sourceRoot":"","sources":["../../src/commands/steal.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,66 @@
1
+ // packages/hzl-cli/src/commands/steal.test.ts
2
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
3
+ import fs from 'fs';
4
+ import path from 'path';
5
+ import os from 'os';
6
+ import { runSteal } from './steal.js';
7
+ import { initializeDb, closeDb } from '../db.js';
8
+ import { TaskStatus } from 'hzl-core/events/types.js';
9
+ describe('runSteal', () => {
10
+ let tempDir;
11
+ let dbPath;
12
+ let services;
13
+ beforeEach(() => {
14
+ tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'hzl-steal-test-'));
15
+ dbPath = path.join(tempDir, 'test.db');
16
+ services = initializeDb(dbPath);
17
+ });
18
+ afterEach(() => {
19
+ closeDb(services);
20
+ fs.rmSync(tempDir, { recursive: true, force: true });
21
+ });
22
+ it('steals a task with --force', () => {
23
+ const task = services.taskService.createTask({ title: 'Test', project: 'inbox' });
24
+ services.taskService.setStatus(task.task_id, TaskStatus.Ready);
25
+ services.taskService.claimTask(task.task_id, { author: 'original-owner' });
26
+ const result = runSteal({
27
+ services,
28
+ taskId: task.task_id,
29
+ newOwner: 'new-owner',
30
+ force: true,
31
+ json: false,
32
+ });
33
+ expect(result.claimed_by_author).toBe('new-owner');
34
+ expect(result.stolen_from).toBe('original-owner');
35
+ });
36
+ it('steals a task with expired lease when --if-expired', () => {
37
+ const task = services.taskService.createTask({ title: 'Test', project: 'inbox' });
38
+ services.taskService.setStatus(task.task_id, TaskStatus.Ready);
39
+ // Claim with lease in the past
40
+ const pastLease = new Date(Date.now() - 60000).toISOString();
41
+ services.taskService.claimTask(task.task_id, { author: 'original-owner', lease_until: pastLease });
42
+ const result = runSteal({
43
+ services,
44
+ taskId: task.task_id,
45
+ newOwner: 'new-owner',
46
+ ifExpired: true,
47
+ json: false,
48
+ });
49
+ expect(result.claimed_by_author).toBe('new-owner');
50
+ });
51
+ it('throws when trying to steal non-expired task with --if-expired', () => {
52
+ const task = services.taskService.createTask({ title: 'Test', project: 'inbox' });
53
+ services.taskService.setStatus(task.task_id, TaskStatus.Ready);
54
+ // Claim with lease in the future
55
+ const futureLease = new Date(Date.now() + 3600000).toISOString();
56
+ services.taskService.claimTask(task.task_id, { author: 'original-owner', lease_until: futureLease });
57
+ expect(() => runSteal({
58
+ services,
59
+ taskId: task.task_id,
60
+ newOwner: 'new-owner',
61
+ ifExpired: true,
62
+ json: false,
63
+ })).toThrow(/lease has not expired/);
64
+ });
65
+ });
66
+ //# sourceMappingURL=steal.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"steal.test.js","sourceRoot":"","sources":["../../src/commands/steal.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,22 @@
1
+ import { Command } from 'commander';
2
+ import { type Services } from '../db.js';
3
+ export interface StuckTask {
4
+ task_id: string;
5
+ title: string;
6
+ project: string;
7
+ claimed_by_author: string | null;
8
+ lease_until: string | null;
9
+ expired_for_ms: number;
10
+ }
11
+ export interface StuckResult {
12
+ tasks: StuckTask[];
13
+ total: number;
14
+ }
15
+ export declare function runStuck(options: {
16
+ services: Services;
17
+ project?: string;
18
+ olderThanMinutes?: number;
19
+ json: boolean;
20
+ }): StuckResult;
21
+ export declare function createStuckCommand(): Command;
22
+ //# sourceMappingURL=stuck.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stuck.d.ts","sourceRoot":"","sources":["../../src/commands/stuck.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAEpC,OAAO,EAAyB,KAAK,QAAQ,EAAE,MAAM,UAAU,CAAC;AAIhE,MAAM,WAAW,SAAS;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,iBAAiB,EAAE,MAAM,GAAG,IAAI,CAAC;IACjC,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,cAAc,EAAE,MAAM,CAAC;CACxB;AAED,MAAM,WAAW,WAAW;IAC1B,KAAK,EAAE,SAAS,EAAE,CAAC;IACnB,KAAK,EAAE,MAAM,CAAC;CACf;AAED,wBAAgB,QAAQ,CAAC,OAAO,EAAE;IAChC,QAAQ,EAAE,QAAQ,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,IAAI,EAAE,OAAO,CAAC;CACf,GAAG,WAAW,CAmEd;AAED,wBAAgB,kBAAkB,IAAI,OAAO,CAsB5C"}
@@ -0,0 +1,91 @@
1
+ // packages/hzl-cli/src/commands/stuck.ts
2
+ import { Command } from 'commander';
3
+ import { resolveDbPath } from '../config.js';
4
+ import { initializeDb, closeDb } from '../db.js';
5
+ import { handleError } from '../errors.js';
6
+ export function runStuck(options) {
7
+ const { services, project, olderThanMinutes = 0, json } = options;
8
+ const db = services.db;
9
+ const now = new Date();
10
+ // Find tasks with expired leases
11
+ let query = `
12
+ SELECT task_id, title, project, claimed_by_author, claimed_by_agent_id, lease_until
13
+ FROM tasks_current
14
+ WHERE status = 'in_progress'
15
+ AND lease_until IS NOT NULL
16
+ AND lease_until < ?
17
+ `;
18
+ const params = [now.toISOString()];
19
+ if (project) {
20
+ query += ' AND project = ?';
21
+ params.push(project);
22
+ }
23
+ query += ' ORDER BY lease_until ASC';
24
+ const rows = db.prepare(query).all(...params);
25
+ const tasks = rows
26
+ .map(row => {
27
+ const leaseExpiry = new Date(row.lease_until);
28
+ const expiredForMs = now.getTime() - leaseExpiry.getTime();
29
+ const expiredForMinutes = expiredForMs / 60000;
30
+ // Filter by olderThanMinutes if specified
31
+ if (expiredForMinutes < olderThanMinutes) {
32
+ return null;
33
+ }
34
+ return {
35
+ task_id: row.task_id,
36
+ title: row.title,
37
+ project: row.project,
38
+ claimed_by_author: row.claimed_by_author,
39
+ lease_until: row.lease_until,
40
+ expired_for_ms: expiredForMs,
41
+ };
42
+ })
43
+ .filter((t) => t !== null);
44
+ const result = {
45
+ tasks,
46
+ total: tasks.length,
47
+ };
48
+ if (json) {
49
+ console.log(JSON.stringify(result));
50
+ }
51
+ else {
52
+ if (tasks.length === 0) {
53
+ console.log('No stuck tasks found');
54
+ }
55
+ else {
56
+ console.log(`Stuck tasks (${tasks.length}):`);
57
+ for (const task of tasks) {
58
+ const expiredMinutes = Math.round(task.expired_for_ms / 60000);
59
+ console.log(` [${task.task_id.slice(0, 8)}] ${task.title} (${task.project})`);
60
+ console.log(` Owner: ${task.claimed_by_author ?? 'unknown'} | Expired: ${expiredMinutes}m ago`);
61
+ }
62
+ }
63
+ }
64
+ return result;
65
+ }
66
+ export function createStuckCommand() {
67
+ return new Command('stuck')
68
+ .description('List tasks with expired leases')
69
+ .option('-p, --project <project>', 'Filter by project')
70
+ .option('--older-than <minutes>', 'Only show tasks expired for more than N minutes', '0')
71
+ .action(function (opts) {
72
+ const globalOpts = this.optsWithGlobals();
73
+ const dbPath = resolveDbPath(globalOpts.db);
74
+ const services = initializeDb(dbPath);
75
+ try {
76
+ runStuck({
77
+ services,
78
+ project: opts.project,
79
+ olderThanMinutes: parseInt(opts.olderThan, 10),
80
+ json: globalOpts.json ?? false,
81
+ });
82
+ }
83
+ catch (e) {
84
+ handleError(e, globalOpts.json);
85
+ }
86
+ finally {
87
+ closeDb(services);
88
+ }
89
+ });
90
+ }
91
+ //# sourceMappingURL=stuck.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stuck.js","sourceRoot":"","sources":["../../src/commands/stuck.ts"],"names":[],"mappings":"AAAA,yCAAyC;AACzC,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,aAAa,EAAE,MAAM,cAAc,CAAC;AAC7C,OAAO,EAAE,YAAY,EAAE,OAAO,EAAiB,MAAM,UAAU,CAAC;AAChE,OAAO,EAAE,WAAW,EAAE,MAAM,cAAc,CAAC;AAiB3C,MAAM,UAAU,QAAQ,CAAC,OAKxB;IACC,MAAM,EAAE,QAAQ,EAAE,OAAO,EAAE,gBAAgB,GAAG,CAAC,EAAE,IAAI,EAAE,GAAG,OAAO,CAAC;IAClE,MAAM,EAAE,GAAG,QAAQ,CAAC,EAAE,CAAC;IACvB,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;IAEvB,iCAAiC;IACjC,IAAI,KAAK,GAAG;;;;;;GAMX,CAAC;IACF,MAAM,MAAM,GAAU,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC;IAE1C,IAAI,OAAO,EAAE,CAAC;QACZ,KAAK,IAAI,kBAAkB,CAAC;QAC5B,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACvB,CAAC;IAED,KAAK,IAAI,2BAA2B,CAAC;IAErC,MAAM,IAAI,GAAG,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,GAAG,MAAM,CAAU,CAAC;IAEvD,MAAM,KAAK,GAAgB,IAAI;SAC5B,GAAG,CAAC,GAAG,CAAC,EAAE;QACT,MAAM,WAAW,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;QAC9C,MAAM,YAAY,GAAG,GAAG,CAAC,OAAO,EAAE,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;QAC3D,MAAM,iBAAiB,GAAG,YAAY,GAAG,KAAK,CAAC;QAE/C,0CAA0C;QAC1C,IAAI,iBAAiB,GAAG,gBAAgB,EAAE,CAAC;YACzC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO;YACL,OAAO,EAAE,GAAG,CAAC,OAAO;YACpB,KAAK,EAAE,GAAG,CAAC,KAAK;YAChB,OAAO,EAAE,GAAG,CAAC,OAAO;YACpB,iBAAiB,EAAE,GAAG,CAAC,iBAAiB;YACxC,WAAW,EAAE,GAAG,CAAC,WAAW;YAC5B,cAAc,EAAE,YAAY;SAC7B,CAAC;IACJ,CAAC,CAAC;SACD,MAAM,CAAC,CAAC,CAAC,EAAkB,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC;IAE7C,MAAM,MAAM,GAAgB;QAC1B,KAAK;QACL,KAAK,EAAE,KAAK,CAAC,MAAM;KACpB,CAAC;IAEF,IAAI,IAAI,EAAE,CAAC;QACT,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC;IACtC,CAAC;SAAM,CAAC;QACN,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACvB,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;QACtC,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,gBAAgB,KAAK,CAAC,MAAM,IAAI,CAAC,CAAC;YAC9C,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,MAAM,cAAc,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC,CAAC;gBAC/D,OAAO,CAAC,GAAG,CAAC,MAAM,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,KAAK,KAAK,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC;gBAC/E,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,CAAC,iBAAiB,IAAI,SAAS,eAAe,cAAc,OAAO,CAAC,CAAC;YACrG,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,UAAU,kBAAkB;IAChC,OAAO,IAAI,OAAO,CAAC,OAAO,CAAC;SACxB,WAAW,CAAC,gCAAgC,CAAC;SAC7C,MAAM,CAAC,yBAAyB,EAAE,mBAAmB,CAAC;SACtD,MAAM,CAAC,wBAAwB,EAAE,iDAAiD,EAAE,GAAG,CAAC;SACxF,MAAM,CAAC,UAAyB,IAAS;QACxC,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,EAAmB,CAAC;QAC3D,MAAM,MAAM,GAAG,aAAa,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;QAC5C,MAAM,QAAQ,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC;QACtC,IAAI,CAAC;YACH,QAAQ,CAAC;gBACP,QAAQ;gBACR,OAAO,EAAE,IAAI,CAAC,OAAO;gBACrB,gBAAgB,EAAE,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,EAAE,CAAC;gBAC9C,IAAI,EAAE,UAAU,CAAC,IAAI,IAAI,KAAK;aAC/B,CAAC,CAAC;QACL,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,WAAW,CAAC,CAAC,EAAE,UAAU,CAAC,IAAI,CAAC,CAAC;QAClC,CAAC;gBAAS,CAAC;YACT,OAAO,CAAC,QAAQ,CAAC,CAAC;QACpB,CAAC;IACH,CAAC,CAAC,CAAC;AACP,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=stuck.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stuck.test.d.ts","sourceRoot":"","sources":["../../src/commands/stuck.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,51 @@
1
+ // packages/hzl-cli/src/commands/stuck.test.ts
2
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
3
+ import fs from 'fs';
4
+ import path from 'path';
5
+ import os from 'os';
6
+ import { runStuck } from './stuck.js';
7
+ import { initializeDb, closeDb } from '../db.js';
8
+ import { TaskStatus } from 'hzl-core/events/types.js';
9
+ describe('runStuck', () => {
10
+ let tempDir;
11
+ let dbPath;
12
+ let services;
13
+ beforeEach(() => {
14
+ tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'hzl-stuck-test-'));
15
+ dbPath = path.join(tempDir, 'test.db');
16
+ services = initializeDb(dbPath);
17
+ });
18
+ afterEach(() => {
19
+ closeDb(services);
20
+ fs.rmSync(tempDir, { recursive: true, force: true });
21
+ });
22
+ it('returns empty list when no stuck tasks', () => {
23
+ const result = runStuck({ services, json: false });
24
+ expect(result.tasks).toHaveLength(0);
25
+ });
26
+ it('finds tasks with expired leases', () => {
27
+ const task = services.taskService.createTask({ title: 'Stuck task', project: 'inbox' });
28
+ services.taskService.setStatus(task.task_id, TaskStatus.Ready);
29
+ // Claim with lease in the past
30
+ const pastLease = new Date(Date.now() - 60000).toISOString();
31
+ services.taskService.claimTask(task.task_id, { author: 'stalled-agent', lease_until: pastLease });
32
+ const result = runStuck({ services, json: false });
33
+ expect(result.tasks).toHaveLength(1);
34
+ expect(result.tasks[0].task_id).toBe(task.task_id);
35
+ });
36
+ it('filters by project', () => {
37
+ // Stuck task in project-a
38
+ const task1 = services.taskService.createTask({ title: 'Stuck task 1', project: 'project-a' });
39
+ services.taskService.setStatus(task1.task_id, TaskStatus.Ready);
40
+ const pastLease = new Date(Date.now() - 60000).toISOString();
41
+ services.taskService.claimTask(task1.task_id, { lease_until: pastLease });
42
+ // Stuck task in project-b
43
+ const task2 = services.taskService.createTask({ title: 'Stuck task 2', project: 'project-b' });
44
+ services.taskService.setStatus(task2.task_id, TaskStatus.Ready);
45
+ services.taskService.claimTask(task2.task_id, { lease_until: pastLease });
46
+ const result = runStuck({ services, project: 'project-a', json: false });
47
+ expect(result.tasks).toHaveLength(1);
48
+ expect(result.tasks[0].project).toBe('project-a');
49
+ });
50
+ });
51
+ //# sourceMappingURL=stuck.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stuck.test.js","sourceRoot":"","sources":["../../src/commands/stuck.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,32 @@
1
+ import { TaskStatus } from 'hzl-core/events/types.js';
2
+ import type { Services } from '../db.js';
3
+ import type { OutputFormatter } from '../output.js';
4
+ export interface CreateTaskInput {
5
+ title: string;
6
+ project?: string;
7
+ description?: string;
8
+ tags?: string[];
9
+ priority?: number;
10
+ depends_on?: string[];
11
+ }
12
+ export declare function createTask(services: Services, input: CreateTaskInput, author?: string, out?: OutputFormatter): string;
13
+ export declare function claimTask(services: Services, taskId: string, author?: string, leaseMinutes?: number, out?: OutputFormatter): void;
14
+ export declare function claimNext(services: Services, opts: {
15
+ project?: string;
16
+ tags?: string[];
17
+ author?: string;
18
+ leaseMinutes?: number;
19
+ }, out?: OutputFormatter): string | null;
20
+ export declare function setStatus(services: Services, taskId: string, status: TaskStatus, author?: string, out?: OutputFormatter): void;
21
+ export declare function completeTask(services: Services, taskId: string, author?: string, out?: OutputFormatter): void;
22
+ export declare function releaseTask(services: Services, taskId: string, reason?: string, author?: string, out?: OutputFormatter): void;
23
+ export declare function archiveTask(services: Services, taskId: string, reason?: string, author?: string, out?: OutputFormatter): void;
24
+ export declare function getTask(services: Services, taskId: string, out: OutputFormatter): void;
25
+ export declare function listTasks(services: Services, opts: {
26
+ project?: string;
27
+ status?: TaskStatus;
28
+ limit?: number;
29
+ }, out: OutputFormatter): void;
30
+ export declare function addComment(services: Services, taskId: string, text: string, author?: string, out?: OutputFormatter): void;
31
+ export declare function addCheckpoint(services: Services, taskId: string, name: string, data?: Record<string, unknown>, author?: string, out?: OutputFormatter): void;
32
+ //# sourceMappingURL=task.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"task.d.ts","sourceRoot":"","sources":["../../src/commands/task.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,0BAA0B,CAAC;AAEtD,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,cAAc,CAAC;AAGpD,MAAM,WAAW,eAAe;IAC9B,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;IAChB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;CACvB;AAED,wBAAgB,UAAU,CAAC,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,eAAe,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,eAAe,GAAG,MAAM,CAcrH;AAED,wBAAgB,SAAS,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,YAAY,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,eAAe,GAAG,IAAI,CAIjI;AAED,wBAAgB,SAAS,CAAC,QAAQ,EAAE,QAAQ,EAAE,IAAI,EAAE;IAAE,OAAO,CAAC,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;IAAC,MAAM,CAAC,EAAE,MAAM,CAAC;IAAC,YAAY,CAAC,EAAE,MAAM,CAAA;CAAE,EAAE,GAAG,CAAC,EAAE,eAAe,GAAG,MAAM,GAAG,IAAI,CAevK;AAED,wBAAgB,SAAS,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,eAAe,GAAG,IAAI,CAG9H;AAED,wBAAgB,YAAY,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,eAAe,GAAG,IAAI,CAG7G;AAED,wBAAgB,WAAW,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,eAAe,GAAG,IAAI,CAG7H;AAED,wBAAgB,WAAW,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,eAAe,GAAG,IAAI,CAG7H;AAED,wBAAgB,OAAO,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,EAAE,eAAe,GAAG,IAAI,CAItF;AAED,wBAAgB,SAAS,CAAC,QAAQ,EAAE,QAAQ,EAAE,IAAI,EAAE;IAAE,OAAO,CAAC,EAAE,MAAM,CAAC;IAAC,MAAM,CAAC,EAAE,UAAU,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAA;CAAE,EAAE,GAAG,EAAE,eAAe,GAAG,IAAI,CAKzI;AAED,wBAAgB,UAAU,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,eAAe,GAAG,IAAI,CAGzH;AAED,wBAAgB,aAAa,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,eAAe,GAAG,IAAI,CAG5J"}
@@ -0,0 +1,72 @@
1
+ import { CLIError, ExitCode } from '../errors.js';
2
+ export function createTask(services, input, author, out) {
3
+ const task = services.taskService.createTask({
4
+ title: input.title,
5
+ project: input.project ?? 'inbox',
6
+ description: input.description,
7
+ tags: input.tags,
8
+ priority: input.priority,
9
+ depends_on: input.depends_on,
10
+ }, { author });
11
+ out?.success(`Created task ${task.task_id}`);
12
+ return task.task_id;
13
+ }
14
+ export function claimTask(services, taskId, author, leaseMinutes, out) {
15
+ const leaseUntil = leaseMinutes ? new Date(Date.now() + leaseMinutes * 60000).toISOString() : undefined;
16
+ const task = services.taskService.claimTask(taskId, { author, lease_until: leaseUntil });
17
+ out?.success(`Claimed task ${task.task_id}`);
18
+ }
19
+ export function claimNext(services, opts, out) {
20
+ const leaseUntil = opts.leaseMinutes ? new Date(Date.now() + opts.leaseMinutes * 60000).toISOString() : undefined;
21
+ const task = services.taskService.claimNext({
22
+ project: opts.project,
23
+ tags: opts.tags,
24
+ author: opts.author,
25
+ lease_until: leaseUntil,
26
+ });
27
+ if (task) {
28
+ out?.success(`Claimed task ${task.task_id}: ${task.title}`);
29
+ return task.task_id;
30
+ }
31
+ else {
32
+ out?.text('No tasks available to claim');
33
+ return null;
34
+ }
35
+ }
36
+ export function setStatus(services, taskId, status, author, out) {
37
+ services.taskService.setStatus(taskId, status, { author });
38
+ out?.success(`Set task ${taskId} status to ${status}`);
39
+ }
40
+ export function completeTask(services, taskId, author, out) {
41
+ services.taskService.completeTask(taskId, { author });
42
+ out?.success(`Completed task ${taskId}`);
43
+ }
44
+ export function releaseTask(services, taskId, reason, author, out) {
45
+ services.taskService.releaseTask(taskId, { reason, author });
46
+ out?.success(`Released task ${taskId}`);
47
+ }
48
+ export function archiveTask(services, taskId, reason, author, out) {
49
+ services.taskService.archiveTask(taskId, { reason, author });
50
+ out?.success(`Archived task ${taskId}`);
51
+ }
52
+ export function getTask(services, taskId, out) {
53
+ const task = services.taskService.getTaskById(taskId);
54
+ if (!task)
55
+ throw new CLIError(`Task not found: ${taskId}`, ExitCode.NotFound);
56
+ out.json(task);
57
+ }
58
+ export function listTasks(services, opts, out) {
59
+ const tasks = services.taskService
60
+ .getAvailableTasks({ project: opts.project, limit: opts.limit ?? 50 })
61
+ .filter((t) => !opts.status || t.status === opts.status);
62
+ out.table(tasks, ['task_id', 'title', 'project', 'status', 'priority']);
63
+ }
64
+ export function addComment(services, taskId, text, author, out) {
65
+ services.taskService.addComment(taskId, text, { author });
66
+ out?.success(`Added comment to task ${taskId}`);
67
+ }
68
+ export function addCheckpoint(services, taskId, name, data, author, out) {
69
+ services.taskService.addCheckpoint(taskId, name, data, { author });
70
+ out?.success(`Added checkpoint "${name}" to task ${taskId}`);
71
+ }
72
+ //# sourceMappingURL=task.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"task.js","sourceRoot":"","sources":["../../src/commands/task.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,23 @@
1
+ import { Command } from 'commander';
2
+ import { type Services } from '../db.js';
3
+ export interface UpdateResult {
4
+ task_id: string;
5
+ title: string;
6
+ description: string | null;
7
+ priority: number;
8
+ tags: string[];
9
+ }
10
+ export interface TaskUpdates {
11
+ title?: string;
12
+ description?: string;
13
+ priority?: number;
14
+ tags?: string[];
15
+ }
16
+ export declare function runUpdate(options: {
17
+ services: Services;
18
+ taskId: string;
19
+ updates: TaskUpdates;
20
+ json: boolean;
21
+ }): UpdateResult;
22
+ export declare function createUpdateCommand(): Command;
23
+ //# sourceMappingURL=update.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"update.d.ts","sourceRoot":"","sources":["../../src/commands/update.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAEpC,OAAO,EAAyB,KAAK,QAAQ,EAAE,MAAM,UAAU,CAAC;AAKhE,MAAM,WAAW,YAAY;IAC3B,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,QAAQ,EAAE,MAAM,CAAC;IACjB,IAAI,EAAE,MAAM,EAAE,CAAC;CAChB;AAED,MAAM,WAAW,WAAW;IAC1B,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;CACjB;AAED,wBAAgB,SAAS,CAAC,OAAO,EAAE;IACjC,QAAQ,EAAE,QAAQ,CAAC;IACnB,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,WAAW,CAAC;IACrB,IAAI,EAAE,OAAO,CAAC;CACf,GAAG,YAAY,CAqEf;AAED,wBAAgB,mBAAmB,IAAI,OAAO,CA0B7C"}
@@ -0,0 +1,105 @@
1
+ // packages/hzl-cli/src/commands/update.ts
2
+ import { Command } from 'commander';
3
+ import { resolveDbPath } from '../config.js';
4
+ import { initializeDb, closeDb } from '../db.js';
5
+ import { handleError, CLIError, ExitCode } from '../errors.js';
6
+ import { EventType } from 'hzl-core/events/types.js';
7
+ export function runUpdate(options) {
8
+ const { services, taskId, updates, json } = options;
9
+ const { eventStore, projectionEngine } = services;
10
+ const task = services.taskService.getTaskById(taskId);
11
+ if (!task) {
12
+ throw new CLIError(`Task not found: ${taskId}`, ExitCode.NotFound);
13
+ }
14
+ // Emit one task_updated event per field change
15
+ // The TaskUpdatedSchema requires: field, old_value (optional), new_value
16
+ if (updates.title !== undefined && updates.title !== task.title) {
17
+ const event = eventStore.append({
18
+ task_id: taskId,
19
+ type: EventType.TaskUpdated,
20
+ data: { field: 'title', old_value: task.title, new_value: updates.title },
21
+ });
22
+ projectionEngine.applyEvent(event);
23
+ }
24
+ if (updates.description !== undefined && updates.description !== task.description) {
25
+ const event = eventStore.append({
26
+ task_id: taskId,
27
+ type: EventType.TaskUpdated,
28
+ data: { field: 'description', old_value: task.description, new_value: updates.description },
29
+ });
30
+ projectionEngine.applyEvent(event);
31
+ }
32
+ if (updates.priority !== undefined && updates.priority !== task.priority) {
33
+ const event = eventStore.append({
34
+ task_id: taskId,
35
+ type: EventType.TaskUpdated,
36
+ data: { field: 'priority', old_value: task.priority, new_value: updates.priority },
37
+ });
38
+ projectionEngine.applyEvent(event);
39
+ }
40
+ if (updates.tags !== undefined) {
41
+ const event = eventStore.append({
42
+ task_id: taskId,
43
+ type: EventType.TaskUpdated,
44
+ data: { field: 'tags', old_value: task.tags, new_value: updates.tags },
45
+ });
46
+ projectionEngine.applyEvent(event);
47
+ }
48
+ // Get updated task
49
+ const updatedTask = services.taskService.getTaskById(taskId);
50
+ const result = {
51
+ task_id: updatedTask.task_id,
52
+ title: updatedTask.title,
53
+ description: updatedTask.description,
54
+ priority: updatedTask.priority,
55
+ tags: updatedTask.tags,
56
+ };
57
+ if (json) {
58
+ console.log(JSON.stringify(result));
59
+ }
60
+ else {
61
+ console.log(`✓ Updated task ${taskId}`);
62
+ if (updates.title)
63
+ console.log(` Title: ${updatedTask.title}`);
64
+ if (updates.description)
65
+ console.log(` Description: ${updatedTask.description}`);
66
+ if (updates.priority !== undefined)
67
+ console.log(` Priority: ${updatedTask.priority}`);
68
+ if (updates.tags)
69
+ console.log(` Tags: ${updatedTask.tags.join(', ')}`);
70
+ }
71
+ return result;
72
+ }
73
+ export function createUpdateCommand() {
74
+ return new Command('update')
75
+ .description('Update task fields')
76
+ .argument('<taskId>', 'Task ID')
77
+ .option('--title <title>', 'New title')
78
+ .option('--desc <description>', 'New description')
79
+ .option('-p, --priority <n>', 'New priority (0-3)')
80
+ .option('-t, --tags <tags>', 'New tags (comma-separated)')
81
+ .action(function (taskId, opts) {
82
+ const globalOpts = this.optsWithGlobals();
83
+ const dbPath = resolveDbPath(globalOpts.db);
84
+ const services = initializeDb(dbPath);
85
+ try {
86
+ const updates = {};
87
+ if (opts.title)
88
+ updates.title = opts.title;
89
+ if (opts.desc)
90
+ updates.description = opts.desc;
91
+ if (opts.priority !== undefined)
92
+ updates.priority = parseInt(opts.priority, 10);
93
+ if (opts.tags)
94
+ updates.tags = opts.tags.split(',');
95
+ runUpdate({ services, taskId, updates, json: globalOpts.json ?? false });
96
+ }
97
+ catch (e) {
98
+ handleError(e, globalOpts.json);
99
+ }
100
+ finally {
101
+ closeDb(services);
102
+ }
103
+ });
104
+ }
105
+ //# sourceMappingURL=update.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"update.js","sourceRoot":"","sources":["../../src/commands/update.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=update.test.d.ts.map