2023-11-20 12:54:47 +00:00
|
|
|
|
use std::fs;
|
2023-11-12 10:00:07 +00:00
|
|
|
|
use std::path::PathBuf;
|
2023-08-07 14:24:04 +00:00
|
|
|
|
use std::string::ToString;
|
2023-11-05 06:00:24 +00:00
|
|
|
|
use std::sync::atomic::{AtomicI64, Ordering};
|
2023-07-29 01:46:24 +00:00
|
|
|
|
use std::sync::{Arc, Weak};
|
2023-04-04 00:41:16 +00:00
|
|
|
|
|
2023-11-12 10:00:07 +00:00
|
|
|
|
use base64::alphabet::URL_SAFE;
|
|
|
|
|
use base64::engine::general_purpose::PAD;
|
|
|
|
|
use base64::engine::GeneralPurpose;
|
2023-08-14 04:57:59 +00:00
|
|
|
|
use collab_user::core::MutexUserAwareness;
|
2023-08-12 09:36:31 +00:00
|
|
|
|
use serde_json::Value;
|
2023-08-14 04:57:59 +00:00
|
|
|
|
use tokio::sync::{Mutex, RwLock};
|
2023-10-10 11:05:55 +00:00
|
|
|
|
use tokio_stream::StreamExt;
|
2023-10-23 03:43:31 +00:00
|
|
|
|
use tracing::{debug, error, event, info, instrument};
|
2023-05-15 14:16:05 +00:00
|
|
|
|
|
2023-09-17 09:14:34 +00:00
|
|
|
|
use collab_integrate::collab_builder::AppFlowyCollabBuilder;
|
|
|
|
|
use collab_integrate::RocksCollabDB;
|
2023-08-17 15:46:39 +00:00
|
|
|
|
use flowy_error::{internal_error, ErrorCode, FlowyResult};
|
2023-11-24 03:54:47 +00:00
|
|
|
|
use flowy_server_config::AuthenticatorType;
|
2023-08-06 03:51:03 +00:00
|
|
|
|
use flowy_sqlite::kv::StorePreferences;
|
2023-08-14 04:57:59 +00:00
|
|
|
|
use flowy_sqlite::schema::user_table;
|
2023-01-31 00:28:31 +00:00
|
|
|
|
use flowy_sqlite::ConnectionPool;
|
2023-08-06 03:51:03 +00:00
|
|
|
|
use flowy_sqlite::{query_dsl::*, DBConnection, ExpressionMethods};
|
2023-08-20 06:13:54 +00:00
|
|
|
|
use flowy_user_deps::cloud::UserUpdate;
|
2023-07-29 01:46:24 +00:00
|
|
|
|
use flowy_user_deps::entities::*;
|
2023-10-30 04:35:06 +00:00
|
|
|
|
use lib_dispatch::prelude::af_spawn;
|
2023-05-21 10:53:59 +00:00
|
|
|
|
use lib_infra::box_any::BoxAny;
|
2023-05-17 01:49:39 +00:00
|
|
|
|
|
2023-12-06 14:54:17 +00:00
|
|
|
|
use crate::anon_user_upgrade::{
|
|
|
|
|
migration_anon_user_on_sign_up, sync_af_user_data_to_cloud, sync_supabase_user_data_to_cloud,
|
|
|
|
|
};
|
2023-08-20 06:13:54 +00:00
|
|
|
|
use crate::entities::{AuthStateChangedPB, AuthStatePB, UserProfilePB, UserSettingPB};
|
2023-08-28 05:28:24 +00:00
|
|
|
|
use crate::event_map::{DefaultUserStatusCallback, UserCloudServiceProvider, UserStatusCallback};
|
2023-11-01 03:45:35 +00:00
|
|
|
|
use crate::migrations::document_empty_content::HistoricalEmptyDocumentMigration;
|
|
|
|
|
use crate::migrations::migration::{UserDataMigration, UserLocalDataMigration};
|
2023-12-21 06:13:21 +00:00
|
|
|
|
use crate::migrations::session_migration::migrate_session_with_user_uuid;
|
2023-11-01 03:45:35 +00:00
|
|
|
|
use crate::migrations::workspace_and_favorite_v1::FavoriteV1AndWorkspaceArrayMigration;
|
2023-08-17 15:46:39 +00:00
|
|
|
|
use crate::migrations::MigrationUser;
|
2023-08-18 14:32:51 +00:00
|
|
|
|
use crate::services::cloud_config::get_cloud_config;
|
2023-10-02 07:12:24 +00:00
|
|
|
|
use crate::services::collab_interact::{CollabInteract, DefaultCollabInteract};
|
2023-12-21 06:13:21 +00:00
|
|
|
|
use crate::services::db::{UserDB, UserDBPath};
|
2023-08-17 15:46:39 +00:00
|
|
|
|
use crate::services::entities::{ResumableSignUp, Session};
|
2023-08-14 04:57:59 +00:00
|
|
|
|
use crate::services::user_awareness::UserAwarenessDataSource;
|
2023-07-29 01:46:24 +00:00
|
|
|
|
use crate::services::user_sql::{UserTable, UserTableChangeset};
|
2023-08-14 04:57:59 +00:00
|
|
|
|
use crate::services::user_workspace::save_user_workspaces;
|
2023-07-29 01:46:24 +00:00
|
|
|
|
use crate::{errors::FlowyError, notification::*};
|
|
|
|
|
|
2023-11-12 10:00:07 +00:00
|
|
|
|
pub const URL_SAFE_ENGINE: GeneralPurpose = GeneralPurpose::new(&URL_SAFE, PAD);
|
2023-11-20 12:54:47 +00:00
|
|
|
|
pub struct UserConfig {
|
|
|
|
|
/// Used to store the user data
|
|
|
|
|
storage_path: String,
|
|
|
|
|
/// application_path is the path of the application binary. By default, the
|
|
|
|
|
/// storage_path is the same as the application_path. However, when the user
|
|
|
|
|
/// choose a custom path for the user data, the storage_path will be different from
|
|
|
|
|
/// the application_path.
|
|
|
|
|
application_path: String,
|
|
|
|
|
pub device_id: String,
|
2023-02-13 01:29:49 +00:00
|
|
|
|
/// Used as the key of `Session` when saving session information to KV.
|
2023-12-21 06:13:21 +00:00
|
|
|
|
pub(crate) session_cache_key: String,
|
2021-07-10 08:27:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-20 12:54:47 +00:00
|
|
|
|
impl UserConfig {
|
2023-02-13 01:29:49 +00:00
|
|
|
|
/// The `root_dir` represents as the root of the user folders. It must be unique for each
|
|
|
|
|
/// users.
|
2023-11-20 12:54:47 +00:00
|
|
|
|
pub fn new(name: &str, storage_path: &str, application_path: &str, device_id: &str) -> Self {
|
2023-02-13 01:29:49 +00:00
|
|
|
|
let session_cache_key = format!("{}_session_cache", name);
|
|
|
|
|
Self {
|
2023-11-20 12:54:47 +00:00
|
|
|
|
storage_path: storage_path.to_owned(),
|
|
|
|
|
application_path: application_path.to_owned(),
|
2023-02-13 01:29:49 +00:00
|
|
|
|
session_cache_key,
|
2023-11-20 12:54:47 +00:00
|
|
|
|
device_id: device_id.to_owned(),
|
2021-07-10 08:27:20 +00:00
|
|
|
|
}
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
2023-11-20 12:54:47 +00:00
|
|
|
|
|
|
|
|
|
/// Returns bool whether the user choose a custom path for the user data.
|
|
|
|
|
pub fn is_custom_storage_path(&self) -> bool {
|
|
|
|
|
self.storage_path != self.application_path
|
|
|
|
|
}
|
2021-07-10 08:27:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-08-14 04:57:59 +00:00
|
|
|
|
pub struct UserManager {
|
2023-11-05 06:00:24 +00:00
|
|
|
|
database: Arc<UserDB>,
|
2023-11-12 10:00:07 +00:00
|
|
|
|
user_paths: UserPaths,
|
2023-11-20 12:54:47 +00:00
|
|
|
|
pub(crate) user_config: UserConfig,
|
2023-08-14 04:57:59 +00:00
|
|
|
|
pub(crate) cloud_services: Arc<dyn UserCloudServiceProvider>,
|
|
|
|
|
pub(crate) store_preferences: Arc<StorePreferences>,
|
|
|
|
|
pub(crate) user_awareness: Arc<Mutex<Option<MutexUserAwareness>>>,
|
2023-07-29 01:46:24 +00:00
|
|
|
|
pub(crate) user_status_callback: RwLock<Arc<dyn UserStatusCallback>>,
|
2023-08-14 04:57:59 +00:00
|
|
|
|
pub(crate) collab_builder: Weak<AppFlowyCollabBuilder>,
|
2023-10-02 07:12:24 +00:00
|
|
|
|
pub(crate) collab_interact: RwLock<Arc<dyn CollabInteract>>,
|
2023-08-17 15:46:39 +00:00
|
|
|
|
resumable_sign_up: Mutex<Option<ResumableSignUp>>,
|
2023-11-05 06:00:24 +00:00
|
|
|
|
current_session: Arc<parking_lot::RwLock<Option<Session>>>,
|
|
|
|
|
refresh_user_profile_since: AtomicI64,
|
2021-07-10 08:27:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-08-14 04:57:59 +00:00
|
|
|
|
impl UserManager {
|
2023-05-21 10:53:59 +00:00
|
|
|
|
pub fn new(
|
2023-11-20 12:54:47 +00:00
|
|
|
|
user_config: UserConfig,
|
2023-05-21 10:53:59 +00:00
|
|
|
|
cloud_services: Arc<dyn UserCloudServiceProvider>,
|
2023-08-06 03:51:03 +00:00
|
|
|
|
store_preferences: Arc<StorePreferences>,
|
2023-08-14 04:57:59 +00:00
|
|
|
|
collab_builder: Weak<AppFlowyCollabBuilder>,
|
2023-08-20 06:13:54 +00:00
|
|
|
|
) -> Arc<Self> {
|
2023-11-20 12:54:47 +00:00
|
|
|
|
let user_paths = UserPaths::new(user_config.storage_path.clone());
|
2023-11-12 10:00:07 +00:00
|
|
|
|
let database = Arc::new(UserDB::new(user_paths.clone()));
|
2023-05-31 09:42:14 +00:00
|
|
|
|
let user_status_callback: RwLock<Arc<dyn UserStatusCallback>> =
|
|
|
|
|
RwLock::new(Arc::new(DefaultUserStatusCallback));
|
2023-12-21 06:13:21 +00:00
|
|
|
|
let current_session = Arc::new(parking_lot::RwLock::new(None));
|
2023-12-23 05:35:24 +00:00
|
|
|
|
migrate_session_with_user_uuid(&user_config, ¤t_session, &store_preferences);
|
2023-08-20 06:13:54 +00:00
|
|
|
|
|
2023-11-05 06:00:24 +00:00
|
|
|
|
let refresh_user_profile_since = AtomicI64::new(0);
|
2023-08-20 06:13:54 +00:00
|
|
|
|
let user_manager = Arc::new(Self {
|
2023-07-29 01:46:24 +00:00
|
|
|
|
database,
|
2023-11-12 10:00:07 +00:00
|
|
|
|
user_paths,
|
2023-11-20 12:54:47 +00:00
|
|
|
|
user_config,
|
2023-05-21 10:53:59 +00:00
|
|
|
|
cloud_services,
|
2023-08-06 03:51:03 +00:00
|
|
|
|
store_preferences,
|
2023-08-14 04:57:59 +00:00
|
|
|
|
user_awareness: Arc::new(Default::default()),
|
2023-02-13 01:29:49 +00:00
|
|
|
|
user_status_callback,
|
2023-08-14 04:57:59 +00:00
|
|
|
|
collab_builder,
|
2023-10-02 07:12:24 +00:00
|
|
|
|
collab_interact: RwLock::new(Arc::new(DefaultCollabInteract)),
|
2023-08-17 15:46:39 +00:00
|
|
|
|
resumable_sign_up: Default::default(),
|
2023-12-21 06:13:21 +00:00
|
|
|
|
current_session,
|
2023-11-05 06:00:24 +00:00
|
|
|
|
refresh_user_profile_since,
|
2023-08-20 06:13:54 +00:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
let weak_user_manager = Arc::downgrade(&user_manager);
|
|
|
|
|
if let Ok(user_service) = user_manager.cloud_services.get_user_service() {
|
|
|
|
|
if let Some(mut rx) = user_service.subscribe_user_update() {
|
2023-10-30 04:35:06 +00:00
|
|
|
|
af_spawn(async move {
|
2023-11-14 06:01:46 +00:00
|
|
|
|
while let Some(update) = rx.recv().await {
|
2023-08-20 06:13:54 +00:00
|
|
|
|
if let Some(user_manager) = weak_user_manager.upgrade() {
|
|
|
|
|
if let Err(err) = user_manager.handler_user_update(update).await {
|
2023-10-24 15:13:51 +00:00
|
|
|
|
error!("handler_user_update failed: {:?}", err);
|
2023-08-20 06:13:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
2021-08-31 15:01:46 +00:00
|
|
|
|
}
|
2023-08-20 06:13:54 +00:00
|
|
|
|
|
|
|
|
|
user_manager
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
2021-08-31 15:01:46 +00:00
|
|
|
|
|
2023-08-06 03:51:03 +00:00
|
|
|
|
pub fn get_store_preferences(&self) -> Weak<StorePreferences> {
|
|
|
|
|
Arc::downgrade(&self.store_preferences)
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-10 11:05:55 +00:00
|
|
|
|
/// Initializes the user session, including data migrations and user awareness configuration. This function
|
|
|
|
|
/// will be invoked each time the user opens the application.
|
2023-08-14 04:57:59 +00:00
|
|
|
|
///
|
2023-10-10 11:05:55 +00:00
|
|
|
|
/// Starts by retrieving the current session. If the session is successfully obtained, it will attempt
|
|
|
|
|
/// a local data migration for the user. After ensuring the user's data is migrated and up-to-date,
|
2023-08-14 04:57:59 +00:00
|
|
|
|
/// the function will set up the collaboration configuration and initialize the user's awareness. Upon successful
|
|
|
|
|
/// completion, a user status callback is invoked to signify that the initialization process is complete.
|
2023-11-05 06:00:24 +00:00
|
|
|
|
#[instrument(level = "debug", skip_all, err)]
|
2023-10-02 07:12:24 +00:00
|
|
|
|
pub async fn init<C: UserStatusCallback + 'static, I: CollabInteract>(
|
|
|
|
|
&self,
|
|
|
|
|
user_status_callback: C,
|
|
|
|
|
collab_interact: I,
|
2023-10-10 11:05:55 +00:00
|
|
|
|
) -> Result<(), FlowyError> {
|
2023-11-05 06:00:24 +00:00
|
|
|
|
let user_status_callback = Arc::new(user_status_callback);
|
|
|
|
|
*self.user_status_callback.write().await = user_status_callback.clone();
|
|
|
|
|
*self.collab_interact.write().await = Arc::new(collab_interact);
|
|
|
|
|
|
2023-02-13 01:29:49 +00:00
|
|
|
|
if let Ok(session) = self.get_session() {
|
2023-11-24 03:54:47 +00:00
|
|
|
|
let user = self.get_user_profile_from_disk(session.user_id).await?;
|
|
|
|
|
|
|
|
|
|
// Get the current authenticator from the environment variable
|
2023-12-21 06:13:21 +00:00
|
|
|
|
let current_authenticator = current_authenticator();
|
2023-11-24 03:54:47 +00:00
|
|
|
|
|
|
|
|
|
// If the current authenticator is different from the authenticator in the session and it's
|
|
|
|
|
// not a local authenticator, we need to sign out the user.
|
|
|
|
|
if user.authenticator != Authenticator::Local && user.authenticator != current_authenticator {
|
|
|
|
|
event!(
|
|
|
|
|
tracing::Level::INFO,
|
|
|
|
|
"Authenticator changed from {:?} to {:?}",
|
|
|
|
|
user.authenticator,
|
|
|
|
|
current_authenticator
|
|
|
|
|
);
|
|
|
|
|
self.sign_out().await?;
|
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
2023-11-05 06:00:24 +00:00
|
|
|
|
|
|
|
|
|
event!(
|
|
|
|
|
tracing::Level::INFO,
|
|
|
|
|
"init user session: {}:{}",
|
|
|
|
|
user.uid,
|
|
|
|
|
user.email
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Set the token if the current cloud service using token to authenticate
|
|
|
|
|
// Currently, only the AppFlowy cloud using token to init the client api.
|
2023-10-10 11:05:55 +00:00
|
|
|
|
if let Err(err) = self.cloud_services.set_token(&user.token) {
|
|
|
|
|
error!("Set token failed: {}", err);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Subscribe the token state
|
|
|
|
|
let weak_pool = Arc::downgrade(&self.db_pool(user.uid)?);
|
|
|
|
|
if let Some(mut token_state_rx) = self.cloud_services.subscribe_token_state() {
|
2023-11-05 06:00:24 +00:00
|
|
|
|
event!(tracing::Level::DEBUG, "Listen token state change");
|
2023-10-30 04:35:06 +00:00
|
|
|
|
af_spawn(async move {
|
2023-10-10 11:05:55 +00:00
|
|
|
|
while let Some(token_state) = token_state_rx.next().await {
|
2023-11-05 06:00:24 +00:00
|
|
|
|
debug!("Token state changed: {:?}", token_state);
|
2023-10-10 11:05:55 +00:00
|
|
|
|
match token_state {
|
|
|
|
|
UserTokenState::Refresh { token } => {
|
2023-11-05 06:00:24 +00:00
|
|
|
|
// Only save the token if the token is different from the current token
|
2023-10-10 11:05:55 +00:00
|
|
|
|
if token != user.token {
|
|
|
|
|
if let Some(pool) = weak_pool.upgrade() {
|
|
|
|
|
// Save the new token
|
|
|
|
|
if let Err(err) = save_user_token(user.uid, pool, token) {
|
|
|
|
|
error!("Save user token failed: {}", err);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
2023-11-05 06:00:24 +00:00
|
|
|
|
UserTokenState::Invalid => {},
|
2023-10-10 11:05:55 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
2023-11-20 12:54:47 +00:00
|
|
|
|
self.prepare_user(&session).await;
|
2023-10-10 11:05:55 +00:00
|
|
|
|
|
2023-08-14 04:57:59 +00:00
|
|
|
|
// Do the user data migration if needed
|
2023-11-05 06:00:24 +00:00
|
|
|
|
event!(tracing::Level::INFO, "Prepare user data migration");
|
2023-08-03 01:14:52 +00:00
|
|
|
|
match (
|
2023-11-24 03:54:47 +00:00
|
|
|
|
self.database.get_collab_db(session.user_id),
|
2023-08-03 01:14:52 +00:00
|
|
|
|
self.database.get_pool(session.user_id),
|
|
|
|
|
) {
|
|
|
|
|
(Ok(collab_db), Ok(sqlite_pool)) => {
|
2023-11-01 03:45:35 +00:00
|
|
|
|
// ⚠️The order of migrations is crucial. If you're adding a new migration, please ensure
|
|
|
|
|
// it's appended to the end of the list.
|
|
|
|
|
let migrations: Vec<Box<dyn UserDataMigration>> = vec![
|
|
|
|
|
Box::new(HistoricalEmptyDocumentMigration),
|
|
|
|
|
Box::new(FavoriteV1AndWorkspaceArrayMigration),
|
|
|
|
|
];
|
2023-12-06 17:14:02 +00:00
|
|
|
|
match UserLocalDataMigration::new(session.clone(), collab_db, sqlite_pool)
|
|
|
|
|
.run(migrations, ¤t_authenticator)
|
2023-08-03 01:14:52 +00:00
|
|
|
|
{
|
|
|
|
|
Ok(applied_migrations) => {
|
2023-08-04 11:27:14 +00:00
|
|
|
|
if !applied_migrations.is_empty() {
|
2023-10-07 01:58:44 +00:00
|
|
|
|
info!("Did apply migrations: {:?}", applied_migrations);
|
2023-08-03 01:14:52 +00:00
|
|
|
|
}
|
|
|
|
|
},
|
2023-10-24 15:13:51 +00:00
|
|
|
|
Err(e) => error!("User data migration failed: {:?}", e),
|
2023-08-03 01:14:52 +00:00
|
|
|
|
}
|
|
|
|
|
},
|
2023-10-24 15:13:51 +00:00
|
|
|
|
_ => error!("Failed to get collab db or sqlite pool"),
|
2023-08-03 01:14:52 +00:00
|
|
|
|
}
|
2023-08-14 04:57:59 +00:00
|
|
|
|
// Init the user awareness
|
|
|
|
|
self
|
|
|
|
|
.initialize_user_awareness(&session, UserAwarenessDataSource::Local)
|
|
|
|
|
.await;
|
2023-08-20 06:13:54 +00:00
|
|
|
|
|
2023-08-18 14:32:51 +00:00
|
|
|
|
let cloud_config = get_cloud_config(session.user_id, &self.store_preferences);
|
2023-07-14 05:37:13 +00:00
|
|
|
|
if let Err(e) = user_status_callback
|
2023-08-18 14:32:51 +00:00
|
|
|
|
.did_init(
|
|
|
|
|
session.user_id,
|
|
|
|
|
&cloud_config,
|
|
|
|
|
&session.user_workspace,
|
2023-11-20 12:54:47 +00:00
|
|
|
|
&self.user_config.device_id,
|
2023-08-18 14:32:51 +00:00
|
|
|
|
)
|
2023-07-14 05:37:13 +00:00
|
|
|
|
.await
|
|
|
|
|
{
|
2023-10-24 15:13:51 +00:00
|
|
|
|
error!("Failed to call did_init callback: {:?}", e);
|
2023-07-14 05:37:13 +00:00
|
|
|
|
}
|
2021-07-11 07:33:19 +00:00
|
|
|
|
}
|
2023-10-10 11:05:55 +00:00
|
|
|
|
Ok(())
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-14 05:37:13 +00:00
|
|
|
|
pub fn db_connection(&self, uid: i64) -> Result<DBConnection, FlowyError> {
|
|
|
|
|
self.database.get_connection(uid)
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-14 05:37:13 +00:00
|
|
|
|
pub fn db_pool(&self, uid: i64) -> Result<Arc<ConnectionPool>, FlowyError> {
|
|
|
|
|
self.database.get_pool(uid)
|
2023-04-04 00:41:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-29 01:46:24 +00:00
|
|
|
|
pub fn get_collab_db(&self, uid: i64) -> Result<Weak<RocksCollabDB>, FlowyError> {
|
|
|
|
|
self
|
|
|
|
|
.database
|
2023-11-24 03:54:47 +00:00
|
|
|
|
.get_collab_db(uid)
|
2023-07-29 01:46:24 +00:00
|
|
|
|
.map(|collab_db| Arc::downgrade(&collab_db))
|
2023-07-14 05:37:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-08-14 04:57:59 +00:00
|
|
|
|
/// Performs a user sign-in, initializing user awareness and sending relevant notifications.
|
|
|
|
|
///
|
|
|
|
|
/// This asynchronous function interacts with an external user service to authenticate and sign in a user
|
|
|
|
|
/// based on provided parameters. Once signed in, it updates the collaboration configuration, logs the user,
|
|
|
|
|
/// saves their workspaces, and initializes their user awareness.
|
|
|
|
|
///
|
|
|
|
|
/// A sign-in notification is also sent after a successful sign-in.
|
|
|
|
|
///
|
2023-05-21 10:53:59 +00:00
|
|
|
|
#[tracing::instrument(level = "debug", skip(self, params))]
|
|
|
|
|
pub async fn sign_in(
|
|
|
|
|
&self,
|
2023-11-28 02:54:31 +00:00
|
|
|
|
params: SignInParams,
|
2023-11-12 10:00:07 +00:00
|
|
|
|
authenticator: Authenticator,
|
2023-05-21 10:53:59 +00:00
|
|
|
|
) -> Result<UserProfile, FlowyError> {
|
2023-11-12 10:00:07 +00:00
|
|
|
|
self.update_authenticator(&authenticator).await;
|
2023-10-02 09:22:22 +00:00
|
|
|
|
let response: AuthResponse = self
|
2023-05-21 10:53:59 +00:00
|
|
|
|
.cloud_services
|
2023-07-29 01:46:24 +00:00
|
|
|
|
.get_user_service()?
|
2023-11-28 02:54:31 +00:00
|
|
|
|
.sign_in(BoxAny::new(params))
|
2023-05-21 10:53:59 +00:00
|
|
|
|
.await?;
|
2023-08-17 15:46:39 +00:00
|
|
|
|
let session = Session::from(&response);
|
2023-11-20 12:54:47 +00:00
|
|
|
|
self.prepare_user(&session).await;
|
2023-08-17 15:46:39 +00:00
|
|
|
|
|
|
|
|
|
let latest_workspace = response.latest_workspace.clone();
|
2023-11-12 10:00:07 +00:00
|
|
|
|
let user_profile = UserProfile::from((&response, &authenticator));
|
|
|
|
|
self
|
|
|
|
|
.save_auth_data(&response, &authenticator, &session)
|
|
|
|
|
.await?;
|
2023-11-20 12:54:47 +00:00
|
|
|
|
|
2023-08-14 04:57:59 +00:00
|
|
|
|
let _ = self
|
|
|
|
|
.initialize_user_awareness(&session, UserAwarenessDataSource::Remote)
|
|
|
|
|
.await;
|
|
|
|
|
|
2023-07-14 05:37:13 +00:00
|
|
|
|
if let Err(e) = self
|
2023-05-21 10:53:59 +00:00
|
|
|
|
.user_status_callback
|
|
|
|
|
.read()
|
|
|
|
|
.await
|
2023-11-20 12:54:47 +00:00
|
|
|
|
.did_sign_in(
|
|
|
|
|
user_profile.uid,
|
|
|
|
|
&latest_workspace,
|
|
|
|
|
&self.user_config.device_id,
|
|
|
|
|
)
|
2023-07-14 05:37:13 +00:00
|
|
|
|
.await
|
|
|
|
|
{
|
2023-10-24 15:13:51 +00:00
|
|
|
|
error!("Failed to call did_sign_in callback: {:?}", e);
|
2023-07-14 05:37:13 +00:00
|
|
|
|
}
|
2023-08-20 06:13:54 +00:00
|
|
|
|
send_auth_state_notification(AuthStateChangedPB {
|
|
|
|
|
state: AuthStatePB::AuthStateSignIn,
|
2023-10-24 12:11:06 +00:00
|
|
|
|
message: "Sign in success".to_string(),
|
2023-08-20 06:13:54 +00:00
|
|
|
|
})
|
|
|
|
|
.send();
|
2023-05-21 10:53:59 +00:00
|
|
|
|
Ok(user_profile)
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-12 10:00:07 +00:00
|
|
|
|
pub(crate) async fn update_authenticator(&self, authenticator: &Authenticator) {
|
2023-05-31 09:42:14 +00:00
|
|
|
|
self
|
|
|
|
|
.user_status_callback
|
|
|
|
|
.read()
|
|
|
|
|
.await
|
2023-11-12 10:00:07 +00:00
|
|
|
|
.authenticator_did_changed(authenticator.clone());
|
|
|
|
|
self.cloud_services.set_authenticator(authenticator.clone());
|
2023-07-14 05:37:13 +00:00
|
|
|
|
}
|
2023-05-21 10:53:59 +00:00
|
|
|
|
|
2023-08-14 04:57:59 +00:00
|
|
|
|
/// Manages the user sign-up process, potentially migrating data if necessary.
|
|
|
|
|
///
|
|
|
|
|
/// This asynchronous function interacts with an external authentication service to register and sign up a user
|
|
|
|
|
/// based on the provided parameters. Following a successful sign-up, it handles configuration updates, logging,
|
|
|
|
|
/// and saving workspace information. If a user is signing up with a new profile and previously had guest data,
|
|
|
|
|
/// this function may migrate that data over to the new account.
|
|
|
|
|
///
|
2023-08-17 15:46:39 +00:00
|
|
|
|
#[tracing::instrument(level = "info", skip(self, params))]
|
2023-07-14 05:37:13 +00:00
|
|
|
|
pub async fn sign_up(
|
|
|
|
|
&self,
|
2023-11-12 10:00:07 +00:00
|
|
|
|
authenticator: Authenticator,
|
2023-07-14 05:37:13 +00:00
|
|
|
|
params: BoxAny,
|
|
|
|
|
) -> Result<UserProfile, FlowyError> {
|
2023-11-29 20:55:13 +00:00
|
|
|
|
// sign out the current user if there is one
|
2023-12-06 14:54:17 +00:00
|
|
|
|
let migration_user = self.get_migration_user(&authenticator).await;
|
2023-11-29 20:55:13 +00:00
|
|
|
|
let _ = self.sign_out().await;
|
2023-07-14 05:37:13 +00:00
|
|
|
|
|
2023-11-29 20:55:13 +00:00
|
|
|
|
self.update_authenticator(&authenticator).await;
|
2023-07-29 01:46:24 +00:00
|
|
|
|
let auth_service = self.cloud_services.get_user_service()?;
|
2023-10-02 09:22:22 +00:00
|
|
|
|
let response: AuthResponse = auth_service.sign_up(params).await?;
|
2023-12-06 14:54:17 +00:00
|
|
|
|
let new_user_profile = UserProfile::from((&response, &authenticator));
|
|
|
|
|
if new_user_profile.encryption_type.require_encrypt_secret() {
|
2023-08-17 15:46:39 +00:00
|
|
|
|
self
|
|
|
|
|
.resumable_sign_up
|
|
|
|
|
.lock()
|
|
|
|
|
.await
|
|
|
|
|
.replace(ResumableSignUp {
|
2023-12-06 14:54:17 +00:00
|
|
|
|
user_profile: new_user_profile.clone(),
|
2023-08-17 15:46:39 +00:00
|
|
|
|
migration_user,
|
|
|
|
|
response,
|
2023-11-12 10:00:07 +00:00
|
|
|
|
authenticator,
|
2023-08-17 15:46:39 +00:00
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
self
|
2023-12-06 14:54:17 +00:00
|
|
|
|
.continue_sign_up(&new_user_profile, migration_user, response, &authenticator)
|
2023-08-17 15:46:39 +00:00
|
|
|
|
.await?;
|
|
|
|
|
}
|
2023-12-06 14:54:17 +00:00
|
|
|
|
Ok(new_user_profile)
|
2023-08-17 15:46:39 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[tracing::instrument(level = "info", skip(self))]
|
|
|
|
|
pub async fn resume_sign_up(&self) -> Result<(), FlowyError> {
|
|
|
|
|
let ResumableSignUp {
|
|
|
|
|
user_profile,
|
|
|
|
|
migration_user,
|
|
|
|
|
response,
|
2023-11-12 10:00:07 +00:00
|
|
|
|
authenticator,
|
2023-08-17 15:46:39 +00:00
|
|
|
|
} = self
|
|
|
|
|
.resumable_sign_up
|
|
|
|
|
.lock()
|
|
|
|
|
.await
|
|
|
|
|
.clone()
|
|
|
|
|
.ok_or(FlowyError::new(
|
|
|
|
|
ErrorCode::Internal,
|
|
|
|
|
"No resumable sign up data",
|
|
|
|
|
))?;
|
|
|
|
|
self
|
2023-11-12 10:00:07 +00:00
|
|
|
|
.continue_sign_up(&user_profile, migration_user, response, &authenticator)
|
2023-08-17 15:46:39 +00:00
|
|
|
|
.await?;
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[tracing::instrument(level = "info", skip_all, err)]
|
|
|
|
|
async fn continue_sign_up(
|
|
|
|
|
&self,
|
2023-12-06 14:54:17 +00:00
|
|
|
|
new_user_profile: &UserProfile,
|
2023-08-17 15:46:39 +00:00
|
|
|
|
migration_user: Option<MigrationUser>,
|
2023-10-02 09:22:22 +00:00
|
|
|
|
response: AuthResponse,
|
2023-11-12 10:00:07 +00:00
|
|
|
|
authenticator: &Authenticator,
|
2023-08-17 15:46:39 +00:00
|
|
|
|
) -> FlowyResult<()> {
|
2023-08-07 14:24:04 +00:00
|
|
|
|
let new_session = Session::from(&response);
|
2023-11-20 12:54:47 +00:00
|
|
|
|
self.prepare_user(&new_session).await;
|
2023-08-17 15:46:39 +00:00
|
|
|
|
|
|
|
|
|
let user_awareness_source = if response.is_new_user {
|
2023-08-14 04:57:59 +00:00
|
|
|
|
UserAwarenessDataSource::Local
|
|
|
|
|
} else {
|
|
|
|
|
UserAwarenessDataSource::Remote
|
|
|
|
|
};
|
2023-08-28 05:28:24 +00:00
|
|
|
|
|
2023-08-17 15:46:39 +00:00
|
|
|
|
if response.is_new_user {
|
|
|
|
|
if let Some(old_user) = migration_user {
|
|
|
|
|
let new_user = MigrationUser {
|
2023-12-06 14:54:17 +00:00
|
|
|
|
user_profile: new_user_profile.clone(),
|
2023-08-17 15:46:39 +00:00
|
|
|
|
session: new_session.clone(),
|
|
|
|
|
};
|
2023-10-23 03:43:31 +00:00
|
|
|
|
event!(
|
|
|
|
|
tracing::Level::INFO,
|
2023-11-01 03:45:35 +00:00
|
|
|
|
"Migrate anon user data from {:?} to {:?}",
|
2023-10-23 03:43:31 +00:00
|
|
|
|
old_user.user_profile.uid,
|
|
|
|
|
new_user.user_profile.uid
|
2023-08-17 15:46:39 +00:00
|
|
|
|
);
|
2023-08-28 05:28:24 +00:00
|
|
|
|
self
|
2023-12-06 14:54:17 +00:00
|
|
|
|
.migrate_anon_user_data_to_cloud(&old_user, &new_user, authenticator)
|
2023-08-28 05:28:24 +00:00
|
|
|
|
.await?;
|
2023-08-17 15:46:39 +00:00
|
|
|
|
let _ = self.database.close(old_user.session.user_id);
|
2023-07-14 05:37:13 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-08-14 04:57:59 +00:00
|
|
|
|
self
|
|
|
|
|
.initialize_user_awareness(&new_session, user_awareness_source)
|
|
|
|
|
.await;
|
|
|
|
|
|
2023-08-28 05:28:24 +00:00
|
|
|
|
self
|
2023-11-12 10:00:07 +00:00
|
|
|
|
.save_auth_data(&response, authenticator, &new_session)
|
2023-08-28 05:28:24 +00:00
|
|
|
|
.await?;
|
2023-11-05 06:00:24 +00:00
|
|
|
|
|
2023-08-17 15:46:39 +00:00
|
|
|
|
self
|
2023-05-21 10:53:59 +00:00
|
|
|
|
.user_status_callback
|
|
|
|
|
.read()
|
|
|
|
|
.await
|
2023-07-29 01:46:24 +00:00
|
|
|
|
.did_sign_up(
|
2023-08-28 05:28:24 +00:00
|
|
|
|
response.is_new_user,
|
2023-12-06 14:54:17 +00:00
|
|
|
|
new_user_profile,
|
2023-07-29 01:46:24 +00:00
|
|
|
|
&new_session.user_workspace,
|
2023-11-20 12:54:47 +00:00
|
|
|
|
&self.user_config.device_id,
|
2023-07-29 01:46:24 +00:00
|
|
|
|
)
|
2023-08-17 15:46:39 +00:00
|
|
|
|
.await?;
|
2023-08-20 06:13:54 +00:00
|
|
|
|
|
|
|
|
|
send_auth_state_notification(AuthStateChangedPB {
|
|
|
|
|
state: AuthStatePB::AuthStateSignIn,
|
2023-10-24 12:11:06 +00:00
|
|
|
|
message: "Sign up success".to_string(),
|
2023-08-20 06:13:54 +00:00
|
|
|
|
})
|
|
|
|
|
.send();
|
2023-08-17 15:46:39 +00:00
|
|
|
|
Ok(())
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-29 01:46:24 +00:00
|
|
|
|
#[tracing::instrument(level = "info", skip(self))]
|
2023-07-14 05:37:13 +00:00
|
|
|
|
pub async fn sign_out(&self) -> Result<(), FlowyError> {
|
2023-11-29 20:55:13 +00:00
|
|
|
|
if let Ok(session) = self.get_session() {
|
|
|
|
|
self.database.close(session.user_id)?;
|
|
|
|
|
self.set_session(None)?;
|
2023-05-23 15:55:21 +00:00
|
|
|
|
|
2023-11-29 20:55:13 +00:00
|
|
|
|
let server = self.cloud_services.get_user_service()?;
|
2023-10-24 15:13:51 +00:00
|
|
|
|
if let Err(err) = server.sign_out(None).await {
|
|
|
|
|
event!(tracing::Level::ERROR, "{:?}", err);
|
2023-05-21 10:53:59 +00:00
|
|
|
|
}
|
2023-11-29 20:55:13 +00:00
|
|
|
|
}
|
2023-02-13 01:29:49 +00:00
|
|
|
|
Ok(())
|
|
|
|
|
}
|
2021-09-01 08:08:32 +00:00
|
|
|
|
|
2023-08-14 04:57:59 +00:00
|
|
|
|
/// Updates the user's profile with the given parameters.
|
|
|
|
|
///
|
|
|
|
|
/// This function modifies the user's profile based on the provided update parameters. After updating, it
|
|
|
|
|
/// sends a notification about the change. It's also responsible for handling interactions with the underlying
|
|
|
|
|
/// database and updates user profile.
|
|
|
|
|
///
|
2023-02-13 01:29:49 +00:00
|
|
|
|
#[tracing::instrument(level = "debug", skip(self))]
|
|
|
|
|
pub async fn update_user_profile(
|
|
|
|
|
&self,
|
|
|
|
|
params: UpdateUserProfileParams,
|
|
|
|
|
) -> Result<(), FlowyError> {
|
|
|
|
|
let changeset = UserTableChangeset::new(params.clone());
|
2023-07-14 05:37:13 +00:00
|
|
|
|
let session = self.get_session()?;
|
2023-10-24 12:11:06 +00:00
|
|
|
|
upsert_user_profile_change(session.user_id, self.db_pool(session.user_id)?, changeset)?;
|
|
|
|
|
|
2023-11-24 03:54:47 +00:00
|
|
|
|
let profile = self.get_user_profile_from_disk(session.user_id).await?;
|
2023-10-24 12:11:06 +00:00
|
|
|
|
self
|
|
|
|
|
.update_user(session.user_id, profile.token, params)
|
|
|
|
|
.await?;
|
2023-02-13 01:29:49 +00:00
|
|
|
|
Ok(())
|
|
|
|
|
}
|
2021-12-09 14:28:11 +00:00
|
|
|
|
|
2023-02-13 01:29:49 +00:00
|
|
|
|
pub async fn init_user(&self) -> Result<(), FlowyError> {
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-20 12:54:47 +00:00
|
|
|
|
pub async fn prepare_user(&self, session: &Session) {
|
2023-12-21 00:12:40 +00:00
|
|
|
|
let _ = self.database.close(session.user_id);
|
2023-11-20 12:54:47 +00:00
|
|
|
|
self.set_collab_config(session);
|
|
|
|
|
// Ensure to backup user data if a cloud drive is used for storage. While using a cloud drive
|
|
|
|
|
// for storing user data is not advised due to potential data corruption risks, in scenarios where
|
|
|
|
|
// users opt for cloud storage, the application should automatically create a backup of the user
|
|
|
|
|
// data. This backup should be in the form of a zip file and stored locally on the user's disk
|
|
|
|
|
// for safety and data integrity purposes
|
|
|
|
|
if self.user_config.is_custom_storage_path() {
|
|
|
|
|
self
|
|
|
|
|
.database
|
|
|
|
|
.backup_or_restore(session.user_id, &session.user_workspace.id);
|
|
|
|
|
} else {
|
|
|
|
|
self
|
|
|
|
|
.database
|
|
|
|
|
.restore_if_need(session.user_id, &session.user_workspace.id);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-14 04:57:59 +00:00
|
|
|
|
/// Fetches the user profile for the given user ID.
|
2023-11-24 03:54:47 +00:00
|
|
|
|
pub async fn get_user_profile_from_disk(&self, uid: i64) -> Result<UserProfile, FlowyError> {
|
2023-08-20 06:13:54 +00:00
|
|
|
|
let user: UserProfile = user_table::dsl::user_table
|
|
|
|
|
.filter(user_table::id.eq(&uid.to_string()))
|
2023-12-23 15:27:15 +00:00
|
|
|
|
.first::<UserTable>(&mut *(self.db_connection(uid)?))
|
2023-11-05 06:00:24 +00:00
|
|
|
|
.map_err(|err| {
|
|
|
|
|
FlowyError::record_not_found().with_context(format!(
|
|
|
|
|
"Can't find the user profile for user id: {}, error: {:?}",
|
|
|
|
|
uid, err
|
|
|
|
|
))
|
|
|
|
|
})?
|
2023-08-20 06:13:54 +00:00
|
|
|
|
.into();
|
|
|
|
|
|
|
|
|
|
Ok(user)
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-05 06:00:24 +00:00
|
|
|
|
#[tracing::instrument(level = "info", skip_all, err)]
|
2023-10-24 15:13:51 +00:00
|
|
|
|
pub async fn refresh_user_profile(&self, old_user_profile: &UserProfile) -> FlowyResult<()> {
|
2023-11-24 03:54:47 +00:00
|
|
|
|
// If the user is a local user, no need to refresh the user profile
|
|
|
|
|
if old_user_profile.authenticator.is_local() {
|
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
2023-11-05 06:00:24 +00:00
|
|
|
|
|
2023-11-24 03:54:47 +00:00
|
|
|
|
let now = chrono::Utc::now().timestamp();
|
2023-11-05 06:00:24 +00:00
|
|
|
|
// Add debounce to avoid too many requests
|
|
|
|
|
if now - self.refresh_user_profile_since.load(Ordering::SeqCst) < 5 {
|
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
|
|
|
|
self.refresh_user_profile_since.store(now, Ordering::SeqCst);
|
2023-11-24 03:54:47 +00:00
|
|
|
|
|
2023-08-20 06:13:54 +00:00
|
|
|
|
let uid = old_user_profile.uid;
|
2023-10-24 15:13:51 +00:00
|
|
|
|
let result: Result<UserProfile, FlowyError> = self
|
2023-08-20 06:13:54 +00:00
|
|
|
|
.cloud_services
|
|
|
|
|
.get_user_service()?
|
|
|
|
|
.get_user_profile(UserCredentials::from_uid(uid))
|
2023-10-24 15:13:51 +00:00
|
|
|
|
.await;
|
2023-10-24 12:11:06 +00:00
|
|
|
|
|
2023-10-24 15:13:51 +00:00
|
|
|
|
match result {
|
|
|
|
|
Ok(new_user_profile) => {
|
|
|
|
|
// If the user profile is updated, save the new user profile
|
|
|
|
|
if new_user_profile.updated_at > old_user_profile.updated_at {
|
2023-11-14 06:01:46 +00:00
|
|
|
|
validate_encryption_sign(old_user_profile, &new_user_profile.encryption_type.sign());
|
2023-10-24 15:13:51 +00:00
|
|
|
|
// Save the new user profile
|
|
|
|
|
let changeset = UserTableChangeset::from_user_profile(new_user_profile);
|
|
|
|
|
let _ = upsert_user_profile_change(uid, self.database.get_pool(uid)?, changeset);
|
|
|
|
|
}
|
|
|
|
|
Ok(())
|
|
|
|
|
},
|
|
|
|
|
Err(err) => {
|
|
|
|
|
// If the user is not found, notify the frontend to logout
|
2023-11-05 06:00:24 +00:00
|
|
|
|
if err.is_unauthorized() {
|
2023-10-24 15:13:51 +00:00
|
|
|
|
event!(
|
2023-11-05 06:00:24 +00:00
|
|
|
|
tracing::Level::ERROR,
|
|
|
|
|
"User is unauthorized, sign out the user"
|
2023-10-24 15:13:51 +00:00
|
|
|
|
);
|
2023-11-24 03:54:47 +00:00
|
|
|
|
|
2023-11-05 06:00:24 +00:00
|
|
|
|
self.sign_out().await?;
|
2023-10-24 15:13:51 +00:00
|
|
|
|
send_auth_state_notification(AuthStateChangedPB {
|
|
|
|
|
state: AuthStatePB::InvalidAuth,
|
|
|
|
|
message: "User is not found on the server".to_string(),
|
|
|
|
|
})
|
|
|
|
|
.send();
|
|
|
|
|
}
|
|
|
|
|
Err(err)
|
|
|
|
|
},
|
2023-07-14 05:37:13 +00:00
|
|
|
|
}
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-12 10:00:07 +00:00
|
|
|
|
#[instrument(level = "info", skip_all)]
|
2023-07-29 01:46:24 +00:00
|
|
|
|
pub fn user_dir(&self, uid: i64) -> String {
|
2023-11-24 03:54:47 +00:00
|
|
|
|
self.user_paths.user_data_dir(uid)
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn user_setting(&self) -> Result<UserSettingPB, FlowyError> {
|
2023-07-29 01:46:24 +00:00
|
|
|
|
let session = self.get_session()?;
|
2023-02-13 01:29:49 +00:00
|
|
|
|
let user_setting = UserSettingPB {
|
2023-07-29 01:46:24 +00:00
|
|
|
|
user_folder: self.user_dir(session.user_id),
|
2023-02-13 01:29:49 +00:00
|
|
|
|
};
|
|
|
|
|
Ok(user_setting)
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-04 00:41:16 +00:00
|
|
|
|
pub fn user_id(&self) -> Result<i64, FlowyError> {
|
2023-02-13 01:29:49 +00:00
|
|
|
|
Ok(self.get_session()?.user_id)
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-23 03:43:31 +00:00
|
|
|
|
pub fn workspace_id(&self) -> Result<String, FlowyError> {
|
|
|
|
|
Ok(self.get_session()?.user_workspace.id)
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-14 05:37:13 +00:00
|
|
|
|
pub fn token(&self) -> Result<Option<String>, FlowyError> {
|
|
|
|
|
Ok(None)
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-05-21 10:53:59 +00:00
|
|
|
|
async fn update_user(
|
2023-02-13 01:29:49 +00:00
|
|
|
|
&self,
|
2023-05-21 10:53:59 +00:00
|
|
|
|
uid: i64,
|
2023-10-24 12:11:06 +00:00
|
|
|
|
token: String,
|
2023-02-13 01:29:49 +00:00
|
|
|
|
params: UpdateUserProfileParams,
|
|
|
|
|
) -> Result<(), FlowyError> {
|
2023-07-29 01:46:24 +00:00
|
|
|
|
let server = self.cloud_services.get_user_service()?;
|
2023-10-30 04:35:06 +00:00
|
|
|
|
af_spawn(async move {
|
2023-10-24 12:11:06 +00:00
|
|
|
|
let credentials = UserCredentials::new(Some(token), Some(uid), None);
|
2023-07-29 01:46:24 +00:00
|
|
|
|
server.update_user(credentials, params).await
|
2023-02-13 01:29:49 +00:00
|
|
|
|
})
|
2023-07-29 01:46:24 +00:00
|
|
|
|
.await
|
|
|
|
|
.map_err(internal_error)??;
|
2023-02-13 01:29:49 +00:00
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-17 15:46:39 +00:00
|
|
|
|
async fn save_user(&self, uid: i64, user: UserTable) -> Result<(), FlowyError> {
|
2023-12-23 15:27:15 +00:00
|
|
|
|
let mut conn = self.db_connection(uid)?;
|
|
|
|
|
conn.immediate_transaction(|conn| {
|
2023-07-05 12:57:09 +00:00
|
|
|
|
// delete old user if exists
|
2023-07-29 01:46:24 +00:00
|
|
|
|
diesel::delete(user_table::dsl::user_table.filter(user_table::dsl::id.eq(&user.id)))
|
2023-12-23 15:27:15 +00:00
|
|
|
|
.execute(conn)?;
|
2023-07-05 12:57:09 +00:00
|
|
|
|
|
|
|
|
|
let _ = diesel::insert_into(user_table::table)
|
2023-08-17 15:46:39 +00:00
|
|
|
|
.values(user)
|
2023-12-23 15:27:15 +00:00
|
|
|
|
.execute(conn)?;
|
2023-07-05 12:57:09 +00:00
|
|
|
|
Ok::<(), FlowyError>(())
|
|
|
|
|
})?;
|
|
|
|
|
|
2023-08-17 15:46:39 +00:00
|
|
|
|
Ok(())
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-08-12 09:36:31 +00:00
|
|
|
|
pub async fn receive_realtime_event(&self, json: Value) {
|
2023-08-20 06:13:54 +00:00
|
|
|
|
if let Ok(user_service) = self.cloud_services.get_user_service() {
|
|
|
|
|
user_service.receive_realtime_event(json)
|
|
|
|
|
}
|
2023-08-12 09:36:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-14 05:37:13 +00:00
|
|
|
|
/// Returns the current user session.
|
|
|
|
|
pub fn get_session(&self) -> Result<Session, FlowyError> {
|
2023-08-20 06:13:54 +00:00
|
|
|
|
if let Some(session) = (self.current_session.read()).clone() {
|
|
|
|
|
return Ok(session);
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-06 03:51:03 +00:00
|
|
|
|
match self
|
|
|
|
|
.store_preferences
|
2023-11-20 12:54:47 +00:00
|
|
|
|
.get_object::<Session>(&self.user_config.session_cache_key)
|
2023-08-06 03:51:03 +00:00
|
|
|
|
{
|
2023-07-14 05:37:13 +00:00
|
|
|
|
None => Err(FlowyError::new(
|
|
|
|
|
ErrorCode::RecordNotFound,
|
2023-08-05 07:02:05 +00:00
|
|
|
|
"User is not logged in",
|
2023-07-14 05:37:13 +00:00
|
|
|
|
)),
|
2023-08-20 06:13:54 +00:00
|
|
|
|
Some(session) => {
|
|
|
|
|
self.current_session.write().replace(session.clone());
|
|
|
|
|
Ok(session)
|
|
|
|
|
},
|
2021-07-18 15:56:36 +00:00
|
|
|
|
}
|
2023-02-13 01:29:49 +00:00
|
|
|
|
}
|
2023-08-14 04:57:59 +00:00
|
|
|
|
|
2023-08-20 06:13:54 +00:00
|
|
|
|
pub(crate) fn set_session(&self, session: Option<Session>) -> Result<(), FlowyError> {
|
2023-10-07 01:58:44 +00:00
|
|
|
|
debug!("Set current user: {:?}", session);
|
2023-08-20 06:13:54 +00:00
|
|
|
|
match &session {
|
|
|
|
|
None => {
|
|
|
|
|
self.current_session.write().take();
|
|
|
|
|
self
|
|
|
|
|
.store_preferences
|
2023-11-20 12:54:47 +00:00
|
|
|
|
.remove(&self.user_config.session_cache_key)
|
2023-08-20 06:13:54 +00:00
|
|
|
|
},
|
|
|
|
|
Some(session) => {
|
|
|
|
|
self.current_session.write().replace(session.clone());
|
|
|
|
|
self
|
|
|
|
|
.store_preferences
|
2023-11-20 12:54:47 +00:00
|
|
|
|
.set_object(&self.user_config.session_cache_key, session.clone())
|
2023-08-20 06:13:54 +00:00
|
|
|
|
.map_err(internal_error)?;
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-07 01:58:44 +00:00
|
|
|
|
pub(crate) async fn generate_sign_in_url_with_email(
|
2023-10-02 09:22:22 +00:00
|
|
|
|
&self,
|
2023-11-12 10:00:07 +00:00
|
|
|
|
authenticator: &Authenticator,
|
2023-10-02 09:22:22 +00:00
|
|
|
|
email: &str,
|
|
|
|
|
) -> Result<String, FlowyError> {
|
2023-11-12 10:00:07 +00:00
|
|
|
|
self.update_authenticator(authenticator).await;
|
2023-10-02 09:22:22 +00:00
|
|
|
|
|
|
|
|
|
let auth_service = self.cloud_services.get_user_service()?;
|
2023-10-07 01:58:44 +00:00
|
|
|
|
let url = auth_service
|
|
|
|
|
.generate_sign_in_url_with_email(email)
|
|
|
|
|
.await
|
|
|
|
|
.map_err(|err| FlowyError::server_error().with_context(err))?;
|
|
|
|
|
Ok(url)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub(crate) async fn generate_oauth_url(
|
|
|
|
|
&self,
|
|
|
|
|
oauth_provider: &str,
|
|
|
|
|
) -> Result<String, FlowyError> {
|
2023-11-24 03:54:47 +00:00
|
|
|
|
self
|
|
|
|
|
.update_authenticator(&Authenticator::AppFlowyCloud)
|
|
|
|
|
.await;
|
2023-10-07 01:58:44 +00:00
|
|
|
|
let auth_service = self.cloud_services.get_user_service()?;
|
|
|
|
|
let url = auth_service
|
|
|
|
|
.generate_oauth_url_with_provider(oauth_provider)
|
|
|
|
|
.await?;
|
2023-10-02 09:22:22 +00:00
|
|
|
|
Ok(url)
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-05 06:00:24 +00:00
|
|
|
|
#[instrument(level = "info", skip_all, err)]
|
2023-08-17 15:46:39 +00:00
|
|
|
|
async fn save_auth_data(
|
|
|
|
|
&self,
|
|
|
|
|
response: &impl UserAuthResponse,
|
2023-11-12 10:00:07 +00:00
|
|
|
|
authenticator: &Authenticator,
|
2023-08-17 15:46:39 +00:00
|
|
|
|
session: &Session,
|
|
|
|
|
) -> Result<(), FlowyError> {
|
2023-11-12 10:00:07 +00:00
|
|
|
|
let user_profile = UserProfile::from((response, authenticator));
|
2023-08-17 15:46:39 +00:00
|
|
|
|
let uid = user_profile.uid;
|
2023-12-21 06:13:21 +00:00
|
|
|
|
if authenticator.is_local() {
|
|
|
|
|
event!(tracing::Level::DEBUG, "Save new anon user: {:?}", uid);
|
|
|
|
|
self.set_anon_user(session.clone());
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-17 15:46:39 +00:00
|
|
|
|
save_user_workspaces(uid, self.db_pool(uid)?, response.user_workspaces())?;
|
2023-10-24 12:11:06 +00:00
|
|
|
|
event!(tracing::Level::INFO, "Save new user profile to disk");
|
2023-08-17 15:46:39 +00:00
|
|
|
|
self
|
2023-11-12 10:00:07 +00:00
|
|
|
|
.save_user(uid, (user_profile, authenticator.clone()).into())
|
2023-08-17 15:46:39 +00:00
|
|
|
|
.await?;
|
2023-08-20 06:13:54 +00:00
|
|
|
|
self.set_session(Some(session.clone()))?;
|
2023-08-17 15:46:39 +00:00
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-14 04:57:59 +00:00
|
|
|
|
fn set_collab_config(&self, session: &Session) {
|
|
|
|
|
let collab_builder = self.collab_builder.upgrade().unwrap();
|
|
|
|
|
collab_builder.initialize(session.user_workspace.id.clone());
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-20 06:13:54 +00:00
|
|
|
|
async fn handler_user_update(&self, user_update: UserUpdate) -> FlowyResult<()> {
|
|
|
|
|
let session = self.get_session()?;
|
|
|
|
|
if session.user_id == user_update.uid {
|
2023-10-07 01:58:44 +00:00
|
|
|
|
debug!("Receive user update: {:?}", user_update);
|
2023-11-24 03:54:47 +00:00
|
|
|
|
let user_profile = self.get_user_profile_from_disk(user_update.uid).await?;
|
2023-11-14 06:01:46 +00:00
|
|
|
|
if !validate_encryption_sign(&user_profile, &user_update.encryption_sign) {
|
2023-08-20 06:13:54 +00:00
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Save the user profile change
|
2023-10-24 12:11:06 +00:00
|
|
|
|
upsert_user_profile_change(
|
2023-08-20 06:13:54 +00:00
|
|
|
|
user_update.uid,
|
|
|
|
|
self.db_pool(user_update.uid)?,
|
|
|
|
|
UserTableChangeset::from(user_update),
|
|
|
|
|
)?;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-12 10:00:07 +00:00
|
|
|
|
async fn migrate_anon_user_data_to_cloud(
|
2023-08-14 04:57:59 +00:00
|
|
|
|
&self,
|
2023-08-17 15:46:39 +00:00
|
|
|
|
old_user: &MigrationUser,
|
|
|
|
|
new_user: &MigrationUser,
|
2023-12-06 14:54:17 +00:00
|
|
|
|
authenticator: &Authenticator,
|
2023-08-28 05:28:24 +00:00
|
|
|
|
) -> Result<(), FlowyError> {
|
2023-11-24 03:54:47 +00:00
|
|
|
|
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)?;
|
2023-11-01 03:45:35 +00:00
|
|
|
|
migration_anon_user_on_sign_up(old_user, &old_collab_db, new_user, &new_collab_db)?;
|
2023-08-28 05:28:24 +00:00
|
|
|
|
|
2023-12-06 14:54:17 +00:00
|
|
|
|
match authenticator {
|
|
|
|
|
Authenticator::Supabase => {
|
|
|
|
|
if let Err(err) = sync_supabase_user_data_to_cloud(
|
|
|
|
|
self.cloud_services.get_user_service()?,
|
|
|
|
|
&self.user_config.device_id,
|
|
|
|
|
new_user,
|
|
|
|
|
&new_collab_db,
|
|
|
|
|
)
|
|
|
|
|
.await
|
|
|
|
|
{
|
|
|
|
|
error!("Sync user data to cloud failed: {:?}", err);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
Authenticator::AppFlowyCloud => {
|
|
|
|
|
if let Err(err) = sync_af_user_data_to_cloud(
|
|
|
|
|
self.cloud_services.get_user_service()?,
|
|
|
|
|
&self.user_config.device_id,
|
|
|
|
|
new_user,
|
|
|
|
|
&new_collab_db,
|
|
|
|
|
)
|
|
|
|
|
.await
|
|
|
|
|
{
|
|
|
|
|
error!("Sync user data to cloud failed: {:?}", err);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
_ => {},
|
2023-08-28 05:28:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-08-17 15:46:39 +00:00
|
|
|
|
// Save the old user workspace setting.
|
|
|
|
|
save_user_workspaces(
|
|
|
|
|
old_user.session.user_id,
|
|
|
|
|
self.database.get_pool(old_user.session.user_id)?,
|
|
|
|
|
&[old_user.session.user_workspace.clone()],
|
|
|
|
|
)?;
|
2023-08-28 05:28:24 +00:00
|
|
|
|
Ok(())
|
2023-08-14 04:57:59 +00:00
|
|
|
|
}
|
2021-07-13 15:08:20 +00:00
|
|
|
|
}
|
2023-08-20 06:13:54 +00:00
|
|
|
|
|
2023-12-21 06:13:21 +00:00
|
|
|
|
fn current_authenticator() -> Authenticator {
|
|
|
|
|
match AuthenticatorType::from_env() {
|
|
|
|
|
AuthenticatorType::Local => Authenticator::Local,
|
|
|
|
|
AuthenticatorType::Supabase => Authenticator::Supabase,
|
|
|
|
|
AuthenticatorType::AppFlowyCloud => Authenticator::AppFlowyCloud,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-14 06:01:46 +00:00
|
|
|
|
fn validate_encryption_sign(user_profile: &UserProfile, encryption_sign: &str) -> bool {
|
2023-08-20 06:13:54 +00:00
|
|
|
|
// If the local user profile's encryption sign is not equal to the user update's encryption sign,
|
|
|
|
|
// which means the user enable encryption in another device, we should logout the current user.
|
|
|
|
|
let is_valid = user_profile.encryption_type.sign() == encryption_sign;
|
|
|
|
|
if !is_valid {
|
|
|
|
|
send_auth_state_notification(AuthStateChangedPB {
|
2023-10-12 12:25:00 +00:00
|
|
|
|
state: AuthStatePB::InvalidAuth,
|
2023-10-24 12:11:06 +00:00
|
|
|
|
message: "Encryption configuration was changed".to_string(),
|
2023-08-20 06:13:54 +00:00
|
|
|
|
})
|
|
|
|
|
.send();
|
|
|
|
|
}
|
|
|
|
|
is_valid
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-24 12:11:06 +00:00
|
|
|
|
fn upsert_user_profile_change(
|
2023-08-20 06:13:54 +00:00
|
|
|
|
uid: i64,
|
|
|
|
|
pool: Arc<ConnectionPool>,
|
|
|
|
|
changeset: UserTableChangeset,
|
|
|
|
|
) -> FlowyResult<()> {
|
2023-10-24 12:11:06 +00:00
|
|
|
|
event!(
|
|
|
|
|
tracing::Level::DEBUG,
|
|
|
|
|
"Update user profile with changeset: {:?}",
|
|
|
|
|
changeset
|
|
|
|
|
);
|
2023-12-23 15:27:15 +00:00
|
|
|
|
let mut conn = pool.get()?;
|
|
|
|
|
diesel_update_table!(user_table, changeset, &mut *conn);
|
2023-08-20 06:13:54 +00:00
|
|
|
|
let user: UserProfile = user_table::dsl::user_table
|
|
|
|
|
.filter(user_table::id.eq(&uid.to_string()))
|
2023-12-23 15:27:15 +00:00
|
|
|
|
.first::<UserTable>(&mut *conn)?
|
2023-08-20 06:13:54 +00:00
|
|
|
|
.into();
|
|
|
|
|
send_notification(&uid.to_string(), UserNotification::DidUpdateUserProfile)
|
|
|
|
|
.payload(UserProfilePB::from(user))
|
|
|
|
|
.send();
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
2023-10-10 11:05:55 +00:00
|
|
|
|
|
|
|
|
|
#[instrument(level = "info", skip_all, err)]
|
|
|
|
|
fn save_user_token(uid: i64, pool: Arc<ConnectionPool>, token: String) -> FlowyResult<()> {
|
|
|
|
|
let params = UpdateUserProfileParams::new(uid).with_token(token);
|
|
|
|
|
let changeset = UserTableChangeset::new(params);
|
2023-10-24 12:11:06 +00:00
|
|
|
|
upsert_user_profile_change(uid, pool, changeset)
|
2023-10-10 11:05:55 +00:00
|
|
|
|
}
|
2023-11-12 10:00:07 +00:00
|
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
|
struct UserPaths {
|
|
|
|
|
root: String,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl UserPaths {
|
2023-11-20 12:54:47 +00:00
|
|
|
|
fn new(root: String) -> Self {
|
|
|
|
|
Self { root }
|
|
|
|
|
}
|
2023-11-24 03:54:47 +00:00
|
|
|
|
|
|
|
|
|
/// Returns the path to the user's data directory.
|
|
|
|
|
fn user_data_dir(&self, uid: i64) -> String {
|
2023-11-12 10:00:07 +00:00
|
|
|
|
format!("{}/{}", self.root, uid)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl UserDBPath for UserPaths {
|
|
|
|
|
fn user_db_path(&self, uid: i64) -> PathBuf {
|
2023-11-24 03:54:47 +00:00
|
|
|
|
PathBuf::from(self.user_data_dir(uid))
|
2023-11-12 10:00:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn collab_db_path(&self, uid: i64) -> PathBuf {
|
2023-11-24 03:54:47 +00:00
|
|
|
|
let mut path = PathBuf::from(self.user_data_dir(uid));
|
2023-11-12 10:00:07 +00:00
|
|
|
|
path.push("collab_db");
|
|
|
|
|
path
|
|
|
|
|
}
|
2023-11-20 12:54:47 +00:00
|
|
|
|
|
|
|
|
|
fn collab_db_history(&self, uid: i64, create_if_not_exist: bool) -> std::io::Result<PathBuf> {
|
2023-11-24 03:54:47 +00:00
|
|
|
|
let path = PathBuf::from(self.user_data_dir(uid)).join("collab_db_history");
|
2023-11-20 12:54:47 +00:00
|
|
|
|
if !path.exists() && create_if_not_exist {
|
|
|
|
|
fs::create_dir_all(&path)?;
|
|
|
|
|
}
|
|
|
|
|
Ok(path)
|
|
|
|
|
}
|
2023-11-12 10:00:07 +00:00
|
|
|
|
}
|