mirror of
				https://github.com/AppFlowy-IO/AppFlowy.git
				synced 2025-10-30 17:38:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			179 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			179 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>,
 | |
|     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();
 | |
| }
 | 
