1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
use std::cmp::Ordering::{Equal, Greater, Less};
const ENTITY_LEN: usize = 16;
pub fn insert_entity_mut(entity_list: &mut Vec<u8>, entity: &[u8; ENTITY_LEN]) {
let len = entity_list.len();
assert_eq!(len.checked_rem(ENTITY_LEN), Some(0));
let mut size = len / ENTITY_LEN;
if size == 0 {
entity_list.splice(0..0, entity.iter().copied());
return;
}
let mut base = 0usize;
while size > 1 {
let half = size / 2;
let mid = base + half;
let start = mid * ENTITY_LEN;
let end = start + ENTITY_LEN;
let cmp = entity_list[start..end].cmp(entity);
base = if cmp == Greater { base } else { mid };
size -= half;
}
let start = base * ENTITY_LEN;
let end = start + ENTITY_LEN;
let cmp = entity_list[start..end].cmp(entity);
let offset = match cmp {
Equal => return,
Less => (base + 1) * ENTITY_LEN,
Greater => base * ENTITY_LEN,
};
entity_list.splice(offset..offset, entity.iter().copied());
}
pub enum ImmutResult {
Changed(Vec<u8>),
Unchanged,
}
pub fn insert_entity_immut(entity_list: &[u8], entity: &[u8; ENTITY_LEN]) -> ImmutResult {
let len = entity_list.len();
assert_eq!(len.checked_rem(ENTITY_LEN), Some(0));
let mut size = len / ENTITY_LEN;
if size == 0 {
let mut entity_list = entity_list.to_vec();
entity_list.splice(0..0, entity.iter().copied());
return ImmutResult::Changed(entity_list);
}
let mut base = 0usize;
while size > 1 {
let half = size / 2;
let mid = base + half;
let start = mid * ENTITY_LEN;
let end = start + ENTITY_LEN;
let cmp = entity_list[start..end].cmp(entity);
base = if cmp == Greater { base } else { mid };
size -= half;
}
let start = base * ENTITY_LEN;
let end = start + ENTITY_LEN;
let cmp = entity_list[start..end].cmp(entity);
let offset = match cmp {
Equal => return ImmutResult::Unchanged,
Less => (base + 1) * ENTITY_LEN,
Greater => base * ENTITY_LEN,
};
let mut entity_list = entity_list.to_vec();
entity_list.splice(offset..offset, entity.iter().copied());
ImmutResult::Changed(entity_list)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn insert_document_mut() {
let mut entity_list: Vec<u8> = Vec::new();
insert_entity_mut(&mut entity_list, b"aaaaaaaaaaaaaaaa");
insert_entity_mut(&mut entity_list, b"cccccccccccccccc");
insert_entity_mut(&mut entity_list, b"aaaaaaaaaaaaaaaa");
insert_entity_mut(&mut entity_list, b"bbbbbbbbbbbbbbbb");
assert_eq!(&entity_list[..], &b"aaaaaaaaaaaaaaaabbbbbbbbbbbbbbbbcccccccccccccccc"[..]);
}
#[test]
fn insert_document_immut() {
let mut entity_list = Vec::new();
match insert_entity_immut(&entity_list, b"aaaaaaaaaaaaaaaa") {
ImmutResult::Changed(l) => entity_list = l,
ImmutResult::Unchanged => {}
};
match insert_entity_immut(&entity_list, b"cccccccccccccccc") {
ImmutResult::Changed(l) => entity_list = l,
ImmutResult::Unchanged => {}
};
match insert_entity_immut(&entity_list, b"aaaaaaaaaaaaaaaa") {
ImmutResult::Changed(l) => entity_list = l,
ImmutResult::Unchanged => {}
};
match insert_entity_immut(&entity_list, b"bbbbbbbbbbbbbbbb") {
ImmutResult::Changed(l) => entity_list = l,
ImmutResult::Unchanged => {}
};
assert_eq!(entity_list, &b"aaaaaaaaaaaaaaaabbbbbbbbbbbbbbbbcccccccccccccccc"[..]);
}
}