ccjk 9.5.6 → 9.7.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 (103) hide show
  1. package/dist/chunks/agent.mjs +1 -1
  2. package/dist/chunks/api-providers.mjs +1 -1
  3. package/dist/chunks/api.mjs +3 -3
  4. package/dist/chunks/auto-bootstrap.mjs +1 -1
  5. package/dist/chunks/auto-updater.mjs +1 -1
  6. package/dist/chunks/boost.mjs +160 -0
  7. package/dist/chunks/ccjk-agents.mjs +1 -1
  8. package/dist/chunks/ccjk-all.mjs +1 -1
  9. package/dist/chunks/ccjk-config.mjs +1 -1
  10. package/dist/chunks/ccjk-hooks.mjs +1 -1
  11. package/dist/chunks/ccjk-mcp.mjs +2 -2
  12. package/dist/chunks/ccjk-setup.mjs +1 -1
  13. package/dist/chunks/ccjk-skills.mjs +1 -1
  14. package/dist/chunks/ccr.mjs +25 -30
  15. package/dist/chunks/ccu.mjs +1 -1
  16. package/dist/chunks/check-updates.mjs +3 -4
  17. package/dist/chunks/claude-code-config-manager.mjs +7 -7
  18. package/dist/chunks/claude-code-incremental-manager.mjs +2 -2
  19. package/dist/chunks/claude-config.mjs +4 -4
  20. package/dist/chunks/claude-wrapper.mjs +2 -2
  21. package/dist/chunks/codex-config-switch.mjs +4 -5
  22. package/dist/chunks/codex-provider-manager.mjs +2 -3
  23. package/dist/chunks/codex-uninstaller.mjs +2 -2
  24. package/dist/chunks/codex.mjs +207 -6
  25. package/dist/chunks/commands.mjs +391 -88
  26. package/dist/chunks/commands2.mjs +88 -391
  27. package/dist/chunks/completion.mjs +1 -1
  28. package/dist/chunks/config-consolidator.mjs +2 -2
  29. package/dist/chunks/config-switch.mjs +3 -4
  30. package/dist/chunks/config.mjs +78 -7
  31. package/dist/chunks/config2.mjs +400 -410
  32. package/dist/chunks/config3.mjs +410 -400
  33. package/dist/chunks/constants.mjs +1 -1
  34. package/dist/chunks/doctor.mjs +4 -4
  35. package/dist/chunks/features.mjs +24 -17
  36. package/dist/chunks/index.mjs +178 -7
  37. package/dist/chunks/index2.mjs +1162 -169
  38. package/dist/chunks/index3.mjs +910 -1076
  39. package/dist/chunks/index4.mjs +137 -947
  40. package/dist/chunks/index5.mjs +635 -167
  41. package/dist/chunks/init.mjs +141 -99
  42. package/dist/chunks/installer.mjs +147 -649
  43. package/dist/chunks/installer2.mjs +649 -147
  44. package/dist/chunks/interview.mjs +2 -2
  45. package/dist/chunks/marketplace.mjs +1 -1
  46. package/dist/chunks/mcp.mjs +1058 -17
  47. package/dist/chunks/menu.mjs +147 -56
  48. package/dist/chunks/monitor.mjs +2 -2
  49. package/dist/chunks/notification.mjs +1 -1
  50. package/dist/chunks/onboarding.mjs +2 -2
  51. package/dist/chunks/package.mjs +2 -210
  52. package/dist/chunks/permission-manager.mjs +2 -2
  53. package/dist/chunks/permissions.mjs +1 -1
  54. package/dist/chunks/platform.mjs +1 -1
  55. package/dist/chunks/plugin.mjs +1 -1
  56. package/dist/chunks/prompts.mjs +1 -1
  57. package/dist/chunks/providers.mjs +1 -1
  58. package/dist/chunks/quick-setup.mjs +16 -20
  59. package/dist/chunks/silent-updater.mjs +1 -1
  60. package/dist/chunks/simple-config.mjs +2 -2
  61. package/dist/chunks/skill.mjs +1 -1
  62. package/dist/chunks/skills-sync.mjs +1 -1
  63. package/dist/chunks/skills.mjs +1 -1
  64. package/dist/chunks/startup.mjs +1 -1
  65. package/dist/chunks/stats.mjs +1 -1
  66. package/dist/chunks/status.mjs +159 -0
  67. package/dist/chunks/team.mjs +1 -1
  68. package/dist/chunks/thinking.mjs +2 -2
  69. package/dist/chunks/uninstall.mjs +6 -6
  70. package/dist/chunks/update.mjs +6 -9
  71. package/dist/chunks/upgrade-manager.mjs +2 -2
  72. package/dist/chunks/version-checker.mjs +3 -3
  73. package/dist/chunks/vim.mjs +1 -1
  74. package/dist/chunks/workflows.mjs +616 -215
  75. package/dist/cli.mjs +70 -121
  76. package/dist/index.d.mts +17 -1482
  77. package/dist/index.d.ts +17 -1482
  78. package/dist/index.mjs +950 -4740
  79. package/dist/shared/{ccjk.zCqdxT2Y.mjs → ccjk.Br91zBIG.mjs} +2 -2
  80. package/dist/shared/ccjk.CSkyCZIM.mjs +638 -0
  81. package/dist/shared/{ccjk.BKoi8-Hy.mjs → ccjk.DE91nClQ.mjs} +1 -1
  82. package/dist/shared/{ccjk.f40us0yY.mjs → ccjk.DvIrK0wz.mjs} +2 -2
  83. package/dist/shared/ccjk.LsPZ2PYo.mjs +1048 -0
  84. package/dist/shared/{ccjk.DRweXU5F.mjs → ccjk.q1koQxEE.mjs} +2 -2
  85. package/package.json +1 -1
  86. package/templates/claude-code/common/settings.json +15 -111
  87. package/dist/chunks/api-adapter.mjs +0 -180
  88. package/dist/chunks/cli.mjs +0 -2227
  89. package/dist/chunks/context-menu.mjs +0 -913
  90. package/dist/chunks/hooks-sync.mjs +0 -1627
  91. package/dist/chunks/index6.mjs +0 -663
  92. package/dist/chunks/mcp-market.mjs +0 -1077
  93. package/dist/chunks/mcp-server.mjs +0 -776
  94. package/dist/chunks/project-detector.mjs +0 -131
  95. package/dist/chunks/provider-registry.mjs +0 -92
  96. package/dist/chunks/setup-wizard.mjs +0 -362
  97. package/dist/chunks/tools.mjs +0 -143
  98. package/dist/chunks/workflows2.mjs +0 -633
  99. package/dist/shared/ccjk.BM_HZogn.mjs +0 -347
  100. package/dist/shared/ccjk.BaEp4UHQ.mjs +0 -75
  101. package/dist/shared/ccjk.CS0ybJCf.mjs +0 -490
  102. package/dist/shared/ccjk.CZgIwikC.mjs +0 -209
  103. package/dist/shared/ccjk.tO8zeFh1.mjs +0 -397
