use giterated_models::{ error::OperationError, instance::Instance, object::{ AnyObject, GiteratedObject, Object, ObjectRequest, ObjectRequestError, ObjectResponse, }, object_backend::ObjectBackend, operation::{AnyOperation, GiteratedOperation}, }; use std::{any::Any, collections::HashMap, fmt::Debug, str::FromStr, sync::Arc}; use tokio::sync::Mutex; use tracing::warn; use crate::{ state::HandlerState, update::{HandleSettingUpdatedFunction, HandleValueUpdatedFunction, ValueUpdateKind}, OperationHandlers, }; use crate::StackOperationState; #[derive(Clone)] pub struct GiteratedBackend { state: S, handlers: Arc>, } impl GiteratedBackend { pub fn new(state: S, handlers: OperationHandlers) -> Self { Self { state, handlers: Arc::new(handlers), } } pub fn state(&self) -> &S { &self.state } } #[async_trait::async_trait] impl ObjectBackend for GiteratedBackend { async fn object_operation( &self, object: O, operation: &str, payload: D, operation_state: &StackOperationState, ) -> Result> where O: GiteratedObject + Debug + 'static, D: GiteratedOperation + Debug, { let serialized = serde_json::to_value(payload).map_err(|e| OperationError::Internal(e.to_string()))?; let object = (Box::new(object) as Box) .downcast::() .unwrap(); if operation == ObjectRequest::operation_name() { // We're doing an object request let raw_result = self .handlers .resolve_object( *(Box::new(object) as Box) .downcast() .unwrap(), serde_json::from_value(serialized).unwrap(), self.state.clone(), &operation_state, ) .await; return match raw_result { Ok(result) => Ok(serde_json::from_slice(&result) .map_err(|e| OperationError::Internal(e.to_string()))?), Err(err) => match err { OperationError::Internal(internal) => { warn!( "Internal Error: {:?}", OperationError::<()>::Internal(internal.clone()) ); Err(OperationError::Internal(internal)) } OperationError::Unhandled => Err(OperationError::Unhandled), OperationError::Operation(err) => Err(OperationError::Operation( serde_json::from_slice(&err) .map_err(|e| OperationError::Internal(e.to_string()))?, )), }, }; } let raw_result = self .handlers .handle( &*object, operation, AnyOperation(serialized), self.state.clone(), &operation_state, ) .await; match raw_result { Ok(result) => Ok(serde_json::from_slice(&result) .map_err(|e| OperationError::Internal(e.to_string()))?), Err(err) => match err { OperationError::Internal(internal) => { warn!( "Internal Error: {:?}", OperationError::<()>::Internal(internal.clone()) ); Err(OperationError::Internal(internal)) } OperationError::Unhandled => Err(OperationError::Unhandled), OperationError::Operation(err) => Err(OperationError::Operation( serde_json::from_slice(&err) .map_err(|e| OperationError::Internal(e.to_string()))?, )), }, } } async fn get_object( &self, object_str: &str, operation_state: &StackOperationState, ) -> Result, OperationError> { let raw_result = self .handlers .resolve_object( Instance::from_str("giterated.dev").unwrap(), ObjectRequest(object_str.to_string()), self.state.clone(), operation_state, ) .await; let object: ObjectResponse = match raw_result { Ok(result) => Ok(serde_json::from_slice(&result) .map_err(|e| OperationError::Internal(e.to_string()))?), Err(err) => match err { OperationError::Internal(internal) => { warn!( "Internal Error: {:?}", OperationError::<()>::Internal(internal.clone()) ); Err(OperationError::Internal(internal)) } OperationError::Unhandled => Err(OperationError::Unhandled), OperationError::Operation(err) => Err(OperationError::Operation( serde_json::from_slice(&err) .map_err(|e| OperationError::Internal(e.to_string()))?, )), }, }?; unsafe { Ok(Object::new_unchecked( O::from_str(&object.0) .map_err(|_| OperationError::Internal("deserialize failure".to_string()))?, self.clone(), )) } } }