| // Licensed to the Apache Software Foundation (ASF) under one |
| // or more contributor license agreements. See the NOTICE file |
| // distributed with this work for additional information |
| // regarding copyright ownership. The ASF licenses this file |
| // to you under the Apache License, Version 2.0 (the |
| // "License"); you may not use this file except in compliance |
| // with the License. You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, |
| // software distributed under the License is distributed on an |
| // "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| // KIND, either express or implied. See the License for the |
| // specific language governing permissions and limitations |
| // under the License. |
| |
| use std::os::raw::c_int; |
| |
| use pyo3::IntoPyObjectExt; |
| use pyo3::ffi; |
| use pyo3::prelude::*; |
| |
| /// A bytes-like object that implements buffer protocol. |
| #[pyclass(module = "opendal")] |
| pub struct Buffer { |
| inner: Vec<u8>, |
| } |
| |
| impl Buffer { |
| pub fn new(inner: Vec<u8>) -> Self { |
| Buffer { inner } |
| } |
| |
| /// Consume self to build a bytes |
| pub fn into_bytes(self, py: Python) -> PyResult<Py<PyAny>> { |
| let buffer = self.into_py_any(py)?; |
| |
| unsafe { Py::<PyAny>::from_owned_ptr_or_err(py, ffi::PyBytes_FromObject(buffer.as_ptr())) } |
| } |
| |
| /// Consume self to build a bytes |
| pub fn into_bytes_ref(self, py: Python) -> PyResult<Bound<PyAny>> { |
| let buffer = self.into_py_any(py)?; |
| let view = |
| unsafe { Bound::from_owned_ptr_or_err(py, ffi::PyBytes_FromObject(buffer.as_ptr()))? }; |
| |
| Ok(view) |
| } |
| } |
| |
| #[pymethods] |
| impl Buffer { |
| unsafe fn __getbuffer__( |
| slf: PyRefMut<Self>, |
| view: *mut ffi::Py_buffer, |
| flags: c_int, |
| ) -> PyResult<()> { |
| let bytes = slf.inner.as_slice(); |
| let ret = unsafe { |
| ffi::PyBuffer_FillInfo( |
| view, |
| slf.as_ptr() as *mut _, |
| bytes.as_ptr() as *mut _, |
| bytes.len().try_into().unwrap(), |
| 1, // read only |
| flags, |
| ) |
| }; |
| if ret == -1 { |
| return Err(PyErr::fetch(slf.py())); |
| } |
| Ok(()) |
| } |
| } |
| |
| /// Macro to create and register a PyO3 submodule with multiple classes. |
| /// |
| /// Example: |
| /// ```rust |
| /// add_pymodule!(py, m, "services", [PyScheme, PyOtherClass]); |
| /// ``` |
| #[macro_export] |
| macro_rules! add_pymodule { |
| ($py:expr, $parent:expr, $name:expr, [$($cls:ty),* $(,)?]) => {{ |
| let sub_module = pyo3::types::PyModule::new($py, $name)?; |
| $( |
| sub_module.add_class::<$cls>()?; |
| )* |
| $parent.add_submodule(&sub_module)?; |
| $py.import("sys")? |
| .getattr("modules")? |
| .set_item(format!("opendal.{}", $name), &sub_module)?; |
| Ok::<_, pyo3::PyErr>(()) |
| }}; |
| } |
| |
| /// Macro to create and register a PyO3 submodule containing exception types. |
| /// |
| /// Example: |
| /// ```rust |
| /// add_pyexceptions!(py, m, "exceptions", [Error, Unexpected]); |
| /// ``` |
| #[macro_export] |
| macro_rules! add_pyexceptions { |
| ($py:expr, $parent:expr, $name:expr, [$($exc:ty),* $(,)?]) => {{ |
| let sub_module = pyo3::types::PyModule::new($py, $name)?; |
| $( |
| sub_module.add(stringify!($exc), $py.get_type::<$exc>())?; |
| )* |
| $parent.add_submodule(&sub_module)?; |
| $py.import("sys")? |
| .getattr("modules")? |
| .set_item(format!("opendal.{}", $name), &sub_module)?; |
| Ok::<_, pyo3::PyErr>(()) |
| }}; |
| } |