use std::error::Error;
use std::any::Any;
pub const OP_GET_CAPABILITY_DESCRIPTOR: &str = "GetCapabilityDescriptor";
pub trait Dispatcher: Any + Send + Sync {
fn dispatch(&self, actor: &str, op: &str, msg: &[u8]) -> Result<Vec<u8>, Box<dyn Error>>;
}
#[repr(C)]
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Default)]
pub struct CapabilityDescriptor {
pub id: String,
pub name: String,
pub version: String,
pub revision: u32,
pub long_description: String,
pub supported_operations: Vec<OperationDescriptor>,
}
impl CapabilityDescriptor {
pub fn builder() -> CapabilityDescriptorBuilder {
CapabilityDescriptorBuilder::new()
}
}
#[derive(Default)]
pub struct CapabilityDescriptorBuilder {
descriptor: CapabilityDescriptor,
}
impl CapabilityDescriptorBuilder {
fn new() -> CapabilityDescriptorBuilder {
CapabilityDescriptorBuilder::default()
}
pub fn id(self, id: &str) -> Self {
CapabilityDescriptorBuilder {
descriptor: CapabilityDescriptor {
id: id.to_string(),
..self.descriptor
},
}
}
pub fn name(self, name: &str) -> Self {
CapabilityDescriptorBuilder {
descriptor: CapabilityDescriptor {
name: name.to_string(),
..self.descriptor
},
}
}
pub fn long_description(self, desc: &str) -> Self {
CapabilityDescriptorBuilder {
descriptor: CapabilityDescriptor {
long_description: desc.to_string(),
..self.descriptor
},
}
}
pub fn version(self, ver: &str) -> Self {
CapabilityDescriptorBuilder {
descriptor: CapabilityDescriptor {
version: ver.to_string(),
..self.descriptor
},
}
}
pub fn revision(self, rev: u32) -> Self {
CapabilityDescriptorBuilder {
descriptor: CapabilityDescriptor {
revision: rev,
..self.descriptor
},
}
}
pub fn with_operation(self, name: &str, direction: OperationDirection, doctext: &str) -> Self {
let mut newops = self.descriptor.supported_operations;
newops.push(OperationDescriptor::new(name, direction, doctext));
CapabilityDescriptorBuilder {
descriptor: CapabilityDescriptor {
supported_operations: newops,
..self.descriptor
},
}
}
pub fn build(self) -> CapabilityDescriptor {
self.descriptor
}
}
#[repr(C)]
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
pub struct OperationDescriptor {
pub name: String,
pub direction: OperationDirection,
pub doctext: String,
}
impl OperationDescriptor {
pub fn new(name: &str, direction: OperationDirection, doctext: &str) -> OperationDescriptor {
OperationDescriptor {
name: name.to_string(),
direction,
doctext: doctext.to_string(),
}
}
}
#[repr(C)]
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
#[serde(rename_all = "snake_case")]
pub enum OperationDirection {
ToActor,
ToProvider,
Both,
}
#[derive(Default)]
pub struct NullDispatcher {}
impl NullDispatcher {
pub fn new() -> NullDispatcher {
NullDispatcher {}
}
}
impl Dispatcher for NullDispatcher {
fn dispatch(&self, _actor: &str, _op: &str, _msg: &[u8]) -> Result<Vec<u8>, Box<dyn Error>> {
unimplemented!()
}
}
pub trait CapabilityProvider: Any + Send + Sync {
fn configure_dispatch(&self, dispatcher: Box<dyn Dispatcher>) -> Result<(), Box<dyn Error>>;
fn handle_call(&self, actor: &str, op: &str, msg: &[u8]) -> Result<Vec<u8>, Box<dyn Error>>;
}
#[macro_export]
macro_rules! capability_provider {
($provider_type:ty, $constructor:path) => {
#[no_mangle]
pub extern "C" fn __capability_provider_create(
) -> *mut $crate::capabilities::CapabilityProvider {
let constructor: fn() -> $provider_type = $constructor;
let object = constructor();
let boxed: Box<$crate::capabilities::CapabilityProvider> = Box::new(object);
Box::into_raw(boxed)
}
};
}
#[cfg(test)]
mod test {
use super::{CapabilityDescriptor, OperationDescriptor, OperationDirection};
#[test]
fn descriptor_certify_desired_json_format() {
let d = CapabilityDescriptor {
name: "test".to_string(),
id: "wascc:testing".to_string(),
version: "0.0.1".to_string(),
revision: 1,
long_description: "this is a test".to_string(),
supported_operations: vec![OperationDescriptor {
direction: OperationDirection::ToActor,
doctext: "this is a test".to_string(),
name: "OperationDumboDrop".to_string(),
}],
};
let s = serde_json::to_string(&d).unwrap();
assert_eq!(s, "{\"id\":\"wascc:testing\",\"name\":\"test\",\"version\":\"0.0.1\",\"revision\":1,\"long_description\":\"this is a test\",\"supported_operations\":[{\"name\":\"OperationDumboDrop\",\"direction\":\"to_actor\",\"doctext\":\"this is a test\"}]}".to_string());
}
}