1
1
//! Core functionality and types for `BonsaiDb`.
2

            
3
#![forbid(unsafe_code)]
4
#![warn(
5
    clippy::cargo,
6
    missing_docs,
7
    // clippy::missing_docs_in_private_items,
8
    clippy::nursery,
9
    clippy::pedantic,
10
    future_incompatible,
11
    rust_2018_idioms,
12
)]
13
#![allow(
14
    clippy::missing_errors_doc, // TODO clippy::missing_errors_doc
15
    clippy::option_if_let_else,
16
    clippy::module_name_repetitions,
17
)]
18

            
19
/// Types for creating and validating permissions.
20
pub mod permissions;
21

            
22
/// Database administration types and functionality.
23
pub mod admin;
24
/// Types for interacting with `BonsaiDb`.
25
pub mod connection;
26
/// Types for interacting with `Document`s.
27
pub mod document;
28
/// Limits used within `BonsaiDb`.
29
pub mod limits;
30
/// Types for defining database schema.
31
pub mod schema;
32
/// Types for executing transactions.
33
pub mod transaction;
34

            
35
/// Types for utilizing a lightweight atomic Key-Value store.
36
pub mod keyvalue;
37

            
38
/// Traits for tailoring a server.
39
pub mod custom_api;
40

            
41
#[cfg(feature = "networking")]
42
/// Types for implementing the `BonsaiDb` network protocol.
43
pub mod networking;
44

            
45
/// Types for Publish/Subscribe (`PubSub`) messaging.
46
pub mod pubsub;
47

            
48
use std::string::FromUtf8Error;
49

            
50
pub use actionable;
51
pub use arc_bytes;
52
pub use async_trait;
53
pub use circulate;
54
pub use num_traits;
55
pub use ordered_varint;
56
use schema::{view, CollectionName, SchemaName, ViewName};
57
use serde::{Deserialize, Serialize};
58
pub use transmog;
59
pub use transmog_pot;
60

            
61
use crate::{document::Header, schema::InsertError};
62

            
63
/// an enumeration of errors that this crate can produce
64
3108
#[derive(Clone, thiserror::Error, Debug, Serialize, Deserialize)]
65
pub enum Error {
66
    /// The database named `database_name` was created with a different schema
67
    /// (`stored_schema`) than provided (`schema`).
68
    #[error(
69
        "database '{database_name}' was created with schema '{stored_schema}', not '{schema}'"
70
    )]
71
    SchemaMismatch {
72
        /// The name of the database being accessed.
73
        database_name: String,
74

            
75
        /// The schema provided for the database.
76
        schema: SchemaName,
77

            
78
        /// The schema stored for the database.
79
        stored_schema: SchemaName,
80
    },
81

            
82
    /// The [`SchemaName`] returned has already been registered.
83
    #[error("schema '{0}' was already registered")]
84
    SchemaAlreadyRegistered(SchemaName),
85

            
86
    /// The [`SchemaName`] requested was not registered.
87
    #[error("schema '{0}' is not registered")]
88
    SchemaNotRegistered(SchemaName),
89

            
90
    /// An invalid database name was specified. See
91
    /// [`StorageConnection::create_database()`](connection::StorageConnection::create_database)
92
    /// for database name requirements.
93
    #[error("invalid database name: {0}")]
94
    InvalidDatabaseName(String),
95

            
96
    /// The database name given was not found.
97
    #[error("database '{0}' was not found")]
98
    DatabaseNotFound(String),
99

            
100
    /// The database name already exists.
101
    #[error("a database with name '{0}' already exists")]
102
    DatabaseNameAlreadyTaken(String),
103

            
104
    /// An error from interacting with local storage.
105
    #[error("error from storage: {0}")]
106
    Database(String),
107

            
108
    /// An error serializing data.
109
    #[error("error serializing: {0}")]
110
    Serialization(String),
111

            
112
    /// An error from interacting with a server.
113
    #[error("error from server: {0}")]
114
    Server(String),
115

            
116
    /// An error occurred from the QUIC transport layer.
117
    #[error("a transport error occurred: '{0}'")]
118
    Transport(String),
119

            
120
    /// An error occurred from the websocket transport layer.
121
    #[cfg(feature = "websockets")]
122
    #[error("a websocket error occurred: '{0}'")]
123
    Websocket(String),
124

            
125
    /// An error occurred from networking.
126
    #[cfg(feature = "networking")]
127
    #[error("a networking error occurred: '{0}'")]
128
    Networking(networking::Error),
129

            
130
    /// An error occurred from IO.
