use crate::local_server::persistence::LocalDocumentCloudPersistence; use async_stream::stream; use bytes::Bytes; use flowy_collaboration::{ client_document::default::initial_delta_string, entities::{ doc::{CreateDocParams, DocumentId, DocumentInfo, ResetDocumentParams}, ws_data::{ClientRevisionWSData, ClientRevisionWSDataType}, }, errors::CollaborateError, protobuf::ClientRevisionWSData as ClientRevisionWSDataPB, server_document::ServerDocumentManager, synchronizer::{RevisionSyncResponse, RevisionUser}, }; use flowy_core::module::{FolderCouldServiceV1, FolderCouldServiceV2}; use flowy_error::{internal_error, FlowyError}; use futures_util::stream::StreamExt; use lib_ws::{WSModule, WebSocketRawMessage}; use parking_lot::RwLock; use std::{ convert::{TryFrom, TryInto}, fmt::Debug, sync::Arc, }; use tokio::sync::{broadcast, mpsc, mpsc::UnboundedSender}; pub struct LocalServer { doc_manager: Arc, stop_tx: RwLock>>, client_ws_sender: mpsc::UnboundedSender, client_ws_receiver: broadcast::Sender, } impl LocalServer { pub fn new( client_ws_sender: mpsc::UnboundedSender, client_ws_receiver: broadcast::Sender, ) -> Self { let persistence = Arc::new(LocalDocumentCloudPersistence::default()); let doc_manager = Arc::new(ServerDocumentManager::new(persistence)); let stop_tx = RwLock::new(None); LocalServer { doc_manager, stop_tx, client_ws_sender, client_ws_receiver, } } pub async fn stop(&self) { if let Some(stop_tx) = self.stop_tx.read().clone() { let _ = stop_tx.send(()).await; } } pub fn run(&self) { let (stop_tx, stop_rx) = mpsc::channel(1); *self.stop_tx.write() = Some(stop_tx); let runner = LocalWebSocketRunner { doc_manager: self.doc_manager.clone(), stop_rx: Some(stop_rx), client_ws_sender: self.client_ws_sender.clone(), client_ws_receiver: Some(self.client_ws_receiver.subscribe()), }; tokio::spawn(runner.run()); } } struct LocalWebSocketRunner { doc_manager: Arc, stop_rx: Option>, client_ws_sender: mpsc::UnboundedSender, client_ws_receiver: Option>, } impl LocalWebSocketRunner { pub async fn run(mut self) { let mut stop_rx = self.stop_rx.take().expect("Only run once"); let mut client_ws_receiver = self.client_ws_receiver.take().expect("Only run once"); let stream = stream! { loop { tokio::select! { result = client_ws_receiver.recv() => { match result { Ok(msg) => yield msg, Err(_e) => {}, } }, _ = stop_rx.recv() => { tracing::trace!("[LocalWebSocketRunner] stop"); break }, }; } }; stream .for_each(|message| async { match self.handle_message(message).await { Ok(_) => {}, Err(e) => tracing::error!("[LocalWebSocketRunner]: {}", e), } }) .await; } async fn handle_message(&self, message: WebSocketRawMessage) -> Result<(), FlowyError> { let bytes = Bytes::from(message.data); let client_data = ClientRevisionWSData::try_from(bytes).map_err(internal_error)?; let _ = self.handle_client_data(client_data, "".to_owned()).await?; Ok(()) } pub async fn handle_client_data( &self, client_data: ClientRevisionWSData, user_id: String, ) -> Result<(), CollaborateError> { tracing::trace!( "[LocalDocumentServer] receive: {}:{}-{:?} ", client_data.object_id, client_data.id(), client_data.ty, ); let client_ws_sender = self.client_ws_sender.clone(); let user = Arc::new(LocalRevisionUser { user_id, client_ws_sender, }); let ty = client_data.ty.clone(); let document_client_data: ClientRevisionWSDataPB = client_data.try_into().unwrap(); match ty { ClientRevisionWSDataType::ClientPushRev => { let _ = self .doc_manager .handle_client_revisions(user, document_client_data) .await?; }, ClientRevisionWSDataType::ClientPing => { let _ = self.doc_manager.handle_client_ping(user, document_client_data).await?; }, } Ok(()) } } #[derive(Debug)] struct LocalRevisionUser { user_id: String, client_ws_sender: mpsc::UnboundedSender, } impl RevisionUser for LocalRevisionUser { fn user_id(&self) -> String { self.user_id.clone() } fn receive(&self, resp: RevisionSyncResponse) { let sender = self.client_ws_sender.clone(); let send_fn = |sender: UnboundedSender, msg: WebSocketRawMessage| match sender.send(msg) { Ok(_) => {}, Err(e) => { tracing::error!("LocalDocumentUser send message failed: {}", e); }, }; tokio::spawn(async move { match resp { RevisionSyncResponse::Pull(data) => { let bytes: Bytes = data.try_into().unwrap(); let msg = WebSocketRawMessage { module: WSModule::Doc, data: bytes.to_vec(), }; send_fn(sender, msg); }, RevisionSyncResponse::Push(data) => { let bytes: Bytes = data.try_into().unwrap(); let msg = WebSocketRawMessage { module: WSModule::Doc, data: bytes.to_vec(), }; send_fn(sender, msg); }, RevisionSyncResponse::Ack(data) => { let bytes: Bytes = data.try_into().unwrap(); let msg = WebSocketRawMessage { module: WSModule::Doc, data: bytes.to_vec(), }; send_fn(sender, msg); }, } }); } } use flowy_core_data_model::entities::{ app::{App, AppId, CreateAppParams, RepeatedApp, UpdateAppParams}, trash::{RepeatedTrash, RepeatedTrashId}, view::{CreateViewParams, RepeatedView, RepeatedViewId, UpdateViewParams, View, ViewId}, workspace::{CreateWorkspaceParams, RepeatedWorkspace, UpdateWorkspaceParams, Workspace, WorkspaceId}, }; use flowy_document::DocumentCloudService; use flowy_user::module::UserCloudService; use flowy_user_data_model::entities::{ SignInParams, SignInResponse, SignUpParams, SignUpResponse, UpdateUserParams, UserProfile, }; use lib_infra::{future::FutureResult, timestamp, uuid_string}; impl FolderCouldServiceV2 for LocalServer {} impl FolderCouldServiceV1 for LocalServer { fn init(&self) {} fn create_workspace(&self, _token: &str, params: CreateWorkspaceParams) -> FutureResult { let time = timestamp(); let workspace = Workspace { id: uuid_string(), name: params.name, desc: params.desc, apps: RepeatedApp::default(), modified_time: time, create_time: time, }; FutureResult::new(async { Ok(workspace) }) } fn read_workspace(&self, _token: &str, _params: WorkspaceId) -> FutureResult { FutureResult::new(async { let repeated_workspace = RepeatedWorkspace { items: vec![] }; Ok(repeated_workspace) }) } fn update_workspace(&self, _token: &str, _params: UpdateWorkspaceParams) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn delete_workspace(&self, _token: &str, _params: WorkspaceId) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn create_view(&self, _token: &str, params: CreateViewParams) -> FutureResult { let time = timestamp(); let view = View { id: params.view_id, belong_to_id: params.belong_to_id, name: params.name, desc: params.desc, view_type: params.view_type, version: 0, belongings: RepeatedView::default(), modified_time: time, create_time: time, }; FutureResult::new(async { Ok(view) }) } fn read_view(&self, _token: &str, _params: ViewId) -> FutureResult, FlowyError> { FutureResult::new(async { Ok(None) }) } fn delete_view(&self, _token: &str, _params: RepeatedViewId) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn update_view(&self, _token: &str, _params: UpdateViewParams) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn create_app(&self, _token: &str, params: CreateAppParams) -> FutureResult { let time = timestamp(); let app = App { id: uuid_string(), workspace_id: params.workspace_id, name: params.name, desc: params.desc, belongings: RepeatedView::default(), version: 0, modified_time: time, create_time: time, }; FutureResult::new(async { Ok(app) }) } fn read_app(&self, _token: &str, _params: AppId) -> FutureResult, FlowyError> { FutureResult::new(async { Ok(None) }) } fn update_app(&self, _token: &str, _params: UpdateAppParams) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn delete_app(&self, _token: &str, _params: AppId) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn create_trash(&self, _token: &str, _params: RepeatedTrashId) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn delete_trash(&self, _token: &str, _params: RepeatedTrashId) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn read_trash(&self, _token: &str) -> FutureResult { FutureResult::new(async { let repeated_trash = RepeatedTrash { items: vec![] }; Ok(repeated_trash) }) } } impl UserCloudService for LocalServer { fn sign_up(&self, params: SignUpParams) -> FutureResult { let uid = uuid_string(); FutureResult::new(async move { Ok(SignUpResponse { user_id: uid.clone(), name: params.name, email: params.email, token: uid, }) }) } fn sign_in(&self, params: SignInParams) -> FutureResult { let user_id = uuid_string(); FutureResult::new(async { Ok(SignInResponse { user_id: user_id.clone(), name: params.name, email: params.email, token: user_id, }) }) } fn sign_out(&self, _token: &str) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn update_user(&self, _token: &str, _params: UpdateUserParams) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn get_user(&self, _token: &str) -> FutureResult { FutureResult::new(async { Ok(UserProfile::default()) }) } fn ws_addr(&self) -> String { "ws://localhost:8000/ws/".to_owned() } } impl DocumentCloudService for LocalServer { fn create_document(&self, _token: &str, _params: CreateDocParams) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } fn read_document(&self, _token: &str, params: DocumentId) -> FutureResult, FlowyError> { let doc = DocumentInfo { doc_id: params.doc_id, text: initial_delta_string(), rev_id: 0, base_rev_id: 0, }; FutureResult::new(async { Ok(Some(doc)) }) } fn update_document(&self, _token: &str, _params: ResetDocumentParams) -> FutureResult<(), FlowyError> { FutureResult::new(async { Ok(()) }) } }