parsers.rs
157 lines
| 4.6 KiB
| application/rls-services+xml
|
RustLexer
Raphaël Gomès
|
r42992 | // parsers.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::parsers` module provided by the | ||||
//! `hg-core` package. | ||||
//! | ||||
//! From Python, this will be seen as `mercurial.rustext.parsers` | ||||
use cpython::{ | ||||
exc, PyBytes, PyDict, PyErr, PyInt, PyModule, PyResult, PyTuple, Python, | ||||
Yuya Nishihara
|
r43061 | PythonObject, ToPyObject, | ||
Raphaël Gomès
|
r42992 | }; | ||
use hg::{ | ||||
Antoine Cezar
|
r45916 | pack_dirstate, parse_dirstate, utils::hg_path::HgPathBuf, DirstateEntry, | ||
Simon Sapin
|
r47169 | DirstateParents, FastHashMap, PARENT_SIZE, | ||
Raphaël Gomès
|
r42992 | }; | ||
Yuya Nishihara
|
r43067 | use std::convert::TryInto; | ||
Raphaël Gomès
|
r42992 | |||
Yuya Nishihara
|
r43479 | use crate::dirstate::{extract_dirstate, make_dirstate_tuple}; | ||
Raphaël Gomès
|
r42993 | use std::time::Duration; | ||
Raphaël Gomès
|
r42992 | |||
fn parse_dirstate_wrapper( | ||||
py: Python, | ||||
dmap: PyDict, | ||||
copymap: PyDict, | ||||
st: PyBytes, | ||||
) -> PyResult<PyTuple> { | ||||
Antoine Cezar
|
r45916 | match parse_dirstate(st.data(py)) { | ||
Ok((parents, entries, copies)) => { | ||||
let dirstate_map: FastHashMap<HgPathBuf, DirstateEntry> = entries | ||||
.into_iter() | ||||
.map(|(path, entry)| (path.to_owned(), entry)) | ||||
.collect(); | ||||
let copy_map: FastHashMap<HgPathBuf, HgPathBuf> = copies | ||||
.into_iter() | ||||
.map(|(path, copy)| (path.to_owned(), copy.to_owned())) | ||||
.collect(); | ||||
Raphaël Gomès
|
r42993 | |||
Yuya Nishihara
|
r43479 | for (filename, entry) in &dirstate_map { | ||
Raphaël Gomès
|
r42992 | dmap.set_item( | ||
py, | ||||
Raphaël Gomès
|
r45500 | PyBytes::new(py, filename.as_bytes()), | ||
Yuya Nishihara
|
r43479 | make_dirstate_tuple(py, entry)?, | ||
Raphaël Gomès
|
r42992 | )?; | ||
} | ||||
Antoine Cezar
|
r45916 | for (path, copy_path) in copy_map { | ||
Raphaël Gomès
|
r42992 | copymap.set_item( | ||
py, | ||||
Raphaël Gomès
|
r45500 | PyBytes::new(py, path.as_bytes()), | ||
PyBytes::new(py, copy_path.as_bytes()), | ||||
Raphaël Gomès
|
r42992 | )?; | ||
} | ||||
Raphaël Gomès
|
r42993 | Ok( | ||
(PyBytes::new(py, &parents.p1), PyBytes::new(py, &parents.p2)) | ||||
.to_py_object(py), | ||||
) | ||||
Raphaël Gomès
|
r42992 | } | ||
Simon Sapin
|
r47169 | Err(e) => Err(PyErr::new::<exc::ValueError, _>(py, e.to_string())), | ||
Raphaël Gomès
|
r42992 | } | ||
} | ||||
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); | ||||
Raphaël Gomès
|
r42993 | let mut dirstate_map = extract_dirstate(py, &dmap)?; | ||
Raphaël Gomès
|
r42992 | |||
Raphaël Gomès
|
r44278 | let copies: Result<FastHashMap<HgPathBuf, HgPathBuf>, PyErr> = copymap | ||
Raphaël Gomès
|
r42992 | .items(py) | ||
.iter() | ||||
.map(|(key, value)| { | ||||
Ok(( | ||||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(key.extract::<PyBytes>(py)?.data(py)), | ||
HgPathBuf::from_bytes(value.extract::<PyBytes>(py)?.data(py)), | ||||
Raphaël Gomès
|
r42992 | )) | ||
}) | ||||
.collect(); | ||||
Raphaël Gomès
|
r42993 | if p1.len() != PARENT_SIZE || p2.len() != PARENT_SIZE { | ||
return Err(PyErr::new::<exc::ValueError, _>( | ||||
py, | ||||
"expected a 20-byte hash".to_string(), | ||||
)); | ||||
} | ||||
Raphaël Gomès
|
r42992 | match pack_dirstate( | ||
Raphaël Gomès
|
r42993 | &mut dirstate_map, | ||
Raphaël Gomès
|
r42992 | &copies?, | ||
Raphaël Gomès
|
r42993 | DirstateParents { | ||
Yuya Nishihara
|
r43067 | p1: p1.try_into().unwrap(), | ||
p2: p2.try_into().unwrap(), | ||||
Raphaël Gomès
|
r42993 | }, | ||
Yuya Nishihara
|
r43061 | Duration::from_secs(now.as_object().extract::<u64>(py)?), | ||
Raphaël Gomès
|
r42992 | ) { | ||
Raphaël Gomès
|
r42993 | Ok(packed) => { | ||
Raphaël Gomès
|
r46185 | for (filename, entry) in dirstate_map.iter() { | ||
Raphaël Gomès
|
r42992 | dmap.set_item( | ||
py, | ||||
Raphaël Gomès
|
r45500 | PyBytes::new(py, filename.as_bytes()), | ||
Raphaël Gomès
|
r46185 | make_dirstate_tuple(py, &entry)?, | ||
Raphaël Gomès
|
r42992 | )?; | ||
} | ||||
Ok(PyBytes::new(py, &packed)) | ||||
} | ||||
Simon Sapin
|
r47168 | Err(error) => { | ||
Err(PyErr::new::<exc::ValueError, _>(py, error.to_string())) | ||||
} | ||||
Raphaël Gomès
|
r42992 | } | ||
} | ||||
/// Create the module, with `__package__` given from parent | ||||
pub fn init_parsers_module(py: Python, package: &str) -> PyResult<PyModule> { | ||||
let dotted_name = &format!("{}.parsers", package); | ||||
let m = PyModule::new(py, dotted_name)?; | ||||
m.add(py, "__package__", package)?; | ||||
m.add(py, "__doc__", "Parsers - 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 | ||||
) | ||||
), | ||||
)?; | ||||
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) | ||||
} | ||||