1use std::iter;
2use std::mem;
3use std::slice;
4
5#[derive(Debug, PartialEq, Eq, Ord, PartialOrd, Default, Copy, Clone)]
7struct Param<'k, 'v> {
8 key: &'k [u8],
9 value: &'v [u8],
10}
11
12impl<'k, 'v> Param<'k, 'v> {
13 fn key_str(&self) -> &'k str {
15 std::str::from_utf8(self.key).unwrap()
16 }
17
18 fn value_str(&self) -> &'v str {
19 std::str::from_utf8(self.value).unwrap()
20 }
21}
22
23#[derive(Debug, PartialEq, Eq, Ord, PartialOrd, Clone)]
43pub struct Params<'k, 'v> {
44 kind: ParamsKind<'k, 'v>,
45}
46
47const SMALL: usize = 3;
49
50#[derive(Debug, PartialEq, Eq, Ord, PartialOrd, Clone)]
51enum ParamsKind<'k, 'v> {
52 None,
53 Small([Param<'k, 'v>; SMALL], usize),
54 Large(Vec<Param<'k, 'v>>),
55}
56
57impl<'k, 'v> Params<'k, 'v> {
58 pub(crate) fn new() -> Self {
59 let kind = ParamsKind::None;
60 Self { kind }
61 }
62
63 pub fn len(&self) -> usize {
65 match &self.kind {
66 ParamsKind::None => 0,
67 ParamsKind::Small(_, len) => *len,
68 ParamsKind::Large(vec) => vec.len(),
69 }
70 }
71
72 pub(crate) fn truncate(&mut self, n: usize) {
73 match &mut self.kind {
74 ParamsKind::None => {}
75 ParamsKind::Small(_, len) => {
76 *len = n;
77 }
78 ParamsKind::Large(vec) => {
79 vec.truncate(n);
80 }
81 }
82 }
83
84 pub fn get(&self, key: impl AsRef<str>) -> Option<&'v str> {
86 let key = key.as_ref().as_bytes();
87
88 match &self.kind {
89 ParamsKind::None => None,
90 ParamsKind::Small(arr, len) => arr
91 .iter()
92 .take(*len)
93 .find(|param| param.key == key)
94 .map(Param::value_str),
95 ParamsKind::Large(vec) => vec
96 .iter()
97 .find(|param| param.key == key)
98 .map(Param::value_str),
99 }
100 }
101
102 pub fn iter(&self) -> ParamsIter<'_, 'k, 'v> {
104 ParamsIter::new(self)
105 }
106
107 pub fn is_empty(&self) -> bool {
109 match &self.kind {
110 ParamsKind::None => true,
111 ParamsKind::Small(_, len) => *len == 0,
112 ParamsKind::Large(vec) => vec.is_empty(),
113 }
114 }
115
116 pub(crate) fn push(&mut self, key: &'k [u8], value: &'v [u8]) {
118 #[cold]
119 fn drain_to_vec<T: Default>(len: usize, elem: T, arr: &mut [T; SMALL]) -> Vec<T> {
120 let mut vec = Vec::with_capacity(len + 1);
121 vec.extend(arr.iter_mut().map(mem::take));
122 vec.push(elem);
123 vec
124 }
125
126 let param = Param { key, value };
127 match &mut self.kind {
128 ParamsKind::None => {
129 self.kind = ParamsKind::Small([param, Param::default(), Param::default()], 1);
130 }
131 ParamsKind::Small(arr, len) => {
132 if *len == SMALL {
133 self.kind = ParamsKind::Large(drain_to_vec(*len, param, arr));
134 return;
135 }
136 arr[*len] = param;
137 *len += 1;
138 }
139 ParamsKind::Large(vec) => vec.push(param),
140 }
141 }
142}
143
144pub struct ParamsIter<'ps, 'k, 'v> {
146 kind: ParamsIterKind<'ps, 'k, 'v>,
147}
148
149impl<'ps, 'k, 'v> ParamsIter<'ps, 'k, 'v> {
150 fn new(params: &'ps Params<'k, 'v>) -> Self {
151 let kind = match ¶ms.kind {
152 ParamsKind::None => ParamsIterKind::None,
153 ParamsKind::Small(arr, len) => ParamsIterKind::Small(arr.iter().take(*len)),
154 ParamsKind::Large(vec) => ParamsIterKind::Large(vec.iter()),
155 };
156 Self { kind }
157 }
158}
159
160enum ParamsIterKind<'ps, 'k, 'v> {
161 None,
162 Small(iter::Take<slice::Iter<'ps, Param<'k, 'v>>>),
163 Large(slice::Iter<'ps, Param<'k, 'v>>),
164}
165
166impl<'ps, 'k, 'v> Iterator for ParamsIter<'ps, 'k, 'v> {
167 type Item = (&'k str, &'v str);
168
169 fn next(&mut self) -> Option<Self::Item> {
170 match self.kind {
171 ParamsIterKind::None => None,
172 ParamsIterKind::Small(ref mut iter) => {
173 iter.next().map(|p| (p.key_str(), p.value_str()))
174 }
175 ParamsIterKind::Large(ref mut iter) => {
176 iter.next().map(|p| (p.key_str(), p.value_str()))
177 }
178 }
179 }
180}
181
182#[cfg(test)]
183mod tests {
184 use super::*;
185
186 #[test]
187 fn no_alloc() {
188 assert_eq!(Params::new().kind, ParamsKind::None);
189 }
190
191 #[test]
192 fn heap_alloc() {
193 let vec = vec![
194 ("hello", "hello"),
195 ("world", "world"),
196 ("foo", "foo"),
197 ("bar", "bar"),
198 ("baz", "baz"),
199 ];
200
201 let mut params = Params::new();
202 for (key, value) in vec.clone() {
203 params.push(key.as_bytes(), value.as_bytes());
204 assert_eq!(params.get(key), Some(value));
205 }
206
207 match params.kind {
208 ParamsKind::Large(..) => {}
209 _ => panic!(),
210 }
211
212 assert!(params.iter().eq(vec.clone()));
213 }
214
215 #[test]
216 fn stack_alloc() {
217 let vec = vec![("hello", "hello"), ("world", "world"), ("baz", "baz")];
218
219 let mut params = Params::new();
220 for (key, value) in vec.clone() {
221 params.push(key.as_bytes(), value.as_bytes());
222 assert_eq!(params.get(key), Some(value));
223 }
224
225 match params.kind {
226 ParamsKind::Small(..) => {}
227 _ => panic!(),
228 }
229
230 assert!(params.iter().eq(vec.clone()));
231 }
232
233 #[test]
234 fn ignore_array_default() {
235 let params = Params::new();
236 assert!(params.get("").is_none());
237 }
238}