mirror of
https://github.com/loro-dev/loro.git
synced 2025-01-23 05:24:51 +00:00
bef39ce6b5
* refactor: add detached editing config and prepare the architecture for editing detached doc * feat: subscribe for peer id change * fix: undo after checkout & add tests for detached editing * test: add fuzzer for detached editing * feat: expose detached editing configure to wasm * test: add wasm test for detached editing
361 lines
13 KiB
Rust
361 lines
13 KiB
Rust
use criterion::{criterion_group, criterion_main, Criterion};
|
|
|
|
#[cfg(feature = "test_utils")]
|
|
mod run {
|
|
use std::sync::Arc;
|
|
|
|
use super::*;
|
|
use bench_utils::TextAction;
|
|
use criterion::black_box;
|
|
use loro_common::LoroValue;
|
|
use loro_internal::LoroDoc;
|
|
|
|
pub fn b4(c: &mut Criterion) {
|
|
let actions = bench_utils::get_automerge_actions();
|
|
let mut b = c.benchmark_group("refactored direct_apply");
|
|
b.sample_size(10);
|
|
b.bench_function("B4", |b| {
|
|
b.iter(|| {
|
|
let loro = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
let mut txn = loro.txn().unwrap();
|
|
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
}
|
|
})
|
|
});
|
|
|
|
b.bench_function("B4 with 100K actors history", |b| {
|
|
let store = LoroDoc::default();
|
|
for i in 0..100_000 {
|
|
store.set_peer_id(i).unwrap();
|
|
let list = store.get_list("list");
|
|
let value: LoroValue = i.to_string().into();
|
|
let mut txn = store.txn().unwrap();
|
|
list.insert_with_txn(&mut txn, 0, value).unwrap();
|
|
txn.commit().unwrap();
|
|
}
|
|
|
|
let update = store.export_snapshot();
|
|
drop(store);
|
|
b.iter_batched(
|
|
|| {
|
|
let loro = LoroDoc::default();
|
|
loro.import(&update).unwrap();
|
|
loro
|
|
},
|
|
|loro| {
|
|
let text = loro.get_text("text");
|
|
let mut txn = loro.txn().unwrap();
|
|
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
}
|
|
},
|
|
criterion::BatchSize::SmallInput,
|
|
)
|
|
});
|
|
|
|
b.bench_function("B4 Obs", |b| {
|
|
b.iter(|| {
|
|
let loro = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
loro.subscribe_root(Arc::new(move |event| {
|
|
black_box(event);
|
|
}));
|
|
let mut txn = loro.txn().unwrap();
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
}
|
|
})
|
|
});
|
|
|
|
b.bench_function("B4 encode snapshot", |b| {
|
|
let loro = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
|
|
let mut n = 0;
|
|
let mut txn = loro.txn().unwrap();
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
if n == 10 {
|
|
n = 0;
|
|
drop(txn);
|
|
txn = loro.txn().unwrap();
|
|
}
|
|
n += 1;
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
}
|
|
txn.commit().unwrap();
|
|
|
|
b.iter(|| {
|
|
loro.export_snapshot();
|
|
});
|
|
});
|
|
|
|
b.bench_function("B4 encode updates", |b| {
|
|
let loro = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
|
|
let mut n = 0;
|
|
let mut txn = loro.txn().unwrap();
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
if n == 10 {
|
|
n = 0;
|
|
drop(txn);
|
|
txn = loro.txn().unwrap();
|
|
}
|
|
n += 1;
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
}
|
|
txn.commit().unwrap();
|
|
|
|
b.iter(|| {
|
|
loro.export_from(&Default::default());
|
|
});
|
|
});
|
|
|
|
b.bench_function("B4 decode snapshot", |b| {
|
|
let loro = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
let mut n = 0;
|
|
let mut txn = loro.txn().unwrap();
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
if n == 10 {
|
|
n = 0;
|
|
drop(txn);
|
|
txn = loro.txn().unwrap();
|
|
}
|
|
n += 1;
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
}
|
|
txn.commit().unwrap();
|
|
|
|
let data = loro.export_snapshot();
|
|
b.iter(|| {
|
|
let l = LoroDoc::new();
|
|
l.import(&data).unwrap();
|
|
});
|
|
});
|
|
|
|
b.bench_function("B4 import updates", |b| {
|
|
let loro = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
|
|
let mut n = 0;
|
|
let mut txn = loro.txn().unwrap();
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
if n == 10 {
|
|
n = 0;
|
|
drop(txn);
|
|
txn = loro.txn().unwrap();
|
|
}
|
|
n += 1;
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
}
|
|
txn.commit().unwrap();
|
|
|
|
let data = loro.export_from(&Default::default());
|
|
b.iter(|| {
|
|
let l = LoroDoc::new();
|
|
l.import(&data).unwrap();
|
|
});
|
|
});
|
|
|
|
// b.bench_function("B4 utf16", |b| {
|
|
// b.iter(|| {
|
|
// let loro = LoroDoc::new();
|
|
// let text = loro.get_text("text");
|
|
// let mut txn = loro.txn().unwrap();
|
|
|
|
// for TextAction { pos, ins, del } in actions.iter() {
|
|
// text.delete_utf16(&mut txn, *pos, *del).unwrap();
|
|
// text.insert_utf16(&mut txn, *pos, ins).unwrap();
|
|
// }
|
|
// })
|
|
// });
|
|
|
|
b.bench_function("B4_Per100_Txn", |b| {
|
|
b.iter(|| {
|
|
let loro = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
let mut n = 0;
|
|
let mut txn = loro.txn().unwrap();
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
if n == 100 {
|
|
n = 0;
|
|
drop(txn);
|
|
txn = loro.txn().unwrap();
|
|
}
|
|
n += 1;
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
}
|
|
})
|
|
});
|
|
|
|
b.bench_function("B4 One Op One Txn", |b| {
|
|
b.iter(|| {
|
|
let loro = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
{
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
let mut txn = loro.txn().unwrap();
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
txn.commit().unwrap();
|
|
}
|
|
}
|
|
})
|
|
});
|
|
|
|
b.bench_function("B4 One Op One Txn Obs", |b| {
|
|
b.iter(|| {
|
|
let loro = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
loro.subscribe_root(Arc::new(move |event| {
|
|
black_box(event);
|
|
}));
|
|
{
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
let mut txn = loro.txn().unwrap();
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
txn.commit().unwrap();
|
|
}
|
|
}
|
|
})
|
|
});
|
|
|
|
b.bench_function("B4DirectSync", |b| {
|
|
b.iter(|| {
|
|
let loro = LoroDoc::default();
|
|
let loro_b = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
{
|
|
let mut txn = loro.txn().unwrap();
|
|
text.delete_with_txn(&mut txn, *pos, *del).unwrap();
|
|
text.insert_with_txn(&mut txn, *pos, ins).unwrap();
|
|
}
|
|
|
|
loro_b
|
|
.import(&loro.export_from(&loro_b.oplog_vv()))
|
|
.unwrap();
|
|
}
|
|
})
|
|
});
|
|
|
|
drop(b);
|
|
let mut b = c.benchmark_group("refactored-sync");
|
|
b.bench_function("B4Parallel", |b| {
|
|
b.iter(|| {
|
|
let loro = LoroDoc::default();
|
|
let loro_b = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
let text2 = loro_b.get_text("text");
|
|
let mut i = 0;
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
let pos = *pos;
|
|
let del = *del;
|
|
i += 1;
|
|
if i > 1000 {
|
|
break;
|
|
}
|
|
|
|
{
|
|
let mut txn = loro.txn().unwrap();
|
|
text.delete_with_txn(&mut txn, pos, del).unwrap();
|
|
text.insert_with_txn(&mut txn, pos, ins).unwrap();
|
|
}
|
|
|
|
{
|
|
let mut txn = loro_b.txn().unwrap();
|
|
text2.delete_with_txn(&mut txn, pos, del).unwrap();
|
|
text2.insert_with_txn(&mut txn, pos, ins).unwrap();
|
|
}
|
|
loro_b
|
|
.import(&loro.export_from(&loro_b.oplog_vv()))
|
|
.unwrap();
|
|
loro.import(&loro_b.export_from(&loro.oplog_vv())).unwrap();
|
|
}
|
|
})
|
|
});
|
|
|
|
let b = b.sample_size(10);
|
|
b.bench_function("DecodeUpdates B4Parallel", |b| {
|
|
let loro = LoroDoc::default();
|
|
let loro_b = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
let text2 = loro_b.get_text("text");
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
let pos = *pos;
|
|
let del = *del;
|
|
{
|
|
let mut txn = loro.txn().unwrap();
|
|
text.delete_with_txn(&mut txn, pos, del).unwrap();
|
|
text.insert_with_txn(&mut txn, pos, ins).unwrap();
|
|
}
|
|
|
|
{
|
|
let mut txn = loro_b.txn().unwrap();
|
|
text2.delete_with_txn(&mut txn, pos, del).unwrap();
|
|
text2.insert_with_txn(&mut txn, pos, ins).unwrap();
|
|
}
|
|
loro_b
|
|
.import(&loro.export_from(&loro_b.oplog_vv()))
|
|
.unwrap();
|
|
loro.import(&loro_b.export_from(&loro.oplog_vv())).unwrap();
|
|
}
|
|
let data = loro.export_from(&Default::default());
|
|
b.iter(|| {
|
|
let loro = LoroDoc::default();
|
|
loro.import(&data).unwrap();
|
|
})
|
|
});
|
|
b.bench_function("DecodeSnapshot B4Parallel", |b| {
|
|
let loro = LoroDoc::default();
|
|
let loro_b = LoroDoc::default();
|
|
let text = loro.get_text("text");
|
|
let text2 = loro_b.get_text("text");
|
|
for TextAction { pos, ins, del } in actions.iter() {
|
|
let pos = *pos;
|
|
let del = *del;
|
|
{
|
|
let mut txn = loro.txn().unwrap();
|
|
text.delete_with_txn(&mut txn, pos, del).unwrap();
|
|
text.insert_with_txn(&mut txn, pos, ins).unwrap();
|
|
}
|
|
|
|
{
|
|
let mut txn = loro_b.txn().unwrap();
|
|
text2.delete_with_txn(&mut txn, pos, del).unwrap();
|
|
text2.insert_with_txn(&mut txn, pos, ins).unwrap();
|
|
}
|
|
loro_b
|
|
.import(&loro.export_from(&loro_b.oplog_vv()))
|
|
.unwrap();
|
|
loro.import(&loro_b.export_from(&loro.oplog_vv())).unwrap();
|
|
}
|
|
let data = loro.export_snapshot();
|
|
b.iter(|| {
|
|
let loro = LoroDoc::default();
|
|
loro.import(&data).unwrap();
|
|
})
|
|
});
|
|
}
|
|
}
|
|
pub fn dumb(_c: &mut Criterion) {}
|
|
|
|
#[cfg(feature = "test_utils")]
|
|
criterion_group!(benches, run::b4);
|
|
#[cfg(not(feature = "test_utils"))]
|
|
criterion_group!(benches, dumb);
|
|
criterion_main!(benches);
|