dirstate_map.rs
594 lines
| 18.7 KiB
| application/rls-services+xml
|
RustLexer
Raphaël Gomès
|
r42999 | // dirstate_map.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::dirstate_map` file provided by the | ||||
//! `hg-core` package. | ||||
Simon Sapin
|
r47882 | use std::cell::{RefCell, RefMut}; | ||
Raphaël Gomès
|
r42999 | use std::convert::TryInto; | ||
use cpython::{ | ||||
Raphaël Gomès
|
r44836 | exc, ObjectProtocol, PyBool, PyBytes, PyClone, PyDict, PyErr, PyList, | ||
Simon Sapin
|
r47891 | PyObject, PyResult, PySet, PyString, Python, PythonObject, ToPyObject, | ||
UnsafePyLeaked, | ||||
Raphaël Gomès
|
r42999 | }; | ||
use crate::{ | ||||
dirstate::copymap::{CopyMap, CopyMapItemsIterator, CopyMapKeysIterator}, | ||||
Simon Sapin
|
r48271 | dirstate::make_dirstate_tuple, | ||
Raphaël Gomès
|
r44903 | dirstate::non_normal_entries::{ | ||
NonNormalEntries, NonNormalEntriesIterator, | ||||
}, | ||||
Simon Sapin
|
r47893 | dirstate::owning::OwningDirstateMap, | ||
Simon Sapin
|
r47337 | parsers::dirstate_parents_to_pytuple, | ||
Raphaël Gomès
|
r42999 | }; | ||
use hg::{ | ||||
Simon Sapin
|
r47871 | dirstate::parsers::Timestamp, | ||
r48310 | dirstate::MTIME_UNSET, | |||
dirstate::SIZE_NON_NORMAL, | ||||
Simon Sapin
|
r47863 | dirstate_tree::dispatch::DirstateMapMethods, | ||
Simon Sapin
|
r48126 | dirstate_tree::on_disk::DirstateV2ParseError, | ||
Simon Sapin
|
r47169 | errors::HgError, | ||
Simon Sapin
|
r47337 | revlog::Node, | ||
Simon Sapin
|
r47879 | utils::files::normalize_case, | ||
Raphaël Gomès
|
r43227 | utils::hg_path::{HgPath, HgPathBuf}, | ||
Simon Sapin
|
r48271 | DirstateEntry, DirstateError, DirstateMap as RustDirstateMap, | ||
DirstateParents, EntryState, StateMapIter, | ||||
Raphaël Gomès
|
r42999 | }; | ||
// TODO | ||||
// This object needs to share references to multiple members of its Rust | ||||
// inner struct, namely `copy_map`, `dirs` and `all_dirs`. | ||||
// Right now `CopyMap` is done, but it needs to have an explicit reference | ||||
// to `RustDirstateMap` which itself needs to have an encapsulation for | ||||
// every method in `CopyMap` (copymapcopy, etc.). | ||||
// This is ugly and hard to maintain. | ||||
// The same logic applies to `dirs` and `all_dirs`, however the `Dirs` | ||||
// `py_class!` is already implemented and does not mention | ||||
// `RustDirstateMap`, rightfully so. | ||||
// All attributes also have to have a separate refcount data attribute for | ||||
// leaks, with all methods that go along for reference sharing. | ||||
py_class!(pub class DirstateMap |py| { | ||||
Simon Sapin
|
r47863 | @shared data inner: Box<dyn DirstateMapMethods + Send>; | ||
Raphaël Gomès
|
r42999 | |||
Simon Sapin
|
r47892 | /// Returns a `(dirstate_map, parents)` tuple | ||
@staticmethod | ||||
Simon Sapin
|
r48055 | def new( | ||
use_dirstate_tree: bool, | ||||
use_dirstate_v2: bool, | ||||
on_disk: PyBytes, | ||||
) -> PyResult<PyObject> { | ||||
let dirstate_error = |e: DirstateError| { | ||||
PyErr::new::<exc::OSError, _>(py, format!("Dirstate error: {:?}", e)) | ||||
Simon Sapin
|
r47892 | }; | ||
Simon Sapin
|
r48055 | let (inner, parents) = if use_dirstate_tree || use_dirstate_v2 { | ||
Simon Sapin
|
r47893 | let (map, parents) = | ||
Simon Sapin
|
r48055 | OwningDirstateMap::new(py, on_disk, use_dirstate_v2) | ||
Simon Sapin
|
r47893 | .map_err(dirstate_error)?; | ||
Simon Sapin
|
r47892 | (Box::new(map) as _, parents) | ||
Simon Sapin
|
r47865 | } else { | ||
Simon Sapin
|
r47893 | let bytes = on_disk.data(py); | ||
Simon Sapin
|
r47892 | let mut map = RustDirstateMap::default(); | ||
let parents = map.read(bytes).map_err(dirstate_error)?; | ||||
(Box::new(map) as _, parents) | ||||
Simon Sapin
|
r47865 | }; | ||
Simon Sapin
|
r47892 | let map = Self::create_instance(py, inner)?; | ||
let parents = parents.map(|p| dirstate_parents_to_pytuple(py, &p)); | ||||
Ok((map, parents).to_py_object(py).into_object()) | ||||
Raphaël Gomès
|
r42999 | } | ||
def clear(&self) -> PyResult<PyObject> { | ||||
Yuya Nishihara
|
r44702 | self.inner(py).borrow_mut().clear(); | ||
Raphaël Gomès
|
r42999 | Ok(py.None()) | ||
} | ||||
def get( | ||||
&self, | ||||
key: PyObject, | ||||
default: Option<PyObject> = None | ||||
) -> PyResult<Option<PyObject>> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
Simon Sapin
|
r48126 | match self | ||
.inner(py) | ||||
.borrow() | ||||
.get(HgPath::new(key.data(py))) | ||||
.map_err(|e| v2_error(py, e))? | ||||
{ | ||||
Raphaël Gomès
|
r42999 | Some(entry) => { | ||
Simon Sapin
|
r48123 | Ok(Some(make_dirstate_tuple(py, &entry)?)) | ||
Raphaël Gomès
|
r42999 | }, | ||
None => Ok(default) | ||||
} | ||||
} | ||||
def addfile( | ||||
&self, | ||||
f: PyObject, | ||||
state: PyObject, | ||||
mode: PyObject, | ||||
size: PyObject, | ||||
r48310 | mtime: PyObject, | |||
r48314 | added: PyObject, | |||
r48310 | from_p2: PyObject, | |||
possibly_dirty: PyObject, | ||||
Raphaël Gomès
|
r42999 | ) -> PyResult<PyObject> { | ||
r48307 | let f = f.extract::<PyBytes>(py)?; | |||
let filename = HgPath::new(f.data(py)); | ||||
r48314 | let state = if state.is_none(py) { | |||
// Arbitrary default value | ||||
EntryState::Normal | ||||
} else { | ||||
state.extract::<PyBytes>(py)?.data(py)[0] | ||||
r48307 | .try_into() | |||
.map_err(|e: HgError| { | ||||
PyErr::new::<exc::ValueError, _>(py, e.to_string()) | ||||
r48314 | })? | |||
}; | ||||
let mode = if mode.is_none(py) { | ||||
// fallback default value | ||||
0 | ||||
} else { | ||||
mode.extract(py)? | ||||
}; | ||||
r48310 | let size = if size.is_none(py) { | |||
// fallback default value | ||||
SIZE_NON_NORMAL | ||||
} else { | ||||
size.extract(py)? | ||||
}; | ||||
let mtime = if mtime.is_none(py) { | ||||
// fallback default value | ||||
MTIME_UNSET | ||||
} else { | ||||
mtime.extract(py)? | ||||
}; | ||||
r48307 | let entry = DirstateEntry { | |||
state: state, | ||||
mode: mode, | ||||
size: size, | ||||
mtime: mtime, | ||||
}; | ||||
r48314 | let added = added.extract::<PyBool>(py)?.is_true(); | |||
r48310 | let from_p2 = from_p2.extract::<PyBool>(py)?.is_true(); | |||
let possibly_dirty = possibly_dirty.extract::<PyBool>(py)?.is_true(); | ||||
Yuya Nishihara
|
r44702 | self.inner(py).borrow_mut().add_file( | ||
r48307 | filename, | |||
entry, | ||||
r48314 | added, | |||
r48310 | from_p2, | |||
possibly_dirty | ||||
Simon Sapin
|
r48126 | ).and(Ok(py.None())).or_else(|e: DirstateError| { | ||
Raphaël Gomès
|
r44227 | Err(PyErr::new::<exc::ValueError, _>(py, e.to_string())) | ||
}) | ||||
Raphaël Gomès
|
r42999 | } | ||
def removefile( | ||||
&self, | ||||
f: PyObject, | ||||
r48300 | in_merge: PyObject | |||
Raphaël Gomès
|
r42999 | ) -> PyResult<PyObject> { | ||
Yuya Nishihara
|
r44702 | self.inner(py).borrow_mut() | ||
Raphaël Gomès
|
r42999 | .remove_file( | ||
Raphaël Gomès
|
r43227 | HgPath::new(f.extract::<PyBytes>(py)?.data(py)), | ||
r48300 | in_merge.extract::<PyBool>(py)?.is_true(), | |||
Raphaël Gomès
|
r42999 | ) | ||
.or_else(|_| { | ||||
Err(PyErr::new::<exc::OSError, _>( | ||||
py, | ||||
"Dirstate error".to_string(), | ||||
)) | ||||
})?; | ||||
Ok(py.None()) | ||||
} | ||||
def dropfile( | ||||
&self, | ||||
f: PyObject, | ||||
oldstate: PyObject | ||||
) -> PyResult<PyBool> { | ||||
Yuya Nishihara
|
r44702 | self.inner(py).borrow_mut() | ||
Raphaël Gomès
|
r42999 | .drop_file( | ||
Raphaël Gomès
|
r43227 | HgPath::new(f.extract::<PyBytes>(py)?.data(py)), | ||
Raphaël Gomès
|
r42999 | oldstate.extract::<PyBytes>(py)?.data(py)[0] | ||
.try_into() | ||||
Simon Sapin
|
r47169 | .map_err(|e: HgError| { | ||
Raphaël Gomès
|
r42999 | PyErr::new::<exc::ValueError, _>(py, e.to_string()) | ||
})?, | ||||
) | ||||
.and_then(|b| Ok(b.to_py_object(py))) | ||||
Raphaël Gomès
|
r46185 | .or_else(|e| { | ||
Raphaël Gomès
|
r42999 | Err(PyErr::new::<exc::OSError, _>( | ||
py, | ||||
Raphaël Gomès
|
r46185 | format!("Dirstate error: {}", e.to_string()), | ||
Raphaël Gomès
|
r42999 | )) | ||
}) | ||||
} | ||||
def clearambiguoustimes( | ||||
&self, | ||||
files: PyObject, | ||||
now: PyObject | ||||
) -> PyResult<PyObject> { | ||||
Raphaël Gomès
|
r43227 | let files: PyResult<Vec<HgPathBuf>> = files | ||
Raphaël Gomès
|
r42999 | .iter(py)? | ||
.map(|filename| { | ||||
Raphaël Gomès
|
r43227 | Ok(HgPathBuf::from_bytes( | ||
filename?.extract::<PyBytes>(py)?.data(py), | ||||
)) | ||||
Raphaël Gomès
|
r42999 | }) | ||
.collect(); | ||||
Simon Sapin
|
r48126 | self.inner(py) | ||
.borrow_mut() | ||||
.clear_ambiguous_times(files?, now.extract(py)?) | ||||
.map_err(|e| v2_error(py, e))?; | ||||
Raphaël Gomès
|
r42999 | Ok(py.None()) | ||
} | ||||
Raphaël Gomès
|
r44836 | def other_parent_entries(&self) -> PyResult<PyObject> { | ||
Raphaël Gomès
|
r44779 | let mut inner_shared = self.inner(py).borrow_mut(); | ||
Simon Sapin
|
r47653 | let set = PySet::empty(py)?; | ||
Simon Sapin
|
r47864 | for path in inner_shared.iter_other_parent_paths() { | ||
Simon Sapin
|
r48126 | let path = path.map_err(|e| v2_error(py, e))?; | ||
Simon Sapin
|
r47653 | set.add(py, PyBytes::new(py, path.as_bytes()))?; | ||
} | ||||
Ok(set.into_object()) | ||||
Raphaël Gomès
|
r44836 | } | ||
def non_normal_entries(&self) -> PyResult<NonNormalEntries> { | ||||
NonNormalEntries::from_inner(py, self.clone_ref(py)) | ||||
} | ||||
def non_normal_entries_contains(&self, key: PyObject) -> PyResult<bool> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
Simon Sapin
|
r48126 | self.inner(py) | ||
Raphaël Gomès
|
r44779 | .borrow_mut() | ||
Simon Sapin
|
r48126 | .non_normal_entries_contains(HgPath::new(key.data(py))) | ||
.map_err(|e| v2_error(py, e)) | ||||
Raphaël Gomès
|
r44836 | } | ||
def non_normal_entries_display(&self) -> PyResult<PyString> { | ||||
Simon Sapin
|
r48126 | let mut inner = self.inner(py).borrow_mut(); | ||
let paths = inner | ||||
.iter_non_normal_paths() | ||||
.collect::<Result<Vec<_>, _>>() | ||||
.map_err(|e| v2_error(py, e))?; | ||||
let formatted = format!("NonNormalEntries: {}", hg::utils::join_display(paths, ", ")); | ||||
Ok(PyString::new(py, &formatted)) | ||||
Raphaël Gomès
|
r44836 | } | ||
def non_normal_entries_remove(&self, key: PyObject) -> PyResult<PyObject> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
self | ||||
Raphaël Gomès
|
r44779 | .inner(py) | ||
.borrow_mut() | ||||
Raphaël Gomès
|
r44836 | .non_normal_entries_remove(HgPath::new(key.data(py))); | ||
Ok(py.None()) | ||||
} | ||||
Simon Sapin
|
r47864 | def non_normal_or_other_parent_paths(&self) -> PyResult<PyList> { | ||
let mut inner = self.inner(py).borrow_mut(); | ||||
Raphaël Gomès
|
r44836 | |||
let ret = PyList::new(py, &[]); | ||||
Simon Sapin
|
r47864 | for filename in inner.non_normal_or_other_parent_paths() { | ||
Simon Sapin
|
r48126 | let filename = filename.map_err(|e| v2_error(py, e))?; | ||
Raphaël Gomès
|
r44836 | let as_pystring = PyBytes::new(py, filename.as_bytes()); | ||
Raphaël Gomès
|
r44779 | ret.append(py, as_pystring.into_object()); | ||
Raphaël Gomès
|
r44836 | } | ||
Ok(ret) | ||||
Raphaël Gomès
|
r42999 | } | ||
Raphaël Gomès
|
r44903 | def non_normal_entries_iter(&self) -> PyResult<NonNormalEntriesIterator> { | ||
// Make sure the sets are defined before we no longer have a mutable | ||||
// reference to the dmap. | ||||
self.inner(py) | ||||
.borrow_mut() | ||||
.set_non_normal_other_parent_entries(false); | ||||
let leaked_ref = self.inner(py).leak_immutable(); | ||||
NonNormalEntriesIterator::from_inner(py, unsafe { | ||||
leaked_ref.map(py, |o| { | ||||
Simon Sapin
|
r47864 | o.iter_non_normal_paths_panic() | ||
Raphaël Gomès
|
r44903 | }) | ||
}) | ||||
} | ||||
Raphaël Gomès
|
r42999 | def hastrackeddir(&self, d: PyObject) -> PyResult<PyBool> { | ||
let d = d.extract::<PyBytes>(py)?; | ||||
Yuya Nishihara
|
r44702 | Ok(self.inner(py).borrow_mut() | ||
Raphaël Gomès
|
r43227 | .has_tracked_dir(HgPath::new(d.data(py))) | ||
Raphaël Gomès
|
r44315 | .map_err(|e| { | ||
PyErr::new::<exc::ValueError, _>(py, e.to_string()) | ||||
})? | ||||
Raphaël Gomès
|
r42999 | .to_py_object(py)) | ||
} | ||||
def hasdir(&self, d: PyObject) -> PyResult<PyBool> { | ||||
let d = d.extract::<PyBytes>(py)?; | ||||
Yuya Nishihara
|
r44702 | Ok(self.inner(py).borrow_mut() | ||
Raphaël Gomès
|
r43227 | .has_dir(HgPath::new(d.data(py))) | ||
Raphaël Gomès
|
r44315 | .map_err(|e| { | ||
PyErr::new::<exc::ValueError, _>(py, e.to_string()) | ||||
})? | ||||
Raphaël Gomès
|
r42999 | .to_py_object(py)) | ||
} | ||||
def write( | ||||
&self, | ||||
Simon Sapin
|
r48055 | use_dirstate_v2: bool, | ||
Raphaël Gomès
|
r42999 | p1: PyObject, | ||
p2: PyObject, | ||||
now: PyObject | ||||
) -> PyResult<PyBytes> { | ||||
Simon Sapin
|
r47871 | let now = Timestamp(now.extract(py)?); | ||
Raphaël Gomès
|
r42999 | let parents = DirstateParents { | ||
Yuya Nishihara
|
r43068 | p1: extract_node_id(py, &p1)?, | ||
p2: extract_node_id(py, &p2)?, | ||||
Raphaël Gomès
|
r42999 | }; | ||
Simon Sapin
|
r48055 | let mut inner = self.inner(py).borrow_mut(); | ||
let result = if use_dirstate_v2 { | ||||
inner.pack_v2(parents, now) | ||||
} else { | ||||
inner.pack_v1(parents, now) | ||||
}; | ||||
match result { | ||||
Raphaël Gomès
|
r42999 | Ok(packed) => Ok(PyBytes::new(py, &packed)), | ||
Err(_) => Err(PyErr::new::<exc::OSError, _>( | ||||
py, | ||||
"Dirstate error".to_string(), | ||||
)), | ||||
} | ||||
} | ||||
def filefoldmapasdict(&self) -> PyResult<PyDict> { | ||||
let dict = PyDict::new(py); | ||||
Simon Sapin
|
r48126 | for item in self.inner(py).borrow_mut().iter() { | ||
let (path, entry) = item.map_err(|e| v2_error(py, e))?; | ||||
Simon Sapin
|
r47879 | if entry.state != EntryState::Removed { | ||
let key = normalize_case(path); | ||||
let value = path; | ||||
dict.set_item( | ||||
py, | ||||
PyBytes::new(py, key.as_bytes()).into_object(), | ||||
PyBytes::new(py, value.as_bytes()).into_object(), | ||||
)?; | ||||
} | ||||
Raphaël Gomès
|
r42999 | } | ||
Ok(dict) | ||||
} | ||||
def __len__(&self) -> PyResult<usize> { | ||||
Yuya Nishihara
|
r44702 | Ok(self.inner(py).borrow().len()) | ||
Raphaël Gomès
|
r42999 | } | ||
def __contains__(&self, key: PyObject) -> PyResult<bool> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
Simon Sapin
|
r48126 | self.inner(py) | ||
.borrow() | ||||
.contains_key(HgPath::new(key.data(py))) | ||||
.map_err(|e| v2_error(py, e)) | ||||
Raphaël Gomès
|
r42999 | } | ||
def __getitem__(&self, key: PyObject) -> PyResult<PyObject> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
Raphaël Gomès
|
r43227 | let key = HgPath::new(key.data(py)); | ||
Simon Sapin
|
r48126 | match self | ||
.inner(py) | ||||
.borrow() | ||||
.get(key) | ||||
.map_err(|e| v2_error(py, e))? | ||||
{ | ||||
Raphaël Gomès
|
r42999 | Some(entry) => { | ||
Simon Sapin
|
r48123 | Ok(make_dirstate_tuple(py, &entry)?) | ||
Raphaël Gomès
|
r42999 | }, | ||
None => Err(PyErr::new::<exc::KeyError, _>( | ||||
py, | ||||
Raphaël Gomès
|
r43227 | String::from_utf8_lossy(key.as_bytes()), | ||
Raphaël Gomès
|
r42999 | )), | ||
} | ||||
} | ||||
def keys(&self) -> PyResult<DirstateMapKeysIterator> { | ||||
Yuya Nishihara
|
r44702 | let leaked_ref = self.inner(py).leak_immutable(); | ||
Raphaël Gomès
|
r42999 | DirstateMapKeysIterator::from_inner( | ||
py, | ||||
Yuya Nishihara
|
r43579 | unsafe { leaked_ref.map(py, |o| o.iter()) }, | ||
Raphaël Gomès
|
r42999 | ) | ||
} | ||||
def items(&self) -> PyResult<DirstateMapItemsIterator> { | ||||
Yuya Nishihara
|
r44702 | let leaked_ref = self.inner(py).leak_immutable(); | ||
Raphaël Gomès
|
r42999 | DirstateMapItemsIterator::from_inner( | ||
py, | ||||
Yuya Nishihara
|
r43579 | unsafe { leaked_ref.map(py, |o| o.iter()) }, | ||
Raphaël Gomès
|
r42999 | ) | ||
} | ||||
def __iter__(&self) -> PyResult<DirstateMapKeysIterator> { | ||||
Yuya Nishihara
|
r44702 | let leaked_ref = self.inner(py).leak_immutable(); | ||
Raphaël Gomès
|
r42999 | DirstateMapKeysIterator::from_inner( | ||
py, | ||||
Yuya Nishihara
|
r43579 | unsafe { leaked_ref.map(py, |o| o.iter()) }, | ||
Raphaël Gomès
|
r42999 | ) | ||
} | ||||
// TODO all copymap* methods, see docstring above | ||||
def copymapcopy(&self) -> PyResult<PyDict> { | ||||
let dict = PyDict::new(py); | ||||
Simon Sapin
|
r48126 | for item in self.inner(py).borrow().copy_map_iter() { | ||
let (key, value) = item.map_err(|e| v2_error(py, e))?; | ||||
Raphaël Gomès
|
r43227 | dict.set_item( | ||
py, | ||||
Raphaël Gomès
|
r45500 | PyBytes::new(py, key.as_bytes()), | ||
PyBytes::new(py, value.as_bytes()), | ||||
Raphaël Gomès
|
r43227 | )?; | ||
Raphaël Gomès
|
r42999 | } | ||
Ok(dict) | ||||
} | ||||
def copymapgetitem(&self, key: PyObject) -> PyResult<PyBytes> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
Simon Sapin
|
r48126 | match self | ||
.inner(py) | ||||
.borrow() | ||||
.copy_map_get(HgPath::new(key.data(py))) | ||||
.map_err(|e| v2_error(py, e))? | ||||
{ | ||||
Raphaël Gomès
|
r45500 | Some(copy) => Ok(PyBytes::new(py, copy.as_bytes())), | ||
Raphaël Gomès
|
r42999 | None => Err(PyErr::new::<exc::KeyError, _>( | ||
py, | ||||
String::from_utf8_lossy(key.data(py)), | ||||
)), | ||||
} | ||||
} | ||||
def copymap(&self) -> PyResult<CopyMap> { | ||||
CopyMap::from_inner(py, self.clone_ref(py)) | ||||
} | ||||
def copymaplen(&self) -> PyResult<usize> { | ||||
Simon Sapin
|
r47863 | Ok(self.inner(py).borrow().copy_map_len()) | ||
Raphaël Gomès
|
r42999 | } | ||
def copymapcontains(&self, key: PyObject) -> PyResult<bool> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
Simon Sapin
|
r48126 | self.inner(py) | ||
Raphaël Gomès
|
r43227 | .borrow() | ||
Simon Sapin
|
r48126 | .copy_map_contains_key(HgPath::new(key.data(py))) | ||
.map_err(|e| v2_error(py, e)) | ||||
Raphaël Gomès
|
r42999 | } | ||
def copymapget( | ||||
&self, | ||||
key: PyObject, | ||||
default: Option<PyObject> | ||||
) -> PyResult<Option<PyObject>> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
Raphaël Gomès
|
r43227 | match self | ||
Yuya Nishihara
|
r44702 | .inner(py) | ||
Raphaël Gomès
|
r43227 | .borrow() | ||
Simon Sapin
|
r47863 | .copy_map_get(HgPath::new(key.data(py))) | ||
Simon Sapin
|
r48126 | .map_err(|e| v2_error(py, e))? | ||
Raphaël Gomès
|
r43227 | { | ||
Some(copy) => Ok(Some( | ||||
Raphaël Gomès
|
r45500 | PyBytes::new(py, copy.as_bytes()).into_object(), | ||
Raphaël Gomès
|
r43227 | )), | ||
Raphaël Gomès
|
r42999 | None => Ok(default), | ||
} | ||||
} | ||||
def copymapsetitem( | ||||
&self, | ||||
key: PyObject, | ||||
value: PyObject | ||||
) -> PyResult<PyObject> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
let value = value.extract::<PyBytes>(py)?; | ||||
Simon Sapin
|
r48126 | self.inner(py) | ||
.borrow_mut() | ||||
.copy_map_insert( | ||||
HgPathBuf::from_bytes(key.data(py)), | ||||
HgPathBuf::from_bytes(value.data(py)), | ||||
) | ||||
.map_err(|e| v2_error(py, e))?; | ||||
Raphaël Gomès
|
r42999 | Ok(py.None()) | ||
} | ||||
def copymappop( | ||||
&self, | ||||
key: PyObject, | ||||
default: Option<PyObject> | ||||
) -> PyResult<Option<PyObject>> { | ||||
let key = key.extract::<PyBytes>(py)?; | ||||
Raphaël Gomès
|
r43227 | match self | ||
Yuya Nishihara
|
r44702 | .inner(py) | ||
Yuya Nishihara
|
r44685 | .borrow_mut() | ||
Simon Sapin
|
r47863 | .copy_map_remove(HgPath::new(key.data(py))) | ||
Simon Sapin
|
r48126 | .map_err(|e| v2_error(py, e))? | ||
Raphaël Gomès
|
r43227 | { | ||
Raphaël Gomès
|
r42999 | Some(_) => Ok(None), | ||
None => Ok(default), | ||||
} | ||||
} | ||||
def copymapiter(&self) -> PyResult<CopyMapKeysIterator> { | ||||
Yuya Nishihara
|
r44702 | let leaked_ref = self.inner(py).leak_immutable(); | ||
Raphaël Gomès
|
r42999 | CopyMapKeysIterator::from_inner( | ||
py, | ||||
Simon Sapin
|
r47863 | unsafe { leaked_ref.map(py, |o| o.copy_map_iter()) }, | ||
Raphaël Gomès
|
r42999 | ) | ||
} | ||||
def copymapitemsiter(&self) -> PyResult<CopyMapItemsIterator> { | ||||
Yuya Nishihara
|
r44702 | let leaked_ref = self.inner(py).leak_immutable(); | ||
Raphaël Gomès
|
r42999 | CopyMapItemsIterator::from_inner( | ||
py, | ||||
Simon Sapin
|
r47863 | unsafe { leaked_ref.map(py, |o| o.copy_map_iter()) }, | ||
Raphaël Gomès
|
r42999 | ) | ||
} | ||||
Simon Sapin
|
r48140 | def directories(&self) -> PyResult<PyList> { | ||
let dirs = PyList::new(py, &[]); | ||||
for item in self.inner(py).borrow().iter_directories() { | ||||
let (path, mtime) = item.map_err(|e| v2_error(py, e))?; | ||||
let path = PyBytes::new(py, path.as_bytes()); | ||||
let mtime = mtime.map(|t| t.0).unwrap_or(-1); | ||||
let tuple = (path, (b'd', 0, 0, mtime)); | ||||
dirs.append(py, tuple.to_py_object(py).into_object()) | ||||
} | ||||
Ok(dirs) | ||||
} | ||||
Raphaël Gomès
|
r42999 | }); | ||
impl DirstateMap { | ||||
Simon Sapin
|
r47882 | pub fn get_inner_mut<'a>( | ||
Raphaël Gomès
|
r43567 | &'a self, | ||
py: Python<'a>, | ||||
Simon Sapin
|
r47882 | ) -> RefMut<'a, Box<dyn DirstateMapMethods + Send>> { | ||
self.inner(py).borrow_mut() | ||||
Raphaël Gomès
|
r43567 | } | ||
Raphaël Gomès
|
r42999 | fn translate_key( | ||
py: Python, | ||||
Simon Sapin
|
r48126 | res: Result<(&HgPath, DirstateEntry), DirstateV2ParseError>, | ||
Raphaël Gomès
|
r42999 | ) -> PyResult<Option<PyBytes>> { | ||
Simon Sapin
|
r48126 | let (f, _entry) = res.map_err(|e| v2_error(py, e))?; | ||
Ok(Some(PyBytes::new(py, f.as_bytes()))) | ||||
Raphaël Gomès
|
r42999 | } | ||
fn translate_key_value( | ||||
py: Python, | ||||
Simon Sapin
|
r48126 | res: Result<(&HgPath, DirstateEntry), DirstateV2ParseError>, | ||
Raphaël Gomès
|
r42999 | ) -> PyResult<Option<(PyBytes, PyObject)>> { | ||
Simon Sapin
|
r48126 | let (f, entry) = res.map_err(|e| v2_error(py, e))?; | ||
Raphaël Gomès
|
r42999 | Ok(Some(( | ||
Raphaël Gomès
|
r45500 | PyBytes::new(py, f.as_bytes()), | ||
Raphaël Gomès
|
r46185 | make_dirstate_tuple(py, &entry)?, | ||
))) | ||||
} | ||||
Raphaël Gomès
|
r42999 | } | ||
Yuya Nishihara
|
r43159 | py_shared_iterator!( | ||
Raphaël Gomès
|
r42999 | DirstateMapKeysIterator, | ||
Yuya Nishihara
|
r44703 | UnsafePyLeaked<StateMapIter<'static>>, | ||
Raphaël Gomès
|
r42999 | DirstateMap::translate_key, | ||
Option<PyBytes> | ||||
); | ||||
Yuya Nishihara
|
r43159 | py_shared_iterator!( | ||
Raphaël Gomès
|
r42999 | DirstateMapItemsIterator, | ||
Yuya Nishihara
|
r44703 | UnsafePyLeaked<StateMapIter<'static>>, | ||
Raphaël Gomès
|
r42999 | DirstateMap::translate_key_value, | ||
Option<(PyBytes, PyObject)> | ||||
); | ||||
Yuya Nishihara
|
r43068 | |||
Simon Sapin
|
r47337 | fn extract_node_id(py: Python, obj: &PyObject) -> PyResult<Node> { | ||
Yuya Nishihara
|
r43068 | let bytes = obj.extract::<PyBytes>(py)?; | ||
match bytes.data(py).try_into() { | ||||
Ok(s) => Ok(s), | ||||
Err(e) => Err(PyErr::new::<exc::ValueError, _>(py, e.to_string())), | ||||
} | ||||
} | ||||
Simon Sapin
|
r48126 | |||
pub(super) fn v2_error(py: Python<'_>, _: DirstateV2ParseError) -> PyErr { | ||||
PyErr::new::<exc::ValueError, _>(py, "corrupted dirstate-v2") | ||||
} | ||||