add(book): Describes custom Testnets in the Zebra book (#8636)

* Adds documention to the Zebra book about custom Testnets, allows for configuring custom Testnet genesis hashes, refactors check for compatible custom Testnets, adds a TODO, and uses the default testnet when there's an empty or default `testnet_parameters` field in Zebra's network config.

* Adds example configs to custom Testnet docs, moves note about inbound connections on Regtest to a footnote.

* Adds a changelog entry

* Apply suggestions from code review

Co-authored-by: Marek <mail@marek.onl>

* Addresses suggestions from code review.

* Apply suggestions from code review

Co-authored-by: Arya <aryasolhi@gmail.com>

* Update book/src/user/custom-testnets.md

---------

Co-authored-by: Marek <mail@marek.onl>
This commit is contained in:
Arya 2024-06-27 20:33:17 -04:00 committed by GitHub
parent 3b49f4212b
commit 979ed0d720
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
8 changed files with 239 additions and 25 deletions

View File

@ -21,6 +21,10 @@ and this project adheres to [Semantic Versioning](https://semver.org).
- Added Windows to the list of supported platforms in Tier 2 ([#8637](https://github.com/ZcashFoundation/zebra/pull/8637))
- Zebra now drops transactions with unpaid actions in block templates and from the mempool.
### Added
- Added a page to the Zebra book describing custom Testnets ([#8636](https://github.com/ZcashFoundation/zebra/pull/8636))
## [Zebra 1.7.0](https://github.com/ZcashFoundation/zebra/releases/tag/v1.7.0) - 2024-05-07
In this release we introduce Regtest functionality to Zebra and restored Windows support. Also adjusted our Zebra release interval from 2 weeks to 6 weeks approximately.

View File

@ -20,7 +20,8 @@
- [Shielded Scanning gRPC Server](user/shielded-scan-grpc-server.md)
- [Kibana blockchain explorer](user/elasticsearch.md)
- [Forking the Zcash Testnet with Zebra](user/fork-zebra-testnet.md)
- [Regtest with Zebra](user/regtest.md)
- [Custom Testnets](user/custom-testnets.md)
- [Regtest with Zebra](user/regtest.md)
- [OpenAPI specification](user/openapi.md)
- [Troubleshooting](user/troubleshooting.md)
- [Developer Documentation](dev.md)

View File

@ -0,0 +1,180 @@
# Custom Testnets
Custom Testnets in Zebra enable testing consensus rule changes on a public, configured Testnet independent of the default public Zcash Testnet.
Zebra's Testnet can be configured with custom:
- Network upgrade activation heights,
- Network names,
- Network magics,
- Slow start intervals,
- Genesis hashes, and
- Target difficulty limits.
It's also possible to disable Proof-of-Work validation by setting `disable_pow` to `true` so that blocks can be mined onto the chain without valid Equihash solutions, nor block hashes below their target difficulties.
Configuring any of those Testnet parameters except the network name with non-default values will result in an incompatible custom Testnet. Incompatible Testnets will fail to successfully complete peer handshakes with one another, or could provide one another with invalid blocks or invalid mempool transactions. Peer node connections that consistently provide invalid blocks or mempool transactions should be considered misbehaving peer connections and dropped.
All of these parameters are optional, if they are all omitted or set to their default values, Zebra will run on the default public Testnet.
## Usage
In order to use a custom Testnet, Zebra must be configured to run on Testnet with non-default Testnet parameters. If the node is meant to mine blocks, it will need a `[mining]` section, and if it's meant to mine blocks with non-coinbase transactions, it will also need the `[rpc]` section so the `send_raw_transaction` RPC method is available.
Relevant parts of the configuration file with practical Testnet parameters:
```toml
[mining]
miner_address = 't27eWDgjFYJGVXmzrXeVjnb5J3uXDM9xH9v'
[network]
network = "Testnet"
[network.testnet_parameters.activation_heights]
network_name = "ConfiguredTestnet_1"
# The Testnet network magic is not reserved, but it's not recommended
# for use with incompatible Testnet parameters like those in this config.
network_magic = [0, 1, 0, 255]
slow_start_interval = 0
target_difficulty_limit = "0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f"
disable_pow = true
# Configured activation heights must be greater than 0, and less than
# 2^31. Block height 0 is reserved for the Genesis network upgrade in Zebra.
#
# Network upgrades must be activated in the order that they were added to Zcash,
# configuring the activation heights of recent network upgrades will activate
# any omitted prior network upgrades at the same height.
#
# For example, configuring the activation height of NU5 to block height 1 without
# configuring any other network upgrade activation heights will set the
# activation heights of BeforeOverwinter, Overwinter, Sapling, Blossom,
# Heartwood, and Canopy at block height 1 as well.
[network.testnet_parameters.activation_heights]
NU5 = 1
# This section may be omitted if it's not necessary to
# add transactions to Zebra's mempool
[rpc]
listen_addr = "0.0.0.0:18232"
```
Relevant parts of the configuration file with some Mainnet parameters[^fn1]:
```toml
[mining]
# The Mainnet address prefixes are reserved, all custom Testnets currently
# use the default public Testnet's address prefixes.
miner_address = 't27eWDgjFYJGVXmzrXeVjnb5J3uXDM9xH9v'
[network]
network = "Testnet"
[network.testnet_parameters.activation_heights]
# The Mainnet, Testnet, and Regtest network names are reserved.
network_name = "ConfiguredTestnet_2"
# The Mainnet and Regtest network magics are reserved.
network_magic = [0, 1, 0, 255]
slow_start_interval = 20_000
genesis_hash = "00040fe8ec8471911baa1db1266ea15dd06b4a8a5c453883c000b031973dce08"
# Note that setting `disable_pow` to `false` with this target difficultly
# limit will make it very difficult to mine valid blocks onto the chain and
# is not recommended.
target_difficulty_limit = "0008000000000000000000000000000000000000000000000000000000000000"
disable_pow = false
# Note that these activation heights are not recommended unless copying the
# Mainnet best chain and checkpoints to a custom Testnet. Custom Testnets will
# only try to use the Testnet checkpoints (if the checkpoint file's genesis hash
# matches the expected network genesis hash), so the Mainnet checkpoints will need
# to be copied into the Testnet checkpoints file as well.
[network.testnet_parameters.activation_heights]
BeforeOverwinter = 1
Overwinter = 347_500
Sapling = 419_200
Blossom = 653_600
Heartwood = 903_000
Canopy = 1_046_400
NU5 = 1_687_104
# This section may be omitted if it's not necessary to
# add transactions to Zebra's mempool
[rpc]
listen_addr = "0.0.0.0:18232"
```
## Caveats and Restrictions
There are a few caveats:
- Configured network upgrade activation heights must be above the genesis block height, which is reserved for Zebra's `Genesis` network upgrade, and must not be above Zebra's max block height of `2^31 - 1`[^fn2].
- While it's possible to activate Canopy and later network upgrades after height 1, Zebra cannot currently produce pre-Canopy block templates, so the `getblocktemplate` RPC method and Zebra's internal miner which depends on the `getblocktemplate` method won't work until Canopy is activated. An alternative block source will be required to mine pre-Canopy blocks onto Zebra's chain.
- While it's possible to use the default Testnet network magic with a configured Testnet, Zebra will panic when configured to use the default initial Testnet peers and Testnet parameters that are incompatible with the default public Testnet[^fn3].
- If the genesis hash is configured, a genesis block will need to be copied into the custom Testnet state or submitted via the `submitblock` RPC method, Zebra cannot currently generate genesis blocks. See the `CreateGenesisBlock()` function in `zcashd/src/chainparams.cpp` for use cases that require a new genesis block.
There are also a few other restrictions on these parameters:
- The network name must:
- not be any of the reserved network names: `Testnet`, `Mainnet`, and `Regtest`,
- contain only alphanumeric characters and underscores, and
- be shorter than the `MAX_NETWORK_NAME_LENGTH` of `30`.
- The network magic must not be any of the reserved network magics: `[36, 233, 39, 100]` and `[170, 232, 63, 95]`, these are the `Mainnet` and `Regtest` network magics respectively.
- The network upgrade activation heights must be in order, such that the activation height for every network upgrade is at or above the activation height of every preceding network upgrade.
## Comparison To Mainnet and Default Public Testnet Consensus Rules
Aside from the configurable parameters, custom Testnets in Zebra validate the same consensus rules as Testnet.
### Differences Between Mainnet and Testnet Consensus Rules
Zebra's Testnet validates almost all of the same consensus rules as Mainnet, the differences are:
- Constants defined in the `zcash_primitives::consensus::Parameters` trait, which includes but may not be limited to:
- Zcash address prefixes (see [`NetworkConstants`](https://docs.rs/zcash_primitives/latest/zcash_primitives/consensus/trait.NetworkConstants.html)), and coin type, which is `133` on `Mainnet` or `1` elsewhere.
- Network upgrade activation heights.
- Constants defined in Zebra:
- `PoWLimit` defined in the Zcash protocol specification, or target difficulty limit, which is `2^243 - 1` on Mainnet and `2^251 - 1` on the default Testnet.
- Expected genesis block hash.
- Number of funding streams, which is 48 on Mainnet and 51 on Testnet.
- The first block subsidy halving height, which is `1,046,400` (Canopy activation) on Mainnet and `1,116,000` on Testnet
- The Testnet minimum difficulty rule, validated by the `difficulty_threshold_and_time_are_valid()` function in `zebra_state::service::check` and applied to block templates by the `adjust_difficulty_and_time_for_testnet()` function in `zebra_state::service::read::difficulty`, both of which use the `AdjustedDifficulty::expected_difficulty_threshold()` method in `zebra_state::service::check::difficulty` to calculate the expected difficulty.
- Max block time is always enforced on Mainnet, but only enforced after block height `653,606` on Testnets (block times later than the median block time + a constant are rejected while this rule is enforced, this is part of the block header consensus rules in the Zcash specification).
### Configuring More Testnet Parameters To Match Mainnet Consensus Rules
The Mainnet Zcash address prefixes, coin type, network name, and network magic should remain reserved for Mainnet.
The network upgrade activation heights, target difficulty limit, slow start interval, and genesis hash of a custom Testnet could currently be configured in Zebra to match Mainnet.
The remaining consensus differences between Mainnet and Testnet could be made configurable in Zebra so that they could be configured to match Mainnet.
## Differences Between Custom Testnets and Regtest
Zebra's Regtest network is a special case of a custom Testnet that:
- Won't make peer connections[^fn4],
- Skips Proof-of-Work validation,
- Uses a reserved network magic and network name,
- Activates network upgrades up to and including Canopy at block height 1,
- Tries to closely match the `zcashd` Regtest parameters, and
- Expects the Regtest genesis hash.
Once Zebra's internal miner can mine Equihash solutions with configurable parameters, Zebra's Regtest should validate Proof-of-Work with the zcashd Regtest Equihash parameters unless it's disabled in its configuration, and custom Testnets should allow for configuring their Equihash parameters as well.
In the future, Zebra may also allow for disabling peers on custom Testnets so that the only unique parameters of Zebra's Regtest will be the network name and magic.
## Peer Connections In Custom Testnets
Aside from the network name, configuring any Testnet parameters in Zebra will result in an incompatible custom Testnet such that it cannot use the default initial Testnet peers.
In the absence of a configurable Zcash DNS seeder, Zebra nodes on custom Testnets will need to know the exact hostname or IP address of other Zebra nodes on the same custom Testnet to make peer connections.
Zebra nodes on custom Testnets will also reject peer connections with nodes that are using a different network magic or network protocol version, but may still make peer connections with other Zcash nodes which have incompatible network parameters. Zebra nodes should eventually drop those peer connections when it reaches its peerset connection limit and has more available peer candidates if they are consistently sending the node invalid blocks.
##### Footnotes
[^fn1]: Only the network upgrade activation heights, target difficulty limit, slow start interval, and genesis hash of a custom Testnet may currently be configured in Zebra to match Mainnet, other differences between Mainnet and Testnet consensus rules still remain with any custom Testnet configuration. More parameters may become configurable in the future, but the Mainnet Zcash address prefixes, coin type, network name, and network magic should remain reserved only for Mainnet.
[^fn2]: Zebra's max on-disk serialized block height is currently `2^24 - 1`, the max block height of `2^31 - 1` can only be represented in-memory, so while an activation height of `2^31 - 1` is valid, Zebra's best chain would not currently be able to reach that activation height.
[^fn3]: Configuring any of the Testnet parameters that are currently configurable except the network name will result in an incompatible custom Testnet, these are: the network magic, network upgrade activation heights, slow start interval, genesis hash, disabled Proof-of-Work and target difficulty limit.
[^fn4]: Zebra won't make outbound peer connections on Regtest, but currently still listens for inbound peer connections, which will be rejected unless they use the Regtest network magic, and Zcash nodes using the Regtest network magic should not be making outbound peer connections. It may be updated to skip initialization of the peerset service altogether so that it won't listen for peer connections at all when support for isolated custom Testnets is added.

View File

@ -1,9 +1,11 @@
# Regtest with Zebra
The Regtest network in Zebra enables testing of custom functionalities in a private testnet environment with configurable network upgrade activation heights. It allows for starting an isolated node which won't connect to any peers and currently allows for committing blocks without validating their Proof of Work (in the future, it may use a very low target difficulty and easier Equihash parameters instead of skipping Proof of Work validation altogether).
The Regtest network in Zebra enables testing of custom functionalities in a private Testnet environment with configurable network upgrade activation heights. It allows for starting an isolated node which won't connect to any peers and currently allows for committing blocks without validating their Proof of Work (in the future, it may use a very low target difficulty and easier Equihash parameters instead of skipping Proof of Work validation altogether).
Zebra always activates the Canopy network upgrade at block height 1 due to limitations on its block construction.
## Usage
In order to use Regtest, Zebra must be configured to run on the Regtest network. The `[mining]` section is also necessary for mining blocks, and the `[rpc]` section is necessary for using the `send_raw_transaction` RPC method to mine non-coinbase transactions onto the chain.
Relevant parts of the configuration file:
@ -40,7 +42,7 @@ There are two ways to commit blocks to Zebra's state on Regtest:
- Using the `getblocktemplate` and `submitblock` RPC methods directly
- Using Zebra's experimental `internal-miner` feature
## Using Zebra's Internal Miner
### Using Zebra's Internal Miner
Zebra can mine blocks on the Regtest network when compiled with the experimental `internal-miner` compilation feature and configured to enable to internal miner.
@ -55,7 +57,7 @@ Zebra should now mine blocks on Regtest when it starts after a short delay (of a
To confirm that it's working, look for `successfully mined a new block` messages in the logs, or that the tip height is increasing.
## Using RPC methods directly
### Using RPC methods directly
Blocks could also be mined outside of Zebra and submitted via Zebra's RPC methods. This requires enabling the RPC server in the configuration by providing a `listen_addr` field:

View File

@ -228,6 +228,7 @@ impl Network {
}
/// Returns the Sapling activation height for this network.
// TODO: Return an `Option` here now that network upgrade activation heights are configurable on Regtest and custom Testnets
pub fn sapling_activation_height(&self) -> Height {
super::NetworkUpgrade::Sapling
.activation_height(self)

View File

@ -43,7 +43,7 @@ const TESTNET_GENESIS_HASH: &str =
"05a60a92d99d85997cce3b87616c089f6124d7342af37106edc76126334a2c38";
/// Configurable activation heights for Regtest and configured Testnets.
#[derive(Deserialize, Default)]
#[derive(Deserialize, Default, Clone)]
#[serde(rename_all = "PascalCase")]
pub struct ConfiguredActivationHeights {
/// Activation height for `BeforeOverwinter` network upgrade.
@ -277,6 +277,26 @@ impl ParametersBuilder {
pub fn to_network(self) -> Network {
Network::new_configured_testnet(self.finish())
}
/// Returns true if these [`Parameters`] should be compatible with the default Testnet parameters.
pub fn is_compatible_with_default_parameters(&self) -> bool {
let Self {
network_name: _,
network_magic,
genesis_hash,
activation_heights,
slow_start_interval,
target_difficulty_limit,
disable_pow,
} = Self::default();
self.activation_heights == activation_heights
&& self.network_magic == network_magic
&& self.genesis_hash == genesis_hash
&& self.slow_start_interval == slow_start_interval
&& self.target_difficulty_limit == target_difficulty_limit
&& self.disable_pow == disable_pow
}
}
/// Network consensus parameters for test networks such as Regtest and the default Testnet.

View File

@ -647,6 +647,7 @@ impl<'de> Deserialize<'de> for Config {
slow_start_interval: Option<u32>,
target_difficulty_limit: Option<String>,
disable_pow: Option<bool>,
genesis_hash: Option<String>,
activation_heights: Option<ConfiguredActivationHeights>,
}
@ -732,28 +733,13 @@ impl<'de> Deserialize<'de> for Config {
slow_start_interval,
target_difficulty_limit,
disable_pow,
genesis_hash,
activation_heights,
}),
) => {
let mut params_builder = testnet::Parameters::build();
// TODO: allow default peers when fields match default testnet values?
let should_avoid_default_peers = network_magic.is_some()
|| slow_start_interval.is_some()
|| target_difficulty_limit.is_some()
|| disable_pow == Some(true)
|| activation_heights.is_some();
// Return an error if the initial testnet peers includes any of the default initial Mainnet or Testnet
// peers while activation heights or a custom network magic is configured.
if should_avoid_default_peers
&& contains_default_initial_peers(&initial_testnet_peers)
{
return Err(de::Error::custom(
"cannot use default initials peers with incompatible testnet",
));
}
if let Some(network_name) = network_name {
if let Some(network_name) = network_name.clone() {
params_builder = params_builder.with_network_name(network_name)
}
@ -761,13 +747,17 @@ impl<'de> Deserialize<'de> for Config {
params_builder = params_builder.with_network_magic(Magic(network_magic));
}
if let Some(genesis_hash) = genesis_hash {
params_builder = params_builder.with_genesis_hash(genesis_hash);
}
if let Some(slow_start_interval) = slow_start_interval {
params_builder = params_builder.with_slow_start_interval(
slow_start_interval.try_into().map_err(de::Error::custom)?,
);
}
if let Some(target_difficulty_limit) = target_difficulty_limit {
if let Some(target_difficulty_limit) = target_difficulty_limit.clone() {
params_builder = params_builder.with_target_difficulty_limit(
target_difficulty_limit
.parse::<U256>()
@ -780,11 +770,26 @@ impl<'de> Deserialize<'de> for Config {
}
// Retain default Testnet activation heights unless there's an empty [testnet_parameters.activation_heights] section.
if let Some(activation_heights) = activation_heights {
if let Some(activation_heights) = activation_heights.clone() {
params_builder = params_builder.with_activation_heights(activation_heights)
}
params_builder.to_network()
// Return an error if the initial testnet peers includes any of the default initial Mainnet or Testnet
// peers and the configured network parameters are incompatible with the default public Testnet.
if !params_builder.is_compatible_with_default_parameters()
&& contains_default_initial_peers(&initial_testnet_peers)
{
return Err(de::Error::custom(
"cannot use default initials peers with incompatible testnet",
));
};
// Return the default Testnet if no network name was configured and all parameters match the default Testnet
if network_name.is_none() && params_builder == testnet::Parameters::build() {
Network::new_default_testnet()
} else {
params_builder.to_network()
}
}
};

View File

@ -62,6 +62,7 @@ network_magic = [0, 0, 0, 0]
slow_start_interval = 0
target_difficulty_limit = "0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f0f"
disable_pow = true
genesis_hash = "00040fe8ec8471911baa1db1266ea15dd06b4a8a5c453883c000b031973dce08"
[network.testnet_parameters.activation_heights]
BeforeOverwinter = 1