##// END OF EJS Templates
rust: implement conversion of RevlogError into HgError...
rust: implement conversion of RevlogError into HgError The conversion already exists in rhg, where we need to convert to CommandError. This commit moves it to hg core. This makes it easier to code some middleware where we need to carry around a type that represents any type of hg error (HgError).

File last commit:

r52939:28a0eb21 default
r53241:09a36de5 default
Show More
layer.rs
384 lines | 12.8 KiB | application/rls-services+xml | RustLexer
Raphaël Gomès
hg-core: add basic config module...
r46803 // layer.rs
//
// Copyright 2020
// Valentin Gatien-Baron,
// Raphaël Gomès <rgomes@octobus.net>
//
// This software may be used and distributed according to the terms of the
// GNU General Public License version 2 or any later version.
Simon Sapin
rhg: Align with Python on some more error messages...
r47469 use crate::errors::HgError;
Raphaël Gomès
rust-cpython: add a util to get a `Repo` from a python path...
r52939 use crate::exit_codes::{CONFIG_ERROR_ABORT, CONFIG_PARSE_ERROR_ABORT};
Simon Sapin
rust: replace read_whole_file with std::fs::read...
r47210 use crate::utils::files::{get_bytes_from_path, get_path_from_bytes};
Simon Sapin
rhg: Align config file parse error formatting with Python...
r47465 use format_bytes::{format_bytes, write_bytes, DisplayBytes};
Raphaël Gomès
hg-core: add basic config module...
r46803 use lazy_static::lazy_static;
use regex::bytes::Regex;
use std::collections::HashMap;
use std::path::{Path, PathBuf};
lazy_static! {
static ref SECTION_RE: Regex = make_regex(r"^\[([^\[]+)\]");
static ref ITEM_RE: Regex = make_regex(r"^([^=\s][^=]*?)\s*=\s*((.*\S)?)");
/// Continuation whitespace
static ref CONT_RE: Regex = make_regex(r"^\s+(\S|\S.*\S)\s*$");
static ref EMPTY_RE: Regex = make_regex(r"^(;|#|\s*$)");
static ref COMMENT_RE: Regex = make_regex(r"^(;|#)");
/// A directive that allows for removing previous entries
static ref UNSET_RE: Regex = make_regex(r"^%unset\s+(\S+)");
/// A directive that allows for including other config files
static ref INCLUDE_RE: Regex = make_regex(r"^%include\s+(\S|\S.*\S)\s*$");
}
/// All config values separated by layers of precedence.
/// Each config source may be split in multiple layers if `%include` directives
/// are used.
/// TODO detail the general precedence
#[derive(Clone)]
pub struct ConfigLayer {
/// Mapping of the sections to their items
sections: HashMap<Vec<u8>, ConfigItem>,
/// All sections (and their items/values) in a layer share the same origin
pub origin: ConfigOrigin,
/// Whether this layer comes from a trusted user or group
pub trusted: bool,
}
impl ConfigLayer {
pub fn new(origin: ConfigOrigin) -> Self {
ConfigLayer {
sections: HashMap::new(),
trusted: true, // TODO check
origin,
}
}
Simon Sapin
rhg: Add support for --config CLI arguments...
r47254 /// Parse `--config` CLI arguments and return a layer if there’s any
pub(crate) fn parse_cli_args(
cli_config_args: impl IntoIterator<Item = impl AsRef<[u8]>>,
) -> Result<Option<Self>, ConfigError> {
fn parse_one(arg: &[u8]) -> Option<(Vec<u8>, Vec<u8>, Vec<u8>)> {
use crate::utils::SliceExt;
Simon Sapin
rhg: add limited support for the `config` sub-command...
r47255 let (section_and_item, value) = arg.split_2(b'=')?;
let (section, item) = section_and_item.trim().split_2(b'.')?;
Simon Sapin
rhg: Add support for --config CLI arguments...
r47254 Some((
section.to_owned(),
item.to_owned(),
value.trim().to_owned(),
))
}
let mut layer = Self::new(ConfigOrigin::CommandLine);
for arg in cli_config_args {
let arg = arg.as_ref();
if let Some((section, item, value)) = parse_one(arg) {
layer.add(section, item, value, None);
} else {
Pulkit Goyal
rhg: add exit code to HgError::Abort()...
r48199 Err(HgError::abort(
format!(
"abort: malformed --config option: '{}' \
Simon Sapin
rhg: Add support for --config CLI arguments...
r47254 (use --config section.name=value)",
Pulkit Goyal
rhg: add exit code to HgError::Abort()...
r48199 String::from_utf8_lossy(arg),
),
CONFIG_PARSE_ERROR_ABORT,
Raphaël Gomès
rust: add support for hints in error messages...
r50382 None,
Pulkit Goyal
rhg: add exit code to HgError::Abort()...
r48199 ))?
Simon Sapin
rhg: Add support for --config CLI arguments...
r47254 }
}
if layer.sections.is_empty() {
Ok(None)
} else {
Ok(Some(layer))
}
}
Simon Sapin
rhg: Parse per-repository configuration...
r47215 /// Returns whether this layer comes from `--config` CLI arguments
pub(crate) fn is_from_command_line(&self) -> bool {
Raphaël Gomès
rust-clippy: fix most warnings in `hg-core`...
r50825 matches!(self.origin, ConfigOrigin::CommandLine)
Simon Sapin
rhg: Parse per-repository configuration...
r47215 }
Raphaël Gomès
hg-core: add basic config module...
r46803 /// Add an entry to the config, overwriting the old one if already present.
pub fn add(
&mut self,
section: Vec<u8>,
item: Vec<u8>,
value: Vec<u8>,
line: Option<usize>,
) {
self.sections
.entry(section)
Raphaël Gomès
rust: run a clippy pass with the latest stable version...
r52013 .or_default()
Raphaël Gomès
hg-core: add basic config module...
r46803 .insert(item, ConfigValue { bytes: value, line });
}
/// Returns the config value in `<section>.<item>` if it exists
pub fn get(&self, section: &[u8], item: &[u8]) -> Option<&ConfigValue> {
Raphaël Gomès
rust-clippy: fix most warnings in `hg-core`...
r50825 self.sections.get(section)?.get(item)
Raphaël Gomès
hg-core: add basic config module...
r46803 }
Simon Sapin
rhg: Fall back to Python if unsupported extensions are enabled...
r47467 /// Returns the keys defined in the given section
pub fn iter_keys(&self, section: &[u8]) -> impl Iterator<Item = &[u8]> {
self.sections
.get(section)
.into_iter()
.flat_map(|section| section.keys().map(|vec| &**vec))
}
Simon Sapin
rhg: Add support for ui.ignore and ui.ignore.* config...
r49282 /// Returns the (key, value) pairs defined in the given section
pub fn iter_section<'layer>(
&'layer self,
section: &[u8],
) -> impl Iterator<Item = (&'layer [u8], &'layer [u8])> {
self.sections
.get(section)
.into_iter()
.flat_map(|section| section.iter().map(|(k, v)| (&**k, &*v.bytes)))
}
Simon Sapin
rhg: [encode] and [decode] config sections are not supported...
r49162 /// Returns whether any key is defined in the given section
pub fn has_non_empty_section(&self, section: &[u8]) -> bool {
self.sections
.get(section)
.map_or(false, |section| !section.is_empty())
}
Raphaël Gomès
hg-core: add basic config module...
r46803 pub fn is_empty(&self) -> bool {
self.sections.is_empty()
}
/// Returns a `Vec` of layers in order of precedence (so, in read order),
/// recursively parsing the `%include` directives if any.
pub fn parse(src: &Path, data: &[u8]) -> Result<Vec<Self>, ConfigError> {
let mut layers = vec![];
// Discard byte order mark if any
let data = if data.starts_with(b"\xef\xbb\xbf") {
&data[3..]
} else {
data
};
// TODO check if it's trusted
let mut current_layer = Self::new(ConfigOrigin::File(src.to_owned()));
let mut lines_iter =
data.split(|b| *b == b'\n').enumerate().peekable();
let mut section = b"".to_vec();
while let Some((index, bytes)) = lines_iter.next() {
Simon Sapin
rhg: Align with Python on some more error messages...
r47469 let line = Some(index + 1);
Raphaël Gomès
rust-clippy: fix most warnings in `hg-core`...
r50825 if let Some(m) = INCLUDE_RE.captures(bytes) {
Raphaël Gomès
hg-core: add basic config module...
r46803 let filename_bytes = &m[1];
Simon Sapin
rhg: Add support for environment variables in config include paths...
r47476 let filename_bytes = crate::utils::expand_vars(filename_bytes);
Simon Sapin
rust: Remove unnecessary check for absolute path before joining...
r47211 // `Path::parent` only fails for the root directory,
Simon Sapin
rhg: Parse per-repository configuration...
r47215 // which `src` can’t be since we’ve managed to open it as a
// file.
Simon Sapin
rust: Remove unnecessary check for absolute path before joining...
r47211 let dir = src
.parent()
.expect("Path::parent fail on a file we’ve read");
Simon Sapin
rhg: Parse per-repository configuration...
r47215 // `Path::join` with an absolute argument correctly ignores the
// base path
Raphaël Gomès
rust: run a clippy pass with the latest stable version...
r52013 let filename = dir.join(get_path_from_bytes(&filename_bytes));
Simon Sapin
rhg: Silently ignore missing files in config %include...
r47477 match std::fs::read(&filename) {
Ok(data) => {
layers.push(current_layer);
layers.extend(Self::parse(&filename, &data)?);
current_layer =
Self::new(ConfigOrigin::File(src.to_owned()));
Simon Sapin
rhg: Align with Python on some more error messages...
r47469 }
Simon Sapin
rhg: Silently ignore missing files in config %include...
r47477 Err(error) => {
if error.kind() != std::io::ErrorKind::NotFound {
return Err(ConfigParseError {
origin: ConfigOrigin::File(src.to_owned()),
line,
message: format_bytes!(
b"cannot include {} ({})",
filename_bytes,
format_bytes::Utf8(error)
),
}
.into());
}
}
}
Raphaël Gomès
rust-clippy: fix most warnings in `hg-core`...
r50825 } else if EMPTY_RE.captures(bytes).is_some() {
} else if let Some(m) = SECTION_RE.captures(bytes) {
Raphaël Gomès
hg-core: add basic config module...
r46803 section = m[1].to_vec();
Raphaël Gomès
rust-clippy: fix most warnings in `hg-core`...
r50825 } else if let Some(m) = ITEM_RE.captures(bytes) {
Raphaël Gomès
hg-core: add basic config module...
r46803 let item = m[1].to_vec();
let mut value = m[2].to_vec();
loop {
match lines_iter.peek() {
None => break,
Some((_, v)) => {
Raphaël Gomès
rust-clippy: fix most warnings in `hg-core`...
r50825 if COMMENT_RE.captures(v).is_some() {
} else if CONT_RE.captures(v).is_some() {
Raphaël Gomès
hg-core: add basic config module...
r46803 value.extend(b"\n");
value.extend(&m[1]);
} else {
break;
}
}
};
lines_iter.next();
}
Simon Sapin
rhg: Align with Python on some more error messages...
r47469 current_layer.add(section.clone(), item, value, line);
Raphaël Gomès
rust-clippy: fix most warnings in `hg-core`...
r50825 } else if let Some(m) = UNSET_RE.captures(bytes) {
Raphaël Gomès
hg-core: add basic config module...
r46803 if let Some(map) = current_layer.sections.get_mut(&section) {
map.remove(&m[1]);
}
} else {
Simon Sapin
rhg: Align config file parse error formatting with Python...
r47465 let message = if bytes.starts_with(b" ") {
format_bytes!(b"unexpected leading whitespace: {}", bytes)
} else {
bytes.to_owned()
};
Simon Sapin
rust: use HgError in ConfigError...
r47176 return Err(ConfigParseError {
Raphaël Gomès
hg-core: add basic config module...
r46803 origin: ConfigOrigin::File(src.to_owned()),
Simon Sapin
rhg: Align with Python on some more error messages...
r47469 line,
Simon Sapin
rhg: Align config file parse error formatting with Python...
r47465 message,
Simon Sapin
rust: use HgError in ConfigError...
r47176 }
.into());
Raphaël Gomès
hg-core: add basic config module...
r46803 }
}
if !current_layer.is_empty() {
layers.push(current_layer);
}
Ok(layers)
}
}
Simon Sapin
rust: Use the DisplayBytes trait in config printing...
r47249 impl DisplayBytes for ConfigLayer {
fn display_bytes(
&self,
out: &mut dyn std::io::Write,
) -> std::io::Result<()> {
Raphaël Gomès
hg-core: add basic config module...
r46803 let mut sections: Vec<_> = self.sections.iter().collect();
sections.sort_by(|e0, e1| e0.0.cmp(e1.0));
for (section, items) in sections.into_iter() {
Raphaël Gomès
rust-clippy: fix most warnings in `hg-core`...
r50825 let mut items: Vec<_> = items.iter().collect();
Raphaël Gomès
hg-core: add basic config module...
r46803 items.sort_by(|e0, e1| e0.0.cmp(e1.0));
for (item, config_entry) in items {
Simon Sapin
rust: Use the DisplayBytes trait in config printing...
r47249 write_bytes!(
out,
b"{}.{}={} # {}\n",
section,
item,
&config_entry.bytes,
&self.origin,
Raphaël Gomès
hg-core: add basic config module...
r46803 )?
}
}
Ok(())
}
}
/// Mapping of section item to value.
/// In the following:
/// ```text
/// [ui]
/// paginate=no
/// ```
/// "paginate" is the section item and "no" the value.
pub type ConfigItem = HashMap<Vec<u8>, ConfigValue>;
#[derive(Clone, Debug, PartialEq)]
pub struct ConfigValue {
/// The raw bytes of the value (be it from the CLI, env or from a file)
pub bytes: Vec<u8>,
/// Only present if the value comes from a file, 1-indexed.
pub line: Option<usize>,
}
Simon Sapin
rhg: Add support for colored output...
r49584 #[derive(Clone, Debug, PartialEq, Eq)]
Raphaël Gomès
hg-core: add basic config module...
r46803 pub enum ConfigOrigin {
Simon Sapin
rhg: Parse per-repository configuration...
r47215 /// From a configuration file
Raphaël Gomès
hg-core: add basic config module...
r46803 File(PathBuf),
Arseniy Alekseyev
rhg: support tweakdefaults
r50409 /// From [ui.tweakdefaults]
Tweakdefaults,
Simon Sapin
rhg: Parse per-repository configuration...
r47215 /// From a `--config` CLI argument
CommandLine,
Simon Sapin
rhg: Add parsing for the --color global CLI argument...
r49583 /// From a `--color` CLI argument
CommandLineColor,
Simon Sapin
rhg: Parse per-repository configuration...
r47215 /// From environment variables like `$PAGER` or `$EDITOR`
Raphaël Gomès
hg-core: add basic config module...
r46803 Environment(Vec<u8>),
Raphaël Gomès
rust-config: add support for default config items...
r51656 /// From configitems.toml
Defaults,
/* TODO extensions
Raphaël Gomès
hg-core: add basic config module...
r46803 * TODO Python resources?
* Others? */
}
Simon Sapin
rust: Use the DisplayBytes trait in config printing...
r47249 impl DisplayBytes for ConfigOrigin {
fn display_bytes(
&self,
out: &mut dyn std::io::Write,
) -> std::io::Result<()> {
Raphaël Gomès
hg-core: add basic config module...
r46803 match self {
Simon Sapin
rust: Use the DisplayBytes trait in config printing...
r47249 ConfigOrigin::File(p) => out.write_all(&get_bytes_from_path(p)),
ConfigOrigin::CommandLine => out.write_all(b"--config"),
Simon Sapin
rhg: Add parsing for the --color global CLI argument...
r49583 ConfigOrigin::CommandLineColor => out.write_all(b"--color"),
Simon Sapin
rust: Use the DisplayBytes trait in config printing...
r47249 ConfigOrigin::Environment(e) => write_bytes!(out, b"${}", e),
Arseniy Alekseyev
rhg: support tweakdefaults
r50409 ConfigOrigin::Tweakdefaults => {
write_bytes!(out, b"ui.tweakdefaults")
}
Raphaël Gomès
rust-cpython: add a util to get a `Repo` from a python path...
r52939 ConfigOrigin::Defaults => write_bytes!(out, b"configitems.toml"),
Raphaël Gomès
hg-core: add basic config module...
r46803 }
}
}
Simon Sapin
rust: use HgError in ConfigError...
r47176 #[derive(Debug)]
pub struct ConfigParseError {
pub origin: ConfigOrigin,
pub line: Option<usize>,
Simon Sapin
rhg: Align config file parse error formatting with Python...
r47465 pub message: Vec<u8>,
Simon Sapin
rust: use HgError in ConfigError...
r47176 }
Raphaël Gomès
rust-cpython: add a util to get a `Repo` from a python path...
r52939 impl From<ConfigParseError> for HgError {
fn from(error: ConfigParseError) -> Self {
let ConfigParseError {
origin,
line,
message,
} = error;
let line_message = if let Some(line_number) = line {
format_bytes!(b":{}", line_number.to_string().into_bytes())
} else {
Vec::new()
};
HgError::Abort {
message: String::from_utf8_lossy(&format_bytes!(
b"config error at {}{}: {}",
origin,
line_message,
message
))
.to_string(),
detailed_exit_code: CONFIG_ERROR_ABORT,
hint: None,
}
}
}
Simon Sapin
rust: replace trivial `impl From …` with `#[derive(derive_more::From)]`...
r47164 #[derive(Debug, derive_more::From)]
Raphaël Gomès
hg-core: add basic config module...
r46803 pub enum ConfigError {
Simon Sapin
rust: use HgError in ConfigError...
r47176 Parse(ConfigParseError),
Other(HgError),
Raphaël Gomès
hg-core: add basic config module...
r46803 }
Raphaël Gomès
rust-cpython: add a util to get a `Repo` from a python path...
r52939 impl From<ConfigError> for HgError {
fn from(error: ConfigError) -> Self {
match error {
ConfigError::Parse(config_parse_error) => {
Self::from(config_parse_error)
}
ConfigError::Other(hg_error) => hg_error,
}
}
}
Raphaël Gomès
hg-core: add basic config module...
r46803 fn make_regex(pattern: &'static str) -> Regex {
Regex::new(pattern).expect("expected a valid regex")
}