@xdevops/issue-auto-finish 1.0.69 → 1.0.70

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 (79) hide show
  1. package/dist/{LockNote-G32LJFVG.js → LockNote-3ATNJJ5K.js} +3 -3
  2. package/dist/{ai-runner-DHH74B4O.js → ai-runner-PJYBQ6LO.js} +2 -2
  3. package/dist/{analyze-STQYS7K4.js → analyze-4MTNYZNK.js} +3 -3
  4. package/dist/{braindump-UOCBJGPF.js → braindump-MBIFTOSK.js} +6 -6
  5. package/dist/{chunk-VFTESEYG.js → chunk-E4Q6KZ23.js} +4 -3
  6. package/dist/chunk-E4Q6KZ23.js.map +1 -0
  7. package/dist/{chunk-EYGZJYIO.js → chunk-ECAOTSHA.js} +2 -2
  8. package/dist/{chunk-RPL2X7KB.js → chunk-KZ6CKYWK.js} +6 -6
  9. package/dist/{chunk-XCHQMDNZ.js → chunk-LD222Y4U.js} +5 -3
  10. package/dist/chunk-LD222Y4U.js.map +1 -0
  11. package/dist/{chunk-JUXD7ZBK.js → chunk-M22VPSJ2.js} +2 -2
  12. package/dist/{chunk-DEM52VS6.js → chunk-OPWGIRIO.js} +167 -36
  13. package/dist/chunk-OPWGIRIO.js.map +1 -0
  14. package/dist/{chunk-2QHSU5YR.js → chunk-QCVJKH52.js} +9 -1
  15. package/dist/chunk-QCVJKH52.js.map +1 -0
  16. package/dist/{chunk-Z75EDBWI.js → chunk-RCAARXN2.js} +3 -3
  17. package/dist/{chunk-TASSKSJ3.js → chunk-WHTYES3V.js} +9 -1
  18. package/dist/{chunk-TASSKSJ3.js.map → chunk-WHTYES3V.js.map} +1 -1
  19. package/dist/{chunk-TGVBX5CF.js → chunk-WVZQMM3R.js} +3 -2
  20. package/dist/chunk-WVZQMM3R.js.map +1 -0
  21. package/dist/{chunk-VJD6TIKY.js → chunk-ZBEFRFEW.js} +80 -7
  22. package/dist/chunk-ZBEFRFEW.js.map +1 -0
  23. package/dist/cli.js +8 -8
  24. package/dist/{config-SONIIB3D.js → config-OQ6ZBQSM.js} +3 -3
  25. package/dist/config-schema.d.ts +2 -0
  26. package/dist/config-schema.d.ts.map +1 -1
  27. package/dist/config.d.ts +2 -0
  28. package/dist/config.d.ts.map +1 -1
  29. package/dist/coordination/LockNote.d.ts +1 -1
  30. package/dist/coordination/LockNote.d.ts.map +1 -1
  31. package/dist/deploy/PreviewReaper.d.ts +38 -0
  32. package/dist/deploy/PreviewReaper.d.ts.map +1 -0
  33. package/dist/deploy/index.d.ts +1 -0
  34. package/dist/deploy/index.d.ts.map +1 -1
  35. package/dist/{doctor-JMAZCEHC.js → doctor-7VTUPXAZ.js} +3 -3
  36. package/dist/events/EventBus.d.ts +1 -1
  37. package/dist/events/EventBus.d.ts.map +1 -1
  38. package/dist/i18n/locales/en.d.ts.map +1 -1
  39. package/dist/i18n/locales/zh-CN.d.ts.map +1 -1
  40. package/dist/index.d.ts.map +1 -1
  41. package/dist/index.js +10 -10
  42. package/dist/{init-G75RO7SX.js → init-7TMZG72C.js} +6 -6
  43. package/dist/lib.js +5 -5
  44. package/dist/lifecycle/ActionLifecycleManager.d.ts.map +1 -1
  45. package/dist/orchestrator/PipelineOrchestrator.d.ts +3 -0
  46. package/dist/orchestrator/PipelineOrchestrator.d.ts.map +1 -1
  47. package/dist/{restart-2K7RHMYC.js → restart-AK2XV3AR.js} +4 -4
  48. package/dist/run.js +10 -10
  49. package/dist/{start-NVOOEF2A.js → start-2OIZUMWY.js} +4 -4
  50. package/dist/tracker/ExecutableTask.d.ts.map +1 -1
  51. package/dist/tracker/IssueState.d.ts +2 -1
  52. package/dist/tracker/IssueState.d.ts.map +1 -1
  53. package/dist/web/WebServer.d.ts +1 -0
  54. package/dist/web/WebServer.d.ts.map +1 -1
  55. package/dist/web/routes/api.d.ts +2 -0
  56. package/dist/web/routes/api.d.ts.map +1 -1
  57. package/package.json +1 -1
  58. package/src/web/frontend/dist/assets/{index-B8u7dV-R.js → index-BH2GGJHH.js} +54 -54
  59. package/src/web/frontend/dist/assets/{index-BQKHLcAS.css → index-CPNbFsHB.css} +1 -1
  60. package/src/web/frontend/dist/index.html +2 -2
  61. package/dist/chunk-2QHSU5YR.js.map +0 -1
  62. package/dist/chunk-DEM52VS6.js.map +0 -1
  63. package/dist/chunk-TGVBX5CF.js.map +0 -1
  64. package/dist/chunk-VFTESEYG.js.map +0 -1
  65. package/dist/chunk-VJD6TIKY.js.map +0 -1
  66. package/dist/chunk-XCHQMDNZ.js.map +0 -1
  67. /package/dist/{LockNote-G32LJFVG.js.map → LockNote-3ATNJJ5K.js.map} +0 -0
  68. /package/dist/{ai-runner-DHH74B4O.js.map → ai-runner-PJYBQ6LO.js.map} +0 -0
  69. /package/dist/{analyze-STQYS7K4.js.map → analyze-4MTNYZNK.js.map} +0 -0
  70. /package/dist/{braindump-UOCBJGPF.js.map → braindump-MBIFTOSK.js.map} +0 -0
  71. /package/dist/{chunk-EYGZJYIO.js.map → chunk-ECAOTSHA.js.map} +0 -0
  72. /package/dist/{chunk-RPL2X7KB.js.map → chunk-KZ6CKYWK.js.map} +0 -0
  73. /package/dist/{chunk-JUXD7ZBK.js.map → chunk-M22VPSJ2.js.map} +0 -0
  74. /package/dist/{chunk-Z75EDBWI.js.map → chunk-RCAARXN2.js.map} +0 -0
  75. /package/dist/{config-SONIIB3D.js.map → config-OQ6ZBQSM.js.map} +0 -0
  76. /package/dist/{doctor-JMAZCEHC.js.map → doctor-7VTUPXAZ.js.map} +0 -0
  77. /package/dist/{init-G75RO7SX.js.map → init-7TMZG72C.js.map} +0 -0
  78. /package/dist/{restart-2K7RHMYC.js.map → restart-AK2XV3AR.js.map} +0 -0
  79. /package/dist/{start-NVOOEF2A.js.map → start-2OIZUMWY.js.map} +0 -0
