-
Notifications
You must be signed in to change notification settings - Fork 963
Expand file tree
/
Copy pathcreate.rs
More file actions
128 lines (121 loc) · 4.43 KB
/
create.rs
File metadata and controls
128 lines (121 loc) · 4.43 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
// Copyright 2020-2023 The Jujutsu Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
use clap_complete::ArgValueCompleter;
use itertools::Itertools as _;
use jj_lib::object_id::ObjectId as _;
use jj_lib::op_store::RefTarget;
use jj_lib::ref_name::RefNameBuf;
use crate::cli_util::CommandHelper;
use crate::cli_util::RevisionArg;
use crate::cli_util::has_tracked_remote_bookmarks;
use crate::command_error::CommandError;
use crate::command_error::user_error;
use crate::complete;
use crate::revset_util;
use crate::ui::Ui;
/// Create a new bookmark
#[derive(clap::Args, Clone, Debug)]
pub struct BookmarkCreateArgs {
/// The bookmark's target revision
#[arg(
long,
short,
default_value = "@",
visible_alias = "to",
value_name = "REVSET"
)]
#[arg(add = ArgValueCompleter::new(complete::revset_expression_all))]
revision: RevisionArg,
/// The bookmarks to create
#[arg(required = true, value_parser = revset_util::parse_bookmark_name)]
names: Vec<RefNameBuf>,
}
pub async fn cmd_bookmark_create(
ui: &mut Ui,
command: &CommandHelper,
args: &BookmarkCreateArgs,
) -> Result<(), CommandError> {
let mut workspace_command = command.workspace_helper(ui)?;
let target_commit = workspace_command.resolve_single_rev(ui, &args.revision)?;
let repo = workspace_command.repo().as_ref();
let view = repo.view();
let bookmark_names = &args.names;
for name in bookmark_names {
if view.get_local_bookmark(name).is_present() {
return Err(user_error(format!(
"Bookmark already exists: {name}",
name = name.as_symbol()
))
.hinted("Use `jj bookmark set` to update it."));
}
if has_tracked_remote_bookmarks(repo, name) {
return Err(user_error(format!(
"Tracked remote bookmarks exist for deleted bookmark: {name}",
name = name.as_symbol()
))
.hinted(format!(
"Use `jj bookmark set` to recreate the local bookmark. Run `jj bookmark untrack \
{name}` to disassociate them.",
name = name.as_symbol()
)));
}
}
if target_commit.is_discardable(repo).await? {
writeln!(ui.warning_default(), "Target revision is empty.")?;
}
let mut tx = workspace_command.start_transaction();
let remote_settings = tx.settings().remote_settings()?;
let remote_auto_track_matchers =
revset_util::parse_remote_auto_track_bookmarks_map_for_new_bookmarks(ui, &remote_settings)?;
let readonly_repo = tx.base_repo().clone();
for name in bookmark_names {
tx.repo_mut()
.set_local_bookmark_target(name, RefTarget::normal(target_commit.id().clone()));
for (remote_name, matcher) in &remote_auto_track_matchers {
if !matcher.is_match(name.as_str()) {
continue;
}
let Some(view) = readonly_repo.view().get_remote_view(remote_name) else {
continue;
};
let symbol = name.to_remote_symbol(remote_name);
if view.bookmarks.contains_key(name) {
writeln!(
ui.warning_default(),
"Auto-tracking bookmark that exists on the remote: {symbol}"
)?;
}
tx.repo_mut().track_remote_bookmark(symbol)?;
}
}
if let Some(mut formatter) = ui.status_formatter() {
write!(
formatter,
"Created {} bookmarks pointing to ",
bookmark_names.len()
)?;
tx.write_commit_summary(formatter.as_mut(), &target_commit)?;
writeln!(formatter)?;
}
tx.finish(
ui,
format!(
"create bookmark {names} pointing to commit {id}",
names = bookmark_names.iter().map(|n| n.as_symbol()).join(", "),
id = target_commit.id().hex()
),
)
.await?;
Ok(())
}