1
use std::{
2
    borrow::Cow,
3
    fmt::{Display, Write},
4
    ops::{Deref, DerefMut},
5
};
6

            
7
use arc_bytes::serde::{Bytes, CowBytes};
8
use serde::{Deserialize, Serialize};
9

            
10
use crate::schema::{view::map::Mappings, Key, Map, SerializedCollection};
11

            
12
mod collection;
13
mod revision;
14
pub use collection::*;
15
pub use revision::Revision;
16

            
17
/// The header of a `Document`.
18
1883374
#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, Eq)]
19
pub struct Header {
20
    /// The id of the Document. Unique across the collection `C`
21
    pub id: u64,
22

            
23
    /// The revision of the stored document.
24
    pub revision: Revision,
25
}
26

            
27
impl AsRef<Self> for Header {
28
11592
    fn as_ref(&self) -> &Self {
29
11592
        self
30
11592
    }
31
}
32

            
33
impl Header {
34
    /// Creates a `Map` result with an empty key and value.
35
    #[must_use]
36
26
    pub fn emit(&self) -> Mappings<(), ()> {
37
26
        self.emit_key_and_value((), ())
38
26
    }
39

            
40
    /// Creates a `Map` result with a `key` and an empty value.
41
    #[must_use]
42
36870
    pub fn emit_key<K: for<'a> Key<'a>>(&self, key: K) -> Mappings<K, ()> {
43
36870
        self.emit_key_and_value(key, ())
44
36870
    }
45

            
46
    /// Creates a `Map` result with `value` and an empty key.
47
    #[must_use]
48
1
    pub fn emit_value<Value>(&self, value: Value) -> Mappings<(), Value> {
49
1
        self.emit_key_and_value((), value)
50
1
    }
51

            
52
    /// Creates a `Map` result with a `key` and `value`.
53
    #[must_use]
54
61969
    pub fn emit_key_and_value<K: for<'a> Key<'a>, Value>(
55
61969
        &self,
56
61969
        key: K,
57
61969
        value: Value,
58
61969
    ) -> Mappings<K, Value> {
59
61969
        Mappings::Simple(Some(Map::new(self.clone(), key, value)))
60
61969
    }
61
}
62

            
63
impl Display for Header {
64
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
65
1
        self.id.fmt(f)?;
66
1
        f.write_char('@')?;
67
1
        self.revision.fmt(f)
68
1
    }
69
}
70

            
71
/// Contains a serialized document in the database.
72
941688
#[derive(Clone, Debug, Serialize, Deserialize)]
73
pub struct BorrowedDocument<'a> {
74
    /// The header of the document, which contains the id and `Revision`.
75
    pub header: Header,
76

            
77
    /// The serialized bytes of the stored item.
78
    #[serde(borrow)]
79
    pub contents: CowBytes<'a>,
80
}
81

            
82
/// Contains a serialized document in the database.
83
22540
#[derive(Clone, Debug, Serialize, Deserialize)]
84
pub struct OwnedDocument {
85
    /// The header of the document, which contains the id and `Revision`.
86
    pub header: Header,
87

            
88
    /// The serialized bytes of the stored item.
89
    pub contents: Bytes,
90
}
91

            
92
/// Common interface of a document in `BonsaiDb`.
93
pub trait Document<'a>:
94
    Deref<Target = Header> + DerefMut + AsRef<Header> + AsRef<[u8]> + Sized
95
{
96
    /// The bytes type used in the interface.
97
    type Bytes;
98

            
99
    /// Creates a new document with `contents`.
100
    #[must_use]
101
    fn new(id: u64, contents: impl Into<Self::Bytes>) -> Self;
102
    /// Creates a new document with serialized bytes from `contents`.
103
    fn with_contents<S: SerializedCollection<Contents = S>>(
104
        id: u64,
105
        contents: &S,
106
    ) -> Result<Self, crate::Error>;
107
    /// Retrieves `contents` through deserialization into the type `D`.
108
    fn contents<D>(&self) -> Result<D::Contents, crate::Error>
109
    where
110
        D: SerializedCollection<Contents = D>;
111
    /// Serializes and stores `contents` into this document.
112
    fn set_contents<S: SerializedCollection<Contents = S>>(
113
        &mut self,
114
        contents: &S,
115
    ) -> Result<(), crate::Error>;
116
    /// Creates a new revision.
117
    ///
118
    /// **WARNING: This normally should not be used** outside of implementing a
119
    /// backend for `BonsaiDb`. To update a document, use `set_contents()` and
120
    /// send the document with the existing `Revision` information.
121
    #[must_use]
122
    fn create_new_revision(&self, contents: impl Into<Self::Bytes>) -> Option<Self>;
123
}
124

            
125
impl<'a> AsRef<[u8]> for BorrowedDocument<'a> {
126
92
    fn as_ref(&self) -> &[u8] {
127
92
        &self.contents
128
92
    }
129
}
130

            
131
impl<'a> Document<'a> for BorrowedDocument<'a> {
132
    type Bytes = CowBytes<'a>;
