From 97024e5be443dc0742419d400c4c0f3b22fa19a5 Mon Sep 17 00:00:00 2001 From: Yuya Nishihara Date: Sat, 2 Mar 2024 13:19:26 +0900 Subject: [PATCH] cli: extract CommandError and helper functions to new module The cli_util module is big enough to slow down Emacs, so let's split it up. This change is an easy one. --- cli/examples/custom-backend/main.rs | 3 +- cli/examples/custom-command/main.rs | 3 +- cli/examples/custom-global-flag/main.rs | 3 +- cli/examples/custom-working-copy/main.rs | 3 +- cli/src/cli_util.rs | 428 +--------------------- cli/src/command_error.rs | 434 +++++++++++++++++++++++ cli/src/commands/abandon.rs | 5 +- cli/src/commands/backout.rs | 3 +- cli/src/commands/bench.rs | 3 +- cli/src/commands/branch.rs | 4 +- cli/src/commands/cat.rs | 3 +- cli/src/commands/checkout.rs | 3 +- cli/src/commands/chmod.rs | 3 +- cli/src/commands/commit.rs | 3 +- cli/src/commands/config.rs | 5 +- cli/src/commands/debug.rs | 3 +- cli/src/commands/describe.rs | 3 +- cli/src/commands/diff.rs | 3 +- cli/src/commands/diffedit.rs | 3 +- cli/src/commands/duplicate.rs | 4 +- cli/src/commands/edit.rs | 3 +- cli/src/commands/files.rs | 3 +- cli/src/commands/git.rs | 7 +- cli/src/commands/init.rs | 3 +- cli/src/commands/interdiff.rs | 3 +- cli/src/commands/log.rs | 3 +- cli/src/commands/merge.rs | 3 +- cli/src/commands/mod.rs | 3 +- cli/src/commands/move.rs | 3 +- cli/src/commands/new.rs | 5 +- cli/src/commands/next.rs | 5 +- cli/src/commands/obslog.rs | 5 +- cli/src/commands/operation.rs | 6 +- cli/src/commands/prev.rs | 3 +- cli/src/commands/rebase.rs | 5 +- cli/src/commands/resolve.rs | 3 +- cli/src/commands/restore.rs | 3 +- cli/src/commands/root.rs | 3 +- cli/src/commands/run.rs | 5 +- cli/src/commands/show.rs | 3 +- cli/src/commands/sparse.rs | 5 +- cli/src/commands/split.rs | 3 +- cli/src/commands/squash.rs | 3 +- cli/src/commands/status.rs | 3 +- cli/src/commands/tag.rs | 3 +- cli/src/commands/unsquash.rs | 3 +- cli/src/commands/untrack.rs | 3 +- cli/src/commands/util.rs | 3 +- cli/src/commands/version.rs | 3 +- cli/src/commands/workspace.rs | 6 +- cli/src/description_util.rs | 3 +- cli/src/diff_util.rs | 3 +- cli/src/git_util.rs | 2 +- cli/src/lib.rs | 1 + cli/src/ui.rs | 2 +- 55 files changed, 557 insertions(+), 488 deletions(-) create mode 100644 cli/src/command_error.rs diff --git a/cli/examples/custom-backend/main.rs b/cli/examples/custom-backend/main.rs index 08145d69b..b636cf17f 100644 --- a/cli/examples/custom-backend/main.rs +++ b/cli/examples/custom-backend/main.rs @@ -18,7 +18,8 @@ use std::path::Path; use std::time::SystemTime; use async_trait::async_trait; -use jj_cli::cli_util::{CliRunner, CommandError, CommandHelper}; +use jj_cli::cli_util::{CliRunner, CommandHelper}; +use jj_cli::command_error::CommandError; use jj_cli::ui::Ui; use jj_lib::backend::{ Backend, BackendInitError, BackendLoadError, BackendResult, ChangeId, Commit, CommitId, diff --git a/cli/examples/custom-command/main.rs b/cli/examples/custom-command/main.rs index bdcf65749..bf8abba7d 100644 --- a/cli/examples/custom-command/main.rs +++ b/cli/examples/custom-command/main.rs @@ -14,7 +14,8 @@ use std::io::Write as _; -use jj_cli::cli_util::{CliRunner, CommandError, CommandHelper}; +use jj_cli::cli_util::{CliRunner, CommandHelper}; +use jj_cli::command_error::CommandError; use jj_cli::ui::Ui; #[derive(clap::Parser, Clone, Debug)] diff --git a/cli/examples/custom-global-flag/main.rs b/cli/examples/custom-global-flag/main.rs index c557fe8bb..50e215399 100644 --- a/cli/examples/custom-global-flag/main.rs +++ b/cli/examples/custom-global-flag/main.rs @@ -14,7 +14,8 @@ use std::io::Write as _; -use jj_cli::cli_util::{CliRunner, CommandError}; +use jj_cli::cli_util::CliRunner; +use jj_cli::command_error::CommandError; use jj_cli::ui::Ui; #[derive(clap::Args, Clone, Debug)] diff --git a/cli/examples/custom-working-copy/main.rs b/cli/examples/custom-working-copy/main.rs index af9e10b96..40bdcf4b4 100644 --- a/cli/examples/custom-working-copy/main.rs +++ b/cli/examples/custom-working-copy/main.rs @@ -17,7 +17,8 @@ use std::path::{Path, PathBuf}; use std::sync::Arc; use itertools::Itertools; -use jj_cli::cli_util::{CliRunner, CommandError, CommandHelper}; +use jj_cli::cli_util::{CliRunner, CommandHelper}; +use jj_cli::command_error::CommandError; use jj_cli::ui::Ui; use jj_lib::backend::{Backend, MergedTreeId}; use jj_lib::commit::Commit; diff --git a/cli/src/cli_util.rs b/cli/src/cli_util.rs index eb49fe6e5..2ab8a0d59 100644 --- a/cli/src/cli_util.rs +++ b/cli/src/cli_util.rs @@ -31,9 +31,8 @@ use clap::builder::{NonEmptyStringValueParser, TypedValueParser, ValueParserFact use clap::{Arg, ArgAction, ArgMatches, Command, FromArgMatches}; use indexmap::{IndexMap, IndexSet}; use itertools::Itertools; -use jj_lib::backend::{BackendError, ChangeId, CommitId, MergedTreeId}; +use jj_lib::backend::{ChangeId, CommitId, MergedTreeId}; use jj_lib::commit::Commit; -use jj_lib::git::{GitConfigParseError, GitExportError, GitImportError, GitRemoteManagementError}; use jj_lib::git_backend::GitBackend; use jj_lib::gitignore::{GitIgnoreError, GitIgnoreFile}; use jj_lib::hex_util::to_reverse_hex; @@ -41,43 +40,42 @@ use jj_lib::id_prefix::IdPrefixContext; use jj_lib::matchers::{EverythingMatcher, Matcher, PrefixMatcher}; use jj_lib::merged_tree::MergedTree; use jj_lib::object_id::ObjectId; -use jj_lib::op_heads_store::{self, OpHeadResolutionError}; use jj_lib::op_store::{OpStoreError, OperationId, WorkspaceId}; use jj_lib::op_walk::OpsetEvaluationError; use jj_lib::operation::Operation; use jj_lib::repo::{ CheckOutCommitError, EditCommitError, MutableRepo, ReadonlyRepo, Repo, RepoLoader, - RepoLoaderError, RewriteRootCommit, StoreFactories, StoreLoadError, + StoreFactories, StoreLoadError, }; use jj_lib::repo_path::{FsPathParseError, RepoPath, RepoPathBuf}; use jj_lib::revset::{ - DefaultSymbolResolver, Revset, RevsetAliasesMap, RevsetCommitRef, RevsetEvaluationError, - RevsetExpression, RevsetFilterPredicate, RevsetIteratorExt, RevsetParseContext, - RevsetParseError, RevsetParseErrorKind, RevsetResolutionError, RevsetWorkspaceContext, + DefaultSymbolResolver, Revset, RevsetAliasesMap, RevsetCommitRef, RevsetExpression, + RevsetFilterPredicate, RevsetIteratorExt, RevsetParseContext, RevsetParseError, + RevsetWorkspaceContext, }; use jj_lib::rewrite::restore_tree; use jj_lib::settings::{ConfigResultExt as _, UserSettings}; use jj_lib::signing::SignInitError; use jj_lib::str_util::{StringPattern, StringPatternParseError}; use jj_lib::transaction::Transaction; -use jj_lib::tree::TreeMergeError; use jj_lib::view::View; use jj_lib::working_copy::{ - CheckoutStats, LockedWorkingCopy, ResetError, SnapshotError, SnapshotOptions, WorkingCopy, - WorkingCopyFactory, WorkingCopyStateError, + CheckoutStats, LockedWorkingCopy, SnapshotOptions, WorkingCopy, WorkingCopyFactory, }; use jj_lib::workspace::{ - default_working_copy_factories, LockedWorkspace, Workspace, WorkspaceInitError, - WorkspaceLoadError, WorkspaceLoader, + default_working_copy_factories, LockedWorkspace, Workspace, WorkspaceLoadError, WorkspaceLoader, }; -use jj_lib::{dag_walk, file_util, git, op_walk, revset}; +use jj_lib::{dag_walk, file_util, git, op_heads_store, op_walk, revset}; use once_cell::unsync::OnceCell; -use thiserror::Error; use toml_edit; use tracing::instrument; use tracing_chrome::ChromeLayerBuilder; use tracing_subscriber::prelude::*; +use crate::command_error::{ + internal_error, internal_error_with_message, user_error, user_error_with_hint, + user_error_with_message, CommandError, +}; use crate::commit_templater::CommitTemplateLanguageExtension; use crate::config::{ new_config_path, AnnotatedValue, CommandNameAndArgs, ConfigSource, LayeredConfigs, @@ -86,110 +84,12 @@ use crate::formatter::{FormatRecorder, Formatter, PlainTextFormatter}; use crate::git_util::{ is_colocated_git_workspace, print_failed_git_export, print_git_import_stats, }; -use crate::merge_tools::{ - ConflictResolveError, DiffEditError, DiffEditor, DiffGenerateError, MergeEditor, - MergeToolConfigError, -}; -use crate::template_parser::{TemplateAliasesMap, TemplateParseError, TemplateParseErrorKind}; +use crate::merge_tools::{DiffEditor, MergeEditor, MergeToolConfigError}; +use crate::template_parser::TemplateAliasesMap; use crate::templater::Template; use crate::ui::{ColorChoice, Ui}; use crate::{commit_templater, text_util}; -#[derive(Clone, Debug)] -pub enum CommandError { - UserError { - err: Arc, - hint: Option, - }, - ConfigError(String), - /// Invalid command line - CliError(String), - /// Invalid command line detected by clap - ClapCliError(Arc), - BrokenPipe, - InternalError(Arc), -} - -/// Wraps error with user-visible message. -#[derive(Debug, Error)] -#[error("{message}")] -struct ErrorWithMessage { - message: String, - source: Box, -} - -impl ErrorWithMessage { - fn new( - message: impl Into, - source: impl Into>, - ) -> Self { - ErrorWithMessage { - message: message.into(), - source: source.into(), - } - } -} - -pub fn user_error(err: impl Into>) -> CommandError { - user_error_with_hint_opt(err, None) -} - -pub fn user_error_with_hint( - err: impl Into>, - hint: impl Into, -) -> CommandError { - user_error_with_hint_opt(err, Some(hint.into())) -} - -pub fn user_error_with_message( - message: impl Into, - source: impl Into>, -) -> CommandError { - user_error_with_hint_opt(ErrorWithMessage::new(message, source), None) -} - -pub fn user_error_with_message_and_hint( - message: impl Into, - hint: impl Into, - source: impl Into>, -) -> CommandError { - user_error_with_hint_opt(ErrorWithMessage::new(message, source), Some(hint.into())) -} - -pub fn user_error_with_hint_opt( - err: impl Into>, - hint: Option, -) -> CommandError { - CommandError::UserError { - err: Arc::from(err.into()), - hint, - } -} - -pub fn internal_error(err: impl Into>) -> CommandError { - CommandError::InternalError(Arc::from(err.into())) -} - -pub fn internal_error_with_message( - message: impl Into, - source: impl Into>, -) -> CommandError { - CommandError::InternalError(Arc::new(ErrorWithMessage::new(message, source))) -} - -fn format_similarity_hint>(candidates: &[S]) -> Option { - match candidates { - [] => None, - names => { - let quoted_names = names - .iter() - .map(|s| format!(r#""{}""#, s.as_ref())) - .join(", "); - Some(format!("Did you mean {quoted_names}?")) - } - } -} - fn print_error_sources(ui: &Ui, source: Option<&dyn std::error::Error>) -> io::Result<()> { let Some(err) = source else { return Ok(()); @@ -205,306 +105,6 @@ fn print_error_sources(ui: &Ui, source: Option<&dyn std::error::Error>) -> io::R Ok(()) } -impl From for CommandError { - fn from(err: std::io::Error) -> Self { - if err.kind() == std::io::ErrorKind::BrokenPipe { - CommandError::BrokenPipe - } else { - user_error(err) - } - } -} - -impl From for CommandError { - fn from(err: config::ConfigError) -> Self { - CommandError::ConfigError(err.to_string()) - } -} - -impl From for CommandError { - fn from(err: crate::config::ConfigError) -> Self { - CommandError::ConfigError(err.to_string()) - } -} - -impl From for CommandError { - fn from(err: RewriteRootCommit) -> Self { - internal_error_with_message("Attempted to rewrite the root commit", err) - } -} - -impl From for CommandError { - fn from(err: EditCommitError) -> Self { - internal_error_with_message("Failed to edit a commit", err) - } -} - -impl From for CommandError { - fn from(err: CheckOutCommitError) -> Self { - internal_error_with_message("Failed to check out a commit", err) - } -} - -impl From for CommandError { - fn from(err: BackendError) -> Self { - internal_error_with_message("Unexpected error from backend", err) - } -} - -impl From for CommandError { - fn from(err: WorkspaceInitError) -> Self { - match err { - WorkspaceInitError::DestinationExists(_) => { - user_error("The target repo already exists") - } - WorkspaceInitError::NonUnicodePath => { - user_error("The target repo path contains non-unicode characters") - } - WorkspaceInitError::CheckOutCommit(err) => { - internal_error_with_message("Failed to check out the initial commit", err) - } - WorkspaceInitError::Path(err) => { - internal_error_with_message("Failed to access the repository", err) - } - WorkspaceInitError::Backend(err) => { - user_error_with_message("Failed to access the repository", err) - } - WorkspaceInitError::WorkingCopyState(err) => { - internal_error_with_message("Failed to access the repository", err) - } - WorkspaceInitError::SignInit(err @ SignInitError::UnknownBackend(_)) => user_error(err), - WorkspaceInitError::SignInit(err) => internal_error(err), - } - } -} - -impl From for CommandError { - fn from(err: OpHeadResolutionError) -> Self { - match err { - OpHeadResolutionError::NoHeads => { - internal_error_with_message("Corrupt repository", err) - } - } - } -} - -impl From for CommandError { - fn from(err: OpsetEvaluationError) -> Self { - match err { - OpsetEvaluationError::OpsetResolution(err) => user_error(err), - OpsetEvaluationError::OpHeadResolution(err) => err.into(), - OpsetEvaluationError::OpStore(err) => err.into(), - } - } -} - -impl From for CommandError { - fn from(err: SnapshotError) -> Self { - match err { - SnapshotError::NewFileTooLarge { .. } => user_error_with_message_and_hint( - "Failed to snapshot the working copy", - r#"Increase the value of the `snapshot.max-new-file-size` config option if you -want this file to be snapshotted. Otherwise add it to your `.gitignore` file."#, - err, - ), - err => internal_error_with_message("Failed to snapshot the working copy", err), - } - } -} - -impl From for CommandError { - fn from(err: TreeMergeError) -> Self { - internal_error_with_message("Merge failed", err) - } -} - -impl From for CommandError { - fn from(err: OpStoreError) -> Self { - internal_error_with_message("Failed to load an operation", err) - } -} - -impl From for CommandError { - fn from(err: RepoLoaderError) -> Self { - internal_error_with_message("Failed to load the repo", err) - } -} - -impl From for CommandError { - fn from(err: ResetError) -> Self { - internal_error_with_message("Failed to reset the working copy", err) - } -} - -impl From for CommandError { - fn from(err: DiffEditError) -> Self { - user_error_with_message("Failed to edit diff", err) - } -} - -impl From for CommandError { - fn from(err: DiffGenerateError) -> Self { - user_error_with_message("Failed to generate diff", err) - } -} - -impl From for CommandError { - fn from(err: ConflictResolveError) -> Self { - user_error_with_message("Failed to resolve conflicts", err) - } -} - -impl From for CommandError { - fn from(err: MergeToolConfigError) -> Self { - user_error_with_message("Failed to load tool configuration", err) - } -} - -impl From for CommandError { - fn from(err: git2::Error) -> Self { - user_error_with_message("Git operation failed", err) - } -} - -impl From for CommandError { - fn from(err: GitImportError) -> Self { - let message = format!("Failed to import refs from underlying Git repo: {err}"); - let hint = match &err { - GitImportError::MissingHeadTarget { .. } - | GitImportError::MissingRefAncestor { .. } => Some( - "\ -Is this Git repository a shallow or partial clone (cloned with the --depth or --filter \ - argument)? -jj currently does not support shallow/partial clones. To use jj with this \ - repository, try -unshallowing the repository (https://stackoverflow.com/q/6802145) or re-cloning with the full -repository contents." - .to_string(), - ), - GitImportError::RemoteReservedForLocalGitRepo => { - Some("Run `jj git remote rename` to give different name.".to_string()) - } - GitImportError::InternalBackend(_) => None, - GitImportError::InternalGitError(_) => None, - GitImportError::UnexpectedBackend => None, - }; - user_error_with_hint_opt(message, hint) - } -} - -impl From for CommandError { - fn from(err: GitExportError) -> Self { - internal_error_with_message("Failed to export refs to underlying Git repo", err) - } -} - -impl From for CommandError { - fn from(err: GitRemoteManagementError) -> Self { - user_error(err) - } -} - -impl From for CommandError { - fn from(err: RevsetEvaluationError) -> Self { - user_error(err) - } -} - -impl From for CommandError { - fn from(err: RevsetParseError) -> Self { - let err_chain = iter::successors(Some(&err), |e| e.origin()); - let message = err_chain.clone().join("\n"); - // Only for the bottom error, which is usually the root cause - let hint = match err_chain.last().unwrap().kind() { - RevsetParseErrorKind::NotPrefixOperator { - op: _, - similar_op, - description, - } - | RevsetParseErrorKind::NotPostfixOperator { - op: _, - similar_op, - description, - } - | RevsetParseErrorKind::NotInfixOperator { - op: _, - similar_op, - description, - } => Some(format!("Did you mean '{similar_op}' for {description}?")), - RevsetParseErrorKind::NoSuchFunction { - name: _, - candidates, - } => format_similarity_hint(candidates), - _ => None, - }; - user_error_with_hint_opt(format!("Failed to parse revset: {message}"), hint) - } -} - -impl From for CommandError { - fn from(err: RevsetResolutionError) -> Self { - let hint = match &err { - RevsetResolutionError::NoSuchRevision { - name: _, - candidates, - } => format_similarity_hint(candidates), - RevsetResolutionError::EmptyString - | RevsetResolutionError::WorkspaceMissingWorkingCopy { .. } - | RevsetResolutionError::AmbiguousCommitIdPrefix(_) - | RevsetResolutionError::AmbiguousChangeIdPrefix(_) - | RevsetResolutionError::StoreError(_) => None, - }; - user_error_with_hint_opt(err, hint) - } -} - -impl From for CommandError { - fn from(err: TemplateParseError) -> Self { - let err_chain = iter::successors(Some(&err), |e| e.origin()); - let message = err_chain.clone().join("\n"); - // Only for the bottom error, which is usually the root cause - let hint = match err_chain.last().unwrap().kind() { - TemplateParseErrorKind::NoSuchKeyword { candidates, .. } - | TemplateParseErrorKind::NoSuchFunction { candidates, .. } - | TemplateParseErrorKind::NoSuchMethod { candidates, .. } => { - format_similarity_hint(candidates) - } - _ => None, - }; - user_error_with_hint_opt(format!("Failed to parse template: {message}"), hint) - } -} - -impl From for CommandError { - fn from(err: FsPathParseError) -> Self { - user_error(err) - } -} - -impl From for CommandError { - fn from(err: clap::Error) -> Self { - CommandError::ClapCliError(Arc::new(err)) - } -} - -impl From for CommandError { - fn from(err: GitConfigParseError) -> Self { - internal_error_with_message("Failed to parse Git config", err) - } -} - -impl From for CommandError { - fn from(err: WorkingCopyStateError) -> Self { - internal_error_with_message("Failed to access working copy state", err) - } -} - -impl From for CommandError { - fn from(err: GitIgnoreError) -> Self { - user_error_with_message("Failed to process .gitignore.", err) - } -} - #[derive(Clone)] struct ChromeTracingFlushGuard { _inner: Option>, diff --git a/cli/src/command_error.rs b/cli/src/command_error.rs new file mode 100644 index 000000000..fa3287b7f --- /dev/null +++ b/cli/src/command_error.rs @@ -0,0 +1,434 @@ +// Copyright 2022-2024 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 std::sync::Arc; +use std::{error, io, iter, str}; + +use itertools::Itertools as _; +use jj_lib::backend::BackendError; +use jj_lib::git::{GitConfigParseError, GitExportError, GitImportError, GitRemoteManagementError}; +use jj_lib::gitignore::GitIgnoreError; +use jj_lib::op_heads_store::OpHeadResolutionError; +use jj_lib::op_store::OpStoreError; +use jj_lib::op_walk::OpsetEvaluationError; +use jj_lib::repo::{CheckOutCommitError, EditCommitError, RepoLoaderError, RewriteRootCommit}; +use jj_lib::repo_path::FsPathParseError; +use jj_lib::revset::{ + RevsetEvaluationError, RevsetParseError, RevsetParseErrorKind, RevsetResolutionError, +}; +use jj_lib::signing::SignInitError; +use jj_lib::tree::TreeMergeError; +use jj_lib::working_copy::{ResetError, SnapshotError, WorkingCopyStateError}; +use jj_lib::workspace::WorkspaceInitError; +use thiserror::Error; + +use crate::merge_tools::{ + ConflictResolveError, DiffEditError, DiffGenerateError, MergeToolConfigError, +}; +use crate::template_parser::{TemplateParseError, TemplateParseErrorKind}; + +#[derive(Clone, Debug)] +pub enum CommandError { + UserError { + err: Arc, + hint: Option, + }, + ConfigError(String), + /// Invalid command line + CliError(String), + /// Invalid command line detected by clap + ClapCliError(Arc), + BrokenPipe, + InternalError(Arc), +} + +/// Wraps error with user-visible message. +#[derive(Debug, Error)] +#[error("{message}")] +struct ErrorWithMessage { + message: String, + source: Box, +} + +impl ErrorWithMessage { + fn new( + message: impl Into, + source: impl Into>, + ) -> Self { + ErrorWithMessage { + message: message.into(), + source: source.into(), + } + } +} + +pub fn user_error(err: impl Into>) -> CommandError { + user_error_with_hint_opt(err, None) +} + +pub fn user_error_with_hint( + err: impl Into>, + hint: impl Into, +) -> CommandError { + user_error_with_hint_opt(err, Some(hint.into())) +} + +pub fn user_error_with_message( + message: impl Into, + source: impl Into>, +) -> CommandError { + user_error_with_hint_opt(ErrorWithMessage::new(message, source), None) +} + +pub fn user_error_with_message_and_hint( + message: impl Into, + hint: impl Into, + source: impl Into>, +) -> CommandError { + user_error_with_hint_opt(ErrorWithMessage::new(message, source), Some(hint.into())) +} + +pub fn user_error_with_hint_opt( + err: impl Into>, + hint: Option, +) -> CommandError { + CommandError::UserError { + err: Arc::from(err.into()), + hint, + } +} + +pub fn internal_error(err: impl Into>) -> CommandError { + CommandError::InternalError(Arc::from(err.into())) +} + +pub fn internal_error_with_message( + message: impl Into, + source: impl Into>, +) -> CommandError { + CommandError::InternalError(Arc::new(ErrorWithMessage::new(message, source))) +} + +fn format_similarity_hint>(candidates: &[S]) -> Option { + match candidates { + [] => None, + names => { + let quoted_names = names + .iter() + .map(|s| format!(r#""{}""#, s.as_ref())) + .join(", "); + Some(format!("Did you mean {quoted_names}?")) + } + } +} + +impl From for CommandError { + fn from(err: io::Error) -> Self { + if err.kind() == io::ErrorKind::BrokenPipe { + CommandError::BrokenPipe + } else { + user_error(err) + } + } +} + +impl From for CommandError { + fn from(err: config::ConfigError) -> Self { + CommandError::ConfigError(err.to_string()) + } +} + +impl From for CommandError { + fn from(err: crate::config::ConfigError) -> Self { + CommandError::ConfigError(err.to_string()) + } +} + +impl From for CommandError { + fn from(err: RewriteRootCommit) -> Self { + internal_error_with_message("Attempted to rewrite the root commit", err) + } +} + +impl From for CommandError { + fn from(err: EditCommitError) -> Self { + internal_error_with_message("Failed to edit a commit", err) + } +} + +impl From for CommandError { + fn from(err: CheckOutCommitError) -> Self { + internal_error_with_message("Failed to check out a commit", err) + } +} + +impl From for CommandError { + fn from(err: BackendError) -> Self { + internal_error_with_message("Unexpected error from backend", err) + } +} + +impl From for CommandError { + fn from(err: WorkspaceInitError) -> Self { + match err { + WorkspaceInitError::DestinationExists(_) => { + user_error("The target repo already exists") + } + WorkspaceInitError::NonUnicodePath => { + user_error("The target repo path contains non-unicode characters") + } + WorkspaceInitError::CheckOutCommit(err) => { + internal_error_with_message("Failed to check out the initial commit", err) + } + WorkspaceInitError::Path(err) => { + internal_error_with_message("Failed to access the repository", err) + } + WorkspaceInitError::Backend(err) => { + user_error_with_message("Failed to access the repository", err) + } + WorkspaceInitError::WorkingCopyState(err) => { + internal_error_with_message("Failed to access the repository", err) + } + WorkspaceInitError::SignInit(err @ SignInitError::UnknownBackend(_)) => user_error(err), + WorkspaceInitError::SignInit(err) => internal_error(err), + } + } +} + +impl From for CommandError { + fn from(err: OpHeadResolutionError) -> Self { + match err { + OpHeadResolutionError::NoHeads => { + internal_error_with_message("Corrupt repository", err) + } + } + } +} + +impl From for CommandError { + fn from(err: OpsetEvaluationError) -> Self { + match err { + OpsetEvaluationError::OpsetResolution(err) => user_error(err), + OpsetEvaluationError::OpHeadResolution(err) => err.into(), + OpsetEvaluationError::OpStore(err) => err.into(), + } + } +} + +impl From for CommandError { + fn from(err: SnapshotError) -> Self { + match err { + SnapshotError::NewFileTooLarge { .. } => user_error_with_message_and_hint( + "Failed to snapshot the working copy", + r#"Increase the value of the `snapshot.max-new-file-size` config option if you +want this file to be snapshotted. Otherwise add it to your `.gitignore` file."#, + err, + ), + err => internal_error_with_message("Failed to snapshot the working copy", err), + } + } +} + +impl From for CommandError { + fn from(err: TreeMergeError) -> Self { + internal_error_with_message("Merge failed", err) + } +} + +impl From for CommandError { + fn from(err: OpStoreError) -> Self { + internal_error_with_message("Failed to load an operation", err) + } +} + +impl From for CommandError { + fn from(err: RepoLoaderError) -> Self { + internal_error_with_message("Failed to load the repo", err) + } +} + +impl From for CommandError { + fn from(err: ResetError) -> Self { + internal_error_with_message("Failed to reset the working copy", err) + } +} + +impl From for CommandError { + fn from(err: DiffEditError) -> Self { + user_error_with_message("Failed to edit diff", err) + } +} + +impl From for CommandError { + fn from(err: DiffGenerateError) -> Self { + user_error_with_message("Failed to generate diff", err) + } +} + +impl From for CommandError { + fn from(err: ConflictResolveError) -> Self { + user_error_with_message("Failed to resolve conflicts", err) + } +} + +impl From for CommandError { + fn from(err: MergeToolConfigError) -> Self { + user_error_with_message("Failed to load tool configuration", err) + } +} + +impl From for CommandError { + fn from(err: git2::Error) -> Self { + user_error_with_message("Git operation failed", err) + } +} + +impl From for CommandError { + fn from(err: GitImportError) -> Self { + let message = format!("Failed to import refs from underlying Git repo: {err}"); + let hint = match &err { + GitImportError::MissingHeadTarget { .. } + | GitImportError::MissingRefAncestor { .. } => Some( + "\ +Is this Git repository a shallow or partial clone (cloned with the --depth or --filter \ + argument)? +jj currently does not support shallow/partial clones. To use jj with this \ + repository, try +unshallowing the repository (https://stackoverflow.com/q/6802145) or re-cloning with the full +repository contents." + .to_string(), + ), + GitImportError::RemoteReservedForLocalGitRepo => { + Some("Run `jj git remote rename` to give different name.".to_string()) + } + GitImportError::InternalBackend(_) => None, + GitImportError::InternalGitError(_) => None, + GitImportError::UnexpectedBackend => None, + }; + user_error_with_hint_opt(message, hint) + } +} + +impl From for CommandError { + fn from(err: GitExportError) -> Self { + internal_error_with_message("Failed to export refs to underlying Git repo", err) + } +} + +impl From for CommandError { + fn from(err: GitRemoteManagementError) -> Self { + user_error(err) + } +} + +impl From for CommandError { + fn from(err: RevsetEvaluationError) -> Self { + user_error(err) + } +} + +impl From for CommandError { + fn from(err: RevsetParseError) -> Self { + let err_chain = iter::successors(Some(&err), |e| e.origin()); + let message = err_chain.clone().join("\n"); + // Only for the bottom error, which is usually the root cause + let hint = match err_chain.last().unwrap().kind() { + RevsetParseErrorKind::NotPrefixOperator { + op: _, + similar_op, + description, + } + | RevsetParseErrorKind::NotPostfixOperator { + op: _, + similar_op, + description, + } + | RevsetParseErrorKind::NotInfixOperator { + op: _, + similar_op, + description, + } => Some(format!("Did you mean '{similar_op}' for {description}?")), + RevsetParseErrorKind::NoSuchFunction { + name: _, + candidates, + } => format_similarity_hint(candidates), + _ => None, + }; + user_error_with_hint_opt(format!("Failed to parse revset: {message}"), hint) + } +} + +impl From for CommandError { + fn from(err: RevsetResolutionError) -> Self { + let hint = match &err { + RevsetResolutionError::NoSuchRevision { + name: _, + candidates, + } => format_similarity_hint(candidates), + RevsetResolutionError::EmptyString + | RevsetResolutionError::WorkspaceMissingWorkingCopy { .. } + | RevsetResolutionError::AmbiguousCommitIdPrefix(_) + | RevsetResolutionError::AmbiguousChangeIdPrefix(_) + | RevsetResolutionError::StoreError(_) => None, + }; + user_error_with_hint_opt(err, hint) + } +} + +impl From for CommandError { + fn from(err: TemplateParseError) -> Self { + let err_chain = iter::successors(Some(&err), |e| e.origin()); + let message = err_chain.clone().join("\n"); + // Only for the bottom error, which is usually the root cause + let hint = match err_chain.last().unwrap().kind() { + TemplateParseErrorKind::NoSuchKeyword { candidates, .. } + | TemplateParseErrorKind::NoSuchFunction { candidates, .. } + | TemplateParseErrorKind::NoSuchMethod { candidates, .. } => { + format_similarity_hint(candidates) + } + _ => None, + }; + user_error_with_hint_opt(format!("Failed to parse template: {message}"), hint) + } +} + +impl From for CommandError { + fn from(err: FsPathParseError) -> Self { + user_error(err) + } +} + +impl From for CommandError { + fn from(err: clap::Error) -> Self { + CommandError::ClapCliError(Arc::new(err)) + } +} + +impl From for CommandError { + fn from(err: GitConfigParseError) -> Self { + internal_error_with_message("Failed to parse Git config", err) + } +} + +impl From for CommandError { + fn from(err: WorkingCopyStateError) -> Self { + internal_error_with_message("Failed to access working copy state", err) + } +} + +impl From for CommandError { + fn from(err: GitIgnoreError) -> Self { + user_error_with_message("Failed to process .gitignore.", err) + } +} diff --git a/cli/src/commands/abandon.rs b/cli/src/commands/abandon.rs index 896e9db49..ee76c0d7d 100644 --- a/cli/src/commands/abandon.rs +++ b/cli/src/commands/abandon.rs @@ -17,9 +17,8 @@ use std::io::Write; use jj_lib::object_id::ObjectId; use tracing::instrument; -use crate::cli_util::{ - resolve_multiple_nonempty_revsets, CommandError, CommandHelper, RevisionArg, -}; +use crate::cli_util::{resolve_multiple_nonempty_revsets, CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::ui::Ui; /// Abandon a revision diff --git a/cli/src/commands/backout.rs b/cli/src/commands/backout.rs index abfd009b1..1538b848e 100644 --- a/cli/src/commands/backout.rs +++ b/cli/src/commands/backout.rs @@ -16,7 +16,8 @@ use jj_lib::object_id::ObjectId; use jj_lib::rewrite::back_out_commit; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::ui::Ui; /// Apply the reverse of a revision on top of another revision diff --git a/cli/src/commands/bench.rs b/cli/src/commands/bench.rs index 49b3c6ab7..865dd8cca 100644 --- a/cli/src/commands/bench.rs +++ b/cli/src/commands/bench.rs @@ -23,7 +23,8 @@ use criterion::{BatchSize, BenchmarkGroup, BenchmarkId, Criterion}; use jj_lib::object_id::HexPrefix; use jj_lib::repo::Repo; -use crate::cli_util::{CommandError, CommandHelper, WorkspaceCommandHelper}; +use crate::cli_util::{CommandHelper, WorkspaceCommandHelper}; +use crate::command_error::CommandError; use crate::ui::Ui; /// Commands for benchmarking internal operations diff --git a/cli/src/commands/branch.rs b/cli/src/commands/branch.rs index b4b5e82c4..9f936ffed 100644 --- a/cli/src/commands/branch.rs +++ b/cli/src/commands/branch.rs @@ -28,9 +28,9 @@ use jj_lib::str_util::StringPattern; use jj_lib::view::View; use crate::cli_util::{ - parse_string_pattern, user_error, user_error_with_hint, CommandError, CommandHelper, - RemoteBranchName, RemoteBranchNamePattern, RevisionArg, + parse_string_pattern, CommandHelper, RemoteBranchName, RemoteBranchNamePattern, RevisionArg, }; +use crate::command_error::{user_error, user_error_with_hint, CommandError}; use crate::formatter::Formatter; use crate::ui::Ui; diff --git a/cli/src/commands/cat.rs b/cli/src/commands/cat.rs index 840b6960f..128ed73a2 100644 --- a/cli/src/commands/cat.rs +++ b/cli/src/commands/cat.rs @@ -19,7 +19,8 @@ use jj_lib::repo::Repo; use pollster::FutureExt; use tracing::instrument; -use crate::cli_util::{user_error, CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::{user_error, CommandError}; use crate::ui::Ui; /// Print contents of a file in a revision diff --git a/cli/src/commands/checkout.rs b/cli/src/commands/checkout.rs index 7d446faed..329f18605 100644 --- a/cli/src/commands/checkout.rs +++ b/cli/src/commands/checkout.rs @@ -15,7 +15,8 @@ use jj_lib::object_id::ObjectId; use tracing::instrument; -use crate::cli_util::{join_message_paragraphs, CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{join_message_paragraphs, CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::ui::Ui; /// Create a new, empty change and edit it in the working copy diff --git a/cli/src/commands/chmod.rs b/cli/src/commands/chmod.rs index 6c91a427f..dfcd0ab1f 100644 --- a/cli/src/commands/chmod.rs +++ b/cli/src/commands/chmod.rs @@ -18,7 +18,8 @@ use jj_lib::merged_tree::MergedTreeBuilder; use jj_lib::object_id::ObjectId; use tracing::instrument; -use crate::cli_util::{user_error, CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::{user_error, CommandError}; use crate::ui::Ui; #[derive(Copy, Clone, Debug, PartialEq, Eq, PartialOrd, Ord, clap::ValueEnum)] diff --git a/cli/src/commands/commit.rs b/cli/src/commands/commit.rs index a6f4906ae..5fa566afe 100644 --- a/cli/src/commands/commit.rs +++ b/cli/src/commands/commit.rs @@ -17,7 +17,8 @@ use jj_lib::repo::Repo; use jj_lib::rewrite::merge_commit_trees; use tracing::instrument; -use crate::cli_util::{join_message_paragraphs, user_error, CommandError, CommandHelper}; +use crate::cli_util::{join_message_paragraphs, CommandHelper}; +use crate::command_error::{user_error, CommandError}; use crate::description_util::{description_template_for_commit, edit_description}; use crate::ui::Ui; diff --git a/cli/src/commands/config.rs b/cli/src/commands/config.rs index 3ed0efbca..e39f27e21 100644 --- a/cli/src/commands/config.rs +++ b/cli/src/commands/config.rs @@ -19,9 +19,10 @@ use itertools::Itertools; use tracing::instrument; use crate::cli_util::{ - get_new_config_file_path, run_ui_editor, serialize_config_value, user_error, - write_config_value_to_file, CommandError, CommandHelper, + get_new_config_file_path, run_ui_editor, serialize_config_value, write_config_value_to_file, + CommandHelper, }; +use crate::command_error::{user_error, CommandError}; use crate::config::{AnnotatedValue, ConfigSource}; use crate::ui::Ui; diff --git a/cli/src/commands/debug.rs b/cli/src/commands/debug.rs index f458b17f1..5224d62fe 100644 --- a/cli/src/commands/debug.rs +++ b/cli/src/commands/debug.rs @@ -23,7 +23,8 @@ use jj_lib::object_id::ObjectId; use jj_lib::working_copy::WorkingCopy; use jj_lib::{op_walk, revset}; -use crate::cli_util::{internal_error, user_error, CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::{internal_error, user_error, CommandError}; use crate::template_parser; use crate::ui::Ui; diff --git a/cli/src/commands/describe.rs b/cli/src/commands/describe.rs index c6d15eb16..99b2e0ec8 100644 --- a/cli/src/commands/describe.rs +++ b/cli/src/commands/describe.rs @@ -17,7 +17,8 @@ use std::io::{self, Read, Write}; use jj_lib::object_id::ObjectId; use tracing::instrument; -use crate::cli_util::{join_message_paragraphs, CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{join_message_paragraphs, CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::description_util::{description_template_for_describe, edit_description}; use crate::ui::Ui; diff --git a/cli/src/commands/diff.rs b/cli/src/commands/diff.rs index 5a072fa01..4677b5071 100644 --- a/cli/src/commands/diff.rs +++ b/cli/src/commands/diff.rs @@ -15,7 +15,8 @@ use jj_lib::rewrite::merge_commit_trees; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::diff_util::{diff_formats_for, show_diff, DiffFormatArgs}; use crate::ui::Ui; diff --git a/cli/src/commands/diffedit.rs b/cli/src/commands/diffedit.rs index 6adf43f72..1345667c7 100644 --- a/cli/src/commands/diffedit.rs +++ b/cli/src/commands/diffedit.rs @@ -19,7 +19,8 @@ use jj_lib::object_id::ObjectId; use jj_lib::rewrite::merge_commit_trees; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::ui::Ui; /// Touch up the content changes in a revision with a diff editor diff --git a/cli/src/commands/duplicate.rs b/cli/src/commands/duplicate.rs index 703bdd959..6a3c09166 100644 --- a/cli/src/commands/duplicate.rs +++ b/cli/src/commands/duplicate.rs @@ -20,9 +20,9 @@ use jj_lib::repo::Repo; use tracing::instrument; use crate::cli_util::{ - resolve_multiple_nonempty_revsets, short_commit_hash, user_error, CommandError, CommandHelper, - RevisionArg, + resolve_multiple_nonempty_revsets, short_commit_hash, CommandHelper, RevisionArg, }; +use crate::command_error::{user_error, CommandError}; use crate::ui::Ui; /// Create a new change with the same content as an existing one diff --git a/cli/src/commands/edit.rs b/cli/src/commands/edit.rs index f48aecb04..9a4aac7c6 100644 --- a/cli/src/commands/edit.rs +++ b/cli/src/commands/edit.rs @@ -17,7 +17,8 @@ use std::io::Write; use jj_lib::object_id::ObjectId; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::ui::Ui; /// Edit a commit in the working copy diff --git a/cli/src/commands/files.rs b/cli/src/commands/files.rs index bbc3df4b8..d058cf843 100644 --- a/cli/src/commands/files.rs +++ b/cli/src/commands/files.rs @@ -16,7 +16,8 @@ use std::io::Write; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::ui::Ui; /// List files in a revision diff --git a/cli/src/commands/git.rs b/cli/src/commands/git.rs index e0cffbcf0..4eb49ce80 100644 --- a/cli/src/commands/git.rs +++ b/cli/src/commands/git.rs @@ -42,10 +42,13 @@ use maplit::hashset; use crate::cli_util::{ parse_string_pattern, print_trackable_remote_branches, resolve_multiple_nonempty_revsets, - short_change_hash, short_commit_hash, start_repo_transaction, user_error, user_error_with_hint, - user_error_with_hint_opt, user_error_with_message, CommandError, CommandHelper, RevisionArg, + short_change_hash, short_commit_hash, start_repo_transaction, CommandHelper, RevisionArg, WorkspaceCommandHelper, }; +use crate::command_error::{ + user_error, user_error_with_hint, user_error_with_hint_opt, user_error_with_message, + CommandError, +}; use crate::git_util::{ get_git_repo, is_colocated_git_workspace, print_failed_git_export, print_git_import_stats, with_remote_git_callbacks, diff --git a/cli/src/commands/init.rs b/cli/src/commands/init.rs index dcd53dffd..6f442a3fe 100644 --- a/cli/src/commands/init.rs +++ b/cli/src/commands/init.rs @@ -20,7 +20,8 @@ use jj_lib::workspace::Workspace; use tracing::instrument; use super::git; -use crate::cli_util::{user_error_with_hint, user_error_with_message, CommandError, CommandHelper}; +use crate::cli_util::CommandHelper; +use crate::command_error::{user_error_with_hint, user_error_with_message, CommandError}; use crate::ui::Ui; /// Create a new repo in the given directory diff --git a/cli/src/commands/interdiff.rs b/cli/src/commands/interdiff.rs index b83f5f3b6..f551a87b1 100644 --- a/cli/src/commands/interdiff.rs +++ b/cli/src/commands/interdiff.rs @@ -16,7 +16,8 @@ use clap::ArgGroup; use jj_lib::rewrite::rebase_to_dest_parent; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::diff_util::{self, DiffFormatArgs}; use crate::ui::Ui; diff --git a/cli/src/commands/log.rs b/cli/src/commands/log.rs index 83ea7f2fe..8bf661fff 100644 --- a/cli/src/commands/log.rs +++ b/cli/src/commands/log.rs @@ -21,7 +21,8 @@ use jj_lib::revset_graph::{ }; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, LogContentFormat, RevisionArg}; +use crate::cli_util::{CommandHelper, LogContentFormat, RevisionArg}; +use crate::command_error::CommandError; use crate::diff_util::{self, DiffFormatArgs}; use crate::graphlog::{get_graphlog, Edge}; use crate::ui::Ui; diff --git a/cli/src/commands/merge.rs b/cli/src/commands/merge.rs index f1e1b7c4c..13fd00e11 100644 --- a/cli/src/commands/merge.rs +++ b/cli/src/commands/merge.rs @@ -15,7 +15,8 @@ use tracing::instrument; use super::new; -use crate::cli_util::{CommandError, CommandHelper}; +use crate::cli_util::CommandHelper; +use crate::command_error::CommandError; use crate::ui::Ui; #[instrument(skip_all)] diff --git a/cli/src/commands/mod.rs b/cli/src/commands/mod.rs index 6acca1c5c..e307e9658 100644 --- a/cli/src/commands/mod.rs +++ b/cli/src/commands/mod.rs @@ -62,7 +62,8 @@ use std::fmt::Debug; use clap::{CommandFactory, FromArgMatches, Subcommand}; use tracing::instrument; -use crate::cli_util::{user_error_with_hint, Args, CommandError, CommandHelper}; +use crate::cli_util::{Args, CommandHelper}; +use crate::command_error::{user_error_with_hint, CommandError}; use crate::ui::Ui; #[derive(clap::Parser, Clone, Debug)] diff --git a/cli/src/commands/move.rs b/cli/src/commands/move.rs index ea47374c9..bfae6f06c 100644 --- a/cli/src/commands/move.rs +++ b/cli/src/commands/move.rs @@ -18,7 +18,8 @@ use jj_lib::repo::Repo; use jj_lib::rewrite::merge_commit_trees; use tracing::instrument; -use crate::cli_util::{user_error, CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::{user_error, CommandError}; use crate::description_util::combine_messages; use crate::ui::Ui; diff --git a/cli/src/commands/new.rs b/cli/src/commands/new.rs index e17657996..17222316e 100644 --- a/cli/src/commands/new.rs +++ b/cli/src/commands/new.rs @@ -22,9 +22,8 @@ use jj_lib::revset::{RevsetExpression, RevsetIteratorExt}; use jj_lib::rewrite::{merge_commit_trees, rebase_commit}; use tracing::instrument; -use crate::cli_util::{ - self, short_commit_hash, user_error, CommandError, CommandHelper, RevisionArg, -}; +use crate::cli_util::{self, short_commit_hash, CommandHelper, RevisionArg}; +use crate::command_error::{user_error, CommandError}; use crate::ui::Ui; /// Create a new, empty change and (by default) edit it in the working copy diff --git a/cli/src/commands/next.rs b/cli/src/commands/next.rs index ead6ecbf9..d1193e4ee 100644 --- a/cli/src/commands/next.rs +++ b/cli/src/commands/next.rs @@ -19,9 +19,8 @@ use jj_lib::commit::Commit; use jj_lib::repo::Repo; use jj_lib::revset::{RevsetExpression, RevsetIteratorExt}; -use crate::cli_util::{ - short_commit_hash, user_error, CommandError, CommandHelper, WorkspaceCommandHelper, -}; +use crate::cli_util::{short_commit_hash, CommandHelper, WorkspaceCommandHelper}; +use crate::command_error::{user_error, CommandError}; use crate::ui::Ui; /// Move the current working copy commit to the next child revision in the diff --git a/cli/src/commands/obslog.rs b/cli/src/commands/obslog.rs index 5994b8fe2..069ef8508 100644 --- a/cli/src/commands/obslog.rs +++ b/cli/src/commands/obslog.rs @@ -18,9 +18,8 @@ use jj_lib::matchers::EverythingMatcher; use jj_lib::rewrite::rebase_to_dest_parent; use tracing::instrument; -use crate::cli_util::{ - CommandError, CommandHelper, LogContentFormat, RevisionArg, WorkspaceCommandHelper, -}; +use crate::cli_util::{CommandHelper, LogContentFormat, RevisionArg, WorkspaceCommandHelper}; +use crate::command_error::CommandError; use crate::diff_util::{self, DiffFormat, DiffFormatArgs}; use crate::formatter::Formatter; use crate::graphlog::{get_graphlog, Edge}; diff --git a/cli/src/commands/operation.rs b/cli/src/commands/operation.rs index f5cf52781..88190f17e 100644 --- a/cli/src/commands/operation.rs +++ b/cli/src/commands/operation.rs @@ -23,10 +23,8 @@ use jj_lib::op_walk; use jj_lib::operation::Operation; use jj_lib::repo::Repo; -use crate::cli_util::{ - short_operation_hash, user_error, user_error_with_hint, CommandError, CommandHelper, - LogContentFormat, -}; +use crate::cli_util::{short_operation_hash, CommandHelper, LogContentFormat}; +use crate::command_error::{user_error, user_error_with_hint, CommandError}; use crate::graphlog::{get_graphlog, Edge}; use crate::operation_templater; use crate::templater::Template as _; diff --git a/cli/src/commands/prev.rs b/cli/src/commands/prev.rs index 2da2fadf2..58af0e1ff 100644 --- a/cli/src/commands/prev.rs +++ b/cli/src/commands/prev.rs @@ -16,7 +16,8 @@ use itertools::Itertools; use jj_lib::repo::Repo; use jj_lib::revset::{RevsetExpression, RevsetIteratorExt}; -use crate::cli_util::{short_commit_hash, user_error, CommandError, CommandHelper}; +use crate::cli_util::{short_commit_hash, CommandHelper}; +use crate::command_error::{user_error, CommandError}; use crate::commands::next::choose_commit; use crate::ui::Ui; diff --git a/cli/src/commands/rebase.rs b/cli/src/commands/rebase.rs index abe6c8152..0e9534e1e 100644 --- a/cli/src/commands/rebase.rs +++ b/cli/src/commands/rebase.rs @@ -29,9 +29,10 @@ use jj_lib::settings::UserSettings; use tracing::instrument; use crate::cli_util::{ - self, resolve_multiple_nonempty_revsets_default_single, short_commit_hash, user_error, - CommandError, CommandHelper, RevisionArg, WorkspaceCommandHelper, + self, resolve_multiple_nonempty_revsets_default_single, short_commit_hash, CommandHelper, + RevisionArg, WorkspaceCommandHelper, }; +use crate::command_error::{user_error, CommandError}; use crate::ui::Ui; /// Move revisions to different parent(s) diff --git a/cli/src/commands/resolve.rs b/cli/src/commands/resolve.rs index 0e5370cc7..3512e0189 100644 --- a/cli/src/commands/resolve.rs +++ b/cli/src/commands/resolve.rs @@ -22,7 +22,8 @@ use jj_lib::object_id::ObjectId; use jj_lib::repo_path::RepoPathBuf; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, WorkspaceCommandHelper}; +use crate::cli_util::{CommandHelper, WorkspaceCommandHelper}; +use crate::command_error::CommandError; use crate::formatter::Formatter; use crate::ui::Ui; diff --git a/cli/src/commands/restore.rs b/cli/src/commands/restore.rs index f9d11bdb6..5c71005f1 100644 --- a/cli/src/commands/restore.rs +++ b/cli/src/commands/restore.rs @@ -18,7 +18,8 @@ use jj_lib::object_id::ObjectId; use jj_lib::rewrite::{merge_commit_trees, restore_tree}; use tracing::instrument; -use crate::cli_util::{user_error, CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::{user_error, CommandError}; use crate::ui::Ui; /// Restore paths from another revision diff --git a/cli/src/commands/root.rs b/cli/src/commands/root.rs index 54794f7a2..c3231dd0d 100644 --- a/cli/src/commands/root.rs +++ b/cli/src/commands/root.rs @@ -14,7 +14,8 @@ use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper}; +use crate::cli_util::CommandHelper; +use crate::command_error::CommandError; use crate::commands::workspace; use crate::ui::Ui; diff --git a/cli/src/commands/run.rs b/cli/src/commands/run.rs index 82f28c5aa..0dae7b3c9 100644 --- a/cli/src/commands/run.rs +++ b/cli/src/commands/run.rs @@ -14,9 +14,8 @@ //! This file contains the internal implementation of `run`. -use crate::cli_util::{ - resolve_multiple_nonempty_revsets, user_error, CommandError, CommandHelper, RevisionArg, -}; +use crate::cli_util::{resolve_multiple_nonempty_revsets, CommandHelper, RevisionArg}; +use crate::command_error::{user_error, CommandError}; use crate::ui::Ui; /// Run a command across a set of revisions. diff --git a/cli/src/commands/show.rs b/cli/src/commands/show.rs index 082d8c64f..19e276a14 100644 --- a/cli/src/commands/show.rs +++ b/cli/src/commands/show.rs @@ -15,7 +15,8 @@ use jj_lib::matchers::EverythingMatcher; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::diff_util::{self, DiffFormatArgs}; use crate::ui::Ui; diff --git a/cli/src/commands/sparse.rs b/cli/src/commands/sparse.rs index bf313789f..fddac7bad 100644 --- a/cli/src/commands/sparse.rs +++ b/cli/src/commands/sparse.rs @@ -24,9 +24,8 @@ use jj_lib::repo_path::RepoPathBuf; use jj_lib::settings::UserSettings; use tracing::instrument; -use crate::cli_util::{ - edit_temp_file, internal_error_with_message, print_checkout_stats, CommandError, CommandHelper, -}; +use crate::cli_util::{edit_temp_file, print_checkout_stats, CommandHelper}; +use crate::command_error::{internal_error_with_message, CommandError}; use crate::ui::Ui; /// Manage which paths from the working-copy commit are present in the working diff --git a/cli/src/commands/split.rs b/cli/src/commands/split.rs index ff3d91862..21c8a091f 100644 --- a/cli/src/commands/split.rs +++ b/cli/src/commands/split.rs @@ -18,7 +18,8 @@ use jj_lib::repo::Repo; use jj_lib::rewrite::merge_commit_trees; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::CommandError; use crate::description_util::{description_template_for_commit, edit_description}; use crate::ui::Ui; diff --git a/cli/src/commands/squash.rs b/cli/src/commands/squash.rs index b66e4c772..1098741e4 100644 --- a/cli/src/commands/squash.rs +++ b/cli/src/commands/squash.rs @@ -17,7 +17,8 @@ use jj_lib::object_id::ObjectId; use jj_lib::revset; use tracing::instrument; -use crate::cli_util::{self, user_error, CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{self, CommandHelper, RevisionArg}; +use crate::command_error::{user_error, CommandError}; use crate::description_util::combine_messages; use crate::ui::Ui; diff --git a/cli/src/commands/status.rs b/cli/src/commands/status.rs index 32bfd644f..07fa3224c 100644 --- a/cli/src/commands/status.rs +++ b/cli/src/commands/status.rs @@ -19,7 +19,8 @@ use jj_lib::rewrite::merge_commit_trees; use tracing::instrument; use super::resolve; -use crate::cli_util::{CommandError, CommandHelper}; +use crate::cli_util::CommandHelper; +use crate::command_error::CommandError; use crate::diff_util; use crate::ui::Ui; diff --git a/cli/src/commands/tag.rs b/cli/src/commands/tag.rs index 1dd7afc6a..6103f0bd8 100644 --- a/cli/src/commands/tag.rs +++ b/cli/src/commands/tag.rs @@ -14,7 +14,8 @@ use jj_lib::str_util::StringPattern; -use crate::cli_util::{parse_string_pattern, CommandError, CommandHelper}; +use crate::cli_util::{parse_string_pattern, CommandHelper}; +use crate::command_error::CommandError; use crate::ui::Ui; /// Manage tags. diff --git a/cli/src/commands/unsquash.rs b/cli/src/commands/unsquash.rs index db7076772..5dc10452b 100644 --- a/cli/src/commands/unsquash.rs +++ b/cli/src/commands/unsquash.rs @@ -17,7 +17,8 @@ use jj_lib::object_id::ObjectId; use jj_lib::rewrite::merge_commit_trees; use tracing::instrument; -use crate::cli_util::{user_error, CommandError, CommandHelper, RevisionArg}; +use crate::cli_util::{CommandHelper, RevisionArg}; +use crate::command_error::{user_error, CommandError}; use crate::description_util::combine_messages; use crate::ui::Ui; diff --git a/cli/src/commands/untrack.rs b/cli/src/commands/untrack.rs index a7c809195..207d5a312 100644 --- a/cli/src/commands/untrack.rs +++ b/cli/src/commands/untrack.rs @@ -21,7 +21,8 @@ use jj_lib::repo::Repo; use jj_lib::working_copy::SnapshotOptions; use tracing::instrument; -use crate::cli_util::{user_error_with_hint, CommandError, CommandHelper}; +use crate::cli_util::CommandHelper; +use crate::command_error::{user_error_with_hint, CommandError}; use crate::ui::Ui; /// Stop tracking specified paths in the working copy diff --git a/cli/src/commands/util.rs b/cli/src/commands/util.rs index d986f5255..2ec9f0c3c 100644 --- a/cli/src/commands/util.rs +++ b/cli/src/commands/util.rs @@ -20,7 +20,8 @@ use clap::{Command, Subcommand}; use jj_lib::repo::Repo; use tracing::instrument; -use crate::cli_util::{user_error, CommandError, CommandHelper}; +use crate::cli_util::CommandHelper; +use crate::command_error::{user_error, CommandError}; use crate::ui::Ui; /// Infrequently used commands such as for generating shell completions diff --git a/cli/src/commands/version.rs b/cli/src/commands/version.rs index f182d561d..9981138b6 100644 --- a/cli/src/commands/version.rs +++ b/cli/src/commands/version.rs @@ -16,7 +16,8 @@ use std::io::Write; use tracing::instrument; -use crate::cli_util::{CommandError, CommandHelper}; +use crate::cli_util::CommandHelper; +use crate::command_error::CommandError; use crate::ui::Ui; /// Display version information diff --git a/cli/src/commands/workspace.rs b/cli/src/commands/workspace.rs index c5f766213..3701df5a8 100644 --- a/cli/src/commands/workspace.rs +++ b/cli/src/commands/workspace.rs @@ -29,10 +29,10 @@ use jj_lib::workspace::Workspace; use tracing::instrument; use crate::cli_util::{ - self, check_stale_working_copy, internal_error_with_message, print_checkout_stats, - short_commit_hash, user_error, CommandError, CommandHelper, RevisionArg, WorkingCopyFreshness, - WorkspaceCommandHelper, + self, check_stale_working_copy, print_checkout_stats, short_commit_hash, CommandHelper, + RevisionArg, WorkingCopyFreshness, WorkspaceCommandHelper, }; +use crate::command_error::{internal_error_with_message, user_error, CommandError}; use crate::ui::Ui; /// Commands for working with workspaces diff --git a/cli/src/description_util.rs b/cli/src/description_util.rs index 745ab53ba..90a9cb95b 100644 --- a/cli/src/description_util.rs +++ b/cli/src/description_util.rs @@ -5,7 +5,8 @@ use jj_lib::merged_tree::MergedTree; use jj_lib::repo::ReadonlyRepo; use jj_lib::settings::UserSettings; -use crate::cli_util::{edit_temp_file, CommandError, WorkspaceCommandHelper}; +use crate::cli_util::{edit_temp_file, WorkspaceCommandHelper}; +use crate::command_error::CommandError; use crate::diff_util::{self, DiffFormat}; use crate::formatter::PlainTextFormatter; use crate::text_util; diff --git a/cli/src/diff_util.rs b/cli/src/diff_util.rs index 6f52ad294..7c35265b8 100644 --- a/cli/src/diff_util.rs +++ b/cli/src/diff_util.rs @@ -37,7 +37,8 @@ use pollster::FutureExt; use tracing::instrument; use unicode_width::UnicodeWidthStr as _; -use crate::cli_util::{CommandError, WorkspaceCommandHelper}; +use crate::cli_util::WorkspaceCommandHelper; +use crate::command_error::CommandError; use crate::config::CommandNameAndArgs; use crate::formatter::Formatter; use crate::merge_tools::{self, ExternalMergeTool}; diff --git a/cli/src/git_util.rs b/cli/src/git_util.rs index 0aeb4ec00..a19b160eb 100644 --- a/cli/src/git_util.rs +++ b/cli/src/git_util.rs @@ -30,7 +30,7 @@ use jj_lib::store::Store; use jj_lib::workspace::Workspace; use unicode_width::UnicodeWidthStr; -use crate::cli_util::{user_error, CommandError}; +use crate::command_error::{user_error, CommandError}; use crate::formatter::Formatter; use crate::progress::Progress; use crate::ui::Ui; diff --git a/cli/src/lib.rs b/cli/src/lib.rs index 32bb24a92..4fc40c934 100644 --- a/cli/src/lib.rs +++ b/cli/src/lib.rs @@ -16,6 +16,7 @@ pub mod cleanup_guard; pub mod cli_util; +pub mod command_error; pub mod commands; pub mod commit_templater; pub mod config; diff --git a/cli/src/ui.rs b/cli/src/ui.rs index d7d19a4a4..3f857bdc3 100644 --- a/cli/src/ui.rs +++ b/cli/src/ui.rs @@ -21,7 +21,7 @@ use std::{env, fmt, io, mem}; use minus::Pager as MinusPager; use tracing::instrument; -use crate::cli_util::CommandError; +use crate::command_error::CommandError; use crate::config::CommandNameAndArgs; use crate::formatter::{Formatter, FormatterFactory, LabeledWriter};