mirror of
https://github.com/hinto-janai/cuprate.git
synced 2025-01-24 11:26:03 +00:00
Compare commits
11 commits
374bc77cfb
...
01f34e5107
Author | SHA1 | Date | |
---|---|---|---|
|
01f34e5107 | ||
|
848a6a71c4 | ||
|
f4c88b6f05 | ||
|
6523afbe13 | ||
|
c840053854 | ||
|
fe365fdda2 | ||
|
99a7cbc93d | ||
|
1ad8b34941 | ||
|
57af45e01d | ||
|
5588671501 | ||
|
19150df355 |
116 changed files with 1013 additions and 900 deletions
8
Cargo.lock
generated
8
Cargo.lock
generated
|
@ -554,6 +554,7 @@ dependencies = [
|
|||
name = "cuprate-consensus"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"cfg-if",
|
||||
"cuprate-consensus-rules",
|
||||
"cuprate-helper",
|
||||
"cuprate-test-utils",
|
||||
|
@ -581,6 +582,7 @@ dependencies = [
|
|||
name = "cuprate-consensus-rules"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"cfg-if",
|
||||
"crypto-bigint",
|
||||
"cuprate-constants",
|
||||
"cuprate-cryptonight",
|
||||
|
@ -685,15 +687,14 @@ dependencies = [
|
|||
"cuprate-blockchain",
|
||||
"cuprate-consensus",
|
||||
"cuprate-consensus-rules",
|
||||
"cuprate-helper",
|
||||
"cuprate-types",
|
||||
"hex",
|
||||
"hex-literal",
|
||||
"monero-serai",
|
||||
"rayon",
|
||||
"sha3",
|
||||
"thiserror",
|
||||
"tokio",
|
||||
"tokio-test",
|
||||
"tower",
|
||||
]
|
||||
|
||||
|
@ -740,6 +741,7 @@ version = "0.1.0"
|
|||
dependencies = [
|
||||
"bitflags 2.6.0",
|
||||
"bytes",
|
||||
"cfg-if",
|
||||
"cuprate-helper",
|
||||
"futures",
|
||||
"proptest",
|
||||
|
@ -791,6 +793,7 @@ version = "0.1.0"
|
|||
dependencies = [
|
||||
"async-trait",
|
||||
"borsh",
|
||||
"cfg-if",
|
||||
"cuprate-helper",
|
||||
"cuprate-pruning",
|
||||
"cuprate-test-utils",
|
||||
|
@ -805,7 +808,6 @@ dependencies = [
|
|||
"tokio-util",
|
||||
"tower",
|
||||
"tracing",
|
||||
"tracing-subscriber",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
|
|
|
@ -211,7 +211,6 @@ unseparated_literal_suffix = "deny"
|
|||
unnecessary_safety_doc = "deny"
|
||||
unnecessary_safety_comment = "deny"
|
||||
unnecessary_self_imports = "deny"
|
||||
tests_outside_test_module = "deny"
|
||||
string_to_string = "deny"
|
||||
rest_pat_in_fully_bound_structs = "deny"
|
||||
redundant_type_annotations = "deny"
|
||||
|
|
|
@ -55,6 +55,7 @@ cargo doc --open --package cuprate-blockchain
|
|||
## 1-off crates
|
||||
| Crate | In-tree path | Purpose |
|
||||
|-------|--------------|---------|
|
||||
| [`cuprate-constants`](https://doc.cuprate.org/cuprate_constants) | [`constants/`](https://github.com/Cuprate/cuprate/tree/main/constants) | Shared `const/static` data across Cuprate
|
||||
| [`cuprate-cryptonight`](https://doc.cuprate.org/cuprate_cryptonight) | [`cryptonight/`](https://github.com/Cuprate/cuprate/tree/main/cryptonight) | CryptoNight hash functions
|
||||
| [`cuprate-pruning`](https://doc.cuprate.org/cuprate_pruning) | [`pruning/`](https://github.com/Cuprate/cuprate/tree/main/pruning) | Monero pruning logic/types
|
||||
| [`cuprate-helper`](https://doc.cuprate.org/cuprate_helper) | [`helper/`](https://github.com/Cuprate/cuprate/tree/main/helper) | Kitchen-sink helper crate for Cuprate
|
||||
|
|
|
@ -12,6 +12,7 @@ cuprate-helper = { path = "../helper", default-features = false, features = ["st
|
|||
cuprate-consensus-rules = { path = "./rules", features = ["rayon"] }
|
||||
cuprate-types = { path = "../types" }
|
||||
|
||||
cfg-if = { workspace = true }
|
||||
thiserror = { workspace = true }
|
||||
tower = { workspace = true, features = ["util"] }
|
||||
tracing = { workspace = true, features = ["std", "attributes"] }
|
||||
|
@ -19,7 +20,6 @@ futures = { workspace = true, features = ["std", "async-await"] }
|
|||
|
||||
randomx-rs = { workspace = true }
|
||||
monero-serai = { workspace = true, features = ["std"] }
|
||||
curve25519-dalek = { workspace = true }
|
||||
|
||||
rayon = { workspace = true }
|
||||
thread_local = { workspace = true }
|
||||
|
@ -34,8 +34,12 @@ cuprate-test-utils = { path = "../test-utils" }
|
|||
cuprate-consensus-rules = {path = "./rules", features = ["proptest"]}
|
||||
|
||||
hex-literal = { workspace = true }
|
||||
curve25519-dalek = { workspace = true }
|
||||
|
||||
tokio = { workspace = true, features = ["rt-multi-thread", "macros"]}
|
||||
tokio-test = { workspace = true }
|
||||
proptest = { workspace = true }
|
||||
proptest-derive = { workspace = true }
|
||||
|
||||
[lints]
|
||||
workspace = true
|
|
@ -9,19 +9,22 @@ name = "cuprate-fast-sync-create-hashes"
|
|||
path = "src/create.rs"
|
||||
|
||||
[dependencies]
|
||||
clap = { workspace = true, features = ["derive", "std"] }
|
||||
cuprate-blockchain = { path = "../../storage/blockchain" }
|
||||
cuprate-consensus = { path = ".." }
|
||||
cuprate-consensus-rules = { path = "../rules" }
|
||||
cuprate-types = { path = "../../types" }
|
||||
hex.workspace = true
|
||||
hex-literal.workspace = true
|
||||
monero-serai.workspace = true
|
||||
rayon.workspace = true
|
||||
sha3 = "0.10.8"
|
||||
thiserror.workspace = true
|
||||
cuprate-helper = { path = "../../helper", features = ["cast"] }
|
||||
|
||||
clap = { workspace = true, features = ["derive", "std"] }
|
||||
hex = { workspace = true }
|
||||
hex-literal = { workspace = true }
|
||||
monero-serai = { workspace = true }
|
||||
sha3 = { version = "0.10.8" }
|
||||
thiserror = { workspace = true }
|
||||
tokio = { workspace = true, features = ["full"] }
|
||||
tower.workspace = true
|
||||
tower = { workspace = true }
|
||||
|
||||
[dev-dependencies]
|
||||
tokio-test = "0.4.4"
|
||||
|
||||
[lints]
|
||||
workspace = true
|
|
@ -1,3 +1,8 @@
|
|||
#![expect(
|
||||
unused_crate_dependencies,
|
||||
reason = "binary shares same Cargo.toml as library"
|
||||
)]
|
||||
|
||||
use std::{fmt::Write, fs::write};
|
||||
|
||||
use clap::Parser;
|
||||
|
@ -70,16 +75,13 @@ async fn main() {
|
|||
let mut height = 0_usize;
|
||||
|
||||
while height < height_target {
|
||||
match read_batch(&mut read_handle, height).await {
|
||||
Ok(block_ids) => {
|
||||
if let Ok(block_ids) = read_batch(&mut read_handle, height).await {
|
||||
let hash = hash_of_hashes(block_ids.as_slice());
|
||||
hashes_of_hashes.push(hash);
|
||||
}
|
||||
Err(_) => {
|
||||
} else {
|
||||
println!("Failed to read next batch from database");
|
||||
break;
|
||||
}
|
||||
}
|
||||
height += BATCH_SIZE;
|
||||
}
|
||||
|
||||
|
@ -88,5 +90,5 @@ async fn main() {
|
|||
let generated = generate_hex(&hashes_of_hashes);
|
||||
write("src/data/hashes_of_hashes", generated).expect("Could not write file");
|
||||
|
||||
println!("Generated hashes up to block height {}", height);
|
||||
println!("Generated hashes up to block height {height}");
|
||||
}
|
||||
|
|
|
@ -1,12 +1,12 @@
|
|||
[
|
||||
hex!("1adffbaf832784406018009e07d3dc3a39da7edb6632523c119ed8acb32eb934"),
|
||||
hex!("ae960265e3398d04f3cd4f949ed13c2689424887c71c1441a03d900a9d3a777f"),
|
||||
hex!("938c72d267bbd3a17cdecbe02443d00012ee62d6e9f3524f5a914192110b1798"),
|
||||
hex!("de0c82e51549b6514b42a591fd5440dddb5cc0118ec461459a99017bf06a0a0a"),
|
||||
hex!("9a50f4586ec7e0fb58c6383048d3b334180235fd34bb714af20f1a3ebce4c911"),
|
||||
hex!("5a3942f9bb318d65997bf57c40e045d62e7edbe35f3dae57499c2c5554896543"),
|
||||
hex!("9dccee3b094cdd1b98e357c2c81bfcea798ea75efd94e67c6f5e86f428c5ec2c"),
|
||||
hex!("620397540d44f21c3c57c20e9d47c6aaf0b1bf4302a4d43e75f2e33edd1a4032"),
|
||||
hex!("ef6c612fb17bd70ac2ac69b2f85a421b138cc3a81daf622b077cb402dbf68377"),
|
||||
hex!("6815ecb2bd73a3ba5f20558bfe1b714c30d6892b290e0d6f6cbf18237cedf75a"),
|
||||
hex_literal::hex!("1adffbaf832784406018009e07d3dc3a39da7edb6632523c119ed8acb32eb934"),
|
||||
hex_literal::hex!("ae960265e3398d04f3cd4f949ed13c2689424887c71c1441a03d900a9d3a777f"),
|
||||
hex_literal::hex!("938c72d267bbd3a17cdecbe02443d00012ee62d6e9f3524f5a914192110b1798"),
|
||||
hex_literal::hex!("de0c82e51549b6514b42a591fd5440dddb5cc0118ec461459a99017bf06a0a0a"),
|
||||
hex_literal::hex!("9a50f4586ec7e0fb58c6383048d3b334180235fd34bb714af20f1a3ebce4c911"),
|
||||
hex_literal::hex!("5a3942f9bb318d65997bf57c40e045d62e7edbe35f3dae57499c2c5554896543"),
|
||||
hex_literal::hex!("9dccee3b094cdd1b98e357c2c81bfcea798ea75efd94e67c6f5e86f428c5ec2c"),
|
||||
hex_literal::hex!("620397540d44f21c3c57c20e9d47c6aaf0b1bf4302a4d43e75f2e33edd1a4032"),
|
||||
hex_literal::hex!("ef6c612fb17bd70ac2ac69b2f85a421b138cc3a81daf622b077cb402dbf68377"),
|
||||
hex_literal::hex!("6815ecb2bd73a3ba5f20558bfe1b714c30d6892b290e0d6f6cbf18237cedf75a"),
|
||||
]
|
||||
|
|
|
@ -6,8 +6,6 @@ use std::{
|
|||
task::{Context, Poll},
|
||||
};
|
||||
|
||||
#[allow(unused_imports)]
|
||||
use hex_literal::hex;
|
||||
use monero_serai::{
|
||||
block::Block,
|
||||
transaction::{Input, Transaction},
|
||||
|
@ -19,6 +17,7 @@ use cuprate_consensus::{
|
|||
transactions::new_tx_verification_data,
|
||||
};
|
||||
use cuprate_consensus_rules::{miner_tx::MinerTxError, ConsensusError};
|
||||
use cuprate_helper::cast::u64_to_usize;
|
||||
use cuprate_types::{VerifiedBlockInformation, VerifiedTransactionInformation};
|
||||
|
||||
use crate::{hash_of_hashes, BlockId, HashOfHashes};
|
||||
|
@ -31,9 +30,9 @@ const BATCH_SIZE: usize = 512;
|
|||
|
||||
#[cfg(test)]
|
||||
static HASHES_OF_HASHES: &[HashOfHashes] = &[
|
||||
hex!("3fdc9032c16d440f6c96be209c36d3d0e1aed61a2531490fe0ca475eb615c40a"),
|
||||
hex!("0102030405060708010203040506070801020304050607080102030405060708"),
|
||||
hex!("0102030405060708010203040506070801020304050607080102030405060708"),
|
||||
hex_literal::hex!("3fdc9032c16d440f6c96be209c36d3d0e1aed61a2531490fe0ca475eb615c40a"),
|
||||
hex_literal::hex!("0102030405060708010203040506070801020304050607080102030405060708"),
|
||||
hex_literal::hex!("0102030405060708010203040506070801020304050607080102030405060708"),
|
||||
];
|
||||
|
||||
#[cfg(test)]
|
||||
|
@ -44,14 +43,14 @@ fn max_height() -> u64 {
|
|||
(HASHES_OF_HASHES.len() * BATCH_SIZE) as u64
|
||||
}
|
||||
|
||||
#[derive(Debug, PartialEq)]
|
||||
#[derive(Debug, PartialEq, Eq)]
|
||||
pub struct ValidBlockId(BlockId);
|
||||
|
||||
fn valid_block_ids(block_ids: &[BlockId]) -> Vec<ValidBlockId> {
|
||||
block_ids.iter().map(|b| ValidBlockId(*b)).collect()
|
||||
}
|
||||
|
||||
#[allow(clippy::large_enum_variant)]
|
||||
#[expect(clippy::large_enum_variant)]
|
||||
pub enum FastSyncRequest {
|
||||
ValidateHashes {
|
||||
start_height: u64,
|
||||
|
@ -64,8 +63,8 @@ pub enum FastSyncRequest {
|
|||
},
|
||||
}
|
||||
|
||||
#[allow(clippy::large_enum_variant)]
|
||||
#[derive(Debug, PartialEq)]
|
||||
#[expect(clippy::large_enum_variant)]
|
||||
#[derive(Debug, PartialEq, Eq)]
|
||||
pub enum FastSyncResponse {
|
||||
ValidateHashes {
|
||||
validated_hashes: Vec<ValidBlockId>,
|
||||
|
@ -74,7 +73,7 @@ pub enum FastSyncResponse {
|
|||
ValidateBlock(VerifiedBlockInformation),
|
||||
}
|
||||
|
||||
#[derive(thiserror::Error, Debug, PartialEq)]
|
||||
#[derive(thiserror::Error, Debug, PartialEq, Eq)]
|
||||
pub enum FastSyncError {
|
||||
#[error("Block does not match its expected hash")]
|
||||
BlockHashMismatch,
|
||||
|
@ -127,9 +126,9 @@ where
|
|||
+ Send
|
||||
+ 'static,
|
||||
{
|
||||
#[allow(dead_code)]
|
||||
pub(crate) fn new(context_svc: C) -> FastSyncService<C> {
|
||||
FastSyncService { context_svc }
|
||||
#[expect(dead_code)]
|
||||
pub(crate) const fn new(context_svc: C) -> Self {
|
||||
Self { context_svc }
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -161,7 +160,7 @@ where
|
|||
FastSyncRequest::ValidateHashes {
|
||||
start_height,
|
||||
block_ids,
|
||||
} => validate_hashes(start_height, &block_ids).await,
|
||||
} => validate_hashes(start_height, &block_ids),
|
||||
FastSyncRequest::ValidateBlock { block, txs, token } => {
|
||||
validate_block(context_svc, block, txs, token).await
|
||||
}
|
||||
|
@ -170,11 +169,13 @@ where
|
|||
}
|
||||
}
|
||||
|
||||
async fn validate_hashes(
|
||||
fn validate_hashes(
|
||||
start_height: u64,
|
||||
block_ids: &[BlockId],
|
||||
) -> Result<FastSyncResponse, FastSyncError> {
|
||||
if start_height as usize % BATCH_SIZE != 0 {
|
||||
let start_height_usize = u64_to_usize(start_height);
|
||||
|
||||
if start_height_usize % BATCH_SIZE != 0 {
|
||||
return Err(FastSyncError::InvalidStartHeight);
|
||||
}
|
||||
|
||||
|
@ -182,9 +183,9 @@ async fn validate_hashes(
|
|||
return Err(FastSyncError::OutOfRange);
|
||||
}
|
||||
|
||||
let stop_height = start_height as usize + block_ids.len();
|
||||
let stop_height = start_height_usize + block_ids.len();
|
||||
|
||||
let batch_from = start_height as usize / BATCH_SIZE;
|
||||
let batch_from = start_height_usize / BATCH_SIZE;
|
||||
let batch_to = cmp::min(stop_height / BATCH_SIZE, HASHES_OF_HASHES.len());
|
||||
let n_batches = batch_to - batch_from;
|
||||
|
||||
|
@ -285,7 +286,7 @@ where
|
|||
block_blob,
|
||||
txs: verified_txs,
|
||||
block_hash,
|
||||
pow_hash: [0u8; 32],
|
||||
pow_hash: [0_u8; 32],
|
||||
height: *height,
|
||||
generated_coins,
|
||||
weight,
|
||||
|
@ -299,46 +300,36 @@ where
|
|||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
use tokio_test::block_on;
|
||||
|
||||
#[test]
|
||||
fn test_validate_hashes_errors() {
|
||||
let ids = [[1u8; 32], [2u8; 32], [3u8; 32], [4u8; 32], [5u8; 32]];
|
||||
let ids = [[1_u8; 32], [2_u8; 32], [3_u8; 32], [4_u8; 32], [5_u8; 32]];
|
||||
assert_eq!(
|
||||
block_on(validate_hashes(3, &[])),
|
||||
validate_hashes(3, &[]),
|
||||
Err(FastSyncError::InvalidStartHeight)
|
||||
);
|
||||
assert_eq!(
|
||||
block_on(validate_hashes(3, &ids)),
|
||||
validate_hashes(3, &ids),
|
||||
Err(FastSyncError::InvalidStartHeight)
|
||||
);
|
||||
|
||||
assert_eq!(
|
||||
block_on(validate_hashes(20, &[])),
|
||||
Err(FastSyncError::OutOfRange)
|
||||
);
|
||||
assert_eq!(
|
||||
block_on(validate_hashes(20, &ids)),
|
||||
Err(FastSyncError::OutOfRange)
|
||||
);
|
||||
assert_eq!(validate_hashes(20, &[]), Err(FastSyncError::OutOfRange));
|
||||
assert_eq!(validate_hashes(20, &ids), Err(FastSyncError::OutOfRange));
|
||||
|
||||
assert_eq!(validate_hashes(4, &[]), Err(FastSyncError::NothingToDo));
|
||||
assert_eq!(
|
||||
block_on(validate_hashes(4, &[])),
|
||||
Err(FastSyncError::NothingToDo)
|
||||
);
|
||||
assert_eq!(
|
||||
block_on(validate_hashes(4, &ids[..3])),
|
||||
validate_hashes(4, &ids[..3]),
|
||||
Err(FastSyncError::NothingToDo)
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_validate_hashes_success() {
|
||||
let ids = [[1u8; 32], [2u8; 32], [3u8; 32], [4u8; 32], [5u8; 32]];
|
||||
let ids = [[1_u8; 32], [2_u8; 32], [3_u8; 32], [4_u8; 32], [5_u8; 32]];
|
||||
let validated_hashes = valid_block_ids(&ids[0..4]);
|
||||
let unknown_hashes = ids[4..].to_vec();
|
||||
assert_eq!(
|
||||
block_on(validate_hashes(0, &ids)),
|
||||
validate_hashes(0, &ids),
|
||||
Ok(FastSyncResponse::ValidateHashes {
|
||||
validated_hashes,
|
||||
unknown_hashes
|
||||
|
@ -349,15 +340,10 @@ mod tests {
|
|||
#[test]
|
||||
fn test_validate_hashes_mismatch() {
|
||||
let ids = [
|
||||
[1u8; 32], [2u8; 32], [3u8; 32], [5u8; 32], [1u8; 32], [2u8; 32], [3u8; 32], [4u8; 32],
|
||||
[1_u8; 32], [2_u8; 32], [3_u8; 32], [5_u8; 32], [1_u8; 32], [2_u8; 32], [3_u8; 32],
|
||||
[4_u8; 32],
|
||||
];
|
||||
assert_eq!(
|
||||
block_on(validate_hashes(0, &ids)),
|
||||
Err(FastSyncError::Mismatch)
|
||||
);
|
||||
assert_eq!(
|
||||
block_on(validate_hashes(4, &ids)),
|
||||
Err(FastSyncError::Mismatch)
|
||||
);
|
||||
assert_eq!(validate_hashes(0, &ids), Err(FastSyncError::Mismatch));
|
||||
assert_eq!(validate_hashes(4, &ids), Err(FastSyncError::Mismatch));
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,3 +1,9 @@
|
|||
// Used in `create.rs`
|
||||
use clap as _;
|
||||
use cuprate_blockchain as _;
|
||||
use hex as _;
|
||||
use tokio as _;
|
||||
|
||||
pub mod fast_sync;
|
||||
pub mod util;
|
||||
|
||||
|
|
|
@ -7,7 +7,7 @@ authors = ["Boog900"]
|
|||
|
||||
[features]
|
||||
default = []
|
||||
proptest = ["dep:proptest", "dep:proptest-derive", "cuprate-types/proptest"]
|
||||
proptest = ["cuprate-types/proptest"]
|
||||
rayon = ["dep:rayon"]
|
||||
|
||||
[dependencies]
|
||||
|
@ -25,15 +25,16 @@ hex = { workspace = true, features = ["std"] }
|
|||
hex-literal = { workspace = true }
|
||||
crypto-bigint = { workspace = true }
|
||||
|
||||
cfg-if = { workspace = true }
|
||||
tracing = { workspace = true, features = ["std"] }
|
||||
thiserror = { workspace = true }
|
||||
|
||||
rayon = { workspace = true, optional = true }
|
||||
|
||||
proptest = {workspace = true, optional = true}
|
||||
proptest-derive = {workspace = true, optional = true}
|
||||
|
||||
[dev-dependencies]
|
||||
proptest = {workspace = true}
|
||||
proptest-derive = {workspace = true}
|
||||
tokio = {version = "1.35.0", features = ["rt-multi-thread", "macros"]}
|
||||
proptest = { workspace = true }
|
||||
proptest-derive = { workspace = true }
|
||||
tokio = { version = "1.35.0", features = ["rt-multi-thread", "macros"] }
|
||||
|
||||
[lints]
|
||||
workspace = true
|
|
@ -44,22 +44,22 @@ pub enum BlockError {
|
|||
MinerTxError(#[from] MinerTxError),
|
||||
}
|
||||
|
||||
/// A trait to represent the RandomX VM.
|
||||
/// A trait to represent the `RandomX` VM.
|
||||
pub trait RandomX {
|
||||
type Error;
|
||||
|
||||
fn calculate_hash(&self, buf: &[u8]) -> Result<[u8; 32], Self::Error>;
|
||||
}
|
||||
|
||||
/// Returns if this height is a RandomX seed height.
|
||||
pub fn is_randomx_seed_height(height: usize) -> bool {
|
||||
/// Returns if this height is a `RandomX` seed height.
|
||||
pub const fn is_randomx_seed_height(height: usize) -> bool {
|
||||
height % RX_SEEDHASH_EPOCH_BLOCKS == 0
|
||||
}
|
||||
|
||||
/// Returns the RandomX seed height for this block.
|
||||
/// Returns the `RandomX` seed height for this block.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/blocks.html#randomx-seed>
|
||||
pub fn randomx_seed_height(height: usize) -> usize {
|
||||
pub const fn randomx_seed_height(height: usize) -> usize {
|
||||
if height <= RX_SEEDHASH_EPOCH_BLOCKS + RX_SEEDHASH_EPOCH_LAG {
|
||||
0
|
||||
} else {
|
||||
|
@ -122,10 +122,10 @@ pub fn check_block_pow(hash: &[u8; 32], difficulty: u128) -> Result<(), BlockErr
|
|||
/// Returns the penalty free zone
|
||||
///
|
||||
/// <https://cuprate.github.io/monero-book/consensus_rules/blocks/weight_limit.html#penalty-free-zone>
|
||||
pub fn penalty_free_zone(hf: &HardFork) -> usize {
|
||||
if hf == &HardFork::V1 {
|
||||
pub fn penalty_free_zone(hf: HardFork) -> usize {
|
||||
if hf == HardFork::V1 {
|
||||
PENALTY_FREE_ZONE_1
|
||||
} else if hf >= &HardFork::V2 && hf < &HardFork::V5 {
|
||||
} else if hf >= HardFork::V2 && hf < HardFork::V5 {
|
||||
PENALTY_FREE_ZONE_2
|
||||
} else {
|
||||
PENALTY_FREE_ZONE_5
|
||||
|
@ -135,7 +135,7 @@ pub fn penalty_free_zone(hf: &HardFork) -> usize {
|
|||
/// Sanity check on the block blob size.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/blocks.html#block-weight-and-size>
|
||||
fn block_size_sanity_check(
|
||||
const fn block_size_sanity_check(
|
||||
block_blob_len: usize,
|
||||
effective_median: usize,
|
||||
) -> Result<(), BlockError> {
|
||||
|
@ -149,7 +149,7 @@ fn block_size_sanity_check(
|
|||
/// Sanity check on the block weight.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/blocks.html#block-weight-and-size>
|
||||
pub fn check_block_weight(
|
||||
pub const fn check_block_weight(
|
||||
block_weight: usize,
|
||||
median_for_block_reward: usize,
|
||||
) -> Result<(), BlockError> {
|
||||
|
@ -163,7 +163,7 @@ pub fn check_block_weight(
|
|||
/// Sanity check on number of txs in the block.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/blocks.html#amount-of-transactions>
|
||||
fn check_amount_txs(number_none_miner_txs: usize) -> Result<(), BlockError> {
|
||||
const fn check_amount_txs(number_none_miner_txs: usize) -> Result<(), BlockError> {
|
||||
if number_none_miner_txs + 1 > 0x10000000 {
|
||||
Err(BlockError::TooManyTxs)
|
||||
} else {
|
||||
|
@ -175,10 +175,10 @@ fn check_amount_txs(number_none_miner_txs: usize) -> Result<(), BlockError> {
|
|||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/blocks.html#previous-id>
|
||||
fn check_prev_id(block: &Block, top_hash: &[u8; 32]) -> Result<(), BlockError> {
|
||||
if &block.header.previous != top_hash {
|
||||
Err(BlockError::PreviousIDIncorrect)
|
||||
} else {
|
||||
if &block.header.previous == top_hash {
|
||||
Ok(())
|
||||
} else {
|
||||
Err(BlockError::PreviousIDIncorrect)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -273,7 +273,7 @@ pub fn check_block(
|
|||
block_weight,
|
||||
block_chain_ctx.median_weight_for_block_reward,
|
||||
block_chain_ctx.already_generated_coins,
|
||||
&block_chain_ctx.current_hf,
|
||||
block_chain_ctx.current_hf,
|
||||
)?;
|
||||
|
||||
Ok((vote, generated_coins))
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
#[rustfmt::skip]
|
||||
/// Decomposed amount table.
|
||||
pub static DECOMPOSED_AMOUNTS: [u64; 172] = [
|
||||
pub(crate) static DECOMPOSED_AMOUNTS: [u64; 172] = [
|
||||
1, 2, 3, 4, 5, 6, 7, 8, 9,
|
||||
10, 20, 30, 40, 50, 60, 70, 80, 90,
|
||||
100, 200, 300, 400, 500, 600, 700, 800, 900,
|
||||
|
@ -40,8 +40,8 @@ mod tests {
|
|||
|
||||
#[test]
|
||||
fn decomposed_amounts_return_decomposed() {
|
||||
for amount in DECOMPOSED_AMOUNTS.iter() {
|
||||
assert!(is_decomposed_amount(amount))
|
||||
for amount in &DECOMPOSED_AMOUNTS {
|
||||
assert!(is_decomposed_amount(amount));
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -8,7 +8,7 @@ use monero_serai::{
|
|||
|
||||
use cuprate_helper::network::Network;
|
||||
|
||||
const fn genesis_nonce(network: &Network) -> u32 {
|
||||
const fn genesis_nonce(network: Network) -> u32 {
|
||||
match network {
|
||||
Network::Mainnet => 10000,
|
||||
Network::Testnet => 10001,
|
||||
|
@ -16,7 +16,7 @@ const fn genesis_nonce(network: &Network) -> u32 {
|
|||
}
|
||||
}
|
||||
|
||||
fn genesis_miner_tx(network: &Network) -> Transaction {
|
||||
fn genesis_miner_tx(network: Network) -> Transaction {
|
||||
Transaction::read(&mut hex::decode(match network {
|
||||
Network::Mainnet | Network::Testnet => "013c01ff0001ffffffffffff03029b2e4c0281c0b02e7c53291a94d1d0cbff8883f8024f5142ee494ffbbd08807121017767aafcde9be00dcfd098715ebcf7f410daebc582fda69d24a28e9d0bc890d1",
|
||||
Network::Stagenet => "013c01ff0001ffffffffffff0302df5d56da0c7d643ddd1ce61901c7bdc5fb1738bfe39fbe69c28a3a7032729c0f2101168d0c4ca86fb55a4cf6a36d31431be1c53a3bd7411bb24e8832410289fa6f3b"
|
||||
|
@ -26,7 +26,7 @@ fn genesis_miner_tx(network: &Network) -> Transaction {
|
|||
/// Generates the Monero genesis block.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/genesis_block.html>
|
||||
pub fn generate_genesis_block(network: &Network) -> Block {
|
||||
pub fn generate_genesis_block(network: Network) -> Block {
|
||||
Block {
|
||||
header: BlockHeader {
|
||||
hardfork_version: 1,
|
||||
|
@ -47,19 +47,19 @@ mod tests {
|
|||
#[test]
|
||||
fn generate_genesis_blocks() {
|
||||
assert_eq!(
|
||||
&generate_genesis_block(&Network::Mainnet).hash(),
|
||||
&generate_genesis_block(Network::Mainnet).hash(),
|
||||
hex::decode("418015bb9ae982a1975da7d79277c2705727a56894ba0fb246adaabb1f4632e3")
|
||||
.unwrap()
|
||||
.as_slice()
|
||||
);
|
||||
assert_eq!(
|
||||
&generate_genesis_block(&Network::Testnet).hash(),
|
||||
&generate_genesis_block(Network::Testnet).hash(),
|
||||
hex::decode("48ca7cd3c8de5b6a4d53d2861fbdaedca141553559f9be9520068053cda8430b")
|
||||
.unwrap()
|
||||
.as_slice()
|
||||
);
|
||||
assert_eq!(
|
||||
&generate_genesis_block(&Network::Stagenet).hash(),
|
||||
&generate_genesis_block(Network::Stagenet).hash(),
|
||||
hex::decode("76ee3cc98646292206cd3e86f74d88b4dcc1d937088645e9b0cbca84b7ce74eb")
|
||||
.unwrap()
|
||||
.as_slice()
|
||||
|
|
|
@ -25,10 +25,10 @@ pub fn check_block_version_vote(
|
|||
) -> Result<(), HardForkError> {
|
||||
// self = current hf
|
||||
if hf != version {
|
||||
Err(HardForkError::VersionIncorrect)?;
|
||||
return Err(HardForkError::VersionIncorrect);
|
||||
}
|
||||
if hf > vote {
|
||||
Err(HardForkError::VoteTooLow)?;
|
||||
return Err(HardForkError::VoteTooLow);
|
||||
}
|
||||
|
||||
Ok(())
|
||||
|
@ -41,8 +41,8 @@ pub struct HFInfo {
|
|||
threshold: usize,
|
||||
}
|
||||
impl HFInfo {
|
||||
pub const fn new(height: usize, threshold: usize) -> HFInfo {
|
||||
HFInfo { height, threshold }
|
||||
pub const fn new(height: usize, threshold: usize) -> Self {
|
||||
Self { height, threshold }
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -51,7 +51,7 @@ impl HFInfo {
|
|||
pub struct HFsInfo([HFInfo; NUMB_OF_HARD_FORKS]);
|
||||
|
||||
impl HFsInfo {
|
||||
pub fn info_for_hf(&self, hf: &HardFork) -> HFInfo {
|
||||
pub const fn info_for_hf(&self, hf: &HardFork) -> HFInfo {
|
||||
self.0[*hf as usize - 1]
|
||||
}
|
||||
|
||||
|
@ -62,7 +62,7 @@ impl HFsInfo {
|
|||
/// Returns the main-net hard-fork information.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/hardforks.html#Mainnet-Hard-Forks>
|
||||
pub const fn main_net() -> HFsInfo {
|
||||
pub const fn main_net() -> Self {
|
||||
Self([
|
||||
HFInfo::new(0, 0),
|
||||
HFInfo::new(1009827, 0),
|
||||
|
@ -86,7 +86,7 @@ impl HFsInfo {
|
|||
/// Returns the test-net hard-fork information.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/hardforks.html#Testnet-Hard-Forks>
|
||||
pub const fn test_net() -> HFsInfo {
|
||||
pub const fn test_net() -> Self {
|
||||
Self([
|
||||
HFInfo::new(0, 0),
|
||||
HFInfo::new(624634, 0),
|
||||
|
@ -110,7 +110,7 @@ impl HFsInfo {
|
|||
/// Returns the test-net hard-fork information.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/hardforks.html#Stagenet-Hard-Forks>
|
||||
pub const fn stage_net() -> HFsInfo {
|
||||
pub const fn stage_net() -> Self {
|
||||
Self([
|
||||
HFInfo::new(0, 0),
|
||||
HFInfo::new(32000, 0),
|
||||
|
@ -165,8 +165,8 @@ impl Display for HFVotes {
|
|||
}
|
||||
|
||||
impl HFVotes {
|
||||
pub fn new(window_size: usize) -> HFVotes {
|
||||
HFVotes {
|
||||
pub fn new(window_size: usize) -> Self {
|
||||
Self {
|
||||
votes: [0; NUMB_OF_HARD_FORKS],
|
||||
vote_list: VecDeque::with_capacity(window_size),
|
||||
window_size,
|
||||
|
@ -251,6 +251,6 @@ impl HFVotes {
|
|||
/// Returns the votes needed for a hard-fork.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/hardforks.html#accepting-a-fork>
|
||||
pub fn votes_needed(threshold: usize, window: usize) -> usize {
|
||||
pub const fn votes_needed(threshold: usize, window: usize) -> usize {
|
||||
(threshold * window).div_ceil(100)
|
||||
}
|
||||
|
|
|
@ -51,7 +51,7 @@ proptest! {
|
|||
prop_assert_eq!(hf_votes.total_votes(), hf_votes.vote_list.len());
|
||||
|
||||
let mut votes = [0_usize; NUMB_OF_HARD_FORKS];
|
||||
for vote in hf_votes.vote_list.iter() {
|
||||
for vote in &hf_votes.vote_list {
|
||||
// manually go through the list of votes tallying
|
||||
votes[*vote as usize - 1] += 1;
|
||||
}
|
||||
|
@ -61,9 +61,9 @@ proptest! {
|
|||
|
||||
#[test]
|
||||
fn window_size_kept_constant(mut hf_votes in arb_full_hf_votes(), new_votes in any::<Vec<HardFork>>()) {
|
||||
for new_vote in new_votes.into_iter() {
|
||||
for new_vote in new_votes {
|
||||
hf_votes.add_vote_for_hf(&new_vote);
|
||||
prop_assert_eq!(hf_votes.total_votes(), TEST_WINDOW_SIZE)
|
||||
prop_assert_eq!(hf_votes.total_votes(), TEST_WINDOW_SIZE);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -1,3 +1,12 @@
|
|||
cfg_if::cfg_if! {
|
||||
// Used in external `tests/`.
|
||||
if #[cfg(test)] {
|
||||
use proptest as _;
|
||||
use proptest_derive as _;
|
||||
use tokio as _;
|
||||
}
|
||||
}
|
||||
|
||||
use std::time::{SystemTime, UNIX_EPOCH};
|
||||
|
||||
pub mod batch_verifier;
|
||||
|
|
|
@ -41,7 +41,7 @@ const MINER_TX_TIME_LOCKED_BLOCKS: usize = 60;
|
|||
/// the block.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/blocks/reward.html#calculating-base-block-reward>
|
||||
fn calculate_base_reward(already_generated_coins: u64, hf: &HardFork) -> u64 {
|
||||
fn calculate_base_reward(already_generated_coins: u64, hf: HardFork) -> u64 {
|
||||
let target_mins = hf.block_time().as_secs() / 60;
|
||||
let emission_speed_factor = 20 - (target_mins - 1);
|
||||
((MONEY_SUPPLY - already_generated_coins) >> emission_speed_factor)
|
||||
|
@ -55,7 +55,7 @@ pub fn calculate_block_reward(
|
|||
block_weight: usize,
|
||||
median_bw: usize,
|
||||
already_generated_coins: u64,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> u64 {
|
||||
let base_reward = calculate_base_reward(already_generated_coins, hf);
|
||||
|
||||
|
@ -76,9 +76,9 @@ pub fn calculate_block_reward(
|
|||
/// Checks the miner transactions version.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/blocks/miner_tx.html#version>
|
||||
fn check_miner_tx_version(tx_version: &TxVersion, hf: &HardFork) -> Result<(), MinerTxError> {
|
||||
fn check_miner_tx_version(tx_version: TxVersion, hf: HardFork) -> Result<(), MinerTxError> {
|
||||
// The TxVersion enum checks if the version is not 1 or 2
|
||||
if hf >= &HardFork::V12 && tx_version != &TxVersion::RingCT {
|
||||
if hf >= HardFork::V12 && tx_version != TxVersion::RingCT {
|
||||
Err(MinerTxError::VersionInvalid)
|
||||
} else {
|
||||
Ok(())
|
||||
|
@ -95,31 +95,31 @@ fn check_inputs(inputs: &[Input], chain_height: usize) -> Result<(), MinerTxErro
|
|||
|
||||
match &inputs[0] {
|
||||
Input::Gen(height) => {
|
||||
if height != &chain_height {
|
||||
Err(MinerTxError::InputsHeightIncorrect)
|
||||
} else {
|
||||
if height == &chain_height {
|
||||
Ok(())
|
||||
} else {
|
||||
Err(MinerTxError::InputsHeightIncorrect)
|
||||
}
|
||||
}
|
||||
_ => Err(MinerTxError::InputNotOfTypeGen),
|
||||
Input::ToKey { .. } => Err(MinerTxError::InputNotOfTypeGen),
|
||||
}
|
||||
}
|
||||
|
||||
/// Checks the miner transaction has a correct time lock.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/blocks/miner_tx.html#unlock-time>
|
||||
fn check_time_lock(time_lock: &Timelock, chain_height: usize) -> Result<(), MinerTxError> {
|
||||
const fn check_time_lock(time_lock: &Timelock, chain_height: usize) -> Result<(), MinerTxError> {
|
||||
match time_lock {
|
||||
&Timelock::Block(till_height) => {
|
||||
// Lock times above this amount are timestamps not blocks.
|
||||
// This is just for safety though and shouldn't actually be hit.
|
||||
if till_height > MAX_BLOCK_HEIGHT_USIZE {
|
||||
Err(MinerTxError::InvalidLockTime)?;
|
||||
return Err(MinerTxError::InvalidLockTime);
|
||||
}
|
||||
if till_height != chain_height + MINER_TX_TIME_LOCKED_BLOCKS {
|
||||
Err(MinerTxError::InvalidLockTime)
|
||||
} else {
|
||||
if till_height == chain_height + MINER_TX_TIME_LOCKED_BLOCKS {
|
||||
Ok(())
|
||||
} else {
|
||||
Err(MinerTxError::InvalidLockTime)
|
||||
}
|
||||
}
|
||||
_ => Err(MinerTxError::InvalidLockTime),
|
||||
|
@ -132,18 +132,18 @@ fn check_time_lock(time_lock: &Timelock, chain_height: usize) -> Result<(), Mine
|
|||
/// && <https://monero-book.cuprate.org/consensus_rules/blocks/miner_tx.html#zero-amount-v1-output>
|
||||
fn sum_outputs(
|
||||
outputs: &[Output],
|
||||
hf: &HardFork,
|
||||
tx_version: &TxVersion,
|
||||
hf: HardFork,
|
||||
tx_version: TxVersion,
|
||||
) -> Result<u64, MinerTxError> {
|
||||
let mut sum: u64 = 0;
|
||||
for out in outputs {
|
||||
let amt = out.amount.unwrap_or(0);
|
||||
|
||||
if tx_version == &TxVersion::RingSignatures && amt == 0 {
|
||||
if tx_version == TxVersion::RingSignatures && amt == 0 {
|
||||
return Err(MinerTxError::OutputAmountIncorrect);
|
||||
}
|
||||
|
||||
if hf == &HardFork::V3 && !is_decomposed_amount(&amt) {
|
||||
if hf == HardFork::V3 && !is_decomposed_amount(&amt) {
|
||||
return Err(MinerTxError::OutputNotDecomposed);
|
||||
}
|
||||
sum = sum.checked_add(amt).ok_or(MinerTxError::OutputsOverflow)?;
|
||||
|
@ -158,9 +158,9 @@ fn check_total_output_amt(
|
|||
total_output: u64,
|
||||
reward: u64,
|
||||
fees: u64,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> Result<u64, MinerTxError> {
|
||||
if hf == &HardFork::V1 || hf >= &HardFork::V12 {
|
||||
if hf == HardFork::V1 || hf >= HardFork::V12 {
|
||||
if total_output != reward + fees {
|
||||
return Err(MinerTxError::OutputAmountIncorrect);
|
||||
}
|
||||
|
@ -186,16 +186,16 @@ pub fn check_miner_tx(
|
|||
block_weight: usize,
|
||||
median_bw: usize,
|
||||
already_generated_coins: u64,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> Result<u64, MinerTxError> {
|
||||
let tx_version = TxVersion::from_raw(tx.version()).ok_or(MinerTxError::VersionInvalid)?;
|
||||
check_miner_tx_version(&tx_version, hf)?;
|
||||
check_miner_tx_version(tx_version, hf)?;
|
||||
|
||||
// ref: <https://monero-book.cuprate.org/consensus_rules/blocks/miner_tx.html#ringct-type>
|
||||
match tx {
|
||||
Transaction::V1 { .. } => (),
|
||||
Transaction::V2 { proofs, .. } => {
|
||||
if hf >= &HardFork::V12 && proofs.is_some() {
|
||||
if hf >= HardFork::V12 && proofs.is_some() {
|
||||
return Err(MinerTxError::RCTTypeNotNULL);
|
||||
}
|
||||
}
|
||||
|
@ -208,7 +208,7 @@ pub fn check_miner_tx(
|
|||
check_output_types(&tx.prefix().outputs, hf).map_err(|_| MinerTxError::InvalidOutputType)?;
|
||||
|
||||
let reward = calculate_block_reward(block_weight, median_bw, already_generated_coins, hf);
|
||||
let total_outs = sum_outputs(&tx.prefix().outputs, hf, &tx_version)?;
|
||||
let total_outs = sum_outputs(&tx.prefix().outputs, hf, tx_version)?;
|
||||
|
||||
check_total_output_amt(total_outs, reward, total_fees, hf)
|
||||
}
|
||||
|
@ -222,7 +222,7 @@ mod tests {
|
|||
proptest! {
|
||||
#[test]
|
||||
fn tail_emission(generated_coins in any::<u64>(), hf in any::<HardFork>()) {
|
||||
prop_assert!(calculate_base_reward(generated_coins, &hf) >= MINIMUM_REWARD_PER_MIN * hf.block_time().as_secs() / 60)
|
||||
prop_assert!(calculate_base_reward(generated_coins, hf) >= MINIMUM_REWARD_PER_MIN * hf.block_time().as_secs() / 60);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -99,11 +99,8 @@ fn check_output_keys(outputs: &[Output]) -> Result<(), TransactionError> {
|
|||
///
|
||||
/// <https://monero-book.cuprate.org/consensus_rules/transactions/outputs.html#output-type>
|
||||
/// <https://monero-book.cuprate.org/consensus_rules/blocks/miner_tx.html#output-type>
|
||||
pub(crate) fn check_output_types(
|
||||
outputs: &[Output],
|
||||
hf: &HardFork,
|
||||
) -> Result<(), TransactionError> {
|
||||
if hf == &HardFork::V15 {
|
||||
pub(crate) fn check_output_types(outputs: &[Output], hf: HardFork) -> Result<(), TransactionError> {
|
||||
if hf == HardFork::V15 {
|
||||
for outs in outputs.windows(2) {
|
||||
if outs[0].view_tag.is_some() != outs[1].view_tag.is_some() {
|
||||
return Err(TransactionError::OutputTypeInvalid);
|
||||
|
@ -113,8 +110,8 @@ pub(crate) fn check_output_types(
|
|||
}
|
||||
|
||||
for out in outputs {
|
||||
if hf <= &HardFork::V14 && out.view_tag.is_some()
|
||||
|| hf >= &HardFork::V16 && out.view_tag.is_none()
|
||||
if hf <= HardFork::V14 && out.view_tag.is_some()
|
||||
|| hf >= HardFork::V16 && out.view_tag.is_none()
|
||||
{
|
||||
return Err(TransactionError::OutputTypeInvalid);
|
||||
}
|
||||
|
@ -125,12 +122,12 @@ pub(crate) fn check_output_types(
|
|||
/// Checks the individual outputs amount for version 1 txs.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/outputs.html#output-amount>
|
||||
fn check_output_amount_v1(amount: u64, hf: &HardFork) -> Result<(), TransactionError> {
|
||||
fn check_output_amount_v1(amount: u64, hf: HardFork) -> Result<(), TransactionError> {
|
||||
if amount == 0 {
|
||||
return Err(TransactionError::ZeroOutputForV1);
|
||||
}
|
||||
|
||||
if hf >= &HardFork::V2 && !is_decomposed_amount(&amount) {
|
||||
if hf >= HardFork::V2 && !is_decomposed_amount(&amount) {
|
||||
return Err(TransactionError::AmountNotDecomposed);
|
||||
}
|
||||
|
||||
|
@ -140,7 +137,7 @@ fn check_output_amount_v1(amount: u64, hf: &HardFork) -> Result<(), TransactionE
|
|||
/// Checks the individual outputs amount for version 2 txs.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/outputs.html#output-amount>
|
||||
fn check_output_amount_v2(amount: u64) -> Result<(), TransactionError> {
|
||||
const fn check_output_amount_v2(amount: u64) -> Result<(), TransactionError> {
|
||||
if amount == 0 {
|
||||
Ok(())
|
||||
} else {
|
||||
|
@ -154,8 +151,8 @@ fn check_output_amount_v2(amount: u64) -> Result<(), TransactionError> {
|
|||
/// && <https://monero-book.cuprate.org/consensus_rules/transactions/outputs.html#outputs-must-not-overflow>
|
||||
fn sum_outputs(
|
||||
outputs: &[Output],
|
||||
hf: &HardFork,
|
||||
tx_version: &TxVersion,
|
||||
hf: HardFork,
|
||||
tx_version: TxVersion,
|
||||
) -> Result<u64, TransactionError> {
|
||||
let mut sum: u64 = 0;
|
||||
|
||||
|
@ -181,15 +178,15 @@ fn sum_outputs(
|
|||
/// && <https://monero-book.cuprate.org/consensus_rules/transactions/ring_ct/bulletproofs+.html#max-outputs>
|
||||
fn check_number_of_outputs(
|
||||
outputs: usize,
|
||||
hf: &HardFork,
|
||||
tx_version: &TxVersion,
|
||||
hf: HardFork,
|
||||
tx_version: TxVersion,
|
||||
bp_or_bpp: bool,
|
||||
) -> Result<(), TransactionError> {
|
||||
if tx_version == &TxVersion::RingSignatures {
|
||||
if tx_version == TxVersion::RingSignatures {
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
if hf >= &HardFork::V12 && outputs < 2 {
|
||||
if hf >= HardFork::V12 && outputs < 2 {
|
||||
return Err(TransactionError::InvalidNumberOfOutputs);
|
||||
}
|
||||
|
||||
|
@ -207,8 +204,8 @@ fn check_number_of_outputs(
|
|||
/// && <https://monero-book.cuprate.org/consensus_rules/transactions/ring_ct/bulletproofs+.html#max-outputs>
|
||||
fn check_outputs_semantics(
|
||||
outputs: &[Output],
|
||||
hf: &HardFork,
|
||||
tx_version: &TxVersion,
|
||||
hf: HardFork,
|
||||
tx_version: TxVersion,
|
||||
bp_or_bpp: bool,
|
||||
) -> Result<u64, TransactionError> {
|
||||
check_output_types(outputs, hf)?;
|
||||
|
@ -223,11 +220,11 @@ fn check_outputs_semantics(
|
|||
/// Checks if an outputs unlock time has passed.
|
||||
///
|
||||
/// <https://monero-book.cuprate.org/consensus_rules/transactions/unlock_time.html>
|
||||
pub fn output_unlocked(
|
||||
pub const fn output_unlocked(
|
||||
time_lock: &Timelock,
|
||||
current_chain_height: usize,
|
||||
current_time_lock_timestamp: u64,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> bool {
|
||||
match *time_lock {
|
||||
Timelock::None => true,
|
||||
|
@ -243,7 +240,7 @@ pub fn output_unlocked(
|
|||
/// Returns if a locked output, which uses a block height, can be spent.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/unlock_time.html#block-height>
|
||||
fn check_block_time_lock(unlock_height: usize, current_chain_height: usize) -> bool {
|
||||
const fn check_block_time_lock(unlock_height: usize, current_chain_height: usize) -> bool {
|
||||
// current_chain_height = 1 + top height
|
||||
unlock_height <= current_chain_height
|
||||
}
|
||||
|
@ -251,10 +248,10 @@ fn check_block_time_lock(unlock_height: usize, current_chain_height: usize) -> b
|
|||
/// Returns if a locked output, which uses a block height, can be spent.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/unlock_time.html#timestamp>
|
||||
fn check_timestamp_time_lock(
|
||||
const fn check_timestamp_time_lock(
|
||||
unlock_timestamp: u64,
|
||||
current_time_lock_timestamp: u64,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> bool {
|
||||
current_time_lock_timestamp + hf.block_time().as_secs() >= unlock_timestamp
|
||||
}
|
||||
|
@ -269,19 +266,19 @@ fn check_all_time_locks(
|
|||
time_locks: &[Timelock],
|
||||
current_chain_height: usize,
|
||||
current_time_lock_timestamp: u64,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> Result<(), TransactionError> {
|
||||
time_locks.iter().try_for_each(|time_lock| {
|
||||
if !output_unlocked(
|
||||
if output_unlocked(
|
||||
time_lock,
|
||||
current_chain_height,
|
||||
current_time_lock_timestamp,
|
||||
hf,
|
||||
) {
|
||||
Ok(())
|
||||
} else {
|
||||
tracing::debug!("Transaction invalid: one or more inputs locked, lock: {time_lock:?}.");
|
||||
Err(TransactionError::OneOrMoreRingMembersLocked)
|
||||
} else {
|
||||
Ok(())
|
||||
}
|
||||
})
|
||||
}
|
||||
|
@ -292,11 +289,11 @@ fn check_all_time_locks(
|
|||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/inputs.html#minimum-decoys>
|
||||
/// && <https://monero-book.cuprate.org/consensus_rules/transactions/inputs.html#equal-number-of-decoys>
|
||||
pub fn check_decoy_info(decoy_info: &DecoyInfo, hf: &HardFork) -> Result<(), TransactionError> {
|
||||
if hf == &HardFork::V15 {
|
||||
pub fn check_decoy_info(decoy_info: &DecoyInfo, hf: HardFork) -> Result<(), TransactionError> {
|
||||
if hf == HardFork::V15 {
|
||||
// Hard-fork 15 allows both v14 and v16 rules
|
||||
return check_decoy_info(decoy_info, &HardFork::V14)
|
||||
.or_else(|_| check_decoy_info(decoy_info, &HardFork::V16));
|
||||
return check_decoy_info(decoy_info, HardFork::V14)
|
||||
.or_else(|_| check_decoy_info(decoy_info, HardFork::V16));
|
||||
}
|
||||
|
||||
let current_minimum_decoys = minimum_decoys(hf);
|
||||
|
@ -310,13 +307,13 @@ pub fn check_decoy_info(decoy_info: &DecoyInfo, hf: &HardFork) -> Result<(), Tra
|
|||
if decoy_info.mixable > 1 {
|
||||
return Err(TransactionError::MoreThanOneMixableInputWithUnmixable);
|
||||
}
|
||||
} else if hf >= &HardFork::V8 && decoy_info.min_decoys != current_minimum_decoys {
|
||||
} else if hf >= HardFork::V8 && decoy_info.min_decoys != current_minimum_decoys {
|
||||
// From V8 enforce the minimum used number of rings is the default minimum.
|
||||
return Err(TransactionError::InputDoesNotHaveExpectedNumbDecoys);
|
||||
}
|
||||
|
||||
// From v12 all inputs must have the same number of decoys.
|
||||
if hf >= &HardFork::V12 && decoy_info.min_decoys != decoy_info.max_decoys {
|
||||
if hf >= HardFork::V12 && decoy_info.min_decoys != decoy_info.max_decoys {
|
||||
return Err(TransactionError::InputDoesNotHaveExpectedNumbDecoys);
|
||||
}
|
||||
|
||||
|
@ -334,19 +331,19 @@ fn check_key_images(input: &Input) -> Result<(), TransactionError> {
|
|||
return Err(TransactionError::KeyImageIsNotInPrimeSubGroup);
|
||||
}
|
||||
}
|
||||
_ => Err(TransactionError::IncorrectInputType)?,
|
||||
Input::Gen(_) => return Err(TransactionError::IncorrectInputType),
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// Checks that the input is of type [`Input::ToKey`] aka txin_to_key.
|
||||
/// Checks that the input is of type [`Input::ToKey`] aka `txin_to_key`.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/inputs.html#input-type>
|
||||
fn check_input_type(input: &Input) -> Result<(), TransactionError> {
|
||||
const fn check_input_type(input: &Input) -> Result<(), TransactionError> {
|
||||
match input {
|
||||
Input::ToKey { .. } => Ok(()),
|
||||
_ => Err(TransactionError::IncorrectInputType)?,
|
||||
Input::Gen(_) => Err(TransactionError::IncorrectInputType),
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -362,15 +359,15 @@ fn check_input_has_decoys(input: &Input) -> Result<(), TransactionError> {
|
|||
Ok(())
|
||||
}
|
||||
}
|
||||
_ => Err(TransactionError::IncorrectInputType)?,
|
||||
Input::Gen(_) => Err(TransactionError::IncorrectInputType),
|
||||
}
|
||||
}
|
||||
|
||||
/// Checks that the ring members for the input are unique after hard-fork 6.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/inputs.html#unique-ring-members>
|
||||
fn check_ring_members_unique(input: &Input, hf: &HardFork) -> Result<(), TransactionError> {
|
||||
if hf >= &HardFork::V6 {
|
||||
fn check_ring_members_unique(input: &Input, hf: HardFork) -> Result<(), TransactionError> {
|
||||
if hf >= HardFork::V6 {
|
||||
match input {
|
||||
Input::ToKey { key_offsets, .. } => key_offsets.iter().skip(1).try_for_each(|offset| {
|
||||
if *offset == 0 {
|
||||
|
@ -379,7 +376,7 @@ fn check_ring_members_unique(input: &Input, hf: &HardFork) -> Result<(), Transac
|
|||
Ok(())
|
||||
}
|
||||
}),
|
||||
_ => Err(TransactionError::IncorrectInputType)?,
|
||||
Input::Gen(_) => Err(TransactionError::IncorrectInputType),
|
||||
}
|
||||
} else {
|
||||
Ok(())
|
||||
|
@ -389,23 +386,22 @@ fn check_ring_members_unique(input: &Input, hf: &HardFork) -> Result<(), Transac
|
|||
/// Checks that from hf 7 the inputs are sorted by key image.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/inputs.html#sorted-inputs>
|
||||
fn check_inputs_sorted(inputs: &[Input], hf: &HardFork) -> Result<(), TransactionError> {
|
||||
fn check_inputs_sorted(inputs: &[Input], hf: HardFork) -> Result<(), TransactionError> {
|
||||
let get_ki = |inp: &Input| match inp {
|
||||
Input::ToKey { key_image, .. } => Ok(key_image.compress().to_bytes()),
|
||||
_ => Err(TransactionError::IncorrectInputType),
|
||||
Input::Gen(_) => Err(TransactionError::IncorrectInputType),
|
||||
};
|
||||
|
||||
if hf >= &HardFork::V7 {
|
||||
if hf >= HardFork::V7 {
|
||||
for inps in inputs.windows(2) {
|
||||
match get_ki(&inps[0])?.cmp(&get_ki(&inps[1])?) {
|
||||
Ordering::Greater => (),
|
||||
_ => return Err(TransactionError::InputsAreNotOrdered),
|
||||
}
|
||||
}
|
||||
Ok(())
|
||||
} else {
|
||||
Ok(())
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// Checks the youngest output is at least 10 blocks old.
|
||||
|
@ -414,9 +410,9 @@ fn check_inputs_sorted(inputs: &[Input], hf: &HardFork) -> Result<(), Transactio
|
|||
fn check_10_block_lock(
|
||||
youngest_used_out_height: usize,
|
||||
current_chain_height: usize,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> Result<(), TransactionError> {
|
||||
if hf >= &HardFork::V12 {
|
||||
if hf >= HardFork::V12 {
|
||||
if youngest_used_out_height + 10 > current_chain_height {
|
||||
tracing::debug!(
|
||||
"Transaction invalid: One or more ring members younger than 10 blocks."
|
||||
|
@ -442,7 +438,7 @@ fn sum_inputs_check_overflow(inputs: &[Input]) -> Result<u64, TransactionError>
|
|||
.checked_add(amount.unwrap_or(0))
|
||||
.ok_or(TransactionError::InputsOverflow)?;
|
||||
}
|
||||
_ => Err(TransactionError::IncorrectInputType)?,
|
||||
Input::Gen(_) => return Err(TransactionError::IncorrectInputType),
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -454,7 +450,7 @@ fn sum_inputs_check_overflow(inputs: &[Input]) -> Result<u64, TransactionError>
|
|||
/// Semantic rules are rules that don't require blockchain context, the hard-fork does not require blockchain context as:
|
||||
/// - The tx-pool will use the current hard-fork
|
||||
/// - When syncing the hard-fork is in the block header.
|
||||
fn check_inputs_semantics(inputs: &[Input], hf: &HardFork) -> Result<u64, TransactionError> {
|
||||
fn check_inputs_semantics(inputs: &[Input], hf: HardFork) -> Result<u64, TransactionError> {
|
||||
// <https://monero-book.cuprate.org/consensus_rules/transactions/inputs.html#no-empty-inputs>
|
||||
if inputs.is_empty() {
|
||||
return Err(TransactionError::NoInputs);
|
||||
|
@ -481,14 +477,14 @@ fn check_inputs_contextual(
|
|||
inputs: &[Input],
|
||||
tx_ring_members_info: &TxRingMembersInfo,
|
||||
current_chain_height: usize,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> Result<(), TransactionError> {
|
||||
// This rule is not contained in monero-core explicitly, but it is enforced by how Monero picks ring members.
|
||||
// When picking ring members monerod will only look in the DB at past blocks so an output has to be younger
|
||||
// than this transaction to be used in this tx.
|
||||
if tx_ring_members_info.youngest_used_out_height >= current_chain_height {
|
||||
tracing::debug!("Transaction invalid: One or more ring members too young.");
|
||||
Err(TransactionError::OneOrMoreRingMembersLocked)?;
|
||||
return Err(TransactionError::OneOrMoreRingMembersLocked);
|
||||
}
|
||||
|
||||
check_10_block_lock(
|
||||
|
@ -500,7 +496,7 @@ fn check_inputs_contextual(
|
|||
if let Some(decoys_info) = &tx_ring_members_info.decoy_info {
|
||||
check_decoy_info(decoys_info, hf)?;
|
||||
} else {
|
||||
assert_eq!(hf, &HardFork::V1);
|
||||
assert_eq!(hf, HardFork::V1);
|
||||
}
|
||||
|
||||
for input in inputs {
|
||||
|
@ -517,22 +513,22 @@ fn check_inputs_contextual(
|
|||
/// <https://monero-book.cuprate.org/consensus_rules/transactions.html#version>
|
||||
fn check_tx_version(
|
||||
decoy_info: &Option<DecoyInfo>,
|
||||
version: &TxVersion,
|
||||
hf: &HardFork,
|
||||
version: TxVersion,
|
||||
hf: HardFork,
|
||||
) -> Result<(), TransactionError> {
|
||||
if let Some(decoy_info) = decoy_info {
|
||||
let max = max_tx_version(hf);
|
||||
if version > &max {
|
||||
if version > max {
|
||||
return Err(TransactionError::TransactionVersionInvalid);
|
||||
}
|
||||
|
||||
let min = min_tx_version(hf);
|
||||
if version < &min && decoy_info.not_mixable == 0 {
|
||||
if version < min && decoy_info.not_mixable == 0 {
|
||||
return Err(TransactionError::TransactionVersionInvalid);
|
||||
}
|
||||
} else {
|
||||
// This will only happen for hard-fork 1 when only RingSignatures are allowed.
|
||||
if version != &TxVersion::RingSignatures {
|
||||
if version != TxVersion::RingSignatures {
|
||||
return Err(TransactionError::TransactionVersionInvalid);
|
||||
}
|
||||
}
|
||||
|
@ -541,8 +537,8 @@ fn check_tx_version(
|
|||
}
|
||||
|
||||
/// Returns the default maximum tx version for the given hard-fork.
|
||||
fn max_tx_version(hf: &HardFork) -> TxVersion {
|
||||
if hf <= &HardFork::V3 {
|
||||
fn max_tx_version(hf: HardFork) -> TxVersion {
|
||||
if hf <= HardFork::V3 {
|
||||
TxVersion::RingSignatures
|
||||
} else {
|
||||
TxVersion::RingCT
|
||||
|
@ -550,15 +546,15 @@ fn max_tx_version(hf: &HardFork) -> TxVersion {
|
|||
}
|
||||
|
||||
/// Returns the default minimum tx version for the given hard-fork.
|
||||
fn min_tx_version(hf: &HardFork) -> TxVersion {
|
||||
if hf >= &HardFork::V6 {
|
||||
fn min_tx_version(hf: HardFork) -> TxVersion {
|
||||
if hf >= HardFork::V6 {
|
||||
TxVersion::RingCT
|
||||
} else {
|
||||
TxVersion::RingSignatures
|
||||
}
|
||||
}
|
||||
|
||||
fn transaction_weight_limit(hf: &HardFork) -> usize {
|
||||
fn transaction_weight_limit(hf: HardFork) -> usize {
|
||||
penalty_free_zone(hf) / 2 - 600
|
||||
}
|
||||
|
||||
|
@ -575,14 +571,14 @@ pub fn check_transaction_semantic(
|
|||
tx_blob_size: usize,
|
||||
tx_weight: usize,
|
||||
tx_hash: &[u8; 32],
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
verifier: impl BatchVerifier,
|
||||
) -> Result<u64, TransactionError> {
|
||||
// <https://monero-book.cuprate.org/consensus_rules/transactions.html#transaction-size>
|
||||
if tx_blob_size > MAX_TX_BLOB_SIZE
|
||||
|| (hf >= &HardFork::V8 && tx_weight > transaction_weight_limit(hf))
|
||||
|| (hf >= HardFork::V8 && tx_weight > transaction_weight_limit(hf))
|
||||
{
|
||||
Err(TransactionError::TooBig)?;
|
||||
return Err(TransactionError::TooBig);
|
||||
}
|
||||
|
||||
let tx_version =
|
||||
|
@ -602,13 +598,13 @@ pub fn check_transaction_semantic(
|
|||
Transaction::V2 { proofs: None, .. } | Transaction::V1 { .. } => false,
|
||||
};
|
||||
|
||||
let outputs_sum = check_outputs_semantics(&tx.prefix().outputs, hf, &tx_version, bp_or_bpp)?;
|
||||
let outputs_sum = check_outputs_semantics(&tx.prefix().outputs, hf, tx_version, bp_or_bpp)?;
|
||||
let inputs_sum = check_inputs_semantics(&tx.prefix().inputs, hf)?;
|
||||
|
||||
let fee = match tx {
|
||||
Transaction::V1 { .. } => {
|
||||
if outputs_sum >= inputs_sum {
|
||||
Err(TransactionError::OutputsTooHigh)?;
|
||||
return Err(TransactionError::OutputsTooHigh);
|
||||
}
|
||||
inputs_sum - outputs_sum
|
||||
}
|
||||
|
@ -633,13 +629,12 @@ pub fn check_transaction_semantic(
|
|||
/// This function also does _not_ check for duplicate key-images: <https://monero-book.cuprate.org/consensus_rules/transactions/inputs.html#unique-key-image>.
|
||||
///
|
||||
/// `current_time_lock_timestamp` must be: <https://monero-book.cuprate.org/consensus_rules/transactions/unlock_time.html#getting-the-current-time>.
|
||||
|
||||
pub fn check_transaction_contextual(
|
||||
tx: &Transaction,
|
||||
tx_ring_members_info: &TxRingMembersInfo,
|
||||
current_chain_height: usize,
|
||||
current_time_lock_timestamp: u64,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> Result<(), TransactionError> {
|
||||
let tx_version =
|
||||
TxVersion::from_raw(tx.version()).ok_or(TransactionError::TransactionVersionInvalid)?;
|
||||
|
@ -650,7 +645,7 @@ pub fn check_transaction_contextual(
|
|||
current_chain_height,
|
||||
hf,
|
||||
)?;
|
||||
check_tx_version(&tx_ring_members_info.decoy_info, &tx_version, hf)?;
|
||||
check_tx_version(&tx_ring_members_info.decoy_info, tx_version, hf)?;
|
||||
|
||||
check_all_time_locks(
|
||||
&tx_ring_members_info.time_locked_outs,
|
||||
|
|
|
@ -26,7 +26,7 @@ pub fn get_absolute_offsets(relative_offsets: &[u64]) -> Result<Vec<u64>, Transa
|
|||
Ok(offsets)
|
||||
}
|
||||
|
||||
/// Inserts the output IDs that are needed to verify the transaction inputs into the provided HashMap.
|
||||
/// Inserts the output IDs that are needed to verify the transaction inputs into the provided `HashMap`.
|
||||
///
|
||||
/// This will error if the inputs are empty
|
||||
/// <https://cuprate.github.io/monero-book/consensus_rules/transactions.html#no-empty-inputs>
|
||||
|
@ -49,7 +49,7 @@ pub fn insert_ring_member_ids(
|
|||
.entry(amount.unwrap_or(0))
|
||||
.or_default()
|
||||
.extend(get_absolute_offsets(key_offsets)?),
|
||||
_ => return Err(TransactionError::IncorrectInputType),
|
||||
Input::Gen(_) => return Err(TransactionError::IncorrectInputType),
|
||||
}
|
||||
}
|
||||
Ok(())
|
||||
|
@ -60,7 +60,7 @@ pub fn insert_ring_member_ids(
|
|||
pub enum Rings {
|
||||
/// Legacy, pre-ringCT, rings.
|
||||
Legacy(Vec<Vec<EdwardsPoint>>),
|
||||
/// RingCT rings, (outkey, amount commitment).
|
||||
/// `RingCT` rings, (outkey, amount commitment).
|
||||
RingCT(Vec<Vec<[EdwardsPoint; 2]>>),
|
||||
}
|
||||
|
||||
|
@ -103,15 +103,15 @@ impl DecoyInfo {
|
|||
///
|
||||
/// So:
|
||||
///
|
||||
/// amount_outs_on_chain(inputs`[X]`) == outputs_with_amount`[X]`
|
||||
/// `amount_outs_on_chain(inputs[X]) == outputs_with_amount[X]`
|
||||
///
|
||||
/// Do not rely on this function to do consensus checks!
|
||||
///
|
||||
pub fn new(
|
||||
inputs: &[Input],
|
||||
outputs_with_amount: impl Fn(u64) -> usize,
|
||||
hf: &HardFork,
|
||||
) -> Result<DecoyInfo, TransactionError> {
|
||||
hf: HardFork,
|
||||
) -> Result<Self, TransactionError> {
|
||||
let mut min_decoys = usize::MAX;
|
||||
let mut max_decoys = usize::MIN;
|
||||
let mut mixable = 0;
|
||||
|
@ -119,7 +119,7 @@ impl DecoyInfo {
|
|||
|
||||
let minimum_decoys = minimum_decoys(hf);
|
||||
|
||||
for inp in inputs.iter() {
|
||||
for inp in inputs {
|
||||
match inp {
|
||||
Input::ToKey {
|
||||
key_offsets,
|
||||
|
@ -149,11 +149,11 @@ impl DecoyInfo {
|
|||
min_decoys = min(min_decoys, numb_decoys);
|
||||
max_decoys = max(max_decoys, numb_decoys);
|
||||
}
|
||||
_ => return Err(TransactionError::IncorrectInputType),
|
||||
Input::Gen(_) => return Err(TransactionError::IncorrectInputType),
|
||||
}
|
||||
}
|
||||
|
||||
Ok(DecoyInfo {
|
||||
Ok(Self {
|
||||
mixable,
|
||||
not_mixable,
|
||||
min_decoys,
|
||||
|
@ -166,7 +166,7 @@ impl DecoyInfo {
|
|||
/// **There are exceptions to this always being the minimum decoys**
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/inputs.html#default-minimum-decoys>
|
||||
pub(crate) fn minimum_decoys(hf: &HardFork) -> usize {
|
||||
pub(crate) fn minimum_decoys(hf: HardFork) -> usize {
|
||||
use HardFork as HF;
|
||||
match hf {
|
||||
HF::V1 => panic!("hard-fork 1 does not use these rules!"),
|
||||
|
|
|
@ -40,10 +40,10 @@ pub enum RingCTError {
|
|||
CLSAGError(#[from] ClsagError),
|
||||
}
|
||||
|
||||
/// Checks the RingCT type is allowed for the current hard fork.
|
||||
/// Checks the `RingCT` type is allowed for the current hard fork.
|
||||
///
|
||||
/// <https://monero-book.cuprate.org/consensus_rules/ring_ct.html#type>
|
||||
fn check_rct_type(ty: &RctType, hf: HardFork, tx_hash: &[u8; 32]) -> Result<(), RingCTError> {
|
||||
fn check_rct_type(ty: RctType, hf: HardFork, tx_hash: &[u8; 32]) -> Result<(), RingCTError> {
|
||||
use HardFork as F;
|
||||
use RctType as T;
|
||||
|
||||
|
@ -125,11 +125,11 @@ pub(crate) fn ring_ct_semantic_checks(
|
|||
proofs: &RctProofs,
|
||||
tx_hash: &[u8; 32],
|
||||
verifier: impl BatchVerifier,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> Result<(), RingCTError> {
|
||||
let rct_type = proofs.rct_type();
|
||||
|
||||
check_rct_type(&rct_type, *hf, tx_hash)?;
|
||||
check_rct_type(rct_type, hf, tx_hash)?;
|
||||
check_output_range_proofs(proofs, verifier)?;
|
||||
|
||||
if rct_type != RctType::AggregateMlsagBorromean {
|
||||
|
@ -154,7 +154,7 @@ pub(crate) fn check_input_signatures(
|
|||
};
|
||||
|
||||
if rings.is_empty() {
|
||||
Err(RingCTError::RingInvalid)?;
|
||||
return Err(RingCTError::RingInvalid);
|
||||
}
|
||||
|
||||
let pseudo_outs = match &proofs.prunable {
|
||||
|
@ -222,20 +222,20 @@ mod tests {
|
|||
#[test]
|
||||
fn grandfathered_bulletproofs2() {
|
||||
assert!(check_rct_type(
|
||||
&RctType::MlsagBulletproofsCompactAmount,
|
||||
RctType::MlsagBulletproofsCompactAmount,
|
||||
HardFork::V14,
|
||||
&[0; 32]
|
||||
)
|
||||
.is_err());
|
||||
|
||||
assert!(check_rct_type(
|
||||
&RctType::MlsagBulletproofsCompactAmount,
|
||||
RctType::MlsagBulletproofsCompactAmount,
|
||||
HardFork::V14,
|
||||
&GRANDFATHERED_TRANSACTIONS[0]
|
||||
)
|
||||
.is_ok());
|
||||
assert!(check_rct_type(
|
||||
&RctType::MlsagBulletproofsCompactAmount,
|
||||
RctType::MlsagBulletproofsCompactAmount,
|
||||
HardFork::V14,
|
||||
&GRANDFATHERED_TRANSACTIONS[1]
|
||||
)
|
||||
|
|
|
@ -17,7 +17,7 @@ use crate::try_par_iter;
|
|||
/// Verifies the ring signature.
|
||||
///
|
||||
/// ref: <https://monero-book.cuprate.org/consensus_rules/transactions/ring_signatures.html>
|
||||
pub fn check_input_signatures(
|
||||
pub(crate) fn check_input_signatures(
|
||||
inputs: &[Input],
|
||||
signatures: &[RingSignature],
|
||||
rings: &Rings,
|
||||
|
@ -45,7 +45,7 @@ pub fn check_input_signatures(
|
|||
Ok(())
|
||||
})?;
|
||||
}
|
||||
_ => panic!("tried to verify v1 tx with a non v1 ring"),
|
||||
Rings::RingCT(_) => panic!("tried to verify v1 tx with a non v1 ring"),
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
|
|
|
@ -17,13 +17,13 @@ use crate::decomposed_amount::DECOMPOSED_AMOUNTS;
|
|||
|
||||
#[test]
|
||||
fn test_check_output_amount_v1() {
|
||||
for amount in DECOMPOSED_AMOUNTS.iter() {
|
||||
assert!(check_output_amount_v1(*amount, &HardFork::V2).is_ok())
|
||||
for amount in &DECOMPOSED_AMOUNTS {
|
||||
assert!(check_output_amount_v1(*amount, HardFork::V2).is_ok());
|
||||
}
|
||||
|
||||
proptest!(|(amount in any::<u64>().prop_filter("value_decomposed", |val| !is_decomposed_amount(val)))| {
|
||||
prop_assert!(check_output_amount_v1(amount, &HardFork::V2).is_err());
|
||||
prop_assert!(check_output_amount_v1(amount, &HardFork::V1).is_ok())
|
||||
prop_assert!(check_output_amount_v1(amount, HardFork::V2).is_err());
|
||||
prop_assert!(check_output_amount_v1(amount, HardFork::V1).is_ok());
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -42,10 +42,10 @@ fn test_sum_outputs() {
|
|||
|
||||
let outs = [output_10, outputs_20];
|
||||
|
||||
let sum = sum_outputs(&outs, &HardFork::V16, &TxVersion::RingSignatures).unwrap();
|
||||
let sum = sum_outputs(&outs, HardFork::V16, TxVersion::RingSignatures).unwrap();
|
||||
assert_eq!(sum, 30);
|
||||
|
||||
assert!(sum_outputs(&outs, &HardFork::V16, &TxVersion::RingCT).is_err())
|
||||
assert!(sum_outputs(&outs, HardFork::V16, TxVersion::RingCT).is_err());
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
@ -53,50 +53,50 @@ fn test_decoy_info() {
|
|||
let decoy_info = DecoyInfo {
|
||||
mixable: 0,
|
||||
not_mixable: 0,
|
||||
min_decoys: minimum_decoys(&HardFork::V8),
|
||||
max_decoys: minimum_decoys(&HardFork::V8) + 1,
|
||||
min_decoys: minimum_decoys(HardFork::V8),
|
||||
max_decoys: minimum_decoys(HardFork::V8) + 1,
|
||||
};
|
||||
|
||||
assert!(check_decoy_info(&decoy_info, &HardFork::V8).is_ok());
|
||||
assert!(check_decoy_info(&decoy_info, &HardFork::V16).is_err());
|
||||
assert!(check_decoy_info(&decoy_info, HardFork::V8).is_ok());
|
||||
assert!(check_decoy_info(&decoy_info, HardFork::V16).is_err());
|
||||
|
||||
let mut decoy_info = DecoyInfo {
|
||||
mixable: 0,
|
||||
not_mixable: 0,
|
||||
min_decoys: minimum_decoys(&HardFork::V8) - 1,
|
||||
max_decoys: minimum_decoys(&HardFork::V8) + 1,
|
||||
min_decoys: minimum_decoys(HardFork::V8) - 1,
|
||||
max_decoys: minimum_decoys(HardFork::V8) + 1,
|
||||
};
|
||||
|
||||
assert!(check_decoy_info(&decoy_info, &HardFork::V8).is_err());
|
||||
assert!(check_decoy_info(&decoy_info, HardFork::V8).is_err());
|
||||
|
||||
decoy_info.not_mixable = 1;
|
||||
assert!(check_decoy_info(&decoy_info, &HardFork::V8).is_ok());
|
||||
assert!(check_decoy_info(&decoy_info, HardFork::V8).is_ok());
|
||||
|
||||
decoy_info.mixable = 2;
|
||||
assert!(check_decoy_info(&decoy_info, &HardFork::V8).is_err());
|
||||
assert!(check_decoy_info(&decoy_info, HardFork::V8).is_err());
|
||||
|
||||
let mut decoy_info = DecoyInfo {
|
||||
mixable: 0,
|
||||
not_mixable: 0,
|
||||
min_decoys: minimum_decoys(&HardFork::V12),
|
||||
max_decoys: minimum_decoys(&HardFork::V12) + 1,
|
||||
min_decoys: minimum_decoys(HardFork::V12),
|
||||
max_decoys: minimum_decoys(HardFork::V12) + 1,
|
||||
};
|
||||
|
||||
assert!(check_decoy_info(&decoy_info, &HardFork::V12).is_err());
|
||||
assert!(check_decoy_info(&decoy_info, HardFork::V12).is_err());
|
||||
|
||||
decoy_info.max_decoys = decoy_info.min_decoys;
|
||||
assert!(check_decoy_info(&decoy_info, &HardFork::V12).is_ok());
|
||||
assert!(check_decoy_info(&decoy_info, HardFork::V12).is_ok());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_torsion_ki() {
|
||||
for &key_image in EIGHT_TORSION[1..].iter() {
|
||||
for &key_image in &EIGHT_TORSION[1..] {
|
||||
assert!(check_key_images(&Input::ToKey {
|
||||
key_image,
|
||||
amount: None,
|
||||
key_offsets: vec![],
|
||||
})
|
||||
.is_err())
|
||||
.is_err());
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -110,7 +110,7 @@ prop_compose! {
|
|||
prop_compose! {
|
||||
/// Returns a valid torsioned point.
|
||||
fn random_torsioned_point()(point in random_point(), torsion in 1..8_usize ) -> EdwardsPoint {
|
||||
point + curve25519_dalek::constants::EIGHT_TORSION[torsion]
|
||||
point + EIGHT_TORSION[torsion]
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -161,7 +161,7 @@ prop_compose! {
|
|||
/// Returns a [`Timelock`] that is locked given a height and time.
|
||||
fn locked_timelock(height: u64, time_for_time_lock: u64)(
|
||||
timebased in any::<bool>(),
|
||||
lock_height in (height+1)..MAX_BLOCK_HEIGHT+1,
|
||||
lock_height in (height+1)..=MAX_BLOCK_HEIGHT,
|
||||
time_for_time_lock in (time_for_time_lock+121)..,
|
||||
) -> Timelock {
|
||||
if timebased || lock_height > MAX_BLOCK_HEIGHT {
|
||||
|
@ -176,7 +176,7 @@ prop_compose! {
|
|||
/// Returns a [`Timelock`] that is unlocked given a height and time.
|
||||
fn unlocked_timelock(height: u64, time_for_time_lock: u64)(
|
||||
ty in 0..3,
|
||||
lock_height in 0..(height+1),
|
||||
lock_height in 0..=height,
|
||||
time_for_time_lock in 0..(time_for_time_lock+121),
|
||||
) -> Timelock {
|
||||
match ty {
|
||||
|
@ -204,33 +204,33 @@ proptest! {
|
|||
hf_no_view_tags in hf_in_range(1..14),
|
||||
hf_view_tags in hf_in_range(16..17),
|
||||
) {
|
||||
prop_assert!(check_output_types(&view_tag_outs, &hf_view_tags).is_ok());
|
||||
prop_assert!(check_output_types(&view_tag_outs, &hf_no_view_tags).is_err());
|
||||
prop_assert!(check_output_types(&view_tag_outs, hf_view_tags).is_ok());
|
||||
prop_assert!(check_output_types(&view_tag_outs, hf_no_view_tags).is_err());
|
||||
|
||||
|
||||
prop_assert!(check_output_types(&non_view_tag_outs, &hf_no_view_tags).is_ok());
|
||||
prop_assert!(check_output_types(&non_view_tag_outs, &hf_view_tags).is_err());
|
||||
prop_assert!(check_output_types(&non_view_tag_outs, hf_no_view_tags).is_ok());
|
||||
prop_assert!(check_output_types(&non_view_tag_outs, hf_view_tags).is_err());
|
||||
|
||||
prop_assert!(check_output_types(&non_view_tag_outs, &HardFork::V15).is_ok());
|
||||
prop_assert!(check_output_types(&view_tag_outs, &HardFork::V15).is_ok());
|
||||
prop_assert!(check_output_types(&non_view_tag_outs, HardFork::V15).is_ok());
|
||||
prop_assert!(check_output_types(&view_tag_outs, HardFork::V15).is_ok());
|
||||
view_tag_outs.append(&mut non_view_tag_outs);
|
||||
prop_assert!(check_output_types(&view_tag_outs, &HardFork::V15).is_err());
|
||||
prop_assert!(check_output_types(&view_tag_outs, HardFork::V15).is_err());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_valid_number_of_outputs(valid_numb_outs in 2..17_usize) {
|
||||
prop_assert!(check_number_of_outputs(valid_numb_outs, &HardFork::V16, &TxVersion::RingCT, true).is_ok());
|
||||
prop_assert!(check_number_of_outputs(valid_numb_outs, HardFork::V16, TxVersion::RingCT, true).is_ok());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_invalid_number_of_outputs(numb_outs in 17..usize::MAX) {
|
||||
prop_assert!(check_number_of_outputs(numb_outs, &HardFork::V16, &TxVersion::RingCT, true).is_err());
|
||||
prop_assert!(check_number_of_outputs(numb_outs, HardFork::V16, TxVersion::RingCT, true).is_err());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_check_output_amount_v2(amt in 1..u64::MAX) {
|
||||
prop_assert!(check_output_amount_v2(amt).is_err());
|
||||
prop_assert!(check_output_amount_v2(0).is_ok())
|
||||
prop_assert!(check_output_amount_v2(0).is_ok());
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
@ -242,9 +242,9 @@ proptest! {
|
|||
|
||||
#[test]
|
||||
fn test_timestamp_time_lock(timestamp in MAX_BLOCK_HEIGHT+1..u64::MAX) {
|
||||
prop_assert!(check_timestamp_time_lock(timestamp, timestamp - 120, &HardFork::V16));
|
||||
prop_assert!(!check_timestamp_time_lock(timestamp, timestamp - 121, &HardFork::V16));
|
||||
prop_assert!(check_timestamp_time_lock(timestamp, timestamp, &HardFork::V16));
|
||||
prop_assert!(check_timestamp_time_lock(timestamp, timestamp - 120, HardFork::V16));
|
||||
prop_assert!(!check_timestamp_time_lock(timestamp, timestamp - 121, HardFork::V16));
|
||||
prop_assert!(check_timestamp_time_lock(timestamp, timestamp, HardFork::V16));
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
@ -252,11 +252,11 @@ proptest! {
|
|||
mut locked_locks in vec(locked_timelock(5_000, 100_000_000), 1..50),
|
||||
mut unlocked_locks in vec(unlocked_timelock(5_000, 100_000_000), 1..50)
|
||||
) {
|
||||
assert!(check_all_time_locks(&locked_locks, 5_000, 100_000_000, &HardFork::V16).is_err());
|
||||
assert!(check_all_time_locks(&unlocked_locks, 5_000, 100_000_000, &HardFork::V16).is_ok());
|
||||
assert!(check_all_time_locks(&locked_locks, 5_000, 100_000_000, HardFork::V16).is_err());
|
||||
assert!(check_all_time_locks(&unlocked_locks, 5_000, 100_000_000, HardFork::V16).is_ok());
|
||||
|
||||
unlocked_locks.append(&mut locked_locks);
|
||||
assert!(check_all_time_locks(&unlocked_locks, 5_000, 100_000_000, &HardFork::V16).is_err());
|
||||
assert!(check_all_time_locks(&unlocked_locks, 5_000, 100_000_000, HardFork::V16).is_err());
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
use std::{cell::RefCell, ops::DerefMut};
|
||||
use std::cell::RefCell;
|
||||
|
||||
use monero_serai::ringct::bulletproofs::BatchVerifier as InternalBatchVerifier;
|
||||
use rayon::prelude::*;
|
||||
|
@ -13,8 +13,8 @@ pub struct MultiThreadedBatchVerifier {
|
|||
|
||||
impl MultiThreadedBatchVerifier {
|
||||
/// Create a new multithreaded batch verifier,
|
||||
pub fn new(numb_threads: usize) -> MultiThreadedBatchVerifier {
|
||||
MultiThreadedBatchVerifier {
|
||||
pub fn new(numb_threads: usize) -> Self {
|
||||
Self {
|
||||
internal: ThreadLocal::with_capacity(numb_threads),
|
||||
}
|
||||
}
|
||||
|
@ -42,6 +42,6 @@ impl BatchVerifier for &'_ MultiThreadedBatchVerifier {
|
|||
.get_or(|| RefCell::new(InternalBatchVerifier::new()))
|
||||
.borrow_mut();
|
||||
|
||||
stmt(verifier.deref_mut())
|
||||
stmt(&mut verifier)
|
||||
}
|
||||
}
|
||||
|
|
|
@ -72,17 +72,17 @@ impl PreparedBlockExPow {
|
|||
/// This errors if either the `block`'s:
|
||||
/// - Hard-fork values are invalid
|
||||
/// - Miner transaction is missing a miner input
|
||||
pub fn new(block: Block) -> Result<PreparedBlockExPow, ConsensusError> {
|
||||
pub fn new(block: Block) -> Result<Self, ConsensusError> {
|
||||
let (hf_version, hf_vote) = HardFork::from_block_header(&block.header)
|
||||
.map_err(|_| BlockError::HardForkError(HardForkError::HardForkUnknown))?;
|
||||
|
||||
let Some(Input::Gen(height)) = block.miner_transaction.prefix().inputs.first() else {
|
||||
Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
return Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
MinerTxError::InputNotOfTypeGen,
|
||||
)))?
|
||||
)));
|
||||
};
|
||||
|
||||
Ok(PreparedBlockExPow {
|
||||
Ok(Self {
|
||||
block_blob: block.serialize(),
|
||||
hf_vote,
|
||||
hf_version,
|
||||
|
@ -123,20 +123,17 @@ impl PreparedBlock {
|
|||
///
|
||||
/// The randomX VM must be Some if RX is needed or this will panic.
|
||||
/// The randomX VM must also be initialised with the correct seed.
|
||||
fn new<R: RandomX>(
|
||||
block: Block,
|
||||
randomx_vm: Option<&R>,
|
||||
) -> Result<PreparedBlock, ConsensusError> {
|
||||
fn new<R: RandomX>(block: Block, randomx_vm: Option<&R>) -> Result<Self, ConsensusError> {
|
||||
let (hf_version, hf_vote) = HardFork::from_block_header(&block.header)
|
||||
.map_err(|_| BlockError::HardForkError(HardForkError::HardForkUnknown))?;
|
||||
|
||||
let [Input::Gen(height)] = &block.miner_transaction.prefix().inputs[..] else {
|
||||
Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
return Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
MinerTxError::InputNotOfTypeGen,
|
||||
)))?
|
||||
)));
|
||||
};
|
||||
|
||||
Ok(PreparedBlock {
|
||||
Ok(Self {
|
||||
block_blob: block.serialize(),
|
||||
hf_vote,
|
||||
hf_version,
|
||||
|
@ -156,17 +153,17 @@ impl PreparedBlock {
|
|||
|
||||
/// Creates a new [`PreparedBlock`] from a [`PreparedBlockExPow`].
|
||||
///
|
||||
/// This function will give an invalid PoW hash if `randomx_vm` is not initialised
|
||||
/// This function will give an invalid proof-of-work hash if `randomx_vm` is not initialised
|
||||
/// with the correct seed.
|
||||
///
|
||||
/// # Panics
|
||||
/// This function will panic if `randomx_vm` is
|
||||
/// [`None`] even though RandomX is needed.
|
||||
/// [`None`] even though `RandomX` is needed.
|
||||
fn new_prepped<R: RandomX>(
|
||||
block: PreparedBlockExPow,
|
||||
randomx_vm: Option<&R>,
|
||||
) -> Result<PreparedBlock, ConsensusError> {
|
||||
Ok(PreparedBlock {
|
||||
) -> Result<Self, ConsensusError> {
|
||||
Ok(Self {
|
||||
block_blob: block.block_blob,
|
||||
hf_vote: block.hf_vote,
|
||||
hf_version: block.hf_version,
|
||||
|
@ -218,7 +215,6 @@ pub enum VerifyBlockRequest {
|
|||
}
|
||||
|
||||
/// A response from a verify block request.
|
||||
#[allow(clippy::large_enum_variant)] // The largest variant is most common ([`MainChain`])
|
||||
pub enum VerifyBlockResponse {
|
||||
/// This block is valid.
|
||||
MainChain(VerifiedBlockInformation),
|
||||
|
@ -254,12 +250,8 @@ where
|
|||
D::Future: Send + 'static,
|
||||
{
|
||||
/// Creates a new block verifier.
|
||||
pub(crate) fn new(
|
||||
context_svc: C,
|
||||
tx_verifier_svc: TxV,
|
||||
database: D,
|
||||
) -> BlockVerifierService<C, TxV, D> {
|
||||
BlockVerifierService {
|
||||
pub(crate) const fn new(context_svc: C, tx_verifier_svc: TxV, database: D) -> Self {
|
||||
Self {
|
||||
context_svc,
|
||||
tx_verifier_svc,
|
||||
_database: database,
|
||||
|
|
|
@ -36,8 +36,8 @@ use crate::{
|
|||
///
|
||||
/// Returns [`AltBlockInformation`], which contains the cumulative difficulty of the alt chain.
|
||||
///
|
||||
/// This function only checks the block's PoW and its weight.
|
||||
pub async fn sanity_check_alt_block<C>(
|
||||
/// This function only checks the block's proof-of-work and its weight.
|
||||
pub(crate) async fn sanity_check_alt_block<C>(
|
||||
block: Block,
|
||||
txs: HashMap<[u8; 32], TransactionVerificationData>,
|
||||
mut context_svc: C,
|
||||
|
@ -66,15 +66,17 @@ where
|
|||
|
||||
// Check if the block's miner input is formed correctly.
|
||||
let [Input::Gen(height)] = &block.miner_transaction.prefix().inputs[..] else {
|
||||
Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
return Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
MinerTxError::InputNotOfTypeGen,
|
||||
)))?
|
||||
))
|
||||
.into());
|
||||
};
|
||||
|
||||
if *height != alt_context_cache.chain_height {
|
||||
Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
return Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
MinerTxError::InputsHeightIncorrect,
|
||||
)))?
|
||||
))
|
||||
.into());
|
||||
}
|
||||
|
||||
// prep the alt block.
|
||||
|
@ -103,10 +105,10 @@ where
|
|||
if let Some(median_timestamp) =
|
||||
difficulty_cache.median_timestamp(u64_to_usize(BLOCKCHAIN_TIMESTAMP_CHECK_WINDOW))
|
||||
{
|
||||
check_timestamp(&prepped_block.block, median_timestamp).map_err(ConsensusError::Block)?
|
||||
check_timestamp(&prepped_block.block, median_timestamp).map_err(ConsensusError::Block)?;
|
||||
};
|
||||
|
||||
let next_difficulty = difficulty_cache.next_difficulty(&prepped_block.hf_version);
|
||||
let next_difficulty = difficulty_cache.next_difficulty(prepped_block.hf_version);
|
||||
// make sure the block's PoW is valid for this difficulty.
|
||||
check_block_pow(&prepped_block.pow_hash, next_difficulty).map_err(ConsensusError::Block)?;
|
||||
|
||||
|
@ -127,12 +129,12 @@ where
|
|||
// Check the block weight is below the limit.
|
||||
check_block_weight(
|
||||
block_weight,
|
||||
alt_weight_cache.median_for_block_reward(&prepped_block.hf_version),
|
||||
alt_weight_cache.median_for_block_reward(prepped_block.hf_version),
|
||||
)
|
||||
.map_err(ConsensusError::Block)?;
|
||||
|
||||
let long_term_weight = weight::calculate_block_long_term_weight(
|
||||
&prepped_block.hf_version,
|
||||
prepped_block.hf_version,
|
||||
block_weight,
|
||||
alt_weight_cache.median_long_term_weight(),
|
||||
);
|
||||
|
@ -232,9 +234,9 @@ where
|
|||
}
|
||||
};
|
||||
|
||||
Ok(Some(
|
||||
alt_chain_context.cached_rx_vm.insert(cached_vm).1.clone(),
|
||||
))
|
||||
Ok(Some(Arc::clone(
|
||||
&alt_chain_context.cached_rx_vm.insert(cached_vm).1,
|
||||
)))
|
||||
}
|
||||
|
||||
/// Returns the [`DifficultyCache`] for the alt chain.
|
||||
|
|
|
@ -68,16 +68,17 @@ where
|
|||
|
||||
// Make sure no blocks in the batch have a higher hard fork than the last block.
|
||||
if block_0.hf_version > top_hf_in_batch {
|
||||
Err(ConsensusError::Block(BlockError::HardForkError(
|
||||
return Err(ConsensusError::Block(BlockError::HardForkError(
|
||||
HardForkError::VersionIncorrect,
|
||||
)))?;
|
||||
))
|
||||
.into());
|
||||
}
|
||||
|
||||
if block_0.block_hash != block_1.block.header.previous
|
||||
|| block_0.height != block_1.height - 1
|
||||
{
|
||||
tracing::debug!("Blocks do not follow each other, verification failed.");
|
||||
Err(ConsensusError::Block(BlockError::PreviousIDIncorrect))?;
|
||||
return Err(ConsensusError::Block(BlockError::PreviousIDIncorrect).into());
|
||||
}
|
||||
|
||||
// Cache any potential RX VM seeds as we may need them for future blocks in the batch.
|
||||
|
@ -85,7 +86,7 @@ where
|
|||
new_rx_vm = Some((block_0.height, block_0.block_hash));
|
||||
}
|
||||
|
||||
timestamps_hfs.push((block_0.block.header.timestamp, block_0.hf_version))
|
||||
timestamps_hfs.push((block_0.block.header.timestamp, block_0.hf_version));
|
||||
}
|
||||
|
||||
// Get the current blockchain context.
|
||||
|
@ -117,15 +118,16 @@ where
|
|||
if context.chain_height != blocks[0].height {
|
||||
tracing::debug!("Blocks do not follow main chain, verification failed.");
|
||||
|
||||
Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
return Err(ConsensusError::Block(BlockError::MinerTxError(
|
||||
MinerTxError::InputsHeightIncorrect,
|
||||
)))?;
|
||||
))
|
||||
.into());
|
||||
}
|
||||
|
||||
if context.top_hash != blocks[0].block.header.previous {
|
||||
tracing::debug!("Blocks do not follow main chain, verification failed.");
|
||||
|
||||
Err(ConsensusError::Block(BlockError::PreviousIDIncorrect))?;
|
||||
return Err(ConsensusError::Block(BlockError::PreviousIDIncorrect).into());
|
||||
}
|
||||
|
||||
let mut rx_vms = if top_hf_in_batch < HardFork::V12 {
|
||||
|
@ -156,7 +158,7 @@ where
|
|||
context_svc
|
||||
.oneshot(BlockChainContextRequest::NewRXVM((
|
||||
new_vm_seed,
|
||||
new_vm.clone(),
|
||||
Arc::clone(&new_vm),
|
||||
)))
|
||||
.await?;
|
||||
|
||||
|
|
|
@ -56,8 +56,8 @@ pub struct ContextConfig {
|
|||
|
||||
impl ContextConfig {
|
||||
/// Get the config for main-net.
|
||||
pub fn main_net() -> ContextConfig {
|
||||
ContextConfig {
|
||||
pub const fn main_net() -> Self {
|
||||
Self {
|
||||
hard_fork_cfg: HardForkConfig::main_net(),
|
||||
difficulty_cfg: DifficultyCacheConfig::main_net(),
|
||||
weights_config: BlockWeightsCacheConfig::main_net(),
|
||||
|
@ -65,8 +65,8 @@ impl ContextConfig {
|
|||
}
|
||||
|
||||
/// Get the config for stage-net.
|
||||
pub fn stage_net() -> ContextConfig {
|
||||
ContextConfig {
|
||||
pub const fn stage_net() -> Self {
|
||||
Self {
|
||||
hard_fork_cfg: HardForkConfig::stage_net(),
|
||||
// These 2 have the same config as main-net.
|
||||
difficulty_cfg: DifficultyCacheConfig::main_net(),
|
||||
|
@ -75,8 +75,8 @@ impl ContextConfig {
|
|||
}
|
||||
|
||||
/// Get the config for test-net.
|
||||
pub fn test_net() -> ContextConfig {
|
||||
ContextConfig {
|
||||
pub const fn test_net() -> Self {
|
||||
Self {
|
||||
hard_fork_cfg: HardForkConfig::test_net(),
|
||||
// These 2 have the same config as main-net.
|
||||
difficulty_cfg: DifficultyCacheConfig::main_net(),
|
||||
|
@ -155,7 +155,7 @@ impl RawBlockChainContext {
|
|||
/// Returns the next blocks long term weight from its block weight.
|
||||
pub fn next_block_long_term_weight(&self, block_weight: usize) -> usize {
|
||||
weight::calculate_block_long_term_weight(
|
||||
&self.current_hf,
|
||||
self.current_hf,
|
||||
block_weight,
|
||||
self.median_long_term_weight,
|
||||
)
|
||||
|
@ -191,7 +191,7 @@ impl BlockChainContext {
|
|||
}
|
||||
|
||||
/// Returns the blockchain context without checking the validity token.
|
||||
pub fn unchecked_blockchain_context(&self) -> &RawBlockChainContext {
|
||||
pub const fn unchecked_blockchain_context(&self) -> &RawBlockChainContext {
|
||||
&self.raw
|
||||
}
|
||||
}
|
||||
|
@ -222,7 +222,7 @@ pub struct NewBlockData {
|
|||
pub enum BlockChainContextRequest {
|
||||
/// Get the current blockchain context.
|
||||
GetContext,
|
||||
/// Gets the current RandomX VM.
|
||||
/// Gets the current `RandomX` VM.
|
||||
GetCurrentRxVm,
|
||||
/// Get the next difficulties for these blocks.
|
||||
///
|
||||
|
@ -288,7 +288,7 @@ pub enum BlockChainContextRequest {
|
|||
/// This variant is private and is not callable from outside this crate, the block verifier service will
|
||||
/// handle getting the randomX VM of an alt chain.
|
||||
AltChainRxVM {
|
||||
/// The height the RandomX VM is needed for.
|
||||
/// The height the `RandomX` VM is needed for.
|
||||
height: usize,
|
||||
/// The chain to look in for the seed.
|
||||
chain: Chain,
|
||||
|
@ -312,7 +312,7 @@ pub enum BlockChainContextRequest {
|
|||
pub enum BlockChainContextResponse {
|
||||
/// Blockchain context response.
|
||||
Context(BlockChainContext),
|
||||
/// A map of seed height to RandomX VMs.
|
||||
/// A map of seed height to `RandomX` VMs.
|
||||
RxVms(HashMap<usize, Arc<RandomXVm>>),
|
||||
/// A list of difficulties.
|
||||
BatchDifficulties(Vec<u128>),
|
||||
|
|
|
@ -68,29 +68,33 @@ impl AltChainContextCache {
|
|||
}
|
||||
|
||||
/// A map of top IDs to alt chains.
|
||||
pub struct AltChainMap {
|
||||
pub(crate) struct AltChainMap {
|
||||
alt_cache_map: HashMap<[u8; 32], Box<AltChainContextCache>>,
|
||||
}
|
||||
|
||||
impl AltChainMap {
|
||||
pub fn new() -> Self {
|
||||
pub(crate) fn new() -> Self {
|
||||
Self {
|
||||
alt_cache_map: HashMap::new(),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn clear(&mut self) {
|
||||
pub(crate) fn clear(&mut self) {
|
||||
self.alt_cache_map.clear();
|
||||
}
|
||||
|
||||
/// Add an alt chain cache to the map.
|
||||
pub fn add_alt_cache(&mut self, prev_id: [u8; 32], alt_cache: Box<AltChainContextCache>) {
|
||||
pub(crate) fn add_alt_cache(
|
||||
&mut self,
|
||||
prev_id: [u8; 32],
|
||||
alt_cache: Box<AltChainContextCache>,
|
||||
) {
|
||||
self.alt_cache_map.insert(prev_id, alt_cache);
|
||||
}
|
||||
|
||||
/// Attempts to take an [`AltChainContextCache`] from the map, returning [`None`] if no cache is
|
||||
/// present.
|
||||
pub async fn get_alt_chain_context<D: Database>(
|
||||
pub(crate) async fn get_alt_chain_context<D: Database>(
|
||||
&mut self,
|
||||
prev_id: [u8; 32],
|
||||
database: D,
|
||||
|
@ -109,7 +113,7 @@ impl AltChainMap {
|
|||
|
||||
let Some((parent_chain, top_height)) = res else {
|
||||
// Couldn't find prev_id
|
||||
Err(ConsensusError::Block(BlockError::PreviousIDIncorrect))?
|
||||
return Err(ConsensusError::Block(BlockError::PreviousIDIncorrect).into());
|
||||
};
|
||||
|
||||
Ok(Box::new(AltChainContextCache {
|
||||
|
@ -125,7 +129,7 @@ impl AltChainMap {
|
|||
}
|
||||
|
||||
/// Builds a [`DifficultyCache`] for an alt chain.
|
||||
pub async fn get_alt_chain_difficulty_cache<D: Database + Clone>(
|
||||
pub(crate) async fn get_alt_chain_difficulty_cache<D: Database + Clone>(
|
||||
prev_id: [u8; 32],
|
||||
main_chain_difficulty_cache: &DifficultyCache,
|
||||
mut database: D,
|
||||
|
@ -142,7 +146,7 @@ pub async fn get_alt_chain_difficulty_cache<D: Database + Clone>(
|
|||
|
||||
let Some((chain, top_height)) = res else {
|
||||
// Can't find prev_id
|
||||
Err(ConsensusError::Block(BlockError::PreviousIDIncorrect))?
|
||||
return Err(ConsensusError::Block(BlockError::PreviousIDIncorrect).into());
|
||||
};
|
||||
|
||||
Ok(match chain {
|
||||
|
@ -172,7 +176,7 @@ pub async fn get_alt_chain_difficulty_cache<D: Database + Clone>(
|
|||
}
|
||||
|
||||
/// Builds a [`BlockWeightsCache`] for an alt chain.
|
||||
pub async fn get_alt_chain_weight_cache<D: Database + Clone>(
|
||||
pub(crate) async fn get_alt_chain_weight_cache<D: Database + Clone>(
|
||||
prev_id: [u8; 32],
|
||||
main_chain_weight_cache: &BlockWeightsCache,
|
||||
mut database: D,
|
||||
|
@ -189,7 +193,7 @@ pub async fn get_alt_chain_weight_cache<D: Database + Clone>(
|
|||
|
||||
let Some((chain, top_height)) = res else {
|
||||
// Can't find prev_id
|
||||
Err(ConsensusError::Block(BlockError::PreviousIDIncorrect))?
|
||||
return Err(ConsensusError::Block(BlockError::PreviousIDIncorrect).into());
|
||||
};
|
||||
|
||||
Ok(match chain {
|
||||
|
|
|
@ -43,24 +43,24 @@ impl DifficultyCacheConfig {
|
|||
///
|
||||
/// # Notes
|
||||
/// You probably do not need this, use [`DifficultyCacheConfig::main_net`] instead.
|
||||
pub const fn new(window: usize, cut: usize, lag: usize) -> DifficultyCacheConfig {
|
||||
DifficultyCacheConfig { window, cut, lag }
|
||||
pub const fn new(window: usize, cut: usize, lag: usize) -> Self {
|
||||
Self { window, cut, lag }
|
||||
}
|
||||
|
||||
/// Returns the total amount of blocks we need to track to calculate difficulty
|
||||
pub fn total_block_count(&self) -> usize {
|
||||
pub const fn total_block_count(&self) -> usize {
|
||||
self.window + self.lag
|
||||
}
|
||||
|
||||
/// The amount of blocks we account for after removing the outliers.
|
||||
pub fn accounted_window_len(&self) -> usize {
|
||||
pub const fn accounted_window_len(&self) -> usize {
|
||||
self.window - 2 * self.cut
|
||||
}
|
||||
|
||||
/// Returns the config needed for [`Mainnet`](cuprate_helper::network::Network::Mainnet). This is also the
|
||||
/// config for all other current networks.
|
||||
pub const fn main_net() -> DifficultyCacheConfig {
|
||||
DifficultyCacheConfig {
|
||||
pub const fn main_net() -> Self {
|
||||
Self {
|
||||
window: DIFFICULTY_WINDOW,
|
||||
cut: DIFFICULTY_CUT,
|
||||
lag: DIFFICULTY_LAG,
|
||||
|
@ -112,7 +112,7 @@ impl DifficultyCache {
|
|||
timestamps.len()
|
||||
);
|
||||
|
||||
let diff = DifficultyCache {
|
||||
let diff = Self {
|
||||
timestamps,
|
||||
cumulative_difficulties,
|
||||
last_accounted_height: chain_height - 1,
|
||||
|
@ -203,8 +203,8 @@ impl DifficultyCache {
|
|||
|
||||
/// Returns the required difficulty for the next block.
|
||||
///
|
||||
/// See: https://cuprate.github.io/monero-book/consensus_rules/blocks/difficulty.html#calculating-difficulty
|
||||
pub fn next_difficulty(&self, hf: &HardFork) -> u128 {
|
||||
/// See: <https://cuprate.github.io/monero-book/consensus_rules/blocks/difficulty.html#calculating-difficulty>
|
||||
pub fn next_difficulty(&self, hf: HardFork) -> u128 {
|
||||
next_difficulty(
|
||||
&self.config,
|
||||
&self.timestamps,
|
||||
|
@ -223,7 +223,7 @@ impl DifficultyCache {
|
|||
pub fn next_difficulties(
|
||||
&self,
|
||||
blocks: Vec<(u64, HardFork)>,
|
||||
current_hf: &HardFork,
|
||||
current_hf: HardFork,
|
||||
) -> Vec<u128> {
|
||||
let mut timestamps = self.timestamps.clone();
|
||||
let mut cumulative_difficulties = self.cumulative_difficulties.clone();
|
||||
|
@ -232,8 +232,6 @@ impl DifficultyCache {
|
|||
|
||||
difficulties.push(self.next_difficulty(current_hf));
|
||||
|
||||
let mut diff_info_popped = Vec::new();
|
||||
|
||||
for (new_timestamp, hf) in blocks {
|
||||
timestamps.push_back(new_timestamp);
|
||||
|
||||
|
@ -241,17 +239,15 @@ impl DifficultyCache {
|
|||
cumulative_difficulties.push_back(last_cum_diff + *difficulties.last().unwrap());
|
||||
|
||||
if timestamps.len() > self.config.total_block_count() {
|
||||
diff_info_popped.push((
|
||||
timestamps.pop_front().unwrap(),
|
||||
cumulative_difficulties.pop_front().unwrap(),
|
||||
));
|
||||
timestamps.pop_front().unwrap();
|
||||
cumulative_difficulties.pop_front().unwrap();
|
||||
}
|
||||
|
||||
difficulties.push(next_difficulty(
|
||||
&self.config,
|
||||
×tamps,
|
||||
&cumulative_difficulties,
|
||||
&hf,
|
||||
hf,
|
||||
));
|
||||
}
|
||||
|
||||
|
@ -295,12 +291,12 @@ impl DifficultyCache {
|
|||
}
|
||||
}
|
||||
|
||||
/// Calculates the next difficulty with the inputted config/timestamps/cumulative_difficulties.
|
||||
/// Calculates the next difficulty with the inputted `config/timestamps/cumulative_difficulties`.
|
||||
fn next_difficulty(
|
||||
config: &DifficultyCacheConfig,
|
||||
timestamps: &VecDeque<u64>,
|
||||
cumulative_difficulties: &VecDeque<u128>,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
) -> u128 {
|
||||
if timestamps.len() <= 1 {
|
||||
return 1;
|
||||
|
|
|
@ -28,7 +28,7 @@ pub struct HardForkConfig {
|
|||
|
||||
impl HardForkConfig {
|
||||
/// Config for main-net.
|
||||
pub const fn main_net() -> HardForkConfig {
|
||||
pub const fn main_net() -> Self {
|
||||
Self {
|
||||
info: HFsInfo::main_net(),
|
||||
window: DEFAULT_WINDOW_SIZE,
|
||||
|
@ -36,7 +36,7 @@ impl HardForkConfig {
|
|||
}
|
||||
|
||||
/// Config for stage-net.
|
||||
pub const fn stage_net() -> HardForkConfig {
|
||||
pub const fn stage_net() -> Self {
|
||||
Self {
|
||||
info: HFsInfo::stage_net(),
|
||||
window: DEFAULT_WINDOW_SIZE,
|
||||
|
@ -44,7 +44,7 @@ impl HardForkConfig {
|
|||
}
|
||||
|
||||
/// Config for test-net.
|
||||
pub const fn test_net() -> HardForkConfig {
|
||||
pub const fn test_net() -> Self {
|
||||
Self {
|
||||
info: HFsInfo::test_net(),
|
||||
window: DEFAULT_WINDOW_SIZE,
|
||||
|
@ -54,7 +54,7 @@ impl HardForkConfig {
|
|||
|
||||
/// A struct that keeps track of the current hard-fork and current votes.
|
||||
#[derive(Debug, Clone, Eq, PartialEq)]
|
||||
pub struct HardForkState {
|
||||
pub(crate) struct HardForkState {
|
||||
/// The current active hard-fork.
|
||||
pub(crate) current_hardfork: HardFork,
|
||||
|
||||
|
@ -83,7 +83,7 @@ impl HardForkState {
|
|||
get_votes_in_range(database.clone(), block_start..chain_height, config.window).await?;
|
||||
|
||||
if chain_height > config.window {
|
||||
debug_assert_eq!(votes.total_votes(), config.window)
|
||||
debug_assert_eq!(votes.total_votes(), config.window);
|
||||
}
|
||||
|
||||
let BlockchainResponse::BlockExtendedHeader(ext_header) = database
|
||||
|
@ -97,7 +97,7 @@ impl HardForkState {
|
|||
|
||||
let current_hardfork = ext_header.version;
|
||||
|
||||
let mut hfs = HardForkState {
|
||||
let mut hfs = Self {
|
||||
config,
|
||||
current_hardfork,
|
||||
votes,
|
||||
|
@ -122,7 +122,7 @@ impl HardForkState {
|
|||
/// # Invariant
|
||||
///
|
||||
/// This _must_ only be used on a main-chain cache.
|
||||
pub async fn pop_blocks_main_chain<D: Database + Clone>(
|
||||
pub(crate) async fn pop_blocks_main_chain<D: Database + Clone>(
|
||||
&mut self,
|
||||
numb_blocks: usize,
|
||||
database: D,
|
||||
|
@ -159,7 +159,7 @@ impl HardForkState {
|
|||
}
|
||||
|
||||
/// Add a new block to the cache.
|
||||
pub fn new_block(&mut self, vote: HardFork, height: usize) {
|
||||
pub(crate) fn new_block(&mut self, vote: HardFork, height: usize) {
|
||||
// We don't _need_ to take in `height` but it's for safety, so we don't silently loose track
|
||||
// of blocks.
|
||||
assert_eq!(self.last_height + 1, height);
|
||||
|
@ -183,7 +183,7 @@ impl HardForkState {
|
|||
|
||||
/// Checks if the next hard-fork should be activated and activates it if it should.
|
||||
///
|
||||
/// https://cuprate.github.io/monero-docs/consensus_rules/hardforks.html#accepting-a-fork
|
||||
/// <https://cuprate.github.io/monero-docs/consensus_rules/hardforks.html#accepting-a-fork>
|
||||
fn check_set_new_hf(&mut self) {
|
||||
self.current_hardfork = self.votes.current_fork(
|
||||
&self.current_hardfork,
|
||||
|
@ -194,7 +194,7 @@ impl HardForkState {
|
|||
}
|
||||
|
||||
/// Returns the current hard-fork.
|
||||
pub fn current_hardfork(&self) -> HardFork {
|
||||
pub(crate) const fn current_hardfork(&self) -> HardFork {
|
||||
self.current_hardfork
|
||||
}
|
||||
}
|
||||
|
@ -218,7 +218,7 @@ async fn get_votes_in_range<D: Database>(
|
|||
panic!("Database sent incorrect response!");
|
||||
};
|
||||
|
||||
for hf_info in vote_list.into_iter() {
|
||||
for hf_info in vote_list {
|
||||
votes.add_vote_for_hf(&HardFork::from_vote(hf_info.vote));
|
||||
}
|
||||
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
//! RandomX VM Cache
|
||||
//! `RandomX` VM Cache
|
||||
//!
|
||||
//! This module keeps track of the RandomX VM to calculate the next blocks PoW, if the block needs a randomX VM and potentially
|
||||
//! This module keeps track of the `RandomX` VM to calculate the next blocks proof-of-work, if the block needs a randomX VM and potentially
|
||||
//! more VMs around this height.
|
||||
//!
|
||||
use std::{
|
||||
|
@ -34,11 +34,11 @@ const RX_SEEDS_CACHED: usize = 2;
|
|||
/// A multithreaded randomX VM.
|
||||
#[derive(Debug)]
|
||||
pub struct RandomXVm {
|
||||
/// These RandomX VMs all share the same cache.
|
||||
/// These `RandomX` VMs all share the same cache.
|
||||
vms: ThreadLocal<VmInner>,
|
||||
/// The RandomX cache.
|
||||
/// The `RandomX` cache.
|
||||
cache: RandomXCache,
|
||||
/// The flags used to start the RandomX VMs.
|
||||
/// The flags used to start the `RandomX` VMs.
|
||||
flags: RandomXFlag,
|
||||
}
|
||||
|
||||
|
@ -50,7 +50,7 @@ impl RandomXVm {
|
|||
|
||||
let cache = RandomXCache::new(flags, seed.as_slice())?;
|
||||
|
||||
Ok(RandomXVm {
|
||||
Ok(Self {
|
||||
vms: ThreadLocal::new(),
|
||||
cache,
|
||||
flags,
|
||||
|
@ -69,10 +69,10 @@ impl RandomX for RandomXVm {
|
|||
}
|
||||
}
|
||||
|
||||
/// The randomX VMs cache, keeps the VM needed to calculate the current block's PoW hash (if a VM is needed) and a
|
||||
/// The randomX VMs cache, keeps the VM needed to calculate the current block's proof-of-work hash (if a VM is needed) and a
|
||||
/// couple more around this VM.
|
||||
#[derive(Clone, Debug)]
|
||||
pub struct RandomXVmCache {
|
||||
pub(crate) struct RandomXVmCache {
|
||||
/// The top [`RX_SEEDS_CACHED`] RX seeds.
|
||||
pub(crate) seeds: VecDeque<(usize, [u8; 32])>,
|
||||
/// The VMs for `seeds` (if after hf 12, otherwise this will be empty).
|
||||
|
@ -117,7 +117,7 @@ impl RandomXVmCache {
|
|||
HashMap::new()
|
||||
};
|
||||
|
||||
Ok(RandomXVmCache {
|
||||
Ok(Self {
|
||||
seeds,
|
||||
vms,
|
||||
cached_vm: None,
|
||||
|
@ -125,14 +125,14 @@ impl RandomXVmCache {
|
|||
}
|
||||
|
||||
/// Add a randomX VM to the cache, with the seed it was created with.
|
||||
pub fn add_vm(&mut self, vm: ([u8; 32], Arc<RandomXVm>)) {
|
||||
pub(crate) fn add_vm(&mut self, vm: ([u8; 32], Arc<RandomXVm>)) {
|
||||
self.cached_vm.replace(vm);
|
||||
}
|
||||
|
||||
/// Creates a RX VM for an alt chain, looking at the main chain RX VMs to see if we can use one
|
||||
/// of them first.
|
||||
pub async fn get_alt_vm<D: Database>(
|
||||
&mut self,
|
||||
pub(crate) async fn get_alt_vm<D: Database>(
|
||||
&self,
|
||||
height: usize,
|
||||
chain: Chain,
|
||||
database: D,
|
||||
|
@ -152,7 +152,7 @@ impl RandomXVmCache {
|
|||
break;
|
||||
};
|
||||
|
||||
return Ok(vm.clone());
|
||||
return Ok(Arc::clone(vm));
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -161,8 +161,8 @@ impl RandomXVmCache {
|
|||
Ok(alt_vm)
|
||||
}
|
||||
|
||||
/// Get the main-chain RandomX VMs.
|
||||
pub async fn get_vms(&mut self) -> HashMap<usize, Arc<RandomXVm>> {
|
||||
/// Get the main-chain `RandomX` VMs.
|
||||
pub(crate) async fn get_vms(&mut self) -> HashMap<usize, Arc<RandomXVm>> {
|
||||
match self.seeds.len().checked_sub(self.vms.len()) {
|
||||
// No difference in the amount of seeds to VMs.
|
||||
Some(0) => (),
|
||||
|
@ -206,23 +206,23 @@ impl RandomXVmCache {
|
|||
})
|
||||
.collect()
|
||||
})
|
||||
.await
|
||||
.await;
|
||||
}
|
||||
}
|
||||
|
||||
self.vms.clone()
|
||||
}
|
||||
|
||||
/// Removes all the RandomX VMs above the `new_height`.
|
||||
pub fn pop_blocks_main_chain(&mut self, new_height: usize) {
|
||||
/// Removes all the `RandomX` VMs above the `new_height`.
|
||||
pub(crate) fn pop_blocks_main_chain(&mut self, new_height: usize) {
|
||||
self.seeds.retain(|(height, _)| *height < new_height);
|
||||
self.vms.retain(|height, _| *height < new_height);
|
||||
}
|
||||
|
||||
/// Add a new block to the VM cache.
|
||||
///
|
||||
/// hash is the block hash not the blocks PoW hash.
|
||||
pub fn new_block(&mut self, height: usize, hash: &[u8; 32]) {
|
||||
/// hash is the block hash not the blocks proof-of-work hash.
|
||||
pub(crate) fn new_block(&mut self, height: usize, hash: &[u8; 32]) {
|
||||
if is_randomx_seed_height(height) {
|
||||
tracing::debug!("Block {height} is a randomX seed height, adding it to the cache.",);
|
||||
|
||||
|
@ -235,7 +235,7 @@ impl RandomXVmCache {
|
|||
self.seeds
|
||||
.iter()
|
||||
.any(|(cached_height, _)| height == cached_height)
|
||||
})
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -258,7 +258,7 @@ pub(crate) fn get_last_rx_seed_heights(mut last_height: usize, mut amount: usize
|
|||
// We don't include the lag as we only want seeds not the specific seed for this height.
|
||||
let seed_height = (last_height - 1) & !(RX_SEEDHASH_EPOCH_BLOCKS - 1);
|
||||
seeds.push(seed_height);
|
||||
last_height = seed_height
|
||||
last_height = seed_height;
|
||||
}
|
||||
|
||||
seeds
|
||||
|
|
|
@ -36,7 +36,7 @@ pub(super) struct ContextTaskRequest {
|
|||
}
|
||||
|
||||
/// The Context task that keeps the blockchain context and handles requests.
|
||||
pub struct ContextTask<D: Database> {
|
||||
pub(crate) struct ContextTask<D: Database> {
|
||||
/// A token used to invalidate previous contexts when a new
|
||||
/// block is added to the chain.
|
||||
current_validity_token: ValidityToken,
|
||||
|
@ -65,7 +65,7 @@ pub struct ContextTask<D: Database> {
|
|||
impl<D: Database + Clone + Send + 'static> ContextTask<D> {
|
||||
/// Initialize the [`ContextTask`], this will need to pull a lot of data from the database so may take a
|
||||
/// while to complete.
|
||||
pub async fn init_context(
|
||||
pub(crate) async fn init_context(
|
||||
cfg: ContextConfig,
|
||||
mut database: D,
|
||||
) -> Result<Self, ExtendedConsensusError> {
|
||||
|
@ -131,7 +131,7 @@ impl<D: Database + Clone + Send + 'static> ContextTask<D> {
|
|||
rx_vms::RandomXVmCache::init_from_chain_height(chain_height, ¤t_hf, db).await
|
||||
});
|
||||
|
||||
let context_svc = ContextTask {
|
||||
let context_svc = Self {
|
||||
current_validity_token: ValidityToken::new(),
|
||||
difficulty_cache: difficulty_cache_handle.await.unwrap()?,
|
||||
weight_cache: weight_cache_handle.await.unwrap()?,
|
||||
|
@ -148,7 +148,7 @@ impl<D: Database + Clone + Send + 'static> ContextTask<D> {
|
|||
}
|
||||
|
||||
/// Handles a [`BlockChainContextRequest`] and returns a [`BlockChainContextResponse`].
|
||||
pub async fn handle_req(
|
||||
pub(crate) async fn handle_req(
|
||||
&mut self,
|
||||
req: BlockChainContextRequest,
|
||||
) -> Result<BlockChainContextResponse, tower::BoxError> {
|
||||
|
@ -164,17 +164,17 @@ impl<D: Database + Clone + Send + 'static> ContextTask<D> {
|
|||
context_to_verify_block: ContextToVerifyBlock {
|
||||
median_weight_for_block_reward: self
|
||||
.weight_cache
|
||||
.median_for_block_reward(¤t_hf),
|
||||
.median_for_block_reward(current_hf),
|
||||
effective_median_weight: self
|
||||
.weight_cache
|
||||
.effective_median_block_weight(¤t_hf),
|
||||
.effective_median_block_weight(current_hf),
|
||||
top_hash: self.top_block_hash,
|
||||
median_block_timestamp: self
|
||||
.difficulty_cache
|
||||
.median_timestamp(u64_to_usize(BLOCKCHAIN_TIMESTAMP_CHECK_WINDOW)),
|
||||
chain_height: self.chain_height,
|
||||
current_hf,
|
||||
next_difficulty: self.difficulty_cache.next_difficulty(¤t_hf),
|
||||
next_difficulty: self.difficulty_cache.next_difficulty(current_hf),
|
||||
already_generated_coins: self.already_generated_coins,
|
||||
},
|
||||
cumulative_difficulty: self.difficulty_cache.cumulative_difficulty(),
|
||||
|
@ -191,7 +191,7 @@ impl<D: Database + Clone + Send + 'static> ContextTask<D> {
|
|||
|
||||
let next_diffs = self
|
||||
.difficulty_cache
|
||||
.next_difficulties(blocks, &self.hardfork_state.current_hardfork());
|
||||
.next_difficulties(blocks, self.hardfork_state.current_hardfork());
|
||||
BlockChainContextResponse::BatchDifficulties(next_diffs)
|
||||
}
|
||||
BlockChainContextRequest::NewRXVM(vm) => {
|
||||
|
@ -330,10 +330,10 @@ impl<D: Database + Clone + Send + 'static> ContextTask<D> {
|
|||
|
||||
/// Run the [`ContextTask`], the task will listen for requests on the passed in channel. When the channel closes the
|
||||
/// task will finish.
|
||||
pub async fn run(mut self, mut rx: mpsc::Receiver<ContextTaskRequest>) {
|
||||
pub(crate) async fn run(mut self, mut rx: mpsc::Receiver<ContextTaskRequest>) {
|
||||
while let Some(req) = rx.recv().await {
|
||||
let res = self.handle_req(req.req).instrument(req.span).await;
|
||||
let _ = req.tx.send(res);
|
||||
drop(req.tx.send(res));
|
||||
}
|
||||
|
||||
tracing::info!("Shutting down blockchain context task.");
|
||||
|
|
|
@ -15,8 +15,8 @@ pub struct ValidityToken {
|
|||
|
||||
impl ValidityToken {
|
||||
/// Creates a new [`ValidityToken`]
|
||||
pub fn new() -> ValidityToken {
|
||||
ValidityToken {
|
||||
pub fn new() -> Self {
|
||||
Self {
|
||||
token: CancellationToken::new(),
|
||||
}
|
||||
}
|
||||
|
@ -28,6 +28,6 @@ impl ValidityToken {
|
|||
|
||||
/// Sets the data to invalid.
|
||||
pub fn set_data_invalid(self) {
|
||||
self.token.cancel()
|
||||
self.token.cancel();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -38,16 +38,16 @@ pub struct BlockWeightsCacheConfig {
|
|||
|
||||
impl BlockWeightsCacheConfig {
|
||||
/// Creates a new [`BlockWeightsCacheConfig`]
|
||||
pub const fn new(short_term_window: usize, long_term_window: usize) -> BlockWeightsCacheConfig {
|
||||
BlockWeightsCacheConfig {
|
||||
pub const fn new(short_term_window: usize, long_term_window: usize) -> Self {
|
||||
Self {
|
||||
short_term_window,
|
||||
long_term_window,
|
||||
}
|
||||
}
|
||||
|
||||
/// Returns the [`BlockWeightsCacheConfig`] for all networks (They are all the same as mainnet).
|
||||
pub fn main_net() -> BlockWeightsCacheConfig {
|
||||
BlockWeightsCacheConfig {
|
||||
pub const fn main_net() -> Self {
|
||||
Self {
|
||||
short_term_window: SHORT_TERM_WINDOW,
|
||||
long_term_window: LONG_TERM_WINDOW,
|
||||
}
|
||||
|
@ -99,7 +99,7 @@ impl BlockWeightsCache {
|
|||
|
||||
tracing::info!("Initialized block weight cache, chain-height: {:?}, long term weights length: {:?}, short term weights length: {:?}", chain_height, long_term_weights.len(), short_term_block_weights.len());
|
||||
|
||||
Ok(BlockWeightsCache {
|
||||
Ok(Self {
|
||||
short_term_block_weights: rayon_spawn_async(move || {
|
||||
RollingMedian::from_vec(short_term_block_weights, config.short_term_window)
|
||||
})
|
||||
|
@ -178,7 +178,7 @@ impl BlockWeightsCache {
|
|||
|
||||
/// Add a new block to the cache.
|
||||
///
|
||||
/// The block_height **MUST** be one more than the last height the cache has
|
||||
/// The `block_height` **MUST** be one more than the last height the cache has
|
||||
/// seen.
|
||||
pub fn new_block(&mut self, block_height: usize, block_weight: usize, long_term_weight: usize) {
|
||||
assert_eq!(self.tip_height + 1, block_height);
|
||||
|
@ -208,8 +208,8 @@ impl BlockWeightsCache {
|
|||
/// Returns the effective median weight, used for block reward calculations and to calculate
|
||||
/// the block weight limit.
|
||||
///
|
||||
/// See: https://cuprate.github.io/monero-book/consensus_rules/blocks/weight_limit.html#calculating-effective-median-weight
|
||||
pub fn effective_median_block_weight(&self, hf: &HardFork) -> usize {
|
||||
/// See: <https://cuprate.github.io/monero-book/consensus_rules/blocks/weight_limit.html#calculating-effective-median-weight>
|
||||
pub fn effective_median_block_weight(&self, hf: HardFork) -> usize {
|
||||
calculate_effective_median_block_weight(
|
||||
hf,
|
||||
self.median_short_term_weight(),
|
||||
|
@ -219,9 +219,9 @@ impl BlockWeightsCache {
|
|||
|
||||
/// Returns the median weight used to calculate block reward punishment.
|
||||
///
|
||||
/// https://cuprate.github.io/monero-book/consensus_rules/blocks/reward.html#calculating-block-reward
|
||||
pub fn median_for_block_reward(&self, hf: &HardFork) -> usize {
|
||||
if hf < &HardFork::V12 {
|
||||
/// <https://cuprate.github.io/monero-book/consensus_rules/blocks/reward.html#calculating-block-reward>
|
||||
pub fn median_for_block_reward(&self, hf: HardFork) -> usize {
|
||||
if hf < HardFork::V12 {
|
||||
self.median_short_term_weight()
|
||||
} else {
|
||||
self.effective_median_block_weight(hf)
|
||||
|
@ -232,17 +232,17 @@ impl BlockWeightsCache {
|
|||
|
||||
/// Calculates the effective median with the long term and short term median.
|
||||
fn calculate_effective_median_block_weight(
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
median_short_term_weight: usize,
|
||||
median_long_term_weight: usize,
|
||||
) -> usize {
|
||||
if hf < &HardFork::V10 {
|
||||
if hf < HardFork::V10 {
|
||||
return median_short_term_weight.max(penalty_free_zone(hf));
|
||||
}
|
||||
|
||||
let long_term_median = median_long_term_weight.max(PENALTY_FREE_ZONE_5);
|
||||
let short_term_median = median_short_term_weight;
|
||||
let effective_median = if hf >= &HardFork::V10 && hf < &HardFork::V15 {
|
||||
let effective_median = if hf >= HardFork::V10 && hf < HardFork::V15 {
|
||||
min(
|
||||
max(PENALTY_FREE_ZONE_5, short_term_median),
|
||||
50 * long_term_median,
|
||||
|
@ -258,19 +258,19 @@ fn calculate_effective_median_block_weight(
|
|||
}
|
||||
|
||||
/// Calculates a blocks long term weight.
|
||||
pub fn calculate_block_long_term_weight(
|
||||
hf: &HardFork,
|
||||
pub(crate) fn calculate_block_long_term_weight(
|
||||
hf: HardFork,
|
||||
block_weight: usize,
|
||||
long_term_median: usize,
|
||||
) -> usize {
|
||||
if hf < &HardFork::V10 {
|
||||
if hf < HardFork::V10 {
|
||||
return block_weight;
|
||||
}
|
||||
|
||||
let long_term_median = max(penalty_free_zone(hf), long_term_median);
|
||||
|
||||
let (short_term_constraint, adjusted_block_weight) =
|
||||
if hf >= &HardFork::V10 && hf < &HardFork::V15 {
|
||||
if hf >= HardFork::V10 && hf < HardFork::V15 {
|
||||
let stc = long_term_median + long_term_median * 2 / 5;
|
||||
(stc, block_weight)
|
||||
} else {
|
||||
|
|
|
@ -10,6 +10,16 @@
|
|||
//! implement a database you need to have a service which accepts [`BlockchainReadRequest`] and responds
|
||||
//! with [`BlockchainResponse`].
|
||||
//!
|
||||
|
||||
cfg_if::cfg_if! {
|
||||
// Used in external `tests/`.
|
||||
if #[cfg(test)] {
|
||||
use cuprate_test_utils as _;
|
||||
use curve25519_dalek as _;
|
||||
use hex_literal as _;
|
||||
}
|
||||
}
|
||||
|
||||
use cuprate_consensus_rules::ConsensusError;
|
||||
|
||||
mod batch_verifier;
|
||||
|
@ -34,6 +44,7 @@ pub use cuprate_types::{
|
|||
|
||||
/// An Error returned from one of the consensus services.
|
||||
#[derive(Debug, thiserror::Error)]
|
||||
#[expect(variant_size_differences)]
|
||||
pub enum ExtendedConsensusError {
|
||||
/// A consensus error.
|
||||
#[error("{0}")]
|
||||
|
@ -53,7 +64,8 @@ pub enum ExtendedConsensusError {
|
|||
}
|
||||
|
||||
/// Initialize the 2 verifier [`tower::Service`]s (block and transaction).
|
||||
pub async fn initialize_verifier<D, Ctx>(
|
||||
#[expect(clippy::type_complexity)]
|
||||
pub fn initialize_verifier<D, Ctx>(
|
||||
database: D,
|
||||
ctx_svc: Ctx,
|
||||
) -> Result<
|
||||
|
@ -112,7 +124,7 @@ pub mod __private {
|
|||
Response = BlockchainResponse,
|
||||
Error = tower::BoxError,
|
||||
>,
|
||||
> crate::Database for T
|
||||
> Database for T
|
||||
where
|
||||
T::Future: Future<Output = Result<Self::Response, Self::Error>> + Send + 'static,
|
||||
{
|
||||
|
|
|
@ -1,2 +1,2 @@
|
|||
mod context;
|
||||
pub mod mock_db;
|
||||
pub(crate) mod mock_db;
|
||||
|
|
|
@ -1,11 +1,12 @@
|
|||
use cuprate_consensus_rules::HardFork;
|
||||
|
||||
pub static HFS_2688888_2689608: [(HardFork, HardFork); 720] =
|
||||
pub(crate) static HFS_2688888_2689608: [(HardFork, HardFork); 720] =
|
||||
include!("./data/hfs_2688888_2689608");
|
||||
|
||||
pub static HFS_2678808_2688888: [(HardFork, HardFork); 10080] =
|
||||
pub(crate) static HFS_2678808_2688888: [(HardFork, HardFork); 10080] =
|
||||
include!("./data/hfs_2678808_2688888");
|
||||
|
||||
pub static BW_2850000_3050000: [(usize, usize); 200_000] = include!("./data/bw_2850000_3050000");
|
||||
pub(crate) static BW_2850000_3050000: [(usize, usize); 200_000] =
|
||||
include!("./data/bw_2850000_3050000");
|
||||
|
||||
pub static DIF_3000000_3002000: [(u128, u64); 2000] = include!("./data/dif_3000000_3002000");
|
||||
pub(crate) static DIF_3000000_3002000: [(u128, u64); 2000] = include!("./data/dif_3000000_3002000");
|
||||
|
|
|
@ -17,7 +17,7 @@ const TEST_LAG: usize = 2;
|
|||
|
||||
const TEST_TOTAL_ACCOUNTED_BLOCKS: usize = TEST_WINDOW + TEST_LAG;
|
||||
|
||||
pub const TEST_DIFFICULTY_CONFIG: DifficultyCacheConfig =
|
||||
pub(crate) const TEST_DIFFICULTY_CONFIG: DifficultyCacheConfig =
|
||||
DifficultyCacheConfig::new(TEST_WINDOW, TEST_CUT, TEST_LAG);
|
||||
|
||||
#[tokio::test]
|
||||
|
@ -35,7 +35,7 @@ async fn first_3_blocks_fixed_difficulty() -> Result<(), tower::BoxError> {
|
|||
.await?;
|
||||
|
||||
for height in 1..3 {
|
||||
assert_eq!(difficulty_cache.next_difficulty(&HardFork::V1), 1);
|
||||
assert_eq!(difficulty_cache.next_difficulty(HardFork::V1), 1);
|
||||
difficulty_cache.new_block(height, 0, u128::MAX);
|
||||
}
|
||||
Ok(())
|
||||
|
@ -66,7 +66,7 @@ async fn calculate_diff_3000000_3002000() -> Result<(), tower::BoxError> {
|
|||
for (cum_dif, timestamp) in DIF_3000000_3002000.iter().take(cfg.total_block_count()) {
|
||||
db_builder.add_block(
|
||||
DummyBlockExtendedHeader::default().with_difficulty_info(*timestamp, *cum_dif),
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
let mut diff_cache = DifficultyCache::init_from_chain_height(
|
||||
|
@ -84,7 +84,7 @@ async fn calculate_diff_3000000_3002000() -> Result<(), tower::BoxError> {
|
|||
{
|
||||
let diff = diff_info[1].0 - diff_info[0].0;
|
||||
|
||||
assert_eq!(diff_cache.next_difficulty(&HardFork::V16), diff);
|
||||
assert_eq!(diff_cache.next_difficulty(HardFork::V16), diff);
|
||||
|
||||
diff_cache.new_block(3_000_720 + i, diff_info[1].1, diff_info[1].0);
|
||||
}
|
||||
|
@ -139,22 +139,22 @@ proptest! {
|
|||
no_lag_cache.cumulative_difficulties.pop_front();
|
||||
}
|
||||
// get the difficulty
|
||||
let next_diff_no_lag = no_lag_cache.next_difficulty(&hf);
|
||||
let next_diff_no_lag = no_lag_cache.next_difficulty(hf);
|
||||
|
||||
for _ in 0..TEST_LAG {
|
||||
// add new blocks to the lagged cache
|
||||
diff_cache.new_block(diff_cache.last_accounted_height+1, timestamp, cumulative_difficulty);
|
||||
}
|
||||
// they both should now be the same
|
||||
prop_assert_eq!(diff_cache.next_difficulty(&hf), next_diff_no_lag)
|
||||
prop_assert_eq!(diff_cache.next_difficulty(hf), next_diff_no_lag);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn next_difficulty_consistent(diff_cache in arb_difficulty_cache(TEST_TOTAL_ACCOUNTED_BLOCKS), hf in any::<HardFork>()) {
|
||||
let first_call = diff_cache.next_difficulty(&hf);
|
||||
prop_assert_eq!(first_call, diff_cache.next_difficulty(&hf));
|
||||
prop_assert_eq!(first_call, diff_cache.next_difficulty(&hf));
|
||||
prop_assert_eq!(first_call, diff_cache.next_difficulty(&hf));
|
||||
let first_call = diff_cache.next_difficulty(hf);
|
||||
prop_assert_eq!(first_call, diff_cache.next_difficulty(hf));
|
||||
prop_assert_eq!(first_call, diff_cache.next_difficulty(hf));
|
||||
prop_assert_eq!(first_call, diff_cache.next_difficulty(hf));
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
@ -178,7 +178,7 @@ proptest! {
|
|||
|
||||
#[test]
|
||||
fn window_size_kept_constant(mut diff_cache in arb_difficulty_cache(TEST_TOTAL_ACCOUNTED_BLOCKS), new_blocks in any::<Vec<(u64, u128)>>()) {
|
||||
for (timestamp, cumulative_difficulty) in new_blocks.into_iter() {
|
||||
for (timestamp, cumulative_difficulty) in new_blocks {
|
||||
diff_cache.new_block(diff_cache.last_accounted_height+1, timestamp, cumulative_difficulty);
|
||||
prop_assert_eq!(diff_cache.timestamps.len(), TEST_TOTAL_ACCOUNTED_BLOCKS);
|
||||
prop_assert_eq!(diff_cache.cumulative_difficulties.len(), TEST_TOTAL_ACCOUNTED_BLOCKS);
|
||||
|
@ -193,7 +193,7 @@ proptest! {
|
|||
) {
|
||||
let cache = diff_cache.clone();
|
||||
|
||||
diff_cache.next_difficulties(timestamps.into_iter().zip([hf].into_iter().cycle()).collect(), &hf);
|
||||
diff_cache.next_difficulties(timestamps.into_iter().zip(std::iter::once(hf).cycle()).collect(), hf);
|
||||
|
||||
prop_assert_eq!(diff_cache, cache);
|
||||
}
|
||||
|
@ -204,12 +204,12 @@ proptest! {
|
|||
timestamps in any_with::<Vec<u64>>(size_range(0..1000).lift()),
|
||||
hf in any::<HardFork>(),
|
||||
) {
|
||||
let timestamps: Vec<_> = timestamps.into_iter().zip([hf].into_iter().cycle()).collect();
|
||||
let timestamps: Vec<_> = timestamps.into_iter().zip(std::iter::once(hf).cycle()).collect();
|
||||
|
||||
let diffs = diff_cache.next_difficulties(timestamps.clone(), &hf);
|
||||
let diffs = diff_cache.next_difficulties(timestamps.clone(), hf);
|
||||
|
||||
for (timestamp, diff) in timestamps.into_iter().zip(diffs.into_iter()) {
|
||||
prop_assert_eq!(diff_cache.next_difficulty(×tamp.1), diff);
|
||||
prop_assert_eq!(diff_cache.next_difficulty(timestamp.1), diff);
|
||||
diff_cache.new_block(diff_cache.last_accounted_height +1, timestamp.0, diff + diff_cache.cumulative_difficulty());
|
||||
}
|
||||
|
||||
|
@ -226,7 +226,7 @@ proptest! {
|
|||
let blocks_to_pop = new_blocks.len();
|
||||
|
||||
let mut new_cache = old_cache.clone();
|
||||
for (timestamp, cumulative_difficulty) in new_blocks.into_iter() {
|
||||
for (timestamp, cumulative_difficulty) in new_blocks {
|
||||
database.add_block(DummyBlockExtendedHeader::default().with_difficulty_info(timestamp, cumulative_difficulty));
|
||||
new_cache.new_block(new_cache.last_accounted_height+1, timestamp, cumulative_difficulty);
|
||||
}
|
||||
|
@ -250,7 +250,7 @@ proptest! {
|
|||
let blocks_to_pop = new_blocks.len();
|
||||
|
||||
let mut new_cache = old_cache.clone();
|
||||
for (timestamp, cumulative_difficulty) in new_blocks.into_iter() {
|
||||
for (timestamp, cumulative_difficulty) in new_blocks {
|
||||
database.add_block(DummyBlockExtendedHeader::default().with_difficulty_info(timestamp, cumulative_difficulty));
|
||||
new_cache.new_block(new_cache.last_accounted_height+1, timestamp, cumulative_difficulty);
|
||||
}
|
||||
|
|
|
@ -31,7 +31,7 @@ const TEST_HFS: [HFInfo; NUMB_OF_HARD_FORKS] = [
|
|||
HFInfo::new(150, 0),
|
||||
];
|
||||
|
||||
pub const TEST_HARD_FORK_CONFIG: HardForkConfig = HardForkConfig {
|
||||
pub(crate) const TEST_HARD_FORK_CONFIG: HardForkConfig = HardForkConfig {
|
||||
window: TEST_WINDOW_SIZE,
|
||||
info: HFsInfo::new(TEST_HFS),
|
||||
};
|
||||
|
|
|
@ -39,6 +39,7 @@ fn rx_heights_consistent() {
|
|||
}
|
||||
|
||||
#[tokio::test]
|
||||
#[expect(unused_qualifications, reason = "false positive in tokio macro")]
|
||||
async fn rx_vm_created_on_hf_12() {
|
||||
let db = DummyDatabaseBuilder::default().finish(Some(10));
|
||||
|
||||
|
|
|
@ -8,7 +8,8 @@ use crate::{
|
|||
};
|
||||
use cuprate_types::Chain;
|
||||
|
||||
pub const TEST_WEIGHT_CONFIG: BlockWeightsCacheConfig = BlockWeightsCacheConfig::new(100, 5000);
|
||||
pub(crate) const TEST_WEIGHT_CONFIG: BlockWeightsCacheConfig =
|
||||
BlockWeightsCacheConfig::new(100, 5000);
|
||||
|
||||
#[tokio::test]
|
||||
async fn blocks_out_of_window_not_counted() -> Result<(), tower::BoxError> {
|
||||
|
@ -157,7 +158,7 @@ async fn calc_bw_ltw_2850000_3050000() {
|
|||
|
||||
for (i, (weight, ltw)) in BW_2850000_3050000.iter().skip(100_000).enumerate() {
|
||||
let calc_ltw = calculate_block_long_term_weight(
|
||||
&HardFork::V16,
|
||||
HardFork::V16,
|
||||
*weight,
|
||||
weight_cache.median_long_term_weight(),
|
||||
);
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(non_local_definitions, reason = "proptest macro")]
|
||||
|
||||
use std::{
|
||||
future::Future,
|
||||
pin::Pin,
|
||||
|
@ -60,7 +62,7 @@ pub struct DummyBlockExtendedHeader {
|
|||
|
||||
impl From<DummyBlockExtendedHeader> for ExtendedBlockHeader {
|
||||
fn from(value: DummyBlockExtendedHeader) -> Self {
|
||||
ExtendedBlockHeader {
|
||||
Self {
|
||||
version: value.version.unwrap_or(HardFork::V1),
|
||||
vote: value.vote.unwrap_or(HardFork::V1).as_u8(),
|
||||
timestamp: value.timestamp.unwrap_or_default(),
|
||||
|
@ -72,31 +74,23 @@ impl From<DummyBlockExtendedHeader> for ExtendedBlockHeader {
|
|||
}
|
||||
|
||||
impl DummyBlockExtendedHeader {
|
||||
pub fn with_weight_into(
|
||||
mut self,
|
||||
weight: usize,
|
||||
long_term_weight: usize,
|
||||
) -> DummyBlockExtendedHeader {
|
||||
pub const fn with_weight_into(mut self, weight: usize, long_term_weight: usize) -> Self {
|
||||
self.block_weight = Some(weight);
|
||||
self.long_term_weight = Some(long_term_weight);
|
||||
self
|
||||
}
|
||||
|
||||
pub fn with_hard_fork_info(
|
||||
mut self,
|
||||
version: HardFork,
|
||||
vote: HardFork,
|
||||
) -> DummyBlockExtendedHeader {
|
||||
pub const fn with_hard_fork_info(mut self, version: HardFork, vote: HardFork) -> Self {
|
||||
self.vote = Some(vote);
|
||||
self.version = Some(version);
|
||||
self
|
||||
}
|
||||
|
||||
pub fn with_difficulty_info(
|
||||
pub const fn with_difficulty_info(
|
||||
mut self,
|
||||
timestamp: u64,
|
||||
cumulative_difficulty: u128,
|
||||
) -> DummyBlockExtendedHeader {
|
||||
) -> Self {
|
||||
self.timestamp = Some(timestamp);
|
||||
self.cumulative_difficulty = Some(cumulative_difficulty);
|
||||
self
|
||||
|
@ -104,16 +98,16 @@ impl DummyBlockExtendedHeader {
|
|||
}
|
||||
|
||||
#[derive(Debug, Default)]
|
||||
pub struct DummyDatabaseBuilder {
|
||||
pub(crate) struct DummyDatabaseBuilder {
|
||||
blocks: Vec<DummyBlockExtendedHeader>,
|
||||
}
|
||||
|
||||
impl DummyDatabaseBuilder {
|
||||
pub fn add_block(&mut self, block: DummyBlockExtendedHeader) {
|
||||
pub(crate) fn add_block(&mut self, block: DummyBlockExtendedHeader) {
|
||||
self.blocks.push(block);
|
||||
}
|
||||
|
||||
pub fn finish(self, dummy_height: Option<usize>) -> DummyDatabase {
|
||||
pub(crate) fn finish(self, dummy_height: Option<usize>) -> DummyDatabase {
|
||||
DummyDatabase {
|
||||
blocks: Arc::new(self.blocks.into()),
|
||||
dummy_height,
|
||||
|
@ -122,14 +116,15 @@ impl DummyDatabaseBuilder {
|
|||
}
|
||||
|
||||
#[derive(Clone, Debug)]
|
||||
pub struct DummyDatabase {
|
||||
pub(crate) struct DummyDatabase {
|
||||
blocks: Arc<RwLock<Vec<DummyBlockExtendedHeader>>>,
|
||||
dummy_height: Option<usize>,
|
||||
}
|
||||
|
||||
impl DummyDatabase {
|
||||
pub fn add_block(&mut self, block: DummyBlockExtendedHeader) {
|
||||
self.blocks.write().unwrap().push(block)
|
||||
#[expect(clippy::needless_pass_by_ref_mut)]
|
||||
pub(crate) fn add_block(&mut self, block: DummyBlockExtendedHeader) {
|
||||
self.blocks.write().unwrap().push(block);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -144,7 +139,7 @@ impl Service<BlockchainReadRequest> for DummyDatabase {
|
|||
}
|
||||
|
||||
fn call(&mut self, req: BlockchainReadRequest) -> Self::Future {
|
||||
let blocks = self.blocks.clone();
|
||||
let blocks = Arc::clone(&self.blocks);
|
||||
let dummy_height = self.dummy_height;
|
||||
|
||||
async move {
|
||||
|
|
|
@ -5,7 +5,6 @@
|
|||
use std::{
|
||||
collections::HashSet,
|
||||
future::Future,
|
||||
ops::Deref,
|
||||
pin::Pin,
|
||||
sync::Arc,
|
||||
task::{Context, Poll},
|
||||
|
@ -102,8 +101,8 @@ where
|
|||
D::Future: Send + 'static,
|
||||
{
|
||||
/// Creates a new [`TxVerifierService`].
|
||||
pub fn new(database: D) -> TxVerifierService<D> {
|
||||
TxVerifierService { database }
|
||||
pub const fn new(database: D) -> Self {
|
||||
Self { database }
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -244,7 +243,7 @@ where
|
|||
|
||||
if kis_spent {
|
||||
tracing::debug!("One or more key images in batch already spent.");
|
||||
Err(ConsensusError::Transaction(TransactionError::KeyImageSpent))?;
|
||||
return Err(ConsensusError::Transaction(TransactionError::KeyImageSpent).into());
|
||||
}
|
||||
|
||||
let mut verified_at_block_hashes = txs
|
||||
|
@ -281,8 +280,8 @@ where
|
|||
let (txs_needing_full_verification, txs_needing_partial_verification) =
|
||||
transactions_needing_verification(
|
||||
txs,
|
||||
verified_at_block_hashes,
|
||||
&hf,
|
||||
&verified_at_block_hashes,
|
||||
hf,
|
||||
current_chain_height,
|
||||
time_for_time_lock,
|
||||
)?;
|
||||
|
@ -302,11 +301,14 @@ where
|
|||
Ok(VerifyTxResponse::Ok)
|
||||
}
|
||||
|
||||
#[allow(clippy::type_complexity)] // I don't think the return is too complex
|
||||
#[expect(
|
||||
clippy::type_complexity,
|
||||
reason = "I don't think the return is too complex"
|
||||
)]
|
||||
fn transactions_needing_verification(
|
||||
txs: &[Arc<TransactionVerificationData>],
|
||||
hashes_in_main_chain: HashSet<[u8; 32]>,
|
||||
current_hf: &HardFork,
|
||||
hashes_in_main_chain: &HashSet<[u8; 32]>,
|
||||
current_hf: HardFork,
|
||||
current_chain_height: usize,
|
||||
time_for_time_lock: u64,
|
||||
) -> Result<
|
||||
|
@ -321,27 +323,28 @@ fn transactions_needing_verification(
|
|||
// txs needing partial _contextual_ validation, not semantic.
|
||||
let mut partial_validation_transactions = Vec::new();
|
||||
|
||||
for tx in txs.iter() {
|
||||
for tx in txs {
|
||||
let guard = tx.cached_verification_state.lock().unwrap();
|
||||
|
||||
match guard.deref() {
|
||||
match &*guard {
|
||||
CachedVerificationState::NotVerified => {
|
||||
drop(guard);
|
||||
full_validation_transactions
|
||||
.push((tx.clone(), VerificationNeeded::SemanticAndContextual));
|
||||
.push((Arc::clone(tx), VerificationNeeded::SemanticAndContextual));
|
||||
continue;
|
||||
}
|
||||
CachedVerificationState::ValidAtHashAndHF { block_hash, hf } => {
|
||||
if current_hf != hf {
|
||||
if current_hf != *hf {
|
||||
drop(guard);
|
||||
full_validation_transactions
|
||||
.push((tx.clone(), VerificationNeeded::SemanticAndContextual));
|
||||
.push((Arc::clone(tx), VerificationNeeded::SemanticAndContextual));
|
||||
continue;
|
||||
}
|
||||
|
||||
if !hashes_in_main_chain.contains(block_hash) {
|
||||
drop(guard);
|
||||
full_validation_transactions.push((tx.clone(), VerificationNeeded::Contextual));
|
||||
full_validation_transactions
|
||||
.push((Arc::clone(tx), VerificationNeeded::Contextual));
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
@ -350,21 +353,22 @@ fn transactions_needing_verification(
|
|||
hf,
|
||||
time_lock,
|
||||
} => {
|
||||
if current_hf != hf {
|
||||
if current_hf != *hf {
|
||||
drop(guard);
|
||||
full_validation_transactions
|
||||
.push((tx.clone(), VerificationNeeded::SemanticAndContextual));
|
||||
.push((Arc::clone(tx), VerificationNeeded::SemanticAndContextual));
|
||||
continue;
|
||||
}
|
||||
|
||||
if !hashes_in_main_chain.contains(block_hash) {
|
||||
drop(guard);
|
||||
full_validation_transactions.push((tx.clone(), VerificationNeeded::Contextual));
|
||||
full_validation_transactions
|
||||
.push((Arc::clone(tx), VerificationNeeded::Contextual));
|
||||
continue;
|
||||
}
|
||||
|
||||
// If the time lock is still locked then the transaction is invalid.
|
||||
if !output_unlocked(time_lock, current_chain_height, time_for_time_lock, hf) {
|
||||
if !output_unlocked(time_lock, current_chain_height, time_for_time_lock, *hf) {
|
||||
return Err(ConsensusError::Transaction(
|
||||
TransactionError::OneOrMoreRingMembersLocked,
|
||||
));
|
||||
|
@ -374,7 +378,7 @@ fn transactions_needing_verification(
|
|||
|
||||
if tx.version == TxVersion::RingSignatures {
|
||||
drop(guard);
|
||||
partial_validation_transactions.push(tx.clone());
|
||||
partial_validation_transactions.push(Arc::clone(tx));
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
@ -400,7 +404,7 @@ where
|
|||
|
||||
batch_get_decoy_info(&txs, hf, database)
|
||||
.await?
|
||||
.try_for_each(|decoy_info| decoy_info.and_then(|di| Ok(check_decoy_info(&di, &hf)?)))?;
|
||||
.try_for_each(|decoy_info| decoy_info.and_then(|di| Ok(check_decoy_info(&di, hf)?)))?;
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
@ -417,7 +421,7 @@ where
|
|||
D: Database + Clone + Sync + Send + 'static,
|
||||
{
|
||||
let txs_ring_member_info =
|
||||
batch_get_ring_member_info(txs.iter().map(|(tx, _)| tx), &hf, database).await?;
|
||||
batch_get_ring_member_info(txs.iter().map(|(tx, _)| tx), hf, database).await?;
|
||||
|
||||
rayon_spawn_async(move || {
|
||||
let batch_verifier = MultiThreadedBatchVerifier::new(rayon::current_num_threads());
|
||||
|
@ -432,7 +436,7 @@ where
|
|||
tx.tx_blob.len(),
|
||||
tx.tx_weight,
|
||||
&tx.tx_hash,
|
||||
&hf,
|
||||
hf,
|
||||
&batch_verifier,
|
||||
)?;
|
||||
// make sure we calculated the right fee.
|
||||
|
@ -445,7 +449,7 @@ where
|
|||
ring,
|
||||
current_chain_height,
|
||||
current_time_lock_timestamp,
|
||||
&hf,
|
||||
hf,
|
||||
)?;
|
||||
|
||||
Ok::<_, ConsensusError>(())
|
||||
|
|
|
@ -57,7 +57,7 @@ fn get_ring_members_for_inputs(
|
|||
})
|
||||
.collect::<Result<_, TransactionError>>()?)
|
||||
}
|
||||
_ => Err(TransactionError::IncorrectInputType),
|
||||
Input::Gen(_) => Err(TransactionError::IncorrectInputType),
|
||||
})
|
||||
.collect::<Result<_, TransactionError>>()
|
||||
}
|
||||
|
@ -143,7 +143,7 @@ fn new_rings(
|
|||
/// them.
|
||||
pub async fn batch_get_ring_member_info<D: Database>(
|
||||
txs_verification_data: impl Iterator<Item = &Arc<TransactionVerificationData>> + Clone,
|
||||
hf: &HardFork,
|
||||
hf: HardFork,
|
||||
mut database: D,
|
||||
) -> Result<Vec<TxRingMembersInfo>, ExtendedConsensusError> {
|
||||
let mut output_ids = HashMap::new();
|
||||
|
@ -183,14 +183,14 @@ pub async fn batch_get_ring_member_info<D: Database>(
|
|||
)
|
||||
.map_err(ConsensusError::Transaction)?;
|
||||
|
||||
let decoy_info = if hf != &HardFork::V1 {
|
||||
let decoy_info = if hf == HardFork::V1 {
|
||||
None
|
||||
} else {
|
||||
// this data is only needed after hard-fork 1.
|
||||
Some(
|
||||
DecoyInfo::new(&tx_v_data.tx.prefix().inputs, numb_outputs, hf)
|
||||
.map_err(ConsensusError::Transaction)?,
|
||||
)
|
||||
} else {
|
||||
None
|
||||
};
|
||||
|
||||
new_ring_member_info(ring_members_for_tx, decoy_info, tx_v_data.version)
|
||||
|
@ -224,7 +224,7 @@ pub async fn batch_get_decoy_info<'a, D: Database + Clone + Send + 'static>(
|
|||
.flat_map(|tx_info| {
|
||||
tx_info.tx.prefix().inputs.iter().map(|input| match input {
|
||||
Input::ToKey { amount, .. } => amount.unwrap_or(0),
|
||||
_ => 0,
|
||||
Input::Gen(_) => 0,
|
||||
})
|
||||
})
|
||||
.collect::<HashSet<_>>();
|
||||
|
@ -249,7 +249,7 @@ pub async fn batch_get_decoy_info<'a, D: Database + Clone + Send + 'static>(
|
|||
DecoyInfo::new(
|
||||
&tx_v_data.tx.prefix().inputs,
|
||||
|amt| outputs_with_amount.get(&amt).copied().unwrap_or(0),
|
||||
&hf,
|
||||
hf,
|
||||
)
|
||||
.map_err(ConsensusError::Transaction)
|
||||
}))
|
||||
|
|
|
@ -39,7 +39,7 @@ pub fn new_tx_verification_data(
|
|||
/// Calculates the weight of a [`Transaction`].
|
||||
///
|
||||
/// This is more efficient that [`Transaction::weight`] if you already have the transaction blob.
|
||||
pub fn tx_weight(tx: &Transaction, tx_blob: &[u8]) -> usize {
|
||||
pub(crate) fn tx_weight(tx: &Transaction, tx_blob: &[u8]) -> usize {
|
||||
// the tx weight is only different from the blobs length for bp(+) txs.
|
||||
|
||||
match &tx {
|
||||
|
@ -64,7 +64,7 @@ pub fn tx_weight(tx: &Transaction, tx_blob: &[u8]) -> usize {
|
|||
}
|
||||
|
||||
/// Calculates the fee of the [`Transaction`].
|
||||
pub fn tx_fee(tx: &Transaction) -> Result<u64, TransactionError> {
|
||||
pub(crate) fn tx_fee(tx: &Transaction) -> Result<u64, TransactionError> {
|
||||
let mut fee = 0_u64;
|
||||
|
||||
match &tx {
|
||||
|
|
|
@ -1,3 +1,6 @@
|
|||
#![expect(unused_crate_dependencies, reason = "external test module")]
|
||||
#![expect(clippy::allow_attributes, reason = "usage inside macro")]
|
||||
|
||||
use std::{
|
||||
collections::{BTreeMap, HashMap},
|
||||
future::ready,
|
||||
|
@ -29,7 +32,7 @@ fn dummy_database(outputs: BTreeMap<u64, OutputOnChain>) -> impl Database + Clon
|
|||
BlockchainResponse::NumberOutputsWithAmount(HashMap::new())
|
||||
}
|
||||
BlockchainReadRequest::Outputs(outs) => {
|
||||
let idxs = outs.get(&0).unwrap();
|
||||
let idxs = &outs[&0];
|
||||
|
||||
let mut ret = HashMap::new();
|
||||
|
||||
|
|
|
@ -22,7 +22,6 @@ hex-literal = { workspace = true, optional = true }
|
|||
paste = { workspace = true, optional = true }
|
||||
|
||||
[dev-dependencies]
|
||||
monero-serai = { workspace = true }
|
||||
hex = { workspace = true, features = ["std"] }
|
||||
pretty_assertions = { workspace = true }
|
||||
|
||||
|
|
|
@ -18,13 +18,10 @@ macro_rules! generate_genesis_consts {
|
|||
$(
|
||||
#[doc = concat!(stringify!([<$network:camel>]), " data.")]
|
||||
pub mod [<$network:lower>] {
|
||||
#[cfg(feature = "monero-serai")]
|
||||
use monero_serai::{block::Block, transaction::Transaction};
|
||||
#[cfg(feature = "monero-serai")]
|
||||
use std::sync::LazyLock;
|
||||
|
||||
#[doc = concat!("The ", stringify!([<$network:lower>]), " genesis block in [`Block`] form.")]
|
||||
#[cfg(feature = "monero-serai")]
|
||||
pub static GENESIS_BLOCK: LazyLock<Block> =
|
||||
LazyLock::new(|| Block::read(&mut GENESIS_BLOCK_BYTES).unwrap());
|
||||
|
||||
|
@ -41,7 +38,6 @@ macro_rules! generate_genesis_consts {
|
|||
pub const GENESIS_BLOCK_HASH_BYTES: [u8; 32] = hex_literal::hex!($block_hash);
|
||||
|
||||
#[doc = concat!("The ", stringify!([<$network:lower>]), " genesis block in [`Transaction`] form.")]
|
||||
#[cfg(feature = "monero-serai")]
|
||||
pub static GENESIS_TX: LazyLock<Transaction> =
|
||||
LazyLock::new(|| Transaction::read(&mut GENESIS_TX_BYTES).unwrap());
|
||||
|
||||
|
|
|
@ -5,7 +5,6 @@ cfg_if::cfg_if! {
|
|||
// Used in test modules.
|
||||
if #[cfg(test)] {
|
||||
use hex as _;
|
||||
use monero_serai as _;
|
||||
use pretty_assertions as _;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -74,9 +74,9 @@ pub const fn combine_low_high_bits_to_u128(low_bits: u64, high_bits: u64) -> u12
|
|||
/// ```rust
|
||||
/// # use cuprate_helper::map::*;
|
||||
/// # use monero_serai::transaction::*;
|
||||
/// use cuprate_constants::block::MAX_BLOCK_HEIGHT;
|
||||
/// use cuprate_constants::block::{MAX_BLOCK_HEIGHT, MAX_BLOCK_HEIGHT_USIZE};
|
||||
/// assert_eq!(u64_to_timelock(0), Timelock::None);
|
||||
/// assert_eq!(u64_to_timelock(MAX_BLOCK_HEIGHT-1), Timelock::Block(MAX_BLOCK_HEIGHT-1));
|
||||
/// assert_eq!(u64_to_timelock(MAX_BLOCK_HEIGHT-1), Timelock::Block(MAX_BLOCK_HEIGHT_USIZE-1));
|
||||
/// assert_eq!(u64_to_timelock(MAX_BLOCK_HEIGHT), Timelock::Time(MAX_BLOCK_HEIGHT));
|
||||
/// ```
|
||||
pub const fn u64_to_timelock(u: u64) -> Timelock {
|
||||
|
@ -96,9 +96,9 @@ pub const fn u64_to_timelock(u: u64) -> Timelock {
|
|||
/// ```rust
|
||||
/// # use cuprate_helper::map::*;
|
||||
/// # use monero_serai::transaction::*;
|
||||
/// use cuprate_constants::block::MAX_BLOCK_HEIGHT;
|
||||
/// use cuprate_constants::block::{MAX_BLOCK_HEIGHT, MAX_BLOCK_HEIGHT_USIZE};
|
||||
/// assert_eq!(timelock_to_u64(Timelock::None), 0);
|
||||
/// assert_eq!(timelock_to_u64(Timelock::Block(MAX_BLOCK_HEIGHT-1)), MAX_BLOCK_HEIGHT-1);
|
||||
/// assert_eq!(timelock_to_u64(Timelock::Block(MAX_BLOCK_HEIGHT_USIZE-1)), MAX_BLOCK_HEIGHT-1);
|
||||
/// assert_eq!(timelock_to_u64(Timelock::Time(MAX_BLOCK_HEIGHT)), MAX_BLOCK_HEIGHT);
|
||||
/// ```
|
||||
pub const fn timelock_to_u64(timelock: Timelock) -> u64 {
|
||||
|
|
|
@ -25,3 +25,6 @@ thiserror = { workspace = true, optional = true}
|
|||
|
||||
[dev-dependencies]
|
||||
hex = { workspace = true, features = ["default"] }
|
||||
|
||||
[lints]
|
||||
workspace = true
|
|
@ -9,7 +9,7 @@ pub struct ContainerAsBlob<T: Containerable + EpeeValue>(Vec<T>);
|
|||
|
||||
impl<T: Containerable + EpeeValue> From<Vec<T>> for ContainerAsBlob<T> {
|
||||
fn from(value: Vec<T>) -> Self {
|
||||
ContainerAsBlob(value)
|
||||
Self(value)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -36,9 +36,7 @@ impl<T: Containerable + EpeeValue> EpeeValue for ContainerAsBlob<T> {
|
|||
));
|
||||
}
|
||||
|
||||
Ok(ContainerAsBlob(
|
||||
bytes.chunks(T::SIZE).map(T::from_bytes).collect(),
|
||||
))
|
||||
Ok(Self(bytes.chunks(T::SIZE).map(T::from_bytes).collect()))
|
||||
}
|
||||
|
||||
fn should_write(&self) -> bool {
|
||||
|
@ -46,10 +44,10 @@ impl<T: Containerable + EpeeValue> EpeeValue for ContainerAsBlob<T> {
|
|||
}
|
||||
|
||||
fn epee_default_value() -> Option<Self> {
|
||||
Some(ContainerAsBlob(vec![]))
|
||||
Some(Self(vec![]))
|
||||
}
|
||||
|
||||
fn write<B: BufMut>(self, w: &mut B) -> crate::Result<()> {
|
||||
fn write<B: BufMut>(self, w: &mut B) -> Result<()> {
|
||||
let mut buf = BytesMut::with_capacity(self.0.len() * T::SIZE);
|
||||
self.0.iter().for_each(|tt| tt.push_bytes(&mut buf));
|
||||
buf.write(w)
|
||||
|
|
|
@ -7,6 +7,7 @@ use core::{
|
|||
pub type Result<T> = core::result::Result<T, Error>;
|
||||
|
||||
#[cfg_attr(feature = "std", derive(thiserror::Error))]
|
||||
#[expect(clippy::error_impl_error, reason = "FIXME: rename this type")]
|
||||
pub enum Error {
|
||||
#[cfg_attr(feature = "std", error("IO error: {0}"))]
|
||||
IO(&'static str),
|
||||
|
@ -17,19 +18,18 @@ pub enum Error {
|
|||
}
|
||||
|
||||
impl Error {
|
||||
fn field_name(&self) -> &'static str {
|
||||
const fn field_name(&self) -> &'static str {
|
||||
match self {
|
||||
Error::IO(_) => "io",
|
||||
Error::Format(_) => "format",
|
||||
Error::Value(_) => "value",
|
||||
Self::IO(_) => "io",
|
||||
Self::Format(_) => "format",
|
||||
Self::Value(_) => "value",
|
||||
}
|
||||
}
|
||||
|
||||
fn field_data(&self) -> &str {
|
||||
match self {
|
||||
Error::IO(data) => data,
|
||||
Error::Format(data) => data,
|
||||
Error::Value(data) => data,
|
||||
Self::IO(data) | Self::Format(data) => data,
|
||||
Self::Value(data) => data,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -44,12 +44,12 @@ impl Debug for Error {
|
|||
|
||||
impl From<TryFromIntError> for Error {
|
||||
fn from(_: TryFromIntError) -> Self {
|
||||
Error::Value("Int is too large".to_string())
|
||||
Self::Value("Int is too large".to_string())
|
||||
}
|
||||
}
|
||||
|
||||
impl From<Utf8Error> for Error {
|
||||
fn from(_: Utf8Error) -> Self {
|
||||
Error::Value("Invalid utf8 str".to_string())
|
||||
Self::Value("Invalid utf8 str".to_string())
|
||||
}
|
||||
}
|
||||
|
|
|
@ -3,7 +3,7 @@ use bytes::{Buf, BufMut};
|
|||
use crate::error::*;
|
||||
|
||||
#[inline]
|
||||
pub fn checked_read_primitive<B: Buf, R: Sized>(
|
||||
pub(crate) fn checked_read_primitive<B: Buf, R: Sized>(
|
||||
b: &mut B,
|
||||
read: impl Fn(&mut B) -> R,
|
||||
) -> Result<R> {
|
||||
|
@ -11,16 +11,20 @@ pub fn checked_read_primitive<B: Buf, R: Sized>(
|
|||
}
|
||||
|
||||
#[inline]
|
||||
pub fn checked_read<B: Buf, R>(b: &mut B, read: impl Fn(&mut B) -> R, size: usize) -> Result<R> {
|
||||
pub(crate) fn checked_read<B: Buf, R>(
|
||||
b: &mut B,
|
||||
read: impl Fn(&mut B) -> R,
|
||||
size: usize,
|
||||
) -> Result<R> {
|
||||
if b.remaining() < size {
|
||||
Err(Error::IO("Not enough bytes in buffer to build object."))?;
|
||||
}
|
||||
|
||||
Err(Error::IO("Not enough bytes in buffer to build object."))
|
||||
} else {
|
||||
Ok(read(b))
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn checked_write_primitive<B: BufMut, T: Sized>(
|
||||
pub(crate) fn checked_write_primitive<B: BufMut, T: Sized>(
|
||||
b: &mut B,
|
||||
write: impl Fn(&mut B, T),
|
||||
t: T,
|
||||
|
@ -29,16 +33,16 @@ pub fn checked_write_primitive<B: BufMut, T: Sized>(
|
|||
}
|
||||
|
||||
#[inline]
|
||||
pub fn checked_write<B: BufMut, T>(
|
||||
pub(crate) fn checked_write<B: BufMut, T>(
|
||||
b: &mut B,
|
||||
write: impl Fn(&mut B, T),
|
||||
t: T,
|
||||
size: usize,
|
||||
) -> Result<()> {
|
||||
if b.remaining_mut() < size {
|
||||
Err(Error::IO("Not enough capacity to write object."))?;
|
||||
}
|
||||
|
||||
Err(Error::IO("Not enough capacity to write object."))
|
||||
} else {
|
||||
write(b, t);
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
|
|
@ -59,9 +59,12 @@
|
|||
//!
|
||||
//! ```
|
||||
|
||||
#[cfg(test)]
|
||||
use hex as _;
|
||||
|
||||
extern crate alloc;
|
||||
|
||||
use core::{ops::Deref, str::from_utf8 as str_from_utf8};
|
||||
use core::str::from_utf8 as str_from_utf8;
|
||||
|
||||
use bytes::{Buf, BufMut, Bytes, BytesMut};
|
||||
|
||||
|
@ -130,7 +133,7 @@ pub fn to_bytes<T: EpeeObject>(val: T) -> Result<BytesMut> {
|
|||
fn read_header<B: Buf>(r: &mut B) -> Result<()> {
|
||||
let buf = checked_read(r, |b: &mut B| b.copy_to_bytes(HEADER.len()), HEADER.len())?;
|
||||
|
||||
if buf.deref() != HEADER {
|
||||
if &*buf != HEADER {
|
||||
return Err(Error::Format("Data does not contain header"));
|
||||
}
|
||||
Ok(())
|
||||
|
@ -185,7 +188,7 @@ fn read_object<T: EpeeObject, B: Buf>(r: &mut B, skipped_objects: &mut u8) -> Re
|
|||
|
||||
for _ in 0..number_o_field {
|
||||
let field_name_bytes = read_field_name_bytes(r)?;
|
||||
let field_name = str_from_utf8(field_name_bytes.deref())?;
|
||||
let field_name = str_from_utf8(&field_name_bytes)?;
|
||||
|
||||
if !object_builder.add_field(field_name, r)? {
|
||||
skip_epee_value(r, skipped_objects)?;
|
||||
|
@ -289,7 +292,7 @@ where
|
|||
B: BufMut,
|
||||
{
|
||||
write_varint(usize_to_u64(iterator.len()), w)?;
|
||||
for item in iterator.into_iter() {
|
||||
for item in iterator {
|
||||
item.write(w)?;
|
||||
}
|
||||
Ok(())
|
||||
|
@ -329,10 +332,7 @@ impl EpeeObject for SkipObject {
|
|||
fn skip_epee_value<B: Buf>(r: &mut B, skipped_objects: &mut u8) -> Result<()> {
|
||||
let marker = read_marker(r)?;
|
||||
|
||||
let mut len = 1;
|
||||
if marker.is_seq {
|
||||
len = read_varint(r)?;
|
||||
}
|
||||
let len = if marker.is_seq { read_varint(r)? } else { 1 };
|
||||
|
||||
if let Some(size) = marker.inner_marker.size() {
|
||||
let bytes_to_skip = size
|
||||
|
|
|
@ -19,13 +19,13 @@ pub enum InnerMarker {
|
|||
}
|
||||
|
||||
impl InnerMarker {
|
||||
pub fn size(&self) -> Option<usize> {
|
||||
pub const fn size(&self) -> Option<usize> {
|
||||
Some(match self {
|
||||
InnerMarker::I64 | InnerMarker::U64 | InnerMarker::F64 => 8,
|
||||
InnerMarker::I32 | InnerMarker::U32 => 4,
|
||||
InnerMarker::I16 | InnerMarker::U16 => 2,
|
||||
InnerMarker::I8 | InnerMarker::U8 | InnerMarker::Bool => 1,
|
||||
InnerMarker::String | InnerMarker::Object => return None,
|
||||
Self::I64 | Self::U64 | Self::F64 => 8,
|
||||
Self::I32 | Self::U32 => 4,
|
||||
Self::I16 | Self::U16 => 2,
|
||||
Self::I8 | Self::U8 | Self::Bool => 1,
|
||||
Self::String | Self::Object => return None,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
@ -40,23 +40,23 @@ pub struct Marker {
|
|||
|
||||
impl Marker {
|
||||
pub(crate) const fn new(inner_marker: InnerMarker) -> Self {
|
||||
Marker {
|
||||
Self {
|
||||
inner_marker,
|
||||
is_seq: false,
|
||||
}
|
||||
}
|
||||
|
||||
#[must_use]
|
||||
pub const fn into_seq(self) -> Self {
|
||||
if self.is_seq {
|
||||
panic!("Sequence of sequence not allowed!");
|
||||
}
|
||||
assert!(!self.is_seq, "Sequence of sequence not allowed!");
|
||||
if matches!(self.inner_marker, InnerMarker::U8) {
|
||||
return Marker {
|
||||
return Self {
|
||||
inner_marker: InnerMarker::String,
|
||||
is_seq: false,
|
||||
};
|
||||
}
|
||||
|
||||
Marker {
|
||||
Self {
|
||||
inner_marker: self.inner_marker,
|
||||
is_seq: true,
|
||||
}
|
||||
|
@ -112,7 +112,7 @@ impl TryFrom<u8> for Marker {
|
|||
_ => return Err(Error::Format("Unknown value Marker")),
|
||||
};
|
||||
|
||||
Ok(Marker {
|
||||
Ok(Self {
|
||||
inner_marker,
|
||||
is_seq,
|
||||
})
|
||||
|
|
|
@ -71,7 +71,7 @@ impl<T: EpeeObject> EpeeValue for Vec<T> {
|
|||
|
||||
let individual_marker = Marker::new(marker.inner_marker);
|
||||
|
||||
let mut res = Vec::with_capacity(len);
|
||||
let mut res = Self::with_capacity(len);
|
||||
for _ in 0..len {
|
||||
res.push(T::read(r, &individual_marker)?);
|
||||
}
|
||||
|
@ -83,7 +83,7 @@ impl<T: EpeeObject> EpeeValue for Vec<T> {
|
|||
}
|
||||
|
||||
fn epee_default_value() -> Option<Self> {
|
||||
Some(Vec::new())
|
||||
Some(Self::new())
|
||||
}
|
||||
|
||||
fn write<B: BufMut>(self, w: &mut B) -> Result<()> {
|
||||
|
@ -181,7 +181,7 @@ impl EpeeValue for Vec<u8> {
|
|||
}
|
||||
|
||||
fn epee_default_value() -> Option<Self> {
|
||||
Some(Vec::new())
|
||||
Some(Self::new())
|
||||
}
|
||||
|
||||
fn should_write(&self) -> bool {
|
||||
|
@ -216,7 +216,7 @@ impl EpeeValue for Bytes {
|
|||
}
|
||||
|
||||
fn epee_default_value() -> Option<Self> {
|
||||
Some(Bytes::new())
|
||||
Some(Self::new())
|
||||
}
|
||||
|
||||
fn should_write(&self) -> bool {
|
||||
|
@ -247,14 +247,14 @@ impl EpeeValue for BytesMut {
|
|||
return Err(Error::IO("Not enough bytes to fill object"));
|
||||
}
|
||||
|
||||
let mut bytes = BytesMut::zeroed(len);
|
||||
let mut bytes = Self::zeroed(len);
|
||||
r.copy_to_slice(&mut bytes);
|
||||
|
||||
Ok(bytes)
|
||||
}
|
||||
|
||||
fn epee_default_value() -> Option<Self> {
|
||||
Some(BytesMut::new())
|
||||
Some(Self::new())
|
||||
}
|
||||
|
||||
fn should_write(&self) -> bool {
|
||||
|
@ -285,12 +285,11 @@ impl<const N: usize> EpeeValue for ByteArrayVec<N> {
|
|||
return Err(Error::IO("Not enough bytes to fill object"));
|
||||
}
|
||||
|
||||
ByteArrayVec::try_from(r.copy_to_bytes(len))
|
||||
.map_err(|_| Error::Format("Field has invalid length"))
|
||||
Self::try_from(r.copy_to_bytes(len)).map_err(|_| Error::Format("Field has invalid length"))
|
||||
}
|
||||
|
||||
fn epee_default_value() -> Option<Self> {
|
||||
Some(ByteArrayVec::try_from(Bytes::new()).unwrap())
|
||||
Some(Self::try_from(Bytes::new()).unwrap())
|
||||
}
|
||||
|
||||
fn should_write(&self) -> bool {
|
||||
|
@ -320,8 +319,7 @@ impl<const N: usize> EpeeValue for ByteArray<N> {
|
|||
return Err(Error::IO("Not enough bytes to fill object"));
|
||||
}
|
||||
|
||||
ByteArray::try_from(r.copy_to_bytes(N))
|
||||
.map_err(|_| Error::Format("Field has invalid length"))
|
||||
Self::try_from(r.copy_to_bytes(N)).map_err(|_| Error::Format("Field has invalid length"))
|
||||
}
|
||||
|
||||
fn write<B: BufMut>(self, w: &mut B) -> Result<()> {
|
||||
|
@ -335,7 +333,7 @@ impl EpeeValue for String {
|
|||
|
||||
fn read<B: Buf>(r: &mut B, marker: &Marker) -> Result<Self> {
|
||||
let bytes = Vec::<u8>::read(r, marker)?;
|
||||
String::from_utf8(bytes).map_err(|_| Error::Format("Invalid string"))
|
||||
Self::from_utf8(bytes).map_err(|_| Error::Format("Invalid string"))
|
||||
}
|
||||
|
||||
fn should_write(&self) -> bool {
|
||||
|
@ -343,7 +341,7 @@ impl EpeeValue for String {
|
|||
}
|
||||
|
||||
fn epee_default_value() -> Option<Self> {
|
||||
Some(String::new())
|
||||
Some(Self::new())
|
||||
}
|
||||
|
||||
fn write<B: BufMut>(self, w: &mut B) -> Result<()> {
|
||||
|
@ -383,7 +381,7 @@ impl<const N: usize> EpeeValue for Vec<[u8; N]> {
|
|||
|
||||
let individual_marker = Marker::new(marker.inner_marker);
|
||||
|
||||
let mut res = Vec::with_capacity(len);
|
||||
let mut res = Self::with_capacity(len);
|
||||
for _ in 0..len {
|
||||
res.push(<[u8; N]>::read(r, &individual_marker)?);
|
||||
}
|
||||
|
@ -395,7 +393,7 @@ impl<const N: usize> EpeeValue for Vec<[u8; N]> {
|
|||
}
|
||||
|
||||
fn epee_default_value() -> Option<Self> {
|
||||
Some(Vec::new())
|
||||
Some(Self::new())
|
||||
}
|
||||
|
||||
fn write<B: BufMut>(self, w: &mut B) -> Result<()> {
|
||||
|
|
|
@ -21,14 +21,14 @@ const FITS_IN_FOUR_BYTES: u64 = 2_u64.pow(32 - SIZE_OF_SIZE_MARKER) - 1;
|
|||
/// ```
|
||||
pub fn read_varint<B: Buf>(r: &mut B) -> Result<u64> {
|
||||
if !r.has_remaining() {
|
||||
Err(Error::IO("Not enough bytes to build VarInt"))?
|
||||
return Err(Error::IO("Not enough bytes to build VarInt"));
|
||||
}
|
||||
|
||||
let vi_start = r.get_u8();
|
||||
let len = 1 << (vi_start & 0b11);
|
||||
|
||||
if r.remaining() < len - 1 {
|
||||
Err(Error::IO("Not enough bytes to build VarInt"))?
|
||||
return Err(Error::IO("Not enough bytes to build VarInt"));
|
||||
}
|
||||
|
||||
let mut vi = u64::from(vi_start >> 2);
|
||||
|
@ -67,12 +67,15 @@ pub fn write_varint<B: BufMut>(number: u64, w: &mut B) -> Result<()> {
|
|||
};
|
||||
|
||||
if w.remaining_mut() < 1 << size_marker {
|
||||
Err(Error::IO("Not enough capacity to write VarInt"))?;
|
||||
return Err(Error::IO("Not enough capacity to write VarInt"));
|
||||
}
|
||||
|
||||
let number = (number << 2) | size_marker;
|
||||
|
||||
// Although `as` is unsafe we just checked the length.
|
||||
#[expect(
|
||||
clippy::cast_possible_truncation,
|
||||
reason = "Although `as` is unsafe we just checked the length."
|
||||
)]
|
||||
match size_marker {
|
||||
0 => w.put_u8(number as u8),
|
||||
1 => w.put_u16_le(number as u16),
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "outer test module")]
|
||||
|
||||
use cuprate_epee_encoding::{epee_object, from_bytes, to_bytes};
|
||||
|
||||
struct AltName {
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "outer test module")]
|
||||
|
||||
use cuprate_epee_encoding::{epee_object, from_bytes};
|
||||
|
||||
struct T {
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "outer test module")]
|
||||
|
||||
use cuprate_epee_encoding::{epee_object, from_bytes, to_bytes};
|
||||
|
||||
pub struct Optional {
|
||||
|
@ -58,7 +60,7 @@ fn epee_non_default_does_encode() {
|
|||
|
||||
let val: Optional = from_bytes(&mut bytes).unwrap();
|
||||
assert_eq!(val.optional_val, -3);
|
||||
assert_eq!(val.val, 8)
|
||||
assert_eq!(val.val, 8);
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
@ -70,5 +72,5 @@ fn epee_value_not_present_with_default() {
|
|||
|
||||
let val: Optional = from_bytes(&mut bytes).unwrap();
|
||||
assert_eq!(val.optional_val, -4);
|
||||
assert_eq!(val.val, 76)
|
||||
assert_eq!(val.val, 76);
|
||||
}
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "outer test module")]
|
||||
|
||||
use cuprate_epee_encoding::{epee_object, from_bytes, to_bytes};
|
||||
|
||||
struct Child {
|
||||
|
@ -37,6 +39,7 @@ epee_object!(
|
|||
);
|
||||
|
||||
#[test]
|
||||
#[expect(clippy::float_cmp)]
|
||||
fn epee_flatten() {
|
||||
let val2 = ParentChild {
|
||||
h: 38.9,
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
#![expect(unused_crate_dependencies, reason = "outer test module")]
|
||||
|
||||
use cuprate_epee_encoding::{epee_object, from_bytes, to_bytes};
|
||||
use std::ops::Deref;
|
||||
|
||||
#[derive(Clone)]
|
||||
struct T {
|
||||
|
@ -28,6 +29,6 @@ fn optional_val_in_data() {
|
|||
];
|
||||
let t: T = from_bytes(&mut &bytes[..]).unwrap();
|
||||
let bytes2 = to_bytes(t.clone()).unwrap();
|
||||
assert_eq!(bytes.as_slice(), bytes2.deref());
|
||||
assert_eq!(bytes.as_slice(), &*bytes2);
|
||||
assert_eq!(t.val.unwrap(), 21);
|
||||
}
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "outer test module")]
|
||||
|
||||
use cuprate_epee_encoding::{epee_object, from_bytes, to_bytes};
|
||||
|
||||
#[derive(Eq, PartialEq, Debug, Clone)]
|
||||
|
@ -5,7 +7,7 @@ pub struct SupportFlags(u32);
|
|||
|
||||
impl From<u32> for SupportFlags {
|
||||
fn from(value: u32) -> Self {
|
||||
SupportFlags(value)
|
||||
Self(value)
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "outer test module")]
|
||||
|
||||
use cuprate_epee_encoding::{epee_object, from_bytes, to_bytes};
|
||||
|
||||
#[derive(Clone, Debug, PartialEq)]
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "outer test module")]
|
||||
|
||||
use cuprate_epee_encoding::{epee_object, from_bytes};
|
||||
|
||||
struct ObjSeq {
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "outer test module")]
|
||||
|
||||
use cuprate_epee_encoding::{epee_object, from_bytes};
|
||||
|
||||
struct D {
|
||||
|
@ -737,5 +739,5 @@ fn stack_overflow() {
|
|||
|
||||
let obj: Result<Q, _> = from_bytes(&mut bytes.as_slice());
|
||||
|
||||
assert!(obj.is_err())
|
||||
assert!(obj.is_err());
|
||||
}
|
||||
|
|
|
@ -14,6 +14,7 @@ tracing = ["dep:tracing", "tokio-util/tracing"]
|
|||
[dependencies]
|
||||
cuprate-helper = { path = "../../helper", default-features = false, features = ["cast"] }
|
||||
|
||||
cfg-if = { workspace = true }
|
||||
thiserror = { workspace = true }
|
||||
bytes = { workspace = true, features = ["std"] }
|
||||
bitflags = { workspace = true }
|
||||
|
@ -27,3 +28,6 @@ rand = { workspace = true, features = ["std", "std_rng"] }
|
|||
tokio-util = { workspace = true, features = ["io-util"]}
|
||||
tokio = { workspace = true, features = ["full"] }
|
||||
futures = { workspace = true, features = ["std"] }
|
||||
|
||||
[lints]
|
||||
workspace = true
|
|
@ -47,7 +47,7 @@ pub struct LevinBucketCodec<C> {
|
|||
|
||||
impl<C> Default for LevinBucketCodec<C> {
|
||||
fn default() -> Self {
|
||||
LevinBucketCodec {
|
||||
Self {
|
||||
state: LevinBucketState::WaitingForHeader,
|
||||
protocol: Protocol::default(),
|
||||
handshake_message_seen: false,
|
||||
|
@ -56,8 +56,8 @@ impl<C> Default for LevinBucketCodec<C> {
|
|||
}
|
||||
|
||||
impl<C> LevinBucketCodec<C> {
|
||||
pub fn new(protocol: Protocol) -> Self {
|
||||
LevinBucketCodec {
|
||||
pub const fn new(protocol: Protocol) -> Self {
|
||||
Self {
|
||||
state: LevinBucketState::WaitingForHeader,
|
||||
protocol,
|
||||
handshake_message_seen: false,
|
||||
|
@ -112,8 +112,10 @@ impl<C: LevinCommand + Debug> Decoder for LevinBucketCodec<C> {
|
|||
}
|
||||
}
|
||||
|
||||
let _ =
|
||||
std::mem::replace(&mut self.state, LevinBucketState::WaitingForBody(head));
|
||||
drop(std::mem::replace(
|
||||
&mut self.state,
|
||||
LevinBucketState::WaitingForBody(head),
|
||||
));
|
||||
}
|
||||
LevinBucketState::WaitingForBody(head) => {
|
||||
let body_len = u64_to_usize(head.size);
|
||||
|
@ -145,7 +147,7 @@ impl<C: LevinCommand> Encoder<Bucket<C>> for LevinBucketCodec<C> {
|
|||
type Error = BucketError;
|
||||
fn encode(&mut self, item: Bucket<C>, dst: &mut BytesMut) -> Result<(), Self::Error> {
|
||||
if let Some(additional) = (HEADER_SIZE + item.body.len()).checked_sub(dst.capacity()) {
|
||||
dst.reserve(additional)
|
||||
dst.reserve(additional);
|
||||
}
|
||||
|
||||
item.header.write_bytes_into(dst);
|
||||
|
|
|
@ -13,7 +13,7 @@
|
|||
// copies or substantial portions of the Software.
|
||||
//
|
||||
|
||||
//! This module provides a struct BucketHead for the header of a levin protocol
|
||||
//! This module provides a struct `BucketHead` for the header of a levin protocol
|
||||
//! message.
|
||||
|
||||
use bitflags::bitflags;
|
||||
|
@ -62,7 +62,7 @@ bitflags! {
|
|||
|
||||
impl From<u32> for Flags {
|
||||
fn from(value: u32) -> Self {
|
||||
Flags(value)
|
||||
Self(value)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -99,9 +99,9 @@ impl<C: LevinCommand> BucketHead<C> {
|
|||
///
|
||||
/// # Panics
|
||||
/// This function will panic if there aren't enough bytes to fill the header.
|
||||
/// Currently [HEADER_SIZE]
|
||||
pub fn from_bytes(buf: &mut BytesMut) -> BucketHead<C> {
|
||||
BucketHead {
|
||||
/// Currently [`HEADER_SIZE`]
|
||||
pub fn from_bytes(buf: &mut BytesMut) -> Self {
|
||||
Self {
|
||||
signature: buf.get_u64_le(),
|
||||
size: buf.get_u64_le(),
|
||||
have_to_return_data: buf.get_u8() != 0,
|
||||
|
|
|
@ -33,6 +33,16 @@
|
|||
#![deny(unused_mut)]
|
||||
//#![deny(missing_docs)]
|
||||
|
||||
cfg_if::cfg_if! {
|
||||
// Used in `tests/`.
|
||||
if #[cfg(test)] {
|
||||
use futures as _;
|
||||
use proptest as _;
|
||||
use rand as _;
|
||||
use tokio as _;
|
||||
}
|
||||
}
|
||||
|
||||
use std::fmt::Debug;
|
||||
|
||||
use bytes::{Buf, Bytes};
|
||||
|
@ -99,7 +109,7 @@ pub struct Protocol {
|
|||
|
||||
impl Default for Protocol {
|
||||
fn default() -> Self {
|
||||
Protocol {
|
||||
Self {
|
||||
version: MONERO_PROTOCOL_VERSION,
|
||||
signature: MONERO_LEVIN_SIGNATURE,
|
||||
max_packet_size_before_handshake: MONERO_MAX_PACKET_SIZE_BEFORE_HANDSHAKE,
|
||||
|
@ -130,22 +140,22 @@ pub enum MessageType {
|
|||
|
||||
impl MessageType {
|
||||
/// Returns if the message requires a response
|
||||
pub fn have_to_return_data(&self) -> bool {
|
||||
pub const fn have_to_return_data(&self) -> bool {
|
||||
match self {
|
||||
MessageType::Request => true,
|
||||
MessageType::Response | MessageType::Notification => false,
|
||||
Self::Request => true,
|
||||
Self::Response | Self::Notification => false,
|
||||
}
|
||||
}
|
||||
|
||||
/// Returns the `MessageType` given the flags and have_to_return_data fields
|
||||
pub fn from_flags_and_have_to_return(
|
||||
/// Returns the `MessageType` given the flags and `have_to_return_data` fields
|
||||
pub const fn from_flags_and_have_to_return(
|
||||
flags: Flags,
|
||||
have_to_return: bool,
|
||||
) -> Result<Self, BucketError> {
|
||||
Ok(match (flags, have_to_return) {
|
||||
(Flags::REQUEST, true) => MessageType::Request,
|
||||
(Flags::REQUEST, false) => MessageType::Notification,
|
||||
(Flags::RESPONSE, false) => MessageType::Response,
|
||||
(Flags::REQUEST, true) => Self::Request,
|
||||
(Flags::REQUEST, false) => Self::Notification,
|
||||
(Flags::RESPONSE, false) => Self::Response,
|
||||
_ => {
|
||||
return Err(BucketError::InvalidHeaderFlags(
|
||||
"Unable to assign a message type to this bucket",
|
||||
|
@ -154,10 +164,10 @@ impl MessageType {
|
|||
})
|
||||
}
|
||||
|
||||
pub fn as_flags(&self) -> header::Flags {
|
||||
pub const fn as_flags(&self) -> Flags {
|
||||
match self {
|
||||
MessageType::Request | MessageType::Notification => header::Flags::REQUEST,
|
||||
MessageType::Response => header::Flags::RESPONSE,
|
||||
Self::Request | Self::Notification => Flags::REQUEST,
|
||||
Self::Response => Flags::RESPONSE,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -173,7 +183,7 @@ pub struct BucketBuilder<C> {
|
|||
}
|
||||
|
||||
impl<C: LevinCommand> BucketBuilder<C> {
|
||||
pub fn new(protocol: &Protocol) -> Self {
|
||||
pub const fn new(protocol: &Protocol) -> Self {
|
||||
Self {
|
||||
signature: Some(protocol.signature),
|
||||
ty: None,
|
||||
|
@ -185,27 +195,27 @@ impl<C: LevinCommand> BucketBuilder<C> {
|
|||
}
|
||||
|
||||
pub fn set_signature(&mut self, sig: u64) {
|
||||
self.signature = Some(sig)
|
||||
self.signature = Some(sig);
|
||||
}
|
||||
|
||||
pub fn set_message_type(&mut self, ty: MessageType) {
|
||||
self.ty = Some(ty)
|
||||
self.ty = Some(ty);
|
||||
}
|
||||
|
||||
pub fn set_command(&mut self, command: C) {
|
||||
self.command = Some(command)
|
||||
self.command = Some(command);
|
||||
}
|
||||
|
||||
pub fn set_return_code(&mut self, code: i32) {
|
||||
self.return_code = Some(code)
|
||||
self.return_code = Some(code);
|
||||
}
|
||||
|
||||
pub fn set_protocol_version(&mut self, version: u32) {
|
||||
self.protocol_version = Some(version)
|
||||
self.protocol_version = Some(version);
|
||||
}
|
||||
|
||||
pub fn set_body(&mut self, body: Bytes) {
|
||||
self.body = Some(body)
|
||||
self.body = Some(body);
|
||||
}
|
||||
|
||||
pub fn finish(self) -> Bucket<C> {
|
||||
|
|
|
@ -33,13 +33,13 @@ pub enum LevinMessage<T: LevinBody> {
|
|||
|
||||
impl<T: LevinBody> From<T> for LevinMessage<T> {
|
||||
fn from(value: T) -> Self {
|
||||
LevinMessage::Body(value)
|
||||
Self::Body(value)
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: LevinBody> From<Bucket<T::Command>> for LevinMessage<T> {
|
||||
fn from(value: Bucket<T::Command>) -> Self {
|
||||
LevinMessage::Bucket(value)
|
||||
Self::Bucket(value)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -58,7 +58,7 @@ pub struct Dummy(pub usize);
|
|||
|
||||
impl<T: LevinBody> From<Dummy> for LevinMessage<T> {
|
||||
fn from(value: Dummy) -> Self {
|
||||
LevinMessage::Dummy(value.0)
|
||||
Self::Dummy(value.0)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -76,12 +76,11 @@ pub fn make_fragmented_messages<T: LevinBody>(
|
|||
fragment_size: usize,
|
||||
message: T,
|
||||
) -> Result<Vec<Bucket<T::Command>>, BucketError> {
|
||||
if fragment_size * 2 < HEADER_SIZE {
|
||||
panic!(
|
||||
assert!(
|
||||
fragment_size * 2 >= HEADER_SIZE,
|
||||
"Fragment size: {fragment_size}, is too small, must be at least {}",
|
||||
2 * HEADER_SIZE
|
||||
);
|
||||
}
|
||||
|
||||
let mut builder = BucketBuilder::new(protocol);
|
||||
message.encode(&mut builder)?;
|
||||
|
|
|
@ -1,3 +1,9 @@
|
|||
#![expect(
|
||||
clippy::tests_outside_test_module,
|
||||
unused_crate_dependencies,
|
||||
reason = "outer test module"
|
||||
)]
|
||||
|
||||
use bytes::{Buf, BufMut, Bytes, BytesMut};
|
||||
use futures::{SinkExt, StreamExt};
|
||||
use proptest::{prelude::any_with, prop_assert_eq, proptest, sample::size_range};
|
||||
|
@ -58,12 +64,12 @@ impl LevinBody for TestBody {
|
|||
) -> Result<Self, BucketError> {
|
||||
let size = u64_to_usize(body.get_u64_le());
|
||||
// bucket
|
||||
Ok(TestBody::Bytes(size, body.copy_to_bytes(size)))
|
||||
Ok(Self::Bytes(size, body.copy_to_bytes(size)))
|
||||
}
|
||||
|
||||
fn encode(self, builder: &mut BucketBuilder<Self::Command>) -> Result<(), BucketError> {
|
||||
match self {
|
||||
TestBody::Bytes(len, bytes) => {
|
||||
Self::Bytes(len, bytes) => {
|
||||
let mut buf = BytesMut::new();
|
||||
buf.put_u64_le(len as u64);
|
||||
buf.extend_from_slice(bytes.as_ref());
|
||||
|
@ -141,12 +147,12 @@ proptest! {
|
|||
message2.extend_from_slice(&fragments[0].body[(33 + 8)..]);
|
||||
|
||||
for frag in fragments.iter().skip(1) {
|
||||
message2.extend_from_slice(frag.body.as_ref())
|
||||
message2.extend_from_slice(frag.body.as_ref());
|
||||
}
|
||||
|
||||
prop_assert_eq!(message.as_slice(), &message2[0..message.len()], "numb_fragments: {}", fragments.len());
|
||||
|
||||
for byte in message2[message.len()..].iter(){
|
||||
for byte in &message2[message.len()..]{
|
||||
prop_assert_eq!(*byte, 0);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -25,3 +25,6 @@ thiserror = { workspace = true }
|
|||
[dev-dependencies]
|
||||
tokio = { workspace = true, features = ["rt-multi-thread", "macros", "sync"] }
|
||||
proptest = { workspace = true, features = ["default"] }
|
||||
|
||||
[lints]
|
||||
workspace = true
|
|
@ -8,7 +8,7 @@ use std::{
|
|||
/// (1 - ep) is the probability that a transaction travels for `k` hops before a nodes embargo timeout fires, this constant is (1 - ep).
|
||||
const EMBARGO_FULL_TRAVEL_PROBABILITY: f64 = 0.90;
|
||||
|
||||
/// The graph type to use for dandelion routing, the dandelion paper recommends [Graph::FourRegular].
|
||||
/// The graph type to use for dandelion routing, the dandelion paper recommends [`Graph::FourRegular`].
|
||||
///
|
||||
/// The decision between line graphs and 4-regular graphs depend on the priorities of the system, if
|
||||
/// linkability of transactions is a first order concern then line graphs may be better, however 4-regular graphs
|
||||
|
@ -66,7 +66,7 @@ impl DandelionConfig {
|
|||
/// Returns the number of outbound peers to use to stem transactions.
|
||||
///
|
||||
/// This value depends on the [`Graph`] chosen.
|
||||
pub fn number_of_stems(&self) -> usize {
|
||||
pub const fn number_of_stems(&self) -> usize {
|
||||
match self.graph {
|
||||
Graph::Line => 1,
|
||||
Graph::FourRegular => 2,
|
||||
|
|
|
@ -26,7 +26,7 @@
|
|||
//! The diffuse service should have a request of [`DiffuseRequest`](traits::DiffuseRequest) and it's error
|
||||
//! should be [`tower::BoxError`].
|
||||
//!
|
||||
//! ## Outbound Peer TryStream
|
||||
//! ## Outbound Peer `TryStream`
|
||||
//!
|
||||
//! The outbound peer [`TryStream`](futures::TryStream) should provide a stream of randomly selected outbound
|
||||
//! peers, these peers will then be used to route stem txs to.
|
||||
|
@ -37,7 +37,7 @@
|
|||
//! ## Peer Service
|
||||
//!
|
||||
//! This service represents a connection to an individual peer, this should be returned from the Outbound Peer
|
||||
//! TryStream. This should immediately send the transaction to the peer when requested, it should _not_ set
|
||||
//! `TryStream`. This should immediately send the transaction to the peer when requested, it should _not_ set
|
||||
//! a timer.
|
||||
//!
|
||||
//! The peer service should have a request of [`StemRequest`](traits::StemRequest) and its error
|
||||
|
|
|
@ -30,7 +30,7 @@ pub struct IncomingTxBuilder<const RS: bool, const DBS: bool, Tx, TxId, PeerId>
|
|||
|
||||
impl<Tx, TxId, PeerId> IncomingTxBuilder<false, false, Tx, TxId, PeerId> {
|
||||
/// Creates a new [`IncomingTxBuilder`].
|
||||
pub fn new(tx: Tx, tx_id: TxId) -> Self {
|
||||
pub const fn new(tx: Tx, tx_id: TxId) -> Self {
|
||||
Self {
|
||||
tx,
|
||||
tx_id,
|
||||
|
|
|
@ -88,9 +88,7 @@ where
|
|||
.insert(peer.clone());
|
||||
}
|
||||
|
||||
let state = from
|
||||
.map(|from| TxState::Stem { from })
|
||||
.unwrap_or(TxState::Local);
|
||||
let state = from.map_or(TxState::Local, |from| TxState::Stem { from });
|
||||
|
||||
let fut = self
|
||||
.dandelion_router
|
||||
|
@ -280,13 +278,15 @@ where
|
|||
};
|
||||
|
||||
if let Err(e) = self.handle_incoming_tx(tx, routing_state, tx_id).await {
|
||||
#[expect(clippy::let_underscore_must_use, reason = "dropped receivers can be ignored")]
|
||||
let _ = res_tx.send(());
|
||||
|
||||
tracing::error!("Error handling transaction in dandelion pool: {e}");
|
||||
return;
|
||||
}
|
||||
let _ = res_tx.send(());
|
||||
|
||||
#[expect(clippy::let_underscore_must_use)]
|
||||
let _ = res_tx.send(());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -140,7 +140,7 @@ where
|
|||
State::Stem
|
||||
};
|
||||
|
||||
DandelionRouter {
|
||||
Self {
|
||||
outbound_peer_discover: Box::pin(outbound_peer_discover),
|
||||
broadcast_svc,
|
||||
current_state,
|
||||
|
@ -198,7 +198,7 @@ where
|
|||
fn stem_tx(
|
||||
&mut self,
|
||||
tx: Tx,
|
||||
from: Id,
|
||||
from: &Id,
|
||||
) -> BoxFuture<'static, Result<State, DandelionRouterError>> {
|
||||
if self.stem_peers.is_empty() {
|
||||
tracing::debug!("Stem peers are empty, fluffing stem transaction.");
|
||||
|
@ -216,7 +216,7 @@ where
|
|||
});
|
||||
|
||||
let Some(peer) = self.stem_peers.get_mut(stem_route) else {
|
||||
self.stem_routes.remove(&from);
|
||||
self.stem_routes.remove(from);
|
||||
continue;
|
||||
};
|
||||
|
||||
|
@ -302,7 +302,7 @@ where
|
|||
tracing::debug!(
|
||||
parent: span,
|
||||
"Peer returned an error on `poll_ready`: {e}, removing from router.",
|
||||
)
|
||||
);
|
||||
})
|
||||
.is_ok(),
|
||||
Poll::Pending => {
|
||||
|
@ -341,7 +341,7 @@ where
|
|||
State::Stem => {
|
||||
tracing::trace!(parent: &self.span, "Steming transaction");
|
||||
|
||||
self.stem_tx(req.tx, from)
|
||||
self.stem_tx(req.tx, &from)
|
||||
}
|
||||
},
|
||||
TxState::Local => {
|
||||
|
|
|
@ -12,7 +12,7 @@ use crate::{
|
|||
OutboundPeer, State,
|
||||
};
|
||||
|
||||
pub fn mock_discover_svc<Req: Send + 'static>() -> (
|
||||
pub(crate) fn mock_discover_svc<Req: Send + 'static>() -> (
|
||||
impl Stream<
|
||||
Item = Result<
|
||||
OutboundPeer<
|
||||
|
@ -49,7 +49,7 @@ pub fn mock_discover_svc<Req: Send + 'static>() -> (
|
|||
(discover, rx)
|
||||
}
|
||||
|
||||
pub fn mock_broadcast_svc<Req: Send + 'static>() -> (
|
||||
pub(crate) fn mock_broadcast_svc<Req: Send + 'static>() -> (
|
||||
impl Service<
|
||||
Req,
|
||||
Future = impl Future<Output = Result<(), tower::BoxError>> + Send + 'static,
|
||||
|
@ -70,8 +70,8 @@ pub fn mock_broadcast_svc<Req: Send + 'static>() -> (
|
|||
)
|
||||
}
|
||||
|
||||
#[allow(clippy::type_complexity)] // just test code.
|
||||
pub fn mock_in_memory_backing_pool<
|
||||
#[expect(clippy::type_complexity, reason = "just test code.")]
|
||||
pub(crate) fn mock_in_memory_backing_pool<
|
||||
Tx: Clone + Send + 'static,
|
||||
TxID: Clone + Hash + Eq + Send + 'static,
|
||||
>() -> (
|
||||
|
@ -85,11 +85,11 @@ pub fn mock_in_memory_backing_pool<
|
|||
Arc<std::sync::Mutex<HashMap<TxID, (Tx, State)>>>,
|
||||
) {
|
||||
let txs = Arc::new(std::sync::Mutex::new(HashMap::new()));
|
||||
let txs_2 = txs.clone();
|
||||
let txs_2 = Arc::clone(&txs);
|
||||
|
||||
(
|
||||
service_fn(move |req: TxStoreRequest<TxID>| {
|
||||
let txs = txs.clone();
|
||||
let txs = Arc::clone(&txs);
|
||||
async move {
|
||||
match req {
|
||||
TxStoreRequest::Get(tx_id) => {
|
||||
|
|
|
@ -39,5 +39,5 @@ async fn basic_functionality() {
|
|||
// TODO: the DandelionPoolManager doesn't handle adding txs to the pool, add more tests here to test
|
||||
// all functionality.
|
||||
//assert!(pool.lock().unwrap().contains_key(&1));
|
||||
assert!(broadcast_rx.try_recv().is_ok())
|
||||
assert!(broadcast_rx.try_recv().is_ok());
|
||||
}
|
||||
|
|
|
@ -14,13 +14,14 @@ cuprate-helper = { path = "../../helper", features = ["asynch"], default-feature
|
|||
cuprate-wire = { path = "../../net/wire", features = ["tracing"] }
|
||||
cuprate-pruning = { path = "../../pruning" }
|
||||
|
||||
tokio = { workspace = true, features = ["net", "sync", "macros", "time"]}
|
||||
tokio = { workspace = true, features = ["net", "sync", "macros", "time", "rt", "rt-multi-thread"]}
|
||||
tokio-util = { workspace = true, features = ["codec"] }
|
||||
tokio-stream = { workspace = true, features = ["sync"]}
|
||||
futures = { workspace = true, features = ["std"] }
|
||||
async-trait = { workspace = true }
|
||||
tower = { workspace = true, features = ["util", "tracing"] }
|
||||
|
||||
cfg-if = { workspace = true }
|
||||
thiserror = { workspace = true }
|
||||
tracing = { workspace = true, features = ["std", "attributes"] }
|
||||
hex-literal = { workspace = true }
|
||||
|
@ -28,9 +29,10 @@ hex-literal = { workspace = true }
|
|||
borsh = { workspace = true, features = ["derive", "std"], optional = true }
|
||||
|
||||
[dev-dependencies]
|
||||
cuprate-test-utils = {path = "../../test-utils"}
|
||||
cuprate-test-utils = { path = "../../test-utils" }
|
||||
|
||||
hex = { workspace = true, features = ["std"] }
|
||||
tokio = { workspace = true, features = ["net", "rt-multi-thread", "rt", "macros"]}
|
||||
tokio-test = { workspace = true }
|
||||
tracing-subscriber = { workspace = true }
|
||||
|
||||
[lints]
|
||||
workspace = true
|
|
@ -43,8 +43,8 @@ pub enum InternalPeerID<A> {
|
|||
impl<A: Display> Display for InternalPeerID<A> {
|
||||
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
||||
match self {
|
||||
InternalPeerID::KnownAddr(addr) => addr.fmt(f),
|
||||
InternalPeerID::Unknown(id) => f.write_str(&format!("Unknown, ID: {id}")),
|
||||
Self::KnownAddr(addr) => addr.fmt(f),
|
||||
Self::Unknown(id) => f.write_str(&format!("Unknown, ID: {id}")),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -113,7 +113,7 @@ impl<Z: NetworkZone> Client<Z> {
|
|||
fn set_err(&self, err: PeerError) -> tower::BoxError {
|
||||
let err_str = err.to_string();
|
||||
match self.error.try_insert_err(err) {
|
||||
Ok(_) => err_str,
|
||||
Ok(()) => err_str,
|
||||
Err(e) => e.to_string(),
|
||||
}
|
||||
.into()
|
||||
|
@ -169,9 +169,8 @@ impl<Z: NetworkZone> Service<PeerRequest> for Client<Z> {
|
|||
TrySendError::Closed(req) | TrySendError::Full(req) => {
|
||||
self.set_err(PeerError::ClientChannelClosed);
|
||||
|
||||
let _ = req
|
||||
.response_channel
|
||||
.send(Err(PeerError::ClientChannelClosed.into()));
|
||||
let resp = Err(PeerError::ClientChannelClosed.into());
|
||||
drop(req.response_channel.send(resp));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -216,7 +215,7 @@ where
|
|||
|
||||
tracing::debug!("Sending back response");
|
||||
|
||||
let _ = req.response_channel.send(Ok(res));
|
||||
drop(req.response_channel.send(Ok(res)));
|
||||
}
|
||||
}
|
||||
.instrument(task_span),
|
||||
|
|
|
@ -26,7 +26,7 @@ use crate::{
|
|||
};
|
||||
|
||||
/// A request to the connection task from a [`Client`](crate::client::Client).
|
||||
pub struct ConnectionTaskRequest {
|
||||
pub(crate) struct ConnectionTaskRequest {
|
||||
/// The request.
|
||||
pub request: PeerRequest,
|
||||
/// The response channel.
|
||||
|
@ -36,7 +36,7 @@ pub struct ConnectionTaskRequest {
|
|||
}
|
||||
|
||||
/// The connection state.
|
||||
pub enum State {
|
||||
pub(crate) enum State {
|
||||
/// Waiting for a request from Cuprate or the connected peer.
|
||||
WaitingForRequest,
|
||||
/// Waiting for a response from the peer.
|
||||
|
@ -53,7 +53,7 @@ pub enum State {
|
|||
/// Returns if the [`LevinCommand`] is the correct response message for our request.
|
||||
///
|
||||
/// e.g. that we didn't get a block for a txs request.
|
||||
fn levin_command_response(message_id: &MessageID, command: LevinCommand) -> bool {
|
||||
const fn levin_command_response(message_id: MessageID, command: LevinCommand) -> bool {
|
||||
matches!(
|
||||
(message_id, command),
|
||||
(MessageID::Handshake, LevinCommand::Handshake)
|
||||
|
@ -71,7 +71,7 @@ fn levin_command_response(message_id: &MessageID, command: LevinCommand) -> bool
|
|||
}
|
||||
|
||||
/// This represents a connection to a peer.
|
||||
pub struct Connection<Z: NetworkZone, A, CS, PS, PR, BrdcstStrm> {
|
||||
pub(crate) struct Connection<Z: NetworkZone, A, CS, PS, PR, BrdcstStrm> {
|
||||
/// The peer sink - where we send messages to the peer.
|
||||
peer_sink: Z::Sink,
|
||||
|
||||
|
@ -104,15 +104,15 @@ where
|
|||
BrdcstStrm: Stream<Item = BroadcastMessage> + Send + 'static,
|
||||
{
|
||||
/// Create a new connection struct.
|
||||
pub fn new(
|
||||
pub(crate) fn new(
|
||||
peer_sink: Z::Sink,
|
||||
client_rx: mpsc::Receiver<ConnectionTaskRequest>,
|
||||
broadcast_stream: BrdcstStrm,
|
||||
peer_request_handler: PeerRequestHandler<Z, A, CS, PS, PR>,
|
||||
connection_guard: ConnectionGuard,
|
||||
error: SharedError<PeerError>,
|
||||
) -> Connection<Z, A, CS, PS, PR, BrdcstStrm> {
|
||||
Connection {
|
||||
) -> Self {
|
||||
Self {
|
||||
peer_sink,
|
||||
state: State::WaitingForRequest,
|
||||
request_timeout: None,
|
||||
|
@ -174,15 +174,14 @@ where
|
|||
if let Err(e) = res {
|
||||
// can't clone the error so turn it to a string first, hacky but oh well.
|
||||
let err_str = e.to_string();
|
||||
let _ = req.response_channel.send(Err(err_str.clone().into()));
|
||||
drop(req.response_channel.send(Err(err_str.into())));
|
||||
return Err(e);
|
||||
} else {
|
||||
// We still need to respond even if the response is this.
|
||||
let _ = req
|
||||
.response_channel
|
||||
.send(Ok(PeerResponse::Protocol(ProtocolResponse::NA)));
|
||||
}
|
||||
|
||||
// We still need to respond even if the response is this.
|
||||
let resp = Ok(PeerResponse::Protocol(ProtocolResponse::NA));
|
||||
drop(req.response_channel.send(resp));
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
|
@ -215,7 +214,7 @@ where
|
|||
};
|
||||
|
||||
// Check if the message is a response to our request.
|
||||
if levin_command_response(request_id, mes.command()) {
|
||||
if levin_command_response(*request_id, mes.command()) {
|
||||
// TODO: Do more checks before returning response.
|
||||
|
||||
let State::WaitingForResponse { tx, .. } =
|
||||
|
@ -224,9 +223,11 @@ where
|
|||
panic!("Not in correct state, can't receive response!")
|
||||
};
|
||||
|
||||
let _ = tx.send(Ok(mes
|
||||
let resp = Ok(mes
|
||||
.try_into()
|
||||
.map_err(|_| PeerError::PeerSentInvalidMessage)?));
|
||||
.map_err(|_| PeerError::PeerSentInvalidMessage)?);
|
||||
|
||||
drop(tx.send(resp));
|
||||
|
||||
self.request_timeout = None;
|
||||
|
||||
|
@ -282,7 +283,7 @@ where
|
|||
|
||||
tokio::select! {
|
||||
biased;
|
||||
_ = self.request_timeout.as_mut().expect("Request timeout was not set!") => {
|
||||
() = self.request_timeout.as_mut().expect("Request timeout was not set!") => {
|
||||
Err(PeerError::ClientChannelClosed)
|
||||
}
|
||||
broadcast_req = self.broadcast_stream.next() => {
|
||||
|
@ -306,8 +307,11 @@ where
|
|||
/// Runs the Connection handler logic, this should be put in a separate task.
|
||||
///
|
||||
/// `eager_protocol_messages` are protocol messages that we received during a handshake.
|
||||
pub async fn run<Str>(mut self, mut stream: Str, eager_protocol_messages: Vec<ProtocolMessage>)
|
||||
where
|
||||
pub(crate) async fn run<Str>(
|
||||
mut self,
|
||||
mut stream: Str,
|
||||
eager_protocol_messages: Vec<ProtocolMessage>,
|
||||
) where
|
||||
Str: FusedStream<Item = Result<Message, cuprate_wire::BucketError>> + Unpin,
|
||||
{
|
||||
tracing::debug!(
|
||||
|
@ -348,6 +352,7 @@ where
|
|||
|
||||
/// Shutdowns the connection, flushing pending requests and setting the error slot, if it hasn't been
|
||||
/// set already.
|
||||
#[expect(clippy::significant_drop_tightening)]
|
||||
fn shutdown(mut self, err: PeerError) {
|
||||
tracing::debug!("Connection task shutting down: {}", err);
|
||||
|
||||
|
@ -362,11 +367,11 @@ where
|
|||
if let State::WaitingForResponse { tx, .. } =
|
||||
std::mem::replace(&mut self.state, State::WaitingForRequest)
|
||||
{
|
||||
let _ = tx.send(Err(err_str.clone().into()));
|
||||
drop(tx.send(Err(err_str.clone().into())));
|
||||
}
|
||||
|
||||
while let Ok(req) = client_rx.try_recv() {
|
||||
let _ = req.response_channel.send(Err(err_str.clone().into()));
|
||||
drop(req.response_channel.send(Err(err_str.clone().into())));
|
||||
}
|
||||
|
||||
self.connection_guard.connection_closed();
|
||||
|
|
|
@ -40,7 +40,9 @@ impl<Z: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
Connector<Z, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
||||
{
|
||||
/// Create a new connector from a handshaker.
|
||||
pub fn new(handshaker: HandShaker<Z, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>) -> Self {
|
||||
pub const fn new(
|
||||
handshaker: HandShaker<Z, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>,
|
||||
) -> Self {
|
||||
Self { handshaker }
|
||||
}
|
||||
}
|
||||
|
|
|
@ -113,7 +113,7 @@ impl<Z: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
HandShaker<Z, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
||||
{
|
||||
/// Creates a new handshaker.
|
||||
fn new(
|
||||
const fn new(
|
||||
address_book: AdrBook,
|
||||
peer_sync_svc: PSync,
|
||||
core_sync_svc: CSync,
|
||||
|
@ -226,11 +226,12 @@ pub async fn ping<N: NetworkZone>(addr: N::Addr) -> Result<u64, HandshakeError>
|
|||
Err(BucketError::IO(std::io::Error::new(
|
||||
std::io::ErrorKind::ConnectionAborted,
|
||||
"The peer stream returned None",
|
||||
)))?
|
||||
))
|
||||
.into())
|
||||
}
|
||||
|
||||
/// This function completes a handshake with the requested peer.
|
||||
#[allow(clippy::too_many_arguments)]
|
||||
#[expect(clippy::too_many_arguments)]
|
||||
async fn handshake<Z: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr, BrdcstStrm>(
|
||||
req: DoHandshakeRequest<Z>,
|
||||
|
||||
|
@ -403,7 +404,10 @@ where
|
|||
break 'check_out_addr None;
|
||||
};
|
||||
|
||||
// u32 does not make sense as a port so just truncate it.
|
||||
#[expect(
|
||||
clippy::cast_possible_truncation,
|
||||
reason = "u32 does not make sense as a port so just truncate it."
|
||||
)]
|
||||
outbound_address.set_port(peer_node_data.my_port as u16);
|
||||
|
||||
let Ok(Ok(ping_peer_id)) = timeout(
|
||||
|
@ -508,7 +512,7 @@ where
|
|||
info.id,
|
||||
info.handle.clone(),
|
||||
connection_tx.clone(),
|
||||
semaphore.clone(),
|
||||
Arc::clone(&semaphore),
|
||||
address_book,
|
||||
core_sync_svc,
|
||||
peer_sync_svc,
|
||||
|
@ -671,7 +675,7 @@ async fn wait_for_message<Z: NetworkZone>(
|
|||
_ => {
|
||||
return Err(HandshakeError::PeerSentInvalidMessage(
|
||||
"Peer sent an admin request before responding to the handshake",
|
||||
))
|
||||
));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -686,16 +690,17 @@ async fn wait_for_message<Z: NetworkZone>(
|
|||
));
|
||||
}
|
||||
|
||||
_ => Err(HandshakeError::PeerSentInvalidMessage(
|
||||
Message::Response(_) => Err(HandshakeError::PeerSentInvalidMessage(
|
||||
"Peer sent an incorrect message",
|
||||
)),
|
||||
}?
|
||||
}?;
|
||||
}
|
||||
|
||||
Err(BucketError::IO(std::io::Error::new(
|
||||
std::io::ErrorKind::ConnectionAborted,
|
||||
"The peer stream returned None",
|
||||
)))?
|
||||
))
|
||||
.into())
|
||||
}
|
||||
|
||||
/// Sends a [`AdminResponseMessage::SupportFlags`] down the peer sink.
|
||||
|
|
|
@ -87,14 +87,13 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
where
|
||||
NAdrBook: AddressBook<N> + Clone,
|
||||
{
|
||||
let HandshakerBuilder {
|
||||
let Self {
|
||||
core_sync_svc,
|
||||
peer_sync_svc,
|
||||
protocol_request_svc,
|
||||
our_basic_node_data,
|
||||
broadcast_stream_maker,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
..
|
||||
} = self;
|
||||
|
||||
|
@ -106,7 +105,7 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
our_basic_node_data,
|
||||
broadcast_stream_maker,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
_zone: PhantomData,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -130,14 +129,13 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
where
|
||||
NCSync: CoreSyncSvc + Clone,
|
||||
{
|
||||
let HandshakerBuilder {
|
||||
let Self {
|
||||
address_book,
|
||||
peer_sync_svc,
|
||||
protocol_request_svc,
|
||||
our_basic_node_data,
|
||||
broadcast_stream_maker,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
..
|
||||
} = self;
|
||||
|
||||
|
@ -149,7 +147,7 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
our_basic_node_data,
|
||||
broadcast_stream_maker,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
_zone: PhantomData,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -167,14 +165,13 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
where
|
||||
NPSync: PeerSyncSvc<N> + Clone,
|
||||
{
|
||||
let HandshakerBuilder {
|
||||
let Self {
|
||||
address_book,
|
||||
core_sync_svc,
|
||||
protocol_request_svc,
|
||||
our_basic_node_data,
|
||||
broadcast_stream_maker,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
..
|
||||
} = self;
|
||||
|
||||
|
@ -186,7 +183,7 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
our_basic_node_data,
|
||||
broadcast_stream_maker,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
_zone: PhantomData,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -204,14 +201,13 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
where
|
||||
NProtoHdlr: ProtocolRequestHandler + Clone,
|
||||
{
|
||||
let HandshakerBuilder {
|
||||
let Self {
|
||||
address_book,
|
||||
core_sync_svc,
|
||||
peer_sync_svc,
|
||||
our_basic_node_data,
|
||||
broadcast_stream_maker,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
..
|
||||
} = self;
|
||||
|
||||
|
@ -223,7 +219,7 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
our_basic_node_data,
|
||||
broadcast_stream_maker,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
_zone: PhantomData,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -242,14 +238,13 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
BrdcstStrm: Stream<Item = BroadcastMessage> + Send + 'static,
|
||||
NBrdcstStrmMkr: Fn(InternalPeerID<N::Addr>) -> BrdcstStrm + Clone + Send + 'static,
|
||||
{
|
||||
let HandshakerBuilder {
|
||||
let Self {
|
||||
address_book,
|
||||
core_sync_svc,
|
||||
peer_sync_svc,
|
||||
protocol_request_svc,
|
||||
our_basic_node_data,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
..
|
||||
} = self;
|
||||
|
||||
|
@ -261,7 +256,7 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
our_basic_node_data,
|
||||
broadcast_stream_maker: new_broadcast_stream_maker,
|
||||
connection_parent_span,
|
||||
_zone,
|
||||
_zone: PhantomData,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -270,6 +265,7 @@ impl<N: NetworkZone, AdrBook, CSync, PSync, ProtoHdlr, BrdcstStrmMkr>
|
|||
/// ## Default Connection Parent Span
|
||||
///
|
||||
/// The default connection span will be [`Span::none`].
|
||||
#[must_use]
|
||||
pub fn with_connection_parent_span(self, connection_parent_span: Span) -> Self {
|
||||
Self {
|
||||
connection_parent_span: Some(connection_parent_span),
|
||||
|
|
|
@ -42,8 +42,8 @@ pub struct DummyCoreSyncSvc(CoreSyncData);
|
|||
|
||||
impl DummyCoreSyncSvc {
|
||||
/// Returns a [`DummyCoreSyncSvc`] that will just return the mainnet genesis [`CoreSyncData`].
|
||||
pub fn static_mainnet_genesis() -> DummyCoreSyncSvc {
|
||||
DummyCoreSyncSvc(CoreSyncData {
|
||||
pub const fn static_mainnet_genesis() -> Self {
|
||||
Self(CoreSyncData {
|
||||
cumulative_difficulty: 1,
|
||||
cumulative_difficulty_top64: 0,
|
||||
current_height: 1,
|
||||
|
@ -56,8 +56,8 @@ impl DummyCoreSyncSvc {
|
|||
}
|
||||
|
||||
/// Returns a [`DummyCoreSyncSvc`] that will just return the testnet genesis [`CoreSyncData`].
|
||||
pub fn static_testnet_genesis() -> DummyCoreSyncSvc {
|
||||
DummyCoreSyncSvc(CoreSyncData {
|
||||
pub const fn static_testnet_genesis() -> Self {
|
||||
Self(CoreSyncData {
|
||||
cumulative_difficulty: 1,
|
||||
cumulative_difficulty_top64: 0,
|
||||
current_height: 1,
|
||||
|
@ -70,8 +70,8 @@ impl DummyCoreSyncSvc {
|
|||
}
|
||||
|
||||
/// Returns a [`DummyCoreSyncSvc`] that will just return the stagenet genesis [`CoreSyncData`].
|
||||
pub fn static_stagenet_genesis() -> DummyCoreSyncSvc {
|
||||
DummyCoreSyncSvc(CoreSyncData {
|
||||
pub const fn static_stagenet_genesis() -> Self {
|
||||
Self(CoreSyncData {
|
||||
cumulative_difficulty: 1,
|
||||
cumulative_difficulty_top64: 0,
|
||||
current_height: 1,
|
||||
|
@ -84,8 +84,8 @@ impl DummyCoreSyncSvc {
|
|||
}
|
||||
|
||||
/// Returns a [`DummyCoreSyncSvc`] that will return the provided [`CoreSyncData`].
|
||||
pub fn static_custom(data: CoreSyncData) -> DummyCoreSyncSvc {
|
||||
DummyCoreSyncSvc(data)
|
||||
pub const fn static_custom(data: CoreSyncData) -> Self {
|
||||
Self(data)
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -46,7 +46,7 @@ pub(crate) struct PeerRequestHandler<Z: NetworkZone, A, CS, PS, PR> {
|
|||
pub peer_info: PeerInformation<Z::Addr>,
|
||||
}
|
||||
|
||||
impl<Z: NetworkZone, A, CS, PS, PR> PeerRequestHandler<Z, A, CS, PS, PR>
|
||||
impl<Z, A, CS, PS, PR> PeerRequestHandler<Z, A, CS, PS, PR>
|
||||
where
|
||||
Z: NetworkZone,
|
||||
A: AddressBook<Z>,
|
||||
|
@ -55,7 +55,7 @@ where
|
|||
PR: ProtocolRequestHandler,
|
||||
{
|
||||
/// Handles an incoming [`PeerRequest`] to our node.
|
||||
pub async fn handle_peer_request(
|
||||
pub(crate) async fn handle_peer_request(
|
||||
&mut self,
|
||||
req: PeerRequest,
|
||||
) -> Result<PeerResponse, tower::BoxError> {
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
//! Timeout Monitor
|
||||
//!
|
||||
//! This module holds the task that sends periodic [TimedSync](PeerRequest::TimedSync) requests to a peer to make
|
||||
//! This module holds the task that sends periodic [`TimedSync`](PeerRequest::TimedSync) requests to a peer to make
|
||||
//! sure the connection is still active.
|
||||
use std::sync::Arc;
|
||||
|
||||
|
@ -64,7 +64,7 @@ where
|
|||
return Ok(());
|
||||
}
|
||||
|
||||
let Ok(permit) = semaphore.clone().try_acquire_owned() else {
|
||||
let Ok(permit) = Arc::clone(&semaphore).try_acquire_owned() else {
|
||||
// If we can't get a permit the connection is currently waiting for a response, so no need to
|
||||
// do a timed sync.
|
||||
continue;
|
||||
|
|
|
@ -4,7 +4,7 @@ pub struct SharedError<T>(Arc<OnceLock<T>>);
|
|||
|
||||
impl<T> Clone for SharedError<T> {
|
||||
fn clone(&self) -> Self {
|
||||
Self(self.0.clone())
|
||||
Self(Arc::clone(&self.0))
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -18,11 +18,12 @@ pub struct HandleBuilder {
|
|||
|
||||
impl HandleBuilder {
|
||||
/// Create a new builder.
|
||||
pub fn new() -> Self {
|
||||
pub const fn new() -> Self {
|
||||
Self { permit: None }
|
||||
}
|
||||
|
||||
/// Sets the permit for this connection.
|
||||
#[must_use]
|
||||
pub fn with_permit(mut self, permit: Option<OwnedSemaphorePermit>) -> Self {
|
||||
self.permit = permit;
|
||||
self
|
||||
|
@ -40,7 +41,7 @@ impl HandleBuilder {
|
|||
_permit: self.permit,
|
||||
},
|
||||
ConnectionHandle {
|
||||
token: token.clone(),
|
||||
token,
|
||||
ban: Arc::new(OnceLock::new()),
|
||||
},
|
||||
)
|
||||
|
@ -66,13 +67,13 @@ impl ConnectionGuard {
|
|||
///
|
||||
/// This will be called on [`Drop::drop`].
|
||||
pub fn connection_closed(&self) {
|
||||
self.token.cancel()
|
||||
self.token.cancel();
|
||||
}
|
||||
}
|
||||
|
||||
impl Drop for ConnectionGuard {
|
||||
fn drop(&mut self) {
|
||||
self.token.cancel()
|
||||
self.token.cancel();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -90,6 +91,10 @@ impl ConnectionHandle {
|
|||
}
|
||||
/// Bans the peer for the given `duration`.
|
||||
pub fn ban_peer(&self, duration: Duration) {
|
||||
#[expect(
|
||||
clippy::let_underscore_must_use,
|
||||
reason = "error means peer is already banned; fine to ignore"
|
||||
)]
|
||||
let _ = self.ban.set(BanPeer(duration));
|
||||
self.token.cancel();
|
||||
}
|
||||
|
@ -103,6 +108,6 @@ impl ConnectionHandle {
|
|||
}
|
||||
/// Sends the signal to the connection task to disconnect.
|
||||
pub fn send_close_signal(&self) {
|
||||
self.token.cancel()
|
||||
self.token.cancel();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
//!
|
||||
//! # Network Zones
|
||||
//!
|
||||
//! This crate abstracts over network zones, Tor/I2p/clearnet with the [NetworkZone] trait. Currently only clearnet is implemented: [ClearNet].
|
||||
//! This crate abstracts over network zones, Tor/I2p/clearnet with the [`NetworkZone`] trait. Currently only clearnet is implemented: [`ClearNet`].
|
||||
//!
|
||||
//! # Usage
|
||||
//!
|
||||
|
@ -56,6 +56,16 @@
|
|||
//! .unwrap();
|
||||
//! # });
|
||||
//! ```
|
||||
|
||||
cfg_if::cfg_if! {
|
||||
// Used in `tests/`
|
||||
if #[cfg(test)] {
|
||||
use cuprate_test_utils as _;
|
||||
use tokio_test as _;
|
||||
use hex as _;
|
||||
}
|
||||
}
|
||||
|
||||
use std::{fmt::Debug, future::Future, hash::Hash};
|
||||
|
||||
use futures::{Sink, Stream};
|
||||
|
@ -102,7 +112,7 @@ pub trait NetZoneAddress:
|
|||
+ Unpin
|
||||
+ 'static
|
||||
{
|
||||
/// Cuprate needs to be able to ban peers by IP addresses and not just by SocketAddr as
|
||||
/// Cuprate needs to be able to ban peers by IP addresses and not just by `SocketAddr` as
|
||||
/// that include the port, to be able to facilitate this network addresses must have a ban ID
|
||||
/// which for hidden services could just be the address it self but for clear net addresses will
|
||||
/// be the IP address.
|
||||
|
|
|
@ -19,7 +19,7 @@ impl NetZoneAddress for SocketAddr {
|
|||
type BanID = IpAddr;
|
||||
|
||||
fn set_port(&mut self, port: u16) {
|
||||
SocketAddr::set_port(self, port)
|
||||
Self::set_port(self, port);
|
||||
}
|
||||
|
||||
fn ban_id(&self) -> Self::BanID {
|
||||
|
|
|
@ -8,7 +8,7 @@
|
|||
//!
|
||||
//! Here is every P2P request/response.
|
||||
//!
|
||||
//! *note admin messages are already request/response so "Handshake" is actually made of a HandshakeRequest & HandshakeResponse
|
||||
//! *note admin messages are already request/response so "Handshake" is actually made of a `HandshakeRequest` & `HandshakeResponse`
|
||||
//!
|
||||
//! ```md
|
||||
//! Admin:
|
||||
|
@ -78,15 +78,15 @@ pub enum PeerRequest {
|
|||
}
|
||||
|
||||
impl PeerRequest {
|
||||
pub fn id(&self) -> MessageID {
|
||||
pub const fn id(&self) -> MessageID {
|
||||
match self {
|
||||
PeerRequest::Admin(admin_req) => match admin_req {
|
||||
Self::Admin(admin_req) => match admin_req {
|
||||
AdminRequestMessage::Handshake(_) => MessageID::Handshake,
|
||||
AdminRequestMessage::TimedSync(_) => MessageID::TimedSync,
|
||||
AdminRequestMessage::Ping => MessageID::Ping,
|
||||
AdminRequestMessage::SupportFlags => MessageID::SupportFlags,
|
||||
},
|
||||
PeerRequest::Protocol(protocol_request) => match protocol_request {
|
||||
Self::Protocol(protocol_request) => match protocol_request {
|
||||
ProtocolRequest::GetObjects(_) => MessageID::GetObjects,
|
||||
ProtocolRequest::GetChain(_) => MessageID::GetChain,
|
||||
ProtocolRequest::FluffyMissingTxs(_) => MessageID::FluffyMissingTxs,
|
||||
|
@ -98,10 +98,10 @@ impl PeerRequest {
|
|||
}
|
||||
}
|
||||
|
||||
pub fn needs_response(&self) -> bool {
|
||||
pub const fn needs_response(&self) -> bool {
|
||||
!matches!(
|
||||
self,
|
||||
PeerRequest::Protocol(
|
||||
Self::Protocol(
|
||||
ProtocolRequest::NewBlock(_)
|
||||
| ProtocolRequest::NewFluffyBlock(_)
|
||||
| ProtocolRequest::NewTransactions(_)
|
||||
|
@ -126,15 +126,15 @@ pub enum PeerResponse {
|
|||
}
|
||||
|
||||
impl PeerResponse {
|
||||
pub fn id(&self) -> Option<MessageID> {
|
||||
pub const fn id(&self) -> Option<MessageID> {
|
||||
Some(match self {
|
||||
PeerResponse::Admin(admin_res) => match admin_res {
|
||||
Self::Admin(admin_res) => match admin_res {
|
||||
AdminResponseMessage::Handshake(_) => MessageID::Handshake,
|
||||
AdminResponseMessage::TimedSync(_) => MessageID::TimedSync,
|
||||
AdminResponseMessage::Ping(_) => MessageID::Ping,
|
||||
AdminResponseMessage::SupportFlags(_) => MessageID::SupportFlags,
|
||||
},
|
||||
PeerResponse::Protocol(protocol_res) => match protocol_res {
|
||||
Self::Protocol(protocol_res) => match protocol_res {
|
||||
ProtocolResponse::GetObjects(_) => MessageID::GetObjects,
|
||||
ProtocolResponse::GetChain(_) => MessageID::GetChain,
|
||||
ProtocolResponse::NewFluffyBlock(_) => MessageID::NewBlock,
|
||||
|
|
|
@ -11,15 +11,13 @@ pub struct MessageConversionError;
|
|||
impl From<ProtocolRequest> for ProtocolMessage {
|
||||
fn from(value: ProtocolRequest) -> Self {
|
||||
match value {
|
||||
ProtocolRequest::GetObjects(val) => ProtocolMessage::GetObjectsRequest(val),
|
||||
ProtocolRequest::GetChain(val) => ProtocolMessage::ChainRequest(val),
|
||||
ProtocolRequest::FluffyMissingTxs(val) => {
|
||||
ProtocolMessage::FluffyMissingTransactionsRequest(val)
|
||||
}
|
||||
ProtocolRequest::GetTxPoolCompliment(val) => ProtocolMessage::GetTxPoolCompliment(val),
|
||||
ProtocolRequest::NewBlock(val) => ProtocolMessage::NewBlock(val),
|
||||
ProtocolRequest::NewFluffyBlock(val) => ProtocolMessage::NewFluffyBlock(val),
|
||||
ProtocolRequest::NewTransactions(val) => ProtocolMessage::NewTransactions(val),
|
||||
ProtocolRequest::GetObjects(val) => Self::GetObjectsRequest(val),
|
||||
ProtocolRequest::GetChain(val) => Self::ChainRequest(val),
|
||||
ProtocolRequest::FluffyMissingTxs(val) => Self::FluffyMissingTransactionsRequest(val),
|
||||
ProtocolRequest::GetTxPoolCompliment(val) => Self::GetTxPoolCompliment(val),
|
||||
ProtocolRequest::NewBlock(val) => Self::NewBlock(val),
|
||||
ProtocolRequest::NewFluffyBlock(val) => Self::NewFluffyBlock(val),
|
||||
ProtocolRequest::NewTransactions(val) => Self::NewTransactions(val),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -29,15 +27,13 @@ impl TryFrom<ProtocolMessage> for ProtocolRequest {
|
|||
|
||||
fn try_from(value: ProtocolMessage) -> Result<Self, Self::Error> {
|
||||
Ok(match value {
|
||||
ProtocolMessage::GetObjectsRequest(val) => ProtocolRequest::GetObjects(val),
|
||||
ProtocolMessage::ChainRequest(val) => ProtocolRequest::GetChain(val),
|
||||
ProtocolMessage::FluffyMissingTransactionsRequest(val) => {
|
||||
ProtocolRequest::FluffyMissingTxs(val)
|
||||
}
|
||||
ProtocolMessage::GetTxPoolCompliment(val) => ProtocolRequest::GetTxPoolCompliment(val),
|
||||
ProtocolMessage::NewBlock(val) => ProtocolRequest::NewBlock(val),
|
||||
ProtocolMessage::NewFluffyBlock(val) => ProtocolRequest::NewFluffyBlock(val),
|
||||
ProtocolMessage::NewTransactions(val) => ProtocolRequest::NewTransactions(val),
|
||||
ProtocolMessage::GetObjectsRequest(val) => Self::GetObjects(val),
|
||||
ProtocolMessage::ChainRequest(val) => Self::GetChain(val),
|
||||
ProtocolMessage::FluffyMissingTransactionsRequest(val) => Self::FluffyMissingTxs(val),
|
||||
ProtocolMessage::GetTxPoolCompliment(val) => Self::GetTxPoolCompliment(val),
|
||||
ProtocolMessage::NewBlock(val) => Self::NewBlock(val),
|
||||
ProtocolMessage::NewFluffyBlock(val) => Self::NewFluffyBlock(val),
|
||||
ProtocolMessage::NewTransactions(val) => Self::NewTransactions(val),
|
||||
ProtocolMessage::GetObjectsResponse(_) | ProtocolMessage::ChainEntryResponse(_) => {
|
||||
return Err(MessageConversionError)
|
||||
}
|
||||
|
@ -48,8 +44,8 @@ impl TryFrom<ProtocolMessage> for ProtocolRequest {
|
|||
impl From<PeerRequest> for Message {
|
||||
fn from(value: PeerRequest) -> Self {
|
||||
match value {
|
||||
PeerRequest::Admin(val) => Message::Request(val),
|
||||
PeerRequest::Protocol(val) => Message::Protocol(val.into()),
|
||||
PeerRequest::Admin(val) => Self::Request(val),
|
||||
PeerRequest::Protocol(val) => Self::Protocol(val.into()),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -59,8 +55,8 @@ impl TryFrom<Message> for PeerRequest {
|
|||
|
||||
fn try_from(value: Message) -> Result<Self, Self::Error> {
|
||||
match value {
|
||||
Message::Request(req) => Ok(PeerRequest::Admin(req)),
|
||||
Message::Protocol(pro) => Ok(PeerRequest::Protocol(pro.try_into()?)),
|
||||
Message::Request(req) => Ok(Self::Admin(req)),
|
||||
Message::Protocol(pro) => Ok(Self::Protocol(pro.try_into()?)),
|
||||
Message::Response(_) => Err(MessageConversionError),
|
||||
}
|
||||
}
|
||||
|
@ -71,10 +67,10 @@ impl TryFrom<ProtocolResponse> for ProtocolMessage {
|
|||
|
||||
fn try_from(value: ProtocolResponse) -> Result<Self, Self::Error> {
|
||||
Ok(match value {
|
||||
ProtocolResponse::NewTransactions(val) => ProtocolMessage::NewTransactions(val),
|
||||
ProtocolResponse::NewFluffyBlock(val) => ProtocolMessage::NewFluffyBlock(val),
|
||||
ProtocolResponse::GetChain(val) => ProtocolMessage::ChainEntryResponse(val),
|
||||
ProtocolResponse::GetObjects(val) => ProtocolMessage::GetObjectsResponse(val),
|
||||
ProtocolResponse::NewTransactions(val) => Self::NewTransactions(val),
|
||||
ProtocolResponse::NewFluffyBlock(val) => Self::NewFluffyBlock(val),
|
||||
ProtocolResponse::GetChain(val) => Self::ChainEntryResponse(val),
|
||||
ProtocolResponse::GetObjects(val) => Self::GetObjectsResponse(val),
|
||||
ProtocolResponse::NA => return Err(MessageConversionError),
|
||||
})
|
||||
}
|
||||
|
@ -85,10 +81,10 @@ impl TryFrom<ProtocolMessage> for ProtocolResponse {
|
|||
|
||||
fn try_from(value: ProtocolMessage) -> Result<Self, Self::Error> {
|
||||
Ok(match value {
|
||||
ProtocolMessage::NewTransactions(val) => ProtocolResponse::NewTransactions(val),
|
||||
ProtocolMessage::NewFluffyBlock(val) => ProtocolResponse::NewFluffyBlock(val),
|
||||
ProtocolMessage::ChainEntryResponse(val) => ProtocolResponse::GetChain(val),
|
||||
ProtocolMessage::GetObjectsResponse(val) => ProtocolResponse::GetObjects(val),
|
||||
ProtocolMessage::NewTransactions(val) => Self::NewTransactions(val),
|
||||
ProtocolMessage::NewFluffyBlock(val) => Self::NewFluffyBlock(val),
|
||||
ProtocolMessage::ChainEntryResponse(val) => Self::GetChain(val),
|
||||
ProtocolMessage::GetObjectsResponse(val) => Self::GetObjects(val),
|
||||
ProtocolMessage::ChainRequest(_)
|
||||
| ProtocolMessage::FluffyMissingTransactionsRequest(_)
|
||||
| ProtocolMessage::GetObjectsRequest(_)
|
||||
|
@ -103,8 +99,8 @@ impl TryFrom<Message> for PeerResponse {
|
|||
|
||||
fn try_from(value: Message) -> Result<Self, Self::Error> {
|
||||
match value {
|
||||
Message::Response(res) => Ok(PeerResponse::Admin(res)),
|
||||
Message::Protocol(pro) => Ok(PeerResponse::Protocol(pro.try_into()?)),
|
||||
Message::Response(res) => Ok(Self::Admin(res)),
|
||||
Message::Protocol(pro) => Ok(Self::Protocol(pro.try_into()?)),
|
||||
Message::Request(_) => Err(MessageConversionError),
|
||||
}
|
||||
}
|
||||
|
@ -115,8 +111,8 @@ impl TryFrom<PeerResponse> for Message {
|
|||
|
||||
fn try_from(value: PeerResponse) -> Result<Self, Self::Error> {
|
||||
Ok(match value {
|
||||
PeerResponse::Admin(val) => Message::Response(val),
|
||||
PeerResponse::Protocol(val) => Message::Protocol(val.try_into()?),
|
||||
PeerResponse::Admin(val) => Self::Response(val),
|
||||
PeerResponse::Protocol(val) => Self::Protocol(val.try_into()?),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
|
|
@ -52,7 +52,7 @@ pub struct ZoneSpecificPeerListEntryBase<A: NetZoneAddress> {
|
|||
pub rpc_credits_per_hash: u32,
|
||||
}
|
||||
|
||||
impl<A: NetZoneAddress> From<ZoneSpecificPeerListEntryBase<A>> for cuprate_wire::PeerListEntryBase {
|
||||
impl<A: NetZoneAddress> From<ZoneSpecificPeerListEntryBase<A>> for PeerListEntryBase {
|
||||
fn from(value: ZoneSpecificPeerListEntryBase<A>) -> Self {
|
||||
Self {
|
||||
adr: value.adr.into(),
|
||||
|
@ -74,9 +74,7 @@ pub enum PeerListConversionError {
|
|||
PruningSeed(#[from] PruningError),
|
||||
}
|
||||
|
||||
impl<A: NetZoneAddress> TryFrom<cuprate_wire::PeerListEntryBase>
|
||||
for ZoneSpecificPeerListEntryBase<A>
|
||||
{
|
||||
impl<A: NetZoneAddress> TryFrom<PeerListEntryBase> for ZoneSpecificPeerListEntryBase<A> {
|
||||
type Error = PeerListConversionError;
|
||||
|
||||
fn try_from(value: PeerListEntryBase) -> Result<Self, Self::Error> {
|
||||
|
|
|
@ -1,4 +1,7 @@
|
|||
//! This file contains a test for a handshake with monerod but uses fragmented messages.
|
||||
|
||||
#![expect(unused_crate_dependencies, reason = "external test module")]
|
||||
|
||||
use std::{
|
||||
net::SocketAddr,
|
||||
pin::Pin,
|
||||
|
@ -21,6 +24,13 @@ use tokio_util::{
|
|||
use tower::{Service, ServiceExt};
|
||||
|
||||
use cuprate_helper::network::Network;
|
||||
use cuprate_test_utils::monerod::monerod;
|
||||
use cuprate_wire::{
|
||||
common::PeerSupportFlags,
|
||||
levin::{message::make_fragmented_messages, LevinMessage, Protocol},
|
||||
BasicNodeData, Message, MoneroWireCodec,
|
||||
};
|
||||
|
||||
use cuprate_p2p_core::{
|
||||
client::{
|
||||
handshaker::HandshakerBuilder, ConnectRequest, Connector, DoHandshakeRequest,
|
||||
|
@ -28,13 +38,6 @@ use cuprate_p2p_core::{
|
|||
},
|
||||
ClearNetServerCfg, ConnectionDirection, NetworkZone,
|
||||
};
|
||||
use cuprate_wire::{
|
||||
common::PeerSupportFlags,
|
||||
levin::{message::make_fragmented_messages, LevinMessage, Protocol},
|
||||
BasicNodeData, Message, MoneroWireCodec,
|
||||
};
|
||||
|
||||
use cuprate_test_utils::monerod::monerod;
|
||||
|
||||
/// A network zone equal to clear net where every message sent is turned into a fragmented message.
|
||||
/// Does not support sending fragmented or dummy messages manually.
|
||||
|
@ -184,7 +187,7 @@ async fn fragmented_handshake_monerod_to_cuprate() {
|
|||
let next_connection_fut = timeout(Duration::from_secs(30), listener.next());
|
||||
|
||||
if let Some(Ok((addr, stream, sink))) = next_connection_fut.await.unwrap() {
|
||||
let _ = handshaker
|
||||
handshaker
|
||||
.ready()
|
||||
.await
|
||||
.unwrap()
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "external test module")]
|
||||
|
||||
use std::{sync::Arc, time::Duration};
|
||||
|
||||
use tokio::sync::Semaphore;
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
#![expect(unused_crate_dependencies, reason = "external test module")]
|
||||
|
||||
use std::time::Duration;
|
||||
|
||||
use futures::StreamExt;
|
||||
|
@ -9,6 +11,10 @@ use tokio_util::codec::{FramedRead, FramedWrite};
|
|||
use tower::{Service, ServiceExt};
|
||||
|
||||
use cuprate_helper::network::Network;
|
||||
use cuprate_test_utils::{
|
||||
monerod::monerod,
|
||||
test_netzone::{TestNetZone, TestNetZoneAddr},
|
||||
};
|
||||
use cuprate_wire::{common::PeerSupportFlags, BasicNodeData, MoneroWireCodec};
|
||||
|
||||
use cuprate_p2p_core::{
|
||||
|
@ -19,12 +25,8 @@ use cuprate_p2p_core::{
|
|||
ClearNet, ClearNetServerCfg, ConnectionDirection, NetworkZone,
|
||||
};
|
||||
|
||||
use cuprate_test_utils::{
|
||||
monerod::monerod,
|
||||
test_netzone::{TestNetZone, TestNetZoneAddr},
|
||||
};
|
||||
|
||||
#[tokio::test]
|
||||
#[expect(clippy::significant_drop_tightening)]
|
||||
async fn handshake_cuprate_to_cuprate() {
|
||||
// Tests a Cuprate <-> Cuprate handshake by making 2 handshake services and making them talk to
|
||||
// each other.
|
||||
|
@ -147,7 +149,7 @@ async fn handshake_monerod_to_cuprate() {
|
|||
let next_connection_fut = timeout(Duration::from_secs(30), listener.next());
|
||||
|
||||
if let Some(Ok((addr, stream, sink))) = next_connection_fut.await.unwrap() {
|
||||
let _ = handshaker
|
||||
handshaker
|
||||
.ready()
|
||||
.await
|
||||
.unwrap()
|
||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue