use crate::common::types::BlockFees; use grin_api::client; use grin_api::json_rpc::Response; use grin_core::core::{FeeFields, Output, OutputFeatures, Transaction, TxKernel}; use grin_core::global::ChainTypes; use grin_core::libtx::tx_fee; use grin_keychain::{BlindingFactor, ExtKeychain, Identifier, Keychain, SwitchCommitmentType}; use grin_onion::crypto::comsig::ComSignature; use grin_onion::onion::Onion; use grin_onion::Hop; use grin_util::{Mutex, ZeroingString}; use grin_wallet_api::Owner; use grin_wallet_config::WalletConfig; use grin_wallet_controller::controller; use grin_wallet_impls::{DefaultLCProvider, DefaultWalletImpl, HTTPNodeClient}; use grin_wallet_libwallet::{InitTxArgs, Slate, VersionedSlate, WalletInfo, WalletInst}; use log::error; use mwixnet::wallet::HttpWallet; use secp256k1zkp::pedersen::Commitment; use secp256k1zkp::{Secp256k1, SecretKey}; use serde_derive::{Deserialize, Serialize}; use serde_json::json; use std::net::{IpAddr, Ipv4Addr, SocketAddr}; use std::sync::Arc; use std::thread; use x25519_dalek::PublicKey as xPublicKey; /// Response to build a coinbase output. #[derive(Serialize, Deserialize, Debug, Clone)] pub struct CbData { /// Output pub output: Output, /// Kernel pub kernel: TxKernel, /// Key Id pub key_id: Option, } pub struct IntegrationGrinWallet { wallet: Arc< Mutex< Box< dyn WalletInst< 'static, DefaultLCProvider<'static, HTTPNodeClient, ExtKeychain>, HTTPNodeClient, ExtKeychain, >, >, >, >, api_listen_port: u16, owner_api: Arc< Owner, HTTPNodeClient, ExtKeychain>, >, http_client: Arc, } impl IntegrationGrinWallet { pub async fn async_new_wallet( wallet_dir: String, api_listen_port: u16, node_api: String, ) -> IntegrationGrinWallet { let node_client = HTTPNodeClient::new(&node_api, None).unwrap(); let mut wallet = Box::new( DefaultWalletImpl::<'static, HTTPNodeClient>::new(node_client.clone()).unwrap(), ) as Box< dyn WalletInst< 'static, DefaultLCProvider, HTTPNodeClient, ExtKeychain, >, >; // Wallet LifeCycle Provider provides all functions init wallet and work with seeds, etc... let lc = wallet.lc_provider().unwrap(); let mut wallet_config = WalletConfig::default(); wallet_config.check_node_api_http_addr = node_api.clone(); wallet_config.owner_api_listen_port = Some(api_listen_port); wallet_config.api_secret_path = None; wallet_config.data_file_dir = wallet_dir.clone(); // The top level wallet directory should be set manually (in the reference implementation, // this is provided in the WalletConfig) let _ = lc.set_top_level_directory(&wallet_config.data_file_dir); lc.create_config( &ChainTypes::AutomatedTesting, "grin-wallet.toml", Some(wallet_config.clone()), None, None, ) .unwrap(); lc.create_wallet(None, None, 12, ZeroingString::from("pass"), false) .unwrap(); // Start owner API let km = Arc::new(Mutex::new(None)); let wallet = Arc::new(Mutex::new(wallet)); let owner_api = Arc::new(Owner::new(wallet.clone(), None)); let address_str = format!("127.0.0.1:{}", api_listen_port); let owner_addr: SocketAddr = address_str.parse().unwrap(); let thr_wallet = wallet.clone(); let _thread_handle = thread::spawn(move || { controller::owner_listener( thr_wallet, km, address_str.as_str(), None, None, Some(true), None, false, ) .unwrap() }); let http_client = Arc::new( HttpWallet::async_open_wallet(&owner_addr, &None, &ZeroingString::from("pass")) .await .unwrap(), ); IntegrationGrinWallet { wallet, api_listen_port, owner_api, http_client, } } pub async fn async_retrieve_summary_info(&self) -> Result { let params = json!({ "token": self.http_client.clone().get_token(), "refresh_from_node": true, "minimum_confirmations": 1 }); let (_, wallet_info): (bool, WalletInfo) = self .http_client .clone() .async_perform_request("retrieve_summary_info", ¶ms) .await?; Ok(wallet_info) } pub async fn async_send( &self, receiving_wallet: &IntegrationGrinWallet, amount: u64, ) -> Result { let slate = self.async_init_send_tx(amount).await.unwrap(); let slate = receiving_wallet.async_receive_tx(&slate).await.unwrap(); let slate = self.async_finalize_tx(&slate).await.unwrap(); let tx = Slate::from(slate).tx_or_err().unwrap().clone(); Ok(tx) } async fn async_init_send_tx( &self, amount: u64, ) -> Result { let args = InitTxArgs { src_acct_name: None, amount, minimum_confirmations: 0, max_outputs: 10, num_change_outputs: 1, selection_strategy_is_use_all: false, ..Default::default() }; let params = json!({ "token": self.http_client.clone().get_token(), "args": args }); let slate: VersionedSlate = self .http_client .clone() .async_perform_request("init_send_tx", ¶ms) .await?; let params = json!({ "token": self.http_client.clone().get_token(), "slate": &slate }); self.http_client .clone() .async_perform_request("tx_lock_outputs", ¶ms) .await?; Ok(slate) } pub async fn async_receive_tx( &self, slate: &VersionedSlate, ) -> Result { let req_body = json!({ "jsonrpc": "2.0", "method": "receive_tx", "id": 1, "params": [slate, null, null] }); let res: Response = client::post_async(self.foreign_api().as_str(), &req_body, None) .await .map_err(|e| { let report = format!("Failed to receive tx. Is the wallet listening? {}", e); error!("{}", report); grin_servers::common::types::Error::WalletComm(report) })?; let parsed: VersionedSlate = res.clone().into_result().map_err(|e| { let report = format!("Error parsing result: {}", e); error!("{}", report); grin_servers::common::types::Error::WalletComm(report) })?; Ok(parsed) } async fn async_finalize_tx( &self, slate: &VersionedSlate, ) -> Result { let params = json!({ "token": self.http_client.clone().get_token(), "slate": slate }); self.http_client .clone() .async_perform_request("finalize_tx", ¶ms) .await } async fn async_post_tx( &self, finalized_slate: &VersionedSlate, fluff: bool, ) -> Result { let params = json!({ "token": self.http_client.clone().get_token(), "slate": finalized_slate, "fluff": fluff }); self.http_client .clone() .async_perform_request("post_tx", ¶ms) .await } /// Call the wallet API to create a coinbase output for the given block_fees. /// Will retry based on default "retry forever with backoff" behavior. pub async fn async_create_coinbase( &self, block_fees: &BlockFees, ) -> Result { let req_body = json!({ "jsonrpc": "2.0", "method": "build_coinbase", "id": 1, "params": { "block_fees": block_fees } }); let res: Response = client::post_async(self.foreign_api().as_str(), &req_body, None) .await .map_err(|e| { let report = format!("Failed to get coinbase. Is the wallet listening? {}", e); error!("{}", report); grin_servers::common::types::Error::WalletComm(report) })?; let parsed: CbData = res.clone().into_result().map_err(|e| { let report = format!("Error parsing result: {}", e); error!("{}", report); grin_servers::common::types::Error::WalletComm(report) })?; Ok(parsed) } pub fn build_onion( &self, commitment: &Commitment, server_pubkeys: &Vec, ) -> Result<(Onion, ComSignature), grin_wallet_libwallet::Error> { let keychain = self .wallet .lock() .lc_provider()? .wallet_inst()? .keychain(self.keychain_mask().as_ref())?; let (_, outputs) = self.owner_api .retrieve_outputs(self.keychain_mask().as_ref(), false, false, None)?; let mut output = None; for o in &outputs { if o.commit == *commitment { output = Some(o.output.clone()); break; } } if output.is_none() { return Err(grin_wallet_libwallet::Error::GenericError(String::from( "output not found", ))); } let amount = output.clone().unwrap().value; let input_blind = keychain.derive_key( amount, &output.clone().unwrap().key_id, SwitchCommitmentType::Regular, )?; let fee = tx_fee(1, 1, 1); let new_amount = amount - (fee * server_pubkeys.len() as u64); let new_output = self.owner_api.build_output( self.keychain_mask().as_ref(), OutputFeatures::Plain, new_amount, )?; let secp = Secp256k1::new(); let mut blind_sum = new_output .blind .split(&BlindingFactor::from_secret_key(input_blind.clone()), &secp)?; let hops = server_pubkeys .iter() .enumerate() .map(|(i, &p)| { if (i + 1) == server_pubkeys.len() { Hop { server_pubkey: p.clone(), excess: blind_sum.secret_key(&secp).unwrap(), fee: FeeFields::from(fee as u32), rangeproof: Some(new_output.output.proof.clone()), } } else { let hop_excess = BlindingFactor::rand(&secp); blind_sum = blind_sum.split(&hop_excess, &secp).unwrap(); Hop { server_pubkey: p.clone(), excess: hop_excess.secret_key(&secp).unwrap(), fee: FeeFields::from(fee as u32), rangeproof: None, } } }) .collect(); let onion = grin_onion::create_onion(&commitment, &hops).unwrap(); let comsig = ComSignature::sign(amount, &input_blind, &onion.serialize().unwrap()).unwrap(); Ok((onion, comsig)) } pub fn owner_api( &self, ) -> Arc< Owner, HTTPNodeClient, ExtKeychain>, > { self.owner_api.clone() } pub fn foreign_api(&self) -> String { format!("http://127.0.0.1:{}/v2/foreign", self.api_listen_port) } pub fn owner_address(&self) -> SocketAddr { SocketAddr::new( IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), self.api_listen_port, ) } pub fn keychain_mask(&self) -> Option { self.http_client.as_ref().get_token().keychain_mask.clone() } pub fn get_client(&self) -> Arc { self.http_client.clone() } } #[allow(dead_code)] pub struct GrinWalletManager { // base directory for the server instance working_dir: String, wallets: Vec>>, } impl GrinWalletManager { pub fn new(test_dir: &str) -> GrinWalletManager { GrinWalletManager { working_dir: String::from(test_dir), wallets: vec![], } } pub async fn async_new_wallet( &mut self, node_api_addr: &SocketAddr, ) -> Arc> { let wallet_dir = format!("{}/wallets/{}", self.working_dir, self.wallets.len()); let wallet = Arc::new(Mutex::new( IntegrationGrinWallet::async_new_wallet( wallet_dir, 21000 + self.wallets.len() as u16, format!("http://{}", node_api_addr), ) .await, )); self.wallets.push(wallet.clone()); wallet } }