infra-kit 0.1.102 → 0.1.105

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 (127) hide show
  1. package/.eslintcache +1 -1
  2. package/.omc/state/agent-replay-0a58307d-2a37-4c69-851c-83a646502d62.jsonl +1 -0
  3. package/.omc/state/agent-replay-11c41aa0-51fa-49e1-a1dc-26dcd6ac26cc.jsonl +16 -0
  4. package/.omc/state/agent-replay-4cf1c186-81b2-497c-b002-d7f84e7839f3.jsonl +9 -0
  5. package/.omc/state/agent-replay-5c4ab554-64f1-42ae-83e3-21e0237e955c.jsonl +11 -0
  6. package/.omc/state/agent-replay-a60ac2ec-afbd-449f-a540-6df287392fc2.jsonl +1 -0
  7. package/.omc/state/agent-replay-be37e426-6fc8-47f4-8178-221c8494551c.jsonl +3 -0
  8. package/.omc/state/agent-replay-c967c819-3d1c-447b-ab48-56a8448ef9f8.jsonl +2 -0
  9. package/.omc/state/idle-notif-cooldown.json +3 -0
  10. package/.omc/state/last-tool-error.json +4 -4
  11. package/.omc/state/mission-state.json +53 -0
  12. package/.omc/state/sessions/0a58307d-2a37-4c69-851c-83a646502d62/pre-tool-advisory-throttle.json +18 -0
  13. package/.omc/state/sessions/0a58307d-2a37-4c69-851c-83a646502d62/subagent-tracking-state.json +7 -0
  14. package/.omc/state/sessions/11c41aa0-51fa-49e1-a1dc-26dcd6ac26cc/last-tool-error-state.json +7 -0
  15. package/.omc/state/sessions/11c41aa0-51fa-49e1-a1dc-26dcd6ac26cc/mission-state.json +117 -0
  16. package/.omc/state/sessions/11c41aa0-51fa-49e1-a1dc-26dcd6ac26cc/pre-tool-advisory-throttle.json +42 -0
  17. package/.omc/state/sessions/11c41aa0-51fa-49e1-a1dc-26dcd6ac26cc/subagent-tracking-state.json +53 -0
  18. package/.omc/state/sessions/4cf1c186-81b2-497c-b002-d7f84e7839f3/last-tool-error-state.json +7 -0
  19. package/.omc/state/sessions/4cf1c186-81b2-497c-b002-d7f84e7839f3/pre-tool-advisory-throttle.json +18 -0
  20. package/.omc/state/sessions/4cf1c186-81b2-497c-b002-d7f84e7839f3/subagent-tracking-state.json +7 -0
  21. package/.omc/state/sessions/5c4ab554-64f1-42ae-83e3-21e0237e955c/mission-state.json +117 -0
  22. package/.omc/state/sessions/5c4ab554-64f1-42ae-83e3-21e0237e955c/pre-tool-advisory-throttle.json +18 -0
  23. package/.omc/state/sessions/5c4ab554-64f1-42ae-83e3-21e0237e955c/subagent-tracking-state.json +17 -0
  24. package/.omc/state/sessions/a60ac2ec-afbd-449f-a540-6df287392fc2/pre-tool-advisory-throttle.json +18 -0
  25. package/.omc/state/sessions/a60ac2ec-afbd-449f-a540-6df287392fc2/subagent-tracking-state.json +7 -0
  26. package/.omc/state/sessions/c967c819-3d1c-447b-ab48-56a8448ef9f8/pre-tool-advisory-throttle.json +10 -0
  27. package/.omc/state/sessions/c967c819-3d1c-447b-ab48-56a8448ef9f8/subagent-tracking-state.json +7 -0
  28. package/.omc/state/subagent-tracking.json +14 -4
  29. package/.turbo/turbo-build.log +7 -0
  30. package/.turbo/turbo-check.log +14 -0
  31. package/.turbo/turbo-prettier-fix.log +2 -1
  32. package/.turbo/turbo-test.log +28 -5
  33. package/.turbo/turbo-validate.log +14 -0
  34. package/dist/cli.js +81 -74
  35. package/dist/cli.js.map +4 -4
  36. package/dist/entry/index.d.ts +2 -0
  37. package/dist/index.js +2 -0
  38. package/dist/index.js.map +7 -0
  39. package/dist/lib/package-config/package-config.d.ts +71 -0
  40. package/dist/mcp.js +43 -41
  41. package/dist/mcp.js.map +4 -4
  42. package/eslint.config.js +1 -1
  43. package/infra-kit.config.ts +5 -0
  44. package/package.json +20 -13
  45. package/scripts/build.js +32 -3
  46. package/src/.omc/state/sessions/0a58307d-2a37-4c69-851c-83a646502d62/pre-tool-advisory-throttle.json +18 -0
  47. package/src/commands/.omc/state/sessions/c967c819-3d1c-447b-ab48-56a8448ef9f8/pre-tool-advisory-throttle.json +18 -0
  48. package/src/commands/audit/__tests__/audit.test.ts +59 -0
  49. package/src/commands/audit/audit.ts +177 -0
  50. package/src/commands/audit/index.ts +1 -0
  51. package/src/commands/config/config.ts +49 -7
  52. package/src/commands/doctor/doctor.ts +3 -3
  53. package/src/commands/env-clear/env-clear.ts +1 -1
  54. package/src/commands/env-list/env-list.ts +3 -3
  55. package/src/commands/env-load/env-load.ts +1 -1
  56. package/src/commands/env-status/env-status.ts +1 -1
  57. package/src/commands/gh-merge-dev/gh-merge-dev.ts +3 -8
  58. package/src/commands/gh-release-deliver/gh-release-deliver.ts +47 -21
  59. package/src/commands/gh-release-deploy-all/gh-release-deploy-all.ts +13 -7
  60. package/src/commands/gh-release-deploy-selected/gh-release-deploy-selected.ts +12 -6
  61. package/src/commands/gh-release-list/gh-release-list.ts +19 -8
  62. package/src/commands/init/__tests__/migrate-config.test.ts +160 -0
  63. package/src/commands/init/init.ts +48 -35
  64. package/src/commands/init/migrate-config.ts +146 -0
  65. package/src/commands/node_modules/.vite/vitest/da39a3ee5e6b4b0d3255bfef95601890afd80709/results.json +1 -0
  66. package/src/commands/release-create/__tests__/release-create.test.ts +55 -0
  67. package/src/commands/release-create/release-create.ts +142 -38
  68. package/src/commands/release-desc-edit/release-desc-edit.ts +28 -8
  69. package/src/commands/version/version.ts +1 -1
  70. package/src/commands/worktrees-add/worktrees-add.ts +7 -12
  71. package/src/commands/worktrees-list/worktrees-list.ts +13 -5
  72. package/src/commands/worktrees-open/worktrees-open.ts +1 -1
  73. package/src/commands/worktrees-remove/worktrees-remove.ts +6 -10
  74. package/src/commands/worktrees-sync/worktrees-sync.ts +3 -5
  75. package/src/entry/cli.ts +49 -6
  76. package/src/entry/index.ts +5 -0
  77. package/src/integrations/cmux/open-workspace-with-layout.ts +4 -4
  78. package/src/integrations/cmux/workspace-title.ts +10 -4
  79. package/src/integrations/doppler/doppler-project.ts +1 -1
  80. package/src/integrations/gh/gh-release-prs/__tests__/gh-release-prs.test.ts +115 -0
  81. package/src/integrations/gh/gh-release-prs/gh-release-prs.ts +49 -32
  82. package/src/lib/.omc/state/sessions/a60ac2ec-afbd-449f-a540-6df287392fc2/pre-tool-advisory-throttle.json +14 -0
  83. package/src/lib/constants/index.ts +15 -0
  84. package/src/lib/git-utils/__tests__/git-utils.test.ts +49 -0
  85. package/src/lib/git-utils/git-utils.ts +3 -1
  86. package/src/lib/infra-kit-config/__tests__/infra-kit-config.test.ts +270 -0
  87. package/src/lib/infra-kit-config/index.ts +7 -1
  88. package/src/lib/infra-kit-config/infra-kit-config.ts +46 -28
  89. package/src/lib/package-config/__tests__/package-config.test.ts +95 -0
  90. package/src/lib/package-config/index.ts +3 -0
  91. package/src/lib/package-config/package-config-schema.ts +19 -0
  92. package/src/lib/package-config/package-config.ts +99 -0
  93. package/src/lib/package-validator/__tests__/package-validator.test.ts +263 -0
  94. package/src/lib/package-validator/checks/__tests__/checks.test.ts +130 -0
  95. package/src/lib/package-validator/checks/config-check.ts +30 -0
  96. package/src/lib/package-validator/checks/files-check.ts +29 -0
  97. package/src/lib/package-validator/checks/index.ts +4 -0
  98. package/src/lib/package-validator/checks/scripts-check.ts +23 -0
  99. package/src/lib/package-validator/checks/turbo-check.ts +47 -0
  100. package/src/lib/package-validator/fs-utils.ts +18 -0
  101. package/src/lib/package-validator/index.ts +3 -0
  102. package/src/lib/package-validator/loader/config-loader.ts +77 -0
  103. package/src/lib/package-validator/loader/index.ts +2 -0
  104. package/src/lib/package-validator/loader/package-discovery.ts +98 -0
  105. package/src/lib/package-validator/package-validator.ts +48 -0
  106. package/src/lib/package-validator/types.ts +15 -0
  107. package/src/lib/release-id/__tests__/release-id.test.ts +351 -0
  108. package/src/lib/release-id/__tests__/versioned-regression.test.ts +69 -0
  109. package/src/lib/release-id/index.ts +15 -0
  110. package/src/lib/release-id/release-id.ts +257 -0
  111. package/src/lib/release-utils/__tests__/release-utils.test.ts +122 -0
  112. package/src/lib/release-utils/index.ts +4 -0
  113. package/src/lib/release-utils/release-utils.ts +85 -17
  114. package/src/lib/version-utils/__tests__/load-existing-versions.test.ts +37 -0
  115. package/src/lib/version-utils/__tests__/next-version.test.ts +119 -13
  116. package/src/lib/version-utils/index.ts +3 -0
  117. package/src/lib/version-utils/load-existing-versions.ts +29 -10
  118. package/src/lib/version-utils/next-version.ts +67 -12
  119. package/src/lib/version-utils/version-utils.ts +13 -4
  120. package/src/mcp/tools/index.ts +2 -0
  121. package/src/node_modules/.vite/vitest/da39a3ee5e6b4b0d3255bfef95601890afd80709/results.json +1 -0
  122. package/src/types.ts +1 -1
  123. package/tsconfig.tsbuildinfo +1 -1
  124. package/src/lib/__tests__/infra-kit-config.test.ts +0 -231
  125. /package/src/integrations/{clickup → linear}/.gitkeep +0 -0
  126. /package/src/lib/{__tests__ → constants/__tests__}/constants.test.ts +0 -0
  127. /package/src/lib/{constants.ts → constants/constants.ts} +0 -0
package/dist/mcp.js.map CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
- "sources": ["../src/entry/mcp.ts", "../src/lib/error-handlers/index.ts", "../src/lib/logger/index.ts", "../src/mcp/server.ts", "../src/mcp/prompts/index.ts", "../src/mcp/resources/index.ts", "../src/commands/env-clear/env-clear.ts", "../src/lib/constants.ts", "../src/types.ts", "../src/commands/env-list/env-list.ts", "../src/lib/infra-kit-config/infra-kit-config.ts", "../src/lib/git-utils/git-utils.ts", "../src/integrations/doppler/doppler-project.ts", "../src/commands/env-load/env-load.ts", "../src/integrations/doppler/doppler-cli-auth.ts", "../src/lib/command-echo/command-echo.ts", "../src/commands/env-status/env-status.ts", "../src/commands/gh-merge-dev/gh-merge-dev.ts", "../src/integrations/gh/gh-cli-auth/gh-cli-auth.ts", "../src/integrations/gh/gh-release-prs/gh-release-prs.ts", "../src/lib/release-utils/release-utils.ts", "../src/integrations/jira/api.ts", "../src/lib/version-utils/load-existing-versions.ts", "../src/lib/version-utils/version-utils.ts", "../src/lib/version-utils/next-version.ts", "../src/lib/errors/operation-error.ts", "../src/commands/gh-release-deliver/gh-release-deliver.ts", "../src/lib/errors/format-zx-error.ts", "../src/lib/worktrees/remove-worktrees.ts", "../src/integrations/cmux/close-workspace-by-title.ts", "../src/integrations/cmux/list-workspace-titles.ts", "../src/integrations/cmux/open-workspace-with-layout.ts", "../src/integrations/cmux/workspace-title.ts", "../src/commands/gh-release-deploy-all/gh-release-deploy-all.ts", "../src/commands/gh-release-deploy-selected/gh-release-deploy-selected.ts", "../src/commands/gh-release-list/gh-release-list.ts", "../src/commands/release-create/release-create.ts", "../src/commands/release-desc-edit/release-desc-edit.ts", "../src/commands/version/version.ts", "../package.json", "../src/commands/worktrees-add/worktrees-add.ts", "../src/integrations/cursor/add-folders-to-workspace.ts", "../src/integrations/cursor/reconcile-workspace-folders.ts", "../src/integrations/cursor/remove-folders-from-workspace.ts", "../src/integrations/cursor/resolve-workspace-path.ts", "../src/commands/worktrees-list/worktrees-list.ts", "../src/commands/worktrees-open/worktrees-open.ts", "../src/commands/worktrees-remove/worktrees-remove.ts", "../src/commands/worktrees-sync/worktrees-sync.ts", "../src/lib/tool-handler/tool-handler.ts", "../src/mcp/tools/index.ts"],
4
- "sourcesContent": ["import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js'\nimport process from 'node:process'\n\nimport { setupErrorHandlers } from 'src/lib/error-handlers'\nimport { initLoggerMcp } from 'src/lib/logger'\n\nimport { createMcpServer } from '../mcp/server'\n\nconst logger = initLoggerMcp()\n\nconst startServer = async () => {\n let server\n\n try {\n server = await createMcpServer()\n\n logger.info('MCP Server instance created')\n } catch (error) {\n logger.error({ err: error, msg: 'Failed to create MCP server' })\n logger.error(`Fatal error during server creation.`)\n\n process.exit(1)\n }\n\n try {\n const transport = new StdioServerTransport()\n\n await server.connect(transport)\n\n logger.info({ msg: 'Server connected to transport. Ready.' })\n } catch (error) {\n logger.error({ err: error, msg: 'Failed to initialize server' })\n logger.error(`Fatal error during server transport init.`)\n\n process.exit(1)\n }\n}\n\n// Setup error handlers\nsetupErrorHandlers(logger)\n\n// Start the server\nstartServer()\n", "import process from 'node:process'\nimport type { Logger } from 'pino'\n\nimport { LOG_FILE_PATH } from '../logger/index'\n\n/**\n * Setup error handlers for the application\n * @param logger - The logger instance\n *\n * ONLY FOR SERVER!\n */\nexport const setupErrorHandlers = (logger: Logger) => {\n process.on('SIGINT', () => {\n logger.info({ msg: 'Received SIGINT. Shutting down...' })\n process.exit(0)\n })\n\n process.on('SIGTERM', () => {\n logger.info({ msg: 'Received SIGTERM. Shutting down...' })\n process.exit(0)\n })\n\n process.on('uncaughtException', (error) => {\n logger.fatal({ err: error, msg: 'Uncaught Exception' })\n logger.error(`Uncaught Exception! Check ${LOG_FILE_PATH}. Shutting down...`)\n logger.flush()\n process.exit(1)\n })\n\n process.on('unhandledRejection', (reason, promise) => {\n logger.fatal({ reason, promise, msg: 'Unhandled Rejection' })\n logger.error(`Unhandled Rejection! Check ${LOG_FILE_PATH}. Shutting down...`)\n logger.flush()\n process.exit(1)\n })\n}\n", "import process from 'node:process'\nimport pino from 'pino'\nimport pretty from 'pino-pretty'\n\n// eslint-disable-next-line sonarjs/publicly-writable-directories\nexport const LOG_FILE_PATH = '/tmp/mcp-infra-kit.log'\n\nexport const initLoggerMcp = () => {\n const logLevel = process.argv.includes('--debug') ? 'debug' : 'info'\n\n const logger = pino({ level: logLevel }, pino.destination({ dest: LOG_FILE_PATH }))\n\n logger.info(`Logger initialized with level: ${logLevel}. Logging to: ${LOG_FILE_PATH}`)\n\n return logger\n}\n\nexport const initLoggerCLI = () => {\n const logLevel = process.argv.includes('--debug') ? 'debug' : 'info'\n\n const ignoreFields = ['time', 'pid', 'hostname']\n\n if (logLevel === 'debug') {\n ignoreFields.push('level')\n }\n\n const logger = pino(\n { level: logLevel },\n pretty({\n destination: 2,\n ignore: ignoreFields.join(','),\n colorize: true,\n }),\n )\n\n return logger\n}\n\n// Singleton logger instance for CLI usage\nexport const logger = initLoggerCLI()\n", "import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\n\nimport { initializePrompts } from './prompts'\nimport { initializeResources } from './resources'\nimport { initializeTools } from './tools'\n\nexport async function createMcpServer() {\n const server = new McpServer(\n {\n name: 'infra-kit',\n version: '1.0.0',\n },\n {\n capabilities: {\n resources: {},\n tools: {},\n prompts: {},\n },\n },\n )\n\n await initializePrompts(server)\n await initializeResources(server)\n await initializeTools(server)\n\n return server\n}\n", "import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\n\nexport const initializePrompts = async (_server: McpServer) => {}\n", "import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\n\nexport const initializeResources = async (_server: McpServer) => {}\n", "import fs from 'node:fs'\nimport path from 'node:path'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\n\nimport {\n ENV_CLEAR_FILE,\n ENV_LOAD_FILE,\n INFRA_KIT_ENV_CONFIG_VAR,\n INFRA_KIT_ENV_LOADED_AT_VAR,\n INFRA_KIT_ENV_PROJECT_VAR,\n atomicWriteFileSync,\n getSessionCacheDir,\n parseVarNamesFromEnvFile,\n} from 'src/lib/constants'\nimport { defineMcpTool, textContent } from 'src/types'\n\n/**\n * Clear loaded env vars. Prints a file path to stdout that must be sourced to apply.\n * The env-clear shell alias does this automatically. Throws when no env is loaded\n * so CLI callers exit non-zero and MCP callers receive a structured tool error.\n */\nexport const envClear = async () => {\n const cacheDir = getSessionCacheDir()\n const envLoadPath = path.join(cacheDir, ENV_LOAD_FILE)\n\n if (!fs.existsSync(envLoadPath)) {\n throw new Error('No loaded environment found. Run `env-load` first.')\n }\n\n const varNames = parseVarNamesFromEnvFile(envLoadPath)\n\n const unsetLines = [\n ...varNames.map((v) => {\n return `unset ${v}`\n }),\n `unset ${INFRA_KIT_ENV_CONFIG_VAR}`,\n `unset ${INFRA_KIT_ENV_PROJECT_VAR}`,\n `unset ${INFRA_KIT_ENV_LOADED_AT_VAR}`,\n ]\n\n const clearFilePath = path.resolve(cacheDir, ENV_CLEAR_FILE)\n\n fs.mkdirSync(cacheDir, { recursive: true, mode: 0o700 })\n\n atomicWriteFileSync(clearFilePath, `${unsetLines.join('\\n')}\\n`, 0o600)\n\n // REQUIRED\n process.stdout.write(`${clearFilePath}\\n`)\n\n // Remove env load file so the next env-clear call correctly reports \"no env loaded\".\n fs.unlinkSync(envLoadPath)\n\n const structuredContent = {\n filePath: clearFilePath,\n variableCount: varNames.length,\n unsetStatements: unsetLines,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const envClearMcpTool = defineMcpTool({\n name: 'env-clear',\n description:\n 'Generate a shell script that unsets every env var previously loaded by env-load for this session, plus the infra-kit session metadata vars. Does NOT mutate the calling process. When `infra-kit init` has installed the zsh shell integration, the user\\'s terminal auto-sources the unset script on its next prompt (precmd hook) \u2014 so calling this via MCP will clear the vars in the shell that launched Claude Code automatically. Other callers must source \"<filePath>\" themselves or surface it to the user. Errors if no env is currently loaded.',\n inputSchema: {},\n outputSchema: {\n filePath: z.string().describe('Path to the file that must be sourced to apply'),\n variableCount: z.number().describe('Number of variables cleared'),\n unsetStatements: z.array(z.string()).describe('Unset statements generated'),\n },\n handler: envClear,\n})\n", "import fs from 'node:fs'\nimport os from 'node:os'\nimport path from 'node:path'\nimport process from 'node:process'\n\nexport const ENV_LOAD_FILE = 'env-load.sh'\nexport const ENV_CLEAR_FILE = 'env-clear.sh'\n\nexport const INFRA_KIT_SESSION_VAR = 'INFRA_KIT_SESSION'\nexport const INFRA_KIT_ENV_CONFIG_VAR = 'INFRA_KIT_ENV_CONFIG'\nexport const INFRA_KIT_ENV_PROJECT_VAR = 'INFRA_KIT_ENV_PROJECT'\nexport const INFRA_KIT_ENV_LOADED_AT_VAR = 'INFRA_KIT_ENV_LOADED_AT'\n\n/**\n * Matches a line of the form `KEY=...` where KEY is an env-var identifier\n * (letter or underscore, then word chars). Capture group 1 is the name. Shared\n * between env-load (validation, var counting) and parseVarNamesFromEnvFile.\n */\nexport const ENV_VAR_LINE_PATTERN = /^([A-Z_]\\w*)=/i\n\nexport const parseVarNamesFromEnvFile = (filePath: string): string[] => {\n if (!fs.existsSync(filePath)) return []\n\n const content = fs.readFileSync(filePath, 'utf-8')\n const names: string[] = []\n\n for (const line of content.split('\\n')) {\n const match = ENV_VAR_LINE_PATTERN.exec(line)\n\n if (match) {\n names.push(match[1]!)\n }\n }\n\n return names\n}\n\n/**\n * Root cache dir for infra-kit across all sessions. Resolved from\n * $XDG_CACHE_HOME when set, falling back to ~/.cache/infra-kit. Keep in sync\n * with the shell block emitted by `infra-kit init` (src/commands/init/init.ts).\n */\nexport const getCacheRoot = (): string => {\n const xdg = process.env.XDG_CACHE_HOME\n const base = xdg && xdg.length > 0 ? xdg : path.join(os.homedir(), '.cache')\n\n return path.join(base, 'infra-kit')\n}\n\nexport const getSessionCacheDir = (): string => {\n const session = process.env[INFRA_KIT_SESSION_VAR]\n\n if (!session) {\n throw new Error(`${INFRA_KIT_SESSION_VAR} is not set. Run \\`infra-kit init\\` then \\`source ~/.zshrc\\`.`)\n }\n\n return path.join(getCacheRoot(), session)\n}\n\n/**\n * Write content atomically: write to a pid-suffixed temp file in the same\n * directory, then rename. fs.renameSync is atomic on a single filesystem, so\n * concurrent writers can't produce a half-written secret file.\n */\nexport const atomicWriteFileSync = (filePath: string, content: string, mode: number): void => {\n const tmpPath = `${filePath}.tmp.${process.pid}`\n\n fs.writeFileSync(tmpPath, content, { mode })\n\n try {\n fs.renameSync(tmpPath, filePath)\n } catch (error) {\n fs.rmSync(tmpPath, { force: true })\n throw error\n }\n}\n\nexport const WORKTREES_DIR_SUFFIX = '-worktrees'\n// eslint-disable-next-line sonarjs/publicly-writable-directories\nexport const LOG_FILE_PATH = '/tmp/mcp-infra-kit.log'\n", "import type { z } from 'zod/v4'\n\nexport interface ToolsExecutionResult<TStructured = Record<string, unknown>> {\n [x: string]: unknown\n content: {\n type: 'text'\n text: string\n }[]\n structuredContent?: TStructured\n}\n\nexport interface RequiredConfirmedOptionArg {\n confirmedCommand: boolean\n}\n\nexport interface McpTool<TIn extends z.ZodRawShape = z.ZodRawShape, TOut extends z.ZodRawShape = z.ZodRawShape> {\n name: string\n description: string\n inputSchema: TIn\n outputSchema: TOut\n handler: (\n params: z.infer<z.ZodObject<TIn>> & RequiredConfirmedOptionArg,\n ) => Promise<ToolsExecutionResult<z.infer<z.ZodObject<TOut>>>>\n}\n\n/**\n * Build the dual-channel content array shared by every MCP tool. Narrows the\n * literal `type: 'text'` so handlers can use inferred return types without TS\n * widening `type` to `string` \u2014 which would otherwise break assignability\n * against the MCP SDK's content union.\n *\n * @example\n * return {\n * content: textContent(JSON.stringify(structuredContent, null, 2)),\n * structuredContent,\n * }\n */\nexport const textContent = (text: string): ToolsExecutionResult['content'] => {\n return [{ type: 'text', text }]\n}\n\n/**\n * Factory that ties the handler's return type to the declared `outputSchema`\n * so `structuredContent` is checked against the schema at compile time. If a\n * handler accidentally drops or renames a field, TS errors at the registration\n * site rather than at runtime in an MCP client.\n *\n * @example\n * export const envLoadMcpTool = defineMcpTool({\n * name: 'env-load',\n * description: '...',\n * inputSchema: { config: z.string() },\n * outputSchema: {\n * filePath: z.string(),\n * variableCount: z.number(),\n * project: z.string(),\n * config: z.string(),\n * },\n * handler: envLoad,\n * })\n */\nexport const defineMcpTool = <TIn extends z.ZodRawShape, TOut extends z.ZodRawShape>(\n tool: McpTool<TIn, TOut>,\n): McpTool<TIn, TOut> => {\n return tool\n}\n", "import { z } from 'zod/v4'\n\nimport { getDopplerProject } from 'src/integrations/doppler/doppler-project'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { defineMcpTool, textContent } from 'src/types'\n\n/**\n * List available Doppler configs for the detected project.\n *\n * Purely local: reads infra-kit.yml and does not call Doppler. We intentionally\n * do not run validateDopplerCliAndAuth here \u2014 users listing envs often do so\n * before `doppler login`, and a spurious auth error would be misleading.\n */\nexport const envList = async () => {\n const project = await getDopplerProject()\n const { environments } = await getInfraKitConfig()\n\n logger.info(`Doppler project: ${project}\\n`)\n logger.info('Available configs:')\n\n for (const env of environments) {\n logger.info(` - ${env}`)\n }\n\n const structuredContent = {\n project,\n configs: environments,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const envListMcpTool = defineMcpTool({\n name: 'env-list',\n description:\n 'List the environments the project is configured to support. Returns the `environments` list declared in infra-kit.yml at the project root (not a live fetch from Doppler) plus the Doppler project name resolved from the same file. Read-only.',\n inputSchema: {},\n outputSchema: {\n project: z.string().describe('Detected Doppler project name'),\n configs: z.array(z.string()).describe('Available environment configs'),\n },\n handler: envList,\n})\n", "import fs from 'node:fs/promises'\nimport os from 'node:os'\nimport path from 'node:path'\nimport yaml from 'yaml'\nimport { z } from 'zod/v4'\n\nimport { getProjectRoot, getRepoName } from 'src/lib/git-utils'\n\nconst INFRA_KIT_CONFIG_FILE = 'infra-kit.yml'\n\nconst USER_CONFIG_DIR_NAME = '.infra-kit'\nconst USER_GLOBAL_CONFIG_FILE = 'config.yml'\nconst USER_PROJECTS_DIR = 'projects'\n\n// envManagement\nconst dopplerEnvManagementSchema = z.object({\n provider: z.literal('doppler'),\n config: z.object({\n name: z.string().min(1),\n }),\n})\n\nconst envManagementSchema = z.discriminatedUnion('provider', [dopplerEnvManagementSchema])\n\n// ide\nconst cursorIdeConfigSchema = z\n .object({\n mode: z.enum(['workspace', 'windows']).default('workspace'),\n workspaceConfigPath: z.string().min(1).optional(),\n })\n .refine(\n (v) => {\n return v.mode !== 'workspace' || !!v.workspaceConfigPath\n },\n {\n message: 'workspaceConfigPath is required when mode is \"workspace\"',\n path: ['workspaceConfigPath'],\n },\n )\n\nconst cursorIdeSchema = z.object({\n provider: z.literal('cursor'),\n config: cursorIdeConfigSchema,\n})\n\nconst ideSchema = z.discriminatedUnion('provider', [cursorIdeSchema])\n\n// taskManager\nconst jiraTaskManagerSchema = z.object({\n provider: z.literal('jira'),\n config: z.object({\n baseUrl: z.string().url(),\n projectId: z.number().int().positive(),\n }),\n})\n\nconst taskManagerSchema = z.discriminatedUnion('provider', [jiraTaskManagerSchema])\n\n// worktrees prompt defaults\nconst worktreesConfigSchema = z.object({\n openInGithubDesktop: z.boolean().optional(),\n openInCmux: z.boolean().optional(),\n})\n\nconst infraKitConfigSchema = z.object({\n environments: z.array(z.string().min(1)).min(1),\n envManagement: envManagementSchema,\n ide: ideSchema.optional(),\n taskManager: taskManagerSchema.optional(),\n worktrees: worktreesConfigSchema.optional(),\n})\n\nconst infraKitOverrideConfigSchema = infraKitConfigSchema.partial()\n\nexport type InfraKitConfig = z.infer<typeof infraKitConfigSchema>\n\nexport interface InfraKitConfigPaths {\n /** Committed project config (required). */\n main: string\n /** User-scope global overrides applied to every project. */\n userGlobal: string\n /** User-scope per-project overrides \u2014 `<userProjectsDir>/<projectName>/infra-kit.yml`. */\n userProject: string\n /** Repo basename (`path.basename(projectRoot)`) used to namespace the user-project file. */\n projectName: string\n}\n\ninterface CacheEntry {\n mtimes: Record<keyof Omit<InfraKitConfigPaths, 'projectName'>, number | null>\n value: InfraKitConfig\n}\n\nlet cached: CacheEntry | null = null\n\n/**\n * Resolve every file path that participates in the config merge chain. Always\n * returns paths even for files that don't yet exist, so callers can use them\n * for \"where would my override go?\" prompts.\n *\n * @example\n * const paths = await getInfraKitConfigPaths()\n * // {\n * // main: '/Users/arthur/projects/api/infra-kit.yml',\n * // userGlobal: '/Users/arthur/.infra-kit/config.yml',\n * // userProject: '/Users/arthur/.infra-kit/projects/api/infra-kit.yml',\n * // projectName: 'api',\n * // }\n */\nexport const getInfraKitConfigPaths = async (): Promise<InfraKitConfigPaths> => {\n const projectRoot = await getProjectRoot()\n const projectName = await getRepoName()\n const userConfigDir = path.join(os.homedir(), USER_CONFIG_DIR_NAME)\n\n return {\n main: path.join(projectRoot, INFRA_KIT_CONFIG_FILE),\n userGlobal: path.join(userConfigDir, USER_GLOBAL_CONFIG_FILE),\n userProject: path.join(userConfigDir, USER_PROJECTS_DIR, projectName, INFRA_KIT_CONFIG_FILE),\n projectName,\n }\n}\n\n/**\n * Read and validate `infra-kit.yml`, with optional override layers shallow-merged\n * on top in this order (later wins):\n * 1. project `infra-kit.yml` \u2014 committed source of truth\n * 2. `~/.infra-kit/config.yml` \u2014 user-global defaults\n * 3. `~/.infra-kit/projects/<repo-name>/infra-kit.yml` \u2014 user-scope per-project overrides\n *\n * Top-level keys (entire capability sections like `ide`, `envManagement`)\n * replace wholesale. Results are cached per file mtimes so the long-running\n * MCP server picks up edits without a restart.\n *\n * @example\n * // infra-kit.yml: { environments: ['dev'], envManagement: { provider: 'doppler', config: { name: 'p' } } }\n * // ~/.infra-kit/config.yml: { ide: { provider: 'cursor', config: { mode: 'windows' } } }\n * const cfg = await getInfraKitConfig()\n * // => { environments: ['dev'], envManagement: {...}, ide: { provider: 'cursor', config: { mode: 'windows' } } }\n */\nexport const getInfraKitConfig = async (): Promise<InfraKitConfig> => {\n const paths = await getInfraKitConfigPaths()\n\n let mainStat: Awaited<ReturnType<typeof fs.stat>>\n\n try {\n mainStat = await fs.stat(paths.main)\n } catch {\n cached = null\n throw new Error(`infra-kit.yml not found at ${paths.main}`)\n }\n\n const [userGlobalStat, userProjectStat] = await Promise.all([\n statIfExists(paths.userGlobal),\n statIfExists(paths.userProject),\n ])\n\n const mtimes = {\n main: Number(mainStat.mtimeMs),\n userGlobal: userGlobalStat ? Number(userGlobalStat.mtimeMs) : null,\n userProject: userProjectStat ? Number(userProjectStat.mtimeMs) : null,\n }\n\n if (cached && shallowEqual(cached.mtimes, mtimes)) {\n return cached.value\n }\n\n const layers: ConfigLayer[] = [\n { label: 'infra-kit.yml', path: paths.main, required: true },\n { label: '~/.infra-kit/config.yml', path: paths.userGlobal, required: false },\n {\n label: `~/.infra-kit/projects/${paths.projectName}/infra-kit.yml`,\n path: paths.userProject,\n required: false,\n },\n ]\n\n let merged: Record<string, unknown> = {}\n\n for (const layer of layers) {\n const data = await loadLayer(layer)\n\n if (data === null) continue\n\n merged = { ...merged, ...data }\n }\n\n const finalResult = infraKitConfigSchema.safeParse(merged)\n\n if (!finalResult.success) {\n throw new Error(`Invalid merged infra-kit config: ${z.prettifyError(finalResult.error)}`)\n }\n\n cached = { mtimes, value: finalResult.data }\n\n return finalResult.data\n}\n\n/**\n * For tests \u2014 drops the in-memory cache so the next read hits disk.\n *\n * @example\n * resetInfraKitConfigCache()\n * await getInfraKitConfig() // re-reads files even if mtimes look unchanged\n */\nexport const resetInfraKitConfigCache = (): void => {\n cached = null\n}\n\n/**\n * `fs.stat` that returns `null` instead of throwing on ENOENT. Used so the\n * resolver can probe optional files in the merge chain without try/catch noise.\n *\n * @example\n * const stat = await statIfExists('/does/not/exist') // => null\n */\nconst statIfExists = async (filePath: string): Promise<Awaited<ReturnType<typeof fs.stat>> | null> => {\n try {\n return await fs.stat(filePath)\n } catch {\n return null\n }\n}\n\n/**\n * `fs.readFile` that returns `null` instead of throwing on ENOENT.\n *\n * @example\n * const raw = await readIfExists('/missing.yml') // => null\n * const raw = await readIfExists('/exists.yml') // => 'environments: [dev]\\n'\n */\nconst readIfExists = async (filePath: string): Promise<string | null> => {\n try {\n return await fs.readFile(filePath, 'utf-8')\n } catch {\n return null\n }\n}\n\n/**\n * Reference-equality comparison of every key in two flat records. Used to\n * cheaply detect whether the cached mtime fingerprint still matches.\n *\n * @example\n * shallowEqual({ a: 1, b: 2 }, { a: 1, b: 2 }) // => true\n * shallowEqual({ a: 1 }, { a: 1, b: 2 }) // => false\n * shallowEqual({ a: 1 }, { a: 2 }) // => false\n */\nconst shallowEqual = <T extends Record<string, unknown>>(a: T, b: T): boolean => {\n const keys = Object.keys(a)\n\n if (keys.length !== Object.keys(b).length) return false\n\n return keys.every((k) => {\n return a[k] === b[k]\n })\n}\n\ninterface ConfigLayer {\n label: string\n path: string\n required: boolean\n}\n\n/**\n * Read a single layer of the merge chain: parse the YAML if the file exists\n * and validate it against the override schema. Returns `null` if an optional\n * layer is missing; throws if the layer is required or invalid.\n *\n * @example\n * await loadLayer({ label: '~/.infra-kit/config.yml', path: '/missing.yml', required: false })\n * // => null\n *\n * @example\n * // /home/me/.infra-kit/config.yml: 'ide:\\n provider: cursor\\n config: { mode: windows }'\n * await loadLayer({ label: '~/.infra-kit/config.yml', path: '/home/me/.infra-kit/config.yml', required: false })\n * // => { ide: { provider: 'cursor', config: { mode: 'windows' } } }\n */\nconst loadLayer = async (layer: ConfigLayer): Promise<Record<string, unknown> | null> => {\n const raw = await readIfExists(layer.path)\n\n if (raw === null) {\n if (layer.required) {\n throw new Error(`${layer.label} not found at ${layer.path}`)\n }\n\n return null\n }\n\n const parsedRaw = yaml.parse(raw) ?? {}\n const result = infraKitOverrideConfigSchema.safeParse(parsedRaw)\n\n if (!result.success) {\n throw new Error(`Invalid ${layer.label} at ${layer.path}: ${z.prettifyError(result.error)}`)\n }\n\n return result.data as Record<string, unknown>\n}\n", "import path from 'node:path'\nimport { $ } from 'zx'\n\n/**\n * Get current git worktrees\n *\n * @returns [release/v1.18.22, release/v1.18.23, release/v1.18.24] or [feature/mobile-app, feature/explore-page, feature/login-page]\n */\nexport const getCurrentWorktrees = async (type: 'release' | 'feature'): Promise<string[]> => {\n const worktreesOutput = await $`git worktree list`\n\n const worktreeLines = worktreesOutput.stdout.split('\\n').filter(Boolean)\n\n const worktreePredicateMap = {\n release: releaseWorktreePredicate,\n feature: featureWorktreePredicate,\n }\n\n return worktreeLines.map(worktreePredicateMap[type]).filter((branch) => {\n return branch !== null\n })\n}\n\n/**\n * Extract the branch name from a `git worktree list` output line.\n *\n * `git worktree list` formats each line as:\n * <path> <hash> [<branch>]\n *\n * Reads the branch from the trailing `[branch]` token so it works for the\n * main checkout too (whose path does not encode the branch name).\n */\nconst parseWorktreeBranch = (line: string): string | null => {\n const trimmed = line.trimEnd()\n\n if (!trimmed.endsWith(']')) return null\n\n const open = trimmed.lastIndexOf('[')\n\n if (open === -1) return null\n\n const branch = trimmed.slice(open + 1, -1)\n\n return branch.length > 0 ? branch : null\n}\n\n/**\n * Extract a release branch name from a `git worktree list` output line.\n *\n * Returns `null` for lines that are not release worktrees.\n *\n * @example\n * releaseWorktreePredicate('/path/to/release/v1.18.22 abc1234 [release/v1.18.22]')\n * // => 'release/v1.18.22'\n *\n * @example\n * releaseWorktreePredicate('/path/to/feature/login abc1234 [feature/login]')\n * // => null\n */\nconst releaseWorktreePredicate = (line: string): string | null => {\n const branch = parseWorktreeBranch(line)\n\n return branch?.startsWith('release/v') ? branch : null\n}\n\n/**\n * Extract a feature branch name from a `git worktree list` output line.\n *\n * Returns `null` for lines that are not feature worktrees.\n *\n * @example\n * featureWorktreePredicate('/path/to/feature/login-page abc1234 [feature/login-page]')\n * // => 'feature/login-page'\n *\n * @example\n * featureWorktreePredicate('/path/to/release/v1.18.22 abc1234 [release/v1.18.22]')\n * // => null\n */\nconst featureWorktreePredicate = (line: string): string | null => {\n const branch = parseWorktreeBranch(line)\n\n return branch?.startsWith('feature/') ? branch : null\n}\n\n/**\n * Get the current project root directory\n */\nexport const getProjectRoot = async (): Promise<string> => {\n const result = await $`git rev-parse --show-toplevel`\n\n return result.stdout.trim()\n}\n\n/**\n * Get the current repository name (basename of the project root)\n */\nexport const getRepoName = async (): Promise<string> => {\n const projectRoot = await getProjectRoot()\n\n return path.basename(projectRoot)\n}\n", "import { getInfraKitConfig } from 'src/lib/infra-kit-config'\n\n/**\n * Resolve Doppler project name from infra-kit.yml at the project root\n */\nexport const getDopplerProject = async (): Promise<string> => {\n const { envManagement } = await getInfraKitConfig()\n\n return envManagement.config.name\n}\n", "import select from '@inquirer/select'\nimport { Buffer } from 'node:buffer'\nimport fs from 'node:fs'\nimport path from 'node:path'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\nimport { $ } from 'zx'\n\nimport { validateDopplerCliAndAuth } from 'src/integrations/doppler'\nimport { getDopplerProject } from 'src/integrations/doppler/doppler-project'\nimport { commandEcho } from 'src/lib/command-echo'\nimport {\n ENV_LOAD_FILE,\n ENV_VAR_LINE_PATTERN,\n INFRA_KIT_ENV_CONFIG_VAR,\n INFRA_KIT_ENV_LOADED_AT_VAR,\n INFRA_KIT_ENV_PROJECT_VAR,\n atomicWriteFileSync,\n getSessionCacheDir,\n} from 'src/lib/constants'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface EnvLoadArgs {\n config?: string\n}\n\n/**\n * Load environment variables from Doppler for the given config\n */\nexport const envLoad = async (args: EnvLoadArgs) => {\n await validateDopplerCliAndAuth()\n\n const { config } = args\n\n commandEcho.start('env-load')\n\n let selectedConfig = ''\n\n if (config) {\n selectedConfig = config\n } else {\n const { environments } = await getInfraKitConfig()\n\n commandEcho.setInteractive()\n selectedConfig = await select(\n {\n message: 'Select environment config',\n choices: environments.map((env) => {\n return { name: env, value: env }\n }),\n },\n // Render to stderr so the prompt is visible when stdout is captured via $() in the shell function.\n // Only env-load and env-clear use the $() stdout-capture shell pattern.\n { output: process.stderr },\n )\n }\n\n commandEcho.addOption('--config', selectedConfig)\n\n const project = await getDopplerProject()\n\n const envContent = await downloadDopplerSecrets(project, selectedConfig)\n\n assertValidEnvContent(envContent)\n\n // Build env file content in dotenv format\n const loadedAt = new Date().toISOString()\n const envFileLines = [\n 'set -a',\n envContent,\n `${INFRA_KIT_ENV_CONFIG_VAR}=${shellSingleQuote(selectedConfig)}`,\n `${INFRA_KIT_ENV_PROJECT_VAR}=${shellSingleQuote(project)}`,\n `${INFRA_KIT_ENV_LOADED_AT_VAR}=${shellSingleQuote(loadedAt)}`,\n 'set +a',\n ]\n\n const cacheDir = getSessionCacheDir()\n const envFilePath = path.resolve(cacheDir, ENV_LOAD_FILE)\n\n fs.mkdirSync(cacheDir, { recursive: true, mode: 0o700 })\n atomicWriteFileSync(envFilePath, `${envFileLines.join('\\n')}\\n`, 0o600)\n\n // REQUIRED\n process.stdout.write(`${envFilePath}\\n`)\n\n // Logs to stderr (pino \u2192 pretty-print), so it doesn't pollute the captured\n // file path that the shell wrapper reads from stdout.\n commandEcho.print()\n\n const varCount = countEnvVarLines(envContent)\n\n const structuredContent = {\n filePath: envFilePath,\n variableCount: varCount,\n project,\n config: selectedConfig,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n/**\n * Cap the Doppler stdout we're willing to accept. A well-formed env bundle is\n * O(10 KB); megabytes would indicate a service regression or the wrong stream\n * being captured, and we don't want to write that to disk or source it.\n */\nexport const DOPPLER_MAX_OUTPUT_BYTES = 1024 * 1024\n\n/**\n * Hard upper bound for the Doppler subprocess. Well under zx's default so a\n * hung call surfaces quickly instead of blocking an interactive shell or an\n * MCP tool handler.\n */\nconst DOPPLER_DOWNLOAD_TIMEOUT_MS = 30_000\n\nconst downloadDopplerSecrets = async (project: string, config: string): Promise<string> => {\n const prevQuiet = $.quiet\n\n $.quiet = true\n try {\n const result =\n await $`doppler secrets download --no-file --format env --project ${project} --config ${config}`.timeout(\n DOPPLER_DOWNLOAD_TIMEOUT_MS,\n )\n\n assertDopplerOutputSize(result.stdout)\n\n return result.stdout.trim()\n } finally {\n $.quiet = prevQuiet\n }\n}\n\nexport const assertDopplerOutputSize = (stdout: string): void => {\n const bytes = Buffer.byteLength(stdout, 'utf-8')\n\n if (bytes > DOPPLER_MAX_OUTPUT_BYTES) {\n throw new Error(\n `doppler returned unexpectedly large output (${bytes} bytes > ${DOPPLER_MAX_OUTPUT_BYTES}) \u2014 refusing to write to disk`,\n )\n }\n}\n\nconst countEnvVarLines = (content: string): number => {\n return content.split('\\n').filter((line) => {\n return ENV_VAR_LINE_PATTERN.test(line)\n }).length\n}\n\nconst SHELL_DIRECTIVE_LINES = new Set(['set -a', 'set +a'])\n\nexport const shellSingleQuote = (value: string): string => {\n const escaped = value.replaceAll(\"'\", \"'\\\\''\")\n\n return `'${escaped}'`\n}\n\n/**\n * Guard against Doppler returning non-env output (auth warnings on stdout,\n * partial downloads, HTML error pages, etc.). Every non-blank, non-directive\n * line must match KEY=VALUE \u2014 skipping directives keeps future format tweaks\n * cheap without loosening the check.\n */\nexport const assertValidEnvContent = (content: string): void => {\n if (content.trim().length === 0) {\n throw new Error('doppler returned empty output for env-load')\n }\n\n for (const line of content.split('\\n')) {\n const trimmed = line.trim()\n\n if (trimmed.length === 0 || SHELL_DIRECTIVE_LINES.has(trimmed)) continue\n\n if (!ENV_VAR_LINE_PATTERN.test(trimmed)) {\n throw new Error(\n `doppler returned unexpected output for env-load (expected KEY=value lines, got: ${JSON.stringify(trimmed.slice(0, 80))})`,\n )\n }\n }\n}\n\n// MCP Tool Registration\nexport const envLoadMcpTool = defineMcpTool({\n name: 'env-load',\n description:\n 'Download the env vars for a Doppler config and write them to a temporary shell script. Does NOT mutate the calling process \u2014 returns the path to a script that must be sourced (\"source <filePath>\") for the vars to take effect. The infra-kit shell wrapper auto-sources; direct MCP callers must handle sourcing themselves or surface filePath to the user. \"config\" is required when invoked via MCP (the CLI interactive picker is unreachable without a TTY).',\n inputSchema: {\n config: z\n .string()\n .describe('Doppler config / environment name to load (e.g. \"dev\", \"arthur\", \"renana\"). Required for MCP calls.'),\n },\n outputSchema: {\n filePath: z.string().describe('Path to the file that must be sourced to apply variables'),\n variableCount: z.number().describe('Number of variables loaded'),\n project: z.string().describe('Doppler project name'),\n config: z.string().describe('Doppler config name'),\n },\n handler: envLoad,\n})\n", "import { $ } from 'zx'\n\n/**\n * Validate Doppler CLI installation and authentication status. Throws on failure\n * so callers (CLI entry, MCP tool handler) can translate to the right surface \u2014\n * CLI exits non-zero; MCP returns a structured tool error instead of tearing\n * down the server.\n */\nexport const validateDopplerCliAndAuth = async (): Promise<void> => {\n try {\n await $`doppler --version`\n } catch (error: unknown) {\n throw new Error('Doppler CLI is not installed. Install it from: https://docs.doppler.com/docs/install-cli', {\n cause: error,\n })\n }\n\n try {\n await $`doppler me`\n } catch (error: unknown) {\n throw new Error('Doppler CLI is not authenticated. Run: doppler login', { cause: error })\n }\n}\n", "import { logger } from 'src/lib/logger'\n\ninterface CommandOption {\n flag: string\n value: string | string[] | boolean\n}\n\nconst createCommandEcho = () => {\n let commandName = ''\n let options: CommandOption[] = []\n let isInteractive = false\n\n return {\n /**\n * Initialize command echo for a new command\n */\n start(name: string): void {\n commandName = name\n options = []\n isInteractive = false\n },\n\n /**\n * Mark that the command had interactive input (prompts)\n * Call this once when ANY prompt happens\n */\n setInteractive(): void {\n isInteractive = true\n },\n\n /**\n * Track an option selection\n * @param flag The CLI flag (e.g., \"--versions\")\n * @param value The selected value\n */\n addOption(flag: string, value: string | string[] | boolean): void {\n options.push({ flag, value })\n },\n\n /**\n * Print the equivalent CLI command if there was interactive input\n */\n print(): void {\n if (!isInteractive || options.length === 0) {\n return\n }\n\n const formattedOptions = options\n .map((opt) => {\n if (typeof opt.value === 'boolean') {\n return opt.value ? opt.flag : ''\n }\n\n if (Array.isArray(opt.value)) {\n return `${opt.flag} \"${opt.value.join(', ')}\"`\n }\n\n return `${opt.flag} \"${opt.value}\"`\n })\n .filter(Boolean)\n .join(' ')\n\n logger.info(`\uD83D\uDCDF Equivalent command: \\npnpm exec infra-kit ${commandName} ${formattedOptions}\\n`)\n },\n\n /**\n * Reset state (useful for testing)\n */\n reset(): void {\n commandName = ''\n options = []\n isInteractive = false\n },\n }\n}\n\n// Singleton instance (same pattern as logger)\nexport const commandEcho = createCommandEcho()\n", "import path from 'node:path'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\n\nimport { validateDopplerCliAndAuth } from 'src/integrations/doppler'\nimport {\n ENV_LOAD_FILE,\n INFRA_KIT_ENV_CONFIG_VAR,\n INFRA_KIT_ENV_LOADED_AT_VAR,\n INFRA_KIT_ENV_PROJECT_VAR,\n INFRA_KIT_SESSION_VAR,\n getSessionCacheDir,\n parseVarNamesFromEnvFile,\n} from 'src/lib/constants'\nimport { logger } from 'src/lib/logger'\nimport { defineMcpTool, textContent } from 'src/types'\n\n/**\n * Show Doppler authentication status and detected project info\n */\nexport const envStatus = async () => {\n await validateDopplerCliAndAuth()\n\n logger.info('Environment session status:')\n\n // Check session-loaded vars \u2014 getSessionCacheDir() throws if INFRA_KIT_SESSION is unset\n const cacheDir = getSessionCacheDir()\n\n const sessionId = process.env[INFRA_KIT_SESSION_VAR]!\n const envLoadPath = path.join(cacheDir, ENV_LOAD_FILE)\n\n let sessionLoadedCount = 0\n let sessionTotalCount = 0\n\n const sessionConfig = process.env[INFRA_KIT_ENV_CONFIG_VAR] ?? null\n const sessionProject = process.env[INFRA_KIT_ENV_PROJECT_VAR] ?? null\n const sessionLoadedAt = process.env[INFRA_KIT_ENV_LOADED_AT_VAR] ?? null\n\n if (sessionConfig) {\n const varNames = parseVarNamesFromEnvFile(envLoadPath)\n\n if (varNames.length > 0) {\n sessionTotalCount = varNames.length\n sessionLoadedCount = varNames.filter((v) => {\n return v in process.env\n }).length\n }\n\n const loadedAtDisplay = sessionLoadedAt?.replace(/\\.\\d{3}Z$/, '') ?? null\n\n logger.info(\n ` ${sessionConfig}: ${sessionLoadedCount} of ${sessionTotalCount} vars loaded (project: ${sessionProject}, loadedAt: ${loadedAtDisplay}, session: ${sessionId})\\n`,\n )\n\n if (sessionTotalCount > 0 && sessionLoadedCount < sessionTotalCount) {\n const missing = sessionTotalCount - sessionLoadedCount\n\n logger.warn(\n ` ${missing} cached var(s) are not present in the current process \u2014 env-load needs to be re-sourced, or vars were unset manually.`,\n )\n }\n } else {\n logger.info(` Session ${sessionId}: no env loaded\\n`)\n }\n\n const structuredContent = {\n sessionId,\n sessionLoadedCount,\n sessionTotalCount,\n sessionConfig,\n sessionProject,\n sessionLoadedAt,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const envStatusMcpTool = defineMcpTool({\n name: 'env-status',\n description:\n 'Report which Doppler project/config is currently loaded in the terminal session, when it was loaded, and how many variables are cached. Read-only \u2014 use env-load / env-clear to change the terminal session.',\n inputSchema: {},\n outputSchema: {\n sessionId: z.string().describe('Current terminal session ID'),\n sessionLoadedCount: z.number().describe('Number of cached vars active in the current session'),\n sessionTotalCount: z.number().describe('Total number of cached var names'),\n sessionConfig: z.string().nullable().describe('Doppler config name of the loaded session (environment name)'),\n sessionProject: z.string().nullable().describe('Doppler project name of the loaded session'),\n sessionLoadedAt: z.string().nullable().describe('ISO 8601 timestamp of when the env was loaded'),\n },\n handler: envStatus,\n})\n", "/* eslint-disable sonarjs/cognitive-complexity */\nimport checkbox from '@inquirer/checkbox'\nimport confirm from '@inquirer/confirm'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\nimport { $ } from 'zx'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\ninterface GhMergeDevArgs extends RequiredConfirmedOptionArg {\n all?: boolean\n}\n\n/**\n * Merge dev into every release branch\n */\nexport const ghMergeDev = async (args: GhMergeDevArgs) => {\n const { all, confirmedCommand } = args\n\n commandEcho.start('merge-dev')\n\n // Only merge dev into regular releases (not hotfixes, which target main)\n const allPRs = await getReleasePRsWithInfo()\n const releasePRsList = allPRs\n .filter((pr) => {\n return detectReleaseType(pr.title) === 'regular'\n })\n .map((pr) => {\n return pr.branch\n })\n\n if (releasePRsList.length === 0) {\n logger.info('\u2139\uFE0F No open release branches found')\n\n commandEcho.print()\n\n return {\n content: textContent(\n JSON.stringify({ successfulMerges: 0, failedMerges: 0, failedBranches: [], totalBranches: 0 }, null, 2),\n ),\n structuredContent: { successfulMerges: 0, failedMerges: 0, failedBranches: [], totalBranches: 0 },\n }\n }\n\n let selectedReleaseBranches: string[] = []\n\n if (all) {\n selectedReleaseBranches = releasePRsList\n } else {\n commandEcho.setInteractive()\n\n const descriptions = await getJiraDescriptions()\n\n selectedReleaseBranches = await checkbox({\n required: true,\n message: '\uD83C\uDF3F Select release branches',\n choices: formatBranchChoices({ branches: releasePRsList, descriptions }),\n })\n }\n\n // Track --all flag if all branches were selected (either via flag or interactively)\n const allSelected = selectedReleaseBranches.length === releasePRsList.length\n\n if (allSelected) {\n commandEcho.addOption('--all', true)\n } else {\n commandEcho.addOption(\n '--versions',\n selectedReleaseBranches.map((branch) => {\n return branch.replace('release/v', '')\n }),\n )\n }\n\n // Validate input\n // if (selectedReleaseBranches.length === 0) {\n // console.error('No branches provided. Exiting...')\n // process.exit(1)\n // }\n\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: `Are you sure you want to merge dev into these branches: ${selectedReleaseBranches.join(', ')}?`,\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n if (!confirmedCommand) {\n commandEcho.addOption('--yes', true)\n }\n\n $.quiet = true\n\n await $`git fetch origin`\n await $`git switch dev`\n await $`git pull origin dev`\n\n const failedBranches: string[] = []\n\n // Merge dev into each branch\n for (const branch of selectedReleaseBranches) {\n const success = await mergeDev(branch)\n\n if (!success) {\n failedBranches.push(branch)\n }\n }\n\n $.quiet = false\n\n if (failedBranches.length > 0) {\n logger.info(`\\n\u26A0\uFE0F ${failedBranches.length} branch(es) failed to merge automatically.\\n`)\n logger.info('\uD83D\uDCCB Manual merge script for failed branches:')\n for (const branch of failedBranches) {\n logger.info(\n `# Merge dev into ${branch} and resolve conflicts if any \\n\\ngit switch ${branch} && git pull origin ${branch} && git merge origin/dev\\ngit push origin ${branch} && git switch dev\\n`,\n )\n }\n logger.info(\n `\u2705 ${selectedReleaseBranches.length - failedBranches.length}/${selectedReleaseBranches.length} merges completed successfully.`,\n )\n } else {\n logger.info('\u2705 All merges completed successfully!\\n')\n }\n\n commandEcho.print()\n\n const structuredContent = {\n successfulMerges: selectedReleaseBranches.length - failedBranches.length,\n failedMerges: failedBranches.length,\n failedBranches,\n totalBranches: selectedReleaseBranches.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\nconst mergeDev = async (branch: string): Promise<boolean> => {\n try {\n await $`git switch ${branch}`\n\n await $`git pull origin ${branch}`\n\n await $`git merge origin/dev --no-edit`\n\n await $`git push origin ${branch}`\n\n await $`git switch dev`\n\n logger.info(`Successfully merged dev into ${branch}`)\n\n return true\n } catch (error: unknown) {\n const err = new OperationError(error, {\n operation: `merge dev into ${branch}`,\n remediation: \"resolve conflicts manually or rerun after 'git fetch origin'\",\n })\n\n logger.error({ error, branch, msg: err.message })\n\n await $`git reset --merge HEAD~1`\n\n return false\n }\n}\n\n// MCP Tool Registration\nexport const ghMergeDevMcpTool = defineMcpTool({\n name: 'gh-merge-dev',\n description:\n 'Merge origin/dev into every open regular (non-hotfix) release branch and push the result. Mutates local git state and the remote release branches. When invoked via MCP, pass all=true \u2014 the branch picker is unreachable without a TTY, and the confirmation prompt is auto-skipped for MCP calls, so the caller is responsible for gating. Irreversible once pushed.',\n inputSchema: {\n all: z\n .boolean()\n .optional()\n .describe(\n 'Target every open regular release branch. Must be true for MCP calls (the interactive picker is unavailable without a TTY).',\n ),\n },\n outputSchema: {\n successfulMerges: z.number().describe('Number of successful merges'),\n failedMerges: z.number().describe('Number of failed merges'),\n failedBranches: z.array(z.string()).describe('List of branches that failed to merge'),\n totalBranches: z.number().describe('Total number of branches processed'),\n },\n handler: ghMergeDev,\n})\n", "import process from 'node:process'\nimport { $ } from 'zx'\n\nimport { logger } from 'src/lib/logger'\n\n/**\n * Validate GitHub CLI installation and authentication status and throw an error if not valid\n */\nexport const validateGitHubCliAndAuth = async () => {\n try {\n await $`gh --version`\n } catch (error: unknown) {\n logger.error({ error }, 'Error: GitHub CLI (gh) is not installed.')\n logger.error('Please install it from: https://cli.github.com/')\n process.exit(1)\n }\n\n try {\n await $`gh auth status`\n } catch (error: unknown) {\n logger.error({ error }, 'Error: GitHub CLI (gh) is not authenticated.')\n logger.error('Please authenticate it from: https://cli.github.com/manual/gh_auth_login or type \"gh auth login\"')\n process.exit(1)\n }\n}\n", "import process from 'node:process'\nimport { $ } from 'zx'\n\nimport { logger } from 'src/lib/logger'\nimport { getBaseBranch } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { sortVersions } from 'src/lib/version-utils'\n\ninterface ReleasePR {\n headRefName: string\n number: number\n state: string\n title: string\n baseRefName: string\n}\n\nexport interface ReleasePRInfo {\n branch: string\n title: string\n}\n\n/**\n * Fetch all open release/hotfix PRs from GitHub.\n * Searches both dev (regular) and main (hotfix) base branches.\n * Returns deduplicated ReleasePR objects.\n */\nconst fetchAllReleasePRs = async (): Promise<ReleasePR[]> => {\n const releasePRs =\n await $`gh pr list --search \"Release in:title\" --base dev --json number,title,headRefName,state,baseRefName`\n\n const hotfixPRs =\n await $`gh pr list --search \"Hotfix in:title\" --base main --json number,title,headRefName,state,baseRefName`\n\n const all: ReleasePR[] = [...JSON.parse(releasePRs.stdout), ...JSON.parse(hotfixPRs.stdout)]\n\n // Deduplicate by headRefName\n const seen = new Set<string>()\n\n return all.filter((pr) => {\n if (seen.has(pr.headRefName)) return false\n\n seen.add(pr.headRefName)\n\n return true\n })\n}\n\n/**\n * Fetch open release PRs from GitHub with 'Release' or 'Hotfix' in the title and base 'dev'.\n * Returns an array of headRefName strings sorted by semver in ascending order.\n *\n * @returns [release/v1.18.22, release/v1.18.23, release/v1.18.24] (sorted by semver)\n */\nexport const getReleasePRs = async (): Promise<string[]> => {\n try {\n const prs = await fetchAllReleasePRs()\n\n if (prs.length === 0) {\n logger.error('\u274C No release PRs found. Check the project folder for the script. Exiting...')\n\n process.exit(1)\n }\n\n return sortVersions(\n prs.map((pr) => {\n return pr.headRefName\n }),\n )\n } catch (error) {\n logger.error({ error }, '\u274C Error fetching release PRs')\n\n process.exit(1)\n }\n}\n\n/**\n * Fetch open release PRs with title info (for detecting release type).\n * Returns ReleasePRInfo objects sorted by semver.\n */\nexport const getReleasePRsWithInfo = async (): Promise<ReleasePRInfo[]> => {\n try {\n const prs = await fetchAllReleasePRs()\n\n if (prs.length === 0) {\n logger.error('\u274C No release PRs found. Check the project folder for the script. Exiting...')\n process.exit(1)\n }\n\n const sortedBranches = sortVersions(\n prs.map((pr) => {\n return pr.headRefName\n }),\n )\n const prByBranch = new Map(\n prs.map((pr) => {\n return [pr.headRefName, pr]\n }),\n )\n\n return sortedBranches.map((branch) => {\n return {\n branch,\n title: prByBranch.get(branch)!.title,\n }\n })\n } catch (error) {\n logger.error({ error }, '\u274C Error fetching release PRs')\n process.exit(1)\n }\n}\n\ninterface UpdateReleasePRBodyArgs {\n branch: string\n body: string\n}\n\n/**\n * Update the body of an open release PR identified by its head branch.\n */\nexport const updateReleasePRBody = async (args: UpdateReleasePRBodyArgs): Promise<void> => {\n const { branch, body } = args\n\n try {\n $.quiet = true\n await $`gh pr edit ${branch} --body ${body}`\n $.quiet = false\n } catch (error: unknown) {\n logger.error({ error, branch }, `Error updating release PR body for ${branch}`)\n throw error\n }\n}\n\ninterface CreateReleaseBranchArgs {\n version: string\n jiraVersionUrl: string\n type: ReleaseType\n description?: string\n}\n\n// Function to create a release branch\nexport const createReleaseBranch = async (\n args: CreateReleaseBranchArgs,\n): Promise<{ branchName: string; prUrl: string }> => {\n const { version, jiraVersionUrl, type, description } = args\n const titlePrefix = type === 'hotfix' ? 'Hotfix' : 'Release'\n const baseBranch = getBaseBranch(type)\n\n const branchName = `release/v${version}`\n\n const body = description && description.trim() !== '' ? `${jiraVersionUrl}\\n\\n${description}` : `${jiraVersionUrl} \\n`\n\n try {\n $.quiet = true\n\n await $`git switch ${baseBranch}`\n await $`git pull origin ${baseBranch}`\n await $`git checkout -b ${branchName}`\n await $`git push -u origin ${branchName}`\n await $`git commit --allow-empty-message --allow-empty --message ''`\n await $`git push origin ${branchName}`\n\n // Create PR and capture URL\n const prResult =\n await $`gh pr create --title \"${titlePrefix} v${version}\" --body ${body} --base ${baseBranch} --head ${branchName}`\n\n const prLink = prResult.stdout.trim()\n\n await $`git switch ${baseBranch}`\n\n $.quiet = false\n\n return {\n branchName,\n prUrl: prLink,\n }\n } catch (error: unknown) {\n logger.error({ error, branchName }, `Error creating release branch ${branchName}`)\n\n throw error\n }\n}\n", "import { $ } from 'zx'\n\nimport { createReleaseBranch } from 'src/integrations/gh'\nimport { createJiraVersion, getProjectVersions, loadJiraConfigOptional } from 'src/integrations/jira'\nimport type { JiraConfig } from 'src/integrations/jira'\n\nexport type ReleaseType = 'regular' | 'hotfix'\n\n/**\n * Get the base branch for a release type.\n * Regular releases branch from/to dev, hotfixes branch from/to main.\n */\nexport const getBaseBranch = (type: ReleaseType): string => {\n return type === 'hotfix' ? 'main' : 'dev'\n}\n\nexport interface ReleaseCreationResult {\n version: string\n type: ReleaseType\n branchName: string\n prUrl: string\n jiraVersionUrl: string\n}\n\n/**\n * Prepare git repository for release creation\n * Fetches latest changes, switches to base branch, and pulls latest\n */\nexport const prepareGitForRelease = async (type: ReleaseType = 'regular'): Promise<void> => {\n const baseBranch = getBaseBranch(type)\n\n $.quiet = true\n\n await $`git fetch origin`\n await $`git switch ${baseBranch}`\n await $`git pull origin ${baseBranch}`\n\n $.quiet = false\n}\n\ninterface CreateSingleReleaseArgs {\n version: string\n jiraConfig: JiraConfig\n description?: string\n type?: ReleaseType\n}\n\n/**\n * Create a single release by creating both Jira version and GitHub release branch\n */\nexport const createSingleRelease = async (args: CreateSingleReleaseArgs): Promise<ReleaseCreationResult> => {\n const { version, jiraConfig, description, type = 'regular' } = args\n // 1. Create Jira version (mandatory)\n const versionName = `v${version}`\n\n const result = await createJiraVersion(\n {\n name: versionName,\n projectId: jiraConfig.projectId,\n description: description || '',\n released: false,\n archived: false,\n },\n jiraConfig,\n )\n\n // Construct user-friendly Jira URL using project key from API response\n const jiraVersionUrl = `${jiraConfig.baseUrl}/projects/${result.version!.projectId}/versions/${result.version!.id}/tab/release-report-all-issues`\n\n // 2. Create GitHub release branch\n const releaseInfo = await createReleaseBranch({ version, jiraVersionUrl, type, description })\n\n return {\n version,\n type,\n branchName: releaseInfo.branchName,\n prUrl: releaseInfo.prUrl,\n jiraVersionUrl,\n }\n}\n\n/**\n * Fetch Jira version descriptions mapped by version name (e.g., \"v1.2.5\" \u2192 \"Some description\")\n * Gracefully returns empty map if Jira is unavailable\n */\nexport const getJiraDescriptions = async (): Promise<Map<string, string>> => {\n const descriptions = new Map<string, string>()\n\n const jiraConfig = await loadJiraConfigOptional()\n\n if (!jiraConfig) return descriptions\n\n try {\n const versions = await getProjectVersions(jiraConfig)\n\n for (const version of versions) {\n if (version.description) {\n descriptions.set(version.name, version.description)\n }\n }\n } catch {\n // Jira fetch failed, continue without descriptions\n }\n\n return descriptions\n}\n\n/**\n * Format a version string with its release type tag, e.g. \"1.2.5 [regular]\"\n * When maxVersionLength is provided, pads the version for alignment.\n */\nexport const formatVersionLabel = (version: string, type: ReleaseType, maxVersionLength?: number): string => {\n const padding = maxVersionLength ? ' '.repeat(maxVersionLength - version.length + 3) : ' '\n const tag = `[${type}]`.padEnd(11)\n\n return `${version}${padding}${tag}`\n}\n\n/**\n * Detect release type from PR title.\n * PRs titled \"Hotfix v...\" are hotfix, everything else is regular.\n */\nexport const detectReleaseType = (title: string): ReleaseType => {\n return title.toLowerCase().startsWith('hotfix') ? 'hotfix' : 'regular'\n}\n\ninterface FormatBranchChoicesArgs {\n branches: string[]\n descriptions: Map<string, string>\n types?: Map<string, ReleaseType>\n}\n\n/**\n * Format release branch names as checkbox choices with aligned type tags and Jira descriptions\n */\nexport const formatBranchChoices = (args: FormatBranchChoicesArgs): { name: string; value: string }[] => {\n const { branches, descriptions, types } = args\n\n const versionNames = branches.map((b) => {\n return b.replace('release/v', '')\n })\n\n const maxLen = Math.max(\n ...versionNames.map((v) => {\n return v.length\n }),\n )\n\n return branches.map((branch, i) => {\n const version = versionNames[i] as string\n const type = types ? types.get(branch) || 'regular' : undefined\n const desc = descriptions.get(`v${version}`)\n const padding = ' '.repeat(maxLen - version.length + 3)\n\n let name = type ? formatVersionLabel(version, type, maxLen) : version\n\n if (desc) {\n name = type ? `${name} ${desc}` : `${version}${padding}${desc}`\n }\n\n return { name, value: branch }\n })\n}\n", "import process from 'node:process'\n\nimport { logger } from 'src/lib/logger'\n\nimport type {\n CreateJiraVersionParams,\n CreateJiraVersionResult,\n DeliverJiraReleaseParams,\n DeliverJiraReleaseResult,\n JiraConfig,\n JiraVersion,\n UpdateJiraVersionParams,\n UpdateJiraVersionResult,\n} from './types.js'\n\n/**\n * Creates a new version in Jira using the REST API\n * @param params - Version creation parameters\n * @param config - Jira configuration (baseUrl, token, projectId)\n * @returns Result containing created version or error\n */\nexport const createJiraVersion = async (\n params: CreateJiraVersionParams,\n config: JiraConfig,\n): Promise<CreateJiraVersionResult> => {\n try {\n const { baseUrl, token, email, projectId } = config\n\n // Use current date if not provided\n // const releaseDate =\n // params.releaseDate || new Date().toISOString().split('T')[0] // 2025-12-06\n\n // Prepare request body\n const requestBody = {\n name: params.name,\n projectId: params.projectId || projectId,\n description: params.description || '',\n // releaseDate,\n released: params.released || false,\n archived: params.archived || false,\n }\n\n // logger.info(\n // { version: params.name, projectId: requestBody.projectId },\n // 'Creating Jira version',\n // )\n\n // Make API request\n const url = `${baseUrl}/rest/api/3/version`\n\n // Create Basic auth credentials\n const credentials = btoa(`${email}:${token}`)\n\n const response = await fetch(url, {\n method: 'POST',\n headers: {\n Accept: 'application/json',\n 'Content-Type': 'application/json',\n Authorization: `Basic ${credentials}`,\n },\n body: JSON.stringify(requestBody),\n })\n\n if (!response.ok) {\n const errorText = await response.text()\n\n logger.error(\n {\n status: response.status,\n statusText: response.statusText,\n error: errorText,\n },\n 'Failed to create Jira version',\n )\n\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const version = (await response.json()) as JiraVersion\n\n // logger.info(\n // { versionId: version.id, versionName: version.name },\n // 'Successfully created Jira version',\n // )\n\n return {\n success: true,\n version,\n }\n } catch (error) {\n logger.error({ error }, 'Error creating Jira version')\n\n throw error\n }\n}\n\n/**\n * Gets all versions for a project from Jira\n * @param config - Jira configuration\n * @returns Array of JiraVersion objects\n */\nexport const getProjectVersions = async (config: JiraConfig): Promise<JiraVersion[]> => {\n try {\n const { baseUrl, token, email, projectId } = config\n\n const url = `${baseUrl}/rest/api/3/project/${projectId}/versions`\n const credentials = btoa(`${email}:${token}`)\n\n const response = await fetch(url, {\n method: 'GET',\n headers: {\n Accept: 'application/json',\n Authorization: `Basic ${credentials}`,\n },\n })\n\n if (!response.ok) {\n const errorText = await response.text()\n\n logger.error(\n {\n status: response.status,\n statusText: response.statusText,\n error: errorText,\n },\n 'Failed to get Jira project versions',\n )\n\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const versions = (await response.json()) as JiraVersion[]\n\n return versions\n } catch (error) {\n logger.error({ error }, 'Error getting Jira project versions')\n\n throw error\n }\n}\n\n/**\n * Finds a Jira version by name in the project\n * @param versionName - Name of the version to find (e.g., \"v1.33.10\")\n * @param config - Jira configuration\n * @returns JiraVersion if found, null otherwise\n */\nexport const findVersionByName = async (versionName: string, config: JiraConfig): Promise<JiraVersion | null> => {\n try {\n const versions = await getProjectVersions(config)\n const version = versions.find((v) => {\n return v.name === versionName\n })\n\n return version || null\n } catch (error) {\n logger.error({ error, versionName }, 'Error finding Jira version by name')\n\n throw error\n }\n}\n\n/**\n * Updates an existing Jira version\n * @param params - Update parameters\n * @param config - Jira configuration\n * @returns Result containing updated version or error\n */\nexport const updateJiraVersion = async (\n params: UpdateJiraVersionParams,\n config: JiraConfig,\n): Promise<UpdateJiraVersionResult> => {\n try {\n const { baseUrl, token, email } = config\n\n // Only include fields the caller explicitly passed.\n const requestBody: Record<string, any> = {}\n\n if (params.released !== undefined) requestBody.released = params.released\n if (params.archived !== undefined) requestBody.archived = params.archived\n if (params.releaseDate !== undefined) requestBody.releaseDate = params.releaseDate\n if (params.description !== undefined) requestBody.description = params.description\n\n const url = `${baseUrl}/rest/api/3/version/${params.versionId}`\n const credentials = btoa(`${email}:${token}`)\n\n const response = await fetch(url, {\n method: 'PUT',\n headers: {\n Accept: 'application/json',\n 'Content-Type': 'application/json',\n Authorization: `Basic ${credentials}`,\n },\n body: JSON.stringify(requestBody),\n })\n\n if (!response.ok) {\n const errorText = await response.text()\n\n logger.error(\n {\n status: response.status,\n statusText: response.statusText,\n error: errorText,\n },\n 'Failed to update Jira version',\n )\n\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const version = (await response.json()) as JiraVersion\n\n return {\n success: true,\n version,\n }\n } catch (error) {\n logger.error({ error }, 'Error updating Jira version')\n\n throw error\n }\n}\n\n/**\n * Delivers a Jira release by marking it as released with the current date\n * @param params - Parameters containing the version name\n * @param config - Jira configuration\n * @returns Result containing updated version\n * @throws Error if version not found or update fails\n */\nexport const deliverJiraRelease = async (\n params: DeliverJiraReleaseParams,\n config: JiraConfig,\n): Promise<DeliverJiraReleaseResult> => {\n try {\n const { versionName } = params\n\n // Find the version by name\n const version = await findVersionByName(versionName, config)\n\n if (!version) {\n logger.error({ versionName }, 'Jira version not found')\n throw new Error(`Version \"${versionName}\" not found in Jira project`)\n }\n\n // Update the version to mark it as released\n const result = await updateJiraVersion(\n {\n versionId: version.id,\n released: true,\n releaseDate: new Date().toISOString().split('T')[0], // Current date in YYYY-MM-DD format\n },\n config,\n )\n\n return result\n } catch (error) {\n logger.error({ error }, 'Error delivering Jira release')\n throw error\n }\n}\n\n/**\n * Loads Jira configuration from environment variables\n * @throws Error with detailed message if configuration is missing or invalid\n * @returns Promise<JiraConfig>\n */\nexport const loadJiraConfig = async (): Promise<JiraConfig> => {\n const baseUrl = process.env.JIRA_BASE_URL\n const token = process.env.JIRA_TOKEN || process.env.JIRA_API_TOKEN\n const projectIdStr = process.env.JIRA_PROJECT_ID\n const email = process.env.JIRA_EMAIL\n\n const missingVars: string[] = []\n\n if (!baseUrl) missingVars.push('JIRA_BASE_URL (e.g., https://your-domain.atlassian.net)')\n if (!token) missingVars.push('JIRA_TOKEN or JIRA_API_TOKEN (your Jira API token)')\n if (!projectIdStr) missingVars.push('JIRA_PROJECT_ID (numeric project ID)')\n if (!email) missingVars.push('JIRA_EMAIL (your Jira email address)')\n\n if (missingVars.length > 0) {\n const errorMessage = [\n 'Jira configuration is required but incomplete.',\n 'Please configure the following environment variables:',\n ...missingVars.map((v) => {\n return ` - ${v}`\n }),\n '',\n 'You can set these in your .env file or as environment variables.',\n ].join('\\n')\n\n throw new Error(errorMessage)\n }\n\n const projectId = Number.parseInt(projectIdStr!, 10)\n\n if (Number.isNaN(projectId)) {\n throw new TypeError(`Invalid JIRA_PROJECT_ID: \"${projectIdStr}\" must be a numeric value (e.g., 10001)`)\n }\n\n return {\n baseUrl: baseUrl!.replace(/\\/$/, ''), // Remove trailing slash\n token: token!,\n projectId,\n email: email!,\n }\n}\n\n/**\n * Attempts to load Jira configuration from environment variables\n * Returns null if configuration is missing or invalid (for optional Jira integration)\n * @returns Promise<JiraConfig | null>\n */\nexport const loadJiraConfigOptional = async (): Promise<JiraConfig | null> => {\n try {\n const config = await loadJiraConfig()\n\n return config\n } catch (error) {\n logger.warn({ error }, 'Jira configuration not available, skipping Jira integration')\n\n return null\n }\n}\n", "import { $ } from 'zx'\n\nimport { getProjectVersions, loadJiraConfigOptional } from 'src/integrations/jira'\nimport { logger } from 'src/lib/logger'\n\nimport { collectKnownVersions } from './next-version'\nimport type { SemVer } from './next-version'\n\nconst parseRemoteRefs = async (): Promise<string[]> => {\n const previousQuiet = $.quiet\n\n try {\n $.quiet = true\n const result = await $`git ls-remote --heads origin 'release/v*'`\n const lines = result.stdout.split('\\n')\n\n return lines\n .map((line) => {\n const tab = line.indexOf('\\t')\n\n if (tab === -1) return ''\n\n return line.slice(tab + 1).replace(/^refs\\/heads\\//, '')\n })\n .filter(Boolean)\n } finally {\n $.quiet = previousQuiet\n }\n}\n\nconst fetchJiraVersionNames = async (): Promise<string[]> => {\n const config = await loadJiraConfigOptional()\n\n if (!config) return []\n\n const versions = await getProjectVersions(config)\n\n return versions.map((v) => {\n return v.name\n })\n}\n\n/**\n * Load known release versions from the union of:\n * - remote release branches (`release/v*` on origin)\n * - Jira fix versions (when configured)\n *\n * Each source is queried in parallel; if either fails, we log a warning\n * and continue with the other so a transient outage doesn't block release\n * creation.\n */\nexport const loadExistingVersions = async (): Promise<SemVer[]> => {\n const [branchesResult, jiraResult] = await Promise.allSettled([parseRemoteRefs(), fetchJiraVersionNames()])\n\n if (branchesResult.status === 'rejected') {\n logger.warn({ error: branchesResult.reason }, 'Failed to list remote release branches; continuing without them')\n }\n\n if (jiraResult.status === 'rejected') {\n logger.warn({ error: jiraResult.reason }, 'Failed to fetch Jira versions; continuing without them')\n }\n\n return collectKnownVersions({\n remoteBranches: branchesResult.status === 'fulfilled' ? branchesResult.value : [],\n jiraVersions: jiraResult.status === 'fulfilled' ? jiraResult.value : [],\n })\n}\n", "/**\n * Parse version string into major, minor, patch numbers\n */\nexport const parseVersion = (versionStr: string): [number, number, number] => {\n return versionStr.replace('release/', '').slice(1).split('.').map(Number) as [number, number, number]\n}\n\n/**\n * Sort version strings in ascending order\n * Note: Returns a new sorted array without mutating the original\n */\nexport const sortVersions = (versions: string[]): string[] => {\n return [...versions].sort((a, b) => {\n const [majA, minA, patchA] = parseVersion(a)\n const [majB, minB, patchB] = parseVersion(b)\n\n if (majA !== majB) return (majA ?? 0) - (majB ?? 0)\n if (minA !== minB) return (minA ?? 0) - (minB ?? 0)\n\n return (patchA ?? 0) - (patchB ?? 0)\n })\n}\n", "import type { ReleaseType } from 'src/lib/release-utils'\n\nimport { parseVersion, sortVersions } from './version-utils'\n\nexport const NEXT_TOKEN = 'next'\n\nexport type SemVer = readonly [number, number, number]\n\nexport interface ExistingVersionsSources {\n remoteBranches?: string[]\n jiraVersions?: string[]\n}\n\nconst VERSION_RE = /^v?(\\d+)\\.(\\d+)\\.(\\d+)$/\n\nconst stripBranchPrefix = (raw: string): string => {\n return raw.replace(/^.*release\\//, '')\n}\n\nconst tryParse = (raw: string): SemVer | null => {\n const cleaned = stripBranchPrefix(raw.trim())\n const match = VERSION_RE.exec(cleaned)\n\n if (!match) return null\n\n return [Number(match[1]), Number(match[2]), Number(match[3])]\n}\n\nconst semverKey = (v: SemVer): string => {\n return `${v[0]}.${v[1]}.${v[2]}`\n}\n\nexport const collectKnownVersions = (sources: ExistingVersionsSources): SemVer[] => {\n const all = [...(sources.remoteBranches ?? []), ...(sources.jiraVersions ?? [])]\n const parsed: SemVer[] = []\n const seen = new Set<string>()\n\n for (const raw of all) {\n const v = tryParse(raw)\n\n if (!v) continue\n\n const key = semverKey(v)\n\n if (seen.has(key)) continue\n\n seen.add(key)\n parsed.push(v)\n }\n\n return sortVersions(\n parsed.map((v) => {\n return semverKey(v)\n }),\n ).map((s) => {\n return parseVersion(`v${s}`)\n })\n}\n\nexport class NoPriorVersionsError extends Error {\n constructor() {\n super('No prior release versions found from git or Jira. Specify the version explicitly.')\n this.name = 'NoPriorVersionsError'\n }\n}\n\n/**\n * Compute the next semantic version based on release type.\n * - regular: bump minor, reset patch to 0\n * - hotfix: bump patch on the highest minor (any patch)\n *\n * Returns the version without the leading \"v\" (e.g. \"1.64.0\").\n */\nexport const computeNextVersion = (known: SemVer[], type: ReleaseType): string => {\n if (known.length === 0) throw new NoPriorVersionsError()\n\n const max = known[known.length - 1] as SemVer\n\n if (type === 'hotfix') {\n const [major, minor] = max\n\n const highestPatchOnMinor = known.reduce((acc, v) => {\n if (v[0] === major && v[1] === minor) return Math.max(acc, v[2])\n\n return acc\n }, 0)\n\n return `${major}.${minor}.${highestPatchOnMinor + 1}`\n }\n\n const [major, minor] = max\n\n return `${major}.${minor + 1}.0`\n}\n\nconst isNextToken = (token: string): boolean => {\n return token.trim().toLowerCase() === NEXT_TOKEN\n}\n\nexport interface ReleaseEntry {\n version: string\n type: ReleaseType\n description?: string\n}\n\nconst isReleaseType = (value: string): value is ReleaseType => {\n return value === 'regular' || value === 'hotfix'\n}\n\n/**\n * Parse a CLI release spec of the form `version[:type[:description]]`.\n * Type defaults to \"regular\". Description is everything after the second\n * colon, so colons inside descriptions are preserved.\n */\nexport const parseReleaseSpec = (raw: string): ReleaseEntry => {\n const spec = raw.trim()\n\n if (spec === '') throw new Error('Release spec is empty')\n\n const firstColon = spec.indexOf(':')\n\n if (firstColon === -1) {\n return { version: spec, type: 'regular' }\n }\n\n const version = spec.slice(0, firstColon).trim()\n const rest = spec.slice(firstColon + 1)\n const secondColon = rest.indexOf(':')\n\n const typeRaw = secondColon === -1 ? rest.trim() : rest.slice(0, secondColon).trim()\n const description = secondColon === -1 ? '' : rest.slice(secondColon + 1).trim()\n const typeLower = typeRaw.toLowerCase()\n\n if (!isReleaseType(typeLower)) {\n throw new Error(`Invalid release type \"${typeRaw}\". Expected \"regular\" or \"hotfix\".`)\n }\n\n const entry: ReleaseEntry = { version, type: typeLower }\n\n if (description !== '') entry.description = description\n\n return entry\n}\n\n/**\n * Resolve a list of release entries (each with its own type and optional\n * \"next\" version token) into entries with concrete versions. Each \"next\"\n * advances based on the running max so successive \"next\" tokens produce\n * sequential versions, even across mixed types.\n */\nexport const resolveReleaseEntries = (entries: ReleaseEntry[], known: SemVer[]): ReleaseEntry[] => {\n const running: SemVer[] = [...known]\n\n return entries.map((entry) => {\n const trimmed = entry.version.trim()\n\n if (trimmed === '') {\n throw new Error('Release entry has an empty version')\n }\n\n if (isNextToken(trimmed)) {\n const next = computeNextVersion(running, entry.type)\n\n running.push(parseVersion(`v${next}`))\n\n return { ...entry, version: next }\n }\n\n const parsed = tryParse(trimmed)\n\n if (!parsed) {\n throw new Error(`Invalid version \"${trimmed}\". Expected semver like \"1.2.5\" or the token \"next\".`)\n }\n\n const explicit = `${parsed[0]}.${parsed[1]}.${parsed[2]}`\n\n running.push(parsed)\n\n return { ...entry, version: explicit }\n })\n}\n\nexport const hasNextToken = (entries: ReleaseEntry[]): boolean => {\n return entries.some((e) => {\n return isNextToken(e.version)\n })\n}\n", "const STDERR_EXCERPT_MAX_BYTES = 200\n\nexport interface OperationErrorContext {\n operation: string\n remediation?: string\n stderrExcerpt?: string\n}\n\n/**\n * Duck-typed read of zx's `ProcessOutput.stderr` (and similar shapes) without\n * importing zx types just for an `instanceof` check.\n *\n * @example\n * extractStderr(new Error('x')) // undefined\n * extractStderr({ stderr: 'fatal: ...' }) // 'fatal: ...'\n * extractStderr({ stderr: '' }) // undefined (empty treated as missing)\n */\nconst extractStderr = (cause: unknown): string | undefined => {\n if (cause === null || typeof cause !== 'object') return undefined\n const stderr = (cause as { stderr?: unknown }).stderr\n\n return typeof stderr === 'string' && stderr.length > 0 ? stderr : undefined\n}\n\n/**\n * Compose the human-and-agent-readable message body for an `OperationError`:\n * `\"failed to <operation> [\u2014 stderr: <excerpt>] [\u2014 try: <remediation>]\"`.\n * `stderrExcerpt` overrides anything duck-typed off `cause`; both are trimmed\n * and capped at {@link STDERR_EXCERPT_MAX_BYTES} so a runaway subprocess can't\n * blow up the message.\n */\nconst buildMessage = (cause: unknown, ctx: OperationErrorContext): string => {\n const stderr = ctx.stderrExcerpt ?? extractStderr(cause)\n const parts = [`failed to ${ctx.operation}`]\n\n if (stderr) parts.push(`stderr: ${stderr.slice(0, STDERR_EXCERPT_MAX_BYTES).trim()}`)\n if (ctx.remediation) parts.push(`try: ${ctx.remediation}`)\n\n return parts.join(' \u2014 ')\n}\n\n/**\n * Error type for any handler-level failure that should surface to the caller\n * (CLI user or MCP-connected agent) with a remediation hint. Wraps an\n * underlying cause and renders a single-line, structured message so logs and\n * agent tool-result text stay scannable.\n *\n * Pattern modeled on the exemplary Doppler errors in\n * `src/integrations/doppler/doppler-cli-auth.ts`.\n *\n * @example\n * // wrap a zx subprocess failure\n * try {\n * await $`git worktree add ${path} ${branch}`\n * } catch (err) {\n * throw new OperationError(err, {\n * operation: `git worktree add for ${branch}`,\n * remediation: 'check the branch name and that the parent dir is writable',\n * })\n * }\n *\n * @example\n * // validation failure with no underlying cause\n * throw new OperationError(undefined, {\n * operation: 'launch deploy-all workflow',\n * remediation: `pass one of: ${environments.join(', ')}`,\n * stderrExcerpt: `invalid environment: ${selectedEnv}`,\n * })\n */\nexport class OperationError extends Error {\n readonly operation: string\n readonly remediation?: string\n\n constructor(cause: unknown, ctx: OperationErrorContext) {\n super(buildMessage(cause, ctx), { cause })\n this.name = 'OperationError'\n this.operation = ctx.operation\n this.remediation = ctx.remediation\n }\n}\n", "import confirm from '@inquirer/confirm'\nimport select from '@inquirer/select'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\nimport { $ } from 'zx'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { deliverJiraRelease, loadJiraConfigOptional } from 'src/integrations/jira'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { formatZxError } from 'src/lib/errors/format-zx-error'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { removeWorktrees } from 'src/lib/worktrees'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\ninterface GhReleaseDeliverArgs extends RequiredConfirmedOptionArg {\n version: string\n}\n\ntype PRState = 'OPEN' | 'MERGED' | 'CLOSED'\n\ninterface PRStatus {\n number: number\n state: PRState\n title: string\n}\n\n/**\n * Wrap a delivery step so its failure logs structured zx fields and surfaces\n * an `OperationError` whose message names the actual step that failed \u2014\n * instead of the previous blanket \"merging release branch into dev\" message.\n */\nconst runStep = async <T>(operation: string, remediation: string, fn: () => Promise<T>): Promise<T> => {\n try {\n return await fn()\n } catch (error) {\n logger.error({ err: formatZxError(error) }, `\u274C Failed to ${operation}`)\n throw new OperationError(error, { operation, remediation })\n }\n}\n\n/**\n * Fetch the (most-recent) PR for the given head branch, across all states, so\n * we can resume a partially-completed delivery: a PR merged on a prior attempt\n * still appears here as `state: 'MERGED'`, letting the caller skip the merge.\n */\nconst fetchPRByHead = async (head: string): Promise<PRStatus | null> => {\n const result = await $`gh pr list --head ${head} --state all --json number,state,title --limit 1`\n const prs = JSON.parse(result.stdout) as PRStatus[]\n\n return prs[0] ?? null\n}\n\n/**\n * Find a MERGED RC PR (dev \u2192 main) whose title matches the given version, used\n * to detect that a prior delivery run already merged the RC and we should skip\n * the merge step on resume. Title-matched on purpose: an older MERGED RC PR\n * from a different release must not short-circuit this version's flow.\n */\nconst fetchMergedRcPRForVersion = async (version: string): Promise<PRStatus | null> => {\n const expectedTitle = `Release v${version} (RC)`\n const result = await $`gh pr list --head dev --base main --state merged --json number,state,title --limit 20`\n const prs = JSON.parse(result.stdout) as PRStatus[]\n const match = prs.find((pr) => {\n return pr.title === expectedTitle\n })\n\n return match ?? null\n}\n\n/**\n * Find an open dev \u2192 main PR, if any. GitHub allows at most one open PR per\n * head/base pair, so an existing open PR here is the RC PR for this release \u2014\n * even if its title was set by a previous (failed) delivery run. Adopting it\n * is what makes the flow recoverable after a mid-run failure.\n */\nconst fetchOpenDevToMainPR = async (): Promise<PRStatus | null> => {\n const result = await $`gh pr list --head dev --base main --state open --json number,state,title --limit 5`\n const prs = JSON.parse(result.stdout) as PRStatus[]\n\n return prs[0] ?? null\n}\n\ninterface ResolvedTarget {\n selectedReleaseBranch: string\n releasePrTitle: string\n}\n\nconst resolveTargetFromVersion = async (version: string): Promise<ResolvedTarget> => {\n const selectedReleaseBranch = `release/v${version}`\n const pr = await fetchPRByHead(selectedReleaseBranch)\n\n if (!pr) {\n logger.error(`\u274C No PR found for branch ${selectedReleaseBranch}.`)\n throw new OperationError(undefined, {\n operation: `deliver release ${selectedReleaseBranch}`,\n remediation: `confirm a PR exists ('gh pr list --head ${selectedReleaseBranch} --state all')`,\n })\n }\n\n return { selectedReleaseBranch, releasePrTitle: pr.title }\n}\n\nconst resolveTargetInteractively = async (): Promise<ResolvedTarget> => {\n const releasePRsInfo = await getReleasePRsWithInfo()\n\n const branches = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n commandEcho.setInteractive()\n\n const descriptions = await getJiraDescriptions()\n\n const selectedReleaseBranch = await select({\n message: '\uD83C\uDF3F Select release branch',\n choices: formatBranchChoices({ branches, descriptions, types: releaseTypes }),\n })\n\n const prInfo = releasePRsInfo.find((pr) => {\n return pr.branch === selectedReleaseBranch\n })\n\n if (!prInfo) {\n logger.error(`\u274C Release branch ${selectedReleaseBranch} not found in open PRs.`)\n throw new OperationError(undefined, {\n operation: `deliver release ${selectedReleaseBranch}`,\n remediation: `confirm an open PR exists for ${selectedReleaseBranch} ('gh pr list')`,\n })\n }\n\n return { selectedReleaseBranch, releasePrTitle: prInfo.title }\n}\n\n/**\n * `gh pr merge --delete-branch` also deletes the local branch, which fails if a\n * worktree has it checked out (the actual root cause of the \"Failed to merge\n * release PR\" surface error). Pre-remove any worktree for the release branch\n * so the local delete can succeed.\n */\nconst removeReleaseWorktreeIfPresent = async (releaseBranch: string): Promise<void> => {\n const worktreeBranches = await getCurrentWorktrees('release')\n\n if (!worktreeBranches.includes(releaseBranch)) return\n\n const [projectRoot, repoName] = await Promise.all([getProjectRoot(), getRepoName()])\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n\n const removed = await removeWorktrees({ branches: [releaseBranch], worktreeDir, repoName })\n\n if (removed.length === 0) {\n throw new OperationError(undefined, {\n operation: `remove worktree for ${releaseBranch} before merge`,\n remediation: `run manually: git worktree remove ${worktreeDir}/${releaseBranch} (use --force if uncommitted changes)`,\n })\n }\n}\n\ninterface MergeReleasePRArgs {\n selectedReleaseBranch: string\n releaseType: ReleaseType\n}\n\nconst mergeReleasePR = async (args: MergeReleasePRArgs): Promise<void> => {\n const { selectedReleaseBranch, releaseType } = args\n\n const mergeTarget = releaseType === 'hotfix' ? 'main' : 'dev'\n\n const releasePr = await fetchPRByHead(selectedReleaseBranch)\n\n if (!releasePr) {\n throw new OperationError(undefined, {\n operation: `look up release PR for ${selectedReleaseBranch}`,\n remediation: `verify the PR exists in GitHub`,\n })\n }\n\n if (releasePr.state === 'MERGED') {\n logger.info(`\u2713 Release PR ${selectedReleaseBranch} already merged \u2014 skipping`)\n\n return\n }\n\n if (releasePr.state === 'CLOSED') {\n throw new OperationError(undefined, {\n operation: `merge release PR ${selectedReleaseBranch} into ${mergeTarget}`,\n remediation: `the PR is closed without merge; reopen it or create a new release`,\n })\n }\n\n await runStep(\n `merge release PR ${selectedReleaseBranch} into ${mergeTarget}`,\n `check 'gh pr view ${selectedReleaseBranch}' for mergeability and required reviews`,\n async () => {\n await $`gh pr merge ${selectedReleaseBranch} --squash --admin --delete-branch`\n },\n )\n}\n\nconst resolveRcPRNumber = async (selectedVersion: string): Promise<number> => {\n const expectedTitle = `Release v${selectedVersion} (RC)`\n const existingOpen = await fetchOpenDevToMainPR()\n\n // Adopt any existing open dev\u2192main PR. GitHub permits only one open PR per\n // head/base pair, so a stale open RC PR (left behind by a prior failed run\n // \u2014 the single most common cause of \"Error merging release branch into\n // dev\") blocks `gh pr create`. Retitle it instead of fighting it.\n if (existingOpen) {\n const rcNumber = existingOpen.number\n\n if (existingOpen.title !== expectedTitle) {\n logger.info(\n `Adopting open dev \u2192 main PR #${rcNumber} (\"${existingOpen.title}\") and retitling for v${selectedVersion}`,\n )\n await runStep(\n `retitle dev \u2192 main PR #${rcNumber} to \"${expectedTitle}\"`,\n `update manually: gh pr edit ${rcNumber} --title \"${expectedTitle}\"`,\n async () => {\n await $`gh pr edit ${rcNumber} --title ${expectedTitle}`\n },\n )\n }\n\n return rcNumber\n }\n\n await runStep(\n `create RC PR (dev \u2192 main) for v${selectedVersion}`,\n `run 'gh pr create --base main --head dev' manually to surface the underlying error (e.g. no commits between dev and main)`,\n async () => {\n await $`gh pr create --base main --head dev --title ${expectedTitle} --body \"\"`\n },\n )\n\n const created = await fetchOpenDevToMainPR()\n\n if (!created) {\n throw new OperationError(undefined, {\n operation: `look up RC PR for v${selectedVersion}`,\n remediation: `verify the RC PR was created ('gh pr list --head dev --base main')`,\n })\n }\n\n return created.number\n}\n\nconst ensureRcPRMerged = async (selectedVersion: string): Promise<void> => {\n const alreadyMerged = await fetchMergedRcPRForVersion(selectedVersion)\n\n if (alreadyMerged) {\n logger.info(`\u2713 RC PR for v${selectedVersion} already merged into main \u2014 skipping`)\n\n return\n }\n\n const rcNumber = await resolveRcPRNumber(selectedVersion)\n\n await runStep(\n `merge RC PR #${rcNumber} (dev \u2192 main) for v${selectedVersion}`,\n `check 'gh pr view ${rcNumber}' for mergeability and required reviews`,\n async () => {\n await $`gh pr merge ${rcNumber} --squash --admin`\n },\n )\n}\n\nconst dispatchDeployWorkflow = async (): Promise<void> => {\n $.quiet = false\n\n await runStep(\n `dispatch deploy-all workflow on main`,\n `check 'gh workflow list' and that you have permission to dispatch deploy-all.yml`,\n async () => {\n await $`gh workflow run deploy-all.yml --ref main -f environment=prod`\n },\n )\n\n $.quiet = true\n}\n\nconst syncMainIntoDev = async (): Promise<void> => {\n await runStep(\n `sync main back into dev`,\n `run manually: git switch main && git pull && git switch dev && git pull && git merge main --no-edit && git push`,\n async () => {\n await $`git switch main && git pull && git switch dev && git pull && git merge main --no-edit && git push`\n },\n )\n}\n\nconst deliverJiraReleaseSafely = async (selectedReleaseBranch: string): Promise<void> => {\n const jiraConfig = await loadJiraConfigOptional()\n\n if (!jiraConfig) {\n logger.info('\uD83D\uDD14 Jira is not configured, skipping Jira release delivery')\n\n return\n }\n\n try {\n const versionName = selectedReleaseBranch.replace('release/', '')\n\n await deliverJiraRelease({ versionName }, jiraConfig)\n } catch (error) {\n logger.error({ err: formatZxError(error) }, 'Failed to deliver Jira release (non-blocking)')\n }\n}\n\n/**\n * Deliver a release branch to production. Each network/git step is run inside\n * `runStep` so the surfaced error names the failing operation and includes the\n * subprocess stderr. PR-merge steps are idempotent: if the release PR or RC PR\n * is already MERGED, the step is skipped, so re-running after a mid-flight\n * failure picks up where it stopped.\n */\nexport const ghReleaseDeliver = async (args: GhReleaseDeliverArgs) => {\n const { version, confirmedCommand } = args\n\n commandEcho.start('release-deliver')\n\n const { selectedReleaseBranch, releasePrTitle } = version\n ? await resolveTargetFromVersion(version)\n : await resolveTargetInteractively()\n\n const selectedVersion = selectedReleaseBranch.replace('release/v', '')\n\n commandEcho.addOption('--version', selectedVersion)\n\n const releaseType: ReleaseType = detectReleaseType(releasePrTitle)\n\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: `Are you sure you want to deliver version ${selectedReleaseBranch} to production?`,\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n commandEcho.addOption('--yes', true)\n\n $.quiet = true\n\n await removeReleaseWorktreeIfPresent(selectedReleaseBranch)\n await mergeReleasePR({ selectedReleaseBranch, releaseType })\n\n if (releaseType !== 'hotfix') {\n await ensureRcPRMerged(selectedVersion)\n }\n\n await dispatchDeployWorkflow()\n await syncMainIntoDev()\n\n $.quiet = false\n\n await deliverJiraReleaseSafely(selectedReleaseBranch)\n\n logger.info(`Successfully delivered ${selectedReleaseBranch} to production!`)\n\n commandEcho.print()\n\n const structuredContent = {\n releaseBranch: selectedReleaseBranch,\n version: selectedVersion,\n type: releaseType,\n success: true,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const ghReleaseDeliverMcpTool = defineMcpTool({\n name: 'gh-release-deliver',\n description:\n 'Deliver a release to production. For hotfixes: squash-merges the release branch to main and dispatches the deploy-all workflow. For regular releases: squash-merges to dev, opens an RC PR, merges dev into main, dispatches the deploy-all workflow, then syncs main back to dev. Also releases the matching Jira fix version if Jira is configured. Dispatches the deploy workflow fire-and-forget \u2014 the tool returns once the workflow is accepted by GitHub, not when the deployment finishes. PR-merge steps are idempotent: re-running after a partial failure skips PRs that are already merged. Irreversible production operation: the confirmation prompt is auto-skipped for MCP calls, so the caller is responsible for gating. \"version\" is required when invoked via MCP (the picker is unreachable without a TTY).',\n inputSchema: {\n version: z.string().describe('Release version to deliver to production (e.g., \"1.2.5\"). Required for MCP calls.'),\n },\n outputSchema: {\n releaseBranch: z.string().describe('The release branch that was delivered'),\n version: z.string().describe('The version that was delivered'),\n type: z.enum(['regular', 'hotfix']).describe('Release type'),\n success: z.boolean().describe('Whether the delivery was successful'),\n },\n handler: ghReleaseDeliver,\n})\n", "const STDERR_EXCERPT_MAX_BYTES = 500\nconst STDOUT_EXCERPT_MAX_BYTES = 200\n\nexport interface ZxErrorFields {\n exitCode?: number | null\n stderr?: string\n stdout?: string\n message?: string\n name?: string\n}\n\nconst readTrimmedString = (value: unknown, max: number): string | undefined => {\n if (typeof value !== 'string' || value.length === 0) return undefined\n\n return value.slice(-max).trim()\n}\n\n/**\n * Extract loggable fields from an error for `logger.error({ err: formatZxError(e) }, ...)`.\n *\n * pino's default `err` serializer handles `Error` subclasses but renders zx's\n * `ProcessOutput` as `{}` because its informative fields (`stderr`, `stdout`,\n * `exitCode`) are non-enumerable / on the prototype. This helper duck-types\n * those fields so subprocess failures surface in logs instead of vanishing.\n */\nexport const formatZxError = (error: unknown): ZxErrorFields => {\n if (error === null || typeof error !== 'object') {\n return { message: String(error) }\n }\n\n const rec = error as Record<string, unknown>\n const fields: ZxErrorFields = {}\n\n if (error instanceof Error) {\n fields.name = error.name\n fields.message = error.message\n } else if (typeof rec.message === 'string') {\n fields.message = rec.message\n }\n\n const exitCode = rec.exitCode\n\n if (typeof exitCode === 'number' || exitCode === null) fields.exitCode = exitCode\n\n const stderr = readTrimmedString(rec.stderr, STDERR_EXCERPT_MAX_BYTES)\n\n if (stderr) fields.stderr = stderr\n\n const stdout = readTrimmedString(rec.stdout, STDOUT_EXCERPT_MAX_BYTES)\n\n if (stdout) fields.stdout = stdout\n\n return fields\n}\n", "import { $ } from 'zx'\n\nimport { buildCmuxWorkspaceTitle, closeCmuxWorkspaceByTitle } from 'src/integrations/cmux'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { logger } from 'src/lib/logger'\n\ninterface RemoveWorktreesArgs {\n branches: string[]\n worktreeDir: string\n repoName: string\n pruneFolder?: boolean\n}\n\n/**\n * Close any cmux workspace for each branch and run `git worktree remove`,\n * returning the branches that were removed cleanly. Failures are logged but\n * never thrown, so a single bad worktree doesn't poison a batch removal.\n *\n * When `pruneFolder` is true and every branch was removed, also prune the\n * worktree metadata and delete the worktrees folder \u2014 used by the\n * `worktrees-remove` \"all\" path to leave the filesystem clean.\n */\nexport const removeWorktrees = async (args: RemoveWorktreesArgs): Promise<string[]> => {\n const { branches, worktreeDir, repoName, pruneFolder = false } = args\n\n const results = await Promise.allSettled(\n branches.map(async (branch) => {\n const worktreePath = `${worktreeDir}/${branch}`\n\n const title = buildCmuxWorkspaceTitle({ repoName, branch })\n\n await closeCmuxWorkspaceByTitle(title)\n\n await $`git worktree remove ${worktreePath}`\n\n return branch\n }),\n )\n\n const removed: string[] = []\n\n for (const [index, result] of results.entries()) {\n if (result.status === 'fulfilled') {\n removed.push(result.value)\n } else {\n const branch = branches[index]\n const err = new OperationError(result.reason, {\n operation: `remove worktree for ${branch}`,\n remediation: \"check 'git worktree list' for the path; uncommitted changes block removal\",\n })\n\n logger.error({ error: result.reason, msg: err.message })\n }\n }\n\n if (pruneFolder && removed.length === branches.length) {\n await $`git worktree prune`\n await $`rm -rf ${worktreeDir}`\n\n logger.info(`\uD83D\uDDD1\uFE0F Removed worktree folder: ${worktreeDir}`)\n logger.info('')\n }\n\n return removed\n}\n", "import { $ } from 'zx'\n\nimport { logger } from 'src/lib/logger'\n\n/**\n * Best-effort close of the cmux workspace whose title exactly matches `title`.\n * Silently no-ops if cmux isn't running, the workspace isn't found, or close fails.\n */\nexport const closeCmuxWorkspaceByTitle = async (title: string): Promise<void> => {\n try {\n const listOutput = (await $`cmux list-workspaces`.quiet()).stdout\n\n const ref = findWorkspaceRefByTitle(listOutput, title)\n\n if (!ref) {\n return\n }\n\n await $`cmux close-workspace --workspace ${ref}`.quiet()\n } catch (error) {\n logger.debug({ error, title }, 'cmux: skipped closing workspace')\n }\n}\n\n/**\n * Parses `cmux list-workspaces` output and returns the workspace ref whose\n * title exactly matches `title`, or undefined if no match.\n *\n * Each line looks like:\n * \" workspace:8 hulyo-monorepo v1.48.0\"\n * \"* workspace:6 obsidian-workspace [selected]\"\n */\nconst findWorkspaceRefByTitle = (output: string, title: string): string | undefined => {\n for (const rawLine of output.split('\\n')) {\n // eslint-disable-next-line sonarjs/slow-regex, regexp/no-super-linear-backtracking\n const match = rawLine.match(/^[* ]\\s*(workspace:\\d+)\\s+(.+?)(?:\\s+\\[selected\\])?\\s*$/)\n\n if (!match) {\n continue\n }\n\n const ref = match[1]\n const lineTitle = match[2]?.trim() ?? ''\n\n if (lineTitle === title) {\n return ref\n }\n }\n\n return undefined\n}\n", "import { $ } from 'zx'\n\nimport { logger } from 'src/lib/logger'\n\n/**\n * Returns the set of titles for all currently-open cmux workspaces.\n * Returns an empty set if cmux isn't running, the call fails, or the\n * output can't be parsed \u2014 callers should treat \"empty\" as \"unknown,\n * proceed as if nothing is open\".\n *\n * Each line of `cmux list-workspaces` looks like:\n * \" workspace:8 hulyo-monorepo v1.48.0\"\n * \"* workspace:6 obsidian-workspace [selected]\"\n */\nexport const listCmuxWorkspaceTitles = async (): Promise<Set<string>> => {\n try {\n const output = (await $`cmux list-workspaces`.quiet()).stdout\n\n const titles = new Set<string>()\n\n for (const rawLine of output.split('\\n')) {\n // eslint-disable-next-line sonarjs/slow-regex, regexp/no-super-linear-backtracking\n const match = rawLine.match(/^[* ]\\s*workspace:\\d+\\s+(.+?)(?:\\s+\\[selected\\])?\\s*$/)\n\n if (!match) {\n continue\n }\n\n const title = match[1]?.trim()\n\n if (title) {\n titles.add(title)\n }\n }\n\n return titles\n } catch (error) {\n logger.debug({ error }, 'cmux: skipped listing workspace titles')\n\n return new Set()\n }\n}\n", "import { $ } from 'zx'\n\ninterface OpenCmuxWorkspaceArgs {\n cwd: string\n title?: string\n}\n\n/**\n * Opens a new cmux workspace rooted at `cwd` with three panes:\n * left-top (primary) | right (full-height)\n * left-bottom |\n * All panes inherit `cwd` from the workspace.\n */\nexport const openCmuxWorkspaceWithLayout = async (args: OpenCmuxWorkspaceArgs): Promise<void> => {\n const { cwd, title } = args\n\n const newWorkspaceOutput = (await $`cmux new-workspace --cwd ${cwd}`).stdout\n\n const workspaceRef = parseWorkspaceRef(newWorkspaceOutput)\n\n const surfacesOutput = (await $`cmux list-pane-surfaces --workspace ${workspaceRef}`).stdout\n\n const leftTopRef = parseFirstSurfaceRef(surfacesOutput)\n\n await $`cmux new-split right --workspace ${workspaceRef} --surface ${leftTopRef}`\n await $`cmux new-split down --workspace ${workspaceRef} --surface ${leftTopRef}`\n\n if (title) {\n await $`cmux rename-workspace --workspace ${workspaceRef} ${title}`\n }\n}\n\n/**\n * Extracts the first `surface:<id>` reference from the output of\n * `cmux list-pane-surfaces`. Used to locate the initial (primary) pane\n * surface so subsequent splits can be anchored relative to it.\n *\n * @example\n * const output = 'surface:12 (active)\\nsurface:13\\n'\n * parseFirstSurfaceRef(output) // => 'surface:12'\n */\nconst parseFirstSurfaceRef = (output: string): string => {\n const match = output.match(/surface:\\d+/)\n\n if (!match) {\n throw new Error('cmux: could not locate initial surface in list-pane-surfaces output')\n }\n\n return match[0]\n}\n\n/**\n * Extracts the `workspace:<id>` reference from the output of\n * `cmux new-workspace`. The returned ref is used to target the newly\n * created workspace in follow-up `cmux` commands (splits, rename, etc.).\n *\n * @example\n * const output = 'created workspace:7\\n'\n * parseWorkspaceRef(output) // => 'workspace:7'\n */\nconst parseWorkspaceRef = (output: string): string => {\n const match = output.match(/workspace:\\d+/)\n\n if (!match) {\n throw new Error('cmux: could not locate workspace ref in new-workspace output')\n }\n\n return match[0]\n}\n", "interface BuildCmuxWorkspaceTitleArgs {\n repoName: string\n branch: string\n}\n\n/**\n * Builds the cmux workspace title used by `worktrees-add` and looked up by\n * `worktrees-remove`. The `release/` prefix is stripped so the title reads\n * e.g. `\"hulyo-monorepo v1.48.0\"` for branch `\"release/v1.48.0\"`.\n */\nexport const buildCmuxWorkspaceTitle = (args: BuildCmuxWorkspaceTitleArgs): string => {\n const { repoName, branch } = args\n\n const version = branch.replace('release/', '')\n\n return `${repoName} ${version}`\n}\n", "import select from '@inquirer/select'\nimport { z } from 'zod/v4'\nimport { $ } from 'zx'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface GhReleaseDeployAllArgs {\n version: string\n env: string\n skipTerraform?: boolean\n}\n\n/**\n * Deploy a release branch to an environment\n */\nexport const ghReleaseDeployAll = async (args: GhReleaseDeployAllArgs) => {\n const { version, env, skipTerraform } = args\n\n commandEcho.start('release-deploy-all')\n\n let selectedReleaseBranch = '' // \"release/v1.8.0\"\n\n if (version) {\n selectedReleaseBranch = version === 'dev' ? 'dev' : `release/v${version}`\n } else {\n commandEcho.setInteractive()\n\n const releasePRsInfo = await getReleasePRsWithInfo()\n\n const branches = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n const descriptions = await getJiraDescriptions()\n\n selectedReleaseBranch = await select({\n message: '\uD83C\uDF3F Select release branch',\n choices: [{ name: 'dev', value: 'dev' }, ...formatBranchChoices({ branches, descriptions, types: releaseTypes })],\n })\n }\n\n const selectedVersion = selectedReleaseBranch === 'dev' ? 'dev' : selectedReleaseBranch.replace('release/v', '')\n\n commandEcho.addOption('--version', selectedVersion)\n\n const { environments } = await getInfraKitConfig()\n\n let selectedEnv = ''\n\n if (env) {\n selectedEnv = env\n } else {\n commandEcho.setInteractive()\n\n selectedEnv = await select({\n message: '\uD83E\uDDEA Select environment',\n choices: environments.map((env) => {\n return {\n name: env,\n value: env,\n }\n }),\n })\n }\n\n commandEcho.addOption('--env', selectedEnv)\n\n if (!environments.includes(selectedEnv)) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-all workflow',\n remediation: `pass one of: ${environments.join(', ')}`,\n stderrExcerpt: `invalid environment: ${selectedEnv}`,\n })\n }\n\n const shouldSkipTerraform = skipTerraform ?? false\n\n if (shouldSkipTerraform) {\n commandEcho.addOption('--skip-terraform', true)\n }\n\n try {\n $.quiet = true\n\n const skipTerraformFlag = shouldSkipTerraform ? ['-f', 'skip_terraform_deploy=true'] : []\n\n await $`gh workflow run deploy-all.yml --ref ${selectedReleaseBranch} -f environment=${selectedEnv} ${skipTerraformFlag}`\n\n $.quiet = false\n\n logger.info(\n `Successfully launched deploy-all workflow_dispatch for release branch: ${selectedReleaseBranch} and environment: ${selectedEnv}`,\n )\n\n commandEcho.print()\n\n const structuredContent = {\n releaseBranch: selectedReleaseBranch,\n version: selectedReleaseBranch.replace('release/v', ''),\n environment: selectedEnv,\n skipTerraformDeploy: shouldSkipTerraform,\n success: true,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error: unknown) {\n logger.error({ error }, '\u274C Error launching workflow')\n throw new OperationError(error, {\n operation: 'launch deploy-all workflow',\n remediation: \"check 'gh workflow list' and that deploy-all.yml exists on the target ref\",\n })\n }\n}\n\n// MCP Tool Registration\nexport const ghReleaseDeployAllMcpTool = defineMcpTool({\n name: 'gh-release-deploy-all',\n description:\n 'Dispatch the deploy-all.yml GitHub Actions workflow to deploy every service from a release branch to the given environment. Fire-and-forget \u2014 returns once GitHub accepts the workflow_dispatch, NOT when the deployment finishes; watch the workflow run for completion status. Use gh-release-deploy-selected for a subset of services. Pass version=\"dev\" to deploy from the dev branch instead of a release branch. Both \"version\" and \"env\" are required when invoked via MCP (interactive pickers are unavailable without a TTY).',\n inputSchema: {\n version: z\n .string()\n .describe(\n 'Release version to deploy from (e.g. \"1.2.5\") \u2014 resolves to the release/vX.Y.Z branch. Pass \"dev\" to deploy from the dev branch instead. Required for MCP calls.',\n ),\n env: z\n .string()\n .describe(\n 'Target environment name \u2014 must match an env configured for the project (e.g. \"dev\", \"renana\", \"oriana\"). Required for MCP calls.',\n ),\n skipTerraform: z.boolean().optional().describe('Skip the terraform deployment stage.'),\n },\n outputSchema: {\n releaseBranch: z.string().describe('The release branch that was deployed'),\n version: z.string().describe('The version that was deployed'),\n environment: z.string().describe('The environment deployed to'),\n skipTerraformDeploy: z.boolean().describe('Whether terraform deployment was skipped'),\n success: z.boolean().describe('Whether the deployment was successful'),\n },\n handler: ghReleaseDeployAll,\n})\n", "import checkbox from '@inquirer/checkbox'\nimport select from '@inquirer/select'\nimport fs from 'node:fs/promises'\nimport { resolve } from 'node:path'\nimport yaml from 'yaml'\nimport { z } from 'zod/v4'\nimport { $ } from 'zx'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getProjectRoot } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface GhReleaseDeploySelectedArgs {\n version: string\n env: string\n services: string[]\n skipTerraform?: boolean\n}\n\n/**\n * Deploy selected services from a release branch to an environment\n */\n// eslint-disable-next-line sonarjs/cognitive-complexity\nexport const ghReleaseDeploySelected = async (args: GhReleaseDeploySelectedArgs) => {\n const { version, env, services, skipTerraform } = args\n\n commandEcho.start('release-deploy-selected')\n\n let selectedReleaseBranch = ''\n\n if (version) {\n selectedReleaseBranch = version === 'dev' ? 'dev' : `release/v${version}`\n } else {\n commandEcho.setInteractive()\n\n const releasePRsInfo = await getReleasePRsWithInfo()\n\n const branches = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n const descriptions = await getJiraDescriptions()\n\n selectedReleaseBranch = await select({\n message: '\uD83C\uDF3F Select release branch',\n choices: [{ name: 'dev', value: 'dev' }, ...formatBranchChoices({ branches, descriptions, types: releaseTypes })],\n })\n }\n\n const selectedVersion = selectedReleaseBranch === 'dev' ? 'dev' : selectedReleaseBranch.replace('release/v', '')\n\n commandEcho.addOption('--version', selectedVersion)\n\n const { environments } = await getInfraKitConfig()\n\n let selectedEnv = ''\n\n if (env) {\n selectedEnv = env\n } else {\n commandEcho.setInteractive()\n\n selectedEnv = await select({\n message: '\uD83E\uDDEA Select environment',\n choices: environments.map((env) => {\n return {\n name: env,\n value: env,\n }\n }),\n })\n }\n\n commandEcho.addOption('--env', selectedEnv)\n\n if (!environments.includes(selectedEnv)) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-selected workflow',\n remediation: `pass one of: ${environments.join(', ')}`,\n stderrExcerpt: `invalid environment: ${selectedEnv}`,\n })\n }\n\n // Parse available services from workflow file\n const availableServices = await parseServicesFromWorkflow()\n\n if (availableServices.length === 0) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-selected workflow',\n remediation: 'check .github/workflows/deploy-selected-services.yml for boolean service inputs',\n stderrExcerpt: 'no services found in workflow file',\n })\n }\n\n let selectedServices: string[] = []\n\n if (services && services.length > 0) {\n selectedServices = services\n } else {\n commandEcho.setInteractive()\n\n selectedServices = await checkbox({\n message: '\uD83D\uDE80 Select services to deploy (space to select, enter to confirm)',\n choices: availableServices.map((svc) => {\n return {\n name: svc,\n value: svc,\n }\n }),\n })\n }\n\n commandEcho.addOption('--services', selectedServices)\n\n if (selectedServices.length === 0) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-selected workflow',\n remediation: `pass at least one service from: ${availableServices.join(', ')}`,\n stderrExcerpt: 'no services selected',\n })\n }\n\n // Validate all selected services\n const invalidServices = selectedServices.filter((svc) => {\n return !availableServices.includes(svc)\n })\n\n if (invalidServices.length > 0) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-selected workflow',\n remediation: `pass services from: ${availableServices.join(', ')}`,\n stderrExcerpt: `invalid services: ${invalidServices.join(', ')}`,\n })\n }\n\n const shouldSkipTerraform = skipTerraform ?? false\n\n if (shouldSkipTerraform) {\n commandEcho.addOption('--skip-terraform', true)\n }\n\n try {\n $.quiet = true\n\n // Build the workflow command with boolean flags for each selected service\n const serviceFlags = selectedServices.flatMap((svc) => {\n return ['-f', `${svc}=true`]\n })\n const skipTerraformFlag = shouldSkipTerraform ? ['-f', 'skip_terraform_deploy=true'] : []\n\n await $`gh workflow run deploy-selected-services.yml --ref ${selectedReleaseBranch} -f environment=${selectedEnv} ${serviceFlags} ${skipTerraformFlag}`\n\n $.quiet = false\n\n logger.info(\n `Successfully launched deploy-selected-services workflow_dispatch for release branch: ${selectedReleaseBranch}, environment: ${selectedEnv}, services: ${selectedServices.join(', ')}`,\n )\n\n commandEcho.print()\n\n const structuredContent = {\n releaseBranch: selectedReleaseBranch,\n version: selectedReleaseBranch.replace('release/v', ''),\n environment: selectedEnv,\n services: selectedServices,\n skipTerraformDeploy: shouldSkipTerraform,\n success: true,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error: unknown) {\n logger.error({ error }, '\u274C Error launching workflow')\n throw new OperationError(error, {\n operation: 'launch deploy-selected workflow',\n remediation: \"check 'gh workflow list' and that deploy-selected-services.yml exists on the target ref\",\n })\n }\n}\n\n/**\n * Parse available services from the workflow file\n * Services are defined as boolean inputs (excluding skip_terraform_deploy)\n */\nconst parseServicesFromWorkflow = async (): Promise<string[]> => {\n const projectRoot = await getProjectRoot()\n\n const workflowPath = resolve(projectRoot, '.github/workflows/deploy-selected-services.yml')\n\n const content = await fs.readFile(workflowPath, 'utf-8')\n const parsed = yaml.parse(content)\n\n const inputs = parsed.on.workflow_dispatch.inputs\n const services: string[] = []\n\n for (const [key, value] of Object.entries(inputs)) {\n // Filter for boolean type inputs, excluding non-service flags\n if ((value as { type: string }).type === 'boolean' && key !== 'skip_terraform_deploy') {\n services.push(key)\n }\n }\n\n return services\n}\n\n// MCP Tool Registration\nexport const ghReleaseDeploySelectedMcpTool = defineMcpTool({\n name: 'gh-release-deploy-selected',\n description:\n 'Dispatch the deploy-selected-services.yml GitHub Actions workflow to deploy a chosen subset of services from a release branch to the given environment. Fire-and-forget \u2014 returns once GitHub accepts the workflow_dispatch, NOT when the deployment finishes; watch the workflow run for completion status. Service names are validated against the boolean inputs declared in the workflow. Use gh-release-deploy-all for every service. \"version\", \"env\", and \"services\" are all required when invoked via MCP (interactive pickers are unavailable without a TTY).',\n inputSchema: {\n version: z\n .string()\n .describe(\n 'Release version to deploy from (e.g. \"1.2.5\") \u2014 resolves to the release/vX.Y.Z branch. Pass \"dev\" to deploy from the dev branch instead. Required for MCP calls.',\n ),\n env: z\n .string()\n .describe(\n 'Target environment name \u2014 must match an env configured for the project (e.g. \"dev\", \"renana\", \"oriana\"). Required for MCP calls.',\n ),\n services: z\n .array(z.string())\n .describe(\n 'Service names to deploy. Each must match a boolean input declared in .github/workflows/deploy-selected-services.yml (e.g. \"client-be\", \"client-fe\"). Required for MCP calls.',\n ),\n skipTerraform: z.boolean().optional().describe('Skip the terraform deployment stage.'),\n },\n outputSchema: {\n releaseBranch: z.string().describe('The release branch that was deployed'),\n version: z.string().describe('The version that was deployed'),\n environment: z.string().describe('The environment deployed to'),\n services: z.array(z.string()).describe('The services that were deployed'),\n skipTerraformDeploy: z.boolean().describe('Whether terraform deployment was skipped'),\n success: z.boolean().describe('Whether the deployment was successful'),\n },\n handler: ghReleaseDeploySelected,\n})\n", "import { z } from 'zod/v4'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatVersionLabel, getJiraDescriptions } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\n\n/**\n * List all open release branches\n */\nexport const ghReleaseList = async () => {\n const releasePRs = await getReleasePRsWithInfo()\n\n const releases = releasePRs.map((pr) => {\n return {\n version: pr.branch.replace('release/', ''),\n type: detectReleaseType(pr.title),\n }\n })\n\n const jiraDescriptions = await getJiraDescriptions()\n\n const maxVersionLength = Math.max(\n ...releases.map((r) => {\n return r.version.length\n }),\n )\n\n const formattedLines = releases.map((release) => {\n const label = formatVersionLabel(release.version, release.type, maxVersionLength)\n const description = jiraDescriptions.get(release.version)\n\n if (description) {\n return `${label} ${description}`\n }\n\n return label\n })\n\n logger.info('All release branches: \\n')\n logger.info(`\\n${formattedLines.join('\\n')}\\n`)\n\n const structuredContent = {\n releases: releases.map((release) => {\n return {\n version: release.version,\n type: release.type,\n description: jiraDescriptions.get(release.version) || null,\n }\n }),\n count: releases.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const ghReleaseListMcpTool = defineMcpTool({\n name: 'gh-release-list',\n description:\n 'List every open release PR with its version, type (regular / hotfix), and associated Jira fix-version description. Read-only; sourced from GitHub and Jira.',\n inputSchema: {},\n outputSchema: {\n releases: z\n .array(\n z.object({\n version: z.string().describe('Release version'),\n type: z.enum(['regular', 'hotfix']).describe('Release type'),\n description: z.string().nullable().describe('Jira version description'),\n }),\n )\n .describe('List of all release branches'),\n count: z.number().describe('Number of release branches'),\n },\n handler: ghReleaseList,\n})\n", "import confirm from '@inquirer/confirm'\nimport select from '@inquirer/select'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\nimport { question } from 'zx'\n\nimport { loadJiraConfig } from 'src/integrations/jira'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { logger } from 'src/lib/logger'\nimport { createSingleRelease, prepareGitForRelease } from 'src/lib/release-utils'\nimport type { ReleaseCreationResult, ReleaseType } from 'src/lib/release-utils'\nimport {\n NoPriorVersionsError,\n computeNextVersion,\n hasNextToken,\n loadExistingVersions,\n parseVersion,\n resolveReleaseEntries,\n} from 'src/lib/version-utils'\nimport type { ReleaseEntry, SemVer } from 'src/lib/version-utils'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\ninterface ReleaseCreateArgs extends RequiredConfirmedOptionArg {\n releases?: ReleaseEntry[]\n}\n\nconst VERSION_PROMPT_HINT = '\"1.2.5\" or \"next\"'\n\nconst trySuggestNext = (known: SemVer[], type: ReleaseType): string | null => {\n try {\n return computeNextVersion(known, type)\n } catch (err) {\n if (err instanceof NoPriorVersionsError) return null\n\n throw err\n }\n}\n\nconst resolveOrExit = (entries: ReleaseEntry[], known: SemVer[]): ReleaseEntry[] => {\n try {\n return resolveReleaseEntries(entries, known)\n } catch (err) {\n if (err instanceof NoPriorVersionsError) {\n throw new OperationError(err, {\n operation: 'resolve release version',\n remediation: 'pass an explicit version (e.g. \"1.2.5\") instead of \"next\" when there are no prior versions',\n })\n }\n\n throw err\n }\n}\n\nconst promptForReleasesInteractive = async (ensureKnown: () => Promise<SemVer[]>): Promise<ReleaseEntry[]> => {\n commandEcho.setInteractive()\n\n const baseKnown = await ensureKnown()\n const running: SemVer[] = [...baseKnown]\n const entries: ReleaseEntry[] = []\n let addAnother = true\n\n while (addAnother) {\n const ordinal = entries.length + 1\n const type = await select<ReleaseType>({\n message: `Release #${ordinal} \u2014 select type:`,\n choices: [\n { name: 'regular', value: 'regular' },\n { name: 'hotfix', value: 'hotfix' },\n ],\n default: 'regular',\n })\n\n const suggestion = trySuggestNext(running, type)\n const defaultHint = suggestion ? ` [${suggestion}]` : ''\n const versionAnswer = (await question(` Version (e.g. ${VERSION_PROMPT_HINT})${defaultHint}: `)).trim()\n const versionInput = versionAnswer === '' ? (suggestion ?? '') : versionAnswer\n\n if (versionInput === '') {\n logger.error('No version provided. Exiting...')\n process.exit(1)\n }\n\n const resolved = resolveOrExit([{ version: versionInput, type }], running)[0] as ReleaseEntry\n\n running.push(parseVersion(`v${resolved.version}`))\n\n const description = (await question(' Description (optional, press Enter to skip): ')).trim()\n\n entries.push({ ...resolved, ...(description !== '' ? { description } : {}) })\n\n addAnother = await confirm({ message: 'Add another release?', default: false })\n }\n\n return entries\n}\n\nconst formatReleaseSummary = (entry: ReleaseEntry): string => {\n const parts = [`v${entry.version}`, entry.type]\n\n if (entry.description) parts.push(entry.description)\n\n return parts.join(' \u00B7 ')\n}\n\nconst echoReleases = (entries: ReleaseEntry[]): void => {\n for (const entry of entries) {\n const spec = entry.description\n ? `${entry.version}:${entry.type}:${entry.description}`\n : `${entry.version}:${entry.type}`\n\n commandEcho.addOption('--release', spec)\n }\n}\n\ninterface FailedRelease {\n version: string\n error: string\n}\n\nconst collectEntries = async (\n inputReleases: ReleaseEntry[] | undefined,\n ensureKnown: () => Promise<SemVer[]>,\n): Promise<ReleaseEntry[]> => {\n if (inputReleases && inputReleases.length > 0) {\n const known = hasNextToken(inputReleases) ? await ensureKnown() : []\n const resolved = resolveOrExit(inputReleases, known)\n\n echoReleases(resolved)\n\n return resolved\n }\n\n const interactive = await promptForReleasesInteractive(ensureKnown)\n\n echoReleases(interactive)\n\n return interactive\n}\n\nconst confirmReleases = async (entries: ReleaseEntry[], confirmedCommand: boolean): Promise<void> => {\n const summary = entries.map(formatReleaseSummary).join('\\n - ')\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: `Create the following ${entries.length} release(s)?\\n - ${summary}\\n`,\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n commandEcho.addOption('--yes', true)\n}\n\ninterface ExecuteOneArgs {\n entry: ReleaseEntry\n jiraConfig: Awaited<ReturnType<typeof loadJiraConfig>>\n}\n\nconst executeOne = async (\n args: ExecuteOneArgs,\n): Promise<{ result?: ReleaseCreationResult; failure?: FailedRelease }> => {\n const { entry, jiraConfig } = args\n\n try {\n await prepareGitForRelease(entry.type)\n\n const result = await createSingleRelease({\n version: entry.version,\n jiraConfig,\n description: entry.description,\n type: entry.type,\n })\n\n logger.info(`\u2705 Successfully created release: v${entry.version} (${entry.type})`)\n logger.info(`\uD83D\uDD17 GitHub PR: ${result.prUrl}`)\n logger.info(`\uD83D\uDD17 Jira Version: ${result.jiraVersionUrl}\\n`)\n\n return { result }\n } catch (error) {\n const err = new OperationError(error, {\n operation: `create release v${entry.version} (${entry.type})`,\n remediation: 'verify the version is unique and the base branch is clean',\n })\n\n logger.error(`\u274C ${err.message}\\n`)\n\n return { failure: { version: entry.version, error: err.message } }\n }\n}\n\nconst logFinalSummary = (total: number, successCount: number, failureCount: number): void => {\n if (successCount === total) {\n logger.info(`\u2705 All ${total} release branch(es) were created successfully.`)\n } else if (successCount > 0) {\n logger.warn(`\u26A0\uFE0F ${successCount} of ${total} release branches were created successfully.`)\n logger.warn(`\u274C ${failureCount} release(s) failed.`)\n } else {\n logger.error(`\u274C All ${total} release branch(es) failed to create.`)\n }\n}\n\n/**\n * Create one or more release branches. Each release carries its own type\n * (regular/hotfix) and optional Jira description, so a single invocation\n * may mix regular and hotfix releases off their respective base branches.\n */\nexport const releaseCreate = async (args: ReleaseCreateArgs) => {\n const { releases: inputReleases, confirmedCommand } = args\n\n commandEcho.start('release-create')\n\n const jiraConfig = await loadJiraConfig()\n\n let known: SemVer[] | null = null\n const ensureKnown = async (): Promise<SemVer[]> => {\n if (known === null) known = await loadExistingVersions()\n\n return known\n }\n\n const entries = await collectEntries(inputReleases, ensureKnown)\n\n if (entries.length === 0) {\n throw new OperationError(undefined, {\n operation: 'create release',\n remediation: 'pass at least one entry in \"releases\" (e.g. [{ version: \"1.2.5\", type: \"regular\" }])',\n stderrExcerpt: 'no releases provided',\n })\n }\n\n await confirmReleases(entries, Boolean(confirmedCommand))\n\n const created: ReleaseCreationResult[] = []\n const failed: FailedRelease[] = []\n\n for (const entry of entries) {\n const { result, failure } = await executeOne({ entry, jiraConfig })\n\n if (result) created.push(result)\n if (failure) failed.push(failure)\n }\n\n logFinalSummary(entries.length, created.length, failed.length)\n\n commandEcho.print()\n\n const structuredContent = {\n createdBranches: created.map((r) => {\n return r.branchName\n }),\n successCount: created.length,\n failureCount: failed.length,\n releases: created,\n failedReleases: failed,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const releaseCreateMcpTool = defineMcpTool({\n name: 'release-create',\n description:\n 'Create one or more releases in a single call. Each entry in \"releases\" carries its own version, type (regular|hotfix, default regular), and optional description, so regular and hotfix releases can be mixed in the same invocation. For each release this tool switches to the appropriate base branch (dev for regular, main for hotfix), cuts the release branch, opens a GitHub release PR, and creates the matching Jira fix version. The literal token \"next\" auto-increments from the union of remote release branches and Jira fix versions (regular bumps minor + resets patch; hotfix bumps patch on the highest minor); multiple \"next\" tokens advance sequentially across mixed types. Confirmation is auto-skipped for MCP calls, so the caller is responsible for gating. Continues on per-release failure and reports successes/failures.',\n inputSchema: {\n releases: z\n .array(\n z.object({\n version: z\n .string()\n .describe('Version to create (e.g., \"1.2.5\") or the literal token \"next\" for auto-increment.'),\n type: z\n .enum(['regular', 'hotfix'])\n .optional()\n .default('regular')\n .describe('Release type: \"regular\" (branches off dev) or \"hotfix\" (branches off main).'),\n description: z.string().optional().describe('Optional description for the Jira version.'),\n }),\n )\n .min(1)\n .describe('One or more releases to create. Each entry has its own version, type, and optional description.'),\n },\n outputSchema: {\n createdBranches: z.array(z.string()).describe('List of created release branch names'),\n successCount: z.number().describe('Number of releases created successfully'),\n failureCount: z.number().describe('Number of releases that failed'),\n releases: z\n .array(\n z.object({\n version: z.string().describe('Version number'),\n type: z.enum(['regular', 'hotfix']).describe('Release type'),\n branchName: z.string().describe('Release branch name'),\n prUrl: z.string().describe('GitHub PR URL'),\n jiraVersionUrl: z.string().describe('Jira version URL'),\n }),\n )\n .describe('Detailed information for each created release with URLs'),\n failedReleases: z\n .array(\n z.object({\n version: z.string().describe('Version number that failed'),\n error: z.string().describe('Error message'),\n }),\n )\n .describe('List of releases that failed with error messages'),\n },\n handler: releaseCreate,\n})\n", "import confirm from '@inquirer/confirm'\nimport select from '@inquirer/select'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\nimport { question } from 'zx'\n\nimport { getReleasePRsWithInfo, updateReleasePRBody } from 'src/integrations/gh'\nimport { findVersionByName, loadJiraConfig, updateJiraVersion } from 'src/integrations/jira'\nimport type { JiraConfig, JiraVersion } from 'src/integrations/jira'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\ninterface ReleaseDescEditArgs extends RequiredConfirmedOptionArg {\n version?: string\n description?: string\n}\n\nconst buildJiraVersionUrl = (jiraConfig: JiraConfig, version: JiraVersion): string => {\n return `${jiraConfig.baseUrl}/projects/${version.projectId}/versions/${version.id}/tab/release-report-all-issues`\n}\n\nconst buildPRBody = (jiraVersionUrl: string, description: string): string => {\n return description.trim() !== '' ? `${jiraVersionUrl}\\n\\n${description}` : `${jiraVersionUrl} \\n`\n}\n\nconst pickReleaseBranch = async (): Promise<{ branch: string; type: ReleaseType }> => {\n const releasePRsInfo = await getReleasePRsWithInfo()\n const branches = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n const types = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n const descriptions = await getJiraDescriptions()\n\n const branch = await select({\n message: '\uD83C\uDF3F Select release branch',\n choices: formatBranchChoices({ branches, descriptions, types }),\n })\n\n return { branch, type: types.get(branch) || 'regular' }\n}\n\nconst verifyReleasePRExists = async (selectedBranch: string): Promise<ReleaseType> => {\n const releasePRsInfo = await getReleasePRsWithInfo()\n const prInfo = releasePRsInfo.find((pr) => {\n return pr.branch === selectedBranch\n })\n\n if (!prInfo) {\n throw new OperationError(undefined, {\n operation: `edit description for ${selectedBranch}`,\n remediation: `confirm an open PR exists for ${selectedBranch} ('gh pr list')`,\n })\n }\n\n return detectReleaseType(prInfo.title)\n}\n\nconst promptDescription = async (current: string): Promise<string> => {\n const hint = current === '' ? '(no current description)' : `current: \"${current}\"`\n const answer = await question(` New description ${hint}\\n (press Enter to keep current): `)\n const trimmed = answer.replace(/\\n$/, '')\n\n return trimmed === '' ? current : trimmed\n}\n\n/**\n * Edit a release's description in Jira (fix version) and in the matching\n * GitHub release PR body. The PR body is rewritten canonically to\n * `<jiraVersionUrl>\\n\\n<description>` (matching `release-create`).\n */\nexport const releaseDescEdit = async (args: ReleaseDescEditArgs) => {\n const { version: versionArg, description: descriptionArg, confirmedCommand } = args\n\n commandEcho.start('release-desc-edit')\n\n const jiraConfig = await loadJiraConfig()\n\n let selectedBranch: string\n\n if (versionArg) {\n selectedBranch = `release/v${versionArg}`\n await verifyReleasePRExists(selectedBranch)\n } else {\n commandEcho.setInteractive()\n const picked = await pickReleaseBranch()\n\n selectedBranch = picked.branch\n }\n\n const selectedVersion = selectedBranch.replace('release/v', '')\n\n commandEcho.addOption('--version', selectedVersion)\n\n const versionName = `v${selectedVersion}`\n const jiraVersion = await findVersionByName(versionName, jiraConfig)\n\n if (!jiraVersion) {\n throw new OperationError(undefined, {\n operation: `edit description for ${versionName}`,\n remediation: `create the Jira fix version \"${versionName}\" first or pick a different release`,\n })\n }\n\n const previousDescription = jiraVersion.description ?? ''\n\n let newDescription: string\n\n if (descriptionArg !== undefined) {\n newDescription = descriptionArg\n commandEcho.addOption('--description', newDescription)\n } else {\n commandEcho.setInteractive()\n newDescription = await promptDescription(previousDescription)\n }\n\n if (newDescription === previousDescription) {\n logger.info(`No change \u2014 description for ${versionName} is already: \"${previousDescription}\"`)\n commandEcho.print()\n\n const structuredContent = {\n version: selectedVersion,\n branch: selectedBranch,\n jiraVersionUrl: buildJiraVersionUrl(jiraConfig, jiraVersion),\n previousDescription,\n newDescription,\n changed: false,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n }\n\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: `Update description for ${versionName}?\\n from: \"${previousDescription}\"\\n to: \"${newDescription}\"\\n`,\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n commandEcho.addOption('--yes', true)\n\n await updateJiraVersion({ versionId: jiraVersion.id, description: newDescription }, jiraConfig)\n\n const jiraVersionUrl = buildJiraVersionUrl(jiraConfig, jiraVersion)\n const body = buildPRBody(jiraVersionUrl, newDescription)\n\n await updateReleasePRBody({ branch: selectedBranch, body })\n\n logger.info(`\u2705 Updated description for ${versionName}`)\n logger.info(`\uD83D\uDD17 Jira Version: ${jiraVersionUrl}`)\n logger.info(`\uD83D\uDD17 PR branch: ${selectedBranch}\\n`)\n\n commandEcho.print()\n\n const structuredContent = {\n version: selectedVersion,\n branch: selectedBranch,\n jiraVersionUrl,\n previousDescription,\n newDescription,\n changed: true,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const releaseDescEditMcpTool = defineMcpTool({\n name: 'release-desc-edit',\n description:\n \"Edit a release's description in Jira and in the matching GitHub release PR body. Targets the Jira fix version named `v<version>` and the open PR on branch `release/v<version>`. The PR body is rewritten canonically to `<jiraVersionUrl>\\\\n\\\\n<description>` \u2014 any prior manual edits to the body are overwritten. Both `version` and `description` are required for MCP calls (the picker/prompt are unreachable without a TTY). Empty `description` clears the description on both sides. Confirmation is auto-skipped for MCP, so the caller is responsible for gating.\",\n inputSchema: {\n version: z.string().describe('Release version, e.g. \"1.2.5\".'),\n description: z.string().describe('New description. Empty string clears the description.'),\n },\n outputSchema: {\n version: z.string().describe('Release version'),\n branch: z.string().describe('Release branch name (e.g. \"release/v1.2.5\")'),\n jiraVersionUrl: z.string().describe('Jira fix version URL'),\n previousDescription: z.string().describe('The description before the update'),\n newDescription: z.string().describe('The description after the update'),\n changed: z.boolean().describe('Whether the description actually changed'),\n },\n handler: releaseDescEdit,\n})\n", "import { z } from 'zod/v4'\n\nimport { logger } from 'src/lib/logger'\nimport { defineMcpTool, textContent } from 'src/types'\n\nimport packageJson from '../../../package.json' with { type: 'json' }\n\n/**\n * Print the infra-kit CLI version\n */\nexport const version = async () => {\n const cliVersion = packageJson.version\n\n logger.info(cliVersion)\n\n const structuredContent = { version: cliVersion }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const versionMcpTool = defineMcpTool({\n name: 'version',\n description: 'Print the installed infra-kit CLI version',\n inputSchema: {},\n outputSchema: {\n version: z.string().describe('Installed infra-kit CLI version (from package.json)'),\n },\n handler: version,\n})\n", "{\n \"name\": \"infra-kit\",\n \"type\": \"module\",\n \"version\": \"0.1.102\",\n \"description\": \"infra-kit\",\n \"main\": \"dist/cli.js\",\n \"module\": \"dist/cli.js\",\n \"bin\": {\n \"infra-kit\": \"dist/cli.js\",\n \"ik\": \"dist/cli.js\"\n },\n \"engines\": {\n \"node\": \">=24.x\"\n },\n \"scripts\": {\n \"inspector\": \"npx @modelcontextprotocol/inspector node ./dist/mcp.js --debug\",\n \"build\": \"pnpm run clean-artifacts && node ./scripts/build.js\",\n \"clean-artifacts\": \"rm -rf dist\",\n \"clean-cache\": \"rm -rf node_modules/.cache .eslintcache tsconfig.tsbuildinfo .turbo .swc\",\n \"prettier-fix\": \"pnpm exec prettier **/* --write --no-error-on-unmatched-pattern --log-level silent --ignore-path ../../../.prettierignore\",\n \"prettier-check\": \"pnpm exec prettier **/* --check --no-error-on-unmatched-pattern --log-level silent --ignore-path ../../../.prettierignore\",\n \"eslint-check\": \"pnpm exec eslint --cache --quiet --report-unused-disable-directives ./src\",\n \"eslint-fix\": \"pnpm exec eslint --cache --quiet --report-unused-disable-directives ./src --fix\",\n \"ts-check\": \"tsc --noEmit\",\n \"test\": \"pnpm exec vitest run --reporter=minimal\",\n \"test-watch\": \"pnpm exec vitest --watch --silent passed-only\",\n \"test-ui\": \"pnpm exec vitest --ui --silent passed-only\",\n \"test-report\": \"pnpm exec vitest run --coverage --silent passed-only\",\n \"qa\": \"pnpm run prettier-check && pnpm run eslint-check && pnpm run ts-check && pnpm run test && echo \u2705 Success\",\n \"fix\": \"pnpm run prettier-fix && pnpm run eslint-fix && pnpm run qa\"\n },\n \"dependencies\": {\n \"@inquirer/checkbox\": \"^5.1.5\",\n \"@inquirer/confirm\": \"^6.0.13\",\n \"@inquirer/select\": \"^5.1.5\",\n \"@modelcontextprotocol/sdk\": \"^1.29.0\",\n \"commander\": \"^14.0.3\",\n \"pino\": \"^10.3.1\",\n \"pino-pretty\": \"^13.1.3\",\n \"yaml\": \"^2.9.0\",\n \"zod\": \"^3.25.76\",\n \"zx\": \"^8.8.5\"\n },\n \"devDependencies\": {\n \"@pkg/eslint-config\": \"workspace:*\",\n \"@pkg/vitest-config\": \"workspace:*\",\n \"esbuild\": \"^0.28.0\",\n \"typescript\": \"^6.0.3\"\n }\n}\n", "/* eslint-disable sonarjs/cognitive-complexity */\nimport checkbox from '@inquirer/checkbox'\nimport confirm from '@inquirer/confirm'\nimport select from '@inquirer/select'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\nimport { $ } from 'zx'\n\nimport { buildCmuxWorkspaceTitle, openCmuxWorkspaceWithLayout } from 'src/integrations/cmux'\nimport { addFoldersToCursorWorkspace, resolveCursorWorkspacePath } from 'src/integrations/cursor'\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\n// Constants\nconst FEATURE_DIR = 'feature'\nconst RELEASE_DIR = 'release'\nconst RELEASE_BRANCH_PREFIX = 'release/v'\n\nexport const CURSOR_MODES = ['workspace', 'windows', 'none'] as const\nexport type CursorMode = (typeof CURSOR_MODES)[number]\n\ninterface WorktreeManagementArgs extends RequiredConfirmedOptionArg {\n all?: boolean\n versions?: string\n cursor?: CursorMode\n githubDesktop?: boolean\n cmux?: boolean\n}\n\n/**\n * Manage git worktrees for release branches\n * Creates worktrees for active release branches and removes unused ones\n */\nexport const worktreesAdd = async (options: WorktreeManagementArgs) => {\n const { confirmedCommand, all, versions, cursor, githubDesktop, cmux } = options\n\n commandEcho.start('worktrees-add')\n\n try {\n const currentWorktrees = await getCurrentWorktrees('release')\n const projectRoot = await getProjectRoot()\n\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n\n await ensureWorktreeDirectory(`${worktreeDir}/${RELEASE_DIR}`)\n await ensureWorktreeDirectory(`${worktreeDir}/${FEATURE_DIR}`)\n\n let selectedReleaseBranches: string[] = []\n\n if (versions) {\n selectedReleaseBranches = versions.split(',').map((v) => {\n return `release/v${v.trim()}`\n })\n } else {\n const releasePRsInfo = await getReleasePRsWithInfo()\n\n const releasePRsList = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n\n if (releasePRsList.length === 0) {\n logger.info('\u2139\uFE0F No open release branches found')\n\n commandEcho.print()\n\n return {\n content: textContent(JSON.stringify({ createdWorktrees: [], count: 0 }, null, 2)),\n structuredContent: { createdWorktrees: [], count: 0 },\n }\n }\n\n if (all) {\n selectedReleaseBranches = releasePRsList\n } else {\n commandEcho.setInteractive()\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n const descriptions = await getJiraDescriptions()\n\n selectedReleaseBranches = await checkbox({\n required: true,\n message: '\uD83C\uDF3F Select release branches',\n choices: formatBranchChoices({ branches: releasePRsList, descriptions, types: releaseTypes }),\n })\n }\n }\n\n // Track --all flag if all branches were selected (either via flag or interactively)\n if (all) {\n commandEcho.addOption('--all', true)\n } else {\n commandEcho.addOption(\n '--versions',\n selectedReleaseBranches.map((branch) => {\n return branch.replace('release/v', '')\n }),\n )\n }\n\n // Ask for confirmation\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: 'Are you sure you want to proceed with these worktree changes?',\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n if (!confirmedCommand) {\n commandEcho.addOption('--yes', true)\n }\n\n const config = await getInfraKitConfig()\n const cursorConfig = config.ide?.provider === 'cursor' ? config.ide.config : undefined\n\n const cursorMode: CursorMode =\n cursor ??\n cursorConfig?.mode ??\n (await select<CursorMode>({\n message: 'Cursor mode for created worktrees?',\n default: 'workspace',\n choices: [\n {\n name: 'Add to workspace file',\n value: 'workspace',\n description: 'Append each worktree as a folder in ide.config.workspaceConfigPath, then open the workspace',\n },\n {\n name: 'Open separate windows',\n value: 'windows',\n description: 'Open each created worktree in its own Cursor window',\n },\n { name: 'Skip', value: 'none', description: 'Do not open Cursor' },\n ],\n }))\n\n if (typeof cursor === 'undefined' && !cursorConfig?.mode) {\n commandEcho.setInteractive()\n }\n\n commandEcho.addOption('--cursor', cursorMode)\n\n const openInGithubDesktop =\n githubDesktop ??\n config.worktrees?.openInGithubDesktop ??\n (await confirm({ message: 'Open created worktrees in GitHub Desktop?' }))\n\n if (typeof githubDesktop === 'undefined' && config.worktrees?.openInGithubDesktop === undefined) {\n commandEcho.setInteractive()\n }\n\n if (openInGithubDesktop) {\n commandEcho.addOption('--github-desktop', true)\n } else {\n commandEcho.addOption('--no-github-desktop', true)\n }\n\n const openInCmux =\n cmux ?? config.worktrees?.openInCmux ?? (await confirm({ message: 'Open created worktrees in cmux?' }))\n\n if (typeof cmux === 'undefined' && config.worktrees?.openInCmux === undefined) {\n commandEcho.setInteractive()\n }\n\n if (openInCmux) {\n commandEcho.addOption('--cmux', true)\n } else {\n commandEcho.addOption('--no-cmux', true)\n }\n\n const { branchesToCreate } = categorizeWorktrees({\n selectedReleaseBranches,\n currentWorktrees,\n })\n\n const createdWorktrees = await createWorktrees(branchesToCreate, worktreeDir)\n\n logResults(createdWorktrees)\n\n if (cursorMode === 'workspace') {\n if (!cursorConfig?.workspaceConfigPath) {\n logger.warn('\u26A0\uFE0F Skipping Cursor: ide.config.workspaceConfigPath is not set in infra-kit config')\n } else {\n const workspacePath = resolveCursorWorkspacePath(cursorConfig.workspaceConfigPath, projectRoot)\n\n const folderPaths = createdWorktrees.map((branch) => {\n return `${worktreeDir}/${branch}`\n })\n\n const { added, skipped } = await addFoldersToCursorWorkspace({ workspacePath, folderPaths })\n\n const skippedSuffix = skipped.length > 0 ? ` (${skipped.length} already present)` : ''\n\n logger.info(`\u2705 Added ${added.length} folder(s) to ${workspacePath}${skippedSuffix}`)\n\n await $`cursor ${workspacePath}`\n }\n } else if (cursorMode === 'windows') {\n for (const branch of createdWorktrees) {\n await $`cursor ${worktreeDir}/${branch}`\n }\n }\n\n if (openInGithubDesktop) {\n for (const branch of createdWorktrees) {\n await $`github ${worktreeDir}/${branch}`\n await $`sleep 5`\n }\n }\n\n if (openInCmux) {\n const repoName = await getRepoName()\n\n for (const branch of createdWorktrees) {\n const title = buildCmuxWorkspaceTitle({ repoName, branch })\n\n await openCmuxWorkspaceWithLayout({\n cwd: `${worktreeDir}/${branch}`,\n title,\n })\n }\n }\n\n commandEcho.print()\n\n const structuredContent = {\n createdWorktrees,\n count: createdWorktrees.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error) {\n logger.error({ error }, '\u274C Error managing worktrees')\n throw new OperationError(error, {\n operation: 'create worktrees',\n remediation: \"verify branches don't already exist as worktrees: 'git worktree list'\",\n })\n }\n}\n\n/**\n * Ensure the worktree directory exists\n */\nconst ensureWorktreeDirectory = async (worktreeDir: string): Promise<void> => {\n await $`mkdir -p ${worktreeDir}`\n}\n\ninterface CategorizeWorktreesArgs {\n selectedReleaseBranches: string[]\n currentWorktrees: string[]\n}\n\n/**\n * Categorize release worktrees into those that need to be created or removed\n */\nconst categorizeWorktrees = (args: CategorizeWorktreesArgs): { branchesToCreate: string[] } => {\n const { selectedReleaseBranches, currentWorktrees } = args\n\n const currentBranchNames = currentWorktrees.filter((branch) => {\n return branch.startsWith(RELEASE_BRANCH_PREFIX)\n })\n\n const branchesToCreate = selectedReleaseBranches.filter((branch) => {\n return !currentBranchNames.includes(branch)\n })\n\n return { branchesToCreate }\n}\n\n/**\n * Create worktrees for the specified branches\n */\nconst createWorktrees = async (branches: string[], worktreeDir: string): Promise<string[]> => {\n const results = await Promise.allSettled(\n branches.map(async (branch) => {\n const worktreePath = `${worktreeDir}/${branch}`\n\n await $`git worktree add ${worktreePath} ${branch}`\n await $({ cwd: worktreePath })`pnpm install`\n\n return branch\n }),\n )\n\n const created: string[] = []\n\n for (const [index, result] of results.entries()) {\n if (result.status === 'fulfilled') {\n created.push(result.value)\n } else {\n const branch = branches[index]\n const err = new OperationError(result.reason, {\n operation: `git worktree add for ${branch}`,\n remediation: 'check the branch name and that the parent dir is writable',\n })\n\n logger.error({ error: result.reason, msg: err.message })\n }\n }\n\n return created\n}\n\n/**\n * Log the results of worktree management\n */\nconst logResults = (created: string[]): void => {\n if (created.length > 0) {\n logger.info('\u2705 Created git worktrees:')\n for (const branch of created) {\n logger.info(branch)\n }\n logger.info('')\n } else {\n logger.info('\u2139\uFE0F No new git worktrees to create')\n }\n}\n\n// MCP Tool Registration\nexport const worktreesAddMcpTool = defineMcpTool({\n name: 'worktrees-add',\n description:\n 'Create local git worktrees for release branches under the worktrees directory and run \"pnpm install\" in each. Mutates the local filesystem. When invoked via MCP, pass either \"versions\" (comma-separated) or all=true \u2014 the branch picker and \"open in Cursor / GitHub Desktop / cmux\" follow-up prompts are unreachable without a TTY, and the CLI confirmation is auto-skipped for MCP calls.',\n inputSchema: {\n all: z\n .boolean()\n .optional()\n .describe(\n 'Add worktrees for every open release branch. Either \"all\" or \"versions\" must be provided for MCP calls (the interactive picker is unavailable without a TTY). Ignored if \"versions\" is provided.',\n ),\n versions: z\n .string()\n .optional()\n .describe(\n 'Comma-separated release versions to target (e.g. \"1.2.5, 1.2.6\"). Either \"versions\" or all=true must be provided for MCP calls. Overrides \"all\" when set.',\n ),\n cursor: z\n .enum(CURSOR_MODES)\n .optional()\n .describe(\n 'Cursor open mode for created worktrees. \"workspace\" appends each worktree as a folder to \"ide.config.workspaceConfigPath\" in infra-kit config and opens the workspace. \"windows\" opens each worktree in its own Cursor window. \"none\" skips Cursor. Resolution order: this flag \u2192 \"ide.config.mode\" from infra-kit config \u2192 interactive prompt (CLI) / \"none\" (MCP, no TTY).',\n ),\n githubDesktop: z\n .boolean()\n .optional()\n .describe(\n 'Open each created worktree in GitHub Desktop. Resolution order: this flag \u2192 \"worktrees.openInGithubDesktop\" from infra-kit config \u2192 interactive prompt (CLI) / false (MCP, no TTY).',\n ),\n cmux: z\n .boolean()\n .optional()\n .describe(\n 'Open each created worktree in a new cmux workspace with a 3-pane layout (left-top, left-bottom, full-height right), all rooted at the worktree directory. Resolution order: this flag \u2192 \"worktrees.openInCmux\" from infra-kit config \u2192 interactive prompt (CLI) / false (MCP, no TTY).',\n ),\n },\n outputSchema: {\n createdWorktrees: z.array(z.string()).describe('List of created git worktree branches'),\n count: z.number().describe('Number of git worktrees created'),\n },\n handler: worktreesAdd,\n})\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\n\ninterface AddFoldersToCursorWorkspaceArgs {\n workspacePath: string\n folderPaths: string[]\n}\n\ninterface AddFoldersToCursorWorkspaceResult {\n added: string[]\n skipped: string[]\n}\n\ninterface WorkspaceFolderEntry {\n path: string\n name?: string\n}\n\ninterface WorkspaceFile {\n folders?: WorkspaceFolderEntry[]\n [key: string]: unknown\n}\n\n/**\n * Adds folders to a Cursor (`.code-workspace`) file's `folders` array, skipping\n * entries that already point to the same absolute path. Folder paths are written\n * as relative to the workspace file's directory to match Cursor's default style.\n */\nexport const addFoldersToCursorWorkspace = async (\n args: AddFoldersToCursorWorkspaceArgs,\n): Promise<AddFoldersToCursorWorkspaceResult> => {\n const { workspacePath, folderPaths } = args\n\n const workspaceDir = path.dirname(workspacePath)\n\n let raw: string\n\n try {\n raw = await fs.readFile(workspacePath, 'utf-8')\n } catch (error) {\n throw new Error(`Cursor workspace file not found at ${workspacePath}: ${(error as Error).message}`)\n }\n\n let parsed: WorkspaceFile\n\n try {\n parsed = JSON.parse(raw) as WorkspaceFile\n } catch (error) {\n throw new Error(\n `Failed to parse ${workspacePath} as JSON. Comments (JSONC) are not supported. ${(error as Error).message}`,\n )\n }\n\n const existingFolders = parsed.folders ?? []\n const existingAbsolutePaths = new Set(\n existingFolders.map((entry) => {\n return path.resolve(workspaceDir, entry.path)\n }),\n )\n\n const added: string[] = []\n const skipped: string[] = []\n\n for (const folderPath of folderPaths) {\n const absolutePath = path.resolve(folderPath)\n\n if (existingAbsolutePaths.has(absolutePath)) {\n skipped.push(folderPath)\n continue\n }\n\n const relativePath = path.relative(workspaceDir, absolutePath)\n\n existingFolders.push({ path: relativePath })\n existingAbsolutePaths.add(absolutePath)\n added.push(folderPath)\n }\n\n parsed.folders = existingFolders\n\n await fs.writeFile(workspacePath, `${JSON.stringify(parsed, null, 2)}\\n`, 'utf-8')\n\n return { added, skipped }\n}\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\n\nimport { addFoldersToCursorWorkspace } from './add-folders-to-workspace'\nimport { removeFoldersFromCursorWorkspace } from './remove-folders-from-workspace'\n\ninterface ReconcileCursorWorkspaceFoldersArgs {\n workspacePath: string\n worktreeDir: string\n currentBranches: string[]\n}\n\ninterface ReconcileCursorWorkspaceFoldersResult {\n added: string[]\n removed: string[]\n}\n\ninterface WorkspaceFolderEntry {\n path: string\n name?: string\n}\n\ninterface WorkspaceFile {\n folders?: WorkspaceFolderEntry[]\n [key: string]: unknown\n}\n\n/**\n * Reconciles the configured Cursor workspace's `folders` array against the\n * actual set of release worktrees on disk:\n * - Adds any worktree folders that aren't already listed.\n * - Removes entries whose absolute path lives under `${worktreeDir}/release/`\n * but no longer corresponds to a current branch (drift from manual\n * `git worktree remove`, deleted branches, etc.).\n *\n * Non-worktree folder entries are left untouched.\n */\nexport const reconcileCursorWorkspaceFolders = async (\n args: ReconcileCursorWorkspaceFoldersArgs,\n): Promise<ReconcileCursorWorkspaceFoldersResult> => {\n const { workspacePath, worktreeDir, currentBranches } = args\n\n const workspaceDir = path.dirname(workspacePath)\n const releaseRoot = path.resolve(`${worktreeDir}/release`)\n\n const raw = await fs.readFile(workspacePath, 'utf-8')\n const parsed = JSON.parse(raw) as WorkspaceFile\n\n const existingFolders = parsed.folders ?? []\n\n const desiredAbsolutePaths = new Set(\n currentBranches.map((branch) => {\n return path.resolve(`${worktreeDir}/${branch}`)\n }),\n )\n\n const danglingFolderPaths: string[] = []\n\n for (const entry of existingFolders) {\n const entryAbsolutePath = path.resolve(workspaceDir, entry.path)\n\n const isReleaseShaped = entryAbsolutePath === releaseRoot || entryAbsolutePath.startsWith(`${releaseRoot}/`)\n\n if (isReleaseShaped && !desiredAbsolutePaths.has(entryAbsolutePath)) {\n danglingFolderPaths.push(entryAbsolutePath)\n }\n }\n\n let removed: string[] = []\n\n if (danglingFolderPaths.length > 0) {\n const result = await removeFoldersFromCursorWorkspace({\n workspacePath,\n folderPaths: danglingFolderPaths,\n })\n\n removed = result.removed\n }\n\n const desiredFolderPaths = currentBranches.map((branch) => {\n return `${worktreeDir}/${branch}`\n })\n\n const { added } =\n desiredFolderPaths.length > 0\n ? await addFoldersToCursorWorkspace({ workspacePath, folderPaths: desiredFolderPaths })\n : { added: [] as string[] }\n\n return { added, removed }\n}\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\n\ninterface RemoveFoldersFromCursorWorkspaceArgs {\n workspacePath: string\n folderPaths: string[]\n}\n\ninterface RemoveFoldersFromCursorWorkspaceResult {\n removed: string[]\n notFound: string[]\n}\n\ninterface WorkspaceFolderEntry {\n path: string\n name?: string\n}\n\ninterface WorkspaceFile {\n folders?: WorkspaceFolderEntry[]\n [key: string]: unknown\n}\n\n/**\n * Removes folders from a Cursor (`.code-workspace`) file's `folders` array. Entries\n * are matched by resolved absolute path, so relative and absolute entries pointing\n * at the same folder are both removed.\n */\nexport const removeFoldersFromCursorWorkspace = async (\n args: RemoveFoldersFromCursorWorkspaceArgs,\n): Promise<RemoveFoldersFromCursorWorkspaceResult> => {\n const { workspacePath, folderPaths } = args\n\n const workspaceDir = path.dirname(workspacePath)\n\n let raw: string\n\n try {\n raw = await fs.readFile(workspacePath, 'utf-8')\n } catch (error) {\n throw new Error(`Cursor workspace file not found at ${workspacePath}: ${(error as Error).message}`)\n }\n\n let parsed: WorkspaceFile\n\n try {\n parsed = JSON.parse(raw) as WorkspaceFile\n } catch (error) {\n throw new Error(\n `Failed to parse ${workspacePath} as JSON. Comments (JSONC) are not supported. ${(error as Error).message}`,\n )\n }\n\n const existingFolders = parsed.folders ?? []\n const targetAbsolutePaths = new Set(\n folderPaths.map((folderPath) => {\n return path.resolve(folderPath)\n }),\n )\n\n const removedAbsolutePaths = new Set<string>()\n\n const filteredFolders = existingFolders.filter((entry) => {\n const entryAbsolutePath = path.resolve(workspaceDir, entry.path)\n\n if (targetAbsolutePaths.has(entryAbsolutePath)) {\n removedAbsolutePaths.add(entryAbsolutePath)\n\n return false\n }\n\n return true\n })\n\n parsed.folders = filteredFolders\n\n await fs.writeFile(workspacePath, `${JSON.stringify(parsed, null, 2)}\\n`, 'utf-8')\n\n const removed: string[] = []\n const notFound: string[] = []\n\n for (const folderPath of folderPaths) {\n const absolutePath = path.resolve(folderPath)\n\n if (removedAbsolutePaths.has(absolutePath)) {\n removed.push(folderPath)\n } else {\n notFound.push(folderPath)\n }\n }\n\n return { removed, notFound }\n}\n", "import path from 'node:path'\n\n/**\n * Resolves the configured Cursor workspace path against the project root.\n * Absolute paths are returned unchanged.\n */\nexport const resolveCursorWorkspacePath = (configValue: string, projectRoot: string): string => {\n if (path.isAbsolute(configValue)) {\n return configValue\n }\n\n return path.resolve(projectRoot, configValue)\n}\n", "import { z } from 'zod/v4'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { getCurrentWorktrees } from 'src/lib/git-utils'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatVersionLabel, getJiraDescriptions } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface WorktreeInfo {\n version: string\n type: ReleaseType\n description: string | null\n}\n\n/**\n * List all release git worktrees with version, type, and Jira description\n */\nexport const worktreesList = async () => {\n const currentWorktrees = await getCurrentWorktrees('release')\n\n if (currentWorktrees.length === 0) {\n logger.info('\u2139\uFE0F No active worktrees found')\n\n return {\n content: textContent(JSON.stringify({ worktrees: [], count: 0 }, null, 2)),\n structuredContent: { worktrees: [], count: 0 },\n }\n }\n\n const [releasePRsInfo, jiraDescriptions] = await Promise.all([getReleasePRsWithInfo(), getJiraDescriptions()])\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n const worktrees: WorktreeInfo[] = currentWorktrees.map((branch) => {\n const version = branch.replace('release/', '')\n const type = releaseTypes.get(branch) || 'regular'\n const description = jiraDescriptions.get(version) || null\n\n return { version, type, description }\n })\n\n // Log formatted output\n const maxVersionLength = Math.max(\n ...worktrees.map((w) => {\n return w.version.length\n }),\n )\n\n const formattedLines = worktrees.map((worktree) => {\n const label = formatVersionLabel(worktree.version, worktree.type, maxVersionLength)\n\n if (worktree.description) {\n return `${label} ${worktree.description}`\n }\n\n return label\n })\n\n logger.info('\uD83C\uDF3F Active worktrees:')\n logger.info(`\\n${formattedLines.join('\\n')}\\n`)\n\n const structuredContent = {\n worktrees,\n count: worktrees.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const worktreesListMcpTool = defineMcpTool({\n name: 'worktrees-list',\n description:\n 'List existing release-branch worktrees with version, release type (regular / hotfix), and Jira fix-version description. Read-only.',\n inputSchema: {},\n outputSchema: {\n worktrees: z\n .array(\n z.object({\n version: z.string().describe('Release version'),\n type: z.enum(['regular', 'hotfix']).describe('Release type'),\n description: z.string().nullable().describe('Jira version description'),\n }),\n )\n .describe('List of all worktrees with details'),\n count: z.number().describe('Number of worktrees'),\n },\n handler: worktreesList,\n})\n", "import { z } from 'zod/v4'\nimport { $ } from 'zx'\n\nimport { buildCmuxWorkspaceTitle, listCmuxWorkspaceTitles, openCmuxWorkspaceWithLayout } from 'src/integrations/cmux'\nimport { reconcileCursorWorkspaceFolders, resolveCursorWorkspacePath } from 'src/integrations/cursor'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface WorktreesOpenResult {\n openedCmux: string[]\n skippedCmux: string[]\n cursorFoldersAdded: number\n cursorFoldersRemoved: number\n}\n\n/**\n * Cold-start restore command: reconciles `Main.code-workspace` against the set\n * of release worktrees on disk, opens Cursor against it, and ensures one cmux\n * workspace exists per worktree. Idempotent and additive \u2014 never removes\n * worktrees, never recreates running cmux workspaces.\n */\nexport const worktreesOpen = async () => {\n commandEcho.start('worktrees-open')\n\n try {\n const projectRoot = await getProjectRoot()\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n const currentBranches = await getCurrentWorktrees('release')\n\n const cursorOutcome = await openCursor({ projectRoot, worktreeDir, currentBranches })\n const cmuxOutcome = await openCmux({ worktreeDir, currentBranches })\n\n const result: WorktreesOpenResult = {\n openedCmux: cmuxOutcome.opened,\n skippedCmux: cmuxOutcome.skipped,\n cursorFoldersAdded: cursorOutcome.added,\n cursorFoldersRemoved: cursorOutcome.removed,\n }\n\n logResults(result, { cursorRan: cursorOutcome.ran, cmuxRan: cmuxOutcome.ran })\n\n commandEcho.print()\n\n return {\n content: textContent(JSON.stringify(result, null, 2)),\n structuredContent: { ...result },\n }\n } catch (error) {\n logger.error({ error }, '\u274C Error opening worktrees')\n throw new OperationError(error, {\n operation: 'open worktrees',\n remediation: \"run 'worktrees-list' to confirm the branches exist\",\n })\n }\n}\n\ninterface OpenCursorArgs {\n projectRoot: string\n worktreeDir: string\n currentBranches: string[]\n}\n\ninterface OpenCursorOutcome {\n ran: boolean\n added: number\n removed: number\n}\n\nconst openCursor = async (args: OpenCursorArgs): Promise<OpenCursorOutcome> => {\n const { projectRoot, worktreeDir, currentBranches } = args\n\n const config = await getInfraKitConfig()\n const cursorConfig = config.ide?.provider === 'cursor' ? config.ide.config : undefined\n\n if (!cursorConfig || cursorConfig.mode !== 'workspace' || !cursorConfig.workspaceConfigPath) {\n logger.warn('\u26A0\uFE0F Skipping Cursor: ide.provider must be \"cursor\", mode \"workspace\", and workspaceConfigPath set.')\n\n return { ran: false, added: 0, removed: 0 }\n }\n\n const workspacePath = resolveCursorWorkspacePath(cursorConfig.workspaceConfigPath, projectRoot)\n\n try {\n const { added, removed } = await reconcileCursorWorkspaceFolders({\n workspacePath,\n worktreeDir,\n currentBranches,\n })\n\n await $`cursor ${workspacePath}`\n\n return { ran: true, added: added.length, removed: removed.length }\n } catch (error) {\n logger.warn({ error }, `\u26A0\uFE0F Failed to reconcile/open Cursor workspace at ${workspacePath}`)\n\n return { ran: false, added: 0, removed: 0 }\n }\n}\n\ninterface OpenCmuxArgs {\n worktreeDir: string\n currentBranches: string[]\n}\n\ninterface OpenCmuxOutcome {\n ran: boolean\n opened: string[]\n skipped: string[]\n}\n\nconst openCmux = async (args: OpenCmuxArgs): Promise<OpenCmuxOutcome> => {\n const { worktreeDir, currentBranches } = args\n\n if (currentBranches.length === 0) {\n return { ran: true, opened: [], skipped: [] }\n }\n\n const repoName = await getRepoName()\n const existingTitles = await listCmuxWorkspaceTitles()\n\n const opened: string[] = []\n const skipped: string[] = []\n\n for (const branch of currentBranches) {\n const title = buildCmuxWorkspaceTitle({ repoName, branch })\n\n if (existingTitles.has(title)) {\n skipped.push(title)\n continue\n }\n\n try {\n await openCmuxWorkspaceWithLayout({ cwd: `${worktreeDir}/${branch}`, title })\n opened.push(title)\n } catch (error) {\n logger.warn({ error, title }, `\u26A0\uFE0F Failed to open cmux workspace for ${branch}`)\n }\n }\n\n return { ran: true, opened, skipped }\n}\n\ninterface LogResultsContext {\n cursorRan: boolean\n cmuxRan: boolean\n}\n\nconst logResults = (result: WorktreesOpenResult, context: LogResultsContext): void => {\n if (context.cursorRan) {\n if (result.cursorFoldersAdded > 0) {\n logger.info(`\u2705 Added ${result.cursorFoldersAdded} folder(s) to Cursor workspace`)\n }\n\n if (result.cursorFoldersRemoved > 0) {\n logger.info(`\uD83E\uDDF9 Removed ${result.cursorFoldersRemoved} dangling folder(s) from Cursor workspace`)\n }\n }\n\n if (result.openedCmux.length > 0) {\n logger.info('\u2705 Opened cmux workspaces:')\n for (const title of result.openedCmux) {\n logger.info(title)\n }\n }\n\n if (result.skippedCmux.length > 0) {\n logger.info(`\u2139\uFE0F Skipped ${result.skippedCmux.length} cmux workspace(s) already open`)\n }\n\n if (\n !context.cursorRan &&\n result.openedCmux.length === 0 &&\n result.skippedCmux.length === 0 &&\n result.cursorFoldersAdded === 0 &&\n result.cursorFoldersRemoved === 0\n ) {\n logger.info('\u2139\uFE0F Nothing to open')\n }\n}\n\n// MCP Tool Registration\nexport const worktreesOpenMcpTool = defineMcpTool({\n name: 'worktrees-open',\n description:\n 'Open Cursor against the configured workspace file and ensure a cmux workspace exists for each existing release worktree. Idempotent and additive \u2014 never removes worktrees, never recreates running cmux workspaces. Use after a cold start (Cursor + cmux closed). For stale-worktree cleanup, use worktrees-sync.',\n inputSchema: {},\n outputSchema: {\n openedCmux: z.array(z.string()).describe('Titles of cmux workspaces opened during this run'),\n skippedCmux: z.array(z.string()).describe('Titles of cmux workspaces that were already open'),\n cursorFoldersAdded: z.number().describe('Number of worktree folders added to the Cursor workspace file'),\n cursorFoldersRemoved: z\n .number()\n .describe('Number of dangling worktree folders removed from the Cursor workspace file'),\n },\n handler: worktreesOpen,\n})\n", "import checkbox from '@inquirer/checkbox'\nimport confirm from '@inquirer/confirm'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\n\nimport { removeFoldersFromCursorWorkspace, resolveCursorWorkspacePath } from 'src/integrations/cursor'\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { removeWorktrees } from 'src/lib/worktrees'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\n// Constants\ninterface WorktreeManagementArgs extends RequiredConfirmedOptionArg {\n all?: boolean\n versions?: string\n}\n\n/**\n * Manage git worktrees for release branches\n * Creates worktrees for active release branches and removes unused ones\n */\nexport const worktreesRemove = async (options: WorktreeManagementArgs) => {\n const { confirmedCommand, all, versions } = options\n\n commandEcho.start('worktrees-remove')\n\n try {\n const currentWorktrees = await getCurrentWorktrees('release')\n\n if (currentWorktrees.length === 0) {\n logger.info('\u2139\uFE0F No active worktrees to remove')\n\n commandEcho.print()\n\n return {\n content: textContent(JSON.stringify({ removedWorktrees: [], count: 0 }, null, 2)),\n structuredContent: { removedWorktrees: [], count: 0 },\n }\n }\n\n const projectRoot = await getProjectRoot()\n\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n\n let selectedReleaseBranches: string[] = []\n\n if (all) {\n selectedReleaseBranches = currentWorktrees\n } else if (versions) {\n selectedReleaseBranches = versions.split(',').map((v) => {\n return `release/v${v.trim()}`\n })\n } else {\n commandEcho.setInteractive()\n\n const [descriptions, prInfo] = await Promise.all([getJiraDescriptions(), getReleasePRsWithInfo()])\n\n const releaseTypes = new Map<string, ReleaseType>(\n prInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n selectedReleaseBranches = await checkbox({\n required: true,\n message: '\uD83C\uDF3F Select release branches',\n choices: formatBranchChoices({ branches: currentWorktrees, descriptions, types: releaseTypes }),\n })\n }\n\n // Track --all flag if all branches were selected (either via flag or interactively)\n const allSelected = selectedReleaseBranches.length === currentWorktrees.length\n\n if (allSelected) {\n commandEcho.addOption('--all', true)\n } else {\n commandEcho.addOption(\n '--versions',\n selectedReleaseBranches.map((branch) => {\n return branch.replace('release/v', '')\n }),\n )\n }\n\n // Ask for confirmation\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: 'Are you sure you want to proceed with these worktree changes?',\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n if (!confirmedCommand) {\n commandEcho.addOption('--yes', true)\n }\n\n const repoName = await getRepoName()\n\n const removedWorktrees = await removeWorktrees({\n branches: selectedReleaseBranches,\n worktreeDir,\n repoName,\n pruneFolder: allSelected,\n })\n\n await syncCursorWorkspaceOnRemove({ removedWorktrees, worktreeDir, projectRoot })\n\n logResults(removedWorktrees)\n\n commandEcho.print()\n\n const structuredContent = {\n removedWorktrees,\n count: removedWorktrees.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error) {\n logger.error({ error }, '\u274C Error managing worktrees')\n throw new OperationError(error, {\n operation: 'remove worktrees',\n remediation: \"check 'git worktree list' for the path; uncommitted changes block removal\",\n })\n }\n}\n\ninterface SyncCursorWorkspaceOnRemoveArgs {\n removedWorktrees: string[]\n worktreeDir: string\n projectRoot: string\n}\n\n/**\n * Strip removed worktrees from the configured Cursor workspace's `folders` array.\n * No-op if Cursor isn't configured, mode isn't \"workspace\", or no worktrees were removed.\n */\nconst syncCursorWorkspaceOnRemove = async (args: SyncCursorWorkspaceOnRemoveArgs): Promise<void> => {\n const { removedWorktrees, worktreeDir, projectRoot } = args\n\n if (removedWorktrees.length === 0) {\n return\n }\n\n const config = await getInfraKitConfig()\n const cursorConfig = config.ide?.provider === 'cursor' ? config.ide.config : undefined\n\n if (!cursorConfig || cursorConfig.mode !== 'workspace' || !cursorConfig.workspaceConfigPath) {\n return\n }\n\n const workspacePath = resolveCursorWorkspacePath(cursorConfig.workspaceConfigPath, projectRoot)\n\n const folderPaths = removedWorktrees.map((branch) => {\n return `${worktreeDir}/${branch}`\n })\n\n try {\n const { removed } = await removeFoldersFromCursorWorkspace({ workspacePath, folderPaths })\n\n if (removed.length > 0) {\n logger.info(`\u2705 Removed ${removed.length} folder(s) from ${workspacePath}`)\n }\n } catch (error) {\n logger.warn({ error }, `\u26A0\uFE0F Failed to update Cursor workspace at ${workspacePath}`)\n }\n}\n\n/**\n * Log the results of worktree management\n */\nconst logResults = (removed: string[]): void => {\n if (removed.length > 0) {\n logger.info('\u274C Removed worktrees:')\n for (const branch of removed) {\n logger.info(branch)\n }\n logger.info('')\n } else {\n logger.info('\u2139\uFE0F No unused worktrees to remove')\n }\n}\n\n// MCP Tool Registration\nexport const worktreesRemoveMcpTool = defineMcpTool({\n name: 'worktrees-remove',\n description:\n 'Remove local git worktrees for release branches. When everything is removed, also runs \"git worktree prune\" and deletes the worktrees directory. When invoked via MCP, pass either \"versions\" (comma-separated) or all=true \u2014 the branch picker is unreachable without a TTY, and the CLI confirmation is auto-skipped for MCP calls, so the caller is responsible for gating.',\n inputSchema: {\n all: z\n .boolean()\n .optional()\n .describe(\n 'Remove every existing worktree. Either \"all\" or \"versions\" must be provided for MCP calls (the interactive picker is unavailable without a TTY). Ignored if \"versions\" is provided.',\n ),\n versions: z\n .string()\n .optional()\n .describe(\n 'Comma-separated release versions to target (e.g. \"1.2.5, 1.2.6\"). Either \"versions\" or all=true must be provided for MCP calls. Overrides \"all\" when set.',\n ),\n },\n outputSchema: {\n removedWorktrees: z.array(z.string()).describe('List of removed git worktree branches'),\n count: z.number().describe('Number of git worktrees removed'),\n },\n handler: worktreesRemove,\n})\n", "import confirm from '@inquirer/confirm'\nimport process from 'node:process'\nimport { z } from 'zod/v4'\nimport { $ } from 'zx'\n\nimport { buildCmuxWorkspaceTitle, closeCmuxWorkspaceByTitle } from 'src/integrations/cmux'\nimport { removeFoldersFromCursorWorkspace, resolveCursorWorkspacePath } from 'src/integrations/cursor'\nimport { getReleasePRs } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\n// Constants\nconst RELEASE_BRANCH_PREFIX = 'release/v'\n\ninterface WorktreeSyncArgs extends RequiredConfirmedOptionArg {}\n\n/**\n * Manage git worktrees for release branches.\n *\n * Creates worktrees for active release branches and removes unused ones\n */\nexport const worktreesSync = async (options: WorktreeSyncArgs) => {\n const { confirmedCommand } = options\n\n commandEcho.start('worktrees-sync')\n\n try {\n const currentWorktrees = await getCurrentWorktrees('release')\n const projectRoot = await getProjectRoot()\n\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n\n const releasePRsList = await getReleasePRs()\n\n // Ask for confirmation\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: 'Are you sure you want to proceed with these worktree changes?',\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n if (!confirmedCommand) {\n commandEcho.addOption('--yes', true)\n }\n\n const { branchesToRemove } = categorizeWorktrees({\n releasePRsList,\n currentWorktrees,\n })\n\n const repoName = await getRepoName()\n\n const removedWorktrees = await removeWorktrees({\n branches: branchesToRemove,\n worktreeDir,\n repoName,\n })\n\n await syncCursorWorkspaceOnRemove({ removedWorktrees, worktreeDir, projectRoot })\n\n logResults(removedWorktrees)\n\n commandEcho.print()\n\n const structuredContent = {\n removedWorktrees,\n count: removedWorktrees.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error) {\n logger.error({ error }, '\u274C Error managing worktrees')\n throw new OperationError(error, {\n operation: 'sync worktrees with remote',\n remediation: \"ensure 'gh auth status' is ok and you can reach origin\",\n })\n }\n}\n\ninterface CategorizeWorktreesArgs {\n releasePRsList: string[]\n currentWorktrees: string[]\n}\n\n/**\n * Categorize worktrees into those that need to be created or removed\n */\nconst categorizeWorktrees = (args: CategorizeWorktreesArgs): { branchesToRemove: string[] } => {\n const { releasePRsList, currentWorktrees } = args\n\n const currentBranchNames = currentWorktrees.filter((branch) => {\n return branch.startsWith(RELEASE_BRANCH_PREFIX)\n })\n\n const branchesToRemove = currentBranchNames.filter((branch) => {\n return !releasePRsList.includes(branch)\n })\n\n return { branchesToRemove }\n}\n\ninterface RemoveWorktreesArgs {\n branches: string[]\n worktreeDir: string\n repoName: string\n}\n\n/**\n * Remove worktrees for the specified branches and close their cmux workspaces\n */\nconst removeWorktrees = async (args: RemoveWorktreesArgs): Promise<string[]> => {\n const { branches, worktreeDir, repoName } = args\n\n const removed: string[] = []\n\n for (const branch of branches) {\n try {\n const worktreePath = `${worktreeDir}/${branch}`\n\n const title = buildCmuxWorkspaceTitle({ repoName, branch })\n\n await closeCmuxWorkspaceByTitle(title)\n\n await $`git worktree remove ${worktreePath}`\n removed.push(branch)\n } catch (error) {\n const err = new OperationError(error, {\n operation: `remove stale worktree for ${branch}`,\n remediation: 'inspect the worktree dir manually; rerun with the branch checked out elsewhere',\n })\n\n logger.error({ error, branch, msg: err.message })\n }\n }\n\n return removed\n}\n\ninterface SyncCursorWorkspaceOnRemoveArgs {\n removedWorktrees: string[]\n worktreeDir: string\n projectRoot: string\n}\n\n/**\n * Strip removed worktrees from the configured Cursor workspace's `folders` array.\n * No-op if Cursor isn't configured, mode isn't \"workspace\", or no worktrees were removed.\n */\nconst syncCursorWorkspaceOnRemove = async (args: SyncCursorWorkspaceOnRemoveArgs): Promise<void> => {\n const { removedWorktrees, worktreeDir, projectRoot } = args\n\n if (removedWorktrees.length === 0) {\n return\n }\n\n const config = await getInfraKitConfig()\n const cursorConfig = config.ide?.provider === 'cursor' ? config.ide.config : undefined\n\n if (!cursorConfig || cursorConfig.mode !== 'workspace' || !cursorConfig.workspaceConfigPath) {\n return\n }\n\n const workspacePath = resolveCursorWorkspacePath(cursorConfig.workspaceConfigPath, projectRoot)\n\n const folderPaths = removedWorktrees.map((branch) => {\n return `${worktreeDir}/${branch}`\n })\n\n try {\n const { removed: removedEntries } = await removeFoldersFromCursorWorkspace({ workspacePath, folderPaths })\n\n if (removedEntries.length > 0) {\n logger.info(`\u2705 Removed ${removedEntries.length} folder(s) from ${workspacePath}`)\n }\n } catch (error) {\n logger.warn({ error }, `\u26A0\uFE0F Failed to update Cursor workspace at ${workspacePath}`)\n }\n}\n\n/**\n * Log the results of worktree management\n */\nconst logResults = (removed: string[]): void => {\n if (removed.length > 0) {\n logger.info('\u274C Removed worktrees:')\n for (const branch of removed) {\n logger.info(branch)\n }\n logger.info('')\n } else {\n logger.info('\u2139\uFE0F No unused worktrees to remove')\n }\n}\n\n// MCP Tool Registration\nexport const worktreesSyncMcpTool = defineMcpTool({\n name: 'worktrees-sync',\n description:\n 'Remove worktrees whose release PR is no longer open (stale cleanup). Only removes \u2014 never creates; use worktrees-add to create worktrees for new releases. The CLI confirmation is auto-skipped for MCP calls, so the caller is responsible for gating.',\n inputSchema: {},\n outputSchema: {\n removedWorktrees: z.array(z.string()).describe('List of removed worktree branches'),\n count: z.number().describe('Number of worktrees removed during sync'),\n },\n handler: worktreesSync,\n})\n", "import { logger } from 'src/lib/logger'\nimport type { ToolsExecutionResult } from 'src/types'\n\ninterface ToolHandlerArgs {\n toolName: string\n handler: (params: any) => Promise<ToolsExecutionResult>\n}\n\nexport const createToolHandler = (args: ToolHandlerArgs) => {\n return async (params: unknown) => {\n const { toolName, handler } = args\n\n logger.info({ msg: `Tool execution started: ${toolName}`, params })\n try {\n const payload = await handler({ ...(params as object), confirmedCommand: true })\n\n logger.info({ msg: `Tool execution successful: ${toolName}` })\n\n return payload\n } catch (error) {\n logger.error({\n err: error,\n params,\n msg: `Tool execution failed: ${toolName}`,\n })\n\n throw error\n }\n }\n}\n", "import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\n\nimport { envClearMcpTool } from 'src/commands/env-clear'\nimport { envListMcpTool } from 'src/commands/env-list'\nimport { envLoadMcpTool } from 'src/commands/env-load'\nimport { envStatusMcpTool } from 'src/commands/env-status'\nimport { ghMergeDevMcpTool } from 'src/commands/gh-merge-dev'\nimport { ghReleaseDeliverMcpTool } from 'src/commands/gh-release-deliver'\nimport { ghReleaseDeployAllMcpTool } from 'src/commands/gh-release-deploy-all'\nimport { ghReleaseDeploySelectedMcpTool } from 'src/commands/gh-release-deploy-selected'\nimport { ghReleaseListMcpTool } from 'src/commands/gh-release-list'\nimport { releaseCreateMcpTool } from 'src/commands/release-create'\nimport { releaseDescEditMcpTool } from 'src/commands/release-desc-edit'\nimport { versionMcpTool } from 'src/commands/version'\nimport { worktreesAddMcpTool } from 'src/commands/worktrees-add'\nimport { worktreesListMcpTool } from 'src/commands/worktrees-list'\nimport { worktreesOpenMcpTool } from 'src/commands/worktrees-open'\nimport { worktreesRemoveMcpTool } from 'src/commands/worktrees-remove'\nimport { worktreesSyncMcpTool } from 'src/commands/worktrees-sync'\nimport { createToolHandler } from 'src/lib/tool-handler'\n\nconst tools = [\n envStatusMcpTool,\n envListMcpTool,\n envLoadMcpTool,\n envClearMcpTool,\n ghMergeDevMcpTool,\n releaseCreateMcpTool,\n releaseDescEditMcpTool,\n ghReleaseDeliverMcpTool,\n ghReleaseDeployAllMcpTool,\n ghReleaseDeploySelectedMcpTool,\n ghReleaseListMcpTool,\n versionMcpTool,\n worktreesAddMcpTool,\n worktreesListMcpTool,\n worktreesOpenMcpTool,\n worktreesRemoveMcpTool,\n worktreesSyncMcpTool,\n]\n\nexport const initializeTools = async (server: McpServer) => {\n for (const tool of tools) {\n server.registerTool(\n tool.name,\n {\n description: tool.description,\n inputSchema: tool.inputSchema,\n outputSchema: tool.outputSchema,\n },\n createToolHandler({ toolName: tool.name, handler: tool.handler }),\n )\n }\n}\n"],
5
- "mappings": "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",
6
- "names": ["StdioServerTransport", "process", "process", "process", "pino", "pretty", "LOG_FILE_PATH", "initLoggerMcp", "logLevel", "logger", "initLoggerCLI", "ignoreFields", "setupErrorHandlers", "logger", "process", "error", "LOG_FILE_PATH", "reason", "promise", "McpServer", "initializePrompts", "_server", "initializeResources", "_server", "fs", "path", "process", "z", "fs", "os", "path", "process", "ENV_LOAD_FILE", "ENV_CLEAR_FILE", "INFRA_KIT_SESSION_VAR", "INFRA_KIT_ENV_CONFIG_VAR", "INFRA_KIT_ENV_PROJECT_VAR", "INFRA_KIT_ENV_LOADED_AT_VAR", "ENV_VAR_LINE_PATTERN", "parseVarNamesFromEnvFile", "filePath", "content", "names", "line", "match", "getCacheRoot", "xdg", "base", "getSessionCacheDir", "session", "atomicWriteFileSync", "mode", "tmpPath", "error", "WORKTREES_DIR_SUFFIX", "textContent", "text", "defineMcpTool", "tool", "envClear", "cacheDir", "getSessionCacheDir", "envLoadPath", "path", "ENV_LOAD_FILE", "fs", "varNames", "parseVarNamesFromEnvFile", "unsetLines", "v", "INFRA_KIT_ENV_CONFIG_VAR", "INFRA_KIT_ENV_PROJECT_VAR", "INFRA_KIT_ENV_LOADED_AT_VAR", "clearFilePath", "ENV_CLEAR_FILE", "atomicWriteFileSync", "process", "structuredContent", "textContent", "envClearMcpTool", "defineMcpTool", "z", "z", "fs", "os", "path", "yaml", "z", "path", "$", "getCurrentWorktrees", "type", "worktreeLines", "worktreePredicateMap", "releaseWorktreePredicate", "featureWorktreePredicate", "branch", "parseWorktreeBranch", "line", "trimmed", "open", "getProjectRoot", "getRepoName", "projectRoot", "INFRA_KIT_CONFIG_FILE", "USER_CONFIG_DIR_NAME", "USER_GLOBAL_CONFIG_FILE", "USER_PROJECTS_DIR", "dopplerEnvManagementSchema", "z", "envManagementSchema", "cursorIdeConfigSchema", "v", "cursorIdeSchema", "ideSchema", "jiraTaskManagerSchema", "taskManagerSchema", "worktreesConfigSchema", "infraKitConfigSchema", "infraKitOverrideConfigSchema", "cached", "getInfraKitConfigPaths", "projectRoot", "getProjectRoot", "projectName", "getRepoName", "userConfigDir", "path", "os", "getInfraKitConfig", "paths", "mainStat", "fs", "userGlobalStat", "userProjectStat", "statIfExists", "mtimes", "shallowEqual", "layers", "merged", "layer", "data", "loadLayer", "finalResult", "statIfExists", "filePath", "fs", "readIfExists", "shallowEqual", "a", "b", "keys", "k", "loadLayer", "layer", "raw", "parsedRaw", "yaml", "result", "infraKitOverrideConfigSchema", "z", "getDopplerProject", "envManagement", "getInfraKitConfig", "envList", "project", "getDopplerProject", "environments", "getInfraKitConfig", "logger", "env", "structuredContent", "textContent", "envListMcpTool", "defineMcpTool", "z", "select", "Buffer", "fs", "path", "process", "z", "$", "$", "validateDopplerCliAndAuth", "error", "createCommandEcho", "commandName", "options", "isInteractive", "name", "flag", "value", "formattedOptions", "opt", "logger", "commandEcho", "envLoad", "args", "validateDopplerCliAndAuth", "config", "commandEcho", "selectedConfig", "environments", "getInfraKitConfig", "select", "env", "process", "project", "getDopplerProject", "envContent", "downloadDopplerSecrets", "assertValidEnvContent", "loadedAt", "envFileLines", "INFRA_KIT_ENV_CONFIG_VAR", "shellSingleQuote", "INFRA_KIT_ENV_PROJECT_VAR", "INFRA_KIT_ENV_LOADED_AT_VAR", "cacheDir", "getSessionCacheDir", "envFilePath", "path", "ENV_LOAD_FILE", "fs", "atomicWriteFileSync", "varCount", "countEnvVarLines", "structuredContent", "textContent", "DOPPLER_MAX_OUTPUT_BYTES", "DOPPLER_DOWNLOAD_TIMEOUT_MS", "prevQuiet", "$", "result", "assertDopplerOutputSize", "stdout", "bytes", "Buffer", "content", "line", "ENV_VAR_LINE_PATTERN", "SHELL_DIRECTIVE_LINES", "value", "trimmed", "envLoadMcpTool", "defineMcpTool", "z", "path", "process", "z", "envStatus", "validateDopplerCliAndAuth", "logger", "cacheDir", "getSessionCacheDir", "sessionId", "process", "INFRA_KIT_SESSION_VAR", "envLoadPath", "path", "ENV_LOAD_FILE", "sessionLoadedCount", "sessionTotalCount", "sessionConfig", "INFRA_KIT_ENV_CONFIG_VAR", "sessionProject", "INFRA_KIT_ENV_PROJECT_VAR", "sessionLoadedAt", "INFRA_KIT_ENV_LOADED_AT_VAR", "varNames", "parseVarNamesFromEnvFile", "v", "loadedAtDisplay", "missing", "structuredContent", "textContent", "envStatusMcpTool", "defineMcpTool", "z", "checkbox", "confirm", "process", "z", "$", "$", "process", "$", "$", "process", "createJiraVersion", "params", "config", "baseUrl", "token", "email", "projectId", "requestBody", "url", "credentials", "response", "errorText", "logger", "error", "getProjectVersions", "findVersionByName", "versionName", "v", "updateJiraVersion", "deliverJiraRelease", "version", "loadJiraConfig", "process", "projectIdStr", "missingVars", "errorMessage", "loadJiraConfigOptional", "getBaseBranch", "type", "prepareGitForRelease", "baseBranch", "$", "createSingleRelease", "args", "version", "jiraConfig", "description", "versionName", "result", "createJiraVersion", "jiraVersionUrl", "releaseInfo", "createReleaseBranch", "getJiraDescriptions", "descriptions", "loadJiraConfigOptional", "versions", "getProjectVersions", "formatVersionLabel", "maxVersionLength", "padding", "tag", "detectReleaseType", "title", "formatBranchChoices", "branches", "types", "versionNames", "b", "maxLen", "v", "branch", "i", "desc", "name", "$", "parseVersion", "versionStr", "sortVersions", "versions", "a", "b", "majA", "minA", "patchA", "majB", "minB", "patchB", "NEXT_TOKEN", "VERSION_RE", "stripBranchPrefix", "raw", "tryParse", "cleaned", "match", "semverKey", "v", "collectKnownVersions", "sources", "all", "parsed", "seen", "key", "sortVersions", "s", "parseVersion", "NoPriorVersionsError", "computeNextVersion", "known", "type", "max", "major", "minor", "highestPatchOnMinor", "acc", "isNextToken", "token", "resolveReleaseEntries", "entries", "known", "running", "entry", "trimmed", "isNextToken", "next", "computeNextVersion", "parseVersion", "parsed", "tryParse", "explicit", "hasNextToken", "e", "parseRemoteRefs", "previousQuiet", "$", "line", "tab", "fetchJiraVersionNames", "config", "loadJiraConfigOptional", "getProjectVersions", "v", "loadExistingVersions", "branchesResult", "jiraResult", "logger", "collectKnownVersions", "fetchAllReleasePRs", "releasePRs", "$", "hotfixPRs", "all", "seen", "pr", "getReleasePRs", "prs", "logger", "process", "sortVersions", "error", "getReleasePRsWithInfo", "sortedBranches", "prByBranch", "branch", "updateReleasePRBody", "args", "body", "createReleaseBranch", "version", "jiraVersionUrl", "type", "description", "titlePrefix", "baseBranch", "getBaseBranch", "branchName", "prLink", "extractStderr", "cause", "stderr", "buildMessage", "ctx", "parts", "OperationError", "ghMergeDev", "args", "all", "confirmedCommand", "commandEcho", "releasePRsList", "getReleasePRsWithInfo", "pr", "detectReleaseType", "logger", "textContent", "selectedReleaseBranches", "descriptions", "getJiraDescriptions", "checkbox", "formatBranchChoices", "branch", "answer", "confirm", "process", "$", "failedBranches", "mergeDev", "structuredContent", "error", "err", "OperationError", "ghMergeDevMcpTool", "defineMcpTool", "z", "confirm", "select", "process", "z", "$", "readTrimmedString", "value", "max", "formatZxError", "error", "rec", "fields", "exitCode", "stderr", "stdout", "$", "$", "closeCmuxWorkspaceByTitle", "title", "listOutput", "$", "ref", "findWorkspaceRefByTitle", "error", "logger", "output", "rawLine", "match", "$", "listCmuxWorkspaceTitles", "output", "$", "titles", "rawLine", "match", "title", "error", "logger", "$", "openCmuxWorkspaceWithLayout", "args", "cwd", "title", "newWorkspaceOutput", "workspaceRef", "parseWorkspaceRef", "surfacesOutput", "leftTopRef", "parseFirstSurfaceRef", "output", "match", "buildCmuxWorkspaceTitle", "args", "repoName", "branch", "version", "removeWorktrees", "args", "branches", "worktreeDir", "repoName", "pruneFolder", "results", "branch", "worktreePath", "title", "buildCmuxWorkspaceTitle", "closeCmuxWorkspaceByTitle", "$", "removed", "index", "result", "err", "OperationError", "logger", "runStep", "operation", "remediation", "fn", "error", "logger", "formatZxError", "OperationError", "fetchPRByHead", "head", "result", "$", "fetchMergedRcPRForVersion", "version", "expectedTitle", "pr", "fetchOpenDevToMainPR", "resolveTargetFromVersion", "selectedReleaseBranch", "resolveTargetInteractively", "releasePRsInfo", "getReleasePRsWithInfo", "branches", "releaseTypes", "detectReleaseType", "commandEcho", "descriptions", "getJiraDescriptions", "select", "formatBranchChoices", "prInfo", "removeReleaseWorktreeIfPresent", "releaseBranch", "getCurrentWorktrees", "projectRoot", "repoName", "getProjectRoot", "getRepoName", "worktreeDir", "WORKTREES_DIR_SUFFIX", "removeWorktrees", "mergeReleasePR", "args", "releaseType", "mergeTarget", "releasePr", "resolveRcPRNumber", "selectedVersion", "existingOpen", "rcNumber", "created", "ensureRcPRMerged", "dispatchDeployWorkflow", "syncMainIntoDev", "deliverJiraReleaseSafely", "jiraConfig", "loadJiraConfigOptional", "versionName", "deliverJiraRelease", "ghReleaseDeliver", "confirmedCommand", "releasePrTitle", "answer", "confirm", "process", "structuredContent", "textContent", "ghReleaseDeliverMcpTool", "defineMcpTool", "z", "select", "z", "$", "ghReleaseDeployAll", "args", "version", "env", "skipTerraform", "commandEcho", "selectedReleaseBranch", "releasePRsInfo", "getReleasePRsWithInfo", "branches", "pr", "releaseTypes", "detectReleaseType", "descriptions", "getJiraDescriptions", "select", "formatBranchChoices", "selectedVersion", "environments", "getInfraKitConfig", "selectedEnv", "OperationError", "shouldSkipTerraform", "$", "logger", "structuredContent", "textContent", "error", "ghReleaseDeployAllMcpTool", "defineMcpTool", "z", "checkbox", "select", "fs", "resolve", "yaml", "z", "$", "ghReleaseDeploySelected", "args", "version", "env", "services", "skipTerraform", "commandEcho", "selectedReleaseBranch", "releasePRsInfo", "getReleasePRsWithInfo", "branches", "pr", "releaseTypes", "detectReleaseType", "descriptions", "getJiraDescriptions", "select", "formatBranchChoices", "selectedVersion", "environments", "getInfraKitConfig", "selectedEnv", "OperationError", "availableServices", "parseServicesFromWorkflow", "selectedServices", "checkbox", "svc", "invalidServices", "shouldSkipTerraform", "$", "serviceFlags", "logger", "structuredContent", "textContent", "error", "projectRoot", "getProjectRoot", "workflowPath", "resolve", "content", "fs", "inputs", "yaml", "key", "value", "ghReleaseDeploySelectedMcpTool", "defineMcpTool", "z", "z", "ghReleaseList", "releases", "getReleasePRsWithInfo", "pr", "detectReleaseType", "jiraDescriptions", "getJiraDescriptions", "maxVersionLength", "r", "formattedLines", "release", "label", "formatVersionLabel", "description", "logger", "structuredContent", "textContent", "ghReleaseListMcpTool", "defineMcpTool", "z", "confirm", "select", "process", "z", "question", "VERSION_PROMPT_HINT", "trySuggestNext", "known", "type", "computeNextVersion", "err", "NoPriorVersionsError", "resolveOrExit", "entries", "resolveReleaseEntries", "OperationError", "promptForReleasesInteractive", "ensureKnown", "commandEcho", "running", "addAnother", "ordinal", "select", "suggestion", "defaultHint", "versionAnswer", "question", "versionInput", "logger", "process", "resolved", "parseVersion", "description", "confirm", "formatReleaseSummary", "entry", "parts", "echoReleases", "spec", "collectEntries", "inputReleases", "hasNextToken", "interactive", "confirmReleases", "confirmedCommand", "summary", "answer", "executeOne", "args", "jiraConfig", "prepareGitForRelease", "result", "createSingleRelease", "error", "logFinalSummary", "total", "successCount", "failureCount", "releaseCreate", "loadJiraConfig", "loadExistingVersions", "created", "failed", "failure", "structuredContent", "r", "textContent", "releaseCreateMcpTool", "defineMcpTool", "z", "confirm", "select", "process", "z", "question", "buildJiraVersionUrl", "jiraConfig", "version", "buildPRBody", "jiraVersionUrl", "description", "pickReleaseBranch", "releasePRsInfo", "getReleasePRsWithInfo", "branches", "pr", "types", "detectReleaseType", "descriptions", "getJiraDescriptions", "branch", "select", "formatBranchChoices", "verifyReleasePRExists", "selectedBranch", "prInfo", "OperationError", "promptDescription", "current", "hint", "trimmed", "question", "releaseDescEdit", "args", "versionArg", "descriptionArg", "confirmedCommand", "commandEcho", "loadJiraConfig", "selectedVersion", "versionName", "jiraVersion", "findVersionByName", "previousDescription", "newDescription", "logger", "structuredContent", "textContent", "answer", "confirm", "process", "updateJiraVersion", "body", "updateReleasePRBody", "releaseDescEditMcpTool", "defineMcpTool", "z", "z", "package_default", "version", "cliVersion", "package_default", "logger", "structuredContent", "textContent", "versionMcpTool", "defineMcpTool", "z", "checkbox", "confirm", "select", "process", "z", "$", "fs", "path", "addFoldersToCursorWorkspace", "args", "workspacePath", "folderPaths", "workspaceDir", "raw", "error", "parsed", "existingFolders", "existingAbsolutePaths", "entry", "added", "skipped", "folderPath", "absolutePath", "relativePath", "fs", "path", "fs", "path", "removeFoldersFromCursorWorkspace", "args", "workspacePath", "folderPaths", "workspaceDir", "raw", "error", "parsed", "existingFolders", "targetAbsolutePaths", "folderPath", "removedAbsolutePaths", "filteredFolders", "entry", "entryAbsolutePath", "removed", "notFound", "absolutePath", "reconcileCursorWorkspaceFolders", "args", "workspacePath", "worktreeDir", "currentBranches", "workspaceDir", "path", "releaseRoot", "raw", "fs", "existingFolders", "desiredAbsolutePaths", "branch", "danglingFolderPaths", "entry", "entryAbsolutePath", "removed", "removeFoldersFromCursorWorkspace", "desiredFolderPaths", "added", "addFoldersToCursorWorkspace", "path", "resolveCursorWorkspacePath", "configValue", "projectRoot", "FEATURE_DIR", "RELEASE_DIR", "RELEASE_BRANCH_PREFIX", "CURSOR_MODES", "worktreesAdd", "options", "confirmedCommand", "all", "versions", "cursor", "githubDesktop", "cmux", "commandEcho", "currentWorktrees", "getCurrentWorktrees", "projectRoot", "getProjectRoot", "worktreeDir", "WORKTREES_DIR_SUFFIX", "ensureWorktreeDirectory", "selectedReleaseBranches", "v", "releasePRsInfo", "getReleasePRsWithInfo", "releasePRsList", "pr", "logger", "textContent", "releaseTypes", "detectReleaseType", "descriptions", "getJiraDescriptions", "checkbox", "formatBranchChoices", "branch", "answer", "confirm", "process", "config", "getInfraKitConfig", "cursorConfig", "cursorMode", "select", "openInGithubDesktop", "openInCmux", "branchesToCreate", "categorizeWorktrees", "createdWorktrees", "createWorktrees", "logResults", "workspacePath", "resolveCursorWorkspacePath", "folderPaths", "added", "skipped", "addFoldersToCursorWorkspace", "skippedSuffix", "$", "repoName", "getRepoName", "title", "buildCmuxWorkspaceTitle", "openCmuxWorkspaceWithLayout", "structuredContent", "error", "OperationError", "args", "currentBranchNames", "branches", "results", "worktreePath", "created", "index", "result", "err", "worktreesAddMcpTool", "defineMcpTool", "z", "z", "worktreesList", "currentWorktrees", "getCurrentWorktrees", "logger", "textContent", "releasePRsInfo", "jiraDescriptions", "getReleasePRsWithInfo", "getJiraDescriptions", "releaseTypes", "pr", "detectReleaseType", "worktrees", "branch", "version", "type", "description", "maxVersionLength", "w", "formattedLines", "worktree", "label", "formatVersionLabel", "structuredContent", "worktreesListMcpTool", "defineMcpTool", "z", "z", "$", "worktreesOpen", "commandEcho", "projectRoot", "getProjectRoot", "worktreeDir", "WORKTREES_DIR_SUFFIX", "currentBranches", "getCurrentWorktrees", "cursorOutcome", "openCursor", "cmuxOutcome", "openCmux", "result", "logResults", "textContent", "error", "logger", "OperationError", "args", "config", "getInfraKitConfig", "cursorConfig", "workspacePath", "resolveCursorWorkspacePath", "added", "removed", "reconcileCursorWorkspaceFolders", "$", "repoName", "getRepoName", "existingTitles", "listCmuxWorkspaceTitles", "opened", "skipped", "branch", "title", "buildCmuxWorkspaceTitle", "openCmuxWorkspaceWithLayout", "context", "worktreesOpenMcpTool", "defineMcpTool", "z", "checkbox", "confirm", "process", "z", "worktreesRemove", "options", "confirmedCommand", "all", "versions", "commandEcho", "currentWorktrees", "getCurrentWorktrees", "logger", "textContent", "projectRoot", "getProjectRoot", "worktreeDir", "WORKTREES_DIR_SUFFIX", "selectedReleaseBranches", "v", "descriptions", "prInfo", "getJiraDescriptions", "getReleasePRsWithInfo", "releaseTypes", "pr", "detectReleaseType", "checkbox", "formatBranchChoices", "allSelected", "branch", "answer", "confirm", "process", "repoName", "getRepoName", "removedWorktrees", "removeWorktrees", "syncCursorWorkspaceOnRemove", "logResults", "structuredContent", "error", "OperationError", "args", "config", "getInfraKitConfig", "cursorConfig", "workspacePath", "resolveCursorWorkspacePath", "folderPaths", "removed", "removeFoldersFromCursorWorkspace", "worktreesRemoveMcpTool", "defineMcpTool", "z", "confirm", "process", "z", "$", "RELEASE_BRANCH_PREFIX", "worktreesSync", "options", "confirmedCommand", "commandEcho", "currentWorktrees", "getCurrentWorktrees", "projectRoot", "getProjectRoot", "worktreeDir", "WORKTREES_DIR_SUFFIX", "releasePRsList", "getReleasePRs", "answer", "confirm", "logger", "process", "branchesToRemove", "categorizeWorktrees", "repoName", "getRepoName", "removedWorktrees", "removeWorktrees", "syncCursorWorkspaceOnRemove", "logResults", "structuredContent", "textContent", "error", "OperationError", "args", "branch", "branches", "removed", "worktreePath", "title", "buildCmuxWorkspaceTitle", "closeCmuxWorkspaceByTitle", "$", "err", "config", "getInfraKitConfig", "cursorConfig", "workspacePath", "resolveCursorWorkspacePath", "folderPaths", "removedEntries", "removeFoldersFromCursorWorkspace", "worktreesSyncMcpTool", "defineMcpTool", "z", "createToolHandler", "args", "params", "toolName", "handler", "logger", "payload", "error", "tools", "envStatusMcpTool", "envListMcpTool", "envLoadMcpTool", "envClearMcpTool", "ghMergeDevMcpTool", "releaseCreateMcpTool", "releaseDescEditMcpTool", "ghReleaseDeliverMcpTool", "ghReleaseDeployAllMcpTool", "ghReleaseDeploySelectedMcpTool", "ghReleaseListMcpTool", "versionMcpTool", "worktreesAddMcpTool", "worktreesListMcpTool", "worktreesOpenMcpTool", "worktreesRemoveMcpTool", "worktreesSyncMcpTool", "initializeTools", "server", "tool", "createToolHandler", "createMcpServer", "server", "McpServer", "initializePrompts", "initializeResources", "initializeTools", "logger", "initLoggerMcp", "startServer", "server", "createMcpServer", "error", "process", "transport", "StdioServerTransport", "setupErrorHandlers"]
3
+ "sources": ["../src/entry/mcp.ts", "../src/lib/error-handlers/index.ts", "../src/lib/logger/index.ts", "../src/mcp/server.ts", "../src/mcp/prompts/index.ts", "../src/mcp/resources/index.ts", "../src/commands/audit/audit.ts", "../src/lib/git-utils/git-utils.ts", "../src/lib/release-id/release-id.ts", "../src/lib/package-config/package-config.ts", "../src/lib/package-config/package-config-schema.ts", "../src/lib/package-validator/fs-utils.ts", "../src/lib/package-validator/package-validator.ts", "../src/lib/package-validator/loader/config-loader.ts", "../src/lib/package-validator/loader/package-discovery.ts", "../src/lib/package-validator/checks/config-check.ts", "../src/lib/package-validator/checks/files-check.ts", "../src/lib/package-validator/checks/scripts-check.ts", "../src/lib/package-validator/checks/turbo-check.ts", "../src/types.ts", "../src/commands/env-clear/env-clear.ts", "../src/lib/constants/constants.ts", "../src/commands/env-list/env-list.ts", "../src/lib/infra-kit-config/infra-kit-config.ts", "../src/integrations/doppler/doppler-project.ts", "../src/commands/env-load/env-load.ts", "../src/integrations/doppler/doppler-cli-auth.ts", "../src/lib/command-echo/command-echo.ts", "../src/commands/env-status/env-status.ts", "../src/commands/gh-merge-dev/gh-merge-dev.ts", "../src/integrations/gh/gh-cli-auth/gh-cli-auth.ts", "../src/integrations/gh/gh-release-prs/gh-release-prs.ts", "../src/lib/release-utils/release-utils.ts", "../src/integrations/jira/api.ts", "../src/lib/errors/operation-error.ts", "../src/commands/gh-release-deliver/gh-release-deliver.ts", "../src/lib/errors/format-zx-error.ts", "../src/lib/worktrees/remove-worktrees.ts", "../src/integrations/cmux/close-workspace-by-title.ts", "../src/integrations/cmux/list-workspace-titles.ts", "../src/integrations/cmux/open-workspace-with-layout.ts", "../src/integrations/cmux/workspace-title.ts", "../src/commands/gh-release-deploy-all/gh-release-deploy-all.ts", "../src/commands/gh-release-deploy-selected/gh-release-deploy-selected.ts", "../src/commands/gh-release-list/gh-release-list.ts", "../src/commands/release-create/release-create.ts", "../src/lib/version-utils/load-existing-versions.ts", "../src/lib/version-utils/version-utils.ts", "../src/lib/version-utils/next-version.ts", "../src/commands/release-desc-edit/release-desc-edit.ts", "../src/commands/version/version.ts", "../package.json", "../src/commands/worktrees-add/worktrees-add.ts", "../src/integrations/cursor/add-folders-to-workspace.ts", "../src/integrations/cursor/reconcile-workspace-folders.ts", "../src/integrations/cursor/remove-folders-from-workspace.ts", "../src/integrations/cursor/resolve-workspace-path.ts", "../src/commands/worktrees-list/worktrees-list.ts", "../src/commands/worktrees-open/worktrees-open.ts", "../src/commands/worktrees-remove/worktrees-remove.ts", "../src/commands/worktrees-sync/worktrees-sync.ts", "../src/lib/tool-handler/tool-handler.ts", "../src/mcp/tools/index.ts"],
4
+ "sourcesContent": ["import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js'\nimport process from 'node:process'\n\nimport { setupErrorHandlers } from 'src/lib/error-handlers'\nimport { initLoggerMcp } from 'src/lib/logger'\n\nimport { createMcpServer } from '../mcp/server'\n\nconst logger = initLoggerMcp()\n\nconst startServer = async () => {\n let server\n\n try {\n server = await createMcpServer()\n\n logger.info('MCP Server instance created')\n } catch (error) {\n logger.error({ err: error, msg: 'Failed to create MCP server' })\n logger.error(`Fatal error during server creation.`)\n\n process.exit(1)\n }\n\n try {\n const transport = new StdioServerTransport()\n\n await server.connect(transport)\n\n logger.info({ msg: 'Server connected to transport. Ready.' })\n } catch (error) {\n logger.error({ err: error, msg: 'Failed to initialize server' })\n logger.error(`Fatal error during server transport init.`)\n\n process.exit(1)\n }\n}\n\n// Setup error handlers\nsetupErrorHandlers(logger)\n\n// Start the server\nstartServer()\n", "import process from 'node:process'\nimport type { Logger } from 'pino'\n\nimport { LOG_FILE_PATH } from '../logger/index'\n\n/**\n * Setup error handlers for the application\n * @param logger - The logger instance\n *\n * ONLY FOR SERVER!\n */\nexport const setupErrorHandlers = (logger: Logger) => {\n process.on('SIGINT', () => {\n logger.info({ msg: 'Received SIGINT. Shutting down...' })\n process.exit(0)\n })\n\n process.on('SIGTERM', () => {\n logger.info({ msg: 'Received SIGTERM. Shutting down...' })\n process.exit(0)\n })\n\n process.on('uncaughtException', (error) => {\n logger.fatal({ err: error, msg: 'Uncaught Exception' })\n logger.error(`Uncaught Exception! Check ${LOG_FILE_PATH}. Shutting down...`)\n logger.flush()\n process.exit(1)\n })\n\n process.on('unhandledRejection', (reason, promise) => {\n logger.fatal({ reason, promise, msg: 'Unhandled Rejection' })\n logger.error(`Unhandled Rejection! Check ${LOG_FILE_PATH}. Shutting down...`)\n logger.flush()\n process.exit(1)\n })\n}\n", "import process from 'node:process'\nimport pino from 'pino'\nimport pretty from 'pino-pretty'\n\n// eslint-disable-next-line sonarjs/publicly-writable-directories\nexport const LOG_FILE_PATH = '/tmp/mcp-infra-kit.log'\n\nexport const initLoggerMcp = () => {\n const logLevel = process.argv.includes('--debug') ? 'debug' : 'info'\n\n const logger = pino({ level: logLevel }, pino.destination({ dest: LOG_FILE_PATH }))\n\n logger.info(`Logger initialized with level: ${logLevel}. Logging to: ${LOG_FILE_PATH}`)\n\n return logger\n}\n\nexport const initLoggerCLI = () => {\n const logLevel = process.argv.includes('--debug') ? 'debug' : 'info'\n\n const ignoreFields = ['time', 'pid', 'hostname']\n\n if (logLevel === 'debug') {\n ignoreFields.push('level')\n }\n\n const logger = pino(\n { level: logLevel },\n pretty({\n destination: 2,\n ignore: ignoreFields.join(','),\n colorize: true,\n }),\n )\n\n return logger\n}\n\n// Singleton logger instance for CLI usage\nexport const logger = initLoggerCLI()\n", "import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\n\nimport { initializePrompts } from './prompts'\nimport { initializeResources } from './resources'\nimport { initializeTools } from './tools'\n\nexport async function createMcpServer() {\n const server = new McpServer(\n {\n name: 'infra-kit',\n version: '1.0.0',\n },\n {\n capabilities: {\n resources: {},\n tools: {},\n prompts: {},\n },\n },\n )\n\n await initializePrompts(server)\n await initializeResources(server)\n await initializeTools(server)\n\n return server\n}\n", "import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\n\nexport const initializePrompts = async (_server: McpServer) => {}\n", "import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\n\nexport const initializeResources = async (_server: McpServer) => {}\n", "import path from 'node:path'\nimport process from 'node:process'\nimport { z } from 'zod'\n\nimport { getProjectRoot } from 'src/lib/git-utils'\nimport { logger } from 'src/lib/logger'\nimport { ROOT_DEFAULT_RULES } from 'src/lib/package-config'\nimport type { ResolvedPackageRules } from 'src/lib/package-config'\nimport { discoverPackages, pathExists, validatePackage } from 'src/lib/package-validator'\nimport type { PackageValidationResult } from 'src/lib/package-validator'\nimport { defineMcpTool, textContent } from 'src/types'\n\n// TODO [DO]: extract `audit` into its own standalone CLI tool, decoupled from infra-kit.\n\ninterface AuditOptions {\n /** Audit every non-vendor workspace package instead of just the current one. */\n all?: boolean\n /** Audit the monorepo root (turbo pipeline + root commands) instead of a package. */\n root?: boolean\n /** Directory to resolve the current package from. Defaults to `process.cwd()`. */\n cwd?: string\n}\n\n/** A directory to audit plus the under-the-hood defaults that apply to it. */\ninterface AuditTarget {\n dir: string\n baseline?: Readonly<ResolvedPackageRules>\n}\n\n/**\n * Walk upward from `start` to the nearest directory containing a package.json.\n * Used so `infra-kit audit` (no `--all`) targets the package whose package.json\n * script invoked it, regardless of the exact working directory.\n *\n * @example\n * await findPackageRoot('/repo/packages/serverless-config/src')\n * // => '/repo/packages/serverless-config'\n */\nconst findPackageRoot = async (start: string): Promise<string> => {\n let current = path.resolve(start)\n\n while (current !== path.dirname(current)) {\n if (await pathExists(path.join(current, 'package.json'))) {\n return current\n }\n\n current = path.dirname(current)\n }\n\n if (await pathExists(path.join(current, 'package.json'))) {\n return current\n }\n\n throw new Error(`No package.json found in or above ${start}`)\n}\n\n/**\n * Resolve which directories to audit, and the baseline defaults each uses:\n * `root` \u2192 the monorepo root with {@link ROOT_DEFAULT_RULES}; `all` \u2192 every\n * discovered non-vendor package; otherwise the package walked up from cwd.\n */\nconst resolveTargets = async (options: AuditOptions): Promise<AuditTarget[]> => {\n if (options.root) {\n return [{ dir: await getProjectRoot(), baseline: ROOT_DEFAULT_RULES }]\n }\n\n if (options.all) {\n const dirs = await discoverPackages(await getProjectRoot())\n\n return dirs.map((dir) => {\n return { dir }\n })\n }\n\n return [{ dir: await findPackageRoot(options.cwd ?? process.cwd()) }]\n}\n\n/**\n * Print a package's audit result as doctor-style `[PASS]`/`[FAIL]` lines.\n */\nconst logResult = (result: PackageValidationResult): void => {\n const header = result.passed ? 'PASS' : 'FAIL'\n\n logger.info(`\\n${result.packageName} \u2014 ${header}`)\n\n for (const check of result.checks) {\n const icon = check.status === 'pass' ? '[PASS]' : '[FAIL]'\n\n logger.info(` ${icon} ${check.name}: ${check.message}`)\n }\n}\n\n/**\n * Audit the monorepo root (`root`), every non-vendor workspace package (`all`),\n * or the package resolved by walking up from the working directory (default \u2014\n * the shape used by a package's `\"check\": \"infra-kit audit\"` script). The\n * returned `structuredContent.allPassed` lets the CLI set a non-zero exit code so\n * the audit fails CI; this function never calls `process.exit` so the MCP tool\n * can reuse it.\n *\n * @example\n * // CLI inside packages/serverless-config: `infra-kit audit`\n * await audit() // audits the current package\n * await audit({ all: true }) // audits every non-vendor workspace package\n * await audit({ root: true }) // audits the monorepo root (turbo + root commands)\n */\nexport const audit = async (options: AuditOptions = {}) => {\n const targets = await resolveTargets(options)\n\n const results: PackageValidationResult[] = []\n\n for (const target of targets) {\n results.push(await validatePackage(target.dir, target.baseline))\n }\n\n for (const result of results) {\n logResult(result)\n }\n\n const allPassed = results.every((result) => {\n return result.passed\n })\n\n logger.info(`\\n${allPassed ? '\u2705 All valid' : '\u274C Audit failed'} (${results.length} checked)`)\n\n const structuredContent = {\n allPassed,\n packages: results.map((result) => {\n return {\n name: result.packageName,\n passed: result.passed,\n checks: result.checks,\n }\n }),\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\nconst auditInputSchema = {\n all: z.boolean().optional().describe('Audit every non-vendor workspace package'),\n root: z.boolean().optional().describe('Audit the monorepo root (turbo pipeline + root commands)'),\n}\n\nconst auditOutputSchema = {\n allPassed: z.boolean().describe('Whether every audited package passed all checks'),\n packages: z\n .array(\n z.object({\n name: z.string(),\n passed: z.boolean(),\n checks: z.array(\n z.object({\n name: z.string(),\n status: z.enum(['pass', 'fail']),\n message: z.string(),\n }),\n ),\n }),\n )\n .describe('Per-package check results'),\n}\n\n// MCP Tool Registration\nexport const auditMcpTool = defineMcpTool({\n name: 'audit',\n description:\n 'Audit packages against infra-kit.config.ts rules (config present and valid, required scripts, required files, and turbo tasks for the root). Defaults to the current package; all=true audits every non-vendor workspace package; root=true audits the monorepo root.',\n inputSchema: auditInputSchema,\n outputSchema: auditOutputSchema,\n handler: (params) => {\n return audit({ all: params.all, root: params.root })\n },\n})\n", "import path from 'node:path'\nimport { $ } from 'zx'\n\nimport { isReleaseBranch } from 'src/lib/release-id'\n\n/**\n * Get current git worktrees\n *\n * @returns [release/v1.18.22, release/v1.18.23, release/v1.18.24] or [feature/mobile-app, feature/explore-page, feature/login-page]\n */\nexport const getCurrentWorktrees = async (type: 'release' | 'feature'): Promise<string[]> => {\n const worktreesOutput = await $`git worktree list`\n\n const worktreeLines = worktreesOutput.stdout.split('\\n').filter(Boolean)\n\n const worktreePredicateMap = {\n release: releaseWorktreePredicate,\n feature: featureWorktreePredicate,\n }\n\n return worktreeLines.map(worktreePredicateMap[type]).filter((branch) => {\n return branch !== null\n })\n}\n\n/**\n * Extract the branch name from a `git worktree list` output line.\n *\n * `git worktree list` formats each line as:\n * <path> <hash> [<branch>]\n *\n * Reads the branch from the trailing `[branch]` token so it works for the\n * main checkout too (whose path does not encode the branch name).\n */\nconst parseWorktreeBranch = (line: string): string | null => {\n const trimmed = line.trimEnd()\n\n if (!trimmed.endsWith(']')) return null\n\n const open = trimmed.lastIndexOf('[')\n\n if (open === -1) return null\n\n const branch = trimmed.slice(open + 1, -1)\n\n return branch.length > 0 ? branch : null\n}\n\n/**\n * Extract a release branch name from a `git worktree list` output line.\n *\n * Returns `null` for lines that are not release worktrees.\n *\n * @example\n * releaseWorktreePredicate('/path/to/release/v1.18.22 abc1234 [release/v1.18.22]')\n * // => 'release/v1.18.22'\n *\n * @example\n * releaseWorktreePredicate('/path/to/feature/login abc1234 [feature/login]')\n * // => null\n */\nconst releaseWorktreePredicate = (line: string): string | null => {\n const branch = parseWorktreeBranch(line)\n\n return isReleaseBranch(branch) ? branch : null\n}\n\n/**\n * Extract a feature branch name from a `git worktree list` output line.\n *\n * Returns `null` for lines that are not feature worktrees.\n *\n * @example\n * featureWorktreePredicate('/path/to/feature/login-page abc1234 [feature/login-page]')\n * // => 'feature/login-page'\n *\n * @example\n * featureWorktreePredicate('/path/to/release/v1.18.22 abc1234 [release/v1.18.22]')\n * // => null\n */\nconst featureWorktreePredicate = (line: string): string | null => {\n const branch = parseWorktreeBranch(line)\n\n return branch?.startsWith('feature/') ? branch : null\n}\n\n/**\n * Get the current project root directory\n */\nexport const getProjectRoot = async (): Promise<string> => {\n const result = await $`git rev-parse --show-toplevel`\n\n return result.stdout.trim()\n}\n\n/**\n * Get the current repository name (basename of the project root)\n */\nexport const getRepoName = async (): Promise<string> => {\n const projectRoot = await getProjectRoot()\n\n return path.basename(projectRoot)\n}\n", "/**\n * A release identity is either a semantic version or a free-form kebab-case\n * name. `raw` is the canonical token for the id: the no-`v` semver string for\n * versions (e.g. `1.2.3`) and the name itself for named releases.\n */\nexport type ReleaseId =\n | { kind: 'version'; semver: { major: number; minor: number; patch: number }; raw: string }\n | { kind: 'name'; name: string; raw: string }\n\n/** Matches a bare or `v`-prefixed semver token, e.g. `1.2.3` or `v1.2.3`. */\nconst VERSION_RE = /^v?(\\d+)\\.(\\d+)\\.(\\d+)$/\n\n/** Matches the semver core after the `v` in a `release/v\u2026` branch. */\nconst BRANCH_SEMVER_RE = /^(\\d+)\\.(\\d+)\\.(\\d+)$/\n\n/** Kebab-case: lowercase alphanumeric segments joined by single hyphens. */\nconst KEBAB_RE = /^[a-z0-9]+(?:-[a-z0-9]+)*$/\n\nconst RELEASE_BRANCH_PREFIX = 'release/'\nconst VERSION_BRANCH_PREFIX = 'release/v'\nconst NAME_BRANCH_PREFIX = 'release/n/'\nconst REFS_HEADS_PREFIX = 'refs/heads/'\n\nconst NEXT_TOKEN = 'next'\nconst MAX_NAME_LENGTH = 50\n\n/**\n * Names that would collide with branch/release semantics or read as a special\n * token. Banned regardless of kebab-case validity.\n */\nconst RESERVED_NAMES: ReadonlySet<string> = new Set(['dev', 'main', 'next', 'hotfix', 'regular', 'release'])\n\n/** Thrown by {@link validateName} when a release name is not acceptable. */\nexport class InvalidReleaseNameError extends Error {\n constructor(message: string) {\n super(message)\n this.name = 'InvalidReleaseNameError'\n }\n}\n\n/** Thrown by {@link parseReleaseRef} when a release ref cannot be parsed. */\nexport class InvalidReleaseRefError extends Error {\n constructor(message: string) {\n super(message)\n this.name = 'InvalidReleaseRefError'\n }\n}\n\nconst stripRefsHeads = (input: string): string => {\n return input.startsWith(REFS_HEADS_PREFIX) ? input.slice(REFS_HEADS_PREFIX.length) : input\n}\n\nconst makeVersion = (major: number, minor: number, patch: number): ReleaseId => {\n return {\n kind: 'version',\n semver: { major, minor, patch },\n raw: `${major}.${minor}.${patch}`,\n }\n}\n\n/**\n * Validate a release name. Throws {@link InvalidReleaseNameError} with a\n * specific message unless the name is kebab-case, at most 50 characters, and\n * not a reserved word. Semver-looking tokens (e.g. `1.2.3`) are already\n * excluded by the kebab-case rule since they contain dots.\n */\nexport const validateName = (name: string): void => {\n if (name.length === 0) {\n throw new InvalidReleaseNameError('Release name is empty. Provide a kebab-case name like \"checkout-redesign\".')\n }\n\n if (name.length > MAX_NAME_LENGTH) {\n throw new InvalidReleaseNameError(\n `Release name \"${name}\" is ${name.length} characters; the maximum is ${MAX_NAME_LENGTH}.`,\n )\n }\n\n if (!KEBAB_RE.test(name)) {\n throw new InvalidReleaseNameError(\n `Release name \"${name}\" is not kebab-case. Use lowercase letters, digits, and single hyphens, e.g. \"checkout-redesign\".`,\n )\n }\n\n if (RESERVED_NAMES.has(name)) {\n throw new InvalidReleaseNameError(\n `Release name \"${name}\" is reserved. Reserved names: ${[...RESERVED_NAMES].join(', ')}.`,\n )\n }\n}\n\n/**\n * Lenient parse of a git branch name into a {@link ReleaseId}. Tolerates a\n * leading `refs/heads/`. Returns `null` for anything that is not a valid\n * `release/v<semver>` or `release/n/<name>` branch. Never throws.\n */\nexport const parseBranchName = (branch: string): ReleaseId | null => {\n const stripped = stripRefsHeads(branch.trim())\n\n if (stripped.startsWith(VERSION_BRANCH_PREFIX)) {\n const semverPart = stripped.slice(VERSION_BRANCH_PREFIX.length)\n const match = BRANCH_SEMVER_RE.exec(semverPart)\n\n if (!match) return null\n\n return makeVersion(Number(match[1]), Number(match[2]), Number(match[3]))\n }\n\n if (stripped.startsWith(NAME_BRANCH_PREFIX)) {\n const namePart = stripped.slice(NAME_BRANCH_PREFIX.length)\n\n try {\n validateName(namePart)\n } catch {\n return null\n }\n\n return { kind: 'name', name: namePart, raw: namePart }\n }\n\n return null\n}\n\n/**\n * Strict parse of a release ref into a {@link ReleaseId}. Throws\n * {@link InvalidReleaseRefError} on invalid input. Precedence (order matters):\n * 1. `release/\u2026` (or `refs/heads/release/\u2026`) \u2192 delegate to parseBranchName.\n * 2. semver token (`1.2.3` / `v1.2.3`) \u2192 version.\n * 3. `next` \u2192 throws; callers must resolve `next` to a concrete version\n * via computeNextVersion before calling this.\n * 4. otherwise \u2192 validateName, returning a named release.\n */\nexport const parseReleaseRef = (input: string): ReleaseId => {\n const trimmed = input.trim()\n const branchCandidate = stripRefsHeads(trimmed)\n\n if (branchCandidate.startsWith(RELEASE_BRANCH_PREFIX)) {\n const parsed = parseBranchName(trimmed)\n\n if (!parsed) {\n throw new InvalidReleaseRefError(\n `\"${input}\" looks like a release branch but is not a valid release/v<semver> or release/n/<name> ref.`,\n )\n }\n\n return parsed\n }\n\n const versionMatch = VERSION_RE.exec(trimmed)\n\n if (versionMatch) {\n return makeVersion(Number(versionMatch[1]), Number(versionMatch[2]), Number(versionMatch[3]))\n }\n\n if (trimmed.toLowerCase() === NEXT_TOKEN) {\n throw new InvalidReleaseRefError(\n 'The \"next\" token must be resolved to a concrete version (via computeNextVersion) before parsing a release ref.',\n )\n }\n\n try {\n validateName(trimmed)\n } catch (err) {\n const reason = err instanceof Error ? err.message : String(err)\n\n throw new InvalidReleaseRefError(`Cannot parse \"${input}\" as a release ref: ${reason}`)\n }\n\n return { kind: 'name', name: trimmed, raw: trimmed }\n}\n\n/** Render the branch name for a release id: `release/v1.2.3` | `release/n/<name>`. */\nexport const formatBranchName = (id: ReleaseId): string => {\n if (id.kind === 'version') return `${VERSION_BRANCH_PREFIX}${id.raw}`\n\n return `${NAME_BRANCH_PREFIX}${id.name}`\n}\n\n/**\n * Render a PR title: `Release v1.2.3` / `Hotfix v1.2.3` for versions,\n * `Release <name>` / `Hotfix <name>` for names.\n */\nexport const formatPrTitle = (id: ReleaseId, type: 'regular' | 'hotfix'): string => {\n const prefix = type === 'hotfix' ? 'Hotfix' : 'Release'\n\n if (id.kind === 'version') return `${prefix} v${id.raw}`\n\n return `${prefix} ${id.name}`\n}\n\n/** Render a release-candidate PR title: `Release v1.2.3 (RC)` | `Release <name> (RC)`. */\nexport const formatRcTitle = (id: ReleaseId): string => {\n if (id.kind === 'version') return `Release v${id.raw} (RC)`\n\n return `Release ${id.name} (RC)`\n}\n\n/** Render the Jira fix-version name: `v1.2.3` | `<name>`. */\nexport const formatJiraName = (id: ReleaseId): string => {\n if (id.kind === 'version') return `v${id.raw}`\n\n return id.name\n}\n\n/** Render a short human display label: `1.2.3` | `<name>`. */\nexport const displayLabel = (id: ReleaseId): string => {\n return id.raw\n}\n\n/** True iff `branch` is a valid release branch under either scheme. */\nexport const isReleaseBranch = (branch: string | null | undefined): boolean => {\n if (branch === null || branch === undefined) return false\n\n return parseBranchName(branch) !== null\n}\n\nconst toTime = (value: string | Date | undefined): number | null => {\n if (value === undefined) return null\n\n const time = value instanceof Date ? value.getTime() : new Date(value).getTime()\n\n return Number.isNaN(time) ? null : time\n}\n\n/**\n * Comparator for {@link ReleaseId} values (locked ordering):\n * - All versions sort before all names.\n * - Versions: semver ascending (major, then minor, then patch; numeric).\n * - Names: by date ascending when both dates are provided, otherwise\n * lexicographic by name. The result is stable and deterministic.\n */\nexport const compareReleaseIds = (\n a: ReleaseId,\n b: ReleaseId,\n dates?: { a?: string | Date; b?: string | Date },\n): number => {\n if (a.kind === 'version' && b.kind === 'version') {\n if (a.semver.major !== b.semver.major) return a.semver.major - b.semver.major\n if (a.semver.minor !== b.semver.minor) return a.semver.minor - b.semver.minor\n\n return a.semver.patch - b.semver.patch\n }\n\n if (a.kind === 'version') return -1\n if (b.kind === 'version') return 1\n\n const timeA = toTime(dates?.a)\n const timeB = toTime(dates?.b)\n\n if (timeA !== null && timeB !== null && timeA !== timeB) {\n return timeA - timeB\n }\n\n if (a.name < b.name) return -1\n if (a.name > b.name) return 1\n\n return 0\n}\n", "/**\n * Validation rules for a single workspace package, declared in its\n * `infra-kit.config.js`. Every field is optional: a key left unset falls back to\n * the active baseline ({@link DEFAULT_RULES} for packages, {@link ROOT_DEFAULT_RULES}\n * for the monorepo root), and a key set replaces that default wholesale (per-key,\n * no array concatenation) so a package can opt out with an explicit empty array.\n *\n * Most packages need none of these \u2014 the standard rules live in the baseline, so\n * a typical config is just `defineConfig(() => ({}))`.\n *\n * @example\n * // infra-kit.config.js\n * import { defineConfig } from 'infra-kit'\n *\n * export default defineConfig(() => ({}))\n */\nexport interface InfraKitPackageConfig {\n /** Scripts that must be present in the package's package.json `scripts` map. */\n requiredScripts?: string[]\n /** Files (relative to the package root) that must exist on disk. */\n requiredFiles?: string[]\n /** Turborepo expectations \u2014 only meaningful where a turbo.json lives (the root). */\n turbo?: {\n /** Tasks that must be defined in turbo.json `tasks`. */\n requiredTasks?: string[]\n }\n}\n\n/**\n * Accepted shapes for a package config's default export \u2014 mirrors Vite's\n * `defineConfig` input: a plain object, a sync factory, or an async factory.\n */\nexport type InfraKitPackageConfigInput =\n | InfraKitPackageConfig\n | (() => InfraKitPackageConfig)\n | (() => Promise<InfraKitPackageConfig>)\n\n/**\n * Identity helper that gives `infra-kit.config.js` authors full type inference\n * and editor autocomplete without changing the value \u2014 exactly like Vite's\n * `defineConfig`. Resolution of the factory form happens in the loader, not here.\n *\n * @example\n * export default defineConfig(() => ({}))\n *\n * @example\n * export default defineConfig(() => ({ requiredScripts: [] }))\n */\nexport const defineConfig = (config: InfraKitPackageConfigInput): InfraKitPackageConfigInput => {\n return config\n}\n\n/** Fully-resolved rules with every defaultable field present. */\nexport interface ResolvedPackageRules {\n requiredScripts: string[]\n requiredFiles: string[]\n turboTasks: string[]\n}\n\n/**\n * Baseline rules for a standard TypeScript workspace package, applied to any key\n * a package leaves unset. These are the \"under the hood\" defaults so a conforming\n * package's config can stay empty; non-standard packages override the relevant key.\n */\nexport const DEFAULT_RULES: Readonly<ResolvedPackageRules> = {\n requiredScripts: ['build', 'ts-check', 'eslint-check', 'prettier-check', 'test'],\n requiredFiles: ['tsconfig.json', 'eslint.config.js', 'readme.md'],\n turboTasks: [],\n}\n\n/**\n * Baseline rules for the monorepo root (`infra-kit audit --root`). Checks the\n * root commands, the workspace/turbo files, and that the turbo pipeline defines\n * the expected tasks \u2014 so the root's own config can also stay empty.\n */\nexport const ROOT_DEFAULT_RULES: Readonly<ResolvedPackageRules> = {\n requiredScripts: ['build', 'dev', 'test', 'qa', 'check', 'fix'],\n requiredFiles: ['turbo.json', 'pnpm-workspace.yaml'],\n turboTasks: ['build', 'test', 'ts-check', 'eslint-check', 'prettier-check', 'check'],\n}\n\n/**\n * Merge a parsed package config over a baseline. Each key is replaced wholesale\n * when the package provides it, otherwise the baseline value is used.\n *\n * @example\n * resolvePackageConfig({ requiredScripts: [] })\n * // => { requiredScripts: [], requiredFiles: [...DEFAULT_RULES.requiredFiles], turboTasks: [] }\n */\nexport const resolvePackageConfig = (\n config: InfraKitPackageConfig,\n baseline: Readonly<ResolvedPackageRules> = DEFAULT_RULES,\n): ResolvedPackageRules => {\n return {\n requiredScripts: config.requiredScripts ?? [...baseline.requiredScripts],\n requiredFiles: config.requiredFiles ?? [...baseline.requiredFiles],\n turboTasks: config.turbo?.requiredTasks ?? [...baseline.turboTasks],\n }\n}\n", "import { z } from 'zod'\n\n/**\n * Schema for the resolved (post-factory) package config object. `strictObject`\n * rejects unknown keys so typos in `infra-kit.config.ts` surface as validation\n * errors instead of being silently ignored.\n *\n * Kept in its own module \u2014 separate from the public `defineConfig`/types entry \u2014\n * so the published `infra-kit` type surface stays free of a `zod` import.\n */\nexport const packageConfigSchema = z.strictObject({\n requiredScripts: z.array(z.string().min(1)).optional(),\n requiredFiles: z.array(z.string().min(1)).optional(),\n turbo: z\n .strictObject({\n requiredTasks: z.array(z.string().min(1)).optional(),\n })\n .optional(),\n})\n", "import fs from 'node:fs/promises'\n\n/**\n * Resolve whether a path is reachable, suppressing ENOENT into a boolean.\n *\n * @example\n * await pathExists('/etc/hosts') // => true\n * await pathExists('/nope') // => false\n */\nexport const pathExists = async (target: string): Promise<boolean> => {\n try {\n await fs.access(target)\n\n return true\n } catch {\n return false\n }\n}\n", "import path from 'node:path'\n\nimport { DEFAULT_RULES } from 'src/lib/package-config'\nimport type { ResolvedPackageRules } from 'src/lib/package-config'\n\nimport { checkConfig, checkFiles, checkScripts, checkTurbo } from './checks'\nimport { readPackageJson } from './loader'\nimport type { PackageCheck, PackageValidationResult } from './types'\n\n// Re-exported on the historical import path so consumers and tests that reach\n// for the loader through `package-validator` keep resolving after the split.\nexport { discoverPackages, loadPackageConfig } from './loader'\nexport type { PackageCheck, PackageValidationResult } from './types'\n\n/**\n * Validate a single directory against its `infra-kit.config.ts` rules: the config\n * must be present and valid, every required script must be declared, every\n * required file must exist, and (root only) every required turbo task must be\n * defined. When the config fails to load, only that check is reported (the rules\n * are unknown, so the rule-based checks are skipped). `baseline` selects which\n * under-the-hood defaults apply \u2014 package defaults or {@link ROOT_DEFAULT_RULES}.\n *\n * @example\n * const result = await validatePackage('/repo/packages/serverless-config')\n * // result.passed reflects the package's conformance; result.checks lists each check\n */\nexport const validatePackage = async (\n packageDir: string,\n baseline: Readonly<ResolvedPackageRules> = DEFAULT_RULES,\n): Promise<PackageValidationResult> => {\n const pkgJson = await readPackageJson(packageDir)\n const packageName = pkgJson.name ?? path.basename(packageDir)\n\n const { check: configCheck, rules } = await checkConfig(packageDir, baseline)\n const checks: PackageCheck[] = [configCheck]\n\n if (rules) {\n checks.push(...checkScripts(pkgJson.scripts ?? {}, rules.requiredScripts))\n checks.push(...(await checkFiles(packageDir, rules.requiredFiles)))\n checks.push(...(await checkTurbo(packageDir, rules.turboTasks)))\n }\n\n const passed = checks.every((check) => {\n return check.status === 'pass'\n })\n\n return { packageDir, packageName, checks, passed }\n}\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\nimport { pathToFileURL } from 'node:url'\nimport { z } from 'zod'\n\nimport { DEFAULT_RULES, packageConfigSchema, resolvePackageConfig } from 'src/lib/package-config'\nimport type { ResolvedPackageRules } from 'src/lib/package-config'\n\nimport { pathExists } from '../fs-utils'\n\n/** Per-package config filename every validated package must provide. */\nexport const PACKAGE_CONFIG_FILE = 'infra-kit.config.ts'\n\ninterface PackageJsonShape {\n name?: string\n scripts?: Record<string, string>\n}\n\n/**\n * Read and JSON-parse a package.json, returning an empty object when it is\n * missing or unreadable so callers can degrade into a clear \"missing\" check.\n */\nexport const readPackageJson = async (packageDir: string): Promise<PackageJsonShape> => {\n try {\n const raw = await fs.readFile(path.join(packageDir, 'package.json'), 'utf-8')\n\n return JSON.parse(raw) as PackageJsonShape\n } catch {\n return {}\n }\n}\n\n/**\n * Load, resolve, and validate a package's `infra-kit.config.ts`.\n *\n * Dynamic-imports the file (ESM), resolves the Vite-style factory or object\n * default export, validates the result against {@link packageConfigSchema}, and\n * merges it over the defaults. Throws a descriptive error when the file is\n * absent or the resolved config violates the schema.\n *\n * @example\n * await loadPackageConfig('/repo/packages/serverless-config')\n * // => { requiredScripts: [], requiredFiles: ['serverless.common.yml'], turboTasks: [] }\n */\nexport const loadPackageConfig = async (\n packageDir: string,\n baseline: Readonly<ResolvedPackageRules> = DEFAULT_RULES,\n): Promise<ResolvedPackageRules> => {\n const configPath = path.join(packageDir, PACKAGE_CONFIG_FILE)\n\n if (!(await pathExists(configPath))) {\n throw new Error(`${PACKAGE_CONFIG_FILE} not found at ${configPath}`)\n }\n\n // Cache-bust with the file mtime so repeated loads (long-running MCP server)\n // pick up edits without a process restart. `.ts` configs load via Node's\n // native type stripping (the repo requires Node >= 24).\n const stat = await fs.stat(configPath)\n const moduleUrl = `${pathToFileURL(configPath).href}?mtime=${Number(stat.mtimeMs)}`\n\n const imported = (await import(moduleUrl)) as { default?: unknown }\n const rawExport = imported.default\n\n if (rawExport === undefined) {\n throw new Error(`${PACKAGE_CONFIG_FILE} at ${configPath} has no default export`)\n }\n\n const resolvedExport = typeof rawExport === 'function' ? await (rawExport as () => unknown)() : rawExport\n\n const parsed = packageConfigSchema.safeParse(resolvedExport)\n\n if (!parsed.success) {\n throw new Error(`Invalid ${PACKAGE_CONFIG_FILE} at ${configPath}: ${z.prettifyError(parsed.error)}`)\n }\n\n return resolvePackageConfig(parsed.data, baseline)\n}\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\nimport yaml from 'yaml'\n\nimport { pathExists } from '../fs-utils'\n\nconst WORKSPACE_FILE = 'pnpm-workspace.yaml'\n\n/**\n * List the immediate child directories of `dir`, returning `[]` when the path\n * can't be read (e.g. the parent glob segment matched a non-existent dir).\n */\nconst listChildDirs = async (dir: string): Promise<string[]> => {\n const entries = await fs.readdir(dir, { withFileTypes: true }).catch(() => {\n return []\n })\n\n return entries\n .filter((entry) => {\n return entry.isDirectory()\n })\n .map((entry) => {\n return path.join(dir, entry.name)\n })\n}\n\n/**\n * Apply one glob segment to a set of directories: `*` fans out to every child\n * directory, a literal segment keeps the dirs where that child path exists.\n */\nconst expandSegment = async (dirs: string[], segment: string): Promise<string[]> => {\n const next: string[] = []\n\n for (const dir of dirs) {\n if (segment === '*') {\n next.push(...(await listChildDirs(dir)))\n\n continue\n }\n\n const candidate = path.join(dir, segment)\n\n if (await pathExists(candidate)) {\n next.push(candidate)\n }\n }\n\n return next\n}\n\n/**\n * Expand a single pnpm-workspace glob (only the `*` segment wildcard is\n * supported, which covers every pattern this monorepo uses) into directories.\n */\nconst expandGlob = async (projectRoot: string, pattern: string): Promise<string[]> => {\n let dirs = [projectRoot]\n\n for (const segment of pattern.split('/')) {\n dirs = await expandSegment(dirs, segment)\n }\n\n return dirs\n}\n\n/**\n * Discover validatable workspace packages from `pnpm-workspace.yaml`.\n *\n * Negation patterns (`!\u2026`) and everything under `vendor/` are excluded \u2014\n * vendor is mirrored from `starter-workspace` and is checksum-enforced by\n * `pnpm vendor:check`, so its configs are owned upstream, not here. Only\n * directories that actually contain a package.json are returned.\n *\n * @example\n * await discoverPackages('/repo')\n * // => ['/repo/apps/infra-kit/cli', '/repo/packages/serverless-config']\n */\nexport const discoverPackages = async (projectRoot: string): Promise<string[]> => {\n const raw = await fs.readFile(path.join(projectRoot, WORKSPACE_FILE), 'utf-8')\n const parsed = (yaml.parse(raw) ?? {}) as { packages?: string[] }\n\n const patterns = (parsed.packages ?? []).filter((pattern) => {\n return !pattern.startsWith('!') && !pattern.startsWith('vendor')\n })\n\n const found = new Set<string>()\n\n for (const pattern of patterns) {\n const dirs = await expandGlob(projectRoot, pattern)\n\n for (const dir of dirs) {\n if (await pathExists(path.join(dir, 'package.json'))) {\n found.add(dir)\n }\n }\n }\n\n return [...found].sort()\n}\n", "import { DEFAULT_RULES } from 'src/lib/package-config'\nimport type { ResolvedPackageRules } from 'src/lib/package-config'\n\nimport { PACKAGE_CONFIG_FILE, loadPackageConfig } from '../loader'\nimport type { PackageCheck } from '../types'\n\n/**\n * Build the \"config present and valid\" check, returning the resolved rules when\n * the load succeeds so the caller can run the rule-based checks against them.\n * When the config fails to load the rules are `null` and the caller skips the\n * rule-based checks (the expectations are unknown).\n */\nexport const checkConfig = async (\n packageDir: string,\n baseline: Readonly<ResolvedPackageRules> = DEFAULT_RULES,\n): Promise<{ check: PackageCheck; rules: ResolvedPackageRules | null }> => {\n try {\n const rules = await loadPackageConfig(packageDir, baseline)\n\n return {\n check: { name: PACKAGE_CONFIG_FILE, status: 'pass', message: 'present and valid' },\n rules,\n }\n } catch (err) {\n return {\n check: { name: PACKAGE_CONFIG_FILE, status: 'fail', message: (err as Error).message },\n rules: null,\n }\n }\n}\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\n\nimport type { PackageCheck } from '../types'\n\n/**\n * Check that every required file exists relative to the package root AND is a\n * regular file. A directory that happens to share the required name fails \u2014 a\n * required `readme.md` must be the file, not a folder.\n */\nexport const checkFiles = async (packageDir: string, requiredFiles: string[]): Promise<PackageCheck[]> => {\n return Promise.all(\n requiredFiles.map(async (file) => {\n const stat = await fs.stat(path.join(packageDir, file)).catch(() => {\n return null\n })\n\n if (!stat) {\n return { name: `file:${file}`, status: 'fail' as const, message: `missing file: ${file}` }\n }\n\n if (!stat.isFile()) {\n return { name: `file:${file}`, status: 'fail' as const, message: `not a file: ${file} (found a directory)` }\n }\n\n return { name: `file:${file}`, status: 'pass' as const, message: 'exists' }\n }),\n )\n}\n", "import type { PackageCheck } from '../types'\n\n/**\n * Check that every required script is present in the package.json `scripts` map\n * and carries a runnable command. A key declared with an empty or whitespace-only\n * value fails as well \u2014 an empty script silently no-ops in CI, so presence alone\n * is not enough.\n */\nexport const checkScripts = (scripts: Record<string, string>, requiredScripts: string[]): PackageCheck[] => {\n return requiredScripts.map((script) => {\n const value = scripts[script]\n\n if (typeof value !== 'string') {\n return { name: `script:${script}`, status: 'fail', message: `missing \"${script}\" in package.json scripts` }\n }\n\n if (value.trim().length === 0) {\n return { name: `script:${script}`, status: 'fail', message: `\"${script}\" is empty in package.json scripts` }\n }\n\n return { name: `script:${script}`, status: 'pass', message: 'defined' }\n })\n}\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\n\nimport type { PackageCheck } from '../types'\n\nconst TURBO_FILE = 'turbo.json'\n\n/**\n * Check that every required turbo task is defined in turbo.json `tasks`. A root\n * task may be keyed as either `name` or `//#name`, so both forms count as present.\n * Runs only when the resolved rules ask for turbo tasks (the monorepo root).\n *\n * Reports a single clear diagnostic when turbo.json is unreadable or carries no\n * `tasks` object, rather than emitting one identical \"missing task\" line per\n * required task.\n */\nexport const checkTurbo = async (packageDir: string, requiredTasks: string[]): Promise<PackageCheck[]> => {\n if (requiredTasks.length === 0) {\n return []\n }\n\n let parsed: { tasks?: Record<string, unknown> }\n\n try {\n const raw = await fs.readFile(path.join(packageDir, TURBO_FILE), 'utf-8')\n\n parsed = JSON.parse(raw) as { tasks?: Record<string, unknown> }\n } catch (err) {\n return [{ name: TURBO_FILE, status: 'fail', message: `cannot read/parse ${TURBO_FILE}: ${(err as Error).message}` }]\n }\n\n const tasks = parsed.tasks\n\n if (tasks === null || typeof tasks !== 'object') {\n return [{ name: TURBO_FILE, status: 'fail', message: `no \"tasks\" object defined in ${TURBO_FILE}` }]\n }\n\n return requiredTasks.map((task) => {\n const defined = task in tasks || `//#${task}` in tasks\n\n return {\n name: `turbo:${task}`,\n status: defined ? 'pass' : 'fail',\n message: defined ? 'defined' : `missing turbo task \"${task}\" in ${TURBO_FILE}`,\n }\n })\n}\n", "import type { z } from 'zod'\n\nexport interface ToolsExecutionResult<TStructured = Record<string, unknown>> {\n [x: string]: unknown\n content: {\n type: 'text'\n text: string\n }[]\n structuredContent?: TStructured\n}\n\nexport interface RequiredConfirmedOptionArg {\n confirmedCommand: boolean\n}\n\nexport interface McpTool<TIn extends z.ZodRawShape = z.ZodRawShape, TOut extends z.ZodRawShape = z.ZodRawShape> {\n name: string\n description: string\n inputSchema: TIn\n outputSchema: TOut\n handler: (\n params: z.infer<z.ZodObject<TIn>> & RequiredConfirmedOptionArg,\n ) => Promise<ToolsExecutionResult<z.infer<z.ZodObject<TOut>>>>\n}\n\n/**\n * Build the dual-channel content array shared by every MCP tool. Narrows the\n * literal `type: 'text'` so handlers can use inferred return types without TS\n * widening `type` to `string` \u2014 which would otherwise break assignability\n * against the MCP SDK's content union.\n *\n * @example\n * return {\n * content: textContent(JSON.stringify(structuredContent, null, 2)),\n * structuredContent,\n * }\n */\nexport const textContent = (text: string): ToolsExecutionResult['content'] => {\n return [{ type: 'text', text }]\n}\n\n/**\n * Factory that ties the handler's return type to the declared `outputSchema`\n * so `structuredContent` is checked against the schema at compile time. If a\n * handler accidentally drops or renames a field, TS errors at the registration\n * site rather than at runtime in an MCP client.\n *\n * @example\n * export const envLoadMcpTool = defineMcpTool({\n * name: 'env-load',\n * description: '...',\n * inputSchema: { config: z.string() },\n * outputSchema: {\n * filePath: z.string(),\n * variableCount: z.number(),\n * project: z.string(),\n * config: z.string(),\n * },\n * handler: envLoad,\n * })\n */\nexport const defineMcpTool = <TIn extends z.ZodRawShape, TOut extends z.ZodRawShape>(\n tool: McpTool<TIn, TOut>,\n): McpTool<TIn, TOut> => {\n return tool\n}\n", "import fs from 'node:fs'\nimport path from 'node:path'\nimport process from 'node:process'\nimport { z } from 'zod'\n\nimport {\n ENV_CLEAR_FILE,\n ENV_LOAD_FILE,\n INFRA_KIT_ENV_CONFIG_VAR,\n INFRA_KIT_ENV_LOADED_AT_VAR,\n INFRA_KIT_ENV_PROJECT_VAR,\n atomicWriteFileSync,\n getSessionCacheDir,\n parseVarNamesFromEnvFile,\n} from 'src/lib/constants'\nimport { defineMcpTool, textContent } from 'src/types'\n\n/**\n * Clear loaded env vars. Prints a file path to stdout that must be sourced to apply.\n * The env-clear shell alias does this automatically. Throws when no env is loaded\n * so CLI callers exit non-zero and MCP callers receive a structured tool error.\n */\nexport const envClear = async () => {\n const cacheDir = getSessionCacheDir()\n const envLoadPath = path.join(cacheDir, ENV_LOAD_FILE)\n\n if (!fs.existsSync(envLoadPath)) {\n throw new Error('No loaded environment found. Run `env-load` first.')\n }\n\n const varNames = parseVarNamesFromEnvFile(envLoadPath)\n\n const unsetLines = [\n ...varNames.map((v) => {\n return `unset ${v}`\n }),\n `unset ${INFRA_KIT_ENV_CONFIG_VAR}`,\n `unset ${INFRA_KIT_ENV_PROJECT_VAR}`,\n `unset ${INFRA_KIT_ENV_LOADED_AT_VAR}`,\n ]\n\n const clearFilePath = path.resolve(cacheDir, ENV_CLEAR_FILE)\n\n fs.mkdirSync(cacheDir, { recursive: true, mode: 0o700 })\n\n atomicWriteFileSync(clearFilePath, `${unsetLines.join('\\n')}\\n`, 0o600)\n\n // REQUIRED\n process.stdout.write(`${clearFilePath}\\n`)\n\n // Remove env load file so the next env-clear call correctly reports \"no env loaded\".\n fs.unlinkSync(envLoadPath)\n\n const structuredContent = {\n filePath: clearFilePath,\n variableCount: varNames.length,\n unsetStatements: unsetLines,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const envClearMcpTool = defineMcpTool({\n name: 'env-clear',\n description:\n 'Generate a shell script that unsets every env var previously loaded by env-load for this session, plus the infra-kit session metadata vars. Does NOT mutate the calling process. When `infra-kit init` has installed the zsh shell integration, the user\\'s terminal auto-sources the unset script on its next prompt (precmd hook) \u2014 so calling this via MCP will clear the vars in the shell that launched Claude Code automatically. Other callers must source \"<filePath>\" themselves or surface it to the user. Errors if no env is currently loaded.',\n inputSchema: {},\n outputSchema: {\n filePath: z.string().describe('Path to the file that must be sourced to apply'),\n variableCount: z.number().describe('Number of variables cleared'),\n unsetStatements: z.array(z.string()).describe('Unset statements generated'),\n },\n handler: envClear,\n})\n", "import fs from 'node:fs'\nimport os from 'node:os'\nimport path from 'node:path'\nimport process from 'node:process'\n\nexport const ENV_LOAD_FILE = 'env-load.sh'\nexport const ENV_CLEAR_FILE = 'env-clear.sh'\n\nexport const INFRA_KIT_SESSION_VAR = 'INFRA_KIT_SESSION'\nexport const INFRA_KIT_ENV_CONFIG_VAR = 'INFRA_KIT_ENV_CONFIG'\nexport const INFRA_KIT_ENV_PROJECT_VAR = 'INFRA_KIT_ENV_PROJECT'\nexport const INFRA_KIT_ENV_LOADED_AT_VAR = 'INFRA_KIT_ENV_LOADED_AT'\n\n/**\n * Matches a line of the form `KEY=...` where KEY is an env-var identifier\n * (letter or underscore, then word chars). Capture group 1 is the name. Shared\n * between env-load (validation, var counting) and parseVarNamesFromEnvFile.\n */\nexport const ENV_VAR_LINE_PATTERN = /^([A-Z_]\\w*)=/i\n\nexport const parseVarNamesFromEnvFile = (filePath: string): string[] => {\n if (!fs.existsSync(filePath)) return []\n\n const content = fs.readFileSync(filePath, 'utf-8')\n const names: string[] = []\n\n for (const line of content.split('\\n')) {\n const match = ENV_VAR_LINE_PATTERN.exec(line)\n\n if (match) {\n names.push(match[1]!)\n }\n }\n\n return names\n}\n\n/**\n * Root cache dir for infra-kit across all sessions. Resolved from\n * $XDG_CACHE_HOME when set, falling back to ~/.cache/infra-kit. Keep in sync\n * with the shell block emitted by `infra-kit init` (src/commands/init/init.ts).\n */\nexport const getCacheRoot = (): string => {\n const xdg = process.env.XDG_CACHE_HOME\n const base = xdg && xdg.length > 0 ? xdg : path.join(os.homedir(), '.cache')\n\n return path.join(base, 'infra-kit')\n}\n\nexport const getSessionCacheDir = (): string => {\n const session = process.env[INFRA_KIT_SESSION_VAR]\n\n if (!session) {\n throw new Error(`${INFRA_KIT_SESSION_VAR} is not set. Run \\`infra-kit init\\` then \\`source ~/.zshrc\\`.`)\n }\n\n return path.join(getCacheRoot(), session)\n}\n\n/**\n * Write content atomically: write to a pid-suffixed temp file in the same\n * directory, then rename. fs.renameSync is atomic on a single filesystem, so\n * concurrent writers can't produce a half-written secret file.\n */\nexport const atomicWriteFileSync = (filePath: string, content: string, mode: number): void => {\n const tmpPath = `${filePath}.tmp.${process.pid}`\n\n fs.writeFileSync(tmpPath, content, { mode })\n\n try {\n fs.renameSync(tmpPath, filePath)\n } catch (error) {\n fs.rmSync(tmpPath, { force: true })\n throw error\n }\n}\n\nexport const WORKTREES_DIR_SUFFIX = '-worktrees'\n// eslint-disable-next-line sonarjs/publicly-writable-directories\nexport const LOG_FILE_PATH = '/tmp/mcp-infra-kit.log'\n", "import { z } from 'zod'\n\nimport { getDopplerProject } from 'src/integrations/doppler/doppler-project'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { defineMcpTool, textContent } from 'src/types'\n\n/**\n * List available Doppler configs for the detected project.\n *\n * Purely local: reads infra-kit.json and does not call Doppler. We intentionally\n * do not run validateDopplerCliAndAuth here \u2014 users listing envs often do so\n * before `doppler login`, and a spurious auth error would be misleading.\n */\nexport const envList = async () => {\n const project = await getDopplerProject()\n const { environments } = await getInfraKitConfig()\n\n logger.info(`Doppler project: ${project}\\n`)\n logger.info('Available configs:')\n\n for (const env of environments) {\n logger.info(` - ${env}`)\n }\n\n const structuredContent = {\n project,\n configs: environments,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const envListMcpTool = defineMcpTool({\n name: 'env-list',\n description:\n 'List the environments the project is configured to support. Returns the `environments` list declared in infra-kit.json at the project root (not a live fetch from Doppler) plus the Doppler project name resolved from the same file. Read-only.',\n inputSchema: {},\n outputSchema: {\n project: z.string().describe('Detected Doppler project name'),\n configs: z.array(z.string()).describe('Available environment configs'),\n },\n handler: envList,\n})\n", "import fs from 'node:fs/promises'\nimport os from 'node:os'\nimport path from 'node:path'\nimport { z } from 'zod'\n\nimport { getProjectRoot, getRepoName } from 'src/lib/git-utils'\n\nconst INFRA_KIT_CONFIG_FILE = 'infra-kit.json'\n\nconst USER_CONFIG_DIR_NAME = '.infra-kit'\nconst USER_GLOBAL_CONFIG_FILE = 'config.json'\nconst USER_PROJECTS_DIR = 'projects'\n\n// envManagement\nconst dopplerEnvManagementSchema = z.object({\n provider: z.literal('doppler'),\n config: z.object({\n name: z.string().min(1),\n }),\n})\n\nconst envManagementSchema = z.discriminatedUnion('provider', [dopplerEnvManagementSchema])\n\n// ide\nconst cursorIdeConfigSchema = z\n .object({\n mode: z.enum(['workspace', 'windows']).default('workspace'),\n workspaceConfigPath: z.string().min(1).optional(),\n })\n .refine(\n (v) => {\n return v.mode !== 'workspace' || !!v.workspaceConfigPath\n },\n {\n message: 'workspaceConfigPath is required when mode is \"workspace\"',\n path: ['workspaceConfigPath'],\n },\n )\n\nconst cursorIdeSchema = z.object({\n provider: z.literal('cursor'),\n config: cursorIdeConfigSchema,\n})\n\nconst ideSchema = z.discriminatedUnion('provider', [cursorIdeSchema])\n\n// taskManager\nconst jiraTaskManagerSchema = z.object({\n provider: z.literal('jira'),\n config: z.object({\n baseUrl: z.string().url(),\n projectId: z.number().int().positive(),\n }),\n})\n\nconst taskManagerSchema = z.discriminatedUnion('provider', [jiraTaskManagerSchema])\n\n// worktrees prompt defaults\nconst worktreesConfigSchema = z.object({\n openInGithubDesktop: z.boolean().optional(),\n openInCmux: z.boolean().optional(),\n})\n\nexport const infraKitConfigSchema = z.object({\n environments: z.array(z.string().min(1)).min(1),\n envManagement: envManagementSchema,\n ide: ideSchema.optional(),\n taskManager: taskManagerSchema.optional(),\n worktrees: worktreesConfigSchema.optional(),\n})\n\nexport const infraKitOverrideConfigSchema = infraKitConfigSchema.partial()\n\nexport type InfraKitConfig = z.infer<typeof infraKitConfigSchema>\n\nexport interface InfraKitConfigPaths {\n /** Committed project config (required). */\n main: string\n /** User-scope global overrides applied to every project. */\n userGlobal: string\n /** User-scope per-project overrides \u2014 `<userProjectsDir>/<projectName>/infra-kit.json`. */\n userProject: string\n /** Repo basename (`path.basename(projectRoot)`) used to namespace the user-project file. */\n projectName: string\n}\n\ninterface CacheEntry {\n mtimes: Record<keyof Omit<InfraKitConfigPaths, 'projectName'>, number | null>\n value: InfraKitConfig\n}\n\nlet cached: CacheEntry | null = null\n\n/**\n * Resolve every file path that participates in the config merge chain. Always\n * returns paths even for files that don't yet exist, so callers can use them\n * for \"where would my override go?\" prompts.\n *\n * @example\n * const paths = await getInfraKitConfigPaths()\n * // {\n * // main: '/Users/arthur/projects/api/infra-kit.json',\n * // userGlobal: '/Users/arthur/.infra-kit/config.json',\n * // userProject: '/Users/arthur/.infra-kit/projects/api/infra-kit.json',\n * // projectName: 'api',\n * // }\n */\nexport const getInfraKitConfigPaths = async (): Promise<InfraKitConfigPaths> => {\n const projectRoot = await getProjectRoot()\n const projectName = await getRepoName()\n const userConfigDir = path.join(os.homedir(), USER_CONFIG_DIR_NAME)\n\n return {\n main: path.join(projectRoot, INFRA_KIT_CONFIG_FILE),\n userGlobal: path.join(userConfigDir, USER_GLOBAL_CONFIG_FILE),\n userProject: path.join(userConfigDir, USER_PROJECTS_DIR, projectName, INFRA_KIT_CONFIG_FILE),\n projectName,\n }\n}\n\n/**\n * Read and validate `infra-kit.json`, with optional override layers shallow-merged\n * on top in this order (later wins):\n * 1. project `infra-kit.json` \u2014 committed source of truth\n * 2. `~/.infra-kit/config.json` \u2014 user-global defaults\n * 3. `~/.infra-kit/projects/<repo-name>/infra-kit.json` \u2014 user-scope per-project overrides\n *\n * Top-level keys (entire capability sections like `ide`, `envManagement`)\n * replace wholesale. Results are cached per file mtimes so the long-running\n * MCP server picks up edits without a restart.\n *\n * @example\n * // infra-kit.json: { \"environments\": [\"dev\"], \"envManagement\": { \"provider\": \"doppler\", \"config\": { \"name\": \"p\" } } }\n * // ~/.infra-kit/config.json: { \"ide\": { \"provider\": \"cursor\", \"config\": { \"mode\": \"windows\" } } }\n * const cfg = await getInfraKitConfig()\n * // => { environments: ['dev'], envManagement: {...}, ide: { provider: 'cursor', config: { mode: 'windows' } } }\n */\nexport const getInfraKitConfig = async (): Promise<InfraKitConfig> => {\n const paths = await getInfraKitConfigPaths()\n\n let mainStat: Awaited<ReturnType<typeof fs.stat>>\n\n try {\n mainStat = await fs.stat(paths.main)\n } catch {\n cached = null\n\n // Bridge the YAML\u2192JSON cutover: if a legacy infra-kit.yml is sitting where\n // the JSON config should be, point the user at the one-shot migration.\n const legacyYmlPath = paths.main.replace(/\\.json$/, '.yml')\n\n if (await statIfExists(legacyYmlPath)) {\n throw new Error(\n `infra-kit.json not found at ${paths.main}. A legacy infra-kit.yml exists \u2014 run \\`infra-kit init\\` to convert it.`,\n )\n }\n\n throw new Error(`infra-kit.json not found at ${paths.main}`)\n }\n\n const [userGlobalStat, userProjectStat] = await Promise.all([\n statIfExists(paths.userGlobal),\n statIfExists(paths.userProject),\n ])\n\n const mtimes = {\n main: Number(mainStat.mtimeMs),\n userGlobal: userGlobalStat ? Number(userGlobalStat.mtimeMs) : null,\n userProject: userProjectStat ? Number(userProjectStat.mtimeMs) : null,\n }\n\n if (cached && shallowEqual(cached.mtimes, mtimes)) {\n return cached.value\n }\n\n const layers: ConfigLayer[] = [\n { label: 'infra-kit.json', path: paths.main, required: true },\n { label: '~/.infra-kit/config.json', path: paths.userGlobal, required: false },\n {\n label: `~/.infra-kit/projects/${paths.projectName}/infra-kit.json`,\n path: paths.userProject,\n required: false,\n },\n ]\n\n let merged: Record<string, unknown> = {}\n\n for (const layer of layers) {\n const data = await loadLayer(layer)\n\n if (data === null) continue\n\n merged = { ...merged, ...data }\n }\n\n const finalResult = infraKitConfigSchema.safeParse(merged)\n\n if (!finalResult.success) {\n throw new Error(`Invalid merged infra-kit config: ${z.prettifyError(finalResult.error)}`)\n }\n\n cached = { mtimes, value: finalResult.data }\n\n return finalResult.data\n}\n\n/**\n * For tests \u2014 drops the in-memory cache so the next read hits disk.\n *\n * @example\n * resetInfraKitConfigCache()\n * await getInfraKitConfig() // re-reads files even if mtimes look unchanged\n */\nexport const resetInfraKitConfigCache = (): void => {\n cached = null\n}\n\n/**\n * `fs.stat` that returns `null` instead of throwing on ENOENT. Used so the\n * resolver can probe optional files in the merge chain without try/catch noise.\n *\n * @example\n * const stat = await statIfExists('/does/not/exist') // => null\n */\nconst statIfExists = async (filePath: string): Promise<Awaited<ReturnType<typeof fs.stat>> | null> => {\n try {\n return await fs.stat(filePath)\n } catch {\n return null\n }\n}\n\n/**\n * `fs.readFile` that returns `null` instead of throwing on ENOENT.\n *\n * @example\n * const raw = await readIfExists('/missing.json') // => null\n * const raw = await readIfExists('/exists.json') // => '{ \"environments\": [\"dev\"] }\\n'\n */\nconst readIfExists = async (filePath: string): Promise<string | null> => {\n try {\n return await fs.readFile(filePath, 'utf-8')\n } catch {\n return null\n }\n}\n\n/**\n * Reference-equality comparison of every key in two flat records. Used to\n * cheaply detect whether the cached mtime fingerprint still matches.\n *\n * @example\n * shallowEqual({ a: 1, b: 2 }, { a: 1, b: 2 }) // => true\n * shallowEqual({ a: 1 }, { a: 1, b: 2 }) // => false\n * shallowEqual({ a: 1 }, { a: 2 }) // => false\n */\nconst shallowEqual = <T extends Record<string, unknown>>(a: T, b: T): boolean => {\n const keys = Object.keys(a)\n\n if (keys.length !== Object.keys(b).length) return false\n\n return keys.every((k) => {\n return a[k] === b[k]\n })\n}\n\ninterface ConfigLayer {\n label: string\n path: string\n required: boolean\n}\n\n/**\n * Read a single layer of the merge chain: parse the JSON if the file exists\n * and validate it against the override schema. Returns `null` if an optional\n * layer is missing; throws if the layer is required, malformed, or invalid.\n * An empty/whitespace-only file is treated as `{}` (JSON.parse would throw).\n *\n * @example\n * await loadLayer({ label: '~/.infra-kit/config.json', path: '/missing.json', required: false })\n * // => null\n *\n * @example\n * // /home/me/.infra-kit/config.json: '{ \"ide\": { \"provider\": \"cursor\", \"config\": { \"mode\": \"windows\" } } }'\n * await loadLayer({ label: '~/.infra-kit/config.json', path: '/home/me/.infra-kit/config.json', required: false })\n * // => { ide: { provider: 'cursor', config: { mode: 'windows' } } }\n */\nconst loadLayer = async (layer: ConfigLayer): Promise<Record<string, unknown> | null> => {\n const raw = await readIfExists(layer.path)\n\n if (raw === null) {\n if (layer.required) {\n throw new Error(`${layer.label} not found at ${layer.path}`)\n }\n\n return null\n }\n\n let parsedRaw: unknown\n\n try {\n parsedRaw = raw.trim() === '' ? {} : JSON.parse(raw)\n } catch (err) {\n throw new Error(`Invalid JSON in ${layer.label} at ${layer.path}: ${(err as Error).message}`)\n }\n\n const result = infraKitOverrideConfigSchema.safeParse(parsedRaw)\n\n if (!result.success) {\n throw new Error(`Invalid ${layer.label} at ${layer.path}: ${z.prettifyError(result.error)}`)\n }\n\n return result.data as Record<string, unknown>\n}\n", "import { getInfraKitConfig } from 'src/lib/infra-kit-config'\n\n/**\n * Resolve Doppler project name from infra-kit.json at the project root\n */\nexport const getDopplerProject = async (): Promise<string> => {\n const { envManagement } = await getInfraKitConfig()\n\n return envManagement.config.name\n}\n", "import select from '@inquirer/select'\nimport { Buffer } from 'node:buffer'\nimport fs from 'node:fs'\nimport path from 'node:path'\nimport process from 'node:process'\nimport { z } from 'zod'\nimport { $ } from 'zx'\n\nimport { validateDopplerCliAndAuth } from 'src/integrations/doppler'\nimport { getDopplerProject } from 'src/integrations/doppler/doppler-project'\nimport { commandEcho } from 'src/lib/command-echo'\nimport {\n ENV_LOAD_FILE,\n ENV_VAR_LINE_PATTERN,\n INFRA_KIT_ENV_CONFIG_VAR,\n INFRA_KIT_ENV_LOADED_AT_VAR,\n INFRA_KIT_ENV_PROJECT_VAR,\n atomicWriteFileSync,\n getSessionCacheDir,\n} from 'src/lib/constants'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface EnvLoadArgs {\n config?: string\n}\n\n/**\n * Load environment variables from Doppler for the given config\n */\nexport const envLoad = async (args: EnvLoadArgs) => {\n await validateDopplerCliAndAuth()\n\n const { config } = args\n\n commandEcho.start('env-load')\n\n let selectedConfig = ''\n\n if (config) {\n selectedConfig = config\n } else {\n const { environments } = await getInfraKitConfig()\n\n commandEcho.setInteractive()\n selectedConfig = await select(\n {\n message: 'Select environment config',\n choices: environments.map((env) => {\n return { name: env, value: env }\n }),\n },\n // Render to stderr so the prompt is visible when stdout is captured via $() in the shell function.\n // Only env-load and env-clear use the $() stdout-capture shell pattern.\n { output: process.stderr },\n )\n }\n\n commandEcho.addOption('--config', selectedConfig)\n\n const project = await getDopplerProject()\n\n const envContent = await downloadDopplerSecrets(project, selectedConfig)\n\n assertValidEnvContent(envContent)\n\n // Build env file content in dotenv format\n const loadedAt = new Date().toISOString()\n const envFileLines = [\n 'set -a',\n envContent,\n `${INFRA_KIT_ENV_CONFIG_VAR}=${shellSingleQuote(selectedConfig)}`,\n `${INFRA_KIT_ENV_PROJECT_VAR}=${shellSingleQuote(project)}`,\n `${INFRA_KIT_ENV_LOADED_AT_VAR}=${shellSingleQuote(loadedAt)}`,\n 'set +a',\n ]\n\n const cacheDir = getSessionCacheDir()\n const envFilePath = path.resolve(cacheDir, ENV_LOAD_FILE)\n\n fs.mkdirSync(cacheDir, { recursive: true, mode: 0o700 })\n atomicWriteFileSync(envFilePath, `${envFileLines.join('\\n')}\\n`, 0o600)\n\n // REQUIRED\n process.stdout.write(`${envFilePath}\\n`)\n\n // Logs to stderr (pino \u2192 pretty-print), so it doesn't pollute the captured\n // file path that the shell wrapper reads from stdout.\n commandEcho.print()\n\n const varCount = countEnvVarLines(envContent)\n\n const structuredContent = {\n filePath: envFilePath,\n variableCount: varCount,\n project,\n config: selectedConfig,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n/**\n * Cap the Doppler stdout we're willing to accept. A well-formed env bundle is\n * O(10 KB); megabytes would indicate a service regression or the wrong stream\n * being captured, and we don't want to write that to disk or source it.\n */\nexport const DOPPLER_MAX_OUTPUT_BYTES = 1024 * 1024\n\n/**\n * Hard upper bound for the Doppler subprocess. Well under zx's default so a\n * hung call surfaces quickly instead of blocking an interactive shell or an\n * MCP tool handler.\n */\nconst DOPPLER_DOWNLOAD_TIMEOUT_MS = 30_000\n\nconst downloadDopplerSecrets = async (project: string, config: string): Promise<string> => {\n const prevQuiet = $.quiet\n\n $.quiet = true\n try {\n const result =\n await $`doppler secrets download --no-file --format env --project ${project} --config ${config}`.timeout(\n DOPPLER_DOWNLOAD_TIMEOUT_MS,\n )\n\n assertDopplerOutputSize(result.stdout)\n\n return result.stdout.trim()\n } finally {\n $.quiet = prevQuiet\n }\n}\n\nexport const assertDopplerOutputSize = (stdout: string): void => {\n const bytes = Buffer.byteLength(stdout, 'utf-8')\n\n if (bytes > DOPPLER_MAX_OUTPUT_BYTES) {\n throw new Error(\n `doppler returned unexpectedly large output (${bytes} bytes > ${DOPPLER_MAX_OUTPUT_BYTES}) \u2014 refusing to write to disk`,\n )\n }\n}\n\nconst countEnvVarLines = (content: string): number => {\n return content.split('\\n').filter((line) => {\n return ENV_VAR_LINE_PATTERN.test(line)\n }).length\n}\n\nconst SHELL_DIRECTIVE_LINES = new Set(['set -a', 'set +a'])\n\nexport const shellSingleQuote = (value: string): string => {\n const escaped = value.replaceAll(\"'\", \"'\\\\''\")\n\n return `'${escaped}'`\n}\n\n/**\n * Guard against Doppler returning non-env output (auth warnings on stdout,\n * partial downloads, HTML error pages, etc.). Every non-blank, non-directive\n * line must match KEY=VALUE \u2014 skipping directives keeps future format tweaks\n * cheap without loosening the check.\n */\nexport const assertValidEnvContent = (content: string): void => {\n if (content.trim().length === 0) {\n throw new Error('doppler returned empty output for env-load')\n }\n\n for (const line of content.split('\\n')) {\n const trimmed = line.trim()\n\n if (trimmed.length === 0 || SHELL_DIRECTIVE_LINES.has(trimmed)) continue\n\n if (!ENV_VAR_LINE_PATTERN.test(trimmed)) {\n throw new Error(\n `doppler returned unexpected output for env-load (expected KEY=value lines, got: ${JSON.stringify(trimmed.slice(0, 80))})`,\n )\n }\n }\n}\n\n// MCP Tool Registration\nexport const envLoadMcpTool = defineMcpTool({\n name: 'env-load',\n description:\n 'Download the env vars for a Doppler config and write them to a temporary shell script. Does NOT mutate the calling process \u2014 returns the path to a script that must be sourced (\"source <filePath>\") for the vars to take effect. The infra-kit shell wrapper auto-sources; direct MCP callers must handle sourcing themselves or surface filePath to the user. \"config\" is required when invoked via MCP (the CLI interactive picker is unreachable without a TTY).',\n inputSchema: {\n config: z\n .string()\n .describe('Doppler config / environment name to load (e.g. \"dev\", \"arthur\", \"renana\"). Required for MCP calls.'),\n },\n outputSchema: {\n filePath: z.string().describe('Path to the file that must be sourced to apply variables'),\n variableCount: z.number().describe('Number of variables loaded'),\n project: z.string().describe('Doppler project name'),\n config: z.string().describe('Doppler config name'),\n },\n handler: envLoad,\n})\n", "import { $ } from 'zx'\n\n/**\n * Validate Doppler CLI installation and authentication status. Throws on failure\n * so callers (CLI entry, MCP tool handler) can translate to the right surface \u2014\n * CLI exits non-zero; MCP returns a structured tool error instead of tearing\n * down the server.\n */\nexport const validateDopplerCliAndAuth = async (): Promise<void> => {\n try {\n await $`doppler --version`\n } catch (error: unknown) {\n throw new Error('Doppler CLI is not installed. Install it from: https://docs.doppler.com/docs/install-cli', {\n cause: error,\n })\n }\n\n try {\n await $`doppler me`\n } catch (error: unknown) {\n throw new Error('Doppler CLI is not authenticated. Run: doppler login', { cause: error })\n }\n}\n", "import { logger } from 'src/lib/logger'\n\ninterface CommandOption {\n flag: string\n value: string | string[] | boolean\n}\n\nconst createCommandEcho = () => {\n let commandName = ''\n let options: CommandOption[] = []\n let isInteractive = false\n\n return {\n /**\n * Initialize command echo for a new command\n */\n start(name: string): void {\n commandName = name\n options = []\n isInteractive = false\n },\n\n /**\n * Mark that the command had interactive input (prompts)\n * Call this once when ANY prompt happens\n */\n setInteractive(): void {\n isInteractive = true\n },\n\n /**\n * Track an option selection\n * @param flag The CLI flag (e.g., \"--versions\")\n * @param value The selected value\n */\n addOption(flag: string, value: string | string[] | boolean): void {\n options.push({ flag, value })\n },\n\n /**\n * Print the equivalent CLI command if there was interactive input\n */\n print(): void {\n if (!isInteractive || options.length === 0) {\n return\n }\n\n const formattedOptions = options\n .map((opt) => {\n if (typeof opt.value === 'boolean') {\n return opt.value ? opt.flag : ''\n }\n\n if (Array.isArray(opt.value)) {\n return `${opt.flag} \"${opt.value.join(', ')}\"`\n }\n\n return `${opt.flag} \"${opt.value}\"`\n })\n .filter(Boolean)\n .join(' ')\n\n logger.info(`\uD83D\uDCDF Equivalent command: \\npnpm exec infra-kit ${commandName} ${formattedOptions}\\n`)\n },\n\n /**\n * Reset state (useful for testing)\n */\n reset(): void {\n commandName = ''\n options = []\n isInteractive = false\n },\n }\n}\n\n// Singleton instance (same pattern as logger)\nexport const commandEcho = createCommandEcho()\n", "import path from 'node:path'\nimport process from 'node:process'\nimport { z } from 'zod'\n\nimport { validateDopplerCliAndAuth } from 'src/integrations/doppler'\nimport {\n ENV_LOAD_FILE,\n INFRA_KIT_ENV_CONFIG_VAR,\n INFRA_KIT_ENV_LOADED_AT_VAR,\n INFRA_KIT_ENV_PROJECT_VAR,\n INFRA_KIT_SESSION_VAR,\n getSessionCacheDir,\n parseVarNamesFromEnvFile,\n} from 'src/lib/constants'\nimport { logger } from 'src/lib/logger'\nimport { defineMcpTool, textContent } from 'src/types'\n\n/**\n * Show Doppler authentication status and detected project info\n */\nexport const envStatus = async () => {\n await validateDopplerCliAndAuth()\n\n logger.info('Environment session status:')\n\n // Check session-loaded vars \u2014 getSessionCacheDir() throws if INFRA_KIT_SESSION is unset\n const cacheDir = getSessionCacheDir()\n\n const sessionId = process.env[INFRA_KIT_SESSION_VAR]!\n const envLoadPath = path.join(cacheDir, ENV_LOAD_FILE)\n\n let sessionLoadedCount = 0\n let sessionTotalCount = 0\n\n const sessionConfig = process.env[INFRA_KIT_ENV_CONFIG_VAR] ?? null\n const sessionProject = process.env[INFRA_KIT_ENV_PROJECT_VAR] ?? null\n const sessionLoadedAt = process.env[INFRA_KIT_ENV_LOADED_AT_VAR] ?? null\n\n if (sessionConfig) {\n const varNames = parseVarNamesFromEnvFile(envLoadPath)\n\n if (varNames.length > 0) {\n sessionTotalCount = varNames.length\n sessionLoadedCount = varNames.filter((v) => {\n return v in process.env\n }).length\n }\n\n const loadedAtDisplay = sessionLoadedAt?.replace(/\\.\\d{3}Z$/, '') ?? null\n\n logger.info(\n ` ${sessionConfig}: ${sessionLoadedCount} of ${sessionTotalCount} vars loaded (project: ${sessionProject}, loadedAt: ${loadedAtDisplay}, session: ${sessionId})\\n`,\n )\n\n if (sessionTotalCount > 0 && sessionLoadedCount < sessionTotalCount) {\n const missing = sessionTotalCount - sessionLoadedCount\n\n logger.warn(\n ` ${missing} cached var(s) are not present in the current process \u2014 env-load needs to be re-sourced, or vars were unset manually.`,\n )\n }\n } else {\n logger.info(` Session ${sessionId}: no env loaded\\n`)\n }\n\n const structuredContent = {\n sessionId,\n sessionLoadedCount,\n sessionTotalCount,\n sessionConfig,\n sessionProject,\n sessionLoadedAt,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const envStatusMcpTool = defineMcpTool({\n name: 'env-status',\n description:\n 'Report which Doppler project/config is currently loaded in the terminal session, when it was loaded, and how many variables are cached. Read-only \u2014 use env-load / env-clear to change the terminal session.',\n inputSchema: {},\n outputSchema: {\n sessionId: z.string().describe('Current terminal session ID'),\n sessionLoadedCount: z.number().describe('Number of cached vars active in the current session'),\n sessionTotalCount: z.number().describe('Total number of cached var names'),\n sessionConfig: z.string().nullable().describe('Doppler config name of the loaded session (environment name)'),\n sessionProject: z.string().nullable().describe('Doppler project name of the loaded session'),\n sessionLoadedAt: z.string().nullable().describe('ISO 8601 timestamp of when the env was loaded'),\n },\n handler: envStatus,\n})\n", "/* eslint-disable sonarjs/cognitive-complexity */\nimport checkbox from '@inquirer/checkbox'\nimport confirm from '@inquirer/confirm'\nimport process from 'node:process'\nimport { z } from 'zod'\nimport { $ } from 'zx'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { logger } from 'src/lib/logger'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions, releaseBranchLabels } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\ninterface GhMergeDevArgs extends RequiredConfirmedOptionArg {\n all?: boolean\n}\n\n/**\n * Merge dev into every release branch\n */\nexport const ghMergeDev = async (args: GhMergeDevArgs) => {\n const { all, confirmedCommand } = args\n\n commandEcho.start('merge-dev')\n\n // Only merge dev into regular releases (not hotfixes, which target main)\n const allPRs = await getReleasePRsWithInfo()\n const releasePRsList = allPRs\n .filter((pr) => {\n return detectReleaseType(pr.title) === 'regular'\n })\n .map((pr) => {\n return pr.branch\n })\n\n if (releasePRsList.length === 0) {\n logger.info('\u2139\uFE0F No open release branches found')\n\n commandEcho.print()\n\n return {\n content: textContent(\n JSON.stringify({ successfulMerges: 0, failedMerges: 0, failedBranches: [], totalBranches: 0 }, null, 2),\n ),\n structuredContent: { successfulMerges: 0, failedMerges: 0, failedBranches: [], totalBranches: 0 },\n }\n }\n\n let selectedReleaseBranches: string[] = []\n\n if (all) {\n selectedReleaseBranches = releasePRsList\n } else {\n commandEcho.setInteractive()\n\n const descriptions = await getJiraDescriptions()\n\n selectedReleaseBranches = await checkbox({\n required: true,\n message: '\uD83C\uDF3F Select release branches',\n choices: formatBranchChoices({ branches: releasePRsList, descriptions }),\n })\n }\n\n // Track --all flag if all branches were selected (either via flag or interactively)\n const allSelected = selectedReleaseBranches.length === releasePRsList.length\n\n if (allSelected) {\n commandEcho.addOption('--all', true)\n } else {\n commandEcho.addOption('--versions', releaseBranchLabels(selectedReleaseBranches))\n }\n\n // Validate input\n // if (selectedReleaseBranches.length === 0) {\n // console.error('No branches provided. Exiting...')\n // process.exit(1)\n // }\n\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: `Are you sure you want to merge dev into these branches: ${selectedReleaseBranches.join(', ')}?`,\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n if (!confirmedCommand) {\n commandEcho.addOption('--yes', true)\n }\n\n $.quiet = true\n\n await $`git fetch origin`\n await $`git switch dev`\n await $`git pull origin dev`\n\n const failedBranches: string[] = []\n\n // Merge dev into each branch\n for (const branch of selectedReleaseBranches) {\n const success = await mergeDev(branch)\n\n if (!success) {\n failedBranches.push(branch)\n }\n }\n\n $.quiet = false\n\n if (failedBranches.length > 0) {\n logger.info(`\\n\u26A0\uFE0F ${failedBranches.length} branch(es) failed to merge automatically.\\n`)\n logger.info('\uD83D\uDCCB Manual merge script for failed branches:')\n for (const branch of failedBranches) {\n logger.info(\n `# Merge dev into ${branch} and resolve conflicts if any \\n\\ngit switch ${branch} && git pull origin ${branch} && git merge origin/dev\\ngit push origin ${branch} && git switch dev\\n`,\n )\n }\n logger.info(\n `\u2705 ${selectedReleaseBranches.length - failedBranches.length}/${selectedReleaseBranches.length} merges completed successfully.`,\n )\n } else {\n logger.info('\u2705 All merges completed successfully!\\n')\n }\n\n commandEcho.print()\n\n const structuredContent = {\n successfulMerges: selectedReleaseBranches.length - failedBranches.length,\n failedMerges: failedBranches.length,\n failedBranches,\n totalBranches: selectedReleaseBranches.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\nconst mergeDev = async (branch: string): Promise<boolean> => {\n try {\n await $`git switch ${branch}`\n\n await $`git pull origin ${branch}`\n\n await $`git merge origin/dev --no-edit`\n\n await $`git push origin ${branch}`\n\n await $`git switch dev`\n\n logger.info(`Successfully merged dev into ${branch}`)\n\n return true\n } catch (error: unknown) {\n const err = new OperationError(error, {\n operation: `merge dev into ${branch}`,\n remediation: \"resolve conflicts manually or rerun after 'git fetch origin'\",\n })\n\n logger.error({ error, branch, msg: err.message })\n\n await $`git reset --merge HEAD~1`\n\n return false\n }\n}\n\n// MCP Tool Registration\nexport const ghMergeDevMcpTool = defineMcpTool({\n name: 'gh-merge-dev',\n description:\n 'Merge origin/dev into every open regular (non-hotfix) release branch and push the result. Mutates local git state and the remote release branches. When invoked via MCP, pass all=true \u2014 the branch picker is unreachable without a TTY, and the confirmation prompt is auto-skipped for MCP calls, so the caller is responsible for gating. Irreversible once pushed.',\n inputSchema: {\n all: z\n .boolean()\n .optional()\n .describe(\n 'Target every open regular release branch. Must be true for MCP calls (the interactive picker is unavailable without a TTY).',\n ),\n },\n outputSchema: {\n successfulMerges: z.number().describe('Number of successful merges'),\n failedMerges: z.number().describe('Number of failed merges'),\n failedBranches: z.array(z.string()).describe('List of branches that failed to merge'),\n totalBranches: z.number().describe('Total number of branches processed'),\n },\n handler: ghMergeDev,\n})\n", "import process from 'node:process'\nimport { $ } from 'zx'\n\nimport { logger } from 'src/lib/logger'\n\n/**\n * Validate GitHub CLI installation and authentication status and throw an error if not valid\n */\nexport const validateGitHubCliAndAuth = async () => {\n try {\n await $`gh --version`\n } catch (error: unknown) {\n logger.error({ error }, 'Error: GitHub CLI (gh) is not installed.')\n logger.error('Please install it from: https://cli.github.com/')\n process.exit(1)\n }\n\n try {\n await $`gh auth status`\n } catch (error: unknown) {\n logger.error({ error }, 'Error: GitHub CLI (gh) is not authenticated.')\n logger.error('Please authenticate it from: https://cli.github.com/manual/gh_auth_login or type \"gh auth login\"')\n process.exit(1)\n }\n}\n", "import process from 'node:process'\nimport { $ } from 'zx'\n\nimport { logger } from 'src/lib/logger'\nimport { compareReleaseIds, formatBranchName, formatPrTitle, parseBranchName } from 'src/lib/release-id'\nimport type { ReleaseId } from 'src/lib/release-id'\nimport { getBaseBranch } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\n\ninterface ReleasePR {\n headRefName: string\n number: number\n state: string\n title: string\n baseRefName: string\n createdAt: string\n}\n\nexport interface ReleasePRInfo {\n branch: string\n title: string\n createdAt: string\n}\n\n/**\n * Sort release head refs in the locked deterministic order (versions block\n * first by semver ascending, then names by PR creation date). Head refs that\n * are not valid release branches (parseBranchName \u2192 null) are filtered out\n * rather than throwing or NaN-sorting, so a stray junk branch can never break\n * discovery. Carries each PR's createdAt for name ordering.\n */\nconst sortReleasePRs = (prs: ReleasePR[]): ReleasePR[] => {\n return prs\n .map((pr) => {\n return { pr, id: parseBranchName(pr.headRefName) }\n })\n .filter((entry): entry is { pr: ReleasePR; id: NonNullable<typeof entry.id> } => {\n return entry.id !== null\n })\n .sort((a, b) => {\n return compareReleaseIds(a.id, b.id, { a: a.pr.createdAt, b: b.pr.createdAt })\n })\n .map((entry) => {\n return entry.pr\n })\n}\n\n/**\n * Fetch all open release/hotfix PRs from GitHub.\n * Searches both dev (regular) and main (hotfix) base branches.\n * Returns deduplicated ReleasePR objects.\n */\nconst fetchAllReleasePRs = async (): Promise<ReleasePR[]> => {\n const releasePRs =\n await $`gh pr list --search \"Release in:title\" --base dev --json number,title,headRefName,state,baseRefName,createdAt`\n\n const hotfixPRs =\n await $`gh pr list --search \"Hotfix in:title\" --base main --json number,title,headRefName,state,baseRefName,createdAt`\n\n const all: ReleasePR[] = [...JSON.parse(releasePRs.stdout), ...JSON.parse(hotfixPRs.stdout)]\n\n // Deduplicate by headRefName\n const seen = new Set<string>()\n\n return all.filter((pr) => {\n if (seen.has(pr.headRefName)) return false\n\n seen.add(pr.headRefName)\n\n return true\n })\n}\n\n/**\n * Fetch open release PRs from GitHub with 'Release' or 'Hotfix' in the title.\n * Returns an array of headRefName strings in the locked deterministic order\n * (version branches first by semver ascending, then named branches by PR\n * creation date). Unparseable head refs are filtered out.\n *\n * @returns [release/v1.18.22, release/v1.18.23, release/n/checkout-redesign]\n */\nexport const getReleasePRs = async (): Promise<string[]> => {\n try {\n const prs = await fetchAllReleasePRs()\n\n if (prs.length === 0) {\n logger.error('\u274C No release PRs found. Check the project folder for the script. Exiting...')\n\n process.exit(1)\n }\n\n return sortReleasePRs(prs).map((pr) => {\n return pr.headRefName\n })\n } catch (error) {\n logger.error({ error }, '\u274C Error fetching release PRs')\n\n process.exit(1)\n }\n}\n\n/**\n * Fetch open release PRs with title info (for detecting release type).\n * Returns ReleasePRInfo objects in the locked deterministic order (version\n * branches first by semver ascending, then named branches by PR creation\n * date). Unparseable head refs are filtered out.\n */\nexport const getReleasePRsWithInfo = async (): Promise<ReleasePRInfo[]> => {\n try {\n const prs = await fetchAllReleasePRs()\n\n if (prs.length === 0) {\n logger.error('\u274C No release PRs found. Check the project folder for the script. Exiting...')\n process.exit(1)\n }\n\n return sortReleasePRs(prs).map((pr) => {\n return {\n branch: pr.headRefName,\n title: pr.title,\n createdAt: pr.createdAt,\n }\n })\n } catch (error) {\n logger.error({ error }, '\u274C Error fetching release PRs')\n process.exit(1)\n }\n}\n\ninterface UpdateReleasePRBodyArgs {\n branch: string\n body: string\n}\n\n/**\n * Update the body of an open release PR identified by its head branch.\n */\nexport const updateReleasePRBody = async (args: UpdateReleasePRBodyArgs): Promise<void> => {\n const { branch, body } = args\n\n try {\n $.quiet = true\n await $`gh pr edit ${branch} --body ${body}`\n $.quiet = false\n } catch (error: unknown) {\n logger.error({ error, branch }, `Error updating release PR body for ${branch}`)\n throw error\n }\n}\n\ninterface CreateReleaseBranchArgs {\n id: ReleaseId\n jiraVersionUrl: string\n type: ReleaseType\n description?: string\n}\n\n// Function to create a release branch\nexport const createReleaseBranch = async (\n args: CreateReleaseBranchArgs,\n): Promise<{ branchName: string; prUrl: string }> => {\n const { id, jiraVersionUrl, type, description } = args\n const prTitle = formatPrTitle(id, type)\n const baseBranch = getBaseBranch(type)\n\n const branchName = formatBranchName(id)\n\n const body = description && description.trim() !== '' ? `${jiraVersionUrl}\\n\\n${description}` : `${jiraVersionUrl} \\n`\n\n try {\n $.quiet = true\n\n await $`git switch ${baseBranch}`\n await $`git pull origin ${baseBranch}`\n await $`git checkout -b ${branchName}`\n await $`git push -u origin ${branchName}`\n await $`git commit --allow-empty-message --allow-empty --message ''`\n await $`git push origin ${branchName}`\n\n // Create PR and capture URL\n const prResult = await $`gh pr create --title \"${prTitle}\" --body ${body} --base ${baseBranch} --head ${branchName}`\n\n const prLink = prResult.stdout.trim()\n\n await $`git switch ${baseBranch}`\n\n $.quiet = false\n\n return {\n branchName,\n prUrl: prLink,\n }\n } catch (error: unknown) {\n logger.error({ error, branchName }, `Error creating release branch ${branchName}`)\n\n throw error\n }\n}\n", "import { $ } from 'zx'\n\nimport { createReleaseBranch } from 'src/integrations/gh'\nimport { createJiraVersion, getProjectVersions, loadJiraConfigOptional } from 'src/integrations/jira'\nimport type { JiraConfig } from 'src/integrations/jira'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { displayLabel, formatBranchName, formatJiraName, parseBranchName, parseReleaseRef } from 'src/lib/release-id'\nimport type { ReleaseId } from 'src/lib/release-id'\n\n/** Sentinel ref for deploying from the `dev` branch instead of a release branch. */\nexport const DEV_REF = 'dev'\n\nexport type ReleaseType = 'regular' | 'hotfix'\n\n/**\n * Get the base branch for a release type.\n * Regular releases branch from/to dev, hotfixes branch from/to main.\n */\nexport const getBaseBranch = (type: ReleaseType): string => {\n return type === 'hotfix' ? 'main' : 'dev'\n}\n\nexport interface ReleaseCreationResult {\n version: string\n type: ReleaseType\n branchName: string\n prUrl: string\n jiraVersionUrl: string\n}\n\n/**\n * Prepare git repository for release creation\n * Fetches latest changes, switches to base branch, and pulls latest\n */\nexport const prepareGitForRelease = async (type: ReleaseType = 'regular'): Promise<void> => {\n const baseBranch = getBaseBranch(type)\n\n $.quiet = true\n\n await $`git fetch origin`\n await $`git switch ${baseBranch}`\n await $`git pull origin ${baseBranch}`\n\n $.quiet = false\n}\n\ninterface CreateSingleReleaseArgs {\n id: ReleaseId\n jiraConfig: JiraConfig\n description?: string\n type?: ReleaseType\n}\n\n/**\n * Create a single release by creating both Jira version and GitHub release branch\n */\nexport const createSingleRelease = async (args: CreateSingleReleaseArgs): Promise<ReleaseCreationResult> => {\n const { id, jiraConfig, description, type = 'regular' } = args\n // 1. Create Jira version (mandatory). For versioned releases this is\n // \"v1.2.3\" (byte-identical to before); for named releases it is \"<name>\".\n const versionName = formatJiraName(id)\n\n const result = await createJiraVersion(\n {\n name: versionName,\n projectId: jiraConfig.projectId,\n description: description || '',\n released: false,\n archived: false,\n },\n jiraConfig,\n )\n\n // Construct user-friendly Jira URL using project key from API response\n const jiraVersionUrl = `${jiraConfig.baseUrl}/projects/${result.version!.projectId}/versions/${result.version!.id}/tab/release-report-all-issues`\n\n // 2. Create GitHub release branch\n const releaseInfo = await createReleaseBranch({ id, jiraVersionUrl, type, description })\n\n return {\n version: displayLabel(id),\n type,\n branchName: releaseInfo.branchName,\n prUrl: releaseInfo.prUrl,\n jiraVersionUrl,\n }\n}\n\n/**\n * Fetch Jira version descriptions mapped by version name (e.g., \"v1.2.5\" \u2192 \"Some description\")\n * Gracefully returns empty map if Jira is unavailable\n */\nexport const getJiraDescriptions = async (): Promise<Map<string, string>> => {\n const descriptions = new Map<string, string>()\n\n const jiraConfig = await loadJiraConfigOptional()\n\n if (!jiraConfig) return descriptions\n\n try {\n const versions = await getProjectVersions(jiraConfig)\n\n for (const version of versions) {\n if (version.description) {\n descriptions.set(version.name, version.description)\n }\n }\n } catch {\n // Jira fetch failed, continue without descriptions\n }\n\n return descriptions\n}\n\n/**\n * Format a version string with its release type tag, e.g. \"1.2.5 [regular]\"\n * When maxVersionLength is provided, pads the version for alignment.\n */\nexport const formatVersionLabel = (version: string, type: ReleaseType, maxVersionLength?: number): string => {\n const padding = maxVersionLength ? ' '.repeat(maxVersionLength - version.length + 3) : ' '\n const tag = `[${type}]`.padEnd(11)\n\n return `${version}${padding}${tag}`\n}\n\n/**\n * Detect release type from PR title.\n * PRs titled \"Hotfix v...\" are hotfix, everything else is regular.\n */\nexport const detectReleaseType = (title: string): ReleaseType => {\n return title.toLowerCase().startsWith('hotfix') ? 'hotfix' : 'regular'\n}\n\ninterface FormatBranchChoicesArgs {\n branches: string[]\n descriptions: Map<string, string>\n types?: Map<string, ReleaseType>\n}\n\ninterface ParsedBranchChoice {\n branch: string\n id: ReleaseId\n /** Human display label: `1.2.3` | `<name>`. */\n label: string\n}\n\n/**\n * Parse branches into release ids, dropping any that do not parse (lenient\n * discovery source). Exported for unit testing the version/name/junk split.\n */\nexport const parseBranchChoices = (branches: string[]): ParsedBranchChoice[] => {\n return branches.flatMap((branch) => {\n const id = parseBranchName(branch)\n\n if (!id) return []\n\n return [{ branch, id, label: displayLabel(id) }]\n })\n}\n\n/**\n * Resolve an operator-supplied release ref (version `1.2.3` / `v1.2.3` or name\n * `checkout-redesign`) to its branch name (`release/v1.2.3` | `release/n/<name>`).\n * Strict: surfaces a parse failure as an OperationError with remediation text.\n */\nexport const resolveReleaseBranch = (versionArg: string): string => {\n try {\n return formatBranchName(parseReleaseRef(versionArg))\n } catch (error) {\n throw new OperationError(error, {\n operation: `resolve release ref \"${versionArg}\"`,\n remediation: 'pass a version (e.g. \"1.2.5\") or a release name (e.g. \"checkout-redesign\")',\n })\n }\n}\n\n/**\n * Render the human display label for a release branch. Returns the `dev`\n * sentinel unchanged; otherwise derives `1.2.3` | `<name>` from the branch.\n * Falls back to the raw branch when it does not parse as a release id.\n */\nexport const releaseLabelFromBranch = (branch: string): string => {\n if (branch === DEV_REF) return DEV_REF\n\n const id = parseBranchName(branch)\n\n return id ? displayLabel(id) : branch\n}\n\n/**\n * Render human display labels for a list of release branches, dropping any\n * branch that does not parse as a release id (lenient discovery contract).\n */\nexport const releaseBranchLabels = (branches: string[]): string[] => {\n return branches.flatMap((branch) => {\n const id = parseBranchName(branch)\n\n return id ? [displayLabel(id)] : []\n })\n}\n\n/**\n * Format release branch names as checkbox choices with aligned type tags and Jira descriptions\n */\nexport const formatBranchChoices = (args: FormatBranchChoicesArgs): { name: string; value: string }[] => {\n const { branches, descriptions, types } = args\n\n const parsed = parseBranchChoices(branches)\n\n const maxLen = Math.max(\n 0,\n ...parsed.map((p) => {\n return p.label.length\n }),\n )\n\n return parsed.map(({ branch, id, label }) => {\n const type = types ? types.get(branch) || 'regular' : undefined\n // Jira-descriptions map is keyed by the Jira version NAME (`v1.2.3` | `<name>`).\n const desc = descriptions.get(formatJiraName(id))\n const padding = ' '.repeat(maxLen - label.length + 3)\n\n let name = type ? formatVersionLabel(label, type, maxLen) : label\n\n if (desc) {\n name = type ? `${name} ${desc}` : `${label}${padding}${desc}`\n }\n\n return { name, value: branch }\n })\n}\n", "import process from 'node:process'\n\nimport { logger } from 'src/lib/logger'\n\nimport type {\n CreateJiraVersionParams,\n CreateJiraVersionResult,\n DeliverJiraReleaseParams,\n DeliverJiraReleaseResult,\n JiraConfig,\n JiraVersion,\n UpdateJiraVersionParams,\n UpdateJiraVersionResult,\n} from './types.js'\n\n/**\n * Creates a new version in Jira using the REST API\n * @param params - Version creation parameters\n * @param config - Jira configuration (baseUrl, token, projectId)\n * @returns Result containing created version or error\n */\nexport const createJiraVersion = async (\n params: CreateJiraVersionParams,\n config: JiraConfig,\n): Promise<CreateJiraVersionResult> => {\n try {\n const { baseUrl, token, email, projectId } = config\n\n // Use current date if not provided\n // const releaseDate =\n // params.releaseDate || new Date().toISOString().split('T')[0] // 2025-12-06\n\n // Prepare request body\n const requestBody = {\n name: params.name,\n projectId: params.projectId || projectId,\n description: params.description || '',\n // releaseDate,\n released: params.released || false,\n archived: params.archived || false,\n }\n\n // logger.info(\n // { version: params.name, projectId: requestBody.projectId },\n // 'Creating Jira version',\n // )\n\n // Make API request\n const url = `${baseUrl}/rest/api/3/version`\n\n // Create Basic auth credentials\n const credentials = btoa(`${email}:${token}`)\n\n const response = await fetch(url, {\n method: 'POST',\n headers: {\n Accept: 'application/json',\n 'Content-Type': 'application/json',\n Authorization: `Basic ${credentials}`,\n },\n body: JSON.stringify(requestBody),\n })\n\n if (!response.ok) {\n const errorText = await response.text()\n\n logger.error(\n {\n status: response.status,\n statusText: response.statusText,\n error: errorText,\n },\n 'Failed to create Jira version',\n )\n\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const version = (await response.json()) as JiraVersion\n\n // logger.info(\n // { versionId: version.id, versionName: version.name },\n // 'Successfully created Jira version',\n // )\n\n return {\n success: true,\n version,\n }\n } catch (error) {\n logger.error({ error }, 'Error creating Jira version')\n\n throw error\n }\n}\n\n/**\n * Gets all versions for a project from Jira\n * @param config - Jira configuration\n * @returns Array of JiraVersion objects\n */\nexport const getProjectVersions = async (config: JiraConfig): Promise<JiraVersion[]> => {\n try {\n const { baseUrl, token, email, projectId } = config\n\n const url = `${baseUrl}/rest/api/3/project/${projectId}/versions`\n const credentials = btoa(`${email}:${token}`)\n\n const response = await fetch(url, {\n method: 'GET',\n headers: {\n Accept: 'application/json',\n Authorization: `Basic ${credentials}`,\n },\n })\n\n if (!response.ok) {\n const errorText = await response.text()\n\n logger.error(\n {\n status: response.status,\n statusText: response.statusText,\n error: errorText,\n },\n 'Failed to get Jira project versions',\n )\n\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const versions = (await response.json()) as JiraVersion[]\n\n return versions\n } catch (error) {\n logger.error({ error }, 'Error getting Jira project versions')\n\n throw error\n }\n}\n\n/**\n * Finds a Jira version by name in the project\n * @param versionName - Name of the version to find (e.g., \"v1.33.10\")\n * @param config - Jira configuration\n * @returns JiraVersion if found, null otherwise\n */\nexport const findVersionByName = async (versionName: string, config: JiraConfig): Promise<JiraVersion | null> => {\n try {\n const versions = await getProjectVersions(config)\n const version = versions.find((v) => {\n return v.name === versionName\n })\n\n return version || null\n } catch (error) {\n logger.error({ error, versionName }, 'Error finding Jira version by name')\n\n throw error\n }\n}\n\n/**\n * Updates an existing Jira version\n * @param params - Update parameters\n * @param config - Jira configuration\n * @returns Result containing updated version or error\n */\nexport const updateJiraVersion = async (\n params: UpdateJiraVersionParams,\n config: JiraConfig,\n): Promise<UpdateJiraVersionResult> => {\n try {\n const { baseUrl, token, email } = config\n\n // Only include fields the caller explicitly passed.\n const requestBody: Record<string, any> = {}\n\n if (params.released !== undefined) requestBody.released = params.released\n if (params.archived !== undefined) requestBody.archived = params.archived\n if (params.releaseDate !== undefined) requestBody.releaseDate = params.releaseDate\n if (params.description !== undefined) requestBody.description = params.description\n\n const url = `${baseUrl}/rest/api/3/version/${params.versionId}`\n const credentials = btoa(`${email}:${token}`)\n\n const response = await fetch(url, {\n method: 'PUT',\n headers: {\n Accept: 'application/json',\n 'Content-Type': 'application/json',\n Authorization: `Basic ${credentials}`,\n },\n body: JSON.stringify(requestBody),\n })\n\n if (!response.ok) {\n const errorText = await response.text()\n\n logger.error(\n {\n status: response.status,\n statusText: response.statusText,\n error: errorText,\n },\n 'Failed to update Jira version',\n )\n\n throw new Error(`HTTP ${response.status}: ${response.statusText}`)\n }\n\n const version = (await response.json()) as JiraVersion\n\n return {\n success: true,\n version,\n }\n } catch (error) {\n logger.error({ error }, 'Error updating Jira version')\n\n throw error\n }\n}\n\n/**\n * Delivers a Jira release by marking it as released with the current date\n * @param params - Parameters containing the version name\n * @param config - Jira configuration\n * @returns Result containing updated version\n * @throws Error if version not found or update fails\n */\nexport const deliverJiraRelease = async (\n params: DeliverJiraReleaseParams,\n config: JiraConfig,\n): Promise<DeliverJiraReleaseResult> => {\n try {\n const { versionName } = params\n\n // Find the version by name\n const version = await findVersionByName(versionName, config)\n\n if (!version) {\n logger.error({ versionName }, 'Jira version not found')\n throw new Error(`Version \"${versionName}\" not found in Jira project`)\n }\n\n // Update the version to mark it as released\n const result = await updateJiraVersion(\n {\n versionId: version.id,\n released: true,\n releaseDate: new Date().toISOString().split('T')[0], // Current date in YYYY-MM-DD format\n },\n config,\n )\n\n return result\n } catch (error) {\n logger.error({ error }, 'Error delivering Jira release')\n throw error\n }\n}\n\n/**\n * Loads Jira configuration from environment variables\n * @throws Error with detailed message if configuration is missing or invalid\n * @returns Promise<JiraConfig>\n */\nexport const loadJiraConfig = async (): Promise<JiraConfig> => {\n const baseUrl = process.env.JIRA_BASE_URL\n const token = process.env.JIRA_TOKEN || process.env.JIRA_API_TOKEN\n const projectIdStr = process.env.JIRA_PROJECT_ID\n const email = process.env.JIRA_EMAIL\n\n const missingVars: string[] = []\n\n if (!baseUrl) missingVars.push('JIRA_BASE_URL (e.g., https://your-domain.atlassian.net)')\n if (!token) missingVars.push('JIRA_TOKEN or JIRA_API_TOKEN (your Jira API token)')\n if (!projectIdStr) missingVars.push('JIRA_PROJECT_ID (numeric project ID)')\n if (!email) missingVars.push('JIRA_EMAIL (your Jira email address)')\n\n if (missingVars.length > 0) {\n const errorMessage = [\n 'Jira configuration is required but incomplete.',\n 'Please configure the following environment variables:',\n ...missingVars.map((v) => {\n return ` - ${v}`\n }),\n '',\n 'You can set these in your .env file or as environment variables.',\n ].join('\\n')\n\n throw new Error(errorMessage)\n }\n\n const projectId = Number.parseInt(projectIdStr!, 10)\n\n if (Number.isNaN(projectId)) {\n throw new TypeError(`Invalid JIRA_PROJECT_ID: \"${projectIdStr}\" must be a numeric value (e.g., 10001)`)\n }\n\n return {\n baseUrl: baseUrl!.replace(/\\/$/, ''), // Remove trailing slash\n token: token!,\n projectId,\n email: email!,\n }\n}\n\n/**\n * Attempts to load Jira configuration from environment variables\n * Returns null if configuration is missing or invalid (for optional Jira integration)\n * @returns Promise<JiraConfig | null>\n */\nexport const loadJiraConfigOptional = async (): Promise<JiraConfig | null> => {\n try {\n const config = await loadJiraConfig()\n\n return config\n } catch (error) {\n logger.warn({ error }, 'Jira configuration not available, skipping Jira integration')\n\n return null\n }\n}\n", "const STDERR_EXCERPT_MAX_BYTES = 200\n\nexport interface OperationErrorContext {\n operation: string\n remediation?: string\n stderrExcerpt?: string\n}\n\n/**\n * Duck-typed read of zx's `ProcessOutput.stderr` (and similar shapes) without\n * importing zx types just for an `instanceof` check.\n *\n * @example\n * extractStderr(new Error('x')) // undefined\n * extractStderr({ stderr: 'fatal: ...' }) // 'fatal: ...'\n * extractStderr({ stderr: '' }) // undefined (empty treated as missing)\n */\nconst extractStderr = (cause: unknown): string | undefined => {\n if (cause === null || typeof cause !== 'object') return undefined\n const stderr = (cause as { stderr?: unknown }).stderr\n\n return typeof stderr === 'string' && stderr.length > 0 ? stderr : undefined\n}\n\n/**\n * Compose the human-and-agent-readable message body for an `OperationError`:\n * `\"failed to <operation> [\u2014 stderr: <excerpt>] [\u2014 try: <remediation>]\"`.\n * `stderrExcerpt` overrides anything duck-typed off `cause`; both are trimmed\n * and capped at {@link STDERR_EXCERPT_MAX_BYTES} so a runaway subprocess can't\n * blow up the message.\n */\nconst buildMessage = (cause: unknown, ctx: OperationErrorContext): string => {\n const stderr = ctx.stderrExcerpt ?? extractStderr(cause)\n const parts = [`failed to ${ctx.operation}`]\n\n if (stderr) parts.push(`stderr: ${stderr.slice(0, STDERR_EXCERPT_MAX_BYTES).trim()}`)\n if (ctx.remediation) parts.push(`try: ${ctx.remediation}`)\n\n return parts.join(' \u2014 ')\n}\n\n/**\n * Error type for any handler-level failure that should surface to the caller\n * (CLI user or MCP-connected agent) with a remediation hint. Wraps an\n * underlying cause and renders a single-line, structured message so logs and\n * agent tool-result text stay scannable.\n *\n * Pattern modeled on the exemplary Doppler errors in\n * `src/integrations/doppler/doppler-cli-auth.ts`.\n *\n * @example\n * // wrap a zx subprocess failure\n * try {\n * await $`git worktree add ${path} ${branch}`\n * } catch (err) {\n * throw new OperationError(err, {\n * operation: `git worktree add for ${branch}`,\n * remediation: 'check the branch name and that the parent dir is writable',\n * })\n * }\n *\n * @example\n * // validation failure with no underlying cause\n * throw new OperationError(undefined, {\n * operation: 'launch deploy-all workflow',\n * remediation: `pass one of: ${environments.join(', ')}`,\n * stderrExcerpt: `invalid environment: ${selectedEnv}`,\n * })\n */\nexport class OperationError extends Error {\n readonly operation: string\n readonly remediation?: string\n\n constructor(cause: unknown, ctx: OperationErrorContext) {\n super(buildMessage(cause, ctx), { cause })\n this.name = 'OperationError'\n this.operation = ctx.operation\n this.remediation = ctx.remediation\n }\n}\n", "import confirm from '@inquirer/confirm'\nimport select from '@inquirer/select'\nimport process from 'node:process'\nimport { z } from 'zod'\nimport { $ } from 'zx'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { deliverJiraRelease, loadJiraConfigOptional } from 'src/integrations/jira'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { formatZxError } from 'src/lib/errors/format-zx-error'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { logger } from 'src/lib/logger'\nimport { displayLabel, formatJiraName, formatRcTitle, parseBranchName } from 'src/lib/release-id'\nimport type { ReleaseId } from 'src/lib/release-id'\nimport {\n detectReleaseType,\n formatBranchChoices,\n getJiraDescriptions,\n resolveReleaseBranch,\n} from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { removeWorktrees } from 'src/lib/worktrees'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\ninterface GhReleaseDeliverArgs extends RequiredConfirmedOptionArg {\n version: string\n}\n\ntype PRState = 'OPEN' | 'MERGED' | 'CLOSED'\n\ninterface PRStatus {\n number: number\n state: PRState\n title: string\n}\n\n/**\n * Wrap a delivery step so its failure logs structured zx fields and surfaces\n * an `OperationError` whose message names the actual step that failed \u2014\n * instead of the previous blanket \"merging release branch into dev\" message.\n */\nconst runStep = async <T>(operation: string, remediation: string, fn: () => Promise<T>): Promise<T> => {\n try {\n return await fn()\n } catch (error) {\n logger.error({ err: formatZxError(error) }, `\u274C Failed to ${operation}`)\n throw new OperationError(error, { operation, remediation })\n }\n}\n\n/**\n * Fetch the (most-recent) PR for the given head branch, across all states, so\n * we can resume a partially-completed delivery: a PR merged on a prior attempt\n * still appears here as `state: 'MERGED'`, letting the caller skip the merge.\n */\nconst fetchPRByHead = async (head: string): Promise<PRStatus | null> => {\n const result = await $`gh pr list --head ${head} --state all --json number,state,title --limit 1`\n const prs = JSON.parse(result.stdout) as PRStatus[]\n\n return prs[0] ?? null\n}\n\n/**\n * Find a MERGED RC PR (dev \u2192 main) whose title matches the given version, used\n * to detect that a prior delivery run already merged the RC and we should skip\n * the merge step on resume. Title-matched on purpose: an older MERGED RC PR\n * from a different release must not short-circuit this version's flow.\n */\nconst fetchMergedRcPRForVersion = async (id: ReleaseId): Promise<PRStatus | null> => {\n const expectedTitle = formatRcTitle(id)\n const result = await $`gh pr list --head dev --base main --state merged --json number,state,title --limit 20`\n const prs = JSON.parse(result.stdout) as PRStatus[]\n const match = prs.find((pr) => {\n return pr.title === expectedTitle\n })\n\n return match ?? null\n}\n\n/**\n * Find an open dev \u2192 main PR, if any. GitHub allows at most one open PR per\n * head/base pair, so an existing open PR here is the RC PR for this release \u2014\n * even if its title was set by a previous (failed) delivery run. Adopting it\n * is what makes the flow recoverable after a mid-run failure.\n */\nconst fetchOpenDevToMainPR = async (): Promise<PRStatus | null> => {\n const result = await $`gh pr list --head dev --base main --state open --json number,state,title --limit 5`\n const prs = JSON.parse(result.stdout) as PRStatus[]\n\n return prs[0] ?? null\n}\n\ninterface ResolvedTarget {\n selectedReleaseBranch: string\n releasePrTitle: string\n}\n\nconst resolveTargetFromVersion = async (version: string): Promise<ResolvedTarget> => {\n const selectedReleaseBranch = resolveReleaseBranch(version)\n const pr = await fetchPRByHead(selectedReleaseBranch)\n\n if (!pr) {\n logger.error(`\u274C No PR found for branch ${selectedReleaseBranch}.`)\n throw new OperationError(undefined, {\n operation: `deliver release ${selectedReleaseBranch}`,\n remediation: `confirm a PR exists ('gh pr list --head ${selectedReleaseBranch} --state all')`,\n })\n }\n\n return { selectedReleaseBranch, releasePrTitle: pr.title }\n}\n\nconst resolveTargetInteractively = async (): Promise<ResolvedTarget> => {\n const releasePRsInfo = await getReleasePRsWithInfo()\n\n const branches = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n commandEcho.setInteractive()\n\n const descriptions = await getJiraDescriptions()\n\n const selectedReleaseBranch = await select({\n message: '\uD83C\uDF3F Select release branch',\n choices: formatBranchChoices({ branches, descriptions, types: releaseTypes }),\n })\n\n const prInfo = releasePRsInfo.find((pr) => {\n return pr.branch === selectedReleaseBranch\n })\n\n if (!prInfo) {\n logger.error(`\u274C Release branch ${selectedReleaseBranch} not found in open PRs.`)\n throw new OperationError(undefined, {\n operation: `deliver release ${selectedReleaseBranch}`,\n remediation: `confirm an open PR exists for ${selectedReleaseBranch} ('gh pr list')`,\n })\n }\n\n return { selectedReleaseBranch, releasePrTitle: prInfo.title }\n}\n\n/**\n * `gh pr merge --delete-branch` also deletes the local branch, which fails if a\n * worktree has it checked out (the actual root cause of the \"Failed to merge\n * release PR\" surface error). Pre-remove any worktree for the release branch\n * so the local delete can succeed.\n */\nconst removeReleaseWorktreeIfPresent = async (releaseBranch: string): Promise<void> => {\n const worktreeBranches = await getCurrentWorktrees('release')\n\n if (!worktreeBranches.includes(releaseBranch)) return\n\n const [projectRoot, repoName] = await Promise.all([getProjectRoot(), getRepoName()])\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n\n const removed = await removeWorktrees({ branches: [releaseBranch], worktreeDir, repoName })\n\n if (removed.length === 0) {\n throw new OperationError(undefined, {\n operation: `remove worktree for ${releaseBranch} before merge`,\n remediation: `run manually: git worktree remove ${worktreeDir}/${releaseBranch} (use --force if uncommitted changes)`,\n })\n }\n}\n\ninterface MergeReleasePRArgs {\n selectedReleaseBranch: string\n releaseType: ReleaseType\n}\n\nconst mergeReleasePR = async (args: MergeReleasePRArgs): Promise<void> => {\n const { selectedReleaseBranch, releaseType } = args\n\n const mergeTarget = releaseType === 'hotfix' ? 'main' : 'dev'\n\n const releasePr = await fetchPRByHead(selectedReleaseBranch)\n\n if (!releasePr) {\n throw new OperationError(undefined, {\n operation: `look up release PR for ${selectedReleaseBranch}`,\n remediation: `verify the PR exists in GitHub`,\n })\n }\n\n if (releasePr.state === 'MERGED') {\n logger.info(`\u2713 Release PR ${selectedReleaseBranch} already merged \u2014 skipping`)\n\n return\n }\n\n if (releasePr.state === 'CLOSED') {\n throw new OperationError(undefined, {\n operation: `merge release PR ${selectedReleaseBranch} into ${mergeTarget}`,\n remediation: `the PR is closed without merge; reopen it or create a new release`,\n })\n }\n\n await runStep(\n `merge release PR ${selectedReleaseBranch} into ${mergeTarget}`,\n `check 'gh pr view ${selectedReleaseBranch}' for mergeability and required reviews`,\n async () => {\n await $`gh pr merge ${selectedReleaseBranch} --squash --admin --delete-branch`\n },\n )\n}\n\nconst resolveRcPRNumber = async (id: ReleaseId): Promise<number> => {\n const selectedLabel = displayLabel(id)\n const expectedTitle = formatRcTitle(id)\n const existingOpen = await fetchOpenDevToMainPR()\n\n // Adopt any existing open dev\u2192main PR. GitHub permits only one open PR per\n // head/base pair, so a stale open RC PR (left behind by a prior failed run\n // \u2014 the single most common cause of \"Error merging release branch into\n // dev\") blocks `gh pr create`. Retitle it instead of fighting it.\n if (existingOpen) {\n const rcNumber = existingOpen.number\n\n if (existingOpen.title !== expectedTitle) {\n logger.info(\n `Adopting open dev \u2192 main PR #${rcNumber} (\"${existingOpen.title}\") and retitling for ${selectedLabel}`,\n )\n await runStep(\n `retitle dev \u2192 main PR #${rcNumber} to \"${expectedTitle}\"`,\n `update manually: gh pr edit ${rcNumber} --title \"${expectedTitle}\"`,\n async () => {\n await $`gh pr edit ${rcNumber} --title ${expectedTitle}`\n },\n )\n }\n\n return rcNumber\n }\n\n await runStep(\n `create RC PR (dev \u2192 main) for ${selectedLabel}`,\n `run 'gh pr create --base main --head dev' manually to surface the underlying error (e.g. no commits between dev and main)`,\n async () => {\n await $`gh pr create --base main --head dev --title ${expectedTitle} --body \"\"`\n },\n )\n\n const created = await fetchOpenDevToMainPR()\n\n if (!created) {\n throw new OperationError(undefined, {\n operation: `look up RC PR for ${selectedLabel}`,\n remediation: `verify the RC PR was created ('gh pr list --head dev --base main')`,\n })\n }\n\n return created.number\n}\n\nconst ensureRcPRMerged = async (id: ReleaseId): Promise<void> => {\n const selectedLabel = displayLabel(id)\n const alreadyMerged = await fetchMergedRcPRForVersion(id)\n\n if (alreadyMerged) {\n logger.info(`\u2713 RC PR for ${selectedLabel} already merged into main \u2014 skipping`)\n\n return\n }\n\n const rcNumber = await resolveRcPRNumber(id)\n\n await runStep(\n `merge RC PR #${rcNumber} (dev \u2192 main) for ${selectedLabel}`,\n `check 'gh pr view ${rcNumber}' for mergeability and required reviews`,\n async () => {\n await $`gh pr merge ${rcNumber} --squash --admin`\n },\n )\n}\n\nconst dispatchDeployWorkflow = async (): Promise<void> => {\n $.quiet = false\n\n await runStep(\n `dispatch deploy-all workflow on main`,\n `check 'gh workflow list' and that you have permission to dispatch deploy-all.yml`,\n async () => {\n await $`gh workflow run deploy-all.yml --ref main -f environment=prod`\n },\n )\n\n $.quiet = true\n}\n\nconst syncMainIntoDev = async (): Promise<void> => {\n await runStep(\n `sync main back into dev`,\n `run manually: git switch main && git pull && git switch dev && git pull && git merge main --no-edit && git push`,\n async () => {\n await $`git switch main && git pull && git switch dev && git pull && git merge main --no-edit && git push`\n },\n )\n}\n\nconst deliverJiraReleaseSafely = async (id: ReleaseId): Promise<void> => {\n const jiraConfig = await loadJiraConfigOptional()\n\n if (!jiraConfig) {\n logger.info('\uD83D\uDD14 Jira is not configured, skipping Jira release delivery')\n\n return\n }\n\n try {\n // Jira fix version name: `v1.2.3` | `<name>` \u2014 must match create-time formatJiraName.\n const versionName = formatJiraName(id)\n\n await deliverJiraRelease({ versionName }, jiraConfig)\n } catch (error) {\n logger.error({ err: formatZxError(error) }, 'Failed to deliver Jira release (non-blocking)')\n }\n}\n\n/**\n * Deliver a release branch to production. Each network/git step is run inside\n * `runStep` so the surfaced error names the failing operation and includes the\n * subprocess stderr. PR-merge steps are idempotent: if the release PR or RC PR\n * is already MERGED, the step is skipped, so re-running after a mid-flight\n * failure picks up where it stopped.\n */\nexport const ghReleaseDeliver = async (args: GhReleaseDeliverArgs) => {\n const { version, confirmedCommand } = args\n\n commandEcho.start('release-deliver')\n\n const { selectedReleaseBranch, releasePrTitle } = version\n ? await resolveTargetFromVersion(version)\n : await resolveTargetInteractively()\n\n // selectedReleaseBranch is always a release branch (operator ref strictly\n // parsed, or picked from discovery-filtered choices) so this cannot be null.\n const releaseId = parseBranchName(selectedReleaseBranch)\n\n if (!releaseId) {\n throw new OperationError(undefined, {\n operation: `deliver release ${selectedReleaseBranch}`,\n remediation: 'pass a version (e.g. \"1.2.5\") or a release name (e.g. \"checkout-redesign\")',\n })\n }\n\n const selectedVersion = displayLabel(releaseId)\n\n commandEcho.addOption('--version', selectedVersion)\n logger.info(`Delivering ${releaseId.kind === 'name' ? 'named release' : 'version'} ${selectedReleaseBranch}`)\n\n const releaseType: ReleaseType = detectReleaseType(releasePrTitle)\n\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: `Are you sure you want to deliver version ${selectedReleaseBranch} to production?`,\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n commandEcho.addOption('--yes', true)\n\n $.quiet = true\n\n await removeReleaseWorktreeIfPresent(selectedReleaseBranch)\n await mergeReleasePR({ selectedReleaseBranch, releaseType })\n\n if (releaseType !== 'hotfix') {\n await ensureRcPRMerged(releaseId)\n }\n\n await dispatchDeployWorkflow()\n await syncMainIntoDev()\n\n $.quiet = false\n\n await deliverJiraReleaseSafely(releaseId)\n\n logger.info(`Successfully delivered ${selectedReleaseBranch} to production!`)\n\n commandEcho.print()\n\n const structuredContent = {\n releaseBranch: selectedReleaseBranch,\n version: selectedVersion,\n type: releaseType,\n success: true,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const ghReleaseDeliverMcpTool = defineMcpTool({\n name: 'gh-release-deliver',\n description:\n 'Deliver a release to production. For hotfixes: squash-merges the release branch to main and dispatches the deploy-all workflow. For regular releases: squash-merges to dev, opens an RC PR, merges dev into main, dispatches the deploy-all workflow, then syncs main back to dev. Also releases the matching Jira fix version if Jira is configured. Dispatches the deploy workflow fire-and-forget \u2014 the tool returns once the workflow is accepted by GitHub, not when the deployment finishes. PR-merge steps are idempotent: re-running after a partial failure skips PRs that are already merged. Irreversible production operation: the confirmation prompt is auto-skipped for MCP calls, so the caller is responsible for gating. \"version\" is required when invoked via MCP (the picker is unreachable without a TTY).',\n inputSchema: {\n version: z\n .string()\n .describe(\n 'Accepts a release version (e.g. \"1.2.5\") OR a release name (e.g. \"checkout-redesign\") to deliver to production. Required for MCP calls.',\n ),\n },\n outputSchema: {\n releaseBranch: z.string().describe('The release branch that was delivered'),\n version: z.string().describe('The version that was delivered'),\n type: z.enum(['regular', 'hotfix']).describe('Release type'),\n success: z.boolean().describe('Whether the delivery was successful'),\n },\n handler: ghReleaseDeliver,\n})\n", "const STDERR_EXCERPT_MAX_BYTES = 500\nconst STDOUT_EXCERPT_MAX_BYTES = 200\n\nexport interface ZxErrorFields {\n exitCode?: number | null\n stderr?: string\n stdout?: string\n message?: string\n name?: string\n}\n\nconst readTrimmedString = (value: unknown, max: number): string | undefined => {\n if (typeof value !== 'string' || value.length === 0) return undefined\n\n return value.slice(-max).trim()\n}\n\n/**\n * Extract loggable fields from an error for `logger.error({ err: formatZxError(e) }, ...)`.\n *\n * pino's default `err` serializer handles `Error` subclasses but renders zx's\n * `ProcessOutput` as `{}` because its informative fields (`stderr`, `stdout`,\n * `exitCode`) are non-enumerable / on the prototype. This helper duck-types\n * those fields so subprocess failures surface in logs instead of vanishing.\n */\nexport const formatZxError = (error: unknown): ZxErrorFields => {\n if (error === null || typeof error !== 'object') {\n return { message: String(error) }\n }\n\n const rec = error as Record<string, unknown>\n const fields: ZxErrorFields = {}\n\n if (error instanceof Error) {\n fields.name = error.name\n fields.message = error.message\n } else if (typeof rec.message === 'string') {\n fields.message = rec.message\n }\n\n const exitCode = rec.exitCode\n\n if (typeof exitCode === 'number' || exitCode === null) fields.exitCode = exitCode\n\n const stderr = readTrimmedString(rec.stderr, STDERR_EXCERPT_MAX_BYTES)\n\n if (stderr) fields.stderr = stderr\n\n const stdout = readTrimmedString(rec.stdout, STDOUT_EXCERPT_MAX_BYTES)\n\n if (stdout) fields.stdout = stdout\n\n return fields\n}\n", "import { $ } from 'zx'\n\nimport { buildCmuxWorkspaceTitle, closeCmuxWorkspaceByTitle } from 'src/integrations/cmux'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { logger } from 'src/lib/logger'\n\ninterface RemoveWorktreesArgs {\n branches: string[]\n worktreeDir: string\n repoName: string\n pruneFolder?: boolean\n}\n\n/**\n * Close any cmux workspace for each branch and run `git worktree remove`,\n * returning the branches that were removed cleanly. Failures are logged but\n * never thrown, so a single bad worktree doesn't poison a batch removal.\n *\n * When `pruneFolder` is true and every branch was removed, also prune the\n * worktree metadata and delete the worktrees folder \u2014 used by the\n * `worktrees-remove` \"all\" path to leave the filesystem clean.\n */\nexport const removeWorktrees = async (args: RemoveWorktreesArgs): Promise<string[]> => {\n const { branches, worktreeDir, repoName, pruneFolder = false } = args\n\n const results = await Promise.allSettled(\n branches.map(async (branch) => {\n const worktreePath = `${worktreeDir}/${branch}`\n\n const title = buildCmuxWorkspaceTitle({ repoName, branch })\n\n await closeCmuxWorkspaceByTitle(title)\n\n await $`git worktree remove ${worktreePath}`\n\n return branch\n }),\n )\n\n const removed: string[] = []\n\n for (const [index, result] of results.entries()) {\n if (result.status === 'fulfilled') {\n removed.push(result.value)\n } else {\n const branch = branches[index]\n const err = new OperationError(result.reason, {\n operation: `remove worktree for ${branch}`,\n remediation: \"check 'git worktree list' for the path; uncommitted changes block removal\",\n })\n\n logger.error({ error: result.reason, msg: err.message })\n }\n }\n\n if (pruneFolder && removed.length === branches.length) {\n await $`git worktree prune`\n await $`rm -rf ${worktreeDir}`\n\n logger.info(`\uD83D\uDDD1\uFE0F Removed worktree folder: ${worktreeDir}`)\n logger.info('')\n }\n\n return removed\n}\n", "import { $ } from 'zx'\n\nimport { logger } from 'src/lib/logger'\n\n/**\n * Best-effort close of the cmux workspace whose title exactly matches `title`.\n * Silently no-ops if cmux isn't running, the workspace isn't found, or close fails.\n */\nexport const closeCmuxWorkspaceByTitle = async (title: string): Promise<void> => {\n try {\n const listOutput = (await $`cmux list-workspaces`.quiet()).stdout\n\n const ref = findWorkspaceRefByTitle(listOutput, title)\n\n if (!ref) {\n return\n }\n\n await $`cmux close-workspace --workspace ${ref}`.quiet()\n } catch (error) {\n logger.debug({ error, title }, 'cmux: skipped closing workspace')\n }\n}\n\n/**\n * Parses `cmux list-workspaces` output and returns the workspace ref whose\n * title exactly matches `title`, or undefined if no match.\n *\n * Each line looks like:\n * \" workspace:8 hulyo-monorepo v1.48.0\"\n * \"* workspace:6 obsidian-workspace [selected]\"\n */\nconst findWorkspaceRefByTitle = (output: string, title: string): string | undefined => {\n for (const rawLine of output.split('\\n')) {\n // eslint-disable-next-line sonarjs/slow-regex, regexp/no-super-linear-backtracking\n const match = rawLine.match(/^[* ]\\s*(workspace:\\d+)\\s+(.+?)(?:\\s+\\[selected\\])?\\s*$/)\n\n if (!match) {\n continue\n }\n\n const ref = match[1]\n const lineTitle = match[2]?.trim() ?? ''\n\n if (lineTitle === title) {\n return ref\n }\n }\n\n return undefined\n}\n", "import { $ } from 'zx'\n\nimport { logger } from 'src/lib/logger'\n\n/**\n * Returns the set of titles for all currently-open cmux workspaces.\n * Returns an empty set if cmux isn't running, the call fails, or the\n * output can't be parsed \u2014 callers should treat \"empty\" as \"unknown,\n * proceed as if nothing is open\".\n *\n * Each line of `cmux list-workspaces` looks like:\n * \" workspace:8 hulyo-monorepo v1.48.0\"\n * \"* workspace:6 obsidian-workspace [selected]\"\n */\nexport const listCmuxWorkspaceTitles = async (): Promise<Set<string>> => {\n try {\n const output = (await $`cmux list-workspaces`.quiet()).stdout\n\n const titles = new Set<string>()\n\n for (const rawLine of output.split('\\n')) {\n // eslint-disable-next-line sonarjs/slow-regex, regexp/no-super-linear-backtracking\n const match = rawLine.match(/^[* ]\\s*workspace:\\d+\\s+(.+?)(?:\\s+\\[selected\\])?\\s*$/)\n\n if (!match) {\n continue\n }\n\n const title = match[1]?.trim()\n\n if (title) {\n titles.add(title)\n }\n }\n\n return titles\n } catch (error) {\n logger.debug({ error }, 'cmux: skipped listing workspace titles')\n\n return new Set()\n }\n}\n", "import { $ } from 'zx'\n\ninterface OpenCmuxWorkspaceArgs {\n cwd: string\n title?: string\n}\n\n/**\n * Opens a new cmux workspace rooted at `cwd` with three panes:\n * left-top (primary) | right (full-height)\n * left-bottom |\n * All panes inherit `cwd` from the workspace.\n */\nexport const openCmuxWorkspaceWithLayout = async (args: OpenCmuxWorkspaceArgs): Promise<void> => {\n const { cwd, title } = args\n\n const newWorkspaceOutput = (await $`cmux workspace create --cwd ${cwd}`).stdout\n\n const workspaceRef = parseWorkspaceRef(newWorkspaceOutput)\n\n const surfacesOutput = (await $`cmux list-pane-surfaces --workspace ${workspaceRef}`).stdout\n\n const leftTopRef = parseFirstSurfaceRef(surfacesOutput)\n\n await $`cmux new-split right --workspace ${workspaceRef} --surface ${leftTopRef}`\n await $`cmux new-split down --workspace ${workspaceRef} --surface ${leftTopRef}`\n\n if (title) {\n await $`cmux workspace rename --workspace ${workspaceRef} ${title}`\n }\n}\n\n/**\n * Extracts the first `surface:<id>` reference from the output of\n * `cmux list-pane-surfaces`. Used to locate the initial (primary) pane\n * surface so subsequent splits can be anchored relative to it.\n *\n * @example\n * const output = 'surface:12 (active)\\nsurface:13\\n'\n * parseFirstSurfaceRef(output) // => 'surface:12'\n */\nconst parseFirstSurfaceRef = (output: string): string => {\n const match = output.match(/surface:\\d+/)\n\n if (!match) {\n throw new Error('cmux: could not locate initial surface in list-pane-surfaces output')\n }\n\n return match[0]\n}\n\n/**\n * Extracts the `workspace:<id>` reference from the output of\n * `cmux workspace create`. The returned ref is used to target the newly\n * created workspace in follow-up `cmux` commands (splits, rename, etc.).\n *\n * @example\n * const output = 'created workspace:7\\n'\n * parseWorkspaceRef(output) // => 'workspace:7'\n */\nconst parseWorkspaceRef = (output: string): string => {\n const match = output.match(/workspace:\\d+/)\n\n if (!match) {\n throw new Error('cmux: could not locate workspace ref in workspace create output')\n }\n\n return match[0]\n}\n", "import { displayLabel, parseBranchName } from 'src/lib/release-id'\n\ninterface BuildCmuxWorkspaceTitleArgs {\n repoName: string\n branch: string\n}\n\n/**\n * Builds the cmux workspace title used by `worktrees-add` and looked up by\n * `worktrees-remove`. Release branches are rendered via their release-id\n * display label so the title reads e.g. `\"hulyo-monorepo 1.48.0\"` for\n * `\"release/v1.48.0\"` and `\"hulyo-monorepo checkout-redesign\"` for\n * `\"release/n/checkout-redesign\"`. Non-release branches (cmux titles them too)\n * fall back to the raw branch string.\n */\nexport const buildCmuxWorkspaceTitle = (args: BuildCmuxWorkspaceTitleArgs): string => {\n const { repoName, branch } = args\n\n const id = parseBranchName(branch)\n const label = id ? displayLabel(id) : branch\n\n return `${repoName} ${label}`\n}\n", "import select from '@inquirer/select'\nimport { z } from 'zod'\nimport { $ } from 'zx'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport {\n detectReleaseType,\n formatBranchChoices,\n getJiraDescriptions,\n releaseLabelFromBranch,\n resolveReleaseBranch,\n} from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface GhReleaseDeployAllArgs {\n version: string\n env: string\n skipTerraform?: boolean\n}\n\n/**\n * Deploy a release branch to an environment\n */\nexport const ghReleaseDeployAll = async (args: GhReleaseDeployAllArgs) => {\n const { version, env, skipTerraform } = args\n\n commandEcho.start('release-deploy-all')\n\n let selectedReleaseBranch = '' // \"release/v1.8.0\" | \"release/n/checkout-redesign\" | \"dev\"\n\n if (version) {\n selectedReleaseBranch = version === 'dev' ? 'dev' : resolveReleaseBranch(version)\n } else {\n commandEcho.setInteractive()\n\n const releasePRsInfo = await getReleasePRsWithInfo()\n\n const branches = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n const descriptions = await getJiraDescriptions()\n\n selectedReleaseBranch = await select({\n message: '\uD83C\uDF3F Select release branch',\n choices: [{ name: 'dev', value: 'dev' }, ...formatBranchChoices({ branches, descriptions, types: releaseTypes })],\n })\n }\n\n const selectedVersion = releaseLabelFromBranch(selectedReleaseBranch)\n\n commandEcho.addOption('--version', selectedVersion)\n\n const { environments } = await getInfraKitConfig()\n\n let selectedEnv = ''\n\n if (env) {\n selectedEnv = env\n } else {\n commandEcho.setInteractive()\n\n selectedEnv = await select({\n message: '\uD83E\uDDEA Select environment',\n choices: environments.map((env) => {\n return {\n name: env,\n value: env,\n }\n }),\n })\n }\n\n commandEcho.addOption('--env', selectedEnv)\n\n if (!environments.includes(selectedEnv)) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-all workflow',\n remediation: `pass one of: ${environments.join(', ')}`,\n stderrExcerpt: `invalid environment: ${selectedEnv}`,\n })\n }\n\n const shouldSkipTerraform = skipTerraform ?? false\n\n if (shouldSkipTerraform) {\n commandEcho.addOption('--skip-terraform', true)\n }\n\n try {\n $.quiet = true\n\n const skipTerraformFlag = shouldSkipTerraform ? ['-f', 'skip_terraform_deploy=true'] : []\n\n await $`gh workflow run deploy-all.yml --ref ${selectedReleaseBranch} -f environment=${selectedEnv} ${skipTerraformFlag}`\n\n $.quiet = false\n\n logger.info(\n `Successfully launched deploy-all workflow_dispatch for release branch: ${selectedReleaseBranch} and environment: ${selectedEnv}`,\n )\n\n commandEcho.print()\n\n const structuredContent = {\n releaseBranch: selectedReleaseBranch,\n version: selectedVersion,\n environment: selectedEnv,\n skipTerraformDeploy: shouldSkipTerraform,\n success: true,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error: unknown) {\n logger.error({ error }, '\u274C Error launching workflow')\n throw new OperationError(error, {\n operation: 'launch deploy-all workflow',\n remediation: \"check 'gh workflow list' and that deploy-all.yml exists on the target ref\",\n })\n }\n}\n\n// MCP Tool Registration\nexport const ghReleaseDeployAllMcpTool = defineMcpTool({\n name: 'gh-release-deploy-all',\n description:\n 'Dispatch the deploy-all.yml GitHub Actions workflow to deploy every service from a release branch to the given environment. Fire-and-forget \u2014 returns once GitHub accepts the workflow_dispatch, NOT when the deployment finishes; watch the workflow run for completion status. Use gh-release-deploy-selected for a subset of services. Pass version=\"dev\" to deploy from the dev branch instead of a release branch. Both \"version\" and \"env\" are required when invoked via MCP (interactive pickers are unavailable without a TTY).',\n inputSchema: {\n version: z\n .string()\n .describe(\n 'Accepts a release version (e.g. \"1.2.5\") OR a release name (e.g. \"checkout-redesign\") \u2014 resolves to the release/vX.Y.Z or release/n/<name> branch. Pass \"dev\" to deploy from the dev branch instead. Required for MCP calls.',\n ),\n env: z\n .string()\n .describe(\n 'Target environment name \u2014 must match an env configured for the project (e.g. \"dev\", \"renana\", \"oriana\"). Required for MCP calls.',\n ),\n skipTerraform: z.boolean().optional().describe('Skip the terraform deployment stage.'),\n },\n outputSchema: {\n releaseBranch: z.string().describe('The release branch that was deployed'),\n version: z.string().describe('The version that was deployed'),\n environment: z.string().describe('The environment deployed to'),\n skipTerraformDeploy: z.boolean().describe('Whether terraform deployment was skipped'),\n success: z.boolean().describe('Whether the deployment was successful'),\n },\n handler: ghReleaseDeployAll,\n})\n", "import checkbox from '@inquirer/checkbox'\nimport select from '@inquirer/select'\nimport fs from 'node:fs/promises'\nimport { resolve } from 'node:path'\nimport yaml from 'yaml'\nimport { z } from 'zod'\nimport { $ } from 'zx'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getProjectRoot } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport {\n detectReleaseType,\n formatBranchChoices,\n getJiraDescriptions,\n releaseLabelFromBranch,\n resolveReleaseBranch,\n} from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface GhReleaseDeploySelectedArgs {\n version: string\n env: string\n services: string[]\n skipTerraform?: boolean\n}\n\n/**\n * Deploy selected services from a release branch to an environment\n */\n// eslint-disable-next-line sonarjs/cognitive-complexity\nexport const ghReleaseDeploySelected = async (args: GhReleaseDeploySelectedArgs) => {\n const { version, env, services, skipTerraform } = args\n\n commandEcho.start('release-deploy-selected')\n\n let selectedReleaseBranch = ''\n\n if (version) {\n selectedReleaseBranch = version === 'dev' ? 'dev' : resolveReleaseBranch(version)\n } else {\n commandEcho.setInteractive()\n\n const releasePRsInfo = await getReleasePRsWithInfo()\n\n const branches = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n const descriptions = await getJiraDescriptions()\n\n selectedReleaseBranch = await select({\n message: '\uD83C\uDF3F Select release branch',\n choices: [{ name: 'dev', value: 'dev' }, ...formatBranchChoices({ branches, descriptions, types: releaseTypes })],\n })\n }\n\n const selectedVersion = releaseLabelFromBranch(selectedReleaseBranch)\n\n commandEcho.addOption('--version', selectedVersion)\n\n const { environments } = await getInfraKitConfig()\n\n let selectedEnv = ''\n\n if (env) {\n selectedEnv = env\n } else {\n commandEcho.setInteractive()\n\n selectedEnv = await select({\n message: '\uD83E\uDDEA Select environment',\n choices: environments.map((env) => {\n return {\n name: env,\n value: env,\n }\n }),\n })\n }\n\n commandEcho.addOption('--env', selectedEnv)\n\n if (!environments.includes(selectedEnv)) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-selected workflow',\n remediation: `pass one of: ${environments.join(', ')}`,\n stderrExcerpt: `invalid environment: ${selectedEnv}`,\n })\n }\n\n // Parse available services from workflow file\n const availableServices = await parseServicesFromWorkflow()\n\n if (availableServices.length === 0) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-selected workflow',\n remediation: 'check .github/workflows/deploy-selected-services.yml for boolean service inputs',\n stderrExcerpt: 'no services found in workflow file',\n })\n }\n\n let selectedServices: string[] = []\n\n if (services && services.length > 0) {\n selectedServices = services\n } else {\n commandEcho.setInteractive()\n\n selectedServices = await checkbox({\n message: '\uD83D\uDE80 Select services to deploy (space to select, enter to confirm)',\n choices: availableServices.map((svc) => {\n return {\n name: svc,\n value: svc,\n }\n }),\n })\n }\n\n commandEcho.addOption('--services', selectedServices)\n\n if (selectedServices.length === 0) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-selected workflow',\n remediation: `pass at least one service from: ${availableServices.join(', ')}`,\n stderrExcerpt: 'no services selected',\n })\n }\n\n // Validate all selected services\n const invalidServices = selectedServices.filter((svc) => {\n return !availableServices.includes(svc)\n })\n\n if (invalidServices.length > 0) {\n throw new OperationError(undefined, {\n operation: 'launch deploy-selected workflow',\n remediation: `pass services from: ${availableServices.join(', ')}`,\n stderrExcerpt: `invalid services: ${invalidServices.join(', ')}`,\n })\n }\n\n const shouldSkipTerraform = skipTerraform ?? false\n\n if (shouldSkipTerraform) {\n commandEcho.addOption('--skip-terraform', true)\n }\n\n try {\n $.quiet = true\n\n // Build the workflow command with boolean flags for each selected service\n const serviceFlags = selectedServices.flatMap((svc) => {\n return ['-f', `${svc}=true`]\n })\n const skipTerraformFlag = shouldSkipTerraform ? ['-f', 'skip_terraform_deploy=true'] : []\n\n await $`gh workflow run deploy-selected-services.yml --ref ${selectedReleaseBranch} -f environment=${selectedEnv} ${serviceFlags} ${skipTerraformFlag}`\n\n $.quiet = false\n\n logger.info(\n `Successfully launched deploy-selected-services workflow_dispatch for release branch: ${selectedReleaseBranch}, environment: ${selectedEnv}, services: ${selectedServices.join(', ')}`,\n )\n\n commandEcho.print()\n\n const structuredContent = {\n releaseBranch: selectedReleaseBranch,\n version: selectedVersion,\n environment: selectedEnv,\n services: selectedServices,\n skipTerraformDeploy: shouldSkipTerraform,\n success: true,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error: unknown) {\n logger.error({ error }, '\u274C Error launching workflow')\n throw new OperationError(error, {\n operation: 'launch deploy-selected workflow',\n remediation: \"check 'gh workflow list' and that deploy-selected-services.yml exists on the target ref\",\n })\n }\n}\n\n/**\n * Parse available services from the workflow file\n * Services are defined as boolean inputs (excluding skip_terraform_deploy)\n */\nconst parseServicesFromWorkflow = async (): Promise<string[]> => {\n const projectRoot = await getProjectRoot()\n\n const workflowPath = resolve(projectRoot, '.github/workflows/deploy-selected-services.yml')\n\n const content = await fs.readFile(workflowPath, 'utf-8')\n const parsed = yaml.parse(content)\n\n const inputs = parsed.on.workflow_dispatch.inputs\n const services: string[] = []\n\n for (const [key, value] of Object.entries(inputs)) {\n // Filter for boolean type inputs, excluding non-service flags\n if ((value as { type: string }).type === 'boolean' && key !== 'skip_terraform_deploy') {\n services.push(key)\n }\n }\n\n return services\n}\n\n// MCP Tool Registration\nexport const ghReleaseDeploySelectedMcpTool = defineMcpTool({\n name: 'gh-release-deploy-selected',\n description:\n 'Dispatch the deploy-selected-services.yml GitHub Actions workflow to deploy a chosen subset of services from a release branch to the given environment. Fire-and-forget \u2014 returns once GitHub accepts the workflow_dispatch, NOT when the deployment finishes; watch the workflow run for completion status. Service names are validated against the boolean inputs declared in the workflow. Use gh-release-deploy-all for every service. \"version\", \"env\", and \"services\" are all required when invoked via MCP (interactive pickers are unavailable without a TTY).',\n inputSchema: {\n version: z\n .string()\n .describe(\n 'Accepts a release version (e.g. \"1.2.5\") OR a release name (e.g. \"checkout-redesign\") \u2014 resolves to the release/vX.Y.Z or release/n/<name> branch. Pass \"dev\" to deploy from the dev branch instead. Required for MCP calls.',\n ),\n env: z\n .string()\n .describe(\n 'Target environment name \u2014 must match an env configured for the project (e.g. \"dev\", \"renana\", \"oriana\"). Required for MCP calls.',\n ),\n services: z\n .array(z.string())\n .describe(\n 'Service names to deploy. Each must match a boolean input declared in .github/workflows/deploy-selected-services.yml (e.g. \"client-be\", \"client-fe\"). Required for MCP calls.',\n ),\n skipTerraform: z.boolean().optional().describe('Skip the terraform deployment stage.'),\n },\n outputSchema: {\n releaseBranch: z.string().describe('The release branch that was deployed'),\n version: z.string().describe('The version that was deployed'),\n environment: z.string().describe('The environment deployed to'),\n services: z.array(z.string()).describe('The services that were deployed'),\n skipTerraformDeploy: z.boolean().describe('Whether terraform deployment was skipped'),\n success: z.boolean().describe('Whether the deployment was successful'),\n },\n handler: ghReleaseDeploySelected,\n})\n", "import { z } from 'zod'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { logger } from 'src/lib/logger'\nimport { displayLabel, formatJiraName, parseBranchName } from 'src/lib/release-id'\nimport { detectReleaseType, formatVersionLabel, getJiraDescriptions } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\n\n/**\n * List all open release branches\n */\nexport const ghReleaseList = async () => {\n const releasePRs = await getReleasePRsWithInfo()\n\n // Skip branches that do not parse as release ids (lenient discovery source).\n const releases = releasePRs.flatMap((pr) => {\n const id = parseBranchName(pr.branch)\n\n if (!id) return []\n\n return [\n {\n // Human display label: `1.2.3` | `<name>`.\n version: displayLabel(id),\n // Jira-descriptions map is keyed by the Jira version NAME (`v1.2.3` | `<name>`).\n jiraKey: formatJiraName(id),\n type: detectReleaseType(pr.title),\n },\n ]\n })\n\n const jiraDescriptions = await getJiraDescriptions()\n\n const maxVersionLength = Math.max(\n ...releases.map((r) => {\n return r.version.length\n }),\n )\n\n const formattedLines = releases.map((release) => {\n const label = formatVersionLabel(release.version, release.type, maxVersionLength)\n const description = jiraDescriptions.get(release.jiraKey)\n\n if (description) {\n return `${label} ${description}`\n }\n\n return label\n })\n\n logger.info('All release branches: \\n')\n logger.info(`\\n${formattedLines.join('\\n')}\\n`)\n\n const structuredContent = {\n releases: releases.map((release) => {\n return {\n version: release.version,\n type: release.type,\n description: jiraDescriptions.get(release.jiraKey) || null,\n }\n }),\n count: releases.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const ghReleaseListMcpTool = defineMcpTool({\n name: 'gh-release-list',\n description:\n 'List every open release PR with its version, type (regular / hotfix), and associated Jira fix-version description. Read-only; sourced from GitHub and Jira.',\n inputSchema: {},\n outputSchema: {\n releases: z\n .array(\n z.object({\n version: z.string().describe('Release version'),\n type: z.enum(['regular', 'hotfix']).describe('Release type'),\n description: z.string().nullable().describe('Jira version description'),\n }),\n )\n .describe('List of all release branches'),\n count: z.number().describe('Number of release branches'),\n },\n handler: ghReleaseList,\n})\n", "import confirm from '@inquirer/confirm'\nimport select from '@inquirer/select'\nimport process from 'node:process'\nimport { z } from 'zod'\nimport { question } from 'zx'\n\nimport { loadJiraConfig } from 'src/integrations/jira'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { logger } from 'src/lib/logger'\nimport { InvalidReleaseNameError, displayLabel, validateName } from 'src/lib/release-id'\nimport { createSingleRelease, prepareGitForRelease } from 'src/lib/release-utils'\nimport type { ReleaseCreationResult, ReleaseType } from 'src/lib/release-utils'\nimport {\n NoPriorVersionsError,\n computeNextVersion,\n hasNextToken,\n loadExistingVersions,\n parseVersion,\n resolveReleaseEntries,\n} from 'src/lib/version-utils'\nimport type { ReleaseEntry, ReleaseInput, SemVer } from 'src/lib/version-utils'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\ninterface ReleaseCreateArgs extends RequiredConfirmedOptionArg {\n releases?: ReleaseInput[]\n}\n\nconst VERSION_PROMPT_HINT = '\"1.2.5\" or \"next\"'\n\nconst trySuggestNext = (known: SemVer[], type: ReleaseType): string | null => {\n try {\n return computeNextVersion(known, type)\n } catch (err) {\n if (err instanceof NoPriorVersionsError) return null\n\n throw err\n }\n}\n\nconst resolveOrExit = (entries: ReleaseInput[], known: SemVer[]): ReleaseEntry[] => {\n try {\n return resolveReleaseEntries(entries, known)\n } catch (err) {\n if (err instanceof NoPriorVersionsError) {\n throw new OperationError(err, {\n operation: 'resolve release version',\n remediation: 'pass an explicit version (e.g. \"1.2.5\") instead of \"next\" when there are no prior versions',\n })\n }\n\n if (err instanceof InvalidReleaseNameError) {\n throw new OperationError(err, {\n operation: 'validate release name',\n remediation:\n 'use a kebab-case name like \"checkout-redesign\" (lowercase, digits, single hyphens, not a reserved word)',\n })\n }\n\n throw err\n }\n}\n\nconst promptForVersionInput = async (running: SemVer[], type: ReleaseType): Promise<string> => {\n const suggestion = trySuggestNext(running, type)\n const defaultHint = suggestion ? ` [${suggestion}]` : ''\n const versionAnswer = (await question(` Version (e.g. ${VERSION_PROMPT_HINT})${defaultHint}: `)).trim()\n const versionInput = versionAnswer === '' ? (suggestion ?? '') : versionAnswer\n\n if (versionInput === '') {\n logger.error('No version provided. Exiting...')\n process.exit(1)\n }\n\n return versionInput\n}\n\nconst promptForNameInput = async (): Promise<string> => {\n const name = (await question(' Name (kebab-case, e.g. \"checkout-redesign\"): ')).trim()\n\n if (name === '') {\n logger.error('No name provided. Exiting...')\n process.exit(1)\n }\n\n try {\n validateName(name)\n } catch (err) {\n const reason = err instanceof Error ? err.message : String(err)\n\n logger.error(`${reason} Exiting...`)\n process.exit(1)\n }\n\n return name\n}\n\nconst promptForReleasesInteractive = async (ensureKnown: () => Promise<SemVer[]>): Promise<ReleaseEntry[]> => {\n commandEcho.setInteractive()\n\n let baseKnown: SemVer[] | null = null\n const running: SemVer[] = []\n const ensureRunning = async (): Promise<SemVer[]> => {\n if (baseKnown === null) {\n baseKnown = await ensureKnown()\n running.push(...baseKnown)\n }\n\n return running\n }\n\n const entries: ReleaseEntry[] = []\n let addAnother = true\n\n while (addAnother) {\n const ordinal = entries.length + 1\n const kind = await select<'version' | 'name'>({\n message: `Release #${ordinal} \u2014 version or name?`,\n choices: [\n { name: 'version (semver / next)', value: 'version' },\n { name: 'name (free-form)', value: 'name' },\n ],\n default: 'version',\n })\n\n const type = await select<ReleaseType>({\n message: `Release #${ordinal} \u2014 select type:`,\n choices: [\n { name: 'regular', value: 'regular' },\n { name: 'hotfix', value: 'hotfix' },\n ],\n default: 'regular',\n })\n\n let resolved: ReleaseEntry\n\n if (kind === 'name') {\n const name = await promptForNameInput()\n\n resolved = resolveOrExit([{ name, type }], [])[0] as ReleaseEntry\n } else {\n // Versions may need prior versions for \"next\"; load lazily.\n const versionInput = await promptForVersionInput(await ensureRunning(), type)\n\n resolved = resolveOrExit([{ version: versionInput, type }], running)[0] as ReleaseEntry\n\n if (resolved.id.kind === 'version') {\n running.push(parseVersion(`v${resolved.id.raw}`))\n }\n }\n\n const description = (await question(' Description (optional, press Enter to skip): ')).trim()\n\n entries.push({ ...resolved, ...(description !== '' ? { description } : {}) })\n\n addAnother = await confirm({ message: 'Add another release?', default: false })\n }\n\n return entries\n}\n\nconst formatReleaseSummary = (entry: ReleaseEntry): string => {\n const label = entry.id.kind === 'version' ? `v${entry.id.raw}` : entry.id.name\n const parts = [label, entry.type]\n\n if (entry.description) parts.push(entry.description)\n\n return parts.join(' \u00B7 ')\n}\n\nconst echoReleases = (entries: ReleaseEntry[]): void => {\n for (const entry of entries) {\n if (entry.id.kind === 'name') {\n // Named releases echo the --name form; description is not part of the\n // flag and is set later via release-desc-edit / interactively.\n commandEcho.addOption('--name', entry.id.name)\n\n continue\n }\n\n const spec = entry.description\n ? `${entry.id.raw}:${entry.type}:${entry.description}`\n : `${entry.id.raw}:${entry.type}`\n\n commandEcho.addOption('--release', spec)\n }\n}\n\ninterface FailedRelease {\n version: string\n error: string\n}\n\nconst collectEntries = async (\n inputReleases: ReleaseInput[] | undefined,\n ensureKnown: () => Promise<SemVer[]>,\n): Promise<ReleaseEntry[]> => {\n if (inputReleases && inputReleases.length > 0) {\n const known = hasNextToken(inputReleases) ? await ensureKnown() : []\n const resolved = resolveOrExit(inputReleases, known)\n\n echoReleases(resolved)\n\n return resolved\n }\n\n const interactive = await promptForReleasesInteractive(ensureKnown)\n\n echoReleases(interactive)\n\n return interactive\n}\n\nconst confirmReleases = async (entries: ReleaseEntry[], confirmedCommand: boolean): Promise<void> => {\n const summary = entries.map(formatReleaseSummary).join('\\n - ')\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: `Create the following ${entries.length} release(s)?\\n - ${summary}\\n`,\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n commandEcho.addOption('--yes', true)\n}\n\ninterface ExecuteOneArgs {\n entry: ReleaseEntry\n jiraConfig: Awaited<ReturnType<typeof loadJiraConfig>>\n}\n\nconst executeOne = async (\n args: ExecuteOneArgs,\n): Promise<{ result?: ReleaseCreationResult; failure?: FailedRelease }> => {\n const { entry, jiraConfig } = args\n const label = displayLabel(entry.id)\n const prTitleLabel = entry.id.kind === 'version' ? `v${entry.id.raw}` : entry.id.name\n\n try {\n await prepareGitForRelease(entry.type)\n\n const result = await createSingleRelease({\n id: entry.id,\n jiraConfig,\n description: entry.description,\n type: entry.type,\n })\n\n logger.info(`\u2705 Successfully created release: ${prTitleLabel} (${entry.type})`)\n logger.info(`\uD83D\uDD17 GitHub PR: ${result.prUrl}`)\n logger.info(`\uD83D\uDD17 Jira Version: ${result.jiraVersionUrl}\\n`)\n\n return { result }\n } catch (error) {\n const err = new OperationError(error, {\n operation: `create release ${prTitleLabel} (${entry.type})`,\n remediation: 'verify the version or name is unique and the base branch is clean',\n })\n\n logger.error(`\u274C ${err.message}\\n`)\n\n return { failure: { version: label, error: err.message } }\n }\n}\n\nconst logFinalSummary = (total: number, successCount: number, failureCount: number): void => {\n if (successCount === total) {\n logger.info(`\u2705 All ${total} release branch(es) were created successfully.`)\n } else if (successCount > 0) {\n logger.warn(`\u26A0\uFE0F ${successCount} of ${total} release branches were created successfully.`)\n logger.warn(`\u274C ${failureCount} release(s) failed.`)\n } else {\n logger.error(`\u274C All ${total} release branch(es) failed to create.`)\n }\n}\n\n/**\n * Create one or more release branches. Each release carries its own type\n * (regular/hotfix) and optional Jira description, so a single invocation\n * may mix regular and hotfix releases off their respective base branches.\n */\nexport const releaseCreate = async (args: ReleaseCreateArgs) => {\n const { releases: inputReleases, confirmedCommand } = args\n\n commandEcho.start('release-create')\n\n const jiraConfig = await loadJiraConfig()\n\n let known: SemVer[] | null = null\n const ensureKnown = async (): Promise<SemVer[]> => {\n if (known === null) known = await loadExistingVersions()\n\n return known\n }\n\n const entries = await collectEntries(inputReleases, ensureKnown)\n\n if (entries.length === 0) {\n throw new OperationError(undefined, {\n operation: 'create release',\n remediation: 'pass at least one entry in \"releases\" (e.g. [{ version: \"1.2.5\", type: \"regular\" }])',\n stderrExcerpt: 'no releases provided',\n })\n }\n\n await confirmReleases(entries, Boolean(confirmedCommand))\n\n const created: ReleaseCreationResult[] = []\n const failed: FailedRelease[] = []\n\n for (const entry of entries) {\n const { result, failure } = await executeOne({ entry, jiraConfig })\n\n if (result) created.push(result)\n if (failure) failed.push(failure)\n }\n\n logFinalSummary(entries.length, created.length, failed.length)\n\n commandEcho.print()\n\n const structuredContent = {\n createdBranches: created.map((r) => {\n return r.branchName\n }),\n successCount: created.length,\n failureCount: failed.length,\n releases: created,\n failedReleases: failed,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const releaseCreateMcpTool = defineMcpTool({\n name: 'release-create',\n description:\n 'Create one or more releases in a single call. Each entry in \"releases\" carries EITHER a \"version\" (semver or the literal token \"next\") OR a \"name\" (free-form kebab-case identifier) \u2014 exactly one is required and they are mutually exclusive. Each entry also has its own type (regular|hotfix, default regular) and optional description, so regular and hotfix releases can be mixed in the same invocation. For each release this tool switches to the appropriate base branch (dev for regular, main for hotfix), cuts the release branch (release/v<semver> for versions, release/n/<name> for names), opens a GitHub release PR, and creates the matching Jira fix version (v<semver> for versions, <name> for names). The literal token \"next\" auto-increments from the union of remote release branches and Jira fix versions (regular bumps minor + resets patch; hotfix bumps patch on the highest minor); multiple \"next\" tokens advance sequentially across mixed types. Named releases never auto-bump and \"next\" is version-only. Confirmation is auto-skipped for MCP calls, so the caller is responsible for gating. Continues on per-release failure and reports successes/failures.',\n inputSchema: {\n releases: z\n .array(\n z\n .object({\n version: z\n .string()\n .optional()\n .describe(\n 'Version to create (e.g., \"1.2.5\") or the literal token \"next\" for auto-increment. Mutually exclusive with \"name\".',\n ),\n name: z\n .string()\n .optional()\n .describe(\n 'Free-form kebab-case release name (e.g., \"checkout-redesign\"). Mutually exclusive with \"version\". Named releases never auto-bump.',\n ),\n type: z\n .enum(['regular', 'hotfix'])\n .optional()\n .default('regular')\n .describe('Release type: \"regular\" (branches off dev) or \"hotfix\" (branches off main).'),\n description: z.string().optional().describe('Optional description for the Jira version.'),\n })\n .refine(\n (entry) => {\n return (entry.version === undefined) !== (entry.name === undefined)\n },\n {\n message: 'Each release entry must have exactly one of \"version\" or \"name\" (they are mutually exclusive).',\n },\n )\n .transform((entry): ReleaseInput => {\n return entry.name !== undefined\n ? { name: entry.name, type: entry.type, ...(entry.description ? { description: entry.description } : {}) }\n : {\n version: entry.version as string,\n type: entry.type,\n ...(entry.description ? { description: entry.description } : {}),\n }\n }),\n )\n .min(1)\n .describe(\n 'One or more releases to create. Each entry has exactly one of \"version\" or \"name\", plus its own type and optional description.',\n ),\n },\n outputSchema: {\n createdBranches: z.array(z.string()).describe('List of created release branch names'),\n successCount: z.number().describe('Number of releases created successfully'),\n failureCount: z.number().describe('Number of releases that failed'),\n releases: z\n .array(\n z.object({\n version: z.string().describe('Version number'),\n type: z.enum(['regular', 'hotfix']).describe('Release type'),\n branchName: z.string().describe('Release branch name'),\n prUrl: z.string().describe('GitHub PR URL'),\n jiraVersionUrl: z.string().describe('Jira version URL'),\n }),\n )\n .describe('Detailed information for each created release with URLs'),\n failedReleases: z\n .array(\n z.object({\n version: z.string().describe('Version number that failed'),\n error: z.string().describe('Error message'),\n }),\n )\n .describe('List of releases that failed with error messages'),\n },\n handler: releaseCreate,\n})\n", "import { $ } from 'zx'\n\nimport { getProjectVersions, loadJiraConfigOptional } from 'src/integrations/jira'\nimport { logger } from 'src/lib/logger'\nimport { parseBranchName } from 'src/lib/release-id'\n\nimport { collectKnownVersions } from './next-version'\nimport type { SemVer } from './next-version'\n\n/**\n * Extract version-branch tokens from raw `git ls-remote` stdout. Each line is\n * `<sha>\\t<ref>`; refs are routed through release-id's lenient\n * {@link parseBranchName} and only `kind: 'version'` ids are kept (named\n * `release/n/*` branches are irrelevant to `next`-bump math and are dropped).\n * Returns the no-`v` semver tokens (e.g. `1.2.3`) that\n * {@link collectKnownVersions} parses as versions. Pure \u2014 no I/O \u2014 so it is\n * unit-testable without the network.\n */\nexport const extractVersionBranches = (lsRemoteStdout: string): string[] => {\n return lsRemoteStdout\n .split('\\n')\n .map((line) => {\n const tab = line.indexOf('\\t')\n\n if (tab === -1) return null\n\n return parseBranchName(line.slice(tab + 1))\n })\n .filter((id): id is NonNullable<typeof id> => {\n return id !== null && id.kind === 'version'\n })\n .map((id) => {\n return id.raw\n })\n}\n\nconst parseRemoteRefs = async (): Promise<string[]> => {\n const previousQuiet = $.quiet\n\n try {\n $.quiet = true\n const result = await $`git ls-remote --heads origin 'release/v*'`\n\n return extractVersionBranches(result.stdout)\n } finally {\n $.quiet = previousQuiet\n }\n}\n\nconst fetchJiraVersionNames = async (): Promise<string[]> => {\n const config = await loadJiraConfigOptional()\n\n if (!config) return []\n\n const versions = await getProjectVersions(config)\n\n return versions.map((v) => {\n return v.name\n })\n}\n\n/**\n * Load known release versions from the union of:\n * - remote release branches (`release/v*` on origin)\n * - Jira fix versions (when configured)\n *\n * Each source is queried in parallel; if either fails, we log a warning\n * and continue with the other so a transient outage doesn't block release\n * creation.\n */\nexport const loadExistingVersions = async (): Promise<SemVer[]> => {\n const [branchesResult, jiraResult] = await Promise.allSettled([parseRemoteRefs(), fetchJiraVersionNames()])\n\n if (branchesResult.status === 'rejected') {\n logger.warn({ error: branchesResult.reason }, 'Failed to list remote release branches; continuing without them')\n }\n\n if (jiraResult.status === 'rejected') {\n logger.warn({ error: jiraResult.reason }, 'Failed to fetch Jira versions; continuing without them')\n }\n\n return collectKnownVersions({\n remoteBranches: branchesResult.status === 'fulfilled' ? branchesResult.value : [],\n jiraVersions: jiraResult.status === 'fulfilled' ? jiraResult.value : [],\n })\n}\n", "/**\n * Parse a `v<semver>` token into major, minor, patch numbers.\n *\n * SemVer-ONLY: this assumes a `v<semver>` token and will produce `NaN`s for\n * anything else. Callers that may see branch names or named releases\n * (`release/n/<name>`) MUST use the `release-id` module (`parseBranchName` /\n * `parseReleaseRef` + `compareReleaseIds`) instead. Only provably names-free\n * callers (the `collectKnownVersions` / `next`-bump math path) may use this.\n */\nexport const parseVersion = (versionStr: string): [number, number, number] => {\n return versionStr.slice(1).split('.').map(Number) as [number, number, number]\n}\n\n/**\n * Sort version strings in ascending order.\n * Note: Returns a new sorted array without mutating the original.\n *\n * SemVer-ONLY: relies on {@link parseVersion} and will NaN-sort named releases.\n * Name-aware callers MUST use `compareReleaseIds` from the `release-id` module.\n */\nexport const sortVersions = (versions: string[]): string[] => {\n return [...versions].sort((a, b) => {\n const [majA, minA, patchA] = parseVersion(a)\n const [majB, minB, patchB] = parseVersion(b)\n\n if (majA !== majB) return (majA ?? 0) - (majB ?? 0)\n if (minA !== minB) return (minA ?? 0) - (minB ?? 0)\n\n return (patchA ?? 0) - (patchB ?? 0)\n })\n}\n", "import { parseReleaseRef, validateName } from 'src/lib/release-id'\nimport type { ReleaseId } from 'src/lib/release-id'\nimport type { ReleaseType } from 'src/lib/release-utils'\n\nimport { parseVersion, sortVersions } from './version-utils'\n\nexport const NEXT_TOKEN = 'next'\n\nexport type SemVer = readonly [number, number, number]\n\nexport interface ExistingVersionsSources {\n remoteBranches?: string[]\n jiraVersions?: string[]\n}\n\nconst VERSION_RE = /^v?(\\d+)\\.(\\d+)\\.(\\d+)$/\n\nconst stripBranchPrefix = (raw: string): string => {\n return raw.replace(/^.*release\\//, '')\n}\n\nconst tryParse = (raw: string): SemVer | null => {\n const cleaned = stripBranchPrefix(raw.trim())\n const match = VERSION_RE.exec(cleaned)\n\n if (!match) return null\n\n return [Number(match[1]), Number(match[2]), Number(match[3])]\n}\n\nconst semverKey = (v: SemVer): string => {\n return `${v[0]}.${v[1]}.${v[2]}`\n}\n\nexport const collectKnownVersions = (sources: ExistingVersionsSources): SemVer[] => {\n const all = [...(sources.remoteBranches ?? []), ...(sources.jiraVersions ?? [])]\n const parsed: SemVer[] = []\n const seen = new Set<string>()\n\n for (const raw of all) {\n const v = tryParse(raw)\n\n if (!v) continue\n\n const key = semverKey(v)\n\n if (seen.has(key)) continue\n\n seen.add(key)\n parsed.push(v)\n }\n\n return sortVersions(\n parsed.map((v) => {\n return semverKey(v)\n }),\n ).map((s) => {\n return parseVersion(`v${s}`)\n })\n}\n\nexport class NoPriorVersionsError extends Error {\n constructor() {\n super('No prior release versions found from git or Jira. Specify the version explicitly.')\n this.name = 'NoPriorVersionsError'\n }\n}\n\n/**\n * Compute the next semantic version based on release type.\n * - regular: bump minor, reset patch to 0\n * - hotfix: bump patch on the highest minor (any patch)\n *\n * Returns the version without the leading \"v\" (e.g. \"1.64.0\").\n */\nexport const computeNextVersion = (known: SemVer[], type: ReleaseType): string => {\n if (known.length === 0) throw new NoPriorVersionsError()\n\n const max = known[known.length - 1] as SemVer\n\n if (type === 'hotfix') {\n const [major, minor] = max\n\n const highestPatchOnMinor = known.reduce((acc, v) => {\n if (v[0] === major && v[1] === minor) return Math.max(acc, v[2])\n\n return acc\n }, 0)\n\n return `${major}.${minor}.${highestPatchOnMinor + 1}`\n }\n\n const [major, minor] = max\n\n return `${major}.${minor + 1}.0`\n}\n\nconst isNextToken = (token: string): boolean => {\n return token.trim().toLowerCase() === NEXT_TOKEN\n}\n\n/**\n * A release spec is the parsed form of a versioned release request: a raw\n * version token (`\"1.2.5\"` or `\"next\"`) plus its type and optional\n * description. This is the unchanged output of {@link parseReleaseSpec} and the\n * versioned-input shape consumed by {@link resolveReleaseEntries}.\n */\nexport interface ReleaseSpec {\n version: string\n type: ReleaseType\n description?: string\n}\n\n/**\n * A named release request: a bare kebab-case name plus its type and optional\n * description. Named releases never auto-bump; `\"next\"` is version-only.\n */\nexport interface NamedReleaseInput {\n name: string\n type: ReleaseType\n description?: string\n}\n\n/** Either a versioned spec or a named release request. */\nexport type ReleaseInput = ReleaseSpec | NamedReleaseInput\n\n/**\n * A fully resolved release entry. The {@link ReleaseId} carries the concrete\n * identity (version or name) and is the single source for branch/PR/Jira\n * formatting via the `release-id` module.\n */\nexport interface ReleaseEntry {\n id: ReleaseId\n type: ReleaseType\n description?: string\n}\n\nconst isReleaseType = (value: string): value is ReleaseType => {\n return value === 'regular' || value === 'hotfix'\n}\n\nconst isNamedReleaseInput = (input: ReleaseInput): input is NamedReleaseInput => {\n return 'name' in input\n}\n\n/**\n * Parse a CLI release spec of the form `version[:type[:description]]`.\n * Type defaults to \"regular\". Description is everything after the second\n * colon, so colons inside descriptions are preserved.\n */\nexport const parseReleaseSpec = (raw: string): ReleaseSpec => {\n const spec = raw.trim()\n\n if (spec === '') throw new Error('Release spec is empty')\n\n const firstColon = spec.indexOf(':')\n\n if (firstColon === -1) {\n return { version: spec, type: 'regular' }\n }\n\n const version = spec.slice(0, firstColon).trim()\n const rest = spec.slice(firstColon + 1)\n const secondColon = rest.indexOf(':')\n\n const typeRaw = secondColon === -1 ? rest.trim() : rest.slice(0, secondColon).trim()\n const description = secondColon === -1 ? '' : rest.slice(secondColon + 1).trim()\n const typeLower = typeRaw.toLowerCase()\n\n if (!isReleaseType(typeLower)) {\n throw new Error(`Invalid release type \"${typeRaw}\". Expected \"regular\" or \"hotfix\".`)\n }\n\n const entry: ReleaseSpec = { version, type: typeLower }\n\n if (description !== '') entry.description = description\n\n return entry\n}\n\nconst withDescription = (base: { id: ReleaseId; type: ReleaseType }, description?: string): ReleaseEntry => {\n return description !== undefined && description !== '' ? { ...base, description } : base\n}\n\nconst resolveNamedInput = (input: NamedReleaseInput): ReleaseEntry => {\n const name = input.name.trim()\n\n // validateName throws InvalidReleaseNameError with a specific message.\n validateName(name)\n\n return withDescription({ id: { kind: 'name', name, raw: name }, type: input.type }, input.description)\n}\n\n/**\n * Resolve a list of release inputs into entries carrying a concrete\n * {@link ReleaseId}. Versioned inputs use the existing spec format: the\n * `\"next\"` token advances based on the running max so successive `\"next\"`\n * tokens produce sequential versions (even across mixed types), then the\n * concrete version is wrapped in a {@link ReleaseId}. Named inputs are\n * validated via `validateName` and never auto-bump.\n */\nexport const resolveReleaseEntries = (entries: ReleaseInput[], known: SemVer[]): ReleaseEntry[] => {\n const running: SemVer[] = [...known]\n\n return entries.map((entry) => {\n if (isNamedReleaseInput(entry)) {\n return resolveNamedInput(entry)\n }\n\n const trimmed = entry.version.trim()\n\n if (trimmed === '') {\n throw new Error('Release entry has an empty version')\n }\n\n if (isNextToken(trimmed)) {\n const next = computeNextVersion(running, entry.type)\n\n running.push(parseVersion(`v${next}`))\n\n return withDescription({ id: parseReleaseRef(next), type: entry.type }, entry.description)\n }\n\n const parsed = tryParse(trimmed)\n\n if (!parsed) {\n throw new Error(`Invalid version \"${trimmed}\". Expected semver like \"1.2.5\" or the token \"next\".`)\n }\n\n const explicit = `${parsed[0]}.${parsed[1]}.${parsed[2]}`\n\n running.push(parsed)\n\n return withDescription({ id: parseReleaseRef(explicit), type: entry.type }, entry.description)\n })\n}\n\nexport const hasNextToken = (entries: ReleaseInput[]): boolean => {\n return entries.some((e) => {\n return !isNamedReleaseInput(e) && isNextToken(e.version)\n })\n}\n", "import confirm from '@inquirer/confirm'\nimport select from '@inquirer/select'\nimport process from 'node:process'\nimport { z } from 'zod'\nimport { question } from 'zx'\n\nimport { getReleasePRsWithInfo, updateReleasePRBody } from 'src/integrations/gh'\nimport { findVersionByName, loadJiraConfig, updateJiraVersion } from 'src/integrations/jira'\nimport type { JiraConfig, JiraVersion } from 'src/integrations/jira'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { logger } from 'src/lib/logger'\nimport { displayLabel, formatJiraName, parseBranchName } from 'src/lib/release-id'\nimport {\n detectReleaseType,\n formatBranchChoices,\n getJiraDescriptions,\n resolveReleaseBranch,\n} from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\ninterface ReleaseDescEditArgs extends RequiredConfirmedOptionArg {\n version?: string\n description?: string\n}\n\nconst buildJiraVersionUrl = (jiraConfig: JiraConfig, version: JiraVersion): string => {\n return `${jiraConfig.baseUrl}/projects/${version.projectId}/versions/${version.id}/tab/release-report-all-issues`\n}\n\nconst buildPRBody = (jiraVersionUrl: string, description: string): string => {\n return description.trim() !== '' ? `${jiraVersionUrl}\\n\\n${description}` : `${jiraVersionUrl} \\n`\n}\n\nconst pickReleaseBranch = async (): Promise<{ branch: string; type: ReleaseType }> => {\n const releasePRsInfo = await getReleasePRsWithInfo()\n const branches = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n const types = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n const descriptions = await getJiraDescriptions()\n\n const branch = await select({\n message: '\uD83C\uDF3F Select release branch',\n choices: formatBranchChoices({ branches, descriptions, types }),\n })\n\n return { branch, type: types.get(branch) || 'regular' }\n}\n\nconst verifyReleasePRExists = async (selectedBranch: string): Promise<ReleaseType> => {\n const releasePRsInfo = await getReleasePRsWithInfo()\n const prInfo = releasePRsInfo.find((pr) => {\n return pr.branch === selectedBranch\n })\n\n if (!prInfo) {\n throw new OperationError(undefined, {\n operation: `edit description for ${selectedBranch}`,\n remediation: `confirm an open PR exists for ${selectedBranch} ('gh pr list')`,\n })\n }\n\n return detectReleaseType(prInfo.title)\n}\n\nconst promptDescription = async (current: string): Promise<string> => {\n const hint = current === '' ? '(no current description)' : `current: \"${current}\"`\n const answer = await question(` New description ${hint}\\n (press Enter to keep current): `)\n const trimmed = answer.replace(/\\n$/, '')\n\n return trimmed === '' ? current : trimmed\n}\n\n/**\n * Edit a release's description in Jira (fix version) and in the matching\n * GitHub release PR body. The PR body is rewritten canonically to\n * `<jiraVersionUrl>\\n\\n<description>` (matching `release-create`).\n */\nexport const releaseDescEdit = async (args: ReleaseDescEditArgs) => {\n const { version: versionArg, description: descriptionArg, confirmedCommand } = args\n\n commandEcho.start('release-desc-edit')\n\n const jiraConfig = await loadJiraConfig()\n\n let selectedBranch: string\n\n if (versionArg) {\n selectedBranch = resolveReleaseBranch(versionArg)\n await verifyReleasePRExists(selectedBranch)\n } else {\n commandEcho.setInteractive()\n const picked = await pickReleaseBranch()\n\n selectedBranch = picked.branch\n }\n\n // selectedBranch is always a release branch here (operator ref strictly parsed,\n // or picked from discovery-filtered choices), so parseBranchName cannot be null.\n const releaseId = parseBranchName(selectedBranch)\n\n if (!releaseId) {\n throw new OperationError(undefined, {\n operation: `edit description for ${selectedBranch}`,\n remediation: 'pass a version (e.g. \"1.2.5\") or a release name (e.g. \"checkout-redesign\")',\n })\n }\n\n const selectedVersion = displayLabel(releaseId)\n\n commandEcho.addOption('--version', selectedVersion)\n\n // Jira fix version is named by the Jira convention: `v1.2.3` | `<name>`.\n const versionName = formatJiraName(releaseId)\n const jiraVersion = await findVersionByName(versionName, jiraConfig)\n\n if (!jiraVersion) {\n throw new OperationError(undefined, {\n operation: `edit description for ${versionName}`,\n remediation: `create the Jira fix version \"${versionName}\" first or pick a different release`,\n })\n }\n\n const previousDescription = jiraVersion.description ?? ''\n\n let newDescription: string\n\n if (descriptionArg !== undefined) {\n newDescription = descriptionArg\n commandEcho.addOption('--description', newDescription)\n } else {\n commandEcho.setInteractive()\n newDescription = await promptDescription(previousDescription)\n }\n\n if (newDescription === previousDescription) {\n logger.info(`No change \u2014 description for ${versionName} is already: \"${previousDescription}\"`)\n commandEcho.print()\n\n const structuredContent = {\n version: selectedVersion,\n branch: selectedBranch,\n jiraVersionUrl: buildJiraVersionUrl(jiraConfig, jiraVersion),\n previousDescription,\n newDescription,\n changed: false,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n }\n\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: `Update description for ${versionName}?\\n from: \"${previousDescription}\"\\n to: \"${newDescription}\"\\n`,\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n commandEcho.addOption('--yes', true)\n\n await updateJiraVersion({ versionId: jiraVersion.id, description: newDescription }, jiraConfig)\n\n const jiraVersionUrl = buildJiraVersionUrl(jiraConfig, jiraVersion)\n const body = buildPRBody(jiraVersionUrl, newDescription)\n\n await updateReleasePRBody({ branch: selectedBranch, body })\n\n logger.info(`\u2705 Updated description for ${versionName}`)\n logger.info(`\uD83D\uDD17 Jira Version: ${jiraVersionUrl}`)\n logger.info(`\uD83D\uDD17 PR branch: ${selectedBranch}\\n`)\n\n commandEcho.print()\n\n const structuredContent = {\n version: selectedVersion,\n branch: selectedBranch,\n jiraVersionUrl,\n previousDescription,\n newDescription,\n changed: true,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const releaseDescEditMcpTool = defineMcpTool({\n name: 'release-desc-edit',\n description:\n \"Edit a release's description in Jira and in the matching GitHub release PR body. Accepts a release version or a release name: targets the Jira fix version named `v<version>` (versioned) or `<name>` (named) and the open PR on branch `release/v<version>` or `release/n/<name>`. The PR body is rewritten canonically to `<jiraVersionUrl>\\\\n\\\\n<description>` \u2014 any prior manual edits to the body are overwritten. Both `version` and `description` are required for MCP calls (the picker/prompt are unreachable without a TTY). Empty `description` clears the description on both sides. Confirmation is auto-skipped for MCP, so the caller is responsible for gating.\",\n inputSchema: {\n version: z\n .string()\n .describe('Accepts a release version (e.g. \"1.2.5\") OR a release name (e.g. \"checkout-redesign\").'),\n description: z.string().describe('New description. Empty string clears the description.'),\n },\n outputSchema: {\n version: z.string().describe('Release version'),\n branch: z.string().describe('Release branch name (e.g. \"release/v1.2.5\" or \"release/n/checkout-redesign\")'),\n jiraVersionUrl: z.string().describe('Jira fix version URL'),\n previousDescription: z.string().describe('The description before the update'),\n newDescription: z.string().describe('The description after the update'),\n changed: z.boolean().describe('Whether the description actually changed'),\n },\n handler: releaseDescEdit,\n})\n", "import { z } from 'zod'\n\nimport { logger } from 'src/lib/logger'\nimport { defineMcpTool, textContent } from 'src/types'\n\nimport packageJson from '../../../package.json' with { type: 'json' }\n\n/**\n * Print the infra-kit CLI version\n */\nexport const version = async () => {\n const cliVersion = packageJson.version\n\n logger.info(cliVersion)\n\n const structuredContent = { version: cliVersion }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const versionMcpTool = defineMcpTool({\n name: 'version',\n description: 'Print the installed infra-kit CLI version',\n inputSchema: {},\n outputSchema: {\n version: z.string().describe('Installed infra-kit CLI version (from package.json)'),\n },\n handler: version,\n})\n", "{\n \"name\": \"infra-kit\",\n \"type\": \"module\",\n \"version\": \"0.1.105\",\n \"description\": \"infra-kit\",\n \"main\": \"dist/index.js\",\n \"module\": \"dist/index.js\",\n \"types\": \"dist/entry/index.d.ts\",\n \"exports\": {\n \".\": {\n \"types\": \"./dist/entry/index.d.ts\",\n \"import\": \"./dist/index.js\"\n }\n },\n \"bin\": {\n \"infra-kit\": \"dist/cli.js\"\n },\n \"engines\": {\n \"node\": \">=24.x\"\n },\n \"scripts\": {\n \"inspector\": \"npx @modelcontextprotocol/inspector node ./dist/mcp.js --debug\",\n \"build\": \"pnpm run clean-artifacts && node ./scripts/build.js\",\n \"check\": \"node ./dist/cli.js check\",\n \"clean-artifacts\": \"rm -rf dist\",\n \"clean-cache\": \"rm -rf node_modules/.cache .eslintcache tsconfig.tsbuildinfo .turbo .swc\",\n \"prettier-fix\": \"pnpm exec prettier **/* --write --no-error-on-unmatched-pattern --log-level silent --ignore-path ../../../.prettierignore\",\n \"prettier-check\": \"pnpm exec prettier **/* --check --no-error-on-unmatched-pattern --log-level silent --ignore-path ../../../.prettierignore\",\n \"eslint-check\": \"pnpm exec eslint --cache --quiet --report-unused-disable-directives ./src\",\n \"eslint-fix\": \"pnpm exec eslint --cache --quiet --report-unused-disable-directives ./src --fix\",\n \"ts-check\": \"tsc --noEmit\",\n \"test\": \"pnpm exec vitest run --reporter=minimal\",\n \"test-watch\": \"pnpm exec vitest --watch --silent passed-only\",\n \"test-ui\": \"pnpm exec vitest --ui --silent passed-only\",\n \"test-report\": \"pnpm exec vitest run --coverage --silent passed-only\",\n \"qa\": \"pnpm run prettier-check && pnpm run eslint-check && pnpm run ts-check && pnpm run test && echo \u2705 Success\",\n \"fix\": \"pnpm run prettier-fix && pnpm run eslint-fix && pnpm run qa\"\n },\n \"dependencies\": {\n \"@inquirer/checkbox\": \"^5.2.1\",\n \"@inquirer/confirm\": \"^6.1.1\",\n \"@inquirer/select\": \"^5.2.1\",\n \"@modelcontextprotocol/sdk\": \"^1.29.0\",\n \"commander\": \"^15.0.0\",\n \"pino\": \"^10.3.1\",\n \"pino-pretty\": \"^13.1.3\",\n \"yaml\": \"^2.9.0\",\n \"zod\": \"^4.4.3\",\n \"zx\": \"^8.8.5\"\n },\n \"devDependencies\": {\n \"@wl/eslint-config\": \"workspace:*\",\n \"@wl/vitest-config\": \"workspace:*\",\n \"esbuild\": \"^0.28.1\",\n \"typescript\": \"^6.0.3\"\n }\n}\n", "/* eslint-disable sonarjs/cognitive-complexity */\nimport checkbox from '@inquirer/checkbox'\nimport confirm from '@inquirer/confirm'\nimport select from '@inquirer/select'\nimport process from 'node:process'\nimport { z } from 'zod'\nimport { $ } from 'zx'\n\nimport { buildCmuxWorkspaceTitle, openCmuxWorkspaceWithLayout } from 'src/integrations/cmux'\nimport { addFoldersToCursorWorkspace, resolveCursorWorkspacePath } from 'src/integrations/cursor'\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { formatBranchName, isReleaseBranch, parseReleaseRef } from 'src/lib/release-id'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions, releaseBranchLabels } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\n// Constants\nconst FEATURE_DIR = 'feature'\nconst RELEASE_DIR = 'release'\n\nexport const CURSOR_MODES = ['workspace', 'windows', 'none'] as const\nexport type CursorMode = (typeof CURSOR_MODES)[number]\n\ninterface WorktreeManagementArgs extends RequiredConfirmedOptionArg {\n all?: boolean\n versions?: string\n cursor?: CursorMode\n githubDesktop?: boolean\n cmux?: boolean\n}\n\n/**\n * Manage git worktrees for release branches\n * Creates worktrees for active release branches and removes unused ones\n */\nexport const worktreesAdd = async (options: WorktreeManagementArgs) => {\n const { confirmedCommand, all, versions, cursor, githubDesktop, cmux } = options\n\n commandEcho.start('worktrees-add')\n\n try {\n const currentWorktrees = await getCurrentWorktrees('release')\n const projectRoot = await getProjectRoot()\n\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n\n await ensureWorktreeDirectory(`${worktreeDir}/${RELEASE_DIR}`)\n await ensureWorktreeDirectory(`${worktreeDir}/${FEATURE_DIR}`)\n\n let selectedReleaseBranches: string[] = []\n\n if (versions) {\n selectedReleaseBranches = versions.split(',').map((v) => {\n return formatBranchName(parseReleaseRef(v.trim()))\n })\n } else {\n const releasePRsInfo = await getReleasePRsWithInfo()\n\n const releasePRsList = releasePRsInfo.map((pr) => {\n return pr.branch\n })\n\n if (releasePRsList.length === 0) {\n logger.info('\u2139\uFE0F No open release branches found')\n\n commandEcho.print()\n\n return {\n content: textContent(JSON.stringify({ createdWorktrees: [], count: 0 }, null, 2)),\n structuredContent: { createdWorktrees: [], count: 0 },\n }\n }\n\n if (all) {\n selectedReleaseBranches = releasePRsList\n } else {\n commandEcho.setInteractive()\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n const descriptions = await getJiraDescriptions()\n\n selectedReleaseBranches = await checkbox({\n required: true,\n message: '\uD83C\uDF3F Select release branches',\n choices: formatBranchChoices({ branches: releasePRsList, descriptions, types: releaseTypes }),\n })\n }\n }\n\n // Track --all flag if all branches were selected (either via flag or interactively)\n if (all) {\n commandEcho.addOption('--all', true)\n } else {\n commandEcho.addOption('--versions', releaseBranchLabels(selectedReleaseBranches))\n }\n\n // Ask for confirmation\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: 'Are you sure you want to proceed with these worktree changes?',\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n if (!confirmedCommand) {\n commandEcho.addOption('--yes', true)\n }\n\n const config = await getInfraKitConfig()\n const cursorConfig = config.ide?.provider === 'cursor' ? config.ide.config : undefined\n\n const cursorMode: CursorMode =\n cursor ??\n cursorConfig?.mode ??\n (await select<CursorMode>({\n message: 'Cursor mode for created worktrees?',\n default: 'workspace',\n choices: [\n {\n name: 'Add to workspace file',\n value: 'workspace',\n description: 'Append each worktree as a folder in ide.config.workspaceConfigPath, then open the workspace',\n },\n {\n name: 'Open separate windows',\n value: 'windows',\n description: 'Open each created worktree in its own Cursor window',\n },\n { name: 'Skip', value: 'none', description: 'Do not open Cursor' },\n ],\n }))\n\n if (typeof cursor === 'undefined' && !cursorConfig?.mode) {\n commandEcho.setInteractive()\n }\n\n commandEcho.addOption('--cursor', cursorMode)\n\n const openInGithubDesktop =\n githubDesktop ??\n config.worktrees?.openInGithubDesktop ??\n (await confirm({ message: 'Open created worktrees in GitHub Desktop?' }))\n\n if (typeof githubDesktop === 'undefined' && config.worktrees?.openInGithubDesktop === undefined) {\n commandEcho.setInteractive()\n }\n\n if (openInGithubDesktop) {\n commandEcho.addOption('--github-desktop', true)\n } else {\n commandEcho.addOption('--no-github-desktop', true)\n }\n\n const openInCmux =\n cmux ?? config.worktrees?.openInCmux ?? (await confirm({ message: 'Open created worktrees in cmux?' }))\n\n if (typeof cmux === 'undefined' && config.worktrees?.openInCmux === undefined) {\n commandEcho.setInteractive()\n }\n\n if (openInCmux) {\n commandEcho.addOption('--cmux', true)\n } else {\n commandEcho.addOption('--no-cmux', true)\n }\n\n const { branchesToCreate } = categorizeWorktrees({\n selectedReleaseBranches,\n currentWorktrees,\n })\n\n const createdWorktrees = await createWorktrees(branchesToCreate, worktreeDir)\n\n logResults(createdWorktrees)\n\n if (cursorMode === 'workspace') {\n if (!cursorConfig?.workspaceConfigPath) {\n logger.warn('\u26A0\uFE0F Skipping Cursor: ide.config.workspaceConfigPath is not set in infra-kit config')\n } else {\n const workspacePath = resolveCursorWorkspacePath(cursorConfig.workspaceConfigPath, projectRoot)\n\n const folderPaths = createdWorktrees.map((branch) => {\n return `${worktreeDir}/${branch}`\n })\n\n const { added, skipped } = await addFoldersToCursorWorkspace({ workspacePath, folderPaths })\n\n const skippedSuffix = skipped.length > 0 ? ` (${skipped.length} already present)` : ''\n\n logger.info(`\u2705 Added ${added.length} folder(s) to ${workspacePath}${skippedSuffix}`)\n\n await $`cursor ${workspacePath}`\n }\n } else if (cursorMode === 'windows') {\n for (const branch of createdWorktrees) {\n await $`cursor ${worktreeDir}/${branch}`\n }\n }\n\n if (openInGithubDesktop) {\n for (const branch of createdWorktrees) {\n await $`github ${worktreeDir}/${branch}`\n await $`sleep 5`\n }\n }\n\n if (openInCmux) {\n const repoName = await getRepoName()\n\n for (const branch of createdWorktrees) {\n const title = buildCmuxWorkspaceTitle({ repoName, branch })\n\n await openCmuxWorkspaceWithLayout({\n cwd: `${worktreeDir}/${branch}`,\n title,\n })\n }\n }\n\n commandEcho.print()\n\n const structuredContent = {\n createdWorktrees,\n count: createdWorktrees.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error) {\n logger.error({ error }, '\u274C Error managing worktrees')\n throw new OperationError(error, {\n operation: 'create worktrees',\n remediation: \"verify branches don't already exist as worktrees: 'git worktree list'\",\n })\n }\n}\n\n/**\n * Ensure the worktree directory exists\n */\nconst ensureWorktreeDirectory = async (worktreeDir: string): Promise<void> => {\n await $`mkdir -p ${worktreeDir}`\n}\n\ninterface CategorizeWorktreesArgs {\n selectedReleaseBranches: string[]\n currentWorktrees: string[]\n}\n\n/**\n * Categorize release worktrees into those that need to be created or removed\n */\nconst categorizeWorktrees = (args: CategorizeWorktreesArgs): { branchesToCreate: string[] } => {\n const { selectedReleaseBranches, currentWorktrees } = args\n\n const currentBranchNames = currentWorktrees.filter((branch) => {\n return isReleaseBranch(branch)\n })\n\n const branchesToCreate = selectedReleaseBranches.filter((branch) => {\n return !currentBranchNames.includes(branch)\n })\n\n return { branchesToCreate }\n}\n\n/**\n * Create worktrees for the specified branches\n */\nconst createWorktrees = async (branches: string[], worktreeDir: string): Promise<string[]> => {\n const results = await Promise.allSettled(\n branches.map(async (branch) => {\n const worktreePath = `${worktreeDir}/${branch}`\n\n await $`git worktree add ${worktreePath} ${branch}`\n await $({ cwd: worktreePath })`pnpm install`\n\n return branch\n }),\n )\n\n const created: string[] = []\n\n for (const [index, result] of results.entries()) {\n if (result.status === 'fulfilled') {\n created.push(result.value)\n } else {\n const branch = branches[index]\n const err = new OperationError(result.reason, {\n operation: `git worktree add for ${branch}`,\n remediation: 'check the branch name and that the parent dir is writable',\n })\n\n logger.error({ error: result.reason, msg: err.message })\n }\n }\n\n return created\n}\n\n/**\n * Log the results of worktree management\n */\nconst logResults = (created: string[]): void => {\n if (created.length > 0) {\n logger.info('\u2705 Created git worktrees:')\n for (const branch of created) {\n logger.info(branch)\n }\n logger.info('')\n } else {\n logger.info('\u2139\uFE0F No new git worktrees to create')\n }\n}\n\n// MCP Tool Registration\nexport const worktreesAddMcpTool = defineMcpTool({\n name: 'worktrees-add',\n description:\n 'Create local git worktrees for release branches under the worktrees directory and run \"pnpm install\" in each. Mutates the local filesystem. When invoked via MCP, pass either \"versions\" (comma-separated) or all=true \u2014 the branch picker and \"open in Cursor / GitHub Desktop / cmux\" follow-up prompts are unreachable without a TTY, and the CLI confirmation is auto-skipped for MCP calls.',\n inputSchema: {\n all: z\n .boolean()\n .optional()\n .describe(\n 'Add worktrees for every open release branch. Either \"all\" or \"versions\" must be provided for MCP calls (the interactive picker is unavailable without a TTY). Ignored if \"versions\" is provided.',\n ),\n versions: z\n .string()\n .optional()\n .describe(\n 'Comma-separated release versions or names to target (e.g. \"1.2.5, 1.2.6\" or \"checkout-redesign, 1.2.5\"). Either \"versions\" or all=true must be provided for MCP calls. Overrides \"all\" when set.',\n ),\n cursor: z\n .enum(CURSOR_MODES)\n .optional()\n .describe(\n 'Cursor open mode for created worktrees. \"workspace\" appends each worktree as a folder to \"ide.config.workspaceConfigPath\" in infra-kit config and opens the workspace. \"windows\" opens each worktree in its own Cursor window. \"none\" skips Cursor. Resolution order: this flag \u2192 \"ide.config.mode\" from infra-kit config \u2192 interactive prompt (CLI) / \"none\" (MCP, no TTY).',\n ),\n githubDesktop: z\n .boolean()\n .optional()\n .describe(\n 'Open each created worktree in GitHub Desktop. Resolution order: this flag \u2192 \"worktrees.openInGithubDesktop\" from infra-kit config \u2192 interactive prompt (CLI) / false (MCP, no TTY).',\n ),\n cmux: z\n .boolean()\n .optional()\n .describe(\n 'Open each created worktree in a new cmux workspace with a 3-pane layout (left-top, left-bottom, full-height right), all rooted at the worktree directory. Resolution order: this flag \u2192 \"worktrees.openInCmux\" from infra-kit config \u2192 interactive prompt (CLI) / false (MCP, no TTY).',\n ),\n },\n outputSchema: {\n createdWorktrees: z.array(z.string()).describe('List of created git worktree branches'),\n count: z.number().describe('Number of git worktrees created'),\n },\n handler: worktreesAdd,\n})\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\n\ninterface AddFoldersToCursorWorkspaceArgs {\n workspacePath: string\n folderPaths: string[]\n}\n\ninterface AddFoldersToCursorWorkspaceResult {\n added: string[]\n skipped: string[]\n}\n\ninterface WorkspaceFolderEntry {\n path: string\n name?: string\n}\n\ninterface WorkspaceFile {\n folders?: WorkspaceFolderEntry[]\n [key: string]: unknown\n}\n\n/**\n * Adds folders to a Cursor (`.code-workspace`) file's `folders` array, skipping\n * entries that already point to the same absolute path. Folder paths are written\n * as relative to the workspace file's directory to match Cursor's default style.\n */\nexport const addFoldersToCursorWorkspace = async (\n args: AddFoldersToCursorWorkspaceArgs,\n): Promise<AddFoldersToCursorWorkspaceResult> => {\n const { workspacePath, folderPaths } = args\n\n const workspaceDir = path.dirname(workspacePath)\n\n let raw: string\n\n try {\n raw = await fs.readFile(workspacePath, 'utf-8')\n } catch (error) {\n throw new Error(`Cursor workspace file not found at ${workspacePath}: ${(error as Error).message}`)\n }\n\n let parsed: WorkspaceFile\n\n try {\n parsed = JSON.parse(raw) as WorkspaceFile\n } catch (error) {\n throw new Error(\n `Failed to parse ${workspacePath} as JSON. Comments (JSONC) are not supported. ${(error as Error).message}`,\n )\n }\n\n const existingFolders = parsed.folders ?? []\n const existingAbsolutePaths = new Set(\n existingFolders.map((entry) => {\n return path.resolve(workspaceDir, entry.path)\n }),\n )\n\n const added: string[] = []\n const skipped: string[] = []\n\n for (const folderPath of folderPaths) {\n const absolutePath = path.resolve(folderPath)\n\n if (existingAbsolutePaths.has(absolutePath)) {\n skipped.push(folderPath)\n continue\n }\n\n const relativePath = path.relative(workspaceDir, absolutePath)\n\n existingFolders.push({ path: relativePath })\n existingAbsolutePaths.add(absolutePath)\n added.push(folderPath)\n }\n\n parsed.folders = existingFolders\n\n await fs.writeFile(workspacePath, `${JSON.stringify(parsed, null, 2)}\\n`, 'utf-8')\n\n return { added, skipped }\n}\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\n\nimport { addFoldersToCursorWorkspace } from './add-folders-to-workspace'\nimport { removeFoldersFromCursorWorkspace } from './remove-folders-from-workspace'\n\ninterface ReconcileCursorWorkspaceFoldersArgs {\n workspacePath: string\n worktreeDir: string\n currentBranches: string[]\n}\n\ninterface ReconcileCursorWorkspaceFoldersResult {\n added: string[]\n removed: string[]\n}\n\ninterface WorkspaceFolderEntry {\n path: string\n name?: string\n}\n\ninterface WorkspaceFile {\n folders?: WorkspaceFolderEntry[]\n [key: string]: unknown\n}\n\n/**\n * Reconciles the configured Cursor workspace's `folders` array against the\n * actual set of release worktrees on disk:\n * - Adds any worktree folders that aren't already listed.\n * - Removes entries whose absolute path lives under `${worktreeDir}/release/`\n * but no longer corresponds to a current branch (drift from manual\n * `git worktree remove`, deleted branches, etc.).\n *\n * Non-worktree folder entries are left untouched.\n */\nexport const reconcileCursorWorkspaceFolders = async (\n args: ReconcileCursorWorkspaceFoldersArgs,\n): Promise<ReconcileCursorWorkspaceFoldersResult> => {\n const { workspacePath, worktreeDir, currentBranches } = args\n\n const workspaceDir = path.dirname(workspacePath)\n const releaseRoot = path.resolve(`${worktreeDir}/release`)\n\n const raw = await fs.readFile(workspacePath, 'utf-8')\n const parsed = JSON.parse(raw) as WorkspaceFile\n\n const existingFolders = parsed.folders ?? []\n\n const desiredAbsolutePaths = new Set(\n currentBranches.map((branch) => {\n return path.resolve(`${worktreeDir}/${branch}`)\n }),\n )\n\n const danglingFolderPaths: string[] = []\n\n for (const entry of existingFolders) {\n const entryAbsolutePath = path.resolve(workspaceDir, entry.path)\n\n const isReleaseShaped = entryAbsolutePath === releaseRoot || entryAbsolutePath.startsWith(`${releaseRoot}/`)\n\n if (isReleaseShaped && !desiredAbsolutePaths.has(entryAbsolutePath)) {\n danglingFolderPaths.push(entryAbsolutePath)\n }\n }\n\n let removed: string[] = []\n\n if (danglingFolderPaths.length > 0) {\n const result = await removeFoldersFromCursorWorkspace({\n workspacePath,\n folderPaths: danglingFolderPaths,\n })\n\n removed = result.removed\n }\n\n const desiredFolderPaths = currentBranches.map((branch) => {\n return `${worktreeDir}/${branch}`\n })\n\n const { added } =\n desiredFolderPaths.length > 0\n ? await addFoldersToCursorWorkspace({ workspacePath, folderPaths: desiredFolderPaths })\n : { added: [] as string[] }\n\n return { added, removed }\n}\n", "import fs from 'node:fs/promises'\nimport path from 'node:path'\n\ninterface RemoveFoldersFromCursorWorkspaceArgs {\n workspacePath: string\n folderPaths: string[]\n}\n\ninterface RemoveFoldersFromCursorWorkspaceResult {\n removed: string[]\n notFound: string[]\n}\n\ninterface WorkspaceFolderEntry {\n path: string\n name?: string\n}\n\ninterface WorkspaceFile {\n folders?: WorkspaceFolderEntry[]\n [key: string]: unknown\n}\n\n/**\n * Removes folders from a Cursor (`.code-workspace`) file's `folders` array. Entries\n * are matched by resolved absolute path, so relative and absolute entries pointing\n * at the same folder are both removed.\n */\nexport const removeFoldersFromCursorWorkspace = async (\n args: RemoveFoldersFromCursorWorkspaceArgs,\n): Promise<RemoveFoldersFromCursorWorkspaceResult> => {\n const { workspacePath, folderPaths } = args\n\n const workspaceDir = path.dirname(workspacePath)\n\n let raw: string\n\n try {\n raw = await fs.readFile(workspacePath, 'utf-8')\n } catch (error) {\n throw new Error(`Cursor workspace file not found at ${workspacePath}: ${(error as Error).message}`)\n }\n\n let parsed: WorkspaceFile\n\n try {\n parsed = JSON.parse(raw) as WorkspaceFile\n } catch (error) {\n throw new Error(\n `Failed to parse ${workspacePath} as JSON. Comments (JSONC) are not supported. ${(error as Error).message}`,\n )\n }\n\n const existingFolders = parsed.folders ?? []\n const targetAbsolutePaths = new Set(\n folderPaths.map((folderPath) => {\n return path.resolve(folderPath)\n }),\n )\n\n const removedAbsolutePaths = new Set<string>()\n\n const filteredFolders = existingFolders.filter((entry) => {\n const entryAbsolutePath = path.resolve(workspaceDir, entry.path)\n\n if (targetAbsolutePaths.has(entryAbsolutePath)) {\n removedAbsolutePaths.add(entryAbsolutePath)\n\n return false\n }\n\n return true\n })\n\n parsed.folders = filteredFolders\n\n await fs.writeFile(workspacePath, `${JSON.stringify(parsed, null, 2)}\\n`, 'utf-8')\n\n const removed: string[] = []\n const notFound: string[] = []\n\n for (const folderPath of folderPaths) {\n const absolutePath = path.resolve(folderPath)\n\n if (removedAbsolutePaths.has(absolutePath)) {\n removed.push(folderPath)\n } else {\n notFound.push(folderPath)\n }\n }\n\n return { removed, notFound }\n}\n", "import path from 'node:path'\n\n/**\n * Resolves the configured Cursor workspace path against the project root.\n * Absolute paths are returned unchanged.\n */\nexport const resolveCursorWorkspacePath = (configValue: string, projectRoot: string): string => {\n if (path.isAbsolute(configValue)) {\n return configValue\n }\n\n return path.resolve(projectRoot, configValue)\n}\n", "import { z } from 'zod'\n\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { getCurrentWorktrees } from 'src/lib/git-utils'\nimport { logger } from 'src/lib/logger'\nimport { displayLabel, formatJiraName, parseBranchName } from 'src/lib/release-id'\nimport { detectReleaseType, formatVersionLabel, getJiraDescriptions } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface WorktreeInfo {\n version: string\n type: ReleaseType\n description: string | null\n}\n\n/**\n * List all release git worktrees with version, type, and Jira description\n */\nexport const worktreesList = async () => {\n const currentWorktrees = await getCurrentWorktrees('release')\n\n if (currentWorktrees.length === 0) {\n logger.info('\u2139\uFE0F No active worktrees found')\n\n return {\n content: textContent(JSON.stringify({ worktrees: [], count: 0 }, null, 2)),\n structuredContent: { worktrees: [], count: 0 },\n }\n }\n\n const [releasePRsInfo, jiraDescriptions] = await Promise.all([getReleasePRsWithInfo(), getJiraDescriptions()])\n\n const releaseTypes = new Map<string, ReleaseType>(\n releasePRsInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n // Skip worktrees whose branch does not parse as a release id (lenient source).\n const worktrees: WorktreeInfo[] = currentWorktrees.flatMap((branch) => {\n const id = parseBranchName(branch)\n\n if (!id) return []\n\n // Human label `1.2.3` | `<name>`; Jira-descriptions map is keyed by the\n // Jira version NAME (`v1.2.3` | `<name>`) \u2014 same split as formatBranchChoices.\n const version = displayLabel(id)\n const type = releaseTypes.get(branch) || 'regular'\n const description = jiraDescriptions.get(formatJiraName(id)) || null\n\n return [{ version, type, description }]\n })\n\n // Log formatted output\n const maxVersionLength = Math.max(\n ...worktrees.map((w) => {\n return w.version.length\n }),\n )\n\n const formattedLines = worktrees.map((worktree) => {\n const label = formatVersionLabel(worktree.version, worktree.type, maxVersionLength)\n\n if (worktree.description) {\n return `${label} ${worktree.description}`\n }\n\n return label\n })\n\n logger.info('\uD83C\uDF3F Active worktrees:')\n logger.info(`\\n${formattedLines.join('\\n')}\\n`)\n\n const structuredContent = {\n worktrees,\n count: worktrees.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n}\n\n// MCP Tool Registration\nexport const worktreesListMcpTool = defineMcpTool({\n name: 'worktrees-list',\n description:\n 'List existing release-branch worktrees with version, release type (regular / hotfix), and Jira fix-version description. Read-only.',\n inputSchema: {},\n outputSchema: {\n worktrees: z\n .array(\n z.object({\n version: z.string().describe('Release version'),\n type: z.enum(['regular', 'hotfix']).describe('Release type'),\n description: z.string().nullable().describe('Jira version description'),\n }),\n )\n .describe('List of all worktrees with details'),\n count: z.number().describe('Number of worktrees'),\n },\n handler: worktreesList,\n})\n", "import { z } from 'zod'\nimport { $ } from 'zx'\n\nimport { buildCmuxWorkspaceTitle, listCmuxWorkspaceTitles, openCmuxWorkspaceWithLayout } from 'src/integrations/cmux'\nimport { reconcileCursorWorkspaceFolders, resolveCursorWorkspacePath } from 'src/integrations/cursor'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { defineMcpTool, textContent } from 'src/types'\n\ninterface WorktreesOpenResult {\n openedCmux: string[]\n skippedCmux: string[]\n cursorFoldersAdded: number\n cursorFoldersRemoved: number\n}\n\n/**\n * Cold-start restore command: reconciles `Main.code-workspace` against the set\n * of release worktrees on disk, opens Cursor against it, and ensures one cmux\n * workspace exists per worktree. Idempotent and additive \u2014 never removes\n * worktrees, never recreates running cmux workspaces.\n */\nexport const worktreesOpen = async () => {\n commandEcho.start('worktrees-open')\n\n try {\n const projectRoot = await getProjectRoot()\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n const currentBranches = await getCurrentWorktrees('release')\n\n const cursorOutcome = await openCursor({ projectRoot, worktreeDir, currentBranches })\n const cmuxOutcome = await openCmux({ worktreeDir, currentBranches })\n\n const result: WorktreesOpenResult = {\n openedCmux: cmuxOutcome.opened,\n skippedCmux: cmuxOutcome.skipped,\n cursorFoldersAdded: cursorOutcome.added,\n cursorFoldersRemoved: cursorOutcome.removed,\n }\n\n logResults(result, { cursorRan: cursorOutcome.ran, cmuxRan: cmuxOutcome.ran })\n\n commandEcho.print()\n\n return {\n content: textContent(JSON.stringify(result, null, 2)),\n structuredContent: { ...result },\n }\n } catch (error) {\n logger.error({ error }, '\u274C Error opening worktrees')\n throw new OperationError(error, {\n operation: 'open worktrees',\n remediation: \"run 'worktrees-list' to confirm the branches exist\",\n })\n }\n}\n\ninterface OpenCursorArgs {\n projectRoot: string\n worktreeDir: string\n currentBranches: string[]\n}\n\ninterface OpenCursorOutcome {\n ran: boolean\n added: number\n removed: number\n}\n\nconst openCursor = async (args: OpenCursorArgs): Promise<OpenCursorOutcome> => {\n const { projectRoot, worktreeDir, currentBranches } = args\n\n const config = await getInfraKitConfig()\n const cursorConfig = config.ide?.provider === 'cursor' ? config.ide.config : undefined\n\n if (!cursorConfig || cursorConfig.mode !== 'workspace' || !cursorConfig.workspaceConfigPath) {\n logger.warn('\u26A0\uFE0F Skipping Cursor: ide.provider must be \"cursor\", mode \"workspace\", and workspaceConfigPath set.')\n\n return { ran: false, added: 0, removed: 0 }\n }\n\n const workspacePath = resolveCursorWorkspacePath(cursorConfig.workspaceConfigPath, projectRoot)\n\n try {\n const { added, removed } = await reconcileCursorWorkspaceFolders({\n workspacePath,\n worktreeDir,\n currentBranches,\n })\n\n await $`cursor ${workspacePath}`\n\n return { ran: true, added: added.length, removed: removed.length }\n } catch (error) {\n logger.warn({ error }, `\u26A0\uFE0F Failed to reconcile/open Cursor workspace at ${workspacePath}`)\n\n return { ran: false, added: 0, removed: 0 }\n }\n}\n\ninterface OpenCmuxArgs {\n worktreeDir: string\n currentBranches: string[]\n}\n\ninterface OpenCmuxOutcome {\n ran: boolean\n opened: string[]\n skipped: string[]\n}\n\nconst openCmux = async (args: OpenCmuxArgs): Promise<OpenCmuxOutcome> => {\n const { worktreeDir, currentBranches } = args\n\n if (currentBranches.length === 0) {\n return { ran: true, opened: [], skipped: [] }\n }\n\n const repoName = await getRepoName()\n const existingTitles = await listCmuxWorkspaceTitles()\n\n const opened: string[] = []\n const skipped: string[] = []\n\n for (const branch of currentBranches) {\n const title = buildCmuxWorkspaceTitle({ repoName, branch })\n\n if (existingTitles.has(title)) {\n skipped.push(title)\n continue\n }\n\n try {\n await openCmuxWorkspaceWithLayout({ cwd: `${worktreeDir}/${branch}`, title })\n opened.push(title)\n } catch (error) {\n logger.warn({ error, title }, `\u26A0\uFE0F Failed to open cmux workspace for ${branch}`)\n }\n }\n\n return { ran: true, opened, skipped }\n}\n\ninterface LogResultsContext {\n cursorRan: boolean\n cmuxRan: boolean\n}\n\nconst logResults = (result: WorktreesOpenResult, context: LogResultsContext): void => {\n if (context.cursorRan) {\n if (result.cursorFoldersAdded > 0) {\n logger.info(`\u2705 Added ${result.cursorFoldersAdded} folder(s) to Cursor workspace`)\n }\n\n if (result.cursorFoldersRemoved > 0) {\n logger.info(`\uD83E\uDDF9 Removed ${result.cursorFoldersRemoved} dangling folder(s) from Cursor workspace`)\n }\n }\n\n if (result.openedCmux.length > 0) {\n logger.info('\u2705 Opened cmux workspaces:')\n for (const title of result.openedCmux) {\n logger.info(title)\n }\n }\n\n if (result.skippedCmux.length > 0) {\n logger.info(`\u2139\uFE0F Skipped ${result.skippedCmux.length} cmux workspace(s) already open`)\n }\n\n if (\n !context.cursorRan &&\n result.openedCmux.length === 0 &&\n result.skippedCmux.length === 0 &&\n result.cursorFoldersAdded === 0 &&\n result.cursorFoldersRemoved === 0\n ) {\n logger.info('\u2139\uFE0F Nothing to open')\n }\n}\n\n// MCP Tool Registration\nexport const worktreesOpenMcpTool = defineMcpTool({\n name: 'worktrees-open',\n description:\n 'Open Cursor against the configured workspace file and ensure a cmux workspace exists for each existing release worktree. Idempotent and additive \u2014 never removes worktrees, never recreates running cmux workspaces. Use after a cold start (Cursor + cmux closed). For stale-worktree cleanup, use worktrees-sync.',\n inputSchema: {},\n outputSchema: {\n openedCmux: z.array(z.string()).describe('Titles of cmux workspaces opened during this run'),\n skippedCmux: z.array(z.string()).describe('Titles of cmux workspaces that were already open'),\n cursorFoldersAdded: z.number().describe('Number of worktree folders added to the Cursor workspace file'),\n cursorFoldersRemoved: z\n .number()\n .describe('Number of dangling worktree folders removed from the Cursor workspace file'),\n },\n handler: worktreesOpen,\n})\n", "import checkbox from '@inquirer/checkbox'\nimport confirm from '@inquirer/confirm'\nimport process from 'node:process'\nimport { z } from 'zod'\n\nimport { removeFoldersFromCursorWorkspace, resolveCursorWorkspacePath } from 'src/integrations/cursor'\nimport { getReleasePRsWithInfo } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { formatBranchName, parseReleaseRef } from 'src/lib/release-id'\nimport { detectReleaseType, formatBranchChoices, getJiraDescriptions, releaseBranchLabels } from 'src/lib/release-utils'\nimport type { ReleaseType } from 'src/lib/release-utils'\nimport { removeWorktrees } from 'src/lib/worktrees'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\n// Constants\ninterface WorktreeManagementArgs extends RequiredConfirmedOptionArg {\n all?: boolean\n versions?: string\n}\n\n/**\n * Manage git worktrees for release branches\n * Creates worktrees for active release branches and removes unused ones\n */\nexport const worktreesRemove = async (options: WorktreeManagementArgs) => {\n const { confirmedCommand, all, versions } = options\n\n commandEcho.start('worktrees-remove')\n\n try {\n const currentWorktrees = await getCurrentWorktrees('release')\n\n if (currentWorktrees.length === 0) {\n logger.info('\u2139\uFE0F No active worktrees to remove')\n\n commandEcho.print()\n\n return {\n content: textContent(JSON.stringify({ removedWorktrees: [], count: 0 }, null, 2)),\n structuredContent: { removedWorktrees: [], count: 0 },\n }\n }\n\n const projectRoot = await getProjectRoot()\n\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n\n let selectedReleaseBranches: string[] = []\n\n if (all) {\n selectedReleaseBranches = currentWorktrees\n } else if (versions) {\n selectedReleaseBranches = versions.split(',').map((v) => {\n return formatBranchName(parseReleaseRef(v.trim()))\n })\n } else {\n commandEcho.setInteractive()\n\n const [descriptions, prInfo] = await Promise.all([getJiraDescriptions(), getReleasePRsWithInfo()])\n\n const releaseTypes = new Map<string, ReleaseType>(\n prInfo.map((pr) => {\n return [pr.branch, detectReleaseType(pr.title)]\n }),\n )\n\n selectedReleaseBranches = await checkbox({\n required: true,\n message: '\uD83C\uDF3F Select release branches',\n choices: formatBranchChoices({ branches: currentWorktrees, descriptions, types: releaseTypes }),\n })\n }\n\n // Track --all flag if all branches were selected (either via flag or interactively)\n const allSelected = selectedReleaseBranches.length === currentWorktrees.length\n\n if (allSelected) {\n commandEcho.addOption('--all', true)\n } else {\n commandEcho.addOption('--versions', releaseBranchLabels(selectedReleaseBranches))\n }\n\n // Ask for confirmation\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: 'Are you sure you want to proceed with these worktree changes?',\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n if (!confirmedCommand) {\n commandEcho.addOption('--yes', true)\n }\n\n const repoName = await getRepoName()\n\n const removedWorktrees = await removeWorktrees({\n branches: selectedReleaseBranches,\n worktreeDir,\n repoName,\n pruneFolder: allSelected,\n })\n\n await syncCursorWorkspaceOnRemove({ removedWorktrees, worktreeDir, projectRoot })\n\n logResults(removedWorktrees)\n\n commandEcho.print()\n\n const structuredContent = {\n removedWorktrees,\n count: removedWorktrees.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error) {\n logger.error({ error }, '\u274C Error managing worktrees')\n throw new OperationError(error, {\n operation: 'remove worktrees',\n remediation: \"check 'git worktree list' for the path; uncommitted changes block removal\",\n })\n }\n}\n\ninterface SyncCursorWorkspaceOnRemoveArgs {\n removedWorktrees: string[]\n worktreeDir: string\n projectRoot: string\n}\n\n/**\n * Strip removed worktrees from the configured Cursor workspace's `folders` array.\n * No-op if Cursor isn't configured, mode isn't \"workspace\", or no worktrees were removed.\n */\nconst syncCursorWorkspaceOnRemove = async (args: SyncCursorWorkspaceOnRemoveArgs): Promise<void> => {\n const { removedWorktrees, worktreeDir, projectRoot } = args\n\n if (removedWorktrees.length === 0) {\n return\n }\n\n const config = await getInfraKitConfig()\n const cursorConfig = config.ide?.provider === 'cursor' ? config.ide.config : undefined\n\n if (!cursorConfig || cursorConfig.mode !== 'workspace' || !cursorConfig.workspaceConfigPath) {\n return\n }\n\n const workspacePath = resolveCursorWorkspacePath(cursorConfig.workspaceConfigPath, projectRoot)\n\n const folderPaths = removedWorktrees.map((branch) => {\n return `${worktreeDir}/${branch}`\n })\n\n try {\n const { removed } = await removeFoldersFromCursorWorkspace({ workspacePath, folderPaths })\n\n if (removed.length > 0) {\n logger.info(`\u2705 Removed ${removed.length} folder(s) from ${workspacePath}`)\n }\n } catch (error) {\n logger.warn({ error }, `\u26A0\uFE0F Failed to update Cursor workspace at ${workspacePath}`)\n }\n}\n\n/**\n * Log the results of worktree management\n */\nconst logResults = (removed: string[]): void => {\n if (removed.length > 0) {\n logger.info('\u274C Removed worktrees:')\n for (const branch of removed) {\n logger.info(branch)\n }\n logger.info('')\n } else {\n logger.info('\u2139\uFE0F No unused worktrees to remove')\n }\n}\n\n// MCP Tool Registration\nexport const worktreesRemoveMcpTool = defineMcpTool({\n name: 'worktrees-remove',\n description:\n 'Remove local git worktrees for release branches. When everything is removed, also runs \"git worktree prune\" and deletes the worktrees directory. When invoked via MCP, pass either \"versions\" (comma-separated) or all=true \u2014 the branch picker is unreachable without a TTY, and the CLI confirmation is auto-skipped for MCP calls, so the caller is responsible for gating.',\n inputSchema: {\n all: z\n .boolean()\n .optional()\n .describe(\n 'Remove every existing worktree. Either \"all\" or \"versions\" must be provided for MCP calls (the interactive picker is unavailable without a TTY). Ignored if \"versions\" is provided.',\n ),\n versions: z\n .string()\n .optional()\n .describe(\n 'Comma-separated release versions or names to target (e.g. \"1.2.5, 1.2.6\" or \"checkout-redesign, 1.2.5\"). Either \"versions\" or all=true must be provided for MCP calls. Overrides \"all\" when set.',\n ),\n },\n outputSchema: {\n removedWorktrees: z.array(z.string()).describe('List of removed git worktree branches'),\n count: z.number().describe('Number of git worktrees removed'),\n },\n handler: worktreesRemove,\n})\n", "import confirm from '@inquirer/confirm'\nimport process from 'node:process'\nimport { z } from 'zod'\nimport { $ } from 'zx'\n\nimport { buildCmuxWorkspaceTitle, closeCmuxWorkspaceByTitle } from 'src/integrations/cmux'\nimport { removeFoldersFromCursorWorkspace, resolveCursorWorkspacePath } from 'src/integrations/cursor'\nimport { getReleasePRs } from 'src/integrations/gh'\nimport { commandEcho } from 'src/lib/command-echo'\nimport { WORKTREES_DIR_SUFFIX } from 'src/lib/constants'\nimport { OperationError } from 'src/lib/errors/operation-error'\nimport { getCurrentWorktrees, getProjectRoot, getRepoName } from 'src/lib/git-utils'\nimport { getInfraKitConfig } from 'src/lib/infra-kit-config'\nimport { logger } from 'src/lib/logger'\nimport { isReleaseBranch } from 'src/lib/release-id'\nimport { defineMcpTool, textContent } from 'src/types'\nimport type { RequiredConfirmedOptionArg } from 'src/types'\n\ninterface WorktreeSyncArgs extends RequiredConfirmedOptionArg {}\n\n/**\n * Manage git worktrees for release branches.\n *\n * Creates worktrees for active release branches and removes unused ones\n */\nexport const worktreesSync = async (options: WorktreeSyncArgs) => {\n const { confirmedCommand } = options\n\n commandEcho.start('worktrees-sync')\n\n try {\n const currentWorktrees = await getCurrentWorktrees('release')\n const projectRoot = await getProjectRoot()\n\n const worktreeDir = `${projectRoot}${WORKTREES_DIR_SUFFIX}`\n\n const releasePRsList = await getReleasePRs()\n\n // Ask for confirmation\n const answer = confirmedCommand\n ? true\n : await confirm({\n message: 'Are you sure you want to proceed with these worktree changes?',\n })\n\n if (!confirmedCommand) {\n commandEcho.setInteractive()\n }\n\n if (!answer) {\n logger.info('Operation cancelled. Exiting...')\n process.exit(0)\n }\n\n // Track --yes flag if confirmation was interactive (user confirmed)\n if (!confirmedCommand) {\n commandEcho.addOption('--yes', true)\n }\n\n const { branchesToRemove } = categorizeWorktrees({\n releasePRsList,\n currentWorktrees,\n })\n\n const repoName = await getRepoName()\n\n const removedWorktrees = await removeWorktrees({\n branches: branchesToRemove,\n worktreeDir,\n repoName,\n })\n\n await syncCursorWorkspaceOnRemove({ removedWorktrees, worktreeDir, projectRoot })\n\n logResults(removedWorktrees)\n\n commandEcho.print()\n\n const structuredContent = {\n removedWorktrees,\n count: removedWorktrees.length,\n }\n\n return {\n content: textContent(JSON.stringify(structuredContent, null, 2)),\n structuredContent,\n }\n } catch (error) {\n logger.error({ error }, '\u274C Error managing worktrees')\n throw new OperationError(error, {\n operation: 'sync worktrees with remote',\n remediation: \"ensure 'gh auth status' is ok and you can reach origin\",\n })\n }\n}\n\ninterface CategorizeWorktreesArgs {\n releasePRsList: string[]\n currentWorktrees: string[]\n}\n\n/**\n * Categorize worktrees into those that need to be created or removed\n */\nconst categorizeWorktrees = (args: CategorizeWorktreesArgs): { branchesToRemove: string[] } => {\n const { releasePRsList, currentWorktrees } = args\n\n const currentBranchNames = currentWorktrees.filter((branch) => {\n return isReleaseBranch(branch)\n })\n\n const branchesToRemove = currentBranchNames.filter((branch) => {\n return !releasePRsList.includes(branch)\n })\n\n return { branchesToRemove }\n}\n\ninterface RemoveWorktreesArgs {\n branches: string[]\n worktreeDir: string\n repoName: string\n}\n\n/**\n * Remove worktrees for the specified branches and close their cmux workspaces\n */\nconst removeWorktrees = async (args: RemoveWorktreesArgs): Promise<string[]> => {\n const { branches, worktreeDir, repoName } = args\n\n const removed: string[] = []\n\n for (const branch of branches) {\n try {\n const worktreePath = `${worktreeDir}/${branch}`\n\n const title = buildCmuxWorkspaceTitle({ repoName, branch })\n\n await closeCmuxWorkspaceByTitle(title)\n\n await $`git worktree remove ${worktreePath}`\n removed.push(branch)\n } catch (error) {\n const err = new OperationError(error, {\n operation: `remove stale worktree for ${branch}`,\n remediation: 'inspect the worktree dir manually; rerun with the branch checked out elsewhere',\n })\n\n logger.error({ error, branch, msg: err.message })\n }\n }\n\n return removed\n}\n\ninterface SyncCursorWorkspaceOnRemoveArgs {\n removedWorktrees: string[]\n worktreeDir: string\n projectRoot: string\n}\n\n/**\n * Strip removed worktrees from the configured Cursor workspace's `folders` array.\n * No-op if Cursor isn't configured, mode isn't \"workspace\", or no worktrees were removed.\n */\nconst syncCursorWorkspaceOnRemove = async (args: SyncCursorWorkspaceOnRemoveArgs): Promise<void> => {\n const { removedWorktrees, worktreeDir, projectRoot } = args\n\n if (removedWorktrees.length === 0) {\n return\n }\n\n const config = await getInfraKitConfig()\n const cursorConfig = config.ide?.provider === 'cursor' ? config.ide.config : undefined\n\n if (!cursorConfig || cursorConfig.mode !== 'workspace' || !cursorConfig.workspaceConfigPath) {\n return\n }\n\n const workspacePath = resolveCursorWorkspacePath(cursorConfig.workspaceConfigPath, projectRoot)\n\n const folderPaths = removedWorktrees.map((branch) => {\n return `${worktreeDir}/${branch}`\n })\n\n try {\n const { removed: removedEntries } = await removeFoldersFromCursorWorkspace({ workspacePath, folderPaths })\n\n if (removedEntries.length > 0) {\n logger.info(`\u2705 Removed ${removedEntries.length} folder(s) from ${workspacePath}`)\n }\n } catch (error) {\n logger.warn({ error }, `\u26A0\uFE0F Failed to update Cursor workspace at ${workspacePath}`)\n }\n}\n\n/**\n * Log the results of worktree management\n */\nconst logResults = (removed: string[]): void => {\n if (removed.length > 0) {\n logger.info('\u274C Removed worktrees:')\n for (const branch of removed) {\n logger.info(branch)\n }\n logger.info('')\n } else {\n logger.info('\u2139\uFE0F No unused worktrees to remove')\n }\n}\n\n// MCP Tool Registration\nexport const worktreesSyncMcpTool = defineMcpTool({\n name: 'worktrees-sync',\n description:\n 'Remove worktrees whose release PR is no longer open (stale cleanup). Only removes \u2014 never creates; use worktrees-add to create worktrees for new releases. The CLI confirmation is auto-skipped for MCP calls, so the caller is responsible for gating.',\n inputSchema: {},\n outputSchema: {\n removedWorktrees: z.array(z.string()).describe('List of removed worktree branches'),\n count: z.number().describe('Number of worktrees removed during sync'),\n },\n handler: worktreesSync,\n})\n", "import { logger } from 'src/lib/logger'\nimport type { ToolsExecutionResult } from 'src/types'\n\ninterface ToolHandlerArgs {\n toolName: string\n handler: (params: any) => Promise<ToolsExecutionResult>\n}\n\nexport const createToolHandler = (args: ToolHandlerArgs) => {\n return async (params: unknown) => {\n const { toolName, handler } = args\n\n logger.info({ msg: `Tool execution started: ${toolName}`, params })\n try {\n const payload = await handler({ ...(params as object), confirmedCommand: true })\n\n logger.info({ msg: `Tool execution successful: ${toolName}` })\n\n return payload\n } catch (error) {\n logger.error({\n err: error,\n params,\n msg: `Tool execution failed: ${toolName}`,\n })\n\n throw error\n }\n }\n}\n", "import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\n\nimport { auditMcpTool } from 'src/commands/audit'\nimport { envClearMcpTool } from 'src/commands/env-clear'\nimport { envListMcpTool } from 'src/commands/env-list'\nimport { envLoadMcpTool } from 'src/commands/env-load'\nimport { envStatusMcpTool } from 'src/commands/env-status'\nimport { ghMergeDevMcpTool } from 'src/commands/gh-merge-dev'\nimport { ghReleaseDeliverMcpTool } from 'src/commands/gh-release-deliver'\nimport { ghReleaseDeployAllMcpTool } from 'src/commands/gh-release-deploy-all'\nimport { ghReleaseDeploySelectedMcpTool } from 'src/commands/gh-release-deploy-selected'\nimport { ghReleaseListMcpTool } from 'src/commands/gh-release-list'\nimport { releaseCreateMcpTool } from 'src/commands/release-create'\nimport { releaseDescEditMcpTool } from 'src/commands/release-desc-edit'\nimport { versionMcpTool } from 'src/commands/version'\nimport { worktreesAddMcpTool } from 'src/commands/worktrees-add'\nimport { worktreesListMcpTool } from 'src/commands/worktrees-list'\nimport { worktreesOpenMcpTool } from 'src/commands/worktrees-open'\nimport { worktreesRemoveMcpTool } from 'src/commands/worktrees-remove'\nimport { worktreesSyncMcpTool } from 'src/commands/worktrees-sync'\nimport { createToolHandler } from 'src/lib/tool-handler'\n\nconst tools = [\n envStatusMcpTool,\n envListMcpTool,\n envLoadMcpTool,\n envClearMcpTool,\n ghMergeDevMcpTool,\n releaseCreateMcpTool,\n releaseDescEditMcpTool,\n ghReleaseDeliverMcpTool,\n ghReleaseDeployAllMcpTool,\n ghReleaseDeploySelectedMcpTool,\n ghReleaseListMcpTool,\n auditMcpTool,\n versionMcpTool,\n worktreesAddMcpTool,\n worktreesListMcpTool,\n worktreesOpenMcpTool,\n worktreesRemoveMcpTool,\n worktreesSyncMcpTool,\n]\n\nexport const initializeTools = async (server: McpServer) => {\n for (const tool of tools) {\n server.registerTool(\n tool.name,\n {\n description: tool.description,\n inputSchema: tool.inputSchema,\n outputSchema: tool.outputSchema,\n },\n createToolHandler({ toolName: tool.name, handler: tool.handler }),\n )\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["StdioServerTransport", "process", "process", "process", "pino", "pretty", "LOG_FILE_PATH", "initLoggerMcp", "logLevel", "logger", "initLoggerCLI", "ignoreFields", "setupErrorHandlers", "logger", "process", "error", "LOG_FILE_PATH", "reason", "promise", "McpServer", "initializePrompts", "_server", "initializeResources", "_server", "path", "process", "z", "path", "$", "VERSION_RE", "BRANCH_SEMVER_RE", "KEBAB_RE", "RELEASE_BRANCH_PREFIX", "VERSION_BRANCH_PREFIX", "NAME_BRANCH_PREFIX", "REFS_HEADS_PREFIX", "NEXT_TOKEN", "RESERVED_NAMES", "InvalidReleaseNameError", "message", "InvalidReleaseRefError", "stripRefsHeads", "input", "REFS_HEADS_PREFIX", "makeVersion", "major", "minor", "patch", "validateName", "name", "KEBAB_RE", "parseBranchName", "branch", "stripped", "VERSION_BRANCH_PREFIX", "semverPart", "match", "BRANCH_SEMVER_RE", "NAME_BRANCH_PREFIX", "namePart", "parseReleaseRef", "trimmed", "RELEASE_BRANCH_PREFIX", "parsed", "versionMatch", "VERSION_RE", "NEXT_TOKEN", "err", "reason", "formatBranchName", "id", "formatPrTitle", "type", "prefix", "formatRcTitle", "formatJiraName", "displayLabel", "isReleaseBranch", "toTime", "value", "time", "compareReleaseIds", "a", "b", "dates", "timeA", "timeB", "getCurrentWorktrees", "type", "worktreeLines", "$", "worktreePredicateMap", "releaseWorktreePredicate", "featureWorktreePredicate", "branch", "parseWorktreeBranch", "line", "trimmed", "open", "isReleaseBranch", "getProjectRoot", "getRepoName", "projectRoot", "path", "DEFAULT_RULES", "ROOT_DEFAULT_RULES", "resolvePackageConfig", "config", "baseline", "z", "packageConfigSchema", "fs", "pathExists", "target", "path", "fs", "path", "pathToFileURL", "z", "PACKAGE_CONFIG_FILE", "readPackageJson", "packageDir", "raw", "fs", "path", "loadPackageConfig", "baseline", "DEFAULT_RULES", "configPath", "pathExists", "stat", "rawExport", "pathToFileURL", "resolvedExport", "parsed", "packageConfigSchema", "z", "resolvePackageConfig", "fs", "path", "yaml", "WORKSPACE_FILE", "listChildDirs", "dir", "fs", "entry", "path", "expandSegment", "dirs", "segment", "next", "candidate", "pathExists", "expandGlob", "projectRoot", "pattern", "discoverPackages", "raw", "patterns", "yaml", "found", "checkConfig", "packageDir", "baseline", "DEFAULT_RULES", "rules", "loadPackageConfig", "PACKAGE_CONFIG_FILE", "err", "fs", "path", "checkFiles", "packageDir", "requiredFiles", "file", "stat", "checkScripts", "scripts", "requiredScripts", "script", "value", "fs", "path", "TURBO_FILE", "checkTurbo", "packageDir", "requiredTasks", "parsed", "raw", "err", "tasks", "task", "defined", "validatePackage", "packageDir", "baseline", "DEFAULT_RULES", "pkgJson", "readPackageJson", "packageName", "path", "configCheck", "rules", "checkConfig", "checks", "checkScripts", "checkFiles", "checkTurbo", "passed", "check", "textContent", "text", "defineMcpTool", "tool", "findPackageRoot", "start", "current", "path", "pathExists", "resolveTargets", "options", "getProjectRoot", "ROOT_DEFAULT_RULES", "discoverPackages", "dir", "process", "logResult", "result", "header", "logger", "check", "icon", "audit", "targets", "results", "target", "validatePackage", "allPassed", "structuredContent", "textContent", "auditInputSchema", "z", "auditOutputSchema", "auditMcpTool", "defineMcpTool", "params", "fs", "path", "process", "z", "fs", "os", "path", "process", "ENV_LOAD_FILE", "ENV_CLEAR_FILE", "INFRA_KIT_SESSION_VAR", "INFRA_KIT_ENV_CONFIG_VAR", "INFRA_KIT_ENV_PROJECT_VAR", "INFRA_KIT_ENV_LOADED_AT_VAR", "ENV_VAR_LINE_PATTERN", "parseVarNamesFromEnvFile", "filePath", "content", "names", "line", "match", "getCacheRoot", "xdg", "base", "getSessionCacheDir", "session", "atomicWriteFileSync", "mode", "tmpPath", "error", "WORKTREES_DIR_SUFFIX", "envClear", "cacheDir", "getSessionCacheDir", "envLoadPath", "path", "ENV_LOAD_FILE", "fs", "varNames", "parseVarNamesFromEnvFile", "unsetLines", "v", "INFRA_KIT_ENV_CONFIG_VAR", "INFRA_KIT_ENV_PROJECT_VAR", "INFRA_KIT_ENV_LOADED_AT_VAR", "clearFilePath", "ENV_CLEAR_FILE", "atomicWriteFileSync", "process", "structuredContent", "textContent", "envClearMcpTool", "defineMcpTool", "z", "z", "fs", "os", "path", "z", "INFRA_KIT_CONFIG_FILE", "USER_CONFIG_DIR_NAME", "USER_GLOBAL_CONFIG_FILE", "USER_PROJECTS_DIR", "dopplerEnvManagementSchema", "z", "envManagementSchema", "cursorIdeConfigSchema", "v", "cursorIdeSchema", "ideSchema", "jiraTaskManagerSchema", "taskManagerSchema", "worktreesConfigSchema", "infraKitConfigSchema", "infraKitOverrideConfigSchema", "cached", "getInfraKitConfigPaths", "projectRoot", "getProjectRoot", "projectName", "getRepoName", "userConfigDir", "path", "os", "getInfraKitConfig", "paths", "mainStat", "fs", "legacyYmlPath", "statIfExists", "userGlobalStat", "userProjectStat", "mtimes", "shallowEqual", "layers", "merged", "layer", "data", "loadLayer", "finalResult", "statIfExists", "filePath", "fs", "readIfExists", "shallowEqual", "a", "b", "keys", "k", "loadLayer", "layer", "raw", "parsedRaw", "err", "result", "infraKitOverrideConfigSchema", "z", "getDopplerProject", "envManagement", "getInfraKitConfig", "envList", "project", "getDopplerProject", "environments", "getInfraKitConfig", "logger", "env", "structuredContent", "textContent", "envListMcpTool", "defineMcpTool", "z", "select", "Buffer", "fs", "path", "process", "z", "$", "$", "validateDopplerCliAndAuth", "error", "createCommandEcho", "commandName", "options", "isInteractive", "name", "flag", "value", "formattedOptions", "opt", "logger", "commandEcho", "envLoad", "args", "validateDopplerCliAndAuth", "config", "commandEcho", "selectedConfig", "environments", "getInfraKitConfig", "select", "env", "process", "project", "getDopplerProject", "envContent", "downloadDopplerSecrets", "assertValidEnvContent", "loadedAt", "envFileLines", "INFRA_KIT_ENV_CONFIG_VAR", "shellSingleQuote", "INFRA_KIT_ENV_PROJECT_VAR", "INFRA_KIT_ENV_LOADED_AT_VAR", "cacheDir", "getSessionCacheDir", "envFilePath", "path", "ENV_LOAD_FILE", "fs", "atomicWriteFileSync", "varCount", "countEnvVarLines", "structuredContent", "textContent", "DOPPLER_MAX_OUTPUT_BYTES", "DOPPLER_DOWNLOAD_TIMEOUT_MS", "prevQuiet", "$", "result", "assertDopplerOutputSize", "stdout", "bytes", "Buffer", "content", "line", "ENV_VAR_LINE_PATTERN", "SHELL_DIRECTIVE_LINES", "value", "trimmed", "envLoadMcpTool", "defineMcpTool", "z", "path", "process", "z", "envStatus", "validateDopplerCliAndAuth", "logger", "cacheDir", "getSessionCacheDir", "sessionId", "process", "INFRA_KIT_SESSION_VAR", "envLoadPath", "path", "ENV_LOAD_FILE", "sessionLoadedCount", "sessionTotalCount", "sessionConfig", "INFRA_KIT_ENV_CONFIG_VAR", "sessionProject", "INFRA_KIT_ENV_PROJECT_VAR", "sessionLoadedAt", "INFRA_KIT_ENV_LOADED_AT_VAR", "varNames", "parseVarNamesFromEnvFile", "v", "loadedAtDisplay", "missing", "structuredContent", "textContent", "envStatusMcpTool", "defineMcpTool", "z", "checkbox", "confirm", "process", "z", "$", "$", "process", "$", "$", "process", "createJiraVersion", "params", "config", "baseUrl", "token", "email", "projectId", "requestBody", "url", "credentials", "response", "errorText", "logger", "error", "getProjectVersions", "findVersionByName", "versionName", "v", "updateJiraVersion", "deliverJiraRelease", "version", "loadJiraConfig", "process", "projectIdStr", "missingVars", "errorMessage", "loadJiraConfigOptional", "extractStderr", "cause", "stderr", "buildMessage", "ctx", "parts", "OperationError", "DEV_REF", "getBaseBranch", "type", "prepareGitForRelease", "baseBranch", "$", "createSingleRelease", "args", "id", "jiraConfig", "description", "versionName", "formatJiraName", "result", "createJiraVersion", "jiraVersionUrl", "releaseInfo", "createReleaseBranch", "displayLabel", "getJiraDescriptions", "descriptions", "loadJiraConfigOptional", "versions", "getProjectVersions", "version", "formatVersionLabel", "maxVersionLength", "padding", "tag", "detectReleaseType", "title", "parseBranchChoices", "branches", "branch", "parseBranchName", "resolveReleaseBranch", "versionArg", "formatBranchName", "parseReleaseRef", "error", "OperationError", "releaseLabelFromBranch", "releaseBranchLabels", "formatBranchChoices", "types", "parsed", "maxLen", "p", "label", "desc", "name", "sortReleasePRs", "prs", "pr", "parseBranchName", "entry", "a", "b", "compareReleaseIds", "fetchAllReleasePRs", "releasePRs", "$", "hotfixPRs", "all", "seen", "getReleasePRs", "logger", "process", "error", "getReleasePRsWithInfo", "updateReleasePRBody", "args", "branch", "body", "createReleaseBranch", "id", "jiraVersionUrl", "type", "description", "prTitle", "formatPrTitle", "baseBranch", "getBaseBranch", "branchName", "formatBranchName", "prLink", "ghMergeDev", "args", "all", "confirmedCommand", "commandEcho", "releasePRsList", "getReleasePRsWithInfo", "pr", "detectReleaseType", "logger", "textContent", "selectedReleaseBranches", "descriptions", "getJiraDescriptions", "checkbox", "formatBranchChoices", "releaseBranchLabels", "answer", "confirm", "process", "$", "failedBranches", "branch", "mergeDev", "structuredContent", "error", "err", "OperationError", "ghMergeDevMcpTool", "defineMcpTool", "z", "confirm", "select", "process", "z", "$", "readTrimmedString", "value", "max", "formatZxError", "error", "rec", "fields", "exitCode", "stderr", "stdout", "$", "$", "closeCmuxWorkspaceByTitle", "title", "listOutput", "$", "ref", "findWorkspaceRefByTitle", "error", "logger", "output", "rawLine", "match", "$", "listCmuxWorkspaceTitles", "output", "$", "titles", "rawLine", "match", "title", "error", "logger", "$", "openCmuxWorkspaceWithLayout", "args", "cwd", "title", "newWorkspaceOutput", "workspaceRef", "parseWorkspaceRef", "surfacesOutput", "leftTopRef", "parseFirstSurfaceRef", "output", "match", "buildCmuxWorkspaceTitle", "args", "repoName", "branch", "id", "parseBranchName", "label", "displayLabel", "removeWorktrees", "args", "branches", "worktreeDir", "repoName", "pruneFolder", "results", "branch", "worktreePath", "title", "buildCmuxWorkspaceTitle", "closeCmuxWorkspaceByTitle", "$", "removed", "index", "result", "err", "OperationError", "logger", "runStep", "operation", "remediation", "fn", "error", "logger", "formatZxError", "OperationError", "fetchPRByHead", "head", "result", "$", "fetchMergedRcPRForVersion", "id", "expectedTitle", "formatRcTitle", "pr", "fetchOpenDevToMainPR", "resolveTargetFromVersion", "version", "selectedReleaseBranch", "resolveReleaseBranch", "resolveTargetInteractively", "releasePRsInfo", "getReleasePRsWithInfo", "branches", "releaseTypes", "detectReleaseType", "commandEcho", "descriptions", "getJiraDescriptions", "select", "formatBranchChoices", "prInfo", "removeReleaseWorktreeIfPresent", "releaseBranch", "getCurrentWorktrees", "projectRoot", "repoName", "getProjectRoot", "getRepoName", "worktreeDir", "WORKTREES_DIR_SUFFIX", "removeWorktrees", "mergeReleasePR", "args", "releaseType", "mergeTarget", "releasePr", "resolveRcPRNumber", "selectedLabel", "displayLabel", "existingOpen", "rcNumber", "created", "ensureRcPRMerged", "dispatchDeployWorkflow", "syncMainIntoDev", "deliverJiraReleaseSafely", "jiraConfig", "loadJiraConfigOptional", "versionName", "formatJiraName", "deliverJiraRelease", "ghReleaseDeliver", "confirmedCommand", "releasePrTitle", "releaseId", "parseBranchName", "selectedVersion", "answer", "confirm", "process", "structuredContent", "textContent", "ghReleaseDeliverMcpTool", "defineMcpTool", "z", "select", "z", "$", "ghReleaseDeployAll", "args", "version", "env", "skipTerraform", "commandEcho", "selectedReleaseBranch", "resolveReleaseBranch", "releasePRsInfo", "getReleasePRsWithInfo", "branches", "pr", "releaseTypes", "detectReleaseType", "descriptions", "getJiraDescriptions", "select", "formatBranchChoices", "selectedVersion", "releaseLabelFromBranch", "environments", "getInfraKitConfig", "selectedEnv", "OperationError", "shouldSkipTerraform", "$", "logger", "structuredContent", "textContent", "error", "ghReleaseDeployAllMcpTool", "defineMcpTool", "z", "checkbox", "select", "fs", "resolve", "yaml", "z", "$", "ghReleaseDeploySelected", "args", "version", "env", "services", "skipTerraform", "commandEcho", "selectedReleaseBranch", "resolveReleaseBranch", "releasePRsInfo", "getReleasePRsWithInfo", "branches", "pr", "releaseTypes", "detectReleaseType", "descriptions", "getJiraDescriptions", "select", "formatBranchChoices", "selectedVersion", "releaseLabelFromBranch", "environments", "getInfraKitConfig", "selectedEnv", "OperationError", "availableServices", "parseServicesFromWorkflow", "selectedServices", "checkbox", "svc", "invalidServices", "shouldSkipTerraform", "$", "serviceFlags", "logger", "structuredContent", "textContent", "error", "projectRoot", "getProjectRoot", "workflowPath", "resolve", "content", "fs", "inputs", "yaml", "key", "value", "ghReleaseDeploySelectedMcpTool", "defineMcpTool", "z", "z", "ghReleaseList", "releases", "getReleasePRsWithInfo", "pr", "id", "parseBranchName", "displayLabel", "formatJiraName", "detectReleaseType", "jiraDescriptions", "getJiraDescriptions", "maxVersionLength", "r", "formattedLines", "release", "label", "formatVersionLabel", "description", "logger", "structuredContent", "textContent", "ghReleaseListMcpTool", "defineMcpTool", "z", "confirm", "select", "process", "z", "question", "$", "parseVersion", "versionStr", "sortVersions", "versions", "a", "b", "majA", "minA", "patchA", "majB", "minB", "patchB", "NEXT_TOKEN", "VERSION_RE", "stripBranchPrefix", "raw", "tryParse", "cleaned", "match", "semverKey", "v", "collectKnownVersions", "sources", "all", "parsed", "seen", "key", "sortVersions", "parseVersion", "NoPriorVersionsError", "computeNextVersion", "known", "type", "max", "major", "minor", "highestPatchOnMinor", "acc", "isNextToken", "token", "isNamedReleaseInput", "input", "withDescription", "base", "description", "resolveNamedInput", "input", "name", "validateName", "resolveReleaseEntries", "entries", "known", "running", "entry", "isNamedReleaseInput", "trimmed", "isNextToken", "next", "computeNextVersion", "parseVersion", "parseReleaseRef", "parsed", "tryParse", "explicit", "hasNextToken", "e", "extractVersionBranches", "lsRemoteStdout", "line", "tab", "parseBranchName", "id", "parseRemoteRefs", "previousQuiet", "$", "result", "fetchJiraVersionNames", "config", "loadJiraConfigOptional", "getProjectVersions", "v", "loadExistingVersions", "branchesResult", "jiraResult", "logger", "collectKnownVersions", "VERSION_PROMPT_HINT", "trySuggestNext", "known", "type", "computeNextVersion", "err", "NoPriorVersionsError", "resolveOrExit", "entries", "resolveReleaseEntries", "OperationError", "InvalidReleaseNameError", "promptForVersionInput", "running", "suggestion", "defaultHint", "versionAnswer", "question", "versionInput", "logger", "process", "promptForNameInput", "name", "validateName", "reason", "promptForReleasesInteractive", "ensureKnown", "commandEcho", "baseKnown", "ensureRunning", "addAnother", "ordinal", "kind", "select", "resolved", "parseVersion", "description", "confirm", "formatReleaseSummary", "entry", "parts", "echoReleases", "spec", "collectEntries", "inputReleases", "hasNextToken", "interactive", "confirmReleases", "confirmedCommand", "summary", "answer", "executeOne", "args", "jiraConfig", "label", "displayLabel", "prTitleLabel", "prepareGitForRelease", "result", "createSingleRelease", "error", "logFinalSummary", "total", "successCount", "failureCount", "releaseCreate", "loadJiraConfig", "loadExistingVersions", "created", "failed", "failure", "structuredContent", "r", "textContent", "releaseCreateMcpTool", "defineMcpTool", "z", "confirm", "select", "process", "z", "question", "buildJiraVersionUrl", "jiraConfig", "version", "buildPRBody", "jiraVersionUrl", "description", "pickReleaseBranch", "releasePRsInfo", "getReleasePRsWithInfo", "branches", "pr", "types", "detectReleaseType", "descriptions", "getJiraDescriptions", "branch", "select", "formatBranchChoices", "verifyReleasePRExists", "selectedBranch", "prInfo", "OperationError", "promptDescription", "current", "hint", "trimmed", "question", "releaseDescEdit", "args", "versionArg", "descriptionArg", "confirmedCommand", "commandEcho", "loadJiraConfig", "resolveReleaseBranch", "releaseId", "parseBranchName", "selectedVersion", "displayLabel", "versionName", "formatJiraName", "jiraVersion", "findVersionByName", "previousDescription", "newDescription", "logger", "structuredContent", "textContent", "answer", "confirm", "process", "updateJiraVersion", "body", "updateReleasePRBody", "releaseDescEditMcpTool", "defineMcpTool", "z", "z", "package_default", "version", "cliVersion", "package_default", "logger", "structuredContent", "textContent", "versionMcpTool", "defineMcpTool", "z", "checkbox", "confirm", "select", "process", "z", "$", "fs", "path", "addFoldersToCursorWorkspace", "args", "workspacePath", "folderPaths", "workspaceDir", "raw", "error", "parsed", "existingFolders", "existingAbsolutePaths", "entry", "added", "skipped", "folderPath", "absolutePath", "relativePath", "fs", "path", "fs", "path", "removeFoldersFromCursorWorkspace", "args", "workspacePath", "folderPaths", "workspaceDir", "raw", "error", "parsed", "existingFolders", "targetAbsolutePaths", "folderPath", "removedAbsolutePaths", "filteredFolders", "entry", "entryAbsolutePath", "removed", "notFound", "absolutePath", "reconcileCursorWorkspaceFolders", "args", "workspacePath", "worktreeDir", "currentBranches", "workspaceDir", "path", "releaseRoot", "raw", "fs", "existingFolders", "desiredAbsolutePaths", "branch", "danglingFolderPaths", "entry", "entryAbsolutePath", "removed", "removeFoldersFromCursorWorkspace", "desiredFolderPaths", "added", "addFoldersToCursorWorkspace", "path", "resolveCursorWorkspacePath", "configValue", "projectRoot", "FEATURE_DIR", "RELEASE_DIR", "CURSOR_MODES", "worktreesAdd", "options", "confirmedCommand", "all", "versions", "cursor", "githubDesktop", "cmux", "commandEcho", "currentWorktrees", "getCurrentWorktrees", "projectRoot", "getProjectRoot", "worktreeDir", "WORKTREES_DIR_SUFFIX", "ensureWorktreeDirectory", "selectedReleaseBranches", "v", "formatBranchName", "parseReleaseRef", "releasePRsInfo", "getReleasePRsWithInfo", "releasePRsList", "pr", "logger", "textContent", "releaseTypes", "detectReleaseType", "descriptions", "getJiraDescriptions", "checkbox", "formatBranchChoices", "releaseBranchLabels", "answer", "confirm", "process", "config", "getInfraKitConfig", "cursorConfig", "cursorMode", "select", "openInGithubDesktop", "openInCmux", "branchesToCreate", "categorizeWorktrees", "createdWorktrees", "createWorktrees", "logResults", "workspacePath", "resolveCursorWorkspacePath", "folderPaths", "branch", "added", "skipped", "addFoldersToCursorWorkspace", "skippedSuffix", "$", "repoName", "getRepoName", "title", "buildCmuxWorkspaceTitle", "openCmuxWorkspaceWithLayout", "structuredContent", "error", "OperationError", "args", "currentBranchNames", "isReleaseBranch", "branches", "results", "worktreePath", "created", "index", "result", "err", "worktreesAddMcpTool", "defineMcpTool", "z", "z", "worktreesList", "currentWorktrees", "getCurrentWorktrees", "logger", "textContent", "releasePRsInfo", "jiraDescriptions", "getReleasePRsWithInfo", "getJiraDescriptions", "releaseTypes", "pr", "detectReleaseType", "worktrees", "branch", "id", "parseBranchName", "version", "displayLabel", "type", "description", "formatJiraName", "maxVersionLength", "w", "formattedLines", "worktree", "label", "formatVersionLabel", "structuredContent", "worktreesListMcpTool", "defineMcpTool", "z", "z", "$", "worktreesOpen", "commandEcho", "projectRoot", "getProjectRoot", "worktreeDir", "WORKTREES_DIR_SUFFIX", "currentBranches", "getCurrentWorktrees", "cursorOutcome", "openCursor", "cmuxOutcome", "openCmux", "result", "logResults", "textContent", "error", "logger", "OperationError", "args", "config", "getInfraKitConfig", "cursorConfig", "workspacePath", "resolveCursorWorkspacePath", "added", "removed", "reconcileCursorWorkspaceFolders", "$", "repoName", "getRepoName", "existingTitles", "listCmuxWorkspaceTitles", "opened", "skipped", "branch", "title", "buildCmuxWorkspaceTitle", "openCmuxWorkspaceWithLayout", "context", "worktreesOpenMcpTool", "defineMcpTool", "z", "checkbox", "confirm", "process", "z", "worktreesRemove", "options", "confirmedCommand", "all", "versions", "commandEcho", "currentWorktrees", "getCurrentWorktrees", "logger", "textContent", "projectRoot", "getProjectRoot", "worktreeDir", "WORKTREES_DIR_SUFFIX", "selectedReleaseBranches", "v", "formatBranchName", "parseReleaseRef", "descriptions", "prInfo", "getJiraDescriptions", "getReleasePRsWithInfo", "releaseTypes", "pr", "detectReleaseType", "checkbox", "formatBranchChoices", "allSelected", "releaseBranchLabels", "answer", "confirm", "process", "repoName", "getRepoName", "removedWorktrees", "removeWorktrees", "syncCursorWorkspaceOnRemove", "logResults", "structuredContent", "error", "OperationError", "args", "config", "getInfraKitConfig", "cursorConfig", "workspacePath", "resolveCursorWorkspacePath", "folderPaths", "branch", "removed", "removeFoldersFromCursorWorkspace", "worktreesRemoveMcpTool", "defineMcpTool", "z", "confirm", "process", "z", "$", "worktreesSync", "options", "confirmedCommand", "commandEcho", "currentWorktrees", "getCurrentWorktrees", "projectRoot", "getProjectRoot", "worktreeDir", "WORKTREES_DIR_SUFFIX", "releasePRsList", "getReleasePRs", "answer", "confirm", "logger", "process", "branchesToRemove", "categorizeWorktrees", "repoName", "getRepoName", "removedWorktrees", "removeWorktrees", "syncCursorWorkspaceOnRemove", "logResults", "structuredContent", "textContent", "error", "OperationError", "args", "branch", "isReleaseBranch", "branches", "removed", "worktreePath", "title", "buildCmuxWorkspaceTitle", "closeCmuxWorkspaceByTitle", "$", "err", "config", "getInfraKitConfig", "cursorConfig", "workspacePath", "resolveCursorWorkspacePath", "folderPaths", "removedEntries", "removeFoldersFromCursorWorkspace", "worktreesSyncMcpTool", "defineMcpTool", "z", "createToolHandler", "args", "params", "toolName", "handler", "logger", "payload", "error", "tools", "envStatusMcpTool", "envListMcpTool", "envLoadMcpTool", "envClearMcpTool", "ghMergeDevMcpTool", "releaseCreateMcpTool", "releaseDescEditMcpTool", "ghReleaseDeliverMcpTool", "ghReleaseDeployAllMcpTool", "ghReleaseDeploySelectedMcpTool", "ghReleaseListMcpTool", "auditMcpTool", "versionMcpTool", "worktreesAddMcpTool", "worktreesListMcpTool", "worktreesOpenMcpTool", "worktreesRemoveMcpTool", "worktreesSyncMcpTool", "initializeTools", "server", "tool", "createToolHandler", "createMcpServer", "server", "McpServer", "initializePrompts", "initializeResources", "initializeTools", "logger", "initLoggerMcp", "startServer", "server", "createMcpServer", "error", "process", "transport", "StdioServerTransport", "setupErrorHandlers"]
7
7
  }