2024-06-05 04:05:51 +02:00
|
|
|
use crate::entities::IndexedWorkspaceIds;
|
2024-04-26 09:44:07 +08:00
|
|
|
use crate::manager::{FolderInitDataSource, FolderManager};
|
|
|
|
use crate::manager_observer::*;
|
|
|
|
use crate::user_default::DefaultFolderBuilder;
|
|
|
|
use collab::core::collab::DataSource;
|
2024-02-08 23:53:05 +08:00
|
|
|
use collab_entity::CollabType;
|
2023-12-18 03:14:05 +08:00
|
|
|
use collab_folder::{Folder, FolderNotify, UserId};
|
2024-01-05 00:05:38 +08:00
|
|
|
use collab_integrate::CollabKVDB;
|
2024-02-08 23:53:05 +08:00
|
|
|
use flowy_error::{FlowyError, FlowyResult};
|
|
|
|
use std::sync::{Arc, Weak};
|
2024-04-26 09:44:07 +08:00
|
|
|
use tokio::task::spawn_blocking;
|
|
|
|
use tracing::{event, info, Level};
|
2023-12-18 03:14:05 +08:00
|
|
|
|
2024-06-05 04:05:51 +02:00
|
|
|
pub const INDEXED_WORKSPACE_KEY: &str = "indexed-workspace-ids";
|
|
|
|
|
2023-12-18 03:14:05 +08:00
|
|
|
impl FolderManager {
|
|
|
|
/// Called immediately after the application launched if the user already sign in/sign up.
|
|
|
|
#[tracing::instrument(level = "info", skip(self, initial_data), err)]
|
|
|
|
pub async fn initialize(
|
|
|
|
&self,
|
|
|
|
uid: i64,
|
|
|
|
workspace_id: &str,
|
|
|
|
initial_data: FolderInitDataSource,
|
|
|
|
) -> FlowyResult<()> {
|
|
|
|
// Update the workspace id
|
|
|
|
event!(
|
|
|
|
Level::INFO,
|
|
|
|
"Init workspace: {} from: {}",
|
|
|
|
workspace_id,
|
|
|
|
initial_data
|
|
|
|
);
|
|
|
|
|
2024-04-26 09:44:07 +08:00
|
|
|
if let Some(old_folder) = self.mutex_folder.write().take() {
|
|
|
|
old_folder.close();
|
|
|
|
info!("remove old folder: {}", old_folder.get_workspace_id());
|
|
|
|
}
|
|
|
|
|
|
|
|
let workspace_id = workspace_id.to_string();
|
2023-12-18 03:14:05 +08:00
|
|
|
// Get the collab db for the user with given user id.
|
|
|
|
let collab_db = self.user.collab_db(uid)?;
|
|
|
|
|
|
|
|
let (view_tx, view_rx) = tokio::sync::broadcast::channel(100);
|
2023-12-26 02:03:42 +08:00
|
|
|
let (section_change_tx, section_change_rx) = tokio::sync::broadcast::channel(100);
|
2023-12-18 03:14:05 +08:00
|
|
|
let folder_notifier = FolderNotify {
|
|
|
|
view_change_tx: view_tx,
|
2023-12-26 02:03:42 +08:00
|
|
|
section_change_tx,
|
2023-12-18 03:14:05 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
let folder = match initial_data {
|
|
|
|
FolderInitDataSource::LocalDisk {
|
|
|
|
create_if_not_exist,
|
|
|
|
} => {
|
2024-02-04 05:50:23 +08:00
|
|
|
let is_exist = self.is_workspace_exist_in_local(uid, &workspace_id).await;
|
2024-02-08 23:53:05 +08:00
|
|
|
// 1. if the folder exists, open it from local disk
|
2023-12-18 03:14:05 +08:00
|
|
|
if is_exist {
|
2024-02-25 07:49:44 +08:00
|
|
|
event!(Level::INFO, "Init folder from local disk");
|
2023-12-18 03:14:05 +08:00
|
|
|
self
|
2024-03-23 09:18:47 +08:00
|
|
|
.make_folder(
|
|
|
|
uid,
|
|
|
|
&workspace_id,
|
|
|
|
collab_db,
|
2024-04-15 14:50:28 +08:00
|
|
|
DataSource::Disk,
|
2024-03-23 09:18:47 +08:00
|
|
|
folder_notifier,
|
|
|
|
)
|
2023-12-18 03:14:05 +08:00
|
|
|
.await?
|
|
|
|
} else if create_if_not_exist {
|
2024-02-08 23:53:05 +08:00
|
|
|
// 2. if the folder doesn't exist and create_if_not_exist is true, create a default folder
|
2023-12-18 03:14:05 +08:00
|
|
|
// Currently, this branch is only used when the server type is supabase. For appflowy cloud,
|
|
|
|
// the default workspace is already created when the user sign up.
|
|
|
|
self
|
|
|
|
.create_default_folder(uid, &workspace_id, collab_db, folder_notifier)
|
|
|
|
.await?
|
|
|
|
} else {
|
2024-02-08 23:53:05 +08:00
|
|
|
// 3. If the folder doesn't exist and create_if_not_exist is false, try to fetch the folder data from cloud/
|
|
|
|
// This will happen user can't fetch the folder data when the user sign in.
|
|
|
|
let doc_state = self
|
|
|
|
.cloud_service
|
|
|
|
.get_folder_doc_state(&workspace_id, uid, CollabType::Folder, &workspace_id)
|
|
|
|
.await?;
|
|
|
|
|
2024-02-25 07:49:44 +08:00
|
|
|
self
|
|
|
|
.make_folder(
|
|
|
|
uid,
|
|
|
|
&workspace_id,
|
|
|
|
collab_db.clone(),
|
2024-04-15 14:50:28 +08:00
|
|
|
DataSource::DocStateV1(doc_state),
|
2024-02-25 07:49:44 +08:00
|
|
|
folder_notifier.clone(),
|
|
|
|
)
|
|
|
|
.await?
|
2023-12-18 03:14:05 +08:00
|
|
|
}
|
|
|
|
},
|
2024-02-08 23:53:05 +08:00
|
|
|
FolderInitDataSource::Cloud(doc_state) => {
|
|
|
|
if doc_state.is_empty() {
|
2023-12-27 11:42:39 +08:00
|
|
|
event!(Level::ERROR, "remote folder data is empty, open from local");
|
2023-12-18 03:14:05 +08:00
|
|
|
self
|
2024-03-23 09:18:47 +08:00
|
|
|
.make_folder(
|
|
|
|
uid,
|
|
|
|
&workspace_id,
|
|
|
|
collab_db,
|
2024-04-15 14:50:28 +08:00
|
|
|
DataSource::Disk,
|
2024-03-23 09:18:47 +08:00
|
|
|
folder_notifier,
|
|
|
|
)
|
2023-12-18 03:14:05 +08:00
|
|
|
.await?
|
|
|
|
} else {
|
2024-02-25 07:49:44 +08:00
|
|
|
event!(Level::INFO, "Restore folder from remote data");
|
|
|
|
self
|
|
|
|
.make_folder(
|
|
|
|
uid,
|
|
|
|
&workspace_id,
|
|
|
|
collab_db.clone(),
|
2024-04-15 14:50:28 +08:00
|
|
|
DataSource::DocStateV1(doc_state),
|
2024-02-25 07:49:44 +08:00
|
|
|
folder_notifier.clone(),
|
|
|
|
)
|
|
|
|
.await?
|
2023-12-18 03:14:05 +08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
let folder_state_rx = folder.subscribe_sync_state();
|
2024-04-12 10:21:41 +02:00
|
|
|
let index_content_rx = folder.subscribe_index_content();
|
|
|
|
self
|
|
|
|
.folder_indexer
|
2024-04-23 15:46:57 +02:00
|
|
|
.set_index_content_receiver(index_content_rx, workspace_id.clone());
|
2024-06-05 04:05:51 +02:00
|
|
|
self.handle_index_folder(workspace_id.clone(), &folder);
|
2024-04-12 10:21:41 +02:00
|
|
|
|
2024-04-23 15:46:57 +02:00
|
|
|
*self.mutex_folder.write() = Some(folder);
|
2023-12-18 03:14:05 +08:00
|
|
|
|
|
|
|
let weak_mutex_folder = Arc::downgrade(&self.mutex_folder);
|
2024-04-26 09:44:07 +08:00
|
|
|
subscribe_folder_sync_state_changed(
|
|
|
|
workspace_id.clone(),
|
|
|
|
folder_state_rx,
|
|
|
|
Arc::downgrade(&self.user),
|
|
|
|
);
|
|
|
|
subscribe_folder_snapshot_state_changed(
|
|
|
|
workspace_id.clone(),
|
|
|
|
&weak_mutex_folder,
|
|
|
|
Arc::downgrade(&self.user),
|
|
|
|
);
|
|
|
|
subscribe_folder_trash_changed(
|
|
|
|
workspace_id.clone(),
|
|
|
|
section_change_rx,
|
|
|
|
&weak_mutex_folder,
|
|
|
|
Arc::downgrade(&self.user),
|
|
|
|
);
|
|
|
|
subscribe_folder_view_changed(
|
|
|
|
workspace_id.clone(),
|
|
|
|
view_rx,
|
|
|
|
&weak_mutex_folder,
|
|
|
|
Arc::downgrade(&self.user),
|
|
|
|
);
|
2024-06-05 04:05:51 +02:00
|
|
|
|
2023-12-18 03:14:05 +08:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2024-02-04 05:50:23 +08:00
|
|
|
async fn is_workspace_exist_in_local(&self, uid: i64, workspace_id: &str) -> bool {
|
|
|
|
if let Ok(weak_collab) = self.user.collab_db(uid) {
|
|
|
|
if let Some(collab_db) = weak_collab.upgrade() {
|
|
|
|
return collab_db.is_exist(uid, workspace_id).await.unwrap_or(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2023-12-18 03:14:05 +08:00
|
|
|
async fn create_default_folder(
|
|
|
|
&self,
|
|
|
|
uid: i64,
|
2023-12-19 04:36:24 +08:00
|
|
|
workspace_id: &str,
|
2024-01-05 00:05:38 +08:00
|
|
|
collab_db: Weak<CollabKVDB>,
|
2023-12-18 03:14:05 +08:00
|
|
|
folder_notifier: FolderNotify,
|
|
|
|
) -> Result<Folder, FlowyError> {
|
2023-12-27 11:42:39 +08:00
|
|
|
event!(
|
|
|
|
Level::INFO,
|
|
|
|
"Create folder:{} with default folder builder",
|
|
|
|
workspace_id
|
|
|
|
);
|
2023-12-18 03:14:05 +08:00
|
|
|
let folder_data =
|
|
|
|
DefaultFolderBuilder::build(uid, workspace_id.to_string(), &self.operation_handlers).await;
|
|
|
|
let collab = self
|
2024-02-25 07:49:44 +08:00
|
|
|
.create_empty_collab(uid, workspace_id, collab_db)
|
2023-12-18 03:14:05 +08:00
|
|
|
.await?;
|
|
|
|
Ok(Folder::create(
|
|
|
|
UserId::from(uid),
|
|
|
|
collab,
|
|
|
|
Some(folder_notifier),
|
|
|
|
folder_data,
|
|
|
|
))
|
|
|
|
}
|
2024-06-05 04:05:51 +02:00
|
|
|
|
|
|
|
fn handle_index_folder(&self, workspace_id: String, folder: &Folder) {
|
|
|
|
let index_all;
|
|
|
|
if self.folder_indexer.is_indexed() {
|
|
|
|
// If indexes already exist, we check if the workspace was
|
|
|
|
// previously indexed, if it wasn't we index all
|
|
|
|
let indexed_ids = self
|
|
|
|
.store_preferences
|
|
|
|
.get_object::<IndexedWorkspaceIds>(INDEXED_WORKSPACE_KEY);
|
|
|
|
if let Some(indexed_ids) = indexed_ids {
|
|
|
|
index_all = !indexed_ids.workspace_ids.contains(&workspace_id.clone());
|
|
|
|
if !index_all {
|
|
|
|
let mut workspace_ids = indexed_ids.workspace_ids.clone();
|
|
|
|
workspace_ids.push(workspace_id.clone());
|
|
|
|
let _ = self
|
|
|
|
.store_preferences
|
|
|
|
.set_object(INDEXED_WORKSPACE_KEY, IndexedWorkspaceIds { workspace_ids });
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
index_all = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If there exists no indexes, we index all views in workspace
|
|
|
|
index_all = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if index_all {
|
|
|
|
let views = folder.views.get_all_views();
|
|
|
|
let folder_indexer = self.folder_indexer.clone();
|
|
|
|
let wid = workspace_id.clone();
|
|
|
|
|
|
|
|
// We spawn a blocking task to index all views in the folder
|
|
|
|
spawn_blocking(move || {
|
|
|
|
folder_indexer.index_all_views(views, wid);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2023-12-18 03:14:05 +08:00
|
|
|
}
|