// 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::io::Write as _; use std::process::ExitCode; use std::sync::Arc; use std::{error, io, iter, str}; use itertools::Itertools as _; use jj_lib::backend::BackendError; use jj_lib::fileset::{FilePatternParseError, FilesetParseError, FilesetParseErrorKind}; 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::{RepoPathBuf, UiPathParseError}; use jj_lib::revset::{ RevsetEvaluationError, RevsetParseError, RevsetParseErrorKind, RevsetResolutionError, }; use jj_lib::signing::SignInitError; use jj_lib::str_util::StringPatternParseError; use jj_lib::working_copy::{ResetError, SnapshotError, WorkingCopyStateError}; use jj_lib::workspace::WorkspaceInitError; use thiserror::Error; use crate::diff_util::DiffRenderError; use crate::formatter::{FormatRecorder, Formatter}; use crate::merge_tools::{ConflictResolveError, DiffEditError, MergeToolConfigError}; use crate::revset_util::UserRevsetEvaluationError; use crate::template_parser::{TemplateParseError, TemplateParseErrorKind}; use crate::ui::Ui; #[derive(Clone, Copy, Debug, Eq, PartialEq)] pub enum CommandErrorKind { User, Config, /// Invalid command line. The inner error type may be `clap::Error`. Cli, BrokenPipe, Internal, } #[derive(Clone, Debug)] pub struct CommandError { pub kind: CommandErrorKind, pub error: Arc, pub hints: Vec, } impl CommandError { pub fn new( kind: CommandErrorKind, err: impl Into>, ) -> Self { CommandError { kind, error: Arc::from(err.into()), hints: vec![], } } pub fn with_message( kind: CommandErrorKind, message: impl Into, source: impl Into>, ) -> Self { Self::new(kind, ErrorWithMessage::new(message, source)) } /// Returns error with the given plain-text `hint` attached. pub fn hinted(mut self, hint: impl Into) -> Self { self.add_hint(hint); self } /// Appends plain-text `hint` to the error. pub fn add_hint(&mut self, hint: impl Into) { self.hints.push(ErrorHint::PlainText(hint.into())); } /// Appends formatted `hint` to the error. pub fn add_formatted_hint(&mut self, hint: FormatRecorder) { self.hints.push(ErrorHint::Formatted(hint)); } /// Constructs formatted hint and appends it to the error. pub fn add_formatted_hint_with( &mut self, write: impl FnOnce(&mut dyn Formatter) -> io::Result<()>, ) { let mut formatter = FormatRecorder::new(); write(&mut formatter).expect("write() to FormatRecorder should never fail"); self.add_formatted_hint(formatter); } /// Appends 0 or more plain-text `hints` to the error. pub fn extend_hints(&mut self, hints: impl IntoIterator) { self.hints .extend(hints.into_iter().map(ErrorHint::PlainText)); } } #[derive(Clone, Debug)] pub enum ErrorHint { PlainText(String), Formatted(FormatRecorder), } /// 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 { CommandError::new(CommandErrorKind::User, err) } pub fn user_error_with_hint( err: impl Into>, hint: impl Into, ) -> CommandError { user_error(err).hinted(hint) } pub fn user_error_with_message( message: impl Into, source: impl Into>, ) -> CommandError { CommandError::with_message(CommandErrorKind::User, message, source) } pub fn config_error(err: impl Into>) -> CommandError { CommandError::new(CommandErrorKind::Config, err) } pub fn config_error_with_message( message: impl Into, source: impl Into>, ) -> CommandError { CommandError::with_message(CommandErrorKind::Config, message, source) } pub fn cli_error(err: impl Into>) -> CommandError { CommandError::new(CommandErrorKind::Cli, err) } pub fn internal_error(err: impl Into>) -> CommandError { CommandError::new(CommandErrorKind::Internal, err) } pub fn internal_error_with_message( message: impl Into, source: impl Into>, ) -> CommandError { CommandError::with_message(CommandErrorKind::Internal, 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 { let kind = match err.kind() { io::ErrorKind::BrokenPipe => CommandErrorKind::BrokenPipe, _ => CommandErrorKind::User, }; CommandError::new(kind, err) } } impl From for CommandError { fn from(err: config::ConfigError) -> Self { config_error(err) } } impl From for CommandError { fn from(err: crate::config::ConfigError) -> Self { config_error(err) } } 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 { match &err { BackendError::Unsupported(_) => user_error(err), _ => 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 { path, size, max_size, } => { // if the size difference is < 1KiB, then show exact bytes. // otherwise, show in human-readable form; this avoids weird cases // where a file is 400 bytes too large but the error says something // like '1.0MiB, maximum size allowed is ~1.0MiB' let size_diff = size.0 - max_size.0; let err_str = if size_diff <= 1024 { format!( "it is {} bytes too large; the maximum size allowed is {} bytes ({}).", size_diff, max_size.0, max_size, ) } else { format!("it is {}; the maximum size allowed is ~{}.", size, max_size,) }; user_error(format!( "Failed to snapshot the working copy\nThe file '{}' is too large to be \ snapshotted: {}", path.display(), err_str, )) .hinted(format!( "This is to prevent large files from being added on accident. You can fix \ this error by: - Adding the file to `.gitignore` - Run `jj config set --repo snapshot.max-new-file-size {}` This will increase the maximum file size allowed for new files, in this repository only. - Run `jj --config-toml 'snapshot.max-new-file-size={}' st` This will increase the maximum file size allowed for new files, for this command only.", size.0, size.0 )) } err => internal_error_with_message("Failed to snapshot the working copy", 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: DiffRenderError) -> Self { match err { DiffRenderError::DiffGenerate(_) => user_error(err), DiffRenderError::Backend(err) => err.into(), DiffRenderError::AccessDenied { .. } => user_error(err), DiffRenderError::Io(err) => err.into(), } } } 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 { match &err { MergeToolConfigError::MergeArgsNotConfigured { tool_name } => { let tool_name = tool_name.clone(); user_error_with_hint( err, format!( "To use `{tool_name}` as a merge tool, the config \ `merge-tools.{tool_name}.merge-args` must be defined (see docs for \ details)" ), ) } _ => 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 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, }; let mut cmd_err = user_error_with_message("Failed to import refs from underlying Git repo", err); cmd_err.extend_hints(hint); cmd_err } } 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: FilesetParseError) -> Self { let hint = fileset_parse_error_hint(&err); let mut cmd_err = user_error_with_message(format!("Failed to parse fileset: {}", err.kind()), err); cmd_err.extend_hints(hint); cmd_err } } impl From for CommandError { fn from(err: RevsetParseError) -> Self { let hint = revset_parse_error_hint(&err); let mut cmd_err = user_error_with_message(format!("Failed to parse revset: {}", err.kind()), err); cmd_err.extend_hints(hint); cmd_err } } impl From for CommandError { fn from(err: RevsetResolutionError) -> Self { let hint = revset_resolution_error_hint(&err); let mut cmd_err = user_error(err); cmd_err.extend_hints(hint); cmd_err } } impl From for CommandError { fn from(err: UserRevsetEvaluationError) -> Self { match err { UserRevsetEvaluationError::Resolution(err) => err.into(), UserRevsetEvaluationError::Evaluation(err) => err.into(), } } } impl From for CommandError { fn from(err: TemplateParseError) -> Self { let hint = template_parse_error_hint(&err); let mut cmd_err = user_error_with_message(format!("Failed to parse template: {}", err.kind()), err); cmd_err.extend_hints(hint); cmd_err } } impl From for CommandError { fn from(err: UiPathParseError) -> Self { user_error(err) } } impl From for CommandError { fn from(err: clap::Error) -> Self { let hint = find_source_parse_error_hint(&err); let mut cmd_err = cli_error(err); cmd_err.extend_hints(hint); cmd_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) } } fn find_source_parse_error_hint(err: &dyn error::Error) -> Option { let source = err.source()?; if let Some(source) = source.downcast_ref() { file_pattern_parse_error_hint(source) } else if let Some(source) = source.downcast_ref() { fileset_parse_error_hint(source) } else if let Some(source) = source.downcast_ref() { revset_parse_error_hint(source) } else if let Some(source) = source.downcast_ref() { revset_resolution_error_hint(source) } else if let Some(UserRevsetEvaluationError::Resolution(source)) = source.downcast_ref() { revset_resolution_error_hint(source) } else if let Some(source) = source.downcast_ref() { string_pattern_parse_error_hint(source) } else if let Some(source) = source.downcast_ref() { template_parse_error_hint(source) } else { None } } fn file_pattern_parse_error_hint(err: &FilePatternParseError) -> Option { match err { FilePatternParseError::InvalidKind(_) => None, // Suggest root:"" if input can be parsed as repo-relative path FilePatternParseError::UiPath(UiPathParseError::Fs(e)) => { RepoPathBuf::from_relative_path(&e.input).ok().map(|path| { format!(r#"Consider using root:{path:?} to specify repo-relative path"#) }) } FilePatternParseError::RelativePath(_) => None, FilePatternParseError::GlobPattern(_) => None, } } fn fileset_parse_error_hint(err: &FilesetParseError) -> Option { match err.kind() { FilesetParseErrorKind::NoSuchFunction { name: _, candidates, } => format_similarity_hint(candidates), FilesetParseErrorKind::InvalidArguments { .. } | FilesetParseErrorKind::Expression(_) => { find_source_parse_error_hint(&err) } _ => None, } } fn revset_parse_error_hint(err: &RevsetParseError) -> Option { // Only for the bottom error, which is usually the root cause let bottom_err = iter::successors(Some(err), |e| e.origin()).last().unwrap(); match bottom_err.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), RevsetParseErrorKind::InvalidFunctionArguments { .. } => { find_source_parse_error_hint(bottom_err) } _ => None, } } fn revset_resolution_error_hint(err: &RevsetResolutionError) -> Option { match err { RevsetResolutionError::NoSuchRevision { name: _, candidates, } => format_similarity_hint(candidates), RevsetResolutionError::EmptyString | RevsetResolutionError::WorkspaceMissingWorkingCopy { .. } | RevsetResolutionError::AmbiguousCommitIdPrefix(_) | RevsetResolutionError::AmbiguousChangeIdPrefix(_) | RevsetResolutionError::StoreError(_) | RevsetResolutionError::Other(_) => None, } } fn string_pattern_parse_error_hint(err: &StringPatternParseError) -> Option { match err { StringPatternParseError::InvalidKind(_) => { Some("Try prefixing with one of `exact:`, `glob:` or `substring:`".into()) } StringPatternParseError::GlobPattern(_) => None, } } fn template_parse_error_hint(err: &TemplateParseError) -> Option { // Only for the bottom error, which is usually the root cause let bottom_err = iter::successors(Some(err), |e| e.origin()).last().unwrap(); match bottom_err.kind() { TemplateParseErrorKind::NoSuchKeyword { candidates, .. } | TemplateParseErrorKind::NoSuchFunction { candidates, .. } | TemplateParseErrorKind::NoSuchMethod { candidates, .. } => { format_similarity_hint(candidates) } TemplateParseErrorKind::InvalidArguments { .. } | TemplateParseErrorKind::Expression(_) => { find_source_parse_error_hint(bottom_err) } _ => None, } } const BROKEN_PIPE_EXIT_CODE: u8 = 3; pub(crate) fn handle_command_result(ui: &mut Ui, result: Result<(), CommandError>) -> ExitCode { try_handle_command_result(ui, result).unwrap_or_else(|_| ExitCode::from(BROKEN_PIPE_EXIT_CODE)) } fn try_handle_command_result( ui: &mut Ui, result: Result<(), CommandError>, ) -> io::Result { let Err(cmd_err) = &result else { return Ok(ExitCode::SUCCESS); }; let err = &cmd_err.error; let hints = &cmd_err.hints; match cmd_err.kind { CommandErrorKind::User => { print_error(ui, "Error: ", err, hints)?; Ok(ExitCode::from(1)) } CommandErrorKind::Config => { print_error(ui, "Config error: ", err, hints)?; writeln!( ui.stderr_formatter().labeled("hint"), "For help, see https://github.com/martinvonz/jj/blob/main/docs/config.md." )?; Ok(ExitCode::from(1)) } CommandErrorKind::Cli => { if let Some(err) = err.downcast_ref::() { handle_clap_error(ui, err, hints) } else { print_error(ui, "Error: ", err, hints)?; Ok(ExitCode::from(2)) } } CommandErrorKind::BrokenPipe => { // A broken pipe is not an error, but a signal to exit gracefully. Ok(ExitCode::from(BROKEN_PIPE_EXIT_CODE)) } CommandErrorKind::Internal => { print_error(ui, "Internal error: ", err, hints)?; Ok(ExitCode::from(255)) } } } fn print_error( ui: &Ui, heading: &str, err: &dyn error::Error, hints: &[ErrorHint], ) -> io::Result<()> { writeln!(ui.error_with_heading(heading), "{err}")?; print_error_sources(ui, err.source())?; print_error_hints(ui, hints)?; Ok(()) } fn print_error_sources(ui: &Ui, source: Option<&dyn error::Error>) -> io::Result<()> { let Some(err) = source else { return Ok(()); }; ui.stderr_formatter() .with_label("error_source", |formatter| { if err.source().is_none() { write!(formatter.labeled("heading"), "Caused by: ")?; writeln!(formatter, "{err}")?; } else { writeln!(formatter.labeled("heading"), "Caused by:")?; for (i, err) in iter::successors(Some(err), |err| err.source()).enumerate() { write!(formatter.labeled("heading"), "{}: ", i + 1)?; writeln!(formatter, "{err}")?; } } Ok(()) }) } fn print_error_hints(ui: &Ui, hints: &[ErrorHint]) -> io::Result<()> { for hint in hints { ui.stderr_formatter().with_label("hint", |formatter| { write!(formatter.labeled("heading"), "Hint: ")?; match hint { ErrorHint::PlainText(message) => { writeln!(formatter, "{message}")?; Ok(()) } ErrorHint::Formatted(recorded) => { recorded.replay(formatter)?; // Formatted hint is usually multi-line text, and it's // convenient if trailing "\n" doesn't have to be omitted. if !recorded.data().ends_with(b"\n") { writeln!(formatter)?; } Ok(()) } } })?; } Ok(()) } fn handle_clap_error(ui: &mut Ui, err: &clap::Error, hints: &[ErrorHint]) -> io::Result { let clap_str = if ui.color() { err.render().ansi().to_string() } else { err.render().to_string() }; match err.kind() { clap::error::ErrorKind::DisplayHelp | clap::error::ErrorKind::DisplayHelpOnMissingArgumentOrSubcommand => ui.request_pager(), _ => {} }; // Definitions for exit codes and streams come from // https://github.com/clap-rs/clap/blob/master/src/error/mod.rs match err.kind() { clap::error::ErrorKind::DisplayHelp | clap::error::ErrorKind::DisplayVersion => { write!(ui.stdout(), "{clap_str}")?; return Ok(ExitCode::SUCCESS); } _ => {} } write!(ui.stderr(), "{clap_str}")?; print_error_hints(ui, hints)?; Ok(ExitCode::from(2)) }