mirror of
https://github.com/Cuprate/cuprate.git
synced 2025-01-27 21:06:00 +00:00
331d4d3e7f
* rename `database` -> `old_database` Keeping it around for reference until new implementation is complete. * create new `database/` skeleton * add `DATABASE.md` design doc skeleton * move design doc to `database/README.md` * add rough code * `lib.rs` -> `gist.rs` * database: use workspace deps * workspace: include `database/` as member CI will now include this crate. * cargo fmt * database: `AGPL` -> `MIT` * readme: add `TODO`s * add base files * cargo.toml: add `heed` feature * fix clippy * lib.rs: add extremely pedantic lints * readme: add `# Backends` * cargo.toml: add `cfg-if` * add `backend/` structure * base `database.rs` * cargo.toml: add `borsh` * backend: add `DATABASE_BACKEND` * base `error.rs` * base `database.rs` * base `transaction.rs` * base `table.rs` * lib.rs: add imports * add `pod.rs` * pod: use `Read/Write`, add tests for all primitive numbers * pod: impl Pod for `Vec<u8>`, `[u8; N]` * pod: add docs, add `private::Sealed` * pod: `to_writer`, `from_reader` The new `as_bytes` + `from_bytes` now allows (de)serializing from bytes directly instead of going through Read/Write. Different array return sizes are abstracted away with `-> impl AsRef<[u8]>` * pod: impl Pod for `Box<[u8]>` * pod: return `Err` on incorrect length in `from_bytes()` * pod: docs * pod: impl Pod for `Arc<[u8]>` * readme: docs * database: add `create_table()`, `get_table()` * table: `Pod` bound * backend: move into directories * pod: add `into_bytes()` * heed: impl `BytesEncode`, `BytesDecode` * add `actor`, `service` features The thread/actor system used will be gated behind `actor`, and the `tower/tokio` integration will be gated behind `service`. * add `lib.rs` docs * service: add `service.rs` * service: add `reader.rs` * service: add `writer.rs` * service: add `request.rs` & `response.rs` * cargo.toml: add `crossbeam` * service: add/use `enum Request`, `enum Response` * service: basic signatures for thread-pools, `Writer` -> `Writers` * service: split `tower::Service<ReadRequest/WriteRequest>` * service: impl `tower::Service` for `Database(Reader|Writer)` * service: add `init()`, impl basic `Reader/Writer` pools * service: add example `Request`'s * service: add example `ReadRequest` handling * temporarily allow clippy lints * readme: update file structure * transaction: add `RoTx::get_range()` * service: module docs * cargo.toml: add `cuprate-helper` * service: scale readers/writers based on thread count * database: change lifetimes * heed: impl Database for `heed` * heed: add `ConcreteRoTx`, `ConcreteRwTx`, impl Tx traits * docs * service: `read.rs` docs * service: `write.rs` docs * service: request/response docs * service: use `OnceLock` in `init()`, add `db_read()`, `db_write()` * service: leak database into `&'static` * database: add `#[inline]`, `#[cold]` * service: `free.rs` docs, more `#[inline]` + `#[cold]` * service: add `shutdown()` * service: add `Request::Shutdown` * service: `shutdown()` docs * heed: hide concrete tx types * lib.rs: add terms * split `Env` <-> `Database` * cargo.toml: add `paste` * database: add `tables/` * impl `serde/borsh` where possible * tables: add `Tables`, add test docs * make db backend mutually exclusive to fix `--all-features` * tables: use `$()*` in `tables!()` * cargo.toml: add `sanakirja 1.4.0` * sanakirja: impl `Env` * sanakirja: impl `Database` * sanakirja: impl `Transaction` * table: temporarily fix `sanakirja` K/V bounds * table: fix `#[cfg]` * cargo.toml: fix deps * lib.rs: docs * service: docs * readme: add files, update `# Documentation`, add `# Layers` * readme: `src/` file purpose * readme: `src/service/` file purpose * readme: `src/backend/` file purpose * fix `Cargo.lock` merge conflict * database: remove `gist.rs` * add to `constants.rs` * add top `//! comments` for files/modules * constants: add sanity-check test * service: add `only_one_database` test in `free.rs` * service: add `tests.rs` * remove unneeded markers + imports * backend: fix `get_range()`'s trait `impl` return * env: add `create_tables_if_needed()`, don't return `Option<Db>` * sort imports by `CONTRIBUTING.md` rules oops sorry boog * add `monero.rs` * monero: docs * database: add missing `RoTx/RwTx` inputs * backend: add missing `RoTx/RwTx` inputs * `monero.rs` trait -> free functions in `ops/` * pod: make methods infallible * ci: add `rustup update` step * service: use `Arc` instead of leaking, remove `db_read/db_write` * service: use `InfallibleOneshotReceiver` for readers * service: shutdown on error, add todos * service: remove `Request` * service: combine `ReadResponse` and `WriteResponse` * service: use `InfallibleOneshotReceiver` for writer * service: only spawn 1 writer, don't allow cloning writer handle * table: add associated `const CONSTANT_SIZE` * add `key.rs` + `trait Key`, add bound to `Table` * fix typos
128 lines
4.1 KiB
Rust
128 lines
4.1 KiB
Rust
//! Database tables.
|
|
//!
|
|
//! This module contains all the table definitions used by `cuprate-database`
|
|
//! and [`Tables`], an `enum` containing all [`Table`]s.
|
|
|
|
//---------------------------------------------------------------------------------------------------- Import
|
|
use crate::table::Table;
|
|
|
|
//---------------------------------------------------------------------------------------------------- Tables
|
|
/// An enumeration of _all_ database tables.
|
|
///
|
|
/// TODO: I don't think we need this.
|
|
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
|
|
#[cfg_attr(
|
|
feature = "borsh",
|
|
derive(borsh::BorshSerialize, borsh::BorshDeserialize)
|
|
)]
|
|
#[derive(Copy, Clone, Debug, PartialEq, PartialOrd, Eq, Ord, Hash)]
|
|
#[allow(missing_docs)]
|
|
pub enum Tables {
|
|
TestTable(TestTable),
|
|
TestTable2(TestTable2),
|
|
}
|
|
|
|
impl Tables {
|
|
/// Get the [`Table::NAME`].
|
|
pub const fn name(&self) -> &'static str {
|
|
/// Hack to access associated trait constant via a variable.
|
|
const fn get<T: Table>(t: &T) -> &'static str {
|
|
T::NAME
|
|
}
|
|
|
|
match self {
|
|
Self::TestTable(t) => get(t),
|
|
Self::TestTable2(t) => get(t),
|
|
}
|
|
}
|
|
|
|
/// Get the [`Table::CONSTANT_SIZE`].
|
|
pub const fn constant_size(&self) -> bool {
|
|
/// Hack to access associated trait constant via a variable.
|
|
const fn get<T: Table>(t: &T) -> bool {
|
|
T::CONSTANT_SIZE
|
|
}
|
|
|
|
match self {
|
|
Self::TestTable(t) => get(t),
|
|
Self::TestTable2(t) => get(t),
|
|
}
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------- Table macro
|
|
/// Create all tables, should be used _once_.
|
|
///
|
|
/// Generating this macro once and using `$()*` is probably
|
|
/// faster for compile times than calling the macro _per_ table.
|
|
///
|
|
/// All tables are zero-sized table structs, and implement the `Table` trait.
|
|
///
|
|
/// Table structs are automatically `CamelCase`,
|
|
/// and their static string names are automatically `snake_case`.
|
|
macro_rules! tables {
|
|
(
|
|
$(
|
|
$(#[$attr:meta])* // Documentation and any `derive`'s.
|
|
$table:ident, // The table name + doubles as the table struct name.
|
|
$size:literal, // Are the table's values all the same size?
|
|
$key:ty => // Key type.
|
|
$value:ty // Value type.
|
|
),* $(,)?
|
|
) => {
|
|
paste::paste! { $(
|
|
// Table struct.
|
|
$(#[$attr])*
|
|
// The below test show the `snake_case` table name in cargo docs.
|
|
/// ## Table Name
|
|
/// ```rust
|
|
/// # use cuprate_database::{*,tables::*};
|
|
#[doc = concat!(
|
|
"assert_eq!(",
|
|
stringify!([<$table:camel>]),
|
|
"::NAME, \"",
|
|
stringify!([<$table:snake>]),
|
|
"\");",
|
|
)]
|
|
/// ```
|
|
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
|
|
#[cfg_attr(feature = "borsh", derive(borsh::BorshSerialize, borsh::BorshDeserialize))]
|
|
#[derive(Copy,Clone,Debug,PartialEq,PartialOrd,Eq,Ord,Hash)]
|
|
pub struct [<$table:camel>];
|
|
|
|
// Table trait impl.
|
|
impl Table for [<$table:camel>] {
|
|
const NAME: &'static str = stringify!([<$table:snake>]);
|
|
const CONSTANT_SIZE: bool = $size;
|
|
type Key = $key;
|
|
type Value = $value;
|
|
}
|
|
|
|
// Table enum.
|
|
impl From<[<$table:camel>]> for Tables {
|
|
fn from(table: [<$table:camel>]) -> Self {
|
|
Self::[<$table:camel>](table)
|
|
}
|
|
}
|
|
)* }
|
|
};
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------- Tables
|
|
tables! {
|
|
/// Test documentation.
|
|
TestTable,
|
|
true,
|
|
i64 => u64,
|
|
|
|
/// Test documentation 2.
|
|
TestTable2,
|
|
true,
|
|
u8 => i8,
|
|
}
|
|
|
|
//---------------------------------------------------------------------------------------------------- Tests
|
|
#[cfg(test)]
|
|
mod test {
|
|
// use super::*;
|
|
}
|