@hua-labs/tap 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/engine/termination.ts","../../src/engine/review.ts","../../src/engine/headless-loop.ts","../../src/bridges/codex-bridge-runner.ts","../../src/config/resolve.ts","../../src/runtime/resolve-node.ts"],"sourcesContent":["/**\n * Termination engine — decides when a review session should stop.\n *\n * Strategies are evaluated in priority order. First non-\"continue\" verdict wins.\n * Default order: manual-stop → round-cap → repetition → quality → diff-insignificance\n */\nimport * as fs from \"node:fs\";\nimport * as crypto from \"node:crypto\";\n\n// ── Types ──────────────────────────────────────────────────────────\n\nexport type TerminationStrategy =\n | \"diff-insignificance\"\n | \"repetition-detection\"\n | \"quality-threshold\"\n | \"round-cap\"\n | \"manual-stop\";\n\nexport type TerminationVerdict = \"continue\" | \"stop\" | \"escalate\";\n\nexport type FindingSeverity =\n | \"critical\"\n | \"high\"\n | \"medium\"\n | \"low\"\n | \"nitpick\";\n\nexport interface ReviewFinding {\n severity: FindingSeverity;\n category: string;\n description: string;\n file?: string;\n line?: number;\n}\n\nexport interface ReviewRound {\n round: number;\n timestamp: string;\n findingCount: number;\n findings: ReviewFinding[];\n suggestedDiffLines: number;\n findingHash: string;\n}\n\nexport interface TerminationConfig {\n strategies: TerminationStrategy[];\n maxRounds: number;\n diffThreshold: number;\n repetitionThreshold: number;\n qualitySeverityFloor: FindingSeverity;\n}\n\nexport interface TerminationContext {\n round: number;\n rounds: ReviewRound[];\n stopSignalPath: string;\n config: TerminationConfig;\n}\n\nexport interface TerminationResult {\n verdict: TerminationVerdict;\n reason: string;\n strategy: TerminationStrategy;\n summary: string;\n}\n\n// ── Defaults ───────────────────────────────────────────────────────\n\nexport const DEFAULT_TERMINATION_CONFIG: TerminationConfig = {\n strategies: [\n \"manual-stop\",\n \"round-cap\",\n \"repetition-detection\",\n \"quality-threshold\",\n \"diff-insignificance\",\n ],\n maxRounds: 5,\n diffThreshold: 3,\n repetitionThreshold: 2,\n qualitySeverityFloor: \"high\",\n};\n\n// ── Severity ranking ───────────────────────────────────────────────\n\nconst SEVERITY_RANK: Record<FindingSeverity, number> = {\n critical: 5,\n high: 4,\n medium: 3,\n low: 2,\n nitpick: 1,\n};\n\nfunction isAtOrAbove(\n severity: FindingSeverity,\n floor: FindingSeverity,\n): boolean {\n return SEVERITY_RANK[severity] >= SEVERITY_RANK[floor];\n}\n\n// ── Finding hash ───────────────────────────────────────────────────\n\nexport function computeFindingHash(findings: ReviewFinding[]): string {\n const normalized = findings\n .filter((f) => isAtOrAbove(f.severity, \"high\"))\n .map((f) => `${f.category}:${f.description.slice(0, 100)}`)\n .sort()\n .join(\"|\");\n\n if (!normalized) return \"empty\";\n\n return crypto\n .createHash(\"sha256\")\n .update(normalized)\n .digest(\"hex\")\n .slice(0, 16);\n}\n\n// ── Strategy evaluators ────────────────────────────────────────────\n\nfunction evalManualStop(ctx: TerminationContext): TerminationResult | null {\n if (fs.existsSync(ctx.stopSignalPath)) {\n return {\n verdict: \"stop\",\n reason: `Manual stop signal found at ${ctx.stopSignalPath}`,\n strategy: \"manual-stop\",\n summary: `Review stopped manually at round ${ctx.round}`,\n };\n }\n return null;\n}\n\nfunction evalRoundCap(ctx: TerminationContext): TerminationResult | null {\n if (ctx.round >= ctx.config.maxRounds) {\n return {\n verdict: \"stop\",\n reason: `Round cap reached (${ctx.round}/${ctx.config.maxRounds})`,\n strategy: \"round-cap\",\n summary: `Review stopped at round cap (${ctx.config.maxRounds})`,\n };\n }\n return null;\n}\n\nfunction evalRepetition(ctx: TerminationContext): TerminationResult | null {\n if (ctx.rounds.length < 2) return null;\n\n const latest = ctx.rounds[ctx.rounds.length - 1];\n if (!latest) return null;\n\n let count = 0;\n for (const round of ctx.rounds) {\n if (round.findingHash === latest.findingHash) count++;\n }\n\n if (count >= ctx.config.repetitionThreshold) {\n return {\n verdict: \"stop\",\n reason: `Same finding hash repeated ${count} times (threshold: ${ctx.config.repetitionThreshold})`,\n strategy: \"repetition-detection\",\n summary: `Review going in circles — same findings repeated ${count}x`,\n };\n }\n\n return null;\n}\n\nfunction evalQualityThreshold(\n ctx: TerminationContext,\n): TerminationResult | null {\n if (ctx.rounds.length === 0) return null;\n\n const latest = ctx.rounds[ctx.rounds.length - 1];\n if (!latest) return null;\n\n // Guard: if parser extracted nothing at all (0 findings + 0 diff lines),\n // treat as inconclusive — not \"clean\". The parser may have failed to\n // extract from malformed output.\n if (\n latest.findingCount === 0 &&\n latest.suggestedDiffLines === 0 &&\n latest.findings.length === 0\n ) {\n return null; // inconclusive — continue to next strategy\n }\n\n const significantFindings = latest.findings.filter((f) =>\n isAtOrAbove(f.severity, ctx.config.qualitySeverityFloor),\n );\n\n if (significantFindings.length === 0) {\n return {\n verdict: \"stop\",\n reason: `No findings at ${ctx.config.qualitySeverityFloor}+ severity in round ${ctx.round}`,\n strategy: \"quality-threshold\",\n summary: `Review clean — no ${ctx.config.qualitySeverityFloor}+ findings in round ${ctx.round}`,\n };\n }\n\n return null;\n}\n\nfunction evalDiffInsignificance(\n ctx: TerminationContext,\n): TerminationResult | null {\n if (ctx.rounds.length === 0) return null;\n\n const latest = ctx.rounds[ctx.rounds.length - 1];\n if (!latest) return null;\n\n // Guard: same as quality-threshold — empty output is inconclusive, not trivial\n if (\n latest.findingCount === 0 &&\n latest.suggestedDiffLines === 0 &&\n latest.findings.length === 0\n ) {\n return null;\n }\n\n if (latest.suggestedDiffLines < ctx.config.diffThreshold) {\n return {\n verdict: \"stop\",\n reason: `Suggested diff (${latest.suggestedDiffLines} lines) below threshold (${ctx.config.diffThreshold})`,\n strategy: \"diff-insignificance\",\n summary: `Review suggestions are trivial (${latest.suggestedDiffLines} lines)`,\n };\n }\n\n return null;\n}\n\nconst STRATEGY_EVALUATORS: Record<\n TerminationStrategy,\n (ctx: TerminationContext) => TerminationResult | null\n> = {\n \"manual-stop\": evalManualStop,\n \"round-cap\": evalRoundCap,\n \"repetition-detection\": evalRepetition,\n \"quality-threshold\": evalQualityThreshold,\n \"diff-insignificance\": evalDiffInsignificance,\n};\n\n// ── Main evaluator ─────────────────────────────────────────────────\n\nexport function evaluate(ctx: TerminationContext): TerminationResult {\n for (const strategy of ctx.config.strategies) {\n const evaluator = STRATEGY_EVALUATORS[strategy];\n if (!evaluator) continue;\n\n const result = evaluator(ctx);\n if (result) return result;\n }\n\n return {\n verdict: \"continue\",\n reason: \"All strategies passed — review continues\",\n strategy:\n ctx.config.strategies[ctx.config.strategies.length - 1] ?? \"round-cap\",\n summary: `Round ${ctx.round} complete, continuing`,\n };\n}\n","/**\n * Review engine — detects review requests, builds prompts, parses output.\n *\n * This module handles the \"what\" of review sessions.\n * The termination engine handles the \"when to stop.\"\n * The bridge handles the \"how to deliver.\"\n */\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport * as crypto from \"node:crypto\";\nimport type {\n ReviewFinding,\n ReviewRound,\n FindingSeverity,\n TerminationConfig,\n} from \"./termination.js\";\nimport { computeFindingHash } from \"./termination.js\";\n\n// ── Types ──────────────────────────────────────────────────────────\n\nexport type AgentRole = \"reviewer\" | \"validator\" | \"long-running\";\n\nexport interface ReviewRequest {\n sourcePath: string;\n sender: string;\n recipient: string;\n prNumber: number;\n branch?: string;\n generation: string;\n isReReview: boolean;\n round: number;\n}\n\nexport interface ReviewSession {\n request: ReviewRequest;\n agentName: string;\n role: AgentRole;\n rounds: ReviewRound[];\n startedAt: string;\n terminatedAt?: string;\n reviewFilePath: string;\n}\n\nexport interface ReviewEngineConfig {\n role: AgentRole;\n generation: string;\n commsDir: string;\n repoRoot: string;\n agentName: string;\n termination: TerminationConfig;\n}\n\nexport interface HeadlessConfig {\n enabled: boolean;\n role: AgentRole;\n termination: TerminationConfig;\n}\n\n// ── Request Detection ──────────────────────────────────────────────\n\nconst REVIEW_KEYWORDS = [/리뷰\\s*요청/, /review[- ]?request/i];\n\nconst REREVIEW_KEYWORDS = [/재리뷰/, /re-?review/i];\n\nconst PR_NUMBER_PATTERNS = [\n /PR\\s*#?\\s*(\\d+)/i,\n /pull\\/(\\d+)/,\n /review[-_ ]?(\\d+)/i,\n];\n\n/**\n * Parse inbox filename to extract routing info.\n * Format: YYYYMMDD-sender-recipient-subject.md\n */\nexport function parseInboxFilename(filename: string): {\n date: string;\n sender: string;\n recipient: string;\n subject: string;\n} | null {\n const base = path.basename(filename, \".md\");\n const match = base.match(/^(\\d{8})-([^-]+)-([^-]+)-(.+)$/);\n if (!match) return null;\n\n return {\n date: match[1],\n sender: match[2],\n recipient: match[3],\n subject: match[4],\n };\n}\n\n/**\n * Extract PR number from text content.\n */\nexport function extractPrNumber(text: string): number | null {\n for (const pattern of PR_NUMBER_PATTERNS) {\n const match = text.match(pattern);\n if (match?.[1]) return parseInt(match[1], 10);\n }\n return null;\n}\n\n/**\n * Detect if a file represents a review request.\n * Returns a ReviewRequest if detected, null otherwise.\n */\nexport function detectReviewRequest(\n filePath: string,\n content: string,\n generation: string,\n): ReviewRequest | null {\n const parsed = parseInboxFilename(filePath);\n if (!parsed) return null;\n\n const fullText = `${parsed.subject} ${content}`;\n\n // Check for review keywords\n const isReview = REVIEW_KEYWORDS.some((re) => re.test(fullText));\n const isReReview = REREVIEW_KEYWORDS.some((re) => re.test(fullText));\n\n if (!isReview && !isReReview) return null;\n\n // Extract PR number\n const prNumber = extractPrNumber(fullText);\n if (!prNumber) return null;\n\n return {\n sourcePath: filePath,\n sender: parsed.sender,\n recipient: parsed.recipient,\n prNumber,\n generation,\n isReReview,\n round: isReReview ? 2 : 1, // Will be adjusted by session tracking\n };\n}\n\n// ── Review Prompt ──────────────────────────────────────────────────\n\nexport function buildReviewPrompt(\n request: ReviewRequest,\n agentName: string,\n round: number,\n): string {\n const roundLabel = round > 1 ? ` (re-review round ${round})` : \"\";\n\n return [\n `You are a code reviewer for the HUA Platform monorepo.`,\n ``,\n `## Task`,\n `Review PR #${request.prNumber}${roundLabel}.`,\n ``,\n `## Instructions`,\n `1. Run: gh pr diff ${request.prNumber}`,\n `2. Read changed files for understanding`,\n `3. Apply review checklist: security > data integrity > performance > error handling > code quality`,\n `4. Write structured findings`,\n ``,\n `## Output`,\n `Write review to: ${path.join(\"reviews\", request.generation, `review-PR${request.prNumber}-${agentName}.md`)}`,\n ``,\n `### Review File Format`,\n `\\`\\`\\`markdown`,\n `---`,\n `date: ${new Date().toISOString().split(\"T\")[0]}`,\n `reviewer: ${agentName}`,\n `pr: ${request.prNumber}`,\n `round: ${round}`,\n `status: clean | p1-Nitems | p2-Nitems`,\n `merge: merge | fix-then-merge | hold`,\n `---`,\n ``,\n `## Findings`,\n ``,\n `### Critical / High`,\n `- [severity] [category] file:line — description`,\n ``,\n `### Medium / Low`,\n `- [severity] [category] file:line — description`,\n ``,\n `## Checks`,\n `- [ ] Build verified`,\n `- [ ] Typecheck passed`,\n `- [ ] Scope check (only expected files changed)`,\n ``,\n `## Suggested Diff Lines`,\n `{number of lines the author should change to address findings}`,\n ``,\n `## Decision`,\n `{one-line merge recommendation}`,\n `\\`\\`\\``,\n ``,\n `## After Review`,\n `- Update reviews/INDEX.md`,\n `- Write inbox reply to ${request.sender}`,\n `- Commit and push comms changes`,\n ].join(\"\\n\");\n}\n\n// ── Review Output Parsing ──────────────────────────────────────────\n\nconst SEVERITY_PATTERNS: Record<FindingSeverity, RegExp> = {\n critical: /\\bcritical\\b/i,\n high: /\\bhigh\\b/i,\n medium: /\\bmedium\\b/i,\n low: /\\blow\\b/i,\n nitpick: /\\bnitpick\\b/i,\n};\n\nconst CATEGORY_PATTERNS = [\n \"security\",\n \"performance\",\n \"correctness\",\n \"data-integrity\",\n \"error-handling\",\n \"code-quality\",\n \"style\",\n];\n\n/**\n * Parse frontmatter from review file.\n */\nexport function parseFrontmatter(\n content: string,\n): Record<string, string> | null {\n const match = content.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (!match?.[1]) return null;\n\n const fields: Record<string, string> = {};\n for (const line of match[1].split(\"\\n\")) {\n const kv = line.match(/^(\\w+):\\s*(.+)$/);\n if (kv?.[1] && kv[2]) {\n fields[kv[1]] = kv[2].trim();\n }\n }\n return fields;\n}\n\n/**\n * Extract suggested diff lines from review content.\n */\nexport function extractSuggestedDiffLines(content: string): number {\n const match = content.match(/## Suggested Diff Lines\\s*\\n\\s*(\\d+)/i);\n if (match?.[1]) return parseInt(match[1], 10);\n\n // Fallback: count lines in code blocks that look like suggestions\n const codeBlocks = content.match(/```[\\s\\S]*?```/g) ?? [];\n let totalLines = 0;\n for (const block of codeBlocks) {\n totalLines += block.split(\"\\n\").length - 2; // minus fences\n }\n return totalLines;\n}\n\n/**\n * Extract findings from review content.\n * Best-effort parsing — reviews may not follow exact format.\n */\nexport function extractFindings(content: string): ReviewFinding[] {\n const findings: ReviewFinding[] = [];\n\n // Match lines that look like finding entries\n const lines = content.split(\"\\n\");\n for (const line of lines) {\n const trimmed = line.trim();\n if (!trimmed.startsWith(\"-\") && !trimmed.startsWith(\"*\")) continue;\n\n // Detect severity\n let severity: FindingSeverity = \"medium\";\n for (const [sev, pattern] of Object.entries(SEVERITY_PATTERNS)) {\n if (pattern.test(trimmed)) {\n severity = sev as FindingSeverity;\n break;\n }\n }\n\n // Detect category\n let category = \"general\";\n for (const cat of CATEGORY_PATTERNS) {\n if (trimmed.toLowerCase().includes(cat)) {\n category = cat;\n break;\n }\n }\n\n // Extract file:line if present\n const fileMatch = trimmed.match(/([a-zA-Z0-9_/.-]+\\.[a-zA-Z]+):(\\d+)/);\n\n // Only include if it looks like an actual finding (has severity keyword or file ref)\n const hasSeverityKeyword = Object.values(SEVERITY_PATTERNS).some((p) =>\n p.test(trimmed),\n );\n if (hasSeverityKeyword || fileMatch) {\n findings.push({\n severity,\n category,\n description: trimmed.replace(/^[-*]\\s*/, \"\").slice(0, 200),\n file: fileMatch?.[1],\n line: fileMatch?.[2] ? parseInt(fileMatch[2], 10) : undefined,\n });\n }\n }\n\n return findings;\n}\n\n/**\n * Parse a review output file into a ReviewRound.\n */\nexport function parseReviewOutput(\n reviewFilePath: string,\n round: number,\n): ReviewRound | null {\n if (!fs.existsSync(reviewFilePath)) return null;\n\n const content = fs.readFileSync(reviewFilePath, \"utf-8\");\n const findings = extractFindings(content);\n const suggestedDiffLines = extractSuggestedDiffLines(content);\n\n return {\n round,\n timestamp: new Date().toISOString(),\n findingCount: findings.length,\n findings,\n suggestedDiffLines,\n findingHash: computeFindingHash(findings),\n };\n}\n\n// ── Review File Path ───────────────────────────────────────────────\n\nexport function reviewFilePath(\n commsDir: string,\n generation: string,\n prNumber: number,\n agentName: string,\n): string {\n return path.join(\n commsDir,\n \"reviews\",\n generation,\n `review-PR${prNumber}-${agentName}.md`,\n );\n}\n\n// ── Stale Detection ────────────────────────────────────────────────\n\n/**\n * Check if a review request is stale (already handled).\n * Mirrors PS1 Test-IsStaleRequest logic.\n */\nexport function isStaleReviewRequest(\n request: ReviewRequest,\n commsDir: string,\n agentName: string,\n): boolean {\n // 1. Check if review file exists and is newer than request\n const revPath = reviewFilePath(\n commsDir,\n request.generation,\n request.prNumber,\n agentName,\n );\n if (fs.existsSync(revPath) && fs.existsSync(request.sourcePath)) {\n const reviewStat = fs.statSync(revPath);\n const requestStat = fs.statSync(request.sourcePath);\n if (reviewStat.mtimeMs > requestStat.mtimeMs) return true;\n }\n\n return false;\n}\n\n// ── Processed Marker ───────────────────────────────────────────────\n\nexport function computeRequestMarkerId(filePath: string): string {\n const stat = fs.statSync(filePath);\n const input = `${filePath}|${stat.mtimeMs}`;\n return crypto.createHash(\"sha1\").update(input).digest(\"hex\");\n}\n\nexport function isAlreadyProcessed(\n stateDir: string,\n filePath: string,\n): boolean {\n const markerId = computeRequestMarkerId(filePath);\n return fs.existsSync(path.join(stateDir, \"processed\", `${markerId}.done`));\n}\n\nexport function unmarkProcessed(\n stateDir: string,\n request: ReviewRequest,\n): void {\n const markerId = computeRequestMarkerId(request.sourcePath);\n const markerPath = path.join(stateDir, \"processed\", `${markerId}.done`);\n if (fs.existsSync(markerPath)) {\n fs.unlinkSync(markerPath);\n }\n}\n\nexport function markAsProcessed(\n stateDir: string,\n request: ReviewRequest,\n): void {\n const markerId = computeRequestMarkerId(request.sourcePath);\n const markerDir = path.join(stateDir, \"processed\");\n fs.mkdirSync(markerDir, { recursive: true });\n const markerPath = path.join(markerDir, `${markerId}.done`);\n const payload = {\n prNumber: request.prNumber,\n sourcePath: request.sourcePath,\n processedAt: new Date().toISOString(),\n };\n const tmp = `${markerPath}.tmp.${process.pid}`;\n fs.writeFileSync(tmp, JSON.stringify(payload, null, 2), \"utf-8\");\n fs.renameSync(tmp, markerPath);\n}\n\n// ── Bridge Receipt ─────────────────────────────────────────────────\n\n/**\n * Write immediate inbox acknowledgment before review starts.\n * Mirrors PS1 Write-BridgeReceipt pattern.\n */\nexport function writeReviewReceipt(\n commsDir: string,\n request: ReviewRequest,\n agentName: string,\n): string {\n const date = new Date().toISOString().split(\"T\")[0].replace(/-/g, \"\");\n const filename = `${date}-${agentName}-${request.sender}-PR${request.prNumber}-ack.md`;\n const content = [\n `## ${agentName} > ${request.sender}`,\n ``,\n `- PR #${request.prNumber} review request received.`,\n `- headless reviewer processing.`,\n `- request: ${path.basename(request.sourcePath)}`,\n ].join(\"\\n\");\n\n const inboxDir = path.join(commsDir, \"inbox\");\n fs.mkdirSync(inboxDir, { recursive: true });\n const inboxPath = path.join(inboxDir, filename);\n const tmp = `${inboxPath}.tmp.${process.pid}`;\n fs.writeFileSync(tmp, content, \"utf-8\");\n fs.renameSync(tmp, inboxPath);\n return inboxPath;\n}\n\n// ── Orchestrator Entry Point ───────────────────────────────────\n\n/**\n * Check if the current bridge process is running in headless reviewer mode.\n * Reads from env vars set by engine/bridge.ts startBridge().\n */\nexport function isHeadlessReviewer(): boolean {\n return process.env.TAP_HEADLESS === \"true\";\n}\n\n/**\n * Get headless reviewer configuration from env vars.\n * Returns null if not in headless mode.\n */\nexport function getHeadlessEnvConfig(): {\n role: string;\n maxRounds: number;\n qualityFloor: string;\n} | null {\n if (!isHeadlessReviewer()) return null;\n return {\n role: process.env.TAP_AGENT_ROLE ?? \"reviewer\",\n maxRounds: parseInt(process.env.TAP_MAX_REVIEW_ROUNDS ?? \"5\", 10),\n qualityFloor: process.env.TAP_QUALITY_FLOOR ?? \"high\",\n };\n}\n\n/**\n * Scan inbox for pending review requests.\n * This is the entry point for the headless review loop.\n *\n * Phase 3 will wire this into the bridge runner's poll cycle:\n * 1. scanInboxForReviews() → detect pending requests\n * 2. For each: writeReviewReceipt() → dispatch to bridge → parseReviewOutput()\n * 3. evaluate() termination → continue or stop\n */\nexport function scanInboxForReviews(\n commsDir: string,\n stateDir: string,\n generation: string,\n agentName: string,\n): ReviewRequest[] {\n const inboxDir = path.join(commsDir, \"inbox\");\n if (!fs.existsSync(inboxDir)) return [];\n\n const files = fs.readdirSync(inboxDir).filter((f) => f.endsWith(\".md\"));\n const requests: ReviewRequest[] = [];\n\n for (const file of files) {\n const filePath = path.join(inboxDir, file);\n const content = fs.readFileSync(filePath, \"utf-8\");\n const request = detectReviewRequest(filePath, content, generation);\n\n if (!request) continue;\n\n // Only process requests addressed to this agent or broadcast (\"전체\"/\"all\")\n const to = request.recipient.toLowerCase();\n if (\n to !== agentName.toLowerCase() &&\n to !== \"전체\" &&\n to !== \"all\" &&\n to !== \"\"\n ) {\n continue;\n }\n\n if (isStaleReviewRequest(request, commsDir, agentName)) continue;\n if (isAlreadyProcessed(stateDir, filePath)) continue;\n\n requests.push(request);\n }\n\n return requests;\n}\n","/**\n * Headless review loop — poll-based review orchestrator for bridge processes.\n *\n * Runs alongside the bridge script. When TAP_HEADLESS=true:\n * 1. Periodically scans inbox for review requests\n * 2. Writes review dispatch files that the bridge picks up\n * 3. Monitors review output for completion\n * 4. Evaluates termination conditions\n * 5. Continues or stops the review session\n *\n * This is a control loop, not a WebSocket client — the bridge handles dispatch.\n */\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport {\n scanInboxForReviews,\n isHeadlessReviewer,\n getHeadlessEnvConfig,\n buildReviewPrompt,\n writeReviewReceipt,\n parseReviewOutput,\n reviewFilePath,\n markAsProcessed,\n unmarkProcessed,\n type ReviewRequest,\n type ReviewSession,\n} from \"./review.js\";\nimport {\n evaluate,\n DEFAULT_TERMINATION_CONFIG,\n type TerminationContext,\n type TerminationConfig,\n type FindingSeverity,\n} from \"./termination.js\";\n\n// ── Types ──────────────────────────────────────────────────────────\n\nexport interface HeadlessLoopOptions {\n commsDir: string;\n stateDir: string;\n repoRoot: string;\n agentName: string;\n generation: string;\n pollIntervalMs: number;\n}\n\nexport interface HeadlessLoopState {\n running: boolean;\n activeSession: ReviewSession | null;\n completedSessions: number;\n lastPollAt: string | null;\n}\n\n// ── Loop implementation ────────────────────────────────────────────\n\nexport function createHeadlessLoop(options: HeadlessLoopOptions): {\n start: () => void;\n stop: () => void;\n getState: () => HeadlessLoopState;\n} {\n const envConfig = getHeadlessEnvConfig();\n const terminationConfig: TerminationConfig = {\n ...DEFAULT_TERMINATION_CONFIG,\n maxRounds: envConfig?.maxRounds ?? DEFAULT_TERMINATION_CONFIG.maxRounds,\n qualitySeverityFloor:\n (envConfig?.qualityFloor as FindingSeverity) ??\n DEFAULT_TERMINATION_CONFIG.qualitySeverityFloor,\n };\n\n const state: HeadlessLoopState = {\n running: false,\n activeSession: null,\n completedSessions: 0,\n lastPollAt: null,\n };\n\n let timer: ReturnType<typeof setInterval> | null = null;\n\n function log(msg: string): void {\n const ts = new Date().toISOString();\n console.error(`[${ts}] [headless-loop] ${msg}`);\n }\n\n function pollOnce(): void {\n state.lastPollAt = new Date().toISOString();\n\n // Skip if already processing a review\n if (state.activeSession) {\n checkActiveSession();\n return;\n }\n\n // Scan for new review requests\n const requests = scanInboxForReviews(\n options.commsDir,\n options.stateDir,\n options.generation,\n options.agentName,\n );\n\n if (requests.length === 0) return;\n\n // Process first request (sequential — one at a time)\n const request = requests[0];\n startReviewSession(request);\n }\n\n function startReviewSession(request: ReviewRequest): void {\n log(`Starting review for PR #${request.prNumber}`);\n\n // Mark as processed EAGERLY to prevent race with generic bridge.\n // If anything fails after this point, we roll back the marker.\n markAsProcessed(options.stateDir, request);\n\n try {\n // Write receipt\n writeReviewReceipt(options.commsDir, request, options.agentName);\n\n // Build review prompt\n const prompt = buildReviewPrompt(request, options.agentName, 1);\n\n // Write dispatch file to commsDir/inbox/ — the bridge watches this\n // directory and will inject it as a turn/start\n const date = new Date().toISOString().split(\"T\")[0].replace(/-/g, \"\");\n const dispatchFilename = `${date}-headless-${options.agentName}-review-PR${request.prNumber}.md`;\n const inboxDir = path.join(options.commsDir, \"inbox\");\n fs.mkdirSync(inboxDir, { recursive: true });\n const dispatchFile = path.join(inboxDir, dispatchFilename);\n const tmp = `${dispatchFile}.tmp.${process.pid}`;\n fs.writeFileSync(tmp, prompt, \"utf-8\");\n fs.renameSync(tmp, dispatchFile);\n\n state.activeSession = {\n request,\n agentName: options.agentName,\n role:\n (envConfig?.role as \"reviewer\" | \"validator\" | \"long-running\") ??\n \"reviewer\",\n rounds: [],\n startedAt: new Date().toISOString(),\n reviewFilePath: reviewFilePath(\n options.commsDir,\n request.generation,\n request.prNumber,\n options.agentName,\n ),\n };\n\n log(`Dispatched review prompt for PR #${request.prNumber} (round 1)`);\n } catch (err) {\n // Roll back processed marker so request can be retried on next poll\n log(\n `Failed to start review for PR #${request.prNumber}: ${err instanceof Error ? err.message : String(err)}`,\n );\n unmarkProcessed(options.stateDir, request);\n }\n }\n\n function checkActiveSession(): void {\n if (!state.activeSession) return;\n\n const session = state.activeSession;\n const revPath = session.reviewFilePath;\n\n // Check if review output file has been updated since last check\n if (!fs.existsSync(revPath)) return;\n\n const stat = fs.statSync(revPath);\n const lastRound = session.rounds[session.rounds.length - 1];\n const lastCheck = lastRound?.timestamp ?? session.startedAt;\n\n // Only process if file is newer than our last check\n if (stat.mtime.toISOString() <= lastCheck) return;\n\n // Parse the review output\n const roundNum = session.rounds.length + 1;\n const round = parseReviewOutput(revPath, roundNum);\n if (!round) return;\n\n session.rounds.push(round);\n log(\n `PR #${session.request.prNumber} round ${roundNum}: ${round.findingCount} findings, ${round.suggestedDiffLines} suggested diff lines`,\n );\n\n // Evaluate termination\n const stopSignalPath = path.join(options.stateDir, \"stop-signal\");\n const ctx: TerminationContext = {\n round: roundNum,\n rounds: session.rounds,\n stopSignalPath,\n config: terminationConfig,\n };\n\n const result = evaluate(ctx);\n\n if (result.verdict === \"stop\") {\n log(\n `PR #${session.request.prNumber} terminated: ${result.reason} (${result.strategy})`,\n );\n completeSession(session);\n } else {\n log(`PR #${session.request.prNumber} continues to round ${roundNum + 1}`);\n dispatchFollowUp(session, roundNum + 1);\n }\n }\n\n function dispatchFollowUp(session: ReviewSession, round: number): void {\n const prompt = buildReviewPrompt(session.request, options.agentName, round);\n\n // Write follow-up dispatch to commsDir/inbox/ for bridge to steer\n const date = new Date().toISOString().split(\"T\")[0].replace(/-/g, \"\");\n const dispatchFilename = `${date}-headless-${options.agentName}-review-PR${session.request.prNumber}-r${round}.md`;\n const inboxDir = path.join(options.commsDir, \"inbox\");\n fs.mkdirSync(inboxDir, { recursive: true });\n const dispatchFile = path.join(inboxDir, dispatchFilename);\n const tmp = `${dispatchFile}.tmp.${process.pid}`;\n fs.writeFileSync(tmp, prompt, \"utf-8\");\n fs.renameSync(tmp, dispatchFile);\n }\n\n function completeSession(session: ReviewSession): void {\n session.terminatedAt = new Date().toISOString();\n\n // Note: request was already marked as processed eagerly in startReviewSession()\n\n // Clean up dispatch files from inbox\n const inboxDir = path.join(options.commsDir, \"inbox\");\n if (fs.existsSync(inboxDir)) {\n const prefix = `headless-${options.agentName}-review-PR${session.request.prNumber}`;\n const files = fs.readdirSync(inboxDir).filter((f) => f.includes(prefix));\n for (const f of files) {\n fs.unlinkSync(path.join(inboxDir, f));\n }\n }\n\n state.activeSession = null;\n state.completedSessions++;\n log(\n `PR #${session.request.prNumber} review complete (${session.rounds.length} rounds)`,\n );\n }\n\n return {\n start() {\n if (!isHeadlessReviewer()) {\n log(\"Not in headless mode — loop not started\");\n return;\n }\n\n state.running = true;\n log(\n `Headless review loop started (${envConfig?.role ?? \"reviewer\"}, poll ${options.pollIntervalMs}ms, max ${terminationConfig.maxRounds} rounds)`,\n );\n\n // Initial poll\n pollOnce();\n\n // Set up interval\n timer = setInterval(pollOnce, options.pollIntervalMs);\n },\n\n stop() {\n state.running = false;\n if (timer) {\n clearInterval(timer);\n timer = null;\n }\n log(\"Headless review loop stopped\");\n },\n\n getState() {\n return { ...state };\n },\n };\n}\n","import * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport { spawn } from \"node:child_process\";\nimport { fileURLToPath } from \"node:url\";\nimport {\n resolveConfig,\n SHARED_CONFIG_FILE,\n LOCAL_CONFIG_FILE,\n} from \"../config/index.js\";\nimport { resolveNodeRuntime, buildRuntimeEnv } from \"../runtime/index.js\";\n\n// ─── Repo root discovery (fallback for unbundled runs) ─────────\n\nfunction findRepoRootFromRunner(): string | null {\n let dir = path.resolve(path.dirname(fileURLToPath(import.meta.url)));\n\n while (true) {\n if (fs.existsSync(path.join(dir, SHARED_CONFIG_FILE))) return dir;\n if (fs.existsSync(path.join(dir, LOCAL_CONFIG_FILE))) return dir;\n if (fs.existsSync(path.join(dir, \"scripts\", \"codex-app-server-bridge.ts\")))\n return dir;\n const parent = path.dirname(dir);\n if (parent === dir) return null;\n dir = parent;\n }\n}\n\n// ─── Headless review loop integration ──────────────────────────\n\nfunction maybeStartHeadlessLoop(\n repoRoot: string,\n commsDir: string,\n stateDir: string | undefined,\n): void {\n if (process.env.TAP_HEADLESS !== \"true\") return;\n\n // Dynamic import to avoid loading review/termination engines in non-headless mode\n import(\"../engine/headless-loop.js\")\n .then(({ createHeadlessLoop }) => {\n const agentName =\n process.env.TAP_AGENT_NAME ??\n process.env.CODEX_TAP_AGENT_NAME ??\n \"reviewer\";\n const generation = process.env.TAP_REVIEW_GENERATION ?? \"gen11\";\n const resolvedStateDir = stateDir ?? path.join(repoRoot, \".tap-comms\");\n\n const loop = createHeadlessLoop({\n commsDir,\n stateDir: resolvedStateDir,\n repoRoot,\n agentName,\n generation,\n pollIntervalMs: 3_000, // Poll faster than generic bridge (5s) for review priority\n });\n\n loop.start();\n\n // Clean shutdown\n process.on(\"SIGTERM\", () => loop.stop());\n process.on(\"SIGINT\", () => loop.stop());\n })\n .catch((err) => {\n console.error(\"[headless-loop] Failed to start:\", err);\n });\n}\n\n// ─── Main ──────────────────────────────────────────────────────\n\nasync function main(): Promise<void> {\n const repoRootHint = findRepoRootFromRunner() ?? undefined;\n const { config } = resolveConfig({}, repoRootHint);\n\n const repoRoot = config.repoRoot;\n const commsDir = config.commsDir;\n let appServerUrl = config.appServerUrl;\n\n // Multi-instance: override port from TAP_BRIDGE_PORT env var\n const instancePort = process.env.TAP_BRIDGE_PORT;\n if (instancePort) {\n try {\n const url = new URL(appServerUrl);\n url.port = instancePort;\n appServerUrl = url.toString().replace(/\\/$/, \"\");\n } catch {\n // Invalid URL — fall through to config default\n }\n }\n\n // Multi-instance: derive instance-specific state dir\n const instanceId = process.env.TAP_BRIDGE_INSTANCE_ID;\n const stateDir = instanceId\n ? path.join(repoRoot, \".tmp\", `codex-app-server-bridge-${instanceId}`)\n : undefined;\n\n // Honor pre-resolved node from parent (2-stage spawn: engine → runner → daemon)\n // TAP_STRIP_TYPES preserves metadata so bun doesn't get --experimental-strip-types.\n const preResolved = process.env.TAP_RESOLVED_NODE;\n const resolved = preResolved\n ? {\n command: preResolved,\n supportsStripTypes: process.env.TAP_STRIP_TYPES === \"1\",\n source: \"env\" as const,\n majorVersion: null,\n }\n : resolveNodeRuntime(config.runtimeCommand, repoRoot);\n\n const command = resolved.command;\n\n // Locate bridge script\n const scriptPath = path.join(\n repoRoot,\n \"scripts\",\n \"codex-app-server-bridge.ts\",\n );\n if (!fs.existsSync(scriptPath)) {\n throw new Error(\n `Bridge script not found: ${scriptPath}\\n` +\n `Ensure scripts/codex-app-server-bridge.ts exists in repo root.`,\n );\n }\n\n // Build args\n const args: string[] = [];\n if (resolved.supportsStripTypes) {\n args.push(\"--experimental-strip-types\");\n }\n args.push(\n scriptPath,\n `--repo-root=${repoRoot}`,\n `--comms-dir=${commsDir}`,\n `--app-server-url=${appServerUrl}`,\n );\n if (stateDir) {\n args.push(`--state-dir=${stateDir}`);\n }\n\n // Forward bridge operational flags from env (set by engine/bridge.ts)\n const busyMode = process.env.TAP_BUSY_MODE;\n if (busyMode) args.push(`--busy-mode=${busyMode}`);\n\n const pollSeconds = process.env.TAP_POLL_SECONDS;\n if (pollSeconds) args.push(`--poll-seconds=${pollSeconds}`);\n\n const reconnectSeconds = process.env.TAP_RECONNECT_SECONDS;\n if (reconnectSeconds) args.push(`--reconnect-seconds=${reconnectSeconds}`);\n\n const lookbackMinutes = process.env.TAP_MESSAGE_LOOKBACK_MINUTES;\n if (lookbackMinutes)\n args.push(`--message-lookback-minutes=${lookbackMinutes}`);\n\n const threadId = process.env.TAP_THREAD_ID;\n if (threadId) args.push(`--thread-id=${threadId}`);\n\n if (process.env.TAP_EPHEMERAL === \"true\") args.push(\"--ephemeral\");\n if (process.env.TAP_PROCESS_EXISTING === \"true\")\n args.push(\"--process-existing-messages\");\n\n // Spawn with fnm-aware PATH so any further child spawns also find the right Node\n const runtimeEnv = buildRuntimeEnv(repoRoot);\n\n const child = spawn(command, args, {\n cwd: repoRoot,\n env: runtimeEnv,\n stdio: \"inherit\",\n });\n\n // Start headless review loop if in headless mode\n maybeStartHeadlessLoop(repoRoot, commsDir, stateDir);\n\n child.on(\"exit\", (code: number | null, signal: NodeJS.Signals | null) => {\n if (signal) {\n process.kill(process.pid, signal);\n return;\n }\n process.exit(code ?? 0);\n });\n\n child.on(\"error\", (error: Error) => {\n console.error(String(error));\n process.exit(1);\n });\n}\n\nmain().catch((error) => {\n console.error(error instanceof Error ? error.message : String(error));\n process.exit(1);\n});\n","import * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type {\n TapSharedConfig,\n TapLocalConfig,\n TapResolvedConfig,\n ConfigSource,\n ConfigResolution,\n} from \"./types.js\";\n\n// ─── File names ────────────────────────────────────────────────\n\nexport const SHARED_CONFIG_FILE = \"tap-config.json\";\nexport const LOCAL_CONFIG_FILE = \"tap-config.local.json\";\n\n// ─── Defaults ──────────────────────────────────────────────────\n\nconst DEFAULT_RUNTIME_COMMAND = \"node\";\nconst DEFAULT_APP_SERVER_URL = \"ws://127.0.0.1:4501\";\n\n// ─── Repo root discovery ───────────────────────────────────────\n\nexport function findRepoRoot(startDir: string = process.cwd()): string {\n let dir = path.resolve(startDir);\n while (true) {\n if (fs.existsSync(path.join(dir, \".git\"))) return dir;\n if (fs.existsSync(path.join(dir, \"package.json\"))) return dir;\n const parent = path.dirname(dir);\n if (parent === dir) break;\n dir = parent;\n }\n return process.cwd();\n}\n\n// ─── JSON file loading ─────────────────────────────────────────\n\nfunction loadJsonFile<T>(filePath: string): T | null {\n if (!fs.existsSync(filePath)) return null;\n try {\n const raw = fs.readFileSync(filePath, \"utf-8\");\n return JSON.parse(raw) as T;\n } catch {\n return null;\n }\n}\n\nexport function loadSharedConfig(repoRoot: string): TapSharedConfig | null {\n return loadJsonFile<TapSharedConfig>(path.join(repoRoot, SHARED_CONFIG_FILE));\n}\n\nexport function loadLocalConfig(repoRoot: string): TapLocalConfig | null {\n return loadJsonFile<TapLocalConfig>(path.join(repoRoot, LOCAL_CONFIG_FILE));\n}\n\n// ─── CLI overrides ─────────────────────────────────────────────\n\nexport interface ConfigOverrides {\n commsDir?: string;\n stateDir?: string;\n runtimeCommand?: string;\n appServerUrl?: string;\n}\n\n// ─── Resolution ────────────────────────────────────────────────\n\n/**\n * Resolve config with priority: CLI flag > env > local config > shared config > auto.\n */\nexport function resolveConfig(\n overrides: ConfigOverrides = {},\n startDir?: string,\n): ConfigResolution {\n const repoRoot = findRepoRoot(startDir);\n const shared = loadSharedConfig(repoRoot) ?? {};\n const local = loadLocalConfig(repoRoot) ?? {};\n\n const sources: Record<keyof TapResolvedConfig, ConfigSource> = {\n repoRoot: \"auto\",\n commsDir: \"auto\",\n stateDir: \"auto\",\n runtimeCommand: \"auto\",\n appServerUrl: \"auto\",\n };\n\n // ─── commsDir ──────────────────────────────────────────────\n let commsDir: string;\n if (overrides.commsDir) {\n commsDir = path.resolve(overrides.commsDir);\n sources.commsDir = \"cli-flag\";\n } else if (process.env.TAP_COMMS_DIR) {\n commsDir = path.resolve(process.env.TAP_COMMS_DIR);\n sources.commsDir = \"env\";\n } else if (local.commsDir) {\n commsDir = resolvePath(repoRoot, local.commsDir);\n sources.commsDir = \"local-config\";\n } else if (shared.commsDir) {\n commsDir = resolvePath(repoRoot, shared.commsDir);\n sources.commsDir = \"shared-config\";\n } else {\n commsDir = path.join(path.dirname(repoRoot), \"tap-comms\");\n }\n\n // ─── stateDir ──────────────────────────────────────────────\n let stateDir: string;\n if (overrides.stateDir) {\n stateDir = path.resolve(overrides.stateDir);\n sources.stateDir = \"cli-flag\";\n } else if (process.env.TAP_STATE_DIR) {\n stateDir = path.resolve(process.env.TAP_STATE_DIR);\n sources.stateDir = \"env\";\n } else if (local.stateDir) {\n stateDir = resolvePath(repoRoot, local.stateDir);\n sources.stateDir = \"local-config\";\n } else if (shared.stateDir) {\n stateDir = resolvePath(repoRoot, shared.stateDir);\n sources.stateDir = \"shared-config\";\n } else {\n stateDir = path.join(repoRoot, \".tap-comms\");\n }\n\n // ─── runtimeCommand ────────────────────────────────────────\n let runtimeCommand: string;\n if (overrides.runtimeCommand) {\n runtimeCommand = overrides.runtimeCommand;\n sources.runtimeCommand = \"cli-flag\";\n } else if (process.env.TAP_RUNTIME_COMMAND) {\n runtimeCommand = process.env.TAP_RUNTIME_COMMAND;\n sources.runtimeCommand = \"env\";\n } else if (local.runtimeCommand) {\n runtimeCommand = local.runtimeCommand;\n sources.runtimeCommand = \"local-config\";\n } else if (shared.runtimeCommand) {\n runtimeCommand = shared.runtimeCommand;\n sources.runtimeCommand = \"shared-config\";\n } else {\n runtimeCommand = DEFAULT_RUNTIME_COMMAND;\n }\n\n // ─── appServerUrl ──────────────────────────────────────────\n let appServerUrl: string;\n if (overrides.appServerUrl) {\n appServerUrl = overrides.appServerUrl;\n sources.appServerUrl = \"cli-flag\";\n } else if (process.env.TAP_APP_SERVER_URL) {\n appServerUrl = process.env.TAP_APP_SERVER_URL;\n sources.appServerUrl = \"env\";\n } else if (local.appServerUrl) {\n appServerUrl = local.appServerUrl;\n sources.appServerUrl = \"local-config\";\n } else if (shared.appServerUrl) {\n appServerUrl = shared.appServerUrl;\n sources.appServerUrl = \"shared-config\";\n } else {\n appServerUrl = DEFAULT_APP_SERVER_URL;\n }\n\n return {\n config: { repoRoot, commsDir, stateDir, runtimeCommand, appServerUrl },\n sources,\n };\n}\n\n// ─── Save helpers ──────────────────────────────────────────────\n\nexport function saveSharedConfig(\n repoRoot: string,\n config: TapSharedConfig,\n): void {\n const filePath = path.join(repoRoot, SHARED_CONFIG_FILE);\n const tmp = `${filePath}.tmp.${process.pid}`;\n fs.writeFileSync(tmp, JSON.stringify(config, null, 2) + \"\\n\", \"utf-8\");\n fs.renameSync(tmp, filePath);\n}\n\nexport function saveLocalConfig(\n repoRoot: string,\n config: TapLocalConfig,\n): void {\n const filePath = path.join(repoRoot, LOCAL_CONFIG_FILE);\n const tmp = `${filePath}.tmp.${process.pid}`;\n fs.writeFileSync(tmp, JSON.stringify(config, null, 2) + \"\\n\", \"utf-8\");\n fs.renameSync(tmp, filePath);\n}\n\n// ─── Helpers ───────────────────────────────────────────────────\n\n/** Resolve a path relative to repoRoot, or keep absolute as-is. */\nfunction resolvePath(repoRoot: string, p: string): string {\n return path.isAbsolute(p) ? p : path.resolve(repoRoot, p);\n}\n","/**\n * Common Node.js runtime resolver for all tap-comms child processes.\n *\n * Resolution chain:\n * .node-version + fnm probe → configured command → tsx fallback\n *\n * Extracted from codex-bridge-runner.ts (M69) to share across:\n * - bridge engine spawn\n * - bridge runner spawn\n * - future CLI commands\n */\n\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport { execSync } from \"node:child_process\";\n\n// ─── Types ─────────────────────────────────────────────────────\n\nexport type RuntimeSource = \"fnm\" | \"config\" | \"path\" | \"tsx-fallback\" | \"bun\";\n\nexport interface ResolvedRuntime {\n /** Absolute path or command name for the resolved runtime. */\n command: string;\n /** Whether --experimental-strip-types is supported and should be used. */\n supportsStripTypes: boolean;\n /** Where the runtime was resolved from (for diagnostics). */\n source: RuntimeSource;\n /** Detected major version, if available. */\n majorVersion: number | null;\n}\n\n// ─── .node-version ─────────────────────────────────────────────\n\nexport function readNodeVersion(repoRoot: string): string | null {\n const nvFile = path.join(repoRoot, \".node-version\");\n if (!fs.existsSync(nvFile)) return null;\n try {\n const raw = fs.readFileSync(nvFile, \"utf-8\").trim();\n return raw.length > 0 ? raw.replace(/^v/, \"\") : null;\n } catch {\n return null;\n }\n}\n\n// ─── fnm probe ─────────────────────────────────────────────────\n\nfunction fnmCandidateDirs(): string[] {\n if (process.platform === \"win32\") {\n return [\n process.env.FNM_DIR,\n process.env.APPDATA ? path.join(process.env.APPDATA, \"fnm\") : null,\n process.env.LOCALAPPDATA\n ? path.join(process.env.LOCALAPPDATA, \"fnm\")\n : null,\n process.env.USERPROFILE\n ? path.join(process.env.USERPROFILE, \"scoop\", \"persist\", \"fnm\")\n : null,\n ].filter(Boolean) as string[];\n }\n // macOS / Linux\n return [\n process.env.FNM_DIR,\n process.env.HOME\n ? path.join(process.env.HOME, \".local\", \"share\", \"fnm\")\n : null,\n process.env.HOME ? path.join(process.env.HOME, \".fnm\") : null,\n process.env.XDG_DATA_HOME\n ? path.join(process.env.XDG_DATA_HOME, \"fnm\")\n : null,\n ].filter(Boolean) as string[];\n}\n\nfunction nodeExecutableName(): string {\n return process.platform === \"win32\" ? \"node.exe\" : \"node\";\n}\n\nexport function probeFnmNode(desiredVersion: string): string | null {\n const dirs = fnmCandidateDirs();\n const exe = nodeExecutableName();\n\n for (const baseDir of dirs) {\n const candidate = path.join(\n baseDir,\n \"node-versions\",\n `v${desiredVersion}`,\n \"installation\",\n exe,\n );\n if (!fs.existsSync(candidate)) continue;\n\n try {\n const v = execSync(`\"${candidate}\" --version`, {\n encoding: \"utf-8\",\n timeout: 5000,\n }).trim();\n if (v.startsWith(`v${desiredVersion.split(\".\")[0]}.`)) {\n return candidate;\n }\n } catch {\n // candidate exists but doesn't work — skip\n }\n }\n\n return null;\n}\n\n// ─── Version detection ─────────────────────────────────────────\n\nexport function detectNodeMajorVersion(command: string): number | null {\n try {\n const version = execSync(`\"${command}\" --version`, {\n encoding: \"utf-8\",\n timeout: 5000,\n }).trim();\n const match = version.match(/^v?(\\d+)\\./);\n return match ? parseInt(match[1], 10) : null;\n } catch {\n return null;\n }\n}\n\nexport function checkStripTypesSupport(command: string): boolean {\n const major = detectNodeMajorVersion(command);\n if (major !== null && major >= 22) return true;\n try {\n execSync(`\"${command}\" --experimental-strip-types -e \"\"`, {\n timeout: 5000,\n stdio: \"pipe\",\n });\n return true;\n } catch {\n return false;\n }\n}\n\n// ─── tsx fallback ──────────────────────────────────────────────\n\nexport function findTsxFallback(repoRoot: string): string | null {\n const candidates = [\n path.join(repoRoot, \"node_modules\", \".bin\", \"tsx.exe\"),\n path.join(repoRoot, \"node_modules\", \".bin\", \"tsx.CMD\"),\n path.join(repoRoot, \"node_modules\", \".bin\", \"tsx\"),\n ];\n for (const c of candidates) {\n if (fs.existsSync(c)) return c;\n }\n return null;\n}\n\n// ─── fnm bin directory (for PATH prepending) ───────────────────\n\n/**\n * Returns the directory containing the fnm-managed node binary,\n * suitable for prepending to PATH in child processes.\n */\nexport function getFnmBinDir(repoRoot: string): string | null {\n const desiredVersion = readNodeVersion(repoRoot);\n if (!desiredVersion) return null;\n\n const nodePath = probeFnmNode(desiredVersion);\n if (!nodePath) return null;\n\n return path.dirname(nodePath);\n}\n\n// ─── Main resolver ─────────────────────────────────────────────\n\n/**\n * Resolve the Node.js runtime to use for spawning child processes.\n *\n * Priority: bun passthrough → .node-version + fnm → configured command → tsx fallback\n */\nexport function resolveNodeRuntime(\n configCommand: string,\n repoRoot: string,\n): ResolvedRuntime {\n // Bun: native TS support, no strip-types needed\n if (configCommand === \"bun\" || configCommand.endsWith(\"bun.exe\")) {\n return {\n command: configCommand,\n supportsStripTypes: false,\n source: \"bun\",\n majorVersion: null,\n };\n }\n\n // .node-version + fnm discovery\n const desiredVersion = readNodeVersion(repoRoot);\n if (desiredVersion) {\n const fnmNode = probeFnmNode(desiredVersion);\n if (fnmNode) {\n const major = detectNodeMajorVersion(fnmNode);\n return {\n command: fnmNode,\n supportsStripTypes: checkStripTypesSupport(fnmNode),\n source: \"fnm\",\n majorVersion: major,\n };\n }\n }\n\n // Configured command (from config or PATH)\n const major = detectNodeMajorVersion(configCommand);\n if (major !== null) {\n return {\n command: configCommand,\n supportsStripTypes: checkStripTypesSupport(configCommand),\n source: major === detectNodeMajorVersion(\"node\") ? \"path\" : \"config\",\n majorVersion: major,\n };\n }\n\n // tsx fallback\n const tsx = findTsxFallback(repoRoot);\n if (tsx) {\n return {\n command: tsx,\n supportsStripTypes: false,\n source: \"tsx-fallback\",\n majorVersion: null,\n };\n }\n\n // Last resort\n return {\n command: configCommand,\n supportsStripTypes: false,\n source: \"path\",\n majorVersion: null,\n };\n}\n\n// ─── Env builder for child processes ───────────────────────────\n\n/**\n * Build an env object with fnm Node prepended to PATH.\n * Use this when spawning child processes that need the correct Node.\n */\nexport function buildRuntimeEnv(\n repoRoot: string,\n baseEnv: NodeJS.ProcessEnv = process.env,\n): NodeJS.ProcessEnv {\n const fnmBin = getFnmBinDir(repoRoot);\n if (!fnmBin) return { ...baseEnv };\n\n const pathKey = process.platform === \"win32\" ? \"Path\" : \"PATH\";\n const currentPath = baseEnv[pathKey] ?? baseEnv.PATH ?? \"\";\n\n return {\n ...baseEnv,\n [pathKey]: `${fnmBin}${path.delimiter}${currentPath}`,\n };\n}\n"],"mappings":"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","names":["fs","fs","path","crypto","reviewFilePath","fs","path","fs","path","fs","path","major","createHeadlessLoop"]}
package/dist/cli.d.mts ADDED
@@ -0,0 +1,2 @@
1
+
2
+ export { }