// status.rs // // Copyright 2019, Raphaël Gomès // // 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::status` module provided by the //! `hg-core` crate. From Python, this will be seen as //! `rustext.dirstate.status`. use crate::dirstate::DirstateMap; use cpython::exc::ValueError; use cpython::{ ObjectProtocol, PyBytes, PyErr, PyList, PyObject, PyResult, PyTuple, Python, PythonObject, ToPyObject, }; use hg::utils::hg_path::HgPathBuf; use hg::{ matchers::{AlwaysMatcher, FileMatcher}, status, utils::{files::get_path_from_bytes, hg_path::HgPath}, StatusResult, }; use std::borrow::Borrow; /// This will be useless once trait impls for collection are added to `PyBytes` /// upstream. fn collect_pybytes_list>( py: Python, collection: &[P], ) -> PyList { let list = PyList::new(py, &[]); for (i, path) in collection.iter().enumerate() { list.insert( py, i, PyBytes::new(py, path.as_ref().as_bytes()).into_object(), ) } list } pub fn status_wrapper( py: Python, dmap: DirstateMap, matcher: PyObject, root_dir: PyObject, list_clean: bool, last_normal_time: i64, check_exec: bool, ) -> PyResult<(PyList, PyList, PyList, PyList, PyList, PyList, PyList)> { let bytes = root_dir.extract::(py)?; let root_dir = get_path_from_bytes(bytes.data(py)); let dmap: DirstateMap = dmap.to_py_object(py); let dmap = dmap.get_inner(py); match matcher.get_type(py).name(py).borrow() { "alwaysmatcher" => { let matcher = AlwaysMatcher; let (lookup, status_res) = status( &dmap, &matcher, &root_dir, list_clean, last_normal_time, check_exec, ) .map_err(|e| PyErr::new::(py, e.to_string()))?; build_response(lookup, status_res, py) } "exactmatcher" => { let files = matcher.call_method( py, "files", PyTuple::new(py, &[]), None, )?; let files: PyList = files.cast_into(py)?; let files: PyResult> = files .iter(py) .map(|f| { Ok(HgPathBuf::from_bytes( f.extract::(py)?.data(py), )) }) .collect(); let files = files?; let matcher = FileMatcher::new(&files) .map_err(|e| PyErr::new::(py, e.to_string()))?; let (lookup, status_res) = status( &dmap, &matcher, &root_dir, list_clean, last_normal_time, check_exec, ) .map_err(|e| PyErr::new::(py, e.to_string()))?; build_response(lookup, status_res, py) } e => { return Err(PyErr::new::( py, format!("Unsupported matcher {}", e), )); } } } fn build_response( lookup: Vec<&HgPath>, status_res: StatusResult, py: Python, ) -> PyResult<(PyList, PyList, PyList, PyList, PyList, PyList, PyList)> { let modified = collect_pybytes_list(py, status_res.modified.as_ref()); let added = collect_pybytes_list(py, status_res.added.as_ref()); let removed = collect_pybytes_list(py, status_res.removed.as_ref()); let deleted = collect_pybytes_list(py, status_res.deleted.as_ref()); let clean = collect_pybytes_list(py, status_res.clean.as_ref()); let lookup = collect_pybytes_list(py, lookup.as_ref()); let unknown = PyList::new(py, &[]); Ok((lookup, modified, added, removed, deleted, unknown, clean)) }