2022-10-20 03:35:11 +00:00
|
|
|
#![allow(clippy::all)]
|
|
|
|
use lib_ot::core::{NodeTreeContext, Transaction};
|
2022-09-11 10:16:19 +00:00
|
|
|
use lib_ot::{
|
2022-10-10 03:56:48 +00:00
|
|
|
core::attributes::AttributeHashMap,
|
2022-10-20 03:35:11 +00:00
|
|
|
core::{Body, Changeset, NodeData, NodeTree, Path, TransactionBuilder},
|
2022-10-10 03:56:48 +00:00
|
|
|
text_delta::TextOperations,
|
2022-09-11 04:59:01 +00:00
|
|
|
};
|
2022-09-13 12:23:56 +00:00
|
|
|
use std::collections::HashMap;
|
2022-09-09 07:05:41 +00:00
|
|
|
|
|
|
|
pub enum NodeScript {
|
2022-09-13 12:23:56 +00:00
|
|
|
InsertNode {
|
|
|
|
path: Path,
|
|
|
|
node_data: NodeData,
|
|
|
|
rev_id: usize,
|
|
|
|
},
|
2022-10-20 03:35:11 +00:00
|
|
|
InsertNodes {
|
|
|
|
path: Path,
|
|
|
|
node_data_list: Vec<NodeData>,
|
|
|
|
rev_id: usize,
|
|
|
|
},
|
2022-09-13 12:23:56 +00:00
|
|
|
UpdateAttributes {
|
|
|
|
path: Path,
|
2022-10-10 03:56:48 +00:00
|
|
|
attributes: AttributeHashMap,
|
2022-09-13 12:23:56 +00:00
|
|
|
},
|
|
|
|
UpdateBody {
|
|
|
|
path: Path,
|
2022-10-20 03:35:11 +00:00
|
|
|
changeset: Changeset,
|
2022-09-13 12:23:56 +00:00
|
|
|
},
|
|
|
|
DeleteNode {
|
|
|
|
path: Path,
|
2022-09-14 01:23:33 +00:00
|
|
|
rev_id: usize,
|
2022-09-13 12:23:56 +00:00
|
|
|
},
|
2022-10-20 03:35:11 +00:00
|
|
|
AssertNumberOfChildrenAtPath {
|
2022-09-13 12:23:56 +00:00
|
|
|
path: Option<Path>,
|
2022-10-20 03:35:11 +00:00
|
|
|
expected: usize,
|
|
|
|
},
|
|
|
|
AssertNodesAtRoot {
|
|
|
|
expected: Vec<NodeData>,
|
2022-09-13 12:23:56 +00:00
|
|
|
},
|
2022-10-20 03:35:11 +00:00
|
|
|
#[allow(dead_code)]
|
|
|
|
AssertNodesAtPath {
|
2022-09-13 12:23:56 +00:00
|
|
|
path: Path,
|
2022-10-20 03:35:11 +00:00
|
|
|
expected: Vec<NodeData>,
|
2022-09-13 12:23:56 +00:00
|
|
|
},
|
|
|
|
AssertNode {
|
|
|
|
path: Path,
|
2022-10-20 03:35:11 +00:00
|
|
|
expected: Option<NodeData>,
|
2022-09-13 12:23:56 +00:00
|
|
|
},
|
|
|
|
AssertNodeDelta {
|
|
|
|
path: Path,
|
2022-10-10 03:56:48 +00:00
|
|
|
expected: TextOperations,
|
2022-09-13 12:23:56 +00:00
|
|
|
},
|
2022-10-20 03:35:11 +00:00
|
|
|
#[allow(dead_code)]
|
|
|
|
AssertTreeJSON {
|
|
|
|
expected: String,
|
|
|
|
},
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct NodeTest {
|
2022-09-13 12:23:56 +00:00
|
|
|
rev_id: usize,
|
|
|
|
rev_operations: HashMap<usize, Transaction>,
|
2022-09-10 00:58:08 +00:00
|
|
|
node_tree: NodeTree,
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl NodeTest {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Self {
|
2022-09-13 12:23:56 +00:00
|
|
|
rev_id: 0,
|
|
|
|
rev_operations: HashMap::new(),
|
2022-10-20 03:35:11 +00:00
|
|
|
node_tree: NodeTree::new(NodeTreeContext::default()),
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
2022-09-13 12:23:56 +00:00
|
|
|
NodeScript::InsertNode {
|
|
|
|
path,
|
|
|
|
node_data: node,
|
|
|
|
rev_id,
|
|
|
|
} => {
|
|
|
|
let mut transaction = TransactionBuilder::new(&self.node_tree)
|
2022-09-09 07:05:41 +00:00
|
|
|
.insert_node_at_path(path, node)
|
|
|
|
.finalize();
|
2022-09-13 12:23:56 +00:00
|
|
|
self.transform_transaction_if_need(&mut transaction, rev_id);
|
|
|
|
self.apply_transaction(transaction);
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
2022-10-20 03:35:11 +00:00
|
|
|
NodeScript::InsertNodes {
|
|
|
|
path,
|
|
|
|
node_data_list,
|
|
|
|
rev_id,
|
|
|
|
} => {
|
|
|
|
let mut transaction = TransactionBuilder::new(&self.node_tree)
|
|
|
|
.insert_nodes_at_path(path, node_data_list)
|
|
|
|
.finalize();
|
|
|
|
self.transform_transaction_if_need(&mut transaction, rev_id);
|
|
|
|
self.apply_transaction(transaction);
|
|
|
|
}
|
2022-09-11 04:59:01 +00:00
|
|
|
NodeScript::UpdateAttributes { path, attributes } => {
|
2022-09-09 07:05:41 +00:00
|
|
|
let transaction = TransactionBuilder::new(&self.node_tree)
|
2022-09-10 00:42:53 +00:00
|
|
|
.update_attributes_at_path(&path, attributes)
|
2022-09-09 07:05:41 +00:00
|
|
|
.finalize();
|
2022-09-13 12:23:56 +00:00
|
|
|
self.apply_transaction(transaction);
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
2022-09-11 04:59:01 +00:00
|
|
|
NodeScript::UpdateBody { path, changeset } => {
|
|
|
|
//
|
|
|
|
let transaction = TransactionBuilder::new(&self.node_tree)
|
|
|
|
.update_body_at_path(&path, changeset)
|
|
|
|
.finalize();
|
2022-09-13 12:23:56 +00:00
|
|
|
self.apply_transaction(transaction);
|
2022-09-11 04:59:01 +00:00
|
|
|
}
|
2022-09-14 01:23:33 +00:00
|
|
|
NodeScript::DeleteNode { path, rev_id } => {
|
|
|
|
let mut transaction = TransactionBuilder::new(&self.node_tree)
|
2022-09-09 07:05:41 +00:00
|
|
|
.delete_node_at_path(&path)
|
|
|
|
.finalize();
|
2022-09-14 01:23:33 +00:00
|
|
|
self.transform_transaction_if_need(&mut transaction, rev_id);
|
2022-09-13 12:23:56 +00:00
|
|
|
self.apply_transaction(transaction);
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
|
|
|
|
2022-10-20 03:35:11 +00:00
|
|
|
NodeScript::AssertNode { path, expected } => {
|
|
|
|
let node = self.node_tree.get_node_data_at_path(&path);
|
|
|
|
assert_eq!(node, expected.map(|e| e.into()));
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
2022-10-20 03:35:11 +00:00
|
|
|
NodeScript::AssertNumberOfChildrenAtPath { path, expected } => match path {
|
2022-09-09 07:05:41 +00:00
|
|
|
None => {
|
|
|
|
let len = self.node_tree.number_of_children(None);
|
2022-10-20 03:35:11 +00:00
|
|
|
assert_eq!(len, expected)
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
|
|
|
Some(path) => {
|
2022-09-11 04:59:01 +00:00
|
|
|
let node_id = self.node_tree.node_id_at_path(path).unwrap();
|
2022-09-09 07:05:41 +00:00
|
|
|
let len = self.node_tree.number_of_children(Some(node_id));
|
2022-10-20 03:35:11 +00:00
|
|
|
assert_eq!(len, expected)
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
|
|
|
},
|
2022-10-20 03:35:11 +00:00
|
|
|
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)
|
|
|
|
}
|
2022-09-11 04:59:01 +00:00
|
|
|
NodeScript::AssertNodeDelta { path, expected } => {
|
|
|
|
let node = self.node_tree.get_node_at_path(&path).unwrap();
|
2022-10-20 03:35:11 +00:00
|
|
|
if let Body::Delta(delta) = node.body.clone() {
|
2022-09-11 04:59:01 +00:00
|
|
|
debug_assert_eq!(delta, expected);
|
|
|
|
} else {
|
|
|
|
panic!("Node body type not match, expect Delta");
|
|
|
|
}
|
|
|
|
}
|
2022-10-20 03:35:11 +00:00
|
|
|
NodeScript::AssertTreeJSON { expected } => {
|
|
|
|
let json = serde_json::to_string(&self.node_tree).unwrap();
|
|
|
|
assert_eq!(json, expected)
|
|
|
|
}
|
2022-09-13 12:23:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn apply_transaction(&mut self, transaction: Transaction) {
|
|
|
|
self.rev_id += 1;
|
|
|
|
self.rev_operations.insert(self.rev_id, transaction.clone());
|
2022-09-14 13:04:12 +00:00
|
|
|
self.node_tree.apply_transaction(transaction).unwrap();
|
2022-09-13 12:23:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn transform_transaction_if_need(&mut self, transaction: &mut Transaction, rev_id: usize) {
|
|
|
|
if self.rev_id >= rev_id {
|
2022-09-14 04:03:52 +00:00
|
|
|
for rev_id in rev_id..=self.rev_id {
|
2022-09-13 12:23:56 +00:00
|
|
|
let old_transaction = self.rev_operations.get(&rev_id).unwrap();
|
2022-09-14 13:04:12 +00:00
|
|
|
*transaction = old_transaction.transform(transaction).unwrap();
|
2022-09-13 12:23:56 +00:00
|
|
|
}
|
2022-09-09 07:05:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|