manager.rs 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. use crate::entities::view::ViewDataType;
  2. use crate::services::folder_editor::FolderRevisionCompactor;
  3. use crate::{
  4. dart_notification::{send_dart_notification, FolderNotification},
  5. entities::workspace::RepeatedWorkspacePB,
  6. errors::FlowyResult,
  7. event_map::{FolderCouldServiceV1, WorkspaceDatabase, WorkspaceUser},
  8. services::{
  9. folder_editor::FolderEditor, persistence::FolderPersistence, set_current_workspace, AppController,
  10. TrashController, ViewController, WorkspaceController,
  11. },
  12. };
  13. use bytes::Bytes;
  14. use flowy_error::FlowyError;
  15. use flowy_folder_data_model::user_default;
  16. use flowy_revision::disk::SQLiteTextBlockRevisionPersistence;
  17. use flowy_revision::{RevisionManager, RevisionPersistence, RevisionWebSocket, SQLiteRevisionSnapshotPersistence};
  18. use flowy_sync::client_document::default::{initial_quill_delta_string, initial_read_me};
  19. use flowy_sync::{client_folder::FolderPad, entities::ws_data::ServerRevisionWSData};
  20. use lazy_static::lazy_static;
  21. use lib_infra::future::FutureResult;
  22. use std::{collections::HashMap, convert::TryInto, fmt::Formatter, sync::Arc};
  23. use tokio::sync::RwLock as TokioRwLock;
  24. lazy_static! {
  25. static ref INIT_FOLDER_FLAG: TokioRwLock<HashMap<String, bool>> = TokioRwLock::new(HashMap::new());
  26. }
  27. const FOLDER_ID: &str = "folder";
  28. const FOLDER_ID_SPLIT: &str = ":";
  29. #[derive(Clone)]
  30. pub struct FolderId(String);
  31. impl FolderId {
  32. pub fn new(user_id: &str) -> Self {
  33. Self(format!("{}{}{}", user_id, FOLDER_ID_SPLIT, FOLDER_ID))
  34. }
  35. }
  36. impl std::fmt::Display for FolderId {
  37. fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
  38. f.write_str(FOLDER_ID)
  39. }
  40. }
  41. impl std::fmt::Debug for FolderId {
  42. fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
  43. f.write_str(FOLDER_ID)
  44. }
  45. }
  46. impl AsRef<str> for FolderId {
  47. fn as_ref(&self) -> &str {
  48. &self.0
  49. }
  50. }
  51. pub struct FolderManager {
  52. pub user: Arc<dyn WorkspaceUser>,
  53. pub(crate) cloud_service: Arc<dyn FolderCouldServiceV1>,
  54. pub(crate) persistence: Arc<FolderPersistence>,
  55. pub(crate) workspace_controller: Arc<WorkspaceController>,
  56. pub(crate) app_controller: Arc<AppController>,
  57. pub(crate) view_controller: Arc<ViewController>,
  58. pub(crate) trash_controller: Arc<TrashController>,
  59. web_socket: Arc<dyn RevisionWebSocket>,
  60. folder_editor: Arc<TokioRwLock<Option<Arc<FolderEditor>>>>,
  61. data_processors: ViewDataProcessorMap,
  62. }
  63. impl FolderManager {
  64. pub async fn new(
  65. user: Arc<dyn WorkspaceUser>,
  66. cloud_service: Arc<dyn FolderCouldServiceV1>,
  67. database: Arc<dyn WorkspaceDatabase>,
  68. data_processors: ViewDataProcessorMap,
  69. web_socket: Arc<dyn RevisionWebSocket>,
  70. ) -> Self {
  71. if let Ok(user_id) = user.user_id() {
  72. // Reset the flag if the folder manager gets initialized, otherwise,
  73. // the folder_editor will not be initialized after flutter hot reload.
  74. INIT_FOLDER_FLAG.write().await.insert(user_id.to_owned(), false);
  75. }
  76. let folder_editor = Arc::new(TokioRwLock::new(None));
  77. let persistence = Arc::new(FolderPersistence::new(database.clone(), folder_editor.clone()));
  78. let trash_controller = Arc::new(TrashController::new(
  79. persistence.clone(),
  80. cloud_service.clone(),
  81. user.clone(),
  82. ));
  83. let view_controller = Arc::new(ViewController::new(
  84. user.clone(),
  85. persistence.clone(),
  86. cloud_service.clone(),
  87. trash_controller.clone(),
  88. data_processors.clone(),
  89. ));
  90. let app_controller = Arc::new(AppController::new(
  91. user.clone(),
  92. persistence.clone(),
  93. trash_controller.clone(),
  94. cloud_service.clone(),
  95. ));
  96. let workspace_controller = Arc::new(WorkspaceController::new(
  97. user.clone(),
  98. persistence.clone(),
  99. trash_controller.clone(),
  100. cloud_service.clone(),
  101. ));
  102. Self {
  103. user,
  104. cloud_service,
  105. persistence,
  106. workspace_controller,
  107. app_controller,
  108. view_controller,
  109. trash_controller,
  110. web_socket,
  111. folder_editor,
  112. data_processors,
  113. }
  114. }
  115. // pub fn network_state_changed(&self, new_type: NetworkType) {
  116. // match new_type {
  117. // NetworkType::UnknownNetworkType => {},
  118. // NetworkType::Wifi => {},
  119. // NetworkType::Cell => {},
  120. // NetworkType::Ethernet => {},
  121. // }
  122. // }
  123. pub async fn did_receive_ws_data(&self, data: Bytes) {
  124. let result: Result<ServerRevisionWSData, protobuf::ProtobufError> = data.try_into();
  125. match result {
  126. Ok(data) => match self.folder_editor.read().await.clone() {
  127. None => {}
  128. Some(editor) => match editor.receive_ws_data(data).await {
  129. Ok(_) => {}
  130. Err(e) => tracing::error!("Folder receive data error: {:?}", e),
  131. },
  132. },
  133. Err(e) => {
  134. tracing::error!("Folder ws data parser failed: {:?}", e);
  135. }
  136. }
  137. }
  138. #[tracing::instrument(level = "trace", skip(self), err)]
  139. pub async fn initialize(&self, user_id: &str, token: &str) -> FlowyResult<()> {
  140. let mut write_guard = INIT_FOLDER_FLAG.write().await;
  141. if let Some(is_init) = write_guard.get(user_id) {
  142. if *is_init {
  143. return Ok(());
  144. }
  145. }
  146. tracing::debug!("Initialize folder editor");
  147. let folder_id = FolderId::new(user_id);
  148. let _ = self.persistence.initialize(user_id, &folder_id).await?;
  149. let pool = self.persistence.db_pool()?;
  150. let object_id = folder_id.as_ref();
  151. let disk_cache = SQLiteTextBlockRevisionPersistence::new(user_id, pool.clone());
  152. let rev_persistence = RevisionPersistence::new(user_id, object_id, disk_cache);
  153. let rev_compactor = FolderRevisionCompactor();
  154. // let history_persistence = SQLiteRevisionHistoryPersistence::new(object_id, pool.clone());
  155. let snapshot_persistence = SQLiteRevisionSnapshotPersistence::new(object_id, pool);
  156. let rev_manager = RevisionManager::new(
  157. user_id,
  158. folder_id.as_ref(),
  159. rev_persistence,
  160. rev_compactor,
  161. // history_persistence,
  162. snapshot_persistence,
  163. );
  164. let folder_editor = FolderEditor::new(user_id, &folder_id, token, rev_manager, self.web_socket.clone()).await?;
  165. *self.folder_editor.write().await = Some(Arc::new(folder_editor));
  166. let _ = self.app_controller.initialize()?;
  167. let _ = self.view_controller.initialize()?;
  168. self.data_processors.iter().for_each(|(_, processor)| {
  169. processor.initialize();
  170. });
  171. write_guard.insert(user_id.to_owned(), true);
  172. Ok(())
  173. }
  174. pub async fn initialize_with_new_user(&self, user_id: &str, token: &str) -> FlowyResult<()> {
  175. DefaultFolderBuilder::build(token, user_id, self.persistence.clone(), self.view_controller.clone()).await?;
  176. self.initialize(user_id, token).await
  177. }
  178. pub async fn clear(&self) {
  179. *self.folder_editor.write().await = None;
  180. }
  181. }
  182. struct DefaultFolderBuilder();
  183. impl DefaultFolderBuilder {
  184. async fn build(
  185. token: &str,
  186. user_id: &str,
  187. persistence: Arc<FolderPersistence>,
  188. view_controller: Arc<ViewController>,
  189. ) -> FlowyResult<()> {
  190. log::debug!("Create user default workspace");
  191. let workspace_rev = user_default::create_default_workspace();
  192. set_current_workspace(&workspace_rev.id);
  193. for app in workspace_rev.apps.iter() {
  194. for (index, view) in app.belongings.iter().enumerate() {
  195. let view_data = if index == 0 {
  196. initial_read_me().json_str()
  197. } else {
  198. initial_quill_delta_string()
  199. };
  200. let _ = view_controller.set_latest_view(&view.id);
  201. let _ = view_controller
  202. .create_view(&view.id, ViewDataType::TextBlock, Bytes::from(view_data))
  203. .await?;
  204. }
  205. }
  206. let folder = FolderPad::new(vec![workspace_rev.clone()], vec![])?;
  207. let folder_id = FolderId::new(user_id);
  208. let _ = persistence.save_folder(user_id, &folder_id, folder).await?;
  209. let repeated_workspace = RepeatedWorkspacePB {
  210. items: vec![workspace_rev.into()],
  211. };
  212. send_dart_notification(token, FolderNotification::UserCreateWorkspace)
  213. .payload(repeated_workspace)
  214. .send();
  215. Ok(())
  216. }
  217. }
  218. #[cfg(feature = "flowy_unit_test")]
  219. impl FolderManager {
  220. pub async fn folder_editor(&self) -> Arc<FolderEditor> {
  221. self.folder_editor.read().await.clone().unwrap()
  222. }
  223. }
  224. pub trait ViewDataProcessor {
  225. fn initialize(&self) -> FutureResult<(), FlowyError>;
  226. fn create_container(&self, user_id: &str, view_id: &str, delta_data: Bytes) -> FutureResult<(), FlowyError>;
  227. fn delete_container(&self, view_id: &str) -> FutureResult<(), FlowyError>;
  228. fn close_container(&self, view_id: &str) -> FutureResult<(), FlowyError>;
  229. fn get_delta_data(&self, view_id: &str) -> FutureResult<Bytes, FlowyError>;
  230. fn create_default_view(&self, user_id: &str, view_id: &str) -> FutureResult<Bytes, FlowyError>;
  231. fn create_view_from_delta_data(
  232. &self,
  233. user_id: &str,
  234. view_id: &str,
  235. data: Vec<u8>,
  236. ) -> FutureResult<Bytes, FlowyError>;
  237. fn data_type(&self) -> ViewDataType;
  238. }
  239. pub type ViewDataProcessorMap = Arc<HashMap<ViewDataType, Arc<dyn ViewDataProcessor + Send + Sync>>>;