filepatterns.rs
134 lines
| 4.2 KiB
| application/rls-services+xml
|
RustLexer
Raphaël Gomès
|
r42515 | // filepatterns.rs | ||
// | ||||
// Copyright 2019, Georges Racinet <gracinet@anybox.fr>, | ||||
// 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::filepatterns` module provided by the | ||||
//! `hg-core` crate. From Python, this will be seen as `rustext.filepatterns` | ||||
Yuya Nishihara
|
r43109 | //! and can be used as replacement for the the pure `filepatterns` Python | ||
//! module. | ||||
Raphaël Gomès
|
r42515 | //! | ||
Raphaël Gomès
|
r42828 | use crate::exceptions::{PatternError, PatternFileError}; | ||
Raphaël Gomès
|
r42515 | use cpython::{ | ||
Raphaël Gomès
|
r44096 | PyBytes, PyDict, PyModule, PyObject, PyResult, PyTuple, Python, ToPyObject, | ||
Raphaël Gomès
|
r42515 | }; | ||
Yuya Nishihara
|
r44210 | use hg::utils::files; | ||
use hg::{build_single_regex, read_pattern_file, LineNumber, PatternTuple}; | ||||
Raphaël Gomès
|
r43227 | use std::path::PathBuf; | ||
Raphaël Gomès
|
r42515 | |||
/// Rust does not like functions with different return signatures. | ||||
/// The 3-tuple version is always returned by the hg-core function, | ||||
/// the (potential) conversion is handled at this level since it is not likely | ||||
/// to have any measurable impact on performance. | ||||
/// | ||||
/// The Python implementation passes a function reference for `warn` instead | ||||
/// of a boolean that is used to emit warnings while parsing. The Rust | ||||
/// implementation chooses to accumulate the warnings and propagate them to | ||||
/// Python upon completion. See the `readpatternfile` function in `match.py` | ||||
/// for more details. | ||||
fn read_pattern_file_wrapper( | ||||
py: Python, | ||||
Raphaël Gomès
|
r42630 | file_path: PyObject, | ||
Raphaël Gomès
|
r42515 | warn: bool, | ||
source_info: bool, | ||||
) -> PyResult<PyTuple> { | ||||
Raphaël Gomès
|
r43227 | let bytes = file_path.extract::<PyBytes>(py)?; | ||
Yuya Nishihara
|
r44210 | let path = files::get_path_from_bytes(bytes.data(py)); | ||
Raphaël Gomès
|
r43227 | match read_pattern_file(path, warn) { | ||
Raphaël Gomès
|
r42515 | Ok((patterns, warnings)) => { | ||
if source_info { | ||||
Raphaël Gomès
|
r42630 | let itemgetter = |x: &PatternTuple| { | ||
(PyBytes::new(py, &x.0), x.1, PyBytes::new(py, &x.2)) | ||||
}; | ||||
let results: Vec<(PyBytes, LineNumber, PyBytes)> = | ||||
patterns.iter().map(itemgetter).collect(); | ||||
Yuya Nishihara
|
r42857 | return Ok((results, warnings_to_py_bytes(py, &warnings)) | ||
.to_py_object(py)); | ||||
Raphaël Gomès
|
r42515 | } | ||
Raphaël Gomès
|
r42630 | let itemgetter = |x: &PatternTuple| PyBytes::new(py, &x.0); | ||
let results: Vec<PyBytes> = | ||||
Raphaël Gomès
|
r42515 | patterns.iter().map(itemgetter).collect(); | ||
Yuya Nishihara
|
r42857 | Ok( | ||
(results, warnings_to_py_bytes(py, &warnings)) | ||||
.to_py_object(py), | ||||
) | ||||
Raphaël Gomès
|
r42515 | } | ||
Err(e) => Err(PatternFileError::pynew(py, e)), | ||||
} | ||||
} | ||||
Yuya Nishihara
|
r42857 | fn warnings_to_py_bytes( | ||
py: Python, | ||||
Raphaël Gomès
|
r43227 | warnings: &[(PathBuf, Vec<u8>)], | ||
Raphaël Gomès
|
r44096 | ) -> Vec<(PyBytes, PyBytes)> { | ||
Yuya Nishihara
|
r42857 | warnings | ||
.iter() | ||||
Raphaël Gomès
|
r43227 | .map(|(path, syn)| { | ||
( | ||||
Raphaël Gomès
|
r44096 | PyBytes::new(py, &path.to_string_lossy().as_bytes()), | ||
Raphaël Gomès
|
r43227 | PyBytes::new(py, syn), | ||
) | ||||
}) | ||||
Yuya Nishihara
|
r42857 | .collect() | ||
} | ||||
Raphaël Gomès
|
r42515 | fn build_single_regex_wrapper( | ||
py: Python, | ||||
Raphaël Gomès
|
r42630 | kind: PyObject, | ||
pat: PyObject, | ||||
globsuffix: PyObject, | ||||
) -> PyResult<PyBytes> { | ||||
Raphaël Gomès
|
r42515 | match build_single_regex( | ||
Raphaël Gomès
|
r42630 | kind.extract::<PyBytes>(py)?.data(py), | ||
pat.extract::<PyBytes>(py)?.data(py), | ||||
globsuffix.extract::<PyBytes>(py)?.data(py), | ||||
Raphaël Gomès
|
r42515 | ) { | ||
Raphaël Gomès
|
r42630 | Ok(regex) => Ok(PyBytes::new(py, ®ex)), | ||
Raphaël Gomès
|
r42515 | Err(e) => Err(PatternError::pynew(py, e)), | ||
} | ||||
} | ||||
pub fn init_module(py: Python, package: &str) -> PyResult<PyModule> { | ||||
let dotted_name = &format!("{}.filepatterns", package); | ||||
let m = PyModule::new(py, dotted_name)?; | ||||
m.add(py, "__package__", package)?; | ||||
m.add( | ||||
py, | ||||
"__doc__", | ||||
"Patterns files parsing - Rust implementation", | ||||
)?; | ||||
m.add( | ||||
py, | ||||
"build_single_regex", | ||||
py_fn!( | ||||
py, | ||||
build_single_regex_wrapper( | ||||
Raphaël Gomès
|
r42630 | kind: PyObject, | ||
pat: PyObject, | ||||
globsuffix: PyObject | ||||
Raphaël Gomès
|
r42515 | ) | ||
), | ||||
)?; | ||||
m.add( | ||||
py, | ||||
"read_pattern_file", | ||||
py_fn!( | ||||
py, | ||||
read_pattern_file_wrapper( | ||||
Raphaël Gomès
|
r42630 | file_path: PyObject, | ||
Raphaël Gomès
|
r42515 | warn: bool, | ||
source_info: bool | ||||
) | ||||
), | ||||
)?; | ||||
Raphaël Gomès
|
r42630 | m.add(py, "PatternError", py.get_type::<PatternError>())?; | ||
Raphaël Gomès
|
r42515 | 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) | ||||
} | ||||