blob: f3e736c2ec1a4d27cb241db5d366cd47e89f91b4 [file] [log] [blame]
// 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>(())
}};
}