@winspan/claude-forge 8.34.0 → 8.36.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 (186) hide show
  1. package/README.md +10 -30
  2. package/dist/capability/index.d.ts +6 -5
  3. package/dist/capability/index.d.ts.map +1 -1
  4. package/dist/capability/index.js +6 -5
  5. package/dist/capability/index.js.map +1 -1
  6. package/dist/capability/types.d.ts +5 -74
  7. package/dist/capability/types.d.ts.map +1 -1
  8. package/dist/capability/types.js +4 -1
  9. package/dist/capability/types.js.map +1 -1
  10. package/dist/core/ai/provider.d.ts +23 -1
  11. package/dist/core/ai/provider.d.ts.map +1 -1
  12. package/dist/core/ai/provider.js +67 -1
  13. package/dist/core/ai/provider.js.map +1 -1
  14. package/dist/core/ai/types.d.ts +28 -0
  15. package/dist/core/ai/types.d.ts.map +1 -1
  16. package/dist/core/storage/rows.d.ts +0 -36
  17. package/dist/core/storage/rows.d.ts.map +1 -1
  18. package/dist/core/storage/schema.sql +1 -45
  19. package/dist/core/storage/sqlite.d.ts +2 -79
  20. package/dist/core/storage/sqlite.d.ts.map +1 -1
  21. package/dist/core/storage/sqlite.js +3 -362
  22. package/dist/core/storage/sqlite.js.map +1 -1
  23. package/dist/core/utils/token-tracker.d.ts +0 -1
  24. package/dist/core/utils/token-tracker.d.ts.map +1 -1
  25. package/dist/core/utils/token-tracker.js +0 -1
  26. package/dist/core/utils/token-tracker.js.map +1 -1
  27. package/dist/daemon/handlers/post-tool-use.d.ts +1 -9
  28. package/dist/daemon/handlers/post-tool-use.d.ts.map +1 -1
  29. package/dist/daemon/handlers/post-tool-use.js +2 -99
  30. package/dist/daemon/handlers/post-tool-use.js.map +1 -1
  31. package/dist/daemon/handlers/stop.d.ts +1 -15
  32. package/dist/daemon/handlers/stop.d.ts.map +1 -1
  33. package/dist/daemon/handlers/stop.js +1 -95
  34. package/dist/daemon/handlers/stop.js.map +1 -1
  35. package/dist/daemon/handlers/user-prompt.d.ts +1 -5
  36. package/dist/daemon/handlers/user-prompt.d.ts.map +1 -1
  37. package/dist/daemon/handlers/user-prompt.js +29 -99
  38. package/dist/daemon/handlers/user-prompt.js.map +1 -1
  39. package/dist/daemon/index.d.ts.map +1 -1
  40. package/dist/daemon/index.js +9 -56
  41. package/dist/daemon/index.js.map +1 -1
  42. package/dist/engine/agent-router.d.ts +37 -0
  43. package/dist/engine/agent-router.d.ts.map +1 -1
  44. package/dist/engine/agent-router.js +58 -0
  45. package/dist/engine/agent-router.js.map +1 -1
  46. package/dist/engine/conventions/routing.yaml +31 -2
  47. package/dist/intelligence/classifier.d.ts +63 -43
  48. package/dist/intelligence/classifier.d.ts.map +1 -1
  49. package/dist/intelligence/classifier.js +256 -191
  50. package/dist/intelligence/classifier.js.map +1 -1
  51. package/dist/intelligence/context-gatherer.d.ts +101 -0
  52. package/dist/intelligence/context-gatherer.d.ts.map +1 -0
  53. package/dist/intelligence/context-gatherer.js +417 -0
  54. package/dist/intelligence/context-gatherer.js.map +1 -0
  55. package/dist/intelligence/cot-classifier.d.ts +95 -0
  56. package/dist/intelligence/cot-classifier.d.ts.map +1 -0
  57. package/dist/intelligence/cot-classifier.js +391 -0
  58. package/dist/intelligence/cot-classifier.js.map +1 -0
  59. package/dist/intelligence/execution-doc-builder.d.ts +90 -0
  60. package/dist/intelligence/execution-doc-builder.d.ts.map +1 -1
  61. package/dist/intelligence/execution-doc-builder.js +459 -42
  62. package/dist/intelligence/execution-doc-builder.js.map +1 -1
  63. package/dist/intelligence/intent-types.d.ts +13 -0
  64. package/dist/intelligence/intent-types.d.ts.map +1 -0
  65. package/dist/intelligence/intent-types.js +19 -0
  66. package/dist/intelligence/intent-types.js.map +1 -0
  67. package/dist/intelligence/multimodal-parser.d.ts +105 -0
  68. package/dist/intelligence/multimodal-parser.d.ts.map +1 -0
  69. package/dist/intelligence/multimodal-parser.js +425 -0
  70. package/dist/intelligence/multimodal-parser.js.map +1 -0
  71. package/dist/skills/official-skills.d.ts.map +1 -1
  72. package/dist/skills/official-skills.js +218 -20
  73. package/dist/skills/official-skills.js.map +1 -1
  74. package/dist/web/routes/status.d.ts.map +1 -1
  75. package/dist/web/routes/status.js +8 -10
  76. package/dist/web/routes/status.js.map +1 -1
  77. package/dist/web/routes/token-usage.d.ts +1 -1
  78. package/dist/web/routes/token-usage.d.ts.map +1 -1
  79. package/dist/web/routes/token-usage.js +1 -16
  80. package/dist/web/routes/token-usage.js.map +1 -1
  81. package/dist/web/routes/types.d.ts +0 -6
  82. package/dist/web/routes/types.d.ts.map +1 -1
  83. package/dist/web/routes/types.js.map +1 -1
  84. package/dist/web/server.d.ts +0 -6
  85. package/dist/web/server.d.ts.map +1 -1
  86. package/dist/web/server.js +0 -5
  87. package/dist/web/server.js.map +1 -1
  88. package/dist/web/static/assets/{AIConfig-DiUFET_Q.js → AIConfig-D4VglzCl.js} +2 -2
  89. package/dist/web/static/assets/{AIConfig-DiUFET_Q.js.map → AIConfig-D4VglzCl.js.map} +1 -1
  90. package/dist/web/static/assets/{Agents-bNNGbQnL.js → Agents-ne5lXc7V.js} +2 -2
  91. package/dist/web/static/assets/{Agents-bNNGbQnL.js.map → Agents-ne5lXc7V.js.map} +1 -1
  92. package/dist/web/static/assets/Dashboard-D4j0Zmek.js +2 -0
  93. package/dist/web/static/assets/Dashboard-D4j0Zmek.js.map +1 -0
  94. package/dist/web/static/assets/{Drawer-DOUcx6m1.js → Drawer-Lo5ihVP-.js} +2 -2
  95. package/dist/web/static/assets/{Drawer-DOUcx6m1.js.map → Drawer-Lo5ihVP-.js.map} +1 -1
  96. package/dist/web/static/assets/{Events-DQHP6Uaq.js → Events-DBJ1B7OW.js} +2 -2
  97. package/dist/web/static/assets/{Events-DQHP6Uaq.js.map → Events-DBJ1B7OW.js.map} +1 -1
  98. package/dist/web/static/assets/{ExecutionTrace-Co8ARdg-.js → ExecutionTrace-Du9XADc1.js} +2 -2
  99. package/dist/web/static/assets/{ExecutionTrace-Co8ARdg-.js.map → ExecutionTrace-Du9XADc1.js.map} +1 -1
  100. package/dist/web/static/assets/{Routing-BW3eGD-8.js → Routing-BNQ09OlH.js} +2 -2
  101. package/dist/web/static/assets/{Routing-BW3eGD-8.js.map → Routing-BNQ09OlH.js.map} +1 -1
  102. package/dist/web/static/assets/{SessionDetail-Cbd7Jwox.js → SessionDetail-BPrPyMNa.js} +2 -2
  103. package/dist/web/static/assets/{SessionDetail-Cbd7Jwox.js.map → SessionDetail-BPrPyMNa.js.map} +1 -1
  104. package/dist/web/static/assets/{Sessions-ZQSCgXyy.js → Sessions-o3EXsXz9.js} +2 -2
  105. package/dist/web/static/assets/{Sessions-ZQSCgXyy.js.map → Sessions-o3EXsXz9.js.map} +1 -1
  106. package/dist/web/static/assets/{Skills-C5-5zOSH.js → Skills-Czt5mkyc.js} +2 -2
  107. package/dist/web/static/assets/{Skills-C5-5zOSH.js.map → Skills-Czt5mkyc.js.map} +1 -1
  108. package/dist/web/static/assets/{export-CbQTOt71.js → export-C0mlC4AT.js} +2 -2
  109. package/dist/web/static/assets/{export-CbQTOt71.js.map → export-C0mlC4AT.js.map} +1 -1
  110. package/dist/web/static/assets/index-B1J7nBu0.js +3 -0
  111. package/dist/web/static/assets/index-B1J7nBu0.js.map +1 -0
  112. package/dist/web/static/assets/index-BVqk4bSO.css +1 -0
  113. package/dist/web/static/assets/{lucide-BanPULT1.js → lucide-Bu44HVAM.js} +33 -73
  114. package/dist/web/static/assets/lucide-Bu44HVAM.js.map +1 -0
  115. package/dist/web/static/index.html +3 -3
  116. package/package.json +2 -2
  117. package/dist/capability/execution-manager.d.ts +0 -96
  118. package/dist/capability/execution-manager.d.ts.map +0 -1
  119. package/dist/capability/execution-manager.js +0 -260
  120. package/dist/capability/execution-manager.js.map +0 -1
  121. package/dist/capability/executor/background-executor.d.ts +0 -58
  122. package/dist/capability/executor/background-executor.d.ts.map +0 -1
  123. package/dist/capability/executor/background-executor.js +0 -322
  124. package/dist/capability/executor/background-executor.js.map +0 -1
  125. package/dist/capability/executor/foreground-executor.d.ts +0 -26
  126. package/dist/capability/executor/foreground-executor.d.ts.map +0 -1
  127. package/dist/capability/executor/foreground-executor.js +0 -82
  128. package/dist/capability/executor/foreground-executor.js.map +0 -1
  129. package/dist/capability/executor/orchestrator.d.ts +0 -38
  130. package/dist/capability/executor/orchestrator.d.ts.map +0 -1
  131. package/dist/capability/executor/orchestrator.js +0 -158
  132. package/dist/capability/executor/orchestrator.js.map +0 -1
  133. package/dist/capability/executor/stream-parser.d.ts +0 -73
  134. package/dist/capability/executor/stream-parser.d.ts.map +0 -1
  135. package/dist/capability/executor/stream-parser.js +0 -56
  136. package/dist/capability/executor/stream-parser.js.map +0 -1
  137. package/dist/capability/executor/types.d.ts +0 -44
  138. package/dist/capability/executor/types.d.ts.map +0 -1
  139. package/dist/capability/executor/types.js +0 -9
  140. package/dist/capability/executor/types.js.map +0 -1
  141. package/dist/capability/executor/worker-auth-probe.d.ts +0 -30
  142. package/dist/capability/executor/worker-auth-probe.d.ts.map +0 -1
  143. package/dist/capability/executor/worker-auth-probe.js +0 -108
  144. package/dist/capability/executor/worker-auth-probe.js.map +0 -1
  145. package/dist/capability/methodologies/bmad.yaml +0 -81
  146. package/dist/capability/methodologies/code-quality-audit.yaml +0 -26
  147. package/dist/capability/methodologies/harness-engineering.yaml +0 -75
  148. package/dist/capability/methodologies/test-coverage-scan.yaml +0 -26
  149. package/dist/capability/methodology-planner.d.ts +0 -49
  150. package/dist/capability/methodology-planner.d.ts.map +0 -1
  151. package/dist/capability/methodology-planner.js +0 -306
  152. package/dist/capability/methodology-planner.js.map +0 -1
  153. package/dist/capability/methodology-registry.d.ts +0 -32
  154. package/dist/capability/methodology-registry.d.ts.map +0 -1
  155. package/dist/capability/methodology-registry.js +0 -113
  156. package/dist/capability/methodology-registry.js.map +0 -1
  157. package/dist/daemon/handlers/methodology-formatter.d.ts +0 -16
  158. package/dist/daemon/handlers/methodology-formatter.d.ts.map +0 -1
  159. package/dist/daemon/handlers/methodology-formatter.js +0 -119
  160. package/dist/daemon/handlers/methodology-formatter.js.map +0 -1
  161. package/dist/daemon/idle-detector.d.ts +0 -35
  162. package/dist/daemon/idle-detector.d.ts.map +0 -1
  163. package/dist/daemon/idle-detector.js +0 -56
  164. package/dist/daemon/idle-detector.js.map +0 -1
  165. package/dist/daemon/idle-trigger.d.ts +0 -53
  166. package/dist/daemon/idle-trigger.d.ts.map +0 -1
  167. package/dist/daemon/idle-trigger.js +0 -153
  168. package/dist/daemon/idle-trigger.js.map +0 -1
  169. package/dist/daemon/methodology-pending-queue.d.ts +0 -33
  170. package/dist/daemon/methodology-pending-queue.d.ts.map +0 -1
  171. package/dist/daemon/methodology-pending-queue.js +0 -120
  172. package/dist/daemon/methodology-pending-queue.js.map +0 -1
  173. package/dist/web/routes/methodology.d.ts +0 -12
  174. package/dist/web/routes/methodology.d.ts.map +0 -1
  175. package/dist/web/routes/methodology.js +0 -228
  176. package/dist/web/routes/methodology.js.map +0 -1
  177. package/dist/web/static/assets/Dashboard-Ciyyw6ph.js +0 -2
  178. package/dist/web/static/assets/Dashboard-Ciyyw6ph.js.map +0 -1
  179. package/dist/web/static/assets/Methodologies-CXNrDXwG.js +0 -5
  180. package/dist/web/static/assets/Methodologies-CXNrDXwG.js.map +0 -1
  181. package/dist/web/static/assets/MethodologyDetail-rV3W1utf.js +0 -2
  182. package/dist/web/static/assets/MethodologyDetail-rV3W1utf.js.map +0 -1
  183. package/dist/web/static/assets/index-DJK5beK6.js +0 -3
  184. package/dist/web/static/assets/index-DJK5beK6.js.map +0 -1
  185. package/dist/web/static/assets/index-phpuytMI.css +0 -1
  186. package/dist/web/static/assets/lucide-BanPULT1.js.map +0 -1
