typed_store/
lib.rs

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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
// Copyright (c) 2021, Facebook, Inc. and its affiliates
// Copyright (c) Mysten Labs, Inc.
// SPDX-License-Identifier: Apache-2.0
#![warn(
    future_incompatible,
    nonstandard_style,
    rust_2018_idioms,
    rust_2021_compatibility
)]

// Re-export rocksdb so that consumers can use the version of rocksdb via typed-store
pub use rocksdb;

pub mod traits;
pub use traits::Map;
pub mod metrics;
pub mod rocks;
pub use metrics::DBMetrics;
pub use typed_store_error::TypedStoreError;

pub type StoreError = typed_store_error::TypedStoreError;

/// A helper macro to simplify common operations for opening and debugging TypedStore (currently internally structs of DBMaps)
/// It operates on a struct where all the members are of Store<K, V> or DBMap<K, V>
/// `TypedStoreDebug` traits are then derived
/// The main features are:
/// 1. Flexible configuration of each table (column family) via defaults and overrides
/// 2. Auto-generated `open` routine
/// 3. Auto-generated `read_only_mode` handle
/// 4. Auto-generated memory stats method
/// 5. Other convenience features
///
/// 1. Flexible configuration:
///     a. Static options specified at struct definition
///
/// The definer of the struct can specify the default options for each table using annotations
/// We can also supply column family options on the default ones
/// A user defined function of signature () -> Options can be provided for each table
/// If an override function is not specified, the default in `typed_store::rocks::default_db_options` is used
/// ```
/// use typed_store::rocks::DBOptions;
/// use typed_store::rocks::DBMap;
/// use typed_store::rocks::MetricConf;
/// use typed_store::DBMapUtils;
/// use typed_store::traits::TypedStoreDebug;
/// use typed_store::traits::TableSummary;
/// use core::fmt::Error;
/// /// Define a struct with all members having type DBMap<K, V>
///
/// fn custom_fn_name1() -> DBOptions {DBOptions::default()}
/// fn custom_fn_name2() -> DBOptions {
///     let mut op = custom_fn_name1();
///     op.options.set_write_buffer_size(123456);
///     op
/// }
/// #[derive(DBMapUtils)]
/// struct Tables {
///     /// Specify custom options function `custom_fn_name1`
///     #[default_options_override_fn = "custom_fn_name1"]
///     table1: DBMap<String, String>,
///     #[default_options_override_fn = "custom_fn_name2"]
///     table2: DBMap<i32, String>,
///     // Nothing specified so `typed_store::rocks::default_db_options` is used
///     table3: DBMap<i32, String>,
///     #[default_options_override_fn = "custom_fn_name1"]
///     table4: DBMap<i32, String>,
/// }
///
/// // b. Options specified by DB opener
/// // For finer control, we also allow the opener of the DB to specify their own options which override the defaults set by the definer
/// // This is done via a configurator which gives one a struct with field similarly named as that of the DB, but of type Options
///
/// #[tokio::main]
/// async fn main() -> Result<(), Error> {
/// // Get a configurator for this table
/// let mut config = Tables::configurator();
/// // Config table 1
/// config.table1 = DBOptions::default();
/// config.table1.options.create_if_missing(true);
/// config.table1.options.set_write_buffer_size(123456);
///
/// let primary_path = tempfile::tempdir().expect("Failed to open temporary directory").into_path();
///
/// // We can then open the DB with the configs
/// let _ = Tables::open_tables_read_write(primary_path, MetricConf::default(), None, Some(config.build()));
/// Ok(())
/// }
///
///```
///
/// 2. Auto-generated `open` routine
///     The function `open_tables_read_write` is generated which allows for specifying DB wide options and custom table configs as mentioned above
///
/// 3. Auto-generated `read_only_mode` handle
///     This mode provides handle struct which opens the DB in read only mode and has certain features like dumping and counting the keys in the tables
///
/// Use the function `Tables::get_read_only_handle` which returns a handle that only allows read only features
///```
/// use typed_store::rocks::DBOptions;
/// use typed_store::rocks::DBMap;
/// use typed_store::DBMapUtils;
/// use typed_store::traits::TypedStoreDebug;
/// use core::fmt::Error;
/// use typed_store::traits::TableSummary;
/// /// Define a struct with all members having type DBMap<K, V>
///
/// fn custom_fn_name1() -> DBOptions {DBOptions::default()}
/// fn custom_fn_name2() -> DBOptions {
///     let mut op = custom_fn_name1();
///     op.options.set_write_buffer_size(123456);
///     op
/// }
/// #[derive(DBMapUtils)]
/// struct Tables {
///     /// Specify custom options function `custom_fn_name1`
///     #[default_options_override_fn = "custom_fn_name1"]
///     table1: DBMap<String, String>,
///     #[default_options_override_fn = "custom_fn_name2"]
///     table2: DBMap<i32, String>,
///     // Nothing specified so `typed_store::rocks::default_db_options` is used
///     table3: DBMap<i32, String>,
///     #[default_options_override_fn = "custom_fn_name1"]
///     table4: DBMap<i32, String>,
/// }
/// #[tokio::main]
/// async fn main() -> Result<(), Error> {
///
/// use typed_store::rocks::MetricConf;let primary_path = tempfile::tempdir().expect("Failed to open temporary directory").into_path();
/// let _ = Tables::open_tables_read_write(primary_path.clone(), typed_store::rocks::MetricConf::default(), None, None);
///
/// // Get the read only handle
/// let read_only_handle = Tables::get_read_only_handle(primary_path, None, None, MetricConf::default());
/// // Use this handle for dumping
/// let ret = read_only_handle.dump("table2", 100, 0).unwrap();
/// let key_count = read_only_handle.count_keys("table1").unwrap();
/// Ok(())
/// }
/// ```
/// 4. Auto-generated memory stats method
///     `self.get_memory_usage` is derived to provide memory and cache usage
///
/// 5. Other convenience features
///     `Tables::describe_tables` is used to get a list of the table names and key-value types as string in a BTreeMap
///
/// // Bad usage example
/// // Structs fields most only be of type Store<K, V> or DMBap<K, V>
/// // This will fail to compile with error `All struct members must be of type Store<K, V> or DMBap<K, V>`
/// // #[derive(DBMapUtils)]
/// // struct BadTables {
/// //     table1: Store<String, String>,
/// //     bad_field: u32,
/// // #}
pub use typed_store_derive::DBMapUtils;