Lines
77.05 %
Functions
37.85 %
Branches
100 %
use std::{
borrow::Cow,
fmt::{Display, Write},
ops::{Deref, DerefMut},
};
use arc_bytes::serde::{Bytes, CowBytes};
use serde::{Deserialize, Serialize};
use crate::schema::{view::map::Mappings, Key, Map, SerializedCollection};
mod collection;
mod revision;
pub use collection::*;
pub use revision::Revision;
/// The header of a `Document`.
#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, Eq)]
pub struct Header {
/// The id of the Document. Unique across the collection `C`
pub id: u64,
/// The revision of the stored document.
pub revision: Revision,
}
impl AsRef<Self> for Header {
fn as_ref(&self) -> &Self {
self
impl Header {
/// Creates a `Map` result with an empty key and value.
#[must_use]
pub fn emit(&self) -> Mappings<(), ()> {
self.emit_key_and_value((), ())
/// Creates a `Map` result with a `key` and an empty value.
pub fn emit_key<K: for<'a> Key<'a>>(&self, key: K) -> Mappings<K, ()> {
self.emit_key_and_value(key, ())
/// Creates a `Map` result with `value` and an empty key.
pub fn emit_value<Value>(&self, value: Value) -> Mappings<(), Value> {
self.emit_key_and_value((), value)
/// Creates a `Map` result with a `key` and `value`.
pub fn emit_key_and_value<K: for<'a> Key<'a>, Value>(
&self,
key: K,
value: Value,
) -> Mappings<K, Value> {
Mappings::Simple(Some(Map::new(self.clone(), key, value)))
impl Display for Header {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.id.fmt(f)?;
f.write_char('@')?;
self.revision.fmt(f)
/// Contains a serialized document in the database.
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct BorrowedDocument<'a> {
/// The header of the document, which contains the id and `Revision`.
pub header: Header,
/// The serialized bytes of the stored item.
#[serde(borrow)]
pub contents: CowBytes<'a>,
pub struct OwnedDocument {
pub contents: Bytes,
/// Common interface of a document in BonsaiDb.
pub trait Document<'a>:
Deref<Target = Header> + DerefMut + AsRef<Header> + AsRef<[u8]> + Sized
{
/// The bytes type used in the interface.
type Bytes;
/// Creates a new document with `contents`.
fn new(id: u64, contents: impl Into<Self::Bytes>) -> Self;
/// Creates a new document with serialized bytes from `contents`.
fn with_contents<S: SerializedCollection<Contents = S>>(
id: u64,
contents: &S,
) -> Result<Self, crate::Error>;
/// Retrieves `contents` through deserialization into the type `D`.
fn contents<D>(&self) -> Result<D::Contents, crate::Error>
where
D: SerializedCollection<Contents = D>;
/// Serializes and stores `contents` into this document.
fn set_contents<S: SerializedCollection<Contents = S>>(
&mut self,
) -> Result<(), crate::Error>;
/// Creates a new revision.
///
/// **WARNING: This normally should not be used** outside of implementing a
/// backend for BonsaiDb. To update a document, use `set_contents()` and
/// send the document with the existing `Revision` information.
fn create_new_revision(&self, contents: impl Into<Self::Bytes>) -> Option<Self>;
impl<'a> AsRef<[u8]> for BorrowedDocument<'a> {
fn as_ref(&self) -> &[u8] {
&self.contents
impl<'a> Document<'a> for BorrowedDocument<'a> {
type Bytes = CowBytes<'a>;
fn new(id: u64, contents: impl Into<CowBytes<'a>>) -> Self {
let contents = contents.into();
let revision = Revision::new(&contents);
Self {
header: Header { id, revision },
contents,
) -> Result<Self, crate::Error> {
let contents = <S as SerializedCollection>::serialize(contents)?;
Ok(Self::new(id, contents))
D: SerializedCollection<Contents = D>,
<D as SerializedCollection>::deserialize(&self.contents)
) -> Result<(), crate::Error> {
self.contents = CowBytes::from(<S as SerializedCollection>::serialize(contents)?);
Ok(())
fn create_new_revision(&self, contents: impl Into<Self::Bytes>) -> Option<Self> {
self.header
.revision
.next_revision(&contents)
.map(|revision| Self {
header: Header {
id: self.header.id,
revision,
},
})
impl Document<'static> for OwnedDocument {
type Bytes = Vec<u8>;
fn new(id: u64, contents: impl Into<Self::Bytes>) -> Self {
let contents = Bytes(contents.into());
id,
revision: Revision::new(&contents),
BorrowedDocument::with_contents(id, contents).map(BorrowedDocument::into_owned)
self.contents = Bytes::from(<S as SerializedCollection>::serialize(contents)?);
impl AsRef<Header> for OwnedDocument {
fn as_ref(&self) -> &Header {
&self.header
impl Deref for OwnedDocument {
type Target = Header;
fn deref(&self) -> &Self::Target {
impl DerefMut for OwnedDocument {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.header
impl AsRef<[u8]> for OwnedDocument {
impl<'a> BorrowedDocument<'a> {
/// Converts this document to an owned document.
pub fn into_owned(self) -> OwnedDocument {
OwnedDocument {
header: self.header,
contents: Bytes::from(self.contents),
impl<'a> AsRef<Header> for BorrowedDocument<'a> {
impl<'a> Deref for BorrowedDocument<'a> {
impl<'a> DerefMut for BorrowedDocument<'a> {
/// The ID of an encryption key.
pub enum KeyId {
/// A key with no id.
None,
/// The master key of the vault.
Master,
/// A specific named key in the vault.
Id(Cow<'static, str>),
#[test]
fn emissions_tests() -> Result<(), crate::Error> {
use crate::{schema::Map, test_util::Basic};
let doc = BorrowedDocument::with_contents(1, &Basic::default())?;
assert_eq!(
doc.emit(),
Mappings::Simple(Some(Map::new(doc.header.clone(), (), ())))
);
doc.emit_key(1),
Mappings::Simple(Some(Map::new(doc.header.clone(), 1, ())))
doc.emit_value(1),
Mappings::Simple(Some(Map::new(doc.header.clone(), (), 1)))
doc.emit_key_and_value(1, 2),
Mappings::Simple(Some(Map::new(doc.header.clone(), 1, 2)))
fn chained_mappings_test() -> Result<(), crate::Error> {
doc.emit().and(doc.emit()),
Mappings::List(vec![
Map::new(doc.header.clone(), (), ()),
Map::new(doc.header.clone(), (), ())
])
fn header_display_test() {
let original_contents = b"one";
let revision = Revision::new(original_contents);
let header = Header { id: 42, revision };
header.to_string(),
"42@0-7692c3ad3540bb803c020b3aee66cd8887123234ea0c6e7143c0add73ff431ed"