##// END OF EJS Templates
rust-cpython: add a util to get a `Repo` from a python path...
rust-cpython: add a util to get a `Repo` from a python path I suspect this will not be the last time we need to do something like this.

File last commit:

r52148:24d32981 default
r52939:28a0eb21 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)
}