| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258 | #![allow(clippy::all)]use lib_ot::core::{NodeTreeContext, OperationTransform, Transaction};use lib_ot::text_delta::DeltaTextOperationBuilder;use lib_ot::{  core::attributes::AttributeHashMap,  core::{Body, Changeset, NodeData, NodeTree, Path, TransactionBuilder},  text_delta::DeltaTextOperations,};use std::collections::HashMap;pub enum NodeScript {  InsertNode {    path: Path,    node_data: NodeData,    rev_id: usize,  },  InsertNodes {    path: Path,    node_data_list: Vec<NodeData>,    rev_id: usize,  },  UpdateAttributes {    path: Path,    attributes: AttributeHashMap,  },  UpdateBody {    path: Path,    changeset: Changeset,  },  DeleteNode {    path: Path,    rev_id: usize,  },  DeleteNodes {    path: Path,    node_data_list: Vec<NodeData>,    rev_id: usize,  },  AssertNumberOfChildrenAtPath {    path: Option<Path>,    expected: usize,  },  AssertNodesAtRoot {    expected: Vec<NodeData>,  },  #[allow(dead_code)]  AssertNodesAtPath {    path: Path,    expected: Vec<NodeData>,  },  AssertNode {    path: Path,    expected: Option<NodeData>,  },  AssertNodeAttributes {    path: Path,    expected: &'static str,  },  AssertNodeDelta {    path: Path,    expected: DeltaTextOperations,  },  AssertNodeDeltaContent {    path: Path,    expected: &'static str,  },  #[allow(dead_code)]  AssertTreeJSON {    expected: String,  },}pub struct NodeTest {  rev_id: usize,  rev_operations: HashMap<usize, Transaction>,  node_tree: NodeTree,}impl NodeTest {  pub fn new() -> Self {    Self {      rev_id: 0,      rev_operations: HashMap::new(),      node_tree: NodeTree::new(NodeTreeContext::default()),    }  }  pub fn run_scripts(&mut self, scripts: Vec<NodeScript>) {    for script in scripts {      self.run_script(script);    }  }  pub fn run_script(&mut self, script: NodeScript) {    match script {      NodeScript::InsertNode {        path,        node_data: node,        rev_id,      } => {        let mut transaction = TransactionBuilder::new()          .insert_node_at_path(path, node)          .build();        self.transform_transaction_if_need(&mut transaction, rev_id);        self.apply_transaction(transaction);      },      NodeScript::InsertNodes {        path,        node_data_list,        rev_id,      } => {        let mut transaction = TransactionBuilder::new()          .insert_nodes_at_path(path, node_data_list)          .build();        self.transform_transaction_if_need(&mut transaction, rev_id);        self.apply_transaction(transaction);      },      NodeScript::UpdateAttributes { path, attributes } => {        let node = self.node_tree.get_node_data_at_path(&path).unwrap();        let transaction = TransactionBuilder::new()          .update_node_at_path(            &path,            Changeset::Attributes {              new: attributes,              old: node.attributes,            },          )          .build();        self.apply_transaction(transaction);      },      NodeScript::UpdateBody { path, changeset } => {        //        let transaction = TransactionBuilder::new()          .update_node_at_path(&path, changeset)          .build();        self.apply_transaction(transaction);      },      NodeScript::DeleteNode { path, rev_id } => {        let mut transaction = TransactionBuilder::new()          .delete_node_at_path(&self.node_tree, &path)          .build();        self.transform_transaction_if_need(&mut transaction, rev_id);        self.apply_transaction(transaction);      },      NodeScript::DeleteNodes {        path,        node_data_list,        rev_id,      } => {        let mut transaction = TransactionBuilder::new()          .delete_nodes_at_path(&self.node_tree, &path, node_data_list.len())          .build();        self.transform_transaction_if_need(&mut transaction, rev_id);        self.apply_transaction(transaction);      },      NodeScript::AssertNode { path, expected } => {        let node = self.node_tree.get_node_data_at_path(&path);        assert_eq!(node, expected.map(|e| e.into()));      },      NodeScript::AssertNodeAttributes { path, expected } => {        let node = self.node_tree.get_node_data_at_path(&path).unwrap();        assert_eq!(node.attributes.to_json().unwrap(), expected);      },      NodeScript::AssertNumberOfChildrenAtPath { path, expected } => match path {        None => {          let len = self.node_tree.number_of_children(None);          assert_eq!(len, expected)        },        Some(path) => {          let node_id = self.node_tree.node_id_at_path(path).unwrap();          let len = self.node_tree.number_of_children(Some(node_id));          assert_eq!(len, expected)        },      },      NodeScript::AssertNodesAtRoot { expected } => {        let nodes = self.node_tree.get_node_data_at_root().unwrap().children;        assert_eq!(nodes, expected)      },      NodeScript::AssertNodesAtPath { path, expected } => {        let nodes = self          .node_tree          .get_node_data_at_path(&path)          .unwrap()          .children;        assert_eq!(nodes, expected)      },      NodeScript::AssertNodeDelta { path, expected } => {        let node = self.node_tree.get_node_at_path(&path).unwrap();        if let Body::Delta(delta) = node.body.clone() {          debug_assert_eq!(delta, expected);        } else {          panic!("Node body type not match, expect Delta");        }      },      NodeScript::AssertNodeDeltaContent { path, expected } => {        let node = self.node_tree.get_node_at_path(&path).unwrap();        if let Body::Delta(delta) = node.body.clone() {          debug_assert_eq!(delta.content().unwrap(), expected);        } else {          panic!("Node body type not match, expect Delta");        }      },      NodeScript::AssertTreeJSON { expected } => {        let json = serde_json::to_string(&self.node_tree).unwrap();        assert_eq!(json, expected)      },    }  }  fn apply_transaction(&mut self, transaction: Transaction) {    self.rev_id += 1;    self.rev_operations.insert(self.rev_id, transaction.clone());    self.node_tree.apply_transaction(transaction).unwrap();  }  fn transform_transaction_if_need(&mut self, transaction: &mut Transaction, rev_id: usize) {    if self.rev_id >= rev_id {      for rev_id in rev_id..=self.rev_id {        let old_transaction = self.rev_operations.get(&rev_id).unwrap();        *transaction = old_transaction.transform(transaction).unwrap();      }    }  }}pub fn edit_node_delta(  delta: &DeltaTextOperations,  new_delta: DeltaTextOperations,) -> (Changeset, DeltaTextOperations) {  let inverted = new_delta.invert(&delta);  let expected = delta.compose(&new_delta).unwrap();  let changeset = Changeset::Delta {    delta: new_delta.clone(),    inverted: inverted.clone(),  };  (changeset, expected)}pub fn make_node_delta_changeset(  initial_content: &str,  insert_str: &str,) -> (DeltaTextOperations, Changeset, DeltaTextOperations) {  let initial_content = initial_content.to_owned();  let initial_delta = DeltaTextOperationBuilder::new()    .insert(&initial_content)    .build();  let delta = DeltaTextOperationBuilder::new()    .retain(initial_content.len())    .insert(insert_str)    .build();  let inverted = delta.invert(&initial_delta);  let expected = initial_delta.compose(&delta).unwrap();  let changeset = Changeset::Delta {    delta: delta.clone(),    inverted: inverted.clone(),  };  (initial_delta, changeset, expected)}
 |