use std::collections::HashMap; use std::fmt::{Display, Formatter}; use std::str::FromStr; use anyhow::Error; use collab_entity::CollabObject; use serde::{Deserialize, Serialize}; use serde_json::Value; use uuid::Uuid; use flowy_error::{ErrorCode, FlowyError}; use lib_infra::box_any::BoxAny; use lib_infra::future::FutureResult; use crate::entities::{ AuthResponse, Role, UpdateUserProfileParams, UserCredentials, UserProfile, UserWorkspace, WorkspaceMember, }; #[derive(Debug, Clone, Serialize, Deserialize)] pub struct UserCloudConfig { pub enable_sync: bool, pub enable_encrypt: bool, // The secret used to encrypt the user's data pub encrypt_secret: String, } impl UserCloudConfig { pub fn new(encrypt_secret: String) -> Self { Self { enable_sync: true, enable_encrypt: false, encrypt_secret, } } pub fn with_enable_encrypt(mut self, enable_encrypt: bool) -> Self { self.enable_encrypt = enable_encrypt; // When the enable_encrypt is true, the encrypt_secret should not be empty debug_assert!(!self.encrypt_secret.is_empty()); self } } impl Display for UserCloudConfig { fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { write!( f, "enable_sync: {}, enable_encrypt: {}", self.enable_sync, self.enable_encrypt ) } } /// Provide the generic interface for the user cloud service /// The user cloud service is responsible for the user authentication and user profile management #[allow(unused_variables)] pub trait UserCloudService: Send + Sync + 'static { /// Sign up a new account. /// The type of the params is defined the this trait's implementation. /// Use the `unbox_or_error` of the [BoxAny] to get the params. fn sign_up(&self, params: BoxAny) -> FutureResult; /// Sign in an account /// The type of the params is defined the this trait's implementation. fn sign_in(&self, params: BoxAny) -> FutureResult; /// Sign out an account fn sign_out(&self, token: Option) -> FutureResult<(), FlowyError>; /// Generate a sign in url for the user with the given email /// Currently, only use the admin client for testing fn generate_sign_in_url_with_email(&self, email: &str) -> FutureResult; /// When the user opens the OAuth URL, it redirects to the corresponding provider's OAuth web page. /// After the user is authenticated, the browser will open a deep link to the AppFlowy app (iOS, macOS, etc.), /// which will call [Client::sign_in_with_url] to sign in. /// /// For example, the OAuth URL on Google looks like `https://appflowy.io/authorize?provider=google`. fn generate_oauth_url_with_provider(&self, provider: &str) -> FutureResult; /// Using the user's token to update the user information fn update_user( &self, credential: UserCredentials, params: UpdateUserProfileParams, ) -> FutureResult<(), FlowyError>; /// Get the user information using the user's token or uid /// return None if the user is not found fn get_user_profile(&self, credential: UserCredentials) -> FutureResult; fn open_workspace(&self, workspace_id: &str) -> FutureResult; /// Return the all the workspaces of the user fn get_all_workspace(&self, uid: i64) -> FutureResult, FlowyError>; fn add_workspace_member( &self, user_email: String, workspace_id: String, ) -> FutureResult<(), Error> { FutureResult::new(async { Ok(()) }) } fn remove_workspace_member( &self, user_email: String, workspace_id: String, ) -> FutureResult<(), Error> { FutureResult::new(async { Ok(()) }) } fn update_workspace_member( &self, user_email: String, workspace_id: String, role: Role, ) -> FutureResult<(), Error> { FutureResult::new(async { Ok(()) }) } fn get_workspace_members( &self, workspace_id: String, ) -> FutureResult, Error> { FutureResult::new(async { Ok(vec![]) }) } fn get_user_awareness_updates(&self, uid: i64) -> FutureResult>, Error>; fn receive_realtime_event(&self, _json: Value) {} fn subscribe_user_update(&self) -> Option { None } fn reset_workspace(&self, collab_object: CollabObject) -> FutureResult<(), Error>; fn create_collab_object( &self, collab_object: &CollabObject, data: Vec, override_if_exist: bool, ) -> FutureResult<(), FlowyError>; } pub type UserUpdateReceiver = tokio::sync::mpsc::Receiver; pub type UserUpdateSender = tokio::sync::mpsc::Sender; #[derive(Debug, Clone)] pub struct UserUpdate { pub uid: i64, pub name: Option, pub email: Option, pub encryption_sign: String, } pub fn uuid_from_map(map: &HashMap) -> Result { let uuid = map .get("uuid") .ok_or_else(|| FlowyError::new(ErrorCode::MissingAuthField, "Missing uuid field"))? .as_str(); let uuid = Uuid::from_str(uuid)?; Ok(uuid) } pub type UserTokenStateReceiver = tokio::sync::broadcast::Receiver; #[derive(Debug, Clone)] pub enum UserTokenState { Refresh, Invalid, }