131
    #[error("an io error occurred: '{0}'")]
132
    Io(String),
133

            
134
    /// An error occurred with the provided configuration options.
135
    #[error("a configuration error occurred: '{0}'")]
136
    Configuration(String),
137

            
138
    /// An error occurred inside of the client.
139
    #[error("an io error in the client: '{0}'")]
140
    Client(String),
141

            
142
    /// An attempt to use a `Collection` with a `Database` that it wasn't defined within.
143
    #[error("attempted to access a collection not registered with this schema")]
144
    CollectionNotFound,
145

            
146
    /// A `Collection` being added already exists. This can be caused by a collection name not being unique.
147
    #[error("attempted to define a collection that already has been defined")]
148
    CollectionAlreadyDefined,
149

            
150
    /// An attempt to update a document that doesn't exist.
151
    #[error("the requested document id {1} from collection {0} was not found")]
152
    DocumentNotFound(CollectionName, u64),
153

            
154
    /// When updating a document, if a situation is detected where the contents have changed on the server since the `Revision` provided, a Conflict error will be returned.
155
    #[error("a conflict was detected while updating document id {1} from collection {0}")]
156
    DocumentConflict(CollectionName, u64),
157

            
158
    /// When saving a document in a collection with unique views, a document emits a key that is already emitted by an existing ocument, this error is returned.
159
    #[error("a unique key violation occurred: document `{existing_document}` already has the same key as `{conflicting_document}` for {view}")]
160
    UniqueKeyViolation {
161
        /// The name of the view that the unique key violation occurred.
162
        view: ViewName,
163
        /// The document that caused the violation.
164
        conflicting_document: Header,
165
        /// The document that already uses the same key.
166
        existing_document: Header,
167
    },
168

            
169
    /// An invalid name was specified during schema creation.
170
    #[error("an invalid name was used in a schema: {0}")]
171
    InvalidName(#[from] schema::InvalidNameError),
172

            
173
    /// Permission was denied.
174
    #[error("permission error: {0}")]
175
    PermissionDenied(#[from] actionable::PermissionDenied),
176

            
177
    /// An internal error handling passwords was encountered.
178
    #[error("error with password: {0}")]
179
    Password(String),
180

            
181
    /// The user specified was not found. This will not be returned in response
182
    /// to an invalid username being used during login. It will be returned in
183
    /// other APIs that operate upon users.
184
    #[error("user not found")]
185
    UserNotFound,
186

            
187
    /// An error occurred converting from bytes to Utf-8.
188
    #[error("invalid string: {0}")]
189
    InvalidUnicode(String),
190

            
191
    /// The credentials specified are not valid.
192
    #[error("invalid credentials")]
193
    InvalidCredentials,
194

            
195
    /// Returned when the a view's reduce() function is unimplemented.
196
    #[error("reduce is unimplemented")]
197
    ReduceUnimplemented,
198

            
199
    /// A floating point operation yielded Not a Number.
200
    #[error("floating point operation yielded NaN")]
201
    NotANumber,
202
}
203

            
204
impl From<pot::Error> for Error {
205
    fn from(err: pot::Error) -> Self {
206
        Self::Serialization(err.to_string())
207
    }
208
}
209

            
210
impl<T> From<InsertError<T>> for Error {
211
    fn from(err: InsertError<T>) -> Self {
212
        err.error
213
    }
214
}
215

            
216
impl From<view::Error> for Error {
217
    fn from(err: view::Error) -> Self {
218
        Self::Database(err.to_string())
219
    }
220
}
221

            
222
impl From<FromUtf8Error> for Error {
223
    fn from(err: FromUtf8Error) -> Self {
224
        Self::InvalidUnicode(err.to_string())
225
    }
226
}
227

            
228
/// Shared schemas and utilities used for unit testing.
229
#[cfg(any(feature = "test-util", test))]
230
#[allow(missing_docs)]
231
pub mod test_util;
232

            
233
/// When true, encryption was enabled at build-time.
234
#[cfg(feature = "encryption")]
235
pub const ENCRYPTION_ENABLED: bool = true;
236

            
237
/// When true, encryption was enabled at build-time.
238
#[cfg(not(feature = "encryption"))]
239
pub const ENCRYPTION_ENABLED: bool = false;
240

            
241
/// A type that implements [`Error`](std::error::Error) and is threadsafe.
242
pub trait AnyError: std::error::Error + Send + Sync + 'static {}
243

            
244
impl<T> AnyError for T where T: std::error::Error + Send + Sync + 'static {}