Skip to content

Commit

Permalink
cli: branch: let find_*_branches() return (name, target) pairs
Browse files Browse the repository at this point in the history
This will help inline view.remove_branch() in cmd_branch_forget(). I don't
care much about owned (String, _) vs (&str, _), but we can't simplify the
lifetime issue in find_forgettable_branches() anyway. So I made all callers
pass cloned Arc<ReadonlyRepo> and borrow (name, target) pairs from there.
  • Loading branch information
yuja committed Jun 26, 2024
1 parent 0f14ea8 commit 4ec170b
Show file tree
Hide file tree
Showing 4 changed files with 58 additions and 46 deletions.
19 changes: 13 additions & 6 deletions cli/src/commands/branch/delete.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
// See the License for the specific language governing permissions and
// limitations under the License.

use itertools::Itertools as _;
use jj_lib::op_store::RefTarget;
use jj_lib::str_util::StringPattern;

Expand Down Expand Up @@ -39,14 +40,20 @@ pub fn cmd_branch_delete(
args: &BranchDeleteArgs,
) -> Result<(), CommandError> {
let mut workspace_command = command.workspace_helper(ui)?;
let view = workspace_command.repo().view();
let names = find_local_branches(view, &args.names)?;
let repo = workspace_command.repo().clone();
let matched_branches = find_local_branches(repo.view(), &args.names)?;
let mut tx = workspace_command.start_transaction();
for branch_name in names.iter() {
for (name, _) in &matched_branches {
tx.mut_repo()
.set_local_branch_target(branch_name, RefTarget::absent());
.set_local_branch_target(name, RefTarget::absent());
}
tx.finish(ui, format!("delete branches {}", names.join(", ")))?;
writeln!(ui.status(), "Deleted {} branches.", names.len())?;
tx.finish(
ui,
format!(
"delete branches {}",
matched_branches.iter().map(|(name, _)| name).join(", ")
),
)?;
writeln!(ui.status(), "Deleted {} branches.", matched_branches.len())?;
Ok(())
}
30 changes: 18 additions & 12 deletions cli/src/commands/branch/forget.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@
// See the License for the specific language governing permissions and
// limitations under the License.

use itertools::Itertools as _;
use jj_lib::op_store::BranchTarget;
use jj_lib::str_util::StringPattern;
use jj_lib::view::View;

Expand Down Expand Up @@ -42,24 +44,28 @@ pub fn cmd_branch_forget(
args: &BranchForgetArgs,
) -> Result<(), CommandError> {
let mut workspace_command = command.workspace_helper(ui)?;
let view = workspace_command.repo().view();
let names = find_forgettable_branches(view, &args.names)?;
let repo = workspace_command.repo().clone();
let matched_branches = find_forgettable_branches(repo.view(), &args.names)?;
let mut tx = workspace_command.start_transaction();
for branch_name in names.iter() {
tx.mut_repo().remove_branch(branch_name);
for (name, _) in &matched_branches {
tx.mut_repo().remove_branch(name);
}
tx.finish(ui, format!("forget branches {}", names.join(", ")))?;
writeln!(ui.status(), "Forgot {} branches.", names.len())?;
tx.finish(
ui,
format!(
"forget branches {}",
matched_branches.iter().map(|(name, _)| name).join(", ")
),
)?;
writeln!(ui.status(), "Forgot {} branches.", matched_branches.len())?;
Ok(())
}

fn find_forgettable_branches(
view: &View,
fn find_forgettable_branches<'a>(
view: &'a View,
name_patterns: &[StringPattern],
) -> Result<Vec<String>, CommandError> {
) -> Result<Vec<(&'a str, BranchTarget<'a>)>, CommandError> {
find_branches_with(name_patterns, |pattern| {
view.branches()
.filter(|(name, _)| pattern.matches(name))
.map(|(name, _)| name.to_owned())
view.branches().filter(|(name, _)| pattern.matches(name))
})
}
27 changes: 13 additions & 14 deletions cli/src/commands/branch/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -85,33 +85,32 @@ pub fn cmd_branch(
}
}

fn find_local_branches(
view: &View,
fn find_local_branches<'a>(
view: &'a View,
name_patterns: &[StringPattern],
) -> Result<Vec<String>, CommandError> {
) -> Result<Vec<(&'a str, &'a RefTarget)>, CommandError> {
find_branches_with(name_patterns, |pattern| {
view.local_branches_matching(pattern)
.map(|(name, _)| name.to_owned())
})
}

fn find_branches_with<'a, I: Iterator<Item = String>>(
name_patterns: &'a [StringPattern],
mut find_matches: impl FnMut(&'a StringPattern) -> I,
) -> Result<Vec<String>, CommandError> {
let mut matching_branches: Vec<String> = vec![];
fn find_branches_with<'a, 'b, V, I: Iterator<Item = (&'a str, V)>>(
name_patterns: &'b [StringPattern],
mut find_matches: impl FnMut(&'b StringPattern) -> I,
) -> Result<Vec<I::Item>, CommandError> {
let mut matching_branches: Vec<I::Item> = vec![];
let mut unmatched_patterns = vec![];
for pattern in name_patterns {
let mut names = find_matches(pattern).peekable();
if names.peek().is_none() {
let mut matches = find_matches(pattern).peekable();
if matches.peek().is_none() {
unmatched_patterns.push(pattern);
}
matching_branches.extend(names);
matching_branches.extend(matches);
}
match &unmatched_patterns[..] {
[] => {
matching_branches.sort_unstable();
matching_branches.dedup();
matching_branches.sort_unstable_by_key(|(name, _)| *name);
matching_branches.dedup_by_key(|(name, _)| *name);
Ok(matching_branches)
}
[pattern] if pattern.is_exact() => Err(user_error(format!("No such branch: {pattern}"))),
Expand Down
28 changes: 14 additions & 14 deletions cli/src/commands/branch/move.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
// See the License for the specific language governing permissions and
// limitations under the License.

use itertools::Itertools as _;
use jj_lib::backend::CommitId;
use jj_lib::object_id::ObjectId as _;
use jj_lib::op_store::RefTarget;
Expand Down Expand Up @@ -63,10 +64,9 @@ pub fn cmd_branch_move(
args: &BranchMoveArgs,
) -> Result<(), CommandError> {
let mut workspace_command = command.workspace_helper(ui)?;
let repo = workspace_command.repo().as_ref();
let view = repo.view();
let repo = workspace_command.repo().clone();

let branch_names = {
let matched_branches = {
let is_source_commit = if !args.from.is_empty() {
workspace_command
.parse_union_revsets(&args.from)?
Expand All @@ -77,38 +77,38 @@ pub fn cmd_branch_move(
};
if !args.names.is_empty() {
find_branches_with(&args.names, |pattern| {
view.local_branches_matching(pattern)
repo.view()
.local_branches_matching(pattern)
.filter(|(_, target)| target.added_ids().any(&is_source_commit))
.map(|(name, _)| name.to_owned())
})?
} else {
view.local_branches()
repo.view()
.local_branches()
.filter(|(_, target)| target.added_ids().any(&is_source_commit))
.map(|(name, _)| name.to_owned())
.collect()
}
};
let target_commit = workspace_command.resolve_single_rev(&args.to)?;

if branch_names.is_empty() {
if matched_branches.is_empty() {
writeln!(ui.status(), "No branches to update.")?;
return Ok(());
}
if branch_names.len() > 1 {
if matched_branches.len() > 1 {
writeln!(
ui.warning_default(),
"Updating multiple branches: {}",
branch_names.join(", "),
matched_branches.iter().map(|(name, _)| name).join(", "),
)?;
if args.names.is_empty() {
writeln!(ui.hint_default(), "Specify branch by name to update one.")?;
}
}

if !args.allow_backwards {
if let Some(name) = branch_names
if let Some((name, _)) = matched_branches
.iter()
.find(|name| !is_fast_forward(repo, view.get_local_branch(name), target_commit.id()))
.find(|(_, old_target)| !is_fast_forward(repo.as_ref(), old_target, target_commit.id()))
{
return Err(user_error_with_hint(
format!("Refusing to move branch backwards or sideways: {name}"),
Expand All @@ -118,15 +118,15 @@ pub fn cmd_branch_move(
}

let mut tx = workspace_command.start_transaction();
for name in &branch_names {
for (name, _) in &matched_branches {
tx.mut_repo()
.set_local_branch_target(name, RefTarget::normal(target_commit.id().clone()));
}
tx.finish(
ui,
format!(
"point branches {names} to commit {id}",
names = branch_names.join(", "),
names = matched_branches.iter().map(|(name, _)| name).join(", "),
id = target_commit.id().hex()
),
)?;
Expand Down

0 comments on commit 4ec170b

Please sign in to comment.