use std::{ cell::{LazyCell, OnceCell}, ops::Deref, time::Instant, }; use bench_utils::TextAction; use criterion::{black_box, criterion_group, criterion_main, Criterion}; use dev_utils::ByteSize; use loro::LoroDoc; fn bench_text(c: &mut Criterion) { use bench_utils::TextAction; let actions = LazyCell::new(bench_utils::get_automerge_actions); let doc: OnceCell = OnceCell::new(); let doc_snapshot: OnceCell> = OnceCell::new(); let doc_x100_snapshot: OnceCell> = OnceCell::new(); let mut g = c.benchmark_group("text"); g.bench_function("B4 apply", |b| { b.iter_custom(|iters| { let start = Instant::now(); let actions: &Vec = actions.deref(); for _ in 0..iters { let loro = apply_text_actions(actions, 1); if doc.get().is_none() { doc.set(loro).unwrap(); } } start.elapsed() }); }); g.bench_function("B4 export fast snapshot (has cache)", |b| { b.iter_batched( || { if doc.get().is_none() { let the_doc = apply_text_actions(&actions, 1); doc.set(the_doc).unwrap(); } let doc = doc.get().unwrap(); doc }, |doc| { doc.export_fast_snapshot(); }, criterion::BatchSize::SmallInput, ) }); g.bench_function("B4 export fast snapshot (no cache)", |b| { b.iter_batched( || apply_text_actions(&actions, 1), |doc| { doc.export_fast_snapshot(); }, criterion::BatchSize::SmallInput, ) }); g.bench_function("B4 load", |b| { b.iter_batched( || { if doc.get().is_none() { let the_doc = apply_text_actions(&actions, 1); doc.set(the_doc).unwrap(); } if doc_snapshot.get().is_none() { let doc = doc.get().unwrap(); let snapshot = doc.export_fast_snapshot(); println!("B4 fast_snapshot size: {:?}", ByteSize(snapshot.len())); doc_snapshot.set(snapshot).unwrap(); } doc_snapshot.get().unwrap() }, |snapshot| { let doc = LoroDoc::new(); doc.import(snapshot).unwrap(); }, criterion::BatchSize::LargeInput, ) }); g.bench_function("B4x100 load", |b| { b.iter_batched( || { if doc_x100_snapshot.get().is_none() { let doc = apply_text_actions(&actions, 100); let snapshot = doc.export_fast_snapshot(); println!("B4x100 fast_snapshot size: {:?}", ByteSize(snapshot.len())); doc_x100_snapshot.set(snapshot).unwrap(); } doc_x100_snapshot.get().unwrap() }, |snapshot| { let doc = LoroDoc::new(); doc.import(snapshot).unwrap(); }, criterion::BatchSize::LargeInput, ) }); g.bench_function("B4x100 load and get value", |b| { b.iter_batched( || { if doc_x100_snapshot.get().is_none() { let doc = apply_text_actions(&actions, 100); let snapshot = doc.export_fast_snapshot(); println!("B4x100 fast_snapshot size: {:?}", ByteSize(snapshot.len())); doc_x100_snapshot.set(snapshot).unwrap(); } doc_x100_snapshot.get().unwrap() }, |snapshot| { let doc = LoroDoc::new(); doc.import(snapshot).unwrap(); let text = doc.get_text("text"); black_box(text.to_string()); }, criterion::BatchSize::LargeInput, ) }); } fn apply_text_actions(actions: &[bench_utils::TextAction], n: usize) -> LoroDoc { let loro = LoroDoc::new(); let text = loro.get_text("text"); for _ in 0..n { for TextAction { del, ins, pos } in actions.iter() { text.delete(*pos, *del).unwrap(); text.insert(*pos, ins).unwrap(); } } loro } criterion_group!(benches, bench_text); criterion_main!(benches);