use std::collections::HashSet; use std::sync::{Arc, Weak}; use collab::core::collab_state::SyncState; use collab_folder::{ Folder, TrashChange, TrashChangeReceiver, View, ViewChange, ViewChangeReceiver, }; use tokio_stream::wrappers::WatchStream; use tokio_stream::StreamExt; use tracing::{event, Level}; use lib_dispatch::prelude::af_spawn; use crate::entities::{ view_pb_with_child_views, view_pb_without_child_views, ChildViewUpdatePB, FolderSnapshotStatePB, FolderSyncStatePB, RepeatedTrashPB, RepeatedViewPB, ViewPB, }; use crate::manager::{get_workspace_view_pbs, MutexFolder}; use crate::notification::{send_notification, FolderNotification}; /// Listen on the [ViewChange] after create/delete/update events happened pub(crate) fn subscribe_folder_view_changed( mut rx: ViewChangeReceiver, weak_mutex_folder: &Weak, ) { let weak_mutex_folder = weak_mutex_folder.clone(); af_spawn(async move { while let Ok(value) = rx.recv().await { if let Some(folder) = weak_mutex_folder.upgrade() { tracing::trace!("Did receive view change: {:?}", value); match value { ViewChange::DidCreateView { view } => { notify_child_views_changed( view_pb_without_child_views(Arc::new(view.clone())), ChildViewChangeReason::DidCreateView, ); notify_parent_view_did_change(folder.clone(), vec![view.parent_view_id]); }, ViewChange::DidDeleteView { views } => { for view in views { notify_child_views_changed( view_pb_without_child_views(view), ChildViewChangeReason::DidDeleteView, ); } }, ViewChange::DidUpdate { view } => { notify_view_did_change(view.clone()); notify_child_views_changed( view_pb_without_child_views(Arc::new(view.clone())), ChildViewChangeReason::DidUpdateView, ); notify_parent_view_did_change(folder.clone(), vec![view.parent_view_id.clone()]); }, }; } } }); } pub(crate) fn subscribe_folder_snapshot_state_changed( workspace_id: String, weak_mutex_folder: &Weak, ) { let weak_mutex_folder = weak_mutex_folder.clone(); af_spawn(async move { if let Some(mutex_folder) = weak_mutex_folder.upgrade() { let stream = mutex_folder .lock() .as_ref() .map(|folder| folder.subscribe_snapshot_state()); if let Some(mut state_stream) = stream { while let Some(snapshot_state) = state_stream.next().await { if let Some(new_snapshot_id) = snapshot_state.snapshot_id() { tracing::debug!("Did create folder remote snapshot: {}", new_snapshot_id); send_notification( &workspace_id, FolderNotification::DidUpdateFolderSnapshotState, ) .payload(FolderSnapshotStatePB { new_snapshot_id }) .send(); } } } } }); } pub(crate) fn subscribe_folder_sync_state_changed( workspace_id: String, mut folder_sync_state_rx: WatchStream, _weak_mutex_folder: &Weak, ) { af_spawn(async move { while let Some(state) = folder_sync_state_rx.next().await { send_notification(&workspace_id, FolderNotification::DidUpdateFolderSyncUpdate) .payload(FolderSyncStatePB::from(state)) .send(); } }); } /// Listen on the [TrashChange]s and notify the frontend some views were changed. pub(crate) fn subscribe_folder_trash_changed( mut rx: TrashChangeReceiver, weak_mutex_folder: &Weak, ) { let weak_mutex_folder = weak_mutex_folder.clone(); af_spawn(async move { while let Ok(value) = rx.recv().await { if let Some(folder) = weak_mutex_folder.upgrade() { let mut unique_ids = HashSet::new(); tracing::trace!("Did receive trash change: {:?}", value); let ids = match value { TrashChange::DidCreateTrash { ids } => ids, TrashChange::DidDeleteTrash { ids } => ids, }; if let Some(folder) = folder.lock().as_ref() { let views = folder.views.get_views(&ids); for view in views { unique_ids.insert(view.parent_view_id.clone()); } let repeated_trash: RepeatedTrashPB = folder.get_all_trash().into(); send_notification("trash", FolderNotification::DidUpdateTrash) .payload(repeated_trash) .send(); } let parent_view_ids = unique_ids.into_iter().collect(); notify_parent_view_did_change(folder.clone(), parent_view_ids); } } }); } /// Notify the the list of parent view ids that its child views were changed. #[tracing::instrument(level = "debug", skip(folder, parent_view_ids))] pub(crate) fn notify_parent_view_did_change>( folder: Arc, parent_view_ids: Vec, ) -> Option<()> { let folder = folder.lock(); let folder = folder.as_ref()?; let workspace_id = folder.get_workspace_id(); let trash_ids = folder .get_all_trash() .into_iter() .map(|trash| trash.id) .collect::>(); for parent_view_id in parent_view_ids { let parent_view_id = parent_view_id.as_ref(); // if the view's parent id equal to workspace id. Then it will fetch the current // workspace views. Because the the workspace is not a view stored in the views map. if parent_view_id == workspace_id { notify_did_update_workspace(&workspace_id, folder) } else { // Parent view can contain a list of child views. Currently, only get the first level // child views. let parent_view = folder.views.get_view(parent_view_id)?; let mut child_views = folder.views.get_views_belong_to(parent_view_id); child_views.retain(|view| !trash_ids.contains(&view.id)); event!(Level::DEBUG, child_views_count = child_views.len()); // Post the notification let parent_view_pb = view_pb_with_child_views(parent_view, child_views); send_notification(parent_view_id, FolderNotification::DidUpdateView) .payload(parent_view_pb) .send(); } } None } pub(crate) fn notify_did_update_workspace(workspace_id: &str, folder: &Folder) { let repeated_view: RepeatedViewPB = get_workspace_view_pbs(workspace_id, folder).into(); tracing::trace!("Did update workspace views: {:?}", repeated_view); send_notification(workspace_id, FolderNotification::DidUpdateWorkspaceViews) .payload(repeated_view) .send(); } fn notify_view_did_change(view: View) -> Option<()> { let view_pb = view_pb_without_child_views(Arc::new(view.clone())); send_notification(&view.id, FolderNotification::DidUpdateView) .payload(view_pb) .send(); None } pub enum ChildViewChangeReason { DidCreateView, DidDeleteView, DidUpdateView, } /// Notify the the list of parent view ids that its child views were changed. #[tracing::instrument(level = "debug", skip_all)] pub(crate) fn notify_child_views_changed(view_pb: ViewPB, reason: ChildViewChangeReason) { let parent_view_id = view_pb.parent_view_id.clone(); let mut payload = ChildViewUpdatePB { parent_view_id: view_pb.parent_view_id.clone(), ..Default::default() }; match reason { ChildViewChangeReason::DidCreateView => { payload.create_child_views.push(view_pb); }, ChildViewChangeReason::DidDeleteView => { payload.delete_child_views.push(view_pb.id); }, ChildViewChangeReason::DidUpdateView => { payload.update_child_views.push(view_pb); }, } send_notification(&parent_view_id, FolderNotification::DidUpdateChildViews) .payload(payload) .send(); }