@@ -1,33 +0,0 @@
1
- /**
2
- * Methodology Pending Inject Queue
3
- *
4
- * 前台模式下,Claude 主对话有时会忽略 UserPromptSubmit 注入的阶段指令
5
- * (3 小时进度 0% 就是这种情况)。Stop hook 检测到这种情况时,把"下
6
- * 一次要注入的指令"写入队列;UserPromptSubmit 在下一轮响应前消费一次。
7
- *
8
- * 进程内 Map + 磁盘 JSON 双层:daemon 重启时从磁盘恢复。
9
- */
10
- export interface PendingDirective {
11
- session_id: string;
12
- execution_id: number;
13
- phase_index: number;
14
- directive: string;
15
- enqueued_at: number;
16
- reason: 'stop-fallback' | 'cold-restart' | 'idle-timeout';
17
- }
18
- export declare class MethodologyPendingQueue {
19
- private memory;
20
- private queueDir;
21
- constructor(queueDir?: string);
22
- private loadFromDisk;
23
- private filePath;
24
- enqueue(entry: PendingDirective): void;
25
- consume(session_id: string): PendingDirective | null;
26
- peek(session_id: string): PendingDirective | null;
27
- size(): number;
28
- purgeByExecutionId(execution_id: number): void;
29
- }
30
- export declare function getMethodologyPendingQueue(): MethodologyPendingQueue;
31
- export declare function _resetMethodologyPendingQueueForTest(): void;
32
- export declare function getPendingQueueDir(): string;
33
- //# sourceMappingURL=methodology-pending-queue.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"methodology-pending-queue.d.ts","sourceRoot":"","sources":["../../src/daemon/methodology-pending-queue.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAaH,MAAM,WAAW,gBAAgB;IAC/B,UAAU,EAAE,MAAM,CAAC;IACnB,YAAY,EAAE,MAAM,CAAC;IACrB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,EAAE,eAAe,GAAG,cAAc,GAAG,cAAc,CAAC;CAC3D;AAED,qBAAa,uBAAuB;IAClC,OAAO,CAAC,MAAM,CAAuC;IACrD,OAAO,CAAC,QAAQ,CAAS;gBAEb,QAAQ,CAAC,EAAE,MAAM;IAK7B,OAAO,CAAC,YAAY;IAwBpB,OAAO,CAAC,QAAQ;IAKhB,OAAO,CAAC,KAAK,EAAE,gBAAgB,GAAG,IAAI;IActC,OAAO,CAAC,UAAU,EAAE,MAAM,GAAG,gBAAgB,GAAG,IAAI;IAiBpD,IAAI,CAAC,UAAU,EAAE,MAAM,GAAG,gBAAgB,GAAG,IAAI;IAIjD,IAAI,IAAI,MAAM;IAKd,kBAAkB,CAAC,YAAY,EAAE,MAAM,GAAG,IAAI;CAW/C;AAID,wBAAgB,0BAA0B,IAAI,uBAAuB,CAGpE;AAGD,wBAAgB,oCAAoC,IAAI,IAAI,CAE3D;AAGD,wBAAgB,kBAAkB,IAAI,MAAM,CAE3C"}
@@ -1,120 +0,0 @@
1
- /**
2
- * Methodology Pending Inject Queue
3
- *
4
- * 前台模式下,Claude 主对话有时会忽略 UserPromptSubmit 注入的阶段指令
5
- * (3 小时进度 0% 就是这种情况)。Stop hook 检测到这种情况时,把"下
6
- * 一次要注入的指令"写入队列;UserPromptSubmit 在下一轮响应前消费一次。
7
- *
8
- * 进程内 Map + 磁盘 JSON 双层:daemon 重启时从磁盘恢复。
9
- */
10
- import { existsSync, mkdirSync, readdirSync, readFileSync, unlinkSync, writeFileSync } from 'fs';
11
- import path from 'path';
12
- import { homedir } from 'os';
13
- import { logger } from '../core/utils/logger.js';
14
- const DEFAULT_QUEUE_DIR = path.join(homedir(), '.claude-forge', 'pending-inject');
15
- function resolveQueueDir() {
16
- return process.env.CLAUDE_FORGE_PENDING_DIR || DEFAULT_QUEUE_DIR;
17
- }
18
- export class MethodologyPendingQueue {
19
- memory = new Map();
20
- queueDir;
21
- constructor(queueDir) {
22
- this.queueDir = queueDir ?? resolveQueueDir();
23
- this.loadFromDisk();
24
- }
25
- loadFromDisk() {
26
- try {
27
- if (!existsSync(this.queueDir)) {
28
- mkdirSync(this.queueDir, { recursive: true });
29
- return;
30
- }
31
- const files = readdirSync(this.queueDir).filter(f => f.endsWith('.json'));
32
- for (const file of files) {
33
- try {
34
- const raw = readFileSync(path.join(this.queueDir, file), 'utf-8');
35
- const entry = JSON.parse(raw);
36
- this.memory.set(entry.session_id, entry);
37
- }
38
- catch (err) {
39
- logger.debug(`[MethodologyPendingQueue] skip corrupt ${file}: ${err}`);
40
- }
41
- }
42
- if (this.memory.size > 0) {
43
- logger.info(`[MethodologyPendingQueue] restored ${this.memory.size} pending directives`);
44
- }
45
- }
46
- catch (err) {
47
- logger.warn(`[MethodologyPendingQueue] loadFromDisk failed: ${err}`);
48
- }
49
- }
50
- filePath(session_id) {
51
- const safe = session_id.replace(/[^a-zA-Z0-9_-]/g, '_');
52
- return path.join(this.queueDir, `${safe}.json`);
53
- }
54
- enqueue(entry) {
55
- this.memory.set(entry.session_id, entry);
56
- try {
57
- if (!existsSync(this.queueDir))
58
- mkdirSync(this.queueDir, { recursive: true });
59
- writeFileSync(this.filePath(entry.session_id), JSON.stringify(entry, null, 2), 'utf-8');
60
- logger.info(`[MethodologyPendingQueue] enqueued ${entry.reason} for session=${entry.session_id.slice(0, 8)} phase=${entry.phase_index}`);
61
- }
62
- catch (err) {
63
- logger.warn(`[MethodologyPendingQueue] persist failed: ${err}`);
64
- }
65
- }
66
- // 取出并移除。返回 null 如果没有待注入。
67
- consume(session_id) {
68
- const entry = this.memory.get(session_id);
69
- if (!entry)
70
- return null;
71
- this.memory.delete(session_id);
72
- try {
73
- const fp = this.filePath(session_id);
74
- if (existsSync(fp))
75
- unlinkSync(fp);
76
- }
77
- catch (err) {
78
- logger.debug(`[MethodologyPendingQueue] cleanup failed: ${err}`);
79
- }
80
- logger.info(`[MethodologyPendingQueue] consumed ${entry.reason} for session=${session_id.slice(0, 8)} phase=${entry.phase_index}`);
81
- return entry;
82
- }
83
- // 对外只用于调试/可观测,不建议修改返回值
84
- peek(session_id) {
85
- return this.memory.get(session_id) ?? null;
86
- }
87
- size() {
88
- return this.memory.size;
89
- }
90
- // 强制清理指定 execution 的所有 pending(execution 被 cancel / 完成时调用)
91
- purgeByExecutionId(execution_id) {
92
- for (const [sid, entry] of this.memory.entries()) {
93
- if (entry.execution_id === execution_id) {
94
- this.memory.delete(sid);
95
- try {
96
- const fp = this.filePath(sid);
97
- if (existsSync(fp))
98
- unlinkSync(fp);
99
- }
100
- catch { /* ignore */ }
101
- }
102
- }
103
- }
104
- }
105
- // Daemon 单例
106
- let _instance = null;
107
- export function getMethodologyPendingQueue() {
108
- if (!_instance)
109
- _instance = new MethodologyPendingQueue();
110
- return _instance;
111
- }
112
- // 测试用 reset
113
- export function _resetMethodologyPendingQueueForTest() {
114
- _instance = null;
115
- }
116
- // 暴露给外部(用于避免测试时写真实 HOME)
117
- export function getPendingQueueDir() {
118
- return resolveQueueDir();
119
- }
120
- //# sourceMappingURL=methodology-pending-queue.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"methodology-pending-queue.js","sourceRoot":"","sources":["../../src/daemon/methodology-pending-queue.ts"],"names":[],"mappings":"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"}
@@ -1,12 +0,0 @@
1
- import type { Application } from 'express';
2
- import type { RouteContext } from './types.js';
3
- /**
4
- * /api/methodology-executions/* and /api/methodologies.
5
- *
6
- * Also owns the SSE stream for live execution updates. Note: the SSE route
7
- * `/api/methodology-executions/events` must be registered before the
8
- * parameterized `/:id` route on Express 4 (Express matches in definition
9
- * order and `:id` would otherwise swallow the literal `events` segment).
10
- */
11
- export declare function registerMethodologyRoutes(app: Application, ctx: RouteContext): void;
12
- //# sourceMappingURL=methodology.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"methodology.d.ts","sourceRoot":"","sources":["../../../src/web/routes/methodology.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAQ3C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAI/C;;;;;;;GAOG;AACH,wBAAgB,yBAAyB,CAAC,GAAG,EAAE,WAAW,EAAE,GAAG,EAAE,YAAY,GAAG,IAAI,CAkOnF"}
@@ -1,228 +0,0 @@
1
- import fs from 'fs';
2
- import path from 'path';
3
- import yaml from 'js-yaml';
4
- import { fileURLToPath } from 'url';
5
- import { WorkerAuthError } from '../../capability/executor/worker-auth-probe.js';
6
- import { logger } from '../../core/utils/logger.js';
7
- const __dirname = path.dirname(fileURLToPath(import.meta.url));
8
- /**
9
- * /api/methodology-executions/* and /api/methodologies.
10
- *
11
- * Also owns the SSE stream for live execution updates. Note: the SSE route
12
- * `/api/methodology-executions/events` must be registered before the
13
- * parameterized `/:id` route on Express 4 (Express matches in definition
14
- * order and `:id` would otherwise swallow the literal `events` segment).
15
- */
16
- export function registerMethodologyRoutes(app, ctx) {
17
- const { storage, executionManager, methodologyRegistry } = ctx;
18
- // List executions
19
- app.get('/api/methodology-executions', (req, res) => {
20
- const limit = parseInt(req.query.limit) || 50;
21
- const sessionId = req.query.session;
22
- const db = storage.getDatabase();
23
- const query = sessionId
24
- ? 'SELECT * FROM methodology_executions WHERE session_id = ? ORDER BY started_at DESC LIMIT ?'
25
- : 'SELECT * FROM methodology_executions ORDER BY started_at DESC LIMIT ?';
26
- const params = sessionId ? [sessionId, limit] : [limit];
27
- const executions = db.prepare(query).all(...params);
28
- res.json(executions);
29
- });
30
- // SSE stream — MUST be before `/:id`
31
- app.get('/api/methodology-executions/events', (req, res) => {
32
- if (!executionManager) {
33
- res.status(503).json({ error: 'Execution manager not available' });
34
- return;
35
- }
36
- const filterId = req.query.execution_id ? parseInt(String(req.query.execution_id)) : null;
37
- res.setHeader('Content-Type', 'text/event-stream');
38
- res.setHeader('Cache-Control', 'no-cache');
39
- res.setHeader('Connection', 'keep-alive');
40
- res.flushHeaders();
41
- const unsub = executionManager.subscribe(ev => {
42
- if (filterId !== null && ev.execution_id !== filterId)
43
- return;
44
- try {
45
- res.write(`event: ${ev.type}\ndata: ${JSON.stringify(ev)}\n\n`);
46
- }
47
- catch { /* client gone */ }
48
- });
49
- const heartbeat = setInterval(() => {
50
- try {
51
- res.write(`: heartbeat ${Date.now()}\n\n`);
52
- }
53
- catch { /* ignore */ }
54
- }, 25_000);
55
- req.on('close', () => {
56
- clearInterval(heartbeat);
57
- unsub();
58
- });
59
- });
60
- // Execution detail: get single execution with phases
61
- app.get('/api/methodology-executions/:id', (req, res) => {
62
- const executionId = parseInt(req.params.id);
63
- const db = storage.getDatabase();
64
- const execution = db.prepare('SELECT * FROM methodology_executions WHERE id = ?').get(executionId);
65
- if (!execution) {
66
- res.status(404).json({ error: 'Execution not found' });
67
- return;
68
- }
69
- const phases = db.prepare(`
70
- SELECT * FROM phase_executions
71
- WHERE methodology_execution_id = ?
72
- ORDER BY phase_index ASC
73
- `).all(executionId);
74
- res.json({
75
- ...execution,
76
- plan: JSON.parse(execution.plan_json),
77
- phases,
78
- });
79
- });
80
- // Cancel a running execution
81
- app.post('/api/methodology-executions/:id/cancel', async (req, res) => {
82
- const executionId = parseInt(req.params.id);
83
- const db = storage.getDatabase();
84
- const execution = db.prepare('SELECT * FROM methodology_executions WHERE id = ?').get(executionId);
85
- if (!execution) {
86
- res.status(404).json({ error: 'Execution not found' });
87
- return;
88
- }
89
- if (execution.status !== 'running') {
90
- res.status(400).json({ error: `Cannot cancel execution in ${execution.status} state` });
91
- return;
92
- }
93
- // Prefer ExecutionManager — it kills the worker process for background mode
94
- if (executionManager) {
95
- try {
96
- await executionManager.cancel(executionId);
97
- }
98
- catch (err) {
99
- logger.warn(`[web] cancel via ExecutionManager failed: ${err}`);
100
- res.status(500).json({ error: `Cancel failed: ${err instanceof Error ? err.message : String(err)}` });
101
- return;
102
- }
103
- }
104
- else {
105
- // Fallback: DB-only cancel (pre-Phase-C behavior)
106
- db.prepare(`
107
- UPDATE methodology_executions
108
- SET status = 'cancelled', completed_at = ?
109
- WHERE id = ?
110
- `).run(Date.now(), executionId);
111
- }
112
- // Mark any still-running phase rows as cancelled
113
- db.prepare(`
114
- UPDATE phase_executions
115
- SET status = 'cancelled', completed_at = ?
116
- WHERE methodology_execution_id = ? AND status = 'running'
117
- `).run(Date.now(), executionId);
118
- res.json({ success: true, message: 'Execution cancelled' });
119
- });
120
- // Start a new execution (Phase C)
121
- app.post('/api/methodology-executions', async (req, res) => {
122
- if (!executionManager || !methodologyRegistry) {
123
- res.status(503).json({ error: 'Execution manager not available' });
124
- return;
125
- }
126
- const { session_id, methodology_id, plan, mode } = req.body ?? {};
127
- if (!session_id) {
128
- res.status(400).json({ error: 'session_id is required' });
129
- return;
130
- }
131
- if (mode !== 'foreground' && mode !== 'background') {
132
- res.status(400).json({ error: 'mode must be foreground or background' });
133
- return;
134
- }
135
- let planObj = null;
136
- if (plan && typeof plan === 'object' && Array.isArray(plan.phases)) {
137
- planObj = plan;
138
- }
139
- else if (methodology_id) {
140
- const methodology = methodologyRegistry.get(methodology_id);
141
- if (!methodology) {
142
- res.status(404).json({ error: `methodology ${methodology_id} not found` });
143
- return;
144
- }
145
- const templates = methodology.phase_templates ?? {};
146
- const phaseIds = Object.keys(templates);
147
- if (phaseIds.length === 0) {
148
- res.status(400).json({ error: `methodology ${methodology_id} has no phase_templates` });
149
- return;
150
- }
151
- planObj = {
152
- methodology_id: methodology.id,
153
- rationale: `Started via API (default phases from ${methodology.id})`,
154
- phases: phaseIds.map(id => {
155
- const tpl = templates[id];
156
- return {
157
- id,
158
- agent: tpl.agent,
159
- prompt: tpl.description ?? tpl.prompt_template.slice(0, 400),
160
- rationale: 'default phase from methodology template',
161
- };
162
- }),
163
- };
164
- }
165
- else {
166
- res.status(400).json({ error: 'either methodology_id or plan must be provided' });
167
- return;
168
- }
169
- try {
170
- const id = executionManager.start({
171
- session_id,
172
- methodology_id: planObj.methodology_id,
173
- plan: planObj,
174
- mode,
175
- });
176
- res.status(201).json({ id, mode });
177
- }
178
- catch (err) {
179
- logger.warn(`[web] start execution failed: ${err}`);
180
- const message = err instanceof Error ? err.message : String(err);
181
- const isAuthError = err instanceof WorkerAuthError
182
- || (err instanceof Error && err.code === 'AUTH_REQUIRED')
183
- || message.includes('Background mode requires');
184
- if (isAuthError) {
185
- res.status(503).json({ error: message, code: 'AUTH_REQUIRED' });
186
- }
187
- else {
188
- res.status(500).json({ error: message });
189
- }
190
- }
191
- });
192
- // Delete an execution
193
- app.delete('/api/methodology-executions/:id', (req, res) => {
194
- const executionId = parseInt(req.params.id);
195
- const db = storage.getDatabase();
196
- db.prepare('DELETE FROM phase_executions WHERE methodology_execution_id = ?').run(executionId);
197
- const result = db.prepare('DELETE FROM methodology_executions WHERE id = ?').run(executionId);
198
- if (result.changes === 0) {
199
- res.status(404).json({ error: 'Execution not found' });
200
- return;
201
- }
202
- res.json({ success: true });
203
- });
204
- // Methodology: list all available methodologies
205
- app.get('/api/methodologies', (_req, res) => {
206
- try {
207
- const candidates = [
208
- path.join(__dirname, '../../capability/methodologies'), // dist/web/routes -> dist/capability/methodologies
209
- path.join(__dirname, '../../../src/capability/methodologies'), // dev: src/web/routes -> src/capability/methodologies
210
- ];
211
- const dir = candidates.find(d => fs.existsSync(d));
212
- if (!dir) {
213
- res.json([]);
214
- return;
215
- }
216
- const files = fs.readdirSync(dir).filter(f => f.endsWith('.yaml'));
217
- const methodologies = files.map(f => {
218
- const content = fs.readFileSync(path.join(dir, f), 'utf-8');
219
- return yaml.load(content);
220
- });
221
- res.json(methodologies);
222
- }
223
- catch (err) {
224
- res.status(500).json({ error: err.message });
225
- }
226
- });
227
- }
228
- //# sourceMappingURL=methodology.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"methodology.js","sourceRoot":"","sources":["../../../src/web/routes/methodology.ts"],"names":[],"mappings":"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"}
@@ -1,2 +0,0 @@
1
- import{j as e}from"./react-vendor-CSp-GLFF.js";import{u as d}from"./query-C99w429o.js";import{b as w}from"./react-router-I-HqunH7.js";import{A as C,a as k,G as L,Z as $,g as q}from"./lucide-BanPULT1.js";import{R as m,A as R,C as j,X as g,Y as y,T as u,a as F,B as K,b as A,P as S,c as D,d as M,L as O}from"./charts-CLrM0_uM.js";import"./vendor-CMMjVdZs.js";async function E(){const s=await fetch("/api/status");if(!s.ok)throw new Error("Failed");return s.json()}async function P(){const s=await fetch("/api/stats");if(!s.ok)throw new Error("Failed");return s.json()}async function U(){const s=await fetch("/api/sessions?limit=1000");return s.ok?(await s.json()).length:0}async function B(){const s=await fetch("/api/routing/stats?window=168");return s.ok?s.json():null}async function I(){const s=await fetch("/api/methodology-executions?limit=1000");return s.ok?(await s.json()).length:0}function z(s){const t=Math.floor(s/86400),a=Math.floor(s%86400/3600),l=Math.floor(s%3600/60);return t>0?`${t}天 ${a}小时`:a>0?`${a}小时 ${l}分钟`:`${l}分钟`}const r={primary:"#6366f1",green:"#10b981",red:"#ef4444",yellow:"#f59e0b",blue:"#3b82f6",purple:"#8b5cf6"},b=[r.primary,r.green,r.yellow,r.red,r.purple,r.blue];function Q(){var f;const s=w(),{data:t}=d({queryKey:["status"],queryFn:E,refetchInterval:5e3}),{data:a}=d({queryKey:["system-stats"],queryFn:P,refetchInterval:3e4}),{data:l}=d({queryKey:["sessions-count"],queryFn:U}),{data:n}=d({queryKey:["dashboard-routing"],queryFn:B}),{data:c}=d({queryKey:["methodology-count"],queryFn:I}),p=(n==null?void 0:n.obedienceRate)!==null&&(n==null?void 0:n.obedienceRate)!==void 0?(n.obedienceRate*100).toFixed(0):"-",v=a!=null&&a.toolUsage?Object.entries(a.toolUsage).sort(([,i],[,o])=>o-i).slice(0,8).map(([i,o])=>({name:i,count:o})):[],h=a!=null&&a.decisionLevels?[{name:"允许",value:a.decisionLevels.allow,fill:r.green},{name:"警告",value:a.decisionLevels.warn,fill:r.yellow},{name:"阻断",value:a.decisionLevels.block,fill:r.red},{name:"确认",value:a.decisionLevels.confirm,fill:r.blue}].filter(i=>i.value>0):[],N=(a==null?void 0:a.dailyActivity)||[];return e.jsxs("div",{children:[e.jsx("h1",{className:"text-2xl font-bold text-gray-900 mb-6",children:"仪表盘"}),e.jsxs("div",{className:"grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4 mb-6",children:[e.jsx(x,{icon:C,label:"总事件",value:((f=t==null?void 0:t.eventCount)==null?void 0:f.toString())||"-",color:"blue",onClick:()=>s("/events")}),e.jsx(x,{icon:k,label:"会话数",value:(l==null?void 0:l.toString())||"-",color:"purple",onClick:()=>s("/sessions")}),e.jsx(x,{icon:L,label:"方法论执行",value:(c==null?void 0:c.toString())||"-",color:"indigo",onClick:()=>s("/methodologies")}),e.jsx(x,{icon:$,label:"路由遵守率",value:`${p}%`,color:"green",onClick:()=>s("/routing")})]}),e.jsxs("div",{className:"grid grid-cols-1 lg:grid-cols-2 gap-4 mb-6",children:[e.jsxs("div",{className:"bg-white rounded-lg shadow p-6",children:[e.jsxs("div",{className:"flex items-center justify-between mb-4",children:[e.jsx("h3",{className:"text-base font-semibold text-gray-900",children:"最近 7 天活动"}),e.jsx("button",{onClick:()=>s("/events"),className:"text-xs text-indigo-600 hover:text-indigo-700",children:"查看全部事件 →"})]}),e.jsx(m,{width:"100%",height:240,children:e.jsxs(R,{data:N,onClick:()=>s("/events"),style:{cursor:"pointer"},children:[e.jsx("defs",{children:e.jsxs("linearGradient",{id:"colorEvents",x1:"0",y1:"0",x2:"0",y2:"1",children:[e.jsx("stop",{offset:"5%",stopColor:r.primary,stopOpacity:.3}),e.jsx("stop",{offset:"95%",stopColor:r.primary,stopOpacity:0})]})}),e.jsx(j,{strokeDasharray:"3 3",stroke:"#f3f4f6"}),e.jsx(g,{dataKey:"date",tick:{fontSize:11},tickFormatter:i=>i==null?void 0:i.slice(5)}),e.jsx(y,{tick:{fontSize:11}}),e.jsx(u,{}),e.jsx(F,{type:"monotone",dataKey:"eventCount",stroke:r.primary,strokeWidth:2,fill:"url(#colorEvents)",name:"事件"})]})})]}),e.jsxs("div",{className:"bg-white rounded-lg shadow p-6",children:[e.jsxs("div",{className:"flex items-center justify-between mb-4",children:[e.jsx("h3",{className:"text-base font-semibold text-gray-900",children:"工具使用分布"}),e.jsx("button",{onClick:()=>s("/events"),className:"text-xs text-indigo-600 hover:text-indigo-700",children:"查看详情 →"})]}),e.jsx(m,{width:"100%",height:240,children:e.jsxs(K,{data:v,layout:"vertical",onClick:i=>{i!=null&&i.activeLabel&&s(`/events?tool=${encodeURIComponent(i.activeLabel)}`)},style:{cursor:"pointer"},children:[e.jsx(j,{strokeDasharray:"3 3",stroke:"#f3f4f6"}),e.jsx(g,{type:"number",tick:{fontSize:11}}),e.jsx(y,{dataKey:"name",type:"category",tick:{fontSize:11},width:80}),e.jsx(u,{}),e.jsx(A,{dataKey:"count",fill:r.primary,radius:[0,4,4,0]})]})})]}),e.jsxs("div",{className:"bg-white rounded-lg shadow p-6",children:[e.jsxs("div",{className:"flex items-center justify-between mb-4",children:[e.jsx("h3",{className:"text-base font-semibold text-gray-900",children:"治理决策分布"}),e.jsx("button",{onClick:()=>s("/execution-trace"),className:"text-xs text-indigo-600 hover:text-indigo-700",children:"查看追踪 →"})]}),h.length>0?e.jsx(m,{width:"100%",height:240,children:e.jsxs(S,{children:[e.jsx(D,{data:h,cx:"50%",cy:"50%",labelLine:!1,label:i=>`${i.name}: ${i.value}`,outerRadius:80,dataKey:"value",style:{cursor:"pointer"},onClick:()=>s("/execution-trace"),children:h.map((i,o)=>e.jsx(M,{fill:b[o%b.length]},o))}),e.jsx(u,{}),e.jsx(O,{})]})}):e.jsx("div",{className:"h-60 flex items-center justify-center text-gray-400 text-sm",children:"暂无决策数据"})]}),e.jsxs("div",{className:"bg-white rounded-lg shadow p-6",children:[e.jsxs("h3",{className:"flex items-center gap-2 text-base font-semibold text-gray-900 mb-4",children:[e.jsx(q,{className:"h-4 w-4 text-indigo-600"}),"系统状态"]}),e.jsxs("dl",{className:"space-y-3 text-sm",children:[e.jsxs("div",{className:"flex justify-between",children:[e.jsx("dt",{className:"text-gray-500",children:"PID"}),e.jsx("dd",{className:"font-mono text-gray-900",children:(t==null?void 0:t.pid)||"-"})]}),e.jsxs("div",{className:"flex justify-between",children:[e.jsx("dt",{className:"text-gray-500",children:"运行时间"}),e.jsx("dd",{className:"text-gray-900",children:t!=null&&t.uptime?z(t.uptime):"-"})]}),e.jsxs("div",{className:"flex justify-between",children:[e.jsx("dt",{className:"text-gray-500",children:"内存使用"}),e.jsx("dd",{className:"text-gray-900",children:t!=null&&t.memory?`${Math.round(t.memory.heapUsed/1024/1024)} MB`:"-"})]}),n&&e.jsxs(e.Fragment,{children:[e.jsx("div",{className:"border-t pt-3 mt-3",children:e.jsx("dt",{className:"text-xs font-semibold text-gray-700 mb-2",children:"路由 (近 7 天)"})}),e.jsxs("div",{className:"flex justify-between",children:[e.jsx("dt",{className:"text-gray-500",children:"总路由"}),e.jsx("dd",{className:"font-mono text-gray-900",children:n.total})]}),e.jsxs("div",{className:"flex justify-between",children:[e.jsx("dt",{className:"text-gray-500",children:"遵守 / 违抗"}),e.jsxs("dd",{className:"font-mono",children:[e.jsx("span",{className:"text-green-600",children:n.obeyedCount}),e.jsx("span",{className:"text-gray-400 mx-1",children:"/"}),e.jsx("span",{className:"text-red-600",children:n.refusedCount})]})]})]})]})]})]})]})}function x({icon:s,label:t,value:a,color:l,onClick:n}){const c={blue:"bg-blue-100 text-blue-600",purple:"bg-purple-100 text-purple-600",indigo:"bg-indigo-100 text-indigo-600",green:"bg-green-100 text-green-600"};return e.jsxs("div",{className:`bg-white rounded-lg shadow p-5 ${n?"cursor-pointer hover:shadow-md transition-shadow":""}`,onClick:n,children:[e.jsxs("div",{className:"flex items-center justify-between mb-3",children:[e.jsx("span",{className:"text-sm text-gray-500",children:t}),e.jsx("div",{className:`w-8 h-8 rounded-lg flex items-center justify-center ${c[l]}`,children:e.jsx(s,{className:"h-4 w-4"})})]}),e.jsx("div",{className:"text-2xl font-bold text-gray-900",children:a})]})}export{Q as default};
2
- //# sourceMappingURL=Dashboard-Ciyyw6ph.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"Dashboard-Ciyyw6ph.js","sources":["../../src/pages/Dashboard.tsx"],"sourcesContent":["import { useQuery } from '@tanstack/react-query'\nimport { useNavigate } from 'react-router-dom'\nimport { Activity, MessageSquare, GitBranch, Clock, Zap } from 'lucide-react'\nimport {\n AreaChart, Area, BarChart, Bar, PieChart, Pie, Cell,\n XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, Legend,\n} from 'recharts'\n\ninterface Stats {\n pid: number\n uptime: number\n memory: any\n eventCount: number\n}\n\ninterface SystemStats {\n totalSessions: number\n totalEvents: number\n totalDecisions: number\n toolUsage: Record<string, number>\n decisionLevels: { block: number; warn: number; confirm: number; allow: number }\n dailyActivity: Array<{ date: string; eventCount: number; sessionCount: number }>\n}\n\nasync function fetchStatus(): Promise<Stats> {\n const res = await fetch('/api/status')\n if (!res.ok) throw new Error('Failed')\n return res.json()\n}\n\nasync function fetchSystemStats(): Promise<SystemStats> {\n const res = await fetch('/api/stats')\n if (!res.ok) throw new Error('Failed')\n return res.json()\n}\n\nasync function fetchSessionsCount(): Promise<number> {\n const res = await fetch('/api/sessions?limit=1000')\n if (!res.ok) return 0\n const sessions = await res.json()\n return sessions.length\n}\n\nasync function fetchRoutingStats() {\n const res = await fetch('/api/routing/stats?window=168')\n if (!res.ok) return null\n return res.json()\n}\n\nasync function fetchMethodologyCount(): Promise<number> {\n const res = await fetch('/api/methodology-executions?limit=1000')\n if (!res.ok) return 0\n const executions = await res.json()\n return executions.length\n}\n\nfunction formatUptime(seconds: number): string {\n const days = Math.floor(seconds / 86400)\n const hours = Math.floor((seconds % 86400) / 3600)\n const minutes = Math.floor((seconds % 3600) / 60)\n if (days > 0) return `${days}天 ${hours}小时`\n if (hours > 0) return `${hours}小时 ${minutes}分钟`\n return `${minutes}分钟`\n}\n\nconst COLORS = {\n primary: '#6366f1',\n green: '#10b981',\n red: '#ef4444',\n yellow: '#f59e0b',\n blue: '#3b82f6',\n purple: '#8b5cf6',\n gray: '#6b7280',\n}\n\nconst PIE_COLORS = [COLORS.primary, COLORS.green, COLORS.yellow, COLORS.red, COLORS.purple, COLORS.blue]\n\nexport default function Dashboard() {\n const navigate = useNavigate()\n const { data: status } = useQuery({ queryKey: ['status'], queryFn: fetchStatus, refetchInterval: 5000 })\n const { data: systemStats } = useQuery({ queryKey: ['system-stats'], queryFn: fetchSystemStats, refetchInterval: 30000 })\n const { data: sessionsCount } = useQuery({ queryKey: ['sessions-count'], queryFn: fetchSessionsCount })\n const { data: routingStats } = useQuery({ queryKey: ['dashboard-routing'], queryFn: fetchRoutingStats })\n const { data: methodologyCount } = useQuery({ queryKey: ['methodology-count'], queryFn: fetchMethodologyCount })\n\n const obediencePercent = routingStats?.obedienceRate !== null && routingStats?.obedienceRate !== undefined\n ? (routingStats.obedienceRate * 100).toFixed(0)\n : '-'\n\n // 准备图表数据\n const toolUsageData = systemStats?.toolUsage\n ? Object.entries(systemStats.toolUsage)\n .sort(([, a], [, b]) => b - a)\n .slice(0, 8)\n .map(([name, count]) => ({ name, count }))\n : []\n\n const decisionData = systemStats?.decisionLevels\n ? [\n { name: '允许', value: systemStats.decisionLevels.allow, fill: COLORS.green },\n { name: '警告', value: systemStats.decisionLevels.warn, fill: COLORS.yellow },\n { name: '阻断', value: systemStats.decisionLevels.block, fill: COLORS.red },\n { name: '确认', value: systemStats.decisionLevels.confirm, fill: COLORS.blue },\n ].filter(d => d.value > 0)\n : []\n\n const dailyData = systemStats?.dailyActivity || []\n\n return (\n <div>\n <h1 className=\"text-2xl font-bold text-gray-900 mb-6\">仪表盘</h1>\n\n {/* Stats grid */}\n <div className=\"grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4 mb-6\">\n <StatCard icon={Activity} label=\"总事件\" value={status?.eventCount?.toString() || '-'} color=\"blue\" onClick={() => navigate('/events')} />\n <StatCard icon={MessageSquare} label=\"会话数\" value={sessionsCount?.toString() || '-'} color=\"purple\" onClick={() => navigate('/sessions')} />\n <StatCard icon={GitBranch} label=\"方法论执行\" value={methodologyCount?.toString() || '-'} color=\"indigo\" onClick={() => navigate('/methodologies')} />\n <StatCard icon={Zap} label=\"路由遵守率\" value={`${obediencePercent}%`} color=\"green\" onClick={() => navigate('/routing')} />\n </div>\n\n {/* Charts */}\n <div className=\"grid grid-cols-1 lg:grid-cols-2 gap-4 mb-6\">\n {/* Daily activity */}\n <div className=\"bg-white rounded-lg shadow p-6\">\n <div className=\"flex items-center justify-between mb-4\">\n <h3 className=\"text-base font-semibold text-gray-900\">最近 7 天活动</h3>\n <button\n onClick={() => navigate('/events')}\n className=\"text-xs text-indigo-600 hover:text-indigo-700\"\n >\n 查看全部事件 →\n </button>\n </div>\n <ResponsiveContainer width=\"100%\" height={240}>\n <AreaChart\n data={dailyData}\n onClick={() => navigate('/events')}\n style={{ cursor: 'pointer' }}\n >\n <defs>\n <linearGradient id=\"colorEvents\" x1=\"0\" y1=\"0\" x2=\"0\" y2=\"1\">\n <stop offset=\"5%\" stopColor={COLORS.primary} stopOpacity={0.3} />\n <stop offset=\"95%\" stopColor={COLORS.primary} stopOpacity={0} />\n </linearGradient>\n </defs>\n <CartesianGrid strokeDasharray=\"3 3\" stroke=\"#f3f4f6\" />\n <XAxis dataKey=\"date\" tick={{ fontSize: 11 }} tickFormatter={(d) => d?.slice(5)} />\n <YAxis tick={{ fontSize: 11 }} />\n <Tooltip />\n <Area\n type=\"monotone\"\n dataKey=\"eventCount\"\n stroke={COLORS.primary}\n strokeWidth={2}\n fill=\"url(#colorEvents)\"\n name=\"事件\"\n />\n </AreaChart>\n </ResponsiveContainer>\n </div>\n\n {/* Tool usage */}\n <div className=\"bg-white rounded-lg shadow p-6\">\n <div className=\"flex items-center justify-between mb-4\">\n <h3 className=\"text-base font-semibold text-gray-900\">工具使用分布</h3>\n <button\n onClick={() => navigate('/events')}\n className=\"text-xs text-indigo-600 hover:text-indigo-700\"\n >\n 查看详情 →\n </button>\n </div>\n <ResponsiveContainer width=\"100%\" height={240}>\n <BarChart\n data={toolUsageData}\n layout=\"vertical\"\n onClick={(e: any) => {\n if (e?.activeLabel) {\n navigate(`/events?tool=${encodeURIComponent(e.activeLabel)}`)\n }\n }}\n style={{ cursor: 'pointer' }}\n >\n <CartesianGrid strokeDasharray=\"3 3\" stroke=\"#f3f4f6\" />\n <XAxis type=\"number\" tick={{ fontSize: 11 }} />\n <YAxis dataKey=\"name\" type=\"category\" tick={{ fontSize: 11 }} width={80} />\n <Tooltip />\n <Bar dataKey=\"count\" fill={COLORS.primary} radius={[0, 4, 4, 0]} />\n </BarChart>\n </ResponsiveContainer>\n </div>\n\n {/* Decision distribution */}\n <div className=\"bg-white rounded-lg shadow p-6\">\n <div className=\"flex items-center justify-between mb-4\">\n <h3 className=\"text-base font-semibold text-gray-900\">治理决策分布</h3>\n <button\n onClick={() => navigate('/execution-trace')}\n className=\"text-xs text-indigo-600 hover:text-indigo-700\"\n >\n 查看追踪 →\n </button>\n </div>\n {decisionData.length > 0 ? (\n <ResponsiveContainer width=\"100%\" height={240}>\n <PieChart>\n <Pie\n data={decisionData}\n cx=\"50%\"\n cy=\"50%\"\n labelLine={false}\n label={(entry) => `${entry.name}: ${entry.value}`}\n outerRadius={80}\n dataKey=\"value\"\n style={{ cursor: 'pointer' }}\n onClick={() => navigate('/execution-trace')}\n >\n {decisionData.map((_, i) => (\n <Cell key={i} fill={PIE_COLORS[i % PIE_COLORS.length]} />\n ))}\n </Pie>\n <Tooltip />\n <Legend />\n </PieChart>\n </ResponsiveContainer>\n ) : (\n <div className=\"h-60 flex items-center justify-center text-gray-400 text-sm\">暂无决策数据</div>\n )}\n </div>\n\n {/* System info */}\n <div className=\"bg-white rounded-lg shadow p-6\">\n <h3 className=\"flex items-center gap-2 text-base font-semibold text-gray-900 mb-4\">\n <Clock className=\"h-4 w-4 text-indigo-600\" />\n 系统状态\n </h3>\n <dl className=\"space-y-3 text-sm\">\n <div className=\"flex justify-between\">\n <dt className=\"text-gray-500\">PID</dt>\n <dd className=\"font-mono text-gray-900\">{status?.pid || '-'}</dd>\n </div>\n <div className=\"flex justify-between\">\n <dt className=\"text-gray-500\">运行时间</dt>\n <dd className=\"text-gray-900\">{status?.uptime ? formatUptime(status.uptime) : '-'}</dd>\n </div>\n <div className=\"flex justify-between\">\n <dt className=\"text-gray-500\">内存使用</dt>\n <dd className=\"text-gray-900\">\n {status?.memory ? `${Math.round(status.memory.heapUsed / 1024 / 1024)} MB` : '-'}\n </dd>\n </div>\n {routingStats && (\n <>\n <div className=\"border-t pt-3 mt-3\">\n <dt className=\"text-xs font-semibold text-gray-700 mb-2\">路由 (近 7 天)</dt>\n </div>\n <div className=\"flex justify-between\">\n <dt className=\"text-gray-500\">总路由</dt>\n <dd className=\"font-mono text-gray-900\">{routingStats.total}</dd>\n </div>\n <div className=\"flex justify-between\">\n <dt className=\"text-gray-500\">遵守 / 违抗</dt>\n <dd className=\"font-mono\">\n <span className=\"text-green-600\">{routingStats.obeyedCount}</span>\n <span className=\"text-gray-400 mx-1\">/</span>\n <span className=\"text-red-600\">{routingStats.refusedCount}</span>\n </dd>\n </div>\n </>\n )}\n </dl>\n </div>\n </div>\n </div>\n )\n}\n\nfunction StatCard({ icon: Icon, label, value, color, onClick }: {\n icon: any, label: string, value: string, color: string, onClick?: () => void\n}) {\n const colorClass: Record<string, string> = {\n blue: 'bg-blue-100 text-blue-600',\n purple: 'bg-purple-100 text-purple-600',\n indigo: 'bg-indigo-100 text-indigo-600',\n green: 'bg-green-100 text-green-600',\n }\n return (\n <div\n className={`bg-white rounded-lg shadow p-5 ${onClick ? 'cursor-pointer hover:shadow-md transition-shadow' : ''}`}\n onClick={onClick}\n >\n <div className=\"flex items-center justify-between mb-3\">\n <span className=\"text-sm text-gray-500\">{label}</span>\n <div className={`w-8 h-8 rounded-lg flex items-center justify-center ${colorClass[color]}`}>\n <Icon className=\"h-4 w-4\" />\n </div>\n </div>\n <div className=\"text-2xl font-bold text-gray-900\">{value}</div>\n </div>\n )\n}\n"],"names":["fetchStatus","res","fetchSystemStats","fetchSessionsCount","fetchRoutingStats","fetchMethodologyCount","formatUptime","seconds","days","hours","minutes","COLORS","PIE_COLORS","Dashboard","navigate","useNavigate","status","useQuery","systemStats","sessionsCount","routingStats","methodologyCount","obediencePercent","toolUsageData","a","b","name","count","decisionData","d","dailyData","jsx","jsxs","StatCard","Activity","_a","MessageSquare","GitBranch","Zap","ResponsiveContainer","AreaChart","CartesianGrid","XAxis","YAxis","Tooltip","Area","BarChart","e","Bar","PieChart","Pie","entry","_","i","Cell","Legend","Clock","Fragment","Icon","label","value","color","onClick","colorClass"],"mappings":"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"}
@@ -1,5 +0,0 @@
1
- import{r as b,j as e}from"./react-vendor-CSp-GLFF.js";import{b as q,u as C,c as E}from"./query-C99w429o.js";import{b as M,L as $}from"./react-router-I-HqunH7.js";import{d as p}from"./vendor-CMMjVdZs.js";import{u as D,a as F}from"./index-DJK5beK6.js";import{a as A}from"./auth-Bnf8ZcqN.js";import{P as K,G as S,C as T,m as P,g as Q,c as O,e as R,d as H,n as B,o as L,f as _}from"./lucide-BanPULT1.js";import{a as U,z as X}from"./date-fns-CZ_bHujz.js";import"./syntax-highlighter-44FakypI.js";async function z(){const t=await fetch("/api/methodology-executions?limit=50");if(!t.ok)throw new Error("Failed to fetch");return t.json()}async function G(){const t=await fetch("/api/methodologies");if(!t.ok)throw new Error("Failed to fetch methodologies");return t.json()}async function J(){const t=await fetch("/api/worker-auth-status");if(!t.ok)throw new Error("Failed to fetch worker auth status");return t.json()}async function Y(t){const a=await A(`/api/methodology-executions/${t}/cancel`,{method:"POST"});if(!a.ok){const o=await a.json().catch(()=>({}));throw new Error(o.error||"Cancel failed")}}async function W(t){if(!(await A(`/api/methodology-executions/${t}`,{method:"DELETE"})).ok)throw new Error("Delete failed")}async function V(t){const a=await A("/api/methodology-executions",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(t)});if(!a.ok){const o=await a.json().catch(()=>({})),i=new Error(o.error||`Start failed (HTTP ${a.status})`);throw typeof o.code=="string"&&(i.code=o.code),i}return a.json()}const Z={"harness-engineering":"Harness Engineering",bmad:"BMad (敏捷开发)"};function ce(){const t=q(),a=D(),o=F(),i=M(),[j,d]=b.useState(!1),{data:r,isLoading:g,error:f}=C({queryKey:["methodology-executions"],queryFn:z,refetchInterval:5e3}),{data:c}=C({queryKey:["methodologies"],queryFn:G}),{data:N}=C({queryKey:["worker-auth-status"],queryFn:J,staleTime:3e4}),m=E({mutationFn:Y,onSuccess:()=>{a.success("方法论执行已取消"),t.invalidateQueries({queryKey:["methodology-executions"]})},onError:s=>a.error(`取消失败: ${s.message}`)}),y=E({mutationFn:W,onSuccess:()=>{a.success("记录已删除"),t.invalidateQueries({queryKey:["methodology-executions"]})},onError:s=>a.error(`删除失败: ${s.message}`)}),x=E({mutationFn:V,onSuccess:s=>{a.success(`已启动执行 #${s.id}`),d(!1),t.invalidateQueries({queryKey:["methodology-executions"]}),t.invalidateQueries({queryKey:["worker-auth-status"]})},onError:async s=>{const l=s.message||"";if(s.code==="AUTH_REQUIRED"||l.includes("Background mode requires")||l.includes("AUTH_REQUIRED")){t.invalidateQueries({queryKey:["worker-auth-status"]}),await o({title:"需要配置 Claude API",message:`后台模式需要 Claude API 认证(ANTHROPIC_API_KEY 或 OAuth)。
2
- 请先在 AI 配置页设置后再启动执行。
3
-
4
- 详细信息:${l}`,confirmText:"去 AI 配置",cancelText:"留在此页",variant:"warning"})&&(d(!1),i("/ai-config"));return}a.error(`启动失败: ${l}`)}}),h=async s=>{await o({title:"取消方法论执行",message:"确定要取消这个运行中的方法论吗?已完成的阶段不会受影响,但当前进行中的阶段将停止。",confirmText:"确认取消",variant:"warning"})&&m.mutate(s)},w=async s=>{await o({title:"删除执行记录",message:"确定要删除这条方法论执行记录吗?此操作不可恢复。",confirmText:"删除",variant:"danger"})&&y.mutate(s)};if(g)return e.jsxs("div",{children:[e.jsx("h1",{className:"text-2xl font-bold text-gray-900 mb-6",children:"方法论执行"}),e.jsx("div",{className:"bg-white rounded-lg shadow p-6",children:e.jsx("p",{className:"text-gray-600",children:"加载中..."})})]});if(f)return e.jsxs("div",{children:[e.jsx("h1",{className:"text-2xl font-bold text-gray-900 mb-6",children:"方法论执行"}),e.jsx("div",{className:"bg-white rounded-lg shadow p-6",children:e.jsxs("p",{className:"text-red-600",children:["加载失败: ",f.message]})})]});const u=(r==null?void 0:r.filter(s=>s.status==="running").length)||0,n=(r==null?void 0:r.filter(s=>s.status==="completed").length)||0;return e.jsxs("div",{children:[e.jsxs("div",{className:"flex items-center justify-between mb-6",children:[e.jsx("h1",{className:"text-2xl font-bold text-gray-900",children:"方法论执行"}),e.jsxs("div",{className:"flex items-center gap-4 text-sm",children:[e.jsxs("span",{className:"text-gray-500",children:["共 ",(r==null?void 0:r.length)||0," 个实例"]}),u>0&&e.jsxs("span",{className:"text-blue-600",children:["· 运行中 ",u]}),n>0&&e.jsxs("span",{className:"text-green-600",children:["· 已完成 ",n]}),e.jsxs("button",{onClick:()=>d(!0),className:"ml-3 inline-flex items-center gap-1 px-3 py-1.5 text-sm bg-indigo-600 text-white rounded-md hover:bg-indigo-700",children:[e.jsx(K,{className:"h-4 w-4"})," 新建执行"]})]})]}),j&&e.jsx(ee,{methodologies:c??[],authStatus:N??null,onCancel:()=>d(!1),onSubmit:s=>x.mutate(s),onGoToAIConfig:()=>{d(!1),i("/ai-config")},submitting:x.isPending}),e.jsxs("div",{className:"space-y-4",children:[r==null?void 0:r.map(s=>{const l=JSON.parse(s.plan_json),v=l.phases[s.current_phase_index],I=s.status==="completed"?100:Math.round(s.current_phase_index/l.phases.length*100);return e.jsx("div",{className:"bg-white rounded-lg shadow",children:e.jsxs("div",{className:"p-6 flex items-start justify-between gap-4",children:[e.jsxs($,{to:`/methodologies/${s.id}`,className:"flex-1 min-w-0",children:[e.jsxs("div",{className:"flex items-center gap-3 mb-2",children:[e.jsx(S,{className:"h-5 w-5 text-indigo-600 flex-shrink-0"}),e.jsx("h3",{className:"text-lg font-semibold text-gray-900",children:Z[s.methodology_id]||s.methodology_id}),e.jsxs("span",{className:p("px-2 py-1 text-xs font-medium rounded-full",s.status==="running"&&"bg-blue-100 text-blue-700",s.status==="completed"&&"bg-green-100 text-green-700",s.status==="failed"&&"bg-red-100 text-red-700",s.status==="cancelled"&&"bg-gray-100 text-gray-600"),children:[s.status==="running"&&"运行中",s.status==="completed"&&"已完成",s.status==="failed"&&"失败",s.status==="cancelled"&&"已取消"]}),s.mode&&e.jsxs("span",{className:p("inline-flex items-center gap-1 px-2 py-1 text-xs font-medium rounded-full",s.mode==="background"?"bg-purple-100 text-purple-700":"bg-sky-100 text-sky-700"),title:s.mode==="background"?"独立 claude -p 子进程":"主会话内执行",children:[s.mode==="background"?e.jsx(T,{className:"h-3 w-3"}):e.jsx(P,{className:"h-3 w-3"}),s.mode==="background"?"后台":"前台"]})]}),e.jsxs("p",{className:"text-sm text-gray-600 mb-3 font-mono",children:["Session: ",s.session_id.slice(0,12),"..."]}),e.jsx("div",{className:"w-full bg-gray-200 rounded-full h-1.5 mb-3",children:e.jsx("div",{className:p("h-1.5 rounded-full transition-all",s.status==="completed"?"bg-green-500":s.status==="cancelled"?"bg-gray-400":"bg-indigo-600"),style:{width:`${I}%`}})}),e.jsxs("div",{className:"flex items-center gap-4 text-sm text-gray-500",children:[e.jsxs("div",{className:"flex items-center gap-1",children:[e.jsx(Q,{className:"h-4 w-4"}),U(s.started_at,{addSuffix:!0,locale:X})]}),s.status==="running"&&v&&e.jsxs("div",{className:"flex items-center gap-1",children:[e.jsx(O,{className:"h-4 w-4"}),"Phase ",s.current_phase_index+1,"/",l.phases.length,": ",v.id]}),s.status==="completed"&&e.jsxs("div",{className:"flex items-center gap-1 text-green-600",children:[e.jsx(R,{className:"h-4 w-4"}),"完成 ",l.phases.length," 个阶段"]}),s.status==="cancelled"&&e.jsxs("div",{className:"flex items-center gap-1 text-gray-500",children:[e.jsx(H,{className:"h-4 w-4"}),"已取消于 Phase ",s.current_phase_index+1]})]})]}),e.jsxs("div",{className:"flex items-center gap-1 flex-shrink-0",children:[s.status==="running"&&e.jsx("button",{onClick:k=>{k.preventDefault(),h(s.id)},className:"p-2 text-orange-600 hover:bg-orange-50 rounded-md",title:"取消执行",children:e.jsx(B,{className:"h-4 w-4"})}),s.status!=="running"&&e.jsx("button",{onClick:k=>{k.preventDefault(),w(s.id)},className:"p-2 text-gray-500 hover:bg-red-50 hover:text-red-600 rounded-md",title:"删除记录",children:e.jsx(L,{className:"h-4 w-4"})})]})]})},s.id)}),(!r||r.length===0)&&e.jsxs("div",{className:"bg-white rounded-lg shadow p-12 text-center",children:[e.jsx(S,{className:"h-12 w-12 text-gray-400 mx-auto mb-4"}),e.jsx("p",{className:"text-gray-600",children:"暂无方法论执行记录"}),e.jsx("p",{className:"text-xs text-gray-500 mt-2",children:'当你输入复杂任务(如"重构 XXX 模块")时,系统会自动匹配方法论并生成执行计划'})]})]})]})}function ee({methodologies:t,authStatus:a,submitting:o,onSubmit:i,onCancel:j,onGoToAIConfig:d}){var u;const r=`manual-${Date.now().toString(36)}`,[g,f]=b.useState(r),[c,N]=b.useState(((u=t[0])==null?void 0:u.id)??""),[m,y]=b.useState("background"),x=a!==null&&a.available===!1,h=x&&m==="background",w=n=>{n.preventDefault(),!(!g.trim()||!c)&&(h||i({session_id:g.trim(),methodology_id:c,mode:m}))};return e.jsx("div",{className:"fixed inset-0 z-50 flex items-center justify-center bg-black/40 p-4",children:e.jsxs("form",{onSubmit:w,className:"w-full max-w-md bg-white rounded-lg shadow-xl border border-gray-200",children:[e.jsxs("div",{className:"px-5 py-4 border-b border-gray-100",children:[e.jsx("h2",{className:"text-lg font-semibold text-gray-900",children:"新建方法论执行"}),e.jsx("p",{className:"text-xs text-gray-500 mt-0.5",children:"选择方法论模板和执行模式"})]}),e.jsxs("div",{className:"px-5 py-4 space-y-4 text-sm",children:[e.jsxs("label",{className:"block",children:[e.jsx("span",{className:"text-gray-700",children:"方法论"}),e.jsxs("select",{className:"mt-1 w-full border border-gray-300 rounded-md px-2 py-1.5",value:c,onChange:n=>N(n.target.value),required:!0,children:[t.length===0&&e.jsx("option",{value:"",children:"(未发现方法论)"}),t.map(n=>e.jsx("option",{value:n.id,children:n.name||n.id},n.id))]})]}),e.jsxs("label",{className:"block",children:[e.jsx("span",{className:"text-gray-700",children:"Session ID"}),e.jsx("input",{type:"text",className:"mt-1 w-full border border-gray-300 rounded-md px-2 py-1.5 font-mono text-xs",value:g,onChange:n=>f(n.target.value),required:!0}),e.jsx("span",{className:"mt-1 block text-xs text-gray-500",children:"后台模式下用于追踪;建议保留默认或自定义一个可识别值"})]}),e.jsxs("div",{children:[e.jsx("span",{className:"text-gray-700",children:"执行模式"}),e.jsxs("div",{className:"mt-2 grid grid-cols-2 gap-2",children:[e.jsxs("button",{type:"button",onClick:()=>y("background"),className:p("rounded-md border px-3 py-2 text-left",m==="background"?"border-indigo-500 bg-indigo-50":"border-gray-200 hover:border-gray-300",x&&"ring-1 ring-amber-200"),children:[e.jsxs("div",{className:"flex items-center gap-2 font-medium text-gray-900",children:[e.jsx(T,{className:"h-4 w-4 text-purple-600"})," 后台",x&&e.jsx("span",{className:"ml-auto text-[10px] font-normal text-amber-700 bg-amber-100 px-1.5 py-0.5 rounded",children:"需先配置 API"})]}),e.jsx("p",{className:"text-xs text-gray-500 mt-1",children:"独立 claude -p 子进程,无需用户在主会话输入"})]}),e.jsxs("button",{type:"button",onClick:()=>y("foreground"),className:p("rounded-md border px-3 py-2 text-left",m==="foreground"?"border-indigo-500 bg-indigo-50":"border-gray-200 hover:border-gray-300"),children:[e.jsxs("div",{className:"flex items-center gap-2 font-medium text-gray-900",children:[e.jsx(P,{className:"h-4 w-4 text-sky-600"})," 前台"]}),e.jsxs("p",{className:"text-xs text-amber-700 mt-1 flex items-start gap-1",children:[e.jsx(_,{className:"h-3.5 w-3.5 mt-0.5 shrink-0"}),e.jsxs("span",{children:["需要你在主会话里输入任意内容才会推进;期间可用 ",e.jsx("code",{className:"px-1 rounded bg-amber-50",children:"/forge methodology status"})," 查看"]})]})]})]})]}),h&&e.jsx("div",{className:"rounded-md border border-amber-300 bg-amber-50 px-3 py-2 text-xs text-amber-800",children:e.jsxs("div",{className:"flex items-start gap-2",children:[e.jsx(_,{className:"h-4 w-4 mt-0.5 shrink-0 text-amber-600"}),e.jsxs("div",{className:"flex-1 space-y-1",children:[e.jsx("div",{className:"font-medium",children:"后台模式需要 Claude API 认证"}),e.jsxs("div",{className:"text-amber-700",children:["未检测到 ",e.jsx("code",{className:"px-1 rounded bg-amber-100",children:"ANTHROPIC_API_KEY"})," 或 OAuth 凭据",a!=null&&a.message?`(${a.message})`:"","。 请先在 AI 配置页完成设置,或切换到前台模式。"]}),e.jsx("button",{type:"button",onClick:d,className:"mt-1 inline-flex items-center gap-1 px-2 py-1 text-xs font-medium text-white bg-amber-600 rounded hover:bg-amber-700",children:"去 AI 配置"})]})]})})]}),e.jsxs("div",{className:"px-5 py-3 border-t border-gray-100 flex justify-end gap-2",children:[e.jsx("button",{type:"button",onClick:j,className:"px-3 py-1.5 text-sm text-gray-700 hover:bg-gray-100 rounded-md",children:"取消"}),e.jsx("button",{type:"submit",disabled:o||!c||h,title:h?"后台模式需要先配置 Claude API":void 0,className:"inline-flex items-center gap-1 px-3 py-1.5 text-sm bg-indigo-600 text-white rounded-md hover:bg-indigo-700 disabled:opacity-50 disabled:cursor-not-allowed",children:o?"启动中...":"启动"})]})]})})}export{ce as default};
5
- //# sourceMappingURL=Methodologies-CXNrDXwG.js.map