@@ -5,8 +5,8 @@ import {
5
5
  findLockNote,
6
6
  isLockStale,
7
7
  parseLockNote
8
- } from "./chunk-VFTESEYG.js";
9
- import "./chunk-2QHSU5YR.js";
8
+ } from "./chunk-E4Q6KZ23.js";
9
+ import "./chunk-QCVJKH52.js";
10
10
  export {
11
11
  buildLockNoteBody,
12
12
  buildReleaseNoteBody,
@@ -15,4 +15,4 @@ export {
15
15
  isLockStale,
16
16
  parseLockNote
17
17
  };
18
- //# sourceMappingURL=LockNote-G32LJFVG.js.map
18
+ //# sourceMappingURL=LockNote-3ATNJJ5K.js.map
@@ -15,7 +15,7 @@ import {
15
15
  resolveModelForRunner,
16
16
  resolveRunnerMode,
17
17
  validateRunnerRegistry
18
- } from "./chunk-TASSKSJ3.js";
18
+ } from "./chunk-WHTYES3V.js";
19
19
  import "./chunk-GF2RRYHB.js";
20
20
  export {
21
21
  BaseAIRunner,
@@ -34,4 +34,4 @@ export {
34
34
  resolveRunnerMode,
35
35
  validateRunnerRegistry
36
36
  };
37
- //# sourceMappingURL=ai-runner-DHH74B4O.js.map
37
+ //# sourceMappingURL=ai-runner-PJYBQ6LO.js.map
@@ -1,6 +1,6 @@
1
1
  import {
2
2
  loadConfig
3
- } from "./chunk-XCHQMDNZ.js";
3
+ } from "./chunk-LD222Y4U.js";
4
4
  import {
5
5
  analyze
6
6
  } from "./chunk-B7TVVODN.js";
@@ -11,7 +11,7 @@ import "./chunk-DAX3FD2O.js";
11
11
  import "./chunk-DADQSKPL.js";
12
12
  import {
13
13
  createAIRunner
14
- } from "./chunk-TASSKSJ3.js";
14
+ } from "./chunk-WHTYES3V.js";
15
15
  import "./chunk-GF2RRYHB.js";
16
16
 
17
17
  // src/cli/commands/analyze.ts
@@ -72,4 +72,4 @@ async function analyzeCommand(opts) {
72
72
  export {
73
73
  analyzeCommand
74
74
  };
75
- //# sourceMappingURL=analyze-STQYS7K4.js.map
75
+ //# sourceMappingURL=analyze-4MTNYZNK.js.map
@@ -9,23 +9,23 @@ import {
9
9
  import {
10
10
  BraindumpOrchestrator,
11
11
  BraindumpTracker
12
- } from "./chunk-Z75EDBWI.js";
12
+ } from "./chunk-RCAARXN2.js";
13
13
  import {
14
14
  AsyncMutex,
15
15
  GitOperations
16
- } from "./chunk-TGVBX5CF.js";
16
+ } from "./chunk-WVZQMM3R.js";
17
17
  import "./chunk-ACVOOHAR.js";
18
- import "./chunk-2QHSU5YR.js";
18
+ import "./chunk-QCVJKH52.js";
19
19
  import {
20
20
  loadConfig
21
- } from "./chunk-XCHQMDNZ.js";
21
+ } from "./chunk-LD222Y4U.js";
22
22
  import "./chunk-B7TVVODN.js";
23
23
  import "./chunk-TN2SYADO.js";
24
24
  import "./chunk-DAX3FD2O.js";
25
25
  import "./chunk-DADQSKPL.js";
26
26
  import {
27
27
  createAIRunner
28
- } from "./chunk-TASSKSJ3.js";
28
+ } from "./chunk-WHTYES3V.js";
29
29
  import "./chunk-GF2RRYHB.js";
30
30
 
31
31
  // src/cli/commands/braindump.ts
