2023-11-14 09:21:09 +00:00
|
|
|
use std::num::NonZeroUsize;
|
|
|
|
use std::sync::Arc;
|
2023-07-29 01:46:24 +00:00
|
|
|
use std::sync::Weak;
|
2023-05-10 05:27:50 +00:00
|
|
|
|
2023-11-05 06:00:24 +00:00
|
|
|
use collab::core::collab::{CollabRawData, MutexCollab};
|
2023-12-27 03:42:39 +00:00
|
|
|
use collab::core::collab_plugin::EncodedCollabV1;
|
|
|
|
use collab::core::origin::CollabOrigin;
|
|
|
|
use collab::preclude::Collab;
|
2023-06-03 05:55:43 +00:00
|
|
|
use collab_document::blocks::DocumentData;
|
2023-07-05 12:57:09 +00:00
|
|
|
use collab_document::document::Document;
|
2023-11-05 06:00:24 +00:00
|
|
|
use collab_document::document_data::{default_document_collab_data, default_document_data};
|
2023-06-05 01:42:11 +00:00
|
|
|
use collab_document::YrsDocAction;
|
2023-10-10 11:05:55 +00:00
|
|
|
use collab_entity::CollabType;
|
2023-11-14 09:21:09 +00:00
|
|
|
use lru::LruCache;
|
|
|
|
use parking_lot::Mutex;
|
2023-11-05 06:00:24 +00:00
|
|
|
use tracing::{event, instrument};
|
2023-04-13 10:53:51 +00:00
|
|
|
|
2023-12-27 03:42:39 +00:00
|
|
|
use collab_integrate::collab_builder::{AppFlowyCollabBuilder, CollabBuilderConfig};
|
2023-09-17 09:14:34 +00:00
|
|
|
use collab_integrate::RocksCollabDB;
|
2023-07-29 01:46:24 +00:00
|
|
|
use flowy_document_deps::cloud::DocumentCloudService;
|
2023-12-27 03:42:39 +00:00
|
|
|
use flowy_error::{internal_error, ErrorCode, FlowyError, FlowyResult};
|
2023-09-01 14:27:29 +00:00
|
|
|
use flowy_storage::FileStorageService;
|
2023-04-28 06:08:53 +00:00
|
|
|
|
2023-08-05 07:02:05 +00:00
|
|
|
use crate::document::MutexDocument;
|
2023-07-05 12:57:09 +00:00
|
|
|
use crate::entities::DocumentSnapshotPB;
|
2023-10-02 07:12:24 +00:00
|
|
|
use crate::reminder::DocumentReminderAction;
|
2023-04-13 10:53:51 +00:00
|
|
|
|
2023-07-29 01:46:24 +00:00
|
|
|
pub trait DocumentUser: Send + Sync {
|
|
|
|
fn user_id(&self) -> Result<i64, FlowyError>;
|
2023-10-23 03:43:31 +00:00
|
|
|
fn workspace_id(&self) -> Result<String, FlowyError>;
|
2023-07-29 01:46:24 +00:00
|
|
|
fn token(&self) -> Result<Option<String>, FlowyError>; // unused now.
|
|
|
|
fn collab_db(&self, uid: i64) -> Result<Weak<RocksCollabDB>, FlowyError>;
|
|
|
|
}
|
|
|
|
|
2023-04-13 10:53:51 +00:00
|
|
|
pub struct DocumentManager {
|
2023-08-17 15:46:39 +00:00
|
|
|
pub user: Arc<dyn DocumentUser>,
|
2023-05-15 14:16:05 +00:00
|
|
|
collab_builder: Arc<AppFlowyCollabBuilder>,
|
2023-11-14 09:21:09 +00:00
|
|
|
documents: Arc<Mutex<LruCache<String, Arc<MutexDocument>>>>,
|
2023-07-05 12:57:09 +00:00
|
|
|
#[allow(dead_code)]
|
|
|
|
cloud_service: Arc<dyn DocumentCloudService>,
|
2023-09-01 14:27:29 +00:00
|
|
|
storage_service: Weak<dyn FileStorageService>,
|
2023-04-13 10:53:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl DocumentManager {
|
2023-07-05 12:57:09 +00:00
|
|
|
pub fn new(
|
|
|
|
user: Arc<dyn DocumentUser>,
|
|
|
|
collab_builder: Arc<AppFlowyCollabBuilder>,
|
|
|
|
cloud_service: Arc<dyn DocumentCloudService>,
|
2023-09-01 14:27:29 +00:00
|
|
|
storage_service: Weak<dyn FileStorageService>,
|
2023-07-05 12:57:09 +00:00
|
|
|
) -> Self {
|
2023-11-14 09:21:09 +00:00
|
|
|
let documents = Arc::new(Mutex::new(LruCache::new(NonZeroUsize::new(10).unwrap())));
|
2023-04-13 10:53:51 +00:00
|
|
|
Self {
|
|
|
|
user,
|
2023-05-15 14:16:05 +00:00
|
|
|
collab_builder,
|
2023-11-14 09:21:09 +00:00
|
|
|
documents,
|
2023-07-05 12:57:09 +00:00
|
|
|
cloud_service,
|
2023-09-01 14:27:29 +00:00
|
|
|
storage_service,
|
2023-04-13 10:53:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-29 01:46:24 +00:00
|
|
|
pub async fn initialize(&self, _uid: i64, _workspace_id: String) -> FlowyResult<()> {
|
2023-11-14 09:21:09 +00:00
|
|
|
self.documents.lock().clear();
|
2023-07-29 01:46:24 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-10-24 12:11:06 +00:00
|
|
|
#[instrument(
|
|
|
|
name = "document_initialize_with_new_user",
|
|
|
|
level = "debug",
|
|
|
|
skip_all,
|
|
|
|
err
|
|
|
|
)]
|
2023-07-29 01:46:24 +00:00
|
|
|
pub async fn initialize_with_new_user(&self, uid: i64, workspace_id: String) -> FlowyResult<()> {
|
|
|
|
self.initialize(uid, workspace_id).await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
2023-10-02 07:12:24 +00:00
|
|
|
|
|
|
|
pub async fn handle_reminder_action(&self, action: DocumentReminderAction) {
|
|
|
|
match action {
|
|
|
|
DocumentReminderAction::Add { reminder: _ } => {},
|
|
|
|
DocumentReminderAction::Remove { reminder_id: _ } => {},
|
|
|
|
DocumentReminderAction::Update { reminder: _ } => {},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-03 12:43:46 +00:00
|
|
|
/// Create a new document.
|
|
|
|
///
|
|
|
|
/// if the document already exists, return the existing document.
|
|
|
|
/// if the data is None, will create a document with default data.
|
2023-08-28 05:28:24 +00:00
|
|
|
pub async fn create_document(
|
2023-06-03 12:43:46 +00:00
|
|
|
&self,
|
2023-08-17 15:46:39 +00:00
|
|
|
uid: i64,
|
2023-06-17 06:25:30 +00:00
|
|
|
doc_id: &str,
|
2023-06-03 12:43:46 +00:00
|
|
|
data: Option<DocumentData>,
|
2023-12-27 03:42:39 +00:00
|
|
|
) -> FlowyResult<()> {
|
2023-07-05 12:57:09 +00:00
|
|
|
tracing::trace!("create a document: {:?}", doc_id);
|
2023-08-28 15:20:56 +00:00
|
|
|
if self.is_doc_exist(doc_id).unwrap_or(false) {
|
2023-12-27 03:42:39 +00:00
|
|
|
Err(FlowyError::new(
|
|
|
|
ErrorCode::RecordAlreadyExists,
|
|
|
|
format!("document {} already exists", doc_id),
|
|
|
|
))
|
2023-08-28 15:20:56 +00:00
|
|
|
} else {
|
2023-12-27 03:42:39 +00:00
|
|
|
let encoded_collab_v1 =
|
|
|
|
doc_state_from_document_data(doc_id, data.unwrap_or_else(default_document_data))?;
|
|
|
|
let collab = self
|
|
|
|
.collab_for_document(
|
|
|
|
uid,
|
|
|
|
doc_id,
|
|
|
|
vec![encoded_collab_v1.doc_state.to_vec()],
|
|
|
|
false,
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
collab.lock().flush();
|
|
|
|
Ok(())
|
2023-08-28 05:28:24 +00:00
|
|
|
}
|
2023-04-17 02:12:04 +00:00
|
|
|
}
|
|
|
|
|
2023-07-05 12:57:09 +00:00
|
|
|
/// Return the document
|
2023-08-28 05:28:24 +00:00
|
|
|
#[tracing::instrument(level = "debug", skip(self), err)]
|
2023-07-14 05:37:13 +00:00
|
|
|
pub async fn get_document(&self, doc_id: &str) -> FlowyResult<Arc<MutexDocument>> {
|
2023-11-14 09:21:09 +00:00
|
|
|
if let Some(doc) = self.documents.lock().get(doc_id).cloned() {
|
|
|
|
return Ok(doc);
|
2023-04-13 10:53:51 +00:00
|
|
|
}
|
2023-12-01 03:17:49 +00:00
|
|
|
|
2023-07-14 05:37:13 +00:00
|
|
|
let mut updates = vec![];
|
2023-07-05 12:57:09 +00:00
|
|
|
if !self.is_doc_exist(doc_id)? {
|
2023-07-14 05:37:13 +00:00
|
|
|
// Try to get the document from the cloud service
|
2023-11-05 06:00:24 +00:00
|
|
|
let result: Result<CollabRawData, FlowyError> = self
|
2023-10-23 03:43:31 +00:00
|
|
|
.cloud_service
|
2023-12-27 03:42:39 +00:00
|
|
|
.get_document_doc_state(doc_id, &self.user.workspace_id()?)
|
2023-11-05 06:00:24 +00:00
|
|
|
.await;
|
|
|
|
|
|
|
|
updates = match result {
|
|
|
|
Ok(data) => data,
|
|
|
|
Err(err) => {
|
|
|
|
if err.is_record_not_found() {
|
|
|
|
// The document's ID exists in the cloud, but its content does not.
|
|
|
|
// This occurs when user A's document hasn't finished syncing and user B tries to open it.
|
|
|
|
// As a result, a blank document is created for user B.
|
|
|
|
event!(
|
|
|
|
tracing::Level::INFO,
|
|
|
|
"can't find the document in the cloud, doc_id: {}",
|
|
|
|
doc_id
|
|
|
|
);
|
2023-11-05 16:47:20 +00:00
|
|
|
vec![default_document_collab_data(doc_id).doc_state.to_vec()]
|
2023-11-05 06:00:24 +00:00
|
|
|
} else {
|
|
|
|
return Err(err);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
2023-07-05 12:57:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let uid = self.user.user_id()?;
|
2023-11-14 09:21:09 +00:00
|
|
|
event!(tracing::Level::DEBUG, "Initialize document: {}", doc_id);
|
2023-12-27 03:42:39 +00:00
|
|
|
let collab = self.collab_for_document(uid, doc_id, updates, true).await?;
|
2023-07-05 12:57:09 +00:00
|
|
|
let document = Arc::new(MutexDocument::open(doc_id, collab)?);
|
|
|
|
|
2023-05-16 06:58:24 +00:00
|
|
|
// save the document to the memory and read it from the memory if we open the same document again.
|
|
|
|
// and we don't want to subscribe to the document changes if we open the same document again.
|
|
|
|
self
|
|
|
|
.documents
|
2023-11-14 09:21:09 +00:00
|
|
|
.lock()
|
|
|
|
.put(doc_id.to_string(), document.clone());
|
2023-06-17 06:25:30 +00:00
|
|
|
Ok(document)
|
|
|
|
}
|
|
|
|
|
2023-07-14 05:37:13 +00:00
|
|
|
pub async fn get_document_data(&self, doc_id: &str) -> FlowyResult<DocumentData> {
|
|
|
|
let mut updates = vec![];
|
2023-07-05 12:57:09 +00:00
|
|
|
if !self.is_doc_exist(doc_id)? {
|
2023-10-23 03:43:31 +00:00
|
|
|
updates = self
|
|
|
|
.cloud_service
|
2023-12-27 03:42:39 +00:00
|
|
|
.get_document_doc_state(doc_id, &self.user.workspace_id()?)
|
2023-10-23 03:43:31 +00:00
|
|
|
.await?;
|
2023-07-05 12:57:09 +00:00
|
|
|
}
|
2023-08-17 15:46:39 +00:00
|
|
|
let uid = self.user.user_id()?;
|
2023-12-27 03:42:39 +00:00
|
|
|
let collab = self
|
|
|
|
.collab_for_document(uid, doc_id, updates, false)
|
|
|
|
.await?;
|
2023-07-05 12:57:09 +00:00
|
|
|
Document::open(collab)?
|
|
|
|
.get_document_data()
|
|
|
|
.map_err(internal_error)
|
2023-04-13 10:53:51 +00:00
|
|
|
}
|
|
|
|
|
2023-11-12 10:00:07 +00:00
|
|
|
#[instrument(level = "debug", skip(self), err)]
|
2023-11-20 12:54:47 +00:00
|
|
|
pub async fn close_document(&self, doc_id: &str) -> FlowyResult<()> {
|
2023-11-14 09:21:09 +00:00
|
|
|
// The lru will pop the least recently used document when the cache is full.
|
2023-11-20 12:54:47 +00:00
|
|
|
if let Ok(doc) = self.get_document(doc_id).await {
|
|
|
|
if let Some(doc) = doc.try_lock() {
|
|
|
|
let _ = doc.flush();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-05 01:42:11 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn delete_document(&self, doc_id: &str) -> FlowyResult<()> {
|
|
|
|
let uid = self.user.user_id()?;
|
2023-07-29 01:46:24 +00:00
|
|
|
if let Some(db) = self.user.collab_db(uid)?.upgrade() {
|
|
|
|
let _ = db.with_write_txn(|txn| {
|
|
|
|
txn.delete_doc(uid, &doc_id)?;
|
|
|
|
Ok(())
|
|
|
|
});
|
2023-11-14 09:21:09 +00:00
|
|
|
|
|
|
|
// When deleting a document, we need to remove it from the cache.
|
|
|
|
self.documents.lock().pop(doc_id);
|
2023-07-29 01:46:24 +00:00
|
|
|
}
|
2023-04-13 10:53:51 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
2023-07-05 12:57:09 +00:00
|
|
|
|
|
|
|
/// Return the list of snapshots of the document.
|
|
|
|
pub async fn get_document_snapshots(
|
|
|
|
&self,
|
|
|
|
document_id: &str,
|
2023-08-17 15:46:39 +00:00
|
|
|
limit: usize,
|
2023-07-05 12:57:09 +00:00
|
|
|
) -> FlowyResult<Vec<DocumentSnapshotPB>> {
|
2023-10-23 03:43:31 +00:00
|
|
|
let workspace_id = self.user.workspace_id()?;
|
2023-08-17 15:46:39 +00:00
|
|
|
let snapshots = self
|
2023-07-05 12:57:09 +00:00
|
|
|
.cloud_service
|
2023-10-23 03:43:31 +00:00
|
|
|
.get_document_snapshots(document_id, limit, &workspace_id)
|
2023-07-05 12:57:09 +00:00
|
|
|
.await?
|
2023-08-17 15:46:39 +00:00
|
|
|
.into_iter()
|
2023-07-05 12:57:09 +00:00
|
|
|
.map(|snapshot| DocumentSnapshotPB {
|
|
|
|
snapshot_id: snapshot.snapshot_id,
|
|
|
|
snapshot_desc: "".to_string(),
|
|
|
|
created_at: snapshot.created_at,
|
|
|
|
data: snapshot.data,
|
|
|
|
})
|
2023-08-17 15:46:39 +00:00
|
|
|
.collect::<Vec<_>>();
|
2023-07-05 12:57:09 +00:00
|
|
|
|
|
|
|
Ok(snapshots)
|
|
|
|
}
|
|
|
|
|
2023-09-06 08:00:23 +00:00
|
|
|
async fn collab_for_document(
|
2023-07-14 05:37:13 +00:00
|
|
|
&self,
|
2023-08-17 15:46:39 +00:00
|
|
|
uid: i64,
|
2023-07-14 05:37:13 +00:00
|
|
|
doc_id: &str,
|
2023-12-27 03:42:39 +00:00
|
|
|
doc_state: Vec<Vec<u8>>,
|
|
|
|
sync_enable: bool,
|
2023-07-14 05:37:13 +00:00
|
|
|
) -> FlowyResult<Arc<MutexCollab>> {
|
|
|
|
let db = self.user.collab_db(uid)?;
|
|
|
|
let collab = self
|
|
|
|
.collab_builder
|
2023-12-27 03:42:39 +00:00
|
|
|
.build(
|
|
|
|
uid,
|
|
|
|
doc_id,
|
|
|
|
CollabType::Document,
|
|
|
|
doc_state,
|
|
|
|
db,
|
|
|
|
CollabBuilderConfig::default().sync_enable(sync_enable),
|
|
|
|
)
|
2023-10-07 01:58:44 +00:00
|
|
|
.await?;
|
2023-07-14 05:37:13 +00:00
|
|
|
Ok(collab)
|
2023-07-05 12:57:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn is_doc_exist(&self, doc_id: &str) -> FlowyResult<bool> {
|
|
|
|
let uid = self.user.user_id()?;
|
2023-07-29 01:46:24 +00:00
|
|
|
if let Some(collab_db) = self.user.collab_db(uid)?.upgrade() {
|
|
|
|
let read_txn = collab_db.read_txn();
|
|
|
|
Ok(read_txn.is_exist(uid, doc_id))
|
|
|
|
} else {
|
|
|
|
Ok(false)
|
|
|
|
}
|
2023-07-05 12:57:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Only expose this method for testing
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
pub fn get_cloud_service(&self) -> &Arc<dyn DocumentCloudService> {
|
|
|
|
&self.cloud_service
|
|
|
|
}
|
2023-09-01 14:27:29 +00:00
|
|
|
/// Only expose this method for testing
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
pub fn get_file_storage_service(&self) -> &Weak<dyn FileStorageService> {
|
|
|
|
&self.storage_service
|
|
|
|
}
|
2023-04-13 10:53:51 +00:00
|
|
|
}
|
2023-12-27 03:42:39 +00:00
|
|
|
|
|
|
|
fn doc_state_from_document_data(
|
|
|
|
doc_id: &str,
|
|
|
|
data: DocumentData,
|
|
|
|
) -> Result<EncodedCollabV1, FlowyError> {
|
|
|
|
let collab = Arc::new(MutexCollab::from_collab(Collab::new_with_origin(
|
|
|
|
CollabOrigin::Empty,
|
|
|
|
doc_id,
|
|
|
|
vec![],
|
|
|
|
)));
|
|
|
|
let _ = Document::create_with_data(collab.clone(), data).map_err(internal_error)?;
|
|
|
|
Ok(collab.encode_collab_v1())
|
|
|
|
}
|