133
396232
    fn new(id: u64, contents: impl Into<CowBytes<'a>>) -> Self {
134
396232
        let contents = contents.into();
135
396232
        let revision = Revision::new(&contents);
136
396232
        Self {
137
396232
            header: Header { id, revision },
138
396232
            contents,
139
396232
        }
140
396232
    }
141

            
142
511
    fn with_contents<S: SerializedCollection<Contents = S>>(
143
511
        id: u64,
144
511
        contents: &S,
145
511
    ) -> Result<Self, crate::Error> {
146
511
        let contents = <S as SerializedCollection>::serialize(contents)?;
147
511
        Ok(Self::new(id, contents))
148
511
    }
149

            
150
2556
    fn contents<D>(&self) -> Result<D::Contents, crate::Error>
151
2556
    where
152
2556
        D: SerializedCollection<Contents = D>,
153
2556
    {
154
2556
        <D as SerializedCollection>::deserialize(&self.contents)
155
2556
    }
156

            
157
    fn set_contents<S: SerializedCollection<Contents = S>>(
158
        &mut self,
159
        contents: &S,
160
    ) -> Result<(), crate::Error> {
161
        self.contents = CowBytes::from(<S as SerializedCollection>::serialize(contents)?);
162
        Ok(())
163
    }
164

            
165
85837
    fn create_new_revision(&self, contents: impl Into<Self::Bytes>) -> Option<Self> {
166
85837
        let contents = contents.into();
167
85837
        self.header
168
85837
            .revision
169
85837
            .next_revision(&contents)
170
85837
            .map(|revision| Self {
171
85770
                header: Header {
172
85770
                    id: self.header.id,
173
85770
                    revision,
174
85770
                },
175
85770
                contents,
176
85837
            })
177
85837
    }
178
}
179

            
180
impl Document<'static> for OwnedDocument {
181
    type Bytes = Vec<u8>;
182

            
183
    fn new(id: u64, contents: impl Into<Self::Bytes>) -> Self {
184
        let contents = Bytes(contents.into());
185
        Self {
186
            header: Header {
187
                id,
188
                revision: Revision::new(&contents),
189
            },
190
            contents,
191
        }
192
    }
193

            
194
    fn with_contents<S: SerializedCollection<Contents = S>>(
195
        id: u64,
196
        contents: &S,
197
    ) -> Result<Self, crate::Error> {
198
        BorrowedDocument::with_contents(id, contents).map(BorrowedDocument::into_owned)
199
    }
200

            
201
1028
    fn contents<D>(&self) -> Result<D::Contents, crate::Error>
202
1028
    where
203
1028
        D: SerializedCollection<Contents = D>,
204
1028
    {
205
1028
        <D as SerializedCollection>::deserialize(&self.contents)
206
1028
    }
207

            
208
517
    fn set_contents<S: SerializedCollection<Contents = S>>(
209
517
        &mut self,
210
517
        contents: &S,
211
517
    ) -> Result<(), crate::Error> {
212
517
        self.contents = Bytes::from(<S as SerializedCollection>::serialize(contents)?);
213
517
        Ok(())
214
517
    }
215

            
216
    fn create_new_revision(&self, contents: impl Into<Self::Bytes>) -> Option<Self> {
217
        let contents = Bytes(contents.into());
218
        self.header
219
            .revision
220
            .next_revision(&contents)
221
            .map(|revision| Self {
222
                header: Header {
223
                    id: self.header.id,
224
                    revision,
225
                },
226
                contents,
227
            })
228
    }
