mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-05 10:29:40 +00:00
Fix handling of output distribution
Some checks are pending
Full Stack Tests / build (push) Waiting to run
Lint / clippy (macos-13) (push) Waiting to run
Lint / clippy (macos-14) (push) Waiting to run
Lint / clippy (ubuntu-latest) (push) Waiting to run
Lint / clippy (windows-latest) (push) Waiting to run
Lint / deny (push) Waiting to run
Lint / fmt (push) Waiting to run
Lint / machete (push) Waiting to run
Monero Tests / unit-tests (push) Waiting to run
Monero Tests / integration-tests (v0.17.3.2) (push) Waiting to run
Monero Tests / integration-tests (v0.18.2.0) (push) Waiting to run
no-std build / build (push) Waiting to run
Processor Tests / build (push) Waiting to run
Tests / test-infra (push) Waiting to run
Tests / test-substrate (push) Waiting to run
Tests / test-serai-client (push) Waiting to run
coins/ Tests / test-coins (push) Waiting to run
Coordinator Tests / build (push) Waiting to run
Some checks are pending
Full Stack Tests / build (push) Waiting to run
Lint / clippy (macos-13) (push) Waiting to run
Lint / clippy (macos-14) (push) Waiting to run
Lint / clippy (ubuntu-latest) (push) Waiting to run
Lint / clippy (windows-latest) (push) Waiting to run
Lint / deny (push) Waiting to run
Lint / fmt (push) Waiting to run
Lint / machete (push) Waiting to run
Monero Tests / unit-tests (push) Waiting to run
Monero Tests / integration-tests (v0.17.3.2) (push) Waiting to run
Monero Tests / integration-tests (v0.18.2.0) (push) Waiting to run
no-std build / build (push) Waiting to run
Processor Tests / build (push) Waiting to run
Tests / test-infra (push) Waiting to run
Tests / test-substrate (push) Waiting to run
Tests / test-serai-client (push) Waiting to run
coins/ Tests / test-coins (push) Waiting to run
Coordinator Tests / build (push) Waiting to run
We prior didn't handle how the output distribution only starts after a specific block.
This commit is contained in:
parent
7a68b065e0
commit
ee10692b23
3 changed files with 64 additions and 33 deletions
|
@ -69,9 +69,9 @@ async fn test_decoy_rpc() {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
// Test get_output_distribution
|
// Test get_output_distribution
|
||||||
// It's documented to take two inclusive block numbers
|
// Our documentation for our Rust fn defines it as taking two block numbers
|
||||||
{
|
{
|
||||||
let distribution_len = rpc.get_output_distribution_len().await.unwrap();
|
let distribution_len = rpc.get_output_distribution_end_height().await.unwrap();
|
||||||
assert_eq!(distribution_len, rpc.get_height().await.unwrap());
|
assert_eq!(distribution_len, rpc.get_height().await.unwrap());
|
||||||
|
|
||||||
rpc.get_output_distribution(0 ..= distribution_len).await.unwrap_err();
|
rpc.get_output_distribution(0 ..= distribution_len).await.unwrap_err();
|
||||||
|
|
|
@ -168,20 +168,20 @@ impl FeePriority {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct EmptyResponse {}
|
struct EmptyResponse {}
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct JsonRpcResponse<T> {
|
struct JsonRpcResponse<T> {
|
||||||
result: T,
|
result: T,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct TransactionResponse {
|
struct TransactionResponse {
|
||||||
tx_hash: String,
|
tx_hash: String,
|
||||||
as_hex: String,
|
as_hex: String,
|
||||||
pruned_as_hex: String,
|
pruned_as_hex: String,
|
||||||
}
|
}
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct TransactionsResponse {
|
struct TransactionsResponse {
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
missed_tx: Vec<String>,
|
missed_tx: Vec<String>,
|
||||||
|
@ -189,7 +189,7 @@ struct TransactionsResponse {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The response to an output query.
|
/// The response to an output query.
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
pub struct OutputResponse {
|
pub struct OutputResponse {
|
||||||
/// The height of the block this output was added to the chain in.
|
/// The height of the block this output was added to the chain in.
|
||||||
pub height: usize,
|
pub height: usize,
|
||||||
|
@ -282,12 +282,12 @@ pub trait Rpc: Sync + Clone + Debug {
|
||||||
///
|
///
|
||||||
/// This is specifically the major version within the most recent block header.
|
/// This is specifically the major version within the most recent block header.
|
||||||
async fn get_hardfork_version(&self) -> Result<u8, RpcError> {
|
async fn get_hardfork_version(&self) -> Result<u8, RpcError> {
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct HeaderResponse {
|
struct HeaderResponse {
|
||||||
major_version: u8,
|
major_version: u8,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct LastHeaderResponse {
|
struct LastHeaderResponse {
|
||||||
block_header: HeaderResponse,
|
block_header: HeaderResponse,
|
||||||
}
|
}
|
||||||
|
@ -306,7 +306,7 @@ pub trait Rpc: Sync + Clone + Debug {
|
||||||
/// The height is defined as the amount of blocks on the blockchain. For a blockchain with only
|
/// The height is defined as the amount of blocks on the blockchain. For a blockchain with only
|
||||||
/// its genesis block, the height will be 1.
|
/// its genesis block, the height will be 1.
|
||||||
async fn get_height(&self) -> Result<usize, RpcError> {
|
async fn get_height(&self) -> Result<usize, RpcError> {
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct HeightResponse {
|
struct HeightResponse {
|
||||||
height: usize,
|
height: usize,
|
||||||
}
|
}
|
||||||
|
@ -398,11 +398,11 @@ pub trait Rpc: Sync + Clone + Debug {
|
||||||
/// `number` is the block's zero-indexed position on the blockchain (`0` for the genesis block,
|
/// `number` is the block's zero-indexed position on the blockchain (`0` for the genesis block,
|
||||||
/// `height - 1` for the latest block).
|
/// `height - 1` for the latest block).
|
||||||
async fn get_block_hash(&self, number: usize) -> Result<[u8; 32], RpcError> {
|
async fn get_block_hash(&self, number: usize) -> Result<[u8; 32], RpcError> {
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct BlockHeaderResponse {
|
struct BlockHeaderResponse {
|
||||||
hash: String,
|
hash: String,
|
||||||
}
|
}
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct BlockHeaderByHeightResponse {
|
struct BlockHeaderByHeightResponse {
|
||||||
block_header: BlockHeaderResponse,
|
block_header: BlockHeaderResponse,
|
||||||
}
|
}
|
||||||
|
@ -416,7 +416,7 @@ pub trait Rpc: Sync + Clone + Debug {
|
||||||
///
|
///
|
||||||
/// The received block will be hashed in order to verify the correct block was returned.
|
/// The received block will be hashed in order to verify the correct block was returned.
|
||||||
async fn get_block(&self, hash: [u8; 32]) -> Result<Block, RpcError> {
|
async fn get_block(&self, hash: [u8; 32]) -> Result<Block, RpcError> {
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct BlockResponse {
|
struct BlockResponse {
|
||||||
blob: String,
|
blob: String,
|
||||||
}
|
}
|
||||||
|
@ -437,7 +437,7 @@ pub trait Rpc: Sync + Clone + Debug {
|
||||||
/// `number` is the block's zero-indexed position on the blockchain (`0` for the genesis block,
|
/// `number` is the block's zero-indexed position on the blockchain (`0` for the genesis block,
|
||||||
/// `height - 1` for the latest block).
|
/// `height - 1` for the latest block).
|
||||||
async fn get_block_by_number(&self, number: usize) -> Result<Block, RpcError> {
|
async fn get_block_by_number(&self, number: usize) -> Result<Block, RpcError> {
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct BlockResponse {
|
struct BlockResponse {
|
||||||
blob: String,
|
blob: String,
|
||||||
}
|
}
|
||||||
|
@ -499,7 +499,7 @@ pub trait Rpc: Sync + Clone + Debug {
|
||||||
/// This MUST NOT be expected to be deterministic in any way.
|
/// This MUST NOT be expected to be deterministic in any way.
|
||||||
// TODO: Take a sanity check argument
|
// TODO: Take a sanity check argument
|
||||||
async fn get_fee_rate(&self, priority: FeePriority) -> Result<FeeRate, RpcError> {
|
async fn get_fee_rate(&self, priority: FeePriority) -> Result<FeeRate, RpcError> {
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct FeeResponse {
|
struct FeeResponse {
|
||||||
status: String,
|
status: String,
|
||||||
fees: Option<Vec<u64>>,
|
fees: Option<Vec<u64>>,
|
||||||
|
@ -555,7 +555,7 @@ pub trait Rpc: Sync + Clone + Debug {
|
||||||
/// Publish a transaction.
|
/// Publish a transaction.
|
||||||
async fn publish_transaction(&self, tx: &Transaction) -> Result<(), RpcError> {
|
async fn publish_transaction(&self, tx: &Transaction) -> Result<(), RpcError> {
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct SendRawResponse {
|
struct SendRawResponse {
|
||||||
status: String,
|
status: String,
|
||||||
double_spend: bool,
|
double_spend: bool,
|
||||||
|
@ -798,22 +798,23 @@ pub trait Rpc: Sync + Clone + Debug {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A trait for any object which can be used to select decoys.
|
/// A trait for any object which can be used to select RingCT decoys.
|
||||||
///
|
///
|
||||||
/// An implementation is provided for any satisfier of `Rpc`. The benefit of this trait is the
|
/// An implementation is provided for any satisfier of `Rpc`. It is not recommended to use an `Rpc`
|
||||||
/// ability to select decoys off of a locally stored output distribution, preventing potential
|
/// object to satisfy this. This should be satisfied by a local store of the output distribution,
|
||||||
/// attacks a remote node can perform.
|
/// both for performance and to prevent potential attacks a remote node can perform.
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
pub trait DecoyRpc: Sync + Clone + Debug {
|
pub trait DecoyRpc: Sync + Clone + Debug {
|
||||||
/// Get the length of the output distribution.
|
/// Get the height the output distribution ends at.
|
||||||
///
|
///
|
||||||
/// This is equivalent to the hight of the blockchain it's for. This is intended to be cheaper
|
/// This is equivalent to the hight of the blockchain it's for. This is intended to be cheaper
|
||||||
/// than fetching the entire output distribution.
|
/// than fetching the entire output distribution.
|
||||||
async fn get_output_distribution_len(&self) -> Result<usize, RpcError>;
|
async fn get_output_distribution_end_height(&self) -> Result<usize, RpcError>;
|
||||||
|
|
||||||
/// Get the output distribution.
|
/// Get the RingCT (zero-amount) output distribution.
|
||||||
///
|
///
|
||||||
/// `range` is in terms of block numbers.
|
/// `range` is in terms of block numbers. The result may be smaller than the requested range if
|
||||||
|
/// the range starts before RingCT outputs were created on-chain.
|
||||||
async fn get_output_distribution(
|
async fn get_output_distribution(
|
||||||
&self,
|
&self,
|
||||||
range: impl Send + RangeBounds<usize>,
|
range: impl Send + RangeBounds<usize>,
|
||||||
|
@ -843,7 +844,7 @@ pub trait DecoyRpc: Sync + Clone + Debug {
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl<R: Rpc> DecoyRpc for R {
|
impl<R: Rpc> DecoyRpc for R {
|
||||||
async fn get_output_distribution_len(&self) -> Result<usize, RpcError> {
|
async fn get_output_distribution_end_height(&self) -> Result<usize, RpcError> {
|
||||||
<Self as Rpc>::get_height(self).await
|
<Self as Rpc>::get_height(self).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -851,14 +852,17 @@ impl<R: Rpc> DecoyRpc for R {
|
||||||
&self,
|
&self,
|
||||||
range: impl Send + RangeBounds<usize>,
|
range: impl Send + RangeBounds<usize>,
|
||||||
) -> Result<Vec<u64>, RpcError> {
|
) -> Result<Vec<u64>, RpcError> {
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Default, Debug, Deserialize)]
|
||||||
struct Distribution {
|
struct Distribution {
|
||||||
distribution: Vec<u64>,
|
distribution: Vec<u64>,
|
||||||
|
// A blockchain with just its genesis block has a height of 1
|
||||||
|
start_height: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct Distributions {
|
struct Distributions {
|
||||||
distributions: [Distribution; 1],
|
distributions: [Distribution; 1],
|
||||||
|
status: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
let from = match range.start_bound() {
|
let from = match range.start_bound() {
|
||||||
|
@ -889,15 +893,39 @@ impl<R: Rpc> DecoyRpc for R {
|
||||||
"binary": false,
|
"binary": false,
|
||||||
"amounts": [0],
|
"amounts": [0],
|
||||||
"cumulative": true,
|
"cumulative": true,
|
||||||
|
// These are actually block numbers, not heights
|
||||||
"from_height": from,
|
"from_height": from,
|
||||||
"to_height": if zero_zero_case { 1 } else { to },
|
"to_height": if zero_zero_case { 1 } else { to },
|
||||||
})),
|
})),
|
||||||
)
|
)
|
||||||
.await?;
|
.await?;
|
||||||
let mut distributions = distributions.distributions;
|
|
||||||
let mut distribution = core::mem::take(&mut distributions[0].distribution);
|
|
||||||
|
|
||||||
let expected_len = if zero_zero_case { 2 } else { (to - from) + 1 };
|
if distributions.status != "OK" {
|
||||||
|
Err(RpcError::ConnectionError(
|
||||||
|
"node couldn't service this request for the output distribution".to_string(),
|
||||||
|
))?;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut distributions = distributions.distributions;
|
||||||
|
let Distribution { start_height, mut distribution } = core::mem::take(&mut distributions[0]);
|
||||||
|
// start_height is also actually a block number, and it should be at least `from`
|
||||||
|
// It may be after depending on when these outputs first appeared on the blockchain
|
||||||
|
// Unfortunately, we can't validate without a binary search to find the RingCT activation block
|
||||||
|
// and an iterative search from there, so we solely sanity check it
|
||||||
|
if start_height < from {
|
||||||
|
Err(RpcError::InvalidNode(format!(
|
||||||
|
"requested distribution from {from} and got from {start_height}"
|
||||||
|
)))?;
|
||||||
|
}
|
||||||
|
// It shouldn't be after `to` though
|
||||||
|
if start_height > to {
|
||||||
|
Err(RpcError::InvalidNode(format!(
|
||||||
|
"requested distribution to {to} and got from {start_height}"
|
||||||
|
)))?;
|
||||||
|
}
|
||||||
|
|
||||||
|
let expected_len = if zero_zero_case { 2 } else { (to - start_height) + 1 };
|
||||||
|
// Yet this is actually a height
|
||||||
if expected_len != distribution.len() {
|
if expected_len != distribution.len() {
|
||||||
Err(RpcError::InvalidNode(format!(
|
Err(RpcError::InvalidNode(format!(
|
||||||
"distribution length ({}) wasn't of the requested length ({})",
|
"distribution length ({}) wasn't of the requested length ({})",
|
||||||
|
@ -905,7 +933,7 @@ impl<R: Rpc> DecoyRpc for R {
|
||||||
expected_len
|
expected_len
|
||||||
)))?;
|
)))?;
|
||||||
}
|
}
|
||||||
// Requesting 0, 0 returns the distribution for the entire chain
|
// Requesting to = 0 returns the distribution for the entire chain
|
||||||
// We work-around this by requesting 0, 1 (yielding two blocks), then popping the second block
|
// We work-around this by requesting 0, 1 (yielding two blocks), then popping the second block
|
||||||
if zero_zero_case {
|
if zero_zero_case {
|
||||||
distribution.pop();
|
distribution.pop();
|
||||||
|
@ -914,7 +942,7 @@ impl<R: Rpc> DecoyRpc for R {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn get_outs(&self, indexes: &[u64]) -> Result<Vec<OutputResponse>, RpcError> {
|
async fn get_outs(&self, indexes: &[u64]) -> Result<Vec<OutputResponse>, RpcError> {
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Debug, Deserialize)]
|
||||||
struct OutsResponse {
|
struct OutsResponse {
|
||||||
status: String,
|
status: String,
|
||||||
outs: Vec<OutputResponse>,
|
outs: Vec<OutputResponse>,
|
||||||
|
|
|
@ -33,7 +33,7 @@ async fn select_n(
|
||||||
if height < DEFAULT_LOCK_WINDOW {
|
if height < DEFAULT_LOCK_WINDOW {
|
||||||
Err(RpcError::InternalError("not enough blocks to select decoys".to_string()))?;
|
Err(RpcError::InternalError("not enough blocks to select decoys".to_string()))?;
|
||||||
}
|
}
|
||||||
if height > rpc.get_output_distribution_len().await? {
|
if height > rpc.get_output_distribution_end_height().await? {
|
||||||
Err(RpcError::InternalError(
|
Err(RpcError::InternalError(
|
||||||
"decoys being requested from blocks this node doesn't have".to_string(),
|
"decoys being requested from blocks this node doesn't have".to_string(),
|
||||||
))?;
|
))?;
|
||||||
|
@ -41,6 +41,9 @@ async fn select_n(
|
||||||
|
|
||||||
// Get the distribution
|
// Get the distribution
|
||||||
let distribution = rpc.get_output_distribution(.. height).await?;
|
let distribution = rpc.get_output_distribution(.. height).await?;
|
||||||
|
if distribution.len() < DEFAULT_LOCK_WINDOW {
|
||||||
|
Err(RpcError::InternalError("not enough blocks to select decoys".to_string()))?;
|
||||||
|
}
|
||||||
let highest_output_exclusive_bound = distribution[distribution.len() - DEFAULT_LOCK_WINDOW];
|
let highest_output_exclusive_bound = distribution[distribution.len() - DEFAULT_LOCK_WINDOW];
|
||||||
// This assumes that each miner TX had one output (as sane) and checks we have sufficient
|
// This assumes that each miner TX had one output (as sane) and checks we have sufficient
|
||||||
// outputs even when excluding them (due to their own timelock requirements)
|
// outputs even when excluding them (due to their own timelock requirements)
|
||||||
|
|
Loading…
Reference in a new issue