metacall 0.5.9

Call NodeJS, TypeScript, Python, C#, Ruby... functions from Rust (a Rust Port for MetaCall).
use super::{MetaCallError, MetaCallNull, MetaCallValue};
use crate::{
    bindings::{metacall_value_destroy, metacall_value_to_function, metacallfv_s},
    cast,
};
use std::{
    ffi::c_void,
    fmt::{self, Debug, Formatter},
};

/// Represents MetaCall function.
pub struct MetaCallFunction {
    leak: bool,
    value: *mut c_void,
}
unsafe impl Send for MetaCallFunction {}
unsafe impl Sync for MetaCallFunction {}
impl Clone for MetaCallFunction {
    fn clone(&self) -> Self {
        Self {
            leak: true,
            value: self.value,
        }
    }
}
impl Debug for MetaCallFunction {
    fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
        write!(f, "MetaCallFunction {{ ... }}")
    }
}

impl MetaCallFunction {
    #[doc(hidden)]
    pub fn new_raw(value: *mut c_void) -> Self {
        Self { leak: false, value }
    }

    #[doc(hidden)]
    pub fn new_raw_leak(value: *mut c_void) -> Self {
        Self { leak: true, value }
    }

    fn value_to_function(&self) -> *mut c_void {
        unsafe { metacall_value_to_function(self.value) }
    }

    fn call_inner<T: MetaCallValue>(&self, args: impl IntoIterator<Item = T>) -> *mut c_void {
        let mut c_args = cast::metacallobj_to_raw_args(args);
        let ret: *mut c_void =
            unsafe { metacallfv_s(self.value_to_function(), c_args.as_mut_ptr(), 0) };

        for c_arg in c_args {
            unsafe { metacall_value_destroy(c_arg) };
        }

        ret
    }
    /// Calls the function with arguments and witout type casting([MetaCallValue](MetaCallValue)).
    pub fn call_untyped<T: MetaCallValue>(
        &self,
        args: impl IntoIterator<Item = T>,
    ) -> Box<dyn MetaCallValue> {
        cast::raw_to_metacallobj_untyped(self.call_inner(args))
    }
    /// Calls the function without passing arguments and witout type
    /// casting([MetaCallValue](MetaCallValue)).
    pub fn call_untyped_no_arg<T: MetaCallValue>(&self) -> Box<dyn MetaCallValue> {
        cast::raw_to_metacallobj_untyped(self.call_inner([] as [MetaCallNull; 0]))
    }
    /// Calls the function with arguments.
    pub fn call<T: MetaCallValue, U: MetaCallValue>(
        &self,
        args: impl IntoIterator<Item = U>,
    ) -> Result<T, MetaCallError> {
        match cast::raw_to_metacallobj::<T>(self.call_inner(args)) {
            Ok(ret) => Ok(ret),
            Err(original) => Err(MetaCallError::FailedCasting(original)),
        }
    }
    /// Calls the function without arguments.
    pub fn call_no_arg<T: MetaCallValue>(&self) -> Result<T, MetaCallError> {
        self.call::<T, MetaCallNull>([])
    }

    #[doc(hidden)]
    pub fn into_raw(mut self) -> *mut c_void {
        self.leak = true;

        self.value
    }
}

impl Drop for MetaCallFunction {
    fn drop(&mut self) {
        if !self.leak {
            unsafe { metacall_value_destroy(self.value) }
        }
    }
}