use std::sync::{Arc, Weak}; use appflowy_integrate::RocksCollabDB; use collab_folder::core::FolderData; use serde::{Deserialize, Serialize}; use tokio::sync::RwLock; use uuid::Uuid; use flowy_error::{internal_error, ErrorCode, FlowyResult}; use flowy_server_config::supabase_config::SupabaseConfiguration; use flowy_sqlite::schema::{user_table, user_workspace_table}; use flowy_sqlite::ConnectionPool; use flowy_sqlite::{kv::KV, query_dsl::*, DBConnection, ExpressionMethods}; use flowy_user_deps::entities::*; use lib_infra::box_any::BoxAny; use lib_infra::util::timestamp; use crate::entities::{AuthTypePB, RepeatedUserWorkspacePB}; use crate::entities::{UserProfilePB, UserSettingPB}; use crate::event_map::{ DefaultUserStatusCallback, SignUpContext, UserCloudServiceProvider, UserStatusCallback, }; use crate::migrations::historical_document::HistoricalEmptyDocumentMigration; use crate::migrations::local_user_to_cloud::migration_user_to_cloud; use crate::migrations::migration::UserLocalDataMigration; use crate::migrations::UserMigrationContext; use crate::services::database::UserDB; use crate::services::session_serde::Session; use crate::services::user_sql::{UserTable, UserTableChangeset}; use crate::services::user_workspace_sql::UserWorkspaceTable; use crate::{errors::FlowyError, notification::*}; const HISTORICAL_USER: &str = "af_historical_users"; const SUPABASE_CONFIG_CACHE_KEY: &str = "af_supabase_config"; pub struct UserSessionConfig { root_dir: String, /// Used as the key of `Session` when saving session information to KV. session_cache_key: String, } impl UserSessionConfig { /// The `root_dir` represents as the root of the user folders. It must be unique for each /// users. pub fn new(name: &str, root_dir: &str) -> Self { let session_cache_key = format!("{}_session_cache", name); Self { root_dir: root_dir.to_owned(), session_cache_key, } } } pub struct UserSession { database: UserDB, session_config: UserSessionConfig, cloud_services: Arc, pub(crate) user_status_callback: RwLock>, } impl UserSession { pub fn new( session_config: UserSessionConfig, cloud_services: Arc, ) -> Self { let database = UserDB::new(&session_config.root_dir); let user_status_callback: RwLock> = RwLock::new(Arc::new(DefaultUserStatusCallback)); Self { database, session_config, cloud_services, user_status_callback, } } pub async fn init(&self, user_status_callback: C) { if let Ok(session) = self.get_session() { match ( self.database.get_collab_db(session.user_id), self.database.get_pool(session.user_id), ) { (Ok(collab_db), Ok(sqlite_pool)) => { match UserLocalDataMigration::new(session.clone(), collab_db, sqlite_pool) .run(vec![Box::new(HistoricalEmptyDocumentMigration)]) { Ok(applied_migrations) => { if applied_migrations.len() > 0 { tracing::info!("Did apply migrations: {:?}", applied_migrations); } }, Err(e) => tracing::error!("User data migration failed: {:?}", e), } }, _ => tracing::error!("Failed to get collab db or sqlite pool"), } if let Err(e) = user_status_callback .did_init(session.user_id, &session.user_workspace) .await { tracing::error!("Failed to call did_sign_in callback: {:?}", e); } } *self.user_status_callback.write().await = Arc::new(user_status_callback); } pub fn db_connection(&self, uid: i64) -> Result { self.database.get_connection(uid) } // The caller will be not 'Sync' before of the return value, // PooledConnection is not sync. You can use // db_connection_pool function to require the ConnectionPool that is 'Sync'. // // let pool = self.db_connection_pool()?; // let conn: PooledConnection = pool.get()?; pub fn db_pool(&self, uid: i64) -> Result, FlowyError> { self.database.get_pool(uid) } pub fn get_collab_db(&self, uid: i64) -> Result, FlowyError> { self .database .get_collab_db(uid) .map(|collab_db| Arc::downgrade(&collab_db)) } async fn migrate_local_user_to_cloud( &self, old_user: &UserMigrationContext, new_user: &UserMigrationContext, ) -> Result, FlowyError> { let old_collab_db = self.database.get_collab_db(old_user.session.user_id)?; let new_collab_db = self.database.get_collab_db(new_user.session.user_id)?; let folder_data = migration_user_to_cloud(old_user, &old_collab_db, new_user, &new_collab_db)?; Ok(folder_data) } #[tracing::instrument(level = "debug", skip(self, params))] pub async fn sign_in( &self, params: BoxAny, auth_type: AuthType, ) -> Result { let resp: SignInResponse = self .cloud_services .get_user_service()? .sign_in(params) .await?; let session: Session = resp.clone().into(); let uid = session.user_id; self.set_session(Some(session))?; self.log_user(uid, self.user_dir(uid)); let user_workspace = resp.latest_workspace.clone(); save_user_workspaces( self.db_pool(uid)?, resp .user_workspaces .iter() .map(|user_workspace| UserWorkspaceTable::from((uid, user_workspace))) .collect(), )?; let user_profile: UserProfile = self.save_user(uid, (resp, auth_type).into()).await?.into(); if let Err(e) = self .user_status_callback .read() .await .did_sign_in(user_profile.id, &user_workspace) .await { tracing::error!("Failed to call did_sign_in callback: {:?}", e); } send_sign_in_notification() .payload::(user_profile.clone().into()) .send(); Ok(user_profile) } pub async fn update_auth_type(&self, auth_type: &AuthType) { self .user_status_callback .read() .await .auth_type_did_changed(auth_type.clone()); self.cloud_services.set_auth_type(auth_type.clone()); } #[tracing::instrument(level = "debug", skip(self, params))] pub async fn sign_up( &self, auth_type: AuthType, params: BoxAny, ) -> Result { let old_user = { if let Ok(old_session) = self.get_session() { self .get_user_profile(old_session.user_id, false) .await .ok() .map(|user_profile| UserMigrationContext { user_profile, session: old_session, }) } else { None } }; let auth_service = self.cloud_services.get_user_service()?; let response: SignUpResponse = auth_service.sign_up(params).await?; let mut sign_up_context = SignUpContext { is_new: response.is_new, local_folder: None, }; let new_session = Session { user_id: response.user_id, user_workspace: response.latest_workspace.clone(), }; let uid = new_session.user_id; self.set_session(Some(new_session.clone()))?; self.log_user(uid, self.user_dir(uid)); save_user_workspaces( self.db_pool(uid)?, response .user_workspaces .iter() .map(|user_workspace| UserWorkspaceTable::from((uid, user_workspace))) .collect(), )?; let user_table = self .save_user(uid, (response, auth_type.clone()).into()) .await?; let new_user_profile: UserProfile = user_table.into(); // Only migrate the data if the user is login in as a guest and sign up as a new user if the current // auth type is not [AuthType::Local]. if sign_up_context.is_new { if let Some(old_user) = old_user { if old_user.user_profile.auth_type == AuthType::Local && !auth_type.is_local() { let new_user = UserMigrationContext { user_profile: new_user_profile.clone(), session: new_session.clone(), }; tracing::info!( "Migrate old user data from {:?} to {:?}", old_user.user_profile.id, new_user.user_profile.id ); match self.migrate_local_user_to_cloud(&old_user, &new_user).await { Ok(folder_data) => sign_up_context.local_folder = folder_data, Err(e) => tracing::error!("{:?}", e), } // close the old user db let _ = self.database.close(old_user.session.user_id); } } } let _ = self .user_status_callback .read() .await .did_sign_up( sign_up_context, &new_user_profile, &new_session.user_workspace, ) .await; Ok(new_user_profile) } #[tracing::instrument(level = "info", skip(self))] pub async fn sign_out(&self) -> Result<(), FlowyError> { let session = self.get_session()?; self.database.close(session.user_id)?; self.set_session(None)?; let server = self.cloud_services.get_user_service()?; tokio::spawn(async move { match server.sign_out(None).await { Ok(_) => {}, Err(e) => tracing::error!("Sign out failed: {:?}", e), } }); Ok(()) } #[tracing::instrument(level = "debug", skip(self))] pub async fn update_user_profile( &self, params: UpdateUserProfileParams, ) -> Result<(), FlowyError> { let auth_type = params.auth_type.clone(); let session = self.get_session()?; let changeset = UserTableChangeset::new(params.clone()); diesel_update_table!( user_table, changeset, &*self.db_connection(session.user_id)? ); let session = self.get_session()?; let user_profile = self.get_user_profile(session.user_id, false).await?; let profile_pb: UserProfilePB = user_profile.into(); send_notification( &session.user_id.to_string(), UserNotification::DidUpdateUserProfile, ) .payload(profile_pb) .send(); self .update_user(&auth_type, session.user_id, None, params) .await?; Ok(()) } pub async fn init_user(&self) -> Result<(), FlowyError> { Ok(()) } pub async fn check_user(&self) -> Result<(), FlowyError> { let user_id = self.get_session()?.user_id; let credential = UserCredentials::from_uid(user_id); let auth_service = self.cloud_services.get_user_service()?; auth_service.check_user(credential).await?; Ok(()) } pub async fn check_user_with_uuid(&self, uuid: &Uuid) -> Result<(), FlowyError> { let credential = UserCredentials::from_uuid(uuid.to_string()); let auth_service = self.cloud_services.get_user_service()?; auth_service.check_user(credential).await?; Ok(()) } pub async fn open_workspace(&self, workspace_id: &str) -> FlowyResult<()> { let uid = self.user_id()?; if let Some(user_workspace) = self.get_user_workspace(uid, workspace_id) { if let Err(err) = self .user_status_callback .read() .await .open_workspace(uid, &user_workspace) .await { tracing::error!("Open workspace failed: {:?}", err); } } Ok(()) } pub async fn add_user_to_workspace( &self, user_email: String, to_workspace_id: String, ) -> FlowyResult<()> { self .cloud_services .get_user_service()? .add_workspace_member(user_email, to_workspace_id) .await?; Ok(()) } pub async fn remove_user_to_workspace( &self, user_email: String, from_workspace_id: String, ) -> FlowyResult<()> { self .cloud_services .get_user_service()? .remove_workspace_member(user_email, from_workspace_id) .await?; Ok(()) } /// Get the user profile from the database /// If the refresh is true, it will try to get the user profile from the server pub async fn get_user_profile(&self, uid: i64, refresh: bool) -> Result { let user_id = uid.to_string(); let user = user_table::dsl::user_table .filter(user_table::id.eq(&user_id)) .first::(&*(self.db_connection(uid)?))?; if refresh { let weak_auth_service = Arc::downgrade(&self.cloud_services.get_user_service()?); let weak_pool = Arc::downgrade(&self.database.get_pool(uid)?); tokio::spawn(async move { if let (Some(auth_service), Some(pool)) = (weak_auth_service.upgrade(), weak_pool.upgrade()) { if let Ok(Some(user_profile)) = auth_service .get_user_profile(UserCredentials::from_uid(uid)) .await { let changeset = UserTableChangeset::from_user_profile(user_profile.clone()); if let Ok(conn) = pool.get() { let filter = user_table::dsl::user_table.filter(user_table::dsl::id.eq(changeset.id.clone())); let _ = diesel::update(filter).set(changeset).execute(&*conn); // Send notification to the client let user_profile_pb: UserProfilePB = user_profile.into(); send_notification(&uid.to_string(), UserNotification::DidUpdateUserProfile) .payload(user_profile_pb) .send(); } } } }); } Ok(user.into()) } pub fn user_dir(&self, uid: i64) -> String { format!("{}/{}", self.session_config.root_dir, uid) } pub fn user_setting(&self) -> Result { let session = self.get_session()?; let user_setting = UserSettingPB { user_folder: self.user_dir(session.user_id), }; Ok(user_setting) } pub fn user_id(&self) -> Result { Ok(self.get_session()?.user_id) } pub fn token(&self) -> Result, FlowyError> { Ok(None) } pub fn save_supabase_config(&self, config: SupabaseConfiguration) { self.cloud_services.update_supabase_config(&config); let _ = KV::set_object(SUPABASE_CONFIG_CACHE_KEY, config); } async fn update_user( &self, _auth_type: &AuthType, uid: i64, token: Option, params: UpdateUserProfileParams, ) -> Result<(), FlowyError> { let server = self.cloud_services.get_user_service()?; let token = token.to_owned(); tokio::spawn(async move { let credentials = UserCredentials::new(token, Some(uid), None); server.update_user(credentials, params).await }) .await .map_err(internal_error)??; Ok(()) } async fn save_user(&self, uid: i64, user: UserTable) -> Result { let conn = self.db_connection(uid)?; conn.immediate_transaction(|| { // delete old user if exists diesel::delete(user_table::dsl::user_table.filter(user_table::dsl::id.eq(&user.id))) .execute(&*conn)?; let _ = diesel::insert_into(user_table::table) .values(user.clone()) .execute(&*conn)?; Ok::<(), FlowyError>(()) })?; Ok(user) } pub fn get_user_workspace(&self, uid: i64, workspace_id: &str) -> Option { let conn = self.db_connection(uid).ok()?; let row = user_workspace_table::dsl::user_workspace_table .filter(user_workspace_table::id.eq(workspace_id)) .first::(&*conn) .ok()?; Some(UserWorkspace::from(row)) } pub fn get_all_user_workspaces(&self, uid: i64) -> FlowyResult> { let conn = self.db_connection(uid)?; let rows = user_workspace_table::dsl::user_workspace_table .filter(user_workspace_table::uid.eq(uid)) .load::(&*conn)?; if let Ok(service) = self.cloud_services.get_user_service() { if let Ok(pool) = self.db_pool(uid) { tokio::spawn(async move { if let Ok(new_user_workspaces) = service.get_user_workspaces(uid).await { let _ = save_user_workspaces( pool, new_user_workspaces .iter() .map(|user_workspace| UserWorkspaceTable::from((uid, user_workspace))) .collect(), ); let repeated_workspace_pbs = RepeatedUserWorkspacePB::from(new_user_workspaces); send_notification(&uid.to_string(), UserNotification::DidUpdateUserWorkspaces) .payload(repeated_workspace_pbs) .send(); } }); } } Ok(rows.into_iter().map(UserWorkspace::from).collect()) } pub async fn save_user_workspaces( &self, uid: i64, user_workspaces: Vec, ) -> FlowyResult<()> { let conn = self.db_connection(uid)?; conn.immediate_transaction(|| { for user_workspace in user_workspaces { if let Err(err) = diesel::update( user_workspace_table::dsl::user_workspace_table .filter(user_workspace_table::id.eq(user_workspace.id.clone())), ) .set(( user_workspace_table::name.eq(&user_workspace.name), user_workspace_table::created_at.eq(&user_workspace.created_at), user_workspace_table::database_storage_id.eq(&user_workspace.database_storage_id), )) .execute(&*conn) .and_then(|rows| { if rows == 0 { let _ = diesel::insert_into(user_workspace_table::table) .values(user_workspace) .execute(&*conn)?; } Ok(()) }) { tracing::error!("Error saving user workspace: {:?}", err); } } Ok::<(), FlowyError>(()) }) } fn set_session(&self, session: Option) -> Result<(), FlowyError> { tracing::debug!("Set user session: {:?}", session); match &session { None => KV::remove(&self.session_config.session_cache_key), Some(session) => { KV::set_object(&self.session_config.session_cache_key, session.clone()) .map_err(internal_error)?; }, } Ok(()) } fn log_user(&self, uid: i64, storage_path: String) { let mut logger_users = KV::get_object::(HISTORICAL_USER).unwrap_or_default(); logger_users.add_user(HistoricalUser { user_id: uid, sign_in_timestamp: timestamp(), storage_path, }); let _ = KV::set_object(HISTORICAL_USER, logger_users); } pub fn get_historical_users(&self) -> Vec { KV::get_object::(HISTORICAL_USER) .unwrap_or_default() .users } /// Returns the current user session. pub fn get_session(&self) -> Result { match KV::get_object::(&self.session_config.session_cache_key) { None => Err(FlowyError::new( ErrorCode::RecordNotFound, format!( "Can't find the value of {}, User is not logged in", self.session_config.session_cache_key ), )), Some(session) => Ok(session), } } } pub fn get_supabase_config() -> Option { KV::get_str(SUPABASE_CONFIG_CACHE_KEY) .and_then(|s| serde_json::from_str(&s).ok()) .unwrap_or_else(|| SupabaseConfiguration::from_env().ok()) } pub fn save_user_workspaces( pool: Arc, user_workspaces: Vec, ) -> FlowyResult<()> { let conn = pool.get()?; conn.immediate_transaction(|| { for user_workspace in user_workspaces { if let Err(err) = diesel::update( user_workspace_table::dsl::user_workspace_table .filter(user_workspace_table::id.eq(user_workspace.id.clone())), ) .set(( user_workspace_table::name.eq(&user_workspace.name), user_workspace_table::created_at.eq(&user_workspace.created_at), user_workspace_table::database_storage_id.eq(&user_workspace.database_storage_id), )) .execute(&*conn) .and_then(|rows| { if rows == 0 { let _ = diesel::insert_into(user_workspace_table::table) .values(user_workspace) .execute(&*conn)?; } Ok(()) }) { tracing::error!("Error saving user workspace: {:?}", err); } } Ok::<(), FlowyError>(()) }) } impl From for AuthType { fn from(pb: AuthTypePB) -> Self { match pb { AuthTypePB::Supabase => AuthType::Supabase, AuthTypePB::Local => AuthType::Local, AuthTypePB::SelfHosted => AuthType::SelfHosted, } } } impl From for AuthTypePB { fn from(auth_type: AuthType) -> Self { match auth_type { AuthType::Supabase => AuthTypePB::Supabase, AuthType::Local => AuthTypePB::Local, AuthType::SelfHosted => AuthTypePB::SelfHosted, } } } #[derive(Debug, Clone, Default, Serialize, Deserialize)] pub struct HistoricalUsers { pub(crate) users: Vec, } impl HistoricalUsers { pub fn add_user(&mut self, new_user: HistoricalUser) { self.users.retain(|user| user.user_id != new_user.user_id); self.users.push(new_user); } } #[derive(Debug, Clone, Default, Serialize, Deserialize)] pub struct HistoricalUser { pub user_id: i64, pub sign_in_timestamp: i64, pub storage_path: String, }