@winspan/claude-forge 8.30.0 → 8.33.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 (195) hide show
  1. package/README.md +4 -4
  2. package/dist/capability/execution-manager.d.ts +38 -1
  3. package/dist/capability/execution-manager.d.ts.map +1 -1
  4. package/dist/capability/execution-manager.js +93 -1
  5. package/dist/capability/execution-manager.js.map +1 -1
  6. package/dist/capability/executor/background-executor.d.ts +1 -0
  7. package/dist/capability/executor/background-executor.d.ts.map +1 -1
  8. package/dist/capability/executor/background-executor.js +27 -4
  9. package/dist/capability/executor/background-executor.js.map +1 -1
  10. package/dist/capability/executor/orchestrator.d.ts +15 -2
  11. package/dist/capability/executor/orchestrator.d.ts.map +1 -1
  12. package/dist/capability/executor/orchestrator.js +82 -3
  13. package/dist/capability/executor/orchestrator.js.map +1 -1
  14. package/dist/capability/executor/worker-auth-probe.d.ts.map +1 -1
  15. package/dist/capability/executor/worker-auth-probe.js +11 -2
  16. package/dist/capability/executor/worker-auth-probe.js.map +1 -1
  17. package/dist/capability/methodologies/bmad.yaml +17 -5
  18. package/dist/capability/methodologies/code-quality-audit.yaml +26 -0
  19. package/dist/capability/methodologies/harness-engineering.yaml +12 -6
  20. package/dist/capability/methodologies/test-coverage-scan.yaml +26 -0
  21. package/dist/capability/methodology-planner.d.ts +17 -1
  22. package/dist/capability/methodology-planner.d.ts.map +1 -1
  23. package/dist/capability/methodology-planner.js +125 -0
  24. package/dist/capability/methodology-planner.js.map +1 -1
  25. package/dist/capability/methodology-registry.d.ts.map +1 -1
  26. package/dist/capability/methodology-registry.js +21 -5
  27. package/dist/capability/methodology-registry.js.map +1 -1
  28. package/dist/capability/types.d.ts +2 -0
  29. package/dist/capability/types.d.ts.map +1 -1
  30. package/dist/core/ai/provider.d.ts +17 -9
  31. package/dist/core/ai/provider.d.ts.map +1 -1
  32. package/dist/core/ai/provider.js +130 -23
  33. package/dist/core/ai/provider.js.map +1 -1
  34. package/dist/core/ai/types.d.ts +26 -5
  35. package/dist/core/ai/types.d.ts.map +1 -1
  36. package/dist/core/storage/rows.d.ts +153 -0
  37. package/dist/core/storage/rows.d.ts.map +1 -0
  38. package/dist/core/storage/rows.js +14 -0
  39. package/dist/core/storage/rows.js.map +1 -0
  40. package/dist/core/storage/schema.sql +26 -2
  41. package/dist/core/storage/sqlite.d.ts +95 -7
  42. package/dist/core/storage/sqlite.d.ts.map +1 -1
  43. package/dist/core/storage/sqlite.js +409 -22
  44. package/dist/core/storage/sqlite.js.map +1 -1
  45. package/dist/core/utils/token-tracker.d.ts +40 -0
  46. package/dist/core/utils/token-tracker.d.ts.map +1 -0
  47. package/dist/core/utils/token-tracker.js +70 -0
  48. package/dist/core/utils/token-tracker.js.map +1 -0
  49. package/dist/daemon/handlers/post-tool-use.d.ts +1 -0
  50. package/dist/daemon/handlers/post-tool-use.d.ts.map +1 -1
  51. package/dist/daemon/handlers/post-tool-use.js +7 -0
  52. package/dist/daemon/handlers/post-tool-use.js.map +1 -1
  53. package/dist/daemon/handlers/stop.d.ts +11 -0
  54. package/dist/daemon/handlers/stop.d.ts.map +1 -1
  55. package/dist/daemon/handlers/stop.js +52 -0
  56. package/dist/daemon/handlers/stop.js.map +1 -1
  57. package/dist/daemon/handlers/user-prompt.d.ts.map +1 -1
  58. package/dist/daemon/handlers/user-prompt.js +63 -4
  59. package/dist/daemon/handlers/user-prompt.js.map +1 -1
  60. package/dist/daemon/idle-detector.d.ts +35 -0
  61. package/dist/daemon/idle-detector.d.ts.map +1 -0
  62. package/dist/daemon/idle-detector.js +56 -0
  63. package/dist/daemon/idle-detector.js.map +1 -0
  64. package/dist/daemon/idle-trigger.d.ts +53 -0
  65. package/dist/daemon/idle-trigger.d.ts.map +1 -0
  66. package/dist/daemon/idle-trigger.js +153 -0
  67. package/dist/daemon/idle-trigger.js.map +1 -0
  68. package/dist/daemon/index.d.ts.map +1 -1
  69. package/dist/daemon/index.js +30 -2
  70. package/dist/daemon/index.js.map +1 -1
  71. package/dist/daemon/routing-observer.d.ts +2 -1
  72. package/dist/daemon/routing-observer.d.ts.map +1 -1
  73. package/dist/daemon/routing-observer.js +117 -39
  74. package/dist/daemon/routing-observer.js.map +1 -1
  75. package/dist/engine/agent-router.d.ts +6 -0
  76. package/dist/engine/agent-router.d.ts.map +1 -1
  77. package/dist/engine/agent-router.js +13 -1
  78. package/dist/engine/agent-router.js.map +1 -1
  79. package/dist/engine/conventions/routing.yaml +15 -0
  80. package/dist/engine/dsl/compiler.d.ts.map +1 -1
  81. package/dist/engine/dsl/compiler.js +85 -3
  82. package/dist/engine/dsl/compiler.js.map +1 -1
  83. package/dist/engine/recommender.d.ts.map +1 -1
  84. package/dist/engine/recommender.js +10 -1
  85. package/dist/engine/recommender.js.map +1 -1
  86. package/dist/intelligence/classifier.d.ts +6 -0
  87. package/dist/intelligence/classifier.d.ts.map +1 -1
  88. package/dist/intelligence/classifier.js +57 -0
  89. package/dist/intelligence/classifier.js.map +1 -1
  90. package/dist/skills/registry.d.ts +6 -0
  91. package/dist/skills/registry.d.ts.map +1 -1
  92. package/dist/skills/registry.js +49 -14
  93. package/dist/skills/registry.js.map +1 -1
  94. package/dist/skills/semantic-matcher.d.ts +1 -0
  95. package/dist/skills/semantic-matcher.d.ts.map +1 -1
  96. package/dist/skills/semantic-matcher.js +6 -1
  97. package/dist/skills/semantic-matcher.js.map +1 -1
  98. package/dist/web/auth-middleware.d.ts +22 -0
  99. package/dist/web/auth-middleware.d.ts.map +1 -0
  100. package/dist/web/auth-middleware.js +51 -0
  101. package/dist/web/auth-middleware.js.map +1 -0
  102. package/dist/web/routes/agents.d.ts +7 -0
  103. package/dist/web/routes/agents.d.ts.map +1 -0
  104. package/dist/web/routes/agents.js +192 -0
  105. package/dist/web/routes/agents.js.map +1 -0
  106. package/dist/web/routes/ai.d.ts +10 -0
  107. package/dist/web/routes/ai.d.ts.map +1 -0
  108. package/dist/web/routes/ai.js +197 -0
  109. package/dist/web/routes/ai.js.map +1 -0
  110. package/dist/web/routes/auth.d.ts +12 -0
  111. package/dist/web/routes/auth.d.ts.map +1 -0
  112. package/dist/web/routes/auth.js +20 -0
  113. package/dist/web/routes/auth.js.map +1 -0
  114. package/dist/web/routes/events.d.ts +11 -0
  115. package/dist/web/routes/events.d.ts.map +1 -0
  116. package/dist/web/routes/events.js +43 -0
  117. package/dist/web/routes/events.js.map +1 -0
  118. package/dist/web/routes/execution-trace.d.ts +13 -0
  119. package/dist/web/routes/execution-trace.d.ts.map +1 -0
  120. package/dist/web/routes/execution-trace.js +308 -0
  121. package/dist/web/routes/execution-trace.js.map +1 -0
  122. package/dist/web/routes/experiments.d.ts +15 -0
  123. package/dist/web/routes/experiments.d.ts.map +1 -0
  124. package/dist/web/routes/experiments.js +187 -0
  125. package/dist/web/routes/experiments.js.map +1 -0
  126. package/dist/web/routes/methodology.d.ts +12 -0
  127. package/dist/web/routes/methodology.d.ts.map +1 -0
  128. package/dist/web/routes/methodology.js +228 -0
  129. package/dist/web/routes/methodology.js.map +1 -0
  130. package/dist/web/routes/patch.d.ts +7 -0
  131. package/dist/web/routes/patch.d.ts.map +1 -0
  132. package/dist/web/routes/patch.js +106 -0
  133. package/dist/web/routes/patch.js.map +1 -0
  134. package/dist/web/routes/routing.d.ts +17 -0
  135. package/dist/web/routes/routing.d.ts.map +1 -0
  136. package/dist/web/routes/routing.js +582 -0
  137. package/dist/web/routes/routing.js.map +1 -0
  138. package/dist/web/routes/rules.d.ts +7 -0
  139. package/dist/web/routes/rules.d.ts.map +1 -0
  140. package/dist/web/routes/rules.js +105 -0
  141. package/dist/web/routes/rules.js.map +1 -0
  142. package/dist/web/routes/sessions.d.ts +10 -0
  143. package/dist/web/routes/sessions.d.ts.map +1 -0
  144. package/dist/web/routes/sessions.js +234 -0
  145. package/dist/web/routes/sessions.js.map +1 -0
  146. package/dist/web/routes/skills.d.ts +10 -0
  147. package/dist/web/routes/skills.d.ts.map +1 -0
  148. package/dist/web/routes/skills.js +272 -0
  149. package/dist/web/routes/skills.js.map +1 -0
  150. package/dist/web/routes/static.d.ts +19 -0
  151. package/dist/web/routes/static.d.ts.map +1 -0
  152. package/dist/web/routes/static.js +61 -0
  153. package/dist/web/routes/static.js.map +1 -0
  154. package/dist/web/routes/status.d.ts +7 -0
  155. package/dist/web/routes/status.d.ts.map +1 -0
  156. package/dist/web/routes/status.js +28 -0
  157. package/dist/web/routes/status.js.map +1 -0
  158. package/dist/web/routes/token-usage.d.ts +7 -0
  159. package/dist/web/routes/token-usage.d.ts.map +1 -0
  160. package/dist/web/routes/token-usage.js +33 -0
  161. package/dist/web/routes/token-usage.js.map +1 -0
  162. package/dist/web/routes/types.d.ts +40 -0
  163. package/dist/web/routes/types.d.ts.map +1 -0
  164. package/dist/web/routes/types.js +52 -0
  165. package/dist/web/routes/types.js.map +1 -0
  166. package/dist/web/server.d.ts +7 -4
  167. package/dist/web/server.d.ts.map +1 -1
  168. package/dist/web/server.js +60 -2330
  169. package/dist/web/server.js.map +1 -1
  170. package/dist/web/ssrf-guard.d.ts +35 -0
  171. package/dist/web/ssrf-guard.d.ts.map +1 -0
  172. package/dist/web/ssrf-guard.js +93 -0
  173. package/dist/web/ssrf-guard.js.map +1 -0
  174. package/dist/web/static/assets/{AIConfig-nZgwaowr.js → AIConfig-D-vrYoJ3.js} +2 -2
  175. package/dist/web/static/assets/{AIConfig-nZgwaowr.js.map → AIConfig-D-vrYoJ3.js.map} +1 -1
  176. package/dist/web/static/assets/{Agents-BZGXKWC7.js → Agents-DAGWYsJj.js} +2 -2
  177. package/dist/web/static/assets/{Agents-BZGXKWC7.js.map → Agents-DAGWYsJj.js.map} +1 -1
  178. package/dist/web/static/assets/{Events-CnA3f740.js → Events-BoQ8Fo5k.js} +2 -2
  179. package/dist/web/static/assets/{Events-CnA3f740.js.map → Events-BoQ8Fo5k.js.map} +1 -1
  180. package/dist/web/static/assets/{ExecutionTrace-ClPfFIQa.js → ExecutionTrace-sFZ_vHNf.js} +2 -2
  181. package/dist/web/static/assets/{ExecutionTrace-ClPfFIQa.js.map → ExecutionTrace-sFZ_vHNf.js.map} +1 -1
  182. package/dist/web/static/assets/Methodologies-C0-Keokj.js +5 -0
  183. package/dist/web/static/assets/Methodologies-C0-Keokj.js.map +1 -0
  184. package/dist/web/static/assets/{Sessions-DwWOKgnl.js → Sessions-Bjf-Mvwb.js} +2 -2
  185. package/dist/web/static/assets/{Sessions-DwWOKgnl.js.map → Sessions-Bjf-Mvwb.js.map} +1 -1
  186. package/dist/web/static/assets/{Skills-DhM6ALhr.js → Skills-CrLshkrJ.js} +2 -2
  187. package/dist/web/static/assets/{Skills-DhM6ALhr.js.map → Skills-CrLshkrJ.js.map} +1 -1
  188. package/dist/web/static/assets/{index-DUYj2ek1.js → index-D23sAOAt.js} +3 -3
  189. package/dist/web/static/assets/{index-DUYj2ek1.js.map → index-D23sAOAt.js.map} +1 -1
  190. package/dist/web/static/assets/index-Drpf7sLl.css +1 -0
  191. package/dist/web/static/index.html +2 -2
  192. package/package.json +3 -2
  193. package/dist/web/static/assets/Methodologies-CAXUXeox.js +0 -2
  194. package/dist/web/static/assets/Methodologies-CAXUXeox.js.map +0 -1
  195. package/dist/web/static/assets/index-CVWult53.css +0 -1
