openspec-mcp 0.3.2 → 0.3.4

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 (57) hide show
  1. package/README.md +14 -15
  2. package/dist/api/routes/kanban.d.ts.map +1 -1
  3. package/dist/api/routes/kanban.js +11 -16
  4. package/dist/api/routes/kanban.js.map +1 -1
  5. package/dist/api/server.d.ts.map +1 -1
  6. package/dist/api/server.js +30 -3
  7. package/dist/api/server.js.map +1 -1
  8. package/dist/core/approval-manager.d.ts +0 -4
  9. package/dist/core/approval-manager.d.ts.map +1 -1
  10. package/dist/core/approval-manager.js +5 -25
  11. package/dist/core/approval-manager.js.map +1 -1
  12. package/dist/core/approval-manager.test.js +6 -9
  13. package/dist/core/approval-manager.test.js.map +1 -1
  14. package/dist/index.js +0 -21
  15. package/dist/index.js.map +1 -1
  16. package/dist/server/tools/approval.js +4 -2
  17. package/dist/server/tools/approval.js.map +1 -1
  18. package/dist/server/tools/context.d.ts.map +1 -1
  19. package/dist/server/tools/context.js +0 -38
  20. package/dist/server/tools/context.js.map +1 -1
  21. package/dist/server/tools/critique.d.ts.map +1 -1
  22. package/dist/server/tools/critique.js +29 -42
  23. package/dist/server/tools/critique.js.map +1 -1
  24. package/dist/server/tools/generator.d.ts.map +1 -1
  25. package/dist/server/tools/generator.js +0 -23
  26. package/dist/server/tools/generator.js.map +1 -1
  27. package/dist/server/tools/reviews.d.ts.map +1 -1
  28. package/dist/server/tools/reviews.js +0 -25
  29. package/dist/server/tools/reviews.js.map +1 -1
  30. package/dist/types/openspec.d.ts +1 -1
  31. package/dist/types/openspec.d.ts.map +1 -1
  32. package/dist/utils/constants.d.ts +2 -3
  33. package/dist/utils/constants.d.ts.map +1 -1
  34. package/dist/utils/constants.js +2 -3
  35. package/dist/utils/constants.js.map +1 -1
  36. package/package.json +1 -1
  37. package/web/dist/assets/index-C5tEIOBf.css +1 -0
  38. package/web/dist/assets/index-D4LCY_M0.js +244 -0
  39. package/web/dist/index.html +2 -2
  40. package/dist/api/routes/qa.d.ts +0 -8
  41. package/dist/api/routes/qa.d.ts.map +0 -1
  42. package/dist/api/routes/qa.js +0 -98
  43. package/dist/api/routes/qa.js.map +0 -1
  44. package/dist/core/prompt-manager.d.ts +0 -50
  45. package/dist/core/prompt-manager.d.ts.map +0 -1
  46. package/dist/core/prompt-manager.js +0 -186
  47. package/dist/core/prompt-manager.js.map +0 -1
  48. package/dist/core/qa-runner.d.ts +0 -134
  49. package/dist/core/qa-runner.d.ts.map +0 -1
  50. package/dist/core/qa-runner.js +0 -299
  51. package/dist/core/qa-runner.js.map +0 -1
  52. package/dist/server/tools/qa.d.ts +0 -12
  53. package/dist/server/tools/qa.d.ts.map +0 -1
  54. package/dist/server/tools/qa.js +0 -248
  55. package/dist/server/tools/qa.js.map +0 -1
  56. package/web/dist/assets/index-Bf5mzJti.css +0 -1
  57. package/web/dist/assets/index-W9UMaaAn.js +0 -244
@@ -1,299 +0,0 @@
1
- /**
2
- * QA Runner - 质量循环模块
3
- *
4
- * 自动运行验证检查,发现问题后可触发修复循环:
5
- * - 语法检查 (syntax)
6
- * - 类型检查 (typecheck)
7
- * - Lint 检查 (lint)
8
- * - 测试运行 (test)
9
- * - 构建验证 (build)
10
- */
11
- import * as fs from 'fs/promises';
12
- import * as path from 'path';
13
- import { exec } from 'child_process';
14
- import { promisify } from 'util';
15
- import { randomUUID } from 'crypto';
16
- const execAsync = promisify(exec);
17
- /**
18
- * 默认 QA 配置
19
- */
20
- const DEFAULT_CONFIG = {
21
- maxIterations: 5,
22
- checks: ['typecheck', 'lint', 'test'],
23
- autoFix: false,
24
- timeout: 60000, // 60 seconds
25
- };
26
- /**
27
- * 默认检查命令
28
- */
29
- const DEFAULT_COMMANDS = {
30
- syntax: 'npx tsc --noEmit --skipLibCheck',
31
- typecheck: 'npx tsc --noEmit',
32
- lint: 'npm run lint --silent 2>/dev/null || echo "lint script not found"',
33
- test: 'npm test --silent 2>/dev/null || echo "test script not found"',
34
- build: 'npm run build --silent',
35
- };
36
- /**
37
- * QARunner 主类
38
- */
39
- export class QARunner {
40
- cwd;
41
- config;
42
- runningQA = new Map();
43
- constructor(options) {
44
- this.cwd = options?.cwd || process.cwd();
45
- this.config = { ...DEFAULT_CONFIG, ...options?.config };
46
- }
47
- /**
48
- * 获取 QA 结果存储目录
49
- */
50
- getQADir() {
51
- return path.join(this.cwd, 'openspec', 'qa');
52
- }
53
- /**
54
- * 获取变更目录
55
- */
56
- getChangeDir(changeName) {
57
- const safeId = this.ensureSafeId(changeName);
58
- return path.join(this.cwd, 'openspec', 'changes', safeId);
59
- }
60
- /**
61
- * ID 安全校验
62
- */
63
- ensureSafeId(id) {
64
- const trimmed = id.trim();
65
- if (!trimmed || trimmed.includes('..') || trimmed.includes('/') || trimmed.includes('\\')) {
66
- throw new Error(`Invalid id: ${id}`);
67
- }
68
- return trimmed;
69
- }
70
- /**
71
- * 确保目录存在
72
- */
73
- async ensureDir(dirPath) {
74
- await fs.mkdir(dirPath, { recursive: true });
75
- }
76
- /**
77
- * 运行 QA 检查
78
- */
79
- async runQA(changeName, options) {
80
- const safeId = this.ensureSafeId(changeName);
81
- // 检查变更是否存在
82
- const changeDir = this.getChangeDir(safeId);
83
- try {
84
- await fs.access(changeDir);
85
- }
86
- catch {
87
- throw new Error(`Change not found: ${changeName}`);
88
- }
89
- // 初始化结果
90
- const result = {
91
- id: randomUUID().substring(0, 8),
92
- changeName: safeId,
93
- status: 'running',
94
- iteration: 1,
95
- maxIterations: this.config.maxIterations,
96
- checks: [],
97
- startedAt: new Date().toISOString(),
98
- summary: { total: 0, passed: 0, failed: 0, skipped: 0 },
99
- };
100
- // 用于控制终止
101
- const control = { aborted: false };
102
- this.runningQA.set(safeId, control);
103
- try {
104
- const checksToRun = options?.checks || this.config.checks;
105
- result.summary.total = checksToRun.length;
106
- for (const checkType of checksToRun) {
107
- if (control.aborted) {
108
- result.status = 'stopped';
109
- break;
110
- }
111
- const checkResult = await this.runCheck(checkType);
112
- result.checks.push(checkResult);
113
- // 更新统计
114
- if (checkResult.status === 'passed') {
115
- result.summary.passed++;
116
- }
117
- else if (checkResult.status === 'failed') {
118
- result.summary.failed++;
119
- }
120
- else if (checkResult.status === 'skipped') {
121
- result.summary.skipped++;
122
- }
123
- }
124
- // 确定最终状态
125
- if (result.status !== 'stopped') {
126
- result.status = result.summary.failed > 0 ? 'failed' : 'passed';
127
- }
128
- }
129
- finally {
130
- this.runningQA.delete(safeId);
131
- result.completedAt = new Date().toISOString();
132
- }
133
- // 保存结果
134
- await this.saveQAResult(result);
135
- return result;
136
- }
137
- /**
138
- * 运行单个检查
139
- */
140
- async runCheck(type) {
141
- const startTime = Date.now();
142
- const command = this.config.commands?.[type] || DEFAULT_COMMANDS[type];
143
- try {
144
- const { stdout, stderr } = await execAsync(command, {
145
- cwd: this.cwd,
146
- timeout: this.config.timeout,
147
- env: { ...process.env, CI: 'true' },
148
- });
149
- const duration = Date.now() - startTime;
150
- const output = stdout + (stderr ? '\n' + stderr : '');
151
- // 检查输出中是否包含错误标识
152
- const hasError = /error|failed|failure/i.test(output) && !/0 error/i.test(output);
153
- return {
154
- type,
155
- status: hasError ? 'failed' : 'passed',
156
- output: output.slice(0, 2000), // 限制输出长度
157
- duration,
158
- };
159
- }
160
- catch (error) {
161
- const duration = Date.now() - startTime;
162
- // 检查是否超时(Node.js exec 超时时设置 killed 或 signal)
163
- const execError = error;
164
- if (execError.killed || execError.signal === 'SIGTERM') {
165
- return {
166
- type,
167
- status: 'timeout',
168
- errors: [`Check timed out after ${this.config.timeout}ms`],
169
- duration,
170
- };
171
- }
172
- // 命令执行失败
173
- return {
174
- type,
175
- status: 'failed',
176
- output: execError.stdout?.slice(0, 1000),
177
- errors: [execError.stderr || execError.message || 'Unknown error'].slice(0, 5),
178
- duration,
179
- };
180
- }
181
- }
182
- /**
183
- * 获取 QA 状态
184
- */
185
- async getQAStatus(changeName) {
186
- const latest = await this.getLatestQAResult(changeName);
187
- return latest;
188
- }
189
- /**
190
- * 获取 QA 历史
191
- */
192
- async getQAHistory(changeName, limit = 10) {
193
- const qaDir = this.getQADir();
194
- const safeId = this.ensureSafeId(changeName);
195
- try {
196
- const files = await fs.readdir(qaDir);
197
- const matchingFiles = files
198
- .filter(f => f.startsWith(safeId + '_') && f.endsWith('.json'))
199
- .sort()
200
- .reverse();
201
- const results = [];
202
- for (const file of matchingFiles.slice(0, limit)) {
203
- try {
204
- const content = await fs.readFile(path.join(qaDir, file), 'utf-8');
205
- results.push(JSON.parse(content));
206
- }
207
- catch {
208
- // 跳过无效文件
209
- }
210
- }
211
- return results;
212
- }
213
- catch {
214
- return [];
215
- }
216
- }
217
- /**
218
- * 获取最新 QA 结果
219
- */
220
- async getLatestQAResult(changeName) {
221
- const history = await this.getQAHistory(changeName, 1);
222
- return history[0] || null;
223
- }
224
- /**
225
- * 停止正在运行的 QA
226
- */
227
- async stopQA(changeName) {
228
- const safeId = this.ensureSafeId(changeName);
229
- const control = this.runningQA.get(safeId);
230
- if (control) {
231
- control.aborted = true;
232
- return true;
233
- }
234
- return false;
235
- }
236
- /**
237
- * 检查 QA 是否正在运行
238
- */
239
- isRunning(changeName) {
240
- const safeId = this.ensureSafeId(changeName);
241
- return this.runningQA.has(safeId);
242
- }
243
- /**
244
- * 保存 QA 结果
245
- */
246
- async saveQAResult(result) {
247
- const qaDir = this.getQADir();
248
- await this.ensureDir(qaDir);
249
- const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
250
- const filePath = path.join(qaDir, `${result.changeName}_${timestamp}.json`);
251
- await fs.writeFile(filePath, JSON.stringify(result, null, 2), 'utf-8');
252
- }
253
- /**
254
- * 获取进度汇总(所有变更)
255
- */
256
- async getQASummary() {
257
- const changesDir = path.join(this.cwd, 'openspec', 'changes');
258
- const summary = {
259
- total: 0,
260
- passed: 0,
261
- failed: 0,
262
- running: 0,
263
- changes: [],
264
- };
265
- try {
266
- const changes = await fs.readdir(changesDir);
267
- for (const change of changes) {
268
- const stat = await fs.stat(path.join(changesDir, change));
269
- if (!stat.isDirectory())
270
- continue;
271
- summary.total++;
272
- const latest = await this.getLatestQAResult(change);
273
- if (this.isRunning(change)) {
274
- summary.running++;
275
- summary.changes.push({ name: change, status: 'running' });
276
- }
277
- else if (latest) {
278
- if (latest.status === 'passed')
279
- summary.passed++;
280
- else if (latest.status === 'failed')
281
- summary.failed++;
282
- summary.changes.push({
283
- name: change,
284
- status: latest.status,
285
- lastRun: latest.completedAt,
286
- });
287
- }
288
- else {
289
- summary.changes.push({ name: change, status: 'pending' });
290
- }
291
- }
292
- }
293
- catch {
294
- // 目录不存在时返回空结果
295
- }
296
- return summary;
297
- }
298
- }
299
- //# sourceMappingURL=qa-runner.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"qa-runner.js","sourceRoot":"","sources":["../../src/core/qa-runner.ts"],"names":[],"mappings":"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"}
@@ -1,12 +0,0 @@
1
- /**
2
- * QA Tools - MCP 工具注册
3
- *
4
- * 提供质量检查相关的 MCP 工具
5
- */
6
- import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
7
- import { QARunner } from '../../core/qa-runner.js';
8
- /**
9
- * 注册 QA 相关工具
10
- */
11
- export declare function registerQATools(server: McpServer, qaRunner: QARunner): void;
12
- //# sourceMappingURL=qa.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"qa.d.ts","sourceRoot":"","sources":["../../../src/server/tools/qa.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAEpE,OAAO,EAAE,QAAQ,EAAyB,MAAM,yBAAyB,CAAC;AAE1E;;GAEG;AACH,wBAAgB,eAAe,CAAC,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,QAAQ,GAAG,IAAI,CAmN3E"}
@@ -1,248 +0,0 @@
1
- /**
2
- * QA Tools - MCP 工具注册
3
- *
4
- * 提供质量检查相关的 MCP 工具
5
- */
6
- import { z } from 'zod';
7
- /**
8
- * 注册 QA 相关工具
9
- */
10
- export function registerQATools(server, qaRunner) {
11
- // 运行 QA 检查
12
- server.registerTool('openspec_run_qa', {
13
- description: '运行质量检查(语法、类型、lint、测试等)',
14
- inputSchema: {
15
- changeName: z.string().describe('变更 ID'),
16
- checks: z.array(z.enum(['syntax', 'typecheck', 'lint', 'test', 'build'])).optional()
17
- .describe('要运行的检查类型,默认: typecheck, lint, test'),
18
- },
19
- }, async ({ changeName, checks }) => {
20
- try {
21
- // 检查是否已在运行
22
- if (qaRunner.isRunning(changeName)) {
23
- return {
24
- content: [{
25
- type: 'text',
26
- text: `QA 正在运行中,请稍候或使用 openspec_stop_qa 停止`,
27
- }],
28
- };
29
- }
30
- const result = await qaRunner.runQA(changeName, {
31
- checks: checks,
32
- });
33
- const output = formatQAResult(result);
34
- return {
35
- content: [{ type: 'text', text: output }],
36
- };
37
- }
38
- catch (error) {
39
- return {
40
- content: [{
41
- type: 'text',
42
- text: `QA 运行失败: ${error instanceof Error ? error.message : 'Unknown error'}`,
43
- }],
44
- };
45
- }
46
- });
47
- // 获取 QA 状态
48
- server.registerTool('openspec_get_qa_status', {
49
- description: '获取变更的 QA 状态',
50
- inputSchema: {
51
- changeName: z.string().describe('变更 ID'),
52
- },
53
- }, async ({ changeName }) => {
54
- try {
55
- // 先检查是否正在运行
56
- if (qaRunner.isRunning(changeName)) {
57
- return {
58
- content: [{ type: 'text', text: `⏳ QA 正在运行中...` }],
59
- };
60
- }
61
- const result = await qaRunner.getQAStatus(changeName);
62
- if (!result) {
63
- return {
64
- content: [{ type: 'text', text: `没有找到 ${changeName} 的 QA 记录,请运行 openspec_run_qa` }],
65
- };
66
- }
67
- const output = formatQAResult(result);
68
- return {
69
- content: [{ type: 'text', text: output }],
70
- };
71
- }
72
- catch (error) {
73
- return {
74
- content: [{
75
- type: 'text',
76
- text: `获取状态失败: ${error instanceof Error ? error.message : 'Unknown error'}`,
77
- }],
78
- };
79
- }
80
- });
81
- // 获取 QA 历史
82
- server.registerTool('openspec_get_qa_history', {
83
- description: '获取变更的 QA 历史记录',
84
- inputSchema: {
85
- changeName: z.string().describe('变更 ID'),
86
- limit: z.number().optional().describe('返回记录数量限制,默认 5'),
87
- },
88
- }, async ({ changeName, limit = 5 }) => {
89
- try {
90
- const history = await qaRunner.getQAHistory(changeName, limit);
91
- if (history.length === 0) {
92
- return {
93
- content: [{ type: 'text', text: `没有找到 ${changeName} 的 QA 历史` }],
94
- };
95
- }
96
- const lines = [`# ${changeName} QA 历史\n`];
97
- for (const r of history) {
98
- const statusEmoji = getStatusEmoji(r.status);
99
- lines.push(`## ${statusEmoji} ${r.completedAt || r.startedAt}`);
100
- lines.push(`- 状态: ${r.status}`);
101
- lines.push(`- 通过: ${r.summary.passed}/${r.summary.total}`);
102
- lines.push('');
103
- }
104
- return {
105
- content: [{ type: 'text', text: lines.join('\n') }],
106
- };
107
- }
108
- catch (error) {
109
- return {
110
- content: [{
111
- type: 'text',
112
- text: `获取历史失败: ${error instanceof Error ? error.message : 'Unknown error'}`,
113
- }],
114
- };
115
- }
116
- });
117
- // 停止 QA
118
- server.registerTool('openspec_stop_qa', {
119
- description: '停止正在运行的 QA 检查',
120
- inputSchema: {
121
- changeName: z.string().describe('变更 ID'),
122
- },
123
- }, async ({ changeName }) => {
124
- try {
125
- const stopped = await qaRunner.stopQA(changeName);
126
- if (stopped) {
127
- return {
128
- content: [{ type: 'text', text: `✅ 已发送停止信号给 ${changeName} 的 QA` }],
129
- };
130
- }
131
- else {
132
- return {
133
- content: [{ type: 'text', text: `ℹ️ ${changeName} 没有正在运行的 QA` }],
134
- };
135
- }
136
- }
137
- catch (error) {
138
- return {
139
- content: [{
140
- type: 'text',
141
- text: `停止失败: ${error instanceof Error ? error.message : 'Unknown error'}`,
142
- }],
143
- };
144
- }
145
- });
146
- // 获取所有变更的 QA 汇总
147
- server.registerTool('openspec_get_qa_summary', {
148
- description: '获取所有变更的 QA 状态汇总',
149
- inputSchema: {},
150
- }, async () => {
151
- try {
152
- const summary = await qaRunner.getQASummary();
153
- const lines = [
154
- `# QA 汇总`,
155
- '',
156
- `| 指标 | 数量 |`,
157
- `|------|------|`,
158
- `| 总变更数 | ${summary.total} |`,
159
- `| ✅ 通过 | ${summary.passed} |`,
160
- `| ❌ 失败 | ${summary.failed} |`,
161
- `| ⏳ 运行中 | ${summary.running} |`,
162
- '',
163
- ];
164
- if (summary.changes.length > 0) {
165
- lines.push('## 变更详情');
166
- lines.push('');
167
- lines.push('| 变更 | 状态 | 最后运行 |');
168
- lines.push('|------|------|----------|');
169
- for (const change of summary.changes) {
170
- const emoji = getStatusEmoji(change.status);
171
- lines.push(`| ${change.name} | ${emoji} ${change.status} | ${change.lastRun || '-'} |`);
172
- }
173
- }
174
- return {
175
- content: [{ type: 'text', text: lines.join('\n') }],
176
- };
177
- }
178
- catch (error) {
179
- return {
180
- content: [{
181
- type: 'text',
182
- text: `获取汇总失败: ${error instanceof Error ? error.message : 'Unknown error'}`,
183
- }],
184
- };
185
- }
186
- });
187
- }
188
- /**
189
- * 格式化 QA 结果
190
- */
191
- function formatQAResult(result) {
192
- const lines = [];
193
- // 标题和状态
194
- const statusEmoji = getStatusEmoji(result.status);
195
- lines.push(`# ${statusEmoji} QA 报告: ${result.changeName}`);
196
- lines.push('');
197
- lines.push(`- **状态**: ${result.status.toUpperCase()}`);
198
- lines.push(`- **开始时间**: ${result.startedAt}`);
199
- if (result.completedAt) {
200
- lines.push(`- **完成时间**: ${result.completedAt}`);
201
- }
202
- lines.push('');
203
- // 统计
204
- lines.push('## 📊 统计');
205
- lines.push('');
206
- lines.push(`| 指标 | 数量 |`);
207
- lines.push(`|------|------|`);
208
- lines.push(`| ✅ 通过 | ${result.summary.passed} |`);
209
- lines.push(`| ❌ 失败 | ${result.summary.failed} |`);
210
- lines.push(`| ⏭️ 跳过 | ${result.summary.skipped} |`);
211
- lines.push(`| **总计** | **${result.summary.total}** |`);
212
- lines.push('');
213
- // 检查详情
214
- if (result.checks.length > 0) {
215
- lines.push('## 🔍 检查详情');
216
- lines.push('');
217
- for (const check of result.checks) {
218
- const checkEmoji = check.status === 'passed' ? '✅' :
219
- check.status === 'failed' ? '❌' :
220
- check.status === 'timeout' ? '⏱️' : '⏭️';
221
- lines.push(`### ${checkEmoji} ${check.type}`);
222
- lines.push('');
223
- lines.push(`- 状态: ${check.status}`);
224
- lines.push(`- 耗时: ${(check.duration / 1000).toFixed(2)}s`);
225
- if (check.errors && check.errors.length > 0) {
226
- lines.push('');
227
- lines.push('**错误:**');
228
- for (const err of check.errors.slice(0, 3)) {
229
- lines.push(`\`\`\`\n${err.slice(0, 500)}\n\`\`\``);
230
- }
231
- }
232
- lines.push('');
233
- }
234
- }
235
- return lines.join('\n');
236
- }
237
- function getStatusEmoji(status) {
238
- switch (status) {
239
- case 'passed': return '✅';
240
- case 'failed': return '❌';
241
- case 'running': return '⏳';
242
- case 'fixing': return '🔧';
243
- case 'timeout': return '⏱️';
244
- case 'stopped': return '🛑';
245
- default: return '⏸️';
246
- }
247
- }
248
- //# sourceMappingURL=qa.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"qa.js","sourceRoot":"","sources":["../../../src/server/tools/qa.ts"],"names":[],"mappings":"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"}