use anyhow::Result;
use clap::ArgAction;
use clap_complete::engine::ArgValueCompleter;
use crate::commands::Run;
use crate::completions;
use crate::providers::{ReviewProvider, ReviewState, detect_provider, review_provider};
use crate::style;
use crate::{git, stack};
#[derive(Debug, clap::Args)]
pub struct Cleanup {
#[arg(add = ArgValueCompleter::new(completions::branch_candidates))]
branch: Option<String>,
#[arg(long, action = ArgAction::SetTrue)]
dry_run: bool,
#[arg(long, action = ArgAction::SetTrue)]
keep_branch: bool,
}
impl Run for Cleanup {
fn run(self) -> Result<()> {
cleanup(self.branch.as_deref(), self.dry_run, self.keep_branch)
}
}
pub fn cleanup(branch: Option<&str>, dry_run: bool, keep_branch: bool) -> Result<()> {
let branch = branch
.map(str::to_owned)
.map_or_else(git::current_branch, Ok)?;
let branches = stack::branch_and_descendants(&branch)?;
let current_branch = git::current_branch()?;
let local_branches = git::local_branches()?;
let provider = detect_provider()?;
let review_provider = review_provider(provider.kind);
let mut cleaned = 0;
let mut skipped = 0;
let mut retargeted = 0;
let branch_parents = stack::branch_parents(&branches)?;
crate::notes::update_stack_notes(review_provider.as_ref(), &branch_parents, dry_run)?;
for branch in branches {
retargeted +=
recover_deleted_parent(review_provider.as_ref(), &branch, &local_branches, dry_run)?;
let Some(review) = review_provider.review_for_branch_including_closed(&branch)? else {
anstream::println!(
"{}",
style::dim(&format!(
"skipped {branch}: no {} review found",
provider.kind
))
);
skipped += 1;
continue;
};
if review.state != ReviewState::Merged {
anstream::println!(
"{}",
style::dim(&format!(
"skipped {branch}: review {} is {}",
review.id, review.state
))
);
skipped += 1;
continue;
}
cleanup_merged_branch(review_provider.as_ref(), &branch, dry_run)?;
cleanup_branch_deletion(&branch, ¤t_branch, dry_run, !keep_branch)?;
cleaned += 1;
}
let retargeted_note = if retargeted > 0 {
format!(", {retargeted} retargeted")
} else {
String::new()
};
anstream::println!(
"{}",
style::success(&format!(
"cleanup complete: {cleaned} cleaned, {skipped} skipped{retargeted_note}"
))
);
Ok(())
}
fn recover_deleted_parent(
review_provider: &dyn ReviewProvider,
branch: &str,
local_branches: &[String],
dry_run: bool,
) -> Result<usize> {
let Some(parent) = stack::parent_for_branch(branch)? else {
return Ok(0);
};
if local_branches.contains(&parent) {
return Ok(0);
}
let Ok(Some(review)) = review_provider.review_for_branch(&parent) else {
return Ok(0);
};
if review.branch != parent
|| review.state != ReviewState::Merged
|| review.base == *branch
|| !local_branches.contains(&review.base)
{
return Ok(0);
}
anstream::println!(
"{}: parent {} is gone, but review {} merged into {}",
style::branch(branch),
style::branch(&parent),
review.id,
style::branch(&review.base)
);
anstream::println!(
"{} retarget {} -> {}",
if dry_run { "would" } else { "will" },
style::branch(branch),
style::branch(&review.base)
);
update_child_review_base(review_provider, branch, &review.base, dry_run)?;
if !dry_run {
stack::set_parent_for_branch(branch, &review.base)?;
}
Ok(1)
}
pub(crate) fn cleanup_merged_branch(
review_provider: &dyn ReviewProvider,
branch: &str,
dry_run: bool,
) -> Result<()> {
let parent = stack::parent_for_branch(branch)?;
let descendants = stack::branch_and_descendants(branch)?;
let direct_children: Vec<_> = descendants
.into_iter()
.skip(1)
.filter_map(|child| match stack::parent_for_branch(&child) {
Ok(Some(child_parent)) if child_parent == branch => Some(Ok(child)),
Ok(_) => None,
Err(error) => Some(Err(error)),
})
.collect::<Result<_>>()?;
for child in direct_children {
match parent.as_deref() {
Some(parent) => {
anstream::println!(
"{} retarget {} -> {}",
if dry_run { "would" } else { "will" },
style::branch(&child),
style::branch(parent)
);
update_child_review_base(review_provider, &child, parent, dry_run)?;
if !dry_run {
if let Ok(base) = git::merge_base(branch, &child) {
stack::set_base_for_branch(&child, &base)?;
}
stack::set_parent_for_branch(&child, parent)?;
}
}
None => {
anstream::println!(
"{} detach {}",
if dry_run { "would" } else { "will" },
style::branch(&child)
);
if !dry_run {
stack::unset_parent_for_branch(&child)?;
stack::unset_base_for_branch(&child)?;
}
}
}
}
anstream::println!(
"{} detach {}",
if dry_run { "would" } else { "will" },
style::branch(branch)
);
if !dry_run {
stack::unset_parent_for_branch(branch)?;
stack::unset_base_for_branch(branch)?;
}
Ok(())
}
pub(crate) fn cleanup_branch_deletion(
branch: &str,
current_branch: &str,
dry_run: bool,
delete_branch: bool,
) -> Result<()> {
if !delete_branch {
return Ok(());
}
if branch == current_branch {
anstream::println!(
"{}",
style::dim(&format!(
"kept {branch}: cannot delete the checked out branch"
))
);
return Ok(());
}
anstream::println!(
"{} delete branch {}",
if dry_run { "would" } else { "will" },
style::branch(branch)
);
if !dry_run {
git::delete_branch(branch)?;
}
Ok(())
}
fn update_child_review_base(
review_provider: &dyn ReviewProvider,
child: &str,
parent: &str,
dry_run: bool,
) -> Result<()> {
let Some(review) = review_provider.review_for_branch(child)? else {
return Ok(());
};
if review.state == ReviewState::Merged || review.base == parent {
return Ok(());
}
anstream::println!(
"{} update review {} -> {} {}",
if dry_run { "would" } else { "will" },
style::branch(&review.branch),
style::branch(parent),
style::dim(&format!("({})", review.id))
);
if !dry_run {
let output = review_provider.update_review_base(&review, parent)?;
if !output.is_empty() {
println!("{output}");
}
}
Ok(())
}