@@ -0,0 +1,192 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import { homedir } from 'os';
4
+ import { logger } from '../../core/utils/logger.js';
5
+ /**
6
+ * /api/agents/* — list, read, update, version, rollback.
7
+ */
8
+ export function registerAgentsRoutes(app, ctx) {
9
+ const agents = ctx.agents;
10
+ // GET /api/agents — list all agents (official + user)
11
+ app.get('/api/agents', (_req, res) => {
12
+ try {
13
+ const agentsDir = path.join(homedir(), '.claude', 'agents');
14
+ const official = agents?.getAll() || [];
15
+ const userAgents = [];
16
+ if (fs.existsSync(agentsDir)) {
17
+ const files = fs.readdirSync(agentsDir).filter(f => f.endsWith('.md'));
18
+ for (const file of files) {
19
+ const name = file.replace('.md', '');
20
+ if (official.some((a) => a.name === name))
21
+ continue;
22
+ const content = fs.readFileSync(path.join(agentsDir, file), 'utf-8');
23
+ const match = content.match(/^---\n([\s\S]*?)\n---/);
24
+ if (match) {
25
+ const fm = match[1];
26
+ const descMatch = fm.match(/description:\s*["']?([^"'\n]+)["']?/);
27
+ const verMatch = fm.match(/version:\s*["']?([^"'\n]+)["']?/);
28
+ userAgents.push({
29
+ name,
30
+ description: descMatch?.[1] || '',
31
+ version: verMatch?.[1],
32
+ });
33
+ }
34
+ }
35
+ }
36
+ res.json({ official, user: userAgents });
37
+ }
38
+ catch (err) {
39
+ logger.warn(`[Web] Failed to list agents: ${err}`);
40
+ res.status(500).json({ error: String(err) });
41
+ }
42
+ });
43
+ // GET /api/agents/:name — get agent details
44
+ app.get('/api/agents/:name', (req, res) => {
45
+ try {
46
+ const { name } = req.params;
47
+ const agentsDir = path.join(homedir(), '.claude', 'agents');
48
+ const filePath = path.join(agentsDir, `${name}.md`);
49
+ if (!fs.existsSync(filePath)) {
50
+ res.status(404).json({ error: 'Agent not found' });
51
+ return;
52
+ }
53
+ const content = fs.readFileSync(filePath, 'utf-8');
54
+ const match = content.match(/^---\n([\s\S]*?)\n---\n([\s\S]*)$/);
55
+ if (!match) {
56
+ res.json({ name, content });
57
+ return;
58
+ }
59
+ const fm = match[1];
60
+ const descMatch = fm.match(/description:\s*["']?([^"'\n]+)["']?/);
61
+ const verMatch = fm.match(/version:\s*["']?([^"'\n]+)["']?/);
62
+ const toolsMatch = fm.match(/tools:\s*(.+)/);
63
+ res.json({
64
+ name,
65
+ description: descMatch?.[1] || '',
66
+ version: verMatch?.[1],
67
+ tools: toolsMatch?.[1],
68
+ content,
69
+ });
70
+ }
71
+ catch (err) {
72
+ logger.warn(`[Web] Failed to get agent ${req.params.name}: ${err}`);
73
+ res.status(500).json({ error: String(err) });
74
+ }
75
+ });
76
+ // PUT /api/agents/:name — update agent content
77
+ app.put('/api/agents/:name', (req, res) => {
78
+ try {
79
+ const { name } = req.params;
80
+ const { content } = req.body;
81
+ if (!content || typeof content !== 'string') {
82
+ res.status(400).json({ error: 'Missing content' });
83
+ return;
84
+ }
85
+ const agentsDir = path.join(homedir(), '.claude', 'agents');
86
+ const filePath = path.join(agentsDir, `${name}.md`);
87
+ if (!fs.existsSync(filePath)) {
88
+ res.status(404).json({ error: 'Agent not found' });
89
+ return;
90
+ }
91
+ const backupDir = path.join(homedir(), '.claude-forge', 'backups', 'agents');
92
+ fs.mkdirSync(backupDir, { recursive: true });
93
+ const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
94
+ const backupPath = path.join(backupDir, `${name}-${timestamp}.md`);
95
+ fs.copyFileSync(filePath, backupPath);
96
+ fs.writeFileSync(filePath, content, 'utf-8');
97
+ logger.info(`[Web] Updated agent ${name} (backup: ${backupPath})`);
98
+ res.json({ success: true, backup: backupPath });
99
+ }
100
+ catch (err) {
101
+ logger.warn(`[Web] Failed to update agent ${req.params.name}: ${err}`);
102
+ res.status(500).json({ error: String(err) });
103
+ }
104
+ });
105
+ // GET /api/agents/:name/versions — list backup versions
106
+ app.get('/api/agents/:name/versions', (req, res) => {
107
+ try {
108
+ const { name } = req.params;
109
+ const backupDir = path.join(homedir(), '.claude-forge', 'backups', 'agents');
110
+ if (!fs.existsSync(backupDir)) {
111
+ res.json({ versions: [] });
112
+ return;
113
+ }
114
+ const files = fs.readdirSync(backupDir)
115
+ .filter(f => f.startsWith(`${name}-`) && f.endsWith('.md'))
116
+ .map(f => {
117
+ const filePath = path.join(backupDir, f);
118
+ const stats = fs.statSync(filePath);
119
+ const timestampMatch = f.match(/-(\d{4}-\d{2}-\d{2}T\d{2}-\d{2}-\d{2}-\d{3}Z)\.md$/);
120
+ return {
121
+ filename: f,
122
+ timestamp: timestampMatch ? timestampMatch[1].replace(/-/g, ':').replace(/T(\d{2}):(\d{2}):(\d{2})/, 'T$1:$2:$3') : '',
123
+ size: stats.size,
124
+ mtime: stats.mtime.toISOString(),
125
+ };
126
+ })
127
+ .sort((a, b) => new Date(b.mtime).getTime() - new Date(a.mtime).getTime());
128
+ res.json({ versions: files });
129
+ }
130
+ catch (err) {
131
+ logger.warn(`[Web] Failed to list agent versions: ${err}`);
132
+ res.status(500).json({ error: String(err) });
133
+ }
134
+ });
135
+ // GET /api/agents/:name/versions/:timestamp — get specific version content
136
+ app.get('/api/agents/:name/versions/:timestamp', (req, res) => {
137
+ try {
138
+ const { name, timestamp } = req.params;
139
+ const backupDir = path.join(homedir(), '.claude-forge', 'backups', 'agents');
140
+ const filename = `${name}-${timestamp}.md`;
141
+ const filePath = path.join(backupDir, filename);
142
+ if (!fs.existsSync(filePath)) {
143
+ res.status(404).json({ error: 'Version not found' });
144
+ return;
145
+ }
146
+ const content = fs.readFileSync(filePath, 'utf-8');
147
+ res.json({ content });
148
+ }
149
+ catch (err) {
150
+ logger.warn(`[Web] Failed to get agent version: ${err}`);
151
+ res.status(500).json({ error: String(err) });
152
+ }
153
+ });
154
+ // POST /api/agents/:name/rollback — rollback to a specific version
155
+ app.post('/api/agents/:name/rollback', (req, res) => {
156
+ try {
157
+ const { name } = req.params;
158
+ const { timestamp } = req.body;
159
+ if (!timestamp) {
160
+ res.status(400).json({ error: 'Missing timestamp' });
161
+ return;
162
+ }
163
+ const agentsDir = path.join(homedir(), '.claude', 'agents');
164
+ const currentPath = path.join(agentsDir, `${name}.md`);
165
+ const backupDir = path.join(homedir(), '.claude-forge', 'backups', 'agents');
166
+ const versionPath = path.join(backupDir, `${name}-${timestamp}.md`);
167
+ if (!fs.existsSync(currentPath)) {
168
+ res.status(404).json({ error: 'Agent not found' });
169
+ return;
170
+ }
171
+ if (!fs.existsSync(versionPath)) {
172
+ res.status(404).json({ error: 'Version not found' });
173
+ return;
174
+ }
175
+ // Backup current version before rollback
176
+ fs.mkdirSync(backupDir, { recursive: true });
177
+ const rollbackTimestamp = new Date().toISOString().replace(/[:.]/g, '-');
178
+ const rollbackBackupPath = path.join(backupDir, `${name}-${rollbackTimestamp}.md`);
179
+ fs.copyFileSync(currentPath, rollbackBackupPath);
180
+ // Restore version
181
+ const versionContent = fs.readFileSync(versionPath, 'utf-8');
182
+ fs.writeFileSync(currentPath, versionContent, 'utf-8');
183
+ logger.info(`[Web] Rolled back agent ${name} to ${timestamp} (backup: ${rollbackBackupPath})`);
184
+ res.json({ success: true, backup: rollbackBackupPath });
185
+ }
186
+ catch (err) {
187
+ logger.warn(`[Web] Failed to rollback agent: ${err}`);
188
+ res.status(500).json({ error: String(err) });
189
+ }
190
+ });
191
+ }
192
+ //# sourceMappingURL=agents.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"agents.js","sourceRoot":"","sources":["../../../src/web/routes/agents.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ import type { Application } from 'express';
2
+ import type { RouteContext } from './types.js';
3
+ /**
4
+ * /api/ai/* and /api/config/ai — AI provider config + upstream connectivity.
5
+ *
6
+ * All outbound calls route through validateBaseUrl + redirect:'manual' so the
7
+ * daemon cannot be tricked into reaching internal hosts (SSRF guard).
8
+ */
9
+ export declare function registerAIRoutes(app: Application, _ctx: RouteContext): void;
10
+ //# sourceMappingURL=ai.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ai.d.ts","sourceRoot":"","sources":["../../../src/web/routes/ai.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAQ3C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAE/C;;;;;GAKG;AACH,wBAAgB,gBAAgB,CAAC,GAAG,EAAE,WAAW,EAAE,IAAI,EAAE,YAAY,GAAG,IAAI,CAoM3E"}
@@ -0,0 +1,197 @@
1
+ import { ConfigManager } from '../../core/config.js';
2
+ import { logger } from '../../core/utils/logger.js';
3
+ import { validateBaseUrl, extractHostFromBaseUrl, UPSTREAM_TIMEOUT_MS, } from '../ssrf-guard.js';
4
+ /**
5
+ * /api/ai/* and /api/config/ai — AI provider config + upstream connectivity.
6
+ *
7
+ * All outbound calls route through validateBaseUrl + redirect:'manual' so the
8
+ * daemon cannot be tricked into reaching internal hosts (SSRF guard).
9
+ */
10
+ export function registerAIRoutes(app, _ctx) {
11
+ // GET /api/config/ai — read current AI config (mask apiKey)
12
+ app.get('/api/config/ai', (_req, res) => {
13
+ const configManager = new ConfigManager();
14
+ const config = configManager.get();
15
+ const maskApiKey = (key) => {
16
+ if (!key || key.length < 10)
17
+ return '***';
18
+ return key.slice(0, 6) + '***' + key.slice(-4);
19
+ };
20
+ res.json({
21
+ api_key: maskApiKey(config.distill.api_key),
22
+ base_url: config.distill.base_url || '',
23
+ model: config.distill.model,
24
+ provider: config.distill.provider,
25
+ classifier_model: config.distill.classifier_model || '',
26
+ classifier_timeout: config.distill.classifier_timeout || 10000,
27
+ });
28
+ });
29
+ // PUT /api/config/ai — update AI config
30
+ app.put('/api/config/ai', (req, res) => {
31
+ const { api_key, base_url, model, provider, classifier_model, classifier_timeout } = req.body ?? {};
32
+ try {
33
+ const configManager = new ConfigManager();
34
+ const config = configManager.get();
35
+ if (typeof api_key === 'string')
36
+ config.distill.api_key = api_key;
37
+ if (typeof base_url === 'string') {
38
+ // Normalize http → https for upstream gateways that force HTTPS redirect
39
+ // (e.g. iflytek one.iflytek.com returns 302 and drops Authorization header)
40
+ let url = base_url.trim();
41
+ if (url.startsWith('http://'))
42
+ url = 'https://' + url.slice('http://'.length);
43
+ config.distill.base_url = url;
44
+ }
45
+ if (typeof model === 'string')
46
+ config.distill.model = model;
47
+ if (typeof provider === 'string')
48
+ config.distill.provider = provider;
49
+ if (typeof classifier_model === 'string')
50
+ config.distill.classifier_model = classifier_model || undefined;
51
+ if (typeof classifier_timeout === 'number')
52
+ config.distill.classifier_timeout = classifier_timeout;
53
+ configManager.save();
54
+ logger.info('[Web] AI config updated');
55
+ res.json({ success: true });
56
+ }
57
+ catch (err) {
58
+ logger.warn(`[Web] Failed to update AI config: ${err}`);
59
+ res.status(500).json({ error: String(err) });
60
+ }
61
+ });
62
+ // GET /api/ai/models — proxy to upstream /v1/models
63
+ app.get('/api/ai/models', async (req, res) => {
64
+ try {
65
+ const configManager = new ConfigManager();
66
+ const config = configManager.get();
67
+ const queryKey = typeof req.query.api_key === 'string' ? req.query.api_key : '';
68
+ const queryUrl = typeof req.query.base_url === 'string' ? req.query.base_url : '';
69
+ const apiKey = queryKey || config.distill.api_key;
70
+ let baseUrl = queryUrl || config.distill.base_url || 'https://api.anthropic.com';
71
+ if (baseUrl.startsWith('http://'))
72
+ baseUrl = 'https://' + baseUrl.slice('http://'.length);
73
+ if (!apiKey) {
74
+ res.status(400).json({ error: 'API key not configured' });
75
+ return;
76
+ }
77
+ const configuredHost = extractHostFromBaseUrl(config.distill.base_url);
78
+ const validation = validateBaseUrl(baseUrl, configuredHost ? [configuredHost] : []);
79
+ if (!validation.ok) {
80
+ res.status(400).json({ error: validation.error });
81
+ return;
82
+ }
83
+ const modelsUrl = baseUrl.endsWith('/v1')
84
+ ? `${baseUrl}/models`
85
+ : `${baseUrl}/v1/models`;
86
+ const controller = new AbortController();
87
+ const timer = setTimeout(() => controller.abort(), UPSTREAM_TIMEOUT_MS);
88
+ let response;
89
+ try {
90
+ response = await fetch(modelsUrl, {
91
+ redirect: 'manual',
92
+ signal: controller.signal,
93
+ headers: {
94
+ 'Authorization': `Bearer ${apiKey}`,
95
+ 'x-api-key': apiKey,
96
+ },
97
+ });
98
+ }
99
+ finally {
100
+ clearTimeout(timer);
101
+ }
102
+ if (response.status >= 300 && response.status < 400) {
103
+ res.status(400).json({
104
+ error: `Upstream tried to redirect (status ${response.status}); refusing to follow`,
105
+ });
106
+ return;
107
+ }
108
+ if (!response.ok) {
109
+ const text = await response.text();
110
+ logger.warn(`[Web] Upstream /v1/models failed: ${response.status} ${text}`);
111
+ res.status(response.status).json({ error: text });
112
+ return;
113
+ }
114
+ const data = await response.json();
115
+ res.json(data);
116
+ }
117
+ catch (err) {
118
+ if (err instanceof Error && err.name === 'AbortError') {
119
+ res.status(504).json({ error: `Upstream timeout (${UPSTREAM_TIMEOUT_MS}ms)` });
120
+ return;
121
+ }
122
+ logger.warn(`[Web] Failed to fetch models: ${err}`);
123
+ res.status(500).json({ error: String(err) });
124
+ }
125
+ });
126
+ // POST /api/ai/test — test AI connection using provided or saved config
127
+ app.post('/api/ai/test', async (req, res) => {
128
+ try {
129
+ const configManager = new ConfigManager();
130
+ const config = configManager.get();
131
+ const { api_key, base_url, model } = req.body ?? {};
132
+ const apiKey = (typeof api_key === 'string' && api_key) || config.distill.api_key;
133
+ let baseUrl = (typeof base_url === 'string' && base_url) || config.distill.base_url || 'https://api.anthropic.com';
134
+ const useModel = (typeof model === 'string' && model) || config.distill.model;
135
+ if (baseUrl.startsWith('http://'))
136
+ baseUrl = 'https://' + baseUrl.slice('http://'.length);
137
+ if (!apiKey) {
138
+ res.status(400).json({ error: 'API key not configured' });
139
+ return;
140
+ }
141
+ const configuredHost = extractHostFromBaseUrl(config.distill.base_url);
142
+ const validation = validateBaseUrl(baseUrl, configuredHost ? [configuredHost] : []);
143
+ if (!validation.ok) {
144
+ res.status(400).json({ error: validation.error });
145
+ return;
146
+ }
147
+ const messagesUrl = baseUrl.endsWith('/v1')
148
+ ? `${baseUrl}/messages`
149
+ : `${baseUrl}/v1/messages`;
150
+ const controller = new AbortController();
151
+ const timer = setTimeout(() => controller.abort(), UPSTREAM_TIMEOUT_MS);
152
+ let response;
153
+ try {
154
+ response = await fetch(messagesUrl, {
155
+ method: 'POST',
156
+ redirect: 'manual',
157
+ signal: controller.signal,
158
+ headers: {
159
+ 'Content-Type': 'application/json',
160
+ 'x-api-key': apiKey,
161
+ 'anthropic-version': '2023-06-01',
162
+ },
163
+ body: JSON.stringify({
164
+ model: useModel,
165
+ max_tokens: 10,
166
+ messages: [{ role: 'user', content: 'ping' }],
167
+ }),
168
+ });
169
+ }
170
+ finally {
171
+ clearTimeout(timer);
172
+ }
173
+ if (response.status >= 300 && response.status < 400) {
174
+ res.status(400).json({
175
+ error: `Upstream tried to redirect (status ${response.status}); refusing to follow`,
176
+ });
177
+ return;
178
+ }
179
+ if (!response.ok) {
180
+ const text = await response.text();
181
+ res.status(response.status).json({ error: text });
182
+ return;
183
+ }
184
+ const data = await response.json();
185
+ res.json({ success: true, model: data.model || useModel });
186
+ }
187
+ catch (err) {
188
+ if (err instanceof Error && err.name === 'AbortError') {
189
+ res.status(504).json({ error: `Upstream timeout (${UPSTREAM_TIMEOUT_MS}ms)` });
190
+ return;
191
+ }
192
+ logger.warn(`[Web] AI connection test failed: ${err}`);
193
+ res.status(500).json({ error: String(err) });
194
+ }
195
+ });
196
+ }
197
+ //# sourceMappingURL=ai.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ai.js","sourceRoot":"","sources":["../../../src/web/routes/ai.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,12 @@
1
+ import type { Application } from 'express';
2
+ import type { RouteContext } from './types.js';
3
+ /**
4
+ * Auth routes — localhost bootstrap endpoint that returns the daemon token.
5
+ *
6
+ * This endpoint is intentionally NOT protected by requireAuth: the token file
7
+ * on disk is chmod 600 (current user only), so anyone who can reach this
8
+ * endpoint already has access to the file. Protecting it would create a
9
+ * chicken-and-egg problem for the UI.
10
+ */
11
+ export declare function registerAuthRoutes(app: Application, _ctx: RouteContext): void;
12
+ //# sourceMappingURL=auth.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.d.ts","sourceRoot":"","sources":["../../../src/web/routes/auth.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAE3C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAE/C;;;;;;;GAOG;AACH,wBAAgB,kBAAkB,CAAC,GAAG,EAAE,WAAW,EAAE,IAAI,EAAE,YAAY,GAAG,IAAI,CAS7E"}
@@ -0,0 +1,20 @@
1
+ import { readAuthToken } from '../auth-middleware.js';
2
+ /**
3
+ * Auth routes — localhost bootstrap endpoint that returns the daemon token.
4
+ *
5
+ * This endpoint is intentionally NOT protected by requireAuth: the token file
6
+ * on disk is chmod 600 (current user only), so anyone who can reach this
7
+ * endpoint already has access to the file. Protecting it would create a
8
+ * chicken-and-egg problem for the UI.
9
+ */
10
+ export function registerAuthRoutes(app, _ctx) {
11
+ app.get('/api/auth/token', (_req, res) => {
12
+ const token = readAuthToken();
13
+ if (!token) {
14
+ res.status(404).json({ error: 'TOKEN_NOT_FOUND' });
15
+ return;
16
+ }
17
+ res.json({ token });
18
+ });
19
+ }
20
+ //# sourceMappingURL=auth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../src/web/routes/auth.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAE,MAAM,uBAAuB,CAAC;AAGtD;;;;;;;GAOG;AACH,MAAM,UAAU,kBAAkB,CAAC,GAAgB,EAAE,IAAkB;IACrE,GAAG,CAAC,GAAG,CAAC,iBAAiB,EAAE,CAAC,IAAI,EAAE,GAAG,EAAE,EAAE;QACvC,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;QAC9B,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,CAAC,CAAC;YACnD,OAAO;QACT,CAAC;QACD,GAAG,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;IACtB,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,11 @@
1
+ import type { Application } from 'express';
2
+ import type { RouteContext } from './types.js';
3
+ /**
4
+ * /api/events — recent events list + SSE live stream.
5
+ *
6
+ * Note: /api/events/stream must be registered BEFORE `/api/events` handlers
7
+ * that might swallow it, but Express matches exact paths so order here is
8
+ * not actually load-bearing for correctness.
9
+ */
10
+ export declare function registerEventsRoutes(app: Application, ctx: RouteContext): void;
11
+ //# sourceMappingURL=events.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"events.d.ts","sourceRoot":"","sources":["../../../src/web/routes/events.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAC3C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAE/C;;;;;;GAMG;AACH,wBAAgB,oBAAoB,CAAC,GAAG,EAAE,WAAW,EAAE,GAAG,EAAE,YAAY,GAAG,IAAI,CAoC9E"}
@@ -0,0 +1,43 @@
1
+ /**
2
+ * /api/events — recent events list + SSE live stream.
3
+ *
4
+ * Note: /api/events/stream must be registered BEFORE `/api/events` handlers
5
+ * that might swallow it, but Express matches exact paths so order here is
6
+ * not actually load-bearing for correctness.
7
+ */
8
+ export function registerEventsRoutes(app, ctx) {
9
+ const { storage } = ctx;
10
+ app.get('/api/events', (req, res) => {
11
+ const limit = parseInt(req.query.limit) || 50;
12
+ const projectPath = req.query.project;
13
+ const sessionId = req.query.session;
14
+ const events = storage.queryEvents({ project_path: projectPath, session_id: sessionId, limit });
15
+ res.json(events);
16
+ });
17
+ // SSE: real-time event stream
18
+ app.get('/api/events/stream', (req, res) => {
19
+ res.writeHead(200, {
20
+ 'Content-Type': 'text/event-stream',
21
+ 'Cache-Control': 'no-cache',
22
+ Connection: 'keep-alive',
23
+ });
24
+ res.write('data: {"type":"connected"}\n\n');
25
+ const filterSession = req.query.session;
26
+ const filterProject = req.query.project;
27
+ const filterHook = req.query.hook;
28
+ const onEvent = (event) => {
29
+ if (filterSession && event.session_id !== filterSession)
30
+ return;
31
+ if (filterProject && event.project_path !== filterProject)
32
+ return;
33
+ if (filterHook && event.hook_type !== filterHook)
34
+ return;
35
+ res.write(`data: ${JSON.stringify(event)}\n\n`);
36
+ };
37
+ storage.on('event', onEvent);
38
+ req.on('close', () => {
39
+ storage.removeListener('event', onEvent);
40
+ });
41
+ });
42
+ }
43
+ //# sourceMappingURL=events.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"events.js","sourceRoot":"","sources":["../../../src/web/routes/events.ts"],"names":[],"mappings":"AAGA;;;;;;GAMG;AACH,MAAM,UAAU,oBAAoB,CAAC,GAAgB,EAAE,GAAiB;IACtE,MAAM,EAAE,OAAO,EAAE,GAAG,GAAG,CAAC;IAExB,GAAG,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;QAClC,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,KAAe,CAAC,IAAI,EAAE,CAAC;QACxD,MAAM,WAAW,GAAG,GAAG,CAAC,KAAK,CAAC,OAA6B,CAAC;QAC5D,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,OAA6B,CAAC;QAC1D,MAAM,MAAM,GAAG,OAAO,CAAC,WAAW,CAAC,EAAE,YAAY,EAAE,WAAW,EAAE,UAAU,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;QAChG,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IACnB,CAAC,CAAC,CAAC;IAEH,8BAA8B;IAC9B,GAAG,CAAC,GAAG,CAAC,oBAAoB,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;QACzC,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE;YACjB,cAAc,EAAE,mBAAmB;YACnC,eAAe,EAAE,UAAU;YAC3B,UAAU,EAAE,YAAY;SACzB,CAAC,CAAC;QACH,GAAG,CAAC,KAAK,CAAC,gCAAgC,CAAC,CAAC;QAE5C,MAAM,aAAa,GAAG,GAAG,CAAC,KAAK,CAAC,OAA6B,CAAC;QAC9D,MAAM,aAAa,GAAG,GAAG,CAAC,KAAK,CAAC,OAA6B,CAAC;QAC9D,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,IAA0B,CAAC;QAExD,MAAM,OAAO,GAAG,CAAC,KAA8B,EAAE,EAAE;YACjD,IAAI,aAAa,IAAI,KAAK,CAAC,UAAU,KAAK,aAAa;gBAAE,OAAO;YAChE,IAAI,aAAa,IAAI,KAAK,CAAC,YAAY,KAAK,aAAa;gBAAE,OAAO;YAClE,IAAI,UAAU,IAAI,KAAK,CAAC,SAAS,KAAK,UAAU;gBAAE,OAAO;YACzD,GAAG,CAAC,KAAK,CAAC,SAAS,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAClD,CAAC,CAAC;QAEF,OAAO,CAAC,EAAE,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAC7B,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;YACnB,OAAO,CAAC,cAAc,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAC3C,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,13 @@
1
+ import type { Application } from 'express';
2
+ import type { RouteContext } from './types.js';
3
+ /**
4
+ * /api/execution-trace/* — routing-event-centric traces + SSE live status.
5
+ *
6
+ * Registration order mirrors the original server.ts exactly. In particular,
7
+ * `/stream` is registered AFTER `/:id`, which means Express matches `/stream`
8
+ * against `/:id` first (Number('stream') = NaN → 400). This preserves the
9
+ * original behaviour; fixing it is out of scope for this split.
10
+ * FIXME: `/api/execution-trace/stream` is shadowed by `/:id` — out of scope for this split.
11
+ */
12
+ export declare function registerExecutionTraceRoutes(app: Application, ctx: RouteContext): void;
13
+ //# sourceMappingURL=execution-trace.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"execution-trace.d.ts","sourceRoot":"","sources":["../../../src/web/routes/execution-trace.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAE3C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAE/C;;;;;;;;GAQG;AACH,wBAAgB,4BAA4B,CAAC,GAAG,EAAE,WAAW,EAAE,GAAG,EAAE,YAAY,GAAG,IAAI,CA6XtF"}