mirror of
https://github.com/Cuprate/cuprate.git
synced 2025-01-18 08:44:36 +00:00
Storage: fix lifetimes (#215)
Some checks failed
Audit / audit (push) Has been cancelled
CI / fmt (push) Has been cancelled
CI / typo (push) Has been cancelled
CI / ci (macos-latest, stable, bash) (push) Has been cancelled
CI / ci (ubuntu-latest, stable, bash) (push) Has been cancelled
CI / ci (windows-latest, stable-x86_64-pc-windows-gnu, msys2 {0}) (push) Has been cancelled
Deny / audit (push) Has been cancelled
Doc / build (push) Has been cancelled
Doc / deploy (push) Has been cancelled
Some checks failed
Audit / audit (push) Has been cancelled
CI / fmt (push) Has been cancelled
CI / typo (push) Has been cancelled
CI / ci (macos-latest, stable, bash) (push) Has been cancelled
CI / ci (ubuntu-latest, stable, bash) (push) Has been cancelled
CI / ci (windows-latest, stable-x86_64-pc-windows-gnu, msys2 {0}) (push) Has been cancelled
Deny / audit (push) Has been cancelled
Doc / build (push) Has been cancelled
Doc / deploy (push) Has been cancelled
* fix db lifetimes * fix redb * fix blockchain with redb * add docs
This commit is contained in:
parent
d5c8eba1d8
commit
a82c08cc80
6 changed files with 59 additions and 61 deletions
|
@ -50,20 +50,12 @@ pub fn open(config: Config) -> Result<ConcreteEnv, InitError> {
|
||||||
// we want since it is agnostic, so we are responsible for this.
|
// we want since it is agnostic, so we are responsible for this.
|
||||||
{
|
{
|
||||||
let env_inner = env.env_inner();
|
let env_inner = env.env_inner();
|
||||||
let tx_rw = env_inner.tx_rw();
|
let tx_rw = env_inner.tx_rw().map_err(runtime_to_init_error)?;
|
||||||
let tx_rw = match tx_rw {
|
|
||||||
Ok(tx_rw) => tx_rw,
|
|
||||||
Err(e) => return Err(runtime_to_init_error(e)),
|
|
||||||
};
|
|
||||||
|
|
||||||
// Create all tables.
|
// Create all tables.
|
||||||
if let Err(e) = OpenTables::create_tables(&env_inner, &tx_rw) {
|
OpenTables::create_tables(&env_inner, &tx_rw).map_err(runtime_to_init_error)?;
|
||||||
return Err(runtime_to_init_error(e));
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Err(e) = tx_rw.commit() {
|
TxRw::commit(tx_rw).map_err(runtime_to_init_error)?;
|
||||||
return Err(runtime_to_init_error(e));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(env)
|
Ok(env)
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
//! TODO
|
//! TODO
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------- Import
|
//---------------------------------------------------------------------------------------------------- Import
|
||||||
use cuprate_database::{EnvInner, RuntimeError, TxRo, TxRw};
|
use cuprate_database::{EnvInner, RuntimeError};
|
||||||
|
|
||||||
use crate::tables::{TablesIter, TablesMut};
|
use crate::tables::{TablesIter, TablesMut};
|
||||||
|
|
||||||
|
@ -84,12 +84,12 @@ pub(crate) use call_fn_on_all_tables_or_early_return;
|
||||||
/// let mut tables = env_inner.open_tables_mut(&tx_rw)?;
|
/// let mut tables = env_inner.open_tables_mut(&tx_rw)?;
|
||||||
/// # Ok(()) }
|
/// # Ok(()) }
|
||||||
/// ```
|
/// ```
|
||||||
pub trait OpenTables<'env, Ro, Rw>
|
pub trait OpenTables<'env> {
|
||||||
where
|
/// The read-only transaction type of the backend.
|
||||||
Self: 'env,
|
type Ro<'a>;
|
||||||
Ro: TxRo<'env>,
|
/// The read-write transaction type of the backend.
|
||||||
Rw: TxRw<'env>,
|
type Rw<'a>;
|
||||||
{
|
|
||||||
/// Open all tables in read/iter mode.
|
/// Open all tables in read/iter mode.
|
||||||
///
|
///
|
||||||
/// This calls [`EnvInner::open_db_ro`] on all database tables
|
/// This calls [`EnvInner::open_db_ro`] on all database tables
|
||||||
|
@ -100,7 +100,7 @@ where
|
||||||
///
|
///
|
||||||
/// As all tables are created upon [`crate::open`],
|
/// As all tables are created upon [`crate::open`],
|
||||||
/// this function will never error because a table doesn't exist.
|
/// this function will never error because a table doesn't exist.
|
||||||
fn open_tables(&'env self, tx_ro: &Ro) -> Result<impl TablesIter, RuntimeError>;
|
fn open_tables(&self, tx_ro: &Self::Ro<'_>) -> Result<impl TablesIter, RuntimeError>;
|
||||||
|
|
||||||
/// Open all tables in read-write mode.
|
/// Open all tables in read-write mode.
|
||||||
///
|
///
|
||||||
|
@ -109,7 +109,7 @@ where
|
||||||
///
|
///
|
||||||
/// # Errors
|
/// # Errors
|
||||||
/// This will only return [`RuntimeError::Io`] on errors.
|
/// This will only return [`RuntimeError::Io`] on errors.
|
||||||
fn open_tables_mut(&'env self, tx_rw: &Rw) -> Result<impl TablesMut, RuntimeError>;
|
fn open_tables_mut(&self, tx_rw: &Self::Rw<'_>) -> Result<impl TablesMut, RuntimeError>;
|
||||||
|
|
||||||
/// Create all database tables.
|
/// Create all database tables.
|
||||||
///
|
///
|
||||||
|
@ -118,28 +118,29 @@ where
|
||||||
///
|
///
|
||||||
/// # Errors
|
/// # Errors
|
||||||
/// This will only return [`RuntimeError::Io`] on errors.
|
/// This will only return [`RuntimeError::Io`] on errors.
|
||||||
fn create_tables(&'env self, tx_rw: &Rw) -> Result<(), RuntimeError>;
|
fn create_tables(&self, tx_rw: &Self::Rw<'_>) -> Result<(), RuntimeError>;
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'env, Ei, Ro, Rw> OpenTables<'env, Ro, Rw> for Ei
|
impl<'env, Ei> OpenTables<'env> for Ei
|
||||||
where
|
where
|
||||||
Ei: EnvInner<'env, Ro, Rw>,
|
Ei: EnvInner<'env>,
|
||||||
Ro: TxRo<'env>,
|
|
||||||
Rw: TxRw<'env>,
|
|
||||||
{
|
{
|
||||||
fn open_tables(&'env self, tx_ro: &Ro) -> Result<impl TablesIter, RuntimeError> {
|
type Ro<'a> = <Ei as EnvInner<'env>>::Ro<'a>;
|
||||||
|
type Rw<'a> = <Ei as EnvInner<'env>>::Rw<'a>;
|
||||||
|
|
||||||
|
fn open_tables(&self, tx_ro: &Self::Ro<'_>) -> Result<impl TablesIter, RuntimeError> {
|
||||||
call_fn_on_all_tables_or_early_return! {
|
call_fn_on_all_tables_or_early_return! {
|
||||||
Self::open_db_ro(self, tx_ro)
|
Self::open_db_ro(self, tx_ro)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn open_tables_mut(&'env self, tx_rw: &Rw) -> Result<impl TablesMut, RuntimeError> {
|
fn open_tables_mut(&self, tx_rw: &Self::Rw<'_>) -> Result<impl TablesMut, RuntimeError> {
|
||||||
call_fn_on_all_tables_or_early_return! {
|
call_fn_on_all_tables_or_early_return! {
|
||||||
Self::open_db_rw(self, tx_rw)
|
Self::open_db_rw(self, tx_rw)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_tables(&'env self, tx_rw: &Rw) -> Result<(), RuntimeError> {
|
fn create_tables(&self, tx_rw: &Self::Rw<'_>) -> Result<(), RuntimeError> {
|
||||||
match call_fn_on_all_tables_or_early_return! {
|
match call_fn_on_all_tables_or_early_return! {
|
||||||
Self::create_db(self, tx_rw)
|
Self::create_db(self, tx_rw)
|
||||||
} {
|
} {
|
||||||
|
|
|
@ -9,7 +9,7 @@ repository = "https://github.com/Cuprate/cuprate/tree/main/storage/database"
|
||||||
keywords = ["cuprate", "database"]
|
keywords = ["cuprate", "database"]
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
default = ["heed"]
|
# default = ["heed"]
|
||||||
# default = ["redb"]
|
# default = ["redb"]
|
||||||
# default = ["redb-memory"]
|
# default = ["redb-memory"]
|
||||||
heed = ["dep:heed"]
|
heed = ["dep:heed"]
|
||||||
|
|
|
@ -244,25 +244,28 @@ impl Env for ConcreteEnv {
|
||||||
}
|
}
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------- EnvInner Impl
|
//---------------------------------------------------------------------------------------------------- EnvInner Impl
|
||||||
impl<'env> EnvInner<'env, heed::RoTxn<'env>, RefCell<heed::RwTxn<'env>>>
|
impl<'env> EnvInner<'env> for RwLockReadGuard<'env, heed::Env>
|
||||||
for RwLockReadGuard<'env, heed::Env>
|
|
||||||
where
|
where
|
||||||
Self: 'env,
|
Self: 'env,
|
||||||
{
|
{
|
||||||
|
type Ro<'a> = heed::RoTxn<'a>;
|
||||||
|
|
||||||
|
type Rw<'a> = RefCell<heed::RwTxn<'a>>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn tx_ro(&'env self) -> Result<heed::RoTxn<'env>, RuntimeError> {
|
fn tx_ro(&self) -> Result<Self::Ro<'_>, RuntimeError> {
|
||||||
Ok(self.read_txn()?)
|
Ok(self.read_txn()?)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn tx_rw(&'env self) -> Result<RefCell<heed::RwTxn<'env>>, RuntimeError> {
|
fn tx_rw(&self) -> Result<Self::Rw<'_>, RuntimeError> {
|
||||||
Ok(RefCell::new(self.write_txn()?))
|
Ok(RefCell::new(self.write_txn()?))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn open_db_ro<T: Table>(
|
fn open_db_ro<T: Table>(
|
||||||
&self,
|
&self,
|
||||||
tx_ro: &heed::RoTxn<'env>,
|
tx_ro: &Self::Ro<'_>,
|
||||||
) -> Result<impl DatabaseRo<T> + DatabaseIter<T>, RuntimeError> {
|
) -> Result<impl DatabaseRo<T> + DatabaseIter<T>, RuntimeError> {
|
||||||
// Open up a read-only database using our table's const metadata.
|
// Open up a read-only database using our table's const metadata.
|
||||||
//
|
//
|
||||||
|
@ -280,7 +283,7 @@ where
|
||||||
#[inline]
|
#[inline]
|
||||||
fn open_db_rw<T: Table>(
|
fn open_db_rw<T: Table>(
|
||||||
&self,
|
&self,
|
||||||
tx_rw: &RefCell<heed::RwTxn<'env>>,
|
tx_rw: &Self::Rw<'_>,
|
||||||
) -> Result<impl DatabaseRw<T>, RuntimeError> {
|
) -> Result<impl DatabaseRw<T>, RuntimeError> {
|
||||||
// Open up a read/write database using our table's const metadata.
|
// Open up a read/write database using our table's const metadata.
|
||||||
//
|
//
|
||||||
|
@ -293,7 +296,7 @@ where
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_db<T: Table>(&self, tx_rw: &RefCell<heed::RwTxn<'env>>) -> Result<(), RuntimeError> {
|
fn create_db<T: Table>(&self, tx_rw: &Self::Rw<'_>) -> Result<(), RuntimeError> {
|
||||||
// Create a database using our:
|
// Create a database using our:
|
||||||
// - [`Table`]'s const metadata.
|
// - [`Table`]'s const metadata.
|
||||||
// - (potentially) our [`Key`] comparison function
|
// - (potentially) our [`Key`] comparison function
|
||||||
|
@ -325,10 +328,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn clear_db<T: Table>(
|
fn clear_db<T: Table>(&self, tx_rw: &mut Self::Rw<'_>) -> Result<(), RuntimeError> {
|
||||||
&self,
|
|
||||||
tx_rw: &mut RefCell<heed::RwTxn<'env>>,
|
|
||||||
) -> Result<(), RuntimeError> {
|
|
||||||
let tx_rw = tx_rw.get_mut();
|
let tx_rw = tx_rw.get_mut();
|
||||||
|
|
||||||
// Open the table. We don't care about flags or key
|
// Open the table. We don't care about flags or key
|
||||||
|
|
|
@ -118,18 +118,20 @@ impl Env for ConcreteEnv {
|
||||||
}
|
}
|
||||||
|
|
||||||
//---------------------------------------------------------------------------------------------------- EnvInner Impl
|
//---------------------------------------------------------------------------------------------------- EnvInner Impl
|
||||||
impl<'env> EnvInner<'env, redb::ReadTransaction, redb::WriteTransaction>
|
impl<'env> EnvInner<'env> for (&'env redb::Database, redb::Durability)
|
||||||
for (&'env redb::Database, redb::Durability)
|
|
||||||
where
|
where
|
||||||
Self: 'env,
|
Self: 'env,
|
||||||
{
|
{
|
||||||
|
type Ro<'a> = redb::ReadTransaction;
|
||||||
|
type Rw<'a> = redb::WriteTransaction;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn tx_ro(&'env self) -> Result<redb::ReadTransaction, RuntimeError> {
|
fn tx_ro(&self) -> Result<redb::ReadTransaction, RuntimeError> {
|
||||||
Ok(self.0.begin_read()?)
|
Ok(self.0.begin_read()?)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn tx_rw(&'env self) -> Result<redb::WriteTransaction, RuntimeError> {
|
fn tx_rw(&self) -> Result<redb::WriteTransaction, RuntimeError> {
|
||||||
// `redb` has sync modes on the TX level, unlike heed,
|
// `redb` has sync modes on the TX level, unlike heed,
|
||||||
// which sets it at the Environment level.
|
// which sets it at the Environment level.
|
||||||
//
|
//
|
||||||
|
@ -142,7 +144,7 @@ where
|
||||||
#[inline]
|
#[inline]
|
||||||
fn open_db_ro<T: Table>(
|
fn open_db_ro<T: Table>(
|
||||||
&self,
|
&self,
|
||||||
tx_ro: &redb::ReadTransaction,
|
tx_ro: &Self::Ro<'_>,
|
||||||
) -> Result<impl DatabaseRo<T> + DatabaseIter<T>, RuntimeError> {
|
) -> Result<impl DatabaseRo<T> + DatabaseIter<T>, RuntimeError> {
|
||||||
// Open up a read-only database using our `T: Table`'s const metadata.
|
// Open up a read-only database using our `T: Table`'s const metadata.
|
||||||
let table: redb::TableDefinition<'static, StorableRedb<T::Key>, StorableRedb<T::Value>> =
|
let table: redb::TableDefinition<'static, StorableRedb<T::Key>, StorableRedb<T::Value>> =
|
||||||
|
@ -154,7 +156,7 @@ where
|
||||||
#[inline]
|
#[inline]
|
||||||
fn open_db_rw<T: Table>(
|
fn open_db_rw<T: Table>(
|
||||||
&self,
|
&self,
|
||||||
tx_rw: &redb::WriteTransaction,
|
tx_rw: &Self::Rw<'_>,
|
||||||
) -> Result<impl DatabaseRw<T>, RuntimeError> {
|
) -> Result<impl DatabaseRw<T>, RuntimeError> {
|
||||||
// Open up a read/write database using our `T: Table`'s const metadata.
|
// Open up a read/write database using our `T: Table`'s const metadata.
|
||||||
let table: redb::TableDefinition<'static, StorableRedb<T::Key>, StorableRedb<T::Value>> =
|
let table: redb::TableDefinition<'static, StorableRedb<T::Key>, StorableRedb<T::Value>> =
|
||||||
|
|
|
@ -62,17 +62,17 @@ pub trait Env: Sized {
|
||||||
// For `heed`, this is just `heed::Env`, for `redb` this is
|
// For `heed`, this is just `heed::Env`, for `redb` this is
|
||||||
// `(redb::Database, redb::Durability)` as each transaction
|
// `(redb::Database, redb::Durability)` as each transaction
|
||||||
// needs the sync mode set during creation.
|
// needs the sync mode set during creation.
|
||||||
type EnvInner<'env>: EnvInner<'env, Self::TxRo<'env>, Self::TxRw<'env>>
|
type EnvInner<'env>: EnvInner<'env>
|
||||||
where
|
where
|
||||||
Self: 'env;
|
Self: 'env;
|
||||||
|
|
||||||
/// The read-only transaction type of the backend.
|
/// The read-only transaction type of the backend.
|
||||||
type TxRo<'env>: TxRo<'env> + 'env
|
type TxRo<'env>: TxRo<'env>
|
||||||
where
|
where
|
||||||
Self: 'env;
|
Self: 'env;
|
||||||
|
|
||||||
/// The read/write transaction type of the backend.
|
/// The read/write transaction type of the backend.
|
||||||
type TxRw<'env>: TxRw<'env> + 'env
|
type TxRw<'env>: TxRw<'env>
|
||||||
where
|
where
|
||||||
Self: 'env;
|
Self: 'env;
|
||||||
|
|
||||||
|
@ -209,23 +209,23 @@ Subsequent table opens will follow the flags/ordering, but only if
|
||||||
///
|
///
|
||||||
/// # Invariant
|
/// # Invariant
|
||||||
#[doc = doc_heed_create_db_invariant!()]
|
#[doc = doc_heed_create_db_invariant!()]
|
||||||
pub trait EnvInner<'env, Ro, Rw>
|
pub trait EnvInner<'env> {
|
||||||
where
|
/// The read-only transaction type of the backend.
|
||||||
Self: 'env,
|
type Ro<'a>: TxRo<'a>;
|
||||||
Ro: TxRo<'env>,
|
/// The read-write transaction type of the backend.
|
||||||
Rw: TxRw<'env>,
|
type Rw<'a>: TxRw<'a>;
|
||||||
{
|
|
||||||
/// Create a read-only transaction.
|
/// Create a read-only transaction.
|
||||||
///
|
///
|
||||||
/// # Errors
|
/// # Errors
|
||||||
/// This will only return [`RuntimeError::Io`] if it errors.
|
/// This will only return [`RuntimeError::Io`] if it errors.
|
||||||
fn tx_ro(&'env self) -> Result<Ro, RuntimeError>;
|
fn tx_ro(&self) -> Result<Self::Ro<'_>, RuntimeError>;
|
||||||
|
|
||||||
/// Create a read/write transaction.
|
/// Create a read/write transaction.
|
||||||
///
|
///
|
||||||
/// # Errors
|
/// # Errors
|
||||||
/// This will only return [`RuntimeError::Io`] if it errors.
|
/// This will only return [`RuntimeError::Io`] if it errors.
|
||||||
fn tx_rw(&'env self) -> Result<Rw, RuntimeError>;
|
fn tx_rw(&self) -> Result<Self::Rw<'_>, RuntimeError>;
|
||||||
|
|
||||||
/// Open a database in read-only mode.
|
/// Open a database in read-only mode.
|
||||||
///
|
///
|
||||||
|
@ -252,7 +252,7 @@ where
|
||||||
#[doc = doc_heed_create_db_invariant!()]
|
#[doc = doc_heed_create_db_invariant!()]
|
||||||
fn open_db_ro<T: Table>(
|
fn open_db_ro<T: Table>(
|
||||||
&self,
|
&self,
|
||||||
tx_ro: &Ro,
|
tx_ro: &Self::Ro<'_>,
|
||||||
) -> Result<impl DatabaseRo<T> + DatabaseIter<T>, RuntimeError>;
|
) -> Result<impl DatabaseRo<T> + DatabaseIter<T>, RuntimeError>;
|
||||||
|
|
||||||
/// Open a database in read/write mode.
|
/// Open a database in read/write mode.
|
||||||
|
@ -271,7 +271,10 @@ where
|
||||||
///
|
///
|
||||||
/// # Invariant
|
/// # Invariant
|
||||||
#[doc = doc_heed_create_db_invariant!()]
|
#[doc = doc_heed_create_db_invariant!()]
|
||||||
fn open_db_rw<T: Table>(&self, tx_rw: &Rw) -> Result<impl DatabaseRw<T>, RuntimeError>;
|
fn open_db_rw<T: Table>(
|
||||||
|
&self,
|
||||||
|
tx_rw: &Self::Rw<'_>,
|
||||||
|
) -> Result<impl DatabaseRw<T>, RuntimeError>;
|
||||||
|
|
||||||
/// Create a database table.
|
/// Create a database table.
|
||||||
///
|
///
|
||||||
|
@ -282,7 +285,7 @@ where
|
||||||
///
|
///
|
||||||
/// # Invariant
|
/// # Invariant
|
||||||
#[doc = doc_heed_create_db_invariant!()]
|
#[doc = doc_heed_create_db_invariant!()]
|
||||||
fn create_db<T: Table>(&self, tx_rw: &Rw) -> Result<(), RuntimeError>;
|
fn create_db<T: Table>(&self, tx_rw: &Self::Rw<'_>) -> Result<(), RuntimeError>;
|
||||||
|
|
||||||
/// Clear all `(key, value)`'s from a database table.
|
/// Clear all `(key, value)`'s from a database table.
|
||||||
///
|
///
|
||||||
|
@ -297,5 +300,5 @@ where
|
||||||
///
|
///
|
||||||
/// If the specified table is not created upon before this function is called,
|
/// If the specified table is not created upon before this function is called,
|
||||||
/// this will return [`RuntimeError::TableNotFound`].
|
/// this will return [`RuntimeError::TableNotFound`].
|
||||||
fn clear_db<T: Table>(&self, tx_rw: &mut Rw) -> Result<(), RuntimeError>;
|
fn clear_db<T: Table>(&self, tx_rw: &mut Self::Rw<'_>) -> Result<(), RuntimeError>;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue