@wandoupeas/coding-forge 0.1.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.
- package/LICENSE +21 -0
- package/README.md +347 -0
- package/dist/agent/adapters/base.d.ts +9 -0
- package/dist/agent/adapters/base.d.ts.map +1 -0
- package/dist/agent/adapters/base.js +2 -0
- package/dist/agent/adapters/base.js.map +1 -0
- package/dist/agent/adapters/claude-code.d.ts +4 -0
- package/dist/agent/adapters/claude-code.d.ts.map +1 -0
- package/dist/agent/adapters/claude-code.js +25 -0
- package/dist/agent/adapters/claude-code.js.map +1 -0
- package/dist/agent/adapters/codex.d.ts +4 -0
- package/dist/agent/adapters/codex.d.ts.map +1 -0
- package/dist/agent/adapters/codex.js +27 -0
- package/dist/agent/adapters/codex.js.map +1 -0
- package/dist/agent/adapters/command-bridge.d.ts +67 -0
- package/dist/agent/adapters/command-bridge.d.ts.map +1 -0
- package/dist/agent/adapters/command-bridge.js +266 -0
- package/dist/agent/adapters/command-bridge.js.map +1 -0
- package/dist/agent/adapters/index.d.ts +7 -0
- package/dist/agent/adapters/index.d.ts.map +1 -0
- package/dist/agent/adapters/index.js +25 -0
- package/dist/agent/adapters/index.js.map +1 -0
- package/dist/agent/adapters/stub.d.ts +3 -0
- package/dist/agent/adapters/stub.d.ts.map +1 -0
- package/dist/agent/adapters/stub.js +66 -0
- package/dist/agent/adapters/stub.js.map +1 -0
- package/dist/agent/context.d.ts +36 -0
- package/dist/agent/context.d.ts.map +1 -0
- package/dist/agent/context.js +16 -0
- package/dist/agent/context.js.map +1 -0
- package/dist/agent/handlers/index.d.ts +6 -0
- package/dist/agent/handlers/index.d.ts.map +1 -0
- package/dist/agent/handlers/index.js +61 -0
- package/dist/agent/handlers/index.js.map +1 -0
- package/dist/agent/index.d.ts +18 -0
- package/dist/agent/index.d.ts.map +1 -0
- package/dist/agent/index.js +37 -0
- package/dist/agent/index.js.map +1 -0
- package/dist/agent/types.d.ts +21 -0
- package/dist/agent/types.d.ts.map +1 -0
- package/dist/agent/types.js +5 -0
- package/dist/agent/types.js.map +1 -0
- package/dist/cli/agent-briefing.d.ts +24 -0
- package/dist/cli/agent-briefing.d.ts.map +1 -0
- package/dist/cli/agent-briefing.js +133 -0
- package/dist/cli/agent-briefing.js.map +1 -0
- package/dist/cli/commands/checkpoint.d.ts +12 -0
- package/dist/cli/commands/checkpoint.d.ts.map +1 -0
- package/dist/cli/commands/checkpoint.js +86 -0
- package/dist/cli/commands/checkpoint.js.map +1 -0
- package/dist/cli/commands/dashboard.d.ts +14 -0
- package/dist/cli/commands/dashboard.d.ts.map +1 -0
- package/dist/cli/commands/dashboard.js +192 -0
- package/dist/cli/commands/dashboard.js.map +1 -0
- package/dist/cli/commands/doctor.d.ts +27 -0
- package/dist/cli/commands/doctor.d.ts.map +1 -0
- package/dist/cli/commands/doctor.js +425 -0
- package/dist/cli/commands/doctor.js.map +1 -0
- package/dist/cli/commands/init.d.ts +19 -0
- package/dist/cli/commands/init.d.ts.map +1 -0
- package/dist/cli/commands/init.js +303 -0
- package/dist/cli/commands/init.js.map +1 -0
- package/dist/cli/commands/knowledge.d.ts +6 -0
- package/dist/cli/commands/knowledge.d.ts.map +1 -0
- package/dist/cli/commands/knowledge.js +408 -0
- package/dist/cli/commands/knowledge.js.map +1 -0
- package/dist/cli/commands/logs.d.ts +50 -0
- package/dist/cli/commands/logs.d.ts.map +1 -0
- package/dist/cli/commands/logs.js +248 -0
- package/dist/cli/commands/logs.js.map +1 -0
- package/dist/cli/commands/mailbox.d.ts +12 -0
- package/dist/cli/commands/mailbox.d.ts.map +1 -0
- package/dist/cli/commands/mailbox.js +131 -0
- package/dist/cli/commands/mailbox.js.map +1 -0
- package/dist/cli/commands/onboard.d.ts +43 -0
- package/dist/cli/commands/onboard.d.ts.map +1 -0
- package/dist/cli/commands/onboard.js +313 -0
- package/dist/cli/commands/onboard.js.map +1 -0
- package/dist/cli/commands/plan.d.ts +16 -0
- package/dist/cli/commands/plan.d.ts.map +1 -0
- package/dist/cli/commands/plan.js +211 -0
- package/dist/cli/commands/plan.js.map +1 -0
- package/dist/cli/commands/resume.d.ts +102 -0
- package/dist/cli/commands/resume.d.ts.map +1 -0
- package/dist/cli/commands/resume.js +256 -0
- package/dist/cli/commands/resume.js.map +1 -0
- package/dist/cli/commands/review.d.ts +13 -0
- package/dist/cli/commands/review.d.ts.map +1 -0
- package/dist/cli/commands/review.js +91 -0
- package/dist/cli/commands/review.js.map +1 -0
- package/dist/cli/commands/run.d.ts +19 -0
- package/dist/cli/commands/run.d.ts.map +1 -0
- package/dist/cli/commands/run.js +161 -0
- package/dist/cli/commands/run.js.map +1 -0
- package/dist/cli/commands/superpowers.d.ts +21 -0
- package/dist/cli/commands/superpowers.d.ts.map +1 -0
- package/dist/cli/commands/superpowers.js +135 -0
- package/dist/cli/commands/superpowers.js.map +1 -0
- package/dist/cli/commands/ui.d.ts +14 -0
- package/dist/cli/commands/ui.d.ts.map +1 -0
- package/dist/cli/commands/ui.js +41 -0
- package/dist/cli/commands/ui.js.map +1 -0
- package/dist/cli/commands/verify.d.ts +11 -0
- package/dist/cli/commands/verify.d.ts.map +1 -0
- package/dist/cli/commands/verify.js +46 -0
- package/dist/cli/commands/verify.js.map +1 -0
- package/dist/cli/index.d.ts +9 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +69 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/cli/utils/logger.d.ts +18 -0
- package/dist/cli/utils/logger.d.ts.map +1 -0
- package/dist/cli/utils/logger.js +57 -0
- package/dist/cli/utils/logger.js.map +1 -0
- package/dist/core/checkpoint.d.ts +54 -0
- package/dist/core/checkpoint.d.ts.map +1 -0
- package/dist/core/checkpoint.js +227 -0
- package/dist/core/checkpoint.js.map +1 -0
- package/dist/core/context.d.ts +36 -0
- package/dist/core/context.d.ts.map +1 -0
- package/dist/core/context.js +71 -0
- package/dist/core/context.js.map +1 -0
- package/dist/core/deliverable.d.ts +34 -0
- package/dist/core/deliverable.d.ts.map +1 -0
- package/dist/core/deliverable.js +125 -0
- package/dist/core/deliverable.js.map +1 -0
- package/dist/core/harness-hints.d.ts +43 -0
- package/dist/core/harness-hints.d.ts.map +1 -0
- package/dist/core/harness-hints.js +162 -0
- package/dist/core/harness-hints.js.map +1 -0
- package/dist/core/logger.d.ts +104 -0
- package/dist/core/logger.d.ts.map +1 -0
- package/dist/core/logger.js +199 -0
- package/dist/core/logger.js.map +1 -0
- package/dist/core/mailbox.d.ts +69 -0
- package/dist/core/mailbox.d.ts.map +1 -0
- package/dist/core/mailbox.js +225 -0
- package/dist/core/mailbox.js.map +1 -0
- package/dist/core/planning.d.ts +110 -0
- package/dist/core/planning.d.ts.map +1 -0
- package/dist/core/planning.js +832 -0
- package/dist/core/planning.js.map +1 -0
- package/dist/core/runtime.d.ts +22 -0
- package/dist/core/runtime.d.ts.map +1 -0
- package/dist/core/runtime.js +364 -0
- package/dist/core/runtime.js.map +1 -0
- package/dist/core/session.d.ts +62 -0
- package/dist/core/session.d.ts.map +1 -0
- package/dist/core/session.js +266 -0
- package/dist/core/session.js.map +1 -0
- package/dist/core/superpowers-registry.d.ts +20 -0
- package/dist/core/superpowers-registry.d.ts.map +1 -0
- package/dist/core/superpowers-registry.js +180 -0
- package/dist/core/superpowers-registry.js.map +1 -0
- package/dist/core/superpowers-runs.d.ts +19 -0
- package/dist/core/superpowers-runs.d.ts.map +1 -0
- package/dist/core/superpowers-runs.js +231 -0
- package/dist/core/superpowers-runs.js.map +1 -0
- package/dist/core/task.d.ts +43 -0
- package/dist/core/task.d.ts.map +1 -0
- package/dist/core/task.js +232 -0
- package/dist/core/task.js.map +1 -0
- package/dist/core/threads.d.ts +23 -0
- package/dist/core/threads.d.ts.map +1 -0
- package/dist/core/threads.js +92 -0
- package/dist/core/threads.js.map +1 -0
- package/dist/core/worker.d.ts +48 -0
- package/dist/core/worker.d.ts.map +1 -0
- package/dist/core/worker.js +181 -0
- package/dist/core/worker.js.map +1 -0
- package/dist/core/workflow-context.d.ts +16 -0
- package/dist/core/workflow-context.d.ts.map +1 -0
- package/dist/core/workflow-context.js +102 -0
- package/dist/core/workflow-context.js.map +1 -0
- package/dist/core/workspace.d.ts +13 -0
- package/dist/core/workspace.d.ts.map +1 -0
- package/dist/core/workspace.js +208 -0
- package/dist/core/workspace.js.map +1 -0
- package/dist/testing/index.d.ts +21 -0
- package/dist/testing/index.d.ts.map +1 -0
- package/dist/testing/index.js +55 -0
- package/dist/testing/index.js.map +1 -0
- package/dist/types/index.d.ts +298 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +5 -0
- package/dist/types/index.js.map +1 -0
- package/dist/ui/assets/index-DLwnhrrc.css +1 -0
- package/dist/ui/assets/index-cDUGEuwi.js +234 -0
- package/dist/ui/http/handlers/project-artifacts.d.ts +3 -0
- package/dist/ui/http/handlers/project-artifacts.d.ts.map +1 -0
- package/dist/ui/http/handlers/project-artifacts.js +20 -0
- package/dist/ui/http/handlers/project-artifacts.js.map +1 -0
- package/dist/ui/http/handlers/project-overview.d.ts +3 -0
- package/dist/ui/http/handlers/project-overview.d.ts.map +1 -0
- package/dist/ui/http/handlers/project-overview.js +20 -0
- package/dist/ui/http/handlers/project-overview.js.map +1 -0
- package/dist/ui/http/handlers/project-recovery.d.ts +3 -0
- package/dist/ui/http/handlers/project-recovery.d.ts.map +1 -0
- package/dist/ui/http/handlers/project-recovery.js +20 -0
- package/dist/ui/http/handlers/project-recovery.js.map +1 -0
- package/dist/ui/http/handlers/project-runtime.d.ts +3 -0
- package/dist/ui/http/handlers/project-runtime.d.ts.map +1 -0
- package/dist/ui/http/handlers/project-runtime.js +20 -0
- package/dist/ui/http/handlers/project-runtime.js.map +1 -0
- package/dist/ui/http/handlers/project-tasks.d.ts +3 -0
- package/dist/ui/http/handlers/project-tasks.d.ts.map +1 -0
- package/dist/ui/http/handlers/project-tasks.js +20 -0
- package/dist/ui/http/handlers/project-tasks.js.map +1 -0
- package/dist/ui/http/handlers/projects.d.ts +23 -0
- package/dist/ui/http/handlers/projects.d.ts.map +1 -0
- package/dist/ui/http/handlers/projects.js +29 -0
- package/dist/ui/http/handlers/projects.js.map +1 -0
- package/dist/ui/http/router.d.ts +5 -0
- package/dist/ui/http/router.d.ts.map +1 -0
- package/dist/ui/http/router.js +81 -0
- package/dist/ui/http/router.js.map +1 -0
- package/dist/ui/http/server.d.ts +29 -0
- package/dist/ui/http/server.d.ts.map +1 -0
- package/dist/ui/http/server.js +143 -0
- package/dist/ui/http/server.js.map +1 -0
- package/dist/ui/index.html +13 -0
- package/dist/ui/project-registry.d.ts +9 -0
- package/dist/ui/project-registry.d.ts.map +1 -0
- package/dist/ui/project-registry.js +19 -0
- package/dist/ui/project-registry.js.map +1 -0
- package/dist/ui/project-scanner.d.ts +10 -0
- package/dist/ui/project-scanner.d.ts.map +1 -0
- package/dist/ui/project-scanner.js +135 -0
- package/dist/ui/project-scanner.js.map +1 -0
- package/dist/ui/read-models/artifacts.d.ts +58 -0
- package/dist/ui/read-models/artifacts.d.ts.map +1 -0
- package/dist/ui/read-models/artifacts.js +97 -0
- package/dist/ui/read-models/artifacts.js.map +1 -0
- package/dist/ui/read-models/overview.d.ts +38 -0
- package/dist/ui/read-models/overview.d.ts.map +1 -0
- package/dist/ui/read-models/overview.js +55 -0
- package/dist/ui/read-models/overview.js.map +1 -0
- package/dist/ui/read-models/recovery.d.ts +18 -0
- package/dist/ui/read-models/recovery.d.ts.map +1 -0
- package/dist/ui/read-models/recovery.js +26 -0
- package/dist/ui/read-models/recovery.js.map +1 -0
- package/dist/ui/read-models/runtime.d.ts +53 -0
- package/dist/ui/read-models/runtime.d.ts.map +1 -0
- package/dist/ui/read-models/runtime.js +86 -0
- package/dist/ui/read-models/runtime.js.map +1 -0
- package/dist/ui/read-models/tasks.d.ts +34 -0
- package/dist/ui/read-models/tasks.d.ts.map +1 -0
- package/dist/ui/read-models/tasks.js +46 -0
- package/dist/ui/read-models/tasks.js.map +1 -0
- package/dist/utils/config.d.ts +30 -0
- package/dist/utils/config.d.ts.map +1 -0
- package/dist/utils/config.js +152 -0
- package/dist/utils/config.js.map +1 -0
- package/dist/utils/file.d.ts +24 -0
- package/dist/utils/file.d.ts.map +1 -0
- package/dist/utils/file.js +42 -0
- package/dist/utils/file.js.map +1 -0
- package/dist/utils/lock.d.ts +7 -0
- package/dist/utils/lock.d.ts.map +1 -0
- package/dist/utils/lock.js +36 -0
- package/dist/utils/lock.js.map +1 -0
- package/package.json +78 -0
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CLI 日志工具
|
|
3
|
+
* 美化终端输出
|
|
4
|
+
*/
|
|
5
|
+
import chalk from 'chalk';
|
|
6
|
+
export const logger = {
|
|
7
|
+
info(message) {
|
|
8
|
+
console.log(chalk.blue('ℹ'), message);
|
|
9
|
+
},
|
|
10
|
+
success(message) {
|
|
11
|
+
console.log(chalk.green('✔'), message);
|
|
12
|
+
},
|
|
13
|
+
warning(message) {
|
|
14
|
+
console.log(chalk.yellow('⚠'), message);
|
|
15
|
+
},
|
|
16
|
+
error(message) {
|
|
17
|
+
console.log(chalk.red('✖'), message);
|
|
18
|
+
},
|
|
19
|
+
debug(message) {
|
|
20
|
+
if (process.env.DEBUG) {
|
|
21
|
+
console.log(chalk.gray('🐛'), chalk.gray(message));
|
|
22
|
+
}
|
|
23
|
+
},
|
|
24
|
+
h1(title) {
|
|
25
|
+
console.log('\n' + chalk.bold.green(title));
|
|
26
|
+
console.log(chalk.green('═'.repeat(title.length * 2)));
|
|
27
|
+
},
|
|
28
|
+
h2(title) {
|
|
29
|
+
console.log('\n' + chalk.bold.cyan(title));
|
|
30
|
+
},
|
|
31
|
+
list(items, bullet = ' •') {
|
|
32
|
+
for (const item of items) {
|
|
33
|
+
console.log(`${bullet} ${item}`);
|
|
34
|
+
}
|
|
35
|
+
},
|
|
36
|
+
table(rows) {
|
|
37
|
+
// 简单表格实现
|
|
38
|
+
if (rows.length === 0)
|
|
39
|
+
return;
|
|
40
|
+
const colWidths = rows[0].map((_, i) => Math.max(...rows.map(row => (row[i] || '').length)));
|
|
41
|
+
for (const row of rows) {
|
|
42
|
+
const line = row.map((cell, i) => (cell || '').padEnd(colWidths[i] + 2)).join('');
|
|
43
|
+
console.log(' ' + line);
|
|
44
|
+
}
|
|
45
|
+
},
|
|
46
|
+
// 进度条
|
|
47
|
+
progress(current, total, width = 30) {
|
|
48
|
+
const ratio = total > 0 ? current / total : 0;
|
|
49
|
+
const filled = Math.round(width * ratio);
|
|
50
|
+
const empty = width - filled;
|
|
51
|
+
const bar = chalk.green('█'.repeat(filled)) + chalk.gray('░'.repeat(empty));
|
|
52
|
+
const percent = Math.round(ratio * 100);
|
|
53
|
+
return `[${bar}] ${percent}% (${current}/${total})`;
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
export default logger;
|
|
57
|
+
//# sourceMappingURL=logger.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../../src/cli/utils/logger.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 检查点系统
|
|
3
|
+
* 关键节点保存快照,支持回滚
|
|
4
|
+
*/
|
|
5
|
+
import { Task, Phase } from '../types/index.js';
|
|
6
|
+
import type { Deliverable } from './deliverable.js';
|
|
7
|
+
export interface CheckpointDeliverableSnapshot extends Deliverable {
|
|
8
|
+
snapshotPath: string;
|
|
9
|
+
}
|
|
10
|
+
export interface Checkpoint {
|
|
11
|
+
id: string;
|
|
12
|
+
name: string;
|
|
13
|
+
createdAt: string;
|
|
14
|
+
description?: string;
|
|
15
|
+
tasks: Task[];
|
|
16
|
+
phases: Phase[];
|
|
17
|
+
gitCommit?: string;
|
|
18
|
+
deliverables: CheckpointDeliverableSnapshot[];
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* 创建检查点
|
|
22
|
+
*/
|
|
23
|
+
export declare function createCheckpoint(name: string, tasks: Task[], phases: Phase[], options?: {
|
|
24
|
+
description?: string;
|
|
25
|
+
gitCommit?: string;
|
|
26
|
+
basePath?: string;
|
|
27
|
+
}): Promise<Checkpoint>;
|
|
28
|
+
/**
|
|
29
|
+
* 列出所有检查点
|
|
30
|
+
*/
|
|
31
|
+
export declare function listCheckpoints(basePath?: string): Promise<Checkpoint[]>;
|
|
32
|
+
/**
|
|
33
|
+
* 加载检查点
|
|
34
|
+
*/
|
|
35
|
+
export declare function loadCheckpoint(checkpointId: string, basePath?: string): Promise<Checkpoint | null>;
|
|
36
|
+
/**
|
|
37
|
+
* 回滚到检查点
|
|
38
|
+
*/
|
|
39
|
+
export declare function rollbackToCheckpoint(checkpointId: string, basePath?: string, options?: {
|
|
40
|
+
restoreDeliverables?: boolean;
|
|
41
|
+
}): Promise<{
|
|
42
|
+
tasks: Task[];
|
|
43
|
+
phases: Phase[];
|
|
44
|
+
deliverables?: Deliverable[];
|
|
45
|
+
} | null>;
|
|
46
|
+
/**
|
|
47
|
+
* 删除检查点
|
|
48
|
+
*/
|
|
49
|
+
export declare function deleteCheckpoint(checkpointId: string, basePath?: string): Promise<boolean>;
|
|
50
|
+
/**
|
|
51
|
+
* 自动在关键节点创建检查点
|
|
52
|
+
*/
|
|
53
|
+
export declare function autoCheckpoint(tasks: Task[], phases: Phase[], basePath?: string): Promise<void>;
|
|
54
|
+
//# sourceMappingURL=checkpoint.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"checkpoint.d.ts","sourceRoot":"","sources":["../../src/core/checkpoint.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAKH,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,mBAAmB,CAAC;AAChD,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,kBAAkB,CAAC;AAQpD,MAAM,WAAW,6BAA8B,SAAQ,WAAW;IAChE,YAAY,EAAE,MAAM,CAAC;CACtB;AAED,MAAM,WAAW,UAAU;IACzB,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,KAAK,EAAE,IAAI,EAAE,CAAC;IACd,MAAM,EAAE,KAAK,EAAE,CAAC;IAChB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,YAAY,EAAE,6BAA6B,EAAE,CAAC;CAC/C;AAED;;GAEG;AACH,wBAAsB,gBAAgB,CACpC,IAAI,EAAE,MAAM,EACZ,KAAK,EAAE,IAAI,EAAE,EACb,MAAM,EAAE,KAAK,EAAE,EACf,OAAO,CAAC,EAAE;IACR,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB,GACA,OAAO,CAAC,UAAU,CAAC,CAuBrB;AAsHD;;GAEG;AACH,wBAAsB,eAAe,CACnC,QAAQ,GAAE,MAAsB,GAC/B,OAAO,CAAC,UAAU,EAAE,CAAC,CA0BvB;AAED;;GAEG;AACH,wBAAsB,cAAc,CAClC,YAAY,EAAE,MAAM,EACpB,QAAQ,GAAE,MAAsB,GAC/B,OAAO,CAAC,UAAU,GAAG,IAAI,CAAC,CAa5B;AAED;;GAEG;AACH,wBAAsB,oBAAoB,CACxC,YAAY,EAAE,MAAM,EACpB,QAAQ,GAAE,MAAsB,EAChC,OAAO,CAAC,EAAE;IACR,mBAAmB,CAAC,EAAE,OAAO,CAAC;CAC/B,GACA,OAAO,CAAC;IACT,KAAK,EAAE,IAAI,EAAE,CAAC;IACd,MAAM,EAAE,KAAK,EAAE,CAAC;IAChB,YAAY,CAAC,EAAE,WAAW,EAAE,CAAC;CAC9B,GAAG,IAAI,CAAC,CA8BR;AAED;;GAEG;AACH,wBAAsB,gBAAgB,CACpC,YAAY,EAAE,MAAM,EACpB,QAAQ,GAAE,MAAsB,GAC/B,OAAO,CAAC,OAAO,CAAC,CAclB;AAED;;GAEG;AACH,wBAAsB,cAAc,CAClC,KAAK,EAAE,IAAI,EAAE,EACb,MAAM,EAAE,KAAK,EAAE,EACf,QAAQ,GAAE,MAAsB,GAC/B,OAAO,CAAC,IAAI,CAAC,CAwBf"}
|
|
@@ -0,0 +1,227 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 检查点系统
|
|
3
|
+
* 关键节点保存快照,支持回滚
|
|
4
|
+
*/
|
|
5
|
+
import { writeFile, readFile, copyFile } from 'fs/promises';
|
|
6
|
+
import { dirname, join } from 'path';
|
|
7
|
+
import { existsSync } from 'fs';
|
|
8
|
+
import { ensureDir, readJson, writeJson } from '../utils/file.js';
|
|
9
|
+
const CHECKPOINTS_DIR = '.webforge/checkpoints';
|
|
10
|
+
const DELIVERABLES_INDEX = '.webforge/deliverables/index.json';
|
|
11
|
+
const TASKS_FILE = '.webforge/tasks.json';
|
|
12
|
+
const PHASES_FILE = '.webforge/phases.json';
|
|
13
|
+
/**
|
|
14
|
+
* 创建检查点
|
|
15
|
+
*/
|
|
16
|
+
export async function createCheckpoint(name, tasks, phases, options) {
|
|
17
|
+
const basePath = options?.basePath ?? process.cwd();
|
|
18
|
+
const checkpointsDir = join(basePath, CHECKPOINTS_DIR);
|
|
19
|
+
await ensureDir(checkpointsDir);
|
|
20
|
+
const id = createCheckpointId();
|
|
21
|
+
const deliverables = await snapshotDeliverables(basePath, id);
|
|
22
|
+
const checkpoint = {
|
|
23
|
+
id,
|
|
24
|
+
name,
|
|
25
|
+
createdAt: new Date().toISOString(),
|
|
26
|
+
description: options?.description,
|
|
27
|
+
tasks: JSON.parse(JSON.stringify(tasks)), // 深拷贝
|
|
28
|
+
phases: JSON.parse(JSON.stringify(phases)),
|
|
29
|
+
gitCommit: options?.gitCommit,
|
|
30
|
+
deliverables
|
|
31
|
+
};
|
|
32
|
+
// 保存检查点
|
|
33
|
+
const checkpointPath = join(checkpointsDir, `${checkpoint.id}.json`);
|
|
34
|
+
await writeFile(checkpointPath, JSON.stringify(checkpoint, null, 2));
|
|
35
|
+
return checkpoint;
|
|
36
|
+
}
|
|
37
|
+
function createCheckpointId() {
|
|
38
|
+
// Tests and rollback flows may create multiple checkpoints under a frozen clock.
|
|
39
|
+
const suffix = Math.random().toString(36).slice(2, 8);
|
|
40
|
+
return `cp-${Date.now()}-${suffix}`;
|
|
41
|
+
}
|
|
42
|
+
async function snapshotDeliverables(basePath, checkpointId) {
|
|
43
|
+
const deliverables = await loadCheckpointDeliverables(basePath);
|
|
44
|
+
if (deliverables.length === 0) {
|
|
45
|
+
return [];
|
|
46
|
+
}
|
|
47
|
+
const snapshots = [];
|
|
48
|
+
for (const deliverable of deliverables) {
|
|
49
|
+
const snapshotRelativePath = buildCheckpointDeliverablePath(checkpointId, deliverable.path);
|
|
50
|
+
const snapshotPath = join(basePath, snapshotRelativePath);
|
|
51
|
+
await ensureDir(dirname(snapshotPath));
|
|
52
|
+
const sourcePath = join(basePath, deliverable.path);
|
|
53
|
+
if (existsSync(sourcePath)) {
|
|
54
|
+
await copyFile(sourcePath, snapshotPath);
|
|
55
|
+
}
|
|
56
|
+
else if (typeof deliverable.content === 'string') {
|
|
57
|
+
await writeFile(snapshotPath, deliverable.content, 'utf-8');
|
|
58
|
+
}
|
|
59
|
+
else {
|
|
60
|
+
continue;
|
|
61
|
+
}
|
|
62
|
+
snapshots.push({
|
|
63
|
+
...deliverable,
|
|
64
|
+
snapshotPath: toPosixPath(snapshotRelativePath)
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
return snapshots;
|
|
68
|
+
}
|
|
69
|
+
async function loadCheckpointDeliverables(basePath) {
|
|
70
|
+
const indexPath = join(basePath, DELIVERABLES_INDEX);
|
|
71
|
+
if (!existsSync(indexPath)) {
|
|
72
|
+
return [];
|
|
73
|
+
}
|
|
74
|
+
try {
|
|
75
|
+
const data = await readJson(indexPath);
|
|
76
|
+
if (Array.isArray(data)) {
|
|
77
|
+
return data;
|
|
78
|
+
}
|
|
79
|
+
if (Array.isArray(data.items)) {
|
|
80
|
+
return data.items;
|
|
81
|
+
}
|
|
82
|
+
return [];
|
|
83
|
+
}
|
|
84
|
+
catch {
|
|
85
|
+
return [];
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
function buildCheckpointDeliverablePath(checkpointId, deliverablePath) {
|
|
89
|
+
const trimmedPath = deliverablePath.replace(/^\.webforge[\\/]+deliverables[\\/]+/, '');
|
|
90
|
+
return join(CHECKPOINTS_DIR, checkpointId, 'deliverables', trimmedPath || deliverablePath.split(/[\\/]/).pop() || 'deliverable');
|
|
91
|
+
}
|
|
92
|
+
function toPosixPath(path) {
|
|
93
|
+
return path.replace(/\\/g, '/');
|
|
94
|
+
}
|
|
95
|
+
async function restoreCheckpointDeliverables(checkpoint, basePath) {
|
|
96
|
+
const restored = [];
|
|
97
|
+
for (const snapshot of checkpoint.deliverables) {
|
|
98
|
+
const targetPath = join(basePath, snapshot.path);
|
|
99
|
+
const snapshotSourcePath = join(basePath, snapshot.snapshotPath);
|
|
100
|
+
await ensureDir(dirname(targetPath));
|
|
101
|
+
if (existsSync(snapshotSourcePath)) {
|
|
102
|
+
await copyFile(snapshotSourcePath, targetPath);
|
|
103
|
+
}
|
|
104
|
+
else if (typeof snapshot.content === 'string') {
|
|
105
|
+
await writeFile(targetPath, snapshot.content, 'utf-8');
|
|
106
|
+
}
|
|
107
|
+
else {
|
|
108
|
+
continue;
|
|
109
|
+
}
|
|
110
|
+
const { snapshotPath, ...deliverable } = snapshot;
|
|
111
|
+
restored.push(deliverable);
|
|
112
|
+
}
|
|
113
|
+
await writeJson(join(basePath, DELIVERABLES_INDEX), { items: restored });
|
|
114
|
+
return restored;
|
|
115
|
+
}
|
|
116
|
+
async function restoreCheckpointGraph(checkpoint, basePath) {
|
|
117
|
+
await writeJson(join(basePath, TASKS_FILE), {
|
|
118
|
+
tasks: checkpoint.tasks
|
|
119
|
+
});
|
|
120
|
+
await writeJson(join(basePath, PHASES_FILE), {
|
|
121
|
+
phases: checkpoint.phases
|
|
122
|
+
});
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* 列出所有检查点
|
|
126
|
+
*/
|
|
127
|
+
export async function listCheckpoints(basePath = process.cwd()) {
|
|
128
|
+
const checkpointsDir = join(basePath, CHECKPOINTS_DIR);
|
|
129
|
+
if (!existsSync(checkpointsDir)) {
|
|
130
|
+
return [];
|
|
131
|
+
}
|
|
132
|
+
const { readdir } = await import('fs/promises');
|
|
133
|
+
const files = await readdir(checkpointsDir);
|
|
134
|
+
const checkpoints = [];
|
|
135
|
+
for (const file of files) {
|
|
136
|
+
if (file.endsWith('.json')) {
|
|
137
|
+
try {
|
|
138
|
+
const content = await readFile(join(checkpointsDir, file), 'utf-8');
|
|
139
|
+
checkpoints.push(JSON.parse(content));
|
|
140
|
+
}
|
|
141
|
+
catch {
|
|
142
|
+
// 忽略损坏的检查点
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
// 按时间倒序
|
|
147
|
+
return checkpoints.sort((a, b) => new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime());
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* 加载检查点
|
|
151
|
+
*/
|
|
152
|
+
export async function loadCheckpoint(checkpointId, basePath = process.cwd()) {
|
|
153
|
+
const checkpointPath = join(basePath, CHECKPOINTS_DIR, `${checkpointId}.json`);
|
|
154
|
+
if (!existsSync(checkpointPath)) {
|
|
155
|
+
return null;
|
|
156
|
+
}
|
|
157
|
+
try {
|
|
158
|
+
const content = await readFile(checkpointPath, 'utf-8');
|
|
159
|
+
return JSON.parse(content);
|
|
160
|
+
}
|
|
161
|
+
catch {
|
|
162
|
+
return null;
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
* 回滚到检查点
|
|
167
|
+
*/
|
|
168
|
+
export async function rollbackToCheckpoint(checkpointId, basePath = process.cwd(), options) {
|
|
169
|
+
const checkpoint = await loadCheckpoint(checkpointId, basePath);
|
|
170
|
+
if (!checkpoint) {
|
|
171
|
+
return null;
|
|
172
|
+
}
|
|
173
|
+
// 创建回滚记录
|
|
174
|
+
const { createCheckpoint } = await import('./checkpoint.js');
|
|
175
|
+
await createCheckpoint(`Rollback from ${checkpoint.name}`, checkpoint.tasks, checkpoint.phases, {
|
|
176
|
+
description: `Auto-created before rollback to ${checkpointId}`,
|
|
177
|
+
basePath
|
|
178
|
+
});
|
|
179
|
+
await restoreCheckpointGraph(checkpoint, basePath);
|
|
180
|
+
const restoredDeliverables = options?.restoreDeliverables
|
|
181
|
+
? await restoreCheckpointDeliverables(checkpoint, basePath)
|
|
182
|
+
: undefined;
|
|
183
|
+
return {
|
|
184
|
+
tasks: checkpoint.tasks,
|
|
185
|
+
phases: checkpoint.phases,
|
|
186
|
+
deliverables: restoredDeliverables
|
|
187
|
+
};
|
|
188
|
+
}
|
|
189
|
+
/**
|
|
190
|
+
* 删除检查点
|
|
191
|
+
*/
|
|
192
|
+
export async function deleteCheckpoint(checkpointId, basePath = process.cwd()) {
|
|
193
|
+
const checkpointPath = join(basePath, CHECKPOINTS_DIR, `${checkpointId}.json`);
|
|
194
|
+
if (!existsSync(checkpointPath)) {
|
|
195
|
+
return false;
|
|
196
|
+
}
|
|
197
|
+
try {
|
|
198
|
+
const { unlink } = await import('fs/promises');
|
|
199
|
+
await unlink(checkpointPath);
|
|
200
|
+
return true;
|
|
201
|
+
}
|
|
202
|
+
catch {
|
|
203
|
+
return false;
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
/**
|
|
207
|
+
* 自动在关键节点创建检查点
|
|
208
|
+
*/
|
|
209
|
+
export async function autoCheckpoint(tasks, phases, basePath = process.cwd()) {
|
|
210
|
+
// 检查是否应该创建检查点
|
|
211
|
+
const completedCount = tasks.filter(t => t.status === 'completed').length;
|
|
212
|
+
const totalCount = tasks.length;
|
|
213
|
+
// 每完成 25% 创建一个检查点
|
|
214
|
+
const milestones = [0.25, 0.5, 0.75];
|
|
215
|
+
for (const milestone of milestones) {
|
|
216
|
+
const targetCount = Math.floor(totalCount * milestone);
|
|
217
|
+
if (completedCount === targetCount && completedCount > 0) {
|
|
218
|
+
const phase = phases.find(p => p.status === 'in_progress');
|
|
219
|
+
await createCheckpoint(`Phase ${phase?.name || 'Milestone'} ${milestone * 100}%`, tasks, phases, {
|
|
220
|
+
description: `Auto-checkpoint at ${completedCount}/${totalCount} tasks`,
|
|
221
|
+
basePath
|
|
222
|
+
});
|
|
223
|
+
break;
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
//# sourceMappingURL=checkpoint.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"checkpoint.js","sourceRoot":"","sources":["../../src/core/checkpoint.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { type ObservationSnapshot, type PermissionHints, type SuperpowersHints } from './harness-hints.js';
|
|
2
|
+
import type { Phase, Task, WorkspaceDeliverableIndexEntry, WorkspaceKnowledgeIndexEntry, WorkspacePaths, WorkspaceRuntime, WorkspaceSessionIndexEntry, WorkspaceWorkflowContext } from '../types/index.js';
|
|
3
|
+
export interface ExecutionContext {
|
|
4
|
+
workspace: {
|
|
5
|
+
basePath: string;
|
|
6
|
+
paths: WorkspacePaths;
|
|
7
|
+
};
|
|
8
|
+
runtime: WorkspaceRuntime;
|
|
9
|
+
task: Task;
|
|
10
|
+
phase: Phase;
|
|
11
|
+
knowledge: {
|
|
12
|
+
items: WorkspaceKnowledgeIndexEntry[];
|
|
13
|
+
requirements: WorkspaceKnowledgeIndexEntry[];
|
|
14
|
+
design: WorkspaceKnowledgeIndexEntry[];
|
|
15
|
+
decisions: WorkspaceKnowledgeIndexEntry[];
|
|
16
|
+
parsed: WorkspaceKnowledgeIndexEntry[];
|
|
17
|
+
notes: WorkspaceKnowledgeIndexEntry[];
|
|
18
|
+
};
|
|
19
|
+
deliverables: {
|
|
20
|
+
items: WorkspaceDeliverableIndexEntry[];
|
|
21
|
+
task: WorkspaceDeliverableIndexEntry[];
|
|
22
|
+
};
|
|
23
|
+
sessions: {
|
|
24
|
+
items: WorkspaceSessionIndexEntry[];
|
|
25
|
+
active: WorkspaceSessionIndexEntry | null;
|
|
26
|
+
related: WorkspaceSessionIndexEntry[];
|
|
27
|
+
};
|
|
28
|
+
superpowers: SuperpowersHints;
|
|
29
|
+
workflowContext: WorkspaceWorkflowContext | null;
|
|
30
|
+
permissions: PermissionHints;
|
|
31
|
+
observation: ObservationSnapshot;
|
|
32
|
+
}
|
|
33
|
+
export declare function buildExecutionContext(basePath: string, taskId: string, options?: {
|
|
34
|
+
workerId?: string;
|
|
35
|
+
}): Promise<ExecutionContext>;
|
|
36
|
+
//# sourceMappingURL=context.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context.d.ts","sourceRoot":"","sources":["../../src/core/context.ts"],"names":[],"mappings":"AACA,OAAO,EAIL,KAAK,mBAAmB,EACxB,KAAK,eAAe,EACpB,KAAK,gBAAgB,EACtB,MAAM,oBAAoB,CAAC;AAQ5B,OAAO,KAAK,EACV,KAAK,EACL,IAAI,EACJ,8BAA8B,EAC9B,4BAA4B,EAC5B,cAAc,EACd,gBAAgB,EAChB,0BAA0B,EAC1B,wBAAwB,EACzB,MAAM,mBAAmB,CAAC;AAE3B,MAAM,WAAW,gBAAgB;IAC/B,SAAS,EAAE;QACT,QAAQ,EAAE,MAAM,CAAC;QACjB,KAAK,EAAE,cAAc,CAAC;KACvB,CAAC;IACF,OAAO,EAAE,gBAAgB,CAAC;IAC1B,IAAI,EAAE,IAAI,CAAC;IACX,KAAK,EAAE,KAAK,CAAC;IACb,SAAS,EAAE;QACT,KAAK,EAAE,4BAA4B,EAAE,CAAC;QACtC,YAAY,EAAE,4BAA4B,EAAE,CAAC;QAC7C,MAAM,EAAE,4BAA4B,EAAE,CAAC;QACvC,SAAS,EAAE,4BAA4B,EAAE,CAAC;QAC1C,MAAM,EAAE,4BAA4B,EAAE,CAAC;QACvC,KAAK,EAAE,4BAA4B,EAAE,CAAC;KACvC,CAAC;IACF,YAAY,EAAE;QACZ,KAAK,EAAE,8BAA8B,EAAE,CAAC;QACxC,IAAI,EAAE,8BAA8B,EAAE,CAAC;KACxC,CAAC;IACF,QAAQ,EAAE;QACR,KAAK,EAAE,0BAA0B,EAAE,CAAC;QACpC,MAAM,EAAE,0BAA0B,GAAG,IAAI,CAAC;QAC1C,OAAO,EAAE,0BAA0B,EAAE,CAAC;KACvC,CAAC;IACF,WAAW,EAAE,gBAAgB,CAAC;IAC9B,eAAe,EAAE,wBAAwB,GAAG,IAAI,CAAC;IACjD,WAAW,EAAE,eAAe,CAAC;IAC7B,WAAW,EAAE,mBAAmB,CAAC;CAClC;AAED,wBAAsB,qBAAqB,CACzC,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,MAAM,EACd,OAAO,GAAE;IAAE,QAAQ,CAAC,EAAE,MAAM,CAAA;CAAO,GAClC,OAAO,CAAC,gBAAgB,CAAC,CAqE3B"}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
import { resolve } from 'path';
|
|
2
|
+
import { buildObservationSnapshot, loadPermissionHints, loadSuperpowersHints } from './harness-hints.js';
|
|
3
|
+
import { getRuntimeWorkflowContext, getSessionWorkflowContext, getTaskWorkflowContext, pickWorkflowContext } from './workflow-context.js';
|
|
4
|
+
import { loadWorkspaceState } from './workspace.js';
|
|
5
|
+
export async function buildExecutionContext(basePath, taskId, options = {}) {
|
|
6
|
+
const workspace = await loadWorkspaceState(basePath);
|
|
7
|
+
const task = workspace.tasks.tasks.find((item) => item.id === taskId);
|
|
8
|
+
if (!task) {
|
|
9
|
+
throw new Error(`Task not found: ${taskId}`);
|
|
10
|
+
}
|
|
11
|
+
const phase = workspace.phases.phases.find((item) => item.id === task.phase);
|
|
12
|
+
if (!phase) {
|
|
13
|
+
throw new Error(`Phase not found for task ${taskId}: ${task.phase}`);
|
|
14
|
+
}
|
|
15
|
+
const knowledgeItems = workspace.indexes.knowledge;
|
|
16
|
+
const deliverables = workspace.indexes.deliverables;
|
|
17
|
+
const sessions = workspace.indexes.sessions;
|
|
18
|
+
const relatedSessions = sessions.filter((session) => session.currentTask === taskId ||
|
|
19
|
+
(!session.currentTask && session.currentPhase === phase.id));
|
|
20
|
+
const activeSession = pickMostRecentSession(relatedSessions.filter((session) => session.status === 'active')) ||
|
|
21
|
+
pickMostRecentSession(relatedSessions);
|
|
22
|
+
const workflowContext = pickWorkflowContext(getTaskWorkflowContext(task), getSessionWorkflowContext(activeSession), getRuntimeWorkflowContext(workspace.runtime));
|
|
23
|
+
const [superpowers, permissions, observation] = await Promise.all([
|
|
24
|
+
loadSuperpowersHints(basePath, task),
|
|
25
|
+
loadPermissionHints(basePath),
|
|
26
|
+
buildObservationSnapshot(basePath, workspace, {
|
|
27
|
+
taskId,
|
|
28
|
+
relatedSessionCount: relatedSessions.length,
|
|
29
|
+
workerId: options.workerId
|
|
30
|
+
})
|
|
31
|
+
]);
|
|
32
|
+
return {
|
|
33
|
+
workspace: {
|
|
34
|
+
basePath: resolve(basePath),
|
|
35
|
+
paths: workspace.paths
|
|
36
|
+
},
|
|
37
|
+
runtime: workspace.runtime,
|
|
38
|
+
task,
|
|
39
|
+
phase,
|
|
40
|
+
knowledge: {
|
|
41
|
+
items: knowledgeItems,
|
|
42
|
+
requirements: knowledgeItems.filter((item) => item.type === 'requirement'),
|
|
43
|
+
design: knowledgeItems.filter((item) => item.type === 'design'),
|
|
44
|
+
decisions: knowledgeItems.filter((item) => item.type === 'decision'),
|
|
45
|
+
parsed: knowledgeItems.filter((item) => item.type === 'parsed'),
|
|
46
|
+
notes: knowledgeItems.filter((item) => item.type === 'note')
|
|
47
|
+
},
|
|
48
|
+
deliverables: {
|
|
49
|
+
items: deliverables,
|
|
50
|
+
task: deliverables.filter((item) => item.taskId === taskId)
|
|
51
|
+
},
|
|
52
|
+
sessions: {
|
|
53
|
+
items: sessions,
|
|
54
|
+
active: activeSession,
|
|
55
|
+
related: relatedSessions
|
|
56
|
+
},
|
|
57
|
+
superpowers,
|
|
58
|
+
workflowContext,
|
|
59
|
+
permissions,
|
|
60
|
+
observation
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
function pickMostRecentSession(sessions) {
|
|
64
|
+
if (sessions.length === 0) {
|
|
65
|
+
return null;
|
|
66
|
+
}
|
|
67
|
+
return sessions.reduce((latest, current) => new Date(current.lastActive).getTime() > new Date(latest.lastActive).getTime()
|
|
68
|
+
? current
|
|
69
|
+
: latest);
|
|
70
|
+
}
|
|
71
|
+
//# sourceMappingURL=context.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context.js","sourceRoot":"","sources":["../../src/core/context.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 交付物管理模块
|
|
3
|
+
* 管理 Worker 产生的可交付物(文档、代码、设计等)
|
|
4
|
+
*/
|
|
5
|
+
export type DeliverableType = 'document' | 'code' | 'test' | 'config' | 'design' | 'review';
|
|
6
|
+
export interface Deliverable {
|
|
7
|
+
id: string;
|
|
8
|
+
taskId: string;
|
|
9
|
+
type: DeliverableType;
|
|
10
|
+
title: string;
|
|
11
|
+
content: string;
|
|
12
|
+
path: string;
|
|
13
|
+
createdBy: string;
|
|
14
|
+
createdAt: string;
|
|
15
|
+
status: 'draft' | 'pending_review' | 'approved' | 'rejected';
|
|
16
|
+
reviewComment?: string;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* 创建交付物
|
|
20
|
+
*/
|
|
21
|
+
export declare function createDeliverable(taskId: string, type: DeliverableType, title: string, content: string, createdBy: string, basePath?: string): Promise<Deliverable>;
|
|
22
|
+
/**
|
|
23
|
+
* 获取交付物内容
|
|
24
|
+
*/
|
|
25
|
+
export declare function getDeliverableContent(deliverableId: string, basePath?: string): Promise<string | null>;
|
|
26
|
+
/**
|
|
27
|
+
* 审核交付物
|
|
28
|
+
*/
|
|
29
|
+
export declare function reviewDeliverable(deliverableId: string, approved: boolean, comment?: string, basePath?: string): Promise<void>;
|
|
30
|
+
/**
|
|
31
|
+
* 列出任务的所有交付物
|
|
32
|
+
*/
|
|
33
|
+
export declare function listDeliverables(taskId?: string, basePath?: string): Promise<Deliverable[]>;
|
|
34
|
+
//# sourceMappingURL=deliverable.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"deliverable.d.ts","sourceRoot":"","sources":["../../src/core/deliverable.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAOH,MAAM,MAAM,eAAe,GACvB,UAAU,GACV,MAAM,GACN,MAAM,GACN,QAAQ,GACR,QAAQ,GACR,QAAQ,CAAC;AAEb,MAAM,WAAW,WAAW;IAC1B,EAAE,EAAE,MAAM,CAAC;IACX,MAAM,EAAE,MAAM,CAAC;IACf,IAAI,EAAE,eAAe,CAAC;IACtB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,OAAO,GAAG,gBAAgB,GAAG,UAAU,GAAG,UAAU,CAAC;IAC7D,aAAa,CAAC,EAAE,MAAM,CAAC;CACxB;AAUD;;GAEG;AACH,wBAAsB,iBAAiB,CACrC,MAAM,EAAE,MAAM,EACd,IAAI,EAAE,eAAe,EACrB,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,MAAM,EACf,SAAS,EAAE,MAAM,EACjB,QAAQ,CAAC,EAAE,MAAM,GAChB,OAAO,CAAC,WAAW,CAAC,CA+BtB;AAED;;GAEG;AACH,wBAAsB,qBAAqB,CACzC,aAAa,EAAE,MAAM,EACrB,QAAQ,CAAC,EAAE,MAAM,GAChB,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAcxB;AAED;;GAEG;AACH,wBAAsB,iBAAiB,CACrC,aAAa,EAAE,MAAM,EACrB,QAAQ,EAAE,OAAO,EACjB,OAAO,CAAC,EAAE,MAAM,EAChB,QAAQ,CAAC,EAAE,MAAM,GAChB,OAAO,CAAC,IAAI,CAAC,CAWf;AAED;;GAEG;AACH,wBAAsB,gBAAgB,CACpC,MAAM,CAAC,EAAE,MAAM,EACf,QAAQ,CAAC,EAAE,MAAM,GAChB,OAAO,CAAC,WAAW,EAAE,CAAC,CAMxB"}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 交付物管理模块
|
|
3
|
+
* 管理 Worker 产生的可交付物(文档、代码、设计等)
|
|
4
|
+
*/
|
|
5
|
+
import { writeText, readText, readJson, writeJson } from '../utils/file.js';
|
|
6
|
+
import { existsSync } from 'fs';
|
|
7
|
+
import { join } from 'path';
|
|
8
|
+
import { withFileLock } from '../utils/lock.js';
|
|
9
|
+
const DELIVERABLES_DIR = '.webforge/deliverables';
|
|
10
|
+
const DELIVERABLES_INDEX = '.webforge/deliverables/index.json';
|
|
11
|
+
const DELIVERABLES_LOCK = '.webforge/deliverables/index.lock';
|
|
12
|
+
/**
|
|
13
|
+
* 创建交付物
|
|
14
|
+
*/
|
|
15
|
+
export async function createDeliverable(taskId, type, title, content, createdBy, basePath) {
|
|
16
|
+
const id = `del-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;
|
|
17
|
+
const timestamp = new Date().toISOString();
|
|
18
|
+
// 确定文件扩展名
|
|
19
|
+
const ext = type === 'code' ? 'ts' :
|
|
20
|
+
type === 'config' ? 'yaml' :
|
|
21
|
+
type === 'test' ? 'test.ts' : 'md';
|
|
22
|
+
const filename = `${id}.${ext}`;
|
|
23
|
+
const filePath = join(DELIVERABLES_DIR, filename);
|
|
24
|
+
// 保存文件
|
|
25
|
+
await writeText(resolveDeliverablePath(basePath, filePath), content);
|
|
26
|
+
const deliverable = {
|
|
27
|
+
id,
|
|
28
|
+
taskId,
|
|
29
|
+
type,
|
|
30
|
+
title,
|
|
31
|
+
content,
|
|
32
|
+
path: filePath,
|
|
33
|
+
createdBy,
|
|
34
|
+
createdAt: timestamp,
|
|
35
|
+
status: 'pending_review' // 默认需要人工审核
|
|
36
|
+
};
|
|
37
|
+
// 更新索引
|
|
38
|
+
await addToIndex(deliverable, basePath);
|
|
39
|
+
return deliverable;
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* 获取交付物内容
|
|
43
|
+
*/
|
|
44
|
+
export async function getDeliverableContent(deliverableId, basePath) {
|
|
45
|
+
try {
|
|
46
|
+
const index = await loadIndex(basePath);
|
|
47
|
+
const del = index.items.find(d => d.id === deliverableId);
|
|
48
|
+
if (!del)
|
|
49
|
+
return null;
|
|
50
|
+
const contentPath = resolveDeliverablePath(basePath, del.path);
|
|
51
|
+
if (existsSync(contentPath)) {
|
|
52
|
+
return await readText(contentPath);
|
|
53
|
+
}
|
|
54
|
+
return del.content; // 回退到内存中的内容
|
|
55
|
+
}
|
|
56
|
+
catch {
|
|
57
|
+
return null;
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* 审核交付物
|
|
62
|
+
*/
|
|
63
|
+
export async function reviewDeliverable(deliverableId, approved, comment, basePath) {
|
|
64
|
+
await withFileLock(resolveDeliverablePath(basePath, DELIVERABLES_LOCK), async () => {
|
|
65
|
+
const index = await loadIndex(basePath);
|
|
66
|
+
const del = index.items.find(d => d.id === deliverableId);
|
|
67
|
+
if (del) {
|
|
68
|
+
del.status = approved ? 'approved' : 'rejected';
|
|
69
|
+
del.reviewComment = comment;
|
|
70
|
+
await saveIndex(index, basePath);
|
|
71
|
+
}
|
|
72
|
+
});
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* 列出任务的所有交付物
|
|
76
|
+
*/
|
|
77
|
+
export async function listDeliverables(taskId, basePath) {
|
|
78
|
+
const index = await loadIndex(basePath);
|
|
79
|
+
if (taskId) {
|
|
80
|
+
return index.items.filter(d => d.taskId === taskId);
|
|
81
|
+
}
|
|
82
|
+
return index.items;
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* 加载索引
|
|
86
|
+
*/
|
|
87
|
+
async function loadIndex(basePath) {
|
|
88
|
+
try {
|
|
89
|
+
const indexPath = resolveDeliverablePath(basePath, DELIVERABLES_INDEX);
|
|
90
|
+
if (!existsSync(indexPath)) {
|
|
91
|
+
return { items: [] };
|
|
92
|
+
}
|
|
93
|
+
const data = await readJson(indexPath);
|
|
94
|
+
if (Array.isArray(data)) {
|
|
95
|
+
return { items: data };
|
|
96
|
+
}
|
|
97
|
+
if (data && Array.isArray(data.items)) {
|
|
98
|
+
return { items: data.items };
|
|
99
|
+
}
|
|
100
|
+
return { items: [] };
|
|
101
|
+
}
|
|
102
|
+
catch {
|
|
103
|
+
return { items: [] };
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* 保存索引
|
|
108
|
+
*/
|
|
109
|
+
async function saveIndex(index, basePath) {
|
|
110
|
+
await writeJson(resolveDeliverablePath(basePath, DELIVERABLES_INDEX), index);
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* 添加到索引
|
|
114
|
+
*/
|
|
115
|
+
async function addToIndex(deliverable, basePath) {
|
|
116
|
+
await withFileLock(resolveDeliverablePath(basePath, DELIVERABLES_LOCK), async () => {
|
|
117
|
+
const index = await loadIndex(basePath);
|
|
118
|
+
index.items.push(deliverable);
|
|
119
|
+
await saveIndex(index, basePath);
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
function resolveDeliverablePath(basePath, relativePath) {
|
|
123
|
+
return basePath ? join(basePath, relativePath) : relativePath;
|
|
124
|
+
}
|
|
125
|
+
//# sourceMappingURL=deliverable.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"deliverable.js","sourceRoot":"","sources":["../../src/core/deliverable.ts"],"names":[],"mappings":"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"}
|