AppFlowy/frontend/rust-lib/flowy-document2/src/manager.rs

105 lines
3.1 KiB
Rust
Raw Normal View History

2023-04-21 09:02:26 +00:00
use collab::plugin_impl::rocks_disk::RocksDiskPlugin;
use collab::preclude::{Collab, CollabBuilder};
use collab_persistence::kv::rocks_kv::RocksCollabDB;
use flowy_error::{FlowyError, FlowyResult};
use parking_lot::RwLock;
2023-04-21 09:02:26 +00:00
use std::{collections::HashMap, sync::Arc};
use crate::{
document::{Document, DocumentDataWrapper},
entities::{BlockEventPB, DocEventPB},
notification::{send_notification, DocumentNotification},
};
pub trait DocumentUser: Send + Sync {
fn user_id(&self) -> Result<i64, FlowyError>;
fn token(&self) -> Result<String, FlowyError>; // unused now.
2023-04-21 09:02:26 +00:00
fn kv_db(&self) -> Result<Arc<RocksCollabDB>, FlowyError>;
}
pub struct DocumentManager {
documents: Arc<RwLock<HashMap<String, Arc<Document>>>>,
user: Arc<dyn DocumentUser>,
}
// unsafe impl Send for DocumentManager {}
// unsafe impl Sync for DocumentManager {}
impl DocumentManager {
pub fn new(user: Arc<dyn DocumentUser>) -> Self {
Self {
documents: Default::default(),
user,
}
}
pub fn create_document(
&self,
doc_id: String,
data: DocumentDataWrapper,
) -> FlowyResult<Arc<Document>> {
self.get_document(doc_id, Some(data))
}
fn get_document(
&self,
doc_id: String,
data: Option<DocumentDataWrapper>,
) -> FlowyResult<Arc<Document>> {
let collab = self.get_collab_for_doc_id(&doc_id)?;
let document = Arc::new(Document::new(collab)?);
self.documents.write().insert(doc_id, document.clone());
if data.is_some() {
// Here use unwrap() is safe, because we have checked data.is_some() before.
// document
// .lock()
// .create_with_data(data.unwrap().0)
// .map_err(|err| FlowyError::internal().context(err))?;
}
Ok(document)
}
pub fn open_document(&self, doc_id: String) -> FlowyResult<Arc<Document>> {
if let Some(doc) = self.documents.read().get(&doc_id) {
return Ok(doc.clone());
}
let document = self.get_document(doc_id.clone(), None)?;
let clone_doc_id = doc_id.clone();
let _document_data = document
.lock()
.open(move |events, is_remote| {
println!("events: {:?}", events);
println!("is_remote: {:?}", is_remote);
send_notification(&clone_doc_id, DocumentNotification::DidReceiveUpdate)
.payload(DocEventPB {
events: events
.iter()
.map(|event| event.to_owned().into())
.collect::<Vec<BlockEventPB>>(),
is_remote: is_remote.to_owned(),
})
.send();
})
.map_err(|err| FlowyError::internal().context(err))?;
Ok(document)
}
pub fn close_document(&self, doc_id: String) -> FlowyResult<()> {
self.documents.write().remove(&doc_id);
Ok(())
}
2023-04-21 09:02:26 +00:00
fn get_collab_for_doc_id(&self, doc_id: &str) -> Result<Collab, FlowyError> {
let uid = self.user.user_id()?;
let kv_db = self.user.kv_db()?;
let mut collab = CollabBuilder::new(uid, doc_id).build();
let disk_plugin = Arc::new(
2023-04-21 09:02:26 +00:00
RocksDiskPlugin::new(uid, kv_db).map_err(|err| FlowyError::internal().context(err))?,
);
collab.add_plugin(disk_plugin);
collab.initial();
Ok(collab)
}
}