# HG changeset patch # User Raphaël Gomès # Date 2024-09-30 14:55:11 # Node ID ae1ab6d71f4a3408f0660c74d9059db758af1693 # Parent 92e23ba257d1c7ec4c7a3df1fef73244291e1613 rust: implement `From` for `HgError` This will be useful in a future patch to avoid a lot of boilerplate. diff --git a/rust/hg-core/src/narrow.rs b/rust/hg-core/src/narrow.rs --- a/rust/hg-core/src/narrow.rs +++ b/rust/hg-core/src/narrow.rs @@ -24,7 +24,7 @@ const DIRSTATE_FILENAME: &str = "narrows /// as part of wire protocol commands. That means that changes to this /// data structure influence the wire protocol and should not be taken /// lightly - especially removals. -const VALID_PREFIXES: [&str; 2] = ["path:", "rootfilesin:"]; +pub const VALID_PREFIXES: [&str; 2] = ["path:", "rootfilesin:"]; /// Return the matcher for the current narrow spec, and all configuration /// warnings to display. diff --git a/rust/hg-core/src/sparse.rs b/rust/hg-core/src/sparse.rs --- a/rust/hg-core/src/sparse.rs +++ b/rust/hg-core/src/sparse.rs @@ -1,14 +1,16 @@ -use std::{collections::HashSet, path::Path}; +use std::{collections::HashSet, fmt::Display, path::Path}; -use format_bytes::{write_bytes, DisplayBytes}; +use format_bytes::{format_bytes, write_bytes, DisplayBytes}; use crate::{ errors::HgError, + exit_codes::STATE_ERROR, filepatterns::parse_pattern_file_contents, matchers::{ AlwaysMatcher, DifferenceMatcher, IncludeMatcher, Matcher, UnionMatcher, }, + narrow::VALID_PREFIXES, operations::cat, repo::Repo, requirements::SPARSE_REQUIREMENT, @@ -36,6 +38,15 @@ impl DisplayBytes for SparseConfigContex } } +impl Display for SparseConfigContext { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + match self { + SparseConfigContext::Sparse => write!(f, "sparse"), + SparseConfigContext::Narrow => write!(f, "narrow"), + } + } +} + /// Possible warnings when reading sparse configuration #[derive(Debug, derive_more::From)] pub enum SparseWarning { @@ -82,6 +93,59 @@ pub enum SparseConfigError { PatternError(PatternError), } +impl From for HgError { + fn from(value: SparseConfigError) -> Self { + match value { + SparseConfigError::IncludesAfterExcludes { context } => { + HgError::Abort { + message: format!( + "{} config cannot have includes after excludes", + context, + ), + detailed_exit_code: STATE_ERROR, + hint: None, + } + } + SparseConfigError::EntryOutsideSection { context, line } => { + HgError::Abort { + message: format!( + "{} config entry outside of section: {}", + context, + String::from_utf8_lossy(&line) + ), + detailed_exit_code: STATE_ERROR, + hint: None, + } + } + SparseConfigError::IncludesInNarrow => HgError::Abort { + message: "including other spec files using '%include' is not \ + supported in narrowspec" + .to_string(), + detailed_exit_code: STATE_ERROR, + hint: None, + }, + SparseConfigError::InvalidNarrowPrefix(vec) => HgError::Abort { + message: String::from_utf8_lossy(&format_bytes!( + b"invalid prefix on narrow pattern: {}", + vec + )) + .to_string(), + detailed_exit_code: STATE_ERROR, + hint: Some(format!( + "narrow patterns must begin with one of the following: {}", + VALID_PREFIXES.join(", ") + )), + }, + SparseConfigError::HgError(hg_error) => hg_error, + SparseConfigError::PatternError(pattern_error) => HgError::Abort { + message: pattern_error.to_string(), + detailed_exit_code: STATE_ERROR, + hint: None, + }, + } + } +} + /// Parse sparse config file content. pub(crate) fn parse_config( raw: &[u8],