##// END OF EJS Templates
rust: make RevlogError AmbiguousPrefix case contain the actual prefix...
rust: make RevlogError AmbiguousPrefix case contain the actual prefix This brings the work started in `652149ed64f0` to its logical conclusion and makes the RevlogError self-sufficient so it can be directly converted to CommandError, without an extra rev text annotation. Without this change, it's confusing that the extra annotation is ignored in most-but-not-all cases.

File last commit:

r52761:db7dbe6f default
r53240:393ad268 default
Show More
requirements.rs
185 lines | 6.7 KiB | application/rls-services+xml | RustLexer
Simon Sapin
rust: use HgError in RevlogError and Vfs...
r47172 use crate::errors::{HgError, HgResultExt};
Simon Sapin
rust: Move VFS code to its own module...
r48764 use crate::repo::Repo;
Simon Sapin
rhg: Check .hg/requires for absence of required features...
r47358 use crate::utils::join_display;
Raphaël Gomès
rust: add Vfs trait...
r52761 use crate::vfs::VfsImpl;
Simon Sapin
rhg: initial support for shared repositories...
r47190 use std::collections::HashSet;
Simon Sapin
requirements: move loading to hg-core and add parsing...
r46536
Simon Sapin
rhg: initial support for shared repositories...
r47190 fn parse(bytes: &[u8]) -> Result<HashSet<String>, HgError> {
Simon Sapin
requirements: move loading to hg-core and add parsing...
r46536 // The Python code reading this file uses `str.splitlines`
// which looks for a number of line separators (even including a couple of
// non-ASCII ones), but Python code writing it always uses `\n`.
let lines = bytes.split(|&byte| byte == b'\n');
lines
.filter(|line| !line.is_empty())
.map(|line| {
// Python uses Unicode `str.isalnum` but feature names are all
// ASCII
Simon Sapin
rhg: check that .hg/requires is ASCII...
r46550 if line[0].is_ascii_alphanumeric() && line.is_ascii() {
Simon Sapin
requirements: move loading to hg-core and add parsing...
r46536 Ok(String::from_utf8(line.into()).unwrap())
} else {
Simon Sapin
rust: replace RequirementsError with HgError...
r47171 Err(HgError::corrupted("parse error in 'requires' file"))
Simon Sapin
requirements: move loading to hg-core and add parsing...
r46536 }
})
.collect()
}
Raphaël Gomès
rust: add Vfs trait...
r52761 pub(crate) fn load(hg_vfs: VfsImpl) -> Result<HashSet<String>, HgError> {
Simon Sapin
rhg: add support for share-safe...
r47191 parse(&hg_vfs.read("requires")?)
}
Raphaël Gomès
rust: add Vfs trait...
r52761 pub(crate) fn load_if_exists(
hg_vfs: &VfsImpl,
) -> Result<HashSet<String>, HgError> {
Simon Sapin
rhg: initial support for shared repositories...
r47190 if let Some(bytes) = hg_vfs.read("requires").io_not_found_as_none()? {
Simon Sapin
rust: replace RequirementsError with HgError...
r47171 parse(&bytes)
} else {
Simon Sapin
requirements: move loading to hg-core and add parsing...
r46536 // Treat a missing file the same as an empty file.
// From `mercurial/localrepo.py`:
// > requires file contains a newline-delimited list of
// > features/capabilities the opener (us) must have in order to use
// > the repository. This file was introduced in Mercurial 0.9.2,
// > which means very old repositories may not have one. We assume
// > a missing file translates to no requirements.
Simon Sapin
rhg: initial support for shared repositories...
r47190 Ok(HashSet::new())
Simon Sapin
requirements: move loading to hg-core and add parsing...
r46536 }
}
Simon Sapin
rhg: exit with relevant code for unsupported requirements...
r46549
Simon Sapin
rhg: initial support for shared repositories...
r47190 pub(crate) fn check(repo: &Repo) -> Result<(), HgError> {
Simon Sapin
rhg: Check .hg/requires for absence of required features...
r47358 let unknown: Vec<_> = repo
.requirements()
.iter()
.map(String::as_str)
// .filter(|feature| !ALL_SUPPORTED.contains(feature.as_str()))
.filter(|feature| {
!REQUIRED.contains(feature) && !SUPPORTED.contains(feature)
})
.collect();
if !unknown.is_empty() {
return Err(HgError::unsupported(format!(
"repository requires feature unknown to this Mercurial: {}",
join_display(&unknown, ", ")
)));
}
let missing: Vec<_> = REQUIRED
.iter()
.filter(|&&feature| !repo.requirements().contains(feature))
.collect();
if !missing.is_empty() {
return Err(HgError::unsupported(format!(
"repository is missing feature required by this Mercurial: {}",
join_display(&missing, ", ")
)));
Simon Sapin
rhg: exit with relevant code for unsupported requirements...
r46549 }
Ok(())
}
Simon Sapin
rhg: Check .hg/requires for absence of required features...
r47358 /// rhg does not support repositories that are *missing* any of these features
const REQUIRED: &[&str] = &["revlogv1", "store", "fncache", "dotencode"];
/// rhg supports repository with or without these
Simon Sapin
rhg: exit with relevant code for unsupported requirements...
r46549 const SUPPORTED: &[&str] = &[
Raphaël Gomès
rust-revlog: teach the revlog opening code to read the repo options...
r52084 GENERALDELTA_REQUIREMENT,
Simon Sapin
rhg: initial support for shared repositories...
r47190 SHARED_REQUIREMENT,
Simon Sapin
rhg: add support for share-safe...
r47191 SHARESAFE_REQUIREMENT,
Simon Sapin
rhg: initial support for shared repositories...
r47190 SPARSEREVLOG_REQUIREMENT,
RELATIVE_SHARED_REQUIREMENT,
rhg: make rhg recognise it supports zstd compression for revlogs...
r47635 REVLOG_COMPRESSION_ZSTD,
Simon Sapin
rhg: Add support for dirstate-v2...
r48165 DIRSTATE_V2_REQUIREMENT,
Arseniy Alekseyev
rhg: fix bugs around [use-dirstate-tracked-hint] and repo auto-upgrade...
r50395 DIRSTATE_TRACKED_HINT_V1,
Simon Sapin
rhg: use persistent nodemap when available...
r46706 // As of this writing everything rhg does is read-only.
// When it starts writing to the repository, it’ll need to either keep the
// persistent nodemap up to date or remove this entry:
Simon Sapin
rhg: Check .hg/requires for absence of required features...
r47358 NODEMAP_REQUIREMENT,
Arseniy Alekseyev
rhg: add support for narrow clones and sparse checkouts...
r49238 // Not all commands support `sparse` and `narrow`. The commands that do
// not should opt out by checking `has_sparse` and `has_narrow`.
SPARSE_REQUIREMENT,
NARROW_REQUIREMENT,
Martin von Zweigbergk
rust-requirements: allow loading repos with `bookmarksinstore` requirement...
r49935 // rhg doesn't care about bookmarks at all yet
BOOKMARKS_IN_STORE_REQUIREMENT,
Simon Sapin
rhg: exit with relevant code for unsupported requirements...
r46549 ];
Simon Sapin
rhg: initial support for shared repositories...
r47190
// Copied from mercurial/requirements.py:
Raphaël Gomès
rust: make requirements public...
r50086 pub const DIRSTATE_V2_REQUIREMENT: &str = "dirstate-v2";
Raphaël Gomès
rust-revlog: teach the revlog opening code to read the repo options...
r52084 pub const GENERALDELTA_REQUIREMENT: &str = "generaldelta";
Simon Sapin
rhg: Add support for dirstate-v2...
r48165
auto-upgrade: introduce a way to auto-upgrade to/from tracked-hint...
r50089 /// A repository that uses the tracked hint dirstate file
#[allow(unused)]
pub const DIRSTATE_TRACKED_HINT_V1: &str = "dirstate-tracked-key-v1";
Simon Sapin
rhg: initial support for shared repositories...
r47190 /// When narrowing is finalized and no longer subject to format changes,
/// we should move this to just "narrow" or similar.
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const NARROW_REQUIREMENT: &str = "narrowhg-experimental";
Simon Sapin
rhg: initial support for shared repositories...
r47190
Martin von Zweigbergk
rust-requirements: allow loading repos with `bookmarksinstore` requirement...
r49935 /// Bookmarks must be stored in the `store` part of the repository and will be
/// share accross shares
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const BOOKMARKS_IN_STORE_REQUIREMENT: &str = "bookmarksinstore";
Martin von Zweigbergk
rust-requirements: allow loading repos with `bookmarksinstore` requirement...
r49935
Simon Sapin
rhg: initial support for shared repositories...
r47190 /// Enables sparse working directory usage
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const SPARSE_REQUIREMENT: &str = "exp-sparse";
Simon Sapin
rhg: initial support for shared repositories...
r47190
/// Enables the internal phase which is used to hide changesets instead
/// of stripping them
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const INTERNAL_PHASE_REQUIREMENT: &str = "internal-phase";
Simon Sapin
rhg: initial support for shared repositories...
r47190
/// Stores manifest in Tree structure
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const TREEMANIFEST_REQUIREMENT: &str = "treemanifest";
Simon Sapin
rhg: initial support for shared repositories...
r47190
Raphaël Gomès
rust-revlog: teach the revlog opening code to read the repo options...
r52084 /// Whether to use the "RevlogNG" or V1 of the revlog format
#[allow(unused)]
pub const REVLOGV1_REQUIREMENT: &str = "revlogv1";
Simon Sapin
rhg: initial support for shared repositories...
r47190 /// Increment the sub-version when the revlog v2 format changes to lock out old
/// clients.
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const REVLOGV2_REQUIREMENT: &str = "exp-revlogv2.1";
Simon Sapin
rhg: initial support for shared repositories...
r47190
Raphaël Gomès
rust-revlog: teach the revlog opening code to read the repo options...
r52084 /// Increment the sub-version when the revlog v2 format changes to lock out old
/// clients.
#[allow(unused)]
pub const CHANGELOGV2_REQUIREMENT: &str = "exp-changelog-v2";
Simon Sapin
rhg: initial support for shared repositories...
r47190 /// A repository with the sparserevlog feature will have delta chains that
/// can spread over a larger span. Sparse reading cuts these large spans into
/// pieces, so that each piece isn't too big.
/// Without the sparserevlog capability, reading from the repository could use
/// huge amounts of memory, because the whole span would be read at once,
/// including all the intermediate revisions that aren't pertinent for the
/// chain. This is why once a repository has enabled sparse-read, it becomes
/// required.
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const SPARSEREVLOG_REQUIREMENT: &str = "sparserevlog";
Simon Sapin
rhg: initial support for shared repositories...
r47190
/// A repository with the the copies-sidedata-changeset requirement will store
/// copies related information in changeset's sidedata.
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const COPIESSDC_REQUIREMENT: &str = "exp-copies-sidedata-changeset";
Simon Sapin
rhg: initial support for shared repositories...
r47190
/// The repository use persistent nodemap for the changelog and the manifest.
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const NODEMAP_REQUIREMENT: &str = "persistent-nodemap";
Simon Sapin
rhg: initial support for shared repositories...
r47190
/// Denotes that the current repository is a share
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const SHARED_REQUIREMENT: &str = "shared";
Simon Sapin
rhg: initial support for shared repositories...
r47190
/// Denotes that current repository is a share and the shared source path is
/// relative to the current repository root path
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const RELATIVE_SHARED_REQUIREMENT: &str = "relshared";
Simon Sapin
rhg: initial support for shared repositories...
r47190
/// A repository with share implemented safely. The repository has different
/// store and working copy requirements i.e. both `.hg/requires` and
/// `.hg/store/requires` are present.
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const SHARESAFE_REQUIREMENT: &str = "share-safe";
rhg: make rhg recognise it supports zstd compression for revlogs...
r47635
/// A repository that use zstd compression inside its revlog
#[allow(unused)]
Raphaël Gomès
rust: make requirements public...
r50086 pub const REVLOG_COMPRESSION_ZSTD: &str = "revlog-compression-zstd";