1
1
use bonsaidb::{
2
    core::{
3
        connection::Connection,
4
        document::{BorrowedDocument, Document},
5
        schema::{
6
            view::{map::ViewMappedValue, EnumKey},
7
            Collection, CollectionName, DefaultSerialization, DefaultViewSerialization, Name,
8
            ReduceResult, View, ViewMapResult, ViewSchema,
9
        },
10
        Error,
11
    },
12
    local::{
13
        config::{Builder, StorageConfiguration},
14
        Database,
15
    },
16
};
17
use serde::{Deserialize, Serialize};
18

            
19
// ANCHOR: enum
20
#[derive(
21
4
    Serialize, Deserialize, Debug, num_derive::FromPrimitive, num_derive::ToPrimitive, Clone,
22
)]
23
pub enum Category {
24
    Rust,
25
    Cooking,
26
}
27

            
28
impl EnumKey for Category {}
29
// ANCHOR_END: enum
30

            
31
// ANCHOR: struct
32
#[derive(Serialize, Deserialize, Debug)]
33
pub struct BlogPost {
34
    pub title: String,
35
    pub body: String,
36
    pub category: Option<Category>,
37
}
38
// ANCHOR_END: struct
39

            
40
impl Collection for BlogPost {
41
30
    fn collection_name() -> CollectionName {
42
30
        CollectionName::new("view-example", "blog-post")
43
30
    }
44

            
45
2
    fn define_views(schema: &mut bonsaidb::core::schema::Schematic) -> Result<(), Error> {
46
2
        schema.define_view(BlogPostsByCategory)
47
2
    }
48
}
49

            
50
impl DefaultSerialization for BlogPost {}
51

            
52
2
#[derive(Debug, Clone)]
53
pub struct BlogPostsByCategory;
54

            
55
// ANCHOR: view
56
impl View for BlogPostsByCategory {
57
    type Collection = BlogPost;
58
    type Key = Option<Category>;
59
    type Value = u32;
60

            
61
13
    fn name(&self) -> Name {
62
13
        Name::new("by-category")
63
13
    }
64
}
65

            
66
impl ViewSchema for BlogPostsByCategory {
67
    type View = Self;
68

            
69
    fn map(&self, document: &BorrowedDocument<'_>) -> ViewMapResult<Self::View> {
70
        let post = document.contents::<BlogPost>()?;
71
        Ok(document.emit_key_and_value(post.category, 1))
72
    }
73

            
74
2
    fn reduce(
75
2
        &self,
76
2
        mappings: &[ViewMappedValue<Self::View>],
77
2
        _rereduce: bool,
78
2
    ) -> ReduceResult<Self::View> {
79
2
        Ok(mappings.iter().map(|mapping| mapping.value).sum())
80
2
    }
81
}
82
// ANCHOR_END: view
83

            
84
impl DefaultViewSerialization for BlogPostsByCategory {}
85

            
86
#[allow(unused_variables)]
87
1
#[tokio::test]
88
1
async fn example() -> Result<(), Error> {
89
1
    drop(tokio::fs::remove_dir_all("example.bonsaidb").await);
90
20
    let db = Database::open::<BlogPost>(StorageConfiguration::new("example.bonsaidb")).await?;
91
    // ANCHOR: query_with_docs
92
1
    let rust_posts = db
93
1
        .view::<BlogPostsByCategory>()
94
1
        .with_key(Some(Category::Rust))
95
2
        .query_with_docs()
96
2
        .await?;
97
    // ANCHOR_END: query_with_docs
98
    // ANCHOR: reduce_one_key
99
1
    let rust_post_count = db
100
1
        .view::<BlogPostsByCategory>()
101
1
        .with_key(Some(Category::Rust))
102
1
        .reduce()
103
        .await?;
104
    // ANCHOR_END: reduce_one_key
105
    // ANCHOR: reduce_multiple_keys
106
1
    let total_post_count = db.view::<BlogPostsByCategory>().reduce().await?;
107
    // ANCHOR_END: reduce_multiple_keys
108
1
    Ok(())
109
1
}