@@ -204,4 +204,4 @@ ${bold("\u786E\u8BA4\u6267\u884C\uFF1F")} (${green("y")}=\u786E\u8BA4 / ${red("q
204
204
  export {
205
205
  braindumpCommand
206
206
  };
207
- //# sourceMappingURL=braindump-UOCBJGPF.js.map
207
+ //# sourceMappingURL=braindump-MBIFTOSK.js.map
@@ -1,6 +1,6 @@
1
1
  import {
2
2
  t
3
- } from "./chunk-2QHSU5YR.js";
3
+ } from "./chunk-QCVJKH52.js";
4
4
 
5
5
  // src/coordination/LockNote.ts
6
6
  var LOCK_NOTE_REGEX = /<!-- iaf-lock:(\{.*?\}) -->/;
@@ -15,7 +15,8 @@ function buildReleaseNoteBody(reason) {
15
15
  completed: t("lock.completedMessage"),
16
16
  failed: t("lock.failedMessage"),
17
17
  cancelled: t("lock.cancelledMessage"),
18
- "crash-recovery": t("lock.crashRecoveryMessage")
18
+ "crash-recovery": t("lock.crashRecoveryMessage"),
19
+ deployed: t("lock.deployedMessage")
19
20
  };
20
21
  return `${messages[reason]}
21
22
 
@@ -65,4 +66,4 @@ export {
65
66
  findAllLockNotes,
66
67
  isLockStale
67
68
  };
68
- //# sourceMappingURL=chunk-VFTESEYG.js.map
69
+ //# sourceMappingURL=chunk-E4Q6KZ23.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/coordination/LockNote.ts"],"sourcesContent":["import type { GongfengNote } from '../clients/GongfengClient.js';\nimport { t } from '../i18n/index.js';\n\n/**\n * 锁评论协议。\n *\n * 锁评论格式(嵌入在可见内容中的 HTML 注释):\n * ```\n * 🚀 **自动处理开始**\n *\n * 已检测到 `auto-finish` 标签,开始自动分析和实施。\n *\n * <!-- iaf-lock:{\"nodeId\":\"devbox-a3f1c902\",\"ts\":1711100000000} -->\n * ```\n *\n * 向后兼容旧格式(纯 HTML 注释,无可见内容)。\n * 不带 AGENT_NOTE_MARKER,不会被 cleanupAgentNotes 清理。\n */\n\nconst LOCK_NOTE_REGEX = /<!-- iaf-lock:(\\{.*?\\}) -->/;\n\nexport interface LockPayload {\n nodeId: string;\n ts: number;\n phase?: string;\n}\n\nexport interface LockNoteInfo {\n noteId: number;\n payload: LockPayload;\n}\n\n/** 释放锁时的原因 */\nexport type ReleaseReason = 'completed' | 'failed' | 'cancelled' | 'crash-recovery' | 'deployed';\n\n/** 构建锁评论正文,可附带可见消息 */\nexport function buildLockNoteBody(payload: LockPayload, message?: string): string {\n const lockTag = `<!-- iaf-lock:${JSON.stringify(payload)} -->`;\n return message ? `${message}\\n\\n${lockTag}` : lockTag;\n}\n\n/** 构建释放后的评论正文(可见状态 + 释放标记) */\nexport function buildReleaseNoteBody(reason: ReleaseReason): string {\n const messages: Record<ReleaseReason, string> = {\n completed: t('lock.completedMessage'),\n failed: t('lock.failedMessage'),\n cancelled: t('lock.cancelledMessage'),\n 'crash-recovery': t('lock.crashRecoveryMessage'),\n deployed: t('lock.deployedMessage'),\n };\n return `${messages[reason]}\\n\\n<!-- iaf-lock-released -->`;\n}\n\n/** 从评论正文中解析锁载荷,非锁评论返回 null。支持嵌入在可见内容中的锁标记。 */\nexport function parseLockNote(body: string): LockPayload | null {\n const match = body.match(LOCK_NOTE_REGEX);\n if (!match) return null;\n try {\n const payload = JSON.parse(match[1]) as LockPayload;\n if (typeof payload.nodeId !== 'string' || typeof payload.ts !== 'number') {\n return null;\n }\n return payload;\n } catch {\n return null;\n }\n}\n\n/** 在评论列表中查找第一条锁评论 */\nexport function findLockNote(notes: GongfengNote[]): LockNoteInfo | null {\n for (const note of notes) {\n const payload = parseLockNote(note.body);\n if (payload) {\n return { noteId: note.id, payload };\n }\n }\n return null;\n}\n\n/** 在评论列表中查找所有锁评论 */\nexport function findAllLockNotes(notes: GongfengNote[]): LockNoteInfo[] {\n const result: LockNoteInfo[] = [];\n for (const note of notes) {\n const payload = parseLockNote(note.body);\n if (payload) {\n result.push({ noteId: note.id, payload });\n }\n }\n return result;\n}\n\n/** 判断锁评论是否过期 */\nexport function isLockStale(payload: LockPayload, staleMs: number): boolean {\n return Date.now() - payload.ts > staleMs;\n}\n"],"mappings":";;;;;AAmBA,IAAM,kBAAkB;AAiBjB,SAAS,kBAAkB,SAAsB,SAA0B;AAChF,QAAM,UAAU,iBAAiB,KAAK,UAAU,OAAO,CAAC;AACxD,SAAO,UAAU,GAAG,OAAO;AAAA;AAAA,EAAO,OAAO,KAAK;AAChD;AAGO,SAAS,qBAAqB,QAA+B;AAClE,QAAM,WAA0C;AAAA,IAC9C,WAAW,EAAE,uBAAuB;AAAA,IACpC,QAAQ,EAAE,oBAAoB;AAAA,IAC9B,WAAW,EAAE,uBAAuB;AAAA,IACpC,kBAAkB,EAAE,2BAA2B;AAAA,IAC/C,UAAU,EAAE,sBAAsB;AAAA,EACpC;AACA,SAAO,GAAG,SAAS,MAAM,CAAC;AAAA;AAAA;AAC5B;AAGO,SAAS,cAAc,MAAkC;AAC9D,QAAM,QAAQ,KAAK,MAAM,eAAe;AACxC,MAAI,CAAC,MAAO,QAAO;AACnB,MAAI;AACF,UAAM,UAAU,KAAK,MAAM,MAAM,CAAC,CAAC;AACnC,QAAI,OAAO,QAAQ,WAAW,YAAY,OAAO,QAAQ,OAAO,UAAU;AACxE,aAAO;AAAA,IACT;AACA,WAAO;AAAA,EACT,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAGO,SAAS,aAAa,OAA4C;AACvE,aAAW,QAAQ,OAAO;AACxB,UAAM,UAAU,cAAc,KAAK,IAAI;AACvC,QAAI,SAAS;AACX,aAAO,EAAE,QAAQ,KAAK,IAAI,QAAQ;AAAA,IACpC;AAAA,EACF;AACA,SAAO;AACT;AAGO,SAAS,iBAAiB,OAAuC;AACtE,QAAM,SAAyB,CAAC;AAChC,aAAW,QAAQ,OAAO;AACxB,UAAM,UAAU,cAAc,KAAK,IAAI;AACvC,QAAI,SAAS;AACX,aAAO,KAAK,EAAE,QAAQ,KAAK,IAAI,QAAQ,CAAC;AAAA,IAC1C;AAAA,EACF;AACA,SAAO;AACT;AAGO,SAAS,YAAY,SAAsB,SAA0B;AAC1E,SAAO,KAAK,IAAI,IAAI,QAAQ,KAAK;AACnC;","names":[]}
@@ -5,7 +5,7 @@ import {
5
5
  } from "./chunk-GXRFUJTN.js";
6
6
  import {
7
7
  resolveConfigFilePath
8
- } from "./chunk-XCHQMDNZ.js";
8
+ } from "./chunk-LD222Y4U.js";
9
9
  import {
10
10
  ensureDir,
11
11
  resolveLogsDir
@@ -89,4 +89,4 @@ async function startDaemon(configPath) {
89
89
  export {
90
90
  startCommand
91
91
  };
92
- //# sourceMappingURL=chunk-EYGZJYIO.js.map
92
+ //# sourceMappingURL=chunk-ECAOTSHA.js.map
@@ -1,13 +1,13 @@
1
1
  import {
2
2
  DependencyChecker
3
- } from "./chunk-JUXD7ZBK.js";
3
+ } from "./chunk-M22VPSJ2.js";
4
4
  import {
5
5
  getProjectKnowledge
6
6
  } from "./chunk-ACVOOHAR.js";
7
7
  import {
8
8
  getLocalIP,
9
9
  resolveConfigFilePath
10
- } from "./chunk-XCHQMDNZ.js";
10
+ } from "./chunk-LD222Y4U.js";
11
11
  import {
12
12
  analyze,
13
13
  collectStaticInfo
@@ -429,7 +429,7 @@ var E2ESetupRunner = class {
429
429
  const step = "reload";
430
430
  yield { step, status: "running", message: "Reloading configuration..." };
431
431
  try {
432
- const { reloadConfig } = await import("./config-SONIIB3D.js");
432
+ const { reloadConfig } = await import("./config-OQ6ZBQSM.js");
433
433
  reloadConfig();
434
434
  yield { step, status: "done", message: "Configuration reloaded" };
435
435
  } catch (err) {
@@ -701,8 +701,8 @@ function createSetupRouter(deps = {}) {
701
701
  sse.write({ step: "collected", message: "Static info collected" });
702
702
  sse.write({ step: "analyzing", message: "Running AI analysis..." });
703
703
  try {
704
- const { loadConfig } = await import("./config-SONIIB3D.js");
705
- const { createAIRunner } = await import("./ai-runner-DHH74B4O.js");
704
+ const { loadConfig } = await import("./config-OQ6ZBQSM.js");
705
+ const { createAIRunner } = await import("./ai-runner-PJYBQ6LO.js");
706
706
  const config = loadConfig();
707
707
  const runner = createAIRunner(config.ai);
708
708
  const knowledge = await analyze({ workDir, aiRunner: runner });
@@ -782,4 +782,4 @@ function createSetupRouter(deps = {}) {
782
782
  export {
783
783
  createSetupRouter
784
784
  };
785
- //# sourceMappingURL=chunk-RPL2X7KB.js.map
785
+ //# sourceMappingURL=chunk-KZ6CKYWK.js.map
@@ -4,7 +4,7 @@ import {
4
4
  import {
5
5
  getBinaryEnvKey,
6
6
  getDefaultBinary
7
- } from "./chunk-TASSKSJ3.js";
7
+ } from "./chunk-WHTYES3V.js";
8
8
 
9
9
  // src/config.ts
10
10
  import { config as loadDotenv } from "dotenv";
@@ -116,6 +116,7 @@ var envSchema = z.object({
116
116
  PREVIEW_HOST: z.string().optional().default(""),
117
117
  PREVIEW_TTL_MS: envMs(String(24 * 60 * 60 * 1e3)),
118
118
  PREVIEW_KEEP_AFTER_COMPLETE: envBoolean("true"),
119
+ PREVIEW_REAP_INTERVAL_MS: envMs("300000"),
119
120
  // --- Brainstorm ---
120
121
  BRAINSTORM_ENABLED: envBoolean("false"),
121
122
  BRAINSTORM_MAX_ROUNDS: envInt("5", { min: 1, max: 20 }),
@@ -295,7 +296,8 @@ function transformEnvToConfig(env, dirname) {
295
296
  enabled: env.PREVIEW_ENABLED,
296
297
  host: env.PREVIEW_HOST,
297
298
  ttlMs: env.PREVIEW_TTL_MS,
298
- keepAfterComplete: env.PREVIEW_KEEP_AFTER_COMPLETE
299
+ keepAfterComplete: env.PREVIEW_KEEP_AFTER_COMPLETE,
300
+ reapIntervalMs: env.PREVIEW_REAP_INTERVAL_MS
299
301
  },
300
302
  brainstorm: {
301
303
  enabled: env.BRAINSTORM_ENABLED,
@@ -441,4 +443,4 @@ export {
441
443
  resetDotenvCache,
442
444
  reloadConfig
443
445
  };
444
- //# sourceMappingURL=chunk-XCHQMDNZ.js.map
446
+ //# sourceMappingURL=chunk-LD222Y4U.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/config.ts","../src/config-schema.ts","../src/utils/network.ts"],"sourcesContent":["import { config as loadDotenv } from 'dotenv';\nimport path from 'node:path';\nimport fs from 'node:fs';\nimport { fileURLToPath } from 'node:url';\nimport {\n envSchema,\n extractEnvSubset,\n transformEnvToConfig,\n ConfigValidationError,\n} from './config-schema.js';\nimport { getGlobalDir } from './paths.js';\n\nexport type {\n Config,\n AIRunnerMode,\n BrainstormAgentConfig,\n ChatAgentConfig,\n // Sub-type aliases for interface segregation\n GongfengConfig,\n ProjectConfig,\n AIConfig,\n PollConfig,\n PipelineConfig,\n ReviewConfig,\n WebConfig,\n IssueNoteSyncConfig,\n WebhookConfig,\n E2eConfig,\n PreviewConfig,\n BrainstormConfig,\n ChatConfig,\n BraindumpConfig,\n AutoUpdateConfig,\n IWikiConfig,\n KnowledgeConfig,\n DistillConfig,\n SyncConfig,\n WorkspaceEnvConfig,\n} from './config-schema.js';\nexport { ConfigValidationError } from './config-schema.js';\n\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\n\n/**\n * Resolve the path to the .env configuration file.\n *\n * Priority (first match wins):\n * 1. Explicit `configPath` argument (e.g. --config CLI flag)\n * 2. `IAF_CONFIG_PATH` environment variable\n * 3. Source-mode: `.env` next to compiled output (`dist/../.env`)\n * 4. Global config written by `issue-auto-finish init` (`{IAF_HOME}/.env`)\n * 5. CWD `.env` (legacy / manual setup)\n * 6. Fallback: global path (even if it doesn't exist yet)\n *\n * `IAF_HOME` defaults to `~/.issue-auto-finish` and can be overridden via env.\n *\n * This is the **single source of truth** for .env path resolution.\n * CLI commands and route handlers should call this instead of\n * rolling their own lookup logic.\n */\nexport function resolveConfigFilePath(configPath?: string): string {\n if (configPath) return configPath;\n if (process.env.IAF_CONFIG_PATH) return process.env.IAF_CONFIG_PATH;\n\n // Source-mode: .env next to the compiled output\n const localEnv = path.resolve(__dirname, '../.env');\n if (fs.existsSync(localEnv)) return localEnv;\n\n // Global config written by `issue-auto-finish init`\n // Checked before CWD to avoid picking up unrelated project .env files\n const globalEnv = path.join(getGlobalDir(), '.env');\n if (fs.existsSync(globalEnv)) return globalEnv;\n\n // Fallback: CWD .env (legacy / manual setup)\n const cwdEnv = path.resolve(process.cwd(), '.env');\n if (fs.existsSync(cwdEnv)) return cwdEnv;\n\n return globalEnv;\n}\n\nlet _dotenvLoaded = false;\nfunction ensureDotenvLoaded(): void {\n if (_dotenvLoaded) return;\n _dotenvLoaded = true;\n loadDotenv({ path: resolveConfigFilePath() });\n}\n\nexport function loadConfig() {\n ensureDotenvLoaded();\n\n const subset = extractEnvSubset(process.env);\n const result = envSchema.safeParse(subset);\n\n if (!result.success) {\n throw new ConfigValidationError(result.error);\n }\n\n return transformEnvToConfig(result.data, __dirname);\n}\n\n/**\n * Reset dotenv cache so the next loadConfig() re-reads the .env file.\n */\nexport function resetDotenvCache(): void {\n _dotenvLoaded = false;\n}\n\n/**\n * Reload config from .env file.\n * Clears dotenv cache, removes stale IAF env vars from process.env,\n * then re-reads and re-parses everything.\n */\nexport function reloadConfig() {\n resetDotenvCache();\n // Clear process.env of IAF-related keys so dotenv can overwrite them\n const iafKeys = extractEnvSubset(process.env);\n for (const key of Object.keys(iafKeys)) {\n delete process.env[key];\n }\n return loadConfig();\n}\n","/**\n * Zod-based configuration schema and transformation layer.\n *\n * Architecture:\n * process.env → extractEnvSubset() → envSchema.safeParse() → transformEnvToConfig() → Config\n *\n * The schema is kept flat (keys = env var names) so that validation errors\n * directly reference environment variable names without extra mapping.\n */\nimport { z } from 'zod';\nimport { getDefaultBinary, getBinaryEnvKey } from './ai-runner/AIRunnerRegistry.js';\nimport { getLocalIP } from './utils/network.js';\nimport path from 'node:path';\n\n// ---------------------------------------------------------------------------\n// Reusable zod helpers\n// ---------------------------------------------------------------------------\n\n/** Boolean env var: only `'true'` (case-sensitive) is truthy. */\nfunction envBoolean(defaultValue: string = 'false') {\n return z\n .string()\n .optional()\n .default(defaultValue)\n .transform((v) => v === 'true');\n}\n\n/** Integer env var with optional min/max bounds. */\nfunction envInt(defaultValue: string, opts?: { min?: number; max?: number }) {\n let schema = z.coerce.number().int();\n if (opts?.min !== undefined) schema = schema.min(opts.min);\n if (opts?.max !== undefined) schema = schema.max(opts.max);\n return schema.optional().default(Number(defaultValue));\n}\n\n/** Port number: integer 1–65535. */\nfunction envPort(defaultValue: string) {\n return z.coerce\n .number()\n .int()\n .min(1, 'Port must be >= 1')\n .max(65535, 'Port must be <= 65535')\n .optional()\n .default(Number(defaultValue));\n}\n\n/** Millisecond duration: integer >= 1000. */\nfunction envMs(defaultValue: string) {\n return z.coerce\n .number()\n .int()\n .min(1000, 'Duration must be >= 1000ms')\n .optional()\n .default(Number(defaultValue));\n}\n\n// ---------------------------------------------------------------------------\n// Flat environment schema\n// ---------------------------------------------------------------------------\n\nexport const envSchema = z.object({\n // --- Required ---\n GONGFENG_API_URL: z.string().url('GONGFENG_API_URL must be a valid URL'),\n GONGFENG_PRIVATE_TOKEN: z.string().min(1, 'GONGFENG_PRIVATE_TOKEN is required'),\n GONGFENG_PROJECT_PATH: z.string().min(1, 'GONGFENG_PROJECT_PATH is required'),\n PROJECT_WORK_DIR: z.string().min(1, 'PROJECT_WORK_DIR is required'),\n\n // --- Paths (override auto-detected data/logs directories) ---\n DATA_DIR: z.string().optional(),\n LOGS_DIR: z.string().optional(),\n\n // --- Project ---\n GIT_ROOT_DIR: z.string().optional(),\n BASE_BRANCH: z.string().optional().default('master'),\n BRANCH_PREFIX: z.string().optional().default('feat/issue'),\n WORKTREE_BASE_DIR: z.string().optional().default(''),\n PROJECT_SUBDIR: z.string().optional().default(''),\n\n // --- AI ---\n AI_RUNNER_MODE: z\n .string()\n .optional()\n .default('claude-internal'),\n AI_MODEL: z.string().optional().default('Claude-4.6-Opus'),\n CLAUDE_BINARY: z.string().optional(),\n CURSOR_BINARY: z.string().optional(),\n CODEBUDDY_BINARY: z.string().optional(),\n CODEBUDDY_ACP_AUTO_APPROVE: envBoolean('true'),\n CLAUDE_PHASE_TIMEOUT_MS: envMs('1800000'),\n AI_IDLE_TIMEOUT_MS: z.coerce\n .number()\n .int()\n .min(0, 'AI_IDLE_TIMEOUT_MS must be >= 0 (0 to disable)')\n .optional()\n .default(1200000),\n NVM_NODE_VERSION: z.string().optional().default('20'),\n\n // --- Pipeline ---\n PIPELINE_MODE: z\n .string()\n .optional()\n .default('auto'),\n\n // --- Poll ---\n POLL_DISCOVERY_INTERVAL_MS: envMs('60000'),\n POLL_DRIVE_INTERVAL_MS: envMs('15000'),\n MAX_RETRIES: envInt('3', { min: 0, max: 100 }),\n MAX_CONCURRENT_ISSUES: envInt('3', { min: 1 }),\n\n // --- Review ---\n REVIEW_ENABLED: envBoolean('true'),\n REVIEW_AUTO_APPROVE_LABELS: z.string().optional().default(''),\n\n // --- Web ---\n WEB_ENABLED: envBoolean('true'),\n WEB_HOST: z.string().optional().default('0.0.0.0'),\n WEB_PORT: envPort('3000'),\n FRONTEND_DIST_DIR: z.string().optional(),\n\n // --- Issue Note Sync ---\n ISSUE_NOTE_SYNC_ENABLED: envBoolean('true'),\n WEB_BASE_URL: z.string().optional(),\n\n // --- Webhook ---\n WEBHOOK_ENABLED: envBoolean('false'),\n WEBHOOK_HOST: z.string().optional().default('0.0.0.0'),\n WEBHOOK_PORT: envPort('8081'),\n WEBHOOK_SECRET: z.string().optional().default(''),\n WEBHOOK_LLM_FALLBACK: envBoolean('true'),\n WEBHOOK_LLM_BINARY: z.string().optional().default('claude-internal'),\n\n // --- E2E ---\n E2E_UI_ENABLED: envBoolean('false'),\n E2E_BASE_URL: z.string().optional().default('https://localhost:8890'),\n E2E_BACKEND_URL: z.string().optional().default('http://127.0.0.1:3000'),\n E2E_AUTH_COOKIES: z.string().optional().default('[]'),\n E2E_BACKEND_PORT_BASE: envPort('4000'),\n E2E_FRONTEND_PORT_BASE: envPort('9000'),\n E2E_UAT_VENDOR_DIR: z.string().optional().default(''),\n E2E_UAT_CONFIG_FILE: z.string().optional().default(''),\n E2E_PYTHON_BIN: z.string().optional().default('python3'),\n\n // --- Preview ---\n PREVIEW_ENABLED: envBoolean('false'),\n PREVIEW_HOST: z.string().optional().default(''),\n PREVIEW_TTL_MS: envMs(String(24 * 60 * 60 * 1000)),\n PREVIEW_KEEP_AFTER_COMPLETE: envBoolean('true'),\n PREVIEW_REAP_INTERVAL_MS: envMs('300000'),\n\n // --- Brainstorm ---\n BRAINSTORM_ENABLED: envBoolean('false'),\n BRAINSTORM_MAX_ROUNDS: envInt('5', { min: 1, max: 20 }),\n BRAINSTORM_TIMEOUT_MS: envMs('600000'),\n BRAINSTORM_GENERATOR_MODE: z\n .string()\n .optional(),\n BRAINSTORM_GENERATOR_BINARY: z.string().optional(),\n BRAINSTORM_GENERATOR_MODEL: z.string().optional(),\n BRAINSTORM_REVIEWER_MODE: z\n .string()\n .optional(),\n BRAINSTORM_REVIEWER_BINARY: z.string().optional(),\n BRAINSTORM_REVIEWER_MODEL: z.string().optional(),\n\n // --- Chat ---\n CHAT_ENABLED: envBoolean('false'),\n CHAT_TIMEOUT_MS: envMs('300000'),\n CHAT_MAX_SESSION_MESSAGES: envInt('100', { min: 1 }),\n CHAT_AGENT_MODE: z\n .string()\n .optional(),\n CHAT_AGENT_BINARY: z.string().optional(),\n CHAT_AGENT_MODEL: z.string().optional(),\n\n // --- Braindump ---\n BRAINDUMP_ENABLED: envBoolean('false'),\n BRAINDUMP_MAX_CONCURRENT: z.coerce.number().int().min(1).optional(),\n BRAINDUMP_SPLIT_TIMEOUT_MS: envMs('300000'),\n BRAINDUMP_TASK_TIMEOUT_MS: z.coerce.number().int().min(1000).optional(),\n BRAINDUMP_MAX_CONFLICT_ATTEMPTS: envInt('20', { min: 1 }),\n BRAINDUMP_CREATE_MR: envBoolean('false'),\n\n // --- Auto Update ---\n AUTO_UPDATE_ENABLED: envBoolean('true'),\n AUTO_UPDATE_INTERVAL_MS: envMs('600000'),\n AUTO_UPDATE_REGISTRY: z\n .string()\n .url('AUTO_UPDATE_REGISTRY must be a valid URL')\n .optional()\n .default('https://registry.npmjs.org'),\n AUTO_UPDATE_DRAIN_TIMEOUT_MS: envMs('300000'),\n\n // --- iWiki ---\n IWIKI_AUTH_COOKIE: z.string().optional(),\n IWIKI_AUTH_TOKEN: z.string().optional(),\n IWIKI_BASE_URL: z.string().optional(),\n\n // --- Locale ---\n LOCALE: z.enum(['zh-CN', 'en']).optional().default('zh-CN'),\n\n // --- Knowledge ---\n KNOWLEDGE_ENABLED: envBoolean('false'),\n KNOWLEDGE_PATH: z.string().optional(),\n KNOWLEDGE_ORPHAN_BRANCH: z.string().optional().default('iaf/knowledge'),\n KNOWLEDGE_SYNC_VECTORS: envBoolean('false'),\n KNOWLEDGE_AUTO_RESTORE: envBoolean('true'),\n\n // --- Distill (知识蒸馏) ---\n DISTILL_ENABLED: envBoolean('false'),\n DISTILL_INTERVAL_MS: envMs('86400000'),\n DISTILL_DIARY_SUMMARIZE: envBoolean('false'),\n DISTILL_MIN_DIARIES_FOR_DISTILL: envInt('3', { min: 1 }),\n DISTILL_MEMORY_CONFIDENCE_THRESHOLD: z.coerce.number().min(0).max(1).optional().default(0.7),\n DISTILL_VECTOR_ENABLED: envBoolean('true'),\n\n // --- Sync (生成文件同步到目标项目) ---\n SYNC_KNOWLEDGE_TO_PROJECT: envBoolean('false'),\n SYNC_RULES_TO_PROJECT: envBoolean('false'),\n\n // --- Release (发布) ---\n RELEASE_ENABLED: envBoolean('false'),\n RELEASE_DETECT_CACHE_TTL_MS: envMs('604800000'),\n\n // --- Verify-Fix Loop (验证-修复循环) ---\n VERIFY_FIX_LOOP_ENABLED: envBoolean('true'),\n VERIFY_FIX_MAX_ITERATIONS: envInt('3', { min: 1, max: 10 }),\n VERIFY_TODOLIST_CHECK_ENABLED: envBoolean('true'),\n\n // --- Analytics (运营分析) ---\n ANALYTICS_ENABLED: envBoolean('false'),\n ANALYTICS_TARGET_API_URL: z.string().url().optional(),\n ANALYTICS_TARGET_API_TOKEN: z.string().optional(),\n ANALYTICS_CACHE_TTL_MS: envMs('300000'),\n ANALYTICS_SKILL_AUTO_DISCOVER: envBoolean('true'),\n\n // --- Coordination (多节点协调) ---\n NODE_ID: z.string().optional(),\n\n // --- Workspace (多仓库) ---\n WORKSPACE_CONFIG_PATH: z.string().optional(),\n\n // --- Tenants (多租户) ---\n TENANTS_CONFIG_PATH: z.string().optional(),\n});\n\nexport type ParsedEnv = z.infer<typeof envSchema>;\n\n// ---------------------------------------------------------------------------\n// Extract env subset\n// ---------------------------------------------------------------------------\n\n/**\n * Picks only the keys declared in `envSchema` from the given env object.\n * Empty strings are converted to `undefined` to preserve the existing\n * \"empty = unset\" behavior.\n */\nexport function extractEnvSubset(\n env: Record<string, string | undefined>,\n): Record<string, string | undefined> {\n const keys = Object.keys(envSchema.shape) as (keyof typeof envSchema.shape)[];\n const subset: Record<string, string | undefined> = {};\n for (const key of keys) {\n const val = env[key as string];\n subset[key as string] = val || undefined; // empty string → undefined\n }\n return subset;\n}\n\n// ---------------------------------------------------------------------------\n// AI Runner helpers\n// ---------------------------------------------------------------------------\n\nexport type AIRunnerMode = string;\n\nfunction resolveAIBinary(\n mode: AIRunnerMode,\n env: ParsedEnv,\n): string {\n const envKey = getBinaryEnvKey(mode);\n if (envKey) {\n const envVal = (env as Record<string, unknown>)[envKey];\n if (typeof envVal === 'string' && envVal) return envVal;\n }\n return getDefaultBinary(mode);\n}\n\nexport interface BrainstormAgentConfig {\n mode: AIRunnerMode;\n binary: string;\n nvmNodeVersion: string;\n model?: string;\n}\n\nexport type ChatAgentConfig = BrainstormAgentConfig;\n\nfunction buildBrainstormAgent(\n role: 'GENERATOR' | 'REVIEWER',\n env: ParsedEnv,\n): BrainstormAgentConfig {\n const roleMode =\n (role === 'GENERATOR' ? env.BRAINSTORM_GENERATOR_MODE : env.BRAINSTORM_REVIEWER_MODE) ??\n env.AI_RUNNER_MODE;\n return {\n mode: roleMode,\n binary:\n (role === 'GENERATOR' ? env.BRAINSTORM_GENERATOR_BINARY : env.BRAINSTORM_REVIEWER_BINARY) ??\n resolveAIBinary(roleMode, env),\n nvmNodeVersion: env.NVM_NODE_VERSION,\n model:\n (role === 'GENERATOR' ? env.BRAINSTORM_GENERATOR_MODEL : env.BRAINSTORM_REVIEWER_MODEL) ??\n env.AI_MODEL,\n };\n}\n\nfunction buildChatAgent(env: ParsedEnv): ChatAgentConfig {\n // Chat defaults to claude-internal for lightweight conversational use,\n // independent of the global AI_RUNNER_MODE (which may be cursor-agent).\n const chatMode = env.CHAT_AGENT_MODE ?? 'claude-internal';\n return {\n mode: chatMode,\n binary: env.CHAT_AGENT_BINARY ?? resolveAIBinary(chatMode, env),\n nvmNodeVersion: env.NVM_NODE_VERSION,\n model: env.CHAT_AGENT_MODEL ?? env.AI_MODEL,\n };\n}\n\n// ---------------------------------------------------------------------------\n// Transform parsed env → nested Config\n// ---------------------------------------------------------------------------\n\nexport function transformEnvToConfig(env: ParsedEnv, dirname: string) {\n const aiMode = env.AI_RUNNER_MODE;\n const webPort = env.WEB_PORT;\n\n const gitRootDir = env.GIT_ROOT_DIR ?? env.PROJECT_WORK_DIR;\n\n return {\n gongfeng: {\n apiUrl: env.GONGFENG_API_URL,\n privateToken: env.GONGFENG_PRIVATE_TOKEN,\n projectPath: env.GONGFENG_PROJECT_PATH,\n },\n project: {\n workDir: env.PROJECT_WORK_DIR,\n gitRootDir,\n baseBranch: env.BASE_BRANCH,\n branchPrefix: env.BRANCH_PREFIX,\n worktreeBaseDir: env.WORKTREE_BASE_DIR,\n projectSubDir: env.PROJECT_SUBDIR,\n },\n ai: {\n mode: aiMode,\n binary: resolveAIBinary(aiMode, env),\n phaseTimeoutMs: env.CLAUDE_PHASE_TIMEOUT_MS,\n idleTimeoutMs: env.AI_IDLE_TIMEOUT_MS || undefined,\n nvmNodeVersion: env.NVM_NODE_VERSION,\n model: env.AI_MODEL,\n codebuddyAcpAutoApprove: env.CODEBUDDY_ACP_AUTO_APPROVE,\n },\n poll: {\n discoveryIntervalMs: env.POLL_DISCOVERY_INTERVAL_MS,\n driveIntervalMs: env.POLL_DRIVE_INTERVAL_MS,\n maxRetries: env.MAX_RETRIES,\n maxConcurrent: env.MAX_CONCURRENT_ISSUES,\n },\n pipeline: {\n mode: env.PIPELINE_MODE,\n },\n review: {\n enabled: env.REVIEW_ENABLED,\n autoApproveLabels: env.REVIEW_AUTO_APPROVE_LABELS\n .split(',')\n .map((s) => s.trim())\n .filter(Boolean),\n },\n web: {\n enabled: env.WEB_ENABLED,\n host: env.WEB_HOST,\n port: webPort,\n frontendDistDir:\n env.FRONTEND_DIST_DIR ??\n path.resolve(dirname, '..', 'src/web/frontend/dist'),\n },\n issueNoteSync: {\n enabled: env.ISSUE_NOTE_SYNC_ENABLED,\n webBaseUrl: env.WEB_BASE_URL ?? `http://${getLocalIP()}:${webPort}`,\n },\n webhook: {\n enabled: env.WEBHOOK_ENABLED,\n host: env.WEBHOOK_HOST,\n port: env.WEBHOOK_PORT,\n secret: env.WEBHOOK_SECRET,\n llmFallback: env.WEBHOOK_LLM_FALLBACK,\n llmBinary: env.WEBHOOK_LLM_BINARY,\n },\n e2e: {\n enabled: env.E2E_UI_ENABLED,\n baseUrl: env.E2E_BASE_URL,\n backendUrl: env.E2E_BACKEND_URL,\n authCookies: env.E2E_AUTH_COOKIES,\n backendPortBase: env.E2E_BACKEND_PORT_BASE,\n frontendPortBase: env.E2E_FRONTEND_PORT_BASE,\n uatVendorDir: env.E2E_UAT_VENDOR_DIR,\n uatConfigFile: env.E2E_UAT_CONFIG_FILE,\n pythonBin: env.E2E_PYTHON_BIN,\n },\n preview: {\n enabled: env.PREVIEW_ENABLED,\n host: env.PREVIEW_HOST,\n ttlMs: env.PREVIEW_TTL_MS,\n keepAfterComplete: env.PREVIEW_KEEP_AFTER_COMPLETE,\n reapIntervalMs: env.PREVIEW_REAP_INTERVAL_MS,\n },\n brainstorm: {\n enabled: env.BRAINSTORM_ENABLED,\n maxRefinementRounds: env.BRAINSTORM_MAX_ROUNDS,\n timeoutMs: env.BRAINSTORM_TIMEOUT_MS,\n generator: buildBrainstormAgent('GENERATOR', env),\n reviewer: buildBrainstormAgent('REVIEWER', env),\n },\n chat: {\n enabled: env.CHAT_ENABLED,\n timeoutMs: env.CHAT_TIMEOUT_MS,\n maxSessionMessages: env.CHAT_MAX_SESSION_MESSAGES,\n agent: buildChatAgent(env),\n },\n braindump: {\n enabled: env.BRAINDUMP_ENABLED,\n maxConcurrent: env.BRAINDUMP_MAX_CONCURRENT ?? env.MAX_CONCURRENT_ISSUES,\n splitTimeoutMs: env.BRAINDUMP_SPLIT_TIMEOUT_MS,\n taskTimeoutMs: env.BRAINDUMP_TASK_TIMEOUT_MS ?? env.CLAUDE_PHASE_TIMEOUT_MS,\n maxConflictAttempts: env.BRAINDUMP_MAX_CONFLICT_ATTEMPTS,\n createMr: env.BRAINDUMP_CREATE_MR,\n },\n autoUpdate: {\n enabled: env.AUTO_UPDATE_ENABLED,\n intervalMs: env.AUTO_UPDATE_INTERVAL_MS,\n registry: env.AUTO_UPDATE_REGISTRY,\n drainTimeoutMs: env.AUTO_UPDATE_DRAIN_TIMEOUT_MS,\n },\n iwiki: {\n authCookie: env.IWIKI_AUTH_COOKIE,\n authToken: env.IWIKI_AUTH_TOKEN,\n baseUrl: env.IWIKI_BASE_URL,\n },\n locale: env.LOCALE,\n knowledge: {\n enabled: env.KNOWLEDGE_ENABLED,\n path: env.KNOWLEDGE_PATH,\n orphanBranch: env.KNOWLEDGE_ORPHAN_BRANCH,\n syncVectors: env.KNOWLEDGE_SYNC_VECTORS,\n autoRestore: env.KNOWLEDGE_AUTO_RESTORE,\n },\n distill: {\n enabled: env.DISTILL_ENABLED,\n intervalMs: env.DISTILL_INTERVAL_MS,\n diarySummarize: env.DISTILL_DIARY_SUMMARIZE,\n minDiariesForDistill: env.DISTILL_MIN_DIARIES_FOR_DISTILL,\n memoryConfidenceThreshold: env.DISTILL_MEMORY_CONFIDENCE_THRESHOLD,\n vectorEnabled: env.DISTILL_VECTOR_ENABLED,\n },\n coordination: {\n nodeId: env.NODE_ID,\n },\n sync: {\n knowledgeToProject: env.SYNC_KNOWLEDGE_TO_PROJECT,\n rulesToProject: env.SYNC_RULES_TO_PROJECT,\n },\n release: {\n enabled: env.RELEASE_ENABLED,\n detectCacheTtlMs: env.RELEASE_DETECT_CACHE_TTL_MS,\n },\n verifyFixLoop: {\n enabled: env.VERIFY_FIX_LOOP_ENABLED,\n maxIterations: env.VERIFY_FIX_MAX_ITERATIONS,\n todolistCheckEnabled: env.VERIFY_TODOLIST_CHECK_ENABLED,\n },\n analytics: {\n enabled: env.ANALYTICS_ENABLED,\n targetApiUrl: env.ANALYTICS_TARGET_API_URL,\n targetApiToken: env.ANALYTICS_TARGET_API_TOKEN,\n cacheTtlMs: env.ANALYTICS_CACHE_TTL_MS,\n skillAutoDiscover: env.ANALYTICS_SKILL_AUTO_DISCOVER,\n },\n workspace: {\n configPath: env.WORKSPACE_CONFIG_PATH,\n },\n tenants: {\n configPath: env.TENANTS_CONFIG_PATH,\n },\n } as const;\n}\n\n// ---------------------------------------------------------------------------\n// Config type (derived from transform)\n// ---------------------------------------------------------------------------\n\n// We use a widened version so the type is writable and compatible with existing\n// code that assigns to Config properties.\ntype TransformResult = ReturnType<typeof transformEnvToConfig>;\n\n// Deeply writable version of the transform result\ntype DeepWritable<T> = {\n -readonly [K in keyof T]: T[K] extends object ? DeepWritable<T[K]> : T[K];\n};\n\nexport type Config = DeepWritable<TransformResult>;\n\n// ---------------------------------------------------------------------------\n// Sub-type aliases for interface segregation\n// ---------------------------------------------------------------------------\n\n/** Gongfeng API connection settings. */\nexport type GongfengConfig = Config['gongfeng'];\n/** Project / repository settings. */\nexport type ProjectConfig = Config['project'];\n/** AI runner settings. */\nexport type AIConfig = Config['ai'];\n/** Polling intervals and concurrency limits. */\nexport type PollConfig = Config['poll'];\n/** Pipeline mode settings. */\nexport type PipelineConfig = Config['pipeline'];\n/** Review / gate settings. */\nexport type ReviewConfig = Config['review'];\n/** Web dashboard settings. */\nexport type WebConfig = Config['web'];\n/** Issue note-sync settings. */\nexport type IssueNoteSyncConfig = Config['issueNoteSync'];\n/** Webhook settings. */\nexport type WebhookConfig = Config['webhook'];\n/** E2E test settings. */\nexport type E2eConfig = Config['e2e'];\n/** Preview environment settings. */\nexport type PreviewConfig = Config['preview'];\n/** Brainstorm agent settings. */\nexport type BrainstormConfig = Config['brainstorm'];\n/** Chat agent settings. */\nexport type ChatConfig = Config['chat'];\n/** Braindump batch settings. */\nexport type BraindumpConfig = Config['braindump'];\n/** Auto-update settings. */\nexport type AutoUpdateConfig = Config['autoUpdate'];\n/** iWiki settings. */\nexport type IWikiConfig = Config['iwiki'];\n/** Knowledge settings. */\nexport type KnowledgeConfig = Config['knowledge'];\n/** Distill (knowledge distillation) settings. */\nexport type DistillConfig = Config['distill'];\n/** Coordination (multi-node claiming) settings. */\nexport type CoordinationConfig = Config['coordination'];\n/** Sync (generated files sync to project) settings. */\nexport type SyncConfig = Config['sync'];\n/** Verify-fix loop settings. */\nexport type VerifyFixLoopConfig = Config['verifyFixLoop'];\n/** Release (发布) settings. */\nexport type ReleaseConfig = Config['release'];\n/** Analytics (operational analysis) settings. */\nexport type AnalyticsConfig = Config['analytics'];\n/** Workspace (multi-repo) settings. */\nexport type WorkspaceEnvConfig = Config['workspace'];\n/** Tenants (multi-tenant) settings. */\nexport type TenantsEnvConfig = Config['tenants'];\n\n// ---------------------------------------------------------------------------\n// ConfigValidationError\n// ---------------------------------------------------------------------------\n\nexport class ConfigValidationError extends Error {\n public readonly issues: z.ZodIssue[];\n\n constructor(zodError: z.ZodError) {\n const lines = zodError.issues.map((issue) => {\n const envVar = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n return ` - ${envVar}: ${issue.message}`;\n });\n super(\n `Configuration validation failed:\\n${lines.join('\\n')}`,\n );\n this.name = 'ConfigValidationError';\n this.issues = zodError.issues;\n }\n}\n","import os from 'node:os';\n\n/** 获取本机第一个非回环 IPv4 地址,无则返回 'localhost' */\nexport function getLocalIP(): string {\n const interfaces = os.networkInterfaces();\n for (const addrs of Object.values(interfaces)) {\n if (!addrs) continue;\n for (const addr of addrs) {\n if (addr.family === 'IPv4' && !addr.internal) return addr.address;\n }\n }\n return 'localhost';\n}\n\n/** 将 0.0.0.0 / 127.0.0.1 / localhost 替换为真实 IP,用于面向用户展示的 URL */\nexport function resolveDisplayHost(host: string): string {\n if (host === '0.0.0.0' || host === '127.0.0.1' || host === 'localhost') {\n return getLocalIP();\n }\n return host;\n}\n"],"mappings":"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","names":["path","path"]}
@@ -1,6 +1,6 @@
1
1
  import {
2
2
  t
3
- } from "./chunk-2QHSU5YR.js";
3
+ } from "./chunk-QCVJKH52.js";
4
4
 
5
5
  // src/cli/setup/DependencyChecker.ts
6
6
  import { spawn } from "child_process";
@@ -381,4 +381,4 @@ async function* streamProcessLines(proc, timeoutMs) {
381
381
  export {
382
382
  DependencyChecker
383
383
  };
384
- //# sourceMappingURL=chunk-JUXD7ZBK.js.map
384
+ //# sourceMappingURL=chunk-M22VPSJ2.js.map