use super::*;
use crate::{
execute_fee,
execute_program,
log,
process_inputs,
OfflineQuery,
PrivateKey,
RecordPlaintext,
Transaction,
};
use crate::types::native::{
CurrentAleo,
IdentifierNative,
ProcessNative,
ProgramNative,
RecordPlaintextNative,
TransactionNative,
};
use js_sys::Array;
use rand::{rngs::StdRng, SeedableRng};
use std::str::FromStr;
#[wasm_bindgen]
impl ProgramManager {
#[wasm_bindgen(js_name = buildJoinTransaction)]
#[allow(clippy::too_many_arguments)]
pub async fn join(
private_key: &PrivateKey,
record_1: RecordPlaintext,
record_2: RecordPlaintext,
fee_credits: f64,
fee_record: Option<RecordPlaintext>,
url: Option<String>,
join_proving_key: Option<ProvingKey>,
join_verifying_key: Option<VerifyingKey>,
fee_proving_key: Option<ProvingKey>,
fee_verifying_key: Option<VerifyingKey>,
offline_query: Option<OfflineQuery>,
) -> Result<Transaction, String> {
log("Executing join program");
let fee_microcredits = match &fee_record {
Some(fee_record) => Self::validate_amount(fee_credits, fee_record, true)?,
None => (fee_credits * 1_000_000.0) as u64,
};
let rng = &mut StdRng::from_entropy();
log("Setup program and inputs");
let node_url = url.as_deref().unwrap_or(DEFAULT_URL);
let program = ProgramNative::credits().unwrap().to_string();
let inputs = Array::new_with_length(2);
inputs.set(0u32, wasm_bindgen::JsValue::from_str(&record_1.to_string()));
inputs.set(1u32, wasm_bindgen::JsValue::from_str(&record_2.to_string()));
let mut process_native = ProcessNative::load_web().map_err(|err| err.to_string())?;
let process = &mut process_native;
let stack = process.get_stack("credits.aleo").map_err(|e| e.to_string())?;
let fee_identifier = if fee_record.is_some() {
IdentifierNative::from_str("fee_private").map_err(|e| e.to_string())?
} else {
IdentifierNative::from_str("fee_public").map_err(|e| e.to_string())?
};
if !stack.contains_proving_key(&fee_identifier) && fee_proving_key.is_some() && fee_verifying_key.is_some() {
let fee_proving_key = fee_proving_key.clone().unwrap();
let fee_verifying_key = fee_verifying_key.clone().unwrap();
stack
.insert_proving_key(&fee_identifier, ProvingKeyNative::from(fee_proving_key))
.map_err(|e| e.to_string())?;
stack
.insert_verifying_key(&fee_identifier, VerifyingKeyNative::from(fee_verifying_key))
.map_err(|e| e.to_string())?;
}
log("Executing the join function");
let (_, mut trace) = execute_program!(
process,
process_inputs!(inputs),
&program,
"join",
private_key,
join_proving_key,
join_verifying_key,
rng
);
log("Preparing inclusion proof for the join execution");
if let Some(offline_query) = offline_query.as_ref() {
trace.prepare_async(offline_query.clone()).await.map_err(|err| err.to_string())?;
} else {
let query = QueryNative::from(node_url);
trace.prepare_async(query).await.map_err(|err| err.to_string())?;
}
log("Proving the join execution");
let execution = trace.prove_execution::<CurrentAleo, _>("credits.aleo/join", rng).map_err(|e| e.to_string())?;
let execution_id = execution.to_execution_id().map_err(|e| e.to_string())?;
log("Verifying the join execution");
process.verify_execution(&execution).map_err(|err| err.to_string())?;
log("Executing the fee");
let fee = execute_fee!(
process,
private_key,
fee_record,
fee_microcredits,
node_url,
fee_proving_key,
fee_verifying_key,
execution_id,
rng,
offline_query
);
log("Creating execution transaction for join");
let transaction = TransactionNative::from_execution(execution, Some(fee)).map_err(|err| err.to_string())?;
Ok(Transaction::from(transaction))
}
}