pub mod handler; pub mod runtime; pub mod state; pub mod update; use std::{any::Any, future::Future, ops::Deref, pin::Pin, sync::Arc}; use core::fmt::Debug; use futures_util::FutureExt; use giterated_models::{ error::OperationError, instance::{ AuthenticationTokenRequest, Instance, RegisterAccountRequest, RepositoryCreateRequest, }, object::GiteratedObject, object_backend::ObjectBackend, operation::GiteratedOperation, repository::{AccessList, Repository}, settings::{GetSetting, SetSetting}, user::User, value::{GetValue, GiteratedObjectValue}, }; use runtime::{GiteratedRuntime, GiteratedRuntimeState}; use serde::{de::DeserializeOwned, Serialize}; #[derive(Clone, Debug, Hash, Eq, PartialEq)] struct ObjectOperationPair { object_name: String, operation_name: String, } pub struct SettingMeta { name: String, deserialize: Box Result, serde_json::Error> + Send + Sync>, } pub struct ValueMeta { name: String, deserialize: Box Result, serde_json::Error> + Send + Sync>, } pub struct ObjectMeta { name: String, from_str: Box Result, ()> + Send + Sync>, any_is_same: Box bool + Send + Sync>, } pub struct OperationMeta { name: String, object_kind: String, deserialize: Box Result, ()> + Send + Sync>, any_is_same: Box bool + Send + Sync>, serialize_success: Box) -> Result, serde_json::Error> + Send + Sync>, serialize_error: Box) -> Result, serde_json::Error> + Send + Sync>, } #[derive(Clone, Debug, Hash, Eq, PartialEq)] pub struct ObjectValuePair { pub object_kind: String, pub value_kind: String, } #[async_trait::async_trait] pub trait GiteratedOperationHandler< L, O: GiteratedObject, D: GiteratedOperation, S: Send + Sync + Clone, >: Send + Sync { fn operation_name(&self) -> &str; fn object_name(&self) -> &str; async fn handle( &self, object: &O, operation: D, state: S, operation_state: &StackOperationState, ) -> Result>; } #[async_trait::async_trait] impl GiteratedOperationHandler<(), O, D, S> for F where F: FnMut( &O, D, S, ) -> Pin< Box>> + Send>, > + Send + Sync + Clone, O: GiteratedObject + Send + Sync, D: GiteratedOperation + 'static, >::Failure: Send, S: Send + Sync + Clone + 'static, { fn operation_name(&self) -> &str { D::operation_name() } fn object_name(&self) -> &str { O::object_name() } async fn handle( &self, object: &O, operation: D, state: S, _operation_state: &StackOperationState, ) -> Result> { self.clone()(object, operation, state).await } } #[async_trait::async_trait] impl GiteratedOperationHandler<(O1,), O, D, S> for F where F: FnMut( &O, D, S, O1, ) -> Pin< Box>> + Send>, > + Send + Sync + Clone, O: GiteratedObject + Send + Sync, D: GiteratedOperation + 'static + Send + Sync, >::Failure: Send, S: Send + Sync + Clone + 'static, O1: FromOperationState, { fn operation_name(&self) -> &str { D::operation_name() } fn object_name(&self) -> &str { O::object_name() } async fn handle( &self, object: &O, operation: D, state: S, operation_state: &StackOperationState, ) -> Result> { let o1 = O1::from_state(object, &operation, operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; self.clone()(object, operation, state, o1).await } } #[async_trait::async_trait] impl GiteratedOperationHandler<(O1, O2), O, D, S> for F where F: FnMut( &O, D, S, O1, O2, ) -> Pin< Box>> + Send>, > + Send + Sync + Clone, O: GiteratedObject + Send + Sync, D: GiteratedOperation + 'static + Send + Sync, >::Failure: Send, S: Send + Sync + Clone + 'static, O1: FromOperationState, O2: FromOperationState, { fn operation_name(&self) -> &str { D::operation_name() } fn object_name(&self) -> &str { O::object_name() } async fn handle( &self, object: &O, operation: D, state: S, operation_state: &StackOperationState, ) -> Result> { let o1 = O1::from_state(object, &operation, operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; let o2 = O2::from_state(object, &operation, operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; self.clone()(object, operation, state, o1, o2).await } } #[async_trait::async_trait] impl GiteratedOperationHandler<(O1, O2, O3), O, D, S> for F where F: FnMut( &O, D, S, O1, O2, O3, ) -> Pin< Box>> + Send>, > + Send + Sync + Clone, O: GiteratedObject + Send + Sync, D: GiteratedOperation + 'static + Send + Sync, >::Failure: Send, S: Send + Sync + Clone + 'static, O1: FromOperationState, O2: FromOperationState, O3: FromOperationState, { fn operation_name(&self) -> &str { D::operation_name() } fn object_name(&self) -> &str { O::object_name() } async fn handle( &self, object: &O, operation: D, state: S, operation_state: &StackOperationState, ) -> Result> { let o1 = O1::from_state(object, &operation, operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; let o2 = O2::from_state(object, &operation, operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; let o3 = O3::from_state(object, &operation, operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; self.clone()(object, operation, state, o1, o2, o3).await } } pub struct OperationWrapper { func: Box< dyn Fn( Box, Box, &(dyn Any + Send + Sync), StackOperationState, ) -> Pin, OperationError>>> + Send>> + Send + Sync, >, state: Box, } impl OperationWrapper { pub fn new< A, O: GiteratedObject + Send + Sync + 'static, D: GiteratedOperation + 'static, F: GiteratedOperationHandler + Send + Sync + 'static + Clone, S: GiteratedRuntimeState + 'static, >( handler: F, state: S, ) -> Self { let handler = Arc::new(Box::pin(handler)); Self { func: Box::new(move |object, operation, state, operation_state| { let handler = handler.clone(); let state = state.downcast_ref::().unwrap().clone(); async move { let handler = handler.clone(); let object: Box = object.downcast().unwrap(); let operation: Box = operation.downcast().unwrap(); let result = handler .handle(&object, *operation, state, &operation_state) .await; result .map(|success| serde_json::to_vec(&success).unwrap()) .map_err(|err| match err { OperationError::Operation(err) => { OperationError::Operation(serde_json::to_vec(&err).unwrap()) } OperationError::Internal(internal) => { OperationError::Internal(internal) } OperationError::Unhandled => OperationError::Unhandled, }) } .boxed() }), state: Box::new(state), } } async fn handle( &self, object: Box, operation: Box, operation_state: &StackOperationState, ) -> Result, OperationError>> { (self.func)(object, operation, &self.state, operation_state.clone()).await } } #[async_trait::async_trait] pub trait FromOperationState + Send + Sync>: Sized + Clone + Send { type Error: Serialize + DeserializeOwned; async fn from_state( object: &O, operation: &D, state: &StackOperationState, ) -> Result>; } #[async_trait::async_trait] impl + Send + Sync> FromOperationState for Arc { type Error = (); async fn from_state( _object: &O, _operation: &D, state: &StackOperationState, ) -> Result> { Ok(state.runtime.clone()) } } #[async_trait::async_trait] impl + Send + Sync> FromOperationState for StackOperationState { type Error = (); async fn from_state( _object: &O, _operation: &D, state: &StackOperationState, ) -> Result> { Ok(state.clone()) } } #[async_trait::async_trait] impl + Send + Sync> FromOperationState for AuthenticatedUser { type Error = (); async fn from_state( _object: &O, _operation: &D, state: &StackOperationState, ) -> Result> { state .user .clone() .ok_or_else(|| OperationError::Operation(())) } } #[async_trait::async_trait] impl + Send + Sync> FromOperationState for AuthenticatedInstance { type Error = (); async fn from_state( _object: &O, _operation: &D, state: &StackOperationState, ) -> Result> { state .instance .clone() .ok_or_else(|| OperationError::Operation(())) } } #[async_trait::async_trait] impl< T: FromOperationState + Send + Sync, O: GiteratedObject + Sync, D: GiteratedOperation + Send + Sync, > FromOperationState for Option { type Error = (); async fn from_state( object: &O, operation: &D, state: &StackOperationState, ) -> Result, OperationError<()>> { Ok(T::from_state(object, operation, state).await.ok()) } } #[derive(Clone)] pub struct AuthorizedUser(AuthenticatedUser); #[derive(Clone)] pub struct AuthorizedInstance(AuthenticatedInstance); #[async_trait::async_trait] pub trait AuthorizedOperation: GiteratedOperation { async fn authorize( &self, authorize_for: &O, state: &StackOperationState, ) -> Result>; } #[async_trait::async_trait] impl< O: GiteratedObject + Send + Sync + Debug, V: GiteratedObjectValue + Send + Sync, > AuthorizedOperation for GetValue { async fn authorize( &self, authorize_for: &O, operation_state: &StackOperationState, ) -> Result> { Ok(operation_state .runtime .get_object::(&authorize_for.to_string(), operation_state) .await .is_ok()) } } #[async_trait::async_trait] impl AuthorizedOperation for SetSetting { async fn authorize( &self, authorize_for: &User, operation_state: &StackOperationState, ) -> Result> { let authenticated_user = operation_state .user .as_ref() .ok_or_else(|| OperationError::Operation(()))?; Ok(authorize_for == authenticated_user.deref()) } } #[async_trait::async_trait] impl AuthorizedOperation for GetSetting { async fn authorize( &self, authorize_for: &User, operation_state: &StackOperationState, ) -> Result> { let authenticated_user = operation_state .user .as_ref() .ok_or_else(|| OperationError::Operation(()))?; Ok(authorize_for == authenticated_user.deref()) } } #[async_trait::async_trait] impl AuthorizedOperation for SetSetting { async fn authorize( &self, authorize_for: &Repository, operation_state: &StackOperationState, ) -> Result> { let authenticated_user = operation_state .user .as_ref() .ok_or_else(|| OperationError::Operation(()))?; let mut object = operation_state .runtime .get_object::(&authorize_for.to_string(), operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; let access_list = object .get_setting::(operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; if access_list .0 .iter() .find(|user| *user == authenticated_user.deref()) .is_some() { Ok(true) } else { Ok(false) } } } #[async_trait::async_trait] impl AuthorizedOperation for GetSetting { async fn authorize( &self, authorize_for: &Repository, operation_state: &StackOperationState, ) -> Result> { let authenticated_user = operation_state .user .as_ref() .ok_or_else(|| OperationError::Operation(()))?; let mut object = operation_state .runtime .get_object::(&authorize_for.to_string(), operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; let access_list = object .get_setting::(operation_state) .await .map_err(|e| OperationError::Internal(e.to_string()))?; if access_list .0 .iter() .find(|user| *user == authenticated_user.deref()) .is_some() { Ok(true) } else { Ok(false) } } } #[async_trait::async_trait] impl AuthorizedOperation for RegisterAccountRequest { async fn authorize( &self, authorize_for: &Instance, state: &StackOperationState, ) -> Result> { if state.our_instance == *authorize_for { Ok(true) } else { Ok(false) } } } #[async_trait::async_trait] impl AuthorizedOperation for AuthenticationTokenRequest { async fn authorize( &self, authorize_for: &Instance, state: &StackOperationState, ) -> Result> { if state.our_instance == *authorize_for { Ok(true) } else { Ok(false) } } } #[async_trait::async_trait] impl AuthorizedOperation for RepositoryCreateRequest { async fn authorize( &self, authorize_for: &Instance, state: &StackOperationState, ) -> Result> { if state.our_instance == *authorize_for { Ok(true) } else { Ok(false) } } } #[async_trait::async_trait] impl + Send + Sync> FromOperationState for AuthorizedUser { type Error = (); async fn from_state( object: &User, operation: &A, state: &StackOperationState, ) -> Result> { let authenticated = AuthenticatedUser::from_state(object, operation, state).await?; match operation.authorize(object, state).await { Ok(authorized) => { assert!(authorized); } Err(err) => return Err(OperationError::Internal(err.to_string())), }; Ok(AuthorizedUser(authenticated)) } } #[async_trait::async_trait] impl + Send + Sync> FromOperationState for AuthorizedInstance { type Error = (); async fn from_state( object: &Instance, operation: &A, state: &StackOperationState, ) -> Result> { let authenticated = AuthenticatedInstance::from_state(object, operation, state).await?; match operation.authorize(object, state).await { Ok(authorized) => { assert!(authorized); } Err(err) => return Err(OperationError::Internal(err.to_string())), }; Ok(AuthorizedInstance(authenticated)) } } // #[async_trait::async_trait> FromOperationState for Option { // type Error = (); // async fn from_state(state: &StackOperationState) -> Result, OperationError<()>> { // Ok(T::from_state(] // impl, pub instance: Option, pub user: Option, } #[derive(Clone, Debug)] pub struct AuthenticatedInstance(Instance); impl AuthenticatedInstance { pub fn new(instance: Instance) -> Self { AuthenticatedInstance(instance) } } impl Deref for AuthenticatedInstance { type Target = Instance; fn deref(&self) -> &Self::Target { &self.0 } } #[derive(Clone, Debug)] pub struct AuthenticatedUser(User); impl AuthenticatedUser { pub fn new(user: User) -> Self { AuthenticatedUser(user) } } impl Deref for AuthenticatedUser { type Target = User; fn deref(&self) -> &Self::Target { &self.0 } }