@@ -1,633 +0,0 @@
1
- import ansis from 'ansis';
2
- import 'inquirer';
3
- import { EventEmitter } from 'node:events';
4
- import { existsSync, mkdirSync, writeFileSync, readFileSync } from 'node:fs';
5
- import { homedir } from 'node:os';
6
- import { join, dirname } from 'pathe';
7
-
8
- const PHASE_CONFIGS = {
9
- brainstorming: {
10
- phase: "brainstorming",
11
- name: {
12
- "en": "Brainstorming",
13
- "zh-CN": "\u5934\u8111\u98CE\u66B4",
14
- "ja-JP": "\u30D6\u30EC\u30A4\u30F3\u30B9\u30C8\u30FC\u30DF\u30F3\u30B0",
15
- "ko-KR": "\uBE0C\uB808\uC778\uC2A4\uD1A0\uBC0D"
16
- },
17
- description: {
18
- "en": "Explore ideas and gather requirements",
19
- "zh-CN": "\u63A2\u7D22\u60F3\u6CD5\uFF0C\u6536\u96C6\u9700\u6C42",
20
- "ja-JP": "\u30A2\u30A4\u30C7\u30A2\u3092\u63A2\u6C42\u3057\u3001\u8981\u4EF6\u3092\u53CE\u96C6\u3059\u308B",
21
- "ko-KR": "\uC544\uC774\uB514\uC5B4 \uD0D0\uC0C9 \uBC0F \uC694\uAD6C\uC0AC\uD56D \uC218\uC9D1"
22
- },
23
- autoActivateSkills: ["brainstorming", "requirements"],
24
- allowedTransitions: ["planning"],
25
- requiresConfirmation: true,
26
- maxDuration: 30
27
- },
28
- planning: {
29
- phase: "planning",
30
- name: {
31
- "en": "Planning",
32
- "zh-CN": "\u89C4\u5212",
33
- "ja-JP": "\u8A08\u753B",
34
- "ko-KR": "\uACC4\uD68D"
35
- },
36
- description: {
37
- "en": "Create detailed implementation plan with bite-sized tasks",
38
- "zh-CN": "\u521B\u5EFA\u8BE6\u7EC6\u7684\u5B9E\u65BD\u8BA1\u5212\uFF0C\u5206\u89E3\u4E3A\u5C0F\u4EFB\u52A1",
39
- "ja-JP": "\u8A73\u7D30\u306A\u5B9F\u88C5\u8A08\u753B\u3092\u4F5C\u6210\u3057\u3001\u5C0F\u3055\u306A\u30BF\u30B9\u30AF\u306B\u5206\u89E3\u3059\u308B",
40
- "ko-KR": "\uC138\uBD80 \uAD6C\uD604 \uACC4\uD68D \uC791\uC131 \uBC0F \uC791\uC740 \uC791\uC5C5\uC73C\uB85C \uBD84\uD574"
41
- },
42
- autoActivateSkills: ["planning", "task-breakdown"],
43
- allowedTransitions: ["implementation", "brainstorming"],
44
- requiresConfirmation: true,
45
- maxDuration: 60
46
- },
47
- implementation: {
48
- phase: "implementation",
49
- name: {
50
- "en": "Implementation",
51
- "zh-CN": "\u5B9E\u73B0",
52
- "ja-JP": "\u5B9F\u88C5",
53
- "ko-KR": "\uAD6C\uD604"
54
- },
55
- description: {
56
- "en": "Execute tasks via subagents with TDD approach",
57
- "zh-CN": "\u901A\u8FC7\u5B50\u4EE3\u7406\u6267\u884C\u4EFB\u52A1\uFF0C\u91C7\u7528 TDD \u65B9\u6CD5",
58
- "ja-JP": "TDD\u30A2\u30D7\u30ED\u30FC\u30C1\u3067\u30B5\u30D6\u30A8\u30FC\u30B8\u30A7\u30F3\u30C8\u3092\u4ECB\u3057\u3066\u30BF\u30B9\u30AF\u3092\u5B9F\u884C",
59
- "ko-KR": "TDD \uC811\uADFC \uBC29\uC2DD\uC73C\uB85C \uC11C\uBE0C\uC5D0\uC774\uC804\uD2B8\uB97C \uD1B5\uD574 \uC791\uC5C5 \uC2E4\uD589"
60
- },
61
- autoActivateSkills: ["implementation", "tdd", "coding"],
62
- allowedTransitions: ["review", "planning"],
63
- requiresConfirmation: false,
64
- maxDuration: 0
65
- // Unlimited
66
- },
67
- review: {
68
- phase: "review",
69
- name: {
70
- "en": "Code Review",
71
- "zh-CN": "\u4EE3\u7801\u5BA1\u67E5",
72
- "ja-JP": "\u30B3\u30FC\u30C9\u30EC\u30D3\u30E5\u30FC",
73
- "ko-KR": "\uCF54\uB4DC \uB9AC\uBDF0"
74
- },
75
- description: {
76
- "en": "Two-stage review: spec compliance + code quality",
77
- "zh-CN": "\u4E24\u9636\u6BB5\u5BA1\u67E5\uFF1A\u89C4\u683C\u7B26\u5408\u6027 + \u4EE3\u7801\u8D28\u91CF",
78
- "ja-JP": "2\u6BB5\u968E\u30EC\u30D3\u30E5\u30FC\uFF1A\u4ED5\u69D8\u6E96\u62E0 + \u30B3\u30FC\u30C9\u54C1\u8CEA",
79
- "ko-KR": "2\uB2E8\uACC4 \uAC80\uD1A0: \uC0AC\uC591 \uC900\uC218 + \uCF54\uB4DC \uD488\uC9C8"
80
- },
81
- autoActivateSkills: ["code-review", "quality-check"],
82
- allowedTransitions: ["finishing", "implementation"],
83
- requiresConfirmation: true,
84
- maxDuration: 30
85
- },
86
- finishing: {
87
- phase: "finishing",
88
- name: {
89
- "en": "Finishing",
90
- "zh-CN": "\u6536\u5C3E",
91
- "ja-JP": "\u4ED5\u4E0A\u3052",
92
- "ko-KR": "\uB9C8\uBB34\uB9AC"
93
- },
94
- description: {
95
- "en": "Final cleanup, documentation, and merge",
96
- "zh-CN": "\u6700\u7EC8\u6E05\u7406\u3001\u6587\u6863\u548C\u5408\u5E76",
97
- "ja-JP": "\u6700\u7D42\u30AF\u30EA\u30FC\u30F3\u30A2\u30C3\u30D7\u3001\u30C9\u30AD\u30E5\u30E1\u30F3\u30C8\u3001\u30DE\u30FC\u30B8",
98
- "ko-KR": "\uCD5C\uC885 \uC815\uB9AC, \uBB38\uC11C\uD654 \uBC0F \uBCD1\uD569"
99
- },
100
- autoActivateSkills: ["finishing", "documentation"],
101
- allowedTransitions: [],
102
- requiresConfirmation: true,
103
- maxDuration: 15
104
- }
105
- };
106
- const WORKFLOW_PERSISTENCE_VERSION = 1;
107
-
108
- function generateId() {
109
- return `wf-${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;
110
- }
111
- const DEFAULT_OPTIONS = {
112
- persistPath: join(homedir(), ".ccjk", "workflow-state.json"),
113
- autoSave: true,
114
- verbose: false
115
- };
116
- class WorkflowStateMachine extends EventEmitter {
117
- sessions = /* @__PURE__ */ new Map();
118
- options;
119
- constructor(options = {}) {
120
- super();
121
- this.options = { ...DEFAULT_OPTIONS, ...options };
122
- this.loadState();
123
- }
124
- // ==========================================================================
125
- // Session Management
126
- // ==========================================================================
127
- /**
128
- * Create a new workflow session
129
- *
130
- * @param params - Session parameters
131
- * @param params.name - Session name
132
- * @param params.description - Optional session description
133
- * @param params.initialPhase - Optional initial workflow phase
134
- * @param params.branch - Optional git branch name
135
- * @param params.skills - Optional list of skill IDs to use
136
- * @param params.metadata - Optional additional metadata
137
- * @returns Created session
138
- */
139
- createSession(params) {
140
- const session = {
141
- id: generateId(),
142
- name: params.name,
143
- description: params.description || "",
144
- currentPhase: params.initialPhase || "brainstorming",
145
- status: "active",
146
- tasks: [],
147
- phaseHistory: [{
148
- from: null,
149
- to: params.initialPhase || "brainstorming",
150
- timestamp: /* @__PURE__ */ new Date(),
151
- reason: "Session created",
152
- triggeredBy: "system"
153
- }],
154
- branch: params.branch,
155
- skills: params.skills || [],
156
- metadata: params.metadata || {},
157
- createdAt: /* @__PURE__ */ new Date(),
158
- updatedAt: /* @__PURE__ */ new Date()
159
- };
160
- this.sessions.set(session.id, session);
161
- this.emit("session:created", session);
162
- this.log(`Created session: ${session.id} - ${session.name}`);
163
- this.saveState();
164
- return session;
165
- }
166
- /**
167
- * Get a session by ID
168
- */
169
- getSession(id) {
170
- return this.sessions.get(id) || null;
171
- }
172
- /**
173
- * Get all sessions
174
- */
175
- getAllSessions() {
176
- return Array.from(this.sessions.values());
177
- }
178
- /**
179
- * Get active sessions
180
- */
181
- getActiveSessions() {
182
- return this.getAllSessions().filter((s) => s.status === "active" || s.status === "paused");
183
- }
184
- /**
185
- * Update session metadata
186
- */
187
- updateSession(id, updates) {
188
- const session = this.getSessionOrThrow(id);
189
- Object.assign(session, updates, { updatedAt: /* @__PURE__ */ new Date() });
190
- this.saveState();
191
- return session;
192
- }
193
- /**
194
- * Delete a session
195
- */
196
- deleteSession(id) {
197
- const session = this.sessions.get(id);
198
- if (!session) {
199
- return false;
200
- }
201
- this.sessions.delete(id);
202
- this.log(`Deleted session: ${id}`);
203
- this.saveState();
204
- return true;
205
- }
206
- // ==========================================================================
207
- // Phase Transitions
208
- // ==========================================================================
209
- /**
210
- * Transition to a new phase
211
- *
212
- * @param sessionId - Session ID
213
- * @param targetPhase - Target phase
214
- * @param reason - Reason for transition
215
- * @returns Updated session
216
- * @throws Error if transition is not allowed
217
- */
218
- transitionTo(sessionId, targetPhase, reason) {
219
- const session = this.getSessionOrThrow(sessionId);
220
- this.validateTransition(session, targetPhase);
221
- const oldPhase = session.currentPhase;
222
- const transition = {
223
- from: oldPhase,
224
- to: targetPhase,
225
- timestamp: /* @__PURE__ */ new Date(),
226
- reason: reason || `Transition from ${oldPhase} to ${targetPhase}`,
227
- triggeredBy: "user"
228
- };
229
- session.currentPhase = targetPhase;
230
- session.phaseHistory.push(transition);
231
- session.updatedAt = /* @__PURE__ */ new Date();
232
- this.emit("phase:changed", session, transition);
233
- this.log(`Phase transition: ${oldPhase} -> ${targetPhase} (${session.name})`);
234
- this.saveState();
235
- return session;
236
- }
237
- /**
238
- * Check if a transition is allowed
239
- */
240
- canTransitionTo(sessionId, targetPhase) {
241
- const session = this.sessions.get(sessionId);
242
- if (!session) {
243
- return false;
244
- }
245
- const currentConfig = PHASE_CONFIGS[session.currentPhase];
246
- return currentConfig.allowedTransitions.includes(targetPhase);
247
- }
248
- /**
249
- * Get allowed transitions for a session
250
- */
251
- getAllowedTransitions(sessionId) {
252
- const session = this.sessions.get(sessionId);
253
- if (!session) {
254
- return [];
255
- }
256
- return PHASE_CONFIGS[session.currentPhase].allowedTransitions;
257
- }
258
- /**
259
- * Auto-advance to next phase if conditions are met
260
- */
261
- autoAdvance(sessionId) {
262
- const session = this.getSessionOrThrow(sessionId);
263
- const currentConfig = PHASE_CONFIGS[session.currentPhase];
264
- if (currentConfig.requiresConfirmation) {
265
- return null;
266
- }
267
- const phaseTasks = this.getTasksForPhase(session, session.currentPhase);
268
- const allCompleted = phaseTasks.every((t) => t.status === "completed");
269
- if (!allCompleted || phaseTasks.length === 0) {
270
- return null;
271
- }
272
- const nextPhase = currentConfig.allowedTransitions[0];
273
- if (!nextPhase) {
274
- return null;
275
- }
276
- return this.transitionTo(sessionId, nextPhase, "Auto-advanced after task completion");
277
- }
278
- // ==========================================================================
279
- // Session Status Management
280
- // ==========================================================================
281
- /**
282
- * Pause a session
283
- */
284
- pauseSession(sessionId) {
285
- const session = this.getSessionOrThrow(sessionId);
286
- if (session.status !== "active") {
287
- throw new Error(`Cannot pause session with status: ${session.status}`);
288
- }
289
- return this.changeSessionStatus(session, "paused");
290
- }
291
- /**
292
- * Resume a paused session
293
- */
294
- resumeSession(sessionId) {
295
- const session = this.getSessionOrThrow(sessionId);
296
- if (session.status !== "paused") {
297
- throw new Error(`Cannot resume session with status: ${session.status}`);
298
- }
299
- return this.changeSessionStatus(session, "active");
300
- }
301
- /**
302
- * Complete a session
303
- */
304
- completeSession(sessionId) {
305
- const session = this.getSessionOrThrow(sessionId);
306
- if (session.status !== "active") {
307
- throw new Error(`Cannot complete session with status: ${session.status}`);
308
- }
309
- session.completedAt = /* @__PURE__ */ new Date();
310
- const updated = this.changeSessionStatus(session, "completed");
311
- this.emit("workflow:completed", updated);
312
- return updated;
313
- }
314
- /**
315
- * Fail a session
316
- */
317
- failSession(sessionId, error) {
318
- const session = this.getSessionOrThrow(sessionId);
319
- session.error = error;
320
- const updated = this.changeSessionStatus(session, "failed");
321
- this.emit("workflow:failed", updated, error);
322
- return updated;
323
- }
324
- /**
325
- * Cancel a session
326
- */
327
- cancelSession(sessionId) {
328
- const session = this.getSessionOrThrow(sessionId);
329
- if (session.status === "completed" || session.status === "failed") {
330
- throw new Error(`Cannot cancel session with status: ${session.status}`);
331
- }
332
- return this.changeSessionStatus(session, "cancelled");
333
- }
334
- // ==========================================================================
335
- // Task Management
336
- // ==========================================================================
337
- /**
338
- * Add a task to a session
339
- */
340
- addTask(sessionId, task) {
341
- const session = this.getSessionOrThrow(sessionId);
342
- const newTask = {
343
- ...task,
344
- id: `task-${Date.now()}-${Math.random().toString(36).substring(2, 7)}`,
345
- status: "pending",
346
- createdAt: /* @__PURE__ */ new Date()
347
- };
348
- session.tasks.push(newTask);
349
- session.updatedAt = /* @__PURE__ */ new Date();
350
- this.emit("task:created", session, newTask);
351
- this.saveState();
352
- return newTask;
353
- }
354
- /**
355
- * Update task status
356
- */
357
- updateTaskStatus(sessionId, taskId, status) {
358
- const session = this.getSessionOrThrow(sessionId);
359
- const task = session.tasks.find((t) => t.id === taskId);
360
- if (!task) {
361
- throw new Error(`Task not found: ${taskId}`);
362
- }
363
- const oldStatus = task.status;
364
- task.status = status;
365
- if (status === "running" && !task.startedAt) {
366
- task.startedAt = /* @__PURE__ */ new Date();
367
- }
368
- if (status === "completed" || status === "failed" || status === "cancelled") {
369
- task.completedAt = /* @__PURE__ */ new Date();
370
- if (task.startedAt) {
371
- task.actualMinutes = Math.round((task.completedAt.getTime() - task.startedAt.getTime()) / 6e4);
372
- }
373
- }
374
- session.updatedAt = /* @__PURE__ */ new Date();
375
- this.emit("task:status", session, task, oldStatus, status);
376
- if (status === "completed") {
377
- this.emit("task:completed", session, task);
378
- } else if (status === "failed") {
379
- this.emit("task:failed", session, task, task.error || "Unknown error");
380
- }
381
- this.saveState();
382
- return task;
383
- }
384
- /**
385
- * Get tasks for a specific phase
386
- */
387
- getTasksForPhase(session, phase) {
388
- return session.tasks.filter((t) => t.metadata?.phase === phase);
389
- }
390
- /**
391
- * Get pending tasks
392
- */
393
- getPendingTasks(sessionId) {
394
- const session = this.getSessionOrThrow(sessionId);
395
- return session.tasks.filter((t) => t.status === "pending" || t.status === "queued");
396
- }
397
- /**
398
- * Get running tasks
399
- */
400
- getRunningTasks(sessionId) {
401
- const session = this.getSessionOrThrow(sessionId);
402
- return session.tasks.filter((t) => t.status === "running");
403
- }
404
- // ==========================================================================
405
- // Persistence
406
- // ==========================================================================
407
- /**
408
- * Save state to disk
409
- */
410
- saveState() {
411
- if (!this.options.autoSave) {
412
- return;
413
- }
414
- const state = {
415
- version: WORKFLOW_PERSISTENCE_VERSION,
416
- sessions: Array.from(this.sessions.values()),
417
- lastUpdated: /* @__PURE__ */ new Date()
418
- };
419
- try {
420
- const dir = dirname(this.options.persistPath);
421
- if (!existsSync(dir)) {
422
- mkdirSync(dir, { recursive: true });
423
- }
424
- writeFileSync(
425
- this.options.persistPath,
426
- JSON.stringify(state, null, 2),
427
- "utf-8"
428
- );
429
- this.log(`State saved to ${this.options.persistPath}`);
430
- } catch (error) {
431
- console.error("Failed to save workflow state:", error);
432
- }
433
- }
434
- /**
435
- * Load state from disk
436
- */
437
- loadState() {
438
- try {
439
- if (!existsSync(this.options.persistPath)) {
440
- this.log("No existing state file found");
441
- return;
442
- }
443
- const content = readFileSync(this.options.persistPath, "utf-8");
444
- const state = JSON.parse(content);
445
- if (state.version !== WORKFLOW_PERSISTENCE_VERSION) {
446
- this.log(`Migrating state from version ${state.version} to ${WORKFLOW_PERSISTENCE_VERSION}`);
447
- }
448
- this.sessions.clear();
449
- for (const session of state.sessions) {
450
- session.createdAt = new Date(session.createdAt);
451
- session.updatedAt = new Date(session.updatedAt);
452
- if (session.completedAt) {
453
- session.completedAt = new Date(session.completedAt);
454
- }
455
- for (const task of session.tasks) {
456
- task.createdAt = new Date(task.createdAt);
457
- if (task.startedAt)
458
- task.startedAt = new Date(task.startedAt);
459
- if (task.completedAt)
460
- task.completedAt = new Date(task.completedAt);
461
- }
462
- for (const transition of session.phaseHistory) {
463
- transition.timestamp = new Date(transition.timestamp);
464
- }
465
- this.sessions.set(session.id, session);
466
- }
467
- this.log(`Loaded ${this.sessions.size} sessions from state file`);
468
- } catch (error) {
469
- console.error("Failed to load workflow state:", error);
470
- }
471
- }
472
- /**
473
- * Clear all state
474
- */
475
- clearState() {
476
- this.sessions.clear();
477
- this.saveState();
478
- this.log("State cleared");
479
- }
480
- // ==========================================================================
481
- // Statistics
482
- // ==========================================================================
483
- /**
484
- * Get workflow statistics
485
- */
486
- getStats() {
487
- const sessions = this.getAllSessions();
488
- const allTasks = sessions.flatMap((s) => s.tasks);
489
- const completedTasks = allTasks.filter((t) => t.status === "completed" && t.actualMinutes);
490
- const avgDuration = completedTasks.length > 0 ? completedTasks.reduce((sum, t) => sum + (t.actualMinutes || 0), 0) / completedTasks.length : 0;
491
- return {
492
- totalSessions: sessions.length,
493
- activeSessions: sessions.filter((s) => s.status === "active").length,
494
- completedSessions: sessions.filter((s) => s.status === "completed").length,
495
- failedSessions: sessions.filter((s) => s.status === "failed").length,
496
- totalTasks: allTasks.length,
497
- completedTasks: completedTasks.length,
498
- averageTaskDuration: Math.round(avgDuration * 10) / 10
499
- };
500
- }
501
- // ==========================================================================
502
- // Private Helpers
503
- // ==========================================================================
504
- getSessionOrThrow(id) {
505
- const session = this.sessions.get(id);
506
- if (!session) {
507
- throw new Error(`Session not found: ${id}`);
508
- }
509
- return session;
510
- }
511
- validateTransition(session, targetPhase) {
512
- if (session.status !== "active") {
513
- throw new Error(`Cannot transition session with status: ${session.status}`);
514
- }
515
- const currentConfig = PHASE_CONFIGS[session.currentPhase];
516
- if (!currentConfig.allowedTransitions.includes(targetPhase)) {
517
- throw new Error(
518
- `Invalid transition: ${session.currentPhase} -> ${targetPhase}. Allowed: ${currentConfig.allowedTransitions.join(", ")}`
519
- );
520
- }
521
- }
522
- changeSessionStatus(session, newStatus) {
523
- const oldStatus = session.status;
524
- session.status = newStatus;
525
- session.updatedAt = /* @__PURE__ */ new Date();
526
- this.emit("session:status", session, oldStatus, newStatus);
527
- this.log(`Session status: ${oldStatus} -> ${newStatus} (${session.name})`);
528
- this.saveState();
529
- return session;
530
- }
531
- log(message) {
532
- if (this.options.verbose) {
533
- console.log(`[WorkflowStateMachine] ${message}`);
534
- }
535
- }
536
- // ==========================================================================
537
- // Type-safe Event Emitter
538
- // ==========================================================================
539
- on(event, listener) {
540
- return super.on(event, listener);
541
- }
542
- emit(event, ...args) {
543
- return super.emit(event, ...args);
544
- }
545
- once(event, listener) {
546
- return super.once(event, listener);
547
- }
548
- off(event, listener) {
549
- return super.off(event, listener);
550
- }
551
- }
552
- let instance = null;
553
- function getWorkflowStateMachine(options) {
554
- if (!instance) {
555
- instance = new WorkflowStateMachine(options);
556
- }
557
- return instance;
558
- }
559
-
560
- function listWorkflows() {
561
- const machine = getWorkflowStateMachine();
562
- return machine.getAllSessions();
563
- }
564
-
565
- const PHASE_ICONS = {
566
- brainstorming: "\u{1F4A1}",
567
- planning: "\u{1F4CB}",
568
- implementation: "\u{1F528}",
569
- review: "\u{1F50D}",
570
- finishing: "\u2705"
571
- };
572
- const PHASE_COLORS = {
573
- brainstorming: ansis.magenta,
574
- planning: ansis.green,
575
- implementation: ansis.yellow,
576
- review: ansis.green,
577
- finishing: ansis.green
578
- };
579
- const STATUS_ICONS = {
580
- active: "\u{1F504}",
581
- paused: "\u23F8\uFE0F",
582
- completed: "\u2705",
583
- failed: "\u274C",
584
- cancelled: "\u{1F6AB}"
585
- };
586
- function formatPhase(phase) {
587
- const icon = PHASE_ICONS[phase];
588
- const color = PHASE_COLORS[phase];
589
- return `${icon} ${color(phase.charAt(0).toUpperCase() + phase.slice(1))}`;
590
- }
591
- function getTaskCounts(session) {
592
- const tasks = session.tasks || [];
593
- return {
594
- completed: tasks.filter((t) => t.status === "completed").length,
595
- failed: tasks.filter((t) => t.status === "failed").length,
596
- total: tasks.length
597
- };
598
- }
599
- async function listAllWorkflows(options = {}) {
600
- const workflows = listWorkflows();
601
- if (options.format === "json") {
602
- console.log(JSON.stringify(workflows, null, 2));
603
- return;
604
- }
605
- if (workflows.length === 0) {
606
- console.log(ansis.yellow("\n \u26A0\uFE0F No workflows found\n"));
607
- return;
608
- }
609
- console.log("");
610
- console.log(ansis.bold.cyan("\u2501".repeat(80)));
611
- console.log(ansis.bold(` ${"ID".padEnd(10)} ${"Name".padEnd(20)} ${"Phase".padEnd(15)} ${"Status".padEnd(10)} ${"Progress".padEnd(15)}`));
612
- console.log(ansis.bold.cyan("\u2501".repeat(80)));
613
- for (const wf of workflows) {
614
- const counts = getTaskCounts(wf);
615
- const progress = `${counts.completed}/${counts.total}`;
616
- console.log(
617
- ` ${ansis.dim(wf.id.slice(0, 8).padEnd(10))} ${wf.name.slice(0, 18).padEnd(20)} ${formatPhase(wf.currentPhase).padEnd(25)} ${STATUS_ICONS[wf.status]} ${wf.status.padEnd(8)} ${progress}`
618
- );
619
- }
620
- console.log(ansis.bold.cyan("\u2501".repeat(80)));
621
- console.log(ansis.dim(` Total: ${workflows.length} workflows`));
622
- console.log("");
623
- }
624
-
625
- async function listWorkflowsQuick(options = {}) {
626
- const opts = {
627
- lang: options.lang,
628
- format: options.format
629
- };
630
- await listAllWorkflows(opts);
631
- }
632
-
633
- export { listWorkflowsQuick };