controller.rs 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. use bytes::Bytes;
  2. use chrono::Utc;
  3. use flowy_collaboration::client_document::default::{initial_delta, initial_read_me};
  4. use flowy_core_data_model::user_default;
  5. use flowy_sync::RevisionWebSocket;
  6. use lazy_static::lazy_static;
  7. use flowy_collaboration::{entities::ws_data::ServerRevisionWSData, folder::FolderPad};
  8. use flowy_document::FlowyDocumentManager;
  9. use std::{collections::HashMap, convert::TryInto, fmt::Formatter, sync::Arc};
  10. use tokio::sync::RwLock as TokioRwLock;
  11. use crate::{
  12. dart_notification::{send_dart_notification, WorkspaceNotification},
  13. entities::workspace::RepeatedWorkspace,
  14. errors::FlowyResult,
  15. module::{FolderCouldServiceV1, WorkspaceDatabase, WorkspaceUser},
  16. services::{
  17. folder_editor::FolderEditor, persistence::FolderPersistence, set_current_workspace, AppController,
  18. TrashController, ViewController, WorkspaceController,
  19. },
  20. };
  21. lazy_static! {
  22. static ref INIT_FOLDER_FLAG: TokioRwLock<HashMap<String, bool>> = TokioRwLock::new(HashMap::new());
  23. }
  24. const FOLDER_ID: &str = "folder";
  25. const FOLDER_ID_SPLIT: &str = ":";
  26. #[derive(Clone)]
  27. pub struct FolderId(String);
  28. impl FolderId {
  29. pub fn new(user_id: &str) -> Self {
  30. Self(format!("{}{}{}", user_id, FOLDER_ID_SPLIT, FOLDER_ID))
  31. }
  32. }
  33. impl std::fmt::Display for FolderId {
  34. fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
  35. f.write_str(FOLDER_ID)
  36. }
  37. }
  38. impl std::fmt::Debug for FolderId {
  39. fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
  40. f.write_str(FOLDER_ID)
  41. }
  42. }
  43. impl AsRef<str> for FolderId {
  44. fn as_ref(&self) -> &str {
  45. &self.0
  46. }
  47. }
  48. pub struct FolderManager {
  49. pub user: Arc<dyn WorkspaceUser>,
  50. pub(crate) cloud_service: Arc<dyn FolderCouldServiceV1>,
  51. pub(crate) persistence: Arc<FolderPersistence>,
  52. pub(crate) workspace_controller: Arc<WorkspaceController>,
  53. pub(crate) app_controller: Arc<AppController>,
  54. pub(crate) view_controller: Arc<ViewController>,
  55. pub(crate) trash_controller: Arc<TrashController>,
  56. web_socket: Arc<dyn RevisionWebSocket>,
  57. folder_editor: Arc<TokioRwLock<Option<Arc<FolderEditor>>>>,
  58. }
  59. impl FolderManager {
  60. pub async fn new(
  61. user: Arc<dyn WorkspaceUser>,
  62. cloud_service: Arc<dyn FolderCouldServiceV1>,
  63. database: Arc<dyn WorkspaceDatabase>,
  64. document_manager: Arc<FlowyDocumentManager>,
  65. web_socket: Arc<dyn RevisionWebSocket>,
  66. ) -> Self {
  67. let folder_editor = Arc::new(TokioRwLock::new(None));
  68. let persistence = Arc::new(FolderPersistence::new(database.clone(), folder_editor.clone()));
  69. let trash_controller = Arc::new(TrashController::new(
  70. persistence.clone(),
  71. cloud_service.clone(),
  72. user.clone(),
  73. ));
  74. let view_controller = Arc::new(ViewController::new(
  75. user.clone(),
  76. persistence.clone(),
  77. cloud_service.clone(),
  78. trash_controller.clone(),
  79. document_manager,
  80. ));
  81. let app_controller = Arc::new(AppController::new(
  82. user.clone(),
  83. persistence.clone(),
  84. trash_controller.clone(),
  85. cloud_service.clone(),
  86. ));
  87. let workspace_controller = Arc::new(WorkspaceController::new(
  88. user.clone(),
  89. persistence.clone(),
  90. trash_controller.clone(),
  91. cloud_service.clone(),
  92. ));
  93. Self {
  94. user,
  95. cloud_service,
  96. persistence,
  97. workspace_controller,
  98. app_controller,
  99. view_controller,
  100. trash_controller,
  101. web_socket,
  102. folder_editor,
  103. }
  104. }
  105. // pub fn network_state_changed(&self, new_type: NetworkType) {
  106. // match new_type {
  107. // NetworkType::UnknownNetworkType => {},
  108. // NetworkType::Wifi => {},
  109. // NetworkType::Cell => {},
  110. // NetworkType::Ethernet => {},
  111. // }
  112. // }
  113. pub async fn did_receive_ws_data(&self, data: Bytes) {
  114. let result: Result<ServerRevisionWSData, protobuf::ProtobufError> = data.try_into();
  115. match result {
  116. Ok(data) => match self.folder_editor.read().await.clone() {
  117. None => {}
  118. Some(editor) => match editor.receive_ws_data(data).await {
  119. Ok(_) => {}
  120. Err(e) => tracing::error!("Folder receive data error: {:?}", e),
  121. },
  122. },
  123. Err(e) => {
  124. tracing::error!("Folder ws data parser failed: {:?}", e);
  125. }
  126. }
  127. }
  128. pub async fn initialize(&self, user_id: &str, token: &str) -> FlowyResult<()> {
  129. let mut write_guard = INIT_FOLDER_FLAG.write().await;
  130. if let Some(is_init) = write_guard.get(user_id) {
  131. if *is_init {
  132. return Ok(());
  133. }
  134. }
  135. let folder_id = FolderId::new(user_id);
  136. let _ = self.persistence.initialize(user_id, &folder_id).await?;
  137. let pool = self.persistence.db_pool()?;
  138. let folder_editor = FolderEditor::new(user_id, &folder_id, token, pool, self.web_socket.clone()).await?;
  139. *self.folder_editor.write().await = Some(Arc::new(folder_editor));
  140. let _ = self.app_controller.initialize()?;
  141. let _ = self.view_controller.initialize()?;
  142. write_guard.insert(user_id.to_owned(), true);
  143. Ok(())
  144. }
  145. pub async fn initialize_with_new_user(&self, user_id: &str, token: &str) -> FlowyResult<()> {
  146. DefaultFolderBuilder::build(token, user_id, self.persistence.clone(), self.view_controller.clone()).await?;
  147. self.initialize(user_id, token).await
  148. }
  149. pub async fn clear(&self) {
  150. *self.folder_editor.write().await = None;
  151. }
  152. }
  153. struct DefaultFolderBuilder();
  154. impl DefaultFolderBuilder {
  155. async fn build(
  156. token: &str,
  157. user_id: &str,
  158. persistence: Arc<FolderPersistence>,
  159. view_controller: Arc<ViewController>,
  160. ) -> FlowyResult<()> {
  161. log::debug!("Create user default workspace");
  162. let time = Utc::now();
  163. let workspace = user_default::create_default_workspace(time);
  164. set_current_workspace(&workspace.id);
  165. for app in workspace.apps.iter() {
  166. for (index, view) in app.belongings.iter().enumerate() {
  167. let view_data = if index == 0 {
  168. initial_read_me().to_json()
  169. } else {
  170. initial_delta().to_json()
  171. };
  172. view_controller.set_latest_view(view);
  173. let _ = view_controller
  174. .create_view_document_content(&view.id, view_data)
  175. .await?;
  176. }
  177. }
  178. let folder = FolderPad::new(vec![workspace.clone()], vec![])?;
  179. let folder_id = FolderId::new(user_id);
  180. let _ = persistence.save_folder(user_id, &folder_id, folder).await?;
  181. let repeated_workspace = RepeatedWorkspace { items: vec![workspace] };
  182. send_dart_notification(token, WorkspaceNotification::UserCreateWorkspace)
  183. .payload(repeated_workspace)
  184. .send();
  185. Ok(())
  186. }
  187. }
  188. #[cfg(feature = "flowy_unit_test")]
  189. impl FolderManager {
  190. pub async fn folder_editor(&self) -> Arc<FolderEditor> {
  191. self.folder_editor.read().await.clone().unwrap()
  192. }
  193. }