cuprate_blockchain

Module ops

Source
Expand description

Abstracted Monero database operations.

This module contains many free functions that use the traits in this crate to generically call Monero-related database operations.

§impl Table

Functions in this module take Tables and TablesMut directly - these are already opened database tables.

As such, the responsibility of transactions, tables, etc, are on the caller.

Notably, this means that these functions are as lean as possible, so calling them in a loop should be okay.

§Atomicity

As transactions are handled by the caller of these functions, it is up to the caller to decide what happens if one them return an error.

To maintain atomicity, transactions should be aborted if one of the functions failed.

For example, if add_block() is called and returns an Err, aborting the transaction that opened the input TableMut would reverse all tables mutated by add_block() up until the error, leaving it in the state it was in before add_block() was called.

§Sub-functions

The main functions within this module are mostly within the block module.

Practically speaking, you should only be using 2 functions for mutation:

The block functions are “parent” functions, calling other sub-functions such as add_output().

add_output() itself only modifies output-related tables, while the block “parent” functions (like add_block and pop_block) modify all tables required.

add_block() makes sure all data related to the input is mutated, while this sub-function do not, it specifically mutates particular tables.

When calling this sub-functions, ensure that either:

  1. This effect (incomplete database mutation) is what is desired, or that…
  2. …the other tables will also be mutated to a correct state

§Example

Simple usage of ops.

use hex_literal::hex;

use cuprate_test_utils::data::BLOCK_V16_TX0;
use cuprate_blockchain::{
    cuprate_database::{
        ConcreteEnv,
        Env, EnvInner,
        DatabaseRo, DatabaseRw, TxRo, TxRw,
    },
    config::ConfigBuilder,
    tables::{Tables, TablesMut, OpenTables},
    ops::block::{add_block, pop_block},
};

// Create a configuration for the database environment.
let tmp_dir = tempfile::tempdir()?;
let db_dir = tmp_dir.path().to_owned();
let config = ConfigBuilder::new()
    .data_directory(db_dir.into())
    .build();

// Initialize the database environment.
let env = cuprate_blockchain::open(config)?;

// Open up a transaction + tables for writing.
let env_inner = env.env_inner();
let tx_rw = env_inner.tx_rw()?;
let mut tables = env_inner.open_tables_mut(&tx_rw)?;

// Write a block to the database.
let mut block = BLOCK_V16_TX0.clone();
add_block(&block, &mut tables)?;

// Commit the data written.
drop(tables);
TxRw::commit(tx_rw)?;

// Read the data, assert it is correct.
let tx_rw = env_inner.tx_rw()?;
let mut tables = env_inner.open_tables_mut(&tx_rw)?;
let (height, hash, serai_block) = pop_block(None, &mut tables)?;

assert_eq!(height, 0);
assert_eq!(serai_block, block.block);
assert_eq!(hash, hex!("43bd1f2b6556dcafa413d8372974af59e4e8f37dbf74dc6b2a9b7212d0577428"));

Modules§

  • Alternative Block/Chain Ops
  • Block functions.
  • Blockchain functions - chain height, generated coins, etc.
  • Key image functions.
  • Output functions.
  • Database properties functions - version, pruning, etc.
  • Transaction functions.