@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.
Files changed (161) hide show
  1. package/bin/issue-auto-finish.js +2 -0
  2. package/dist/ai-runner/AIRunner.d.ts +27 -0
  3. package/dist/ai-runner/AIRunner.d.ts.map +1 -0
  4. package/dist/ai-runner/BaseAIRunner.d.ts +19 -0
  5. package/dist/ai-runner/BaseAIRunner.d.ts.map +1 -0
  6. package/dist/ai-runner/ClaudeInternalRunner.d.ts +13 -0
  7. package/dist/ai-runner/ClaudeInternalRunner.d.ts.map +1 -0
  8. package/dist/ai-runner/CodebuddyRunner.d.ts +13 -0
  9. package/dist/ai-runner/CodebuddyRunner.d.ts.map +1 -0
  10. package/dist/ai-runner/CursorAgentRunner.d.ts +13 -0
  11. package/dist/ai-runner/CursorAgentRunner.d.ts.map +1 -0
  12. package/dist/ai-runner/index.d.ts +15 -0
  13. package/dist/ai-runner/index.d.ts.map +1 -0
  14. package/dist/chunk-HCHEFK4Z.js +80 -0
  15. package/dist/chunk-HCHEFK4Z.js.map +1 -0
  16. package/dist/chunk-I3T573SU.js +153 -0
  17. package/dist/chunk-I3T573SU.js.map +1 -0
  18. package/dist/chunk-IDUKWCC2.js +1995 -0
  19. package/dist/chunk-IDUKWCC2.js.map +1 -0
  20. package/dist/chunk-OWVT3Z34.js +770 -0
  21. package/dist/chunk-OWVT3Z34.js.map +1 -0
  22. package/dist/chunk-RIUI4ROA.js +180 -0
  23. package/dist/chunk-RIUI4ROA.js.map +1 -0
  24. package/dist/chunk-TBIEB3JY.js +3295 -0
  25. package/dist/chunk-TBIEB3JY.js.map +1 -0
  26. package/dist/cli/commands/doctor.d.ts +2 -0
  27. package/dist/cli/commands/doctor.d.ts.map +1 -0
  28. package/dist/cli/commands/init.d.ts +6 -0
  29. package/dist/cli/commands/init.d.ts.map +1 -0
  30. package/dist/cli/commands/start.d.ts +5 -0
  31. package/dist/cli/commands/start.d.ts.map +1 -0
  32. package/dist/cli/index.d.ts +3 -0
  33. package/dist/cli/index.d.ts.map +1 -0
  34. package/dist/cli/setup/ConfigGenerator.d.ts +44 -0
  35. package/dist/cli/setup/ConfigGenerator.d.ts.map +1 -0
  36. package/dist/cli/setup/DependencyChecker.d.ts +21 -0
  37. package/dist/cli/setup/DependencyChecker.d.ts.map +1 -0
  38. package/dist/cli.js +73 -0
  39. package/dist/cli.js.map +1 -0
  40. package/dist/clients/GongfengClient.d.ts +86 -0
  41. package/dist/clients/GongfengClient.d.ts.map +1 -0
  42. package/dist/config.d.ts +92 -0
  43. package/dist/config.d.ts.map +1 -0
  44. package/dist/deploy/DevServerManager.d.ts +21 -0
  45. package/dist/deploy/DevServerManager.d.ts.map +1 -0
  46. package/dist/deploy/PortAllocator.d.ts +20 -0
  47. package/dist/deploy/PortAllocator.d.ts.map +1 -0
  48. package/dist/deploy/index.d.ts +3 -0
  49. package/dist/deploy/index.d.ts.map +1 -0
  50. package/dist/doctor-B26Q6JWI.js +33 -0
  51. package/dist/doctor-B26Q6JWI.js.map +1 -0
  52. package/dist/e2e/E2eSettings.d.ts +6 -0
  53. package/dist/e2e/E2eSettings.d.ts.map +1 -0
  54. package/dist/e2e/ScreenshotCollector.d.ts +11 -0
  55. package/dist/e2e/ScreenshotCollector.d.ts.map +1 -0
  56. package/dist/e2e/ScreenshotPublisher.d.ts +16 -0
  57. package/dist/e2e/ScreenshotPublisher.d.ts.map +1 -0
  58. package/dist/events/EventBus.d.ts +14 -0
  59. package/dist/events/EventBus.d.ts.map +1 -0
  60. package/dist/git/GitOperations.d.ts +30 -0
  61. package/dist/git/GitOperations.d.ts.map +1 -0
  62. package/dist/git/WorktreeContext.d.ts +9 -0
  63. package/dist/git/WorktreeContext.d.ts.map +1 -0
  64. package/dist/i18n/index.d.ts +8 -0
  65. package/dist/i18n/index.d.ts.map +1 -0
  66. package/dist/i18n/locales/en.d.ts +2 -0
  67. package/dist/i18n/locales/en.d.ts.map +1 -0
  68. package/dist/i18n/locales/zh-CN.d.ts +2 -0
  69. package/dist/i18n/locales/zh-CN.d.ts.map +1 -0
  70. package/dist/index.d.ts +2 -0
  71. package/dist/index.d.ts.map +1 -0
  72. package/dist/index.js +12 -0
  73. package/dist/index.js.map +1 -0
  74. package/dist/init-L3VIWCOV.js +65 -0
  75. package/dist/init-L3VIWCOV.js.map +1 -0
  76. package/dist/lib.d.ts +26 -0
  77. package/dist/lib.d.ts.map +1 -0
  78. package/dist/lib.js +50 -0
  79. package/dist/lib.js.map +1 -0
  80. package/dist/logger.d.ts +15 -0
  81. package/dist/logger.d.ts.map +1 -0
  82. package/dist/notesync/NoteSyncSettings.d.ts +12 -0
  83. package/dist/notesync/NoteSyncSettings.d.ts.map +1 -0
  84. package/dist/orchestrator/PipelineOrchestrator.d.ts +53 -0
  85. package/dist/orchestrator/PipelineOrchestrator.d.ts.map +1 -0
  86. package/dist/persistence/PlanPersistence.d.ts +37 -0
  87. package/dist/persistence/PlanPersistence.d.ts.map +1 -0
  88. package/dist/phases/AnalysisPhase.d.ts +13 -0
  89. package/dist/phases/AnalysisPhase.d.ts.map +1 -0
  90. package/dist/phases/BasePhase.d.ts +44 -0
  91. package/dist/phases/BasePhase.d.ts.map +1 -0
  92. package/dist/phases/BuildPhase.d.ts +9 -0
  93. package/dist/phases/BuildPhase.d.ts.map +1 -0
  94. package/dist/phases/DesignPhase.d.ts +13 -0
  95. package/dist/phases/DesignPhase.d.ts.map +1 -0
  96. package/dist/phases/ImplementPhase.d.ts +9 -0
  97. package/dist/phases/ImplementPhase.d.ts.map +1 -0
  98. package/dist/phases/PhaseFactory.d.ts +13 -0
  99. package/dist/phases/PhaseFactory.d.ts.map +1 -0
  100. package/dist/phases/PlanPhase.d.ts +14 -0
  101. package/dist/phases/PlanPhase.d.ts.map +1 -0
  102. package/dist/phases/VerifyPhase.d.ts +13 -0
  103. package/dist/phases/VerifyPhase.d.ts.map +1 -0
  104. package/dist/pipeline/PipelineDefinition.d.ts +40 -0
  105. package/dist/pipeline/PipelineDefinition.d.ts.map +1 -0
  106. package/dist/poller/IssuePoller.d.ts +26 -0
  107. package/dist/poller/IssuePoller.d.ts.map +1 -0
  108. package/dist/prompts/brainstorm-templates.d.ts +4 -0
  109. package/dist/prompts/brainstorm-templates.d.ts.map +1 -0
  110. package/dist/prompts/templates.d.ts +27 -0
  111. package/dist/prompts/templates.d.ts.map +1 -0
  112. package/dist/rules/RuleResolver.d.ts +13 -0
  113. package/dist/rules/RuleResolver.d.ts.map +1 -0
  114. package/dist/rules/index.d.ts +3 -0
  115. package/dist/rules/index.d.ts.map +1 -0
  116. package/dist/run.d.ts +2 -0
  117. package/dist/run.d.ts.map +1 -0
  118. package/dist/run.js +15 -0
  119. package/dist/run.js.map +1 -0
  120. package/dist/services/BrainstormService.d.ts +39 -0
  121. package/dist/services/BrainstormService.d.ts.map +1 -0
  122. package/dist/start-TVN4SS6E.js +25 -0
  123. package/dist/start-TVN4SS6E.js.map +1 -0
  124. package/dist/supplement/SupplementStore.d.ts +21 -0
  125. package/dist/supplement/SupplementStore.d.ts.map +1 -0
  126. package/dist/tracker/IssueState.d.ts +63 -0
  127. package/dist/tracker/IssueState.d.ts.map +1 -0
  128. package/dist/tracker/IssueTracker.d.ts +29 -0
  129. package/dist/tracker/IssueTracker.d.ts.map +1 -0
  130. package/dist/utils/AsyncMutex.d.ts +14 -0
  131. package/dist/utils/AsyncMutex.d.ts.map +1 -0
  132. package/dist/utils/MergeRequestHelper.d.ts +10 -0
  133. package/dist/utils/MergeRequestHelper.d.ts.map +1 -0
  134. package/dist/web/AgentLogStore.d.ts +22 -0
  135. package/dist/web/AgentLogStore.d.ts.map +1 -0
  136. package/dist/web/WebServer.d.ts +26 -0
  137. package/dist/web/WebServer.d.ts.map +1 -0
  138. package/dist/web/routes/api.d.ts +20 -0
  139. package/dist/web/routes/api.d.ts.map +1 -0
  140. package/dist/web/routes/brainstorm.d.ts +9 -0
  141. package/dist/web/routes/brainstorm.d.ts.map +1 -0
  142. package/dist/web/routes/setup.d.ts +8 -0
  143. package/dist/web/routes/setup.d.ts.map +1 -0
  144. package/dist/webhook/CommandExecutor.d.ts +45 -0
  145. package/dist/webhook/CommandExecutor.d.ts.map +1 -0
  146. package/dist/webhook/CommandParser.d.ts +24 -0
  147. package/dist/webhook/CommandParser.d.ts.map +1 -0
  148. package/dist/webhook/IntentRecognizer.d.ts +35 -0
  149. package/dist/webhook/IntentRecognizer.d.ts.map +1 -0
  150. package/dist/webhook/NoteDeduplicator.d.ts +18 -0
  151. package/dist/webhook/NoteDeduplicator.d.ts.map +1 -0
  152. package/dist/webhook/WebhookHandler.d.ts +30 -0
  153. package/dist/webhook/WebhookHandler.d.ts.map +1 -0
  154. package/dist/webhook/WebhookServer.d.ts +21 -0
  155. package/dist/webhook/WebhookServer.d.ts.map +1 -0
  156. package/dist/webhook/index.d.ts +12 -0
  157. package/dist/webhook/index.d.ts.map +1 -0
  158. package/package.json +82 -0
  159. package/src/web/frontend/dist/assets/index-CQdlU9PE.js +65 -0
  160. package/src/web/frontend/dist/assets/index-CgMEkyZJ.css +1 -0
  161. package/src/web/frontend/dist/index.html +13 -0
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ import '../dist/cli.js';
@@ -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":[]}