use crate::cli::commands::template::publish::TemplatePublishArgs;
use crate::cli::config::Config;
use crate::cli::util;
use crate::{loading, project};
use anyhow::{Context, anyhow};
use cargo_toml::Manifest;
use clap::Parser;
use std::path::{Path, PathBuf};
use std::process::Stdio;
use tari_ootle_common_types::Network;
use tari_ootle_publish_lib::walletd_client::ComponentAddressOrName;
use tokio::fs;
use tokio::process::Command;
#[derive(Clone, Parser, Debug)]
pub struct PublishArgs {
#[arg(default_value = ".")]
pub path: PathBuf,
#[arg(short = 'a', long)]
pub account: Option<ComponentAddressOrName>,
#[arg(short = 'c', long)]
pub custom_network: Option<String>,
#[arg(short = 'y', long, default_value_t = false)]
pub yes: bool,
#[arg(short = 'f', long)]
pub max_fee: Option<u64>,
#[arg(long, alias = "bin")]
pub binary: Option<PathBuf>,
#[arg(long)]
pub wallet_daemon_url: Option<url::Url>,
#[arg(long, default_value_t = false)]
pub publish_metadata: bool,
#[arg(long)]
pub metadata_server_url: Option<url::Url>,
}
pub async fn build_template(crate_dir: &Path) -> anyhow::Result<PathBuf> {
let cargo_path = crate_dir.join("Cargo.toml");
if !cargo_path.exists() {
return Err(anyhow!("No Cargo.toml found at {}", cargo_path.display()));
}
let manifest = Manifest::from_path(&cargo_path)?;
let crate_name = manifest
.package
.ok_or_else(|| anyhow!("No [package] section in {}", cargo_path.display()))?
.name;
let template_bin = loading!(
format!("Building WASM template project **{}**", crate_name),
build_project(crate_dir, &crate_name).await
)?;
Ok(template_bin)
}
pub async fn handle(config: Config, network_override: Option<Network>, args: PublishArgs) -> anyhow::Result<()> {
let template_args = TemplatePublishArgs {
path: args.path,
account: args.account,
custom_network: args.custom_network,
yes: args.yes,
max_fee: args.max_fee,
binary: args.binary,
wallet_daemon_url: args.wallet_daemon_url,
publish_metadata: args.publish_metadata,
metadata_server_url: args.metadata_server_url,
};
crate::cli::commands::template::publish::handle(config, network_override, template_args).await
}
async fn build_project(dir: &Path, name: &str) -> anyhow::Result<PathBuf> {
let mut cmd = Command::new("cargo");
cmd.arg("build")
.arg("--target=wasm32-unknown-unknown")
.arg("--release")
.current_dir(dir)
.stdout(Stdio::piped())
.stderr(Stdio::piped());
let process = cmd.spawn()?;
let output = process.wait_with_output().await?;
if !output.status.success() {
return Err(anyhow!(
"Failed to build project: {dir:?}\nBuild Output:\n\n{}",
String::from_utf8_lossy(&output.stderr)
));
}
let target_dir = find_target_dir(dir).await?;
let wasm_name = name.replace('-', "_");
let output_bin = target_dir
.join("wasm32-unknown-unknown")
.join("release")
.join(format!("{wasm_name}.wasm"));
if !util::file_exists(&output_bin).await? {
return Err(anyhow!(
"Binary is not present after build at {:?}\n\nBuild Output:\n{}",
output_bin,
String::from_utf8_lossy(&output.stderr)
));
}
Ok(output_bin)
}
pub async fn find_target_dir(dir: &Path) -> anyhow::Result<PathBuf> {
let output = Command::new("cargo")
.args(["metadata", "--format-version=1", "--no-deps"])
.current_dir(dir)
.stdout(Stdio::piped())
.stderr(Stdio::piped())
.output()
.await?;
if !output.status.success() {
return Err(anyhow!(
"Failed to get cargo metadata: {}",
String::from_utf8_lossy(&output.stderr)
));
}
let metadata: serde_json::Value = serde_json::from_slice(&output.stdout).context("parsing cargo metadata")?;
metadata["target_directory"]
.as_str()
.map(PathBuf::from)
.ok_or_else(|| anyhow!("cargo metadata missing target_directory"))
}
const METADATA_CBOR_FILENAME: &str = "template_metadata.cbor";
pub async fn find_metadata_cbor(project_dir: &Path) -> anyhow::Result<PathBuf> {
let target_dir = find_target_dir(project_dir).await?;
let build_dir = target_dir.join("wasm32-unknown-unknown").join("release").join("build");
if !build_dir.exists() {
return Err(anyhow!(
"Build output directory not found at {}. Run `tari build` first.",
build_dir.display()
));
}
let mut newest: Option<(PathBuf, std::time::SystemTime)> = None;
for entry in std::fs::read_dir(&build_dir).context("reading build directory")? {
let entry = entry?;
let out_file = entry.path().join("out").join(METADATA_CBOR_FILENAME);
if out_file.exists() {
let modified = std::fs::metadata(&out_file)
.and_then(|m| m.modified())
.unwrap_or(std::time::SystemTime::UNIX_EPOCH);
if newest.as_ref().is_none_or(|(_, t)| modified > *t) {
newest = Some((out_file, modified));
}
}
}
newest.map(|(path, _)| path).ok_or_else(|| {
anyhow!(
"No {METADATA_CBOR_FILENAME} found in build output. \
Make sure the template uses tari_ootle_template_build in build.rs \
and has been built with `tari build`."
)
})
}
pub async fn load_project_config(project_folder: &Path) -> anyhow::Result<project::ProjectConfig> {
let mut search_dir = project_folder.to_path_buf();
loop {
let config_file = search_dir.join(project::CONFIG_FILE_NAME);
if config_file.exists() {
let content = fs::read_to_string(&config_file).await.map_err(|error| {
anyhow!(
"Failed to load project config file (at {}): {}",
config_file.display(),
error
)
})?;
return toml::from_str::<project::ProjectConfig>(content.as_str()).context("parsing config toml");
}
if !search_dir.pop() {
break;
}
}
Ok(project::ProjectConfig::default())
}
pub fn resolve_active_network(
cli_override: Option<Network>,
project: &project::ProjectConfig,
global: &Config,
) -> Network {
cli_override
.or_else(|| project.default_network())
.or(global.default_network)
.unwrap_or_default()
}
pub fn resolve_wallet_daemon_url(
cli_override: Option<&url::Url>,
project: &project::ProjectConfig,
global: &Config,
network: Network,
) -> url::Url {
cli_override
.cloned()
.or_else(|| project.wallet_daemon_url(network).cloned())
.or_else(|| global.wallet_daemon_url(network).cloned())
.unwrap_or_else(|| {
url::Url::parse(project::DEFAULT_WALLET_DAEMON_URL).expect("default wallet daemon URL is valid")
})
}