229
}
230

            
231
impl AsRef<Header> for OwnedDocument {
232
11707
    fn as_ref(&self) -> &Header {
233
11707
        &self.header
234
11707
    }
235
}
236

            
237
impl Deref for OwnedDocument {
238
    type Target = Header;
239

            
240
90459
    fn deref(&self) -> &Self::Target {
241
90459
        &self.header
242
90459
    }
243
}
244

            
245
impl DerefMut for OwnedDocument {
246
89884
    fn deref_mut(&mut self) -> &mut Self::Target {
247
89884
        &mut self.header
248
89884
    }
249
}
250

            
251
impl AsRef<[u8]> for OwnedDocument {
252
90252
    fn as_ref(&self) -> &[u8] {
253
90252
        &self.contents
254
90252
    }
255
}
256

            
257
impl<'a> BorrowedDocument<'a> {
258
    /// Converts this document to an owned document.
259
    #[must_use]
260
375544
    pub fn into_owned(self) -> OwnedDocument {
261
375544
        OwnedDocument {
262
375544
            header: self.header,
263
375544
            contents: Bytes::from(self.contents),
264
375544
        }
265
375544
    }
266
}
267

            
268
impl<'a> AsRef<Header> for BorrowedDocument<'a> {
269
    fn as_ref(&self) -> &Header {
270
        &self.header
271
    }
272
}
273

            
274
impl<'a> Deref for BorrowedDocument<'a> {
275
    type Target = Header;
276

            
277
14335
    fn deref(&self) -> &Self::Target {
278
14335
        &self.header
279
14335
    }
280
}
281

            
282
impl<'a> DerefMut for BorrowedDocument<'a> {
283
    fn deref_mut(&mut self) -> &mut Self::Target {
284
        &mut self.header
285
    }
286
}
287

            
288
/// The ID of an encryption key.
289
76438
#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, Eq)]
290
pub enum KeyId {
291
    /// A key with no id.
292
    None,
293
    /// The master key of the vault.
294
    Master,
295
    /// A specific named key in the vault.
296
    Id(Cow<'static, str>),
297
}
298

            
299
1
#[test]
300
1
fn emissions_tests() -> Result<(), crate::Error> {
301
    use crate::{schema::Map, test_util::Basic};
302

            
303
1
    let doc = BorrowedDocument::with_contents(1, &Basic::default())?;
304

            
305
1
    assert_eq!(
306
1
        doc.emit(),
307
1
        Mappings::Simple(Some(Map::new(doc.header.clone(), (), ())))
308
1
    );
309

            
310
1
    assert_eq!(
311
1
        doc.emit_key(1),
312
1
        Mappings::Simple(Some(Map::new(doc.header.clone(), 1, ())))
313
1
    );
314

            
315
1
    assert_eq!(
316
1
        doc.emit_value(1),
317
1
        Mappings::Simple(Some(Map::new(doc.header.clone(), (), 1)))
318
1
    );
319

            
320
1
    assert_eq!(
321
1
        doc.emit_key_and_value(1, 2),
322
1
        Mappings::Simple(Some(Map::new(doc.header.clone(), 1, 2)))
323
1
    );
324

            
325
1
    Ok(())
326
1
}
327

            
328
1
#[test]
329
1
fn chained_mappings_test() -> Result<(), crate::Error> {
330
    use crate::{schema::Map, test_util::Basic};
331

            
332
1
    let doc = BorrowedDocument::with_contents(1, &Basic::default())?;
333

            
334
1
    assert_eq!(
335
1
        doc.emit().and(doc.emit()),
336
1
        Mappings::List(vec![
337
1
            Map::new(doc.header.clone(), (), ()),
338
1
            Map::new(doc.header.clone(), (), ())
339
1
        ])
340
1
    );
341

            
342
1
    Ok(())
343
1
}
344

            
345
1
#[test]
346
1
fn header_display_test() {
347
1
    let original_contents = b"one";
348
1
    let revision = Revision::new(original_contents);
349
1
    let header = Header { id: 42, revision };
350
1
    assert_eq!(
351
1
        header.to_string(),
352
1
        "42@0-7692c3ad3540bb803c020b3aee66cd8887123234ea0c6e7143c0add73ff431ed"
353
1
    );
354
1
}