pub mod handle; pub mod operation_walker; pub mod runtime_handler; use std::{collections::HashMap, fmt::Debug, mem::transmute, ptr::null_mut, sync::Arc}; use giterated_models::{ error::OperationError, object::{GiteratedObject, ObjectRequestError}, operation::{GiteratedOperation, OperationState}, }; use semver::Version; use tracing::{debug, debug_span, trace, trace_span, warn}; use crate::{ callback::{ OperationHandlerCallback, RuntimeState, SettingChangeCallback, SettingGetterCallback, ValueChangeCallback, ValueGetterCallback, }, future::RuntimeFuture, handle::PluginHandle, vtable::{ IntoRuntimeVtable, ObjectVtable, OperationVTable, RuntimeVTable, SettingVtable, ValueVTable, }, AnyFailure, AnySuccess, }; use self::operation_walker::OperationHandlerRules; pub struct State(pub S); impl std::ops::Deref for State { type Target = S; fn deref(&self) -> &Self::Target { &self.0 } } #[derive(Default, Clone)] pub struct TypeMetadata { pub objects: HashMap<&'static str, ObjectVtable>, pub operations: HashMap, OperationVTable>, pub settings: HashMap, SettingVtable>, pub values: HashMap, ValueVTable>, } impl TypeMetadata { pub unsafe fn from_static() -> &'static Self { giterated_static_runtime::get_type_metadata_reference() .cast::() .as_ref() } pub fn register_object(&mut self, object_kind: &'static str, vtable: ObjectVtable) { trace!("Registering type metadata for {}", object_kind); self.objects.insert(object_kind, vtable); } pub fn register_operation( &mut self, object_kind: &'static str, operation_name: &'static str, vtable: OperationVTable, ) { trace!( "Registering operation metadata for {}::{}", object_kind, operation_name ); self.operations.insert( ObjectOperationPair { object_kind, operation_name, }, vtable, ); } pub fn register_setting( &mut self, object_kind: &'static str, setting_name: &'static str, vtable: SettingVtable, ) { trace!("Registering setting {}::{}", object_kind, setting_name); self.settings.insert( ObjectSettingPair { object_kind, setting_name, }, vtable, ); } pub fn register_value( &mut self, object_kind: &'static str, value_name: &'static str, vtable: ValueVTable, ) { trace!("Registering value {}::{}", object_kind, value_name); self.values.insert( ObjectValuePair { object_kind, value_name, }, vtable, ); } } #[derive(Debug, Clone, Copy, Hash, PartialEq, Eq)] pub struct ObjectOperationPair<'s> { pub object_kind: &'s str, pub operation_name: &'s str, } impl<'s> ObjectOperationPair<'s> { pub fn new(object_kind: &'s str, operation_name: &'s str) -> Self { Self { object_kind, operation_name, } } } #[derive(Clone, Copy, Hash, PartialEq, Eq, Debug)] pub struct ObjectSettingPair<'s> { pub object_kind: &'s str, pub setting_name: &'s str, } impl<'s> ObjectSettingPair<'s> { pub fn new(object_kind: &'s str, setting_name: &'s str) -> Self { Self { object_kind, setting_name, } } } #[derive(Clone, Copy, Hash, PartialEq, Eq, Debug)] pub struct ObjectValuePair<'s> { pub object_kind: &'s str, pub value_name: &'s str, } impl<'s> ObjectValuePair<'s> { pub fn new(object_kind: &'s str, value_name: &'s str) -> Self { Self { object_kind, value_name, } } } #[derive(Clone, Copy)] #[repr(C)] pub struct PluginState { pub inner: *mut (), } unsafe impl Send for PluginState {} unsafe impl Sync for PluginState {} impl PluginState { pub unsafe fn transmute_owned(&mut self) -> Box { Box::from_raw(self.inner as *mut T) } pub unsafe fn transmute_ref(&self) -> &T { let ptr: *const T = transmute(self.inner); ptr.as_ref().unwrap() } } impl PluginState { pub fn from(source: S) -> Self { Self { inner: Box::into_raw(Box::new(source)) as *mut _, } } pub unsafe fn from_box(raw: Box) -> Self { Self { inner: Box::into_raw(raw) as *mut _, } } pub unsafe fn from_raw_ptr(raw: *const ()) -> Self { Self { inner: raw as *mut (), } } pub fn null() -> Self { Self { inner: null_mut() } } } pub struct Runtime { plugins: Vec, handlers: RuntimeHandlers, } pub struct FfiRuntimeMetadata { runtime: PluginState, } impl IntoRuntimeVtable for Runtime { unsafe extern "C" fn handle( _this: PluginState, _object_kind: crate::FFIBox, _operation_name: crate::FFIBox, _object: crate::FFIBox, _operation_payload: crate::FFIBox<[u8]>, _operation_state: crate::FFIBox<[u8]>, ) -> RuntimeFuture>> { todo!() } unsafe extern "C" fn get_object( this: PluginState, object_str: &str, operation_state: *mut OperationState, ) -> Result> { let runtime_state = unsafe { RuntimeState::from_static() }; let type_metada = runtime_state .vtable .type_metadata .as_ref() .unwrap_or_else(|| { let runtime = this.transmute_ref::(); &runtime .plugins .first() .unwrap() .initialization .type_metadata }); for (object_type, object_vtable) in &type_metada.objects { if let Ok(object) = (object_vtable.from_str)(object_str) { return Ok(object); } } Err(OperationError::Operation(ObjectRequestError::Invalid)) } } impl Runtime { pub fn new() -> Box { Box::new(Self { plugins: vec![], handlers: RuntimeHandlers::default(), }) } pub fn state(self: &Box) -> RuntimeState { RuntimeState { vtable: RuntimeVTable { runtime: PluginState::from(self), handle_fn: ::handle, get_object: ::get_object, type_metadata: unsafe { TypeMetadata::from_static() }, }, } } pub fn insert_plugin(&mut self, plugin: PluginHandle) { let _guard = debug_span!("inserting plugin", meta = debug(&plugin.meta)).entered(); for (pair, callback) in &plugin.initialization.operation_handlers { let _guard = trace_span!("processing operation handler callbacks", pair = debug(pair)).entered(); if self .handlers .operation_handlers .insert(*pair, (RuntimeDomain::from_plugin(&plugin), *callback)) .is_some() { warn!("Warning! Insertion of handler for overwrote a previous handler.") } trace!("Insertion of operation handler successful") } for (pair, callback) in &plugin.initialization.value_getters { let _guard = trace_span!("processing value getter callbacks", pair = debug(pair)).entered(); if self .handlers .value_getters .insert(*pair, (RuntimeDomain::from_plugin(&plugin), *callback)) .is_some() { warn!("Warning! Insertion of handler for overwrote a previous handler.") } trace!("Insertion of operation handler successful") } for (pair, callback) in &plugin.initialization.setting_getters { let _guard = trace_span!("processing setting getter callbacks", pair = debug(pair)).entered(); if self .handlers .setting_getters .insert(*pair, (RuntimeDomain::from_plugin(&plugin), *callback)) .is_some() { warn!("Warning! Insertion of setting handler for overwrote a previous handler.") } trace!("Insertion of setting handler successful") } self.plugins.push(plugin); } pub fn init(self: Box) { unsafe { giterated_static_runtime::initialize_runtime(Box::into_raw(self).cast::<()>()) } } // pub async fn handle( // &self, // object_kind: &str, // operation_name: &str, // object: &str, // operation_payload: &[u8], // operation_state: &OperationState, // ) -> Result> { // let rules = self.handlers.handle_operation(object_kind, operation_name); // let state = self.state(operation_state); // rules.handle(&state, object, operation_payload).await // } pub fn handle_typed>( &self, _object: O, _operation: D, ) -> Result> { todo!() } } #[derive(Default)] pub struct RuntimeHandlers { operation_handlers: HashMap, (RuntimeDomain, OperationHandlerCallback)>, value_getters: HashMap, (RuntimeDomain, ValueGetterCallback)>, setting_getters: HashMap, (RuntimeDomain, SettingGetterCallback)>, value_change: HashMap, (RuntimeDomain, ValueChangeCallback)>, setting_change: HashMap, (RuntimeDomain, SettingChangeCallback)>, } unsafe impl Send for RuntimeHandlers {} unsafe impl Sync for RuntimeHandlers {} impl RuntimeHandlers { pub fn operation_handler( &mut self, pair: ObjectOperationPair<'static>, handler: OperationHandlerCallback, domain: RuntimeDomain, ) { trace!( "Inserting operation handler for {}::{}", pair.object_kind, pair.operation_name ); // There can only be one handler per operation (at least for now?), send a warning if // a newly registered handler overwrites the previous handler. if self .operation_handlers .insert(pair, (domain, handler)) .is_some() { debug!("Warning! A newly inserted operation handler for {}::{} overwrites a previous handler.", pair.object_kind, pair.operation_name); } } pub fn value_getter( &mut self, pair: ObjectValuePair<'static>, handler: ValueGetterCallback, domain: RuntimeDomain, ) { trace!( "Inserting value getter for {}::{}", pair.object_kind, pair.value_name ); if self.value_getters.insert(pair, (domain, handler)).is_some() { debug!( "Warning! A newly inserted value getter for {}::{} overwrites a previous handler.", pair.object_kind, pair.value_name ); } } pub fn setting_getter( &mut self, pair: ObjectSettingPair<'static>, handler: SettingGetterCallback, domain: RuntimeDomain, ) { trace!( "Inserting setting getter for {}::{}", pair.object_kind, pair.setting_name ); if self .setting_getters .insert(pair, (domain, handler)) .is_some() { debug!("Warning! A newly inserted setting getter for {}::{} overwrites a previous handler.", pair.object_kind, pair.setting_name); } } pub fn value_change( &mut self, pair: ObjectValuePair<'static>, handler: ValueChangeCallback, domain: RuntimeDomain, ) { trace!( "Inserting value change handler for {}::{}", pair.object_kind, pair.value_name ); if self.value_change.insert(pair, (domain, handler)).is_some() { debug!("Warning! A newly inserted value change handler for {}::{} overwrites a previous handler.", pair.object_kind, pair.value_name); panic!("Not intended"); } } pub fn setting_change( &mut self, pair: ObjectSettingPair<'static>, handler: SettingChangeCallback, domain: RuntimeDomain, ) { trace!( "Inserting setting change handler for {}::{}", pair.object_kind, pair.setting_name ); if self .setting_change .insert(pair, (domain, handler)) .is_some() { debug!("Warning! A newly inserted setting change handler for {}::{} overwrites a previous handler.", pair.object_kind, pair.setting_name); panic!("Not intended"); } } } impl RuntimeHandlers { pub fn handle_operation<'o>( &'o self, object_kind: &'o str, operation_name: &'o str, ) -> OperationHandlerRules<'o> { OperationHandlerRules::new(object_kind, operation_name, self) } } pub struct RuntimeDomain { plugin: PluginHandle, } impl RuntimeDomain { pub fn from_plugin(plugin: &PluginHandle) -> Self { Self { plugin: plugin.clone(), } } pub fn object_vtable(&self, object_kind: &str) -> Option { self.plugin .initialization .type_metadata .objects .get(object_kind) .copied() } pub fn operation_vtable( &self, object_kind: &str, operation_name: &str, ) -> Option { self.plugin .initialization .type_metadata .operations .get(&ObjectOperationPair::new(object_kind, operation_name)) .copied() } pub fn setting_vtable(&self, object_kind: &str, setting_name: &str) -> Option { self.plugin .initialization .type_metadata .settings .get(&ObjectSettingPair::new(object_kind, setting_name)) .copied() } pub fn value_vtable(&self, object_kind: &str, value_name: &str) -> Option { self.plugin .initialization .type_metadata .values .get(&ObjectValuePair::new(object_kind, value_name)) .copied() } } #[derive(Clone, Debug)] pub struct PluginMeta { pub name: String, pub version: Version, } #[repr(C)] pub struct FFIPluginMeta { pub name: *const u8, pub name_len: usize, pub version: *const u8, pub version_len: usize, } pub struct RuntimePlugin { handle: PluginHandle, type_metadata: Arc, } impl RuntimePlugin { pub fn plugin_meta(&self) -> PluginMeta { let meta = unsafe { self.handle.raw.plugin_meta() }; let name = unsafe { std::slice::from_raw_parts(meta.name, meta.name_len) }; let version = unsafe { std::slice::from_raw_parts(meta.version, meta.version_len) }; let name = std::str::from_utf8(name).unwrap(); let version = std::str::from_utf8(version).unwrap(); PluginMeta { name: String::from(name), version: Version::parse(version).unwrap(), } } }