2023-12-17 19:14:05 +00:00
|
|
|
use std::collections::HashSet;
|
|
|
|
use std::sync::{Arc, Weak};
|
|
|
|
|
|
|
|
use collab::core::collab_state::SyncState;
|
2023-12-20 02:08:35 +00:00
|
|
|
use collab_folder::{
|
|
|
|
Folder, TrashChange, TrashChangeReceiver, View, ViewChange, ViewChangeReceiver,
|
|
|
|
};
|
2023-12-17 19:14:05 +00:00
|
|
|
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<MutexFolder>,
|
|
|
|
) {
|
|
|
|
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 } => {
|
2023-12-20 02:08:35 +00:00
|
|
|
notify_view_did_change(view.clone());
|
2023-12-17 19:14:05 +00:00
|
|
|
notify_child_views_changed(
|
|
|
|
view_pb_without_child_views(Arc::new(view.clone())),
|
|
|
|
ChildViewChangeReason::DidUpdateView,
|
|
|
|
);
|
2023-12-20 02:08:35 +00:00
|
|
|
notify_parent_view_did_change(folder.clone(), vec![view.parent_view_id.clone()]);
|
2023-12-17 19:14:05 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn subscribe_folder_snapshot_state_changed(
|
|
|
|
workspace_id: String,
|
|
|
|
weak_mutex_folder: &Weak<MutexFolder>,
|
|
|
|
) {
|
|
|
|
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<SyncState>,
|
|
|
|
_weak_mutex_folder: &Weak<MutexFolder>,
|
|
|
|
) {
|
|
|
|
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<MutexFolder>,
|
|
|
|
) {
|
|
|
|
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<T: AsRef<str>>(
|
|
|
|
folder: Arc<MutexFolder>,
|
|
|
|
parent_view_ids: Vec<T>,
|
|
|
|
) -> 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::<Vec<String>>();
|
|
|
|
|
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2023-12-20 02:08:35 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2023-12-17 19:14:05 +00:00
|
|
|
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();
|
|
|
|
}
|