From d1cf48d83d8d6560e5da557758038b8f4f1185bf Mon Sep 17 00:00:00 2001 From: Alex Bean Date: Tue, 3 Dec 2024 12:35:49 +0100 Subject: [PATCH] fix: support new substrate-contracts-node structure and stabilize integration tests (#360) * fix: parse new structure substrate-contracts-node * fix: paseo+coretime integration test * fix: sourcing latest version substrate-contracts-node * refactor: set_executable_permission function * fix: clippy * chore: CI configuration * test: specify port in run_contracts_node * fix: use random ports instead of hardcoded ones --- .github/workflows/ci.yml | 2 +- crates/pop-cli/src/commands/up/contract.rs | 4 ++- crates/pop-cli/src/common/contracts.rs | 5 ++-- crates/pop-cli/tests/contract.rs | 6 ++-- crates/pop-common/src/lib.rs | 1 + crates/pop-common/src/sourcing/mod.rs | 13 +++++++-- crates/pop-contracts/src/call.rs | 22 +++++++++------ crates/pop-contracts/src/lib.rs | 2 +- crates/pop-contracts/src/node/mod.rs | 33 +++++++++++++++++----- crates/pop-contracts/src/testing.rs | 10 +++++++ crates/pop-contracts/src/up.rs | 19 ++++++++----- crates/pop-parachains/src/build.rs | 15 ++-------- crates/pop-parachains/tests/parachain.rs | 2 +- 13 files changed, 90 insertions(+), 44 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 0e68a3ac..bda33794 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -4,7 +4,7 @@ on: push: branches: [ "main" ] pull_request: - branches: [ "main" ] + types: [ opened, synchronize, reopened, ready_for_review ] env: CARGO_TERM_COLOR: always diff --git a/crates/pop-cli/src/commands/up/contract.rs b/crates/pop-cli/src/commands/up/contract.rs index 3b7353b3..96423764 100644 --- a/crates/pop-cli/src/commands/up/contract.rs +++ b/crates/pop-cli/src/commands/up/contract.rs @@ -24,6 +24,7 @@ use url::Url; const COMPLETE: &str = "🚀 Deployment complete"; const DEFAULT_URL: &str = "ws://localhost:9944/"; +const DEFAULT_PORT: u16 = 9944; const FAILED: &str = "🚫 Deployment failed."; #[derive(Args, Clone)] @@ -140,7 +141,8 @@ impl UpContractCommand { let spinner = spinner(); spinner.start("Starting local node..."); - let process = run_contracts_node(binary_path, Some(log.as_file())).await?; + let process = + run_contracts_node(binary_path, Some(log.as_file()), DEFAULT_PORT).await?; let bar = Style::new().magenta().dim().apply_to(Emoji("│", "|")); spinner.stop(format!( "Local node started successfully:{}", diff --git a/crates/pop-cli/src/common/contracts.rs b/crates/pop-cli/src/common/contracts.rs index 338ca84b..3d0be11f 100644 --- a/crates/pop-cli/src/common/contracts.rs +++ b/crates/pop-cli/src/common/contracts.rs @@ -1,7 +1,7 @@ // SPDX-License-Identifier: GPL-3.0 use cliclack::{confirm, log::warning, spinner}; -use pop_common::manifest::from_path; +use pop_common::{manifest::from_path, sourcing::set_executable_permission}; use pop_contracts::contracts_node_generator; use std::path::{Path, PathBuf}; @@ -53,8 +53,9 @@ pub async fn check_contracts_node_and_prompt(skip_confirm: bool) -> anyhow::Resu let spinner = spinner(); spinner.start("📦 Sourcing substrate-contracts-node..."); - binary.use_latest(); + binary = contracts_node_generator(crate::cache()?, binary.latest()).await?; binary.source(false, &(), true).await?; + set_executable_permission(binary.path())?; spinner.stop(format!( "✅ substrate-contracts-node successfully sourced. Cached at: {}", diff --git a/crates/pop-cli/tests/contract.rs b/crates/pop-cli/tests/contract.rs index 3a51f122..77c8dba5 100644 --- a/crates/pop-cli/tests/contract.rs +++ b/crates/pop-cli/tests/contract.rs @@ -2,7 +2,7 @@ use anyhow::Result; use assert_cmd::Command; -use pop_common::templates::Template; +use pop_common::{set_executable_permission, templates::Template}; use pop_contracts::{ contracts_node_generator, dry_run_gas_estimate_instantiate, instantiate_smart_contract, run_contracts_node, set_up_deployment, Contract, UpOpts, @@ -14,6 +14,7 @@ use url::Url; /// Test the contract lifecycle: new, build, up, call #[tokio::test] async fn contract_lifecycle() -> Result<()> { + const DEFAULT_PORT: u16 = 9944; let temp = tempfile::tempdir().unwrap(); let temp_dir = temp.path(); //let temp_dir = Path::new("./"); //For testing locally @@ -44,7 +45,8 @@ async fn contract_lifecycle() -> Result<()> { let binary = contracts_node_generator(temp_dir.to_path_buf().clone(), None).await?; binary.source(false, &(), true).await?; - let process = run_contracts_node(binary.path(), None).await?; + set_executable_permission(binary.path())?; + let process = run_contracts_node(binary.path(), None, DEFAULT_PORT).await?; // Only upload the contract // pop up contract --upload-only diff --git a/crates/pop-common/src/lib.rs b/crates/pop-common/src/lib.rs index bf2d6632..b70dd121 100644 --- a/crates/pop-common/src/lib.rs +++ b/crates/pop-common/src/lib.rs @@ -12,6 +12,7 @@ pub use errors::Error; pub use git::{Git, GitHub, Release}; pub use helpers::{get_project_name_from_path, prefix_with_current_dir_if_needed, replace_in_file}; pub use manifest::{add_crate_to_workspace, find_workspace_toml}; +pub use sourcing::set_executable_permission; pub use templates::extractor::extract_template_files; static APP_USER_AGENT: &str = concat!(env!("CARGO_PKG_NAME"), "/", env!("CARGO_PKG_VERSION")); diff --git a/crates/pop-common/src/sourcing/mod.rs b/crates/pop-common/src/sourcing/mod.rs index 177337c9..b42283be 100644 --- a/crates/pop-common/src/sourcing/mod.rs +++ b/crates/pop-common/src/sourcing/mod.rs @@ -495,9 +495,18 @@ async fn download(url: &str, dest: &Path) -> Result<(), Error> { let mut file = File::create(dest)?; file.write_all(&response.bytes().await?)?; // Make executable - let mut perms = metadata(dest)?.permissions(); + set_executable_permission(dest)?; + Ok(()) +} + +/// Sets the executable permission for a given file. +/// +/// # Arguments +/// * `path` - The file path to which permissions should be granted. +pub fn set_executable_permission>(path: P) -> Result<(), Error> { + let mut perms = metadata(&path)?.permissions(); perms.set_mode(0o755); - std::fs::set_permissions(dest, perms)?; + std::fs::set_permissions(path, perms)?; Ok(()) } diff --git a/crates/pop-contracts/src/call.rs b/crates/pop-contracts/src/call.rs index a554bc2f..1886f3be 100644 --- a/crates/pop-contracts/src/call.rs +++ b/crates/pop-contracts/src/call.rs @@ -180,11 +180,13 @@ mod tests { use crate::{ contracts_node_generator, dry_run_gas_estimate_instantiate, errors::Error, instantiate_smart_contract, mock_build_process, new_environment, run_contracts_node, - set_up_deployment, UpOpts, + set_up_deployment, testing::find_free_port, UpOpts, }; use anyhow::Result; + use pop_common::set_executable_permission; use sp_core::Bytes; - use std::{env, process::Command}; + use std::{env, process::Command, time::Duration}; + use tokio::time::sleep; const CONTRACTS_NETWORK_URL: &str = "wss://rpc2.paseo.popnetwork.xyz"; @@ -334,7 +336,8 @@ mod tests { #[tokio::test] async fn call_works() -> Result<()> { - const LOCALHOST_URL: &str = "ws://127.0.0.1:9944"; + let random_port = find_free_port(); + let localhost_url = format!("ws://127.0.0.1:{}", random_port); let temp_dir = new_environment("testing")?; let current_dir = env::current_dir().expect("Failed to get current directory"); mock_build_process( @@ -347,7 +350,10 @@ mod tests { let binary = contracts_node_generator(cache.clone(), None).await?; binary.source(false, &(), true).await?; - let process = run_contracts_node(binary.path(), None).await?; + set_executable_permission(binary.path())?; + let process = run_contracts_node(binary.path(), None, random_port).await?; + // Wait 5 secs more to give time for the node to be ready + sleep(Duration::from_millis(5000)).await; // Instantiate a Smart Contract. let instantiate_exec = set_up_deployment(UpOpts { path: Some(temp_dir.path().join("testing")), @@ -357,7 +363,7 @@ mod tests { gas_limit: None, proof_size: None, salt: Some(Bytes::from(vec![0x00])), - url: Url::parse(LOCALHOST_URL)?, + url: Url::parse(&localhost_url)?, suri: "//Alice".to_string(), }) .await?; @@ -372,7 +378,7 @@ mod tests { value: "0".to_string(), gas_limit: None, proof_size: None, - url: Url::parse(LOCALHOST_URL)?, + url: Url::parse(&localhost_url)?, suri: "//Alice".to_string(), execute: false, }) @@ -388,7 +394,7 @@ mod tests { value: "0".to_string(), gas_limit: None, proof_size: None, - url: Url::parse(LOCALHOST_URL)?, + url: Url::parse(&localhost_url)?, suri: "//Alice".to_string(), execute: false, }) @@ -396,7 +402,7 @@ mod tests { let weight = dry_run_gas_estimate_call(&call_exec).await?; assert!(weight.ref_time() > 0); assert!(weight.proof_size() > 0); - call_smart_contract(call_exec, weight, &Url::parse(LOCALHOST_URL)?).await?; + call_smart_contract(call_exec, weight, &Url::parse(&localhost_url)?).await?; // Assert that the value has been flipped. query = dry_run_call(&query_exec).await?; assert_eq!(query, "Ok(true)"); diff --git a/crates/pop-contracts/src/lib.rs b/crates/pop-contracts/src/lib.rs index 14228399..005ec91a 100644 --- a/crates/pop-contracts/src/lib.rs +++ b/crates/pop-contracts/src/lib.rs @@ -20,7 +20,7 @@ pub use new::{create_smart_contract, is_valid_contract_name}; pub use node::{contracts_node_generator, is_chain_alive, run_contracts_node}; pub use templates::{Contract, ContractType}; pub use test::{test_e2e_smart_contract, test_smart_contract}; -pub use testing::{mock_build_process, new_environment}; +pub use testing::{find_free_port, mock_build_process, new_environment}; pub use up::{ dry_run_gas_estimate_instantiate, dry_run_upload, instantiate_smart_contract, set_up_deployment, set_up_upload, upload_smart_contract, UpOpts, diff --git a/crates/pop-contracts/src/node/mod.rs b/crates/pop-contracts/src/node/mod.rs index 6618381b..77ac97cd 100644 --- a/crates/pop-contracts/src/node/mod.rs +++ b/crates/pop-contracts/src/node/mod.rs @@ -64,7 +64,7 @@ impl TryInto for Chain { /// * `latest` - If applicable, some specifier used to determine the latest source. fn try_into(&self, tag: Option, latest: Option) -> Result { let archive = archive_name_by_target()?; - let archive_bin_path = release_directory_by_target()?; + let archive_bin_path = release_directory_by_target(tag.as_deref())?; Ok(match self { &Chain::ContractsNode => { // Source from GitHub release asset @@ -115,12 +115,15 @@ pub async fn contracts_node_generator( /// /// * `binary_path` - The path where the binary is stored. Can be the binary name itself if in PATH. /// * `output` - The optional log file for node output. +/// * `port` - The WebSocket port on which the node will listen for connections. pub async fn run_contracts_node( binary_path: PathBuf, output: Option<&File>, + port: u16, ) -> Result { let mut command = Command::new(binary_path); command.arg("-linfo,runtime::contracts=debug"); + command.arg(format!("--rpc-port={}", port)); if let Some(output) = output { command.stdout(Stdio::from(output.try_clone()?)); command.stderr(Stdio::from(output.try_clone()?)); @@ -141,16 +144,30 @@ fn archive_name_by_target() -> Result { } } -fn release_directory_by_target() -> Result<&'static str, Error> { +fn release_directory_by_target(tag: Option<&str>) -> Result<&'static str, Error> { + // The structure of the binary changed in v0.42.0 + let is_old_structure = matches!(tag, Some(tag) if tag < "v0.42.0"); match OS { - "macos" => Ok("artifacts/substrate-contracts-node-mac/substrate-contracts-node"), - "linux" => Ok("artifacts/substrate-contracts-node-linux/substrate-contracts-node"), + "macos" => + if is_old_structure { + Ok("artifacts/substrate-contracts-node-mac/substrate-contracts-node") + } else { + Ok("substrate-contracts-node-mac/substrate-contracts-node") + }, + "linux" => + if is_old_structure { + Ok("artifacts/substrate-contracts-node-linux/substrate-contracts-node") + } else { + Ok("substrate-contracts-node-linux/substrate-contracts-node") + }, _ => Err(Error::UnsupportedPlatform { arch: ARCH, os: OS }), } } #[cfg(test)] mod tests { + use crate::testing::find_free_port; + use super::*; use anyhow::{Error, Result}; use std::process::Command; @@ -185,7 +202,7 @@ mod tests { let version = "v0.40.0"; let binary = contracts_node_generator(cache.clone(), Some(version)).await?; let archive = archive_name_by_target()?; - let archive_bin_path = release_directory_by_target()?; + let archive_bin_path = release_directory_by_target(Some(version))?; assert!(matches!(binary, Binary::Source { name, source, cache} if name == expected.binary() && source == Source::GitHub(ReleaseArchive { @@ -206,7 +223,9 @@ mod tests { #[ignore = "Works fine locally but is causing issues when running tests in parallel in the CI environment."] #[tokio::test] async fn run_contracts_node_works() -> Result<(), Error> { - let local_url = url::Url::parse("ws://localhost:9944")?; + let random_port = find_free_port(); + let localhost_url = format!("ws://127.0.0.1:{}", random_port); + let local_url = url::Url::parse(&localhost_url)?; let temp_dir = tempfile::tempdir().expect("Could not create temp dir"); let cache = temp_dir.path().join(""); @@ -214,7 +233,7 @@ mod tests { let version = "v0.40.0"; let binary = contracts_node_generator(cache.clone(), Some(version)).await?; binary.source(false, &(), true).await?; - let process = run_contracts_node(binary.path(), None).await?; + let process = run_contracts_node(binary.path(), None, 9947).await?; // Check if the node is alive assert!(is_chain_alive(local_url).await?); diff --git a/crates/pop-contracts/src/testing.rs b/crates/pop-contracts/src/testing.rs index 6a8abfcd..c10bd4e5 100644 --- a/crates/pop-contracts/src/testing.rs +++ b/crates/pop-contracts/src/testing.rs @@ -4,6 +4,7 @@ use crate::{create_smart_contract, Contract}; use anyhow::Result; use std::{ fs::{copy, create_dir}, + net::TcpListener, path::Path, }; @@ -37,3 +38,12 @@ where copy(metadata_file, target_contract_dir.join("ink/testing.json"))?; Ok(()) } + +/// Finds an available port by binding to port 0 and retrieving the assigned port. +pub fn find_free_port() -> u16 { + TcpListener::bind("127.0.0.1:0") + .expect("Failed to bind to an available port") + .local_addr() + .expect("Failed to retrieve local address") + .port() +} diff --git a/crates/pop-contracts/src/up.rs b/crates/pop-contracts/src/up.rs index 586a962f..90e0b35a 100644 --- a/crates/pop-contracts/src/up.rs +++ b/crates/pop-contracts/src/up.rs @@ -224,10 +224,12 @@ mod tests { use super::*; use crate::{ contracts_node_generator, errors::Error, mock_build_process, new_environment, - run_contracts_node, + run_contracts_node, testing::find_free_port, }; use anyhow::Result; - use std::{env, process::Command}; + use pop_common::set_executable_permission; + use std::{env, process::Command, time::Duration}; + use tokio::time::sleep; use url::Url; const CONTRACTS_NETWORK_URL: &str = "wss://rpc2.paseo.popnetwork.xyz"; @@ -364,7 +366,8 @@ mod tests { #[tokio::test] async fn instantiate_and_upload() -> Result<()> { - const LOCALHOST_URL: &str = "ws://127.0.0.1:9944"; + let random_port = find_free_port(); + let localhost_url = format!("ws://127.0.0.1:{}", random_port); let temp_dir = new_environment("testing")?; let current_dir = env::current_dir().expect("Failed to get current directory"); mock_build_process( @@ -377,8 +380,10 @@ mod tests { let binary = contracts_node_generator(cache.clone(), None).await?; binary.source(false, &(), true).await?; - let process = run_contracts_node(binary.path(), None).await?; - + set_executable_permission(binary.path())?; + let process = run_contracts_node(binary.path(), None, random_port).await?; + // Wait 5 secs more to give time for the node to be ready + sleep(Duration::from_millis(5000)).await; let upload_exec = set_up_upload(UpOpts { path: Some(temp_dir.path().join("testing")), constructor: "new".to_string(), @@ -387,7 +392,7 @@ mod tests { gas_limit: None, proof_size: None, salt: None, - url: Url::parse(LOCALHOST_URL)?, + url: Url::parse(&localhost_url)?, suri: "//Alice".to_string(), }) .await?; @@ -411,7 +416,7 @@ mod tests { gas_limit: None, proof_size: None, salt: Some(Bytes::from(vec![0x00])), - url: Url::parse(LOCALHOST_URL)?, + url: Url::parse(&localhost_url)?, suri: "//Alice".to_string(), }) .await?; diff --git a/crates/pop-parachains/src/build.rs b/crates/pop-parachains/src/build.rs index 27a2acd9..3c25b2a0 100644 --- a/crates/pop-parachains/src/build.rs +++ b/crates/pop-parachains/src/build.rs @@ -321,14 +321,8 @@ mod tests { Zombienet, }; use anyhow::Result; - use pop_common::manifest::Dependency; - use std::{ - fs, - fs::{metadata, write}, - io::Write, - os::unix::fs::PermissionsExt, - path::Path, - }; + use pop_common::{manifest::Dependency, set_executable_permission}; + use std::{fs, fs::write, io::Write, path::Path}; use tempfile::{tempdir, Builder}; fn setup_template_and_instantiate() -> Result { @@ -405,10 +399,7 @@ default_command = "pop-node" let content = fs::read(&binary_path)?; write(temp_dir.join("target/release/parachain-template-node"), content)?; // Make executable - let mut perms = - metadata(temp_dir.join("target/release/parachain-template-node"))?.permissions(); - perms.set_mode(0o755); - fs::set_permissions(temp_dir.join("target/release/parachain-template-node"), perms)?; + set_executable_permission(temp_dir.join("target/release/parachain-template-node"))?; Ok(binary_path) } diff --git a/crates/pop-parachains/tests/parachain.rs b/crates/pop-parachains/tests/parachain.rs index 1b03c294..3581d243 100644 --- a/crates/pop-parachains/tests/parachain.rs +++ b/crates/pop-parachains/tests/parachain.rs @@ -112,7 +112,7 @@ async fn launch_paseo_and_system_parachain() -> Result<()> { Some(BINARY_VERSION), None, Some(BINARY_VERSION), - None, + Some("v1.3.3"), // 1.3.3 is where coretime-paseo-local was introduced. None, ) .await?;