| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233 | #![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,    },    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::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)}
 |