gen-pr 4.0.1 → 4.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/action.js CHANGED
@@ -1,2 +1,2 @@
1
- import e from"node:fs";import r from"node:os";import t from"node:path";import o from"@actions/core";import{l as i,D as n,a,m as d,n as s,b as u}from"./main-DA3u-sLa.js";import"yaml";import"ansis";import"dotenv";import"node:child_process";import"@octokit/graphql";import"@octokit/rest";import"@ai-sdk/amazon-bedrock";import"@ai-sdk/anthropic";import"@ai-sdk/azure";import"@ai-sdk/google";import"@ai-sdk/google-vertex";import"@ai-sdk/openai";import"@ai-sdk/xai";import"@openrouter/ai-sdk-provider";import"ai";import"ai-v4";import"ollama-ai-provider-v2";const m=i(),g=o.getInput("issue-number",{required:!0}),p=o.getInput("planning-model",{required:!1})||m["planning-model"],l="false"!==(o.getInput("two-staged-planning",{required:!1})||m["two-staged-planning"]),c=o.getInput("reasoning-effort",{required:!1})||m["reasoning-effort"],x="true"===(o.getInput("dry-run",{required:!1})||m["dry-run"]),f=o.getInput("coding-tool",{required:!1})||m["coding-tool"]||n,I=o.getInput("aider-extra-args",{required:!1})||m["aider-extra-args"],v=o.getInput("claude-code-extra-args",{required:!1})||m["claude-code-extra-args"],q=o.getInput("codex-extra-args",{required:!1})||m["codex-extra-args"],h=o.getInput("gemini-extra-args",{required:!1})||m["gemini-extra-args"],b=o.getInput("repomix-extra-args",{required:!1})||m["repomix-extra-args"],k=o.getInput("test-command",{required:!1})||m["test-command"],A=o.getInput("max-test-attempts",{required:!1})||m["max-test-attempts"],E=A?Number.parseInt(String(A),10):u,w=o.getInput("remove-pattern",{required:!1})||m["remove-pattern"],y="true"===(o.getInput("no-branch",{required:!1})||m["no-branch"]),N=o.getInput("node-runtime",{required:!1})||m["node-runtime"]||a;c&&!["low","medium","high"].includes(c)&&(console.error(`Invalid reasoning-effort value: ${c}. Using default. Valid values are: low, medium, high`),process.exit(1)),["aider","claude-code","codex-cli","gemini-cli"].includes(f)||(console.error(`Invalid coding-tool value: ${f}. Using default. Valid values are: aider, claude-code, codex-cli, gemini-cli`),process.exit(1)),N&&!["node","bun","npx","bunx"].includes(N)&&(console.error(`Invalid node-runtime value: ${N}. Using default. Valid values are: node (npx) and bun (bunx)`),process.exit(1));const S=s(N);e.rmSync(t.join(r.homedir(),".config","gh"),{force:!0,recursive:!0}),d({aiderExtraArgs:I,claudeCodeExtraArgs:v,codexExtraArgs:q,geminiExtraArgs:h,codingTool:f,twoStagePlanning:l,dryRun:x,noBranch:y,nodeRuntime:S,issueNumber:Number(g),maxTestAttempts:E,planningModel:p,reasoningEffort:c,repomixExtraArgs:b,testCommand:k,removePattern:w});
2
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"action.js","sources":["../src/action.ts"],"sourcesContent":["import fs from 'node:fs';\nimport os from 'node:os';\nimport path from 'node:path';\nimport core from '@actions/core';\nimport { loadConfigFile } from './config.js';\nimport { DEFAULT_CODING_TOOL, DEFAULT_MAX_TEST_ATTEMPTS, DEFAULT_NODE_RUNTIME } from './defaultOptions.js';\nimport { main } from './main.js';\nimport { normalizeNodeRuntime } from './spawn.js';\nimport type { CodingTool, NodeRuntime, NodeRuntimeActual, ReasoningEffort } from './types.js';\n\nconst configOptions = loadConfigFile();\n\n// Get inputs (GitHub Action inputs override config file values)\nconst issueNumber = core.getInput('issue-number', { required: true });\nconst planningModel =\n  core.getInput('planning-model', { required: false }) || (configOptions['planning-model'] as string);\nconst twoStagePlanningInput =\n  core.getInput('two-staged-planning', { required: false }) || (configOptions['two-staged-planning'] as string);\nconst twoStagePlanning = twoStagePlanningInput !== 'false';\nconst reasoningEffort = (core.getInput('reasoning-effort', { required: false }) ||\n  (configOptions['reasoning-effort'] as string)) as ReasoningEffort | undefined;\nconst dryRunInput = core.getInput('dry-run', { required: false }) || (configOptions['dry-run'] as string);\nconst dryRun = dryRunInput === 'true';\nconst codingTool = (core.getInput('coding-tool', { required: false }) ||\n  (configOptions['coding-tool'] as string) ||\n  DEFAULT_CODING_TOOL) as CodingTool;\nconst aiderExtraArgs =\n  core.getInput('aider-extra-args', { required: false }) || (configOptions['aider-extra-args'] as string);\nconst claudeCodeExtraArgs =\n  core.getInput('claude-code-extra-args', { required: false }) || (configOptions['claude-code-extra-args'] as string);\nconst codexExtraArgs =\n  core.getInput('codex-extra-args', { required: false }) || (configOptions['codex-extra-args'] as string);\nconst geminiExtraArgs =\n  core.getInput('gemini-extra-args', { required: false }) || (configOptions['gemini-extra-args'] as string);\nconst repomixExtraArgs =\n  core.getInput('repomix-extra-args', { required: false }) || (configOptions['repomix-extra-args'] as string);\nconst testCommand = core.getInput('test-command', { required: false }) || (configOptions['test-command'] as string);\nconst maxTestAttemptsInput =\n  core.getInput('max-test-attempts', { required: false }) || (configOptions['max-test-attempts'] as string | number);\nconst maxTestAttempts = maxTestAttemptsInput\n  ? Number.parseInt(String(maxTestAttemptsInput), 10)\n  : DEFAULT_MAX_TEST_ATTEMPTS;\nconst removePattern =\n  core.getInput('remove-pattern', { required: false }) || (configOptions['remove-pattern'] as string);\nconst noBranchInput = core.getInput('no-branch', { required: false }) || (configOptions['no-branch'] as string);\nconst noBranch = noBranchInput === 'true';\nconst nodeRuntimeInput = (core.getInput('node-runtime', { required: false }) ||\n  (configOptions['node-runtime'] as string) ||\n  DEFAULT_NODE_RUNTIME) as NodeRuntime;\n\nif (reasoningEffort && !['low', 'medium', 'high'].includes(reasoningEffort)) {\n  console.error(\n    `Invalid reasoning-effort value: ${reasoningEffort}. Using default. Valid values are: low, medium, high`\n  );\n  process.exit(1);\n}\n\nif (!['aider', 'claude-code', 'codex-cli', 'gemini-cli'].includes(codingTool)) {\n  console.error(\n    `Invalid coding-tool value: ${codingTool}. Using default. Valid values are: aider, claude-code, codex-cli, gemini-cli`\n  );\n  process.exit(1);\n}\n\nif (nodeRuntimeInput && !['node', 'bun', 'npx', 'bunx'].includes(nodeRuntimeInput)) {\n  console.error(\n    `Invalid node-runtime value: ${nodeRuntimeInput}. Using default. Valid values are: node (npx) and bun (bunx)`\n  );\n  process.exit(1);\n}\n\n// Normalize the runtime value (convert aliases to actual commands)\nconst nodeRuntime: NodeRuntimeActual = normalizeNodeRuntime(nodeRuntimeInput);\n\n// cf. https://github.com/cli/cli/issues/8441#issuecomment-1870271857\nfs.rmSync(path.join(os.homedir(), '.config', 'gh'), { force: true, recursive: true });\n\nvoid main({\n  aiderExtraArgs,\n  claudeCodeExtraArgs,\n  codexExtraArgs,\n  geminiExtraArgs,\n  codingTool,\n  twoStagePlanning,\n  dryRun,\n  noBranch,\n  nodeRuntime,\n  issueNumber: Number(issueNumber),\n  maxTestAttempts,\n  planningModel,\n  reasoningEffort,\n  repomixExtraArgs,\n  testCommand,\n  removePattern,\n});\n"],"names":["configOptions","loadConfigFile","issueNumber","core","getInput","required","planningModel","twoStagePlanning","reasoningEffort","dryRun","codingTool","DEFAULT_CODING_TOOL","aiderExtraArgs","claudeCodeExtraArgs","codexExtraArgs","geminiExtraArgs","repomixExtraArgs","testCommand","maxTestAttemptsInput","maxTestAttempts","Number","parseInt","String","DEFAULT_MAX_TEST_ATTEMPTS","removePattern","noBranch","nodeRuntimeInput","DEFAULT_NODE_RUNTIME","includes","console","error","process","exit","nodeRuntime","normalizeNodeRuntime","fs","rmSync","path","join","os","homedir","force","recursive","main"],"mappings":"uiBAUA,MAAMA,EAAgBC,IAGhBC,EAAcC,EAAKC,SAAS,eAAgB,CAAEC,UAAU,IACxDC,EACJH,EAAKC,SAAS,iBAAkB,CAAEC,UAAU,KAAaL,EAAc,kBAGnEO,EAA6C,WADjDJ,EAAKC,SAAS,sBAAuB,CAAEC,UAAU,KAAaL,EAAc,wBAExEQ,EAAmBL,EAAKC,SAAS,mBAAoB,CAAEC,UAAU,KACpEL,EAAc,oBAEXS,EAAyB,UADXN,EAAKC,SAAS,UAAW,CAAEC,UAAU,KAAaL,EAAc,YAE9EU,EAAcP,EAAKC,SAAS,cAAe,CAAEC,UAAU,KAC1DL,EAAc,gBACfW,EACIC,EACJT,EAAKC,SAAS,mBAAoB,CAAEC,UAAU,KAAaL,EAAc,oBACrEa,EACJV,EAAKC,SAAS,yBAA0B,CAAEC,UAAU,KAAaL,EAAc,0BAC3Ec,EACJX,EAAKC,SAAS,mBAAoB,CAAEC,UAAU,KAAaL,EAAc,oBACrEe,EACJZ,EAAKC,SAAS,oBAAqB,CAAEC,UAAU,KAAaL,EAAc,qBACtEgB,EACJb,EAAKC,SAAS,qBAAsB,CAAEC,UAAU,KAAaL,EAAc,sBACvEiB,EAAcd,EAAKC,SAAS,eAAgB,CAAEC,UAAU,KAAaL,EAAc,gBACnFkB,EACJf,EAAKC,SAAS,oBAAqB,CAAEC,UAAU,KAAaL,EAAc,qBACtEmB,EAAkBD,EACpBE,OAAOC,SAASC,OAAOJ,GAAuB,IAC9CK,EACEC,EACJrB,EAAKC,SAAS,iBAAkB,CAAEC,UAAU,KAAaL,EAAc,kBAEnEyB,EAA6B,UADbtB,EAAKC,SAAS,YAAa,CAAEC,UAAU,KAAaL,EAAc,cAElF0B,EAAoBvB,EAAKC,SAAS,eAAgB,CAAEC,UAAU,KACjEL,EAAc,iBACf2B,EAEEnB,IAAoB,CAAC,MAAO,SAAU,QAAQoB,SAASpB,KACzDqB,QAAQC,MACN,mCAAmCtB,yDAErCuB,QAAQC,KAAK,IAGV,CAAC,QAAS,cAAe,YAAa,cAAcJ,SAASlB,KAChEmB,QAAQC,MACN,8BAA8BpB,iFAEhCqB,QAAQC,KAAK,IAGXN,IAAqB,CAAC,OAAQ,MAAO,MAAO,QAAQE,SAASF,KAC/DG,QAAQC,MACN,+BAA+BJ,iEAEjCK,QAAQC,KAAK,IAIf,MAAMC,EAAiCC,EAAqBR,GAG5DS,EAAGC,OAAOC,EAAKC,KAAKC,EAAGC,UAAW,UAAW,MAAO,CAAEC,OAAO,EAAMC,WAAW,IAEzEC,EAAK,CACR/B,iBACAC,sBACAC,iBACAC,kBACAL,aACAH,mBACAE,SACAgB,WACAQ,cACA/B,YAAakB,OAAOlB,GACpBiB,kBACAb,gBACAE,kBACAQ,mBACAC,cACAO"}
1
+ import e from"node:fs";import r from"node:os";import t from"node:path";import o from"@actions/core";import{l as i,D as n,a,m as d,b as s}from"./main-DcT2jGgm.js";import"yaml";import"ansis";import"dotenv";import"node:child_process";import"@octokit/graphql";import"@octokit/rest";import"@ai-sdk/amazon-bedrock";import"@ai-sdk/anthropic";import"@ai-sdk/azure";import"@ai-sdk/google";import"@ai-sdk/google-vertex";import"@ai-sdk/openai";import"@ai-sdk/xai";import"@openrouter/ai-sdk-provider";import"ai";import"ai-v4";import"ollama-ai-provider-v2";const g=["low","medium","high"],m=["aider","claude-code","codex-cli","gemini-cli"],u=["node","bun","npx","bunx"];function p(e,r,t){e&&!r.includes(e)&&(console.error(`Invalid ${t} value: ${e}. Valid values are: ${r.join(", ")}`),process.exit(1))}const c=i(),l=o.getInput("issue-number",{required:!0}),x=o.getInput("planning-model",{required:!1})||c["planning-model"],f="false"!==(o.getInput("two-staged-planning",{required:!1})||String(c["two-staged-planning"]??"")),I=o.getInput("reasoning-effort",{required:!1})||c["reasoning-effort"],q="true"===(o.getInput("dry-run",{required:!1})||String(c["dry-run"]??"")),v=o.getInput("coding-tool",{required:!1})||c["coding-tool"]||n,b=o.getInput("aider-extra-args",{required:!1})||c["aider-extra-args"],h=o.getInput("claude-code-extra-args",{required:!1})||c["claude-code-extra-args"],k=o.getInput("codex-extra-args",{required:!1})||c["codex-extra-args"],E=o.getInput("gemini-extra-args",{required:!1})||c["gemini-extra-args"],S=o.getInput("repomix-extra-args",{required:!1})||c["repomix-extra-args"],A=o.getInput("test-command",{required:!1})||c["test-command"],y=o.getInput("max-test-attempts",{required:!1})||c["max-test-attempts"],w=y?Number.parseInt(String(y),10):s,R=o.getInput("remove-pattern",{required:!1})||c["remove-pattern"],T="true"===(o.getInput("no-branch",{required:!1})||String(c["no-branch"]??"")),j="true"===(o.getInput("verbose",{required:!1})||String(c.verbose??"")),N=o.getInput("node-runtime",{required:!1})||c["node-runtime"]||a;var $;p(($={reasoningEffort:I,codingTool:v,nodeRuntime:N}).reasoningEffort,g,"reasoning-effort"),p($.codingTool,m,"coding-tool"),p($.nodeRuntime,u,"node-runtime"),e.rmSync(t.join(r.homedir(),".config","gh"),{force:!0,recursive:!0}),d({aiderExtraArgs:b,claudeCodeExtraArgs:h,codexExtraArgs:k,geminiExtraArgs:E,codingTool:v,twoStagePlanning:f,dryRun:q,noBranch:T,nodeRuntime:N,issueNumber:Number(l),maxTestAttempts:w,planningModel:x,reasoningEffort:I,repomixExtraArgs:S,testCommand:A,removePattern:R,verbose:j});
2
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"action.js","sources":["../src/utils/validation.ts","../src/action.ts"],"sourcesContent":["/**\n * Validation constants and utilities\n */\n\nexport const VALID_REASONING_EFFORTS = ['low', 'medium', 'high'] as const;\nexport const VALID_CODING_TOOLS = ['aider', 'claude-code', 'codex-cli', 'gemini-cli'] as const;\nexport const VALID_NODE_RUNTIMES = ['node', 'bun', 'npx', 'bunx'] as const;\n\n/**\n * Validate a choice against allowed values\n */\nexport function validateChoice<T extends readonly string[]>(\n  value: string | undefined,\n  choices: T,\n  optionName: string\n): void {\n  if (value && !choices.includes(value as T[number])) {\n    console.error(`Invalid ${optionName} value: ${value}. Valid values are: ${choices.join(', ')}`);\n    process.exit(1);\n  }\n}\n\n/**\n * Validate all main options\n */\nexport function validateMainOptions(options: {\n  reasoningEffort?: string;\n  codingTool?: string;\n  nodeRuntime?: string;\n}): void {\n  validateChoice(options.reasoningEffort, VALID_REASONING_EFFORTS, 'reasoning-effort');\n  validateChoice(options.codingTool, VALID_CODING_TOOLS, 'coding-tool');\n  validateChoice(options.nodeRuntime, VALID_NODE_RUNTIMES, 'node-runtime');\n}\n","import fs from 'node:fs';\nimport os from 'node:os';\nimport path from 'node:path';\nimport core from '@actions/core';\nimport { loadConfigFile } from './config.js';\nimport { DEFAULT_CODING_TOOL, DEFAULT_MAX_TEST_ATTEMPTS, DEFAULT_NODE_RUNTIME } from './defaultOptions.js';\nimport { main } from './main.js';\nimport type { CodingTool, NodeRuntime, ReasoningEffort } from './types.js';\nimport { validateMainOptions } from './utils/validation.js';\n\nconst configOptions = loadConfigFile();\n\n// Get inputs (GitHub Action inputs override config file values)\nconst issueNumber = core.getInput('issue-number', { required: true });\nconst planningModel =\n  core.getInput('planning-model', { required: false }) || (configOptions['planning-model'] as string);\nconst twoStagePlanningInput =\n  core.getInput('two-staged-planning', { required: false }) || String(configOptions['two-staged-planning'] ?? '');\nconst twoStagePlanning = twoStagePlanningInput !== 'false';\nconst reasoningEffort = (core.getInput('reasoning-effort', { required: false }) ||\n  (configOptions['reasoning-effort'] as string)) as ReasoningEffort | undefined;\nconst dryRunInput = core.getInput('dry-run', { required: false }) || String(configOptions['dry-run'] ?? '');\nconst dryRun = dryRunInput === 'true';\nconst codingTool = (core.getInput('coding-tool', { required: false }) ||\n  (configOptions['coding-tool'] as string) ||\n  DEFAULT_CODING_TOOL) as CodingTool;\nconst aiderExtraArgs =\n  core.getInput('aider-extra-args', { required: false }) || (configOptions['aider-extra-args'] as string);\nconst claudeCodeExtraArgs =\n  core.getInput('claude-code-extra-args', { required: false }) || (configOptions['claude-code-extra-args'] as string);\nconst codexExtraArgs =\n  core.getInput('codex-extra-args', { required: false }) || (configOptions['codex-extra-args'] as string);\nconst geminiExtraArgs =\n  core.getInput('gemini-extra-args', { required: false }) || (configOptions['gemini-extra-args'] as string);\nconst repomixExtraArgs =\n  core.getInput('repomix-extra-args', { required: false }) || (configOptions['repomix-extra-args'] as string);\nconst testCommand = core.getInput('test-command', { required: false }) || (configOptions['test-command'] as string);\nconst maxTestAttemptsInput =\n  core.getInput('max-test-attempts', { required: false }) || (configOptions['max-test-attempts'] as string | number);\nconst maxTestAttempts = maxTestAttemptsInput\n  ? Number.parseInt(String(maxTestAttemptsInput), 10)\n  : DEFAULT_MAX_TEST_ATTEMPTS;\nconst removePattern =\n  core.getInput('remove-pattern', { required: false }) || (configOptions['remove-pattern'] as string);\nconst noBranchInput = core.getInput('no-branch', { required: false }) || String(configOptions['no-branch'] ?? '');\nconst noBranch = noBranchInput === 'true';\nconst verboseInput = core.getInput('verbose', { required: false }) || String(configOptions.verbose ?? '');\nconst verbose = verboseInput === 'true';\nconst nodeRuntime = (core.getInput('node-runtime', { required: false }) ||\n  (configOptions['node-runtime'] as string) ||\n  DEFAULT_NODE_RUNTIME) as NodeRuntime;\n\n// Validate all options using shared utility\nvalidateMainOptions({\n  reasoningEffort,\n  codingTool,\n  nodeRuntime,\n});\n\n// cf. https://github.com/cli/cli/issues/8441#issuecomment-1870271857\nfs.rmSync(path.join(os.homedir(), '.config', 'gh'), { force: true, recursive: true });\n\nvoid main({\n  aiderExtraArgs,\n  claudeCodeExtraArgs,\n  codexExtraArgs,\n  geminiExtraArgs,\n  codingTool,\n  twoStagePlanning,\n  dryRun,\n  noBranch,\n  nodeRuntime,\n  issueNumber: Number(issueNumber),\n  maxTestAttempts,\n  planningModel,\n  reasoningEffort,\n  repomixExtraArgs,\n  testCommand,\n  removePattern,\n  verbose,\n});\n"],"names":["VALID_REASONING_EFFORTS","VALID_CODING_TOOLS","VALID_NODE_RUNTIMES","validateChoice","value","choices","optionName","includes","console","error","join","process","exit","configOptions","loadConfigFile","issueNumber","core","getInput","required","planningModel","twoStagePlanning","String","reasoningEffort","dryRun","codingTool","DEFAULT_CODING_TOOL","aiderExtraArgs","claudeCodeExtraArgs","codexExtraArgs","geminiExtraArgs","repomixExtraArgs","testCommand","maxTestAttemptsInput","maxTestAttempts","Number","parseInt","DEFAULT_MAX_TEST_ATTEMPTS","removePattern","noBranch","verbose","nodeRuntime","DEFAULT_NODE_RUNTIME","options","fs","rmSync","path","os","homedir","force","recursive","main"],"mappings":"giBAIO,MAAMA,EAA0B,CAAC,MAAO,SAAU,QAC5CC,EAAqB,CAAC,QAAS,cAAe,YAAa,cAC3DC,EAAsB,CAAC,OAAQ,MAAO,MAAO,QAKnD,SAASC,EACdC,EACAC,EACAC,GAEIF,IAAUC,EAAQE,SAASH,KAC7BI,QAAQC,MAAM,WAAWH,YAAqBF,wBAA4BC,EAAQK,KAAK,SACvFC,QAAQC,KAAK,GAEjB,CCVA,MAAMC,EAAgBC,IAGhBC,EAAcC,EAAKC,SAAS,eAAgB,CAAEC,UAAU,IACxDC,EACJH,EAAKC,SAAS,iBAAkB,CAAEC,UAAU,KAAaL,EAAc,kBAGnEO,EAA6C,WADjDJ,EAAKC,SAAS,sBAAuB,CAAEC,UAAU,KAAYG,OAAOR,EAAc,wBAA0B,KAExGS,EAAmBN,EAAKC,SAAS,mBAAoB,CAAEC,UAAU,KACpEL,EAAc,oBAEXU,EAAyB,UADXP,EAAKC,SAAS,UAAW,CAAEC,UAAU,KAAYG,OAAOR,EAAc,YAAc,KAElGW,EAAcR,EAAKC,SAAS,cAAe,CAAEC,UAAU,KAC1DL,EAAc,gBACfY,EACIC,EACJV,EAAKC,SAAS,mBAAoB,CAAEC,UAAU,KAAaL,EAAc,oBACrEc,EACJX,EAAKC,SAAS,yBAA0B,CAAEC,UAAU,KAAaL,EAAc,0BAC3Ee,EACJZ,EAAKC,SAAS,mBAAoB,CAAEC,UAAU,KAAaL,EAAc,oBACrEgB,EACJb,EAAKC,SAAS,oBAAqB,CAAEC,UAAU,KAAaL,EAAc,qBACtEiB,EACJd,EAAKC,SAAS,qBAAsB,CAAEC,UAAU,KAAaL,EAAc,sBACvEkB,EAAcf,EAAKC,SAAS,eAAgB,CAAEC,UAAU,KAAaL,EAAc,gBACnFmB,EACJhB,EAAKC,SAAS,oBAAqB,CAAEC,UAAU,KAAaL,EAAc,qBACtEoB,EAAkBD,EACpBE,OAAOC,SAASd,OAAOW,GAAuB,IAC9CI,EACEC,EACJrB,EAAKC,SAAS,iBAAkB,CAAEC,UAAU,KAAaL,EAAc,kBAEnEyB,EAA6B,UADbtB,EAAKC,SAAS,YAAa,CAAEC,UAAU,KAAYG,OAAOR,EAAc,cAAgB,KAGxG0B,EAA2B,UADZvB,EAAKC,SAAS,UAAW,CAAEC,UAAU,KAAYG,OAAOR,EAAc0B,SAAW,KAEhGC,EAAexB,EAAKC,SAAS,eAAgB,CAAEC,UAAU,KAC5DL,EAAc,iBACf4B,EDzBK,IAA6BC,EAKlCvC,GALkCuC,EC4BhB,CAClBpB,kBACAE,aACAgB,gBD1BuBlB,gBAAiBtB,EAAyB,oBACjEG,EAAeuC,EAAQlB,WAAYvB,EAAoB,eACvDE,EAAeuC,EAAQF,YAAatC,EAAqB,gBC4B3DyC,EAAGC,OAAOC,EAAKnC,KAAKoC,EAAGC,UAAW,UAAW,MAAO,CAAEC,OAAO,EAAMC,WAAW,IAEzEC,EAAK,CACRxB,iBACAC,sBACAC,iBACAC,kBACAL,aACAJ,mBACAG,SACAe,WACAE,cACAzB,YAAamB,OAAOnB,GACpBkB,kBACAd,gBACAG,kBACAQ,mBACAC,cACAM,gBACAE"}
package/dist/cli.js CHANGED
@@ -1,2 +1,2 @@
1
- import e from"node:fs";import o from"node:path";import t from"node:process";import i from"yargs";import{hideBin as a}from"yargs/helpers";import{l as n,D as r,c as s,d as p,e as d,f as m,g as l,b as g,a as c,n as u,m as f}from"./main-DA3u-sLa.js";import"yaml";import"ansis";import"dotenv";import"node:child_process";import"@octokit/graphql";import"@octokit/rest";import"@ai-sdk/amazon-bedrock";import"@ai-sdk/anthropic";import"@ai-sdk/azure";import"@ai-sdk/google";import"@ai-sdk/google-vertex";import"@ai-sdk/openai";import"@ai-sdk/xai";import"@openrouter/ai-sdk-provider";import"ai";import"ai-v4";import"ollama-ai-provider-v2";const h=await i(a(t.argv)).config(n()).option("issue-number",{alias:"i",description:"GitHub issue number to process",type:"number",demandOption:!0}).option("planning-model",{alias:"m",description:"LLM for planning code changes. Must use llmlite format: provider/model (e.g., openai/gpt-4.1, azure/gpt-4.1, gemini/gemini-2.5-pro, anthropic/claude-4-sonnet-latest, bedrock/us.anthropic.claude-sonnet-4-20250514-v1:0, vertex/gemini-2.5-pro, xai/grok-4)",type:"string"}).option("two-staged-planning",{alias:"p",description:"Enable two-staged planning: first select relevant files, then generate detailed implementation plans (increases LLM cost but improves code quality)",type:"boolean",default:!0}).option("reasoning-effort",{alias:"e",description:"Constrains effort on reasoning for planning models. Supported values are low, medium, and high.",type:"string",choices:["low","medium","high"]}).option("coding-tool",{alias:"c",description:"Coding tool to use for making changes",type:"string",choices:["aider","claude-code","codex-cli","gemini-cli"],default:r}).option("aider-extra-args",{alias:"a",description:'Additional arguments to pass to Aider ("--yes-always --no-check-update --no-show-release-notes" is always applied)',type:"string",default:s}).option("claude-code-extra-args",{description:'Additional arguments to pass to Claude Code ("--dangerously-skip-permissions" is always applied, "--print" is applied only in CI)',type:"string",default:p}).option("codex-extra-args",{description:"Additional arguments to pass to Codex CLI (nothing is always applied)",type:"string",default:d}).option("gemini-extra-args",{description:'Additional arguments to pass to Gemini CLI ("--yolo" is always applied)',type:"string",default:m}).option("repomix-extra-args",{alias:"r",description:"Additional arguments for repomix when generating context",type:"string",default:l}).option("test-command",{alias:"t",description:"Command to run after the coding tool applies changes. If it fails, the assistant will try to fix it.",type:"string"}).option("max-test-attempts",{description:"Maximum number of attempts to fix test failures",type:"number",default:g}).option("dry-run",{alias:"d",description:"Run without making actual changes (no branch creation, no PR)",type:"boolean",default:!1}).option("remove-pattern",{description:"RegExp pattern to remove from issue and PR descriptions",type:"string"}).option("no-branch",{alias:"n",description:"Do not create a new branch, commit changes directly to the base branch",type:"boolean",default:!1}).option("node-runtime",{description:"Node.js runtime to use for running tools",type:"string",choices:["node","bun","npx","bunx"],default:c}).option("working-dir",{alias:"w",description:"Working directory path for commands",type:"string"}).version(function(){let t=import.meta.dir||o.dirname(new URL(import.meta.url).pathname);for(;!e.existsSync(o.join(t,"package.json"));)t=o.dirname(t);return JSON.parse(e.readFileSync(o.join(t,"package.json"),"utf8")).version}()).help().argv;h["working-dir"]&&(t.chdir(h["working-dir"]),console.info(`Changed working directory to: ${t.cwd()}`));const x=u(h["node-runtime"]);await f({aiderExtraArgs:h["aider-extra-args"],claudeCodeExtraArgs:h["claude-code-extra-args"],codexExtraArgs:h["codex-extra-args"],geminiExtraArgs:h["gemini-extra-args"],codingTool:h["coding-tool"],dryRun:h["dry-run"],noBranch:h["no-branch"],nodeRuntime:x,twoStagePlanning:h["two-staged-planning"],issueNumber:h["issue-number"],maxTestAttempts:h["max-test-attempts"],planningModel:h["planning-model"],reasoningEffort:h["reasoning-effort"],repomixExtraArgs:h["repomix-extra-args"],testCommand:h["test-command"],removePattern:h["remove-pattern"]});
2
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"cli.js","sources":["../src/cli.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport process from 'node:process';\nimport yargs from 'yargs';\nimport { hideBin } from 'yargs/helpers';\nimport { loadConfigFile } from './config.js';\nimport {\n  DEFAULT_AIDER_EXTRA_ARGS,\n  DEFAULT_CLAUDE_CODE_EXTRA_ARGS,\n  DEFAULT_CODEX_EXTRA_ARGS,\n  DEFAULT_CODING_TOOL,\n  DEFAULT_GEMINI_EXTRA_ARGS,\n  DEFAULT_MAX_TEST_ATTEMPTS,\n  DEFAULT_NODE_RUNTIME,\n  DEFAULT_REPOMIX_EXTRA_ARGS,\n} from './defaultOptions.js';\nimport { main } from './main.js';\nimport { normalizeNodeRuntime } from './spawn.js';\nimport type { CodingTool, NodeRuntime, NodeRuntimeActual, ReasoningEffort } from './types.js';\n\n// Parse command line arguments using yargs (CLI options override config)\nconst argv = await yargs(hideBin(process.argv))\n  .config(loadConfigFile())\n  // Options same with the GitHub Actions workflow\n  .option('issue-number', {\n    alias: 'i',\n    description: 'GitHub issue number to process',\n    type: 'number',\n    demandOption: true,\n  })\n  .option('planning-model', {\n    alias: 'm',\n    description:\n      'LLM for planning code changes. Must use llmlite format: provider/model (e.g., openai/gpt-4.1, azure/gpt-4.1, gemini/gemini-2.5-pro, anthropic/claude-4-sonnet-latest, bedrock/us.anthropic.claude-sonnet-4-20250514-v1:0, vertex/gemini-2.5-pro, xai/grok-4)',\n    type: 'string',\n  })\n  .option('two-staged-planning', {\n    alias: 'p',\n    description:\n      'Enable two-staged planning: first select relevant files, then generate detailed implementation plans (increases LLM cost but improves code quality)',\n    type: 'boolean',\n    default: true,\n  })\n  .option('reasoning-effort', {\n    alias: 'e',\n    description: 'Constrains effort on reasoning for planning models. Supported values are low, medium, and high.',\n    type: 'string',\n    choices: ['low', 'medium', 'high'],\n  })\n  .option('coding-tool', {\n    alias: 'c',\n    description: 'Coding tool to use for making changes',\n    type: 'string',\n    choices: ['aider', 'claude-code', 'codex-cli', 'gemini-cli'],\n    default: DEFAULT_CODING_TOOL,\n  })\n  .option('aider-extra-args', {\n    alias: 'a',\n    description:\n      'Additional arguments to pass to Aider (\"--yes-always --no-check-update --no-show-release-notes\" is always applied)',\n    type: 'string',\n    default: DEFAULT_AIDER_EXTRA_ARGS,\n  })\n  .option('claude-code-extra-args', {\n    description:\n      'Additional arguments to pass to Claude Code (\"--dangerously-skip-permissions\" is always applied, \"--print\" is applied only in CI)',\n    type: 'string',\n    default: DEFAULT_CLAUDE_CODE_EXTRA_ARGS,\n  })\n  .option('codex-extra-args', {\n    description: 'Additional arguments to pass to Codex CLI (nothing is always applied)',\n    type: 'string',\n    default: DEFAULT_CODEX_EXTRA_ARGS,\n  })\n  .option('gemini-extra-args', {\n    description: 'Additional arguments to pass to Gemini CLI (\"--yolo\" is always applied)',\n    type: 'string',\n    default: DEFAULT_GEMINI_EXTRA_ARGS,\n  })\n  .option('repomix-extra-args', {\n    alias: 'r',\n    description: 'Additional arguments for repomix when generating context',\n    type: 'string',\n    default: DEFAULT_REPOMIX_EXTRA_ARGS,\n  })\n  .option('test-command', {\n    alias: 't',\n    description: 'Command to run after the coding tool applies changes. If it fails, the assistant will try to fix it.',\n    type: 'string',\n  })\n  .option('max-test-attempts', {\n    description: 'Maximum number of attempts to fix test failures',\n    type: 'number',\n    default: DEFAULT_MAX_TEST_ATTEMPTS,\n  })\n  .option('dry-run', {\n    alias: 'd',\n    description: 'Run without making actual changes (no branch creation, no PR)',\n    type: 'boolean',\n    default: false,\n  })\n  .option('remove-pattern', {\n    description: 'RegExp pattern to remove from issue and PR descriptions',\n    type: 'string',\n  })\n  .option('no-branch', {\n    alias: 'n',\n    description: 'Do not create a new branch, commit changes directly to the base branch',\n    type: 'boolean',\n    default: false,\n  })\n  .option('node-runtime', {\n    description: 'Node.js runtime to use for running tools',\n    type: 'string',\n    choices: ['node', 'bun', 'npx', 'bunx'],\n    default: DEFAULT_NODE_RUNTIME,\n  })\n  // Options only for this standalone tool --------------------\n  .option('working-dir', {\n    alias: 'w',\n    description: 'Working directory path for commands',\n    type: 'string',\n  })\n  // ----------------------------------------------------------\n  .version(getVersion())\n  .help().argv;\n\nfunction getVersion(): string {\n  let packageJsonDir = import.meta.dir || path.dirname(new URL(import.meta.url).pathname);\n  while (!fs.existsSync(path.join(packageJsonDir, 'package.json'))) {\n    packageJsonDir = path.dirname(packageJsonDir);\n  }\n  return JSON.parse(fs.readFileSync(path.join(packageJsonDir, 'package.json'), 'utf8')).version;\n}\n\nif (argv['working-dir']) {\n  process.chdir(argv['working-dir']);\n  console.info(`Changed working directory to: ${process.cwd()}`);\n}\n\n// Normalize the runtime value (convert aliases to actual commands)\nconst nodeRuntime: NodeRuntimeActual = normalizeNodeRuntime(argv['node-runtime'] as NodeRuntime);\n\nawait main({\n  aiderExtraArgs: argv['aider-extra-args'],\n  claudeCodeExtraArgs: argv['claude-code-extra-args'],\n  codexExtraArgs: argv['codex-extra-args'],\n  geminiExtraArgs: argv['gemini-extra-args'],\n  codingTool: argv['coding-tool'] as CodingTool,\n  dryRun: argv['dry-run'],\n  noBranch: argv['no-branch'],\n  nodeRuntime,\n  twoStagePlanning: argv['two-staged-planning'],\n  issueNumber: argv['issue-number'],\n  maxTestAttempts: argv['max-test-attempts'],\n  planningModel: argv['planning-model'],\n  reasoningEffort: argv['reasoning-effort'] as ReasoningEffort,\n  repomixExtraArgs: argv['repomix-extra-args'],\n  testCommand: argv['test-command'],\n  removePattern: argv['remove-pattern'],\n});\n"],"names":["argv","yargs","hideBin","process","config","loadConfigFile","option","alias","description","type","demandOption","default","choices","DEFAULT_CODING_TOOL","DEFAULT_AIDER_EXTRA_ARGS","DEFAULT_CLAUDE_CODE_EXTRA_ARGS","DEFAULT_CODEX_EXTRA_ARGS","DEFAULT_GEMINI_EXTRA_ARGS","DEFAULT_REPOMIX_EXTRA_ARGS","DEFAULT_MAX_TEST_ATTEMPTS","DEFAULT_NODE_RUNTIME","version","packageJsonDir","dir","path","dirname","URL","url","pathname","fs","existsSync","join","JSON","parse","readFileSync","getVersion","help","chdir","console","info","cwd","nodeRuntime","normalizeNodeRuntime","main","aiderExtraArgs","claudeCodeExtraArgs","codexExtraArgs","geminiExtraArgs","codingTool","dryRun","noBranch","twoStagePlanning","issueNumber","maxTestAttempts","planningModel","reasoningEffort","repomixExtraArgs","testCommand","removePattern"],"mappings":"onBAqBA,MAAMA,QAAaC,EAAMC,EAAQC,EAAQH,OACtCI,OAAOC,KAEPC,OAAO,eAAgB,CACtBC,MAAO,IACPC,YAAa,iCACbC,KAAM,SACNC,cAAc,IAEfJ,OAAO,iBAAkB,CACxBC,MAAO,IACPC,YACE,+PACFC,KAAM,WAEPH,OAAO,sBAAuB,CAC7BC,MAAO,IACPC,YACE,sJACFC,KAAM,UACNE,SAAS,IAEVL,OAAO,mBAAoB,CAC1BC,MAAO,IACPC,YAAa,kGACbC,KAAM,SACNG,QAAS,CAAC,MAAO,SAAU,UAE5BN,OAAO,cAAe,CACrBC,MAAO,IACPC,YAAa,wCACbC,KAAM,SACNG,QAAS,CAAC,QAAS,cAAe,YAAa,cAC/CD,QAASE,IAEVP,OAAO,mBAAoB,CAC1BC,MAAO,IACPC,YACE,qHACFC,KAAM,SACNE,QAASG,IAEVR,OAAO,yBAA0B,CAChCE,YACE,oIACFC,KAAM,SACNE,QAASI,IAEVT,OAAO,mBAAoB,CAC1BE,YAAa,wEACbC,KAAM,SACNE,QAASK,IAEVV,OAAO,oBAAqB,CAC3BE,YAAa,0EACbC,KAAM,SACNE,QAASM,IAEVX,OAAO,qBAAsB,CAC5BC,MAAO,IACPC,YAAa,2DACbC,KAAM,SACNE,QAASO,IAEVZ,OAAO,eAAgB,CACtBC,MAAO,IACPC,YAAa,uGACbC,KAAM,WAEPH,OAAO,oBAAqB,CAC3BE,YAAa,kDACbC,KAAM,SACNE,QAASQ,IAEVb,OAAO,UAAW,CACjBC,MAAO,IACPC,YAAa,gEACbC,KAAM,UACNE,SAAS,IAEVL,OAAO,iBAAkB,CACxBE,YAAa,0DACbC,KAAM,WAEPH,OAAO,YAAa,CACnBC,MAAO,IACPC,YAAa,yEACbC,KAAM,UACNE,SAAS,IAEVL,OAAO,eAAgB,CACtBE,YAAa,2CACbC,KAAM,SACNG,QAAS,CAAC,OAAQ,MAAO,MAAO,QAChCD,QAASS,IAGVd,OAAO,cAAe,CACrBC,MAAO,IACPC,YAAa,sCACbC,KAAM,WAGPY,QAGH,WACE,IAAIC,cAA6BC,KAAOC,EAAKC,QAAQ,IAAIC,gBAAgBC,KAAKC,UAC9E,MAAQC,EAAGC,WAAWN,EAAKO,KAAKT,EAAgB,kBAC9CA,EAAiBE,EAAKC,QAAQH,GAEhC,OAAOU,KAAKC,MAAMJ,EAAGK,aAAaV,EAAKO,KAAKT,EAAgB,gBAAiB,SAASD,OACxF,CATWc,IACRC,OAAOpC,KAUNA,EAAK,iBACPG,EAAQkC,MAAMrC,EAAK,gBACnBsC,QAAQC,KAAK,iCAAiCpC,EAAQqC,UAIxD,MAAMC,EAAiCC,EAAqB1C,EAAK,uBAE3D2C,EAAK,CACTC,eAAgB5C,EAAK,oBACrB6C,oBAAqB7C,EAAK,0BAC1B8C,eAAgB9C,EAAK,oBACrB+C,gBAAiB/C,EAAK,qBACtBgD,WAAYhD,EAAK,eACjBiD,OAAQjD,EAAK,WACbkD,SAAUlD,EAAK,aACfyC,cACAU,iBAAkBnD,EAAK,uBACvBoD,YAAapD,EAAK,gBAClBqD,gBAAiBrD,EAAK,qBACtBsD,cAAetD,EAAK,kBACpBuD,gBAAiBvD,EAAK,oBACtBwD,iBAAkBxD,EAAK,sBACvByD,YAAazD,EAAK,gBAClB0D,cAAe1D,EAAK"}
1
+ import e from"node:fs";import o from"node:path";import t from"node:process";import i from"yargs";import{hideBin as a}from"yargs/helpers";import{l as n,D as r,c as s,d as p,e as d,f as m,g as l,b as g,a as c,m as u}from"./main-DcT2jGgm.js";import"yaml";import"ansis";import"dotenv";import"node:child_process";import"@octokit/graphql";import"@octokit/rest";import"@ai-sdk/amazon-bedrock";import"@ai-sdk/anthropic";import"@ai-sdk/azure";import"@ai-sdk/google";import"@ai-sdk/google-vertex";import"@ai-sdk/openai";import"@ai-sdk/xai";import"@openrouter/ai-sdk-provider";import"ai";import"ai-v4";import"ollama-ai-provider-v2";const f=await i(a(t.argv)).config(n()).option("issue-number",{alias:"i",description:"GitHub issue number to process",type:"number",demandOption:!0}).option("planning-model",{alias:"m",description:"LLM for planning code changes. Must use llmlite format: provider/model (e.g., openai/gpt-4.1, azure/gpt-4.1, gemini/gemini-2.5-pro, anthropic/claude-4-sonnet-latest, bedrock/us.anthropic.claude-sonnet-4-20250514-v1:0, vertex/gemini-2.5-pro, xai/grok-4)",type:"string"}).option("two-staged-planning",{alias:"p",description:"Enable two-staged planning: first select relevant files, then generate detailed implementation plans (increases LLM cost but improves code quality)",type:"boolean",default:!0}).option("reasoning-effort",{alias:"e",description:"Constrains effort on reasoning for planning models. Supported values are low, medium, and high.",type:"string",choices:["low","medium","high"]}).option("coding-tool",{alias:"c",description:"Coding tool to use for making changes",type:"string",choices:["aider","claude-code","codex-cli","gemini-cli"],default:r}).option("aider-extra-args",{alias:"a",description:'Additional arguments to pass to Aider ("--yes-always --no-check-update --no-show-release-notes" is always applied)',type:"string",default:s}).option("claude-code-extra-args",{description:'Additional arguments to pass to Claude Code ("--dangerously-skip-permissions" is always applied, "--print" is applied only in CI)',type:"string",default:p}).option("codex-extra-args",{description:"Additional arguments to pass to Codex CLI (nothing is always applied)",type:"string",default:d}).option("gemini-extra-args",{description:'Additional arguments to pass to Gemini CLI ("--yolo" is always applied)',type:"string",default:m}).option("repomix-extra-args",{alias:"r",description:"Additional arguments for repomix when generating context",type:"string",default:l}).option("test-command",{alias:"t",description:"Command to run after the coding tool applies changes. If it fails, the assistant will try to fix it.",type:"string"}).option("max-test-attempts",{description:"Maximum number of attempts to fix test failures",type:"number",default:g}).option("dry-run",{alias:"d",description:"Run without making actual changes (no branch creation, no PR)",type:"boolean",default:!1}).option("remove-pattern",{description:"RegExp pattern to remove from issue and PR descriptions",type:"string"}).option("no-branch",{alias:"n",description:"Do not create a new branch, commit changes directly to the base branch",type:"boolean",default:!1}).option("node-runtime",{description:"Node.js runtime to use for running tools",type:"string",choices:["node","bun","npx","bunx"],default:c}).option("working-dir",{alias:"w",description:"Working directory path for commands",type:"string"}).option("verbose",{alias:"v",description:"Print parsed options at start",type:"boolean",default:!1}).version(function(){let t=import.meta.dir||o.dirname(new URL(import.meta.url).pathname);for(;!e.existsSync(o.join(t,"package.json"));)t=o.dirname(t);return JSON.parse(e.readFileSync(o.join(t,"package.json"),"utf8")).version}()).help().argv;f["working-dir"]&&(t.chdir(f["working-dir"]),console.info(`Changed working directory to: ${t.cwd()}`)),await u({aiderExtraArgs:f["aider-extra-args"],claudeCodeExtraArgs:f["claude-code-extra-args"],codexExtraArgs:f["codex-extra-args"],geminiExtraArgs:f["gemini-extra-args"],codingTool:f["coding-tool"],dryRun:f["dry-run"],noBranch:f["no-branch"],nodeRuntime:f["node-runtime"],twoStagePlanning:f["two-staged-planning"],issueNumber:f["issue-number"],maxTestAttempts:f["max-test-attempts"],planningModel:f["planning-model"],reasoningEffort:f["reasoning-effort"],repomixExtraArgs:f["repomix-extra-args"],testCommand:f["test-command"],removePattern:f["remove-pattern"],verbose:f.verbose});
2
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"cli.js","sources":["../src/cli.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport process from 'node:process';\nimport yargs from 'yargs';\nimport { hideBin } from 'yargs/helpers';\nimport { loadConfigFile } from './config.js';\nimport {\n  DEFAULT_AIDER_EXTRA_ARGS,\n  DEFAULT_CLAUDE_CODE_EXTRA_ARGS,\n  DEFAULT_CODEX_EXTRA_ARGS,\n  DEFAULT_CODING_TOOL,\n  DEFAULT_GEMINI_EXTRA_ARGS,\n  DEFAULT_MAX_TEST_ATTEMPTS,\n  DEFAULT_NODE_RUNTIME,\n  DEFAULT_REPOMIX_EXTRA_ARGS,\n} from './defaultOptions.js';\nimport { main } from './main.js';\nimport type { CodingTool, NodeRuntime, ReasoningEffort } from './types.js';\n\n// Parse command line arguments using yargs (CLI options override config)\nconst argv = await yargs(hideBin(process.argv))\n  .config(loadConfigFile())\n  // Options same with the GitHub Actions workflow\n  .option('issue-number', {\n    alias: 'i',\n    description: 'GitHub issue number to process',\n    type: 'number',\n    demandOption: true,\n  })\n  .option('planning-model', {\n    alias: 'm',\n    description:\n      'LLM for planning code changes. Must use llmlite format: provider/model (e.g., openai/gpt-4.1, azure/gpt-4.1, gemini/gemini-2.5-pro, anthropic/claude-4-sonnet-latest, bedrock/us.anthropic.claude-sonnet-4-20250514-v1:0, vertex/gemini-2.5-pro, xai/grok-4)',\n    type: 'string',\n  })\n  .option('two-staged-planning', {\n    alias: 'p',\n    description:\n      'Enable two-staged planning: first select relevant files, then generate detailed implementation plans (increases LLM cost but improves code quality)',\n    type: 'boolean',\n    default: true,\n  })\n  .option('reasoning-effort', {\n    alias: 'e',\n    description: 'Constrains effort on reasoning for planning models. Supported values are low, medium, and high.',\n    type: 'string',\n    choices: ['low', 'medium', 'high'],\n  })\n  .option('coding-tool', {\n    alias: 'c',\n    description: 'Coding tool to use for making changes',\n    type: 'string',\n    choices: ['aider', 'claude-code', 'codex-cli', 'gemini-cli'],\n    default: DEFAULT_CODING_TOOL,\n  })\n  .option('aider-extra-args', {\n    alias: 'a',\n    description:\n      'Additional arguments to pass to Aider (\"--yes-always --no-check-update --no-show-release-notes\" is always applied)',\n    type: 'string',\n    default: DEFAULT_AIDER_EXTRA_ARGS,\n  })\n  .option('claude-code-extra-args', {\n    description:\n      'Additional arguments to pass to Claude Code (\"--dangerously-skip-permissions\" is always applied, \"--print\" is applied only in CI)',\n    type: 'string',\n    default: DEFAULT_CLAUDE_CODE_EXTRA_ARGS,\n  })\n  .option('codex-extra-args', {\n    description: 'Additional arguments to pass to Codex CLI (nothing is always applied)',\n    type: 'string',\n    default: DEFAULT_CODEX_EXTRA_ARGS,\n  })\n  .option('gemini-extra-args', {\n    description: 'Additional arguments to pass to Gemini CLI (\"--yolo\" is always applied)',\n    type: 'string',\n    default: DEFAULT_GEMINI_EXTRA_ARGS,\n  })\n  .option('repomix-extra-args', {\n    alias: 'r',\n    description: 'Additional arguments for repomix when generating context',\n    type: 'string',\n    default: DEFAULT_REPOMIX_EXTRA_ARGS,\n  })\n  .option('test-command', {\n    alias: 't',\n    description: 'Command to run after the coding tool applies changes. If it fails, the assistant will try to fix it.',\n    type: 'string',\n  })\n  .option('max-test-attempts', {\n    description: 'Maximum number of attempts to fix test failures',\n    type: 'number',\n    default: DEFAULT_MAX_TEST_ATTEMPTS,\n  })\n  .option('dry-run', {\n    alias: 'd',\n    description: 'Run without making actual changes (no branch creation, no PR)',\n    type: 'boolean',\n    default: false,\n  })\n  .option('remove-pattern', {\n    description: 'RegExp pattern to remove from issue and PR descriptions',\n    type: 'string',\n  })\n  .option('no-branch', {\n    alias: 'n',\n    description: 'Do not create a new branch, commit changes directly to the base branch',\n    type: 'boolean',\n    default: false,\n  })\n  .option('node-runtime', {\n    description: 'Node.js runtime to use for running tools',\n    type: 'string',\n    choices: ['node', 'bun', 'npx', 'bunx'],\n    default: DEFAULT_NODE_RUNTIME,\n  })\n  // Options only for this standalone tool --------------------\n  .option('working-dir', {\n    alias: 'w',\n    description: 'Working directory path for commands',\n    type: 'string',\n  })\n  .option('verbose', {\n    alias: 'v',\n    description: 'Print parsed options at start',\n    type: 'boolean',\n    default: false,\n  })\n  // ----------------------------------------------------------\n  .version(getVersion())\n  .help().argv;\n\nfunction getVersion(): string {\n  let packageJsonDir = import.meta.dir || path.dirname(new URL(import.meta.url).pathname);\n  while (!fs.existsSync(path.join(packageJsonDir, 'package.json'))) {\n    packageJsonDir = path.dirname(packageJsonDir);\n  }\n  return JSON.parse(fs.readFileSync(path.join(packageJsonDir, 'package.json'), 'utf8')).version;\n}\n\nif (argv['working-dir']) {\n  process.chdir(argv['working-dir']);\n  console.info(`Changed working directory to: ${process.cwd()}`);\n}\n\nawait main({\n  aiderExtraArgs: argv['aider-extra-args'],\n  claudeCodeExtraArgs: argv['claude-code-extra-args'],\n  codexExtraArgs: argv['codex-extra-args'],\n  geminiExtraArgs: argv['gemini-extra-args'],\n  codingTool: argv['coding-tool'] as CodingTool,\n  dryRun: argv['dry-run'],\n  noBranch: argv['no-branch'],\n  nodeRuntime: argv['node-runtime'] as NodeRuntime,\n  twoStagePlanning: argv['two-staged-planning'],\n  issueNumber: argv['issue-number'],\n  maxTestAttempts: argv['max-test-attempts'],\n  planningModel: argv['planning-model'],\n  reasoningEffort: argv['reasoning-effort'] as ReasoningEffort,\n  repomixExtraArgs: argv['repomix-extra-args'],\n  testCommand: argv['test-command'],\n  removePattern: argv['remove-pattern'],\n  verbose: argv.verbose,\n});\n"],"names":["argv","yargs","hideBin","process","config","loadConfigFile","option","alias","description","type","demandOption","default","choices","DEFAULT_CODING_TOOL","DEFAULT_AIDER_EXTRA_ARGS","DEFAULT_CLAUDE_CODE_EXTRA_ARGS","DEFAULT_CODEX_EXTRA_ARGS","DEFAULT_GEMINI_EXTRA_ARGS","DEFAULT_REPOMIX_EXTRA_ARGS","DEFAULT_MAX_TEST_ATTEMPTS","DEFAULT_NODE_RUNTIME","version","packageJsonDir","dir","path","dirname","URL","url","pathname","fs","existsSync","join","JSON","parse","readFileSync","getVersion","help","chdir","console","info","cwd","main","aiderExtraArgs","claudeCodeExtraArgs","codexExtraArgs","geminiExtraArgs","codingTool","dryRun","noBranch","nodeRuntime","twoStagePlanning","issueNumber","maxTestAttempts","planningModel","reasoningEffort","repomixExtraArgs","testCommand","removePattern","verbose"],"mappings":"6mBAoBA,MAAMA,QAAaC,EAAMC,EAAQC,EAAQH,OACtCI,OAAOC,KAEPC,OAAO,eAAgB,CACtBC,MAAO,IACPC,YAAa,iCACbC,KAAM,SACNC,cAAc,IAEfJ,OAAO,iBAAkB,CACxBC,MAAO,IACPC,YACE,+PACFC,KAAM,WAEPH,OAAO,sBAAuB,CAC7BC,MAAO,IACPC,YACE,sJACFC,KAAM,UACNE,SAAS,IAEVL,OAAO,mBAAoB,CAC1BC,MAAO,IACPC,YAAa,kGACbC,KAAM,SACNG,QAAS,CAAC,MAAO,SAAU,UAE5BN,OAAO,cAAe,CACrBC,MAAO,IACPC,YAAa,wCACbC,KAAM,SACNG,QAAS,CAAC,QAAS,cAAe,YAAa,cAC/CD,QAASE,IAEVP,OAAO,mBAAoB,CAC1BC,MAAO,IACPC,YACE,qHACFC,KAAM,SACNE,QAASG,IAEVR,OAAO,yBAA0B,CAChCE,YACE,oIACFC,KAAM,SACNE,QAASI,IAEVT,OAAO,mBAAoB,CAC1BE,YAAa,wEACbC,KAAM,SACNE,QAASK,IAEVV,OAAO,oBAAqB,CAC3BE,YAAa,0EACbC,KAAM,SACNE,QAASM,IAEVX,OAAO,qBAAsB,CAC5BC,MAAO,IACPC,YAAa,2DACbC,KAAM,SACNE,QAASO,IAEVZ,OAAO,eAAgB,CACtBC,MAAO,IACPC,YAAa,uGACbC,KAAM,WAEPH,OAAO,oBAAqB,CAC3BE,YAAa,kDACbC,KAAM,SACNE,QAASQ,IAEVb,OAAO,UAAW,CACjBC,MAAO,IACPC,YAAa,gEACbC,KAAM,UACNE,SAAS,IAEVL,OAAO,iBAAkB,CACxBE,YAAa,0DACbC,KAAM,WAEPH,OAAO,YAAa,CACnBC,MAAO,IACPC,YAAa,yEACbC,KAAM,UACNE,SAAS,IAEVL,OAAO,eAAgB,CACtBE,YAAa,2CACbC,KAAM,SACNG,QAAS,CAAC,OAAQ,MAAO,MAAO,QAChCD,QAASS,IAGVd,OAAO,cAAe,CACrBC,MAAO,IACPC,YAAa,sCACbC,KAAM,WAEPH,OAAO,UAAW,CACjBC,MAAO,IACPC,YAAa,gCACbC,KAAM,UACNE,SAAS,IAGVU,QAGH,WACE,IAAIC,cAA6BC,KAAOC,EAAKC,QAAQ,IAAIC,gBAAgBC,KAAKC,UAC9E,MAAQC,EAAGC,WAAWN,EAAKO,KAAKT,EAAgB,kBAC9CA,EAAiBE,EAAKC,QAAQH,GAEhC,OAAOU,KAAKC,MAAMJ,EAAGK,aAAaV,EAAKO,KAAKT,EAAgB,gBAAiB,SAASD,OACxF,CATWc,IACRC,OAAOpC,KAUNA,EAAK,iBACPG,EAAQkC,MAAMrC,EAAK,gBACnBsC,QAAQC,KAAK,iCAAiCpC,EAAQqC,gBAGlDC,EAAK,CACTC,eAAgB1C,EAAK,oBACrB2C,oBAAqB3C,EAAK,0BAC1B4C,eAAgB5C,EAAK,oBACrB6C,gBAAiB7C,EAAK,qBACtB8C,WAAY9C,EAAK,eACjB+C,OAAQ/C,EAAK,WACbgD,SAAUhD,EAAK,aACfiD,YAAajD,EAAK,gBAClBkD,iBAAkBlD,EAAK,uBACvBmD,YAAanD,EAAK,gBAClBoD,gBAAiBpD,EAAK,qBACtBqD,cAAerD,EAAK,kBACpBsD,gBAAiBtD,EAAK,oBACtBuD,iBAAkBvD,EAAK,sBACvBwD,YAAaxD,EAAK,gBAClByD,cAAezD,EAAK,kBACpB0D,QAAS1D,EAAK0D"}
@@ -0,0 +1,2 @@
1
+ import e from"node:fs";import t from"node:path";import n from"yaml";import o from"ansis";import{config as i}from"dotenv";import r,{spawn as s}from"node:child_process";import{graphql as a}from"@octokit/graphql";import{Octokit as c}from"@octokit/rest";import{createAmazonBedrock as l}from"@ai-sdk/amazon-bedrock";import{createAnthropic as u}from"@ai-sdk/anthropic";import{createAzure as d}from"@ai-sdk/azure";import{createGoogleGenerativeAI as f}from"@ai-sdk/google";import{createVertex as m}from"@ai-sdk/google-vertex";import{createOpenAI as p}from"@ai-sdk/openai";import{createXai as g}from"@ai-sdk/xai";import{createOpenRouter as h}from"@openrouter/ai-sdk-provider";import{generateText as w}from"ai";import{generateText as y}from"ai-v4";import{createOllama as $}from"ollama-ai-provider-v2";function b(){let o={};for(const i of["gen-pr.config.yml","gen-pr.config.yaml"]){const r=t.resolve(process.cwd(),i);if(e.existsSync(r)){try{o=n.parse(e.readFileSync(r,"utf8")),console.info(`Loaded gen-pr config from ${i}`)}catch(e){console.error(`Failed to parse config file ${i}:`,e),process.exit(1)}break}}return o}const v="--model gemini/gemini-2.5-pro --edit-format diff-fenced",x="--allowedTools Bash Edit Write",E="--full-auto",A="",P='--compress --remove-empty-lines --include "src/**/*.{ts,tsx},**/*.md"',R=5,_="aider",I="npx";const S="## gen-pr Metadata";function k(e,t){if(e.length>t){const n=e.slice(0,t),o=e.length-t;return`${n}\n\n... (${Math.floor(o)} characters truncated) ...`}return e}function O(e){return e.replaceAll("\r\n","\n").trim()}function N(e){switch(e){case"node":case"npx":return"npx";case"bun":case"bunx":return"bunx"}}async function G(e,t,n){const{ignoreExitStatus:i,...r}=n??{},s=t.map(e=>e.includes(" ")?`"${e.replaceAll('"','"')}"`:e).join(" ");console.info(o.green(`$ ${e} ${s}`)),console.info("stdout: ---------------------");const a=await C(e,t,r);r.truncateStdout&&console.info(k(a.stdout,3e3));const c=a.stderr.trim();if(c){console.info("stderr: ---------------------");const e=k(c,3e3);console.info(o.yellow(e))}return console.info("-----------------------------"),console.info(o.magenta(`Exit code: ${a.status}\n`)),i||0===a.status||null===a.status||process.exit(a.status),a}async function C(e,t,n){return new Promise((o,i)=>{try{const r=(t??[]).map(e=>e.replace(/\0/g,"")),a=s(e,r,n);a.stdout?.setEncoding?.("utf8"),a.stderr?.setEncoding?.("utf8");let c="",l="";a.stdout?.on("data",e=>{n?.truncateStdout||process.stdout.write(e),c+=e}),a.stderr?.on("data",e=>{l+=e}),a.on("error",e=>{i(e)}),a.on("close",(e,t)=>{void 0===a.pid?i(new Error("Process has no pid.")):o({pid:a.pid,stdout:c,stderr:l,status:e,signal:t})})}catch(e){i(e)}})}function T(e){if(!e)return[];const t=[];let n="",o=!1,i=!1;for(let r=0;r<e.length;r++){const s=e[r];'"'!==s||i?"'"!==s||o?" "!==s||o||i?n+=s:n&&(t.push(n),n=""):i=!i:o=!o}return n&&t.push(n),t}const M=process.env.GH_TOKEN||process.env.GITHUB_TOKEN||r.spawnSync("gh",["auth","token"],{encoding:"utf-8"}).stdout.trim();if(!M)throw new Error("GitHub token not found. Please set GH_TOKEN or GITHUB_TOKEN environment variable, or authenticate with gh CLI");const W=new c({auth:M}),H=a.defaults({headers:{authorization:`token ${M}`}});let L,Y;async function q(){if(!L||!Y){const{stdout:e}=await G("git",["remote","get-url","origin"],{ignoreExitStatus:!0}),t=e.trim().match(/github\.com[:/]([^/]+)\/(.+)/);if(!t?.[1]||!t[2])throw new Error("Could not parse GitHub repository from remote URL");L=t[1],Y=t[2].replace(/\.git$/,"")}if(!L||!Y)throw new Error("Repository information not properly initialized");return{owner:L,repo:Y}}async function F(e,t){try{const n=await async function(e){const{owner:t,repo:n}=await q();return(await W.pulls.get({owner:t,repo:n,pull_number:e,mediaType:{format:"diff"}})).data}(e);if(!n.trim())return;t.code_changes=function(e){const t=5e4,n=1e4,o=[/^diff --git a\/dist\//m,/^diff --git a\/build\//m,/^diff --git a\/.*\.bundle\./m,/^diff --git a\/.*\.min\./m,/^diff --git a\/node_modules\//m];if(e.length<=t)return e;const i=e.split(/(?=^diff --git)/m),r=[];let s=0;for(const e of i){if(!e.trim())continue;if(o.some(t=>t.test(e))){const t=[...e.split("\n").slice(0,4),"@@ ... @@","... (large bundled/compiled file diff truncated) ...",""].join("\n");r.push(t),s+=t.length}else if(e.length>n){const t=`${e.slice(0,n)}\n... (diff truncated) ...\n`;r.push(t),s+=t.length}else r.push(e),s+=e.length;if(s>.9*t){r.push("\n... (remaining diffs truncated) ...\n");break}}return r.join("")}(n.trim())}catch(e){console.warn("Failed to fetch PR diff:",e)}}async function K(e,t,n,o=!1){if(t.has(e))return;let i;t.add(e);try{i=await async function(e){const{owner:t,repo:n}=await q(),{data:o}=await W.issues.get({owner:t,repo:n,issue_number:e});let i=o;if(o.pull_request){const{data:o}=await W.pulls.get({owner:t,repo:n,pull_number:e});i=o}const r=await W.issues.listComments({owner:t,repo:n,issue_number:e});return{author:i.user?.login||"",title:i.title,body:i.body||"",labels:i.labels.map(e=>({name:"string"==typeof e?e:e.name||""})),comments:r.data.map(e=>({author:e.user?.login||"",body:e.body||"",createdAt:e.created_at})),url:i.html_url}}(e)}catch(t){return void console.warn(`Failed to fetch issue #${e}:`,t)}const r=function(e){const t=/(?:^|\s)#(\d+)/g,n=[];for(;;){const o=t.exec(e);if(!o)break;const i=Number.parseInt(o[1]??"",10);Number.isInteger(i)&&n.push(i)}return[...new Set(n)]}([i.body,...i.comments.map(e=>e.body)].join("\n")),s=i.body.replace(/<!--[\s\S]*?-->/g,"");const a=i.url?.includes("/pull/")?function(e){const t=e.indexOf(S);return t>=0?e.substring(0,t):e}(s):s,c=function(e,t){if(!t)return e;try{const n=new RegExp(t,"g");return e.replace(n,"")}catch(n){return console.warn(`Invalid regex pattern "${t}":`,n),e}}(a,n.removePattern||""),l=i.comments.map(e=>({author:e.author,body:O(e.body),createdAt:new Date(e.createdAt).getTime()})),u={author:i.author,title:i.title,description:O(c),comments:[]};return i.url?.includes("/pull/")&&!o&&(await F(e,u),await async function(e,t){try{const n=await async function(e){const{owner:t,repo:n}=await q();return await H("\n query($owner: String!, $repo: String!, $pr: Int!) {\n repository(owner: $owner, name: $repo) {\n pullRequest(number: $pr) {\n reviewThreads(first: 100) {\n nodes {\n isResolved\n comments(first: 100) {\n nodes {\n author {\n login\n }\n body\n path\n line\n diffHunk\n createdAt\n }\n }\n }\n }\n }\n }\n }\n ",{owner:t,repo:n,pr:e})}(e),o=n?.repository?.pullRequest?.reviewThreads?.nodes||[];for(const e of o)!e.isResolved&&e.comments?.nodes&&B(e.comments.nodes,t)}catch(e){console.warn("Failed to fetch PR review threads:",e)}}(e,l),await async function(e,t){try{const n=(await async function(e){const{owner:t,repo:n}=await q();return(await W.pulls.listReviews({owner:t,repo:n,pull_number:e})).data.map(e=>({user:{login:e.user?.login||""},state:e.state,body:e.body||"",submitted_at:e.submitted_at||""}))}(e)).map(e=>({author:e.user.login,reviewState:e.state,body:O(e.body),createdAt:new Date(e.submitted_at).getTime()}));t.push(...n)}catch(e){console.warn("Failed to fetch PR reviews:",e)}}(e,l)),r.length>0&&await async function(e,t,n,o){const i=e.map(e=>K(e,t,n,!0)),r=(await Promise.all(i)).filter(e=>!!e);if(0===r.length)return;o.referenced_issues=r}(r,t,n,u),u.comments=l.filter(e=>e.body).sort((e,t)=>e.createdAt-t.createdAt).map(({createdAt:e,...t})=>t),u}function B(e,t){for(const n of e){if(!n.author||!n.body)continue;const e=D(n.diffHunk),o={author:n.author.login,codeLocation:n.path&&n.line?`${n.path}:${n.line}`:void 0,codeContent:e||void 0,body:O(n.body),createdAt:new Date(n.createdAt).getTime()};Object.keys(o).forEach(e=>{void 0===o[e]&&delete o[e]}),t.push(o)}}function D(e){if(!e)return"";const t=e.split("\n").find(e=>(e.startsWith("+")||e.startsWith("-"))&&!e.startsWith("@@")&&e.trim().length>1);return t?.trim()||""}function j(e,t){const n=`\n${e}`,o=t.map(e=>n.indexOf(`\n${e}`));if(!o.some(e=>-1===e)&&o.every((e,t)=>0===t||e>o[t-1]))return t.map((e,i)=>{const r=o[i]+1+e.length,s=i+1<t.length?o[i+1]+1:n.length;return n.slice(r,s).trim()})}function z(e,t){const n=new RegExp(`${t}{3,}`,"g"),o=e.match(n),i=o?Math.max(...o.map(e=>e.length)):0,r=Math.max(3,i+1);return t.repeat(r)}function U(e){return e.trim().replace(/^(`{3,}|~{3,})[\s\S]*?\n([\s\S]*?)\n\1\s*$/,"$2")}function J(e){return e.map((e,t)=>({id:`msg-${t}`,role:"tool"===e.role?"data":e.role,content:"string"==typeof e.content?e.content:JSON.stringify(e.content)}))}const V={blockQuote:"literal",lineWidth:0};async function Q(e,t,n){try{if(e.startsWith("ollama/"))return await async function(e,t,n){const[o,...i]=e.split("/");try{const r=i.join("/");r||(console.error(`Invalid ${o} model format: ${e}. Expected format: ${o}/model-name`),process.exit(1));const s=`${process.env.OLLAMA_BASE_URL||"http://localhost:11434"}/api`,a=$({baseURL:s,...process.env.OLLAMA_API_KEY&&{apiKey:process.env.OLLAMA_API_KEY}})(r),c=await y({model:a,providerOptions:n?{ollama:{think:!0}}:void 0,messages:J(t)});return X(e,c),c.text}catch(t){console.error(`${o.charAt(0).toUpperCase()+o.slice(1)} API error for model ${e}:`,t),process.exit(1)}}(e,t,n);const[o,i,r]=function(e,t){e.includes("/")||(console.error(`Model must be in format 'provider/model'. Got: ${e}`),process.exit(1));const[n,...o]=e.split("/"),i=o.join("/");switch(n){case"openai":return[p()(i),n,i];case"anthropic":return[u()(i),n,i];case"gemini":return[f()(i),n,i];case"azure":return[d()(i),n,i];case"bedrock":return[l()(i),n,i];case"vertex":return[m()(i),n,i];case"xai":return[g()(i),n,i];case"openrouter":return[h({apiKey:process.env.OPENROUTER_API_KEY,headers:{"HTTP-Referer":"https://github.com/WillBooster/gen-pr","X-Title":"gen-pr"}})(i,t?{reasoning:{effort:t}}:{}),n,i];default:console.error(`Unsupported provider: ${n}. Supported providers: openai, azure, google, anthropic, bedrock, vertex, grok, openrouter, ollama`),process.exit(1)}}(e,n),s={model:o,messages:t};if(n){const t=function(e,t){switch(e){case"openai":case"azure":return/^(o1|o3|o4)/.test(t);case"anthropic":return/^claude-(opus-4|sonnet-4|3-7-sonnet)/.test(t);case"gemini":return/^gemini-2\.5/.test(t);case"bedrock":return/^(us\.)?anthropic\.claude-(opus-4|sonnet-4|3-7-sonnet)/.test(t);case"vertex":return/^gemini-2\.5/.test(t)||/^claude-(3-7-sonnet|opus-4|sonnet-4)/.test(t);case"xai":return/^grok-3/.test(t);case"openrouter":return!0;default:return!1}}(i,r);if(t){const e=function(e){return{low:4e3,medium:8e3,high:24e3}[e]}(n);"openai"===i?s.providerOptions={openai:{reasoningEffort:n}}:"anthropic"===i?s.providerOptions={anthropic:{thinking:{type:"enabled",budgetTokens:e}}}:"gemini"===i?s.providerOptions={google:{thinkingConfig:{thinkingBudget:e}}}:"bedrock"===i?console.warn(`Note: The current AI SDK doesn't work on Bedrock with reasoning. Model ${r} will use default reasoning settings.`):"xai"===i&&(s.providerOptions={xai:{reasoningEffort:n}})}else console.warn(`Model ${e} does not support reasoning/thinking options. Ignoring reasoning effort parameter.`)}const a=await w(s);return X(e,a),a.text}catch(t){console.error(`LLM API error for model ${e}:`,t),process.exit(1)}}function X(e,t){console.info(`${e}:`,n.stringify({text:t.text,usage:t.usage,finishReason:t.finishReason},V))}const Z="repomix.result",ee="## File Paths to be Modified",te="## File Paths to be Referred",ne="## Implementation Plan",oe="## Commit Message";function ie(e,t=!1){const n=t?"pull request":"issue";return`\nYou are an expert software developer tasked with analyzing GitHub ${n}s and identifying relevant files for code changes.\n\nReview the following GitHub ${n} and the list of available file paths and their contents (which will be provided in a separate message).${t?" Consider the comments on the pull request when identifying files.":""}\nYour task is to identify:\n1. Files that need to be MODIFIED to resolve the ${n}\n2. Files that should be REFERRED to (but not modified) to understand the codebase better\n\nGitHub ${t?"Pull Request":"Issue"}:\n${e}\n\nPlease format your response without any explanatory text as follows:\n\`\`\`md\n${ee}\n\n- \`[filePath1]\`\n- \`[filePath2]\`\n- ...\n\n${te}\n\n- \`[filePath1]\`\n- \`[filePath2]\`\n- ...\n\`\`\`\n`}function re(e,t=!1){const n=t?"pull request":"issue";return`\nYou are an expert software developer tasked with creating an implementation plan based on GitHub ${n}s.\n\nReview the following GitHub ${n} and the provided file contents (which will be provided in a separate message).${t?" Consider the comments on the pull request when creating the plan.":""}\nCreate a detailed, step-by-step plan outlining how to address the ${n} effectively.\nAlso, provide a concise and descriptive commit message for the changes, following the Conventional Commits specification.\n\nYour plan should:\n- Focus on implementation details for each file that needs modification\n- Be clear and actionable for a developer to follow\n- Prefer showing diffs rather than complete file contents when describing changes\n- Exclude testing procedures unless users explicitly request\n\nGitHub ${t?"Pull Request":"Issue"}:\n${e}\n\nPlease format your response without any explanatory text as follows:\n\`\`\`md\n${ne}\n\n1. [Specific implementation step]\n2. [Next implementation step]\n...\n\n${oe}\n\n[commit message]\n\`\`\`\n`.trim()}function se(e,t=!1){const n=t?"pull request":"issue";return`\nYou are an expert software developer tasked with analyzing GitHub ${n}s and creating an implementation plan.\n\nReview the following GitHub ${n} and the list of available file paths and their contents (which will be provided in a separate message).${t?" Consider the comments on the pull request when creating the plan.":""}\nYour task is to:\n1. Create a detailed, step-by-step plan outlining how to resolve the ${n} effectively.\n2. Identify files that need to be modified to resolve the ${n}.\n3. Provide a concise and descriptive commit message for the changes, following the Conventional Commits specification.\n\nYour plan should:\n- Focus on implementation details for each file that needs modification\n- Be clear and actionable for a developer to follow\n- Prefer showing diffs rather than complete file contents when describing changes\n- Exclude testing procedures as those will be handled separately\n\nGitHub ${t?"Pull Request":"Issue"}:\n${e}\n\nPlease format your response without any explanatory text as follows:\n\`\`\`md\n${ne}\n\n1. [Specific implementation step]\n2. [Next implementation step]\n...\n\n${ee}\n\n- \`[filePath1]\`\n- \`[filePath2]\`\n- ...\n\n${oe}\n\n[commit message]\n\`\`\`\n`}const ae=process.env.CI||process.env.GITHUB_ACTIONS;const ce={aider:{name:"Aider",buildArgs:function(e,t){const n=["--yes-always","--no-check-update","--no-gitignore","--no-show-model-warnings","--no-show-release-notes",...T(e.aiderExtraArgs||v),"--message",t.prompt];return e.dryRun&&n.push("--dry-run"),t.resolutionPlan&&"filePaths"in t.resolutionPlan&&n.push(...t.resolutionPlan.filePaths),n},getCommand:()=>"aider"},"claude-code":{name:"Claude Code",buildArgs:function(e,t){const n=["--yes","@anthropic-ai/claude-code@latest",...T(e.claudeCodeExtraArgs||x),"--dangerously-skip-permissions"];return ae&&n.push("--print"),n.push(t.prompt),n},getCommand:e=>e,getRunOptions:()=>({stdio:"inherit"})},"codex-cli":{name:"Codex CLI",buildArgs:function(e,t){return["--yes","@openai/codex@latest","exec",...T(e.codexExtraArgs||E),t.prompt]},getCommand:e=>e},"gemini-cli":{name:"Gemini CLI",buildArgs:function(e,t){return["--yes","@google/gemini-cli@latest","--yolo",...T(e.geminiExtraArgs||""),"--prompt",t.prompt]},getCommand:e=>e}};function le(e){return ce[e].name}function ue(e,t,n,o){const i=ce[e];return{command:i.getCommand(n),args:i.buildArgs(t,o),runOptions:i.getRunOptions?.()}}function de(){return{env:{...process.env,NO_COLOR:"1"},ignoreExitStatus:!0}}async function fe(e,t,n){const i=le(e.codingTool),r=N(e.nodeRuntime),{command:s,args:a,runOptions:c}=ue(e.codingTool,e,r,{prompt:t,resolutionPlan:n}),l={...de(),...c};console.info(o.cyan(`Asking ${i} to fix "${e.testCommand}"...`));const u=(await G(s,a,l)).stdout;return`\n\n## ${i} fix attempt for "${e.testCommand}"\n\n${u.trim()}`}async function me(t){i(),!process.env.AWS_REGION&&process.env.AWS_REGION_NAME?process.env.AWS_REGION=process.env.AWS_REGION_NAME:process.env.AWS_REGION&&!process.env.AWS_REGION_NAME&&(process.env.AWS_REGION_NAME=process.env.AWS_REGION),!process.env.GEMINI_API_KEY&&process.env.GOOGLE_GENERATIVE_AI_API_KEY?process.env.GEMINI_API_KEY=process.env.GOOGLE_GENERATIVE_AI_API_KEY:process.env.GEMINI_API_KEY&&!process.env.GOOGLE_GENERATIVE_AI_API_KEY&&(process.env.GOOGLE_GENERATIVE_AI_API_KEY=process.env.GEMINI_API_KEY),function(e,t){t&&(console.info("Parsed options:"),console.info(JSON.stringify(e,null,2)))}(t,t.verbose);const s=N(t.nodeRuntime);t.dryRun?console.info(o.yellow("Running in dry-run mode. No branches or PRs will be created.")):await async function(){const e=(await G("git",["config","user.name"],{ignoreExitStatus:!0})).stdout.trim(),t=(await G("git",["config","user.email"],{ignoreExitStatus:!0})).stdout.trim();if(!e||!t){const e=(await G("gh",["api","user"],{ignoreExitStatus:!0})).stdout.trim();try{const{name:t,email:n}=JSON.parse(e);t&&await G("git",["config","user.name",t]),n&&await G("git",["config","user.email",n])}catch{}}}(),"aider"===t.codingTool&&(await G("python",["-m","pip","install","aider-install"]),await pe(),await G("uv",["tool","uninstall","aider-chat"],{ignoreExitStatus:!0}),await G("aider-install",[]),await pe(),t.aiderExtraArgs?.includes("bedrock/")&&await G("uv",["tool","run","--from","aider-chat","pip","install","--upgrade","--upgrade-strategy","only-if-needed","boto3"]));const a=await async function(e){const{stdout:t}=await G("gh",["pr","view",e.issueNumber.toString(),"--json","headRefName"],{ignoreExitStatus:!0});try{if(t)return JSON.parse(t).headRefName}catch{}}(t),c=!!a,l=a||await async function(){return(await G("git",["branch","--show-current"])).stdout.trim()}(),u=await async function(e){const t=new Set,n=await K(e.issueNumber,t,e,!1);if(!n)throw new Error(`Failed to fetch issue data for issue #${e.issueNumber}`);return n}(t),d=n.stringify(u,V).trim(),f=t.planningModel&&await async function(t,o,i,r,s,a=!1){const c=z(o,"~"),l=`${c}yaml\n${n.stringify(o,V).trim()}\n${c}`,u=["--yes","repomix@latest","--output",Z];u.push(...T(s||P)),await G("npx",u);const d=e.readFileSync(Z,"utf8");if(e.promises.rm(Z,{force:!0}),i){console.info(`Selecting files with ${t} (reasoning effort: ${r}) ...`);const n=await Q(t,[{role:"system",content:ie(l,a).trim()},{role:"user",content:d}],r);console.info("Selecting complete!");const o=j(U(n),[ee,te]);if(!o)return{filePaths:[]};const[i,s]=o.map(e=>[...e.matchAll(/\B-\s*`?([^`\n]+)`?/g)].map(e=>e[1]?.trim()??"").filter(Boolean)),c=[...i,...s].map(t=>{const n=e.existsSync(t)?e.readFileSync(t,"utf8").trim():"",o=z(n,"~");return`## \`${t}\`\n\n${o}\n${n}\n${o}`}).join("\n\n");console.info(`Planning code changes with ${t} (reasoning effort: ${r}) ...`);const u=await Q(t,[{role:"system",content:re(l,a)},{role:"user",content:c}],r);console.info("Planning complete!");const f=j(U(u),[oe,ne]);if(!f)return{filePaths:i};const[m,p]=f;return{plan:p,commitMessage:m?.trim(),filePaths:i}}console.info(`Planning code changes with ${t} (reasoning effort: ${r}) ...`);const f=await Q(t,[{role:"system",content:se(l,a).trim()},{role:"user",content:d}],r);console.info("Planning complete!");const m=j(U(f),[oe,ne,ee]);if(!m)return{filePaths:[]};const[p,g,h]=m,w=[...(h??"").matchAll(/\B-\s*`?([^`\n]+)`?/g)].map(e=>e[1]?.trim()??"").filter(Boolean);return{plan:g,commitMessage:p?.trim(),filePaths:w}}(t.planningModel,d,t.twoStagePlanning,t.reasoningEffort,t.repomixExtraArgs,c)||void 0;console.info("Resolution plan:",f);const m=f&&"plan"in f&&f.plan||"",p=z(d,"~"),g=`\nModify the code to resolve ${c?"the comments on the following GitHub pull request":"the following GitHub issue"}${m?" based on the plan":""}.${"aider"!==t.codingTool?" After that, commit your changes with a message, following the Conventional Commits specification.":""}\n\n## ${c?"Pull Request":"Issue"}\n\n${p}yml\n${d}\n${p}\n\n${m&&`## Plan\n\n${m}`}\n`.trim(),h=new Date,w=`gen-pr-${t.issueNumber}-${t.codingTool}-${h.getFullYear()}_${ge(h.getMonth()+1)}${ge(h.getDate())}_${ge(h.getHours())}${ge(h.getMinutes())}${ge(h.getSeconds())}`;t.noBranch?t.dryRun?console.info(o.yellow(`Would commit directly to base branch: ${l}`)):(c&&await G("git",["fetch","origin",l]),await G("git",["switch",l])):t.dryRun?console.info(o.yellow(`Would create branch: ${w}`)):(c&&(await G("git",["fetch","origin",l]),await G("git",["switch",l])),await G("git",["switch","--force-create",w]));let y,$="",b="",v=!0;const x=le(t.codingTool),{command:E,args:A,runOptions:R}=ue(t.codingTool,t,s,{prompt:g,resolutionPlan:f}),_={...de(),...R};if(y=function(e,t,n){const o=t.map(e=>e===n?"...":e.includes(" ")||e.includes('"')||e.includes("'")?`"${e.replace(/"/g,'\\"')}"`:e);return`${e} ${o.join(" ")}`}(E,A,g),t.dryRun)console.info(`\n=== DRY MODE: ${x} Prompt ===`),console.info(g),console.info(`=== End ${x} Prompt ===\n`),console.info(o.yellow(`Would run: ${y}`)),$="Skipped due to dry-run mode";else{const e=await G(E,A,_);$=e.stdout||"",0!==e.status&&(v=!1,b=`${x} failed with exit code ${e.status}\n${e.stderr}`,console.error(o.red(`${x} execution failed: ${b}`)))}let I=($||"").trim(),O=!0,M="";if(t.dryRun)console.info(o.yellow("Would run test command"));else{const e=await async function(e,t){const[n,...i]=T(e.testCommand||"");if(!n)return{fixResult:"",success:!0};const r=e.maxTestAttempts;let s=0,a="",c=!1,l="";for(;s<r;){s++,console.info(o.cyan(`Executing test command (attempt ${s}/${r}): ${e.testCommand}`));const u=await C(n,i,{cwd:process.cwd()});if(0===u.status){console.info(o.green("Test command passed successfully.")),c=!0;break}if(console.warn(o.yellow(`Test command failed with exit code ${u.status}.`)),l=`Test command failed with exit code ${u.status}\n\nStdout:\n${u.stdout}\n\nStderr:\n${u.stderr}`,s>=r){console.warn(o.yellow(`Maximum fix attempts (${r}) reached. Giving up.`));break}const d=z(u.stdout,"~"),f=z(u.stderr,"~"),m=`\nThe previous changes were applied, but the test command \`${e.testCommand}\` failed.\n\nExit code: ${u.status}\n\nStdout:\n${d}\n${u.stdout}\n${d}\n\nStderr:\n${f}\n${u.stderr}\n${f}\n\nPlease analyze the output and fix the errors.\n`.trim();a+=await fe(e,m,t)}return{fixResult:a,success:c,error:c?void 0:l}}(t,f);I+=e.fixResult,O=e.success,M=e.error||"",O||console.warn(o.yellow("Tests failed after all fix attempts. Will create a draft PR."))}v||console.warn(o.yellow(`${x} execution failed. Will create a draft PR.`));const H=!v||!O;let L="";H&&(b&&(L+=`\n\n### ❌ ${x} Execution Error\n\n\`\`\`\n${b}\n\`\`\``),M&&(L+=`\n\n### ❌ Test Execution Error\n\n\`\`\`\n${M}\n\`\`\``));const Y=f?.commitMessage||`fix: Close #${t.issueNumber}`;if(await G("git",["add","-A"],{ignoreExitStatus:!0}),0!==(await G("git",["commit","-m",Y],{ignoreExitStatus:!0})).status&&await G("git",["commit","-m",Y,"--no-verify"],{ignoreExitStatus:!0}),t.noBranch?t.dryRun?console.info(o.yellow(`Would push changes directly to base branch: ${l}`)):await G("git",["push","origin",l,"--no-verify"]):t.dryRun?console.info(o.yellow(`Would push branch: ${w} to origin`)):await G("git",["push","origin",w,"--no-verify"]),t.noBranch)t.dryRun?console.info(o.yellow("Skipping PR creation due to --no-branch option")):console.info(`Changes committed directly to base branch: ${l}`);else{const e=function(e){return r.spawnSync("git",["log",`${e}..HEAD`,"--reverse","--pretty=%s"],{encoding:"utf8",stdio:"pipe"}).stdout.trim().split("\n").find(e=>e.trim())??""}(l)||Y;let n=c?"":`Close #${t.issueNumber}`;if(t.planningModel&&(n+=`\n\n${S}\n\n- **Planning Model:** ${t.planningModel}`),n+=`\n- **Coding Tool:** ${x}\n- **Coding Command:** \`${y}\``,m){const e=z(m,"~");n+=`\n\n### Plan\n\n${e}\n${k(m,I.length/(m.length+I.length)*3e4)}\n${e}`}if(I){const e=z(I,"~");n+=`\n\n### ${x} Log\n\n${e}\n${k(I,I.length/(m.length+I.length)*3e4)}\n${e}`}n=n.replaceAll(/(?:\s*\n){2,}/g,"\n\n").trim(),H&&(n+=L),t.dryRun?console.info(o.yellow(`Would create PR with title: ${e}${H?" (as draft)":""}`)):await async function(e){const{owner:t,repo:n}=await q();await W.pulls.create({owner:t,repo:n,title:e.title,body:e.body,head:e.head,base:e.base,draft:e.draft})}({title:e,body:n,head:w,base:l,draft:H})}console.info(`\n${c?"Pull request":"Issue"} #${t.issueNumber} processed successfully.`)}async function pe(){try{await G("asdf",["reshim"],{ignoreExitStatus:!0})}catch{}}function ge(e){return String(e).padStart(2,"0")}export{_ as D,I as a,R as b,v as c,x as d,E as e,A as f,P as g,b as l,me as m};
2
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"main-DcT2jGgm.js","sources":["../src/config.ts","../src/defaultOptions.ts","../src/text.ts","../src/spawn.ts","../src/octokit.ts","../src/issue.ts","../src/markdown.ts","../src/llmv4.ts","../src/yaml.ts","../src/llm.ts","../src/plan.ts","../src/ci.ts","../src/utils/toolRegistry.ts","../src/tools/aider.ts","../src/tools/claudeCode.ts","../src/tools/codex.ts","../src/tools/gemini.ts","../src/test.ts","../src/main.ts","../src/env.ts","../src/utils/logging.ts","../src/git.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport YAML from 'yaml';\n\nexport function loadConfigFile(): Record<string, unknown> {\n  // Load config file (YAML) from repository root to set default option values\n  let configOptions: Record<string, unknown> = {};\n  for (const name of ['gen-pr.config.yml', 'gen-pr.config.yaml']) {\n    const cfgPath = path.resolve(process.cwd(), name);\n    if (fs.existsSync(cfgPath)) {\n      try {\n        configOptions = YAML.parse(fs.readFileSync(cfgPath, 'utf8')) as Record<string, unknown>;\n        console.info(`Loaded gen-pr config from ${name}`);\n      } catch (err) {\n        console.error(`Failed to parse config file ${name}:`, err);\n        process.exit(1);\n      }\n      break;\n    }\n  }\n  return configOptions;\n}\n","import type { NodeRuntimeActual } from './types.js';\n\n/** \"--yes-always --no-check-update --no-gitignore --no-show-model-warnings --no-show-release-notes\" is always applied */\nexport const DEFAULT_AIDER_EXTRA_ARGS = '--model gemini/gemini-2.5-pro --edit-format diff-fenced';\n/** \"--dangerously-skip-permissions --print\" is always applied */\nexport const DEFAULT_CLAUDE_CODE_EXTRA_ARGS = '--allowedTools Bash Edit Write';\n/** exec is always applied */\nexport const DEFAULT_CODEX_EXTRA_ARGS = '--full-auto';\n/** \"--yolo\" is always applied */\nexport const DEFAULT_GEMINI_EXTRA_ARGS = '';\nexport const DEFAULT_REPOMIX_EXTRA_ARGS = '--compress --remove-empty-lines --include \"src/**/*.{ts,tsx},**/*.md\"';\nexport const DEFAULT_MAX_TEST_ATTEMPTS = 5;\nexport const DEFAULT_CODING_TOOL = 'aider';\nexport const DEFAULT_NODE_RUNTIME: NodeRuntimeActual = 'npx';\n","export const HEADING_OF_GEN_PR_METADATA = '## gen-pr Metadata';\n\n/**\n * Truncate long text.\n */\nexport function truncateText(text: string, maxLength: number): string {\n  if (text.length > maxLength) {\n    const truncated = text.slice(0, maxLength);\n    const omitted = text.length - maxLength;\n    return `${truncated}\\n\\n... (${Math.floor(omitted)} characters truncated) ...`;\n  }\n\n  return text;\n}\n\n/**\n * Removes HTML-style comments from a string.\n *\n * @param markdownContent The string containing markdown content\n * @returns The string with HTML comments removed\n */\nexport function stripHtmlComments(markdownContent: string): string {\n  return markdownContent.replace(/<!--[\\s\\S]*?-->/g, '');\n}\n\n/**\n * Removes gen-pr metadata sections from a PR body markdown.\n */\nexport function stripMetadataSections(markdownContent: string): string {\n  const index = markdownContent.indexOf(HEADING_OF_GEN_PR_METADATA);\n  return index >= 0 ? markdownContent.substring(0, index) : markdownContent;\n}\n\nexport function normalizeNewLines(text: string): string {\n  return text.replaceAll('\\r\\n', '\\n').trim();\n}\n\n/**\n * Removes text matching the specified regex pattern.\n *\n * @param text The input text\n * @param pattern The regex pattern string to remove\n * @returns The text with matched patterns removed\n */\nexport function removeRegexPattern(text: string, pattern: string): string {\n  if (!pattern) return text;\n  try {\n    const regex = new RegExp(pattern, 'g');\n    return text.replace(regex, '');\n  } catch (error) {\n    console.warn(`Invalid regex pattern \"${pattern}\":`, error);\n    return text;\n  }\n}\n","import type { SpawnOptions, SpawnSyncReturns } from 'node:child_process';\nimport { spawn } from 'node:child_process';\nimport ansis from 'ansis';\nimport { truncateText } from './text.js';\nimport type { NodeRuntime, NodeRuntimeActual } from './types.js';\n\nconst MAX_LOG_LENGTH = 3000;\n\n/**\n * Normalizes node runtime values to their actual executable commands.\n * 'node' is an alias for 'npx' and 'bun' is an alias for 'bunx'.\n *\n * @param runtime The runtime value to normalize\n * @returns The normalized runtime command\n */\nexport function normalizeNodeRuntime(runtime: NodeRuntime): NodeRuntimeActual {\n  switch (runtime) {\n    case 'node':\n    case 'npx':\n      return 'npx';\n    case 'bun':\n    case 'bunx':\n      return 'bunx';\n  }\n}\n\nexport async function runCommand(\n  command: string,\n  args: string[],\n  options?: SpawnOptions & { ignoreExitStatus?: boolean; truncateStdout?: boolean }\n): Promise<Omit<SpawnSyncReturns<string>, 'output' | 'error'>> {\n  const { ignoreExitStatus, ...spawnOptions } = options ?? {};\n  const argsText = args.map((a) => (a.includes(' ') ? `\"${a.replaceAll('\"', '\"')}\"` : a)).join(' ');\n  console.info(ansis.green(`$ ${command} ${argsText}`));\n\n  console.info('stdout: ---------------------');\n  const ret = await spawnAsync(command, args, spawnOptions);\n  if (spawnOptions.truncateStdout) console.info(truncateText(ret.stdout, MAX_LOG_LENGTH));\n  const stderr = ret.stderr.trim();\n  if (stderr) {\n    console.info('stderr: ---------------------');\n    const truncatedStderr = truncateText(stderr, MAX_LOG_LENGTH);\n    console.info(ansis.yellow(truncatedStderr));\n  }\n  console.info('-----------------------------');\n  console.info(ansis.magenta(`Exit code: ${ret.status}\\n`));\n  if (!ignoreExitStatus && ret.status !== 0 && ret.status !== null) {\n    process.exit(ret.status);\n  }\n  return ret;\n}\n\nexport async function spawnAsync(\n  command: string,\n  args: readonly string[],\n  options: SpawnOptions & { truncateStdout?: boolean }\n): Promise<Omit<SpawnSyncReturns<string>, 'output' | 'error'>> {\n  return new Promise((resolve, reject) => {\n    try {\n      // Sanitize args to remove null bytes\n      const sanitizedArgs = (args ?? []).map((arg) => arg.replace(/\\0/g, ''));\n      const proc = spawn(command, sanitizedArgs, options);\n      // `setEncoding` is undefined in Bun\n      proc.stdout?.setEncoding?.('utf8');\n      proc.stderr?.setEncoding?.('utf8');\n\n      let stdout = '';\n      let stderr = '';\n      proc.stdout?.on('data', (data) => {\n        if (!options?.truncateStdout) process.stdout.write(data);\n        stdout += data;\n      });\n      proc.stderr?.on('data', (data) => {\n        stderr += data;\n      });\n\n      proc.on('error', (error) => {\n        reject(error);\n      });\n      proc.on('close', (code: number | null, signal: NodeJS.Signals | null) => {\n        if (proc.pid === undefined) {\n          reject(new Error('Process has no pid.'));\n        } else {\n          resolve({\n            pid: proc.pid,\n            stdout,\n            stderr,\n            status: code,\n            signal,\n          });\n        }\n      });\n    } catch (error) {\n      // eslint-disable-next-line @typescript-eslint/prefer-promise-reject-errors\n      reject(error);\n    }\n  });\n}\n\n/**\n * Parses a command line string into an array of arguments, preserving quoted strings.\n *\n * This function handles:\n * - Space-separated arguments\n * - Double-quoted strings (preserves spaces within)\n * - Single-quoted strings (preserves spaces within)\n *\n * @param argsString The command line string to parse\n * @returns An array of parsed arguments\n */\nexport function parseCommandLineArgs(argsString: string): string[] {\n  if (!argsString) return [];\n\n  const result: string[] = [];\n  let current = '';\n  let inDoubleQuote = false;\n  let inSingleQuote = false;\n\n  for (let i = 0; i < argsString.length; i++) {\n    const char = argsString[i];\n\n    // Handle quotes\n    if (char === '\"' && !inSingleQuote) {\n      inDoubleQuote = !inDoubleQuote;\n      continue;\n    }\n\n    if (char === \"'\" && !inDoubleQuote) {\n      inSingleQuote = !inSingleQuote;\n      continue;\n    }\n\n    // Handle spaces (only split on spaces outside of quotes)\n    if (char === ' ' && !inDoubleQuote && !inSingleQuote) {\n      if (current) {\n        result.push(current);\n        current = '';\n      }\n      continue;\n    }\n\n    // Add character to current argument\n    current += char;\n  }\n\n  // Add the last argument if there is one\n  if (current) {\n    result.push(current);\n  }\n\n  return result;\n}\n","import child_process from 'node:child_process';\nimport { graphql } from '@octokit/graphql';\nimport type { RestEndpointMethodTypes } from '@octokit/rest';\nimport { Octokit } from '@octokit/rest';\nimport { runCommand } from './spawn.js';\nimport type {\n  LabelInfo,\n  PullRequestParams,\n  PullRequestReview,\n  PullRequestReviewThreadsResponse,\n  RepositoryInfo,\n  SimpleComment,\n} from './types.js';\n\nconst token =\n  process.env.GH_TOKEN ||\n  process.env.GITHUB_TOKEN ||\n  child_process.spawnSync('gh', ['auth', 'token'], { encoding: 'utf-8' }).stdout.trim();\nif (!token) {\n  throw new Error(\n    'GitHub token not found. Please set GH_TOKEN or GITHUB_TOKEN environment variable, or authenticate with gh CLI'\n  );\n}\nconst octokit = new Octokit({\n  auth: token,\n});\nconst graphqlClient = graphql.defaults({\n  headers: {\n    authorization: `token ${token}`,\n  },\n});\n\nlet repoOwner: string | undefined;\nlet repoName: string | undefined;\nasync function getRepoInfo(): Promise<RepositoryInfo> {\n  if (!repoOwner || !repoName) {\n    // Get repo info from git remote\n    const { stdout } = await runCommand('git', ['remote', 'get-url', 'origin'], { ignoreExitStatus: true });\n    const remoteUrl = stdout.trim();\n\n    // Parse GitHub repo URL (supports both https and ssh formats)\n    const match = remoteUrl.match(/github\\.com[:/]([^/]+)\\/(.+)/);\n    if (match?.[1] && match[2]) {\n      repoOwner = match[1];\n      repoName = match[2].replace(/\\.git$/, '');\n    } else {\n      throw new Error('Could not parse GitHub repository from remote URL');\n    }\n  }\n\n  if (!repoOwner || !repoName) {\n    throw new Error('Repository information not properly initialized');\n  }\n  return { owner: repoOwner, repo: repoName };\n}\n\nexport async function createPullRequest(params: PullRequestParams): Promise<void> {\n  const { owner, repo } = await getRepoInfo();\n\n  await octokit.pulls.create({\n    owner,\n    repo,\n    title: params.title,\n    body: params.body,\n    head: params.head,\n    base: params.base,\n    draft: params.draft,\n  });\n}\n\nexport async function getPullRequestDiff(pullNumber: number): Promise<string> {\n  const { owner, repo } = await getRepoInfo();\n\n  const response = await octokit.pulls.get({\n    owner,\n    repo,\n    pull_number: pullNumber,\n    mediaType: {\n      format: 'diff',\n    },\n  });\n\n  return response.data as unknown as string;\n}\n\nexport async function getIssue(issueNumber: number): Promise<{\n  author: string;\n  title: string;\n  body: string;\n  labels: LabelInfo[];\n  comments: SimpleComment[];\n  url: string;\n}> {\n  const { owner, repo } = await getRepoInfo();\n\n  // Get issue data first - this works for both issues and PRs\n  const { data: issueData } = await octokit.issues.get({\n    owner,\n    repo,\n    issue_number: issueNumber,\n  });\n  // If it has pull_request field, get the full PR data\n  let issueOrPullRequest:\n    | RestEndpointMethodTypes['pulls']['get']['response']['data']\n    | RestEndpointMethodTypes['issues']['get']['response']['data'] = issueData;\n  if (issueData.pull_request) {\n    const { data: prData } = await octokit.pulls.get({\n      owner,\n      repo,\n      pull_number: issueNumber,\n    });\n    issueOrPullRequest = prData;\n  }\n\n  const commentsResponse = await octokit.issues.listComments({\n    owner,\n    repo,\n    issue_number: issueNumber,\n  });\n\n  return {\n    author: issueOrPullRequest.user?.login || '',\n    title: issueOrPullRequest.title,\n    body: issueOrPullRequest.body || '',\n    labels: issueOrPullRequest.labels.map((label: string | { name?: string }) => ({\n      name: typeof label === 'string' ? label : label.name || '',\n    })),\n    comments: commentsResponse.data.map((comment) => ({\n      author: comment.user?.login || '',\n      body: comment.body || '',\n      createdAt: comment.created_at,\n    })),\n    url: issueOrPullRequest.html_url,\n  };\n}\n\nexport async function getPullRequestReviewThreads(pullNumber: number): Promise<PullRequestReviewThreadsResponse> {\n  const { owner, repo } = await getRepoInfo();\n\n  const MAX_MESSAGE_COUNT = 100;\n  const query = `\n    query($owner: String!, $repo: String!, $pr: Int!) {\n      repository(owner: $owner, name: $repo) {\n        pullRequest(number: $pr) {\n          reviewThreads(first: ${MAX_MESSAGE_COUNT}) {\n            nodes {\n              isResolved\n              comments(first: ${MAX_MESSAGE_COUNT}) {\n                nodes {\n                  author {\n                    login\n                  }\n                  body\n                  path\n                  line\n                  diffHunk\n                  createdAt\n                }\n              }\n            }\n          }\n        }\n      }\n    }\n  `;\n\n  const result = await graphqlClient(query, {\n    owner,\n    repo,\n    pr: pullNumber,\n  });\n\n  return result as PullRequestReviewThreadsResponse;\n}\n\nexport async function getPullRequestReviews(pullNumber: number): Promise<PullRequestReview[]> {\n  const { owner, repo } = await getRepoInfo();\n\n  const response = await octokit.pulls.listReviews({\n    owner,\n    repo,\n    pull_number: pullNumber,\n  });\n\n  return response.data.map((review) => ({\n    user: { login: review.user?.login || '' },\n    state: review.state,\n    body: review.body || '',\n    submitted_at: review.submitted_at || '',\n  }));\n}\n","import type { MainOptions } from './main.js';\nimport { getIssue, getPullRequestDiff, getPullRequestReviews, getPullRequestReviewThreads } from './octokit.js';\nimport { normalizeNewLines, removeRegexPattern, stripHtmlComments, stripMetadataSections } from './text.js';\nimport type { IssueComment, IssueInfo, UserLogin } from './types.js';\n\n// Temporary interface for sorting comments with date information\ninterface IssueCommentWithDate extends IssueComment {\n  createdAt: number;\n}\n\nexport async function createIssueInfo(options: MainOptions): Promise<IssueInfo> {\n  const processedIssues = new Set<number>();\n  const issueInfo = await fetchIssueData(options.issueNumber, processedIssues, options, false);\n  if (!issueInfo) {\n    throw new Error(`Failed to fetch issue data for issue #${options.issueNumber}`);\n  }\n  return issueInfo;\n}\n\nasync function fetchPRDiff(issueNumber: number, issueInfo: IssueInfo): Promise<void> {\n  try {\n    const prDiff = await getPullRequestDiff(issueNumber);\n    if (!prDiff.trim()) return;\n    issueInfo.code_changes = processDiffContent(prDiff.trim());\n  } catch (error) {\n    console.warn('Failed to fetch PR diff:', error);\n  }\n}\n\nasync function fetchIssueData(\n  issueNumber: number,\n  processedIssues: Set<number>,\n  options: MainOptions,\n  isReferenced = false\n): Promise<IssueInfo | undefined> {\n  if (processedIssues.has(issueNumber)) {\n    return;\n  }\n  processedIssues.add(issueNumber);\n\n  let issue: Awaited<ReturnType<typeof getIssue>>;\n  try {\n    issue = await getIssue(issueNumber);\n  } catch (error) {\n    console.warn(`Failed to fetch issue #${issueNumber}:`, error);\n    return;\n  }\n\n  // Extract issue/PR references from the issue body and comments\n  const allText = [issue.body, ...issue.comments.map((c) => c.body)].join('\\n');\n  const referencedNumbers = extractIssueReferences(allText);\n\n  const rawBody = stripHtmlComments(issue.body);\n  const processedBody = issue.url?.includes('/pull/') ? stripMetadataSections(rawBody) : rawBody;\n  const description = removeRegexPattern(processedBody, options.removePattern || '');\n  const commentsWithDate: IssueCommentWithDate[] = issue.comments.map((c) => ({\n    author: c.author,\n    body: normalizeNewLines(c.body),\n    createdAt: new Date(c.createdAt).getTime(),\n  }));\n\n  const issueInfo: IssueInfo = {\n    author: issue.author,\n    title: issue.title,\n    description: normalizeNewLines(description),\n    comments: [], // Will be populated after sorting\n  };\n\n  if (issue.url?.includes('/pull/') && !isReferenced) {\n    await fetchPRDiff(issueNumber, issueInfo);\n    await fetchPRReviewThreads(issueNumber, commentsWithDate);\n    await fetchPRReviews(issueNumber, commentsWithDate);\n  }\n\n  if (referencedNumbers.length > 0) {\n    await fetchReferencedIssues(referencedNumbers, processedIssues, options, issueInfo);\n  }\n\n  // Sort comments by creation date (oldest first) and remove createdAt field\n  issueInfo.comments = commentsWithDate\n    .filter((c) => c.body)\n    .sort((a, b) => a.createdAt - b.createdAt)\n    .map(({ createdAt, ...comment }) => comment);\n\n  return issueInfo;\n}\n\nasync function fetchPRReviewThreads(issueNumber: number, commentsWithDate: IssueCommentWithDate[]): Promise<void> {\n  try {\n    const graphqlData = await getPullRequestReviewThreads(issueNumber);\n    const reviewThreads = graphqlData?.repository?.pullRequest?.reviewThreads?.nodes || [];\n\n    for (const thread of reviewThreads) {\n      if (!thread.isResolved && thread.comments?.nodes) {\n        processReviewThreadComments(thread.comments.nodes, commentsWithDate);\n      }\n    }\n  } catch (error) {\n    console.warn('Failed to fetch PR review threads:', error);\n  }\n}\n\nfunction processReviewThreadComments(\n  comments: Array<{\n    author?: UserLogin;\n    body?: string;\n    path?: string;\n    line?: number;\n    diffHunk?: string;\n    createdAt: string;\n  }>,\n  commentsWithDate: IssueCommentWithDate[]\n): void {\n  for (const comment of comments) {\n    if (!comment.author || !comment.body) continue;\n\n    const codeContent = extractCodeFromDiffHunk(comment.diffHunk);\n    const reviewComment: IssueCommentWithDate = {\n      author: comment.author.login,\n      codeLocation: comment.path && comment.line ? `${comment.path}:${comment.line}` : undefined,\n      codeContent: codeContent || undefined,\n      body: normalizeNewLines(comment.body),\n      createdAt: new Date(comment.createdAt).getTime(),\n    };\n\n    // Remove undefined properties\n    Object.keys(reviewComment).forEach((key) => {\n      if (reviewComment[key as keyof IssueCommentWithDate] === undefined) {\n        delete reviewComment[key as keyof IssueCommentWithDate];\n      }\n    });\n\n    commentsWithDate.push(reviewComment);\n  }\n}\n\nfunction extractCodeFromDiffHunk(diffHunk: string | undefined): string {\n  if (!diffHunk) return '';\n\n  const lines = diffHunk.split('\\n');\n  const codeLine = lines.find(\n    (line: string) => (line.startsWith('+') || line.startsWith('-')) && !line.startsWith('@@') && line.trim().length > 1\n  );\n  return codeLine?.trim() || '';\n}\n\nasync function fetchPRReviews(issueNumber: number, commentsWithDate: IssueCommentWithDate[]): Promise<void> {\n  try {\n    const reviews = await getPullRequestReviews(issueNumber);\n    const reviewResultComments: IssueCommentWithDate[] = reviews.map((review) => ({\n      author: review.user.login,\n      reviewState: review.state,\n      body: normalizeNewLines(review.body),\n      createdAt: new Date(review.submitted_at).getTime(),\n    }));\n    commentsWithDate.push(...reviewResultComments);\n  } catch (error) {\n    console.warn('Failed to fetch PR reviews:', error);\n  }\n}\n\nfunction extractIssueReferences(text: string): number[] {\n  const regex = /(?:^|\\s)#(\\d+)/g;\n  const numbers: number[] = [];\n  for (;;) {\n    const match = regex.exec(text);\n    if (!match) break;\n\n    const number = Number.parseInt(match[1] ?? '', 10);\n    if (Number.isInteger(number)) {\n      numbers.push(number);\n    }\n  }\n  return [...new Set(numbers)]; // Remove duplicates\n}\n\nasync function fetchReferencedIssues(\n  referencedNumbers: number[],\n  processedIssues: Set<number>,\n  options: MainOptions,\n  issueInfo: IssueInfo\n): Promise<void> {\n  const referencedIssuesPromises = referencedNumbers.map((num) => fetchIssueData(num, processedIssues, options, true));\n  const referencedIssues = (await Promise.all(referencedIssuesPromises)).filter((issue): issue is IssueInfo => !!issue);\n\n  if (referencedIssues.length === 0) return;\n\n  issueInfo.referenced_issues = referencedIssues;\n}\n\n/**\n * Process diff content to handle large diffs by truncating or omitting large fragments\n */\nfunction processDiffContent(diffContent: string): string {\n  const MAX_TOTAL_DIFF_SIZE = 50000;\n  const MAX_FILE_DIFF_SIZE = 10000;\n  const LARGE_FILE_PATTERNS = [\n    /^diff --git a\\/dist\\//m,\n    /^diff --git a\\/build\\//m,\n    /^diff --git a\\/.*\\.bundle\\./m,\n    /^diff --git a\\/.*\\.min\\./m,\n    /^diff --git a\\/node_modules\\//m,\n  ];\n\n  // If the entire diff is small enough, return as-is\n  if (diffContent.length <= MAX_TOTAL_DIFF_SIZE) {\n    return diffContent;\n  }\n\n  // Split diff into individual file sections\n  const fileSections = diffContent.split(/(?=^diff --git)/m);\n  const processedSections: string[] = [];\n  let totalSize = 0;\n\n  for (const section of fileSections) {\n    if (!section.trim()) continue;\n\n    const isLargeFile = LARGE_FILE_PATTERNS.some((pattern) => pattern.test(section));\n\n    if (isLargeFile) {\n      // For large/bundled files, include only the header and a truncation notice\n      const lines = section.split('\\n');\n      const headerLines = lines.slice(0, 4); // diff --git, index, ---, +++\n      const truncatedSection = [\n        ...headerLines,\n        '@@ ... @@',\n        '... (large bundled/compiled file diff truncated) ...',\n        '',\n      ].join('\\n');\n\n      processedSections.push(truncatedSection);\n      totalSize += truncatedSection.length;\n    } else if (section.length > MAX_FILE_DIFF_SIZE) {\n      // For other large files, truncate but keep some content\n      const truncatedSection = `${section.slice(0, MAX_FILE_DIFF_SIZE)}\\n... (diff truncated) ...\\n`;\n      processedSections.push(truncatedSection);\n      totalSize += truncatedSection.length;\n    } else {\n      // Small files, include as-is\n      processedSections.push(section);\n      totalSize += section.length;\n    }\n\n    // Stop if we're approaching the total size limit\n    if (totalSize > MAX_TOTAL_DIFF_SIZE * 0.9) {\n      processedSections.push('\\n... (remaining diffs truncated) ...\\n');\n      break;\n    }\n  }\n\n  return processedSections.join('');\n}\n","/**\n * Extracts content between specified headers from a text\n * @param text The text to parse\n * @param headers Array of header strings to extract content for\n * @returns Array of content strings, or undefined if any headers are missing or in different order\n */\nexport function extractHeaderContents(text: string, headers: string[]): string[] | undefined {\n  const modifiedResponse = `\\n${text}`;\n  const indices = headers.map((header) => modifiedResponse.indexOf(`\\n${header}`));\n\n  // Return undefined if any headers are missing or not in order\n  if (\n    indices.some((index) => index === -1) ||\n    !indices.every((index, i) => i === 0 || index > (indices[i - 1] as number))\n  ) {\n    return undefined;\n  }\n\n  return headers.map((header, i) => {\n    const start = (indices[i] as number) + 1 + header.length;\n    const end = i + 1 < headers.length ? (indices[i + 1] as number) + 1 : modifiedResponse.length;\n    return modifiedResponse.slice(start, end).trim();\n  });\n}\n\nexport function findDistinctFence(content: string, fenceChar: '`' | '~'): string {\n  const regex = new RegExp(`${fenceChar}{3,}`, 'g');\n  const matches = content.match(regex);\n  const maxLength = matches ? Math.max(...matches.map((seq) => seq.length)) : 0;\n  const fenceLength = Math.max(3, maxLength + 1);\n  return fenceChar.repeat(fenceLength);\n}\n\nexport function trimCodeBlockFences(content: string): string {\n  // Remove code block fences with any number of backticks or tildes from the beginning and end\n  return content.trim().replace(/^(`{3,}|~{3,})[\\s\\S]*?\\n([\\s\\S]*?)\\n\\1\\s*$/, '$2');\n}\n","import type { ModelMessage } from 'ai';\nimport { generateText, type Message } from 'ai-v4';\nimport { createOllama } from 'ollama-ai-provider-v2';\nimport { logResult } from './llm.js';\nimport type { ReasoningEffort } from './types.js';\n\n/**\n * Call AI SDK v4 provider API (for Ollama)\n */\nexport async function callV4ProviderApi(\n  model: string,\n  messages: ModelMessage[],\n  reasoningEffort?: ReasoningEffort\n): Promise<string> {\n  const [provider, ...modelParts] = model.split('/') as [string, ...string[]];\n  try {\n    const modelName = modelParts.join('/'); // Handle cases where model name itself contains '/'\n    if (!modelName) {\n      console.error(`Invalid ${provider} model format: ${model}. Expected format: ${provider}/model-name`);\n      process.exit(1);\n    }\n\n    const ollamaBaseURL = `${process.env.OLLAMA_BASE_URL || 'http://localhost:11434'}/api`;\n    const ollama = createOllama({\n      baseURL: ollamaBaseURL,\n      ...(process.env.OLLAMA_API_KEY && { apiKey: process.env.OLLAMA_API_KEY }),\n    });\n    const providerModel = ollama(modelName);\n\n    const result = await generateText({\n      model: providerModel,\n      providerOptions: reasoningEffort ? { ollama: { think: true } } : undefined,\n      messages: convertToV4Messages(messages),\n    });\n    logResult(model, result);\n    return result.text;\n  } catch (error) {\n    console.error(`${provider.charAt(0).toUpperCase() + provider.slice(1)} API error for model ${model}:`, error);\n    process.exit(1);\n  }\n}\n\n/**\n * Convert AI SDK v5 ModelMessage[] to AI SDK v4 Message[] format\n */\nfunction convertToV4Messages(messages: ModelMessage[]): Message[] {\n  return messages.map((msg, index) => ({\n    id: `msg-${index}`,\n    role: msg.role === 'tool' ? 'data' : msg.role,\n    content: typeof msg.content === 'string' ? msg.content : JSON.stringify(msg.content),\n  }));\n}\n","/**\n * `blockQuote: 'literal'`: Use a pipe (`prop: |`) to represent multi-line strings.\n * `lineWidth: 0`: No limit on the line width.\n */\nexport const yamlStringifyOptions = { blockQuote: 'literal', lineWidth: 0 } as const;\n","import { createAmazonBedrock } from '@ai-sdk/amazon-bedrock';\nimport { type AnthropicProviderOptions, createAnthropic } from '@ai-sdk/anthropic';\nimport { createAzure } from '@ai-sdk/azure';\nimport { createGoogleGenerativeAI, type GoogleGenerativeAIProviderOptions } from '@ai-sdk/google';\nimport { createVertex } from '@ai-sdk/google-vertex';\nimport { createOpenAI, type OpenAIResponsesProviderOptions } from '@ai-sdk/openai';\nimport type { LanguageModelV2 } from '@ai-sdk/provider';\nimport { createXai } from '@ai-sdk/xai';\nimport { createOpenRouter } from '@openrouter/ai-sdk-provider';\nimport { generateText, type ModelMessage } from 'ai';\nimport YAML from 'yaml';\nimport { callV4ProviderApi } from './llmv4.js';\nimport type { ReasoningEffort } from './types.js';\nimport { yamlStringifyOptions } from './yaml.js';\n\n/**\n * Call LLM API using AI SDK 5\n */\nexport async function callLlmApi(\n  model: string,\n  messages: ModelMessage[],\n  reasoningEffort?: ReasoningEffort\n): Promise<string> {\n  try {\n    // Special handling for Ollama using AI SDK v4\n    if (model.startsWith('ollama/')) {\n      return await callV4ProviderApi(model, messages, reasoningEffort);\n    }\n\n    const [modelInstance, provider, modelName] = getModelInstance(model, reasoningEffort);\n\n    // Build the request parameters\n    const requestParams: Parameters<typeof generateText>[0] = {\n      model: modelInstance,\n      messages: messages,\n    };\n\n    if (reasoningEffort) {\n      // Check if the model supports reasoning/thinking options\n      const modelSupportsReasoning = supportsReasoningOptions(provider, modelName);\n\n      if (!modelSupportsReasoning) {\n        console.warn(\n          `Model ${model} does not support reasoning/thinking options. Ignoring reasoning effort parameter.`\n        );\n      } else {\n        const thinkingBudget = getThinkingBudget(reasoningEffort);\n        if (provider === 'openai') {\n          requestParams.providerOptions = {\n            openai: {\n              reasoningEffort,\n            } satisfies OpenAIResponsesProviderOptions,\n          };\n        } else if (provider === 'anthropic') {\n          requestParams.providerOptions = {\n            anthropic: {\n              thinking: {\n                type: 'enabled',\n                budgetTokens: thinkingBudget,\n              },\n            } satisfies AnthropicProviderOptions,\n          };\n        } else if (provider === 'gemini') {\n          requestParams.providerOptions = {\n            google: {\n              thinkingConfig: {\n                thinkingBudget,\n              },\n            } satisfies GoogleGenerativeAIProviderOptions,\n          };\n        } else if (provider === 'bedrock') {\n          // The latest AI SDK doesn't work on Bedrock with reasoning.\n          console.warn(\n            `Note: The current AI SDK doesn't work on Bedrock with reasoning. Model ${modelName} will use default reasoning settings.`\n          );\n          // requestParams.providerOptions = {\n          //   bedrock: {\n          //     reasoningConfig: { type: 'enabled', budgetTokens: thinkingBudget },\n          //   } satisfies BedrockProviderOptions,\n          // };\n        } else if (provider === 'xai') {\n          requestParams.providerOptions = {\n            xai: {\n              reasoningEffort,\n            },\n          };\n        }\n      }\n    }\n\n    const result = await generateText(requestParams);\n    logResult(model, result);\n\n    return result.text;\n  } catch (error) {\n    console.error(`LLM API error for model ${model}:`, error);\n    process.exit(1);\n  }\n}\n\n/**\n * Log the result of an LLM API call\n */\nexport function logResult(model: string, result: { text: string; usage?: unknown; finishReason?: string }): void {\n  console.info(\n    `${model}:`,\n    YAML.stringify(\n      {\n        text: result.text,\n        usage: result.usage,\n        finishReason: result.finishReason,\n      },\n      yamlStringifyOptions\n    )\n  );\n}\n\nfunction getModelInstance(model: string, reasoningEffort?: ReasoningEffort): [LanguageModelV2, string, string] {\n  // Only support llmlite format (provider/model)\n  if (!model.includes('/')) {\n    console.error(`Model must be in format 'provider/model'. Got: ${model}`);\n    process.exit(1);\n  }\n\n  const [provider, ...modelParts] = model.split('/');\n  const modelName = modelParts.join('/'); // Handle cases where model name itself contains '/'\n\n  switch (provider) {\n    case 'openai': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/openai\n      const openaiProvider = createOpenAI();\n      return [openaiProvider(modelName), provider, modelName];\n    }\n\n    case 'anthropic': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/anthropic\n      const anthropicProvider = createAnthropic();\n      return [anthropicProvider(modelName), provider, modelName];\n    }\n\n    case 'gemini': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/google-generative-ai\n      const googleProvider = createGoogleGenerativeAI();\n      return [googleProvider(modelName), provider, modelName];\n    }\n\n    case 'azure': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/azure\n      const azureProvider = createAzure();\n      return [azureProvider(modelName), provider, modelName];\n    }\n\n    case 'bedrock': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/amazon-bedrock\n      const bedrockProvider = createAmazonBedrock();\n      return [bedrockProvider(modelName), provider, modelName];\n    }\n\n    case 'vertex': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/google-vertex\n      const vertexProvider = createVertex();\n      return [vertexProvider(modelName), provider, modelName];\n    }\n\n    case 'xai': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/xai\n      const grokProvider = createXai();\n      return [grokProvider(modelName), provider, modelName];\n    }\n\n    case 'openrouter': {\n      // cf. https://github.com/OpenRouterTeam/ai-sdk-provider\n      const openrouterProvider = createOpenRouter({\n        apiKey: process.env.OPENROUTER_API_KEY,\n        headers: {\n          'HTTP-Referer': 'https://github.com/WillBooster/gen-pr',\n          'X-Title': 'gen-pr',\n        },\n      });\n      const modelOptions = reasoningEffort\n        ? {\n            reasoning: {\n              effort: reasoningEffort,\n            },\n          }\n        : {};\n      return [openrouterProvider(modelName, modelOptions), provider, modelName];\n    }\n\n    default:\n      console.error(\n        `Unsupported provider: ${provider}. Supported providers: openai, azure, google, anthropic, bedrock, vertex, grok, openrouter, ollama`\n      );\n      process.exit(1);\n  }\n}\n\n/**\n * Check if a model supports reasoning/thinking options\n */\nexport function supportsReasoningOptions(provider: string, modelName: string): boolean {\n  switch (provider) {\n    case 'openai':\n    case 'azure':\n      // OpenAI and Azure: only o1, o3, o4 series models support reasoning effort\n      return /^(o1|o3|o4)/.test(modelName);\n\n    case 'anthropic':\n      // Anthropic: only Claude 3.7 and Claude 4 models support thinking budget\n      return /^claude-(opus-4|sonnet-4|3-7-sonnet)/.test(modelName);\n\n    case 'gemini':\n      // Google: only Gemini 2.5 models support thinking budget\n      return /^gemini-2\\.5/.test(modelName);\n\n    case 'bedrock':\n      // Bedrock: only Anthropic Claude 3.7 and 4 models support reasoning\n      return /^(us\\.)?anthropic\\.claude-(opus-4|sonnet-4|3-7-sonnet)/.test(modelName);\n\n    case 'vertex':\n      // Vertex: Gemini 2.5 models and Claude 3.7/4 models support thinking budget\n      return /^gemini-2\\.5/.test(modelName) || /^claude-(3-7-sonnet|opus-4|sonnet-4)/.test(modelName);\n\n    case 'xai':\n      // Grok: Grok 3 models support reasoning effort\n      return /^grok-3/.test(modelName);\n\n    // We can always pass reasoningEffort to OpenRouter without errors.\n    case 'openrouter':\n      return true;\n\n    default:\n      return false;\n  }\n}\n\n/**\n * Get thinking budget token count based on reasoning effort level\n */\nfunction getThinkingBudget(reasoningEffort: ReasoningEffort): number {\n  const tokenBudgets = {\n    low: 4000, // 4K tokens\n    medium: 8000, // 8K tokens\n    high: 24000, // 24K tokens\n  };\n\n  return tokenBudgets[reasoningEffort];\n}\n","import fs from 'node:fs';\nimport YAML from 'yaml';\nimport { DEFAULT_REPOMIX_EXTRA_ARGS } from './defaultOptions.js';\nimport { callLlmApi } from './llm.js';\nimport { extractHeaderContents, findDistinctFence, trimCodeBlockFences } from './markdown.js';\nimport { parseCommandLineArgs, runCommand } from './spawn.js';\nimport type { ReasoningEffort } from './types.js';\nimport { yamlStringifyOptions } from './yaml.js';\n\nconst REPOMIX_FILE_NAME = 'repomix.result';\n\nexport type ResolutionPlan = {\n  plan?: string;\n  commitMessage?: string;\n  filePaths: string[];\n};\n\nconst HEADING_OF_FILE_PATHS_TO_BE_MODIFIED = '## File Paths to be Modified';\nconst HEADING_OF_FILE_PATHS_TO_BE_REFERRED = '## File Paths to be Referred';\nconst HEADING_OF_PLAN = '## Implementation Plan';\nconst HEADING_OF_COMMIT_MESSAGE = '## Commit Message';\n\nexport async function planCodeChanges(\n  model: string,\n  issueContent: string,\n  twoStagePlanning: boolean,\n  reasoningEffort?: ReasoningEffort,\n  repomixExtraArgs?: string,\n  isPullRequest = false\n): Promise<ResolutionPlan> {\n  const issueFence = findDistinctFence(issueContent, '~');\n  const issueYamlText = `${issueFence}yaml\n${YAML.stringify(issueContent, yamlStringifyOptions).trim()}\n${issueFence}`;\n\n  // Base repomix command arguments\n  const repomixArgs = ['--yes', 'repomix@latest', '--output', REPOMIX_FILE_NAME];\n  repomixArgs.push(...parseCommandLineArgs(repomixExtraArgs || DEFAULT_REPOMIX_EXTRA_ARGS));\n\n  await runCommand('npx', repomixArgs);\n  const repomixResult = fs.readFileSync(REPOMIX_FILE_NAME, 'utf8');\n  void fs.promises.rm(REPOMIX_FILE_NAME, { force: true });\n\n  if (twoStagePlanning) {\n    console.info(`Selecting files with ${model} (reasoning effort: ${reasoningEffort}) ...`);\n    const filesResponse = await callLlmApi(\n      model,\n      [\n        {\n          role: 'system',\n          content: buildPromptForSelectingFiles(issueYamlText, isPullRequest).trim(),\n        },\n        {\n          role: 'user',\n          content: repomixResult,\n        },\n      ],\n      reasoningEffort\n    );\n    console.info('Selecting complete!');\n\n    const extractedFilePathLists = extractHeaderContents(trimCodeBlockFences(filesResponse), [\n      HEADING_OF_FILE_PATHS_TO_BE_MODIFIED,\n      HEADING_OF_FILE_PATHS_TO_BE_REFERRED,\n    ]) as [string, string] | undefined;\n    if (!extractedFilePathLists) {\n      return { filePaths: [] };\n    }\n    const [filePathsToBeModified, filePathsToBeReferred] = extractedFilePathLists.map((filesContent: string) => {\n      const filePathRegex = /\\B-\\s*`?([^`\\n]+)`?/g;\n      const matches = [...filesContent.matchAll(filePathRegex)];\n      return matches.map((match) => match[1]?.trim() ?? '').filter(Boolean);\n    }) as [string[], string[]];\n\n    const fileContents = [...filePathsToBeModified, ...filePathsToBeReferred]\n      .map((filePath) => {\n        const content = fs.existsSync(filePath) ? fs.readFileSync(filePath, 'utf8').trim() : '';\n        const fence = findDistinctFence(content, '~');\n        return `## \\`${filePath}\\`\n\n${fence}\n${content}\n${fence}`;\n      })\n      .join('\\n\\n');\n\n    console.info(`Planning code changes with ${model} (reasoning effort: ${reasoningEffort}) ...`);\n    const planResponse = await callLlmApi(\n      model,\n      [\n        {\n          role: 'system',\n          content: buildPromptForPlanningCodeChanges(issueYamlText, isPullRequest),\n        },\n        {\n          role: 'user',\n          content: fileContents,\n        },\n      ],\n      reasoningEffort\n    );\n    console.info('Planning complete!');\n\n    const extractedContents = extractHeaderContents(trimCodeBlockFences(planResponse), [\n      HEADING_OF_COMMIT_MESSAGE,\n      HEADING_OF_PLAN,\n    ]);\n    if (!extractedContents) {\n      return { filePaths: filePathsToBeModified };\n    }\n    const [commitMessage, plan] = extractedContents;\n    return { plan, commitMessage: commitMessage?.trim(), filePaths: filePathsToBeModified };\n  }\n  console.info(`Planning code changes with ${model} (reasoning effort: ${reasoningEffort}) ...`);\n  const filesResponse = await callLlmApi(\n    model,\n    [\n      {\n        role: 'system',\n        content: buildPromptForSelectingFilesAndPlanningCodeChanges(issueYamlText, isPullRequest).trim(),\n      },\n      {\n        role: 'user',\n        content: repomixResult,\n      },\n    ],\n    reasoningEffort\n  );\n  console.info('Planning complete!');\n\n  const extractedContents = extractHeaderContents(trimCodeBlockFences(filesResponse), [\n    HEADING_OF_COMMIT_MESSAGE,\n    HEADING_OF_PLAN,\n    HEADING_OF_FILE_PATHS_TO_BE_MODIFIED,\n  ]);\n  if (!extractedContents) {\n    return { filePaths: [] };\n  }\n\n  const [commitMessage, plan, filePathsText] = extractedContents;\n\n  const filePathRegex = /\\B-\\s*`?([^`\\n]+)`?/g;\n  const matches = [...(filePathsText ?? '').matchAll(filePathRegex)];\n  const filePathsToBeModified = matches.map((match) => match[1]?.trim() ?? '').filter(Boolean);\n  return { plan, commitMessage: commitMessage?.trim(), filePaths: filePathsToBeModified };\n}\n\nfunction buildPromptForSelectingFiles(issueYamlText: string, isPullRequest = false): string {\n  const itemType = isPullRequest ? 'pull request' : 'issue';\n  const extraInstruction = isPullRequest ? ' Consider the comments on the pull request when identifying files.' : '';\n  return `\nYou are an expert software developer tasked with analyzing GitHub ${itemType}s and identifying relevant files for code changes.\n\nReview the following GitHub ${itemType} and the list of available file paths and their contents (which will be provided in a separate message).${extraInstruction}\nYour task is to identify:\n1. Files that need to be MODIFIED to resolve the ${itemType}\n2. Files that should be REFERRED to (but not modified) to understand the codebase better\n\nGitHub ${isPullRequest ? 'Pull Request' : 'Issue'}:\n${issueYamlText}\n\nPlease format your response without any explanatory text as follows:\n\\`\\`\\`md\n${HEADING_OF_FILE_PATHS_TO_BE_MODIFIED}\n\n- \\`[filePath1]\\`\n- \\`[filePath2]\\`\n- ...\n\n${HEADING_OF_FILE_PATHS_TO_BE_REFERRED}\n\n- \\`[filePath1]\\`\n- \\`[filePath2]\\`\n- ...\n\\`\\`\\`\n`;\n}\n\nfunction buildPromptForPlanningCodeChanges(issueYamlText: string, isPullRequest = false): string {\n  const itemType = isPullRequest ? 'pull request' : 'issue';\n  const extraInstruction = isPullRequest ? ' Consider the comments on the pull request when creating the plan.' : '';\n  return `\nYou are an expert software developer tasked with creating an implementation plan based on GitHub ${itemType}s.\n\nReview the following GitHub ${itemType} and the provided file contents (which will be provided in a separate message).${extraInstruction}\nCreate a detailed, step-by-step plan outlining how to address the ${itemType} effectively.\nAlso, provide a concise and descriptive commit message for the changes, following the Conventional Commits specification.\n\nYour plan should:\n- Focus on implementation details for each file that needs modification\n- Be clear and actionable for a developer to follow\n- Prefer showing diffs rather than complete file contents when describing changes\n- Exclude testing procedures unless users explicitly request\n\nGitHub ${isPullRequest ? 'Pull Request' : 'Issue'}:\n${issueYamlText}\n\nPlease format your response without any explanatory text as follows:\n\\`\\`\\`md\n${HEADING_OF_PLAN}\n\n1. [Specific implementation step]\n2. [Next implementation step]\n...\n\n${HEADING_OF_COMMIT_MESSAGE}\n\n[commit message]\n\\`\\`\\`\n`.trim();\n}\n\nfunction buildPromptForSelectingFilesAndPlanningCodeChanges(issueYamlText: string, isPullRequest = false): string {\n  const itemType = isPullRequest ? 'pull request' : 'issue';\n  const extraInstruction = isPullRequest ? ' Consider the comments on the pull request when creating the plan.' : '';\n  return `\nYou are an expert software developer tasked with analyzing GitHub ${itemType}s and creating an implementation plan.\n\nReview the following GitHub ${itemType} and the list of available file paths and their contents (which will be provided in a separate message).${extraInstruction}\nYour task is to:\n1. Create a detailed, step-by-step plan outlining how to resolve the ${itemType} effectively.\n2. Identify files that need to be modified to resolve the ${itemType}.\n3. Provide a concise and descriptive commit message for the changes, following the Conventional Commits specification.\n\nYour plan should:\n- Focus on implementation details for each file that needs modification\n- Be clear and actionable for a developer to follow\n- Prefer showing diffs rather than complete file contents when describing changes\n- Exclude testing procedures as those will be handled separately\n\nGitHub ${isPullRequest ? 'Pull Request' : 'Issue'}:\n${issueYamlText}\n\nPlease format your response without any explanatory text as follows:\n\\`\\`\\`md\n${HEADING_OF_PLAN}\n\n1. [Specific implementation step]\n2. [Next implementation step]\n...\n\n${HEADING_OF_FILE_PATHS_TO_BE_MODIFIED}\n\n- \\`[filePath1]\\`\n- \\`[filePath2]\\`\n- ...\n\n${HEADING_OF_COMMIT_MESSAGE}\n\n[commit message]\n\\`\\`\\`\n`;\n}\n","export const isCI = process.env.CI || process.env.GITHUB_ACTIONS;\n","import type { SpawnOptions } from 'node:child_process';\nimport type { MainOptions } from '../main.js';\nimport type { ResolutionPlan } from '../plan.js';\nimport { buildAiderArgs } from '../tools/aider.js';\nimport { buildClaudeCodeArgs } from '../tools/claudeCode.js';\nimport { buildCodexArgs } from '../tools/codex.js';\nimport { buildGeminiArgs } from '../tools/gemini.js';\nimport type { CodingTool, NodeRuntimeActual } from '../types.js';\n\nexport interface ToolArgs {\n  prompt: string;\n  resolutionPlan?: ResolutionPlan;\n}\n\nexport interface ToolConfig {\n  name: string;\n  buildArgs: (options: MainOptions, args: ToolArgs) => string[];\n  getCommand: (nodeRuntime: NodeRuntimeActual) => string;\n  getRunOptions?: () => SpawnOptions;\n}\n\nexport const TOOL_REGISTRY: Record<CodingTool, ToolConfig> = {\n  aider: {\n    name: 'Aider',\n    buildArgs: buildAiderArgs,\n    getCommand: () => 'aider',\n  },\n  'claude-code': {\n    name: 'Claude Code',\n    buildArgs: buildClaudeCodeArgs,\n    getCommand: (nodeRuntime) => nodeRuntime,\n    getRunOptions: () => ({ stdio: 'inherit' }),\n  },\n  'codex-cli': {\n    name: 'Codex CLI',\n    buildArgs: buildCodexArgs,\n    getCommand: (nodeRuntime) => nodeRuntime,\n  },\n  'gemini-cli': {\n    name: 'Gemini CLI',\n    buildArgs: buildGeminiArgs,\n    getCommand: (nodeRuntime) => nodeRuntime,\n  },\n};\n\n/**\n * Get the display name for a coding tool\n */\nexport function getToolName(tool: CodingTool): string {\n  return TOOL_REGISTRY[tool].name;\n}\n\n/**\n * Get the command and arguments for a coding tool\n */\nexport function getToolCommandAndArgs(\n  tool: CodingTool,\n  options: MainOptions,\n  nodeRuntime: NodeRuntimeActual,\n  args: ToolArgs\n): { command: string; args: string[]; runOptions?: SpawnOptions } {\n  const config = TOOL_REGISTRY[tool];\n  return {\n    command: config.getCommand(nodeRuntime),\n    args: config.buildArgs(options, args),\n    runOptions: config.getRunOptions?.(),\n  };\n}\n\n/**\n * Create standard run options for tools\n */\nexport function createStandardRunOptions(): SpawnOptions & { ignoreExitStatus?: boolean } {\n  return {\n    env: { ...process.env, NO_COLOR: '1' },\n    ignoreExitStatus: true,\n  };\n}\n","import { DEFAULT_AIDER_EXTRA_ARGS } from '../defaultOptions.js';\nimport type { MainOptions } from '../main.js';\nimport type { ResolutionPlan } from '../plan.js';\n\nimport { parseCommandLineArgs } from '../spawn.js';\n\n/**\n * Builds the command line arguments for the Aider command\n *\n * @param options The main options object\n * @param args Arguments to include\n * @returns An array of command line arguments for aider\n */\nexport function buildAiderArgs(\n  options: MainOptions,\n  args: { prompt: string; resolutionPlan?: ResolutionPlan }\n): string[] {\n  const aiderArgs = [\n    '--yes-always',\n    '--no-check-update',\n    '--no-gitignore',\n    '--no-show-model-warnings',\n    '--no-show-release-notes',\n    ...parseCommandLineArgs(options.aiderExtraArgs || DEFAULT_AIDER_EXTRA_ARGS),\n    '--message',\n    args.prompt,\n  ];\n\n  if (options.dryRun) {\n    aiderArgs.push('--dry-run');\n  }\n  if (args.resolutionPlan && 'filePaths' in args.resolutionPlan) {\n    aiderArgs.push(...args.resolutionPlan.filePaths);\n  }\n\n  return aiderArgs;\n}\n","import { isCI } from '../ci.js';\nimport { DEFAULT_CLAUDE_CODE_EXTRA_ARGS } from '../defaultOptions.js';\nimport type { MainOptions } from '../main.js';\nimport type { ResolutionPlan } from '../plan.js';\n\nimport { parseCommandLineArgs } from '../spawn.js';\n\n/**\n * Builds the command line arguments for the Claude Code command\n *\n * @param options The main options object\n * @param args Arguments to include\n * @returns An array of command line arguments for @anthropic-ai/claude-code@latest\n */\nexport function buildClaudeCodeArgs(\n  options: MainOptions,\n  args: { prompt: string; resolutionPlan?: ResolutionPlan }\n): string[] {\n  // cf. https://docs.anthropic.com/en/docs/claude-code/cli-usage\n  const baseArgs = [\n    '--yes',\n    '@anthropic-ai/claude-code@latest',\n    ...parseCommandLineArgs(options.claudeCodeExtraArgs || DEFAULT_CLAUDE_CODE_EXTRA_ARGS),\n    // Bypass all permission checks\n    '--dangerously-skip-permissions',\n  ];\n  if (isCI) {\n    baseArgs.push('--print');\n  }\n  baseArgs.push(args.prompt);\n  return baseArgs;\n}\n","import { DEFAULT_CODEX_EXTRA_ARGS } from '../defaultOptions.js';\nimport type { MainOptions } from '../main.js';\nimport type { ResolutionPlan } from '../plan.js';\n\nimport { parseCommandLineArgs } from '../spawn.js';\n\n/**\n * Builds the command line arguments for the Codex CLI command\n *\n * @param options The main options object\n * @param args Arguments to include\n * @returns An array of command line arguments for @openai/codex@latest\n */\nexport function buildCodexArgs(\n  options: MainOptions,\n  args: { prompt: string; resolutionPlan?: ResolutionPlan }\n): string[] {\n  return [\n    '--yes',\n    '@openai/codex@latest',\n    'exec',\n    ...parseCommandLineArgs(options.codexExtraArgs || DEFAULT_CODEX_EXTRA_ARGS),\n    args.prompt,\n  ];\n}\n","import { DEFAULT_GEMINI_EXTRA_ARGS } from '../defaultOptions.js';\nimport type { MainOptions } from '../main.js';\nimport type { ResolutionPlan } from '../plan.js';\n\nimport { parseCommandLineArgs } from '../spawn.js';\n\n/**\n * Builds the command line arguments for the Gemini CLI command\n *\n * @param options The main options object\n * @param args Arguments to include\n * @returns An array of command line arguments for @google/gemini-cli@latest\n */\nexport function buildGeminiArgs(\n  options: MainOptions,\n  args: { prompt: string; resolutionPlan?: ResolutionPlan }\n): string[] {\n  return [\n    '--yes',\n    '@google/gemini-cli@latest',\n    '--yolo',\n    ...parseCommandLineArgs(options.geminiExtraArgs || DEFAULT_GEMINI_EXTRA_ARGS),\n    '--prompt',\n    args.prompt,\n  ];\n}\n","import ansis from 'ansis';\nimport type { MainOptions } from './main.js';\nimport { findDistinctFence } from './markdown.js';\nimport type { ResolutionPlan } from './plan.js';\nimport { normalizeNodeRuntime, parseCommandLineArgs, runCommand, spawnAsync } from './spawn.js';\nimport { createStandardRunOptions, getToolCommandAndArgs, getToolName } from './utils/toolRegistry.js';\n\nexport interface TestResult {\n  fixResult: string;\n  success: boolean;\n  error?: string;\n}\n\nexport async function testAndFix(options: MainOptions, resolutionPlan?: ResolutionPlan): Promise<TestResult> {\n  const [commandProgram, ...commandArgs] = parseCommandLineArgs(options.testCommand || '');\n  if (!commandProgram) return { fixResult: '', success: true };\n\n  const maxAttempts = options.maxTestAttempts;\n  let attempts = 0;\n  let fixResult = '';\n  let success = false;\n  let lastError = '';\n\n  while (attempts < maxAttempts) {\n    attempts++;\n    console.info(ansis.cyan(`Executing test command (attempt ${attempts}/${maxAttempts}): ${options.testCommand}`));\n\n    const testResult = await spawnAsync(commandProgram, commandArgs, {\n      cwd: process.cwd(),\n    });\n\n    if (testResult.status === 0) {\n      console.info(ansis.green('Test command passed successfully.'));\n      success = true;\n      break;\n    }\n\n    console.warn(ansis.yellow(`Test command failed with exit code ${testResult.status}.`));\n\n    // Capture the error details\n    lastError = `Test command failed with exit code ${testResult.status}\\n\\nStdout:\\n${testResult.stdout}\\n\\nStderr:\\n${testResult.stderr}`;\n\n    // Only try to fix if we haven't reached the maximum attempts\n    if (attempts >= maxAttempts) {\n      console.warn(ansis.yellow(`Maximum fix attempts (${maxAttempts}) reached. Giving up.`));\n      break;\n    }\n\n    const stdoutFence = findDistinctFence(testResult.stdout, '~');\n    const stderrFence = findDistinctFence(testResult.stderr, '~');\n    const prompt = `\nThe previous changes were applied, but the test command \\`${options.testCommand}\\` failed.\n\nExit code: ${testResult.status}\n\nStdout:\n${stdoutFence}\n${testResult.stdout}\n${stdoutFence}\n\nStderr:\n${stderrFence}\n${testResult.stderr}\n${stderrFence}\n\nPlease analyze the output and fix the errors.\n`.trim();\n\n    fixResult += await runToolFix(options, prompt, resolutionPlan);\n  }\n\n  return { fixResult, success, error: success ? undefined : lastError };\n}\n\n/**\n * Helper function to run coding tool with a fix prompt\n */\nexport async function runToolFix(\n  options: MainOptions,\n  prompt: string,\n  resolutionPlan?: ResolutionPlan\n): Promise<string> {\n  const toolName = getToolName(options.codingTool);\n\n  // Normalize the runtime value (convert aliases to actual commands)\n  const nodeRuntime = normalizeNodeRuntime(options.nodeRuntime);\n\n  // Build tool configuration using registry\n  const {\n    command,\n    args: toolArgs,\n    runOptions,\n  } = getToolCommandAndArgs(options.codingTool, options, nodeRuntime, {\n    prompt,\n    resolutionPlan,\n  });\n\n  const runOpts = {\n    ...createStandardRunOptions(),\n    ...runOptions,\n  };\n\n  console.info(ansis.cyan(`Asking ${toolName} to fix \"${options.testCommand}\"...`));\n  const assistantResult = (await runCommand(command, toolArgs, runOpts)).stdout;\n\n  return `\\n\\n## ${toolName} fix attempt for \"${options.testCommand}\"\\n\\n${assistantResult.trim()}`;\n}\n","import type { SpawnOptions } from 'node:child_process';\nimport ansis from 'ansis';\nimport YAML from 'yaml';\nimport { configureEnvVars } from './env.js';\nimport { configureGitUserDetailsIfNeeded, getBaseBranch, getCurrentBranch, getHeaderOfFirstCommit } from './git.js';\nimport { createIssueInfo } from './issue.js';\nimport { findDistinctFence } from './markdown.js';\nimport { createPullRequest } from './octokit.js';\nimport { planCodeChanges } from './plan.js';\nimport { normalizeNodeRuntime, runCommand } from './spawn.js';\nimport { testAndFix } from './test.js';\nimport { HEADING_OF_GEN_PR_METADATA, truncateText } from './text.js';\nimport type { CodingTool, NodeRuntime, NodeRuntimeActual, ReasoningEffort } from './types.js';\nimport { logVerboseOptions } from './utils/logging.js';\nimport { createStandardRunOptions, getToolCommandAndArgs, getToolName } from './utils/toolRegistry.js';\nimport { yamlStringifyOptions } from './yaml.js';\n\n/**\n * Options for the main function\n */\nexport interface MainOptions {\n  /** Additional arguments to pass to Aider */\n  aiderExtraArgs?: string;\n  /** Additional arguments to pass to Claude Code */\n  claudeCodeExtraArgs?: string;\n  /** Additional arguments to pass to Codex CLI */\n  codexExtraArgs?: string;\n  /** Additional arguments to pass to Gemini CLI */\n  geminiExtraArgs?: string;\n  /** Coding tool to use */\n  codingTool: CodingTool;\n  /** Enable two-staged planning: first select relevant files, then generate detailed implementation plans */\n  twoStagePlanning: boolean;\n  /** Run without making actual changes (no branch creation, no PR) */\n  dryRun: boolean;\n  /** Do not create a new branch, commit changes directly to the base branch */\n  noBranch: boolean;\n  /** Node.js runtime to use */\n  nodeRuntime: NodeRuntime;\n  /** GitHub issue number to process */\n  issueNumber: number;\n  /** Maximum number of attempts to fix test failures */\n  maxTestAttempts: number;\n  /** LLM model to use for planning code changes */\n  planningModel?: string;\n  /** Level of reasoning effort for the LLM */\n  reasoningEffort?: ReasoningEffort;\n  /** Extra arguments for repomix when generating context */\n  repomixExtraArgs?: string;\n  /** Command to run after coding tool applies changes. If it fails, the assistant will try to fix it. */\n  testCommand?: string;\n  /** RegExp pattern to remove from issue and PR descriptions */\n  removePattern?: string;\n  /** Print parsed options at start */\n  verbose?: boolean;\n}\n\nconst MAX_PR_BODY_LENGTH = 30000; // GitHub's limit is 65536, leave some buffer\n\nexport async function main(options: MainOptions): Promise<void> {\n  configureEnvVars();\n\n  // Print parsed options if verbose flag is set\n  logVerboseOptions(options, options.verbose);\n\n  // Normalize the runtime value (convert aliases to actual commands)\n  const nodeRuntime: NodeRuntimeActual = normalizeNodeRuntime(options.nodeRuntime);\n\n  if (options.dryRun) {\n    console.info(ansis.yellow('Running in dry-run mode. No branches or PRs will be created.'));\n  } else {\n    await configureGitUserDetailsIfNeeded();\n  }\n\n  // Install coding tools\n  if (options.codingTool === 'aider') {\n    await runCommand('python', ['-m', 'pip', 'install', 'aider-install']);\n    await reshimToDetectNewTools();\n    await runCommand('uv', ['tool', 'uninstall', 'aider-chat'], { ignoreExitStatus: true });\n    await runCommand('aider-install', []);\n    await reshimToDetectNewTools();\n\n    if (options.aiderExtraArgs?.includes('bedrock/')) {\n      await runCommand('uv', [\n        'tool',\n        'run',\n        '--from',\n        'aider-chat',\n        'pip',\n        'install',\n        '--upgrade',\n        '--upgrade-strategy',\n        'only-if-needed',\n        'boto3',\n      ]);\n      // await runCommand('aider', ['--install-main-branch', '--yes-always']);\n    }\n  }\n\n  const prBaseBranch = await getBaseBranch(options);\n  const isPullRequest = !!prBaseBranch;\n  const baseBranch = prBaseBranch || (await getCurrentBranch());\n\n  const issueInfo = await createIssueInfo(options);\n  const issueText = YAML.stringify(issueInfo, yamlStringifyOptions).trim();\n\n  const resolutionPlan =\n    (options.planningModel &&\n      (await planCodeChanges(\n        options.planningModel,\n        issueText,\n        options.twoStagePlanning,\n        options.reasoningEffort,\n        options.repomixExtraArgs,\n        isPullRequest\n      ))) ||\n    undefined;\n  console.info('Resolution plan:', resolutionPlan);\n\n  const planText = (resolutionPlan && 'plan' in resolutionPlan && resolutionPlan.plan) || '';\n  const issueFence = findDistinctFence(issueText, '~');\n  const isAgentic = options.codingTool !== 'aider';\n  const target = isPullRequest ? 'the comments on the following GitHub pull request' : 'the following GitHub issue';\n  const prompt = `\nModify the code to resolve ${target}${planText ? ' based on the plan' : ''}.${isAgentic ? ' After that, commit your changes with a message, following the Conventional Commits specification.' : ''}\n\n## ${isPullRequest ? 'Pull Request' : 'Issue'}\n\n${issueFence}yml\n${issueText}\n${issueFence}\n\n${\n  planText &&\n  `## Plan\n\n${planText}`\n}\n`.trim();\n\n  const now = new Date();\n  const newBranchName = `gen-pr-${options.issueNumber}-${options.codingTool}-${now.getFullYear()}_${getTwoDigits(now.getMonth() + 1)}${getTwoDigits(now.getDate())}_${getTwoDigits(now.getHours())}${getTwoDigits(now.getMinutes())}${getTwoDigits(now.getSeconds())}`;\n\n  if (options.noBranch) {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would commit directly to base branch: ${baseBranch}`));\n    } else {\n      if (isPullRequest) {\n        await runCommand('git', ['fetch', 'origin', baseBranch]);\n      }\n      await runCommand('git', ['switch', baseBranch]);\n    }\n  } else {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would create branch: ${newBranchName}`));\n    } else {\n      if (isPullRequest) {\n        await runCommand('git', ['fetch', 'origin', baseBranch]);\n        await runCommand('git', ['switch', baseBranch]);\n      }\n      await runCommand('git', ['switch', '--force-create', newBranchName]);\n    }\n  }\n\n  // Execute coding tool\n  let toolResult = '';\n  let toolCommand: string;\n  let toolError = '';\n  let toolSuccess = true;\n  const toolName = getToolName(options.codingTool);\n\n  // Build tool configuration using registry\n  const {\n    command,\n    args: toolArgs,\n    runOptions,\n  } = getToolCommandAndArgs(options.codingTool, options, nodeRuntime, {\n    prompt,\n    resolutionPlan,\n  });\n\n  const runOpts: SpawnOptions & { ignoreExitStatus?: boolean } = {\n    ...createStandardRunOptions(),\n    ...runOptions,\n  };\n\n  toolCommand = buildToolCommandString(command, toolArgs, prompt);\n\n  // Execute tool command\n  if (options.dryRun) {\n    console.info(`\\n=== DRY MODE: ${toolName} Prompt ===`);\n    console.info(prompt);\n    console.info(`=== End ${toolName} Prompt ===\\n`);\n    console.info(ansis.yellow(`Would run: ${toolCommand}`));\n    toolResult = 'Skipped due to dry-run mode';\n  } else {\n    const toolRunResult = await runCommand(command, toolArgs, runOpts);\n    toolResult = toolRunResult.stdout || '';\n    if (toolRunResult.status !== 0) {\n      toolSuccess = false;\n      toolError = `${toolName} failed with exit code ${toolRunResult.status}\\n${toolRunResult.stderr}`;\n      console.error(ansis.red(`${toolName} execution failed: ${toolError}`));\n    }\n  }\n\n  let toolResponse = (toolResult || '').trim();\n  let testSuccess = true;\n  let testError = '';\n  if (options.dryRun) {\n    console.info(ansis.yellow(`Would run test command`));\n  } else {\n    const testResult = await testAndFix(options, resolutionPlan);\n    toolResponse += testResult.fixResult;\n    testSuccess = testResult.success;\n    testError = testResult.error || '';\n    if (!testSuccess) {\n      console.warn(ansis.yellow('Tests failed after all fix attempts. Will create a draft PR.'));\n    }\n  }\n\n  if (!toolSuccess) {\n    console.warn(ansis.yellow(`${toolName} execution failed. Will create a draft PR.`));\n  }\n\n  // Determine if PR should be a draft\n  const shouldBeDraft = !toolSuccess || !testSuccess;\n  let errorLogs = '';\n  if (shouldBeDraft) {\n    if (toolError) {\n      errorLogs += `\\n\\n### ❌ ${toolName} Execution Error\\n\\n\\`\\`\\`\\n${toolError}\\n\\`\\`\\``;\n    }\n    if (testError) {\n      errorLogs += `\\n\\n### ❌ Test Execution Error\\n\\n\\`\\`\\`\\n${testError}\\n\\`\\`\\``;\n    }\n  }\n\n  // Try commiting changes because coding tool may fail to commit changes due to pre-commit hooks\n  const commitMessage = resolutionPlan?.commitMessage || `fix: Close #${options.issueNumber}`;\n  await runCommand('git', ['add', '-A'], { ignoreExitStatus: true });\n  if (\n    (\n      await runCommand('git', ['commit', '-m', commitMessage], {\n        ignoreExitStatus: true,\n      })\n    ).status !== 0\n  ) {\n    await runCommand('git', ['commit', '-m', commitMessage, '--no-verify'], {\n      ignoreExitStatus: true,\n    });\n  }\n  if (options.noBranch) {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would push changes directly to base branch: ${baseBranch}`));\n    } else {\n      await runCommand('git', ['push', 'origin', baseBranch, '--no-verify']);\n    }\n  } else {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would push branch: ${newBranchName} to origin`));\n    } else {\n      await runCommand('git', ['push', 'origin', newBranchName, '--no-verify']);\n    }\n  }\n\n  // Create a PR using GitHub CLI (unless noBranch is enabled)\n  if (options.noBranch) {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Skipping PR creation due to --no-branch option`));\n    } else {\n      console.info(`Changes committed directly to base branch: ${baseBranch}`);\n    }\n  } else {\n    const prTitle = getHeaderOfFirstCommit(baseBranch) || commitMessage;\n    let prBody = isPullRequest ? '' : `Close #${options.issueNumber}`;\n\n    if (options.planningModel) {\n      prBody += `\n\n${HEADING_OF_GEN_PR_METADATA}\n\n- **Planning Model:** ${options.planningModel}`;\n    }\n\n    prBody += `\n- **Coding Tool:** ${toolName}\n- **Coding Command:** \\`${toolCommand}\\``;\n    if (planText) {\n      const responseFence = findDistinctFence(planText, '~');\n      prBody += `\n\n### Plan\n\n${responseFence}\n${truncateText(planText, (toolResponse.length / (planText.length + toolResponse.length)) * MAX_PR_BODY_LENGTH)}\n${responseFence}`;\n    }\n    if (toolResponse) {\n      const responseFence = findDistinctFence(toolResponse, '~');\n      prBody += `\n\n### ${toolName} Log\n\n${responseFence}\n${truncateText(toolResponse, (toolResponse.length / (planText.length + toolResponse.length)) * MAX_PR_BODY_LENGTH)}\n${responseFence}`;\n    }\n    prBody = prBody.replaceAll(/(?:\\s*\\n){2,}/g, '\\n\\n').trim();\n\n    // Add error logs to PR body if it's a draft due to failures\n    if (shouldBeDraft) {\n      prBody += errorLogs;\n    }\n\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would create PR with title: ${prTitle}${shouldBeDraft ? ' (as draft)' : ''}`));\n    } else {\n      await createPullRequest({\n        title: prTitle,\n        body: prBody,\n        head: newBranchName,\n        base: baseBranch,\n        draft: shouldBeDraft,\n      });\n    }\n  }\n\n  console.info(`\\n${isPullRequest ? 'Pull request' : 'Issue'} #${options.issueNumber} processed successfully.`);\n}\n\nasync function reshimToDetectNewTools() {\n  try {\n    // Make uv available on asdf environment\n    await runCommand('asdf', ['reshim'], { ignoreExitStatus: true });\n  } catch {\n    // do nothing\n  }\n}\n\nfunction getTwoDigits(value: number): string {\n  return String(value).padStart(2, '0');\n}\n\n/**\n * Builds a command string for display, replacing the prompt argument with ...\n */\nfunction buildToolCommandString(command: string, args: string[], prompt: string): string {\n  const escapedArgs = args.map((arg) => {\n    if (arg === prompt) {\n      return '...';\n    }\n    if (arg.includes(' ') || arg.includes('\"') || arg.includes(\"'\")) {\n      return `\"${arg.replace(/\"/g, '\\\\\"')}\"`;\n    }\n    return arg;\n  });\n  return `${command} ${escapedArgs.join(' ')}`;\n}\n","import { config } from 'dotenv';\n\nexport function configureEnvVars(): void {\n  config();\n\n  if (!process.env.AWS_REGION && process.env.AWS_REGION_NAME) {\n    process.env.AWS_REGION = process.env.AWS_REGION_NAME;\n  } else if (process.env.AWS_REGION && !process.env.AWS_REGION_NAME) {\n    process.env.AWS_REGION_NAME = process.env.AWS_REGION;\n  }\n\n  if (!process.env.GEMINI_API_KEY && process.env.GOOGLE_GENERATIVE_AI_API_KEY) {\n    process.env.GEMINI_API_KEY = process.env.GOOGLE_GENERATIVE_AI_API_KEY;\n  } else if (process.env.GEMINI_API_KEY && !process.env.GOOGLE_GENERATIVE_AI_API_KEY) {\n    process.env.GOOGLE_GENERATIVE_AI_API_KEY = process.env.GEMINI_API_KEY;\n  }\n}\n","/**\n * Utility functions for logging\n */\n\n/**\n * Log options in verbose mode\n */\nexport function logVerboseOptions(options: unknown, verbose?: boolean): void {\n  if (verbose) {\n    console.info('Parsed options:');\n    console.info(JSON.stringify(options, null, 2));\n  }\n}\n","import child_process from 'node:child_process';\nimport type { MainOptions } from './main.js';\nimport { runCommand } from './spawn.js';\n\nexport async function configureGitUserDetailsIfNeeded(): Promise<void> {\n  const gitUserName = (await runCommand('git', ['config', 'user.name'], { ignoreExitStatus: true })).stdout.trim();\n  const gitUserEmail = (await runCommand('git', ['config', 'user.email'], { ignoreExitStatus: true })).stdout.trim();\n  if (!gitUserName || !gitUserEmail) {\n    const githubUserInfoJson = (await runCommand('gh', ['api', 'user'], { ignoreExitStatus: true })).stdout.trim();\n    try {\n      const { name, email } = JSON.parse(githubUserInfoJson);\n      if (name) {\n        await runCommand('git', ['config', 'user.name', name]);\n      }\n      if (email) {\n        await runCommand('git', ['config', 'user.email', email]);\n      }\n    } catch {}\n  }\n}\n\nexport async function getCurrentBranch(): Promise<string> {\n  const currentBranchResult = await runCommand('git', ['branch', '--show-current']);\n  return currentBranchResult.stdout.trim();\n}\n\nexport async function getBaseBranch(options: MainOptions): Promise<string | undefined> {\n  const { stdout: prViewResult } = await runCommand(\n    'gh',\n    ['pr', 'view', options.issueNumber.toString(), '--json', 'headRefName'],\n    { ignoreExitStatus: true }\n  );\n  try {\n    if (prViewResult) {\n      const prData = JSON.parse(prViewResult);\n      return prData.headRefName;\n    }\n  } catch {\n    // Not a PR or error parsing\n  }\n}\n\nexport function getGitRepoName(): string {\n  const repoUrlResult = child_process.spawnSync('git', ['remote', 'get-url', 'origin'], {\n    encoding: 'utf8',\n    stdio: 'pipe',\n  });\n  const repoUrl = repoUrlResult.stdout.trim();\n  const repoMatch = repoUrl.match(/github\\.com[/:]([\\w-]+\\/[\\w-]+)(\\.git)?$/);\n  return repoMatch?.[1] ?? '';\n}\n\nexport function getHeaderOfFirstCommit(baseBranch: string): string {\n  const firstCommitResult = child_process.spawnSync('git', ['log', `${baseBranch}..HEAD`, '--reverse', '--pretty=%s'], {\n    encoding: 'utf8',\n    stdio: 'pipe',\n  });\n  return (\n    firstCommitResult.stdout\n      .trim()\n      .split('\\n')\n      .find((line) => line.trim()) ?? ''\n  );\n}\n"],"names":["loadConfigFile","configOptions","name","cfgPath","path","resolve","process","cwd","fs","existsSync","YAML","parse","readFileSync","console","info","err","error","exit","DEFAULT_AIDER_EXTRA_ARGS","DEFAULT_CLAUDE_CODE_EXTRA_ARGS","DEFAULT_CODEX_EXTRA_ARGS","DEFAULT_GEMINI_EXTRA_ARGS","DEFAULT_REPOMIX_EXTRA_ARGS","DEFAULT_MAX_TEST_ATTEMPTS","DEFAULT_CODING_TOOL","DEFAULT_NODE_RUNTIME","HEADING_OF_GEN_PR_METADATA","truncateText","text","maxLength","length","truncated","slice","omitted","Math","floor","normalizeNewLines","replaceAll","trim","normalizeNodeRuntime","runtime","async","runCommand","command","args","options","ignoreExitStatus","spawnOptions","argsText","map","a","includes","join","ansis","green","ret","spawnAsync","truncateStdout","stdout","stderr","truncatedStderr","yellow","magenta","status","Promise","reject","sanitizedArgs","arg","replace","proc","spawn","setEncoding","on","data","write","code","signal","undefined","pid","Error","parseCommandLineArgs","argsString","result","current","inDoubleQuote","inSingleQuote","i","char","push","token","env","GH_TOKEN","GITHUB_TOKEN","child_process","spawnSync","encoding","octokit","Octokit","auth","graphqlClient","graphql","defaults","headers","authorization","repoOwner","repoName","getRepoInfo","match","owner","repo","fetchPRDiff","issueNumber","issueInfo","prDiff","pullNumber","pulls","get","pull_number","mediaType","format","getPullRequestDiff","code_changes","diffContent","MAX_TOTAL_DIFF_SIZE","MAX_FILE_DIFF_SIZE","LARGE_FILE_PATTERNS","fileSections","split","processedSections","totalSize","section","some","pattern","test","truncatedSection","processDiffContent","warn","fetchIssueData","processedIssues","isReferenced","has","issue","add","issueData","issues","issue_number","issueOrPullRequest","pull_request","prData","commentsResponse","listComments","author","user","login","title","body","labels","label","comments","comment","createdAt","created_at","url","html_url","getIssue","referencedNumbers","regex","numbers","exec","number","Number","parseInt","isInteger","Set","extractIssueReferences","c","rawBody","processedBody","markdownContent","index","indexOf","substring","stripMetadataSections","description","RegExp","removeRegexPattern","removePattern","commentsWithDate","Date","getTime","graphqlData","pr","getPullRequestReviewThreads","reviewThreads","repository","pullRequest","nodes","thread","isResolved","processReviewThreadComments","fetchPRReviewThreads","reviewResultComments","listReviews","review","state","submitted_at","getPullRequestReviews","reviewState","fetchPRReviews","referencedIssuesPromises","num","referencedIssues","all","filter","referenced_issues","fetchReferencedIssues","sort","b","codeContent","extractCodeFromDiffHunk","diffHunk","reviewComment","codeLocation","line","Object","keys","forEach","key","codeLine","find","startsWith","extractHeaderContents","modifiedResponse","indices","header","every","start","end","findDistinctFence","content","fenceChar","matches","max","seq","fenceLength","repeat","trimCodeBlockFences","convertToV4Messages","messages","msg","id","role","JSON","stringify","yamlStringifyOptions","blockQuote","lineWidth","callLlmApi","model","reasoningEffort","provider","modelParts","modelName","ollamaBaseURL","OLLAMA_BASE_URL","providerModel","createOllama","baseURL","OLLAMA_API_KEY","apiKey","ollama","generateText","providerOptions","think","logResult","charAt","toUpperCase","callV4ProviderApi","modelInstance","createOpenAI","openaiProvider","createAnthropic","anthropicProvider","createGoogleGenerativeAI","googleProvider","createAzure","azureProvider","createAmazonBedrock","bedrockProvider","createVertex","vertexProvider","createXai","grokProvider","createOpenRouter","OPENROUTER_API_KEY","openrouterProvider","reasoning","effort","getModelInstance","requestParams","modelSupportsReasoning","supportsReasoningOptions","thinkingBudget","low","medium","high","getThinkingBudget","openai","anthropic","thinking","type","budgetTokens","google","thinkingConfig","xai","usage","finishReason","REPOMIX_FILE_NAME","HEADING_OF_FILE_PATHS_TO_BE_MODIFIED","HEADING_OF_FILE_PATHS_TO_BE_REFERRED","HEADING_OF_PLAN","HEADING_OF_COMMIT_MESSAGE","buildPromptForSelectingFiles","issueYamlText","isPullRequest","itemType","buildPromptForPlanningCodeChanges","buildPromptForSelectingFilesAndPlanningCodeChanges","isCI","CI","GITHUB_ACTIONS","TOOL_REGISTRY","aider","buildArgs","aiderArgs","aiderExtraArgs","prompt","dryRun","resolutionPlan","filePaths","getCommand","baseArgs","claudeCodeExtraArgs","nodeRuntime","getRunOptions","stdio","codexExtraArgs","geminiExtraArgs","getToolName","tool","getToolCommandAndArgs","config","runOptions","createStandardRunOptions","NO_COLOR","runToolFix","toolName","codingTool","toolArgs","runOpts","cyan","testCommand","assistantResult","main","AWS_REGION","AWS_REGION_NAME","GEMINI_API_KEY","GOOGLE_GENERATIVE_AI_API_KEY","verbose","logVerboseOptions","gitUserName","gitUserEmail","githubUserInfoJson","email","configureGitUserDetailsIfNeeded","reshimToDetectNewTools","prBaseBranch","prViewResult","toString","headRefName","getBaseBranch","baseBranch","getCurrentBranch","createIssueInfo","issueText","planningModel","issueContent","twoStagePlanning","repomixExtraArgs","issueFence","repomixArgs","repomixResult","promises","rm","force","filesResponse","extractedFilePathLists","filePathsToBeModified","filePathsToBeReferred","filesContent","matchAll","Boolean","fileContents","filePath","fence","planResponse","extractedContents","commitMessage","plan","filePathsText","planCodeChanges","planText","now","newBranchName","getFullYear","getTwoDigits","getMonth","getDate","getHours","getMinutes","getSeconds","noBranch","toolCommand","toolResult","toolError","toolSuccess","escapedArgs","buildToolCommandString","toolRunResult","red","toolResponse","testSuccess","testError","testResult","commandProgram","commandArgs","fixResult","success","maxAttempts","maxTestAttempts","attempts","lastError","stdoutFence","stderrFence","testAndFix","shouldBeDraft","errorLogs","prTitle","getHeaderOfFirstCommit","prBody","responseFence","params","create","head","base","draft","createPullRequest","value","String","padStart"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "gen-pr",
3
- "version": "4.0.1",
3
+ "version": "4.1.1",
4
4
  "repository": "github:WillBooster/gen-pr",
5
5
  "license": "Apache-2.0",
6
6
  "author": "WillBooster Inc.",
@@ -1,2 +0,0 @@
1
- import e from"node:fs";import t from"node:path";import n from"yaml";import o from"ansis";import{config as i}from"dotenv";import s,{spawn as r}from"node:child_process";import{graphql as a}from"@octokit/graphql";import{Octokit as c}from"@octokit/rest";import{createAmazonBedrock as l}from"@ai-sdk/amazon-bedrock";import{createAnthropic as u}from"@ai-sdk/anthropic";import{createAzure as d}from"@ai-sdk/azure";import{createGoogleGenerativeAI as f}from"@ai-sdk/google";import{createVertex as p}from"@ai-sdk/google-vertex";import{createOpenAI as m}from"@ai-sdk/openai";import{createXai as g}from"@ai-sdk/xai";import{createOpenRouter as h}from"@openrouter/ai-sdk-provider";import{generateText as w}from"ai";import{generateText as y}from"ai-v4";import{createOllama as $}from"ollama-ai-provider-v2";function b(){let o={};for(const i of["gen-pr.config.yml","gen-pr.config.yaml"]){const s=t.resolve(process.cwd(),i);if(e.existsSync(s)){try{o=n.parse(e.readFileSync(s,"utf8")),console.info(`Loaded gen-pr config from ${i}`)}catch(e){console.error(`Failed to parse config file ${i}:`,e),process.exit(1)}break}}return o}const x="--model gemini/gemini-2.5-pro --edit-format diff-fenced",v="--allowedTools Bash Edit Write",E="--full-auto",A="",R='--compress --remove-empty-lines --include "src/**/*.{ts,tsx},**/*.md"',P=5,_="aider",I="npx";const S="## gen-pr Metadata";function C(e,t){if(e.length>t){const n=e.slice(0,t),o=e.length-t;return`${n}\n\n... (${Math.floor(o)} characters truncated) ...`}return e}function O(e){return e.replaceAll("\r\n","\n").trim()}function k(e){switch(e){case"node":case"npx":return"npx";case"bun":case"bunx":return"bunx"}}async function N(e,t,n){const{ignoreExitStatus:i,...s}=n??{},r=t.map(e=>e.includes(" ")?`"${e.replaceAll('"','"')}"`:e).join(" ");console.info(o.green(`$ ${e} ${r}`)),console.info("stdout: ---------------------");const a=await T(e,t,s);s.truncateStdout&&console.info(C(a.stdout,3e3));const c=a.stderr.trim();if(c){console.info("stderr: ---------------------");const e=C(c,3e3);console.info(o.yellow(e))}return console.info("-----------------------------"),console.info(o.magenta(`Exit code: ${a.status}\n`)),i||0===a.status||null===a.status||process.exit(a.status),a}async function T(e,t,n){return new Promise((o,i)=>{try{const s=(t??[]).map(e=>e.replace(/\0/g,"")),a=r(e,s,n);a.stdout?.setEncoding?.("utf8"),a.stderr?.setEncoding?.("utf8");let c="",l="";a.stdout?.on("data",e=>{n?.truncateStdout||process.stdout.write(e),c+=e}),a.stderr?.on("data",e=>{l+=e}),a.on("error",e=>{i(e)}),a.on("close",(e,t)=>{void 0===a.pid?i(new Error("Process has no pid.")):o({pid:a.pid,stdout:c,stderr:l,status:e,signal:t})})}catch(e){i(e)}})}function G(e){if(!e)return[];const t=[];let n="",o=!1,i=!1;for(let s=0;s<e.length;s++){const r=e[s];'"'!==r||i?"'"!==r||o?" "!==r||o||i?n+=r:n&&(t.push(n),n=""):i=!i:o=!o}return n&&t.push(n),t}const M=process.env.GH_TOKEN||process.env.GITHUB_TOKEN||s.spawnSync("gh",["auth","token"],{encoding:"utf-8"}).stdout.trim();if(!M)throw new Error("GitHub token not found. Please set GH_TOKEN or GITHUB_TOKEN environment variable, or authenticate with gh CLI");const L=new c({auth:M}),W=a.defaults({headers:{authorization:`token ${M}`}});let H,Y;async function q(){if(!H||!Y){const{stdout:e}=await N("git",["remote","get-url","origin"],{ignoreExitStatus:!0}),t=e.trim().match(/github\.com[:/]([^/]+)\/(.+)/);if(!t?.[1]||!t[2])throw new Error("Could not parse GitHub repository from remote URL");H=t[1],Y=t[2].replace(/\.git$/,"")}if(!H||!Y)throw new Error("Repository information not properly initialized");return{owner:H,repo:Y}}async function F(e,t){try{const n=await async function(e){const{owner:t,repo:n}=await q();return(await L.pulls.get({owner:t,repo:n,pull_number:e,mediaType:{format:"diff"}})).data}(e);if(!n.trim())return;t.code_changes=function(e){const t=5e4,n=1e4,o=[/^diff --git a\/dist\//m,/^diff --git a\/build\//m,/^diff --git a\/.*\.bundle\./m,/^diff --git a\/.*\.min\./m,/^diff --git a\/node_modules\//m];if(e.length<=t)return e;const i=e.split(/(?=^diff --git)/m),s=[];let r=0;for(const e of i){if(!e.trim())continue;if(o.some(t=>t.test(e))){const t=[...e.split("\n").slice(0,4),"@@ ... @@","... (large bundled/compiled file diff truncated) ...",""].join("\n");s.push(t),r+=t.length}else if(e.length>n){const t=`${e.slice(0,n)}\n... (diff truncated) ...\n`;s.push(t),r+=t.length}else s.push(e),r+=e.length;if(r>.9*t){s.push("\n... (remaining diffs truncated) ...\n");break}}return s.join("")}(n.trim())}catch(e){console.warn("Failed to fetch PR diff:",e)}}async function K(e,t,n,o=!1){if(t.has(e))return;let i;t.add(e);try{i=await async function(e){const{owner:t,repo:n}=await q(),{data:o}=await L.issues.get({owner:t,repo:n,issue_number:e});let i=o;if(o.pull_request){const{data:o}=await L.pulls.get({owner:t,repo:n,pull_number:e});i=o}const s=await L.issues.listComments({owner:t,repo:n,issue_number:e});return{author:i.user?.login||"",title:i.title,body:i.body||"",labels:i.labels.map(e=>({name:"string"==typeof e?e:e.name||""})),comments:s.data.map(e=>({author:e.user?.login||"",body:e.body||"",createdAt:e.created_at})),url:i.html_url}}(e)}catch(t){return void console.warn(`Failed to fetch issue #${e}:`,t)}const s=function(e){const t=/(?:^|\s)#(\d+)/g,n=[];for(;;){const o=t.exec(e);if(!o)break;const i=Number.parseInt(o[1]??"",10);Number.isInteger(i)&&n.push(i)}return[...new Set(n)]}([i.body,...i.comments.map(e=>e.body)].join("\n")),r=i.body.replace(/<!--[\s\S]*?-->/g,"");const a=i.url?.includes("/pull/")?function(e){const t=e.indexOf(S);return t>=0?e.substring(0,t):e}(r):r,c=function(e,t){if(!t)return e;try{const n=new RegExp(t,"g");return e.replace(n,"")}catch(n){return console.warn(`Invalid regex pattern "${t}":`,n),e}}(a,n.removePattern||""),l=i.comments.map(e=>({author:e.author,body:O(e.body),createdAt:new Date(e.createdAt).getTime()})),u={author:i.author,title:i.title,description:O(c),comments:[]};return i.url?.includes("/pull/")&&!o&&(await F(e,u),await async function(e,t){try{const n=await async function(e){const{owner:t,repo:n}=await q();return await W("\n query($owner: String!, $repo: String!, $pr: Int!) {\n repository(owner: $owner, name: $repo) {\n pullRequest(number: $pr) {\n reviewThreads(first: 100) {\n nodes {\n isResolved\n comments(first: 100) {\n nodes {\n author {\n login\n }\n body\n path\n line\n diffHunk\n createdAt\n }\n }\n }\n }\n }\n }\n }\n ",{owner:t,repo:n,pr:e})}(e),o=n?.repository?.pullRequest?.reviewThreads?.nodes||[];for(const e of o)!e.isResolved&&e.comments?.nodes&&B(e.comments.nodes,t)}catch(e){console.warn("Failed to fetch PR review threads:",e)}}(e,l),await async function(e,t){try{const n=(await async function(e){const{owner:t,repo:n}=await q();return(await L.pulls.listReviews({owner:t,repo:n,pull_number:e})).data.map(e=>({user:{login:e.user?.login||""},state:e.state,body:e.body||"",submitted_at:e.submitted_at||""}))}(e)).map(e=>({author:e.user.login,reviewState:e.state,body:O(e.body),createdAt:new Date(e.submitted_at).getTime()}));t.push(...n)}catch(e){console.warn("Failed to fetch PR reviews:",e)}}(e,l)),s.length>0&&await async function(e,t,n,o){const i=e.map(e=>K(e,t,n,!0)),s=(await Promise.all(i)).filter(e=>!!e);if(0===s.length)return;o.referenced_issues=s}(s,t,n,u),u.comments=l.filter(e=>e.body).sort((e,t)=>e.createdAt-t.createdAt).map(({createdAt:e,...t})=>t),u}function B(e,t){for(const n of e){if(!n.author||!n.body)continue;const e=D(n.diffHunk),o={author:n.author.login,codeLocation:n.path&&n.line?`${n.path}:${n.line}`:void 0,codeContent:e||void 0,body:O(n.body),createdAt:new Date(n.createdAt).getTime()};Object.keys(o).forEach(e=>{void 0===o[e]&&delete o[e]}),t.push(o)}}function D(e){if(!e)return"";const t=e.split("\n").find(e=>(e.startsWith("+")||e.startsWith("-"))&&!e.startsWith("@@")&&e.trim().length>1);return t?.trim()||""}function j(e,t){const n=`\n${e}`,o=t.map(e=>n.indexOf(`\n${e}`));if(!o.some(e=>-1===e)&&o.every((e,t)=>0===t||e>o[t-1]))return t.map((e,i)=>{const s=o[i]+1+e.length,r=i+1<t.length?o[i+1]+1:n.length;return n.slice(s,r).trim()})}function z(e,t){const n=new RegExp(`${t}{3,}`,"g"),o=e.match(n),i=o?Math.max(...o.map(e=>e.length)):0,s=Math.max(3,i+1);return t.repeat(s)}function U(e){return e.trim().replace(/^(`{3,}|~{3,})[\s\S]*?\n([\s\S]*?)\n\1\s*$/,"$2")}function V(e){return e.map((e,t)=>({id:`msg-${t}`,role:"tool"===e.role?"data":e.role,content:"string"==typeof e.content?e.content:JSON.stringify(e.content)}))}const J={blockQuote:"literal",lineWidth:0};async function Q(e,t,n){try{if(e.startsWith("ollama/"))return await async function(e,t,n){const[o,...i]=e.split("/");try{const s=i.join("/");s||(console.error(`Invalid ${o} model format: ${e}. Expected format: ${o}/model-name`),process.exit(1));const r=`${process.env.OLLAMA_BASE_URL||"http://localhost:11434"}/api`,a=$({baseURL:r,...process.env.OLLAMA_API_KEY&&{apiKey:process.env.OLLAMA_API_KEY}})(s),c=await y({model:a,providerOptions:n?{ollama:{think:!0}}:void 0,messages:V(t)});return X(e,c),c.text}catch(t){console.error(`${o.charAt(0).toUpperCase()+o.slice(1)} API error for model ${e}:`,t),process.exit(1)}}(e,t,n);const[o,i,s]=function(e,t){e.includes("/")||(console.error(`Model must be in format 'provider/model'. Got: ${e}`),process.exit(1));const[n,...o]=e.split("/"),i=o.join("/");switch(n){case"openai":return[m()(i),n,i];case"anthropic":return[u()(i),n,i];case"gemini":return[f()(i),n,i];case"azure":return[d()(i),n,i];case"bedrock":return[l()(i),n,i];case"vertex":return[p()(i),n,i];case"xai":return[g()(i),n,i];case"openrouter":return[h({apiKey:process.env.OPENROUTER_API_KEY,headers:{"HTTP-Referer":"https://github.com/WillBooster/gen-pr","X-Title":"gen-pr"}})(i,t?{reasoning:{effort:t}}:{}),n,i];default:console.error(`Unsupported provider: ${n}. Supported providers: openai, azure, google, anthropic, bedrock, vertex, grok, openrouter, ollama`),process.exit(1)}}(e,n),r={model:o,messages:t};if(n){const t=function(e,t){switch(e){case"openai":case"azure":return/^(o1|o3|o4)/.test(t);case"anthropic":return/^claude-(opus-4|sonnet-4|3-7-sonnet)/.test(t);case"gemini":return/^gemini-2\.5/.test(t);case"bedrock":return/^(us\.)?anthropic\.claude-(opus-4|sonnet-4|3-7-sonnet)/.test(t);case"vertex":return/^gemini-2\.5/.test(t)||/^claude-(3-7-sonnet|opus-4|sonnet-4)/.test(t);case"xai":return/^grok-3/.test(t);case"openrouter":return!0;default:return!1}}(i,s);if(t){const e=function(e){return{low:4e3,medium:8e3,high:24e3}[e]}(n);"openai"===i?r.providerOptions={openai:{reasoningEffort:n}}:"anthropic"===i?r.providerOptions={anthropic:{thinking:{type:"enabled",budgetTokens:e}}}:"gemini"===i?r.providerOptions={google:{thinkingConfig:{thinkingBudget:e}}}:"bedrock"===i?console.warn(`Note: The current AI SDK doesn't work on Bedrock with reasoning. Model ${s} will use default reasoning settings.`):"xai"===i&&(r.providerOptions={xai:{reasoningEffort:n}})}else console.warn(`Model ${e} does not support reasoning/thinking options. Ignoring reasoning effort parameter.`)}const a=await w(r);return X(e,a),a.text}catch(t){console.error(`LLM API error for model ${e}:`,t),process.exit(1)}}function X(e,t){console.info(`${e}:`,n.stringify({text:t.text,usage:t.usage,finishReason:t.finishReason},J))}const Z="repomix.result",ee="## File Paths to be Modified",te="## File Paths to be Referred",ne="## Implementation Plan",oe="## Commit Message";function ie(e,t=!1){const n=t?"pull request":"issue";return`\nYou are an expert software developer tasked with analyzing GitHub ${n}s and identifying relevant files for code changes.\n\nReview the following GitHub ${n} and the list of available file paths and their contents (which will be provided in a separate message).${t?" Consider the comments on the pull request when identifying files.":""}\nYour task is to identify:\n1. Files that need to be MODIFIED to resolve the ${n}\n2. Files that should be REFERRED to (but not modified) to understand the codebase better\n\nGitHub ${t?"Pull Request":"Issue"}:\n${e}\n\nPlease format your response without any explanatory text as follows:\n\`\`\`md\n${ee}\n\n- \`[filePath1]\`\n- \`[filePath2]\`\n- ...\n\n${te}\n\n- \`[filePath1]\`\n- \`[filePath2]\`\n- ...\n\`\`\`\n`}function se(e,t=!1){const n=t?"pull request":"issue";return`\nYou are an expert software developer tasked with creating an implementation plan based on GitHub ${n}s.\n\nReview the following GitHub ${n} and the provided file contents (which will be provided in a separate message).${t?" Consider the comments on the pull request when creating the plan.":""}\nCreate a detailed, step-by-step plan outlining how to address the ${n} effectively.\nAlso, provide a concise and descriptive commit message for the changes, following the Conventional Commits specification.\n\nYour plan should:\n- Focus on implementation details for each file that needs modification\n- Be clear and actionable for a developer to follow\n- Prefer showing diffs rather than complete file contents when describing changes\n- Exclude testing procedures unless users explicitly request\n\nGitHub ${t?"Pull Request":"Issue"}:\n${e}\n\nPlease format your response without any explanatory text as follows:\n\`\`\`md\n${ne}\n\n1. [Specific implementation step]\n2. [Next implementation step]\n...\n\n${oe}\n\n[commit message]\n\`\`\`\n`.trim()}function re(e,t=!1){const n=t?"pull request":"issue";return`\nYou are an expert software developer tasked with analyzing GitHub ${n}s and creating an implementation plan.\n\nReview the following GitHub ${n} and the list of available file paths and their contents (which will be provided in a separate message).${t?" Consider the comments on the pull request when creating the plan.":""}\nYour task is to:\n1. Create a detailed, step-by-step plan outlining how to resolve the ${n} effectively.\n2. Identify files that need to be modified to resolve the ${n}.\n3. Provide a concise and descriptive commit message for the changes, following the Conventional Commits specification.\n\nYour plan should:\n- Focus on implementation details for each file that needs modification\n- Be clear and actionable for a developer to follow\n- Prefer showing diffs rather than complete file contents when describing changes\n- Exclude testing procedures as those will be handled separately\n\nGitHub ${t?"Pull Request":"Issue"}:\n${e}\n\nPlease format your response without any explanatory text as follows:\n\`\`\`md\n${ne}\n\n1. [Specific implementation step]\n2. [Next implementation step]\n...\n\n${ee}\n\n- \`[filePath1]\`\n- \`[filePath2]\`\n- ...\n\n${oe}\n\n[commit message]\n\`\`\`\n`}function ae(e,t){const n=["--yes-always","--no-check-update","--no-gitignore","--no-show-model-warnings","--no-show-release-notes",...G(e.aiderExtraArgs||x),"--message",t.prompt];return e.dryRun&&n.push("--dry-run"),t.resolutionPlan&&"filePaths"in t.resolutionPlan&&n.push(...t.resolutionPlan.filePaths),n}const ce=process.env.CI||process.env.GITHUB_ACTIONS;function le(e,t){const n=["--yes","@anthropic-ai/claude-code@latest",...G(e.claudeCodeExtraArgs||v),"--dangerously-skip-permissions"];return ce&&n.push("--print"),n.push(t.prompt),n}function ue(e,t){return["--yes","@openai/codex@latest","exec",...G(e.codexExtraArgs||E),t.prompt]}function de(e,t){return["--yes","@google/gemini-cli@latest","--yolo",...G(e.geminiExtraArgs||""),"--prompt",t.prompt]}async function fe(e,t,n){const i="aider"===e.codingTool?"Aider":"claude-code"===e.codingTool?"Claude Code":"codex-cli"===e.codingTool?"Codex CLI":"Gemini CLI";let s;if("aider"===e.codingTool){const i=ae(e,{prompt:t,resolutionPlan:n});console.info(o.cyan(`Asking Aider to fix "${e.testCommand}"...`)),s=(await N("aider",i,{env:{...process.env,NO_COLOR:"1"},ignoreExitStatus:!0})).stdout}else if("claude-code"===e.codingTool){const n=le(e,{prompt:t});console.info(o.cyan(`Asking Claude Code to fix "${e.testCommand}"...`)),s=(await N(e.nodeRuntime,n,{env:{...process.env,NO_COLOR:"1"},stdio:"inherit",ignoreExitStatus:!0})).stdout}else if("codex-cli"===e.codingTool){const n=ue(e,{prompt:t});console.info(o.cyan(`Asking Codex to fix "${e.testCommand}"...`)),s=(await N(e.nodeRuntime,n,{env:{...process.env,NO_COLOR:"1"},ignoreExitStatus:!0})).stdout}else{const n=de(e,{prompt:t});console.info(o.cyan(`Asking Gemini CLI to fix "${e.testCommand}"...`)),s=(await N(e.nodeRuntime,n,{env:{...process.env,NO_COLOR:"1"},ignoreExitStatus:!0})).stdout}return`\n\n## ${i} fix attempt for "${e.testCommand}"\n\n${s.trim()}`}async function pe(t){i(),!process.env.AWS_REGION&&process.env.AWS_REGION_NAME?process.env.AWS_REGION=process.env.AWS_REGION_NAME:process.env.AWS_REGION&&!process.env.AWS_REGION_NAME&&(process.env.AWS_REGION_NAME=process.env.AWS_REGION),!process.env.GEMINI_API_KEY&&process.env.GOOGLE_GENERATIVE_AI_API_KEY?process.env.GEMINI_API_KEY=process.env.GOOGLE_GENERATIVE_AI_API_KEY:process.env.GEMINI_API_KEY&&!process.env.GOOGLE_GENERATIVE_AI_API_KEY&&(process.env.GOOGLE_GENERATIVE_AI_API_KEY=process.env.GEMINI_API_KEY),t.dryRun?console.info(o.yellow("Running in dry-run mode. No branches or PRs will be created.")):await async function(){const e=(await N("git",["config","user.name"],{ignoreExitStatus:!0})).stdout.trim(),t=(await N("git",["config","user.email"],{ignoreExitStatus:!0})).stdout.trim();if(!e||!t){const e=(await N("gh",["api","user"],{ignoreExitStatus:!0})).stdout.trim();try{const{name:t,email:n}=JSON.parse(e);t&&await N("git",["config","user.name",t]),n&&await N("git",["config","user.email",n])}catch{}}}(),"aider"===t.codingTool&&(await N("python",["-m","pip","install","aider-install"]),await me(),await N("uv",["tool","uninstall","aider-chat"],{ignoreExitStatus:!0}),await N("aider-install",[]),await me(),t.aiderExtraArgs?.includes("bedrock/")&&await N("uv",["tool","run","--from","aider-chat","pip","install","--upgrade","--upgrade-strategy","only-if-needed","boto3"]));const r=await async function(e){const{stdout:t}=await N("gh",["pr","view",e.issueNumber.toString(),"--json","headRefName"],{ignoreExitStatus:!0});try{if(t)return JSON.parse(t).headRefName}catch{}}(t),a=!!r,c=r||await async function(){return(await N("git",["branch","--show-current"])).stdout.trim()}(),l=await async function(e){const t=new Set,n=await K(e.issueNumber,t,e,!1);if(!n)throw new Error(`Failed to fetch issue data for issue #${e.issueNumber}`);return n}(t),u=n.stringify(l,J).trim(),d=t.planningModel&&await async function(t,o,i,s,r,a=!1){const c=z(o,"~"),l=`${c}yaml\n${n.stringify(o,J).trim()}\n${c}`,u=["--yes","repomix@latest","--output",Z];u.push(...G(r||R)),await N("npx",u);const d=e.readFileSync(Z,"utf8");if(e.promises.rm(Z,{force:!0}),i){console.info(`Selecting files with ${t} (reasoning effort: ${s}) ...`);const n=await Q(t,[{role:"system",content:ie(l,a).trim()},{role:"user",content:d}],s);console.info("Selecting complete!");const o=j(U(n),[ee,te]);if(!o)return{filePaths:[]};const[i,r]=o.map(e=>[...e.matchAll(/\B-\s*`?([^`\n]+)`?/g)].map(e=>e[1]?.trim()??"").filter(Boolean)),c=[...i,...r].map(t=>{const n=e.existsSync(t)?e.readFileSync(t,"utf8").trim():"",o=z(n,"~");return`## \`${t}\`\n\n${o}\n${n}\n${o}`}).join("\n\n");console.info(`Planning code changes with ${t} (reasoning effort: ${s}) ...`);const u=await Q(t,[{role:"system",content:se(l,a)},{role:"user",content:c}],s);console.info("Planning complete!");const f=j(U(u),[oe,ne]);if(!f)return{filePaths:i};const[p,m]=f;return{plan:m,commitMessage:p?.trim(),filePaths:i}}console.info(`Planning code changes with ${t} (reasoning effort: ${s}) ...`);const f=await Q(t,[{role:"system",content:re(l,a).trim()},{role:"user",content:d}],s);console.info("Planning complete!");const p=j(U(f),[oe,ne,ee]);if(!p)return{filePaths:[]};const[m,g,h]=p,w=[...(h??"").matchAll(/\B-\s*`?([^`\n]+)`?/g)].map(e=>e[1]?.trim()??"").filter(Boolean);return{plan:g,commitMessage:m?.trim(),filePaths:w}}(t.planningModel,u,t.twoStagePlanning,t.reasoningEffort,t.repomixExtraArgs,a)||void 0;console.info("Resolution plan:",d);const f=d&&"plan"in d&&d.plan||"",p=z(u,"~"),m=`\nModify the code to resolve ${a?"the comments on the following GitHub pull request":"the following GitHub issue"}${f?" based on the plan":""}.${"aider"!==t.codingTool?" After that, commit your changes with a message, following the Conventional Commits specification.":""}\n\n## ${a?"Pull Request":"Issue"}\n\n${p}yml\n${u}\n${p}\n\n${f&&`## Plan\n\n${f}`}\n`.trim(),g=new Date,h=`gen-pr-${t.issueNumber}-${t.codingTool}-${g.getFullYear()}_${ge(g.getMonth()+1)}${ge(g.getDate())}_${ge(g.getHours())}${ge(g.getMinutes())}${ge(g.getSeconds())}`;t.noBranch?t.dryRun?console.info(o.yellow(`Would commit directly to base branch: ${c}`)):(a&&await N("git",["fetch","origin",c]),await N("git",["switch",c])):t.dryRun?console.info(o.yellow(`Would create branch: ${h}`)):(a&&(await N("git",["fetch","origin",c]),await N("git",["switch",c])),await N("git",["switch","--force-create",h]));let w,y="",$="",b=!0;const x="aider"===t.codingTool?"Aider":"claude-code"===t.codingTool?"Claude Code":"codex-cli"===t.codingTool?"Codex CLI":"Gemini CLI";let v,E,A={env:{...process.env,NO_COLOR:"1"},ignoreExitStatus:!0};if("aider"===t.codingTool?(v=ae(t,{prompt:m,resolutionPlan:d}),E="aider"):"claude-code"===t.codingTool?(v=le(t,{prompt:m}),E=t.nodeRuntime,A={...A,stdio:"inherit"}):"codex-cli"===t.codingTool?(v=ue(t,{prompt:m}),E=t.nodeRuntime):(v=de(t,{prompt:m}),E=t.nodeRuntime),w=function(e,t,n){const o=t.map(e=>e===n?"...":e.includes(" ")||e.includes('"')||e.includes("'")?`"${e.replace(/"/g,'\\"')}"`:e);return`${e} ${o.join(" ")}`}(E,v,m),t.dryRun)console.info(`\n=== DRY MODE: ${x} Prompt ===`),console.info(m),console.info(`=== End ${x} Prompt ===\n`),console.info(o.yellow(`Would run: ${w}`)),y="Skipped due to dry-run mode";else{const e=await N(E,v,A);y=e.stdout||"",0!==e.status&&(b=!1,$=`${x} failed with exit code ${e.status}\n${e.stderr}`,console.error(o.red(`${x} execution failed: ${$}`)))}let P=(y||"").trim(),_=!0,I="";if(t.dryRun)console.info(o.yellow("Would run test command"));else{const e=await async function(e,t){const[n,...i]=G(e.testCommand||"");if(!n)return{fixResult:"",success:!0};const s=e.maxTestAttempts;let r=0,a="",c=!1,l="";for(;r<s;){r++,console.info(o.cyan(`Executing test command (attempt ${r}/${s}): ${e.testCommand}`));const u=await T(n,i,{cwd:process.cwd()});if(0===u.status){console.info(o.green("Test command passed successfully.")),c=!0;break}if(console.warn(o.yellow(`Test command failed with exit code ${u.status}.`)),l=`Test command failed with exit code ${u.status}\n\nStdout:\n${u.stdout}\n\nStderr:\n${u.stderr}`,r>=s){console.warn(o.yellow(`Maximum fix attempts (${s}) reached. Giving up.`));break}const d=z(u.stdout,"~"),f=z(u.stderr,"~"),p=`\nThe previous changes were applied, but the test command \`${e.testCommand}\` failed.\n\nExit code: ${u.status}\n\nStdout:\n${d}\n${u.stdout}\n${d}\n\nStderr:\n${f}\n${u.stderr}\n${f}\n\nPlease analyze the output and fix the errors.\n`.trim();a+=await fe(e,p,t)}return{fixResult:a,success:c,error:c?void 0:l}}(t,d);P+=e.fixResult,_=e.success,I=e.error||"",_||console.warn(o.yellow("Tests failed after all fix attempts. Will create a draft PR."))}b||console.warn(o.yellow(`${x} execution failed. Will create a draft PR.`));const O=!b||!_;let k="";O&&($&&(k+=`\n\n### ❌ ${x} Execution Error\n\n\`\`\`\n${$}\n\`\`\``),I&&(k+=`\n\n### ❌ Test Execution Error\n\n\`\`\`\n${I}\n\`\`\``));const M=d?.commitMessage||`fix: Close #${t.issueNumber}`;if(await N("git",["add","-A"],{ignoreExitStatus:!0}),0!==(await N("git",["commit","-m",M],{ignoreExitStatus:!0})).status&&await N("git",["commit","-m",M,"--no-verify"],{ignoreExitStatus:!0}),t.noBranch?t.dryRun?console.info(o.yellow(`Would push changes directly to base branch: ${c}`)):await N("git",["push","origin",c,"--no-verify"]):t.dryRun?console.info(o.yellow(`Would push branch: ${h} to origin`)):await N("git",["push","origin",h,"--no-verify"]),t.noBranch)t.dryRun?console.info(o.yellow("Skipping PR creation due to --no-branch option")):console.info(`Changes committed directly to base branch: ${c}`);else{const e=function(e){return s.spawnSync("git",["log",`${e}..HEAD`,"--reverse","--pretty=%s"],{encoding:"utf8",stdio:"pipe"}).stdout.trim().split("\n").find(e=>e.trim())??""}(c)||M;let n=a?"":`Close #${t.issueNumber}`;if(t.planningModel&&(n+=`\n\n${S}\n\n- **Planning Model:** ${t.planningModel}`),n+=`\n- **Coding Tool:** ${x}\n- **Coding Command:** \`${w}\``,f){const e=z(f,"~");n+=`\n\n### Plan\n\n${e}\n${C(f,P.length/(f.length+P.length)*3e4)}\n${e}`}if(P){const e=z(P,"~");n+=`\n\n### ${x} Log\n\n${e}\n${C(P,P.length/(f.length+P.length)*3e4)}\n${e}`}n=n.replaceAll(/(?:\s*\n){2,}/g,"\n\n").trim(),O&&(n+=k),t.dryRun?console.info(o.yellow(`Would create PR with title: ${e}${O?" (as draft)":""}`)):await async function(e){const{owner:t,repo:n}=await q();await L.pulls.create({owner:t,repo:n,title:e.title,body:e.body,head:e.head,base:e.base,draft:e.draft})}({title:e,body:n,head:h,base:c,draft:O})}console.info(`\n${a?"Pull request":"Issue"} #${t.issueNumber} processed successfully.`)}async function me(){try{await N("asdf",["reshim"],{ignoreExitStatus:!0})}catch{}}function ge(e){return String(e).padStart(2,"0")}export{_ as D,I as a,P as b,x as c,v as d,E as e,A as f,R as g,b as l,pe as m,k as n};
2
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"main-DA3u-sLa.js","sources":["../src/config.ts","../src/defaultOptions.ts","../src/text.ts","../src/spawn.ts","../src/octokit.ts","../src/issue.ts","../src/markdown.ts","../src/llmv4.ts","../src/yaml.ts","../src/llm.ts","../src/plan.ts","../src/tools/aider.ts","../src/ci.ts","../src/tools/claudeCode.ts","../src/tools/codex.ts","../src/tools/gemini.ts","../src/test.ts","../src/main.ts","../src/env.ts","../src/git.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport YAML from 'yaml';\n\nexport function loadConfigFile(): Record<string, unknown> {\n  // Load config file (YAML) from repository root to set default option values\n  let configOptions: Record<string, unknown> = {};\n  for (const name of ['gen-pr.config.yml', 'gen-pr.config.yaml']) {\n    const cfgPath = path.resolve(process.cwd(), name);\n    if (fs.existsSync(cfgPath)) {\n      try {\n        configOptions = YAML.parse(fs.readFileSync(cfgPath, 'utf8')) as Record<string, unknown>;\n        console.info(`Loaded gen-pr config from ${name}`);\n      } catch (err) {\n        console.error(`Failed to parse config file ${name}:`, err);\n        process.exit(1);\n      }\n      break;\n    }\n  }\n  return configOptions;\n}\n","import type { NodeRuntimeActual } from './types.js';\n\n/** \"--yes-always --no-check-update --no-gitignore --no-show-model-warnings --no-show-release-notes\" is always applied */\nexport const DEFAULT_AIDER_EXTRA_ARGS = '--model gemini/gemini-2.5-pro --edit-format diff-fenced';\n/** \"--dangerously-skip-permissions --print\" is always applied */\nexport const DEFAULT_CLAUDE_CODE_EXTRA_ARGS = '--allowedTools Bash Edit Write';\n/** exec is always applied */\nexport const DEFAULT_CODEX_EXTRA_ARGS = '--full-auto';\n/** \"--yolo\" is always applied */\nexport const DEFAULT_GEMINI_EXTRA_ARGS = '';\nexport const DEFAULT_REPOMIX_EXTRA_ARGS = '--compress --remove-empty-lines --include \"src/**/*.{ts,tsx},**/*.md\"';\nexport const DEFAULT_MAX_TEST_ATTEMPTS = 5;\nexport const DEFAULT_CODING_TOOL = 'aider';\nexport const DEFAULT_NODE_RUNTIME: NodeRuntimeActual = 'npx';\n","export const HEADING_OF_GEN_PR_METADATA = '## gen-pr Metadata';\n\n/**\n * Truncate long text.\n */\nexport function truncateText(text: string, maxLength: number): string {\n  if (text.length > maxLength) {\n    const truncated = text.slice(0, maxLength);\n    const omitted = text.length - maxLength;\n    return `${truncated}\\n\\n... (${Math.floor(omitted)} characters truncated) ...`;\n  }\n\n  return text;\n}\n\n/**\n * Removes HTML-style comments from a string.\n *\n * @param markdownContent The string containing markdown content\n * @returns The string with HTML comments removed\n */\nexport function stripHtmlComments(markdownContent: string): string {\n  return markdownContent.replace(/<!--[\\s\\S]*?-->/g, '');\n}\n\n/**\n * Removes gen-pr metadata sections from a PR body markdown.\n */\nexport function stripMetadataSections(markdownContent: string): string {\n  const index = markdownContent.indexOf(HEADING_OF_GEN_PR_METADATA);\n  return index >= 0 ? markdownContent.substring(0, index) : markdownContent;\n}\n\nexport function normalizeNewLines(text: string): string {\n  return text.replaceAll('\\r\\n', '\\n').trim();\n}\n\n/**\n * Removes text matching the specified regex pattern.\n *\n * @param text The input text\n * @param pattern The regex pattern string to remove\n * @returns The text with matched patterns removed\n */\nexport function removeRegexPattern(text: string, pattern: string): string {\n  if (!pattern) return text;\n  try {\n    const regex = new RegExp(pattern, 'g');\n    return text.replace(regex, '');\n  } catch (error) {\n    console.warn(`Invalid regex pattern \"${pattern}\":`, error);\n    return text;\n  }\n}\n","import type { SpawnOptions, SpawnSyncReturns } from 'node:child_process';\nimport { spawn } from 'node:child_process';\nimport ansis from 'ansis';\nimport { truncateText } from './text.js';\nimport type { NodeRuntime, NodeRuntimeActual } from './types.js';\n\nconst MAX_LOG_LENGTH = 3000;\n\n/**\n * Normalizes node runtime values to their actual executable commands.\n * 'node' is an alias for 'npx' and 'bun' is an alias for 'bunx'.\n *\n * @param runtime The runtime value to normalize\n * @returns The normalized runtime command\n */\nexport function normalizeNodeRuntime(runtime: NodeRuntime): NodeRuntimeActual {\n  switch (runtime) {\n    case 'node':\n    case 'npx':\n      return 'npx';\n    case 'bun':\n    case 'bunx':\n      return 'bunx';\n  }\n}\n\nexport async function runCommand(\n  command: string,\n  args: string[],\n  options?: SpawnOptions & { ignoreExitStatus?: boolean; truncateStdout?: boolean }\n): Promise<Omit<SpawnSyncReturns<string>, 'output' | 'error'>> {\n  const { ignoreExitStatus, ...spawnOptions } = options ?? {};\n  const argsText = args.map((a) => (a.includes(' ') ? `\"${a.replaceAll('\"', '\"')}\"` : a)).join(' ');\n  console.info(ansis.green(`$ ${command} ${argsText}`));\n\n  console.info('stdout: ---------------------');\n  const ret = await spawnAsync(command, args, spawnOptions);\n  if (spawnOptions.truncateStdout) console.info(truncateText(ret.stdout, MAX_LOG_LENGTH));\n  const stderr = ret.stderr.trim();\n  if (stderr) {\n    console.info('stderr: ---------------------');\n    const truncatedStderr = truncateText(stderr, MAX_LOG_LENGTH);\n    console.info(ansis.yellow(truncatedStderr));\n  }\n  console.info('-----------------------------');\n  console.info(ansis.magenta(`Exit code: ${ret.status}\\n`));\n  if (!ignoreExitStatus && ret.status !== 0 && ret.status !== null) {\n    process.exit(ret.status);\n  }\n  return ret;\n}\n\nexport async function spawnAsync(\n  command: string,\n  args: readonly string[],\n  options: SpawnOptions & { truncateStdout?: boolean }\n): Promise<Omit<SpawnSyncReturns<string>, 'output' | 'error'>> {\n  return new Promise((resolve, reject) => {\n    try {\n      // Sanitize args to remove null bytes\n      const sanitizedArgs = (args ?? []).map((arg) => arg.replace(/\\0/g, ''));\n      const proc = spawn(command, sanitizedArgs, options);\n      // `setEncoding` is undefined in Bun\n      proc.stdout?.setEncoding?.('utf8');\n      proc.stderr?.setEncoding?.('utf8');\n\n      let stdout = '';\n      let stderr = '';\n      proc.stdout?.on('data', (data) => {\n        if (!options?.truncateStdout) process.stdout.write(data);\n        stdout += data;\n      });\n      proc.stderr?.on('data', (data) => {\n        stderr += data;\n      });\n\n      proc.on('error', (error) => {\n        reject(error);\n      });\n      proc.on('close', (code: number | null, signal: NodeJS.Signals | null) => {\n        if (proc.pid === undefined) {\n          reject(new Error('Process has no pid.'));\n        } else {\n          resolve({\n            pid: proc.pid,\n            stdout,\n            stderr,\n            status: code,\n            signal,\n          });\n        }\n      });\n    } catch (error) {\n      // eslint-disable-next-line @typescript-eslint/prefer-promise-reject-errors\n      reject(error);\n    }\n  });\n}\n\n/**\n * Parses a command line string into an array of arguments, preserving quoted strings.\n *\n * This function handles:\n * - Space-separated arguments\n * - Double-quoted strings (preserves spaces within)\n * - Single-quoted strings (preserves spaces within)\n *\n * @param argsString The command line string to parse\n * @returns An array of parsed arguments\n */\nexport function parseCommandLineArgs(argsString: string): string[] {\n  if (!argsString) return [];\n\n  const result: string[] = [];\n  let current = '';\n  let inDoubleQuote = false;\n  let inSingleQuote = false;\n\n  for (let i = 0; i < argsString.length; i++) {\n    const char = argsString[i];\n\n    // Handle quotes\n    if (char === '\"' && !inSingleQuote) {\n      inDoubleQuote = !inDoubleQuote;\n      continue;\n    }\n\n    if (char === \"'\" && !inDoubleQuote) {\n      inSingleQuote = !inSingleQuote;\n      continue;\n    }\n\n    // Handle spaces (only split on spaces outside of quotes)\n    if (char === ' ' && !inDoubleQuote && !inSingleQuote) {\n      if (current) {\n        result.push(current);\n        current = '';\n      }\n      continue;\n    }\n\n    // Add character to current argument\n    current += char;\n  }\n\n  // Add the last argument if there is one\n  if (current) {\n    result.push(current);\n  }\n\n  return result;\n}\n","import child_process from 'node:child_process';\nimport { graphql } from '@octokit/graphql';\nimport type { RestEndpointMethodTypes } from '@octokit/rest';\nimport { Octokit } from '@octokit/rest';\nimport { runCommand } from './spawn.js';\nimport type {\n  LabelInfo,\n  PullRequestParams,\n  PullRequestReview,\n  PullRequestReviewThreadsResponse,\n  RepositoryInfo,\n  SimpleComment,\n} from './types.js';\n\nconst token =\n  process.env.GH_TOKEN ||\n  process.env.GITHUB_TOKEN ||\n  child_process.spawnSync('gh', ['auth', 'token'], { encoding: 'utf-8' }).stdout.trim();\nif (!token) {\n  throw new Error(\n    'GitHub token not found. Please set GH_TOKEN or GITHUB_TOKEN environment variable, or authenticate with gh CLI'\n  );\n}\nconst octokit = new Octokit({\n  auth: token,\n});\nconst graphqlClient = graphql.defaults({\n  headers: {\n    authorization: `token ${token}`,\n  },\n});\n\nlet repoOwner: string | undefined;\nlet repoName: string | undefined;\nasync function getRepoInfo(): Promise<RepositoryInfo> {\n  if (!repoOwner || !repoName) {\n    // Get repo info from git remote\n    const { stdout } = await runCommand('git', ['remote', 'get-url', 'origin'], { ignoreExitStatus: true });\n    const remoteUrl = stdout.trim();\n\n    // Parse GitHub repo URL (supports both https and ssh formats)\n    const match = remoteUrl.match(/github\\.com[:/]([^/]+)\\/(.+)/);\n    if (match?.[1] && match[2]) {\n      repoOwner = match[1];\n      repoName = match[2].replace(/\\.git$/, '');\n    } else {\n      throw new Error('Could not parse GitHub repository from remote URL');\n    }\n  }\n\n  if (!repoOwner || !repoName) {\n    throw new Error('Repository information not properly initialized');\n  }\n  return { owner: repoOwner, repo: repoName };\n}\n\nexport async function createPullRequest(params: PullRequestParams): Promise<void> {\n  const { owner, repo } = await getRepoInfo();\n\n  await octokit.pulls.create({\n    owner,\n    repo,\n    title: params.title,\n    body: params.body,\n    head: params.head,\n    base: params.base,\n    draft: params.draft,\n  });\n}\n\nexport async function getPullRequestDiff(pullNumber: number): Promise<string> {\n  const { owner, repo } = await getRepoInfo();\n\n  const response = await octokit.pulls.get({\n    owner,\n    repo,\n    pull_number: pullNumber,\n    mediaType: {\n      format: 'diff',\n    },\n  });\n\n  return response.data as unknown as string;\n}\n\nexport async function getIssue(issueNumber: number): Promise<{\n  author: string;\n  title: string;\n  body: string;\n  labels: LabelInfo[];\n  comments: SimpleComment[];\n  url: string;\n}> {\n  const { owner, repo } = await getRepoInfo();\n\n  // Get issue data first - this works for both issues and PRs\n  const { data: issueData } = await octokit.issues.get({\n    owner,\n    repo,\n    issue_number: issueNumber,\n  });\n  // If it has pull_request field, get the full PR data\n  let issueOrPullRequest:\n    | RestEndpointMethodTypes['pulls']['get']['response']['data']\n    | RestEndpointMethodTypes['issues']['get']['response']['data'] = issueData;\n  if (issueData.pull_request) {\n    const { data: prData } = await octokit.pulls.get({\n      owner,\n      repo,\n      pull_number: issueNumber,\n    });\n    issueOrPullRequest = prData;\n  }\n\n  const commentsResponse = await octokit.issues.listComments({\n    owner,\n    repo,\n    issue_number: issueNumber,\n  });\n\n  return {\n    author: issueOrPullRequest.user?.login || '',\n    title: issueOrPullRequest.title,\n    body: issueOrPullRequest.body || '',\n    labels: issueOrPullRequest.labels.map((label: string | { name?: string }) => ({\n      name: typeof label === 'string' ? label : label.name || '',\n    })),\n    comments: commentsResponse.data.map((comment) => ({\n      author: comment.user?.login || '',\n      body: comment.body || '',\n      createdAt: comment.created_at,\n    })),\n    url: issueOrPullRequest.html_url,\n  };\n}\n\nexport async function getPullRequestReviewThreads(pullNumber: number): Promise<PullRequestReviewThreadsResponse> {\n  const { owner, repo } = await getRepoInfo();\n\n  const MAX_MESSAGE_COUNT = 100;\n  const query = `\n    query($owner: String!, $repo: String!, $pr: Int!) {\n      repository(owner: $owner, name: $repo) {\n        pullRequest(number: $pr) {\n          reviewThreads(first: ${MAX_MESSAGE_COUNT}) {\n            nodes {\n              isResolved\n              comments(first: ${MAX_MESSAGE_COUNT}) {\n                nodes {\n                  author {\n                    login\n                  }\n                  body\n                  path\n                  line\n                  diffHunk\n                  createdAt\n                }\n              }\n            }\n          }\n        }\n      }\n    }\n  `;\n\n  const result = await graphqlClient(query, {\n    owner,\n    repo,\n    pr: pullNumber,\n  });\n\n  return result as PullRequestReviewThreadsResponse;\n}\n\nexport async function getPullRequestReviews(pullNumber: number): Promise<PullRequestReview[]> {\n  const { owner, repo } = await getRepoInfo();\n\n  const response = await octokit.pulls.listReviews({\n    owner,\n    repo,\n    pull_number: pullNumber,\n  });\n\n  return response.data.map((review) => ({\n    user: { login: review.user?.login || '' },\n    state: review.state,\n    body: review.body || '',\n    submitted_at: review.submitted_at || '',\n  }));\n}\n","import type { MainOptions } from './main.js';\nimport { getIssue, getPullRequestDiff, getPullRequestReviews, getPullRequestReviewThreads } from './octokit.js';\nimport { normalizeNewLines, removeRegexPattern, stripHtmlComments, stripMetadataSections } from './text.js';\nimport type { IssueComment, IssueInfo, UserLogin } from './types.js';\n\n// Temporary interface for sorting comments with date information\ninterface IssueCommentWithDate extends IssueComment {\n  createdAt: number;\n}\n\nexport async function createIssueInfo(options: MainOptions): Promise<IssueInfo> {\n  const processedIssues = new Set<number>();\n  const issueInfo = await fetchIssueData(options.issueNumber, processedIssues, options, false);\n  if (!issueInfo) {\n    throw new Error(`Failed to fetch issue data for issue #${options.issueNumber}`);\n  }\n  return issueInfo;\n}\n\nasync function fetchPRDiff(issueNumber: number, issueInfo: IssueInfo): Promise<void> {\n  try {\n    const prDiff = await getPullRequestDiff(issueNumber);\n    if (!prDiff.trim()) return;\n    issueInfo.code_changes = processDiffContent(prDiff.trim());\n  } catch (error) {\n    console.warn('Failed to fetch PR diff:', error);\n  }\n}\n\nasync function fetchIssueData(\n  issueNumber: number,\n  processedIssues: Set<number>,\n  options: MainOptions,\n  isReferenced = false\n): Promise<IssueInfo | undefined> {\n  if (processedIssues.has(issueNumber)) {\n    return;\n  }\n  processedIssues.add(issueNumber);\n\n  let issue: Awaited<ReturnType<typeof getIssue>>;\n  try {\n    issue = await getIssue(issueNumber);\n  } catch (error) {\n    console.warn(`Failed to fetch issue #${issueNumber}:`, error);\n    return;\n  }\n\n  // Extract issue/PR references from the issue body and comments\n  const allText = [issue.body, ...issue.comments.map((c) => c.body)].join('\\n');\n  const referencedNumbers = extractIssueReferences(allText);\n\n  const rawBody = stripHtmlComments(issue.body);\n  const processedBody = issue.url?.includes('/pull/') ? stripMetadataSections(rawBody) : rawBody;\n  const description = removeRegexPattern(processedBody, options.removePattern || '');\n  const commentsWithDate: IssueCommentWithDate[] = issue.comments.map((c) => ({\n    author: c.author,\n    body: normalizeNewLines(c.body),\n    createdAt: new Date(c.createdAt).getTime(),\n  }));\n\n  const issueInfo: IssueInfo = {\n    author: issue.author,\n    title: issue.title,\n    description: normalizeNewLines(description),\n    comments: [], // Will be populated after sorting\n  };\n\n  if (issue.url?.includes('/pull/') && !isReferenced) {\n    await fetchPRDiff(issueNumber, issueInfo);\n    await fetchPRReviewThreads(issueNumber, commentsWithDate);\n    await fetchPRReviews(issueNumber, commentsWithDate);\n  }\n\n  if (referencedNumbers.length > 0) {\n    await fetchReferencedIssues(referencedNumbers, processedIssues, options, issueInfo);\n  }\n\n  // Sort comments by creation date (oldest first) and remove createdAt field\n  issueInfo.comments = commentsWithDate\n    .filter((c) => c.body)\n    .sort((a, b) => a.createdAt - b.createdAt)\n    .map(({ createdAt, ...comment }) => comment);\n\n  return issueInfo;\n}\n\nasync function fetchPRReviewThreads(issueNumber: number, commentsWithDate: IssueCommentWithDate[]): Promise<void> {\n  try {\n    const graphqlData = await getPullRequestReviewThreads(issueNumber);\n    const reviewThreads = graphqlData?.repository?.pullRequest?.reviewThreads?.nodes || [];\n\n    for (const thread of reviewThreads) {\n      if (!thread.isResolved && thread.comments?.nodes) {\n        processReviewThreadComments(thread.comments.nodes, commentsWithDate);\n      }\n    }\n  } catch (error) {\n    console.warn('Failed to fetch PR review threads:', error);\n  }\n}\n\nfunction processReviewThreadComments(\n  comments: Array<{\n    author?: UserLogin;\n    body?: string;\n    path?: string;\n    line?: number;\n    diffHunk?: string;\n    createdAt: string;\n  }>,\n  commentsWithDate: IssueCommentWithDate[]\n): void {\n  for (const comment of comments) {\n    if (!comment.author || !comment.body) continue;\n\n    const codeContent = extractCodeFromDiffHunk(comment.diffHunk);\n    const reviewComment: IssueCommentWithDate = {\n      author: comment.author.login,\n      codeLocation: comment.path && comment.line ? `${comment.path}:${comment.line}` : undefined,\n      codeContent: codeContent || undefined,\n      body: normalizeNewLines(comment.body),\n      createdAt: new Date(comment.createdAt).getTime(),\n    };\n\n    // Remove undefined properties\n    Object.keys(reviewComment).forEach((key) => {\n      if (reviewComment[key as keyof IssueCommentWithDate] === undefined) {\n        delete reviewComment[key as keyof IssueCommentWithDate];\n      }\n    });\n\n    commentsWithDate.push(reviewComment);\n  }\n}\n\nfunction extractCodeFromDiffHunk(diffHunk: string | undefined): string {\n  if (!diffHunk) return '';\n\n  const lines = diffHunk.split('\\n');\n  const codeLine = lines.find(\n    (line: string) => (line.startsWith('+') || line.startsWith('-')) && !line.startsWith('@@') && line.trim().length > 1\n  );\n  return codeLine?.trim() || '';\n}\n\nasync function fetchPRReviews(issueNumber: number, commentsWithDate: IssueCommentWithDate[]): Promise<void> {\n  try {\n    const reviews = await getPullRequestReviews(issueNumber);\n    const reviewResultComments: IssueCommentWithDate[] = reviews.map((review) => ({\n      author: review.user.login,\n      reviewState: review.state,\n      body: normalizeNewLines(review.body),\n      createdAt: new Date(review.submitted_at).getTime(),\n    }));\n    commentsWithDate.push(...reviewResultComments);\n  } catch (error) {\n    console.warn('Failed to fetch PR reviews:', error);\n  }\n}\n\nfunction extractIssueReferences(text: string): number[] {\n  const regex = /(?:^|\\s)#(\\d+)/g;\n  const numbers: number[] = [];\n  for (;;) {\n    const match = regex.exec(text);\n    if (!match) break;\n\n    const number = Number.parseInt(match[1] ?? '', 10);\n    if (Number.isInteger(number)) {\n      numbers.push(number);\n    }\n  }\n  return [...new Set(numbers)]; // Remove duplicates\n}\n\nasync function fetchReferencedIssues(\n  referencedNumbers: number[],\n  processedIssues: Set<number>,\n  options: MainOptions,\n  issueInfo: IssueInfo\n): Promise<void> {\n  const referencedIssuesPromises = referencedNumbers.map((num) => fetchIssueData(num, processedIssues, options, true));\n  const referencedIssues = (await Promise.all(referencedIssuesPromises)).filter((issue): issue is IssueInfo => !!issue);\n\n  if (referencedIssues.length === 0) return;\n\n  issueInfo.referenced_issues = referencedIssues;\n}\n\n/**\n * Process diff content to handle large diffs by truncating or omitting large fragments\n */\nfunction processDiffContent(diffContent: string): string {\n  const MAX_TOTAL_DIFF_SIZE = 50000;\n  const MAX_FILE_DIFF_SIZE = 10000;\n  const LARGE_FILE_PATTERNS = [\n    /^diff --git a\\/dist\\//m,\n    /^diff --git a\\/build\\//m,\n    /^diff --git a\\/.*\\.bundle\\./m,\n    /^diff --git a\\/.*\\.min\\./m,\n    /^diff --git a\\/node_modules\\//m,\n  ];\n\n  // If the entire diff is small enough, return as-is\n  if (diffContent.length <= MAX_TOTAL_DIFF_SIZE) {\n    return diffContent;\n  }\n\n  // Split diff into individual file sections\n  const fileSections = diffContent.split(/(?=^diff --git)/m);\n  const processedSections: string[] = [];\n  let totalSize = 0;\n\n  for (const section of fileSections) {\n    if (!section.trim()) continue;\n\n    const isLargeFile = LARGE_FILE_PATTERNS.some((pattern) => pattern.test(section));\n\n    if (isLargeFile) {\n      // For large/bundled files, include only the header and a truncation notice\n      const lines = section.split('\\n');\n      const headerLines = lines.slice(0, 4); // diff --git, index, ---, +++\n      const truncatedSection = [\n        ...headerLines,\n        '@@ ... @@',\n        '... (large bundled/compiled file diff truncated) ...',\n        '',\n      ].join('\\n');\n\n      processedSections.push(truncatedSection);\n      totalSize += truncatedSection.length;\n    } else if (section.length > MAX_FILE_DIFF_SIZE) {\n      // For other large files, truncate but keep some content\n      const truncatedSection = `${section.slice(0, MAX_FILE_DIFF_SIZE)}\\n... (diff truncated) ...\\n`;\n      processedSections.push(truncatedSection);\n      totalSize += truncatedSection.length;\n    } else {\n      // Small files, include as-is\n      processedSections.push(section);\n      totalSize += section.length;\n    }\n\n    // Stop if we're approaching the total size limit\n    if (totalSize > MAX_TOTAL_DIFF_SIZE * 0.9) {\n      processedSections.push('\\n... (remaining diffs truncated) ...\\n');\n      break;\n    }\n  }\n\n  return processedSections.join('');\n}\n","/**\n * Extracts content between specified headers from a text\n * @param text The text to parse\n * @param headers Array of header strings to extract content for\n * @returns Array of content strings, or undefined if any headers are missing or in different order\n */\nexport function extractHeaderContents(text: string, headers: string[]): string[] | undefined {\n  const modifiedResponse = `\\n${text}`;\n  const indices = headers.map((header) => modifiedResponse.indexOf(`\\n${header}`));\n\n  // Return undefined if any headers are missing or not in order\n  if (\n    indices.some((index) => index === -1) ||\n    !indices.every((index, i) => i === 0 || index > (indices[i - 1] as number))\n  ) {\n    return undefined;\n  }\n\n  return headers.map((header, i) => {\n    const start = (indices[i] as number) + 1 + header.length;\n    const end = i + 1 < headers.length ? (indices[i + 1] as number) + 1 : modifiedResponse.length;\n    return modifiedResponse.slice(start, end).trim();\n  });\n}\n\nexport function findDistinctFence(content: string, fenceChar: '`' | '~'): string {\n  const regex = new RegExp(`${fenceChar}{3,}`, 'g');\n  const matches = content.match(regex);\n  const maxLength = matches ? Math.max(...matches.map((seq) => seq.length)) : 0;\n  const fenceLength = Math.max(3, maxLength + 1);\n  return fenceChar.repeat(fenceLength);\n}\n\nexport function trimCodeBlockFences(content: string): string {\n  // Remove code block fences with any number of backticks or tildes from the beginning and end\n  return content.trim().replace(/^(`{3,}|~{3,})[\\s\\S]*?\\n([\\s\\S]*?)\\n\\1\\s*$/, '$2');\n}\n","import type { ModelMessage } from 'ai';\nimport { generateText, type Message } from 'ai-v4';\nimport { createOllama } from 'ollama-ai-provider-v2';\nimport { logResult } from './llm.js';\nimport type { ReasoningEffort } from './types.js';\n\n/**\n * Call AI SDK v4 provider API (for Ollama)\n */\nexport async function callV4ProviderApi(\n  model: string,\n  messages: ModelMessage[],\n  reasoningEffort?: ReasoningEffort\n): Promise<string> {\n  const [provider, ...modelParts] = model.split('/') as [string, ...string[]];\n  try {\n    const modelName = modelParts.join('/'); // Handle cases where model name itself contains '/'\n    if (!modelName) {\n      console.error(`Invalid ${provider} model format: ${model}. Expected format: ${provider}/model-name`);\n      process.exit(1);\n    }\n\n    const ollamaBaseURL = `${process.env.OLLAMA_BASE_URL || 'http://localhost:11434'}/api`;\n    const ollama = createOllama({\n      baseURL: ollamaBaseURL,\n      ...(process.env.OLLAMA_API_KEY && { apiKey: process.env.OLLAMA_API_KEY }),\n    });\n    const providerModel = ollama(modelName);\n\n    const result = await generateText({\n      model: providerModel,\n      providerOptions: reasoningEffort ? { ollama: { think: true } } : undefined,\n      messages: convertToV4Messages(messages),\n    });\n    logResult(model, result);\n    return result.text;\n  } catch (error) {\n    console.error(`${provider.charAt(0).toUpperCase() + provider.slice(1)} API error for model ${model}:`, error);\n    process.exit(1);\n  }\n}\n\n/**\n * Convert AI SDK v5 ModelMessage[] to AI SDK v4 Message[] format\n */\nfunction convertToV4Messages(messages: ModelMessage[]): Message[] {\n  return messages.map((msg, index) => ({\n    id: `msg-${index}`,\n    role: msg.role === 'tool' ? 'data' : msg.role,\n    content: typeof msg.content === 'string' ? msg.content : JSON.stringify(msg.content),\n  }));\n}\n","/**\n * `blockQuote: 'literal'`: Use a pipe (`prop: |`) to represent multi-line strings.\n * `lineWidth: 0`: No limit on the line width.\n */\nexport const yamlStringifyOptions = { blockQuote: 'literal', lineWidth: 0 } as const;\n","import { createAmazonBedrock } from '@ai-sdk/amazon-bedrock';\nimport { type AnthropicProviderOptions, createAnthropic } from '@ai-sdk/anthropic';\nimport { createAzure } from '@ai-sdk/azure';\nimport { createGoogleGenerativeAI, type GoogleGenerativeAIProviderOptions } from '@ai-sdk/google';\nimport { createVertex } from '@ai-sdk/google-vertex';\nimport { createOpenAI, type OpenAIResponsesProviderOptions } from '@ai-sdk/openai';\nimport type { LanguageModelV2 } from '@ai-sdk/provider';\nimport { createXai } from '@ai-sdk/xai';\nimport { createOpenRouter } from '@openrouter/ai-sdk-provider';\nimport { generateText, type ModelMessage } from 'ai';\nimport YAML from 'yaml';\nimport { callV4ProviderApi } from './llmv4.js';\nimport type { ReasoningEffort } from './types.js';\nimport { yamlStringifyOptions } from './yaml.js';\n\n/**\n * Call LLM API using AI SDK 5\n */\nexport async function callLlmApi(\n  model: string,\n  messages: ModelMessage[],\n  reasoningEffort?: ReasoningEffort\n): Promise<string> {\n  try {\n    // Special handling for Ollama using AI SDK v4\n    if (model.startsWith('ollama/')) {\n      return await callV4ProviderApi(model, messages, reasoningEffort);\n    }\n\n    const [modelInstance, provider, modelName] = getModelInstance(model, reasoningEffort);\n\n    // Build the request parameters\n    const requestParams: Parameters<typeof generateText>[0] = {\n      model: modelInstance,\n      messages: messages,\n    };\n\n    if (reasoningEffort) {\n      // Check if the model supports reasoning/thinking options\n      const modelSupportsReasoning = supportsReasoningOptions(provider, modelName);\n\n      if (!modelSupportsReasoning) {\n        console.warn(\n          `Model ${model} does not support reasoning/thinking options. Ignoring reasoning effort parameter.`\n        );\n      } else {\n        const thinkingBudget = getThinkingBudget(reasoningEffort);\n        if (provider === 'openai') {\n          requestParams.providerOptions = {\n            openai: {\n              reasoningEffort,\n            } satisfies OpenAIResponsesProviderOptions,\n          };\n        } else if (provider === 'anthropic') {\n          requestParams.providerOptions = {\n            anthropic: {\n              thinking: {\n                type: 'enabled',\n                budgetTokens: thinkingBudget,\n              },\n            } satisfies AnthropicProviderOptions,\n          };\n        } else if (provider === 'gemini') {\n          requestParams.providerOptions = {\n            google: {\n              thinkingConfig: {\n                thinkingBudget,\n              },\n            } satisfies GoogleGenerativeAIProviderOptions,\n          };\n        } else if (provider === 'bedrock') {\n          // The latest AI SDK doesn't work on Bedrock with reasoning.\n          console.warn(\n            `Note: The current AI SDK doesn't work on Bedrock with reasoning. Model ${modelName} will use default reasoning settings.`\n          );\n          // requestParams.providerOptions = {\n          //   bedrock: {\n          //     reasoningConfig: { type: 'enabled', budgetTokens: thinkingBudget },\n          //   } satisfies BedrockProviderOptions,\n          // };\n        } else if (provider === 'xai') {\n          requestParams.providerOptions = {\n            xai: {\n              reasoningEffort,\n            },\n          };\n        }\n      }\n    }\n\n    const result = await generateText(requestParams);\n    logResult(model, result);\n\n    return result.text;\n  } catch (error) {\n    console.error(`LLM API error for model ${model}:`, error);\n    process.exit(1);\n  }\n}\n\n/**\n * Log the result of an LLM API call\n */\nexport function logResult(model: string, result: { text: string; usage?: unknown; finishReason?: string }): void {\n  console.info(\n    `${model}:`,\n    YAML.stringify(\n      {\n        text: result.text,\n        usage: result.usage,\n        finishReason: result.finishReason,\n      },\n      yamlStringifyOptions\n    )\n  );\n}\n\nfunction getModelInstance(model: string, reasoningEffort?: ReasoningEffort): [LanguageModelV2, string, string] {\n  // Only support llmlite format (provider/model)\n  if (!model.includes('/')) {\n    console.error(`Model must be in format 'provider/model'. Got: ${model}`);\n    process.exit(1);\n  }\n\n  const [provider, ...modelParts] = model.split('/');\n  const modelName = modelParts.join('/'); // Handle cases where model name itself contains '/'\n\n  switch (provider) {\n    case 'openai': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/openai\n      const openaiProvider = createOpenAI();\n      return [openaiProvider(modelName), provider, modelName];\n    }\n\n    case 'anthropic': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/anthropic\n      const anthropicProvider = createAnthropic();\n      return [anthropicProvider(modelName), provider, modelName];\n    }\n\n    case 'gemini': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/google-generative-ai\n      const googleProvider = createGoogleGenerativeAI();\n      return [googleProvider(modelName), provider, modelName];\n    }\n\n    case 'azure': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/azure\n      const azureProvider = createAzure();\n      return [azureProvider(modelName), provider, modelName];\n    }\n\n    case 'bedrock': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/amazon-bedrock\n      const bedrockProvider = createAmazonBedrock();\n      return [bedrockProvider(modelName), provider, modelName];\n    }\n\n    case 'vertex': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/google-vertex\n      const vertexProvider = createVertex();\n      return [vertexProvider(modelName), provider, modelName];\n    }\n\n    case 'xai': {\n      // cf. https://ai-sdk.dev/providers/ai-sdk-providers/xai\n      const grokProvider = createXai();\n      return [grokProvider(modelName), provider, modelName];\n    }\n\n    case 'openrouter': {\n      // cf. https://github.com/OpenRouterTeam/ai-sdk-provider\n      const openrouterProvider = createOpenRouter({\n        apiKey: process.env.OPENROUTER_API_KEY,\n        headers: {\n          'HTTP-Referer': 'https://github.com/WillBooster/gen-pr',\n          'X-Title': 'gen-pr',\n        },\n      });\n      const modelOptions = reasoningEffort\n        ? {\n            reasoning: {\n              effort: reasoningEffort,\n            },\n          }\n        : {};\n      return [openrouterProvider(modelName, modelOptions), provider, modelName];\n    }\n\n    default:\n      console.error(\n        `Unsupported provider: ${provider}. Supported providers: openai, azure, google, anthropic, bedrock, vertex, grok, openrouter, ollama`\n      );\n      process.exit(1);\n  }\n}\n\n/**\n * Check if a model supports reasoning/thinking options\n */\nexport function supportsReasoningOptions(provider: string, modelName: string): boolean {\n  switch (provider) {\n    case 'openai':\n    case 'azure':\n      // OpenAI and Azure: only o1, o3, o4 series models support reasoning effort\n      return /^(o1|o3|o4)/.test(modelName);\n\n    case 'anthropic':\n      // Anthropic: only Claude 3.7 and Claude 4 models support thinking budget\n      return /^claude-(opus-4|sonnet-4|3-7-sonnet)/.test(modelName);\n\n    case 'gemini':\n      // Google: only Gemini 2.5 models support thinking budget\n      return /^gemini-2\\.5/.test(modelName);\n\n    case 'bedrock':\n      // Bedrock: only Anthropic Claude 3.7 and 4 models support reasoning\n      return /^(us\\.)?anthropic\\.claude-(opus-4|sonnet-4|3-7-sonnet)/.test(modelName);\n\n    case 'vertex':\n      // Vertex: Gemini 2.5 models and Claude 3.7/4 models support thinking budget\n      return /^gemini-2\\.5/.test(modelName) || /^claude-(3-7-sonnet|opus-4|sonnet-4)/.test(modelName);\n\n    case 'xai':\n      // Grok: Grok 3 models support reasoning effort\n      return /^grok-3/.test(modelName);\n\n    // We can always pass reasoningEffort to OpenRouter without errors.\n    case 'openrouter':\n      return true;\n\n    default:\n      return false;\n  }\n}\n\n/**\n * Get thinking budget token count based on reasoning effort level\n */\nfunction getThinkingBudget(reasoningEffort: ReasoningEffort): number {\n  const tokenBudgets = {\n    low: 4000, // 4K tokens\n    medium: 8000, // 8K tokens\n    high: 24000, // 24K tokens\n  };\n\n  return tokenBudgets[reasoningEffort];\n}\n","import fs from 'node:fs';\nimport YAML from 'yaml';\nimport { DEFAULT_REPOMIX_EXTRA_ARGS } from './defaultOptions.js';\nimport { callLlmApi } from './llm.js';\nimport { extractHeaderContents, findDistinctFence, trimCodeBlockFences } from './markdown.js';\nimport { parseCommandLineArgs, runCommand } from './spawn.js';\nimport type { ReasoningEffort } from './types.js';\nimport { yamlStringifyOptions } from './yaml.js';\n\nconst REPOMIX_FILE_NAME = 'repomix.result';\n\nexport type ResolutionPlan = {\n  plan?: string;\n  commitMessage?: string;\n  filePaths: string[];\n};\n\nconst HEADING_OF_FILE_PATHS_TO_BE_MODIFIED = '## File Paths to be Modified';\nconst HEADING_OF_FILE_PATHS_TO_BE_REFERRED = '## File Paths to be Referred';\nconst HEADING_OF_PLAN = '## Implementation Plan';\nconst HEADING_OF_COMMIT_MESSAGE = '## Commit Message';\n\nexport async function planCodeChanges(\n  model: string,\n  issueContent: string,\n  twoStagePlanning: boolean,\n  reasoningEffort?: ReasoningEffort,\n  repomixExtraArgs?: string,\n  isPullRequest = false\n): Promise<ResolutionPlan> {\n  const issueFence = findDistinctFence(issueContent, '~');\n  const issueYamlText = `${issueFence}yaml\n${YAML.stringify(issueContent, yamlStringifyOptions).trim()}\n${issueFence}`;\n\n  // Base repomix command arguments\n  const repomixArgs = ['--yes', 'repomix@latest', '--output', REPOMIX_FILE_NAME];\n  repomixArgs.push(...parseCommandLineArgs(repomixExtraArgs || DEFAULT_REPOMIX_EXTRA_ARGS));\n\n  await runCommand('npx', repomixArgs);\n  const repomixResult = fs.readFileSync(REPOMIX_FILE_NAME, 'utf8');\n  void fs.promises.rm(REPOMIX_FILE_NAME, { force: true });\n\n  if (twoStagePlanning) {\n    console.info(`Selecting files with ${model} (reasoning effort: ${reasoningEffort}) ...`);\n    const filesResponse = await callLlmApi(\n      model,\n      [\n        {\n          role: 'system',\n          content: buildPromptForSelectingFiles(issueYamlText, isPullRequest).trim(),\n        },\n        {\n          role: 'user',\n          content: repomixResult,\n        },\n      ],\n      reasoningEffort\n    );\n    console.info('Selecting complete!');\n\n    const extractedFilePathLists = extractHeaderContents(trimCodeBlockFences(filesResponse), [\n      HEADING_OF_FILE_PATHS_TO_BE_MODIFIED,\n      HEADING_OF_FILE_PATHS_TO_BE_REFERRED,\n    ]) as [string, string] | undefined;\n    if (!extractedFilePathLists) {\n      return { filePaths: [] };\n    }\n    const [filePathsToBeModified, filePathsToBeReferred] = extractedFilePathLists.map((filesContent: string) => {\n      const filePathRegex = /\\B-\\s*`?([^`\\n]+)`?/g;\n      const matches = [...filesContent.matchAll(filePathRegex)];\n      return matches.map((match) => match[1]?.trim() ?? '').filter(Boolean);\n    }) as [string[], string[]];\n\n    const fileContents = [...filePathsToBeModified, ...filePathsToBeReferred]\n      .map((filePath) => {\n        const content = fs.existsSync(filePath) ? fs.readFileSync(filePath, 'utf8').trim() : '';\n        const fence = findDistinctFence(content, '~');\n        return `## \\`${filePath}\\`\n\n${fence}\n${content}\n${fence}`;\n      })\n      .join('\\n\\n');\n\n    console.info(`Planning code changes with ${model} (reasoning effort: ${reasoningEffort}) ...`);\n    const planResponse = await callLlmApi(\n      model,\n      [\n        {\n          role: 'system',\n          content: buildPromptForPlanningCodeChanges(issueYamlText, isPullRequest),\n        },\n        {\n          role: 'user',\n          content: fileContents,\n        },\n      ],\n      reasoningEffort\n    );\n    console.info('Planning complete!');\n\n    const extractedContents = extractHeaderContents(trimCodeBlockFences(planResponse), [\n      HEADING_OF_COMMIT_MESSAGE,\n      HEADING_OF_PLAN,\n    ]);\n    if (!extractedContents) {\n      return { filePaths: filePathsToBeModified };\n    }\n    const [commitMessage, plan] = extractedContents;\n    return { plan, commitMessage: commitMessage?.trim(), filePaths: filePathsToBeModified };\n  }\n  console.info(`Planning code changes with ${model} (reasoning effort: ${reasoningEffort}) ...`);\n  const filesResponse = await callLlmApi(\n    model,\n    [\n      {\n        role: 'system',\n        content: buildPromptForSelectingFilesAndPlanningCodeChanges(issueYamlText, isPullRequest).trim(),\n      },\n      {\n        role: 'user',\n        content: repomixResult,\n      },\n    ],\n    reasoningEffort\n  );\n  console.info('Planning complete!');\n\n  const extractedContents = extractHeaderContents(trimCodeBlockFences(filesResponse), [\n    HEADING_OF_COMMIT_MESSAGE,\n    HEADING_OF_PLAN,\n    HEADING_OF_FILE_PATHS_TO_BE_MODIFIED,\n  ]);\n  if (!extractedContents) {\n    return { filePaths: [] };\n  }\n\n  const [commitMessage, plan, filePathsText] = extractedContents;\n\n  const filePathRegex = /\\B-\\s*`?([^`\\n]+)`?/g;\n  const matches = [...(filePathsText ?? '').matchAll(filePathRegex)];\n  const filePathsToBeModified = matches.map((match) => match[1]?.trim() ?? '').filter(Boolean);\n  return { plan, commitMessage: commitMessage?.trim(), filePaths: filePathsToBeModified };\n}\n\nfunction buildPromptForSelectingFiles(issueYamlText: string, isPullRequest = false): string {\n  const itemType = isPullRequest ? 'pull request' : 'issue';\n  const extraInstruction = isPullRequest ? ' Consider the comments on the pull request when identifying files.' : '';\n  return `\nYou are an expert software developer tasked with analyzing GitHub ${itemType}s and identifying relevant files for code changes.\n\nReview the following GitHub ${itemType} and the list of available file paths and their contents (which will be provided in a separate message).${extraInstruction}\nYour task is to identify:\n1. Files that need to be MODIFIED to resolve the ${itemType}\n2. Files that should be REFERRED to (but not modified) to understand the codebase better\n\nGitHub ${isPullRequest ? 'Pull Request' : 'Issue'}:\n${issueYamlText}\n\nPlease format your response without any explanatory text as follows:\n\\`\\`\\`md\n${HEADING_OF_FILE_PATHS_TO_BE_MODIFIED}\n\n- \\`[filePath1]\\`\n- \\`[filePath2]\\`\n- ...\n\n${HEADING_OF_FILE_PATHS_TO_BE_REFERRED}\n\n- \\`[filePath1]\\`\n- \\`[filePath2]\\`\n- ...\n\\`\\`\\`\n`;\n}\n\nfunction buildPromptForPlanningCodeChanges(issueYamlText: string, isPullRequest = false): string {\n  const itemType = isPullRequest ? 'pull request' : 'issue';\n  const extraInstruction = isPullRequest ? ' Consider the comments on the pull request when creating the plan.' : '';\n  return `\nYou are an expert software developer tasked with creating an implementation plan based on GitHub ${itemType}s.\n\nReview the following GitHub ${itemType} and the provided file contents (which will be provided in a separate message).${extraInstruction}\nCreate a detailed, step-by-step plan outlining how to address the ${itemType} effectively.\nAlso, provide a concise and descriptive commit message for the changes, following the Conventional Commits specification.\n\nYour plan should:\n- Focus on implementation details for each file that needs modification\n- Be clear and actionable for a developer to follow\n- Prefer showing diffs rather than complete file contents when describing changes\n- Exclude testing procedures unless users explicitly request\n\nGitHub ${isPullRequest ? 'Pull Request' : 'Issue'}:\n${issueYamlText}\n\nPlease format your response without any explanatory text as follows:\n\\`\\`\\`md\n${HEADING_OF_PLAN}\n\n1. [Specific implementation step]\n2. [Next implementation step]\n...\n\n${HEADING_OF_COMMIT_MESSAGE}\n\n[commit message]\n\\`\\`\\`\n`.trim();\n}\n\nfunction buildPromptForSelectingFilesAndPlanningCodeChanges(issueYamlText: string, isPullRequest = false): string {\n  const itemType = isPullRequest ? 'pull request' : 'issue';\n  const extraInstruction = isPullRequest ? ' Consider the comments on the pull request when creating the plan.' : '';\n  return `\nYou are an expert software developer tasked with analyzing GitHub ${itemType}s and creating an implementation plan.\n\nReview the following GitHub ${itemType} and the list of available file paths and their contents (which will be provided in a separate message).${extraInstruction}\nYour task is to:\n1. Create a detailed, step-by-step plan outlining how to resolve the ${itemType} effectively.\n2. Identify files that need to be modified to resolve the ${itemType}.\n3. Provide a concise and descriptive commit message for the changes, following the Conventional Commits specification.\n\nYour plan should:\n- Focus on implementation details for each file that needs modification\n- Be clear and actionable for a developer to follow\n- Prefer showing diffs rather than complete file contents when describing changes\n- Exclude testing procedures as those will be handled separately\n\nGitHub ${isPullRequest ? 'Pull Request' : 'Issue'}:\n${issueYamlText}\n\nPlease format your response without any explanatory text as follows:\n\\`\\`\\`md\n${HEADING_OF_PLAN}\n\n1. [Specific implementation step]\n2. [Next implementation step]\n...\n\n${HEADING_OF_FILE_PATHS_TO_BE_MODIFIED}\n\n- \\`[filePath1]\\`\n- \\`[filePath2]\\`\n- ...\n\n${HEADING_OF_COMMIT_MESSAGE}\n\n[commit message]\n\\`\\`\\`\n`;\n}\n","import { DEFAULT_AIDER_EXTRA_ARGS } from '../defaultOptions.js';\nimport type { MainOptions } from '../main.js';\nimport type { ResolutionPlan } from '../plan.js';\n\nimport { parseCommandLineArgs } from '../spawn.js';\n\n/**\n * Builds the command line arguments for the Aider command\n *\n * @param options The main options object\n * @param args Arguments to include\n * @returns An array of command line arguments for aider\n */\nexport function buildAiderArgs(\n  options: MainOptions,\n  args: { prompt: string; resolutionPlan?: ResolutionPlan }\n): string[] {\n  const aiderArgs = [\n    '--yes-always',\n    '--no-check-update',\n    '--no-gitignore',\n    '--no-show-model-warnings',\n    '--no-show-release-notes',\n    ...parseCommandLineArgs(options.aiderExtraArgs || DEFAULT_AIDER_EXTRA_ARGS),\n    '--message',\n    args.prompt,\n  ];\n\n  if (options.dryRun) {\n    aiderArgs.push('--dry-run');\n  }\n  if (args.resolutionPlan && 'filePaths' in args.resolutionPlan) {\n    aiderArgs.push(...args.resolutionPlan.filePaths);\n  }\n\n  return aiderArgs;\n}\n","export const isCI = process.env.CI || process.env.GITHUB_ACTIONS;\n","import { isCI } from '../ci.js';\nimport { DEFAULT_CLAUDE_CODE_EXTRA_ARGS } from '../defaultOptions.js';\nimport type { MainOptions } from '../main.js';\nimport type { ResolutionPlan } from '../plan.js';\n\nimport { parseCommandLineArgs } from '../spawn.js';\n\n/**\n * Builds the command line arguments for the Claude Code command\n *\n * @param options The main options object\n * @param args Arguments to include\n * @returns An array of command line arguments for @anthropic-ai/claude-code@latest\n */\nexport function buildClaudeCodeArgs(\n  options: MainOptions,\n  args: { prompt: string; resolutionPlan?: ResolutionPlan }\n): string[] {\n  // cf. https://docs.anthropic.com/en/docs/claude-code/cli-usage\n  const baseArgs = [\n    '--yes',\n    '@anthropic-ai/claude-code@latest',\n    ...parseCommandLineArgs(options.claudeCodeExtraArgs || DEFAULT_CLAUDE_CODE_EXTRA_ARGS),\n    // Bypass all permission checks\n    '--dangerously-skip-permissions',\n  ];\n  if (isCI) {\n    baseArgs.push('--print');\n  }\n  baseArgs.push(args.prompt);\n  return baseArgs;\n}\n","import { DEFAULT_CODEX_EXTRA_ARGS } from '../defaultOptions.js';\nimport type { MainOptions } from '../main.js';\nimport type { ResolutionPlan } from '../plan.js';\n\nimport { parseCommandLineArgs } from '../spawn.js';\n\n/**\n * Builds the command line arguments for the Codex CLI command\n *\n * @param options The main options object\n * @param args Arguments to include\n * @returns An array of command line arguments for @openai/codex@latest\n */\nexport function buildCodexArgs(\n  options: MainOptions,\n  args: { prompt: string; resolutionPlan?: ResolutionPlan }\n): string[] {\n  return [\n    '--yes',\n    '@openai/codex@latest',\n    'exec',\n    ...parseCommandLineArgs(options.codexExtraArgs || DEFAULT_CODEX_EXTRA_ARGS),\n    args.prompt,\n  ];\n}\n","import { DEFAULT_GEMINI_EXTRA_ARGS } from '../defaultOptions.js';\nimport type { MainOptions } from '../main.js';\nimport type { ResolutionPlan } from '../plan.js';\n\nimport { parseCommandLineArgs } from '../spawn.js';\n\n/**\n * Builds the command line arguments for the Gemini CLI command\n *\n * @param options The main options object\n * @param args Arguments to include\n * @returns An array of command line arguments for @google/gemini-cli@latest\n */\nexport function buildGeminiArgs(\n  options: MainOptions,\n  args: { prompt: string; resolutionPlan?: ResolutionPlan }\n): string[] {\n  return [\n    '--yes',\n    '@google/gemini-cli@latest',\n    '--yolo',\n    ...parseCommandLineArgs(options.geminiExtraArgs || DEFAULT_GEMINI_EXTRA_ARGS),\n    '--prompt',\n    args.prompt,\n  ];\n}\n","import ansis from 'ansis';\nimport type { MainOptions } from './main.js';\nimport { findDistinctFence } from './markdown.js';\nimport type { ResolutionPlan } from './plan.js';\nimport { parseCommandLineArgs, runCommand, spawnAsync } from './spawn.js';\nimport { buildAiderArgs } from './tools/aider.js';\nimport { buildClaudeCodeArgs } from './tools/claudeCode.js';\nimport { buildCodexArgs } from './tools/codex.js';\nimport { buildGeminiArgs } from './tools/gemini.js';\n\nexport interface TestResult {\n  fixResult: string;\n  success: boolean;\n  error?: string;\n}\n\nexport async function testAndFix(options: MainOptions, resolutionPlan?: ResolutionPlan): Promise<TestResult> {\n  const [commandProgram, ...commandArgs] = parseCommandLineArgs(options.testCommand || '');\n  if (!commandProgram) return { fixResult: '', success: true };\n\n  const maxAttempts = options.maxTestAttempts;\n  let attempts = 0;\n  let fixResult = '';\n  let success = false;\n  let lastError = '';\n\n  while (attempts < maxAttempts) {\n    attempts++;\n    console.info(ansis.cyan(`Executing test command (attempt ${attempts}/${maxAttempts}): ${options.testCommand}`));\n\n    const testResult = await spawnAsync(commandProgram, commandArgs, {\n      cwd: process.cwd(),\n    });\n\n    if (testResult.status === 0) {\n      console.info(ansis.green('Test command passed successfully.'));\n      success = true;\n      break;\n    }\n\n    console.warn(ansis.yellow(`Test command failed with exit code ${testResult.status}.`));\n\n    // Capture the error details\n    lastError = `Test command failed with exit code ${testResult.status}\\n\\nStdout:\\n${testResult.stdout}\\n\\nStderr:\\n${testResult.stderr}`;\n\n    // Only try to fix if we haven't reached the maximum attempts\n    if (attempts >= maxAttempts) {\n      console.warn(ansis.yellow(`Maximum fix attempts (${maxAttempts}) reached. Giving up.`));\n      break;\n    }\n\n    const stdoutFence = findDistinctFence(testResult.stdout, '~');\n    const stderrFence = findDistinctFence(testResult.stderr, '~');\n    const prompt = `\nThe previous changes were applied, but the test command \\`${options.testCommand}\\` failed.\n\nExit code: ${testResult.status}\n\nStdout:\n${stdoutFence}\n${testResult.stdout}\n${stdoutFence}\n\nStderr:\n${stderrFence}\n${testResult.stderr}\n${stderrFence}\n\nPlease analyze the output and fix the errors.\n`.trim();\n\n    fixResult += await runToolFix(options, prompt, resolutionPlan);\n  }\n\n  return { fixResult, success, error: success ? undefined : lastError };\n}\n\n/**\n * Helper function to run coding tool with a fix prompt\n */\nexport async function runToolFix(\n  options: MainOptions,\n  prompt: string,\n  resolutionPlan?: ResolutionPlan\n): Promise<string> {\n  const toolName =\n    options.codingTool === 'aider'\n      ? 'Aider'\n      : options.codingTool === 'claude-code'\n        ? 'Claude Code'\n        : options.codingTool === 'codex-cli'\n          ? 'Codex CLI'\n          : 'Gemini CLI';\n  let assistantResult: string;\n\n  if (options.codingTool === 'aider') {\n    const aiderArgs = buildAiderArgs(options, { prompt, resolutionPlan });\n    console.info(ansis.cyan(`Asking Aider to fix \"${options.testCommand}\"...`));\n    assistantResult = (\n      await runCommand('aider', aiderArgs, {\n        env: { ...process.env, NO_COLOR: '1' },\n        ignoreExitStatus: true,\n      })\n    ).stdout;\n  } else if (options.codingTool === 'claude-code') {\n    const claudeCodeArgs = buildClaudeCodeArgs(options, { prompt, resolutionPlan });\n    console.info(ansis.cyan(`Asking Claude Code to fix \"${options.testCommand}\"...`));\n    assistantResult = (\n      await runCommand(options.nodeRuntime, claudeCodeArgs, {\n        env: { ...process.env, NO_COLOR: '1' },\n        stdio: 'inherit',\n        ignoreExitStatus: true,\n      })\n    ).stdout;\n  } else if (options.codingTool === 'codex-cli') {\n    const codexArgs = buildCodexArgs(options, { prompt, resolutionPlan });\n    console.info(ansis.cyan(`Asking Codex to fix \"${options.testCommand}\"...`));\n    assistantResult = (\n      await runCommand(options.nodeRuntime, codexArgs, {\n        env: { ...process.env, NO_COLOR: '1' },\n        ignoreExitStatus: true,\n      })\n    ).stdout;\n  } else {\n    const geminiArgs = buildGeminiArgs(options, { prompt, resolutionPlan });\n    console.info(ansis.cyan(`Asking Gemini CLI to fix \"${options.testCommand}\"...`));\n    assistantResult = (\n      await runCommand(options.nodeRuntime, geminiArgs, {\n        env: { ...process.env, NO_COLOR: '1' },\n        ignoreExitStatus: true,\n      })\n    ).stdout;\n  }\n\n  return `\\n\\n## ${toolName} fix attempt for \"${options.testCommand}\"\\n\\n${assistantResult.trim()}`;\n}\n","import type { SpawnOptions } from 'node:child_process';\nimport ansis from 'ansis';\nimport YAML from 'yaml';\nimport { configureEnvVars } from './env.js';\nimport { configureGitUserDetailsIfNeeded, getBaseBranch, getCurrentBranch, getHeaderOfFirstCommit } from './git.js';\nimport { createIssueInfo } from './issue.js';\nimport { findDistinctFence } from './markdown.js';\nimport { createPullRequest } from './octokit.js';\nimport { planCodeChanges } from './plan.js';\nimport { runCommand } from './spawn.js';\nimport { testAndFix } from './test.js';\nimport { HEADING_OF_GEN_PR_METADATA, truncateText } from './text.js';\nimport { buildAiderArgs } from './tools/aider.js';\nimport { buildClaudeCodeArgs } from './tools/claudeCode.js';\nimport { buildCodexArgs } from './tools/codex.js';\nimport { buildGeminiArgs } from './tools/gemini.js';\nimport type { CodingTool, NodeRuntimeActual, ReasoningEffort } from './types.js';\nimport { yamlStringifyOptions } from './yaml.js';\n\n/**\n * Options for the main function\n */\nexport interface MainOptions {\n  /** Additional arguments to pass to Aider */\n  aiderExtraArgs?: string;\n  /** Additional arguments to pass to Claude Code */\n  claudeCodeExtraArgs?: string;\n  /** Additional arguments to pass to Codex CLI */\n  codexExtraArgs?: string;\n  /** Additional arguments to pass to Gemini CLI */\n  geminiExtraArgs?: string;\n  /** Coding tool to use */\n  codingTool: CodingTool;\n  /** Enable two-staged planning: first select relevant files, then generate detailed implementation plans */\n  twoStagePlanning: boolean;\n  /** Run without making actual changes (no branch creation, no PR) */\n  dryRun: boolean;\n  /** Do not create a new branch, commit changes directly to the base branch */\n  noBranch: boolean;\n  /** Node.js runtime to use (already normalized, without aliases) */\n  nodeRuntime: NodeRuntimeActual;\n  /** GitHub issue number to process */\n  issueNumber: number;\n  /** Maximum number of attempts to fix test failures */\n  maxTestAttempts: number;\n  /** LLM model to use for planning code changes */\n  planningModel?: string;\n  /** Level of reasoning effort for the LLM */\n  reasoningEffort?: ReasoningEffort;\n  /** Extra arguments for repomix when generating context */\n  repomixExtraArgs?: string;\n  /** Command to run after coding tool applies changes. If it fails, the assistant will try to fix it. */\n  testCommand?: string;\n  /** RegExp pattern to remove from issue and PR descriptions */\n  removePattern?: string;\n}\n\nconst MAX_PR_BODY_LENGTH = 30000; // GitHub's limit is 65536, leave some buffer\n\nexport async function main(options: MainOptions): Promise<void> {\n  configureEnvVars();\n\n  if (options.dryRun) {\n    console.info(ansis.yellow('Running in dry-run mode. No branches or PRs will be created.'));\n  } else {\n    await configureGitUserDetailsIfNeeded();\n  }\n\n  // Install coding tools\n  if (options.codingTool === 'aider') {\n    await runCommand('python', ['-m', 'pip', 'install', 'aider-install']);\n    await reshimToDetectNewTools();\n    await runCommand('uv', ['tool', 'uninstall', 'aider-chat'], { ignoreExitStatus: true });\n    await runCommand('aider-install', []);\n    await reshimToDetectNewTools();\n\n    if (options.aiderExtraArgs?.includes('bedrock/')) {\n      await runCommand('uv', [\n        'tool',\n        'run',\n        '--from',\n        'aider-chat',\n        'pip',\n        'install',\n        '--upgrade',\n        '--upgrade-strategy',\n        'only-if-needed',\n        'boto3',\n      ]);\n      // await runCommand('aider', ['--install-main-branch', '--yes-always']);\n    }\n  }\n\n  const prBaseBranch = await getBaseBranch(options);\n  const isPullRequest = !!prBaseBranch;\n  const baseBranch = prBaseBranch || (await getCurrentBranch());\n\n  const issueInfo = await createIssueInfo(options);\n  const issueText = YAML.stringify(issueInfo, yamlStringifyOptions).trim();\n\n  const resolutionPlan =\n    (options.planningModel &&\n      (await planCodeChanges(\n        options.planningModel,\n        issueText,\n        options.twoStagePlanning,\n        options.reasoningEffort,\n        options.repomixExtraArgs,\n        isPullRequest\n      ))) ||\n    undefined;\n  console.info('Resolution plan:', resolutionPlan);\n\n  const planText = (resolutionPlan && 'plan' in resolutionPlan && resolutionPlan.plan) || '';\n  const issueFence = findDistinctFence(issueText, '~');\n  const isAgentic = options.codingTool !== 'aider';\n  const target = isPullRequest ? 'the comments on the following GitHub pull request' : 'the following GitHub issue';\n  const prompt = `\nModify the code to resolve ${target}${planText ? ' based on the plan' : ''}.${isAgentic ? ' After that, commit your changes with a message, following the Conventional Commits specification.' : ''}\n\n## ${isPullRequest ? 'Pull Request' : 'Issue'}\n\n${issueFence}yml\n${issueText}\n${issueFence}\n\n${\n  planText &&\n  `## Plan\n\n${planText}`\n}\n`.trim();\n\n  const now = new Date();\n  const newBranchName = `gen-pr-${options.issueNumber}-${options.codingTool}-${now.getFullYear()}_${getTwoDigits(now.getMonth() + 1)}${getTwoDigits(now.getDate())}_${getTwoDigits(now.getHours())}${getTwoDigits(now.getMinutes())}${getTwoDigits(now.getSeconds())}`;\n\n  if (options.noBranch) {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would commit directly to base branch: ${baseBranch}`));\n    } else {\n      if (isPullRequest) {\n        await runCommand('git', ['fetch', 'origin', baseBranch]);\n      }\n      await runCommand('git', ['switch', baseBranch]);\n    }\n  } else {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would create branch: ${newBranchName}`));\n    } else {\n      if (isPullRequest) {\n        await runCommand('git', ['fetch', 'origin', baseBranch]);\n        await runCommand('git', ['switch', baseBranch]);\n      }\n      await runCommand('git', ['switch', '--force-create', newBranchName]);\n    }\n  }\n\n  // Execute coding tool\n  let toolResult = '';\n  let toolCommand: string;\n  let toolError = '';\n  let toolSuccess = true;\n  const toolName =\n    options.codingTool === 'aider'\n      ? 'Aider'\n      : options.codingTool === 'claude-code'\n        ? 'Claude Code'\n        : options.codingTool === 'codex-cli'\n          ? 'Codex CLI'\n          : 'Gemini CLI';\n\n  // Build tool configuration\n  let toolArgs: string[];\n  let command: string;\n  let runOpts: SpawnOptions & { ignoreExitStatus?: boolean } = {\n    env: { ...process.env, NO_COLOR: '1' },\n    ignoreExitStatus: true,\n  };\n\n  if (options.codingTool === 'aider') {\n    toolArgs = buildAiderArgs(options, { prompt: prompt, resolutionPlan });\n    command = 'aider';\n  } else if (options.codingTool === 'claude-code') {\n    toolArgs = buildClaudeCodeArgs(options, { prompt: prompt, resolutionPlan });\n    command = options.nodeRuntime;\n    runOpts = { ...runOpts, stdio: 'inherit' };\n  } else if (options.codingTool === 'codex-cli') {\n    toolArgs = buildCodexArgs(options, { prompt: prompt, resolutionPlan });\n    command = options.nodeRuntime;\n  } else {\n    toolArgs = buildGeminiArgs(options, { prompt: prompt, resolutionPlan });\n    command = options.nodeRuntime;\n  }\n\n  toolCommand = buildToolCommandString(command, toolArgs, prompt);\n\n  // Execute tool command\n  if (options.dryRun) {\n    console.info(`\\n=== DRY MODE: ${toolName} Prompt ===`);\n    console.info(prompt);\n    console.info(`=== End ${toolName} Prompt ===\\n`);\n    console.info(ansis.yellow(`Would run: ${toolCommand}`));\n    toolResult = 'Skipped due to dry-run mode';\n  } else {\n    const toolRunResult = await runCommand(command, toolArgs, runOpts);\n    toolResult = toolRunResult.stdout || '';\n    if (toolRunResult.status !== 0) {\n      toolSuccess = false;\n      toolError = `${toolName} failed with exit code ${toolRunResult.status}\\n${toolRunResult.stderr}`;\n      console.error(ansis.red(`${toolName} execution failed: ${toolError}`));\n    }\n  }\n\n  let toolResponse = (toolResult || '').trim();\n  let testSuccess = true;\n  let testError = '';\n  if (options.dryRun) {\n    console.info(ansis.yellow(`Would run test command`));\n  } else {\n    const testResult = await testAndFix(options, resolutionPlan);\n    toolResponse += testResult.fixResult;\n    testSuccess = testResult.success;\n    testError = testResult.error || '';\n    if (!testSuccess) {\n      console.warn(ansis.yellow('Tests failed after all fix attempts. Will create a draft PR.'));\n    }\n  }\n\n  if (!toolSuccess) {\n    console.warn(ansis.yellow(`${toolName} execution failed. Will create a draft PR.`));\n  }\n\n  // Determine if PR should be a draft\n  const shouldBeDraft = !toolSuccess || !testSuccess;\n  let errorLogs = '';\n  if (shouldBeDraft) {\n    if (toolError) {\n      errorLogs += `\\n\\n### ❌ ${toolName} Execution Error\\n\\n\\`\\`\\`\\n${toolError}\\n\\`\\`\\``;\n    }\n    if (testError) {\n      errorLogs += `\\n\\n### ❌ Test Execution Error\\n\\n\\`\\`\\`\\n${testError}\\n\\`\\`\\``;\n    }\n  }\n\n  // Try commiting changes because coding tool may fail to commit changes due to pre-commit hooks\n  const commitMessage = resolutionPlan?.commitMessage || `fix: Close #${options.issueNumber}`;\n  await runCommand('git', ['add', '-A'], { ignoreExitStatus: true });\n  if (\n    (\n      await runCommand('git', ['commit', '-m', commitMessage], {\n        ignoreExitStatus: true,\n      })\n    ).status !== 0\n  ) {\n    await runCommand('git', ['commit', '-m', commitMessage, '--no-verify'], {\n      ignoreExitStatus: true,\n    });\n  }\n  if (options.noBranch) {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would push changes directly to base branch: ${baseBranch}`));\n    } else {\n      await runCommand('git', ['push', 'origin', baseBranch, '--no-verify']);\n    }\n  } else {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would push branch: ${newBranchName} to origin`));\n    } else {\n      await runCommand('git', ['push', 'origin', newBranchName, '--no-verify']);\n    }\n  }\n\n  // Create a PR using GitHub CLI (unless noBranch is enabled)\n  if (options.noBranch) {\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Skipping PR creation due to --no-branch option`));\n    } else {\n      console.info(`Changes committed directly to base branch: ${baseBranch}`);\n    }\n  } else {\n    const prTitle = getHeaderOfFirstCommit(baseBranch) || commitMessage;\n    let prBody = isPullRequest ? '' : `Close #${options.issueNumber}`;\n\n    if (options.planningModel) {\n      prBody += `\n\n${HEADING_OF_GEN_PR_METADATA}\n\n- **Planning Model:** ${options.planningModel}`;\n    }\n\n    prBody += `\n- **Coding Tool:** ${toolName}\n- **Coding Command:** \\`${toolCommand}\\``;\n    if (planText) {\n      const responseFence = findDistinctFence(planText, '~');\n      prBody += `\n\n### Plan\n\n${responseFence}\n${truncateText(planText, (toolResponse.length / (planText.length + toolResponse.length)) * MAX_PR_BODY_LENGTH)}\n${responseFence}`;\n    }\n    if (toolResponse) {\n      const responseFence = findDistinctFence(toolResponse, '~');\n      prBody += `\n\n### ${toolName} Log\n\n${responseFence}\n${truncateText(toolResponse, (toolResponse.length / (planText.length + toolResponse.length)) * MAX_PR_BODY_LENGTH)}\n${responseFence}`;\n    }\n    prBody = prBody.replaceAll(/(?:\\s*\\n){2,}/g, '\\n\\n').trim();\n\n    // Add error logs to PR body if it's a draft due to failures\n    if (shouldBeDraft) {\n      prBody += errorLogs;\n    }\n\n    if (options.dryRun) {\n      console.info(ansis.yellow(`Would create PR with title: ${prTitle}${shouldBeDraft ? ' (as draft)' : ''}`));\n    } else {\n      await createPullRequest({\n        title: prTitle,\n        body: prBody,\n        head: newBranchName,\n        base: baseBranch,\n        draft: shouldBeDraft,\n      });\n    }\n  }\n\n  console.info(`\\n${isPullRequest ? 'Pull request' : 'Issue'} #${options.issueNumber} processed successfully.`);\n}\n\nasync function reshimToDetectNewTools() {\n  try {\n    // Make uv available on asdf environment\n    await runCommand('asdf', ['reshim'], { ignoreExitStatus: true });\n  } catch {\n    // do nothing\n  }\n}\n\nfunction getTwoDigits(value: number): string {\n  return String(value).padStart(2, '0');\n}\n\n/**\n * Builds a command string for display, replacing the prompt argument with ...\n */\nfunction buildToolCommandString(command: string, args: string[], prompt: string): string {\n  const escapedArgs = args.map((arg) => {\n    if (arg === prompt) {\n      return '...';\n    }\n    if (arg.includes(' ') || arg.includes('\"') || arg.includes(\"'\")) {\n      return `\"${arg.replace(/\"/g, '\\\\\"')}\"`;\n    }\n    return arg;\n  });\n  return `${command} ${escapedArgs.join(' ')}`;\n}\n","import { config } from 'dotenv';\n\nexport function configureEnvVars(): void {\n  config();\n\n  if (!process.env.AWS_REGION && process.env.AWS_REGION_NAME) {\n    process.env.AWS_REGION = process.env.AWS_REGION_NAME;\n  } else if (process.env.AWS_REGION && !process.env.AWS_REGION_NAME) {\n    process.env.AWS_REGION_NAME = process.env.AWS_REGION;\n  }\n\n  if (!process.env.GEMINI_API_KEY && process.env.GOOGLE_GENERATIVE_AI_API_KEY) {\n    process.env.GEMINI_API_KEY = process.env.GOOGLE_GENERATIVE_AI_API_KEY;\n  } else if (process.env.GEMINI_API_KEY && !process.env.GOOGLE_GENERATIVE_AI_API_KEY) {\n    process.env.GOOGLE_GENERATIVE_AI_API_KEY = process.env.GEMINI_API_KEY;\n  }\n}\n","import child_process from 'node:child_process';\nimport type { MainOptions } from './main.js';\nimport { runCommand } from './spawn.js';\n\nexport async function configureGitUserDetailsIfNeeded(): Promise<void> {\n  const gitUserName = (await runCommand('git', ['config', 'user.name'], { ignoreExitStatus: true })).stdout.trim();\n  const gitUserEmail = (await runCommand('git', ['config', 'user.email'], { ignoreExitStatus: true })).stdout.trim();\n  if (!gitUserName || !gitUserEmail) {\n    const githubUserInfoJson = (await runCommand('gh', ['api', 'user'], { ignoreExitStatus: true })).stdout.trim();\n    try {\n      const { name, email } = JSON.parse(githubUserInfoJson);\n      if (name) {\n        await runCommand('git', ['config', 'user.name', name]);\n      }\n      if (email) {\n        await runCommand('git', ['config', 'user.email', email]);\n      }\n    } catch {}\n  }\n}\n\nexport async function getCurrentBranch(): Promise<string> {\n  const currentBranchResult = await runCommand('git', ['branch', '--show-current']);\n  return currentBranchResult.stdout.trim();\n}\n\nexport async function getBaseBranch(options: MainOptions): Promise<string | undefined> {\n  const { stdout: prViewResult } = await runCommand(\n    'gh',\n    ['pr', 'view', options.issueNumber.toString(), '--json', 'headRefName'],\n    { ignoreExitStatus: true }\n  );\n  try {\n    if (prViewResult) {\n      const prData = JSON.parse(prViewResult);\n      return prData.headRefName;\n    }\n  } catch {\n    // Not a PR or error parsing\n  }\n}\n\nexport function getGitRepoName(): string {\n  const repoUrlResult = child_process.spawnSync('git', ['remote', 'get-url', 'origin'], {\n    encoding: 'utf8',\n    stdio: 'pipe',\n  });\n  const repoUrl = repoUrlResult.stdout.trim();\n  const repoMatch = repoUrl.match(/github\\.com[/:]([\\w-]+\\/[\\w-]+)(\\.git)?$/);\n  return repoMatch?.[1] ?? '';\n}\n\nexport function getHeaderOfFirstCommit(baseBranch: string): string {\n  const firstCommitResult = child_process.spawnSync('git', ['log', `${baseBranch}..HEAD`, '--reverse', '--pretty=%s'], {\n    encoding: 'utf8',\n    stdio: 'pipe',\n  });\n  return (\n    firstCommitResult.stdout\n      .trim()\n      .split('\\n')\n      .find((line) => line.trim()) ?? ''\n  );\n}\n"],"names":["loadConfigFile","configOptions","name","cfgPath","path","resolve","process","cwd","fs","existsSync","YAML","parse","readFileSync","console","info","err","error","exit","DEFAULT_AIDER_EXTRA_ARGS","DEFAULT_CLAUDE_CODE_EXTRA_ARGS","DEFAULT_CODEX_EXTRA_ARGS","DEFAULT_GEMINI_EXTRA_ARGS","DEFAULT_REPOMIX_EXTRA_ARGS","DEFAULT_MAX_TEST_ATTEMPTS","DEFAULT_CODING_TOOL","DEFAULT_NODE_RUNTIME","HEADING_OF_GEN_PR_METADATA","truncateText","text","maxLength","length","truncated","slice","omitted","Math","floor","normalizeNewLines","replaceAll","trim","normalizeNodeRuntime","runtime","async","runCommand","command","args","options","ignoreExitStatus","spawnOptions","argsText","map","a","includes","join","ansis","green","ret","spawnAsync","truncateStdout","stdout","stderr","truncatedStderr","yellow","magenta","status","Promise","reject","sanitizedArgs","arg","replace","proc","spawn","setEncoding","on","data","write","code","signal","undefined","pid","Error","parseCommandLineArgs","argsString","result","current","inDoubleQuote","inSingleQuote","i","char","push","token","env","GH_TOKEN","GITHUB_TOKEN","child_process","spawnSync","encoding","octokit","Octokit","auth","graphqlClient","graphql","defaults","headers","authorization","repoOwner","repoName","getRepoInfo","match","owner","repo","fetchPRDiff","issueNumber","issueInfo","prDiff","pullNumber","pulls","get","pull_number","mediaType","format","getPullRequestDiff","code_changes","diffContent","MAX_TOTAL_DIFF_SIZE","MAX_FILE_DIFF_SIZE","LARGE_FILE_PATTERNS","fileSections","split","processedSections","totalSize","section","some","pattern","test","truncatedSection","processDiffContent","warn","fetchIssueData","processedIssues","isReferenced","has","issue","add","issueData","issues","issue_number","issueOrPullRequest","pull_request","prData","commentsResponse","listComments","author","user","login","title","body","labels","label","comments","comment","createdAt","created_at","url","html_url","getIssue","referencedNumbers","regex","numbers","exec","number","Number","parseInt","isInteger","Set","extractIssueReferences","c","rawBody","processedBody","markdownContent","index","indexOf","substring","stripMetadataSections","description","RegExp","removeRegexPattern","removePattern","commentsWithDate","Date","getTime","graphqlData","pr","getPullRequestReviewThreads","reviewThreads","repository","pullRequest","nodes","thread","isResolved","processReviewThreadComments","fetchPRReviewThreads","reviewResultComments","listReviews","review","state","submitted_at","getPullRequestReviews","reviewState","fetchPRReviews","referencedIssuesPromises","num","referencedIssues","all","filter","referenced_issues","fetchReferencedIssues","sort","b","codeContent","extractCodeFromDiffHunk","diffHunk","reviewComment","codeLocation","line","Object","keys","forEach","key","codeLine","find","startsWith","extractHeaderContents","modifiedResponse","indices","header","every","start","end","findDistinctFence","content","fenceChar","matches","max","seq","fenceLength","repeat","trimCodeBlockFences","convertToV4Messages","messages","msg","id","role","JSON","stringify","yamlStringifyOptions","blockQuote","lineWidth","callLlmApi","model","reasoningEffort","provider","modelParts","modelName","ollamaBaseURL","OLLAMA_BASE_URL","providerModel","createOllama","baseURL","OLLAMA_API_KEY","apiKey","ollama","generateText","providerOptions","think","logResult","charAt","toUpperCase","callV4ProviderApi","modelInstance","createOpenAI","openaiProvider","createAnthropic","anthropicProvider","createGoogleGenerativeAI","googleProvider","createAzure","azureProvider","createAmazonBedrock","bedrockProvider","createVertex","vertexProvider","createXai","grokProvider","createOpenRouter","OPENROUTER_API_KEY","openrouterProvider","reasoning","effort","getModelInstance","requestParams","modelSupportsReasoning","supportsReasoningOptions","thinkingBudget","low","medium","high","getThinkingBudget","openai","anthropic","thinking","type","budgetTokens","google","thinkingConfig","xai","usage","finishReason","REPOMIX_FILE_NAME","HEADING_OF_FILE_PATHS_TO_BE_MODIFIED","HEADING_OF_FILE_PATHS_TO_BE_REFERRED","HEADING_OF_PLAN","HEADING_OF_COMMIT_MESSAGE","buildPromptForSelectingFiles","issueYamlText","isPullRequest","itemType","buildPromptForPlanningCodeChanges","buildPromptForSelectingFilesAndPlanningCodeChanges","buildAiderArgs","aiderArgs","aiderExtraArgs","prompt","dryRun","resolutionPlan","filePaths","isCI","CI","GITHUB_ACTIONS","buildClaudeCodeArgs","baseArgs","claudeCodeExtraArgs","buildCodexArgs","codexExtraArgs","buildGeminiArgs","geminiExtraArgs","runToolFix","toolName","codingTool","assistantResult","cyan","testCommand","NO_COLOR","claudeCodeArgs","nodeRuntime","stdio","codexArgs","geminiArgs","main","config","AWS_REGION","AWS_REGION_NAME","GEMINI_API_KEY","GOOGLE_GENERATIVE_AI_API_KEY","gitUserName","gitUserEmail","githubUserInfoJson","email","configureGitUserDetailsIfNeeded","reshimToDetectNewTools","prBaseBranch","prViewResult","toString","headRefName","getBaseBranch","baseBranch","getCurrentBranch","createIssueInfo","issueText","planningModel","issueContent","twoStagePlanning","repomixExtraArgs","issueFence","repomixArgs","repomixResult","promises","rm","force","filesResponse","extractedFilePathLists","filePathsToBeModified","filePathsToBeReferred","filesContent","matchAll","Boolean","fileContents","filePath","fence","planResponse","extractedContents","commitMessage","plan","filePathsText","planCodeChanges","planText","now","newBranchName","getFullYear","getTwoDigits","getMonth","getDate","getHours","getMinutes","getSeconds","noBranch","toolCommand","toolResult","toolError","toolSuccess","toolArgs","runOpts","escapedArgs","buildToolCommandString","toolRunResult","red","toolResponse","testSuccess","testError","testResult","commandProgram","commandArgs","fixResult","success","maxAttempts","maxTestAttempts","attempts","lastError","stdoutFence","stderrFence","testAndFix","shouldBeDraft","errorLogs","prTitle","getHeaderOfFirstCommit","prBody","responseFence","params","create","head","base","draft","createPullRequest","value","String","padStart"],"mappings":"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"}