2023-05-21 18:53:59 +08:00
|
|
|
use std::collections::HashMap;
|
2023-08-06 11:51:03 +08:00
|
|
|
use std::sync::{Arc, Weak};
|
2023-05-21 18:53:59 +08:00
|
|
|
|
2023-07-05 20:57:09 +08:00
|
|
|
use appflowy_integrate::collab_builder::{CollabStorageProvider, CollabStorageType};
|
2023-07-29 09:46:24 +08:00
|
|
|
use appflowy_integrate::{CollabType, RemoteCollabStorage, YrsDocAction};
|
2023-05-21 18:53:59 +08:00
|
|
|
use parking_lot::RwLock;
|
2023-07-05 20:57:09 +08:00
|
|
|
use serde_repr::*;
|
2023-05-21 18:53:59 +08:00
|
|
|
|
2023-07-29 09:46:24 +08:00
|
|
|
use flowy_database_deps::cloud::*;
|
|
|
|
use flowy_document2::deps::DocumentData;
|
|
|
|
use flowy_document_deps::cloud::{DocumentCloudService, DocumentSnapshot};
|
2023-05-23 23:55:21 +08:00
|
|
|
use flowy_error::{ErrorCode, FlowyError, FlowyResult};
|
2023-07-29 09:46:24 +08:00
|
|
|
use flowy_folder_deps::cloud::*;
|
|
|
|
use flowy_server::local_server::{LocalServer, LocalServerDB};
|
2023-05-21 18:53:59 +08:00
|
|
|
use flowy_server::self_host::configuration::self_host_server_configuration;
|
|
|
|
use flowy_server::self_host::SelfHostServer;
|
2023-07-14 13:37:13 +08:00
|
|
|
use flowy_server::supabase::SupabaseServer;
|
2023-05-21 18:53:59 +08:00
|
|
|
use flowy_server::AppFlowyServer;
|
2023-07-14 13:37:13 +08:00
|
|
|
use flowy_server_config::supabase_config::SupabaseConfiguration;
|
2023-08-06 11:51:03 +08:00
|
|
|
use flowy_sqlite::kv::StorePreferences;
|
2023-07-29 09:46:24 +08:00
|
|
|
use flowy_user::event_map::UserCloudServiceProvider;
|
|
|
|
use flowy_user::services::database::{
|
|
|
|
get_user_profile, get_user_workspace, open_collab_db, open_user_db,
|
|
|
|
};
|
|
|
|
use flowy_user_deps::cloud::UserService;
|
|
|
|
use flowy_user_deps::entities::*;
|
2023-07-05 20:57:09 +08:00
|
|
|
use lib_infra::future::FutureResult;
|
2023-05-23 23:55:21 +08:00
|
|
|
|
2023-07-14 13:37:13 +08:00
|
|
|
use crate::AppFlowyCoreConfig;
|
|
|
|
|
2023-05-23 23:55:21 +08:00
|
|
|
const SERVER_PROVIDER_TYPE_KEY: &str = "server_provider_type";
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Hash, Eq, PartialEq, Serialize_repr, Deserialize_repr)]
|
|
|
|
#[repr(u8)]
|
|
|
|
pub enum ServerProviderType {
|
|
|
|
/// Local server provider.
|
|
|
|
/// Offline mode, no user authentication and the data is stored locally.
|
|
|
|
Local = 0,
|
|
|
|
/// Self-hosted server provider.
|
|
|
|
/// The [AppFlowy-Server](https://github.com/AppFlowy-IO/AppFlowy-Server) is still a work in
|
|
|
|
/// progress.
|
|
|
|
SelfHosted = 1,
|
|
|
|
/// Supabase server provider.
|
|
|
|
/// It uses supabase's postgresql database to store data and user authentication.
|
|
|
|
Supabase = 2,
|
|
|
|
}
|
|
|
|
|
2023-05-21 18:53:59 +08:00
|
|
|
/// The [AppFlowyServerProvider] provides list of [AppFlowyServer] base on the [AuthType]. Using
|
|
|
|
/// the auth type, the [AppFlowyServerProvider] will create a new [AppFlowyServer] if it doesn't
|
|
|
|
/// exist.
|
2023-07-29 09:46:24 +08:00
|
|
|
/// Each server implements the [AppFlowyServer] trait, which provides the [UserService], etc.
|
2023-05-21 18:53:59 +08:00
|
|
|
pub struct AppFlowyServerProvider {
|
2023-07-14 13:37:13 +08:00
|
|
|
config: AppFlowyCoreConfig,
|
2023-05-23 23:55:21 +08:00
|
|
|
provider_type: RwLock<ServerProviderType>,
|
|
|
|
providers: RwLock<HashMap<ServerProviderType, Arc<dyn AppFlowyServer>>>,
|
2023-07-14 13:37:13 +08:00
|
|
|
supabase_config: RwLock<Option<SupabaseConfiguration>>,
|
2023-08-06 11:51:03 +08:00
|
|
|
store_preferences: Weak<StorePreferences>,
|
2023-05-21 18:53:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl AppFlowyServerProvider {
|
2023-08-06 11:51:03 +08:00
|
|
|
pub fn new(
|
|
|
|
config: AppFlowyCoreConfig,
|
|
|
|
provider_type: ServerProviderType,
|
|
|
|
supabase_config: Option<SupabaseConfiguration>,
|
|
|
|
store_preferences: Weak<StorePreferences>,
|
|
|
|
) -> Self {
|
2023-07-14 13:37:13 +08:00
|
|
|
Self {
|
|
|
|
config,
|
2023-08-06 11:51:03 +08:00
|
|
|
provider_type: RwLock::new(provider_type),
|
2023-07-14 13:37:13 +08:00
|
|
|
providers: RwLock::new(HashMap::new()),
|
|
|
|
supabase_config: RwLock::new(supabase_config),
|
2023-08-06 11:51:03 +08:00
|
|
|
store_preferences,
|
2023-07-14 13:37:13 +08:00
|
|
|
}
|
2023-05-21 18:53:59 +08:00
|
|
|
}
|
2023-05-23 23:55:21 +08:00
|
|
|
|
|
|
|
pub fn provider_type(&self) -> ServerProviderType {
|
|
|
|
self.provider_type.read().clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a [AppFlowyServer] trait implementation base on the provider_type.
|
|
|
|
fn get_provider(
|
|
|
|
&self,
|
|
|
|
provider_type: &ServerProviderType,
|
|
|
|
) -> FlowyResult<Arc<dyn AppFlowyServer>> {
|
|
|
|
if let Some(provider) = self.providers.read().get(provider_type) {
|
|
|
|
return Ok(provider.clone());
|
|
|
|
}
|
|
|
|
|
2023-07-14 13:37:13 +08:00
|
|
|
let server = match provider_type {
|
|
|
|
ServerProviderType::Local => {
|
2023-07-29 09:46:24 +08:00
|
|
|
let local_db = Arc::new(LocalServerDBImpl {
|
|
|
|
storage_path: self.config.storage_path.clone(),
|
|
|
|
});
|
|
|
|
let server = Arc::new(LocalServer::new(local_db));
|
|
|
|
|
2023-07-14 13:37:13 +08:00
|
|
|
Ok::<Arc<dyn AppFlowyServer>, FlowyError>(server)
|
|
|
|
},
|
|
|
|
ServerProviderType::SelfHosted => {
|
|
|
|
let config = self_host_server_configuration().map_err(|e| {
|
|
|
|
FlowyError::new(
|
|
|
|
ErrorCode::InvalidAuthConfig,
|
|
|
|
format!(
|
|
|
|
"Missing self host config: {:?}. Error: {:?}",
|
|
|
|
provider_type, e
|
|
|
|
),
|
|
|
|
)
|
|
|
|
})?;
|
|
|
|
let server = Arc::new(SelfHostServer::new(config));
|
|
|
|
Ok::<Arc<dyn AppFlowyServer>, FlowyError>(server)
|
|
|
|
},
|
|
|
|
ServerProviderType::Supabase => {
|
|
|
|
let config = self.supabase_config.read().clone().ok_or(FlowyError::new(
|
|
|
|
ErrorCode::InvalidAuthConfig,
|
|
|
|
"Missing supabase config".to_string(),
|
|
|
|
))?;
|
|
|
|
Ok::<Arc<dyn AppFlowyServer>, FlowyError>(Arc::new(SupabaseServer::new(config)))
|
|
|
|
},
|
|
|
|
}?;
|
|
|
|
|
2023-05-23 23:55:21 +08:00
|
|
|
self
|
|
|
|
.providers
|
|
|
|
.write()
|
|
|
|
.insert(provider_type.clone(), server.clone());
|
|
|
|
Ok(server)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-14 13:37:13 +08:00
|
|
|
impl UserCloudServiceProvider for AppFlowyServerProvider {
|
|
|
|
fn update_supabase_config(&self, supabase_config: &SupabaseConfiguration) {
|
|
|
|
self
|
|
|
|
.supabase_config
|
|
|
|
.write()
|
|
|
|
.replace(supabase_config.clone());
|
|
|
|
|
|
|
|
supabase_config.write_env();
|
|
|
|
if let Ok(provider) = self.get_provider(&self.provider_type.read()) {
|
|
|
|
provider.enable_sync(supabase_config.enable_sync);
|
2023-05-23 23:55:21 +08:00
|
|
|
}
|
|
|
|
}
|
2023-05-21 18:53:59 +08:00
|
|
|
|
2023-05-31 17:42:14 +08:00
|
|
|
/// When user login, the provider type is set by the [AuthType] and save to disk for next use.
|
|
|
|
///
|
2023-05-23 23:55:21 +08:00
|
|
|
/// Each [AuthType] has a corresponding [ServerProviderType]. The [ServerProviderType] is used
|
|
|
|
/// to create a new [AppFlowyServer] if it doesn't exist. Once the [ServerProviderType] is set,
|
|
|
|
/// it will be used when user open the app again.
|
2023-05-31 17:42:14 +08:00
|
|
|
///
|
2023-05-23 23:55:21 +08:00
|
|
|
fn set_auth_type(&self, auth_type: AuthType) {
|
|
|
|
let provider_type: ServerProviderType = auth_type.into();
|
2023-05-31 17:42:14 +08:00
|
|
|
*self.provider_type.write() = provider_type.clone();
|
|
|
|
|
2023-08-06 11:51:03 +08:00
|
|
|
match self.store_preferences.upgrade() {
|
|
|
|
None => tracing::error!("🔴Failed to update server provider type: store preferences is drop"),
|
|
|
|
Some(store_preferences) => {
|
|
|
|
match store_preferences.set_object(SERVER_PROVIDER_TYPE_KEY, provider_type.clone()) {
|
|
|
|
Ok(_) => tracing::trace!("Update server provider type to: {:?}", provider_type),
|
|
|
|
Err(e) => {
|
|
|
|
tracing::error!("🔴Failed to update server provider type: {:?}", e);
|
|
|
|
},
|
|
|
|
}
|
2023-05-23 23:55:21 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-29 09:46:24 +08:00
|
|
|
/// Returns the [UserService] base on the current [ServerProviderType].
|
2023-05-21 18:53:59 +08:00
|
|
|
/// Creates a new [AppFlowyServer] if it doesn't exist.
|
2023-07-29 09:46:24 +08:00
|
|
|
fn get_user_service(&self) -> Result<Arc<dyn UserService>, FlowyError> {
|
2023-05-31 17:42:14 +08:00
|
|
|
Ok(
|
|
|
|
self
|
|
|
|
.get_provider(&self.provider_type.read())?
|
|
|
|
.user_service(),
|
|
|
|
)
|
2023-05-23 23:55:21 +08:00
|
|
|
}
|
|
|
|
}
|
2023-05-21 18:53:59 +08:00
|
|
|
|
2023-05-23 23:55:21 +08:00
|
|
|
impl FolderCloudService for AppFlowyServerProvider {
|
2023-07-29 09:46:24 +08:00
|
|
|
fn create_workspace(&self, uid: i64, name: &str) -> FutureResult<Workspace, Error> {
|
2023-05-23 23:55:21 +08:00
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
|
|
|
let name = name.to_string();
|
|
|
|
FutureResult::new(async move { server?.folder_service().create_workspace(uid, &name).await })
|
2023-05-21 18:53:59 +08:00
|
|
|
}
|
2023-07-05 20:57:09 +08:00
|
|
|
|
2023-07-29 09:46:24 +08:00
|
|
|
fn get_folder_data(&self, workspace_id: &str) -> FutureResult<Option<FolderData>, Error> {
|
2023-07-14 13:37:13 +08:00
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
|
|
|
let workspace_id = workspace_id.to_string();
|
|
|
|
FutureResult::new(async move {
|
|
|
|
server?
|
|
|
|
.folder_service()
|
|
|
|
.get_folder_data(&workspace_id)
|
|
|
|
.await
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-07-05 20:57:09 +08:00
|
|
|
fn get_folder_latest_snapshot(
|
|
|
|
&self,
|
|
|
|
workspace_id: &str,
|
2023-07-29 09:46:24 +08:00
|
|
|
) -> FutureResult<Option<FolderSnapshot>, Error> {
|
2023-07-05 20:57:09 +08:00
|
|
|
let workspace_id = workspace_id.to_string();
|
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
|
|
|
FutureResult::new(async move {
|
|
|
|
server?
|
|
|
|
.folder_service()
|
|
|
|
.get_folder_latest_snapshot(&workspace_id)
|
|
|
|
.await
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-07-29 09:46:24 +08:00
|
|
|
fn get_folder_updates(&self, workspace_id: &str, uid: i64) -> FutureResult<Vec<Vec<u8>>, Error> {
|
2023-07-05 20:57:09 +08:00
|
|
|
let workspace_id = workspace_id.to_string();
|
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
|
|
|
FutureResult::new(async move {
|
|
|
|
server?
|
|
|
|
.folder_service()
|
2023-07-14 13:37:13 +08:00
|
|
|
.get_folder_updates(&workspace_id, uid)
|
2023-07-05 20:57:09 +08:00
|
|
|
.await
|
|
|
|
})
|
|
|
|
}
|
2023-07-14 13:37:13 +08:00
|
|
|
|
|
|
|
fn service_name(&self) -> String {
|
|
|
|
self
|
|
|
|
.get_provider(&self.provider_type.read())
|
|
|
|
.map(|provider| provider.folder_service().service_name())
|
|
|
|
.unwrap_or_default()
|
|
|
|
}
|
2023-07-05 20:57:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl DatabaseCloudService for AppFlowyServerProvider {
|
2023-07-29 09:46:24 +08:00
|
|
|
fn get_collab_update(
|
|
|
|
&self,
|
|
|
|
object_id: &str,
|
|
|
|
object_ty: CollabType,
|
|
|
|
) -> FutureResult<CollabObjectUpdate, Error> {
|
2023-07-05 20:57:09 +08:00
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
2023-07-14 13:37:13 +08:00
|
|
|
let database_id = object_id.to_string();
|
2023-07-05 20:57:09 +08:00
|
|
|
FutureResult::new(async move {
|
|
|
|
server?
|
|
|
|
.database_service()
|
2023-07-29 09:46:24 +08:00
|
|
|
.get_collab_update(&database_id, object_ty)
|
2023-07-05 20:57:09 +08:00
|
|
|
.await
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-07-14 13:37:13 +08:00
|
|
|
fn batch_get_collab_updates(
|
2023-07-05 20:57:09 +08:00
|
|
|
&self,
|
2023-07-14 13:37:13 +08:00
|
|
|
object_ids: Vec<String>,
|
2023-07-29 09:46:24 +08:00
|
|
|
object_ty: CollabType,
|
|
|
|
) -> FutureResult<CollabObjectUpdateByOid, Error> {
|
2023-07-14 13:37:13 +08:00
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
|
|
|
FutureResult::new(async move {
|
|
|
|
server?
|
|
|
|
.database_service()
|
2023-07-29 09:46:24 +08:00
|
|
|
.batch_get_collab_updates(object_ids, object_ty)
|
2023-07-14 13:37:13 +08:00
|
|
|
.await
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_collab_latest_snapshot(
|
|
|
|
&self,
|
|
|
|
object_id: &str,
|
2023-07-29 09:46:24 +08:00
|
|
|
) -> FutureResult<Option<DatabaseSnapshot>, Error> {
|
2023-07-05 20:57:09 +08:00
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
2023-07-14 13:37:13 +08:00
|
|
|
let database_id = object_id.to_string();
|
2023-07-05 20:57:09 +08:00
|
|
|
FutureResult::new(async move {
|
|
|
|
server?
|
|
|
|
.database_service()
|
2023-07-14 13:37:13 +08:00
|
|
|
.get_collab_latest_snapshot(&database_id)
|
2023-07-05 20:57:09 +08:00
|
|
|
.await
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DocumentCloudService for AppFlowyServerProvider {
|
2023-07-29 09:46:24 +08:00
|
|
|
fn get_document_updates(&self, document_id: &str) -> FutureResult<Vec<Vec<u8>>, Error> {
|
2023-07-05 20:57:09 +08:00
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
|
|
|
let document_id = document_id.to_string();
|
|
|
|
FutureResult::new(async move {
|
|
|
|
server?
|
|
|
|
.document_service()
|
|
|
|
.get_document_updates(&document_id)
|
|
|
|
.await
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_document_latest_snapshot(
|
|
|
|
&self,
|
|
|
|
document_id: &str,
|
2023-07-29 09:46:24 +08:00
|
|
|
) -> FutureResult<Option<DocumentSnapshot>, Error> {
|
2023-07-05 20:57:09 +08:00
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
|
|
|
let document_id = document_id.to_string();
|
|
|
|
FutureResult::new(async move {
|
|
|
|
server?
|
|
|
|
.document_service()
|
|
|
|
.get_document_latest_snapshot(&document_id)
|
|
|
|
.await
|
|
|
|
})
|
|
|
|
}
|
2023-07-14 13:37:13 +08:00
|
|
|
|
2023-07-29 09:46:24 +08:00
|
|
|
fn get_document_data(&self, document_id: &str) -> FutureResult<Option<DocumentData>, Error> {
|
2023-07-14 13:37:13 +08:00
|
|
|
let server = self.get_provider(&self.provider_type.read());
|
|
|
|
let document_id = document_id.to_string();
|
|
|
|
FutureResult::new(async move {
|
|
|
|
server?
|
|
|
|
.document_service()
|
|
|
|
.get_document_data(&document_id)
|
|
|
|
.await
|
|
|
|
})
|
|
|
|
}
|
2023-07-05 20:57:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl CollabStorageProvider for AppFlowyServerProvider {
|
|
|
|
fn storage_type(&self) -> CollabStorageType {
|
|
|
|
self.provider_type().into()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_storage(&self, storage_type: &CollabStorageType) -> Option<Arc<dyn RemoteCollabStorage>> {
|
|
|
|
match storage_type {
|
|
|
|
CollabStorageType::Local => None,
|
|
|
|
CollabStorageType::AWS => None,
|
|
|
|
CollabStorageType::Supabase => self
|
|
|
|
.get_provider(&ServerProviderType::Supabase)
|
|
|
|
.ok()
|
|
|
|
.and_then(|provider| provider.collab_storage()),
|
|
|
|
}
|
|
|
|
}
|
2023-05-21 18:53:59 +08:00
|
|
|
|
2023-07-14 13:37:13 +08:00
|
|
|
fn is_sync_enabled(&self) -> bool {
|
|
|
|
self
|
|
|
|
.supabase_config
|
|
|
|
.read()
|
|
|
|
.as_ref()
|
|
|
|
.map(|config| config.enable_sync)
|
|
|
|
.unwrap_or(false)
|
2023-05-21 18:53:59 +08:00
|
|
|
}
|
|
|
|
}
|
2023-05-23 23:55:21 +08:00
|
|
|
|
|
|
|
impl From<AuthType> for ServerProviderType {
|
|
|
|
fn from(auth_provider: AuthType) -> Self {
|
|
|
|
match auth_provider {
|
|
|
|
AuthType::Local => ServerProviderType::Local,
|
|
|
|
AuthType::SelfHosted => ServerProviderType::SelfHosted,
|
|
|
|
AuthType::Supabase => ServerProviderType::Supabase,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<&AuthType> for ServerProviderType {
|
|
|
|
fn from(auth_provider: &AuthType) -> Self {
|
|
|
|
Self::from(auth_provider.clone())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-06 11:51:03 +08:00
|
|
|
pub fn current_server_provider(store_preferences: &Arc<StorePreferences>) -> ServerProviderType {
|
|
|
|
match store_preferences.get_object::<ServerProviderType>(SERVER_PROVIDER_TYPE_KEY) {
|
2023-05-23 23:55:21 +08:00
|
|
|
None => ServerProviderType::Local,
|
|
|
|
Some(provider_type) => provider_type,
|
|
|
|
}
|
|
|
|
}
|
2023-07-29 09:46:24 +08:00
|
|
|
|
|
|
|
struct LocalServerDBImpl {
|
|
|
|
storage_path: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl LocalServerDB for LocalServerDBImpl {
|
|
|
|
fn get_user_profile(&self, uid: i64) -> Result<Option<UserProfile>, FlowyError> {
|
|
|
|
let sqlite_db = open_user_db(&self.storage_path, uid)?;
|
|
|
|
let user_profile = get_user_profile(&sqlite_db, uid).ok();
|
|
|
|
Ok(user_profile)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_user_workspace(&self, uid: i64) -> Result<Option<UserWorkspace>, FlowyError> {
|
|
|
|
let sqlite_db = open_user_db(&self.storage_path, uid)?;
|
|
|
|
let user_workspace = get_user_workspace(&sqlite_db, uid)?;
|
|
|
|
Ok(user_workspace)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_collab_updates(&self, uid: i64, object_id: &str) -> Result<Vec<Vec<u8>>, FlowyError> {
|
|
|
|
let collab_db = open_collab_db(&self.storage_path, uid)?;
|
|
|
|
let read_txn = collab_db.read_txn();
|
|
|
|
let updates = read_txn
|
|
|
|
.get_all_updates(uid, object_id)
|
|
|
|
.map_err(|e| FlowyError::internal().context(format!("Failed to open collab db: {:?}", e)))?;
|
|
|
|
|
|
|
|
Ok(updates)
|
|
|
|
}
|
|
|
|
}
|