mirror of
https://github.com/AppFlowy-IO/AppFlowy.git
synced 2024-08-30 18:12:39 +00:00
180 lines
6.8 KiB
Rust
180 lines
6.8 KiB
Rust
#![allow(clippy::all)]
|
|
#![cfg_attr(rustfmt, rustfmt::skip)]
|
|
use std::convert::TryInto;
|
|
use actix_web::web::Data;
|
|
use flowy_document::core::ClientDocumentEditor;
|
|
use flowy_test::{helper::ViewTest, FlowySDKTest};
|
|
use flowy_user::services::UserSession;
|
|
use futures_util::{stream, stream::StreamExt};
|
|
use std::sync::Arc;
|
|
use bytes::Bytes;
|
|
use tokio::time::{sleep, Duration};
|
|
use crate::util::helper::{spawn_server, TestServer};
|
|
use flowy_collaboration::{entities::document_info::DocumentId, protobuf::ResetDocumentParams as ResetDocumentParamsPB};
|
|
use lib_ot::rich_text::{RichTextAttribute, RichTextDelta};
|
|
use parking_lot::RwLock;
|
|
use backend::services::document::persistence::{read_document, reset_document};
|
|
use flowy_collaboration::entities::revision::{RepeatedRevision, Revision};
|
|
use flowy_collaboration::protobuf::{RepeatedRevision as RepeatedRevisionPB, DocumentId as DocumentIdPB};
|
|
use flowy_collaboration::server_document::ServerDocumentManager;
|
|
use flowy_net::ws::connection::FlowyWebSocketConnect;
|
|
use lib_ot::core::Interval;
|
|
|
|
pub struct DocumentTest {
|
|
server: TestServer,
|
|
flowy_test: FlowySDKTest,
|
|
}
|
|
#[derive(Clone)]
|
|
pub enum DocScript {
|
|
ClientInsertText(usize, &'static str),
|
|
ClientFormatText(Interval, RichTextAttribute),
|
|
ClientOpenDoc,
|
|
AssertClient(&'static str),
|
|
AssertServer(&'static str, i64),
|
|
ServerResetDocument(String, i64), // delta_json, rev_id
|
|
}
|
|
|
|
impl DocumentTest {
|
|
pub async fn new() -> Self {
|
|
let server = spawn_server().await;
|
|
let flowy_test = FlowySDKTest::new(server.client_server_config.clone());
|
|
Self { server, flowy_test }
|
|
}
|
|
|
|
pub async fn run_scripts(self, scripts: Vec<DocScript>) {
|
|
let _ = self.flowy_test.sign_up().await;
|
|
let DocumentTest { server, flowy_test } = self;
|
|
let script_context = Arc::new(RwLock::new(ScriptContext::new(flowy_test, server).await));
|
|
run_scripts(script_context, scripts).await;
|
|
sleep(Duration::from_secs(5)).await;
|
|
}
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
struct ScriptContext {
|
|
client_editor: Option<Arc<ClientDocumentEditor>>,
|
|
client_sdk: FlowySDKTest,
|
|
client_user_session: Arc<UserSession>,
|
|
#[allow(dead_code)]
|
|
ws_conn: Arc<FlowyWebSocketConnect>,
|
|
server: TestServer,
|
|
doc_id: String,
|
|
}
|
|
|
|
impl ScriptContext {
|
|
async fn new(client_sdk: FlowySDKTest, server: TestServer) -> Self {
|
|
let user_session = client_sdk.user_session.clone();
|
|
let ws_conn = client_sdk.ws_conn.clone();
|
|
let doc_id = create_doc(&client_sdk).await;
|
|
|
|
Self {
|
|
client_editor: None,
|
|
client_sdk,
|
|
client_user_session: user_session,
|
|
ws_conn,
|
|
server,
|
|
doc_id,
|
|
}
|
|
}
|
|
|
|
async fn open_doc(&mut self) {
|
|
let doc_id = self.doc_id.clone();
|
|
let edit_context = self.client_sdk.document_manager.open_document(doc_id).await.unwrap();
|
|
self.client_editor = Some(edit_context);
|
|
}
|
|
|
|
fn client_editor(&self) -> Arc<ClientDocumentEditor> { self.client_editor.as_ref().unwrap().clone() }
|
|
}
|
|
|
|
async fn run_scripts(context: Arc<RwLock<ScriptContext>>, scripts: Vec<DocScript>) {
|
|
let mut fut_scripts = vec![];
|
|
for script in scripts {
|
|
let context = context.clone();
|
|
let fut = async move {
|
|
let doc_id = context.read().doc_id.clone();
|
|
match script {
|
|
DocScript::ClientOpenDoc => {
|
|
context.write().open_doc().await;
|
|
},
|
|
DocScript::ClientInsertText(index, s) => {
|
|
context.read().client_editor().insert(index, s).await.unwrap();
|
|
},
|
|
DocScript::ClientFormatText(interval, attribute) => {
|
|
context
|
|
.read()
|
|
.client_editor()
|
|
.format(interval, attribute)
|
|
.await
|
|
.unwrap();
|
|
},
|
|
DocScript::AssertClient(s) => {
|
|
sleep(Duration::from_millis(2000)).await;
|
|
let json = context.read().client_editor().doc_json().await.unwrap();
|
|
assert_eq(s, &json);
|
|
},
|
|
DocScript::AssertServer(s, rev_id) => {
|
|
sleep(Duration::from_millis(2000)).await;
|
|
let persistence = Data::new(context.read().server.app_ctx.persistence.document_kv_store());
|
|
let doc_identifier: DocumentIdPB = DocumentId {
|
|
doc_id
|
|
}.try_into().unwrap();
|
|
|
|
let document_info = read_document(persistence.get_ref(), doc_identifier).await.unwrap();
|
|
assert_eq(s, &document_info.text);
|
|
assert_eq!(document_info.rev_id, rev_id);
|
|
},
|
|
DocScript::ServerResetDocument(document_json, rev_id) => {
|
|
let delta_data = Bytes::from(document_json);
|
|
let user_id = context.read().client_user_session.user_id().unwrap();
|
|
let md5 = format!("{:x}", md5::compute(&delta_data));
|
|
let base_rev_id = if rev_id == 0 { rev_id } else { rev_id - 1 };
|
|
let revision = Revision::new(
|
|
&doc_id,
|
|
base_rev_id,
|
|
rev_id,
|
|
delta_data,
|
|
&user_id,
|
|
md5,
|
|
);
|
|
|
|
let document_manager = context.read().server.app_ctx.document_manager.clone();
|
|
reset_doc(&doc_id, RepeatedRevision::new(vec![revision]), document_manager.get_ref()).await;
|
|
sleep(Duration::from_millis(2000)).await;
|
|
},
|
|
}
|
|
};
|
|
fut_scripts.push(fut);
|
|
}
|
|
|
|
let mut stream = stream::iter(fut_scripts);
|
|
while let Some(script) = stream.next().await {
|
|
let _ = script.await;
|
|
}
|
|
|
|
std::mem::forget(context);
|
|
}
|
|
|
|
fn assert_eq(expect: &str, receive: &str) {
|
|
let expected_delta: RichTextDelta = serde_json::from_str(expect).unwrap();
|
|
let target_delta: RichTextDelta = serde_json::from_str(receive).unwrap();
|
|
|
|
if expected_delta != target_delta {
|
|
log::error!("✅ expect: {}", expect,);
|
|
log::error!("❌ receive: {}", receive);
|
|
}
|
|
assert_eq!(target_delta, expected_delta);
|
|
}
|
|
|
|
async fn create_doc(flowy_test: &FlowySDKTest) -> String {
|
|
let view_test = ViewTest::new(flowy_test).await;
|
|
view_test.view.id
|
|
}
|
|
|
|
async fn reset_doc(doc_id: &str, repeated_revision: RepeatedRevision, document_manager: &Arc<ServerDocumentManager>) {
|
|
let pb: RepeatedRevisionPB = repeated_revision.try_into().unwrap();
|
|
let mut params = ResetDocumentParamsPB::new();
|
|
params.set_doc_id(doc_id.to_owned());
|
|
params.set_revisions(pb);
|
|
let _ = reset_document(document_manager, params).await.unwrap();
|
|
}
|