numcodecs_python/
codec.rs1use pyo3::{
2 ffi::PyTypeObject,
3 intern,
4 prelude::*,
5 sync::GILOnceCell,
6 types::{DerefToPyAny, IntoPyDict, PyDict, PyType},
7 PyTypeInfo,
8};
9
10#[expect(unused_imports)] use crate::PyCodecClassMethods;
12use crate::{sealed::Sealed, PyCodecClass};
13
14#[repr(transparent)]
20pub struct PyCodec {
21 _codec: PyAny,
22}
23
24pub trait PyCodecMethods<'py>: Sealed {
26 fn encode(&self, buf: Borrowed<'_, 'py, PyAny>) -> Result<Bound<'py, PyAny>, PyErr>;
37
38 fn decode(
56 &self,
57 buf: Borrowed<'_, 'py, PyAny>,
58 out: Option<Borrowed<'_, 'py, PyAny>>,
59 ) -> Result<Bound<'py, PyAny>, PyErr>;
60
61 fn get_config(&self) -> Result<Bound<'py, PyDict>, PyErr>;
72
73 fn class(&self) -> Bound<'py, PyCodecClass>;
75}
76
77impl<'py> PyCodecMethods<'py> for Bound<'py, PyCodec> {
78 fn encode(&self, buf: Borrowed<'_, 'py, PyAny>) -> Result<Bound<'py, PyAny>, PyErr> {
79 let py = self.py();
80
81 self.as_any().call_method1(intern!(py, "encode"), (buf,))
82 }
83
84 fn decode(
85 &self,
86 buf: Borrowed<'_, 'py, PyAny>,
87 out: Option<Borrowed<'_, 'py, PyAny>>,
88 ) -> Result<Bound<'py, PyAny>, PyErr> {
89 let py = self.as_any().py();
90
91 self.as_any().call_method(
92 intern!(py, "decode"),
93 (buf,),
94 Some(&[(intern!(py, "out"), out)].into_py_dict(py)?),
95 )
96 }
97
98 fn get_config(&self) -> Result<Bound<'py, PyDict>, PyErr> {
99 let py = self.as_any().py();
100
101 self.as_any()
102 .call_method0(intern!(py, "get_config"))?
103 .extract()
104 }
105
106 #[expect(clippy::expect_used)]
107 fn class(&self) -> Bound<'py, PyCodecClass> {
108 self.as_any()
110 .get_type()
111 .extract()
112 .expect("Codec's class must be a CodecClass")
113 }
114}
115
116impl Sealed for Bound<'_, PyCodec> {}
117
118#[doc(hidden)]
119impl DerefToPyAny for PyCodec {}
120
121#[doc(hidden)]
122#[expect(unsafe_code)]
123unsafe impl PyTypeInfo for PyCodec {
124 const MODULE: Option<&'static str> = Some("numcodecs.abc");
125 const NAME: &'static str = "Codec";
126
127 #[inline]
128 fn type_object_raw(py: Python) -> *mut PyTypeObject {
129 static CODEC_TYPE: GILOnceCell<Py<PyType>> = GILOnceCell::new();
130
131 let ty = CODEC_TYPE.import(py, "numcodecs.abc", "Codec");
132
133 #[expect(clippy::expect_used)]
134 let ty = ty.expect("failed to access the `numpy.abc.Codec` type object");
135
136 ty.as_type_ptr()
137 }
138}