use cargo_metadata::{Metadata, MetadataCommand, Package};
use clap::Parser;
use itertools::Itertools;
use std::{
borrow::Cow,
collections::HashSet,
env,
ffi::OsStr,
fmt::Debug,
fs, io,
path::{self, Path, PathBuf},
process::{Command, ExitStatus, Stdio},
};
use stellar_xdr::curr::{Limits, ScMetaEntry, ScMetaV0, StringM, WriteXdr};
use crate::{commands::global, print::Print};
#[derive(Parser, Debug, Clone)]
pub struct Cmd {
#[arg(long)]
pub manifest_path: Option<std::path::PathBuf>,
#[arg(long)]
pub package: Option<String>,
#[arg(long, default_value = "release")]
pub profile: String,
#[arg(long, help_heading = "Features")]
pub features: Option<String>,
#[arg(
long,
conflicts_with = "features",
conflicts_with = "no_default_features",
help_heading = "Features"
)]
pub all_features: bool,
#[arg(long, help_heading = "Features")]
pub no_default_features: bool,
#[arg(long)]
pub out_dir: Option<std::path::PathBuf>,
#[arg(long, conflicts_with = "out_dir", help_heading = "Other")]
pub print_commands_only: bool,
#[arg(long, num_args=1, value_parser=parse_meta_arg, action=clap::ArgAction::Append, help_heading = "Metadata")]
pub meta: Vec<(String, String)>,
}
fn parse_meta_arg(s: &str) -> Result<(String, String), Error> {
let parts = s.splitn(2, '=');
let (key, value) = parts
.map(str::trim)
.next_tuple()
.ok_or_else(|| Error::MetaArg("must be in the form 'key=value'".to_string()))?;
Ok((key.to_string(), value.to_string()))
}
#[derive(thiserror::Error, Debug)]
pub enum Error {
#[error(transparent)]
Metadata(#[from] cargo_metadata::Error),
#[error(transparent)]
CargoCmd(io::Error),
#[error("exit status {0}")]
Exit(ExitStatus),
#[error("package {package} not found")]
PackageNotFound { package: String },
#[error("finding absolute path of Cargo.toml: {0}")]
AbsolutePath(io::Error),
#[error("creating out directory: {0}")]
CreatingOutDir(io::Error),
#[error("copying wasm file: {0}")]
CopyingWasmFile(io::Error),
#[error("getting the current directory: {0}")]
GettingCurrentDir(io::Error),
#[error("retreiving CARGO_HOME: {0}")]
CargoHome(io::Error),
#[error("reading wasm file: {0}")]
ReadingWasmFile(io::Error),
#[error("writing wasm file: {0}")]
WritingWasmFile(io::Error),
#[error("invalid meta entry: {0}")]
MetaArg(String),
}
const WASM_TARGET: &str = "wasm32-unknown-unknown";
const META_CUSTOM_SECTION_NAME: &str = "contractmetav0";
impl Cmd {
pub fn run(&self, global_args: &global::Args) -> Result<(), Error> {
let print = Print::new(global_args.quiet);
let working_dir = env::current_dir().map_err(Error::GettingCurrentDir)?;
let metadata = self.metadata()?;
let packages = self.packages(&metadata)?;
let target_dir = &metadata.target_directory;
if let Some(package) = &self.package {
if packages.is_empty() {
return Err(Error::PackageNotFound {
package: package.clone(),
});
}
}
for p in packages {
let mut cmd = Command::new("cargo");
cmd.stdout(Stdio::piped());
cmd.arg("rustc");
let manifest_path = pathdiff::diff_paths(&p.manifest_path, &working_dir)
.unwrap_or(p.manifest_path.clone().into());
cmd.arg(format!(
"--manifest-path={}",
manifest_path.to_string_lossy()
));
cmd.arg("--crate-type=cdylib");
cmd.arg(format!("--target={WASM_TARGET}"));
if self.profile == "release" {
cmd.arg("--release");
} else {
cmd.arg(format!("--profile={}", self.profile));
}
if self.all_features {
cmd.arg("--all-features");
}
if self.no_default_features {
cmd.arg("--no-default-features");
}
if let Some(features) = self.features() {
let requested: HashSet<String> = features.iter().cloned().collect();
let available = p.features.iter().map(|f| f.0).cloned().collect();
let activate = requested.intersection(&available).join(",");
if !activate.is_empty() {
cmd.arg(format!("--features={activate}"));
}
}
if let Some(rustflags) = make_rustflags_to_remap_absolute_paths(&print)? {
cmd.env("CARGO_BUILD_RUSTFLAGS", rustflags);
}
let mut cmd_str_parts = Vec::<String>::new();
cmd_str_parts.extend(cmd.get_envs().map(|(key, val)| {
format!(
"{}={}",
key.to_string_lossy(),
shell_escape::escape(val.unwrap_or_default().to_string_lossy())
)
}));
cmd_str_parts.push("cargo".to_string());
cmd_str_parts.extend(
cmd.get_args()
.map(OsStr::to_string_lossy)
.map(Cow::into_owned),
);
let cmd_str = cmd_str_parts.join(" ");
if self.print_commands_only {
println!("{cmd_str}");
} else {
print.infoln(cmd_str);
let status = cmd.status().map_err(Error::CargoCmd)?;
if !status.success() {
return Err(Error::Exit(status));
}
let file = format!("{}.wasm", p.name.replace('-', "_"));
let target_file_path = Path::new(target_dir)
.join(WASM_TARGET)
.join(&self.profile)
.join(&file);
self.handle_contract_metadata_args(&target_file_path)?;
if let Some(out_dir) = &self.out_dir {
fs::create_dir_all(out_dir).map_err(Error::CreatingOutDir)?;
let out_file_path = Path::new(out_dir).join(&file);
fs::copy(target_file_path, out_file_path).map_err(Error::CopyingWasmFile)?;
}
}
}
Ok(())
}
fn features(&self) -> Option<Vec<String>> {
self.features
.as_ref()
.map(|f| f.split(&[',', ' ']).map(String::from).collect())
}
fn packages(&self, metadata: &Metadata) -> Result<Vec<Package>, Error> {
let name = if let Some(name) = self.package.clone() {
Some(name)
} else {
let manifest_path = path::absolute(
self.manifest_path
.clone()
.unwrap_or(PathBuf::from("Cargo.toml")),
)
.map_err(Error::AbsolutePath)?;
metadata
.packages
.iter()
.find(|p| p.manifest_path == manifest_path)
.map(|p| p.name.clone())
};
let packages = metadata
.packages
.iter()
.filter(|p|
if let Some(name) = &name {
&p.name == name
} else {
metadata.workspace_default_members.contains(&p.id)
&& p.targets
.iter()
.any(|t| t.crate_types.iter().any(|c| c == "cdylib"))
}
)
.cloned()
.collect();
Ok(packages)
}
fn metadata(&self) -> Result<Metadata, cargo_metadata::Error> {
let mut cmd = MetadataCommand::new();
cmd.no_deps();
if let Some(manifest_path) = &self.manifest_path {
cmd.manifest_path(manifest_path);
}
cmd.exec()
}
fn handle_contract_metadata_args(&self, target_file_path: &PathBuf) -> Result<(), Error> {
if self.meta.is_empty() {
return Ok(());
}
let mut wasm_bytes = fs::read(target_file_path).map_err(Error::ReadingWasmFile)?;
for (k, v) in self.meta.clone() {
let key: StringM = k
.clone()
.try_into()
.map_err(|e| Error::MetaArg(format!("{k} is an invalid metadata key: {e}")))?;
let val: StringM = v
.clone()
.try_into()
.map_err(|e| Error::MetaArg(format!("{v} is an invalid metadata value: {e}")))?;
let meta_entry = ScMetaEntry::ScMetaV0(ScMetaV0 { key, val });
let xdr: Vec<u8> = meta_entry
.to_xdr(Limits::none())
.map_err(|e| Error::MetaArg(format!("failed to encode metadata entry: {e}")))?;
wasm_gen::write_custom_section(&mut wasm_bytes, META_CUSTOM_SECTION_NAME, &xdr);
}
fs::write(target_file_path, wasm_bytes).map_err(Error::WritingWasmFile)
}
}
fn make_rustflags_to_remap_absolute_paths(print: &Print) -> Result<Option<String>, Error> {
let cargo_home = home::cargo_home().map_err(Error::CargoHome)?;
let cargo_home = format!("{}", cargo_home.display());
if cargo_home.find(|c: char| c.is_whitespace()).is_some() {
print.warnln("Cargo home directory contains whitespace. Dependency paths will not be remapped; builds may not be reproducible.");
return Ok(None);
}
if env::var("RUSTFLAGS").is_ok() {
print.warnln("`RUSTFLAGS` set. Dependency paths will not be remapped; builds may not be reproducible.");
return Ok(None);
}
if env::var("CARGO_ENCODED_RUSTFLAGS").is_ok() {
print.warnln("`CARGO_ENCODED_RUSTFLAGS` set. Dependency paths will not be remapped; builds may not be reproducible.");
return Ok(None);
}
if env::var("TARGET_wasm32-unknown-unknown_RUSTFLAGS").is_ok() {
print.warnln("`TARGET_wasm32-unknown-unknown_RUSTFLAGS` set. Dependency paths will not be remapped; builds may not be reproducible.");
return Ok(None);
}
let registry_prefix = format!("{cargo_home}/registry/src/");
let new_rustflag = format!("--remap-path-prefix={registry_prefix}=");
let mut rustflags = get_rustflags().unwrap_or_default();
rustflags.push(new_rustflag);
let rustflags = rustflags.join(" ");
Ok(Some(rustflags))
}
fn get_rustflags() -> Option<Vec<String>> {
if let Ok(a) = env::var("CARGO_BUILD_RUSTFLAGS") {
let args = a
.split_whitespace()
.map(str::trim)
.filter(|s| !s.is_empty())
.map(str::to_string);
return Some(args.collect());
}
None
}