use anyhow::{Context as _, Result, bail};
use std::path::Path;
use std::process::Command;
use super::git_output_in;
use super::remote::detect_github_repo_in;
use super::tags::create_and_push_tag_in;
pub fn gh_api_get(endpoint: &str, token: Option<&str>) -> Result<serde_json::Value> {
gh_api_get_with_binary(Path::new("gh"), endpoint, token)
}
pub fn gh_api_get_with_binary(
gh_binary: &Path,
endpoint: &str,
token: Option<&str>,
) -> Result<serde_json::Value> {
let mut cmd = Command::new(gh_binary);
cmd.args(["api", endpoint]);
if let Some(tok) = token {
cmd.env("GITHUB_TOKEN", tok);
}
let output = cmd
.stdout(std::process::Stdio::piped())
.stderr(std::process::Stdio::piped())
.output()
.with_context(|| format!("failed to spawn gh CLI ({})", gh_binary.display()))?;
if !output.status.success() {
let stderr_raw = String::from_utf8_lossy(&output.stderr);
let raw = format!("gh api GET {} failed: {}", endpoint, stderr_raw.trim());
bail!("{}", redact_gh_stderr(&raw, token));
}
let stdout = String::from_utf8_lossy(&output.stdout);
serde_json::from_str(&stdout).context("failed to parse gh api response")
}
fn redact_gh_stderr(stderr: &str, token: Option<&str>) -> String {
let stripped = crate::redact::redact_url_credentials(stderr);
let mut env: Vec<(String, String)> = std::env::vars().collect();
if let Some(tok) = token
&& !tok.is_empty()
{
env.push(("GITHUB_TOKEN".to_string(), tok.to_string()));
}
crate::redact::string(&stripped, &env)
}
pub fn gh_api_get_paginated(endpoint: &str, token: Option<&str>) -> Result<Vec<serde_json::Value>> {
gh_api_get_paginated_with_binary(Path::new("gh"), endpoint, token)
}
pub fn gh_api_get_paginated_with_binary(
gh_binary: &Path,
endpoint: &str,
token: Option<&str>,
) -> Result<Vec<serde_json::Value>> {
let mut cmd = Command::new(gh_binary);
cmd.args(["api", "--paginate", endpoint]);
if let Some(tok) = token {
cmd.env("GITHUB_TOKEN", tok);
}
let output = cmd
.stdout(std::process::Stdio::piped())
.stderr(std::process::Stdio::piped())
.output()
.with_context(|| format!("failed to spawn gh CLI ({})", gh_binary.display()))?;
if !output.status.success() {
let stderr_raw = String::from_utf8_lossy(&output.stderr);
let raw = format!("gh api GET {} failed: {}", endpoint, stderr_raw.trim());
bail!("{}", redact_gh_stderr(&raw, token));
}
let stdout = String::from_utf8_lossy(&output.stdout);
if let Ok(serde_json::Value::Array(arr)) = serde_json::from_str::<serde_json::Value>(&stdout) {
return Ok(arr);
}
if let Ok(val) = serde_json::from_str::<serde_json::Value>(&stdout) {
return Ok(vec![val]);
}
let mut all_items = Vec::new();
for chunk in stdout.split_inclusive(']') {
let trimmed = chunk.trim();
if trimmed.is_empty() {
continue;
}
if let Ok(serde_json::Value::Array(arr)) =
serde_json::from_str::<serde_json::Value>(trimmed)
{
all_items.extend(arr);
} else if let Ok(val) = serde_json::from_str::<serde_json::Value>(trimmed) {
all_items.push(val);
} else {
tracing::warn!(
len = trimmed.len(),
chunk = ?&trimmed[..trimmed.len().min(200)],
"gh_api_get_paginated: failed to parse JSON chunk",
);
}
}
Ok(all_items)
}
fn gh_api_post_with_binary(
gh_binary: &Path,
endpoint: &str,
body: &serde_json::Value,
) -> Result<serde_json::Value> {
use std::io::Write;
let body_str = serde_json::to_string(body)?;
let mut child = Command::new(gh_binary)
.args(["api", "--method", "POST", endpoint, "--input", "-"])
.stdin(std::process::Stdio::piped())
.stdout(std::process::Stdio::piped())
.stderr(std::process::Stdio::piped())
.spawn()
.with_context(|| format!("failed to spawn gh CLI ({})", gh_binary.display()))?;
if let Some(ref mut stdin) = child.stdin {
stdin.write_all(body_str.as_bytes())?;
}
child.stdin.take();
let output = child.wait_with_output()?;
if !output.status.success() {
let stderr_raw = String::from_utf8_lossy(&output.stderr);
let raw = format!("gh api POST {} failed: {}", endpoint, stderr_raw.trim());
bail!("{}", crate::redact::redact_process_env(&raw));
}
let response: serde_json::Value = serde_json::from_slice(&output.stdout)
.with_context(|| format!("failed to parse GitHub API response from {}", endpoint))?;
Ok(response)
}
pub fn create_tag_via_github_api(
tag: &str,
message: &str,
dry_run: bool,
log: &crate::log::StageLogger,
strict: bool,
) -> Result<()> {
create_tag_via_github_api_in(
&std::env::current_dir()?,
Path::new("gh"),
tag,
message,
dry_run,
log,
strict,
)
}
#[allow(clippy::too_many_arguments)]
pub fn create_tag_via_github_api_in(
cwd: &Path,
gh_binary: &Path,
tag: &str,
message: &str,
dry_run: bool,
log: &crate::log::StageLogger,
strict: bool,
) -> Result<()> {
if dry_run {
log.status(&format!(
"(dry-run) would create tag via GitHub API: {} (\"{}\")",
tag, message
));
return Ok(());
}
let (owner, repo) = detect_github_repo_in(cwd)?;
let sha = git_output_in(cwd, &["rev-parse", "HEAD"])?;
let body = serde_json::json!({
"tag": tag,
"message": message,
"object": sha,
"type": "commit",
"tagger": {
"name": git_output_in(cwd, &["config", "user.name"]).unwrap_or_else(|_| "anodizer".to_string()),
"email": git_output_in(cwd, &["config", "user.email"]).unwrap_or_else(|_| "anodizer@users.noreply.github.com".to_string()),
"date": crate::sde::resolve_now().to_rfc3339(),
}
});
let tag_endpoint = format!("/repos/{owner}/{repo}/git/tags");
let response = match gh_api_post_with_binary(gh_binary, &tag_endpoint, &body) {
Ok(resp) => resp,
Err(e) => {
if e.to_string().contains("failed to spawn gh CLI") {
if strict {
anyhow::bail!(
"gh CLI not found, cannot create tag via GitHub API (strict mode)"
);
}
log.warn("gh CLI not found, falling back to local git tag + push");
return create_and_push_tag_in(cwd, tag, message, dry_run, log, strict);
}
return Err(e);
}
};
let tag_sha = response["sha"]
.as_str()
.ok_or_else(|| anyhow::anyhow!("GitHub API response missing 'sha' field"))?;
let ref_body = serde_json::json!({
"ref": format!("refs/tags/{}", tag),
"sha": tag_sha,
});
let ref_endpoint = format!("/repos/{owner}/{repo}/git/refs");
gh_api_post_with_binary(gh_binary, &ref_endpoint, &ref_body)?;
Ok(())
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn create_tag_dry_run_short_circuits() {
let log = crate::log::StageLogger::new("test", crate::log::Verbosity::Quiet);
let result = create_tag_via_github_api("v1.0.0", "msg", true, &log, false);
assert!(result.is_ok(), "dry-run must succeed: {result:?}");
}
#[test]
fn redact_gh_stderr_replaces_token_value() {
let secret = "ghp_abcdefghijklmnopqrstuvwxyz0123456789";
let stderr = format!("HTTP 401: token {secret} is invalid");
let redacted = redact_gh_stderr(&stderr, Some(secret));
assert!(
!redacted.contains(secret),
"token leaked into redacted output: {redacted}"
);
}
#[test]
fn redact_gh_stderr_with_no_token_still_strips_url_creds() {
let stderr = "auth failed: https://user:secret-pw@github.com/o/r.git rejected";
let redacted = redact_gh_stderr(stderr, None);
assert!(
!redacted.contains("secret-pw"),
"URL credential leaked: {redacted}"
);
}
#[test]
fn redact_gh_stderr_empty_token_is_noop_on_token_field() {
let stderr = "plain error message without credentials";
let redacted = redact_gh_stderr(stderr, Some(""));
assert_eq!(redacted, stderr);
}
#[test]
fn gh_api_get_with_binary_bails_when_binary_missing() {
let tmp = tempfile::tempdir().unwrap();
let missing = tmp.path().join("nonexistent-gh");
let err = gh_api_get_with_binary(&missing, "/repos/x/y", None)
.expect_err("missing binary must error");
let msg = err.to_string();
assert!(
msg.contains("spawn gh") || msg.contains(&missing.display().to_string()),
"expected actionable error mentioning spawn gh or the binary path, got: {msg}"
);
}
#[test]
fn gh_api_get_paginated_with_binary_bails_when_binary_missing() {
let tmp = tempfile::tempdir().unwrap();
let missing = tmp.path().join("nonexistent-gh");
let err = gh_api_get_paginated_with_binary(&missing, "/repos/x/y", None)
.expect_err("missing binary must error");
let msg = err.to_string();
assert!(
msg.contains("spawn gh") || msg.contains(&missing.display().to_string()),
"expected actionable error mentioning spawn gh or the binary path, got: {msg}"
);
}
#[test]
fn create_tag_via_github_api_in_bails_when_not_a_git_repo() {
if Command::new("git")
.arg("--version")
.output()
.map(|o| !o.status.success())
.unwrap_or(true)
{
return;
}
let tmp = tempfile::tempdir().unwrap();
let log = crate::log::StageLogger::new("test", crate::log::Verbosity::Quiet);
let err = create_tag_via_github_api_in(
tmp.path(),
Path::new("gh"),
"v1.0.0",
"msg",
false,
&log,
true,
)
.expect_err("non-git cwd must error");
let msg = err.to_string();
assert!(
msg.contains("git") || msg.contains("remote"),
"expected error to mention git or remote, got: {msg}"
);
}
#[test]
fn create_tag_via_github_api_in_dry_run_short_circuits() {
let tmp = tempfile::tempdir().unwrap();
let log = crate::log::StageLogger::new("test", crate::log::Verbosity::Quiet);
let result = create_tag_via_github_api_in(
tmp.path(),
Path::new("gh"),
"v1.0.0",
"msg",
true,
&log,
false,
);
assert!(result.is_ok(), "dry-run must succeed: {result:?}");
}
}