feature: Add network upgrade activation heights
This commit is contained in:
parent
4a98b8fa0d
commit
c9ee85c3b5
|
@ -9,45 +9,11 @@
|
||||||
//! Typically, consensus parameters are accessed via a function that takes a
|
//! Typically, consensus parameters are accessed via a function that takes a
|
||||||
//! `Network` and `BlockHeight`.
|
//! `Network` and `BlockHeight`.
|
||||||
|
|
||||||
use zebra_chain::block::BlockHeaderHash;
|
pub mod genesis;
|
||||||
use zebra_chain::{Network, Network::*};
|
pub mod network_upgrade;
|
||||||
|
|
||||||
/// A Zcash network protocol upgrade.
|
pub use genesis::*;
|
||||||
//
|
pub use network_upgrade::*;
|
||||||
// TODO: are new network upgrades a breaking change, or should we make this
|
|
||||||
// enum non-exhaustive?
|
|
||||||
pub enum NetworkUpgrade {
|
|
||||||
/// The Zcash protocol before the Overwinter upgrade.
|
|
||||||
///
|
|
||||||
/// We avoid using `Sprout`, because the specification says that Sprout
|
|
||||||
/// is the name of the pre-Sapling protocol, before and after Overwinter.
|
|
||||||
BeforeOverwinter,
|
|
||||||
/// The Zcash protocol after the Overwinter upgrade.
|
|
||||||
Overwinter,
|
|
||||||
/// The Zcash protocol after the Sapling upgrade.
|
|
||||||
Sapling,
|
|
||||||
/// The Zcash protocol after the Blossom upgrade.
|
|
||||||
Blossom,
|
|
||||||
/// The Zcash protocol after the Heartwood upgrade.
|
|
||||||
Heartwood,
|
|
||||||
/// The Zcash protocol after the Canopy upgrade.
|
|
||||||
Canopy,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// The previous block hash for the genesis block.
|
#[cfg(test)]
|
||||||
///
|
mod tests;
|
||||||
/// All known networks use the Bitcoin `null` value for the parent of the
|
|
||||||
/// genesis block. (In Bitcoin, `null` is `[0; 32]`.)
|
|
||||||
pub const GENESIS_PREVIOUS_BLOCK_HASH: BlockHeaderHash = BlockHeaderHash([0; 32]);
|
|
||||||
|
|
||||||
/// Returns the hash for the genesis block in `network`.
|
|
||||||
pub fn genesis_hash(network: Network) -> BlockHeaderHash {
|
|
||||||
match network {
|
|
||||||
// zcash-cli getblockhash 0 | zebrad revhex
|
|
||||||
Mainnet => "08ce3d9731b000c08338455c8a4a6bd05da16e26b11daa1b917184ece80f0400",
|
|
||||||
// zcash-cli -testnet getblockhash 0 | zebrad revhex
|
|
||||||
Testnet => "382c4a332661c7ed0671f32a34d724619f086c61873bce7c99859dd9920aa605",
|
|
||||||
}
|
|
||||||
.parse()
|
|
||||||
.expect("hard-coded hash parses")
|
|
||||||
}
|
|
||||||
|
|
|
@ -0,0 +1,22 @@
|
||||||
|
//! Genesis consensus parameters for each Zcash network.
|
||||||
|
|
||||||
|
use zebra_chain::block::BlockHeaderHash;
|
||||||
|
use zebra_chain::{Network, Network::*};
|
||||||
|
|
||||||
|
/// The previous block hash for the genesis block.
|
||||||
|
///
|
||||||
|
/// All known networks use the Bitcoin `null` value for the parent of the
|
||||||
|
/// genesis block. (In Bitcoin, `null` is `[0; 32]`.)
|
||||||
|
pub const GENESIS_PREVIOUS_BLOCK_HASH: BlockHeaderHash = BlockHeaderHash([0; 32]);
|
||||||
|
|
||||||
|
/// Returns the hash for the genesis block in `network`.
|
||||||
|
pub fn genesis_hash(network: Network) -> BlockHeaderHash {
|
||||||
|
match network {
|
||||||
|
// zcash-cli getblockhash 0 | zebrad revhex
|
||||||
|
Mainnet => "08ce3d9731b000c08338455c8a4a6bd05da16e26b11daa1b917184ece80f0400",
|
||||||
|
// zcash-cli -testnet getblockhash 0 | zebrad revhex
|
||||||
|
Testnet => "382c4a332661c7ed0671f32a34d724619f086c61873bce7c99859dd9920aa605",
|
||||||
|
}
|
||||||
|
.parse()
|
||||||
|
.expect("hard-coded hash parses")
|
||||||
|
}
|
|
@ -0,0 +1,109 @@
|
||||||
|
//! Network upgrade consensus parameters for Zcash.
|
||||||
|
|
||||||
|
use NetworkUpgrade::*;
|
||||||
|
|
||||||
|
use std::collections::BTreeMap;
|
||||||
|
use std::ops::Bound::*;
|
||||||
|
|
||||||
|
use zebra_chain::types::BlockHeight;
|
||||||
|
use zebra_chain::{Network, Network::*};
|
||||||
|
|
||||||
|
/// A Zcash network protocol upgrade.
|
||||||
|
//
|
||||||
|
// TODO: are new network upgrades a breaking change, or should we make this
|
||||||
|
// enum non-exhaustive?
|
||||||
|
#[derive(Copy, Clone, Debug, Eq, Hash, PartialEq)]
|
||||||
|
pub enum NetworkUpgrade {
|
||||||
|
/// The Zcash protocol before the Overwinter upgrade.
|
||||||
|
///
|
||||||
|
/// We avoid using `Sprout`, because the specification says that Sprout
|
||||||
|
/// is the name of the pre-Sapling protocol, before and after Overwinter.
|
||||||
|
BeforeOverwinter,
|
||||||
|
/// The Zcash protocol after the Overwinter upgrade.
|
||||||
|
Overwinter,
|
||||||
|
/// The Zcash protocol after the Sapling upgrade.
|
||||||
|
Sapling,
|
||||||
|
/// The Zcash protocol after the Blossom upgrade.
|
||||||
|
Blossom,
|
||||||
|
/// The Zcash protocol after the Heartwood upgrade.
|
||||||
|
Heartwood,
|
||||||
|
/// The Zcash protocol after the Canopy upgrade.
|
||||||
|
Canopy,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Mainnet network upgrade activation heights.
|
||||||
|
///
|
||||||
|
/// This is actually a bijective map, but it is const, so we use a vector, and
|
||||||
|
/// do the uniqueness check in the unit tests.
|
||||||
|
pub(crate) const MAINNET_ACTIVATION_HEIGHTS: &[(BlockHeight, NetworkUpgrade)] = &[
|
||||||
|
(BlockHeight(0), BeforeOverwinter),
|
||||||
|
(BlockHeight(347_500), Overwinter),
|
||||||
|
(BlockHeight(419_200), Sapling),
|
||||||
|
(BlockHeight(653_600), Blossom),
|
||||||
|
(BlockHeight(903_000), Heartwood),
|
||||||
|
(BlockHeight(1_046_400), Canopy),
|
||||||
|
];
|
||||||
|
|
||||||
|
/// Testnet network upgrade activation heights.
|
||||||
|
///
|
||||||
|
/// This is actually a bijective map, but it is const, so we use a vector, and
|
||||||
|
/// do the uniqueness check in the unit tests.
|
||||||
|
pub(crate) const TESTNET_ACTIVATION_HEIGHTS: &[(BlockHeight, NetworkUpgrade)] = &[
|
||||||
|
(BlockHeight(0), BeforeOverwinter),
|
||||||
|
(BlockHeight(207_500), Overwinter),
|
||||||
|
(BlockHeight(280_000), Sapling),
|
||||||
|
(BlockHeight(584_000), Blossom),
|
||||||
|
(BlockHeight(903_800), Heartwood),
|
||||||
|
// As of 21 July 2020, the Canopy testnet height has not been decided.
|
||||||
|
// See ZIP 251 for updates.
|
||||||
|
];
|
||||||
|
|
||||||
|
impl NetworkUpgrade {
|
||||||
|
/// Returns a BTreeMap of activation heights and network upgrades for
|
||||||
|
/// `network`.
|
||||||
|
///
|
||||||
|
/// If the activation height of a future upgrade is not known, that
|
||||||
|
/// network upgrade does not appear in the list.
|
||||||
|
///
|
||||||
|
/// This is actually a bijective map.
|
||||||
|
pub(crate) fn activation_list(network: Network) -> BTreeMap<BlockHeight, NetworkUpgrade> {
|
||||||
|
match network {
|
||||||
|
Mainnet => MAINNET_ACTIVATION_HEIGHTS,
|
||||||
|
Testnet => TESTNET_ACTIVATION_HEIGHTS,
|
||||||
|
}
|
||||||
|
.iter()
|
||||||
|
.cloned()
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the current network upgrade for `network` and `height`.
|
||||||
|
pub fn current(network: Network, height: BlockHeight) -> NetworkUpgrade {
|
||||||
|
NetworkUpgrade::activation_list(network)
|
||||||
|
.range(..=height)
|
||||||
|
.map(|(_, nu)| *nu)
|
||||||
|
.next_back()
|
||||||
|
.expect("every height has a current network upgrade")
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the next network upgrade for `network` and `height`.
|
||||||
|
///
|
||||||
|
/// Returns None if the name of the next upgrade has not been decided yet.
|
||||||
|
pub fn next(network: Network, height: BlockHeight) -> Option<NetworkUpgrade> {
|
||||||
|
NetworkUpgrade::activation_list(network)
|
||||||
|
.range((Excluded(height), Unbounded))
|
||||||
|
.map(|(_, nu)| *nu)
|
||||||
|
.next()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the activation height for this network upgrade on `network`.
|
||||||
|
///
|
||||||
|
/// Returns None if this network upgrade is a future upgrade, and its
|
||||||
|
/// activation height has not been set yet.
|
||||||
|
pub fn activation_height(&self, network: Network) -> Option<BlockHeight> {
|
||||||
|
NetworkUpgrade::activation_list(network)
|
||||||
|
.iter()
|
||||||
|
.filter(|(_, nu)| nu == &self)
|
||||||
|
.map(|(height, _)| *height)
|
||||||
|
.next()
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,105 @@
|
||||||
|
//! Consensus parameter tests for Zebra.
|
||||||
|
|
||||||
|
use super::*;
|
||||||
|
use NetworkUpgrade::*;
|
||||||
|
|
||||||
|
use std::collections::HashSet;
|
||||||
|
|
||||||
|
use zebra_chain::types::BlockHeight;
|
||||||
|
use zebra_chain::{Network, Network::*};
|
||||||
|
|
||||||
|
/// Check that the activation heights and network upgrades are unique.
|
||||||
|
#[test]
|
||||||
|
fn activation_bijective() {
|
||||||
|
let mainnet_activations = NetworkUpgrade::activation_list(Mainnet);
|
||||||
|
let mainnet_heights: HashSet<&BlockHeight> = mainnet_activations.keys().collect();
|
||||||
|
assert_eq!(MAINNET_ACTIVATION_HEIGHTS.len(), mainnet_heights.len());
|
||||||
|
|
||||||
|
let mainnet_nus: HashSet<&NetworkUpgrade> = mainnet_activations.values().collect();
|
||||||
|
assert_eq!(MAINNET_ACTIVATION_HEIGHTS.len(), mainnet_nus.len());
|
||||||
|
|
||||||
|
let testnet_activations = NetworkUpgrade::activation_list(Testnet);
|
||||||
|
let testnet_heights: HashSet<&BlockHeight> = testnet_activations.keys().collect();
|
||||||
|
assert_eq!(TESTNET_ACTIVATION_HEIGHTS.len(), testnet_heights.len());
|
||||||
|
|
||||||
|
let testnet_nus: HashSet<&NetworkUpgrade> = testnet_activations.values().collect();
|
||||||
|
assert_eq!(TESTNET_ACTIVATION_HEIGHTS.len(), testnet_nus.len());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn activation_extremes_mainnet() {
|
||||||
|
activation_extremes(Mainnet)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn activation_extremes_testnet() {
|
||||||
|
activation_extremes(Testnet)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Test the activation_list, activation_height, current, and next functions
|
||||||
|
/// for `network` with extreme values.
|
||||||
|
fn activation_extremes(network: Network) {
|
||||||
|
// The first two upgrades are BeforeOverwinter and Overwinter
|
||||||
|
assert_eq!(
|
||||||
|
NetworkUpgrade::activation_list(network).get(&BlockHeight(0)),
|
||||||
|
Some(&BeforeOverwinter)
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
BeforeOverwinter.activation_height(network),
|
||||||
|
Some(BlockHeight(0))
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
NetworkUpgrade::current(network, BlockHeight(0)),
|
||||||
|
BeforeOverwinter
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
NetworkUpgrade::next(network, BlockHeight(0)),
|
||||||
|
Some(Overwinter)
|
||||||
|
);
|
||||||
|
|
||||||
|
// We assume that the last upgrade we know about continues forever
|
||||||
|
// (even if we suspect that won't be true)
|
||||||
|
assert_ne!(
|
||||||
|
NetworkUpgrade::activation_list(network).get(&BlockHeight::MAX),
|
||||||
|
Some(&BeforeOverwinter)
|
||||||
|
);
|
||||||
|
assert_ne!(
|
||||||
|
NetworkUpgrade::current(network, BlockHeight::MAX),
|
||||||
|
BeforeOverwinter
|
||||||
|
);
|
||||||
|
assert_eq!(NetworkUpgrade::next(network, BlockHeight::MAX), None);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn activation_consistent_mainnet() {
|
||||||
|
activation_consistent(Mainnet)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn activation_consistent_testnet() {
|
||||||
|
activation_consistent(Testnet)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Check that the activation_height, current, and next functions are
|
||||||
|
/// consistent for `network`.
|
||||||
|
fn activation_consistent(network: Network) {
|
||||||
|
let activation_list = NetworkUpgrade::activation_list(network);
|
||||||
|
let network_upgrades: HashSet<&NetworkUpgrade> = activation_list.values().collect();
|
||||||
|
|
||||||
|
for &network_upgrade in network_upgrades {
|
||||||
|
let height = network_upgrade
|
||||||
|
.activation_height(network)
|
||||||
|
.expect("activations must have a height");
|
||||||
|
assert_eq!(NetworkUpgrade::current(network, height), network_upgrade);
|
||||||
|
// Network upgrades don't repeat
|
||||||
|
assert_ne!(NetworkUpgrade::next(network, height), Some(network_upgrade));
|
||||||
|
assert_ne!(
|
||||||
|
NetworkUpgrade::next(network, BlockHeight(height.0 + 1)),
|
||||||
|
Some(network_upgrade)
|
||||||
|
);
|
||||||
|
assert_ne!(
|
||||||
|
NetworkUpgrade::next(network, BlockHeight::MAX),
|
||||||
|
Some(network_upgrade)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -46,7 +46,7 @@ pub const CURRENT_VERSION: Version = Version(170_011);
|
||||||
///
|
///
|
||||||
/// Used to select the minimum supported version for peer connections.
|
/// Used to select the minimum supported version for peer connections.
|
||||||
//
|
//
|
||||||
// TODO: dynamically choose the minimum network upgrade based on block height.
|
// TODO: replace with NetworkUpgrade::current(network, height).
|
||||||
// See the detailed comment in handshake.rs, where this constant is used.
|
// See the detailed comment in handshake.rs, where this constant is used.
|
||||||
pub const MIN_NETWORK_UPGRADE: NetworkUpgrade = Heartwood;
|
pub const MIN_NETWORK_UPGRADE: NetworkUpgrade = Heartwood;
|
||||||
|
|
||||||
|
|
|
@ -194,8 +194,15 @@ where
|
||||||
// if (pfrom->nVersion < consensusParams.vUpgrades[currentEpoch].nProtocolVersion)
|
// if (pfrom->nVersion < consensusParams.vUpgrades[currentEpoch].nProtocolVersion)
|
||||||
//
|
//
|
||||||
// For approximately 1.5 days before a network upgrade, we also need to:
|
// For approximately 1.5 days before a network upgrade, we also need to:
|
||||||
// - prefer evicting pre-upgrade peers from the peer set, and
|
// - avoid old peers, and
|
||||||
// - prefer choosing post-upgrade ready peers for queries
|
// - prefer updated peers.
|
||||||
|
// For example, we could reject old peers with probability 0.5.
|
||||||
|
//
|
||||||
|
// At the network upgrade, we also need to disconnect from old peers.
|
||||||
|
// TODO: replace MIN_NETWORK_UPGRADE with
|
||||||
|
// NetworkUpgrade::current(network, height) where network is
|
||||||
|
// the configured network, and height is the best tip's block
|
||||||
|
// height.
|
||||||
|
|
||||||
if remote_version < Version::min_version(network, constants::MIN_NETWORK_UPGRADE) {
|
if remote_version < Version::min_version(network, constants::MIN_NETWORK_UPGRADE) {
|
||||||
// Disconnect if peer is using an obsolete version.
|
// Disconnect if peer is using an obsolete version.
|
||||||
|
|
Loading…
Reference in New Issue