//! Database abstraction and utilities. //! //! This documentation is mostly for practical usage of `cuprate_database`. //! //! For a high-level overview, //! see [`database/README.md`](https://github.com/Cuprate/cuprate/blob/main/database/README.md). //! //! # Purpose //! This crate does 3 things: //! 1. Abstracts various databases with the [`Env`], [`Database`], [`Table`], [`Key`], [`RoTx`], and [`RwTx`] traits //! 2. Implements various `Monero` related [functions](ops) & [`tables`] //! 3. Exposes a [`tower::Service`] backed by a thread-pool //! //! # Terminology //! To be more clear on some terms used in this crate: //! //! | Term | Meaning | //! |------------|--------------------------------------| //! | `Env` | The 1 database environment, the "whole" thing //! | `Database` | A `key/value` store //! | `Table` | Solely the metadata of a `Database` (the `key` and `value` types, and the name) //! | `RoTx` | Read only transaction //! | `RwTx` | Read/write transaction //! //! The dataflow is `Env` -> `Tx` -> `Database` //! //! Which reads as: //! 1. You have a database `Environment` //! 2. You open up a `Transaction` //! 2. You get a particular `Database` from that `Environment` //! 3. You can now read/write data from/to that `Database` //! //! # `ConcreteEnv` //! This crate exposes [`ConcreteEnv`], which is a non-generic/non-dynamic, //! concrete object representing a database [`Env`]ironment. //! //! The actual backend for this type is determined via feature flags. //! //! This object existing means `E: Env` doesn't need to be spread all through the codebase, //! however, it also means some small invariants should be kept in mind. //! //! As `ConcreteEnv` is just a re-exposed type which has varying inner types, //! it means some properties will change depending on the backend used. //! //! For example: //! - [`std::mem::size_of::`] //! - [`std::mem::align_of::`] //! //! Things like these functions are affected by the backend and inner data, //! and should not be relied upon. This extends to any `struct/enum` that contains `ConcreteEnv`. //! //! `ConcreteEnv` invariants you can rely on: //! - It implements [`Env`] //! - Upon [`Drop::drop`], all database data will sync to disk //! //! Note that `ConcreteEnv` itself is not a clonable type, //! it should be wrapped in [`std::sync::Arc`]. //! //! TODO: we could also expose `ConcreteDatabase` if we're //! going to be storing any databases in structs, to lessen //! the generic `` pain. //! //! # Feature flags //! The `service` module requires the `service` feature to be enabled. //! See the module for more documentation. //! //! Different database backends are enabled by the feature flags: //! - `heed` //! - `sanakirja` //! //! The default is `heed`. //! //! # Invariants when not using `service` //! `cuprate_database` can be used without the `service` feature enabled but //! there are some things that must be kept in mind when doing so: //! //! TODO: make pretty. these will need to be updated //! as things change and as more backends are added. //! //! 1. Memory map resizing (must resize as needed) //! 1. Must not exceed `Config`'s maximum reader count //! 1. Avoid many nested transactions //! 1. `heed::MdbError::BadValSize` //! 1. `heed::Error::InvalidDatabaseTyping` //! 1. `heed::Error::BadOpenOptions` //! 1. Encoding/decoding into `[u8]` //! //! # Example //! Simple usage of this crate. //! //! ```rust //! use cuprate_database::{ //! config::Config, //! ConcreteEnv, //! Env, Key, RoTx, RwTx, //! service::{ReadRequest, WriteRequest, Response}, //! }; //! //! // Create a configuration for the database environment. //! let db_dir = tempfile::tempdir().unwrap(); //! let config = Config::new(Some(db_dir.path().to_path_buf())); //! //! // Initialize the database thread-pool. //! //! // TODO: //! // 1. let (read_handle, write_handle) = cuprate_database::service::init(config).unwrap(); //! // 2. Send write/read requests //! // 3. Use some other `Env` functions //! // 4. Shutdown //! ``` //---------------------------------------------------------------------------------------------------- Lints // Forbid lints. // Our code, and code generated (e.g macros) cannot overrule these. #![forbid( // `unsafe` is allowed but it _must_ be // commented with `SAFETY: reason`. clippy::undocumented_unsafe_blocks, // Never. unused_unsafe, redundant_semicolons, unused_allocation, coherence_leak_check, single_use_lifetimes, while_true, clippy::missing_docs_in_private_items, // Maybe can be put into `#[deny]`. unconditional_recursion, for_loops_over_fallibles, unused_braces, unused_doc_comments, unused_labels, keyword_idents, non_ascii_idents, variant_size_differences, unused_mut, // Annoying when debugging, maybe put in allow. // Probably can be put into `#[deny]`. future_incompatible, let_underscore, break_with_label_and_loop, duplicate_macro_attributes, exported_private_dependencies, large_assignments, overlapping_range_endpoints, semicolon_in_expressions_from_macros, noop_method_call, unreachable_pub, )] // Deny lints. // Some of these are `#[allow]`'ed on a per-case basis. #![deny( clippy::all, clippy::correctness, clippy::suspicious, clippy::style, clippy::complexity, clippy::perf, clippy::pedantic, clippy::nursery, clippy::cargo, missing_docs, deprecated, unused_comparisons, nonstandard_style )] #![allow(unreachable_code, unused_variables, dead_code)] // TODO: remove #![allow( // FIXME: this lint affects crates outside of // `database/` for some reason, allow for now. clippy::cargo_common_metadata, // FIXME: adding `#[must_use]` onto everything // might just be more annoying than useful... // although it is sometimes nice. clippy::must_use_candidate, // TODO: should be removed after all `todo!()`'s are gone. clippy::diverging_sub_expression, // FIXME: // If #[deny(clippy::restriction)] is used, it // enables a whole bunch of very subjective lints. // The below disables most of the ones that are // a bit too unwieldy. // // Figure out if if `clippy::restriction` should be // used (it enables a bunch of good lints but has // many false positives). // clippy::single_char_lifetime_names, // clippy::implicit_return, // clippy::std_instead_of_alloc, // clippy::std_instead_of_core, // clippy::unwrap_used, // clippy::min_ident_chars, // clippy::absolute_paths, // clippy::missing_inline_in_public_items, // clippy::shadow_reuse, // clippy::shadow_unrelated, // clippy::missing_trait_methods, // clippy::pub_use, // clippy::pub_with_shorthand, // clippy::blanket_clippy_restriction_lints, // clippy::exhaustive_structs, // clippy::exhaustive_enums, // clippy::unsafe_derive_deserialize, // clippy::multiple_inherent_impl, // clippy::unreadable_literal, // clippy::indexing_slicing, // clippy::float_arithmetic, // clippy::cast_possible_truncation, // clippy::as_conversions, // clippy::cast_precision_loss, // clippy::cast_sign_loss, // clippy::missing_asserts_for_indexing, // clippy::default_numeric_fallback, // clippy::module_inception, // clippy::mod_module_files, // clippy::multiple_unsafe_ops_per_block, // clippy::too_many_lines, // clippy::missing_assert_message, // clippy::len_zero, // clippy::separated_literal_suffix, // clippy::single_call_fn, // clippy::unreachable, // clippy::many_single_char_names, // clippy::redundant_pub_crate, // clippy::decimal_literal_representation, // clippy::option_if_let_else, // clippy::lossy_float_literal, // clippy::modulo_arithmetic, // clippy::print_stdout, // clippy::module_name_repetitions, // clippy::no_effect, // clippy::semicolon_outside_block, // clippy::panic, // clippy::question_mark_used, // clippy::expect_used, // clippy::integer_division, // clippy::type_complexity, // clippy::pattern_type_mismatch, // clippy::arithmetic_side_effects, // clippy::default_trait_access, // clippy::similar_names, // clippy::needless_pass_by_value, // clippy::inline_always, // clippy::if_then_some_else_none, // clippy::arithmetic_side_effects, // clippy::float_cmp, // clippy::items_after_statements, // clippy::use_debug, // clippy::mem_forget, // clippy::else_if_without_else, // clippy::str_to_string, // clippy::branches_sharing_code, // clippy::impl_trait_in_params, // clippy::struct_excessive_bools, // clippy::exit, // // This lint is actually good but // // it sometimes hits false positive. // clippy::self_named_module_files clippy::module_name_repetitions, clippy::module_inception, clippy::redundant_pub_crate, clippy::option_if_let_else, )] // Allow some lints when running in debug mode. #![cfg_attr(debug_assertions, allow(clippy::todo, clippy::multiple_crate_versions))] // Only allow building 64-bit targets. // // This allows us to assume 64-bit // invariants in code, e.g. `usize as u64`. #[cfg(not(target_pointer_width = "64"))] compile_error!("Cuprate is only compatible with 64-bit CPUs"); //---------------------------------------------------------------------------------------------------- Public API // Import private modules, export public types. // // Documentation for each module is located in the respective file. mod backend; pub use backend::ConcreteEnv; pub mod config; mod constants; pub use constants::{DATABASE_BACKEND, DATABASE_CORRUPT_MSG, DATABASE_FILENAME}; mod database; pub use database::Database; mod env; pub use env::Env; mod error; pub use error::{InitError, RuntimeError}; mod free; pub mod resize; mod key; pub use key::{DupKey, Key}; mod macros; pub mod ops; mod pod; pub use pod::Pod; mod table; pub use table::Table; pub mod tables; mod transaction; pub use transaction::{RoTx, RwTx}; //---------------------------------------------------------------------------------------------------- Feature-gated #[cfg(feature = "service")] pub mod service; //---------------------------------------------------------------------------------------------------- Private