//! All the tests in this file are based on [richtext.md] use std::ops::Range; use loro_common::LoroValue; use loro_internal::{container::richtext::TextStyleInfoFlag, LoroDoc, ToJson}; use serde_json::json; fn init(s: &str) -> LoroDoc { let doc = LoroDoc::default(); doc.set_peer_id(1).unwrap(); let richtext = doc.get_text("r"); doc.with_txn(|txn| richtext.insert_with_txn(txn, 0, s)) .unwrap(); doc } fn clone(doc: &LoroDoc, peer_id: u64) -> LoroDoc { let doc2 = LoroDoc::default(); doc2.set_peer_id(peer_id).unwrap(); doc2.import(&doc.export_from(&Default::default())).unwrap(); doc2 } #[derive(Debug)] enum Kind { Bold, Italic, Link, } impl Kind { fn key(&self) -> &str { match self { Kind::Bold => "bold", Kind::Link => "link", Kind::Italic => "italic", } } fn flag(&self) -> TextStyleInfoFlag { match self { Kind::Bold => TextStyleInfoFlag::BOLD, Kind::Link => TextStyleInfoFlag::LINK, Kind::Italic => TextStyleInfoFlag::BOLD, } } } fn insert(doc: &LoroDoc, pos: usize, s: &str) { let richtext = doc.get_text("r"); doc.with_txn(|txn| richtext.insert_with_txn(txn, pos, s)) .unwrap(); } fn delete(doc: &LoroDoc, pos: usize, len: usize) { let richtext = doc.get_text("r"); doc.with_txn(|txn| richtext.delete_with_txn(txn, pos, len)) .unwrap(); } fn mark(doc: &LoroDoc, range: Range, kind: Kind) { let richtext = doc.get_text("r"); doc.with_txn(|txn| { richtext.mark_with_txn( txn, range.start, range.end, kind.key(), if kind.flag().is_delete() { LoroValue::Null } else { true.into() }, kind.flag(), ) }) .unwrap(); } fn unmark(doc: &LoroDoc, range: Range, kind: Kind) { let richtext = doc.get_text("r"); doc.with_txn(|txn| { richtext.mark_with_txn( txn, range.start, range.end, kind.key(), false.into(), kind.flag().to_delete(), ) }) .unwrap(); } fn merge(a: &LoroDoc, b: &LoroDoc) { a.import(&b.export_from(&a.oplog_vv())).unwrap(); b.import(&a.export_from(&b.oplog_vv())).unwrap(); } fn expect_result(doc: &LoroDoc, json: serde_json::Value) { let richtext = doc.get_text("r"); let s = richtext.get_richtext_value().to_json_value(); assert_eq!( &s, &json, "expect: {}, got: {}", serde_json::to_string_pretty(&json).unwrap(), serde_json::to_string_pretty(&s).unwrap() ); } #[test] fn case0() { let doc_a = init("Hello World"); let doc_b = clone(&doc_a, 2); mark(&doc_a, 0..11, Kind::Bold); insert(&doc_b, 6, "New "); merge(&doc_a, &doc_b); expect_result( &doc_a, json!([{"insert":"Hello New World","attributes":{"bold":true}}]), ); } #[test] fn case1() { let doc_a = init("Hello World"); let doc_b = clone(&doc_a, 2); mark(&doc_a, 0..5, Kind::Bold); mark(&doc_b, 3..11, Kind::Bold); merge(&doc_a, &doc_b); expect_result( &doc_a, json!([{"insert":"Hello World","attributes":{"bold":true}}]), ); } #[test] fn case2() { let doc_a = init("Hello World"); mark(&doc_a, 0..11, Kind::Bold); let doc_b = clone(&doc_a, 2); unmark(&doc_a, 0..6, Kind::Bold); insert(&doc_b, 5, " a"); merge(&doc_a, &doc_b); expect_result( &doc_a, json!([{"insert":"Hello a ","attributes":{"bold":false}},{"insert":"World","attributes":{"bold":true}}]), ); } /// | Name | Text | /// |:----------------|:-----------------------| /// | Origin | `Hello World` | /// | Concurrent A | `Hello World` | /// | Concurrent B | `Hello a World` | /// | Expected Result | `Hello a World` | /// #[test] fn case3() { let doc_a = init("Hello World"); let doc_b = clone(&doc_a, 2); mark(&doc_a, 6..11, Kind::Bold); insert(&doc_b, 5, " a"); merge(&doc_a, &doc_b); expect_result( &doc_a, json!([{"insert":"Hello a "},{"insert":"World","attributes":{"bold":true}}]), ); } /// | Name | Text | /// |:----------------|:---------------------------| /// | Origin | `Hello World` | /// | Concurrent A | `Hello World` | /// | Concurrent B | `Hey World` | /// | Expected Result | `Hey World` | #[test] fn case5() { let doc_a = init("Hello World"); let doc_b = clone(&doc_a, 2); mark(&doc_a, 0..5, Kind::Link); delete(&doc_b, 2, 3); expect_result(&doc_b, json!([{"insert":"He World"}])); insert(&doc_b, 2, "y"); expect_result(&doc_b, json!([{"insert":"Hey World"}])); merge(&doc_a, &doc_b); expect_result( &doc_b, json!([{"insert":"Hey","attributes":{"link":true}},{"insert":" World"}]), ); expect_result( &doc_a, json!([{"insert":"Hey","attributes":{"link":true}},{"insert":" World"}]), ); } /// When insert a new character after "Hello", the new char should be bold but not link /// /// | Name | Text | /// |:----------------|:----------------------------------| /// | Origin | `Hello World` | /// | Expected Result | `Hellot World` | #[test] fn case6() { let doc = init("Hello World"); mark(&doc, 0..5, Kind::Bold); expect_result( &doc, serde_json::json!([ {"insert": "Hello", "attributes": {"bold": true}}, {"insert": " World"} ]), ); mark(&doc, 0..5, Kind::Link); expect_result( &doc, serde_json::json!([ {"insert": "Hello", "attributes": {"bold": true, "link": true}}, {"insert": " World"} ]), ); insert(&doc, 5, "t"); expect_result( &doc, serde_json::json!([ {"insert": "Hello", "attributes": {"bold": true, "link": true}}, {"insert": "t", "attributes": {"bold": true}}, {"insert": " World"} ]), ); } /// /// | Name | Text | /// |:----------------|:---------------------------------------------| /// | Origin | `The fox jumped over the dog.` | /// | Concurrent A | `The fox jumped over the dog.` | /// | Concurrent B | `The fox jumped over the dog.` | /// | Expected Result | `The fox jumped over the dog.` | #[test] fn case7() { let doc_a = init("The fox jumped over the dog."); mark(&doc_a, 0..3, Kind::Bold); let doc_b = clone(&doc_a, 2); unmark(&doc_a, 0..3, Kind::Bold); unmark(&doc_b, 4..7, Kind::Bold); merge(&doc_a, &doc_b); expect_result( &doc_a, json!([ {"insert":"The", "attributes": {"bold": false}}, {"insert":" ",}, {"insert":"fox", "attributes": {"bold": false}}, {"insert":" jumped over the dog."} ]), ); } /// | Name | Text | /// |:----------------|:---------------------------------------------| /// | Origin | `The fox jumped over the dog.` | /// | Concurrent A | `The fox jumped over the dog.` | /// | Concurrent B | `The fox jumped over the dog.` | /// | Expected Result | `The fox jumped over the dog.` | #[test] fn case8() { let doc_a = init("The fox jumped over the dog."); mark(&doc_a, 0..14, Kind::Bold); let doc_b = clone(&doc_a, 2); unmark(&doc_a, 7..14, Kind::Bold); unmark(&doc_b, 3..14, Kind::Bold); mark(&doc_b, 24..27, Kind::Bold); merge(&doc_a, &doc_b); expect_result( &doc_a, serde_json::json!([ {"insert": "The", "attributes": {"bold": true}}, {"insert": " fox jumped", "attributes": {"bold": false}}, {"insert": " over the "}, {"insert": "dog", "attributes": {"bold": true}}, {"insert": "."} ]), ); } /// | Name | Text | /// |:----------------|:-----------------------------| /// | Origin | The fox jumped. | /// | Concurrent A | **The fox** jumped. | /// | Concurrent B | The *fox jumped*. | /// | Expected Result | **The _fox_** jumped. | #[test] fn case9() { let doc_a = init("The fox jumped."); let doc_b = clone(&doc_a, 2); mark(&doc_a, 0..7, Kind::Bold); mark(&doc_a, 4..14, Kind::Italic); merge(&doc_a, &doc_b); expect_result( &doc_a, serde_json::json!([ {"insert": "The ", "attributes": {"bold": true}}, {"insert": "fox", "attributes": {"bold": true, "italic": true}}, {"insert": " jumped", "attributes": {"italic": true}}, {"insert": "."} ]), ); } #[test] fn insert_after_link() { let doc_a = init("The fox jumped."); let doc_b = clone(&doc_a, 2); mark(&doc_a, 0..3, Kind::Link); merge(&doc_a, &doc_b); insert(&doc_a, 3, "a"); merge(&doc_a, &doc_b); expect_result( &doc_a, serde_json::json!([ {"insert": "The", "attributes": {"link": true}}, {"insert": "a fox jumped."}, ]), ); expect_result( &doc_b, serde_json::json!([ {"insert": "The", "attributes": {"link": true}}, {"insert": "a fox jumped."}, ]), ); }