use super::config::{ClusterTestOpt, Env};
use async_trait::async_trait;
use std::net::SocketAddr;
use std::path::Path;
use sui_config::local_ip_utils::get_available_port;
use sui_config::Config;
use sui_config::{PersistedConfig, SUI_KEYSTORE_FILENAME, SUI_NETWORK_CONFIG};
use sui_graphql_rpc::config::{ConnectionConfig, ServiceConfig};
use sui_graphql_rpc::test_infra::cluster::start_graphql_server_with_fn_rpc;
use sui_indexer::test_utils::{
start_indexer_jsonrpc_for_testing, start_indexer_writer_for_testing,
};
use sui_keys::keystore::{AccountKeystore, FileBasedKeystore, Keystore};
use sui_pg_db::temp::TempDb;
use sui_sdk::sui_client_config::{SuiClientConfig, SuiEnv};
use sui_sdk::wallet_context::WalletContext;
use sui_swarm::memory::Swarm;
use sui_swarm_config::genesis_config::GenesisConfig;
use sui_swarm_config::network_config::NetworkConfig;
use sui_types::base_types::SuiAddress;
use sui_types::crypto::KeypairTraits;
use sui_types::crypto::SuiKeyPair;
use sui_types::crypto::{get_key_pair, AccountKeyPair};
use tempfile::tempdir;
use test_cluster::{TestCluster, TestClusterBuilder};
use tracing::info;
const DEVNET_FAUCET_ADDR: &str = "https://faucet.devnet.sui.io:443";
const STAGING_FAUCET_ADDR: &str = "https://faucet.staging.sui.io:443";
const CONTINUOUS_FAUCET_ADDR: &str = "https://faucet.ci.sui.io:443";
const CONTINUOUS_NOMAD_FAUCET_ADDR: &str = "https://faucet.nomad.ci.sui.io:443";
const TESTNET_FAUCET_ADDR: &str = "https://faucet.testnet.sui.io:443";
const DEVNET_FULLNODE_ADDR: &str = "https://rpc.devnet.sui.io:443";
const STAGING_FULLNODE_ADDR: &str = "https://fullnode.staging.sui.io:443";
const CONTINUOUS_FULLNODE_ADDR: &str = "https://fullnode.ci.sui.io:443";
const CONTINUOUS_NOMAD_FULLNODE_ADDR: &str = "https://fullnode.nomad.ci.sui.io:443";
const TESTNET_FULLNODE_ADDR: &str = "https://fullnode.testnet.sui.io:443";
pub struct ClusterFactory;
impl ClusterFactory {
pub async fn start(
options: &ClusterTestOpt,
) -> Result<Box<dyn Cluster + Sync + Send>, anyhow::Error> {
Ok(match &options.env {
Env::NewLocal => Box::new(LocalNewCluster::start(options).await?),
_ => Box::new(RemoteRunningCluster::start(options).await?),
})
}
}
#[async_trait]
pub trait Cluster {
async fn start(options: &ClusterTestOpt) -> Result<Self, anyhow::Error>
where
Self: Sized;
fn fullnode_url(&self) -> &str;
fn user_key(&self) -> AccountKeyPair;
fn indexer_url(&self) -> &Option<String>;
fn remote_faucet_url(&self) -> Option<&str>;
fn local_faucet_key(&self) -> Option<&AccountKeyPair>;
fn config_directory(&self) -> &Path;
}
pub struct RemoteRunningCluster {
fullnode_url: String,
faucet_url: String,
config_directory: tempfile::TempDir,
}
#[async_trait]
impl Cluster for RemoteRunningCluster {
async fn start(options: &ClusterTestOpt) -> Result<Self, anyhow::Error> {
let (fullnode_url, faucet_url) = match options.env {
Env::Devnet => (
String::from(DEVNET_FULLNODE_ADDR),
String::from(DEVNET_FAUCET_ADDR),
),
Env::Staging => (
String::from(STAGING_FULLNODE_ADDR),
String::from(STAGING_FAUCET_ADDR),
),
Env::Ci => (
String::from(CONTINUOUS_FULLNODE_ADDR),
String::from(CONTINUOUS_FAUCET_ADDR),
),
Env::CiNomad => (
String::from(CONTINUOUS_NOMAD_FULLNODE_ADDR),
String::from(CONTINUOUS_NOMAD_FAUCET_ADDR),
),
Env::Testnet => (
String::from(TESTNET_FULLNODE_ADDR),
String::from(TESTNET_FAUCET_ADDR),
),
Env::CustomRemote => (
options
.fullnode_address
.clone()
.expect("Expect 'fullnode_address' for Env::Custom"),
options
.faucet_address
.clone()
.expect("Expect 'faucet_address' for Env::Custom"),
),
Env::NewLocal => unreachable!("NewLocal shouldn't use RemoteRunningCluster"),
};
Ok(Self {
fullnode_url,
faucet_url,
config_directory: tempfile::tempdir()?,
})
}
fn fullnode_url(&self) -> &str {
&self.fullnode_url
}
fn indexer_url(&self) -> &Option<String> {
&None
}
fn user_key(&self) -> AccountKeyPair {
get_key_pair().1
}
fn remote_faucet_url(&self) -> Option<&str> {
Some(&self.faucet_url)
}
fn local_faucet_key(&self) -> Option<&AccountKeyPair> {
None
}
fn config_directory(&self) -> &Path {
self.config_directory.path()
}
}
pub struct LocalNewCluster {
test_cluster: TestCluster,
fullnode_url: String,
indexer_url: Option<String>,
faucet_key: AccountKeyPair,
config_directory: tempfile::TempDir,
#[allow(unused)]
data_ingestion_path: tempfile::TempDir,
#[allow(unused)]
cancellation_tokens: Vec<tokio_util::sync::DropGuard>,
#[allow(unused)]
database: Option<TempDb>,
graphql_url: Option<String>,
}
impl LocalNewCluster {
#[allow(unused)]
pub fn swarm(&self) -> &Swarm {
&self.test_cluster.swarm
}
pub fn graphql_url(&self) -> &Option<String> {
&self.graphql_url
}
}
#[async_trait]
impl Cluster for LocalNewCluster {
async fn start(options: &ClusterTestOpt) -> Result<Self, anyhow::Error> {
let data_ingestion_path = tempdir()?;
let mut cluster_builder = TestClusterBuilder::new()
.enable_fullnode_events()
.with_data_ingestion_dir(data_ingestion_path.path().to_path_buf());
if let Some(config_dir) = options.config_dir.clone() {
assert!(options.epoch_duration_ms.is_none());
let network_config_path = config_dir.join(SUI_NETWORK_CONFIG);
let network_config: NetworkConfig = PersistedConfig::read(&network_config_path)
.map_err(|err| {
err.context(format!(
"Cannot open Sui network config file at {:?}",
network_config_path
))
})?;
cluster_builder = cluster_builder.set_network_config(network_config);
cluster_builder = cluster_builder.with_config_dir(config_dir);
} else {
let genesis_config = GenesisConfig::custom_genesis(1, 100);
cluster_builder = cluster_builder.set_genesis_config(genesis_config);
if let Some(epoch_duration_ms) = options.epoch_duration_ms {
cluster_builder = cluster_builder.with_epoch_duration_ms(epoch_duration_ms);
}
}
let mut test_cluster = cluster_builder.build().await;
let faucet_key = test_cluster.swarm.config_mut().account_keys.swap_remove(0);
let faucet_address = SuiAddress::from(faucet_key.public());
info!(?faucet_address, "faucet_address");
let fullnode_url = test_cluster.fullnode_handle.rpc_url.clone();
let mut cancellation_tokens = vec![];
let (database, indexer_url, graphql_url) = if options.with_indexer_and_graphql {
let database = TempDb::new()?;
let pg_address = database.database().url().as_str().to_owned();
let indexer_jsonrpc_address = format!("127.0.0.1:{}", get_available_port("127.0.0.1"));
let graphql_address = format!("127.0.0.1:{}", get_available_port("127.0.0.1"));
let graphql_url = format!("http://{graphql_address}");
let (_, _, writer_token) = start_indexer_writer_for_testing(
pg_address.clone(),
None,
None,
Some(data_ingestion_path.path().to_path_buf()),
None, None, None, )
.await;
cancellation_tokens.push(writer_token.drop_guard());
let (_, reader_token) = start_indexer_jsonrpc_for_testing(
pg_address.clone(),
fullnode_url.clone(),
indexer_jsonrpc_address.clone(),
None, )
.await;
cancellation_tokens.push(reader_token.drop_guard());
let graphql_address = graphql_address.parse::<SocketAddr>()?;
let graphql_connection_config = ConnectionConfig {
port: graphql_address.port(),
host: graphql_address.ip().to_string(),
db_url: pg_address,
..Default::default()
};
start_graphql_server_with_fn_rpc(
graphql_connection_config.clone(),
Some(fullnode_url.clone()),
None,
ServiceConfig::test_defaults(),
)
.await;
(
Some(database),
Some(indexer_jsonrpc_address),
Some(graphql_url),
)
} else {
(None, None, None)
};
tokio::time::sleep(tokio::time::Duration::from_secs(2)).await;
Ok(Self {
test_cluster,
fullnode_url,
faucet_key,
config_directory: tempfile::tempdir()?,
data_ingestion_path,
indexer_url,
cancellation_tokens,
database,
graphql_url,
})
}
fn fullnode_url(&self) -> &str {
&self.fullnode_url
}
fn indexer_url(&self) -> &Option<String> {
&self.indexer_url
}
fn user_key(&self) -> AccountKeyPair {
get_key_pair().1
}
fn remote_faucet_url(&self) -> Option<&str> {
None
}
fn local_faucet_key(&self) -> Option<&AccountKeyPair> {
Some(&self.faucet_key)
}
fn config_directory(&self) -> &Path {
self.config_directory.path()
}
}
#[async_trait]
impl Cluster for Box<dyn Cluster + Send + Sync> {
async fn start(_options: &ClusterTestOpt) -> Result<Self, anyhow::Error> {
unreachable!(
"If we already have a boxed Cluster trait object we wouldn't have to call this function"
);
}
fn fullnode_url(&self) -> &str {
(**self).fullnode_url()
}
fn indexer_url(&self) -> &Option<String> {
(**self).indexer_url()
}
fn user_key(&self) -> AccountKeyPair {
(**self).user_key()
}
fn remote_faucet_url(&self) -> Option<&str> {
(**self).remote_faucet_url()
}
fn local_faucet_key(&self) -> Option<&AccountKeyPair> {
(**self).local_faucet_key()
}
fn config_directory(&self) -> &Path {
(**self).config_directory()
}
}
pub fn new_wallet_context_from_cluster(
cluster: &(dyn Cluster + Sync + Send),
key_pair: AccountKeyPair,
) -> WalletContext {
let config_dir = cluster.config_directory();
let wallet_config_path = config_dir.join("client.yaml");
let fullnode_url = cluster.fullnode_url();
info!("Use RPC: {}", &fullnode_url);
let keystore_path = config_dir.join(SUI_KEYSTORE_FILENAME);
let mut keystore = Keystore::from(FileBasedKeystore::new(&keystore_path).unwrap());
let address: SuiAddress = key_pair.public().into();
keystore
.add_key(None, SuiKeyPair::Ed25519(key_pair))
.unwrap();
SuiClientConfig {
keystore,
envs: vec![SuiEnv {
alias: "localnet".to_string(),
rpc: fullnode_url.into(),
ws: None,
basic_auth: None,
}],
active_address: Some(address),
active_env: Some("localnet".to_string()),
}
.persisted(&wallet_config_path)
.save()
.unwrap();
info!(
"Initialize wallet from config path: {:?}",
wallet_config_path
);
WalletContext::new(&wallet_config_path, None, None).unwrap_or_else(|e| {
panic!(
"Failed to init wallet context from path {:?}, error: {e}",
wallet_config_path
)
})
}