1use crate::err::{PyErr, PyResult};
2use crate::ffi;
3use crate::ffi_ptr_ext::FfiPtrExt;
4use crate::types::any::PyAnyMethods;
5use crate::{Bound, IntoPyObject, PyAny, Python};
6use std::convert::Infallible;
7
8#[repr(transparent)]
18pub struct PySlice(PyAny);
19
20pyobject_native_type!(
21 PySlice,
22 ffi::PySliceObject,
23 pyobject_native_static_type_object!(ffi::PySlice_Type),
24 #checkfunction=ffi::PySlice_Check
25);
26
27#[derive(Debug, Eq, PartialEq)]
29pub struct PySliceIndices {
30 pub start: isize,
34 pub stop: isize,
38 pub step: isize,
40 pub slicelength: usize,
42}
43
44impl PySliceIndices {
45 pub fn new(start: isize, stop: isize, step: isize) -> PySliceIndices {
47 PySliceIndices {
48 start,
49 stop,
50 step,
51 slicelength: 0,
52 }
53 }
54}
55
56impl PySlice {
57 pub fn new(py: Python<'_>, start: isize, stop: isize, step: isize) -> Bound<'_, PySlice> {
59 unsafe {
60 ffi::PySlice_New(
61 ffi::PyLong_FromSsize_t(start),
62 ffi::PyLong_FromSsize_t(stop),
63 ffi::PyLong_FromSsize_t(step),
64 )
65 .assume_owned(py)
66 .downcast_into_unchecked()
67 }
68 }
69
70 pub fn full(py: Python<'_>) -> Bound<'_, PySlice> {
72 unsafe {
73 ffi::PySlice_New(ffi::Py_None(), ffi::Py_None(), ffi::Py_None())
74 .assume_owned(py)
75 .downcast_into_unchecked()
76 }
77 }
78}
79
80#[doc(alias = "PySlice")]
86pub trait PySliceMethods<'py>: crate::sealed::Sealed {
87 fn indices(&self, length: isize) -> PyResult<PySliceIndices>;
91}
92
93impl<'py> PySliceMethods<'py> for Bound<'py, PySlice> {
94 fn indices(&self, length: isize) -> PyResult<PySliceIndices> {
95 unsafe {
96 let mut slicelength: isize = 0;
97 let mut start: isize = 0;
98 let mut stop: isize = 0;
99 let mut step: isize = 0;
100 let r = ffi::PySlice_GetIndicesEx(
101 self.as_ptr(),
102 length,
103 &mut start,
104 &mut stop,
105 &mut step,
106 &mut slicelength,
107 );
108 if r == 0 {
109 Ok(PySliceIndices {
110 start,
111 stop,
112 step,
113 slicelength: slicelength as _,
115 })
116 } else {
117 Err(PyErr::fetch(self.py()))
118 }
119 }
120 }
121}
122
123impl<'py> IntoPyObject<'py> for PySliceIndices {
124 type Target = PySlice;
125 type Output = Bound<'py, Self::Target>;
126 type Error = Infallible;
127
128 fn into_pyobject(self, py: Python<'py>) -> Result<Self::Output, Self::Error> {
129 Ok(PySlice::new(py, self.start, self.stop, self.step))
130 }
131}
132
133impl<'py> IntoPyObject<'py> for &PySliceIndices {
134 type Target = PySlice;
135 type Output = Bound<'py, Self::Target>;
136 type Error = Infallible;
137
138 fn into_pyobject(self, py: Python<'py>) -> Result<Self::Output, Self::Error> {
139 Ok(PySlice::new(py, self.start, self.stop, self.step))
140 }
141}
142
143#[cfg(test)]
144mod tests {
145 use super::*;
146
147 #[test]
148 fn test_py_slice_new() {
149 Python::with_gil(|py| {
150 let slice = PySlice::new(py, isize::MIN, isize::MAX, 1);
151 assert_eq!(
152 slice.getattr("start").unwrap().extract::<isize>().unwrap(),
153 isize::MIN
154 );
155 assert_eq!(
156 slice.getattr("stop").unwrap().extract::<isize>().unwrap(),
157 isize::MAX
158 );
159 assert_eq!(
160 slice.getattr("step").unwrap().extract::<isize>().unwrap(),
161 1
162 );
163 });
164 }
165
166 #[test]
167 fn test_py_slice_full() {
168 Python::with_gil(|py| {
169 let slice = PySlice::full(py);
170 assert!(slice.getattr("start").unwrap().is_none(),);
171 assert!(slice.getattr("stop").unwrap().is_none(),);
172 assert!(slice.getattr("step").unwrap().is_none(),);
173 assert_eq!(
174 slice.indices(0).unwrap(),
175 PySliceIndices {
176 start: 0,
177 stop: 0,
178 step: 1,
179 slicelength: 0,
180 },
181 );
182 assert_eq!(
183 slice.indices(42).unwrap(),
184 PySliceIndices {
185 start: 0,
186 stop: 42,
187 step: 1,
188 slicelength: 42,
189 },
190 );
191 });
192 }
193
194 #[test]
195 fn test_py_slice_indices_new() {
196 let start = 0;
197 let stop = 0;
198 let step = 0;
199 assert_eq!(
200 PySliceIndices::new(start, stop, step),
201 PySliceIndices {
202 start,
203 stop,
204 step,
205 slicelength: 0
206 }
207 );
208
209 let start = 0;
210 let stop = 100;
211 let step = 10;
212 assert_eq!(
213 PySliceIndices::new(start, stop, step),
214 PySliceIndices {
215 start,
216 stop,
217 step,
218 slicelength: 0
219 }
220 );
221
222 let start = 0;
223 let stop = -10;
224 let step = -1;
225 assert_eq!(
226 PySliceIndices::new(start, stop, step),
227 PySliceIndices {
228 start,
229 stop,
230 step,
231 slicelength: 0
232 }
233 );
234
235 let start = 0;
236 let stop = -10;
237 let step = 20;
238 assert_eq!(
239 PySliceIndices::new(start, stop, step),
240 PySliceIndices {
241 start,
242 stop,
243 step,
244 slicelength: 0
245 }
246 );
247 }
248}