##// END OF EJS Templates
rust: normalize `_for_unchecked_rev` naming among revlogs and the index...
rust: normalize `_for_unchecked_rev` naming among revlogs and the index This normalizes the naming scheme between the `Revlog`, `Changelog`, etc. which is less suprising, though no real bugs could stem from this because of the type signature mismatch. The very high-level `Repo` object still uses an `UncheckedRevision` parameter for its methods because that's what most callers will want.

File last commit:

r52518:aa23b19e default
r53187:a3fa37bd default
Show More
status.rs
324 lines | 10.2 KiB | application/rls-services+xml | RustLexer
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 // status.rs
//
// Copyright 2019, 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.
//! Bindings for the `hg::status` module provided by the
Yuya Nishihara
rust-cpython: run cargo fmt
r43612 //! `hg-core` crate. From Python, this will be seen as
//! `rustext.dirstate.status`.
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 use crate::{dirstate::DirstateMap, exceptions::FallbackError};
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 use cpython::{
Arseniy Alekseyev
match: avoid rust fast path if the matcher was tampered with...
r52495 exc::ValueError, ObjectProtocol, PyBool, PyBytes, PyErr, PyList, PyObject,
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 PyResult, PyTuple, Python, PythonObject, ToPyObject,
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 };
Simon Sapin
rhg: Add support for `rhg status --copies`...
r49285 use hg::dirstate::status::StatusPath;
Raphaël Gomès
rust-status: expose DifferenceMatcher from Rust to Python
r50374 use hg::matchers::{
DifferenceMatcher, IntersectionMatcher, Matcher, NeverMatcher,
Arseniy Alekseyev
matchers: support patternmatcher in rust
r52496 PatternMatcher, UnionMatcher,
Raphaël Gomès
rust-status: expose DifferenceMatcher from Rust to Python
r50374 };
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 use hg::{
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 matchers::{AlwaysMatcher, FileMatcher, IncludeMatcher},
Arseniy Alekseyev
match: simplify the rust-side file pattern kind parsing...
r52498 parse_pattern_syntax_kind,
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 utils::{
files::{get_bytes_from_path, get_path_from_bytes},
hg_path::{HgPath, HgPathBuf},
},
Arseniy Alekseyev
match: avoid rust fast path if the matcher was tampered with...
r52495 BadMatch, DirstateStatus, IgnorePattern, PatternError, PatternFileWarning,
StatusError, StatusOptions,
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 };
Simon Sapin
rust: Move "lookup" a.k.a. "unsure" paths into `DirstateStatus` struct...
r47880 use std::borrow::Borrow;
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567
Simon Sapin
rhg: Add support for `rhg status --copies`...
r49285 fn collect_status_path_list(py: Python, paths: &[StatusPath<'_>]) -> PyList {
collect_pybytes_list(py, paths.iter().map(|item| &*item.path))
}
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 /// This will be useless once trait impls for collection are added to `PyBytes`
/// upstream.
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 fn collect_pybytes_list(
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 py: Python,
Simon Sapin
rhg: Add support for `rhg status --copies`...
r49285 iter: impl Iterator<Item = impl AsRef<HgPath>>,
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 ) -> PyList {
let list = PyList::new(py, &[]);
Simon Sapin
rhg: Add support for `rhg status --copies`...
r49285 for path in iter {
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 list.append(
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 py,
PyBytes::new(py, path.as_ref().as_bytes()).into_object(),
)
}
list
}
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 fn collect_bad_matches(
py: Python,
collection: &[(impl AsRef<HgPath>, BadMatch)],
) -> PyResult<PyList> {
let list = PyList::new(py, &[]);
let os = py.import("os")?;
let get_error_message = |code: i32| -> PyResult<_> {
os.call(
py,
"strerror",
PyTuple::new(py, &[code.to_py_object(py).into_object()]),
None,
)
};
for (path, bad_match) in collection.iter() {
let message = match bad_match {
BadMatch::OsError(code) => get_error_message(*code)?,
Raphaël Gomès
rust: run `cargo clippy`...
r50809 BadMatch::BadType(bad_type) => {
format!("unsupported file type (type is {})", bad_type)
.to_py_object(py)
.into_object()
}
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 };
list.append(
py,
(PyBytes::new(py, path.as_ref().as_bytes()), message)
.to_py_object(py)
.into_object(),
)
}
Ok(list)
}
fn handle_fallback(py: Python, err: StatusError) -> PyErr {
match err {
StatusError::Pattern(e) => {
Raphaël Gomès
rust-status: add trace-level logging for Rust status fallback for debugging...
r45062 let as_string = e.to_string();
log::trace!("Rust status fallback: `{}`", &as_string);
PyErr::new::<FallbackError, _>(py, &as_string)
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 }
e => PyErr::new::<ValueError, _>(py, e.to_string()),
}
}
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 pub fn status_wrapper(
py: Python,
dmap: DirstateMap,
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 matcher: PyObject,
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 root_dir: PyObject,
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 ignore_files: PyList,
check_exec: bool,
list_clean: bool,
list_ignored: bool,
list_unknown: bool,
Raphaël Gomès
rust-hg-cpython: update status bridge with the new `traversedir` support...
r45354 collect_traversed_dirs: bool,
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 ) -> PyResult<PyTuple> {
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 let bytes = root_dir.extract::<PyBytes>(py)?;
let root_dir = get_path_from_bytes(bytes.data(py));
let dmap: DirstateMap = dmap.to_py_object(py);
Simon Sapin
dirstate-tree: Give to `status()` mutable access to the `DirstateMap`...
r47882 let mut dmap = dmap.get_inner_mut(py);
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 let ignore_files: PyResult<Vec<_>> = ignore_files
.iter(py)
.map(|b| {
let file = b.extract::<PyBytes>(py)?;
Ok(get_path_from_bytes(file.data(py)).to_owned())
})
.collect();
let ignore_files = ignore_files?;
Simon Sapin
rhg: Add support for `rhg status --copies`...
r49285 // The caller may call `copymap.items()` separately
let list_copies = false;
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016
Raphaël Gomès
rust: fix unsound `OwningDirstateMap`...
r49864 let after_status = |res: Result<(DirstateStatus<'_>, _), StatusError>| {
let (status_res, warnings) =
res.map_err(|e| handle_fallback(py, e))?;
build_response(py, status_res, warnings)
};
Raphaël Gomès
hg-cpython: refactor matcher transformation logic...
r50242 let matcher = extract_matcher(py, matcher)?;
dmap.with_status(
&*matcher,
root_dir.to_path_buf(),
ignore_files,
StatusOptions {
check_exec,
list_clean,
list_ignored,
list_unknown,
list_copies,
collect_traversed_dirs,
},
after_status,
)
}
Arseniy Alekseyev
match: share code between includematcher and patternmatcher...
r52497 fn collect_kindpats(
py: Python,
matcher: PyObject,
) -> PyResult<Vec<IgnorePattern>> {
matcher
.getattr(py, "_kindpats")?
.iter(py)?
.map(|k| {
let k = k?;
Arseniy Alekseyev
match: simplify the rust-side file pattern kind parsing...
r52498 let syntax = parse_pattern_syntax_kind(
k.get_item(py, 0)?.extract::<PyBytes>(py)?.data(py),
Arseniy Alekseyev
match: share code between includematcher and patternmatcher...
r52497 )
.map_err(|e| handle_fallback(py, StatusError::Pattern(e)))?;
let pattern = k.get_item(py, 1)?.extract::<PyBytes>(py)?;
let pattern = pattern.data(py);
let source = k.get_item(py, 2)?.extract::<PyBytes>(py)?;
let source = get_path_from_bytes(source.data(py));
let new = IgnorePattern::new(syntax, pattern, source);
Ok(new)
})
.collect()
}
Raphaël Gomès
hg-cpython: refactor matcher transformation logic...
r50242 /// Transform a Python matcher into a Rust matcher.
fn extract_matcher(
py: Python,
matcher: PyObject,
) -> PyResult<Box<dyn Matcher + Sync>> {
Arseniy Alekseyev
match: avoid rust fast path if the matcher was tampered with...
r52495 let tampered = matcher
Arseniy Alekseyev
match: make `was_tampered_with` work recursively...
r52518 .call_method(py, "was_tampered_with_nonrec", PyTuple::empty(py), None)?
Arseniy Alekseyev
match: avoid rust fast path if the matcher was tampered with...
r52495 .extract::<PyBool>(py)?
.is_true();
if tampered {
return Err(handle_fallback(
py,
StatusError::Pattern(PatternError::UnsupportedSyntax(
"Pattern matcher was tampered with!".to_string(),
)),
));
};
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 match matcher.get_type(py).name(py).borrow() {
Raphaël Gomès
hg-cpython: refactor matcher transformation logic...
r50242 "alwaysmatcher" => Ok(Box::new(AlwaysMatcher)),
Raphaël Gomès
rust-dirstate: add support for nevermatcher...
r50247 "nevermatcher" => Ok(Box::new(NeverMatcher)),
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 "exactmatcher" => {
let files = matcher.call_method(
py,
"files",
PyTuple::new(py, &[]),
None,
)?;
let files: PyList = files.cast_into(py)?;
let files: PyResult<Vec<HgPathBuf>> = files
.iter(py)
.map(|f| {
Ok(HgPathBuf::from_bytes(
f.extract::<PyBytes>(py)?.data(py),
))
})
.collect();
Raphaël Gomès
rust-dirstate-status: add `walk_explicit` implementation, use `Matcher` trait...
r44367
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 let files = files?;
Raphaël Gomès
hg-cpython: refactor matcher transformation logic...
r50242 let file_matcher = FileMatcher::new(files)
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 .map_err(|e| PyErr::new::<ValueError, _>(py, e.to_string()))?;
Raphaël Gomès
hg-cpython: refactor matcher transformation logic...
r50242 Ok(Box::new(file_matcher))
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 }
"includematcher" => {
// Get the patterns from Python even though most of them are
// redundant with those we will parse later on, as they include
// those passed from the command line.
Arseniy Alekseyev
match: share code between includematcher and patternmatcher...
r52497 let ignore_patterns = collect_kindpats(py, matcher)?;
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016
Simon Sapin
rust: Parse "subinclude"d files along the way, not later...
r48170 let matcher = IncludeMatcher::new(ignore_patterns)
.map_err(|e| handle_fallback(py, e.into()))?;
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016
Raphaël Gomès
hg-cpython: refactor matcher transformation logic...
r50242 Ok(Box::new(matcher))
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 }
Raphaël Gomès
rust-dirstate: add `unionmatcher` to the allowed matchers...
r50244 "unionmatcher" => {
let matchers: PyResult<Vec<_>> = matcher
.getattr(py, "_matchers")?
.iter(py)?
.map(|py_matcher| extract_matcher(py, py_matcher?))
.collect();
Ok(Box::new(UnionMatcher::new(matchers?)))
}
Raphaël Gomès
rust-dirstate: add `intersectionmatcher` to the allowed matchers...
r50246 "intersectionmatcher" => {
let m1 = extract_matcher(py, matcher.getattr(py, "_m1")?)?;
let m2 = extract_matcher(py, matcher.getattr(py, "_m2")?)?;
Ok(Box::new(IntersectionMatcher::new(m1, m2)))
}
Raphaël Gomès
rust-status: expose DifferenceMatcher from Rust to Python
r50374 "differencematcher" => {
let m1 = extract_matcher(py, matcher.getattr(py, "_m1")?)?;
let m2 = extract_matcher(py, matcher.getattr(py, "_m2")?)?;
Ok(Box::new(DifferenceMatcher::new(m1, m2)))
}
Arseniy Alekseyev
matchers: support patternmatcher in rust
r52496 "patternmatcher" => {
Arseniy Alekseyev
match: share code between includematcher and patternmatcher...
r52497 let patterns = collect_kindpats(py, matcher)?;
Arseniy Alekseyev
matchers: support patternmatcher in rust
r52496
Arseniy Alekseyev
match: share code between includematcher and patternmatcher...
r52497 let matcher = PatternMatcher::new(patterns)
Arseniy Alekseyev
matchers: support patternmatcher in rust
r52496 .map_err(|e| handle_fallback(py, e.into()))?;
Ok(Box::new(matcher))
}
Raphaël Gomès
hg-cpython: fallback when encountering an unknown matcher...
r50240 e => Err(PyErr::new::<FallbackError, _>(
Raphaël Gomès
rust: do a clippy pass...
r45500 py,
format!("Unsupported matcher {}", e),
)),
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 }
}
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567
Raphaël Gomès
rust-dirstate-status: update bridge for new rust version of `dirstate.status`...
r44368 fn build_response(
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 py: Python,
Raphaël Gomès
rust-status: rename `StatusResult` to `DirstateStatus`...
r45012 status_res: DirstateStatus,
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 warnings: Vec<PatternFileWarning>,
) -> PyResult<PyTuple> {
Simon Sapin
rhg: Add support for `rhg status --copies`...
r49285 let modified = collect_status_path_list(py, &status_res.modified);
let added = collect_status_path_list(py, &status_res.added);
let removed = collect_status_path_list(py, &status_res.removed);
let deleted = collect_status_path_list(py, &status_res.deleted);
let clean = collect_status_path_list(py, &status_res.clean);
let ignored = collect_status_path_list(py, &status_res.ignored);
let unknown = collect_status_path_list(py, &status_res.unknown);
let unsure = collect_status_path_list(py, &status_res.unsure);
let bad = collect_bad_matches(py, &status_res.bad)?;
let traversed = collect_pybytes_list(py, status_res.traversed.iter());
Simon Sapin
dirstate-v2: Write .hg/dirstate back to disk on directory cache changes...
r48139 let dirty = status_res.dirty.to_py_object(py);
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 let py_warnings = PyList::new(py, &[]);
for warning in warnings.iter() {
// We use duck-typing on the Python side for dispatch, good enough for
// now.
match warning {
PatternFileWarning::InvalidSyntax(file, syn) => {
py_warnings.append(
py,
(
Raphaël Gomès
rust: run a clippy pass with the latest stable version...
r52013 PyBytes::new(py, &get_bytes_from_path(file)),
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 PyBytes::new(py, syn),
)
.to_py_object(py)
.into_object(),
);
}
PatternFileWarning::NoSuchFile(file) => py_warnings.append(
py,
Raphaël Gomès
rust: run a clippy pass with the latest stable version...
r52013 PyBytes::new(py, &get_bytes_from_path(file)).into_object(),
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 ),
}
}
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 Ok(PyTuple::new(
py,
&[
Simon Sapin
rust: Move "lookup" a.k.a. "unsure" paths into `DirstateStatus` struct...
r47880 unsure.into_object(),
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 modified.into_object(),
added.into_object(),
removed.into_object(),
deleted.into_object(),
clean.into_object(),
ignored.into_object(),
unknown.into_object(),
py_warnings.into_object(),
bad.into_object(),
Raphaël Gomès
rust-hg-cpython: update status bridge with the new `traversedir` support...
r45354 traversed.into_object(),
Simon Sapin
dirstate-v2: Write .hg/dirstate back to disk on directory cache changes...
r48139 dirty.into_object(),
Raphaël Gomès
rust-status: update rust-cpython bridge to account for the changes in core...
r45016 ][..],
))
Raphaël Gomès
rust-dirstate-status: rust-cpython bindings for `dirstate.status`...
r43567 }