diff --git a/.github/workflows/e2e.yml b/.github/workflows/e2e.yml index 738ae32f12..9204a1c175 100644 --- a/.github/workflows/e2e.yml +++ b/.github/workflows/e2e.yml @@ -53,9 +53,9 @@ jobs: matrix: # We build a dynamic-linked linux binary because otherwise HSM support fails with: # Error: IO: Dynamic loading not supported - os: [macos-12, ubuntu-20.04, ubuntu-22.04, windows-2022] + os: [macos-13-large, ubuntu-20.04, ubuntu-22.04, windows-2022] include: - - os: macos-12 + - os: macos-13-large target: x86_64-apple-darwin binary_path: target/x86_64-apple-darwin/release/dfx - os: ubuntu-20.04 @@ -109,7 +109,7 @@ jobs: strategy: fail-fast: false matrix: - os: [macos-12, ubuntu-20.04, ubuntu-22.04] + os: [macos-13-large, ubuntu-20.04, ubuntu-22.04] steps: - uses: actions/checkout@v4 - name: Download dfx binary @@ -182,7 +182,7 @@ jobs: strategy: fail-fast: false matrix: - os: [macos-12, ubuntu-20.04, ubuntu-22.04] + os: [macos-13-large, ubuntu-20.04, ubuntu-22.04] steps: - name: Checking out repo uses: actions/checkout@v4 diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index 06c4dbba6a..25a2c23b2b 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -41,7 +41,7 @@ jobs: strategy: fail-fast: false matrix: - os: [ ubuntu-latest, macos-12, windows-latest ] + os: [ ubuntu-latest, macos-13-large, windows-latest ] steps: - uses: actions/checkout@v4 diff --git a/.github/workflows/prepare-dfx-assets.yml b/.github/workflows/prepare-dfx-assets.yml index 73a7aa4171..52a768be7a 100644 --- a/.github/workflows/prepare-dfx-assets.yml +++ b/.github/workflows/prepare-dfx-assets.yml @@ -24,7 +24,7 @@ jobs: strategy: fail-fast: false matrix: - os: [ ubuntu-latest, macos-12 ] + os: [ ubuntu-latest, macos-13-large ] steps: - uses: actions/checkout@v4 - uses: actions/cache@v4 diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml index 3b235fc519..e95094effb 100644 --- a/.github/workflows/publish.yml +++ b/.github/workflows/publish.yml @@ -31,7 +31,7 @@ jobs: # Error: IO: Dynamic loading not supported target: [ x86_64-apple-darwin, x86_64-unknown-linux-gnu ] include: - - os: macos-12 + - os: macos-13-large target: x86_64-apple-darwin binary_path: target/x86_64-apple-darwin/release name: x86_64-darwin diff --git a/.github/workflows/unit.yml b/.github/workflows/unit.yml index 22ee0a7f90..18b6987b41 100644 --- a/.github/workflows/unit.yml +++ b/.github/workflows/unit.yml @@ -43,7 +43,7 @@ jobs: strategy: fail-fast: false matrix: - os: [ ubuntu-latest, macos-12 ] + os: [ ubuntu-latest, macos-13-large ] steps: - uses: actions/checkout@v4 - uses: actions/cache@v4 diff --git a/CHANGELOG.md b/CHANGELOG.md index 7e949acd6c..2b4112edb2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,10 @@ # UNRELEASED +### chore: update agent version in frontend templates, and include `resolve.dedupe` in Vite config + +### chore: improve error message when trying to use the local replica when it is not running + ### Frontend canister Allow setting permissions lists in init arguments just like in upgrade arguments. diff --git a/Cargo.lock b/Cargo.lock index 36a23f6c0b..72347e06e8 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1608,7 +1608,7 @@ dependencies = [ [[package]] name = "dfx-core" -version = "0.1.0" +version = "0.1.1" dependencies = [ "aes-gcm", "argon2", diff --git a/e2e/tests-dfx/error_context.bash b/e2e/tests-dfx/error_context.bash index 622116ae47..a09aaf365d 100644 --- a/e2e/tests-dfx/error_context.bash +++ b/e2e/tests-dfx/error_context.bash @@ -211,3 +211,13 @@ teardown() { assert_contains "it did not contain a function that dfx was looking for" assert_contains "dfx identity set-wallet --identity " } + +@test "Local replica not running has nice error messages" { + dfx_new + assert_command_fail dfx ping local + assert_contains "You are trying to connect to the local replica but dfx cannot connect to it." + assert_command_fail dfx deploy + assert_contains "You are trying to connect to the local replica but dfx cannot connect to it." + assert_command_fail dfx canister call um5iw-rqaaa-aaaaq-qaaba-cai some_method + assert_contains "You are trying to connect to the local replica but dfx cannot connect to it." +} diff --git a/scripts/workflows/e2e-matrix.py b/scripts/workflows/e2e-matrix.py index f65d6a832d..c48f831211 100755 --- a/scripts/workflows/e2e-matrix.py +++ b/scripts/workflows/e2e-matrix.py @@ -16,7 +16,7 @@ def test_scripts(prefix): matrix = { "test": test, "backend": ["pocketic", "replica"], - "os": ["macos-12", "ubuntu-20.04"], + "os": ["macos-13-large", "ubuntu-20.04"], "exclude": [ { "backend": "pocketic", diff --git a/scripts/workflows/provision-darwin.sh b/scripts/workflows/provision-darwin.sh index c85ea49273..279c4cc1b1 100755 --- a/scripts/workflows/provision-darwin.sh +++ b/scripts/workflows/provision-darwin.sh @@ -20,7 +20,7 @@ if [ "$E2E_TEST" = "tests-dfx/bitcoin.bash" ]; then brew fetch --retry bitcoin brew install bitcoin fi -if [ "$E2E_TEST" = "tests-dfx/build_rust.bash" ]; then +if [ "$E2E_TEST" = "tests-dfx/build_rust.bash" ] && command -v cargo-audit &>/dev/null; then cargo uninstall cargo-audit fi if [ "$E2E_TEST" = "tests-dfx/certificate.bash" ]; then diff --git a/src/dfx-core/CHANGELOG.md b/src/dfx-core/CHANGELOG.md new file mode 100644 index 0000000000..8caf055ffc --- /dev/null +++ b/src/dfx-core/CHANGELOG.md @@ -0,0 +1,19 @@ +# Changelog + +All notable changes to this project will be documented in this file. + +The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/), +and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). + +## [Unreleased] + +## [0.1.1] - 2024-11-08 + +### Added + + - get_version_from_cache_path function to get the version from the cache path + - DfxInterfaceBuilder: with_extension_manager and with_extension_manager_from_cache_path methods + +## [0.1.0] - 2024-09-01 + +Initial Version diff --git a/src/dfx-core/Cargo.toml b/src/dfx-core/Cargo.toml index 5bbfec0a45..577b717fbf 100644 --- a/src/dfx-core/Cargo.toml +++ b/src/dfx-core/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "dfx-core" -version = "0.1.0" +version = "0.1.1" authors.workspace = true edition.workspace = true repository.workspace = true diff --git a/src/dfx-core/src/config/cache.rs b/src/dfx-core/src/config/cache.rs index 11c813f810..13e39a2b61 100644 --- a/src/dfx-core/src/config/cache.rs +++ b/src/dfx-core/src/config/cache.rs @@ -2,13 +2,13 @@ use crate::config::directories::project_dirs; use crate::error::cache::{ DeleteCacheError, EnsureCacheVersionsDirError, GetBinaryCommandPathError, GetCacheRootError, - IsCacheInstalledError, ListCacheVersionsError, + GetVersionFromCachePathError, IsCacheInstalledError, ListCacheVersionsError, }; #[cfg(not(windows))] use crate::foundation::get_user_home; use crate::fs::composite::ensure_dir_exists; use semver::Version; -use std::path::PathBuf; +use std::path::{Path, PathBuf}; pub trait Cache { fn version_str(&self) -> String; @@ -50,6 +50,21 @@ pub fn get_cache_path_for_version(v: &str) -> Result Ok(p) } +pub fn get_version_from_cache_path( + cache_path: &Path, +) -> Result { + let version = cache_path + .file_name() + .ok_or(GetVersionFromCachePathError::NoCachePathFilename( + cache_path.to_path_buf(), + ))? + .to_str() + .ok_or(GetVersionFromCachePathError::CachePathFilenameNotUtf8( + cache_path.to_path_buf(), + ))?; + Ok(Version::parse(version)?) +} + /// Return the binary cache root. It constructs it if not present /// already. pub fn ensure_cache_versions_dir() -> Result { diff --git a/src/dfx-core/src/error/cache.rs b/src/dfx-core/src/error/cache.rs index f74e684eb0..b431c4eecc 100644 --- a/src/dfx-core/src/error/cache.rs +++ b/src/dfx-core/src/error/cache.rs @@ -5,6 +5,7 @@ use crate::error::fs::{ }; use crate::error::get_current_exe::GetCurrentExeError; use crate::error::get_user_home::GetUserHomeError; +use std::path::PathBuf; use thiserror::Error; #[derive(Error, Debug)] @@ -34,6 +35,18 @@ pub enum EnsureCacheVersionsDirError { GetCacheRoot(#[from] GetCacheRootError), } +#[derive(Error, Debug)] +pub enum GetVersionFromCachePathError { + #[error("no filename in cache path '{0}'")] + NoCachePathFilename(PathBuf), + + #[error("filename in cache path '{0}' is not valid UTF-8")] + CachePathFilenameNotUtf8(PathBuf), + + #[error("cannot parse version from cache path filename")] + ParseVersion(#[from] semver::Error), +} + #[derive(Error, Debug)] pub enum GetCacheRootError { #[error(transparent)] diff --git a/src/dfx-core/src/error/interface.rs b/src/dfx-core/src/error/interface.rs new file mode 100644 index 0000000000..79a7929949 --- /dev/null +++ b/src/dfx-core/src/error/interface.rs @@ -0,0 +1,12 @@ +use crate::error::cache::GetVersionFromCachePathError; +use crate::error::extension::NewExtensionManagerError; +use thiserror::Error; + +#[derive(Error, Debug)] +pub enum NewExtensionManagerFromCachePathError { + #[error(transparent)] + GetVersionFromCachePath(#[from] GetVersionFromCachePathError), + + #[error(transparent)] + NewExtensionManager(#[from] NewExtensionManagerError), +} diff --git a/src/dfx-core/src/error/mod.rs b/src/dfx-core/src/error/mod.rs index 5d720b50ce..ff13e4e49e 100644 --- a/src/dfx-core/src/error/mod.rs +++ b/src/dfx-core/src/error/mod.rs @@ -12,6 +12,7 @@ pub mod fs; pub mod get_current_exe; pub mod get_user_home; pub mod identity; +pub mod interface; pub mod keyring; pub mod load_dfx_config; pub mod load_networks_config; diff --git a/src/dfx-core/src/interface/builder.rs b/src/dfx-core/src/interface/builder.rs index 20776e6552..e7390e2c93 100644 --- a/src/dfx-core/src/interface/builder.rs +++ b/src/dfx-core/src/interface/builder.rs @@ -1,12 +1,16 @@ use crate::{ + config::cache::get_version_from_cache_path, config::model::{ dfinity::{Config, NetworksConfig}, network_descriptor::NetworkDescriptor, }, error::{ builder::{BuildAgentError, BuildDfxInterfaceError, BuildIdentityError}, + extension::NewExtensionManagerError, + interface::NewExtensionManagerFromCachePathError, network_config::NetworkConfigError, }, + extension::manager::ExtensionManager, identity::{identity_manager::InitializeIdentity, IdentityManager}, network::{ provider::{create_network_descriptor, LocalBindDetermination}, @@ -16,6 +20,8 @@ use crate::{ }; use ic_agent::{agent::route_provider::RoundRobinRouteProvider, Agent, Identity}; use reqwest::Client; +use semver::Version; +use std::path::Path; use std::sync::Arc; #[derive(PartialEq)] @@ -42,14 +48,17 @@ pub struct DfxInterfaceBuilder { /// There is no need to set this for the local network, where the root key is fetched by default. /// This would typically be set for a testnet, or an alias for the local network. force_fetch_root_key_insecure_non_mainnet_only: bool, + + extension_manager: Option, } impl DfxInterfaceBuilder { - pub(crate) fn new() -> Self { + pub fn new() -> Self { Self { identity: IdentityPicker::Selected, network: NetworkPicker::Local, force_fetch_root_key_insecure_non_mainnet_only: false, + extension_manager: None, } } @@ -77,6 +86,26 @@ impl DfxInterfaceBuilder { self.with_network(NetworkPicker::Named(name.to_string())) } + pub fn with_extension_manager( + self, + version: Version, + ) -> Result { + let extension_manager = Some(ExtensionManager::new(&version)?); + Ok(Self { + extension_manager, + ..self + }) + } + + pub fn with_extension_manager_from_cache_path( + self, + cache_path: &Path, + ) -> Result { + let version = get_version_from_cache_path(cache_path)?; + + Ok(self.with_extension_manager(version)?) + } + pub fn with_force_fetch_root_key_insecure_non_mainnet_only(self) -> Self { Self { force_fetch_root_key_insecure_non_mainnet_only: true, @@ -88,7 +117,7 @@ impl DfxInterfaceBuilder { let fetch_root_key = self.network == NetworkPicker::Local || self.force_fetch_root_key_insecure_non_mainnet_only; let networks_config = NetworksConfig::new()?; - let config = Config::from_current_dir(None)?.map(Arc::new); + let config = Config::from_current_dir(self.extension_manager.as_ref())?.map(Arc::new); let network_descriptor = self.build_network_descriptor(config.clone(), &networks_config)?; let identity = self.build_identity()?; let agent = self.build_agent(identity.clone(), &network_descriptor)?; diff --git a/src/dfx/assets/project_templates/react/src/__frontend_name__/package.json b/src/dfx/assets/project_templates/react/src/__frontend_name__/package.json index 04d9cd02c5..6c1c237ecd 100644 --- a/src/dfx/assets/project_templates/react/src/__frontend_name__/package.json +++ b/src/dfx/assets/project_templates/react/src/__frontend_name__/package.json @@ -13,9 +13,9 @@ "dependencies": { "react": "^18.2.0", "react-dom": "^18.2.0", - "@dfinity/agent": "^1.4.0", - "@dfinity/candid": "^1.4.0", - "@dfinity/principal": "^1.4.0" + "@dfinity/agent": "^2.1.3", + "@dfinity/candid": "^2.1.3", + "@dfinity/principal": "^2.1.3" }, "devDependencies": { "@types/react": "^18.2.14", diff --git a/src/dfx/assets/project_templates/react/src/__frontend_name__/vite.config.js b/src/dfx/assets/project_templates/react/src/__frontend_name__/vite.config.js index 670b0347b6..9755aa9919 100644 --- a/src/dfx/assets/project_templates/react/src/__frontend_name__/vite.config.js +++ b/src/dfx/assets/project_templates/react/src/__frontend_name__/vite.config.js @@ -39,5 +39,6 @@ export default defineConfig({ ), }, ], + dedupe: ['@dfinity/agent'], }, }); diff --git a/src/dfx/assets/project_templates/svelte/src/__frontend_name__/package.json b/src/dfx/assets/project_templates/svelte/src/__frontend_name__/package.json index 69b8edd5af..85cf87751c 100644 --- a/src/dfx/assets/project_templates/svelte/src/__frontend_name__/package.json +++ b/src/dfx/assets/project_templates/svelte/src/__frontend_name__/package.json @@ -11,9 +11,9 @@ "format": "prettier --write \"src/**/*.{json,js,jsx,ts,tsx,css,scss}\"" }, "dependencies": { - "@dfinity/agent": "^1.4.0", - "@dfinity/candid": "^1.4.0", - "@dfinity/principal": "^1.4.0" + "@dfinity/agent": "^2.1.3", + "@dfinity/candid": "^2.1.3", + "@dfinity/principal": "^2.1.3" }, "devDependencies": { "@sveltejs/adapter-static": "^3.0.4", diff --git a/src/dfx/assets/project_templates/svelte/src/__frontend_name__/vite.config.js b/src/dfx/assets/project_templates/svelte/src/__frontend_name__/vite.config.js index 51010df640..70d491d6cd 100644 --- a/src/dfx/assets/project_templates/svelte/src/__frontend_name__/vite.config.js +++ b/src/dfx/assets/project_templates/svelte/src/__frontend_name__/vite.config.js @@ -39,5 +39,6 @@ export default defineConfig({ ), }, ], + dedupe: ['@dfinity/agent'], }, }); diff --git a/src/dfx/assets/project_templates/vanilla_js/src/__frontend_name__/package.json b/src/dfx/assets/project_templates/vanilla_js/src/__frontend_name__/package.json index e96aeb2059..36d9b2fef0 100644 --- a/src/dfx/assets/project_templates/vanilla_js/src/__frontend_name__/package.json +++ b/src/dfx/assets/project_templates/vanilla_js/src/__frontend_name__/package.json @@ -21,9 +21,9 @@ "vitest": "^2.0.5" }, "dependencies": { - "@dfinity/agent": "^1.4.0", - "@dfinity/candid": "^1.4.0", - "@dfinity/principal": "^1.4.0", + "@dfinity/agent": "^2.1.3", + "@dfinity/candid": "^2.1.3", + "@dfinity/principal": "^2.1.3", "lit-html": "^2.8.0" } } diff --git a/src/dfx/assets/project_templates/vanilla_js/src/__frontend_name__/vite.config.js b/src/dfx/assets/project_templates/vanilla_js/src/__frontend_name__/vite.config.js index 18ceb500c7..9d6f44396e 100644 --- a/src/dfx/assets/project_templates/vanilla_js/src/__frontend_name__/vite.config.js +++ b/src/dfx/assets/project_templates/vanilla_js/src/__frontend_name__/vite.config.js @@ -38,5 +38,6 @@ export default defineConfig({ ), }, ], + dedupe: ['@dfinity/agent'], }, }); diff --git a/src/dfx/assets/project_templates/vue/src/__frontend_name__/package.json b/src/dfx/assets/project_templates/vue/src/__frontend_name__/package.json index daa41481a1..8367abd0d5 100644 --- a/src/dfx/assets/project_templates/vue/src/__frontend_name__/package.json +++ b/src/dfx/assets/project_templates/vue/src/__frontend_name__/package.json @@ -23,8 +23,8 @@ "dependencies": { "pinia": "^2.1.6", "vue": "^3.3.4", - "@dfinity/agent": "^1.4.0", - "@dfinity/candid": "^1.4.0", - "@dfinity/principal": "^1.4.0" + "@dfinity/agent": "^2.1.3", + "@dfinity/candid": "^2.1.3", + "@dfinity/principal": "^2.1.3" } } \ No newline at end of file diff --git a/src/dfx/assets/project_templates/vue/src/__frontend_name__/vite.config.js b/src/dfx/assets/project_templates/vue/src/__frontend_name__/vite.config.js index 014c37999a..92afec3511 100644 --- a/src/dfx/assets/project_templates/vue/src/__frontend_name__/vite.config.js +++ b/src/dfx/assets/project_templates/vue/src/__frontend_name__/vite.config.js @@ -34,6 +34,7 @@ export default defineConfig({ alias: [ { find: 'declarations', replacement: fileURLToPath(new URL('../declarations', import.meta.url)) }, { find: '@', replacement: fileURLToPath(new URL('./src', import.meta.url)) }, - ] + ], + dedupe: ['@dfinity/agent'], } }); diff --git a/src/dfx/src/lib/diagnosis.rs b/src/dfx/src/lib/diagnosis.rs index 5e76f5f97d..e59a810da7 100644 --- a/src/dfx/src/lib/diagnosis.rs +++ b/src/dfx/src/lib/diagnosis.rs @@ -1,5 +1,6 @@ use crate::lib::error_code; use anyhow::Error as AnyhowError; +use dfx_core::error::root_key::FetchRootKeyError; use ic_agent::agent::{RejectCode, RejectResponse}; use ic_agent::AgentError; use ic_asset::error::{GatherAssetDescriptorsError, SyncError, UploadContentError}; @@ -58,6 +59,10 @@ pub fn diagnose(err: &AnyhowError) -> Diagnosis { } } + if local_replica_not_running(err) { + return diagnose_local_replica_not_running(); + } + if let Some(sync_error) = err.downcast_ref::() { if duplicate_asset_key_dist_and_src(sync_error) { return diagnose_duplicate_asset_key_dist_and_src(); @@ -67,6 +72,32 @@ pub fn diagnose(err: &AnyhowError) -> Diagnosis { NULL_DIAGNOSIS } +fn local_replica_not_running(err: &AnyhowError) -> bool { + let maybe_agent_error = { + if let Some(FetchRootKeyError::AgentError(agent_error)) = + err.downcast_ref::() + { + Some(agent_error) + } else { + err.downcast_ref::() + } + }; + if let Some(AgentError::TransportError(transport_error)) = maybe_agent_error { + transport_error.is_connect() + && transport_error + .url() + .and_then(|url| url.host()) + .map(|host| match host { + url::Host::Domain(domain) => domain == "localhost", + url::Host::Ipv4(ipv4_addr) => ipv4_addr.is_loopback(), + url::Host::Ipv6(ipv6_addr) => ipv6_addr.is_loopback(), + }) + .unwrap_or(false) + } else { + false + } +} + fn not_a_controller(err: &AgentError) -> bool { // Newer replicas include the error code in the reject response. if matches!( @@ -122,6 +153,17 @@ The most common way this error is solved is by running 'dfx canister update-sett ) } +fn diagnose_local_replica_not_running() -> Diagnosis { + let error_explanation = + "You are trying to connect to the local replica but dfx cannot connect to it."; + let action_suggestion = + "Target a different network or run 'dfx start' to start the local replica."; + ( + Some(error_explanation.to_string()), + Some(action_suggestion.to_string()), + ) +} + fn subnet_not_authorized() -> Diagnosis { let action_suggestion = "If you are connecting to a node directly instead of a boundary node, try using --provisional-create-canister-effective-canister-id with a canister id in the subnet's canister range. First non-root subnet: 5v3p4-iyaaa-aaaaa-qaaaa-cai, second non-root subnet: jrlun-jiaaa-aaaab-aaaaa-cai"; (None, Some(action_suggestion.to_string()))