mirror of
https://github.com/RustPython/RustPython.git
synced 2026-06-02 19:39:49 +09:00
132 lines
3.9 KiB
Rust
132 lines
3.9 KiB
Rust
use super::super::pyobject::{
|
|
AttributeProtocol, PyContext, PyFuncArgs, PyObjectKind, PyObjectRef, PyResult, TypeProtocol,
|
|
};
|
|
use super::super::vm::VirtualMachine;
|
|
use super::objint;
|
|
use super::objsequence::PySliceableSequence;
|
|
use super::objtype;
|
|
|
|
pub fn init(context: &PyContext) {
|
|
let ref str_type = context.str_type;
|
|
str_type.set_attr("__eq__", context.new_rustfunc(str_eq));
|
|
str_type.set_attr("__add__", context.new_rustfunc(str_add));
|
|
str_type.set_attr("__len__", context.new_rustfunc(str_len));
|
|
str_type.set_attr("__mul__", context.new_rustfunc(str_mul));
|
|
str_type.set_attr("__new__", context.new_rustfunc(str_new));
|
|
str_type.set_attr("__str__", context.new_rustfunc(str_str));
|
|
}
|
|
|
|
pub fn get_value(obj: &PyObjectRef) -> String {
|
|
if let PyObjectKind::String { value } = &obj.borrow().kind {
|
|
value.to_string()
|
|
} else {
|
|
panic!("Inner error getting str");
|
|
}
|
|
}
|
|
|
|
fn str_eq(vm: &mut VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
arg_check!(
|
|
vm,
|
|
args,
|
|
required = [(a, Some(vm.ctx.str_type())), (b, None)]
|
|
);
|
|
|
|
let result = if objtype::isinstance(b.clone(), vm.ctx.str_type()) {
|
|
get_value(a) == get_value(b)
|
|
} else {
|
|
false
|
|
};
|
|
Ok(vm.ctx.new_bool(result))
|
|
}
|
|
|
|
fn str_str(vm: &mut VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
arg_check!(vm, args, required = [(s, Some(vm.ctx.str_type()))]);
|
|
Ok(s.clone())
|
|
}
|
|
|
|
fn str_add(vm: &mut VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
arg_check!(
|
|
vm,
|
|
args,
|
|
required = [(s, Some(vm.ctx.str_type())), (s2, None)]
|
|
);
|
|
if objtype::isinstance(s2.clone(), vm.ctx.str_type()) {
|
|
Ok(vm
|
|
.ctx
|
|
.new_str(format!("{}{}", get_value(&s), get_value(&s2))))
|
|
} else {
|
|
Err(vm.new_type_error(format!("Cannot add {:?} and {:?}", s, s2)))
|
|
}
|
|
}
|
|
|
|
fn str_len(vm: &mut VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
arg_check!(vm, args, required = [(s, Some(vm.ctx.str_type()))]);
|
|
let sv = get_value(s);
|
|
Ok(vm.ctx.new_int(sv.len() as i32))
|
|
}
|
|
|
|
fn str_mul(vm: &mut VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
arg_check!(
|
|
vm,
|
|
args,
|
|
required = [(s, Some(vm.ctx.str_type())), (s2, None)]
|
|
);
|
|
if objtype::isinstance(s2.clone(), vm.ctx.int_type()) {
|
|
let value1 = get_value(&s);
|
|
let value2 = objint::get_value(s2);
|
|
let mut result = String::new();
|
|
for _x in 0..value2 {
|
|
result.push_str(value1.as_str());
|
|
}
|
|
Ok(vm.ctx.new_str(result))
|
|
} else {
|
|
Err(vm.new_type_error(format!("Cannot multiply {:?} and {:?}", s, s2)))
|
|
}
|
|
}
|
|
|
|
// TODO: should with following format
|
|
// class str(object='')
|
|
// class str(object=b'', encoding='utf-8', errors='strict')
|
|
fn str_new(vm: &mut VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
if args.args.len() == 1 {
|
|
return Ok(vm.new_str("".to_string()));
|
|
}
|
|
|
|
if args.args.len() > 2 {
|
|
panic!("str expects exactly one parameter");
|
|
};
|
|
|
|
vm.to_str(args.args[1].clone())
|
|
}
|
|
|
|
impl PySliceableSequence for String {
|
|
fn do_slice(&self, start: usize, stop: usize) -> Self {
|
|
self[start..stop].to_string()
|
|
}
|
|
fn do_stepped_slice(&self, start: usize, stop: usize, step: usize) -> Self {
|
|
self[start..stop].chars().step_by(step).collect()
|
|
}
|
|
fn len(&self) -> usize {
|
|
self.len()
|
|
}
|
|
}
|
|
|
|
pub fn subscript(vm: &mut VirtualMachine, value: &String, b: PyObjectRef) -> PyResult {
|
|
// let value = a
|
|
match &(*b.borrow()).kind {
|
|
&PyObjectKind::Integer { value: ref pos } => {
|
|
let idx = value.get_pos(*pos);
|
|
Ok(vm.new_str(value[idx..idx + 1].to_string()))
|
|
}
|
|
&PyObjectKind::Slice {
|
|
start: _,
|
|
stop: _,
|
|
step: _,
|
|
} => Ok(vm.new_str(value.get_slice_items(&b))),
|
|
_ => panic!(
|
|
"TypeError: indexing type {:?} with index {:?} is not supported (yet?)",
|
|
value, b
|
|
),
|
|
}
|
|
}
|