From edded24d2256a074e8e390285e123e39f926551d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=C3=81lvaro=20Rodr=C3=ADguez?= Date: Mon, 14 Aug 2023 17:01:17 +0200 Subject: [PATCH 1/3] feat: optionally output a debug artifact on compile (#2260) Co-authored-by: Tom French <15848336+TomAFrench@users.noreply.github.com> --- crates/fm/src/file_map.rs | 4 +- crates/fm/src/lib.rs | 2 +- crates/nargo/src/artifacts/debug.rs | 50 ++++++++++ crates/nargo/src/artifacts/mod.rs | 1 + crates/nargo/src/ops/preprocess.rs | 23 +++-- crates/nargo_cli/src/cli/compile_cmd.rs | 97 +++++++++++++------ crates/nargo_cli/src/cli/fs/program.rs | 19 +++- crates/noirc_driver/src/contract.rs | 3 + crates/noirc_driver/src/lib.rs | 1 + crates/noirc_errors/src/debug_info.rs | 8 +- .../ssa/acir_gen/acir_ir/generated_acir.rs | 4 +- 11 files changed, 162 insertions(+), 50 deletions(-) create mode 100644 crates/nargo/src/artifacts/debug.rs diff --git a/crates/fm/src/file_map.rs b/crates/fm/src/file_map.rs index 8bbfcf99dd7..22ac6d4e179 100644 --- a/crates/fm/src/file_map.rs +++ b/crates/fm/src/file_map.rs @@ -34,7 +34,9 @@ impl From<&PathBuf> for PathString { pub struct FileMap(SimpleFiles); // XXX: Note that we derive Default here due to ModuleOrigin requiring us to set a FileId -#[derive(Default, Debug, Clone, PartialEq, Eq, Copy, Hash, Serialize, Deserialize)] +#[derive( + Default, Debug, Clone, PartialEq, Eq, Copy, Hash, Serialize, Deserialize, PartialOrd, Ord, +)] pub struct FileId(usize); impl FileId { diff --git a/crates/fm/src/lib.rs b/crates/fm/src/lib.rs index 4c2ce39dd40..ef976a80dbd 100644 --- a/crates/fm/src/lib.rs +++ b/crates/fm/src/lib.rs @@ -70,7 +70,7 @@ impl FileManager { assert!(old_value.is_none(), "ice: the same path was inserted into the file manager twice"); } - pub fn fetch_file(&mut self, file_id: FileId) -> File { + pub fn fetch_file(&self, file_id: FileId) -> File { // Unwrap as we ensure that all file_id's map to a corresponding file in the file map self.file_map.get_file(file_id).unwrap() } diff --git a/crates/nargo/src/artifacts/debug.rs b/crates/nargo/src/artifacts/debug.rs new file mode 100644 index 00000000000..31c15311f06 --- /dev/null +++ b/crates/nargo/src/artifacts/debug.rs @@ -0,0 +1,50 @@ +use serde::{Deserialize, Serialize}; +use std::{ + collections::{BTreeMap, BTreeSet}, + path::PathBuf, +}; + +use fm::FileId; +use noirc_errors::debug_info::DebugInfo; +use noirc_frontend::hir::Context; + +/// For a given file, we store the source code and the path to the file +/// so consumers of the debug artifact can reconstruct the original source code structure. +#[derive(Debug, Serialize, Deserialize)] +pub struct DebugFile { + pub source: String, + pub path: PathBuf, +} + +/// A Debug Artifact stores, for a given program, the debug info for every function +/// along with a map of file Id to the source code so locations in debug info can be mapped to source code they point to. +#[derive(Debug, Serialize, Deserialize)] +pub struct DebugArtifact { + pub debug_symbols: Vec, + pub file_map: BTreeMap, +} + +impl DebugArtifact { + pub fn new(debug_symbols: Vec, compilation_context: &Context) -> Self { + let mut file_map = BTreeMap::new(); + + let files_with_debug_symbols: BTreeSet = debug_symbols + .iter() + .flat_map(|function_symbols| function_symbols.locations.values().map(|loc| loc.file)) + .collect(); + + for file_id in files_with_debug_symbols { + let file_source = compilation_context.file_manager.fetch_file(file_id).source(); + + file_map.insert( + file_id, + DebugFile { + source: file_source.to_string(), + path: compilation_context.file_manager.path(file_id).to_path_buf(), + }, + ); + } + + Self { debug_symbols, file_map } + } +} diff --git a/crates/nargo/src/artifacts/mod.rs b/crates/nargo/src/artifacts/mod.rs index 1ba12c49af7..33311e0856e 100644 --- a/crates/nargo/src/artifacts/mod.rs +++ b/crates/nargo/src/artifacts/mod.rs @@ -8,6 +8,7 @@ use base64::Engine; use serde::{Deserializer, Serializer}; pub mod contract; +pub mod debug; pub mod program; // TODO: move these down into ACVM. diff --git a/crates/nargo/src/ops/preprocess.rs b/crates/nargo/src/ops/preprocess.rs index d07da256ede..0ee4e2590f9 100644 --- a/crates/nargo/src/ops/preprocess.rs +++ b/crates/nargo/src/ops/preprocess.rs @@ -42,7 +42,7 @@ pub fn preprocess_contract_function( include_keys: bool, common_reference_string: &[u8], func: ContractFunction, -) -> Result { +) -> Result<(PreprocessedContractFunction, DebugInfo), B::Error> { // TODO: currently `func`'s bytecode is already optimized for the backend. // In future we'll need to apply those optimizations here. let optimized_bytecode = func.bytecode; @@ -54,14 +54,17 @@ pub fn preprocess_contract_function( (None, None) }; - Ok(PreprocessedContractFunction { - name: func.name, - function_type: func.function_type, - is_internal: func.is_internal, - abi: func.abi, + Ok(( + PreprocessedContractFunction { + name: func.name, + function_type: func.function_type, + is_internal: func.is_internal, + abi: func.abi, - bytecode: optimized_bytecode, - proving_key, - verification_key, - }) + bytecode: optimized_bytecode, + proving_key, + verification_key, + }, + func.debug, + )) } diff --git a/crates/nargo_cli/src/cli/compile_cmd.rs b/crates/nargo_cli/src/cli/compile_cmd.rs index fad9783e654..2a6f802f748 100644 --- a/crates/nargo_cli/src/cli/compile_cmd.rs +++ b/crates/nargo_cli/src/cli/compile_cmd.rs @@ -2,14 +2,16 @@ use acvm::acir::circuit::OpcodeLabel; use acvm::{acir::circuit::Circuit, Backend}; use iter_extended::try_vecmap; use iter_extended::vecmap; +use nargo::artifacts::debug::DebugArtifact; use nargo::package::Package; use nargo::prepare_package; use nargo::{artifacts::contract::PreprocessedContract, NargoError}; use nargo_toml::{find_package_manifest, resolve_workspace_from_toml}; use noirc_driver::{ compile_contracts, compile_main, CompileOptions, CompiledContract, CompiledProgram, - ContractFunction, ErrorsAndWarnings, Warnings, + ErrorsAndWarnings, Warnings, }; +use noirc_errors::debug_info::DebugInfo; use noirc_frontend::graph::CrateName; use noirc_frontend::hir::Context; @@ -19,6 +21,7 @@ use nargo::ops::{preprocess_contract_function, preprocess_program}; use crate::errors::{CliError, CompileError}; +use super::fs::program::save_debug_artifact_to_file; use super::fs::{ common_reference_string::{ read_cached_common_reference_string, update_common_reference_string, @@ -38,6 +41,10 @@ pub(crate) struct CompileCommand { #[arg(long)] include_keys: bool, + /// Output debug files + #[arg(long, hide = true)] + output_debug: bool, + /// The name of the package to compile #[clap(long)] package: Option, @@ -70,49 +77,72 @@ pub(crate) fn run( // As can be seen here, It seems like a leaky abstraction where ContractFunctions (essentially CompiledPrograms) // are compiled via nargo-core and then the PreprocessedContract is constructed here. // This is due to EACH function needing it's own CRS, PKey, and VKey from the backend. - let preprocessed_contracts: Result, CliError> = - try_vecmap(optimized_contracts, |contract| { - let preprocessed_contract_functions = try_vecmap(contract.functions, |func| { - common_reference_string = update_common_reference_string( - backend, - &common_reference_string, - &func.bytecode, - ) - .map_err(CliError::CommonReferenceStringError)?; - - preprocess_contract_function( - backend, - args.include_keys, - &common_reference_string, - func, - ) - .map_err(CliError::ProofSystemCompilerError) - })?; - - Ok(PreprocessedContract { + let preprocessed_contracts: Result< + Vec<(PreprocessedContract, Vec)>, + CliError, + > = try_vecmap(optimized_contracts, |contract| { + let preprocess_result = try_vecmap(contract.functions, |func| { + common_reference_string = update_common_reference_string( + backend, + &common_reference_string, + &func.bytecode, + ) + .map_err(CliError::CommonReferenceStringError)?; + + preprocess_contract_function( + backend, + args.include_keys, + &common_reference_string, + func, + ) + .map_err(CliError::ProofSystemCompilerError) + })?; + + let (preprocessed_contract_functions, debug_infos): (Vec<_>, Vec<_>) = + preprocess_result.into_iter().unzip(); + + Ok(( + PreprocessedContract { name: contract.name, backend: String::from(BACKEND_IDENTIFIER), functions: preprocessed_contract_functions, - }) - }); - for contract in preprocessed_contracts? { + }, + debug_infos, + )) + }); + for (contract, debug_infos) in preprocessed_contracts? { save_contract_to_file( &contract, &format!("{}-{}", package.name, contract.name), &circuit_dir, ); + + if args.output_debug { + let debug_artifact = DebugArtifact::new(debug_infos, &context); + save_debug_artifact_to_file( + &debug_artifact, + &format!("{}-{}", package.name, contract.name), + &circuit_dir, + ); + } } } else { - let (_, program) = compile_package(backend, package, &args.compile_options)?; + let (context, program) = compile_package(backend, package, &args.compile_options)?; common_reference_string = update_common_reference_string(backend, &common_reference_string, &program.circuit) .map_err(CliError::CommonReferenceStringError)?; - let (preprocessed_program, _) = + let (preprocessed_program, debug_info) = preprocess_program(backend, args.include_keys, &common_reference_string, program) .map_err(CliError::ProofSystemCompilerError)?; save_program_to_file(&preprocessed_program, &package.name, &circuit_dir); + + if args.output_debug { + let debug_artifact = DebugArtifact::new(vec![debug_info], &context); + let circuit_name: String = (&package.name).into(); + save_debug_artifact_to_file(&debug_artifact, &circuit_name, &circuit_dir); + } } } @@ -167,9 +197,18 @@ pub(super) fn optimize_contract( backend: &B, contract: CompiledContract, ) -> Result> { - let functions = try_vecmap(contract.functions, |func| { - let optimized_bytecode = optimize_circuit(backend, func.bytecode)?.0; - Ok::<_, CliError>(ContractFunction { bytecode: optimized_bytecode, ..func }) + let functions = try_vecmap(contract.functions, |mut func| { + let (optimized_bytecode, opcode_labels) = optimize_circuit(backend, func.bytecode)?; + let opcode_ids = vecmap(opcode_labels, |label| match label { + OpcodeLabel::Unresolved => { + unreachable!("Compiled circuit opcodes must resolve to some index") + } + OpcodeLabel::Resolved(index) => index as usize, + }); + + func.bytecode = optimized_bytecode; + func.debug.update_acir(opcode_ids); + Ok::<_, CliError>(func) })?; Ok(CompiledContract { functions, ..contract }) diff --git a/crates/nargo_cli/src/cli/fs/program.rs b/crates/nargo_cli/src/cli/fs/program.rs index 311923a6686..3f7107de667 100644 --- a/crates/nargo_cli/src/cli/fs/program.rs +++ b/crates/nargo_cli/src/cli/fs/program.rs @@ -1,6 +1,8 @@ use std::path::{Path, PathBuf}; -use nargo::artifacts::{contract::PreprocessedContract, program::PreprocessedProgram}; +use nargo::artifacts::{ + contract::PreprocessedContract, debug::DebugArtifact, program::PreprocessedProgram, +}; use noirc_frontend::graph::CrateName; use crate::errors::FilesystemError; @@ -15,6 +17,7 @@ pub(crate) fn save_program_to_file>( let circuit_name: String = crate_name.into(); save_build_artifact_to_file(compiled_program, &circuit_name, circuit_dir) } + pub(crate) fn save_contract_to_file>( compiled_contract: &PreprocessedContract, circuit_name: &str, @@ -22,13 +25,23 @@ pub(crate) fn save_contract_to_file>( ) -> PathBuf { save_build_artifact_to_file(compiled_contract, circuit_name, circuit_dir) } + +pub(crate) fn save_debug_artifact_to_file>( + debug_artifact: &DebugArtifact, + circuit_name: &str, + circuit_dir: P, +) -> PathBuf { + let artifact_name = format!("debug_{}", circuit_name); + save_build_artifact_to_file(debug_artifact, &artifact_name, circuit_dir) +} + fn save_build_artifact_to_file, T: ?Sized + serde::Serialize>( build_artifact: &T, - circuit_name: &str, + artifact_name: &str, circuit_dir: P, ) -> PathBuf { create_named_dir(circuit_dir.as_ref(), "target"); - let circuit_path = circuit_dir.as_ref().join(circuit_name).with_extension("json"); + let circuit_path = circuit_dir.as_ref().join(artifact_name).with_extension("json"); write_to_file(&serde_json::to_vec(build_artifact).unwrap(), &circuit_path); diff --git a/crates/noirc_driver/src/contract.rs b/crates/noirc_driver/src/contract.rs index a25d258c9be..a1820ff2e47 100644 --- a/crates/noirc_driver/src/contract.rs +++ b/crates/noirc_driver/src/contract.rs @@ -1,6 +1,7 @@ use crate::program::{deserialize_circuit, serialize_circuit}; use acvm::acir::circuit::Circuit; use noirc_abi::Abi; +use noirc_errors::debug_info::DebugInfo; use serde::{Deserialize, Serialize}; /// Describes the types of smart contract functions that are allowed. @@ -47,6 +48,8 @@ pub struct ContractFunction { #[serde(serialize_with = "serialize_circuit", deserialize_with = "deserialize_circuit")] pub bytecode: Circuit, + + pub debug: DebugInfo, } impl ContractFunctionType { diff --git a/crates/noirc_driver/src/lib.rs b/crates/noirc_driver/src/lib.rs index 2919d2fd3ea..0e6a3c519b5 100644 --- a/crates/noirc_driver/src/lib.rs +++ b/crates/noirc_driver/src/lib.rs @@ -249,6 +249,7 @@ fn compile_contract( is_internal: func_meta.is_internal.unwrap_or(false), abi: function.abi, bytecode: function.circuit, + debug: function.debug, }); } diff --git a/crates/noirc_errors/src/debug_info.rs b/crates/noirc_errors/src/debug_info.rs index 3217dba7a38..0c05bee61d5 100644 --- a/crates/noirc_errors/src/debug_info.rs +++ b/crates/noirc_errors/src/debug_info.rs @@ -1,4 +1,4 @@ -use std::collections::HashMap; +use std::collections::BTreeMap; use crate::Location; use serde::{Deserialize, Serialize}; @@ -6,11 +6,11 @@ use serde::{Deserialize, Serialize}; #[derive(Default, Debug, Clone, Deserialize, Serialize)] pub struct DebugInfo { /// Map opcode index of an ACIR circuit into the source code location - pub locations: HashMap, + pub locations: BTreeMap, } impl DebugInfo { - pub fn new(locations: HashMap) -> Self { + pub fn new(locations: BTreeMap) -> Self { DebugInfo { locations } } @@ -24,7 +24,7 @@ impl DebugInfo { /// This is the case during fallback or width 'optimization' /// opcode_indices is this list of mixed indices pub fn update_acir(&mut self, opcode_indices: Vec) { - let mut new_locations = HashMap::new(); + let mut new_locations = BTreeMap::new(); for (i, idx) in opcode_indices.iter().enumerate() { if self.locations.contains_key(idx) { new_locations.insert(i, self.locations[idx]); diff --git a/crates/noirc_evaluator/src/ssa/acir_gen/acir_ir/generated_acir.rs b/crates/noirc_evaluator/src/ssa/acir_gen/acir_ir/generated_acir.rs index b425eab42d3..fd88921bb3d 100644 --- a/crates/noirc_evaluator/src/ssa/acir_gen/acir_ir/generated_acir.rs +++ b/crates/noirc_evaluator/src/ssa/acir_gen/acir_ir/generated_acir.rs @@ -1,6 +1,6 @@ //! `GeneratedAcir` is constructed as part of the `acir_gen` pass to accumulate all of the ACIR //! program as it is being converted from SSA form. -use std::collections::HashMap; +use std::collections::BTreeMap; use crate::{ brillig::brillig_gen::brillig_directive, @@ -46,7 +46,7 @@ pub(crate) struct GeneratedAcir { pub(crate) input_witnesses: Vec, /// Correspondance between an opcode index (in opcodes) and the source code location which generated it - pub(crate) locations: HashMap, + pub(crate) locations: BTreeMap, /// Source code location of the current instruction being processed /// None if we do not know the location From cc2af74e586bbbba0c45aa0b7c9f9a9e6480f851 Mon Sep 17 00:00:00 2001 From: Alex Vitkov <44268717+alexvitkov@users.noreply.github.com> Date: Mon, 14 Aug 2023 18:07:00 +0300 Subject: [PATCH 2/3] fix(nargo)!: Remove `-p` short flag from the `--program-dir` flag (#2300) --- crates/nargo_cli/src/cli/mod.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/crates/nargo_cli/src/cli/mod.rs b/crates/nargo_cli/src/cli/mod.rs index 2ab34c11e40..df5eed7b297 100644 --- a/crates/nargo_cli/src/cli/mod.rs +++ b/crates/nargo_cli/src/cli/mod.rs @@ -40,7 +40,7 @@ struct NargoCli { #[derive(Args, Clone, Debug)] pub(crate) struct NargoConfig { // REMINDER: Also change this flag in the LSP test lens if renamed - #[arg(short, long, hide=true, global=true, default_value_os_t = std::env::current_dir().unwrap())] + #[arg(long, hide=true, global=true, default_value_os_t = std::env::current_dir().unwrap())] program_dir: PathBuf, } From 87174ac4927c4e237a2d0dbd6290da309e9f70c0 Mon Sep 17 00:00:00 2001 From: Tom French <15848336+TomAFrench@users.noreply.github.com> Date: Mon, 14 Aug 2023 16:42:13 +0100 Subject: [PATCH 3/3] fix: Require package names to be non-empty (#2293) Co-authored-by: Blaine Bublitz --- crates/nargo_cli/src/cli/init_cmd.rs | 17 ++++++---- crates/nargo_cli/src/cli/new_cmd.rs | 14 ++++++--- crates/nargo_cli/src/errors.rs | 3 ++ .../invalid_dependency_name/Nargo.toml | 8 +++++ .../invalid_dependency_name/src/main.nr | 1 + .../package_name_empty/Nargo.toml | 7 +++++ .../package_name_empty/src/main.nr | 1 + .../package_name_hyphen/Nargo.toml | 7 +++++ .../package_name_hyphen/src/main.nr | 1 + .../tests/test_libraries/bad_name/Nargo.toml | 7 +++++ .../tests/test_libraries/bad_name/src/lib.nr | 0 crates/nargo_toml/src/errors.rs | 8 +++-- crates/nargo_toml/src/lib.rs | 10 ++++-- crates/noirc_frontend/src/graph/mod.rs | 31 ++++++++++++++++--- 14 files changed, 96 insertions(+), 19 deletions(-) create mode 100644 crates/nargo_cli/tests/compile_failure/invalid_dependency_name/Nargo.toml create mode 100644 crates/nargo_cli/tests/compile_failure/invalid_dependency_name/src/main.nr create mode 100644 crates/nargo_cli/tests/compile_failure/package_name_empty/Nargo.toml create mode 100644 crates/nargo_cli/tests/compile_failure/package_name_empty/src/main.nr create mode 100644 crates/nargo_cli/tests/compile_failure/package_name_hyphen/Nargo.toml create mode 100644 crates/nargo_cli/tests/compile_failure/package_name_hyphen/src/main.nr create mode 100644 crates/nargo_cli/tests/test_libraries/bad_name/Nargo.toml create mode 100644 crates/nargo_cli/tests/test_libraries/bad_name/src/lib.nr diff --git a/crates/nargo_cli/src/cli/init_cmd.rs b/crates/nargo_cli/src/cli/init_cmd.rs index d6ad172ce26..e6020e3cfd9 100644 --- a/crates/nargo_cli/src/cli/init_cmd.rs +++ b/crates/nargo_cli/src/cli/init_cmd.rs @@ -6,6 +6,7 @@ use acvm::Backend; use clap::Args; use nargo::constants::{PKG_FILE, SRC_DIR}; use nargo::package::PackageType; +use noirc_frontend::graph::CrateName; use std::path::PathBuf; /// Create a Noir project in the current directory. @@ -13,7 +14,7 @@ use std::path::PathBuf; pub(crate) struct InitCommand { /// Name of the package [default: current directory name] #[clap(long)] - name: Option, + name: Option, /// Use a library template #[arg(long, conflicts_with = "bin", conflicts_with = "contract")] @@ -67,9 +68,13 @@ pub(crate) fn run( args: InitCommand, config: NargoConfig, ) -> Result<(), CliError> { - let package_name = args - .name - .unwrap_or_else(|| config.program_dir.file_name().unwrap().to_str().unwrap().to_owned()); + let package_name = match args.name { + Some(name) => name, + None => { + let name = config.program_dir.file_name().unwrap().to_str().unwrap(); + name.parse().map_err(|_| CliError::InvalidPackageName(name.into()))? + } + }; let package_type = if args.lib { PackageType::Library @@ -78,14 +83,14 @@ pub(crate) fn run( } else { PackageType::Binary }; - initialize_project(config.program_dir, &package_name, package_type); + initialize_project(config.program_dir, package_name, package_type); Ok(()) } /// Initializes a new Noir project in `package_dir`. pub(crate) fn initialize_project( package_dir: PathBuf, - package_name: &str, + package_name: CrateName, package_type: PackageType, ) { let src_dir = package_dir.join(SRC_DIR); diff --git a/crates/nargo_cli/src/cli/new_cmd.rs b/crates/nargo_cli/src/cli/new_cmd.rs index dbdeddef712..d6a9d00257b 100644 --- a/crates/nargo_cli/src/cli/new_cmd.rs +++ b/crates/nargo_cli/src/cli/new_cmd.rs @@ -4,6 +4,7 @@ use super::{init_cmd::initialize_project, NargoConfig}; use acvm::Backend; use clap::Args; use nargo::package::PackageType; +use noirc_frontend::graph::CrateName; use std::path::PathBuf; /// Create a Noir project in a new directory. @@ -14,7 +15,7 @@ pub(crate) struct NewCommand { /// Name of the package [default: package directory name] #[clap(long)] - name: Option, + name: Option, /// Use a library template #[arg(long, conflicts_with = "bin", conflicts_with = "contract")] @@ -41,8 +42,13 @@ pub(crate) fn run( return Err(CliError::DestinationAlreadyExists(package_dir)); } - let package_name = - args.name.unwrap_or_else(|| args.path.file_name().unwrap().to_str().unwrap().to_owned()); + let package_name = match args.name { + Some(name) => name, + None => { + let name = args.path.file_name().unwrap().to_str().unwrap(); + name.parse().map_err(|_| CliError::InvalidPackageName(name.into()))? + } + }; let package_type = if args.lib { PackageType::Library } else if args.contract { @@ -50,6 +56,6 @@ pub(crate) fn run( } else { PackageType::Binary }; - initialize_project(package_dir, &package_name, package_type); + initialize_project(package_dir, package_name, package_type); Ok(()) } diff --git a/crates/nargo_cli/src/errors.rs b/crates/nargo_cli/src/errors.rs index d801a56d3f5..dade4262431 100644 --- a/crates/nargo_cli/src/errors.rs +++ b/crates/nargo_cli/src/errors.rs @@ -41,6 +41,9 @@ pub(crate) enum CliError { #[error("Failed to verify proof {}", .0.display())] InvalidProof(PathBuf), + #[error("Invalid package name {0}. Did you mean to use `--name`?")] + InvalidPackageName(String), + /// ABI encoding/decoding error #[error(transparent)] AbiError(#[from] AbiError), diff --git a/crates/nargo_cli/tests/compile_failure/invalid_dependency_name/Nargo.toml b/crates/nargo_cli/tests/compile_failure/invalid_dependency_name/Nargo.toml new file mode 100644 index 00000000000..69b56a2e7d0 --- /dev/null +++ b/crates/nargo_cli/tests/compile_failure/invalid_dependency_name/Nargo.toml @@ -0,0 +1,8 @@ +[package] +name = "invalid_dependency_name" +type = "bin" +authors = [""] +compiler_version = "0.9.0" + +[dependencies] +bad_name = { path = "../../test_libraries/bad_name" } diff --git a/crates/nargo_cli/tests/compile_failure/invalid_dependency_name/src/main.nr b/crates/nargo_cli/tests/compile_failure/invalid_dependency_name/src/main.nr new file mode 100644 index 00000000000..faf1ba0045a --- /dev/null +++ b/crates/nargo_cli/tests/compile_failure/invalid_dependency_name/src/main.nr @@ -0,0 +1 @@ +fn main(x: Field) { } diff --git a/crates/nargo_cli/tests/compile_failure/package_name_empty/Nargo.toml b/crates/nargo_cli/tests/compile_failure/package_name_empty/Nargo.toml new file mode 100644 index 00000000000..7c4dbd0c994 --- /dev/null +++ b/crates/nargo_cli/tests/compile_failure/package_name_empty/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "" +type = "bin" +authors = [""] +compiler_version = "0.9.0" + +[dependencies] diff --git a/crates/nargo_cli/tests/compile_failure/package_name_empty/src/main.nr b/crates/nargo_cli/tests/compile_failure/package_name_empty/src/main.nr new file mode 100644 index 00000000000..faf1ba0045a --- /dev/null +++ b/crates/nargo_cli/tests/compile_failure/package_name_empty/src/main.nr @@ -0,0 +1 @@ +fn main(x: Field) { } diff --git a/crates/nargo_cli/tests/compile_failure/package_name_hyphen/Nargo.toml b/crates/nargo_cli/tests/compile_failure/package_name_hyphen/Nargo.toml new file mode 100644 index 00000000000..f8d0a85db4a --- /dev/null +++ b/crates/nargo_cli/tests/compile_failure/package_name_hyphen/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "hyphenated-name" +type = "bin" +authors = [""] +compiler_version = "0.9.0" + +[dependencies] diff --git a/crates/nargo_cli/tests/compile_failure/package_name_hyphen/src/main.nr b/crates/nargo_cli/tests/compile_failure/package_name_hyphen/src/main.nr new file mode 100644 index 00000000000..faf1ba0045a --- /dev/null +++ b/crates/nargo_cli/tests/compile_failure/package_name_hyphen/src/main.nr @@ -0,0 +1 @@ +fn main(x: Field) { } diff --git a/crates/nargo_cli/tests/test_libraries/bad_name/Nargo.toml b/crates/nargo_cli/tests/test_libraries/bad_name/Nargo.toml new file mode 100644 index 00000000000..313e2411e83 --- /dev/null +++ b/crates/nargo_cli/tests/test_libraries/bad_name/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "bad-name" +type = "lib" +authors = [""] +compiler_version = "0.7.1" + +[dependencies] diff --git a/crates/nargo_cli/tests/test_libraries/bad_name/src/lib.nr b/crates/nargo_cli/tests/test_libraries/bad_name/src/lib.nr new file mode 100644 index 00000000000..e69de29bb2d diff --git a/crates/nargo_toml/src/errors.rs b/crates/nargo_toml/src/errors.rs index 42c5a59cba6..e986a181e43 100644 --- a/crates/nargo_toml/src/errors.rs +++ b/crates/nargo_toml/src/errors.rs @@ -38,9 +38,11 @@ pub enum ManifestError { )] MissingDefaultEntryFile { toml: PathBuf, entry: PathBuf, package_type: PackageType }, - /// Invalid character `-` in package name - #[error("invalid character `-` in package name")] - InvalidPackageName, + #[error("{} found in {toml}", if name.is_empty() { "Empty package name".into() } else { format!("Invalid package name `{name}`") })] + InvalidPackageName { toml: PathBuf, name: String }, + + #[error("{} found in {toml}", if name.is_empty() { "Empty dependency name".into() } else { format!("Invalid dependency name `{name}`") })] + InvalidDependencyName { toml: PathBuf, name: String }, /// Encountered error while downloading git repository. #[error("{0}")] diff --git a/crates/nargo_toml/src/lib.rs b/crates/nargo_toml/src/lib.rs index 4a9238ee021..0e05a0a7901 100644 --- a/crates/nargo_toml/src/lib.rs +++ b/crates/nargo_toml/src/lib.rs @@ -72,14 +72,20 @@ struct PackageConfig { impl PackageConfig { fn resolve_to_package(&self, root_dir: &Path) -> Result { let name = if let Some(name) = &self.package.name { - name.parse().map_err(|_| ManifestError::InvalidPackageName)? + name.parse().map_err(|_| ManifestError::InvalidPackageName { + toml: root_dir.join("Nargo.toml"), + name: name.into(), + })? } else { return Err(ManifestError::MissingNameField { toml: root_dir.join("Nargo.toml") }); }; let mut dependencies: BTreeMap = BTreeMap::new(); for (name, dep_config) in self.dependencies.iter() { - let name = name.parse().map_err(|_| ManifestError::InvalidPackageName)?; + let name = name.parse().map_err(|_| ManifestError::InvalidDependencyName { + toml: root_dir.join("Nargo.toml"), + name: name.into(), + })?; let resolved_dep = dep_config.resolve_to_dependency(root_dir)?; dependencies.insert(name, resolved_dep); diff --git a/crates/noirc_frontend/src/graph/mod.rs b/crates/noirc_frontend/src/graph/mod.rs index 2c251432e9b..7a1daef45fd 100644 --- a/crates/noirc_frontend/src/graph/mod.rs +++ b/crates/noirc_frontend/src/graph/mod.rs @@ -29,6 +29,12 @@ impl CrateId { #[derive(Debug, Clone, PartialEq, Eq, Hash, Ord, PartialOrd)] pub struct CrateName(SmolStr); +impl CrateName { + fn is_valid_name(name: &str) -> bool { + !name.is_empty() && name.chars().all(|n| !CHARACTER_BLACK_LIST.contains(&n)) + } +} + impl Display for CrateName { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { self.0.fmt(f) @@ -54,11 +60,28 @@ impl FromStr for CrateName { type Err = String; fn from_str(name: &str) -> Result { - let is_invalid = name.chars().any(|n| CHARACTER_BLACK_LIST.contains(&n)); - if is_invalid { - Err(name.into()) - } else { + if Self::is_valid_name(name) { Ok(Self(SmolStr::new(name))) + } else { + Err("Package names must be non-empty and cannot contain hyphens".into()) + } + } +} + +#[cfg(test)] +mod crate_name { + use super::{CrateName, CHARACTER_BLACK_LIST}; + + #[test] + fn it_rejects_empty_string() { + assert!(!CrateName::is_valid_name("")); + } + + #[test] + fn it_rejects_blacklisted_chars() { + for bad_char in CHARACTER_BLACK_LIST { + let bad_char_string = bad_char.to_string(); + assert!(!CrateName::is_valid_name(&bad_char_string)); } } }