Consensus Protocol

The Dusk Network utilizes a consensus protocol called Succinct Attestation (SA). SA is a permissionless proof-of-stake (PoS) consensus mechanism that provides statistical finality guarantees[^1]. SA belongs to the committee-based PoS category[^2] because it uses committees to finalize blocks for a given round.

SA is permissionless, meaning that any eligible participant in the Dusk Network protocol can join and participate in the consensus process. To be eligible, a participant must meet the following requirements:

Other terms used in the context of SA:

Repository structure

Example Node

The minimalistic and stateless version of the dusk-blockchain node allows for testing and diagnosing compatibility issues using the Consensus protocol in conjunction with Kadcast[^3]. It enables the node to join and participate in the dusk-blockchain/test-harness. Once the dusk-blockchain is fully migrated, this executable will no longer be needed and can be deprecated.

Example Testbed

A multi-instance setup running 10 SA instances provisioned with 10 eligible participants. The setup is configured to run for up to 1000 rounds. Useful for any kind of testing (issues, stress and performance testing).

Consensus crate

A full implementation of SA mechanism.

Implementation details

The implementation of SA consists of two main tokio-rs tasks, the Main_Loop and Agreement_Loop, which communicate with external components through message queues/channels. The protocol parameters for SA are located in src/config.rs.

How to use (example code)

```rust let mut consensus = Consensus::new( // Inbound messages for Main Loop inboundmsgs, // Outbound messages for Main Loop outboundmsgs, // Inbound messages for Agreement Loop agrinboundqueue, // Outbound messages for Agreement Loop agroutboundqueue, // Implements Operations trait Arc::new(Mutex::new(crate::mocks::Executor {})), // Implements Database trait Arc::new(Mutex::new(crate::mocks::SimpleDB::default())), );

let mut mostrecentblock = Block::default();

loop { /// Provisioners list is retrieved from contract storage state. let provisioners = rusk::get_provisioners();

// Round update is the input data for any consensus round execution.
// Round update includes mostly data from most recent block. 
let round_update = from(most_recent_block);

/// Consensus::Spin call initializes a consensus round
/// and spawns main consensus tokio::tasks.
let ret = consensus.spin(
        round_update
        provisioners,
        cancel_rx,
    )
    .await;

/// Consensus spin output/ret can be a winner block or an error. 
match ret {
    Ok(winner_block) => { 
        println!("new block produced");
    }
    Err(_) => {
        // Cancelled from outside by cancel_rx chan.
        // Max Step Reached - happens only if no consensus is reached for up to 213 steps/71 iterations.
    }
}
most_recent_block = winner;

/// Internally, consensus instance may accept future messages for next round. 
/// They will be drained on running the round, that's why same consensus instance is used for all round executions.

} ```

### Tokio runtime

The implementation is fully based on Tokio-rs/Runtime. That said the recommended way of setting up the runtime is shown below.

rust tokio::runtime::Builder::new_multi_thread() // A thread per an accumulator worker so that CPU-bound operations // (the agreement verification) does not impact negatively main tokio tasks. .worker_threads(2 + consensus::config::ACCUMULATOR_WORKERS_AMOUNT) // Enable the time driver so we can use timeout feature in all steps execution. .enable_time() .build() .unwrap() .block_on(async { ... } )

Build, Run and Test

```bash

Run unit tests

cargo test ```

```bash

Build consensus

cargo b --release ```

```bash

Build and Run in-process testbed example

cargo run --release --example testbed ```

```bash

Build example node

cargo b --release --example node

Run example node

export DUSKWALLETDIR="TBD" export DUSKCONSENSUSKEYS_PASS="TBD"

USAGE: node --bootstrap ... --address --preloaded-num --provisioner-unique-id --log-level

```