// ancestors.rs // // Copyright 2018 Georges Racinet // // 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::ancestors module provided by the //! `hg-core` crate. From Python, this will be seen as `rustext.ancestor` use cindex::Index; use cpython::{ ObjectProtocol, PyClone, PyDict, PyModule, PyObject, PyResult, Python, }; use exceptions::GraphError; use hg; use hg::AncestorsIterator as CoreIterator; use hg::Revision; use std::cell::RefCell; /// Utility function to convert a Python iterable into a Vec /// /// We need this to feed to AncestorIterators constructors because /// a PyErr can arise at each step of iteration, whereas our inner objects /// expect iterables over Revision, not over some Result fn reviter_to_revvec(py: Python, revs: PyObject) -> PyResult> { revs.iter(py)? .map(|r| r.and_then(|o| o.extract::(py))) .collect() } py_class!(class AncestorsIterator |py| { // TODO RW lock ? data inner: RefCell>>; def __next__(&self) -> PyResult> { match self.inner(py).borrow_mut().next() { Some(Err(e)) => Err(GraphError::pynew(py, e)), None => Ok(None), Some(Ok(r)) => Ok(Some(r)), } } def __contains__(&self, rev: Revision) -> PyResult { self.inner(py).borrow_mut().contains(rev).map_err(|e| GraphError::pynew(py, e)) } def __iter__(&self) -> PyResult { Ok(self.clone_ref(py)) } def __new__(_cls, index: PyObject, initrevs: PyObject, stoprev: Revision, inclusive: bool) -> PyResult { let initvec = reviter_to_revvec(py, initrevs)?; let ait = match hg::AncestorsIterator::new(Index::new(py, index)?, initvec, stoprev, inclusive) { Ok(ait) => ait, Err(e) => { return Err(GraphError::pynew(py, e)); } }; AncestorsIterator::from_inner(py, ait) } }); impl AncestorsIterator { pub fn from_inner(py: Python, ait: CoreIterator) -> PyResult { Self::create_instance(py, RefCell::new(Box::new(ait))) } } /// Create the module, with __package__ given from parent pub fn init_module(py: Python, package: &str) -> PyResult { let dotted_name = &format!("{}.ancestor", package); let m = PyModule::new(py, dotted_name)?; m.add(py, "__package__", package)?; m.add( py, "__doc__", "Generic DAG ancestor algorithms - Rust implementation", )?; m.add_class::(py)?; 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) }