cli: make WorkspaceCommandHelper keep a CommandHelper

This avoids cloning `UserSettings` and some other data. I haven't
attempted to measure the performance impact (I expect it's tiny); this
is more about clarifying that there are not multiple different
versions of these fields.
This commit is contained in:
Martin von Zweigbergk 2024-09-01 13:42:37 -07:00 committed by Martin von Zweigbergk
parent 34425a2501
commit 6fdef8bf67

View file

@ -591,15 +591,11 @@ impl AdvanceBranchesSettings {
/// Provides utilities for writing a command that works on a [`Workspace`] /// Provides utilities for writing a command that works on a [`Workspace`]
/// (which most commands do). /// (which most commands do).
pub struct WorkspaceCommandHelper { pub struct WorkspaceCommandHelper {
string_args: Vec<String>, command: CommandHelper,
global_args: GlobalArgs,
settings: UserSettings,
workspace: Workspace, workspace: Workspace,
user_repo: ReadonlyUserRepo, user_repo: ReadonlyUserRepo,
revset_extensions: Arc<RevsetExtensions>,
// TODO: Parsed template can be cached if it doesn't capture 'repo lifetime // TODO: Parsed template can be cached if it doesn't capture 'repo lifetime
commit_summary_template_text: String, commit_summary_template_text: String,
commit_template_extensions: Vec<Arc<dyn CommitTemplateLanguageExtension>>,
revset_aliases_map: RevsetAliasesMap, revset_aliases_map: RevsetAliasesMap,
template_aliases_map: TemplateAliasesMap, template_aliases_map: TemplateAliasesMap,
may_update_working_copy: bool, may_update_working_copy: bool,
@ -616,28 +612,23 @@ impl WorkspaceCommandHelper {
repo: Arc<ReadonlyRepo>, repo: Arc<ReadonlyRepo>,
loaded_at_head: bool, loaded_at_head: bool,
) -> Result<Self, CommandError> { ) -> Result<Self, CommandError> {
let settings = command.data.settings.clone(); let settings = command.settings();
let commit_summary_template_text = let commit_summary_template_text =
settings.config().get_string("templates.commit_summary")?; settings.config().get_string("templates.commit_summary")?;
let revset_aliases_map = let revset_aliases_map =
revset_util::load_revset_aliases(ui, &command.data.layered_configs)?; revset_util::load_revset_aliases(ui, &command.data.layered_configs)?;
let template_aliases_map = command.load_template_aliases(ui)?; let template_aliases_map = command.load_template_aliases(ui)?;
let may_update_working_copy = let may_update_working_copy = loaded_at_head && !command.global_args().ignore_working_copy;
loaded_at_head && !command.data.global_args.ignore_working_copy;
let working_copy_shared_with_git = is_colocated_git_workspace(&workspace, &repo); let working_copy_shared_with_git = is_colocated_git_workspace(&workspace, &repo);
let path_converter = RepoPathUiConverter::Fs { let path_converter = RepoPathUiConverter::Fs {
cwd: command.data.cwd.clone(), cwd: command.cwd().to_owned(),
base: workspace.workspace_root().clone(), base: workspace.workspace_root().clone(),
}; };
let helper = Self { let helper = Self {
string_args: command.data.string_args.clone(), command: command.clone(),
global_args: command.data.global_args.clone(),
settings,
workspace, workspace,
user_repo: ReadonlyUserRepo::new(repo), user_repo: ReadonlyUserRepo::new(repo),
revset_extensions: command.data.revset_extensions.clone(),
commit_summary_template_text, commit_summary_template_text,
commit_template_extensions: command.data.commit_template_extensions.clone(),
revset_aliases_map, revset_aliases_map,
template_aliases_map, template_aliases_map,
may_update_working_copy, may_update_working_copy,
@ -650,6 +641,10 @@ impl WorkspaceCommandHelper {
Ok(helper) Ok(helper)
} }
pub fn settings(&self) -> &UserSettings {
self.command.settings()
}
pub fn git_backend(&self) -> Option<&GitBackend> { pub fn git_backend(&self) -> Option<&GitBackend> {
self.user_repo.git_backend() self.user_repo.git_backend()
} }
@ -658,7 +653,7 @@ impl WorkspaceCommandHelper {
if self.may_update_working_copy { if self.may_update_working_copy {
Ok(()) Ok(())
} else { } else {
let hint = if self.global_args.ignore_working_copy { let hint = if self.command.global_args().ignore_working_copy {
"Don't use --ignore-working-copy." "Don't use --ignore-working-copy."
} else { } else {
"Don't use --at-op." "Don't use --at-op."
@ -700,6 +695,7 @@ impl WorkspaceCommandHelper {
#[instrument(skip_all)] #[instrument(skip_all)]
fn import_git_head(&mut self, ui: &mut Ui) -> Result<(), CommandError> { fn import_git_head(&mut self, ui: &mut Ui) -> Result<(), CommandError> {
assert!(self.may_update_working_copy); assert!(self.may_update_working_copy);
let command = self.command.clone();
let mut tx = self.start_transaction(); let mut tx = self.start_transaction();
git::import_head(tx.mut_repo())?; git::import_head(tx.mut_repo())?;
if !tx.mut_repo().has_changes() { if !tx.mut_repo().has_changes() {
@ -723,13 +719,13 @@ impl WorkspaceCommandHelper {
let workspace_id = self.workspace_id().to_owned(); let workspace_id = self.workspace_id().to_owned();
let new_git_head_commit = tx.mut_repo().store().get_commit(new_git_head_id)?; let new_git_head_commit = tx.mut_repo().store().get_commit(new_git_head_id)?;
tx.mut_repo() tx.mut_repo()
.check_out(workspace_id, &self.settings, &new_git_head_commit)?; .check_out(workspace_id, command.settings(), &new_git_head_commit)?;
let mut locked_ws = self.workspace.start_working_copy_mutation()?; let mut locked_ws = self.workspace.start_working_copy_mutation()?;
// The working copy was presumably updated by the git command that updated // The working copy was presumably updated by the git command that updated
// HEAD, so we just need to reset our working copy // HEAD, so we just need to reset our working copy
// state to it without updating working copy files. // state to it without updating working copy files.
locked_ws.locked_wc().reset(&new_git_head_commit)?; locked_ws.locked_wc().reset(&new_git_head_commit)?;
tx.mut_repo().rebase_descendants(&self.settings)?; tx.mut_repo().rebase_descendants(command.settings())?;
self.user_repo = ReadonlyUserRepo::new(tx.commit("import git head")); self.user_repo = ReadonlyUserRepo::new(tx.commit("import git head"));
locked_ws.finish(self.user_repo.repo.op_id().clone())?; locked_ws.finish(self.user_repo.repo.op_id().clone())?;
if old_git_head.is_present() { if old_git_head.is_present() {
@ -757,7 +753,7 @@ impl WorkspaceCommandHelper {
/// the working copy parent if the repository is colocated. /// the working copy parent if the repository is colocated.
#[instrument(skip_all)] #[instrument(skip_all)]
fn import_git_refs(&mut self, ui: &mut Ui) -> Result<(), CommandError> { fn import_git_refs(&mut self, ui: &mut Ui) -> Result<(), CommandError> {
let git_settings = self.settings.git_settings(); let git_settings = self.settings().git_settings();
let mut tx = self.start_transaction(); let mut tx = self.start_transaction();
// Automated import shouldn't fail because of reserved remote name. // Automated import shouldn't fail because of reserved remote name.
let stats = git::import_some_refs(tx.mut_repo(), &git_settings, |ref_name| { let stats = git::import_some_refs(tx.mut_repo(), &git_settings, |ref_name| {
@ -770,7 +766,7 @@ impl WorkspaceCommandHelper {
print_git_import_stats(ui, tx.repo(), &stats, false)?; print_git_import_stats(ui, tx.repo(), &stats, false)?;
let mut tx = tx.into_inner(); let mut tx = tx.into_inner();
// Rebase here to show slightly different status message. // Rebase here to show slightly different status message.
let num_rebased = tx.mut_repo().rebase_descendants(&self.settings)?; let num_rebased = tx.mut_repo().rebase_descendants(self.settings())?;
if num_rebased > 0 { if num_rebased > 0 {
writeln!( writeln!(
ui.status(), ui.status(),
@ -854,7 +850,7 @@ impl WorkspaceCommandHelper {
// empty arguments. // empty arguments.
if values.is_empty() { if values.is_empty() {
Ok(FilesetExpression::all()) Ok(FilesetExpression::all())
} else if self.settings.config().get_bool("ui.allow-filesets")? { } else if self.settings().config().get_bool("ui.allow-filesets")? {
self.parse_union_filesets(values) self.parse_union_filesets(values)
} else { } else {
let expressions = values let expressions = values
@ -934,7 +930,7 @@ impl WorkspaceCommandHelper {
&self, &self,
args: &DiffFormatArgs, args: &DiffFormatArgs,
) -> Result<DiffRenderer<'_>, CommandError> { ) -> Result<DiffRenderer<'_>, CommandError> {
let formats = diff_util::diff_formats_for(&self.settings, args)?; let formats = diff_util::diff_formats_for(self.settings(), args)?;
Ok(self.diff_renderer(formats)) Ok(self.diff_renderer(formats))
} }
@ -946,7 +942,7 @@ impl WorkspaceCommandHelper {
args: &DiffFormatArgs, args: &DiffFormatArgs,
patch: bool, patch: bool,
) -> Result<Option<DiffRenderer<'_>>, CommandError> { ) -> Result<Option<DiffRenderer<'_>>, CommandError> {
let formats = diff_util::diff_formats_for_log(&self.settings, args, patch)?; let formats = diff_util::diff_formats_for_log(self.settings(), args, patch)?;
Ok((!formats.is_empty()).then(|| self.diff_renderer(formats))) Ok((!formats.is_empty()).then(|| self.diff_renderer(formats)))
} }
@ -960,9 +956,13 @@ impl WorkspaceCommandHelper {
) -> Result<DiffEditor, CommandError> { ) -> Result<DiffEditor, CommandError> {
let base_ignores = self.base_ignores()?; let base_ignores = self.base_ignores()?;
if let Some(name) = tool_name { if let Some(name) = tool_name {
Ok(DiffEditor::with_name(name, &self.settings, base_ignores)?) Ok(DiffEditor::with_name(name, self.settings(), base_ignores)?)
} else { } else {
Ok(DiffEditor::from_settings(ui, &self.settings, base_ignores)?) Ok(DiffEditor::from_settings(
ui,
self.settings(),
base_ignores,
)?)
} }
} }
@ -991,9 +991,9 @@ impl WorkspaceCommandHelper {
tool_name: Option<&str>, tool_name: Option<&str>,
) -> Result<MergeEditor, MergeToolConfigError> { ) -> Result<MergeEditor, MergeToolConfigError> {
if let Some(name) = tool_name { if let Some(name) = tool_name {
MergeEditor::with_name(name, &self.settings) MergeEditor::with_name(name, self.settings())
} else { } else {
MergeEditor::from_settings(ui, &self.settings) MergeEditor::from_settings(ui, self.settings())
} }
} }
@ -1029,7 +1029,7 @@ impl WorkspaceCommandHelper {
let all = match modifier { let all = match modifier {
Some(RevsetModifier::All) => true, Some(RevsetModifier::All) => true,
None => self None => self
.settings .settings()
.config() .config()
.get_bool("ui.always-allow-large-revsets")?, .get_bool("ui.always-allow-large-revsets")?,
}; };
@ -1101,7 +1101,7 @@ impl WorkspaceCommandHelper {
) -> Result<RevsetExpressionEvaluator<'_>, CommandError> { ) -> Result<RevsetExpressionEvaluator<'_>, CommandError> {
Ok(RevsetExpressionEvaluator::new( Ok(RevsetExpressionEvaluator::new(
self.repo().as_ref(), self.repo().as_ref(),
self.revset_extensions.clone(), self.command.revset_extensions().clone(),
self.id_prefix_context()?, self.id_prefix_context()?,
expression, expression,
)) ))
@ -1112,7 +1112,7 @@ impl WorkspaceCommandHelper {
path_converter: &self.path_converter, path_converter: &self.path_converter,
workspace_id: self.workspace_id(), workspace_id: self.workspace_id(),
}; };
let now = if let Some(timestamp) = self.settings.commit_timestamp() { let now = if let Some(timestamp) = self.settings().commit_timestamp() {
chrono::Local chrono::Local
.timestamp_millis_opt(timestamp.timestamp.0) .timestamp_millis_opt(timestamp.timestamp.0)
.unwrap() .unwrap()
@ -1121,20 +1121,21 @@ impl WorkspaceCommandHelper {
}; };
RevsetParseContext::new( RevsetParseContext::new(
&self.revset_aliases_map, &self.revset_aliases_map,
self.settings.user_email(), self.settings().user_email(),
now.into(), now.into(),
&self.revset_extensions, self.command.revset_extensions(),
Some(workspace_context), Some(workspace_context),
) )
} }
fn new_id_prefix_context(&self) -> Result<IdPrefixContext, CommandError> { fn new_id_prefix_context(&self) -> Result<IdPrefixContext, CommandError> {
let mut context: IdPrefixContext = IdPrefixContext::new(self.revset_extensions.clone()); let mut context: IdPrefixContext =
IdPrefixContext::new(self.command.revset_extensions().clone());
let revset_string: String = self let revset_string: String = self
.settings .settings()
.config() .config()
.get_string("revsets.short-prefixes") .get_string("revsets.short-prefixes")
.unwrap_or_else(|_| self.settings.default_revset()); .unwrap_or_else(|_| self.settings().default_revset());
if !revset_string.is_empty() { if !revset_string.is_empty() {
let (expression, modifier) = let (expression, modifier) =
revset::parse_with_modifier(&revset_string, &self.revset_parse_context()).map_err( revset::parse_with_modifier(&revset_string, &self.revset_parse_context()).map_err(
@ -1196,7 +1197,7 @@ impl WorkspaceCommandHelper {
self.workspace_id(), self.workspace_id(),
self.revset_parse_context(), self.revset_parse_context(),
self.id_prefix_context()?, self.id_prefix_context()?,
&self.commit_template_extensions, &self.command.data.commit_template_extensions,
)) ))
} }
@ -1235,7 +1236,7 @@ impl WorkspaceCommandHelper {
repo: &dyn Repo, repo: &dyn Repo,
commits: impl IntoIterator<Item = &'a CommitId>, commits: impl IntoIterator<Item = &'a CommitId>,
) -> Result<(), CommandError> { ) -> Result<(), CommandError> {
if self.global_args.ignore_immutable { if self.command.global_args().ignore_immutable {
let root_id = self.repo().store().root_commit_id(); let root_id = self.repo().store().root_commit_id();
return if commits.into_iter().contains(root_id) { return if commits.into_iter().contains(root_id) {
Err(user_error(format!( Err(user_error(format!(
@ -1250,7 +1251,7 @@ impl WorkspaceCommandHelper {
// Not using self.id_prefix_context() because the disambiguation data // Not using self.id_prefix_context() because the disambiguation data
// must not be calculated and cached against arbitrary repo. It's also // must not be calculated and cached against arbitrary repo. It's also
// unlikely that the immutable expression contains short hashes. // unlikely that the immutable expression contains short hashes.
let id_prefix_context = IdPrefixContext::new(self.revset_extensions.clone()); let id_prefix_context = IdPrefixContext::new(self.command.revset_extensions().clone());
let to_rewrite_revset = let to_rewrite_revset =
RevsetExpression::commits(commits.into_iter().cloned().collect_vec()); RevsetExpression::commits(commits.into_iter().cloned().collect_vec());
let immutable = revset_util::parse_immutable_expression(&self.revset_parse_context()) let immutable = revset_util::parse_immutable_expression(&self.revset_parse_context())
@ -1259,7 +1260,7 @@ impl WorkspaceCommandHelper {
})?; })?;
let mut expression = RevsetExpressionEvaluator::new( let mut expression = RevsetExpressionEvaluator::new(
repo, repo,
self.revset_extensions.clone(), self.command.revset_extensions().clone(),
&id_prefix_context, &id_prefix_context,
immutable, immutable,
); );
@ -1313,6 +1314,9 @@ impl WorkspaceCommandHelper {
let base_ignores = self.base_ignores()?; let base_ignores = self.base_ignores()?;
// Compare working-copy tree and operation with repo's, and reload as needed. // Compare working-copy tree and operation with repo's, and reload as needed.
let fsmonitor_settings = self.settings().fsmonitor_settings()?;
let max_new_file_size = self.settings().max_new_file_size()?;
let command = self.command.clone();
let mut locked_ws = self.workspace.start_working_copy_mutation()?; let mut locked_ws = self.workspace.start_working_copy_mutation()?;
let old_op_id = locked_ws.locked_wc().old_operation_id().clone(); let old_op_id = locked_ws.locked_wc().old_operation_id().clone();
let (repo, wc_commit) = let (repo, wc_commit) =
@ -1361,24 +1365,27 @@ See https://github.com/martinvonz/jj/blob/main/docs/working-copy.md#stale-workin
let progress = crate::progress::snapshot_progress(ui); let progress = crate::progress::snapshot_progress(ui);
let new_tree_id = locked_ws.locked_wc().snapshot(SnapshotOptions { let new_tree_id = locked_ws.locked_wc().snapshot(SnapshotOptions {
base_ignores, base_ignores,
fsmonitor_settings: self.settings.fsmonitor_settings()?, fsmonitor_settings,
progress: progress.as_ref().map(|x| x as _), progress: progress.as_ref().map(|x| x as _),
max_new_file_size: self.settings.max_new_file_size()?, max_new_file_size,
})?; })?;
drop(progress); drop(progress);
if new_tree_id != *wc_commit.tree_id() { if new_tree_id != *wc_commit.tree_id() {
let mut tx = let mut tx = start_repo_transaction(
start_repo_transaction(&self.user_repo.repo, &self.settings, &self.string_args); &self.user_repo.repo,
command.settings(),
command.string_args(),
);
tx.set_is_snapshot(true); tx.set_is_snapshot(true);
let mut_repo = tx.mut_repo(); let mut_repo = tx.mut_repo();
let commit = mut_repo let commit = mut_repo
.rewrite_commit(&self.settings, &wc_commit) .rewrite_commit(command.settings(), &wc_commit)
.set_tree_id(new_tree_id) .set_tree_id(new_tree_id)
.write()?; .write()?;
mut_repo.set_wc_commit(workspace_id, commit.id().clone())?; mut_repo.set_wc_commit(workspace_id, commit.id().clone())?;
// Rebase descendants // Rebase descendants
let num_rebased = mut_repo.rebase_descendants(&self.settings)?; let num_rebased = mut_repo.rebase_descendants(command.settings())?;
if num_rebased > 0 { if num_rebased > 0 {
writeln!( writeln!(
ui.status(), ui.status(),
@ -1441,7 +1448,7 @@ See https://github.com/martinvonz/jj/blob/main/docs/working-copy.md#stale-workin
} }
pub fn start_transaction(&mut self) -> WorkspaceCommandTransaction { pub fn start_transaction(&mut self) -> WorkspaceCommandTransaction {
let tx = start_repo_transaction(self.repo(), &self.settings, &self.string_args); let tx = start_repo_transaction(self.repo(), self.settings(), self.command.string_args());
let id_prefix_context = mem::take(&mut self.user_repo.id_prefix_context); let id_prefix_context = mem::take(&mut self.user_repo.id_prefix_context);
WorkspaceCommandTransaction { WorkspaceCommandTransaction {
helper: self, helper: self,
@ -1460,7 +1467,7 @@ See https://github.com/martinvonz/jj/blob/main/docs/working-copy.md#stale-workin
writeln!(ui.status(), "Nothing changed.")?; writeln!(ui.status(), "Nothing changed.")?;
return Ok(()); return Ok(());
} }
let num_rebased = tx.mut_repo().rebase_descendants(&self.settings)?; let num_rebased = tx.mut_repo().rebase_descendants(self.settings())?;
if num_rebased > 0 { if num_rebased > 0 {
writeln!(ui.status(), "Rebased {num_rebased} descendant commits")?; writeln!(ui.status(), "Rebased {num_rebased} descendant commits")?;
} }
@ -1475,7 +1482,7 @@ See https://github.com/martinvonz/jj/blob/main/docs/working-copy.md#stale-workin
{ {
let wc_commit = tx.repo().store().get_commit(wc_commit_id)?; let wc_commit = tx.repo().store().get_commit(wc_commit_id)?;
tx.mut_repo() tx.mut_repo()
.check_out(workspace_id.clone(), &self.settings, &wc_commit)?; .check_out(workspace_id.clone(), self.settings(), &wc_commit)?;
writeln!( writeln!(
ui.warning_default(), ui.warning_default(),
"The working-copy commit in workspace '{}' became immutable, so a new commit \ "The working-copy commit in workspace '{}' became immutable, so a new commit \
@ -1520,7 +1527,7 @@ See https://github.com/martinvonz/jj/blob/main/docs/working-copy.md#stale-workin
} }
} }
let settings = &self.settings; let settings = self.settings();
if settings.user_name().is_empty() || settings.user_email().is_empty() { if settings.user_name().is_empty() || settings.user_email().is_empty() {
writeln!( writeln!(
ui.warning_default(), ui.warning_default(),
@ -1706,7 +1713,7 @@ Then run `jj squash` to move the resolution into the conflicted commit."#,
&self, &self,
from: impl IntoIterator<Item = &'a CommitId>, from: impl IntoIterator<Item = &'a CommitId>,
) -> Result<Vec<AdvanceableBranch>, CommandError> { ) -> Result<Vec<AdvanceableBranch>, CommandError> {
let ab_settings = AdvanceBranchesSettings::from_config(self.settings.config())?; let ab_settings = AdvanceBranchesSettings::from_config(self.settings().config())?;
if !ab_settings.feature_enabled() { if !ab_settings.feature_enabled() {
// Return early if we know that there's no work to do. // Return early if we know that there's no work to do.
return Ok(Vec::new()); return Ok(Vec::new());
@ -1749,7 +1756,7 @@ impl WorkspaceCommandTransaction<'_> {
} }
pub fn settings(&self) -> &UserSettings { pub fn settings(&self) -> &UserSettings {
&self.helper.settings self.helper.settings()
} }
pub fn base_repo(&self) -> &Arc<ReadonlyRepo> { pub fn base_repo(&self) -> &Arc<ReadonlyRepo> {
@ -1767,7 +1774,7 @@ impl WorkspaceCommandTransaction<'_> {
pub fn check_out(&mut self, commit: &Commit) -> Result<Commit, CheckOutCommitError> { pub fn check_out(&mut self, commit: &Commit) -> Result<Commit, CheckOutCommitError> {
let workspace_id = self.helper.workspace_id().to_owned(); let workspace_id = self.helper.workspace_id().to_owned();
let settings = &self.helper.settings; let settings = self.helper.settings();
self.id_prefix_context.take(); // invalidate self.id_prefix_context.take(); // invalidate
self.tx.mut_repo().check_out(workspace_id, settings, commit) self.tx.mut_repo().check_out(workspace_id, settings, commit)
} }
@ -1813,7 +1820,7 @@ impl WorkspaceCommandTransaction<'_> {
self.helper.workspace_id(), self.helper.workspace_id(),
self.helper.revset_parse_context(), self.helper.revset_parse_context(),
id_prefix_context, id_prefix_context,
&self.helper.commit_template_extensions, &self.helper.command.data.commit_template_extensions,
) )
} }