dirstate.rs
335 lines
| 10.3 KiB
| application/rls-services+xml
|
RustLexer
Raphaël Gomès
|
r42489 | // dirstate.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::dirstate` module provided by the | ||||
//! `hg-core` package. | ||||
//! | ||||
//! From Python, this will be seen as `mercurial.rustext.dirstate` | ||||
use cpython::{ | ||||
Raphaël Gomès
|
r42737 | exc, ObjectProtocol, PyBytes, PyDict, PyErr, PyInt, PyModule, PyObject, | ||
PyResult, PySequence, PyTuple, Python, PythonObject, ToPyObject, | ||||
Raphaël Gomès
|
r42489 | }; | ||
use hg::{ | ||||
Raphaël Gomès
|
r42737 | pack_dirstate, parse_dirstate, CopyVecEntry, DirsIterable, DirsMultiset, | ||
DirstateEntry, DirstateMapError, DirstatePackError, DirstateParents, | ||||
DirstateParseError, DirstateVec, | ||||
Raphaël Gomès
|
r42489 | }; | ||
use std::collections::HashMap; | ||||
use std::ffi::CStr; | ||||
Raphaël Gomès
|
r42737 | |||
Raphaël Gomès
|
r42489 | #[cfg(feature = "python27")] | ||
extern crate python27_sys as python_sys; | ||||
#[cfg(feature = "python3")] | ||||
extern crate python3_sys as python_sys; | ||||
Raphaël Gomès
|
r42737 | |||
Raphaël Gomès
|
r42489 | use self::python_sys::PyCapsule_Import; | ||
use libc::{c_char, c_int}; | ||||
Raphaël Gomès
|
r42737 | use std::cell::RefCell; | ||
Raphaël Gomès
|
r42489 | use std::mem::transmute; | ||
/// C code uses a custom `dirstate_tuple` type, checks in multiple instances | ||||
/// for this type, and raises a Python `Exception` if the check does not pass. | ||||
/// Because this type differs only in name from the regular Python tuple, it | ||||
/// would be a good idea in the near future to remove it entirely to allow | ||||
/// for a pure Python tuple of the same effective structure to be used, | ||||
/// rendering this type and the capsule below useless. | ||||
type MakeDirstateTupleFn = extern "C" fn( | ||||
state: c_char, | ||||
mode: c_int, | ||||
size: c_int, | ||||
mtime: c_int, | ||||
) -> PyObject; | ||||
/// This is largely a copy/paste from cindex.rs, pending the merge of a | ||||
/// `py_capsule_fn!` macro in the rust-cpython project: | ||||
/// https://github.com/dgrunwald/rust-cpython/pull/169 | ||||
fn decapsule_make_dirstate_tuple(py: Python) -> PyResult<MakeDirstateTupleFn> { | ||||
unsafe { | ||||
let caps_name = CStr::from_bytes_with_nul_unchecked( | ||||
b"mercurial.cext.parsers.make_dirstate_tuple_CAPI\0", | ||||
); | ||||
let from_caps = PyCapsule_Import(caps_name.as_ptr(), 0); | ||||
if from_caps.is_null() { | ||||
return Err(PyErr::fetch(py)); | ||||
} | ||||
Ok(transmute(from_caps)) | ||||
} | ||||
} | ||||
fn parse_dirstate_wrapper( | ||||
py: Python, | ||||
dmap: PyDict, | ||||
copymap: PyDict, | ||||
st: PyBytes, | ||||
) -> PyResult<PyTuple> { | ||||
match parse_dirstate(st.data(py)) { | ||||
Ok((parents, dirstate_vec, copies)) => { | ||||
for (filename, entry) in dirstate_vec { | ||||
dmap.set_item( | ||||
py, | ||||
PyBytes::new(py, &filename[..]), | ||||
decapsule_make_dirstate_tuple(py)?( | ||||
Georges Racinet
|
r42600 | entry.state as c_char, | ||
Raphaël Gomès
|
r42489 | entry.mode, | ||
entry.size, | ||||
entry.mtime, | ||||
), | ||||
)?; | ||||
} | ||||
for CopyVecEntry { path, copy_path } in copies { | ||||
copymap.set_item( | ||||
py, | ||||
PyBytes::new(py, path), | ||||
PyBytes::new(py, copy_path), | ||||
)?; | ||||
} | ||||
Ok((PyBytes::new(py, parents.p1), PyBytes::new(py, parents.p2)) | ||||
.to_py_object(py)) | ||||
} | ||||
Err(e) => Err(PyErr::new::<exc::ValueError, _>( | ||||
py, | ||||
match e { | ||||
DirstateParseError::TooLittleData => { | ||||
"too little data for parents".to_string() | ||||
} | ||||
DirstateParseError::Overflow => { | ||||
"overflow in dirstate".to_string() | ||||
} | ||||
DirstateParseError::CorruptedEntry(e) => e, | ||||
}, | ||||
)), | ||||
} | ||||
} | ||||
Raphaël Gomès
|
r42737 | fn extract_dirstate_vec( | ||
Raphaël Gomès
|
r42489 | py: Python, | ||
Raphaël Gomès
|
r42737 | dmap: &PyDict, | ||
) -> Result<DirstateVec, PyErr> { | ||||
dmap.items(py) | ||||
Raphaël Gomès
|
r42489 | .iter() | ||
.map(|(filename, stats)| { | ||||
let stats = stats.extract::<PySequence>(py)?; | ||||
let state = stats.get_item(py, 0)?.extract::<PyBytes>(py)?; | ||||
let state = state.data(py)[0] as i8; | ||||
let mode = stats.get_item(py, 1)?.extract(py)?; | ||||
let size = stats.get_item(py, 2)?.extract(py)?; | ||||
let mtime = stats.get_item(py, 3)?.extract(py)?; | ||||
let filename = filename.extract::<PyBytes>(py)?; | ||||
let filename = filename.data(py); | ||||
Ok(( | ||||
filename.to_owned(), | ||||
DirstateEntry { | ||||
state, | ||||
mode, | ||||
size, | ||||
mtime, | ||||
}, | ||||
)) | ||||
}) | ||||
Raphaël Gomès
|
r42737 | .collect() | ||
} | ||||
fn pack_dirstate_wrapper( | ||||
py: Python, | ||||
dmap: PyDict, | ||||
copymap: PyDict, | ||||
pl: PyTuple, | ||||
now: PyInt, | ||||
) -> PyResult<PyBytes> { | ||||
let p1 = pl.get_item(py, 0).extract::<PyBytes>(py)?; | ||||
let p1: &[u8] = p1.data(py); | ||||
let p2 = pl.get_item(py, 1).extract::<PyBytes>(py)?; | ||||
let p2: &[u8] = p2.data(py); | ||||
let dirstate_vec = extract_dirstate_vec(py, &dmap)?; | ||||
Raphaël Gomès
|
r42489 | |||
let copies: Result<HashMap<Vec<u8>, Vec<u8>>, PyErr> = copymap | ||||
.items(py) | ||||
.iter() | ||||
.map(|(key, value)| { | ||||
Ok(( | ||||
key.extract::<PyBytes>(py)?.data(py).to_owned(), | ||||
value.extract::<PyBytes>(py)?.data(py).to_owned(), | ||||
)) | ||||
}) | ||||
.collect(); | ||||
match pack_dirstate( | ||||
Raphaël Gomès
|
r42737 | &dirstate_vec, | ||
Raphaël Gomès
|
r42489 | &copies?, | ||
DirstateParents { p1, p2 }, | ||||
Georges Racinet
|
r42548 | now.as_object().extract::<i32>(py)?, | ||
Raphaël Gomès
|
r42489 | ) { | ||
Ok((packed, new_dirstate_vec)) => { | ||||
for ( | ||||
filename, | ||||
DirstateEntry { | ||||
state, | ||||
mode, | ||||
size, | ||||
mtime, | ||||
}, | ||||
) in new_dirstate_vec | ||||
{ | ||||
dmap.set_item( | ||||
py, | ||||
PyBytes::new(py, &filename[..]), | ||||
decapsule_make_dirstate_tuple(py)?( | ||||
Raphaël Gomès
|
r42737 | state as c_char, | ||
mode, | ||||
size, | ||||
mtime, | ||||
Raphaël Gomès
|
r42489 | ), | ||
)?; | ||||
} | ||||
Ok(PyBytes::new(py, &packed)) | ||||
} | ||||
Err(error) => Err(PyErr::new::<exc::ValueError, _>( | ||||
py, | ||||
match error { | ||||
DirstatePackError::CorruptedParent => { | ||||
"expected a 20-byte hash".to_string() | ||||
} | ||||
DirstatePackError::CorruptedEntry(e) => e, | ||||
DirstatePackError::BadSize(expected, actual) => { | ||||
format!("bad dirstate size: {} != {}", actual, expected) | ||||
} | ||||
}, | ||||
)), | ||||
} | ||||
} | ||||
Raphaël Gomès
|
r42737 | py_class!(pub class Dirs |py| { | ||
data dirs_map: RefCell<DirsMultiset>; | ||||
// `map` is either a `dict` or a flat iterator (usually a `set`, sometimes | ||||
// a `list`) | ||||
def __new__( | ||||
_cls, | ||||
map: PyObject, | ||||
skip: Option<PyObject> = None | ||||
) -> PyResult<Self> { | ||||
let mut skip_state: Option<i8> = None; | ||||
if let Some(skip) = skip { | ||||
skip_state = Some(skip.extract::<PyBytes>(py)?.data(py)[0] as i8); | ||||
} | ||||
let dirs_map; | ||||
if let Ok(map) = map.cast_as::<PyDict>(py) { | ||||
let dirstate_vec = extract_dirstate_vec(py, &map)?; | ||||
dirs_map = DirsMultiset::new( | ||||
DirsIterable::Dirstate(dirstate_vec), | ||||
skip_state, | ||||
) | ||||
} else { | ||||
let map: Result<Vec<Vec<u8>>, PyErr> = map | ||||
.iter(py)? | ||||
.map(|o| Ok(o?.extract::<PyBytes>(py)?.data(py).to_owned())) | ||||
.collect(); | ||||
dirs_map = DirsMultiset::new( | ||||
DirsIterable::Manifest(map?), | ||||
skip_state, | ||||
) | ||||
} | ||||
Self::create_instance(py, RefCell::new(dirs_map)) | ||||
} | ||||
def addpath(&self, path: PyObject) -> PyResult<PyObject> { | ||||
self.dirs_map(py).borrow_mut().add_path( | ||||
path.extract::<PyBytes>(py)?.data(py), | ||||
); | ||||
Ok(py.None()) | ||||
} | ||||
def delpath(&self, path: PyObject) -> PyResult<PyObject> { | ||||
self.dirs_map(py).borrow_mut().delete_path( | ||||
path.extract::<PyBytes>(py)?.data(py), | ||||
) | ||||
.and(Ok(py.None())) | ||||
.or_else(|e| { | ||||
match e { | ||||
DirstateMapError::PathNotFound(_p) => { | ||||
Err(PyErr::new::<exc::ValueError, _>( | ||||
py, | ||||
"expected a value, found none".to_string(), | ||||
)) | ||||
} | ||||
DirstateMapError::EmptyPath => { | ||||
Ok(py.None()) | ||||
} | ||||
} | ||||
}) | ||||
} | ||||
// This is really inefficient on top of being ugly, but it's an easy way | ||||
// of having it work to continue working on the rest of the module | ||||
// hopefully bypassing Python entirely pretty soon. | ||||
def __iter__(&self) -> PyResult<PyObject> { | ||||
let dict = PyDict::new(py); | ||||
for (key, value) in self.dirs_map(py).borrow().iter() { | ||||
dict.set_item( | ||||
py, | ||||
PyBytes::new(py, &key[..]), | ||||
value.to_py_object(py), | ||||
)?; | ||||
} | ||||
let locals = PyDict::new(py); | ||||
locals.set_item(py, "obj", dict)?; | ||||
py.eval("iter(obj)", None, Some(&locals)) | ||||
} | ||||
def __contains__(&self, item: PyObject) -> PyResult<bool> { | ||||
Ok(self | ||||
.dirs_map(py) | ||||
.borrow() | ||||
.get(&item.extract::<PyBytes>(py)?.data(py).to_owned()) | ||||
.is_some()) | ||||
} | ||||
}); | ||||
Raphaël Gomès
|
r42489 | /// Create the module, with `__package__` given from parent | ||
pub fn init_module(py: Python, package: &str) -> PyResult<PyModule> { | ||||
let dotted_name = &format!("{}.dirstate", package); | ||||
let m = PyModule::new(py, dotted_name)?; | ||||
Raphaël Gomès
|
r42737 | |||
Raphaël Gomès
|
r42489 | m.add(py, "__package__", package)?; | ||
m.add(py, "__doc__", "Dirstate - Rust implementation")?; | ||||
m.add( | ||||
py, | ||||
"parse_dirstate", | ||||
py_fn!( | ||||
py, | ||||
parse_dirstate_wrapper(dmap: PyDict, copymap: PyDict, st: PyBytes) | ||||
), | ||||
)?; | ||||
m.add( | ||||
py, | ||||
"pack_dirstate", | ||||
py_fn!( | ||||
py, | ||||
pack_dirstate_wrapper( | ||||
dmap: PyDict, | ||||
copymap: PyDict, | ||||
pl: PyTuple, | ||||
now: PyInt | ||||
) | ||||
), | ||||
)?; | ||||
Raphaël Gomès
|
r42737 | m.add_class::<Dirs>(py)?; | ||
Raphaël Gomès
|
r42489 | let sys = PyModule::import(py, "sys")?; | ||
let sys_modules: PyDict = sys.get(py, "modules")?.extract(py)?; | ||||
sys_modules.set_item(py, dotted_name, &m)?; | ||||
Ok(m) | ||||
} | ||||