mirror of
https://github.com/RustPython/RustPython.git
synced 2026-06-09 22:49:57 +09:00
126 lines
3.5 KiB
Rust
126 lines
3.5 KiB
Rust
use crate::function::{OptionalArg, PyFuncArgs};
|
|
use crate::pyobject::{IdProtocol, PyContext, PyObjectRef, PyRef, PyResult, PyValue, TypeProtocol};
|
|
use crate::vm::VirtualMachine;
|
|
|
|
use crate::obj::objint::PyInt;
|
|
use crate::obj::objtype::{class_has_attr, PyClassRef};
|
|
use num_bigint::BigInt;
|
|
|
|
#[derive(Debug)]
|
|
pub struct PySlice {
|
|
pub start: Option<PyObjectRef>,
|
|
pub stop: PyObjectRef,
|
|
pub step: Option<PyObjectRef>,
|
|
}
|
|
|
|
impl PyValue for PySlice {
|
|
fn class(vm: &VirtualMachine) -> PyClassRef {
|
|
vm.ctx.slice_type()
|
|
}
|
|
}
|
|
|
|
pub type PySliceRef = PyRef<PySlice>;
|
|
|
|
fn slice_new(cls: PyClassRef, args: PyFuncArgs, vm: &VirtualMachine) -> PyResult<PySliceRef> {
|
|
let slice: PySlice = match args.args.len() {
|
|
0 => {
|
|
return Err(vm.new_type_error("slice() must have at least one arguments.".to_owned()));
|
|
}
|
|
1 => {
|
|
let stop = args.bind(vm)?;
|
|
PySlice {
|
|
start: None,
|
|
stop,
|
|
step: None,
|
|
}
|
|
}
|
|
_ => {
|
|
let (start, stop, step): (PyObjectRef, PyObjectRef, OptionalArg<PyObjectRef>) =
|
|
args.bind(vm)?;
|
|
PySlice {
|
|
start: Some(start),
|
|
stop,
|
|
step: step.into_option(),
|
|
}
|
|
}
|
|
};
|
|
slice.into_ref_with_type(vm, cls)
|
|
}
|
|
|
|
fn get_property_value(vm: &VirtualMachine, value: &Option<PyObjectRef>) -> PyObjectRef {
|
|
if let Some(value) = value {
|
|
value.clone()
|
|
} else {
|
|
vm.get_none()
|
|
}
|
|
}
|
|
|
|
impl PySliceRef {
|
|
fn start(self, vm: &VirtualMachine) -> PyObjectRef {
|
|
get_property_value(vm, &self.start)
|
|
}
|
|
|
|
fn stop(self, _vm: &VirtualMachine) -> PyObjectRef {
|
|
self.stop.clone()
|
|
}
|
|
|
|
fn step(self, vm: &VirtualMachine) -> PyObjectRef {
|
|
get_property_value(vm, &self.step)
|
|
}
|
|
|
|
pub fn start_index(&self, vm: &VirtualMachine) -> PyResult<Option<BigInt>> {
|
|
if let Some(obj) = &self.start {
|
|
to_index_value(vm, obj)
|
|
} else {
|
|
Ok(None)
|
|
}
|
|
}
|
|
|
|
pub fn stop_index(&self, vm: &VirtualMachine) -> PyResult<Option<BigInt>> {
|
|
to_index_value(vm, &self.stop)
|
|
}
|
|
|
|
pub fn step_index(&self, vm: &VirtualMachine) -> PyResult<Option<BigInt>> {
|
|
if let Some(obj) = &self.step {
|
|
to_index_value(vm, obj)
|
|
} else {
|
|
Ok(None)
|
|
}
|
|
}
|
|
}
|
|
|
|
fn to_index_value(vm: &VirtualMachine, obj: &PyObjectRef) -> PyResult<Option<BigInt>> {
|
|
if obj.is(&vm.ctx.none) {
|
|
return Ok(None);
|
|
}
|
|
|
|
if let Some(val) = obj.payload::<PyInt>() {
|
|
Ok(Some(val.as_bigint().clone()))
|
|
} else {
|
|
let cls = obj.class();
|
|
if class_has_attr(&cls, "__index__") {
|
|
let index_result = vm.call_method(obj, "__index__", vec![])?;
|
|
if let Some(val) = index_result.payload::<PyInt>() {
|
|
Ok(Some(val.as_bigint().clone()))
|
|
} else {
|
|
Err(vm.new_type_error("__index__ method returned non integer".to_string()))
|
|
}
|
|
} else {
|
|
Err(vm.new_type_error(
|
|
"slice indices must be integers or None or have an __index__ method".to_string(),
|
|
))
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn init(context: &PyContext) {
|
|
let slice_type = &context.types.slice_type;
|
|
|
|
extend_class!(context, slice_type, {
|
|
"__new__" => context.new_rustfunc(slice_new),
|
|
"start" => context.new_property(PySliceRef::start),
|
|
"stop" => context.new_property(PySliceRef::stop),
|
|
"step" => context.new_property(PySliceRef::step)
|
|
});
|
|
}
|