@xdevops/issue-auto-finish 1.0.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/bin/issue-auto-finish.js +2 -0
- package/dist/ai-runner/AIRunner.d.ts +27 -0
- package/dist/ai-runner/AIRunner.d.ts.map +1 -0
- package/dist/ai-runner/BaseAIRunner.d.ts +19 -0
- package/dist/ai-runner/BaseAIRunner.d.ts.map +1 -0
- package/dist/ai-runner/ClaudeInternalRunner.d.ts +13 -0
- package/dist/ai-runner/ClaudeInternalRunner.d.ts.map +1 -0
- package/dist/ai-runner/CodebuddyRunner.d.ts +13 -0
- package/dist/ai-runner/CodebuddyRunner.d.ts.map +1 -0
- package/dist/ai-runner/CursorAgentRunner.d.ts +13 -0
- package/dist/ai-runner/CursorAgentRunner.d.ts.map +1 -0
- package/dist/ai-runner/index.d.ts +15 -0
- package/dist/ai-runner/index.d.ts.map +1 -0
- package/dist/chunk-HCHEFK4Z.js +80 -0
- package/dist/chunk-HCHEFK4Z.js.map +1 -0
- package/dist/chunk-I3T573SU.js +153 -0
- package/dist/chunk-I3T573SU.js.map +1 -0
- package/dist/chunk-IDUKWCC2.js +1995 -0
- package/dist/chunk-IDUKWCC2.js.map +1 -0
- package/dist/chunk-OWVT3Z34.js +770 -0
- package/dist/chunk-OWVT3Z34.js.map +1 -0
- package/dist/chunk-RIUI4ROA.js +180 -0
- package/dist/chunk-RIUI4ROA.js.map +1 -0
- package/dist/chunk-TBIEB3JY.js +3295 -0
- package/dist/chunk-TBIEB3JY.js.map +1 -0
- package/dist/cli/commands/doctor.d.ts +2 -0
- package/dist/cli/commands/doctor.d.ts.map +1 -0
- package/dist/cli/commands/init.d.ts +6 -0
- package/dist/cli/commands/init.d.ts.map +1 -0
- package/dist/cli/commands/start.d.ts +5 -0
- package/dist/cli/commands/start.d.ts.map +1 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/setup/ConfigGenerator.d.ts +44 -0
- package/dist/cli/setup/ConfigGenerator.d.ts.map +1 -0
- package/dist/cli/setup/DependencyChecker.d.ts +21 -0
- package/dist/cli/setup/DependencyChecker.d.ts.map +1 -0
- package/dist/cli.js +73 -0
- package/dist/cli.js.map +1 -0
- package/dist/clients/GongfengClient.d.ts +86 -0
- package/dist/clients/GongfengClient.d.ts.map +1 -0
- package/dist/config.d.ts +92 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/deploy/DevServerManager.d.ts +21 -0
- package/dist/deploy/DevServerManager.d.ts.map +1 -0
- package/dist/deploy/PortAllocator.d.ts +20 -0
- package/dist/deploy/PortAllocator.d.ts.map +1 -0
- package/dist/deploy/index.d.ts +3 -0
- package/dist/deploy/index.d.ts.map +1 -0
- package/dist/doctor-B26Q6JWI.js +33 -0
- package/dist/doctor-B26Q6JWI.js.map +1 -0
- package/dist/e2e/E2eSettings.d.ts +6 -0
- package/dist/e2e/E2eSettings.d.ts.map +1 -0
- package/dist/e2e/ScreenshotCollector.d.ts +11 -0
- package/dist/e2e/ScreenshotCollector.d.ts.map +1 -0
- package/dist/e2e/ScreenshotPublisher.d.ts +16 -0
- package/dist/e2e/ScreenshotPublisher.d.ts.map +1 -0
- package/dist/events/EventBus.d.ts +14 -0
- package/dist/events/EventBus.d.ts.map +1 -0
- package/dist/git/GitOperations.d.ts +30 -0
- package/dist/git/GitOperations.d.ts.map +1 -0
- package/dist/git/WorktreeContext.d.ts +9 -0
- package/dist/git/WorktreeContext.d.ts.map +1 -0
- package/dist/i18n/index.d.ts +8 -0
- package/dist/i18n/index.d.ts.map +1 -0
- package/dist/i18n/locales/en.d.ts +2 -0
- package/dist/i18n/locales/en.d.ts.map +1 -0
- package/dist/i18n/locales/zh-CN.d.ts +2 -0
- package/dist/i18n/locales/zh-CN.d.ts.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +12 -0
- package/dist/index.js.map +1 -0
- package/dist/init-L3VIWCOV.js +65 -0
- package/dist/init-L3VIWCOV.js.map +1 -0
- package/dist/lib.d.ts +26 -0
- package/dist/lib.d.ts.map +1 -0
- package/dist/lib.js +50 -0
- package/dist/lib.js.map +1 -0
- package/dist/logger.d.ts +15 -0
- package/dist/logger.d.ts.map +1 -0
- package/dist/notesync/NoteSyncSettings.d.ts +12 -0
- package/dist/notesync/NoteSyncSettings.d.ts.map +1 -0
- package/dist/orchestrator/PipelineOrchestrator.d.ts +53 -0
- package/dist/orchestrator/PipelineOrchestrator.d.ts.map +1 -0
- package/dist/persistence/PlanPersistence.d.ts +37 -0
- package/dist/persistence/PlanPersistence.d.ts.map +1 -0
- package/dist/phases/AnalysisPhase.d.ts +13 -0
- package/dist/phases/AnalysisPhase.d.ts.map +1 -0
- package/dist/phases/BasePhase.d.ts +44 -0
- package/dist/phases/BasePhase.d.ts.map +1 -0
- package/dist/phases/BuildPhase.d.ts +9 -0
- package/dist/phases/BuildPhase.d.ts.map +1 -0
- package/dist/phases/DesignPhase.d.ts +13 -0
- package/dist/phases/DesignPhase.d.ts.map +1 -0
- package/dist/phases/ImplementPhase.d.ts +9 -0
- package/dist/phases/ImplementPhase.d.ts.map +1 -0
- package/dist/phases/PhaseFactory.d.ts +13 -0
- package/dist/phases/PhaseFactory.d.ts.map +1 -0
- package/dist/phases/PlanPhase.d.ts +14 -0
- package/dist/phases/PlanPhase.d.ts.map +1 -0
- package/dist/phases/VerifyPhase.d.ts +13 -0
- package/dist/phases/VerifyPhase.d.ts.map +1 -0
- package/dist/pipeline/PipelineDefinition.d.ts +40 -0
- package/dist/pipeline/PipelineDefinition.d.ts.map +1 -0
- package/dist/poller/IssuePoller.d.ts +26 -0
- package/dist/poller/IssuePoller.d.ts.map +1 -0
- package/dist/prompts/brainstorm-templates.d.ts +4 -0
- package/dist/prompts/brainstorm-templates.d.ts.map +1 -0
- package/dist/prompts/templates.d.ts +27 -0
- package/dist/prompts/templates.d.ts.map +1 -0
- package/dist/rules/RuleResolver.d.ts +13 -0
- package/dist/rules/RuleResolver.d.ts.map +1 -0
- package/dist/rules/index.d.ts +3 -0
- package/dist/rules/index.d.ts.map +1 -0
- package/dist/run.d.ts +2 -0
- package/dist/run.d.ts.map +1 -0
- package/dist/run.js +15 -0
- package/dist/run.js.map +1 -0
- package/dist/services/BrainstormService.d.ts +39 -0
- package/dist/services/BrainstormService.d.ts.map +1 -0
- package/dist/start-TVN4SS6E.js +25 -0
- package/dist/start-TVN4SS6E.js.map +1 -0
- package/dist/supplement/SupplementStore.d.ts +21 -0
- package/dist/supplement/SupplementStore.d.ts.map +1 -0
- package/dist/tracker/IssueState.d.ts +63 -0
- package/dist/tracker/IssueState.d.ts.map +1 -0
- package/dist/tracker/IssueTracker.d.ts +29 -0
- package/dist/tracker/IssueTracker.d.ts.map +1 -0
- package/dist/utils/AsyncMutex.d.ts +14 -0
- package/dist/utils/AsyncMutex.d.ts.map +1 -0
- package/dist/utils/MergeRequestHelper.d.ts +10 -0
- package/dist/utils/MergeRequestHelper.d.ts.map +1 -0
- package/dist/web/AgentLogStore.d.ts +22 -0
- package/dist/web/AgentLogStore.d.ts.map +1 -0
- package/dist/web/WebServer.d.ts +26 -0
- package/dist/web/WebServer.d.ts.map +1 -0
- package/dist/web/routes/api.d.ts +20 -0
- package/dist/web/routes/api.d.ts.map +1 -0
- package/dist/web/routes/brainstorm.d.ts +9 -0
- package/dist/web/routes/brainstorm.d.ts.map +1 -0
- package/dist/web/routes/setup.d.ts +8 -0
- package/dist/web/routes/setup.d.ts.map +1 -0
- package/dist/webhook/CommandExecutor.d.ts +45 -0
- package/dist/webhook/CommandExecutor.d.ts.map +1 -0
- package/dist/webhook/CommandParser.d.ts +24 -0
- package/dist/webhook/CommandParser.d.ts.map +1 -0
- package/dist/webhook/IntentRecognizer.d.ts +35 -0
- package/dist/webhook/IntentRecognizer.d.ts.map +1 -0
- package/dist/webhook/NoteDeduplicator.d.ts +18 -0
- package/dist/webhook/NoteDeduplicator.d.ts.map +1 -0
- package/dist/webhook/WebhookHandler.d.ts +30 -0
- package/dist/webhook/WebhookHandler.d.ts.map +1 -0
- package/dist/webhook/WebhookServer.d.ts +21 -0
- package/dist/webhook/WebhookServer.d.ts.map +1 -0
- package/dist/webhook/index.d.ts +12 -0
- package/dist/webhook/index.d.ts.map +1 -0
- package/package.json +82 -0
- package/src/web/frontend/dist/assets/index-CQdlU9PE.js +65 -0
- package/src/web/frontend/dist/assets/index-CgMEkyZJ.css +1 -0
- package/src/web/frontend/dist/index.html +13 -0
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AI Runner 接口:统一 claude-internal 与 cursor agent 两种 CLI 的调用契约。
|
|
3
|
+
*/
|
|
4
|
+
export interface StreamEvent {
|
|
5
|
+
type: string;
|
|
6
|
+
content: unknown;
|
|
7
|
+
timestamp: string;
|
|
8
|
+
}
|
|
9
|
+
export interface RunOptions {
|
|
10
|
+
prompt: string;
|
|
11
|
+
workDir: string;
|
|
12
|
+
timeoutMs: number;
|
|
13
|
+
sessionId?: string;
|
|
14
|
+
continueSession?: boolean;
|
|
15
|
+
mode?: 'plan' | 'agent';
|
|
16
|
+
onStreamEvent?: (event: StreamEvent) => void;
|
|
17
|
+
}
|
|
18
|
+
export interface RunResult {
|
|
19
|
+
success: boolean;
|
|
20
|
+
output: string;
|
|
21
|
+
sessionId?: string;
|
|
22
|
+
exitCode: number | null;
|
|
23
|
+
}
|
|
24
|
+
export interface AIRunner {
|
|
25
|
+
run(options: RunOptions): Promise<RunResult>;
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=AIRunner.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"AIRunner.d.ts","sourceRoot":"","sources":["../../src/ai-runner/AIRunner.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,OAAO,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,UAAU;IACzB,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,eAAe,CAAC,EAAE,OAAO,CAAC;IAC1B,IAAI,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC;IACxB,aAAa,CAAC,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,IAAI,CAAC;CAC9C;AAED,MAAM,WAAW,SAAS;IACxB,OAAO,EAAE,OAAO,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAC;CACzB;AAED,MAAM,WAAW,QAAQ;IACvB,GAAG,CAAC,OAAO,EAAE,UAAU,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC;CAC9C"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { type SpawnOptions } from 'node:child_process';
|
|
2
|
+
import type { RunOptions, RunResult } from './AIRunner.js';
|
|
3
|
+
export declare abstract class BaseAIRunner {
|
|
4
|
+
protected abstract getBinary(): string;
|
|
5
|
+
protected abstract buildArgs(options: RunOptions): string[];
|
|
6
|
+
protected abstract getSpawnOptions(options: RunOptions): SpawnOptions;
|
|
7
|
+
run(options: RunOptions): Promise<RunResult>;
|
|
8
|
+
private emitStreamLine;
|
|
9
|
+
/**
|
|
10
|
+
* Handles both stream-json (one JSON object per line) and legacy single-JSON formats.
|
|
11
|
+
*/
|
|
12
|
+
parseOutput(rawOutput: string, _sessionId?: string): {
|
|
13
|
+
output: string;
|
|
14
|
+
resolvedSessionId?: string;
|
|
15
|
+
};
|
|
16
|
+
private summarizeStreamEvents;
|
|
17
|
+
private tryParseStreamJson;
|
|
18
|
+
}
|
|
19
|
+
//# sourceMappingURL=BaseAIRunner.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"BaseAIRunner.d.ts","sourceRoot":"","sources":["../../src/ai-runner/BaseAIRunner.ts"],"names":[],"mappings":"AAAA,OAAO,EAAS,KAAK,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAE9D,OAAO,KAAK,EAAE,UAAU,EAAE,SAAS,EAAe,MAAM,eAAe,CAAC;AAIxE,8BAAsB,YAAY;IAChC,SAAS,CAAC,QAAQ,CAAC,SAAS,IAAI,MAAM;IACtC,SAAS,CAAC,QAAQ,CAAC,SAAS,CAAC,OAAO,EAAE,UAAU,GAAG,MAAM,EAAE;IAC3D,SAAS,CAAC,QAAQ,CAAC,eAAe,CAAC,OAAO,EAAE,UAAU,GAAG,YAAY;IAE/D,GAAG,CAAC,OAAO,EAAE,UAAU,GAAG,OAAO,CAAC,SAAS,CAAC;IAyGlD,OAAO,CAAC,cAAc;IAatB;;OAEG;IACH,WAAW,CAAC,SAAS,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,MAAM,GAAG;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,iBAAiB,CAAC,EAAE,MAAM,CAAA;KAAE;IA8CnG,OAAO,CAAC,qBAAqB;IAsB7B,OAAO,CAAC,kBAAkB;CAc3B"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { SpawnOptions } from 'node:child_process';
|
|
2
|
+
import { BaseAIRunner } from './BaseAIRunner.js';
|
|
3
|
+
import type { RunOptions } from './AIRunner.js';
|
|
4
|
+
export declare class ClaudeInternalRunner extends BaseAIRunner {
|
|
5
|
+
private readonly binary;
|
|
6
|
+
private readonly nvmNodeVersion;
|
|
7
|
+
private readonly model?;
|
|
8
|
+
constructor(binary: string, nvmNodeVersion: string, model?: string);
|
|
9
|
+
protected getBinary(): string;
|
|
10
|
+
protected buildArgs(options: RunOptions): string[];
|
|
11
|
+
protected getSpawnOptions(options: RunOptions): SpawnOptions;
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=ClaudeInternalRunner.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ClaudeInternalRunner.d.ts","sourceRoot":"","sources":["../../src/ai-runner/ClaudeInternalRunner.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAEhD,qBAAa,oBAAqB,SAAQ,YAAY;IACpD,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAS;IAChC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;IACxC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAS;gBAEpB,MAAM,EAAE,MAAM,EAAE,cAAc,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM;IAOlE,SAAS,CAAC,SAAS,IAAI,MAAM;IAI7B,SAAS,CAAC,SAAS,CAAC,OAAO,EAAE,UAAU,GAAG,MAAM,EAAE;IAgBlD,SAAS,CAAC,eAAe,CAAC,OAAO,EAAE,UAAU,GAAG,YAAY;CAU7D"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { SpawnOptions } from 'node:child_process';
|
|
2
|
+
import { BaseAIRunner } from './BaseAIRunner.js';
|
|
3
|
+
import type { RunOptions } from './AIRunner.js';
|
|
4
|
+
export declare class CodebuddyRunner extends BaseAIRunner {
|
|
5
|
+
private readonly binary;
|
|
6
|
+
private readonly nvmNodeVersion;
|
|
7
|
+
private readonly model?;
|
|
8
|
+
constructor(binary: string, nvmNodeVersion: string, model?: string);
|
|
9
|
+
protected getBinary(): string;
|
|
10
|
+
protected buildArgs(options: RunOptions): string[];
|
|
11
|
+
protected getSpawnOptions(options: RunOptions): SpawnOptions;
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=CodebuddyRunner.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"CodebuddyRunner.d.ts","sourceRoot":"","sources":["../../src/ai-runner/CodebuddyRunner.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAEhD,qBAAa,eAAgB,SAAQ,YAAY;IAC/C,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAS;IAChC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;IACxC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAS;gBAEpB,MAAM,EAAE,MAAM,EAAE,cAAc,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM;IAOlE,SAAS,CAAC,SAAS,IAAI,MAAM;IAI7B,SAAS,CAAC,SAAS,CAAC,OAAO,EAAE,UAAU,GAAG,MAAM,EAAE;IAgBlD,SAAS,CAAC,eAAe,CAAC,OAAO,EAAE,UAAU,GAAG,YAAY;CAS7D"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { SpawnOptions } from 'node:child_process';
|
|
2
|
+
import { BaseAIRunner } from './BaseAIRunner.js';
|
|
3
|
+
import type { RunOptions } from './AIRunner.js';
|
|
4
|
+
export declare class CursorAgentRunner extends BaseAIRunner {
|
|
5
|
+
private readonly binary;
|
|
6
|
+
private readonly nvmNodeVersion;
|
|
7
|
+
private readonly model?;
|
|
8
|
+
constructor(binary: string, nvmNodeVersion: string, model?: string);
|
|
9
|
+
protected getBinary(): string;
|
|
10
|
+
protected buildArgs(options: RunOptions): string[];
|
|
11
|
+
protected getSpawnOptions(_options: RunOptions): SpawnOptions;
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=CursorAgentRunner.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"CursorAgentRunner.d.ts","sourceRoot":"","sources":["../../src/ai-runner/CursorAgentRunner.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAEhD,qBAAa,iBAAkB,SAAQ,YAAY;IACjD,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAS;IAChC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;IACxC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAS;gBAEpB,MAAM,EAAE,MAAM,EAAE,cAAc,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM;IAOlE,SAAS,CAAC,SAAS,IAAI,MAAM;IAI7B,SAAS,CAAC,SAAS,CAAC,OAAO,EAAE,UAAU,GAAG,MAAM,EAAE;IAuBlD,SAAS,CAAC,eAAe,CAAC,QAAQ,EAAE,UAAU,GAAG,YAAY;CAS9D"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type { AIRunner } from './AIRunner.js';
|
|
2
|
+
export type { AIRunner, RunOptions, RunResult, StreamEvent } from './AIRunner.js';
|
|
3
|
+
export { BaseAIRunner } from './BaseAIRunner.js';
|
|
4
|
+
export { ClaudeInternalRunner } from './ClaudeInternalRunner.js';
|
|
5
|
+
export { CodebuddyRunner } from './CodebuddyRunner.js';
|
|
6
|
+
export { CursorAgentRunner } from './CursorAgentRunner.js';
|
|
7
|
+
export interface AIConfig {
|
|
8
|
+
mode: 'claude-internal' | 'cursor-agent' | 'codebuddy';
|
|
9
|
+
binary: string;
|
|
10
|
+
phaseTimeoutMs: number;
|
|
11
|
+
nvmNodeVersion: string;
|
|
12
|
+
model?: string;
|
|
13
|
+
}
|
|
14
|
+
export declare function createAIRunner(ai: AIConfig): AIRunner;
|
|
15
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/ai-runner/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAK9C,YAAY,EAAE,QAAQ,EAAE,UAAU,EAAE,SAAS,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAClF,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC;AACjE,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;AACvD,OAAO,EAAE,iBAAiB,EAAE,MAAM,wBAAwB,CAAC;AAE3D,MAAM,WAAW,QAAQ;IACvB,IAAI,EAAE,iBAAiB,GAAG,cAAc,GAAG,WAAW,CAAC;IACvD,MAAM,EAAE,MAAM,CAAC;IACf,cAAc,EAAE,MAAM,CAAC;IACvB,cAAc,EAAE,MAAM,CAAC;IACvB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,wBAAgB,cAAc,CAAC,EAAE,EAAE,QAAQ,GAAG,QAAQ,CASrD"}
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
// src/cli/setup/ConfigGenerator.ts
|
|
2
|
+
import fs from "fs";
|
|
3
|
+
import path from "path";
|
|
4
|
+
import os from "os";
|
|
5
|
+
var ConfigGenerator = class _ConfigGenerator {
|
|
6
|
+
static getGlobalConfigDir() {
|
|
7
|
+
return path.join(os.homedir(), ".issue-auto-finish");
|
|
8
|
+
}
|
|
9
|
+
static getGlobalConfigPath() {
|
|
10
|
+
return path.join(_ConfigGenerator.getGlobalConfigDir(), ".env");
|
|
11
|
+
}
|
|
12
|
+
static isInitialized(configPath) {
|
|
13
|
+
const envPath = configPath ?? _ConfigGenerator.getGlobalConfigPath();
|
|
14
|
+
return fs.existsSync(envPath);
|
|
15
|
+
}
|
|
16
|
+
static resolveConfigPath(configPath) {
|
|
17
|
+
if (configPath) return configPath;
|
|
18
|
+
const cwdEnv = path.join(process.cwd(), ".env");
|
|
19
|
+
if (fs.existsSync(cwdEnv)) return cwdEnv;
|
|
20
|
+
return _ConfigGenerator.getGlobalConfigPath();
|
|
21
|
+
}
|
|
22
|
+
static generate(config) {
|
|
23
|
+
const lines = [];
|
|
24
|
+
const addSection = (title) => {
|
|
25
|
+
if (lines.length > 0) lines.push("");
|
|
26
|
+
lines.push("# " + title);
|
|
27
|
+
};
|
|
28
|
+
const addVar = (key, value, comment) => {
|
|
29
|
+
if (value === void 0) return;
|
|
30
|
+
if (comment) lines.push("# " + comment);
|
|
31
|
+
lines.push(key + "=" + String(value));
|
|
32
|
+
};
|
|
33
|
+
addSection("Gongfeng");
|
|
34
|
+
addVar("GONGFENG_API_URL", config.gongfeng.apiUrl);
|
|
35
|
+
addVar("GONGFENG_PRIVATE_TOKEN", config.gongfeng.privateToken);
|
|
36
|
+
addVar("GONGFENG_PROJECT_PATH", config.gongfeng.projectPath);
|
|
37
|
+
addSection("Project");
|
|
38
|
+
addVar("PROJECT_WORK_DIR", config.project.workDir);
|
|
39
|
+
addVar("GIT_ROOT_DIR", config.project.gitRootDir);
|
|
40
|
+
addVar("BASE_BRANCH", config.project.baseBranch ?? "master");
|
|
41
|
+
addVar("BRANCH_PREFIX", config.project.branchPrefix ?? "feat/issue");
|
|
42
|
+
addVar("WORKTREE_BASE_DIR", config.project.worktreeBaseDir);
|
|
43
|
+
addVar("PROJECT_SUBDIR", config.project.projectSubDir);
|
|
44
|
+
addSection("AI");
|
|
45
|
+
addVar("AI_RUNNER_MODE", config.ai.mode);
|
|
46
|
+
addVar("AI_MODEL", config.ai.model);
|
|
47
|
+
addVar("CLAUDE_PHASE_TIMEOUT_MS", config.ai.phaseTimeoutMs ?? 18e5);
|
|
48
|
+
addSection("Poll");
|
|
49
|
+
addVar("POLL_DISCOVERY_INTERVAL_MS", config.poll?.discoveryIntervalMs ?? 6e4);
|
|
50
|
+
addVar("POLL_DRIVE_INTERVAL_MS", config.poll?.driveIntervalMs ?? 15e3);
|
|
51
|
+
addVar("MAX_RETRIES", config.poll?.maxRetries ?? 3);
|
|
52
|
+
addVar("MAX_CONCURRENT_ISSUES", config.poll?.maxConcurrent ?? 3);
|
|
53
|
+
addSection("Pipeline");
|
|
54
|
+
addVar("PIPELINE_MODE", config.pipeline?.mode ?? "auto");
|
|
55
|
+
addSection("Review");
|
|
56
|
+
addVar("REVIEW_ENABLED", config.review?.enabled ?? true);
|
|
57
|
+
addSection("Web UI");
|
|
58
|
+
addVar("WEB_ENABLED", true);
|
|
59
|
+
addVar("WEB_PORT", config.web?.port ?? 3e3);
|
|
60
|
+
addVar("WEB_BASE_URL", "http://localhost:" + (config.web?.port ?? 3e3));
|
|
61
|
+
addSection("Issue Note Sync");
|
|
62
|
+
addVar("ISSUE_NOTE_SYNC_ENABLED", true);
|
|
63
|
+
return lines.join("\n") + "\n";
|
|
64
|
+
}
|
|
65
|
+
static write(config, outputPath) {
|
|
66
|
+
const targetPath = outputPath ?? _ConfigGenerator.getGlobalConfigPath();
|
|
67
|
+
const dir = path.dirname(targetPath);
|
|
68
|
+
if (!fs.existsSync(dir)) {
|
|
69
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
70
|
+
}
|
|
71
|
+
const content = _ConfigGenerator.generate(config);
|
|
72
|
+
fs.writeFileSync(targetPath, content, "utf-8");
|
|
73
|
+
return targetPath;
|
|
74
|
+
}
|
|
75
|
+
};
|
|
76
|
+
|
|
77
|
+
export {
|
|
78
|
+
ConfigGenerator
|
|
79
|
+
};
|
|
80
|
+
//# sourceMappingURL=chunk-HCHEFK4Z.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/cli/setup/ConfigGenerator.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport os from 'node:os';\n\nexport interface SetupConfig {\n gongfeng: {\n apiUrl: string;\n privateToken: string;\n projectPath: string;\n };\n project: {\n workDir: string;\n gitRootDir?: string;\n baseBranch?: string;\n branchPrefix?: string;\n worktreeBaseDir?: string;\n projectSubDir?: string;\n };\n ai: {\n mode: 'claude-internal' | 'cursor-agent' | 'codebuddy';\n model?: string;\n phaseTimeoutMs?: number;\n };\n poll?: {\n discoveryIntervalMs?: number;\n driveIntervalMs?: number;\n maxRetries?: number;\n maxConcurrent?: number;\n };\n web?: {\n port?: number;\n };\n pipeline?: {\n mode?: string;\n };\n review?: {\n enabled?: boolean;\n };\n}\n\nexport class ConfigGenerator {\n static getGlobalConfigDir(): string {\n return path.join(os.homedir(), '.issue-auto-finish');\n }\n\n static getGlobalConfigPath(): string {\n return path.join(ConfigGenerator.getGlobalConfigDir(), '.env');\n }\n\n static isInitialized(configPath?: string): boolean {\n const envPath = configPath ?? ConfigGenerator.getGlobalConfigPath();\n return fs.existsSync(envPath);\n }\n\n static resolveConfigPath(configPath?: string): string {\n if (configPath) return configPath;\n const cwdEnv = path.join(process.cwd(), '.env');\n if (fs.existsSync(cwdEnv)) return cwdEnv;\n return ConfigGenerator.getGlobalConfigPath();\n }\n\n static generate(config: SetupConfig): string {\n const lines: string[] = [];\n\n const addSection = (title: string) => {\n if (lines.length > 0) lines.push('');\n lines.push('# ' + title);\n };\n\n const addVar = (\n key: string,\n value: string | number | boolean | undefined,\n comment?: string,\n ) => {\n if (value === undefined) return;\n if (comment) lines.push('# ' + comment);\n lines.push(key + '=' + String(value));\n };\n\n addSection('Gongfeng');\n addVar('GONGFENG_API_URL', config.gongfeng.apiUrl);\n addVar('GONGFENG_PRIVATE_TOKEN', config.gongfeng.privateToken);\n addVar('GONGFENG_PROJECT_PATH', config.gongfeng.projectPath);\n\n addSection('Project');\n addVar('PROJECT_WORK_DIR', config.project.workDir);\n addVar('GIT_ROOT_DIR', config.project.gitRootDir);\n addVar('BASE_BRANCH', config.project.baseBranch ?? 'master');\n addVar('BRANCH_PREFIX', config.project.branchPrefix ?? 'feat/issue');\n addVar('WORKTREE_BASE_DIR', config.project.worktreeBaseDir);\n addVar('PROJECT_SUBDIR', config.project.projectSubDir);\n\n addSection('AI');\n addVar('AI_RUNNER_MODE', config.ai.mode);\n addVar('AI_MODEL', config.ai.model);\n addVar('CLAUDE_PHASE_TIMEOUT_MS', config.ai.phaseTimeoutMs ?? 1800000);\n\n addSection('Poll');\n addVar('POLL_DISCOVERY_INTERVAL_MS', config.poll?.discoveryIntervalMs ?? 60000);\n addVar('POLL_DRIVE_INTERVAL_MS', config.poll?.driveIntervalMs ?? 15000);\n addVar('MAX_RETRIES', config.poll?.maxRetries ?? 3);\n addVar('MAX_CONCURRENT_ISSUES', config.poll?.maxConcurrent ?? 3);\n\n addSection('Pipeline');\n addVar('PIPELINE_MODE', config.pipeline?.mode ?? 'auto');\n\n addSection('Review');\n addVar('REVIEW_ENABLED', config.review?.enabled ?? true);\n\n addSection('Web UI');\n addVar('WEB_ENABLED', true);\n addVar('WEB_PORT', config.web?.port ?? 3000);\n addVar('WEB_BASE_URL', 'http://localhost:' + (config.web?.port ?? 3000));\n\n addSection('Issue Note Sync');\n addVar('ISSUE_NOTE_SYNC_ENABLED', true);\n\n return lines.join('\\n') + '\\n';\n }\n\n static write(config: SetupConfig, outputPath?: string): string {\n const targetPath = outputPath ?? ConfigGenerator.getGlobalConfigPath();\n const dir = path.dirname(targetPath);\n if (!fs.existsSync(dir)) {\n fs.mkdirSync(dir, { recursive: true });\n }\n const content = ConfigGenerator.generate(config);\n fs.writeFileSync(targetPath, content, 'utf-8');\n return targetPath;\n }\n}\n"],"mappings":"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","names":[]}
|
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
import {
|
|
2
|
+
ConfigGenerator
|
|
3
|
+
} from "./chunk-HCHEFK4Z.js";
|
|
4
|
+
import {
|
|
5
|
+
DependencyChecker
|
|
6
|
+
} from "./chunk-RIUI4ROA.js";
|
|
7
|
+
|
|
8
|
+
// src/web/routes/setup.ts
|
|
9
|
+
import { Router } from "express";
|
|
10
|
+
import fs from "fs";
|
|
11
|
+
function createSetupRouter(deps = {}) {
|
|
12
|
+
const router = Router();
|
|
13
|
+
const checker = new DependencyChecker();
|
|
14
|
+
router.get("/api/setup/status", (_req, res) => {
|
|
15
|
+
const initialized = deps.serviceMode || ConfigGenerator.isInitialized(deps.configPath);
|
|
16
|
+
res.json({
|
|
17
|
+
initialized,
|
|
18
|
+
configPath: deps.configPath ?? ConfigGenerator.getGlobalConfigPath(),
|
|
19
|
+
globalConfigDir: ConfigGenerator.getGlobalConfigDir()
|
|
20
|
+
});
|
|
21
|
+
});
|
|
22
|
+
router.get("/api/setup/check-env", async (_req, res) => {
|
|
23
|
+
try {
|
|
24
|
+
const aiMode = _req.query.aiMode || void 0;
|
|
25
|
+
const results = await checker.checkAll(aiMode);
|
|
26
|
+
res.json({ dependencies: results });
|
|
27
|
+
} catch (err) {
|
|
28
|
+
res.status(500).json({ error: err.message });
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
router.post("/api/setup/validate-token", async (req, res) => {
|
|
32
|
+
const { apiUrl, token } = req.body;
|
|
33
|
+
if (!apiUrl || !token) {
|
|
34
|
+
res.status(400).json({ error: "apiUrl and token are required" });
|
|
35
|
+
return;
|
|
36
|
+
}
|
|
37
|
+
try {
|
|
38
|
+
const url = `${apiUrl.replace(/\/$/, "")}/api/v3/user`;
|
|
39
|
+
const resp = await fetch(url, {
|
|
40
|
+
headers: { "PRIVATE-TOKEN": token },
|
|
41
|
+
signal: AbortSignal.timeout(1e4)
|
|
42
|
+
});
|
|
43
|
+
if (resp.ok) {
|
|
44
|
+
const user = await resp.json();
|
|
45
|
+
res.json({ valid: true, user: { username: user.username, name: user.name } });
|
|
46
|
+
} else {
|
|
47
|
+
res.json({ valid: false, error: `HTTP ${resp.status}: ${resp.statusText}` });
|
|
48
|
+
}
|
|
49
|
+
} catch (err) {
|
|
50
|
+
res.json({ valid: false, error: err.message });
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
router.post("/api/setup/validate-repo", async (req, res) => {
|
|
54
|
+
const { apiUrl, token, projectPath } = req.body;
|
|
55
|
+
if (!apiUrl || !token || !projectPath) {
|
|
56
|
+
res.status(400).json({ error: "apiUrl, token, and projectPath are required" });
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
try {
|
|
60
|
+
const encoded = encodeURIComponent(projectPath);
|
|
61
|
+
const url = `${apiUrl.replace(/\/$/, "")}/api/v3/projects/${encoded}`;
|
|
62
|
+
const resp = await fetch(url, {
|
|
63
|
+
headers: { "PRIVATE-TOKEN": token },
|
|
64
|
+
signal: AbortSignal.timeout(1e4)
|
|
65
|
+
});
|
|
66
|
+
if (resp.ok) {
|
|
67
|
+
const project = await resp.json();
|
|
68
|
+
res.json({ valid: true, project });
|
|
69
|
+
} else {
|
|
70
|
+
res.json({ valid: false, error: `HTTP ${resp.status}: ${resp.statusText}` });
|
|
71
|
+
}
|
|
72
|
+
} catch (err) {
|
|
73
|
+
res.json({ valid: false, error: err.message });
|
|
74
|
+
}
|
|
75
|
+
});
|
|
76
|
+
router.post("/api/setup/install-dep", (req, res) => {
|
|
77
|
+
const { name } = req.body;
|
|
78
|
+
if (!name) {
|
|
79
|
+
res.status(400).json({ error: "name is required" });
|
|
80
|
+
return;
|
|
81
|
+
}
|
|
82
|
+
res.setHeader("Content-Type", "text/event-stream");
|
|
83
|
+
res.setHeader("Cache-Control", "no-cache");
|
|
84
|
+
res.setHeader("Connection", "keep-alive");
|
|
85
|
+
res.flushHeaders();
|
|
86
|
+
(async () => {
|
|
87
|
+
try {
|
|
88
|
+
for await (const line of checker.install(name)) {
|
|
89
|
+
res.write(`data: ${JSON.stringify({ line })}
|
|
90
|
+
|
|
91
|
+
`);
|
|
92
|
+
}
|
|
93
|
+
res.write(`data: ${JSON.stringify({ done: true })}
|
|
94
|
+
|
|
95
|
+
`);
|
|
96
|
+
} catch (err) {
|
|
97
|
+
res.write(`data: ${JSON.stringify({ error: err.message })}
|
|
98
|
+
|
|
99
|
+
`);
|
|
100
|
+
} finally {
|
|
101
|
+
res.end();
|
|
102
|
+
}
|
|
103
|
+
})();
|
|
104
|
+
req.on("close", () => {
|
|
105
|
+
res.end();
|
|
106
|
+
});
|
|
107
|
+
});
|
|
108
|
+
router.post("/api/setup/generate-preview", (req, res) => {
|
|
109
|
+
const config = req.body;
|
|
110
|
+
try {
|
|
111
|
+
const preview = ConfigGenerator.generate(config);
|
|
112
|
+
res.json({ preview });
|
|
113
|
+
} catch (err) {
|
|
114
|
+
res.status(400).json({ error: err.message });
|
|
115
|
+
}
|
|
116
|
+
});
|
|
117
|
+
router.post("/api/setup/complete", (req, res) => {
|
|
118
|
+
const { config, outputPath } = req.body;
|
|
119
|
+
if (!config) {
|
|
120
|
+
res.status(400).json({ error: "config is required" });
|
|
121
|
+
return;
|
|
122
|
+
}
|
|
123
|
+
try {
|
|
124
|
+
const targetPath = outputPath ?? deps.configPath;
|
|
125
|
+
const savedPath = ConfigGenerator.write(config, targetPath);
|
|
126
|
+
res.json({ success: true, configPath: savedPath });
|
|
127
|
+
deps.onComplete?.(savedPath);
|
|
128
|
+
} catch (err) {
|
|
129
|
+
res.status(500).json({ error: err.message });
|
|
130
|
+
}
|
|
131
|
+
});
|
|
132
|
+
router.post("/api/setup/check-path", (req, res) => {
|
|
133
|
+
const { path: dirPath } = req.body;
|
|
134
|
+
if (!dirPath) {
|
|
135
|
+
res.status(400).json({ error: "path is required" });
|
|
136
|
+
return;
|
|
137
|
+
}
|
|
138
|
+
try {
|
|
139
|
+
const exists = fs.existsSync(dirPath);
|
|
140
|
+
const isDirectory = exists && fs.statSync(dirPath).isDirectory();
|
|
141
|
+
const isGitRepo = exists && fs.existsSync(dirPath + "/.git");
|
|
142
|
+
res.json({ exists, isDirectory, isGitRepo });
|
|
143
|
+
} catch (err) {
|
|
144
|
+
res.json({ exists: false, isDirectory: false, isGitRepo: false, error: err.message });
|
|
145
|
+
}
|
|
146
|
+
});
|
|
147
|
+
return router;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
export {
|
|
151
|
+
createSetupRouter
|
|
152
|
+
};
|
|
153
|
+
//# sourceMappingURL=chunk-I3T573SU.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/web/routes/setup.ts"],"sourcesContent":["import { Router, Request, Response } from 'express';\nimport fs from 'node:fs';\nimport { DependencyChecker } from '../../cli/setup/DependencyChecker.js';\nimport { ConfigGenerator, SetupConfig } from '../../cli/setup/ConfigGenerator.js';\n\nexport interface SetupRouterDeps {\n configPath?: string;\n onComplete?: (configPath: string) => void;\n serviceMode?: boolean;\n}\n\nexport function createSetupRouter(deps: SetupRouterDeps = {}): Router {\n const router = Router();\n const checker = new DependencyChecker();\n\n router.get('/api/setup/status', (_req: Request, res: Response) => {\n const initialized = deps.serviceMode || ConfigGenerator.isInitialized(deps.configPath);\n res.json({\n initialized,\n configPath: deps.configPath ?? ConfigGenerator.getGlobalConfigPath(),\n globalConfigDir: ConfigGenerator.getGlobalConfigDir(),\n });\n });\n\n router.get('/api/setup/check-env', async (_req: Request, res: Response) => {\n try {\n const aiMode = ((_req.query.aiMode as string) || undefined);\n const results = await checker.checkAll(aiMode);\n res.json({ dependencies: results });\n } catch (err) {\n res.status(500).json({ error: (err as Error).message });\n }\n });\n\n router.post('/api/setup/validate-token', async (req: Request, res: Response) => {\n const { apiUrl, token } = req.body as { apiUrl?: string; token?: string };\n if (!apiUrl || !token) {\n res.status(400).json({ error: 'apiUrl and token are required' });\n return;\n }\n try {\n const url = `${apiUrl.replace(/\\/$/, '')}/api/v3/user`;\n const resp = await fetch(url, {\n headers: { 'PRIVATE-TOKEN': token },\n signal: AbortSignal.timeout(10000),\n });\n if (resp.ok) {\n const user = (await resp.json()) as { username?: string; name?: string };\n res.json({ valid: true, user: { username: user.username, name: user.name } });\n } else {\n res.json({ valid: false, error: `HTTP ${resp.status}: ${resp.statusText}` });\n }\n } catch (err) {\n res.json({ valid: false, error: (err as Error).message });\n }\n });\n\n router.post('/api/setup/validate-repo', async (req: Request, res: Response) => {\n const { apiUrl, token, projectPath } = req.body as {\n apiUrl?: string; token?: string; projectPath?: string;\n };\n if (!apiUrl || !token || !projectPath) {\n res.status(400).json({ error: 'apiUrl, token, and projectPath are required' });\n return;\n }\n try {\n const encoded = encodeURIComponent(projectPath);\n const url = `${apiUrl.replace(/\\/$/, '')}/api/v3/projects/${encoded}`;\n const resp = await fetch(url, {\n headers: { 'PRIVATE-TOKEN': token },\n signal: AbortSignal.timeout(10000),\n });\n if (resp.ok) {\n const project = (await resp.json()) as {\n id?: number; name?: string; path_with_namespace?: string;\n };\n res.json({ valid: true, project });\n } else {\n res.json({ valid: false, error: `HTTP ${resp.status}: ${resp.statusText}` });\n }\n } catch (err) {\n res.json({ valid: false, error: (err as Error).message });\n }\n });\n\n router.post('/api/setup/install-dep', (req: Request, res: Response) => {\n const { name } = req.body as { name?: string };\n if (!name) {\n res.status(400).json({ error: 'name is required' });\n return;\n }\n\n res.setHeader('Content-Type', 'text/event-stream');\n res.setHeader('Cache-Control', 'no-cache');\n res.setHeader('Connection', 'keep-alive');\n res.flushHeaders();\n\n (async () => {\n try {\n for await (const line of checker.install(name)) {\n res.write(`data: ${JSON.stringify({ line })}\\n\\n`);\n }\n res.write(`data: ${JSON.stringify({ done: true })}\\n\\n`);\n } catch (err) {\n res.write(`data: ${JSON.stringify({ error: (err as Error).message })}\\n\\n`);\n } finally {\n res.end();\n }\n })();\n\n req.on('close', () => {\n res.end();\n });\n });\n\n router.post('/api/setup/generate-preview', (req: Request, res: Response) => {\n const config = req.body as SetupConfig;\n try {\n const preview = ConfigGenerator.generate(config);\n res.json({ preview });\n } catch (err) {\n res.status(400).json({ error: (err as Error).message });\n }\n });\n\n router.post('/api/setup/complete', (req: Request, res: Response) => {\n const { config, outputPath } = req.body as {\n config?: SetupConfig;\n outputPath?: string;\n };\n if (!config) {\n res.status(400).json({ error: 'config is required' });\n return;\n }\n try {\n const targetPath = outputPath ?? deps.configPath;\n const savedPath = ConfigGenerator.write(config, targetPath);\n res.json({ success: true, configPath: savedPath });\n deps.onComplete?.(savedPath);\n } catch (err) {\n res.status(500).json({ error: (err as Error).message });\n }\n });\n\n router.post('/api/setup/check-path', (req: Request, res: Response) => {\n const { path: dirPath } = req.body as { path?: string };\n if (!dirPath) {\n res.status(400).json({ error: 'path is required' });\n return;\n }\n try {\n const exists = fs.existsSync(dirPath);\n const isDirectory = exists && fs.statSync(dirPath).isDirectory();\n const isGitRepo = exists && fs.existsSync(dirPath + '/.git');\n res.json({ exists, isDirectory, isGitRepo });\n } catch (err) {\n res.json({ exists: false, isDirectory: false, isGitRepo: false, error: (err as Error).message });\n }\n });\n\n return router;\n}\n"],"mappings":"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","names":[]}
|