use std::sync::{ Arc, Mutex };
use std::collections::HashMap ;
use wasmtime::component::{ Linker, Val };
use crate::{ Interface, PluginContext };
use crate::cardinality::{ Any, AtLeastOne, AtMostOne, Cardinality, ExactlyOne };
use crate::plugin_instance::PluginInstance ;
type PluginSockets<PluginId, Ctx, Plugins> =
<Plugins as Cardinality<PluginId, PluginInstance<Ctx>>>::Rebind<Mutex<PluginInstance<Ctx>>> ;
type DispatchResults<PluginId, Ctx, Plugins> =
<PluginSockets<PluginId, Ctx, Plugins> as Cardinality<PluginId, Mutex<PluginInstance<Ctx>>>>::Rebind<
Result<wasmtime::component::Val, crate::DispatchError>
>;
type DispatchVals<PluginId, Ctx, Plugins> =
<PluginSockets<PluginId, Ctx, Plugins> as Cardinality<PluginId, Mutex<PluginInstance<Ctx>>>>::Rebind<
wasmtime::component::Val
>;
struct BindingData<PluginId, Ctx, Plugins>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
Plugins: Cardinality<PluginId, PluginInstance<Ctx>>,
PluginSockets<PluginId, Ctx, Plugins>: Send + Sync,
{
package_name: String,
interfaces: HashMap<String, Interface>,
plugins: PluginSockets<PluginId, Ctx, Plugins>,
}
pub struct Binding<PluginId, Ctx, Plugins = ExactlyOne<PluginId, PluginInstance<Ctx>>>(Arc<BindingData<PluginId, Ctx, Plugins>>)
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
Plugins: Cardinality<PluginId, PluginInstance<Ctx>> + 'static,
PluginSockets<PluginId, Ctx, Plugins>: Send + Sync;
impl<PluginId, Ctx, Plugins> Clone for Binding<PluginId, Ctx, Plugins>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
Plugins: Cardinality<PluginId, PluginInstance<Ctx>> + 'static,
PluginSockets<PluginId, Ctx, Plugins>: Send + Sync,
{
fn clone( &self ) -> Self {
Self( Arc::clone( &self.0 ))
}
}
impl<PluginId, Ctx, Plugins> std::fmt::Debug for Binding<PluginId, Ctx, Plugins>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + std::fmt::Debug + 'static,
Ctx: PluginContext + std::fmt::Debug + 'static,
Plugins: Cardinality<PluginId, PluginInstance<Ctx>> + 'static,
PluginSockets<PluginId, Ctx, Plugins>: Send + Sync,
PluginSockets<PluginId, Ctx, Plugins>: std::fmt::Debug,
{
fn fmt( &self, f: &mut std::fmt::Formatter<'_> ) -> std::fmt::Result {
f.debug_struct( "Binding" )
.field( "package_name", &self.0.package_name )
.field( "interfaces", &self.0.interfaces )
.field( "plugins", &self.0.plugins )
.finish()
}
}
impl<PluginId, Ctx, Plugins> Binding<PluginId, Ctx, Plugins>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
Plugins: Cardinality<PluginId, PluginInstance<Ctx>> + 'static,
PluginSockets<PluginId, Ctx, Plugins>: Cardinality<PluginId, Mutex<PluginInstance<Ctx>>>,
PluginSockets<PluginId, Ctx, Plugins>: Send + Sync,
{
pub fn new(
package_name: impl Into<String>,
interfaces: HashMap<String, Interface>,
plugins: Plugins
) -> Self {
Self( Arc::new( BindingData {
package_name: package_name.into(),
interfaces,
plugins: plugins.map_mut( Mutex::new ),
}))
}
pub(crate) fn add_to_linker( binding: &Binding<PluginId, Ctx, Plugins>, linker: &mut Linker<Ctx> ) -> Result<(), wasmtime::Error>
where
PluginId: Into<Val>,
DispatchVals<PluginId, Ctx, Plugins>: Into<Val>,
{
binding.0.interfaces.iter().try_for_each(|( name, interface )| {
let interface_ident = format!( "{}/{}", binding.0.package_name, name );
interface.add_to_linker( linker, &binding.0.package_name, &interface_ident, name, binding )
})
}
pub(crate) fn plugins( &self ) -> &PluginSockets<PluginId, Ctx, Plugins> {
&self.0.plugins
}
pub fn dispatch(
&self,
interface_name: &str,
function_name: &str,
args: &[wasmtime::component::Val],
) -> Result<DispatchResults<PluginId, Ctx, Plugins>, crate::DispatchError> {
let interface = self.0.interfaces.get( interface_name )
.ok_or_else(|| crate::DispatchError::InvalidInterfacePath( format!( "{}/{}", self.0.package_name, interface_name )))?;
let function = interface.function( function_name )
.ok_or_else(|| crate::DispatchError::InvalidFunction( function_name.to_string() ))?;
Ok( self.0.plugins.map(| _, plugin | plugin
.lock().map_err(|_| crate::DispatchError::LockRejected )
.and_then(| mut lock | lock.dispatch(
&self.0.package_name,
interface_name,
function_name,
function,
args,
))
))
}
}
#[derive( Debug )]
pub enum BindingAny<PluginId, Ctx>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
{
ExactlyOne( Binding<PluginId, Ctx, ExactlyOne<PluginId, PluginInstance<Ctx>>> ),
AtMostOne( Binding<PluginId, Ctx, AtMostOne<PluginId, PluginInstance<Ctx>>> ),
AtLeastOne( Binding<PluginId, Ctx, AtLeastOne<PluginId, PluginInstance<Ctx>>> ),
Any( Binding<PluginId, Ctx, Any<PluginId, PluginInstance<Ctx>>> ),
}
impl<PluginId, Ctx> BindingAny<PluginId, Ctx>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + Into<Val> + 'static,
Ctx: PluginContext + 'static,
{
pub(crate) fn add_to_linker( &self, linker: &mut Linker<Ctx> ) -> Result<(), wasmtime::Error> {
match self {
Self::ExactlyOne( binding ) => Binding::add_to_linker( binding, linker ),
Self::AtMostOne( binding ) => Binding::add_to_linker( binding, linker ),
Self::AtLeastOne( binding ) => Binding::add_to_linker( binding, linker ),
Self::Any( binding ) => Binding::add_to_linker( binding, linker ),
}
}
}
impl<PluginId, Ctx> From<Binding<PluginId, Ctx, ExactlyOne<PluginId, PluginInstance<Ctx>>>> for BindingAny<PluginId, Ctx>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
{
fn from( binding: Binding<PluginId, Ctx, ExactlyOne<PluginId, PluginInstance<Ctx>>> ) -> Self {
Self::ExactlyOne( binding )
}
}
impl<PluginId, Ctx> From<Binding<PluginId, Ctx, AtMostOne<PluginId, PluginInstance<Ctx>>>> for BindingAny<PluginId, Ctx>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
{
fn from( binding: Binding<PluginId, Ctx, AtMostOne<PluginId, PluginInstance<Ctx>>> ) -> Self {
Self::AtMostOne( binding )
}
}
impl<PluginId, Ctx> From<Binding<PluginId, Ctx, AtLeastOne<PluginId, PluginInstance<Ctx>>>> for BindingAny<PluginId, Ctx>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
{
fn from( binding: Binding<PluginId, Ctx, AtLeastOne<PluginId, PluginInstance<Ctx>>> ) -> Self {
Self::AtLeastOne( binding )
}
}
impl<PluginId, Ctx> From<Binding<PluginId, Ctx, Any<PluginId, PluginInstance<Ctx>>>> for BindingAny<PluginId, Ctx>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
{
fn from( binding: Binding<PluginId, Ctx, Any<PluginId, PluginInstance<Ctx>>> ) -> Self {
Self::Any( binding )
}
}
impl<PluginId, Ctx, Plugins> Binding<PluginId, Ctx, Plugins>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
Plugins: Cardinality<PluginId, PluginInstance<Ctx>>,
PluginSockets<PluginId, Ctx, Plugins>: Send + Sync,
BindingAny<PluginId, Ctx>: From<Binding<PluginId, Ctx, Plugins>>,
{
pub fn into_any( self ) -> BindingAny<PluginId, Ctx> {
self.into()
}
}
impl<PluginId, Ctx> Clone for BindingAny<PluginId, Ctx>
where
PluginId: std::hash::Hash + Eq + Clone + Send + Sync + 'static,
Ctx: PluginContext + 'static,
{
fn clone( &self ) -> Self {
match self {
Self::ExactlyOne( binding ) => Self::ExactlyOne( binding.clone() ),
Self::AtMostOne( binding ) => Self::AtMostOne( binding.clone() ),
Self::AtLeastOne( binding ) => Self::AtLeastOne( binding.clone() ),
Self::Any( binding ) => Self::Any( binding.clone() ),
}
}
}