##// END OF EJS Templates
dirstate: remove the python-side whitelist of allowed matchers...
dirstate: remove the python-side whitelist of allowed matchers This whitelist is too permissive because it allows matchers that contain disallowed ones deep inside, for example through `intersectionmatcher`. It is also too restrictive because it doesn't pass through some of the matchers we support, such as `patternmatcher`. It's also unnecessary because unsupported matchers raise `FallbackError` and we fall back anyway. Making this change makes more of the tests use rust code path, and therefore subtly change behavior. For example, rust status in largefiles repos seems to have strange behavior.

File last commit:

r52148:24d32981 default
r52519:865efc02 default
Show More
dagops.rs
84 lines | 2.8 KiB | application/rls-services+xml | RustLexer
Georges Racinet
rust-cpython: binding for headrevs()...
r41843 // dagops.rs
//
// Copyright 2019 Georges Racinet <georges.racinet@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::dagops` module provided by the
//! `hg-core` package.
//!
//! From Python, this will be seen as `mercurial.rustext.dagop`
Raphaël Gomès
rust: make `Revision` a newtype...
r51872 use crate::PyRevision;
rust-index: add a function to convert PyObject index for hg-core...
r44398 use crate::{conversion::rev_pyiter_collect, exceptions::GraphError};
Georges Racinet
rust-cpython: binding for headrevs()...
r41843 use cpython::{PyDict, PyModule, PyObject, PyResult, Python};
use hg::dagops;
use hg::Revision;
use std::collections::HashSet;
Georges Racinet
rust-index: using `hg::index::Index` in `hg-cpython::dagops`...
r52134 use vcsgraph::graph::Rank;
Georges Racinet
rust-cpython: binding for headrevs()...
r41843
Georges Racinet
rust-index: using `hg::index::Index` in `hg-cpython::dagops`...
r52134 use crate::revlog::py_rust_index_to_graph;
rust-index: add a function to convert PyObject index for hg-core...
r44398
Georges Racinet
rust-cpython: binding for headrevs()...
r41843 /// Using the the `index`, return heads out of any Python iterable of Revisions
///
/// This is the Rust counterpart for `mercurial.dagop.headrevs`
pub fn headrevs(
py: Python,
index: PyObject,
revs: PyObject,
Raphaël Gomès
rust: make `Revision` a newtype...
r51872 ) -> PyResult<HashSet<PyRevision>> {
Georges Racinet
rust-index: using `hg::index::Index` in `hg-cpython::dagops`...
r52134 let py_leaked = py_rust_index_to_graph(py, index)?;
Raphaël Gomès
rust-index: document safety invariants being upheld for every `unsafe` block...
r52148 // Safety: we don't leak the "faked" reference out of `UnsafePyLeaked`
Georges Racinet
rust-index: using `hg::index::Index` in `hg-cpython::dagops`...
r52134 let index = &*unsafe { py_leaked.try_borrow(py)? };
let mut as_set: HashSet<Revision> = rev_pyiter_collect(py, &revs, index)?;
dagops::retain_heads(index, &mut as_set)
Georges Racinet
rust-cpython: binding for headrevs()...
r41843 .map_err(|e| GraphError::pynew(py, e))?;
Raphaël Gomès
rust: make `Revision` a newtype...
r51872 Ok(as_set.into_iter().map(Into::into).collect())
Georges Racinet
rust-cpython: binding for headrevs()...
r41843 }
pacien
rust: expose rank computation function to python...
r49709 /// Computes the rank, i.e. the number of ancestors including itself,
/// of a node represented by its parents.
Georges Racinet
rust-index: using `hg::index::Index` in `hg-cpython::dagops`...
r52134 ///
/// Currently, the pure Rust index supports only the REVLOGV1 format, hence
/// the only possible return value is that the rank is unknown.
///
/// References:
/// - C implementation, function `index_fast_rank()`.
/// - `impl vcsgraph::graph::RankedGraph for Index` in `crate::cindex`.
pacien
rust: expose rank computation function to python...
r49709 pub fn rank(
py: Python,
Georges Racinet
rust-index: using `hg::index::Index` in `hg-cpython::dagops`...
r52134 _index: PyObject,
_p1r: PyRevision,
_p2r: PyRevision,
pacien
rust: expose rank computation function to python...
r49709 ) -> PyResult<Rank> {
Georges Racinet
rust-index: using `hg::index::Index` in `hg-cpython::dagops`...
r52134 Err(GraphError::pynew_from_vcsgraph(
py,
vcsgraph::graph::GraphReadError::InconsistentGraphData,
))
pacien
rust: expose rank computation function to python...
r49709 }
Georges Racinet
rust-cpython: binding for headrevs()...
r41843 /// Create the module, with `__package__` given from parent
pub fn init_module(py: Python, package: &str) -> PyResult<PyModule> {
let dotted_name = &format!("{}.dagop", package);
let m = PyModule::new(py, dotted_name)?;
m.add(py, "__package__", package)?;
m.add(py, "__doc__", "DAG operations - Rust implementation")?;
m.add(
py,
"headrevs",
py_fn!(py, headrevs(index: PyObject, revs: PyObject)),
)?;
pacien
rust: expose rank computation function to python...
r49709 m.add(
py,
"rank",
Raphaël Gomès
rust: make `Revision` a newtype...
r51872 py_fn!(py, rank(index: PyObject, p1r: PyRevision, p2r: PyRevision)),
pacien
rust: expose rank computation function to python...
r49709 )?;
Georges Racinet
rust-cpython: binding for headrevs()...
r41843
let sys = PyModule::import(py, "sys")?;
let sys_modules: PyDict = sys.get(py, "modules")?.extract(py)?;
sys_modules.set_item(py, dotted_name, &m)?;
// Example C code (see pyexpat.c and import.c) will "give away the
// reference", but we won't because it will be consumed once the
// Rust PyObject is dropped.
Ok(m)
}