// Copyright (C) 2019-2023 Aleo Systems Inc.
// This file is part of the Aleo SDK library.
// The Aleo SDK library is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// The Aleo SDK library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with the Aleo SDK library. If not, see .
use crate::{Aleo, CurrentNetwork};
use snarkvm::{
package::Package,
prelude::{Identifier, Locator, Value},
};
use anyhow::Result;
use clap::Parser;
use colored::Colorize;
use core::str::FromStr;
use snarkvm::console::account::PrivateKey;
use std::collections::HashMap;
pub const LOCALE: &num_format::Locale = &num_format::Locale::en;
/// Executes an Aleo program function locally
#[derive(Debug, Parser)]
pub struct Run {
/// The function name.
function: Identifier,
/// The function inputs.
inputs: Vec>,
/// Optional Private key to run the program with
#[clap(short = 'k', long)]
private_key: Option>,
}
impl Run {
/// Compiles an Aleo program function with the specified name.
#[allow(clippy::format_in_format_args)]
pub fn parse(self) -> Result {
// Derive the program directory path.
let path = std::env::current_dir()?;
// Load the package.
let package = Package::open(&path)?;
// Use a provided private key or generate a temporary one
let private_key = self.private_key.unwrap_or(PrivateKey::new(&mut rand::thread_rng()).unwrap());
// Initialize an RNG.
let rng = &mut rand::thread_rng();
// Execute the request.
let (response, metrics) = package.run::(&private_key, self.function, &self.inputs, rng)?;
// Count the number of times a function is called.
let mut program_frequency = HashMap::::new();
for metric in metrics.iter() {
// Prepare the function name string.
let function_name_string = format!("'{}/{}'", metric.program_id, metric.function_name).bold();
// Prepare the function constraints string
let function_constraints_string = format!(
"{function_name_string} - {} constraints",
metric.num_function_constraints.to_formatted_string(LOCALE)
);
// Increment the counter for the function call.
match program_frequency.get_mut(&function_constraints_string) {
Some(counter) => *counter += 1,
None => {
let _ = program_frequency.insert(function_constraints_string, 1);
}
}
}
// Log the metrics.
use num_format::ToFormattedString;
println!("⛓ Constraints\n");
for (function_constraints, counter) in program_frequency {
// Log the constraints
let counter_string = match counter {
1 => "(called 1 time)".to_string().dimmed(),
counter => format!("(called {counter} times)").dimmed(),
};
println!(" • {function_constraints} {counter_string}",)
}
// Log the outputs.
match response.outputs().len() {
0 => (),
1 => println!("\n➡️ Output\n"),
_ => println!("\n➡️ Outputs\n"),
};
for output in response.outputs() {
println!("{}", format!(" • {output}"));
}
println!();
// Prepare the locator.
let locator = Locator::::from_str(&format!("{}/{}", package.program_id(), self.function))?;
// Prepare the path string.
let path_string = format!("(in \"{}\")", path.display());
Ok(format!("✅ Finished '{}' {}", locator.to_string().bold(), path_string.dimmed()))
}
}