139 lines
4.4 KiB
TypeScript
139 lines
4.4 KiB
TypeScript
import { mkdirSync, writeFileSync } from "node:fs";
|
|
import { join } from "node:path";
|
|
import type { Role, RoleResult, StartStep, WorkflowMessage } from "@uncaged/nerve-core";
|
|
import { isDryRun, spawnSafe } from "@uncaged/nerve-workflow-utils";
|
|
import { cfgGet } from "../../lib/provider.js";
|
|
import { lastParseFromMessages, lastRepoFromMessages, resolveRepoCwd } from "../../lib/repo-context.js";
|
|
import { formatSpawnFailure } from "../../lib/spawn-utils.js";
|
|
|
|
export type PublishMeta = {
|
|
success: boolean;
|
|
};
|
|
|
|
export type BuildPublishDeps = {
|
|
nerveRoot: string;
|
|
};
|
|
|
|
function logPath(nerveRoot: string): string {
|
|
return join(nerveRoot, "logs", `solve-issue-publish-${Date.now()}.log`);
|
|
}
|
|
|
|
export function buildPublishRole({ nerveRoot }: BuildPublishDeps): Role<PublishMeta> {
|
|
return async (start: StartStep, messages: WorkflowMessage[]): Promise<RoleResult<PublishMeta>> => {
|
|
const dry = isDryRun(start);
|
|
const file = logPath(nerveRoot);
|
|
mkdirSync(join(file, ".."), { recursive: true });
|
|
|
|
const cwd = resolveRepoCwd(messages);
|
|
const parsed = lastParseFromMessages(messages);
|
|
const repoInfo = lastRepoFromMessages(messages);
|
|
|
|
if (dry) {
|
|
const msg = "[dry-run] publish skipped (no git push / PR)";
|
|
writeFileSync(file, `${msg}\n`, "utf-8");
|
|
return {
|
|
content: `[dry-run] publish skipped — log: ${file}`,
|
|
meta: { success: true },
|
|
};
|
|
}
|
|
|
|
if (cwd === null || parsed === null) {
|
|
writeFileSync(file, "missing cwd or SOLVE_ISSUE_PARSE\n", "utf-8");
|
|
return {
|
|
content: `publish failed: missing repo path or issue markers\nLog: ${file}`,
|
|
meta: { success: false },
|
|
};
|
|
}
|
|
|
|
const branchRun = await spawnSafe("git", ["rev-parse", "--abbrev-ref", "HEAD"], {
|
|
cwd,
|
|
env: null,
|
|
timeoutMs: 15_000,
|
|
});
|
|
if (!branchRun.ok) {
|
|
writeFileSync(file, formatSpawnFailure(branchRun.error), "utf-8");
|
|
return {
|
|
content: `publish failed: cannot read branch — ${formatSpawnFailure(branchRun.error)}`,
|
|
meta: { success: false },
|
|
};
|
|
}
|
|
const branch = branchRun.value.stdout.trim();
|
|
if (branch.length === 0) {
|
|
return { content: "publish failed: empty branch name", meta: { success: false } };
|
|
}
|
|
|
|
const pushRun = await spawnSafe("git", ["push", "-u", "origin", branch], {
|
|
cwd,
|
|
env: null,
|
|
timeoutMs: 180_000,
|
|
});
|
|
const lines: string[] = [];
|
|
if (!pushRun.ok) {
|
|
lines.push(`git push failed: ${formatSpawnFailure(pushRun.error)}`);
|
|
writeFileSync(file, lines.join("\n"), "utf-8");
|
|
return {
|
|
content: `publish: push failed\nLog: ${file}`,
|
|
meta: { success: false },
|
|
};
|
|
}
|
|
lines.push(`$ git push -u origin ${branch}`);
|
|
lines.push(pushRun.value.stdout);
|
|
lines.push(pushRun.value.stderr);
|
|
|
|
const token = process.env.GITEA_TOKEN ?? (await cfgGet(nerveRoot, "GITEA_TOKEN"));
|
|
if (token === null || token.length === 0) {
|
|
lines.push("missing GITEA_TOKEN");
|
|
writeFileSync(file, lines.join("\n"), "utf-8");
|
|
return {
|
|
content: `publish: push ok but no GITEA_TOKEN for PR API\nLog: ${file}`,
|
|
meta: { success: false },
|
|
};
|
|
}
|
|
|
|
const base = repoInfo?.defaultBranch ?? "main";
|
|
const title = `fix: issue #${parsed.number}`;
|
|
const body = [`Automated PR from solve-issue workflow.`, `Issue: https://${parsed.host}/${parsed.owner}/${parsed.repo}/issues/${parsed.number}`].join(
|
|
"\n",
|
|
);
|
|
const apiUrl = `https://${parsed.host}/api/v1/repos/${parsed.owner}/${parsed.repo}/pulls`;
|
|
|
|
let prOk = false;
|
|
let prBody = "";
|
|
try {
|
|
const res = await fetch(apiUrl, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
Authorization: `token ${token}`,
|
|
},
|
|
body: JSON.stringify({
|
|
title,
|
|
body,
|
|
head: branch,
|
|
base,
|
|
}),
|
|
});
|
|
prBody = await res.text();
|
|
prOk = res.ok;
|
|
lines.push(`POST ${apiUrl} -> ${res.status}`);
|
|
lines.push(prBody.slice(0, 4000));
|
|
} catch (e) {
|
|
lines.push(`fetch error: ${e instanceof Error ? e.message : String(e)}`);
|
|
}
|
|
|
|
writeFileSync(file, lines.join("\n"), "utf-8");
|
|
|
|
if (!prOk) {
|
|
return {
|
|
content: `publish: push ok but PR API failed\nLog: ${file}`,
|
|
meta: { success: false },
|
|
};
|
|
}
|
|
|
|
return {
|
|
content: `publish: pushed ${branch} and opened PR\nLog: ${file}`,
|
|
meta: { success: true },
|
|
};
|
|
};
|
|
}
|