edit_script.rs 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. #![allow(clippy::all)]
  2. #![cfg_attr(rustfmt, rustfmt::skip)]
  3. use std::convert::TryInto;
  4. use actix_web::web::Data;
  5. use flowy_document::services::doc::edit::ClientDocumentEditor;
  6. use flowy_test::{helper::ViewTest, FlowySDKTest};
  7. use flowy_user::services::user::UserSession;
  8. use futures_util::{stream, stream::StreamExt};
  9. use std::sync::Arc;
  10. use bytes::Bytes;
  11. use tokio::time::{sleep, Duration};
  12. use crate::util::helper::{spawn_server, TestServer};
  13. use flowy_collaboration::{entities::doc::DocumentId, protobuf::ResetDocumentParams};
  14. use lib_ot::rich_text::{RichTextAttribute, RichTextDelta};
  15. use parking_lot::RwLock;
  16. use backend::services::document::persistence::{read_document, reset_document};
  17. use flowy_collaboration::entities::revision::{RepeatedRevision, Revision};
  18. use flowy_collaboration::sync::ServerDocumentManager;
  19. use lib_ot::core::Interval;
  20. use flowy_net::services::ws::FlowyWSConnect;
  21. pub struct DocumentTest {
  22. server: TestServer,
  23. flowy_test: FlowySDKTest,
  24. }
  25. #[derive(Clone)]
  26. pub enum DocScript {
  27. ClientInsertText(usize, &'static str),
  28. ClientFormatText(Interval, RichTextAttribute),
  29. ClientOpenDoc,
  30. AssertClient(&'static str),
  31. AssertServer(&'static str, i64),
  32. ServerResetDocument(String, i64), // delta_json, rev_id
  33. }
  34. impl DocumentTest {
  35. pub async fn new() -> Self {
  36. let server = spawn_server().await;
  37. let flowy_test = FlowySDKTest::setup_with(server.client_server_config.clone());
  38. Self { server, flowy_test }
  39. }
  40. pub async fn run_scripts(self, scripts: Vec<DocScript>) {
  41. let _ = self.flowy_test.sign_up().await;
  42. let DocumentTest { server, flowy_test } = self;
  43. let script_context = Arc::new(RwLock::new(ScriptContext::new(flowy_test, server).await));
  44. run_scripts(script_context, scripts).await;
  45. sleep(Duration::from_secs(5)).await;
  46. }
  47. }
  48. #[derive(Clone)]
  49. struct ScriptContext {
  50. client_editor: Option<Arc<ClientDocumentEditor>>,
  51. client_sdk: FlowySDKTest,
  52. client_user_session: Arc<UserSession>,
  53. ws_conn: Arc<FlowyWSConnect>,
  54. server: TestServer,
  55. doc_id: String,
  56. }
  57. impl ScriptContext {
  58. async fn new(client_sdk: FlowySDKTest, server: TestServer) -> Self {
  59. let user_session = client_sdk.user_session.clone();
  60. let ws_manager = client_sdk.ws_manager.clone();
  61. let doc_id = create_doc(&client_sdk).await;
  62. Self {
  63. client_editor: None,
  64. client_sdk,
  65. client_user_session: user_session,
  66. ws_conn: ws_manager,
  67. server,
  68. doc_id,
  69. }
  70. }
  71. async fn open_doc(&mut self) {
  72. let doc_id = self.doc_id.clone();
  73. let edit_context = self.client_sdk.document_ctx.controller.open(doc_id).await.unwrap();
  74. self.client_editor = Some(edit_context);
  75. }
  76. fn client_editor(&self) -> Arc<ClientDocumentEditor> { self.client_editor.as_ref().unwrap().clone() }
  77. }
  78. async fn run_scripts(context: Arc<RwLock<ScriptContext>>, scripts: Vec<DocScript>) {
  79. let mut fut_scripts = vec![];
  80. for script in scripts {
  81. let context = context.clone();
  82. let fut = async move {
  83. let doc_id = context.read().doc_id.clone();
  84. match script {
  85. DocScript::ClientOpenDoc => {
  86. context.write().open_doc().await;
  87. },
  88. DocScript::ClientInsertText(index, s) => {
  89. context.read().client_editor().insert(index, s).await.unwrap();
  90. },
  91. DocScript::ClientFormatText(interval, attribute) => {
  92. context
  93. .read()
  94. .client_editor()
  95. .format(interval, attribute)
  96. .await
  97. .unwrap();
  98. },
  99. DocScript::AssertClient(s) => {
  100. sleep(Duration::from_millis(2000)).await;
  101. let json = context.read().client_editor().doc_json().await.unwrap();
  102. assert_eq(s, &json);
  103. },
  104. DocScript::AssertServer(s, rev_id) => {
  105. sleep(Duration::from_millis(2000)).await;
  106. let persistence = Data::new(context.read().server.app_ctx.persistence.kv_store());
  107. let doc_identifier: flowy_collaboration::protobuf::DocumentId = DocumentId {
  108. doc_id
  109. }.try_into().unwrap();
  110. let document_info = read_document(persistence.get_ref(), doc_identifier).await.unwrap();
  111. assert_eq(s, &document_info.text);
  112. assert_eq!(document_info.rev_id, rev_id);
  113. },
  114. DocScript::ServerResetDocument(document_json, rev_id) => {
  115. let delta_data = Bytes::from(document_json);
  116. let user_id = context.read().client_user_session.user_id().unwrap();
  117. let md5 = format!("{:x}", md5::compute(&delta_data));
  118. let base_rev_id = if rev_id == 0 { rev_id } else { rev_id - 1 };
  119. let revision = Revision::new(
  120. &doc_id,
  121. base_rev_id,
  122. rev_id,
  123. delta_data,
  124. &user_id,
  125. md5,
  126. );
  127. let document_manager = context.read().server.app_ctx.document_manager.clone();
  128. reset_doc(&doc_id, RepeatedRevision::new(vec![revision]), document_manager.get_ref()).await;
  129. sleep(Duration::from_millis(2000)).await;
  130. },
  131. }
  132. };
  133. fut_scripts.push(fut);
  134. }
  135. let mut stream = stream::iter(fut_scripts);
  136. while let Some(script) = stream.next().await {
  137. let _ = script.await;
  138. }
  139. std::mem::forget(context);
  140. }
  141. fn assert_eq(expect: &str, receive: &str) {
  142. let expected_delta: RichTextDelta = serde_json::from_str(expect).unwrap();
  143. let target_delta: RichTextDelta = serde_json::from_str(receive).unwrap();
  144. if expected_delta != target_delta {
  145. log::error!("✅ expect: {}", expect,);
  146. log::error!("❌ receive: {}", receive);
  147. }
  148. assert_eq!(target_delta, expected_delta);
  149. }
  150. async fn create_doc(flowy_test: &FlowySDKTest) -> String {
  151. let view_test = ViewTest::new(flowy_test).await;
  152. view_test.view.id
  153. }
  154. async fn reset_doc(doc_id: &str, repeated_revision: RepeatedRevision, document_manager: &Arc<ServerDocumentManager>) {
  155. let pb: flowy_collaboration::protobuf::RepeatedRevision = repeated_revision.try_into().unwrap();
  156. let mut params = ResetDocumentParams::new();
  157. params.set_doc_id(doc_id.to_owned());
  158. params.set_revisions(pb);
  159. let _ = reset_document(document_manager, params).await.unwrap();
  160. }