mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-05 10:29:40 +00:00
Add bootnode code prior used in testnet-internal (#554)
* Add bootnode code prior used in testnet-internal Also performs the devnet/testnet differentation done since the testnet branch. * Fixes * fmt
This commit is contained in:
parent
fcad402186
commit
33dd412e67
6 changed files with 169 additions and 13 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -7613,6 +7613,7 @@ dependencies = [
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"hex",
|
"hex",
|
||||||
"jsonrpsee",
|
"jsonrpsee",
|
||||||
|
"libp2p",
|
||||||
"pallet-transaction-payment-rpc",
|
"pallet-transaction-payment-rpc",
|
||||||
"rand_core",
|
"rand_core",
|
||||||
"sc-authority-discovery",
|
"sc-authority-discovery",
|
||||||
|
|
|
@ -26,6 +26,8 @@ hex = "0.4"
|
||||||
rand_core = "0.6"
|
rand_core = "0.6"
|
||||||
schnorrkel = "0.11"
|
schnorrkel = "0.11"
|
||||||
|
|
||||||
|
libp2p = "0.52"
|
||||||
|
|
||||||
sp-core = { git = "https://github.com/serai-dex/substrate" }
|
sp-core = { git = "https://github.com/serai-dex/substrate" }
|
||||||
sp-keystore = { git = "https://github.com/serai-dex/substrate" }
|
sp-keystore = { git = "https://github.com/serai-dex/substrate" }
|
||||||
sp-timestamp = { git = "https://github.com/serai-dex/substrate" }
|
sp-timestamp = { git = "https://github.com/serai-dex/substrate" }
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
use core::marker::PhantomData;
|
use core::marker::PhantomData;
|
||||||
|
use std::collections::HashSet;
|
||||||
|
|
||||||
use sp_core::Pair as PairTrait;
|
use sp_core::{Decode, Pair as PairTrait, sr25519::Public};
|
||||||
|
|
||||||
use sc_service::ChainType;
|
use sc_service::ChainType;
|
||||||
|
|
||||||
|
@ -23,7 +24,7 @@ fn wasm_binary() -> Vec<u8> {
|
||||||
WASM_BINARY.ok_or("compiled in wasm not available").unwrap().to_vec()
|
WASM_BINARY.ok_or("compiled in wasm not available").unwrap().to_vec()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn testnet_genesis(
|
fn devnet_genesis(
|
||||||
wasm_binary: &[u8],
|
wasm_binary: &[u8],
|
||||||
validators: &[&'static str],
|
validators: &[&'static str],
|
||||||
endowed_accounts: Vec<PublicKey>,
|
endowed_accounts: Vec<PublicKey>,
|
||||||
|
@ -72,6 +73,57 @@ fn testnet_genesis(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn testnet_genesis(wasm_binary: &[u8], validators: Vec<&'static str>) -> RuntimeGenesisConfig {
|
||||||
|
let validators = validators
|
||||||
|
.into_iter()
|
||||||
|
.map(|validator| Public::decode(&mut hex::decode(validator).unwrap().as_slice()).unwrap())
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
assert_eq!(validators.iter().collect::<HashSet<_>>().len(), validators.len());
|
||||||
|
|
||||||
|
RuntimeGenesisConfig {
|
||||||
|
system: SystemConfig { code: wasm_binary.to_vec(), _config: PhantomData },
|
||||||
|
|
||||||
|
transaction_payment: Default::default(),
|
||||||
|
|
||||||
|
coins: CoinsConfig {
|
||||||
|
accounts: validators
|
||||||
|
.iter()
|
||||||
|
.map(|a| (*a, Balance { coin: Coin::Serai, amount: Amount(5_000_000 * 10_u64.pow(8)) }))
|
||||||
|
.collect(),
|
||||||
|
_ignore: Default::default(),
|
||||||
|
},
|
||||||
|
|
||||||
|
dex: DexConfig {
|
||||||
|
pools: vec![Coin::Bitcoin, Coin::Ether, Coin::Dai, Coin::Monero],
|
||||||
|
_ignore: Default::default(),
|
||||||
|
},
|
||||||
|
|
||||||
|
validator_sets: ValidatorSetsConfig {
|
||||||
|
networks: serai_runtime::primitives::NETWORKS
|
||||||
|
.iter()
|
||||||
|
.map(|network| match network {
|
||||||
|
NetworkId::Serai => (NetworkId::Serai, Amount(50_000 * 10_u64.pow(8))),
|
||||||
|
NetworkId::Bitcoin => (NetworkId::Bitcoin, Amount(1_000_000 * 10_u64.pow(8))),
|
||||||
|
NetworkId::Ethereum => (NetworkId::Ethereum, Amount(1_000_000 * 10_u64.pow(8))),
|
||||||
|
NetworkId::Monero => (NetworkId::Monero, Amount(100_000 * 10_u64.pow(8))),
|
||||||
|
})
|
||||||
|
.collect(),
|
||||||
|
participants: validators.clone(),
|
||||||
|
},
|
||||||
|
signals: SignalsConfig::default(),
|
||||||
|
babe: BabeConfig {
|
||||||
|
authorities: validators.iter().map(|validator| ((*validator).into(), 1)).collect(),
|
||||||
|
epoch_config: Some(BABE_GENESIS_EPOCH_CONFIG),
|
||||||
|
_config: PhantomData,
|
||||||
|
},
|
||||||
|
grandpa: GrandpaConfig {
|
||||||
|
authorities: validators.into_iter().map(|validator| (validator.into(), 1)).collect(),
|
||||||
|
_config: PhantomData,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn development_config() -> ChainSpec {
|
pub fn development_config() -> ChainSpec {
|
||||||
let wasm_binary = wasm_binary();
|
let wasm_binary = wasm_binary();
|
||||||
|
|
||||||
|
@ -82,7 +134,7 @@ pub fn development_config() -> ChainSpec {
|
||||||
"devnet",
|
"devnet",
|
||||||
ChainType::Development,
|
ChainType::Development,
|
||||||
move || {
|
move || {
|
||||||
testnet_genesis(
|
devnet_genesis(
|
||||||
&wasm_binary,
|
&wasm_binary,
|
||||||
&["Alice"],
|
&["Alice"],
|
||||||
vec![
|
vec![
|
||||||
|
@ -100,7 +152,7 @@ pub fn development_config() -> ChainSpec {
|
||||||
// Telemetry
|
// Telemetry
|
||||||
None,
|
None,
|
||||||
// Protocol ID
|
// Protocol ID
|
||||||
Some("serai"),
|
Some("serai-devnet"),
|
||||||
// Fork ID
|
// Fork ID
|
||||||
None,
|
None,
|
||||||
// Properties
|
// Properties
|
||||||
|
@ -110,7 +162,7 @@ pub fn development_config() -> ChainSpec {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn testnet_config() -> ChainSpec {
|
pub fn local_config() -> ChainSpec {
|
||||||
let wasm_binary = wasm_binary();
|
let wasm_binary = wasm_binary();
|
||||||
|
|
||||||
ChainSpec::from_genesis(
|
ChainSpec::from_genesis(
|
||||||
|
@ -120,7 +172,7 @@ pub fn testnet_config() -> ChainSpec {
|
||||||
"local",
|
"local",
|
||||||
ChainType::Local,
|
ChainType::Local,
|
||||||
move || {
|
move || {
|
||||||
testnet_genesis(
|
devnet_genesis(
|
||||||
&wasm_binary,
|
&wasm_binary,
|
||||||
&["Alice", "Bob", "Charlie", "Dave"],
|
&["Alice", "Bob", "Charlie", "Dave"],
|
||||||
vec![
|
vec![
|
||||||
|
@ -138,7 +190,7 @@ pub fn testnet_config() -> ChainSpec {
|
||||||
// Telemetry
|
// Telemetry
|
||||||
None,
|
None,
|
||||||
// Protocol ID
|
// Protocol ID
|
||||||
Some("serai"),
|
Some("serai-local"),
|
||||||
// Fork ID
|
// Fork ID
|
||||||
None,
|
None,
|
||||||
// Properties
|
// Properties
|
||||||
|
@ -147,3 +199,39 @@ pub fn testnet_config() -> ChainSpec {
|
||||||
None,
|
None,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn testnet_config() -> ChainSpec {
|
||||||
|
let wasm_binary = wasm_binary();
|
||||||
|
|
||||||
|
ChainSpec::from_genesis(
|
||||||
|
// Name
|
||||||
|
"Test Network 2",
|
||||||
|
// ID
|
||||||
|
"testnet-2",
|
||||||
|
ChainType::Live,
|
||||||
|
move || {
|
||||||
|
let _ = testnet_genesis(&wasm_binary, vec![]);
|
||||||
|
todo!()
|
||||||
|
},
|
||||||
|
// Bootnodes
|
||||||
|
vec![],
|
||||||
|
// Telemetry
|
||||||
|
None,
|
||||||
|
// Protocol ID
|
||||||
|
Some("serai-testnet-2"),
|
||||||
|
// Fork ID
|
||||||
|
None,
|
||||||
|
// Properties
|
||||||
|
None,
|
||||||
|
// Extensions
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn bootnode_multiaddrs(id: &str) -> Vec<libp2p::Multiaddr> {
|
||||||
|
match id {
|
||||||
|
"devnet" | "local" => vec![],
|
||||||
|
"testnet-2" => todo!(),
|
||||||
|
_ => panic!("unrecognized network ID"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -40,7 +40,8 @@ impl SubstrateCli for Cli {
|
||||||
fn load_spec(&self, id: &str) -> Result<Box<dyn sc_service::ChainSpec>, String> {
|
fn load_spec(&self, id: &str) -> Result<Box<dyn sc_service::ChainSpec>, String> {
|
||||||
match id {
|
match id {
|
||||||
"dev" | "devnet" => Ok(Box::new(chain_spec::development_config())),
|
"dev" | "devnet" => Ok(Box::new(chain_spec::development_config())),
|
||||||
"local" => Ok(Box::new(chain_spec::testnet_config())),
|
"local" => Ok(Box::new(chain_spec::local_config())),
|
||||||
|
"testnet" => Ok(Box::new(chain_spec::testnet_config())),
|
||||||
_ => panic!("Unknown network ID"),
|
_ => panic!("Unknown network ID"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,7 @@ pub use sc_rpc_api::DenyUnsafe;
|
||||||
use sc_transaction_pool_api::TransactionPool;
|
use sc_transaction_pool_api::TransactionPool;
|
||||||
|
|
||||||
pub struct FullDeps<C, P> {
|
pub struct FullDeps<C, P> {
|
||||||
|
pub id: String,
|
||||||
pub client: Arc<C>,
|
pub client: Arc<C>,
|
||||||
pub pool: Arc<P>,
|
pub pool: Arc<P>,
|
||||||
pub deny_unsafe: DenyUnsafe,
|
pub deny_unsafe: DenyUnsafe,
|
||||||
|
@ -46,18 +47,19 @@ where
|
||||||
use pallet_transaction_payment_rpc::{TransactionPayment, TransactionPaymentApiServer};
|
use pallet_transaction_payment_rpc::{TransactionPayment, TransactionPaymentApiServer};
|
||||||
|
|
||||||
let mut module = RpcModule::new(());
|
let mut module = RpcModule::new(());
|
||||||
let FullDeps { client, pool, deny_unsafe, authority_discovery } = deps;
|
let FullDeps { id, client, pool, deny_unsafe, authority_discovery } = deps;
|
||||||
|
|
||||||
module.merge(System::new(client.clone(), pool, deny_unsafe).into_rpc())?;
|
module.merge(System::new(client.clone(), pool, deny_unsafe).into_rpc())?;
|
||||||
module.merge(TransactionPayment::new(client.clone()).into_rpc())?;
|
module.merge(TransactionPayment::new(client.clone()).into_rpc())?;
|
||||||
|
|
||||||
if let Some(authority_discovery) = authority_discovery {
|
if let Some(authority_discovery) = authority_discovery {
|
||||||
let mut authority_discovery_module = RpcModule::new((client, RwLock::new(authority_discovery)));
|
let mut authority_discovery_module =
|
||||||
|
RpcModule::new((id, client, RwLock::new(authority_discovery)));
|
||||||
authority_discovery_module.register_async_method(
|
authority_discovery_module.register_async_method(
|
||||||
"p2p_validators",
|
"p2p_validators",
|
||||||
|params, context| async move {
|
|params, context| async move {
|
||||||
let network: NetworkId = params.parse()?;
|
let network: NetworkId = params.parse()?;
|
||||||
let (client, authority_discovery) = &*context;
|
let (id, client, authority_discovery) = &*context;
|
||||||
let latest_block = client.info().best_hash;
|
let latest_block = client.info().best_hash;
|
||||||
|
|
||||||
let validators = client.runtime_api().validators(latest_block, network).map_err(|_| {
|
let validators = client.runtime_api().validators(latest_block, network).map_err(|_| {
|
||||||
|
@ -66,7 +68,9 @@ where
|
||||||
"please report this at https://github.com/serai-dex/serai",
|
"please report this at https://github.com/serai-dex/serai",
|
||||||
)))
|
)))
|
||||||
})?;
|
})?;
|
||||||
let mut all_p2p_addresses = vec![];
|
// Always return the protocol's bootnodes
|
||||||
|
let mut all_p2p_addresses = crate::chain_spec::bootnode_multiaddrs(id);
|
||||||
|
// Additionally returns validators found over the DHT
|
||||||
for validator in validators {
|
for validator in validators {
|
||||||
let mut returned_addresses = authority_discovery
|
let mut returned_addresses = authority_discovery
|
||||||
.write()
|
.write()
|
||||||
|
|
|
@ -161,7 +161,7 @@ pub fn new_partial(
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_full(config: Configuration) -> Result<TaskManager, ServiceError> {
|
pub fn new_full(mut config: Configuration) -> Result<TaskManager, ServiceError> {
|
||||||
let (
|
let (
|
||||||
sc_service::PartialComponents {
|
sc_service::PartialComponents {
|
||||||
client,
|
client,
|
||||||
|
@ -176,6 +176,11 @@ pub fn new_full(config: Configuration) -> Result<TaskManager, ServiceError> {
|
||||||
keystore_container,
|
keystore_container,
|
||||||
) = new_partial(&config)?;
|
) = new_partial(&config)?;
|
||||||
|
|
||||||
|
config.network.node_name = "serai".to_string();
|
||||||
|
config.network.client_version = "0.1.0".to_string();
|
||||||
|
config.network.listen_addresses =
|
||||||
|
vec!["/ip4/0.0.0.0/tcp/30333".parse().unwrap(), "/ip6/::/tcp/30333".parse().unwrap()];
|
||||||
|
|
||||||
let mut net_config = sc_network::config::FullNetworkConfiguration::new(&config.network);
|
let mut net_config = sc_network::config::FullNetworkConfiguration::new(&config.network);
|
||||||
let grandpa_protocol_name =
|
let grandpa_protocol_name =
|
||||||
grandpa::protocol_standard_name(&client.block_hash(0).unwrap().unwrap(), &config.chain_spec);
|
grandpa::protocol_standard_name(&client.block_hash(0).unwrap().unwrap(), &config.chain_spec);
|
||||||
|
@ -203,6 +208,59 @@ pub fn new_full(config: Configuration) -> Result<TaskManager, ServiceError> {
|
||||||
warp_sync_params: Some(WarpSyncParams::WithProvider(warp_sync)),
|
warp_sync_params: Some(WarpSyncParams::WithProvider(warp_sync)),
|
||||||
})?;
|
})?;
|
||||||
|
|
||||||
|
task_manager.spawn_handle().spawn("bootnodes", "bootnodes", {
|
||||||
|
let network = network.clone();
|
||||||
|
let id = config.chain_spec.id().to_string();
|
||||||
|
|
||||||
|
async move {
|
||||||
|
// Transforms the above Multiaddrs into MultiaddrWithPeerIds
|
||||||
|
// While the PeerIds *should* be known in advance and hardcoded, that data wasn't collected in
|
||||||
|
// time and this fine for a testnet
|
||||||
|
let bootnodes = || async {
|
||||||
|
use libp2p::{Transport as TransportTrait, tcp::tokio::Transport, noise::Config};
|
||||||
|
|
||||||
|
let bootnode_multiaddrs = crate::chain_spec::bootnode_multiaddrs(&id);
|
||||||
|
|
||||||
|
let mut tasks = vec![];
|
||||||
|
for multiaddr in bootnode_multiaddrs {
|
||||||
|
tasks.push(tokio::time::timeout(
|
||||||
|
core::time::Duration::from_secs(10),
|
||||||
|
tokio::task::spawn(async move {
|
||||||
|
let Ok(noise) = Config::new(&sc_network::Keypair::generate_ed25519()) else { None? };
|
||||||
|
let mut transport = Transport::default()
|
||||||
|
.upgrade(libp2p::core::upgrade::Version::V1)
|
||||||
|
.authenticate(noise)
|
||||||
|
.multiplex(libp2p::yamux::Config::default());
|
||||||
|
let Ok(transport) = transport.dial(multiaddr.clone()) else { None? };
|
||||||
|
let Ok((peer_id, _)) = transport.await else { None? };
|
||||||
|
Some(sc_network::config::MultiaddrWithPeerId { multiaddr, peer_id })
|
||||||
|
}),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut res = vec![];
|
||||||
|
for task in tasks {
|
||||||
|
if let Ok(Ok(Some(bootnode))) = task.await {
|
||||||
|
res.push(bootnode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
res
|
||||||
|
};
|
||||||
|
|
||||||
|
use sc_network::{NetworkStatusProvider, NetworkPeers};
|
||||||
|
loop {
|
||||||
|
if let Ok(status) = network.status().await {
|
||||||
|
if status.num_connected_peers < 3 {
|
||||||
|
for bootnode in bootnodes().await {
|
||||||
|
let _ = network.add_reserved_peer(bootnode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tokio::time::sleep(core::time::Duration::from_secs(60)).await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
if config.offchain_worker.enabled {
|
if config.offchain_worker.enabled {
|
||||||
task_manager.spawn_handle().spawn(
|
task_manager.spawn_handle().spawn(
|
||||||
"offchain-workers-runner",
|
"offchain-workers-runner",
|
||||||
|
@ -258,11 +316,13 @@ pub fn new_full(config: Configuration) -> Result<TaskManager, ServiceError> {
|
||||||
};
|
};
|
||||||
|
|
||||||
let rpc_builder = {
|
let rpc_builder = {
|
||||||
|
let id = config.chain_spec.id().to_string();
|
||||||
let client = client.clone();
|
let client = client.clone();
|
||||||
let pool = transaction_pool.clone();
|
let pool = transaction_pool.clone();
|
||||||
|
|
||||||
Box::new(move |deny_unsafe, _| {
|
Box::new(move |deny_unsafe, _| {
|
||||||
crate::rpc::create_full(crate::rpc::FullDeps {
|
crate::rpc::create_full(crate::rpc::FullDeps {
|
||||||
|
id: id.clone(),
|
||||||
client: client.clone(),
|
client: client.clone(),
|
||||||
pool: pool.clone(),
|
pool: pool.clone(),
|
||||||
deny_unsafe,
|
deny_unsafe,
|
||||||
|
|
Loading…
Reference in a new issue