You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

3831 lines
182 KiB

/**
* Copyright (c) woodser
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*
* Parts of this file are originally copyright (c) 2014-2019, The Monero Project
*
* Redistribution and use in source and binary forms, with or without modification, are
* permitted provided that the following conditions are met:
*
* All rights reserved.
*
* 1. Redistributions of source code must retain the above copyright notice, this std::list of
* conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice, this std::list
* of conditions and the following disclaimer in the documentation and/or other
* materials provided with the distribution.
*
* 3. Neither the name of the copyright holder nor the names of its contributors may be
* used to endorse or promote products derived from this software without specific
* prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
* THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* Parts of this file are originally copyright (c) 2012-2013 The Cryptonote developers
*/
#include "monero_wallet_full.h"
#ifdef WIN32_LEAN_AND_MEAN
#include <boost/locale.hpp>
#include <boost/filesystem.hpp>
#include <thread>
#endif
#include "utils/monero_utils.h"
#include <chrono>
#include <iostream>
#include "mnemonics/electrum-words.h"
#include "mnemonics/english.h"
#include "wallet/wallet_rpc_server_commands_defs.h"
#include "serialization/binary_utils.h"
#include "serialization/string.h"
#include "common/threadpool.h"
using namespace tools;
/**
* Implements a monero_wallet.h by wrapping wallet2.h.
*/
namespace monero {
// ------------------------- INITIALIZE CONSTANTS ---------------------------
static const int DEFAULT_CONNECTION_TIMEOUT_MILLIS = 1000 * 30; // default connection timeout 30 sec
static const bool STRICT_ = false; // relies exclusively on blockchain data if true, includes local wallet data if false TODO: good use case to expose externally? (note: cannot use `STRICT` due to namespace collision on Windows)
// ----------------------- INTERNAL PRIVATE HELPERS -----------------------
struct key_image_list
{
std::list<std::string> key_images;
BEGIN_KV_SERIALIZE_MAP()
KV_SERIALIZE(key_images)
END_KV_SERIALIZE_MAP()
};
/**
* Remove query criteria which require looking up other transfers/outputs to
* fulfill query.
*
* @param query the query to decontextualize
* @return a reference to the query for convenience
*/
std::shared_ptr<monero_tx_query> decontextualize(std::shared_ptr<monero_tx_query> query) {
query->m_is_incoming = boost::none;
query->m_is_outgoing = boost::none;
query->m_transfer_query = boost::none;
query->m_input_query = boost::none;
query->m_output_query = boost::none;
return query;
}
bool is_contextual(const monero_transfer_query& query) {
if (query.m_tx_query == boost::none) return false;
if (query.m_tx_query.get()->m_is_incoming != boost::none) return true; // requires context of all transfers
if (query.m_tx_query.get()->m_is_outgoing != boost::none) return true;
if (query.m_tx_query.get()->m_input_query != boost::none) return true; // requires context of inputs
if (query.m_tx_query.get()->m_output_query != boost::none) return true; // requires context of outputs
return false;
}
bool is_contextual(const monero_output_query& query) {
if (query.m_tx_query == boost::none) return false;
if (query.m_tx_query.get()->m_is_incoming != boost::none) return true; // requires context of all transfers
if (query.m_tx_query.get()->m_is_outgoing != boost::none) return true;
if (query.m_tx_query.get()->m_transfer_query != boost::none) return true; // requires context of transfers
return false;
}
bool bool_equals(bool val, const boost::optional<bool>& opt_val) {
return opt_val == boost::none ? false : val == *opt_val;
}
// compute m_num_confirmations TODO monero-project: this logic is based on wallet_rpc_server.cpp `set_confirmations` but it should be encapsulated in wallet2
void set_num_confirmations(std::shared_ptr<monero_tx_wallet>& tx, uint64_t blockchain_height) {
std::shared_ptr<monero_block>& block = tx->m_block.get();
if (block->m_height.get() >= blockchain_height || (block->m_height.get() == 0 && !tx->m_in_tx_pool.get())) tx->m_num_confirmations = 0;
else tx->m_num_confirmations = blockchain_height - block->m_height.get();
}
// compute m_num_suggested_confirmations TODO monero-project: this logic is based on wallet_rpc_server.cpp `set_confirmations` but it should be encapsulated in wallet2
void set_num_suggested_confirmations(std::shared_ptr<monero_incoming_transfer>& incoming_transfer, uint64_t blockchain_height, uint64_t block_reward, uint64_t unlock_time) {
if (block_reward == 0) incoming_transfer->m_num_suggested_confirmations = 0;
else incoming_transfer->m_num_suggested_confirmations = (incoming_transfer->m_amount.get() + block_reward - 1) / block_reward;
if (unlock_time < CRYPTONOTE_MAX_BLOCK_NUMBER) {
if (unlock_time > blockchain_height) incoming_transfer->m_num_suggested_confirmations = std::max(incoming_transfer->m_num_suggested_confirmations.get(), unlock_time - blockchain_height);
} else {
const uint64_t now = time(NULL);
if (unlock_time > now) incoming_transfer->m_num_suggested_confirmations = std::max(incoming_transfer->m_num_suggested_confirmations.get(), (unlock_time - now + DIFFICULTY_TARGET_V2 - 1) / DIFFICULTY_TARGET_V2);
}
}
std::shared_ptr<monero_tx_wallet> build_tx_with_incoming_transfer(tools::wallet2& m_w2, uint64_t height, const crypto::hash &payment_id, const tools::wallet2::payment_details &pd) {
// construct block
std::shared_ptr<monero_block> block = std::make_shared<monero_block>();
block->m_height = pd.m_block_height;
block->m_timestamp = pd.m_timestamp;
// construct tx
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
tx->m_block = block;
block->m_txs.push_back(tx);
tx->m_hash = epee::string_tools::pod_to_hex(pd.m_tx_hash);
tx->m_is_incoming = true;
tx->m_payment_id = epee::string_tools::pod_to_hex(payment_id);
if (tx->m_payment_id->substr(16).find_first_not_of('0') == std::string::npos) tx->m_payment_id = tx->m_payment_id->substr(0, 16); // TODO monero-project: this should be part of core wallet
if (tx->m_payment_id == monero_tx::DEFAULT_PAYMENT_ID) tx->m_payment_id = boost::none; // clear default payment id
tx->m_unlock_height = pd.m_unlock_time;
tx->m_is_locked = !m_w2.is_transfer_unlocked(pd.m_unlock_time, pd.m_block_height);
tx->m_fee = pd.m_fee;
tx->m_note = m_w2.get_tx_note(pd.m_tx_hash);
if (tx->m_note->empty()) tx->m_note = boost::none; // clear empty note
tx->m_is_miner_tx = pd.m_coinbase ? true : false;
tx->m_is_confirmed = true;
tx->m_is_failed = false;
tx->m_is_relayed = true;
tx->m_in_tx_pool = false;
tx->m_relay = true;
tx->m_is_double_spend_seen = false;
set_num_confirmations(tx, height);
// construct transfer
std::shared_ptr<monero_incoming_transfer> incoming_transfer = std::make_shared<monero_incoming_transfer>();
incoming_transfer->m_tx = tx;
tx->m_incoming_transfers.push_back(incoming_transfer);
incoming_transfer->m_amount = pd.m_amount;
incoming_transfer->m_account_index = pd.m_subaddr_index.major;
incoming_transfer->m_subaddress_index = pd.m_subaddr_index.minor;
incoming_transfer->m_address = m_w2.get_subaddress_as_str(pd.m_subaddr_index);
set_num_suggested_confirmations(incoming_transfer, height, m_w2.get_last_block_reward(), pd.m_unlock_time);
// return pointer to new tx
return tx;
}
std::shared_ptr<monero_tx_wallet> build_tx_with_outgoing_transfer(tools::wallet2& m_w2, uint64_t height, const crypto::hash &txid, const tools::wallet2::confirmed_transfer_details &pd) {
// construct block
std::shared_ptr<monero_block> block = std::make_shared<monero_block>();
block->m_height = pd.m_block_height;
block->m_timestamp = pd.m_timestamp;
// construct tx
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
tx->m_block = block;
block->m_txs.push_back(tx);
tx->m_hash = epee::string_tools::pod_to_hex(txid);
tx->m_is_outgoing = true;
tx->m_payment_id = epee::string_tools::pod_to_hex(pd.m_payment_id);
if (tx->m_payment_id->substr(16).find_first_not_of('0') == std::string::npos) tx->m_payment_id = tx->m_payment_id->substr(0, 16); // TODO monero-project: this should be part of core wallet
if (tx->m_payment_id == monero_tx::DEFAULT_PAYMENT_ID) tx->m_payment_id = boost::none; // clear default payment id
tx->m_unlock_height = pd.m_unlock_time;
tx->m_is_locked = !m_w2.is_transfer_unlocked(pd.m_unlock_time, pd.m_block_height);
tx->m_fee = pd.m_amount_in - pd.m_amount_out;
tx->m_note = m_w2.get_tx_note(txid);
if (tx->m_note->empty()) tx->m_note = boost::none; // clear empty note
tx->m_is_miner_tx = false;
tx->m_is_confirmed = true;
tx->m_is_failed = false;
tx->m_is_relayed = true;
tx->m_in_tx_pool = false;
tx->m_relay = true;
tx->m_is_double_spend_seen = false;
set_num_confirmations(tx, height);
// construct transfer
std::shared_ptr<monero_outgoing_transfer> outgoing_transfer = std::make_shared<monero_outgoing_transfer>();
outgoing_transfer->m_tx = tx;
tx->m_outgoing_transfer = outgoing_transfer;
uint64_t change = pd.m_change == (uint64_t)-1 ? 0 : pd.m_change; // change may not be known
outgoing_transfer->m_amount = pd.m_amount_in - change - *tx->m_fee;
outgoing_transfer->m_account_index = pd.m_subaddr_account;
std::vector<uint32_t> subaddress_indices;
std::vector<std::string> addresses;
for (uint32_t i: pd.m_subaddr_indices) {
subaddress_indices.push_back(i);
addresses.push_back(m_w2.get_subaddress_as_str({pd.m_subaddr_account, i}));
}
outgoing_transfer->m_subaddress_indices = subaddress_indices;
outgoing_transfer->m_addresses = addresses;
// initialize destinations
for (const auto &d: pd.m_dests) {
std::shared_ptr<monero_destination> destination = std::make_shared<monero_destination>();
destination->m_amount = d.amount;
destination->m_address = d.address(m_w2.nettype(), pd.m_payment_id);
outgoing_transfer->m_destinations.push_back(destination);
}
// replace transfer amount with destination sum
// TODO monero-project: confirmed tx from/to same account has amount 0 but cached transfer destinations
if (*outgoing_transfer->m_amount == 0 && !outgoing_transfer->m_destinations.empty()) {
uint64_t amount = 0;
for (const std::shared_ptr<monero_destination>& destination : outgoing_transfer->m_destinations) amount += *destination->m_amount;
outgoing_transfer->m_amount = amount;
}
// return pointer to new tx
return tx;
}
std::shared_ptr<monero_tx_wallet> build_tx_with_incoming_transfer_unconfirmed(const tools::wallet2& m_w2, uint64_t height, const crypto::hash &payment_id, const tools::wallet2::pool_payment_details &ppd) {
// construct tx
const tools::wallet2::payment_details &pd = ppd.m_pd;
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
tx->m_hash = epee::string_tools::pod_to_hex(pd.m_tx_hash);
tx->m_is_incoming = true;
tx->m_payment_id = epee::string_tools::pod_to_hex(payment_id);
if (tx->m_payment_id->substr(16).find_first_not_of('0') == std::string::npos) tx->m_payment_id = tx->m_payment_id->substr(0, 16); // TODO monero-project: this should be part of core wallet
if (tx->m_payment_id == monero_tx::DEFAULT_PAYMENT_ID) tx->m_payment_id = boost::none; // clear default payment id
tx->m_unlock_height = pd.m_unlock_time;
tx->m_is_locked = true;
tx->m_fee = pd.m_fee;
tx->m_note = m_w2.get_tx_note(pd.m_tx_hash);
if (tx->m_note->empty()) tx->m_note = boost::none; // clear empty note
tx->m_is_miner_tx = false;
tx->m_is_confirmed = false;
tx->m_is_failed = false;
tx->m_is_relayed = true;
tx->m_in_tx_pool = true;
tx->m_relay = true;
tx->m_is_double_spend_seen = ppd.m_double_spend_seen;
tx->m_num_confirmations = 0;
// construct transfer
std::shared_ptr<monero_incoming_transfer> incoming_transfer = std::make_shared<monero_incoming_transfer>();
incoming_transfer->m_tx = tx;
tx->m_incoming_transfers.push_back(incoming_transfer);
incoming_transfer->m_amount = pd.m_amount;
incoming_transfer->m_account_index = pd.m_subaddr_index.major;
incoming_transfer->m_subaddress_index = pd.m_subaddr_index.minor;
incoming_transfer->m_address = m_w2.get_subaddress_as_str(pd.m_subaddr_index);
set_num_suggested_confirmations(incoming_transfer, height, m_w2.get_last_block_reward(), pd.m_unlock_time);
// return pointer to new tx
return tx;
}
std::shared_ptr<monero_tx_wallet> build_tx_with_outgoing_transfer_unconfirmed(const tools::wallet2& m_w2, const crypto::hash &txid, const tools::wallet2::unconfirmed_transfer_details &pd) {
// construct tx
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
tx->m_is_failed = pd.m_state == tools::wallet2::unconfirmed_transfer_details::failed;
tx->m_hash = epee::string_tools::pod_to_hex(txid);
tx->m_is_outgoing = true;
tx->m_payment_id = epee::string_tools::pod_to_hex(pd.m_payment_id);
if (tx->m_payment_id->substr(16).find_first_not_of('0') == std::string::npos) tx->m_payment_id = tx->m_payment_id->substr(0, 16); // TODO monero-project: this should be part of core wallet
if (tx->m_payment_id == monero_tx::DEFAULT_PAYMENT_ID) tx->m_payment_id = boost::none; // clear default payment id
tx->m_unlock_height = pd.m_tx.unlock_time;
tx->m_is_locked = true;
tx->m_fee = pd.m_amount_in - pd.m_amount_out;
tx->m_note = m_w2.get_tx_note(txid);
if (tx->m_note->empty()) tx->m_note = boost::none; // clear empty note
tx->m_is_miner_tx = false;
tx->m_is_confirmed = false;
tx->m_is_relayed = !tx->m_is_failed.get();
tx->m_in_tx_pool = !tx->m_is_failed.get();
tx->m_relay = true;
if (!tx->m_is_failed.get() && tx->m_is_relayed.get()) tx->m_is_double_spend_seen = false; // TODO: test and handle if true
tx->m_num_confirmations = 0;
// construct transfer
std::shared_ptr<monero_outgoing_transfer> outgoing_transfer = std::make_shared<monero_outgoing_transfer>();
outgoing_transfer->m_tx = tx;
tx->m_outgoing_transfer = outgoing_transfer;
outgoing_transfer->m_amount = pd.m_amount_in - pd.m_change - tx->m_fee.get();
outgoing_transfer->m_account_index = pd.m_subaddr_account;
std::vector<uint32_t> subaddress_indices;
std::vector<std::string> addresses;
for (uint32_t i: pd.m_subaddr_indices) {
subaddress_indices.push_back(i);
addresses.push_back(m_w2.get_subaddress_as_str({pd.m_subaddr_account, i}));
}
outgoing_transfer->m_subaddress_indices = subaddress_indices;
outgoing_transfer->m_addresses = addresses;
// initialize destinations
for (const auto &d: pd.m_dests) {
std::shared_ptr<monero_destination> destination = std::make_shared<monero_destination>();
destination->m_amount = d.amount;
destination->m_address = d.address(m_w2.nettype(), pd.m_payment_id);
outgoing_transfer->m_destinations.push_back(destination);
}
// replace transfer amount with destination sum
// TODO monero-project: confirmed tx from/to same account has amount 0 but cached transfer destinations
if (*outgoing_transfer->m_amount == 0 && !outgoing_transfer->m_destinations.empty()) {
uint64_t amount = 0;
for (const std::shared_ptr<monero_destination>& destination : outgoing_transfer->m_destinations) amount += *destination->m_amount;
outgoing_transfer->m_amount = amount;
}
// return pointer to new tx
return tx;
}
std::shared_ptr<monero_tx_wallet> build_tx_with_vout(tools::wallet2& m_w2, const tools::wallet2::transfer_details& td) {
// construct block
std::shared_ptr<monero_block> block = std::make_shared<monero_block>();
block->m_height = td.m_block_height;
// construct tx
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
tx->m_block = block;
block->m_txs.push_back(tx);
tx->m_hash = epee::string_tools::pod_to_hex(td.m_txid);
tx->m_is_confirmed = true;
tx->m_is_failed = false;
tx->m_is_relayed = true;
tx->m_in_tx_pool = false;
tx->m_relay = true;
tx->m_is_double_spend_seen = false;
tx->m_is_locked = !m_w2.is_transfer_unlocked(td);
// construct output
std::shared_ptr<monero_output_wallet> output = std::make_shared<monero_output_wallet>();
output->m_tx = tx;
tx->m_outputs.push_back(output);
output->m_amount = td.amount();
output->m_index = td.m_global_output_index;
output->m_account_index = td.m_subaddr_index.major;
output->m_subaddress_index = td.m_subaddr_index.minor;
output->m_is_spent = td.m_spent;
output->m_is_frozen = td.m_frozen;
output->m_stealth_public_key = epee::string_tools::pod_to_hex(td.get_public_key());
if (td.m_key_image_known) {
output->m_key_image = std::make_shared<monero_key_image>();
output->m_key_image.get()->m_hex = epee::string_tools::pod_to_hex(td.m_key_image);
}
// return pointer to new tx
return tx;
}
/**
* Merges a transaction into a unique set of transactions.
*
* @param tx is the transaction to merge into the existing txs
* @param tx_map maps tx hashes to txs
* @param block_map maps block heights to blocks
*/
void merge_tx(const std::shared_ptr<monero_tx_wallet>& tx, std::map<std::string, std::shared_ptr<monero_tx_wallet>>& tx_map, std::map<uint64_t, std::shared_ptr<monero_block>>& block_map) {
if (tx->m_hash == boost::none) throw std::runtime_error("Tx hash is not initialized");
// merge tx
std::map<std::string, std::shared_ptr<monero_tx_wallet>>::const_iterator tx_iter = tx_map.find(*tx->m_hash);
if (tx_iter == tx_map.end()) {
tx_map[*tx->m_hash] = tx; // cache new tx
} else {
std::shared_ptr<monero_tx_wallet>& a_tx = tx_map[*tx->m_hash];
a_tx->merge(a_tx, tx); // merge with existing tx
}
// merge tx's block if confirmed
if (tx->get_height() != boost::none) {
std::map<uint64_t, std::shared_ptr<monero_block>>::const_iterator block_iter = block_map.find(tx->get_height().get());
if (block_iter == block_map.end()) {
block_map[tx->get_height().get()] = tx->m_block.get(); // cache new block
} else {
std::shared_ptr<monero_block>& a_block = block_map[tx->get_height().get()];
a_block->merge(a_block, tx->m_block.get()); // merge with existing block
}
}
}
/**
* Returns true iff tx1's height is known to be less than tx2's height for sorting.
*/
bool tx_height_less_than(const std::shared_ptr<monero_tx>& tx1, const std::shared_ptr<monero_tx>& tx2) {
if (tx1->m_block != boost::none && tx2->m_block != boost::none) return tx1->get_height() < tx2->get_height();
else if (tx1->m_block == boost::none) return false;
else return true;
}
/**
* Returns true iff transfer1 is ordered before transfer2 by ascending account and subaddress indices.
*/
bool incoming_transfer_before(const std::shared_ptr<monero_incoming_transfer>& transfer1, const std::shared_ptr<monero_incoming_transfer>& transfer2) {
// compare by height
if (tx_height_less_than(transfer1->m_tx, transfer2->m_tx)) return true;
// compare by account and subaddress index
if (transfer1->m_account_index.get() < transfer2->m_account_index.get()) return true;
else if (transfer1->m_account_index.get() == transfer2->m_account_index.get()) return transfer1->m_subaddress_index.get() < transfer2->m_subaddress_index.get();
else return false;
}
/**
* Returns true iff wallet vout1 is ordered before vout2 by ascending account and subaddress indices then index.
*/
bool vout_before(const std::shared_ptr<monero_output>& o1, const std::shared_ptr<monero_output>& o2) {
if (o1 == o2) return false; // ignore equal references
std::shared_ptr<monero_output_wallet> ow1 = std::static_pointer_cast<monero_output_wallet>(o1);
std::shared_ptr<monero_output_wallet> ow2 = std::static_pointer_cast<monero_output_wallet>(o2);
// compare by height
if (tx_height_less_than(ow1->m_tx, ow2->m_tx)) return true;
// compare by account index, subaddress index, output index, then key image hex
if (ow1->m_account_index.get() < ow2->m_account_index.get()) return true;
if (ow1->m_account_index.get() == ow2->m_account_index.get()) {
if (ow1->m_subaddress_index.get() < ow2->m_subaddress_index.get()) return true;
if (ow1->m_subaddress_index.get() == ow2->m_subaddress_index.get()) {
if (ow1->m_index.get() < ow2->m_index.get()) return true;
if (ow1->m_index.get() == ow2->m_index.get()) throw std::runtime_error("Should never sort outputs with duplicate indices");
}
}
return false;
}
std::string get_default_ringdb_path(cryptonote::network_type nettype)
{
boost::filesystem::path dir = tools::get_default_data_dir();
// remove .bitmonero, replace with .shared-ringdb
dir = dir.remove_filename();
dir /= ".shared-ringdb";
if (nettype == cryptonote::TESTNET)
dir /= "testnet";
else if (nettype == cryptonote::STAGENET)
dir /= "stagenet";
return dir.string();
}
/**
* ---------------- DUPLICATED WALLET RPC TRANSFER CODE ---------------------
*
* These functions are duplicated from private functions in wallet rpc
* on_transfer/on_transfer_split, with minor modifications to not be class members.
*
* This code is used to generate and send transactions with equivalent functionality as
* wallet rpc.
*
* Duplicated code is not ideal. Solutions considered:
*
* (1) Duplicate wallet rpc code as done here.
* (2) Modify monero-wallet-rpc on_transfer() / on_transfer_split() to be public.
* (3) Modify monero-wallet-rpc to make this class a friend.
* (4) Move all logic in monero-wallet-rpc to wallet2 so all users can access.
*
* Options 2-4 require modification of monero-project C++. Of those, (4) is probably ideal.
* TODO: open patch on monero-project which moves common wallet rpc logic (e.g. on_transfer, on_transfer_split) to m_w2.
*
* Until then, option (1) is used because it allows monero-project binaries to be used without modification, it's easy, and
* anything other than (4) is temporary.
*/
//------------------------------------------------------------------------------------------------------------------------------
bool validate_transfer(wallet2* m_w2, const std::list<wallet_rpc::transfer_destination>& destinations, const std::string& payment_id, std::vector<cryptonote::tx_destination_entry>& dsts, std::vector<uint8_t>& extra, bool at_least_one_destination, epee::json_rpc::error& er)
{
crypto::hash8 integrated_payment_id = crypto::null_hash8;
std::string extra_nonce;
for (auto it = destinations.begin(); it != destinations.end(); it++)
{
cryptonote::address_parse_info info;
cryptonote::tx_destination_entry de;
er.message = "";
if(!get_account_address_from_str_or_url(info, m_w2->nettype(), it->address,
[&er](const std::string &url, const std::vector<std::string> &addresses, bool dnssec_valid)->std::string {
if (!dnssec_valid)
{
er.message = std::string("Invalid DNSSEC for ") + url;
return {};
}
if (addresses.empty())
{
er.message = std::string("No Monero address found at ") + url;
return {};
}
return addresses[0];
}))
{
er.code = WALLET_RPC_ERROR_CODE_WRONG_ADDRESS;
if (er.message.empty())
er.message = std::string("Invalid destination address");
return false;
}
de.original = it->address;
de.addr = info.address;
de.is_subaddress = info.is_subaddress;
de.amount = it->amount;
de.is_integrated = info.has_payment_id;
dsts.push_back(de);
if (info.has_payment_id)
{
if (!payment_id.empty() || integrated_payment_id != crypto::null_hash8)
{
er.code = WALLET_RPC_ERROR_CODE_WRONG_PAYMENT_ID;
er.message = "A single payment id is allowed per transaction";
return false;
}
integrated_payment_id = info.payment_id;
cryptonote::set_encrypted_payment_id_to_tx_extra_nonce(extra_nonce, integrated_payment_id);
/* Append Payment ID data into extra */
if (!cryptonote::add_extra_nonce_to_tx_extra(extra, extra_nonce)) {
er.code = WALLET_RPC_ERROR_CODE_WRONG_PAYMENT_ID;
er.message = "Something went wrong with integrated payment_id.";
return false;
}
}
}
if (at_least_one_destination && dsts.empty())
{
er.code = WALLET_RPC_ERROR_CODE_ZERO_DESTINATION;
er.message = "No destinations for this transfer";
return false;
}
if (!payment_id.empty())
{
er.code = WALLET_RPC_ERROR_CODE_WRONG_PAYMENT_ID;
er.message = "Standalone payment IDs are obsolete. Use subaddresses or integrated addresses instead";
return false;
}
return true;
}
//------------------------------------------------------------------------------------------------------------------------------
static std::string ptx_to_string(const tools::wallet2::pending_tx &ptx)
{
std::ostringstream oss;
boost::archive::portable_binary_oarchive ar(oss);
try
{
ar << ptx;
}
catch (...)
{
return "";
}
return epee::string_tools::buff_to_hex_nodelimer(oss.str());
}
//------------------------------------------------------------------------------------------------------------------------------
template<typename T> static bool is_error_value(const T &val) { return false; }
static bool is_error_value(const std::string &s) { return s.empty(); }
//------------------------------------------------------------------------------------------------------------------------------
template<typename T, typename V>
static bool fill(T &where, V s)
{
if (is_error_value(s)) return false;
where = std::move(s);
return true;
}
//------------------------------------------------------------------------------------------------------------------------------
template<typename T, typename V>
static bool fill(std::list<T> &where, V s)
{
if (is_error_value(s)) return false;
where.emplace_back(std::move(s));
return true;
}
//------------------------------------------------------------------------------------------------------------------------------
static uint64_t total_amount(const tools::wallet2::pending_tx &ptx)
{
uint64_t amount = 0;
for (const auto &dest: ptx.dests) amount += dest.amount;
return amount;
}
//------------------------------------------------------------------------------------------------------------------------------
template<typename Ts, typename Tu, typename Tk>
bool fill_response(wallet2* m_w2, std::vector<tools::wallet2::pending_tx> &ptx_vector,
bool get_tx_key, Ts& tx_key, Tu &amount, Tu &fee, Tu &weight, std::string &multisig_txset, std::string &unsigned_txset, bool do_not_relay,
Ts &tx_hash, bool get_tx_hex, Ts &tx_blob, bool get_tx_metadata, Ts &tx_metadata, Tk &spent_key_images, epee::json_rpc::error &er)
{
for (const auto & ptx : ptx_vector)
{
if (get_tx_key)
{
epee::wipeable_string s = epee::to_hex::wipeable_string(ptx.tx_key);
for (const crypto::secret_key& additional_tx_key : ptx.additional_tx_keys)
s += epee::to_hex::wipeable_string(additional_tx_key);
fill(tx_key, std::string(s.data(), s.size()));
}
// Compute amount leaving wallet in tx. By convention dests does not include change outputs
fill(amount, total_amount(ptx));
fill(fee, ptx.fee);
fill(weight, cryptonote::get_transaction_weight(ptx.tx));
// add spent key images
key_image_list key_image_list;
bool all_are_txin_to_key = std::all_of(ptx.tx.vin.begin(), ptx.tx.vin.end(), [&](const cryptonote::txin_v& s_e) -> bool
{
CHECKED_GET_SPECIFIC_VARIANT(s_e, const cryptonote::txin_to_key, in, false);
key_image_list.key_images.push_back(epee::string_tools::pod_to_hex(in.k_image));
return true;
});
THROW_WALLET_EXCEPTION_IF(!all_are_txin_to_key, error::unexpected_txin_type, ptx.tx);
fill(spent_key_images, key_image_list);
}
if (m_w2->multisig())
{
multisig_txset = epee::string_tools::buff_to_hex_nodelimer(m_w2->save_multisig_tx(ptx_vector));
if (multisig_txset.empty())
{
er.code = WALLET_RPC_ERROR_CODE_UNKNOWN_ERROR;
er.message = "Failed to save multisig tx set after creation";
return false;
}
}
else
{
if (m_w2->watch_only()){
unsigned_txset = epee::string_tools::buff_to_hex_nodelimer(m_w2->dump_tx_to_str(ptx_vector));
if (unsigned_txset.empty())
{
er.code = WALLET_RPC_ERROR_CODE_UNKNOWN_ERROR;
er.message = "Failed to save unsigned tx set after creation";
return false;
}
}
else if (!do_not_relay)
m_w2->commit_tx(ptx_vector);
// populate response with tx hashes
for (auto & ptx : ptx_vector)
{
bool r = fill(tx_hash, epee::string_tools::pod_to_hex(cryptonote::get_transaction_hash(ptx.tx)));
r = r && (!get_tx_hex || fill(tx_blob, epee::string_tools::buff_to_hex_nodelimer(tx_to_blob(ptx.tx))));
r = r && (!get_tx_metadata || fill(tx_metadata, ptx_to_string(ptx)));
if (!r)
{
er.code = WALLET_RPC_ERROR_CODE_UNKNOWN_ERROR;
er.message = "Failed to save tx info";
return false;
}
}
}
return true;
}
// ----------------------------- WALLET LISTENER ----------------------------
/**
* Listens to wallet2 notifications in order to notify external wallet listeners.
*/
struct wallet2_listener : public tools::i_wallet2_callback {
public:
/**
* Constructs the listener.
*
* @param wallet provides context to notify external listeners
* @param wallet2 provides source notifications which this listener propagates to external listeners
*/
wallet2_listener(monero_wallet_full& wallet, tools::wallet2& wallet2) : m_wallet(wallet), m_w2(wallet2) {
this->m_sync_start_height = boost::none;
this->m_sync_end_height = boost::none;
m_prev_balance = wallet.get_balance();
m_prev_unlocked_balance = wallet.get_unlocked_balance();
m_notification_pool = std::unique_ptr<tools::threadpool>(tools::threadpool::getNewForUnitTests(1)); // TODO (monero-project): utility can be for general use
}
~wallet2_listener() {
MTRACE("~wallet2_listener()");
m_w2.callback(nullptr);
m_notification_pool->recycle();
}
void update_listening() {
boost::lock_guard<boost::mutex> guarg(m_listener_mutex);
// if starting to listen, cache locked txs for later comparison
if (!m_wallet.get_listeners().empty() && m_w2.callback() == nullptr) check_for_changed_unlocked_txs();
// update callback
m_w2.callback(m_wallet.get_listeners().empty() ? nullptr : this);
}
void on_sync_start(uint64_t start_height) {
tools::threadpool::waiter waiter(*m_notification_pool);
m_notification_pool->submit(&waiter, [this, start_height]() {
if (m_sync_start_height != boost::none || m_sync_end_height != boost::none) throw std::runtime_error("Sync start or end height should not already be allocated, is previous sync in progress?");
m_sync_start_height = start_height;
m_sync_end_height = m_wallet.get_daemon_height();
});
}
void on_sync_end() {
tools::threadpool::waiter waiter(*m_notification_pool);
m_notification_pool->submit(&waiter, [this]() {
check_for_changed_balances();
if (m_prev_locked_tx_hashes.size() > 0) check_for_changed_unlocked_txs();
m_sync_start_height = boost::none;
m_sync_end_height = boost::none;
});
m_notification_pool->recycle();
}
void on_new_block(uint64_t height, const cryptonote::block& cn_block) override {
if (m_wallet.get_listeners().empty()) return;
// ignore notifications before sync start height, irrelevant to clients
if (m_sync_start_height == boost::none || height < *m_sync_start_height) return;
// queue notification processing off main thread
tools::threadpool::waiter waiter(*m_notification_pool);
m_notification_pool->submit(&waiter, [this, height]() {
// notify listeners of new block
for (monero_wallet_listener* listener : m_wallet.get_listeners()) {
listener->on_new_block(height);
}
// notify listeners of sync progress
if (height >= *m_sync_end_height) m_sync_end_height = height + 1; // increase end height if necessary
double percent_done = (double) (height - *m_sync_start_height + 1) / (double) (*m_sync_end_height - *m_sync_start_height);
std::string message = std::string("Synchronizing");
for (monero_wallet_listener* listener : m_wallet.get_listeners()) {
listener->on_sync_progress(height, *m_sync_start_height, *m_sync_end_height, percent_done, message);
}
// notify if balances change
bool balances_changed = check_for_changed_balances();
// notify when txs unlock after wallet is synced
if (balances_changed && m_wallet.is_synced()) check_for_changed_unlocked_txs();
});
}
void on_unconfirmed_money_received(uint64_t height, const crypto::hash &txid, const cryptonote::transaction& cn_tx, uint64_t amount, const cryptonote::subaddress_index& subaddr_index) override {
if (m_wallet.get_listeners().empty()) return;
// queue notification processing off main thread
tools::threadpool::waiter waiter(*m_notification_pool);
m_notification_pool->submit(&waiter, [this, height, txid, cn_tx, amount, subaddr_index]() {
try {
// create library tx
std::shared_ptr<monero_tx_wallet> tx = std::static_pointer_cast<monero_tx_wallet>(monero_utils::cn_tx_to_tx(cn_tx, true));
tx->m_hash = epee::string_tools::pod_to_hex(txid);
tx->m_is_confirmed = false;
tx->m_is_locked = true;
std::shared_ptr<monero_output_wallet> output = std::make_shared<monero_output_wallet>();
tx->m_outputs.push_back(output);
output->m_tx = tx;
output->m_amount = amount;
output->m_account_index = subaddr_index.major;
output->m_subaddress_index = subaddr_index.minor;
// notify listeners of output
for (monero_wallet_listener* listener : m_wallet.get_listeners()) {
listener->on_output_received(*output);
}
// notify if balances changed
check_for_changed_balances();
// watch for unlock
m_prev_locked_tx_hashes.insert(tx->m_hash.get());
// free memory
output.reset();
tx.reset();
} catch (std::exception& e) {
std::cout << "Error processing unconfirmed output received: " << std::string(e.what()) << std::endl;
}
});
}
void on_money_received(uint64_t height, const crypto::hash &txid, const cryptonote::transaction& cn_tx, uint64_t amount, uint64_t burnt, const cryptonote::subaddress_index& subaddr_index, bool is_change, uint64_t unlock_height) override {
if (m_wallet.get_listeners().empty()) return;
// queue notification processing off main thread
tools::threadpool::waiter waiter(*m_notification_pool);
m_notification_pool->submit(&waiter, [this, height, txid, cn_tx, amount, burnt, subaddr_index, is_change, unlock_height]() {
try {
// create native library tx
std::shared_ptr<monero_block> block = std::make_shared<monero_block>();
block->m_height = height;
std::shared_ptr<monero_tx_wallet> tx = std::static_pointer_cast<monero_tx_wallet>(monero_utils::cn_tx_to_tx(cn_tx, true));
block->m_txs.push_back(tx);
tx->m_block = block;
tx->m_hash = epee::string_tools::pod_to_hex(txid);
tx->m_is_confirmed = true;
tx->m_is_locked = true;
tx->m_unlock_height = unlock_height;
std::shared_ptr<monero_output_wallet> output = std::make_shared<monero_output_wallet>();
tx->m_outputs.push_back(output);
output->m_tx = tx;
output->m_amount = amount - burnt;
output->m_account_index = subaddr_index.major;
output->m_subaddress_index = subaddr_index.minor;
// notify listeners of output
for (monero_wallet_listener* listener : m_wallet.get_listeners()) {
listener->on_output_received(*output);
}
// watch for unlock
m_prev_locked_tx_hashes.insert(tx->m_hash.get());
// free memory
monero_utils::free(block);
output.reset();
tx.reset();
} catch (std::exception& e) {
std::cout << "Error processing confirmed output received: " << std::string(e.what()) << std::endl;
}
});
}
void on_money_spent(uint64_t height, const crypto::hash &txid, const cryptonote::transaction& cn_tx_in, uint64_t amount, const cryptonote::transaction& cn_tx_out, const cryptonote::subaddress_index& subaddr_index) override {
if (m_wallet.get_listeners().empty()) return;
if (&cn_tx_in != &cn_tx_out) throw std::runtime_error("on_money_spent() in tx is different than out tx");
// queue notification processing off main thread
tools::threadpool::waiter waiter(*m_notification_pool);
m_notification_pool->submit(&waiter, [this, height, txid, cn_tx_in, amount, cn_tx_out, subaddr_index]() {
try {
// create native library tx
std::shared_ptr<monero_block> block = std::make_shared<monero_block>();
block->m_height = height;
std::shared_ptr<monero_tx_wallet> tx = std::static_pointer_cast<monero_tx_wallet>(monero_utils::cn_tx_to_tx(cn_tx_in, true));
block->m_txs.push_back(tx);
tx->m_block = block;
tx->m_hash = epee::string_tools::pod_to_hex(txid);
tx->m_is_confirmed = true;
tx->m_is_locked = true;
std::shared_ptr<monero_output_wallet> output = std::make_shared<monero_output_wallet>();
tx->m_inputs.push_back(output);
output->m_tx = tx;
output->m_amount = amount;
output->m_account_index = subaddr_index.major;
output->m_subaddress_index = subaddr_index.minor;
// notify listeners of output
for (monero_wallet_listener* listener : m_wallet.get_listeners()) {
listener->on_output_spent(*output);
}
// watch for unlock
m_prev_locked_tx_hashes.insert(tx->m_hash.get());
// free memory
monero_utils::free(block);
output.reset();
tx.reset();
} catch (std::exception& e) {
std::cout << "Error processing confirmed output spent: " << std::string(e.what()) << std::endl;
}
});
}
void on_spend_tx_hashes(const std::vector<std::string>& tx_hashes) {
if (m_wallet.get_listeners().empty()) return;
monero_tx_query tx_query;
tx_query.m_hashes = tx_hashes;
tx_query.m_include_outputs = true;
tx_query.m_is_locked = true;
std::vector<std::string> missing_tx_hashes;
on_spend_txs(m_wallet.get_txs(tx_query, missing_tx_hashes));
}
void on_spend_txs(const std::vector<std::shared_ptr<monero_tx_wallet>>& txs) {
if (m_wallet.get_listeners().empty()) return;
tools::threadpool::waiter waiter(*m_notification_pool);
m_notification_pool->submit(&waiter, [this, txs]() {
check_for_changed_balances();
for (const std::shared_ptr<monero_tx_wallet>& tx : txs) notify_outputs(tx);
});
}
private:
monero_wallet_full& m_wallet; // wallet to provide context for notifications
tools::wallet2& m_w2; // internal wallet implementation to listen to
boost::optional<uint64_t> m_sync_start_height;
boost::optional<uint64_t> m_sync_end_height;
boost::mutex m_listener_mutex;
uint64_t m_prev_balance;
uint64_t m_prev_unlocked_balance;
std::set<std::string> m_prev_locked_tx_hashes;
std::unique_ptr<tools::threadpool> m_notification_pool; // threadpool of size 1 to queue notifications for external announcement
bool check_for_changed_balances() {
uint64_t balance = m_wallet.get_balance();
uint64_t unlocked_balance = m_wallet.get_unlocked_balance();
if (balance != m_prev_balance || unlocked_balance != m_prev_unlocked_balance) {
m_prev_balance = balance;
m_prev_unlocked_balance = unlocked_balance;
for (monero_wallet_listener* listener : m_wallet.get_listeners()) {
listener->on_balances_changed(balance, unlocked_balance);
}
return true;
}
return false;
}
// TODO: this can probably be optimized using e.g. wallet2.get_num_rct_outputs() or wallet2.get_num_transfer_details(), or by retaining confirmed block height and only checking on or after unlock height, etc
void check_for_changed_unlocked_txs() {
// get confirmed and locked txs
monero_tx_query query = monero_tx_query();
query.m_is_locked = true;
query.m_is_confirmed = true;
query.m_min_height = m_wallet.get_height() - 70; // only monitor recent txs
std::vector<std::shared_ptr<monero_tx_wallet>> locked_txs = m_wallet.get_txs(query);
// collect hashes of txs no longer locked
std::vector<std::string> tx_hashes_no_longer_locked;
for (const std::string prev_locked_tx_hash : m_prev_locked_tx_hashes) {
bool found = false;
for (const std::shared_ptr<monero_tx_wallet>& locked_tx : locked_txs) {
if (locked_tx->m_hash.get() == prev_locked_tx_hash) {
found = true;
break;
}
}
if (!found) tx_hashes_no_longer_locked.push_back(prev_locked_tx_hash);
}
// fetch txs that are no longer locked
std::vector<std::shared_ptr<monero_tx_wallet>> txs_no_longer_locked;
if (!tx_hashes_no_longer_locked.empty()) {
query.m_hashes = tx_hashes_no_longer_locked;
query.m_is_locked = false;
query.m_include_outputs = true;
std::vector<std::string> missing_tx_hashes;
txs_no_longer_locked = m_wallet.get_txs(query, missing_tx_hashes);
}
// notify listeners of newly unlocked inputs and outputs
for (const std::shared_ptr<monero_tx_wallet>& unlocked_tx : txs_no_longer_locked) {
notify_outputs(unlocked_tx);
}
// re-assign currently locked tx hashes // TODO: needs mutex for thread safety?
m_prev_locked_tx_hashes.clear();
for (const std::shared_ptr<monero_tx_wallet>& locked_tx : locked_txs) {
m_prev_locked_tx_hashes.insert(locked_tx->m_hash.get());
}
}
void notify_outputs(const std::shared_ptr<monero_tx_wallet>& tx) {
// notify spent outputs
if (tx->m_outgoing_transfer != boost::none) {
// build dummy input for notification // TODO: this provides one input with outgoing amount like monero-wallet-rpc client, use real inputs instead
std::shared_ptr<monero_output_wallet> input = std::make_shared<monero_output_wallet>();
input->m_amount = tx->m_outgoing_transfer.get()->m_amount.get() + tx->m_fee.get();
input->m_account_index = tx->m_outgoing_transfer.get()->m_account_index;
if (tx->m_outgoing_transfer.get()->m_subaddress_indices.size() == 1) input->m_subaddress_index = tx->m_outgoing_transfer.get()->m_subaddress_indices[0]; // initialize if transfer sourced from single subaddress
std::shared_ptr<monero_tx_wallet> tx_notify = std::make_shared<monero_tx_wallet>();
input->m_tx = tx_notify;
tx_notify->m_inputs.push_back(input);
tx_notify->m_hash = tx->m_hash;
tx_notify->m_is_locked = tx->m_is_locked;
tx_notify->m_unlock_height = tx->m_unlock_height;
if (tx->m_block != boost::none) {
std::shared_ptr<monero_block> block_notify = std::make_shared<monero_block>();
tx_notify->m_block = block_notify;
block_notify->m_height = tx->get_height();
block_notify->m_txs.push_back(tx_notify);
}
// notify listeners
for (monero_wallet_listener* listener : m_wallet.get_listeners()) listener->on_output_spent(*input);
}
// notify received outputs
if (!tx->m_incoming_transfers.empty()) {
for (const std::shared_ptr<monero_output_wallet>& output : tx->get_outputs_wallet()) {
for (monero_wallet_listener* listener : m_wallet.get_listeners()) listener->on_output_received(*output);
}
}
}
};
// --------------------------- STATIC WALLET UTILS --------------------------
bool monero_wallet_full::wallet_exists(const std::string& path) {
MTRACE("wallet_exists(" << path << ")");
bool key_file_exists;
bool wallet_file_exists;
tools::wallet2::wallet_exists(path, key_file_exists, wallet_file_exists);
return key_file_exists;
}
monero_wallet_full* monero_wallet_full::open_wallet(const std::string& path, const std::string& password, const monero_network_type network_type) {
MTRACE("open_wallet(" << path << ", ***, " << network_type << ")");
monero_wallet_full* wallet = new monero_wallet_full();
wallet->m_w2 = std::unique_ptr<tools::wallet2>(new tools::wallet2(static_cast<cryptonote::network_type>(network_type), 1, true));
wallet->m_w2->load(path, password);
wallet->m_w2->init("");
wallet->init_common();
return wallet;
}
monero_wallet_full* monero_wallet_full::open_wallet_data(const std::string& password, const monero_network_type network_type, const std::string& keys_data, const std::string& cache_data, const monero_rpc_connection& daemon_connection, std::unique_ptr<epee::net_utils::http::http_client_factory> http_client_factory) {
MTRACE("open_wallet_data(...)");
monero_wallet_full* wallet = new monero_wallet_full();
if (http_client_factory == nullptr) wallet->m_w2 = std::unique_ptr<tools::wallet2>(new tools::wallet2(static_cast<cryptonote::network_type>(network_type), 1, true));
else wallet->m_w2 = std::unique_ptr<tools::wallet2>(new tools::wallet2(static_cast<cryptonote::network_type>(network_type), 1, true, std::move(http_client_factory)));
wallet->m_w2->load("", password, keys_data, cache_data);
wallet->m_w2->init("");
wallet->set_daemon_connection(daemon_connection);
wallet->init_common();
return wallet;
}
monero_wallet_full* monero_wallet_full::create_wallet(const monero_wallet_config& config, std::unique_ptr<epee::net_utils::http::http_client_factory> http_client_factory) {
MTRACE("create_wallet(config)");
// validate and normalize config
monero_wallet_config config_normalized = config.copy();
if (config.m_path == boost::none) config_normalized.m_path = std::string("");
if (config.m_password == boost::none) config_normalized.m_password = std::string("");
if (config.m_language == boost::none) config_normalized.m_language = std::string("");
if (config.m_mnemonic == boost::none) config_normalized.m_mnemonic = std::string("");
if (config.m_primary_address == boost::none) config_normalized.m_primary_address = std::string("");
if (config.m_private_spend_key == boost::none) config_normalized.m_private_spend_key = std::string("");
if (config.m_private_view_key == boost::none) config_normalized.m_private_view_key = std::string("");
if (config.m_seed_offset == boost::none) config_normalized.m_seed_offset = std::string("");
if (config.m_account_lookahead != boost::none && config.m_subaddress_lookahead == boost::none) throw std::runtime_error("No subaddress lookahead provided with account lookahead");
if (config.m_account_lookahead == boost::none && config.m_subaddress_lookahead != boost::none) throw std::runtime_error("No account lookahead provided with subaddress lookahead");
// create wallet
if (!config_normalized.m_mnemonic.get().empty()) {
if (config.m_restore_height == boost::none) config_normalized.m_restore_height = 0;
if (!config_normalized.m_language.get().empty()) throw std::runtime_error("Cannot specify language when creating wallet from mnemonic");
return create_wallet_from_mnemonic(config_normalized, std::move(http_client_factory));
} else if (!config_normalized.m_primary_address.get().empty() || !config_normalized.m_private_spend_key.get().empty() || !config_normalized.m_private_view_key.get().empty()) {
if (config.m_restore_height == boost::none) config_normalized.m_restore_height = 0;
if (!config_normalized.m_seed_offset.get().empty()) throw std::runtime_error("Cannot specify seed offset when creating wallet from keys");
if (config_normalized.m_language.get().empty()) config_normalized.m_language = std::string("English");
if (!monero_utils::is_valid_language(config_normalized.m_language.get())) throw std::runtime_error("Unknown language: " + config_normalized.m_language.get());
return create_wallet_from_keys(config_normalized, std::move(http_client_factory));
} else {
if (!config_normalized.m_seed_offset.get().empty()) throw std::runtime_error("Cannot specify seed offset when creating random wallet");
if (config_normalized.m_restore_height != boost::none) throw std::runtime_error("Cannot specify restore height when creating random wallet");
if (config_normalized.m_language.get().empty()) config_normalized.m_language = std::string("English");
if (!monero_utils::is_valid_language(config_normalized.m_language.get())) throw std::runtime_error("Unknown language: " + config_normalized.m_language.get());
return monero_wallet_full::create_wallet_random(config_normalized, std::move(http_client_factory));
}
}
monero_wallet_full* monero_wallet_full::create_wallet_from_mnemonic(const monero_wallet_config& config, std::unique_ptr<epee::net_utils::http::http_client_factory> http_client_factory) {
MTRACE("create_wallet_from_mnemonic(...)");
monero_wallet_full* wallet = new monero_wallet_full();
// validate mnemonic and get recovery key and language
crypto::secret_key recovery_key;
std::string language;
bool is_valid = crypto::ElectrumWords::words_to_bytes(config.m_mnemonic.get(), recovery_key, language);
if (!is_valid) throw std::runtime_error("Invalid mnemonic");
if (language == crypto::ElectrumWords::old_language_name) language = Language::English().get_language_name();
// apply offset if given
if (!config.m_seed_offset.get().empty()) recovery_key = cryptonote::decrypt_key(recovery_key, config.m_seed_offset.get());
// initialize wallet
if (http_client_factory == nullptr) wallet->m_w2 = std::unique_ptr<tools::wallet2>(new tools::wallet2(static_cast<cryptonote::network_type>(config.m_network_type.get()), 1, true));
else wallet->m_w2 = std::unique_ptr<tools::wallet2>(new tools::wallet2(static_cast<cryptonote::network_type>(config.m_network_type.get()), 1, true, std::move(http_client_factory)));
wallet->set_daemon_connection(config.get_server());
wallet->m_w2->set_seed_language(language);
if (config.m_account_lookahead != boost::none) wallet->m_w2->set_subaddress_lookahead(config.m_account_lookahead.get(), config.m_subaddress_lookahead.get());
wallet->m_w2->generate(config.m_path.get(), config.m_password.get(), recovery_key, true, false);
wallet->m_w2->set_refresh_from_block_height(config.m_restore_height.get());
wallet->init_common();
return wallet;
}
monero_wallet_full* monero_wallet_full::create_wallet_from_keys(const monero_wallet_config& config, std::unique_ptr<epee::net_utils::http::http_client_factory> http_client_factory) {
MTRACE("create_wallet_from_keys(...)");
monero_wallet_full* wallet = new monero_wallet_full();
// parse and validate private spend key
crypto::secret_key spend_key_sk;
bool has_spend_key = false;
if (!config.m_private_spend_key.get().empty()) {
cryptonote::blobdata spend_key_data;
if (!epee::string_tools::parse_hexstr_to_binbuff(config.m_private_spend_key.get(), spend_key_data) || spend_key_data.size() != sizeof(crypto::secret_key)) {
throw std::runtime_error("failed to parse secret spend key");
}
has_spend_key = true;
spend_key_sk = *reinterpret_cast<const crypto::secret_key*>(spend_key_data.data());
}
// parse and validate private view key
bool has_view_key = true;
crypto::secret_key view_key_sk;
if (config.m_private_view_key.get().empty()) {
if (has_spend_key) has_view_key = false;
else throw std::runtime_error("Neither spend key nor view key supplied");
}
if (has_view_key) {
cryptonote::blobdata view_key_data;
if (!epee::string_tools::parse_hexstr_to_binbuff(config.m_private_view_key.get(), view_key_data) || view_key_data.size() != sizeof(crypto::secret_key)) {
throw std::runtime_error("failed to parse secret view key");
}
view_key_sk = *reinterpret_cast<const crypto::secret_key*>(view_key_data.data());
}
// parse and validate address
cryptonote::address_parse_info address_info;
if (config.m_primary_address.get().empty()) {
if (has_view_key) throw std::runtime_error("must provide primary address if providing private view key");
} else {
if (!get_account_address_from_str(address_info, static_cast<cryptonote::network_type>(config.m_network_type.get()), config.m_primary_address.get())) throw std::runtime_error("failed to parse address");
// check the spend and view keys match the given address
crypto::public_key pkey;
if (has_spend_key) {
if (!crypto::secret_key_to_public_key(spend_key_sk, pkey)) throw std::runtime_error("failed to verify secret spend key");
if (address_info.address.m_spend_public_key != pkey) throw std::runtime_error("spend key does not match address");
}
if (has_view_key) {
if (!crypto::secret_key_to_public_key(view_key_sk, pkey)) throw std::runtime_error("failed to verify secret view key");
if (address_info.address.m_view_public_key != pkey) throw std::runtime_error("view key does not match address");
}
}
// validate language
if (!monero_utils::is_valid_language(config.m_language.get())) throw std::runtime_error("Unknown language: " + config.m_language.get());
// initialize wallet
if (http_client_factory == nullptr) wallet->m_w2 = std::unique_ptr<tools::wallet2>(new tools::wallet2(static_cast<cryptonote::network_type>(config.m_network_type.get()), 1, true));
else wallet->m_w2 = std::unique_ptr<tools::wallet2>(new tools::wallet2(static_cast<cryptonote::network_type>(config.m_network_type.get()), 1, true, std::move(http_client_factory)));
if (config.m_account_lookahead != boost::none) wallet->m_w2->set_subaddress_lookahead(config.m_account_lookahead.get(), config.m_subaddress_lookahead.get());
if (has_spend_key && has_view_key) wallet->m_w2->generate(config.m_path.get(), config.m_password.get(), address_info.address, spend_key_sk, view_key_sk);
else if (has_spend_key) wallet->m_w2->generate(config.m_path.get(), config.m_password.get(), spend_key_sk, true, false);
else wallet->m_w2->generate(config.m_path.get(), config.m_password.get(), address_info.address, view_key_sk);
wallet->set_daemon_connection(config.get_server());
wallet->m_w2->set_refresh_from_block_height(config.m_restore_height.get());
wallet->m_w2->set_seed_language(config.m_language.get());
wallet->init_common();
return wallet;
}
monero_wallet_full* monero_wallet_full::create_wallet_random(const monero_wallet_config& config, std::unique_ptr<epee::net_utils::http::http_client_factory> http_client_factory) {
MTRACE("create_wallet_random(...)");
monero_wallet_full* wallet = new monero_wallet_full();
if (http_client_factory == nullptr) wallet->m_w2 = std::unique_ptr<tools::wallet2>(new tools::wallet2(static_cast<cryptonote::network_type>(config.m_network_type.get()), 1, true));
else wallet->m_w2 = std::unique_ptr<tools::wallet2>(new tools::wallet2(static_cast<cryptonote::network_type>(config.m_network_type.get()), 1, true, std::move(http_client_factory)));
wallet->set_daemon_connection(config.get_server());
wallet->m_w2->set_seed_language(config.m_language.get());
crypto::secret_key secret_key;
if (config.m_account_lookahead != boost::none) wallet->m_w2->set_subaddress_lookahead(config.m_account_lookahead.get(), config.m_subaddress_lookahead.get());
wallet->m_w2->generate(config.m_path.get(), config.m_password.get(), secret_key, false, false);
wallet->init_common();
if (wallet->is_connected_to_daemon()) wallet->m_w2->set_refresh_from_block_height(wallet->get_daemon_height());
return wallet;
}
std::vector<std::string> monero_wallet_full::get_mnemonic_languages() {
std::vector<std::string> languages;
crypto::ElectrumWords::get_language_list(languages, true);
return languages;
}
// ------------------------------- Deprecated -------------------------------
monero_wallet_full* monero_wallet_full::create_wallet_random(const std::string& path, const std::string& password, const monero_network_type network_type, const monero_rpc_connection& daemon_connection, const std::string& language, std::unique_ptr<epee::net_utils::http::http_client_factory> http_client_factory) {
std::cout << "Warning: monero_wallet_full::create_wallet_random() is deprecated and will be removed soon. Use monero_wallet_full::create_wallet(config) instead" << std::endl;
monero_wallet_config config;
config.m_path = path;
config.m_password = password;
config.m_network_type = network_type;
config.set_server(daemon_connection);
config.m_language = language;
return create_wallet(config, std::move(http_client_factory));
}
monero_wallet_full* monero_wallet_full::create_wallet_from_mnemonic(const std::string& path, const std::string& password, const monero_network_type network_type, const std::string& mnemonic, const monero_rpc_connection& daemon_connection, uint64_t restore_height, const std::string& seed_offset, std::unique_ptr<epee::net_utils::http::http_client_factory> http_client_factory) {
std::cout << "Warning: monero_wallet_full::create_wallet_from_mnemonic() is deprecated and will be removed soon. Use monero_wallet_full::create_wallet(config) instead" << std::endl;
monero_wallet_config config;
config.m_path = path;
config.m_password = password;
config.m_network_type = network_type;
config.m_mnemonic = mnemonic;
config.set_server(daemon_connection);
config.m_restore_height = restore_height;
config.m_seed_offset = seed_offset;
return create_wallet(config, std::move(http_client_factory));
}
monero_wallet_full* monero_wallet_full::create_wallet_from_keys(const std::string& path, const std::string& password, const monero_network_type network_type, const std::string& address, const std::string& view_key, const std::string& spend_key, const monero_rpc_connection& daemon_connection, uint64_t restore_height, const std::string& language, std::unique_ptr<epee::net_utils::http::http_client_factory> http_client_factory) {
std::cout << "Warning: monero_wallet_full::create_wallet_from_keys() is deprecated and will be removed soon. Use monero_wallet_full::create_wallet(config) instead" << std::endl;
monero_wallet_config config;
config.m_path = path;
config.m_password = password;
config.m_network_type = network_type;
config.m_primary_address = address;
config.m_private_view_key = view_key;
config.m_private_spend_key = spend_key;
config.set_server(daemon_connection);
config.m_restore_height = restore_height;
config.m_language = language;
return create_wallet(config, std::move(http_client_factory));
}
// ----------------------------- WALLET METHODS -----------------------------
monero_wallet_full::~monero_wallet_full() {
MTRACE("~monero_wallet_full()");
close(false);
}
void monero_wallet_full::set_daemon_connection(const std::string& uri, const std::string& username, const std::string& password) {
MTRACE("set_daemon_connection(" << uri << ", " << username << ", " << "***" << ")");
// prepare uri, login, and is_trusted for wallet2
boost::optional<epee::net_utils::http::login> login{};
login.emplace(username, password);
bool is_trusted = true;
// TODO: is_local_address() uses common/util which requires libunbound
// bool is_trusted = false;
// try { is_trusted = tools::is_local_address(uri); } // wallet is trusted iff local
// catch (const exception &e) { }
// detect ssl TODO: wallet2 does not detect ssl from uri
epee::net_utils::ssl_support_t ssl = uri.rfind("https", 0) == 0 ? epee::net_utils::ssl_support_t::e_ssl_support_enabled : epee::net_utils::ssl_support_t::e_ssl_support_disabled;
// init wallet2 and set daemon connection
if (!m_w2->init(uri, login, {}, 0, is_trusted, ssl)) throw std::runtime_error("Failed to initialize wallet with daemon connection");
is_connected_to_daemon(); // update m_is_connected cache // TODO: better naming?
}
void monero_wallet_full::set_daemon_proxy(const std::string& uri) {
m_w2->set_proxy(uri);
}
void monero_wallet_full::set_daemon_connection(const boost::optional<monero_rpc_connection>& connection) {
if (connection == boost::none) set_daemon_connection("");
else set_daemon_connection(connection->m_uri == boost::none ? "" : connection->m_uri.get(), connection->m_username == boost::none ? "" : connection->m_username.get(), connection->m_password == boost::none ? "" : connection->m_password.get());
}
boost::optional<monero_rpc_connection> monero_wallet_full::get_daemon_connection() const {
MTRACE("monero_wallet_full::get_daemon_connection()");
if (m_w2->get_daemon_address().empty()) return boost::none;
boost::optional<monero_rpc_connection> connection = monero_rpc_connection();
connection->m_uri = m_w2->get_daemon_address();
if (m_w2->get_daemon_login()) {
if (!m_w2->get_daemon_login()->username.empty()) connection->m_username = m_w2->get_daemon_login()->username;
epee::wipeable_string wipeablePassword = m_w2->get_daemon_login()->password;
std::string password = std::string(wipeablePassword.data(), wipeablePassword.size());
if (!password.empty()) connection->m_password = password;
}
return connection;
}
// TODO: could return Wallet::ConnectionStatus_Disconnected, Wallet::ConnectionStatus_WrongVersion, Wallet::ConnectionStatus_Connected like wallet.cpp::connected()
bool monero_wallet_full::is_connected_to_daemon() const {
uint32_t version = 0;
m_is_connected = m_w2->check_connection(&version, NULL, DEFAULT_CONNECTION_TIMEOUT_MILLIS); // TODO: should this be updated elsewhere?
if (!m_is_connected) return false;
//if (!m_w2->light_wallet() && (version >> 16) != CORE_RPC_VERSION_MAJOR) m_is_connected = false; // wrong network type // TODO: disallow rpc version mismatch by configuration
return m_is_connected;
}
bool monero_wallet_full::is_daemon_synced() const {
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
uint64_t daemonHeight = get_daemon_height();
return daemonHeight >= get_daemon_max_peer_height() && daemonHeight > 1;
}
bool monero_wallet_full::is_daemon_trusted() const {
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
return m_w2->is_trusted_daemon();
}
bool monero_wallet_full::is_synced() const {
return m_is_synced;
}
monero_version monero_wallet_full::get_version() const {
monero_version version;
version.m_number = 65552; // same as monero-wallet-rpc v0.15.0.1 release
version.m_is_release = false; // TODO: could pull from MONERO_VERSION_IS_RELEASE in version.cpp
return version;
}
std::string monero_wallet_full::get_path() const {
return m_w2->path();
}
monero_network_type monero_wallet_full::get_network_type() const {
return static_cast<monero_network_type>(m_w2->nettype());
}
std::string monero_wallet_full::get_mnemonic() const {
epee::wipeable_string seed;
bool ready;
if (m_w2->multisig(&ready)) {
if (!ready) throw std::runtime_error("This wallet is multisig, but not yet finalized");
if (!m_w2->get_multisig_seed(seed)) throw std::runtime_error("Failed to get multisig seed.");
} else {
if (m_w2->watch_only()) return "";
if (!m_w2->is_deterministic()) return "";
if (!m_w2->get_seed(seed)) throw std::runtime_error("Failed to get seed.");
}
return std::string(seed.data(), seed.size());
}
std::string monero_wallet_full::get_mnemonic_language() const {
if (m_w2->watch_only()) return "";
return m_w2->get_seed_language();
}
std::string monero_wallet_full::get_public_view_key() const {
MTRACE("get_private_view_key()");
return epee::string_tools::pod_to_hex(m_w2->get_account().get_keys().m_account_address.m_view_public_key);
}
std::string monero_wallet_full::get_private_view_key() const {
MTRACE("get_private_view_key()");
return epee::string_tools::pod_to_hex(m_w2->get_account().get_keys().m_view_secret_key);
}
std::string monero_wallet_full::get_public_spend_key() const {
MTRACE("get_public_spend_key()");
return epee::string_tools::pod_to_hex(m_w2->get_account().get_keys().m_account_address.m_spend_public_key);
}
std::string monero_wallet_full::get_private_spend_key() const {
MTRACE("get_private_spend_key()");
std::string spend_key = epee::string_tools::pod_to_hex(m_w2->get_account().get_keys().m_spend_secret_key);
if (spend_key == "0000000000000000000000000000000000000000000000000000000000000000") spend_key = "";
return spend_key;
}
std::string monero_wallet_full::get_address(uint32_t account_idx, uint32_t subaddress_idx) const {
return m_w2->get_subaddress_as_str({account_idx, subaddress_idx});
}
monero_subaddress monero_wallet_full::get_address_index(const std::string& address) const {
MTRACE("get_address_index(" << address << ")");
// validate address
cryptonote::address_parse_info info;
if (!get_account_address_from_str(info, m_w2->nettype(), address)) {
throw std::runtime_error("Invalid address");
}
// get index of address in wallet
auto index = m_w2->get_subaddress_index(info.address);
if (!index) throw std::runtime_error("Address doesn't belong to the wallet");
// return indices in subaddress
monero_subaddress subaddress;
cryptonote::subaddress_index cn_index = *index;
subaddress.m_account_index = cn_index.major;
subaddress.m_index = cn_index.minor;
return subaddress;
}
monero_integrated_address monero_wallet_full::get_integrated_address(const std::string& standard_address, const std::string& payment_id) const {
MTRACE("get_integrated_address(" << standard_address << ", " << payment_id << ")");
// TODO monero-project: this logic is based on wallet_rpc_server::on_make_integrated_address() and should be moved to wallet so this is unecessary for api users
// randomly generate payment id if not given, else validate
crypto::hash8 payment_id_h8;
if (payment_id.empty()) {
payment_id_h8 = crypto::rand<crypto::hash8>();
} else {
if (!tools::wallet2::parse_short_payment_id(payment_id, payment_id_h8)) throw std::runtime_error("Invalid payment ID: " + payment_id);
}
// use primary address if standard address not given, else validate
if (standard_address.empty()) {
return decode_integrated_address(m_w2->get_integrated_address_as_str(payment_id_h8));
} else {
// validate standard address
cryptonote::address_parse_info info;
if (!cryptonote::get_account_address_from_str(info, m_w2->nettype(), standard_address)) throw std::runtime_error("Invalid address");
if (info.is_subaddress) throw std::runtime_error("Subaddress shouldn't be used");
if (info.has_payment_id) throw std::runtime_error("Already integrated address");
if (payment_id.empty()) throw std::runtime_error("Payment ID shouldn't be left unspecified");
// create integrated address from given standard address
return decode_integrated_address(cryptonote::get_account_integrated_address_as_str(m_w2->nettype(), info.address, payment_id_h8));
}
}
monero_integrated_address monero_wallet_full::decode_integrated_address(const std::string& integrated_address) const {
MTRACE("decode_integrated_address(" << integrated_address << ")");
// validate integrated address
cryptonote::address_parse_info info;
if (!cryptonote::get_account_address_from_str(info, m_w2->nettype(), integrated_address)) throw std::runtime_error("Invalid address");
if (!info.has_payment_id) throw std::runtime_error("Address is not an integrated address");
// initialize and return result
monero_integrated_address result;
result.m_standard_address = cryptonote::get_account_address_as_str(m_w2->nettype(), info.is_subaddress, info.address);
result.m_payment_id = epee::string_tools::pod_to_hex(info.payment_id);
result.m_integrated_address = integrated_address;
return result;
}
uint64_t monero_wallet_full::get_height() const {
return m_w2->get_blockchain_current_height();
}
uint64_t monero_wallet_full::get_restore_height() const { // TODO: rename to get_sync_from_height()
return m_w2->get_refresh_from_block_height();
}
void monero_wallet_full::set_restore_height(uint64_t restore_height) {
m_w2->set_refresh_from_block_height(restore_height);
}
uint64_t monero_wallet_full::get_daemon_height() const {
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
std::string err;
uint64_t result = m_w2->get_daemon_blockchain_height(err);
if (!err.empty()) throw std::runtime_error(err);
return result;
}
uint64_t monero_wallet_full::get_daemon_max_peer_height() const {
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
std::string err;
uint64_t result = m_w2->get_daemon_blockchain_target_height(err);
if (!err.empty()) throw std::runtime_error(err);
if (result == 0) result = get_daemon_height(); // TODO monero-project: target height can be 0 when daemon is synced. Use blockchain height instead
return result;
}
uint64_t monero_wallet_full::get_height_by_date(uint16_t year, uint8_t month, uint8_t day) const {
return m_w2->get_blockchain_height_by_date(year, month, day);
}
void monero_wallet_full::add_listener(monero_wallet_listener& listener) {
MTRACE("add_listener()");
m_listeners.insert(&listener);
m_w2_listener->update_listening();
}
void monero_wallet_full::remove_listener(monero_wallet_listener& listener) {
MTRACE("remove_listener()");
m_listeners.erase(&listener);
if (!m_sync_loop_running) m_w2_listener->update_listening(); // listener is unregistered after sync to avoid segfault
}
std::set<monero_wallet_listener*> monero_wallet_full::get_listeners() {
MTRACE("get_listeners()");
return m_listeners;
}
monero_sync_result monero_wallet_full::sync() {
MTRACE("sync()");
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
return lock_and_sync();
}
monero_sync_result monero_wallet_full::sync(monero_wallet_listener& listener) {
MTRACE("sync(listener)");
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
// register listener
add_listener(listener);
// sync wallet
monero_sync_result result = lock_and_sync(boost::none);
// unregister listener
remove_listener(listener);
// return sync result
return result;
}
monero_sync_result monero_wallet_full::sync(uint64_t start_height) {
MTRACE("sync(" << start_height << ")");
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
return lock_and_sync(start_height);
}
monero_sync_result monero_wallet_full::sync(uint64_t start_height, monero_wallet_listener& listener) {
MTRACE("sync(" << start_height << ", listener)");
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
// wrap and register sync listener as wallet listener
add_listener(listener);
// sync wallet
monero_sync_result result = lock_and_sync(start_height);
// unregister sync listener
remove_listener(listener);
// return sync result
return result;
}
void monero_wallet_full::start_syncing(uint64_t sync_period_in_ms) {
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
m_syncing_interval = sync_period_in_ms;
if (!m_syncing_enabled) {
m_syncing_enabled = true;
run_sync_loop(); // sync wallet on loop in background
}
}
void monero_wallet_full::stop_syncing() {
m_syncing_enabled = false;
m_w2->stop();
}
void monero_wallet_full::scan_txs(const std::vector<std::string>& tx_ids) {
MTRACE("scan_txs()");
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
// convert string ids to crypto hashes
std::vector<crypto::hash> tx_hashes;
std::vector<std::string>::const_iterator i = tx_ids.begin();
while (i != tx_ids.end()) {
cryptonote::blobdata tx_hash_blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(*i++, tx_hash_blob) || tx_hash_blob.size() != sizeof(crypto::hash)) throw std::runtime_error("TX ID has invalid format");
crypto::hash tx_hash = *reinterpret_cast<const crypto::hash*>(tx_hash_blob.data());
tx_hashes.push_back(tx_hash);
}
// scan txs
m_w2->scan_tx(tx_hashes);
}
void monero_wallet_full::rescan_spent() {
MTRACE("rescan_spent()");
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
if (!is_daemon_trusted()) throw std::runtime_error("Rescan spent can only be used with a trusted daemon");
m_w2->rescan_spent();
}
// TODO: support arguments bool hard, bool refresh = true, bool keep_key_images = false
void monero_wallet_full::rescan_blockchain() {
MTRACE("rescan_blockchain()");
if (!m_is_connected) throw std::runtime_error("Wallet is not connected to daemon");
m_rescan_on_sync = true;
lock_and_sync();
}
// isMultisigImportNeeded
uint64_t monero_wallet_full::get_balance() const {
return m_w2->balance_all(STRICT_);
}
uint64_t monero_wallet_full::get_balance(uint32_t account_idx) const {
return m_w2->balance(account_idx, STRICT_);
}
uint64_t monero_wallet_full::get_balance(uint32_t account_idx, uint32_t subaddress_idx) const {
std::map<uint32_t, uint64_t> balance_per_subaddress = m_w2->balance_per_subaddress(account_idx, STRICT_);
auto iter = balance_per_subaddress.find(subaddress_idx);
return iter == balance_per_subaddress.end() ? 0 : iter->second;
}
uint64_t monero_wallet_full::get_unlocked_balance() const {
return m_w2->unlocked_balance_all(STRICT_);
}
uint64_t monero_wallet_full::get_unlocked_balance(uint32_t account_idx) const {
return m_w2->unlocked_balance(account_idx, STRICT_);
}
uint64_t monero_wallet_full::get_unlocked_balance(uint32_t account_idx, uint32_t subaddress_idx) const {
std::map<uint32_t, std::pair<uint64_t, std::pair<uint64_t, uint64_t>>> unlocked_balance_per_subaddress = m_w2->unlocked_balance_per_subaddress(account_idx, STRICT_);
auto iter = unlocked_balance_per_subaddress.find(subaddress_idx);
return iter == unlocked_balance_per_subaddress.end() ? 0 : iter->second.first;
}
std::vector<monero_account> monero_wallet_full::get_accounts(bool include_subaddresses, const std::string& tag) const {
MTRACE("get_accounts(" << include_subaddresses << ", " << tag << ")");
// need transfers to inform if subaddresses used
std::vector<tools::wallet2::transfer_details> transfers;
if (include_subaddresses) m_w2->get_transfers(transfers);
// build accounts
std::vector<monero_account> accounts;
for (uint32_t account_idx = 0; account_idx < m_w2->get_num_subaddress_accounts(); account_idx++) {
monero_account account;
account.m_index = account_idx;
account.m_primary_address = get_address(account_idx, 0);
account.m_balance = m_w2->balance(account_idx, STRICT_);
account.m_unlocked_balance = m_w2->unlocked_balance(account_idx, STRICT_);
if (include_subaddresses) account.m_subaddresses = get_subaddresses_aux(account_idx, std::vector<uint32_t>(), transfers);
accounts.push_back(account);
}
return accounts;
}
monero_account monero_wallet_full::get_account(uint32_t account_idx, bool include_subaddresses) const {
MTRACE("get_account(" << account_idx << ", " << include_subaddresses << ")");
// need transfers to inform if subaddresses used
std::vector<tools::wallet2::transfer_details> transfers;
if (include_subaddresses) m_w2->get_transfers(transfers);
// build and return account
monero_account account;
account.m_index = account_idx;
account.m_primary_address = get_address(account_idx, 0);
account.m_balance = m_w2->balance(account_idx, STRICT_);
account.m_unlocked_balance = m_w2->unlocked_balance(account_idx, STRICT_);
if (include_subaddresses) account.m_subaddresses = get_subaddresses_aux(account_idx, std::vector<uint32_t>(), transfers);
return account;
}
monero_account monero_wallet_full::create_account(const std::string& label) {
MTRACE("create_account(" << label << ")");
// create account
m_w2->add_subaddress_account(label);
// initialize and return result
monero_account account;
account.m_index = m_w2->get_num_subaddress_accounts() - 1;
account.m_primary_address = m_w2->get_subaddress_as_str({account.m_index.get(), 0});
account.m_balance = 0;
account.m_unlocked_balance = 0;
return account;
}
std::vector<monero_subaddress> monero_wallet_full::get_subaddresses(const uint32_t account_idx, const std::vector<uint32_t>& subaddress_indices) const {
MTRACE("get_subaddresses(" << account_idx << ", ...)");
MTRACE("Subaddress indices size: " << subaddress_indices.size());
std::vector<tools::wallet2::transfer_details> transfers;
m_w2->get_transfers(transfers);
return get_subaddresses_aux(account_idx, subaddress_indices, transfers);
}
monero_subaddress monero_wallet_full::create_subaddress(const uint32_t account_idx, const std::string& label) {
MTRACE("create_subaddress(" << account_idx << ", " << label << ")");
// create subaddress
m_w2->add_subaddress(account_idx, label);
// initialize and return result
monero_subaddress subaddress;
subaddress.m_account_index = account_idx;
subaddress.m_index = m_w2->get_num_subaddresses(account_idx) - 1;
subaddress.m_address = m_w2->get_subaddress_as_str({account_idx, subaddress.m_index.get()});
subaddress.m_label = label;
subaddress.m_balance = 0;
subaddress.m_unlocked_balance = 0;
subaddress.m_num_unspent_outputs = 0;
subaddress.m_is_used = false;
subaddress.m_num_blocks_to_unlock = 0;
return subaddress;
}
void monero_wallet_full::set_subaddress_label(const uint32_t account_idx, const uint32_t subaddress_idx, const std::string& label) {
MTRACE("set_subaddress_label(" << account_idx << ", " << subaddress_idx << ", " << label << ")");
cryptonote::subaddress_index index = {account_idx, subaddress_idx};
m_w2->set_subaddress_label(index, label);
}
std::vector<std::shared_ptr<monero_tx_wallet>> monero_wallet_full::get_txs() const {
return get_txs(monero_tx_query());
}
std::vector<std::shared_ptr<monero_tx_wallet>> monero_wallet_full::get_txs(const monero_tx_query& query) const {
std::vector<std::string> missing_tx_hashes;
std::vector<std::shared_ptr<monero_tx_wallet>> txs = get_txs(query, missing_tx_hashes);
if (!missing_tx_hashes.empty()) throw std::runtime_error("Tx not found in wallet: " + missing_tx_hashes[0]);
return txs;
}
std::vector<std::shared_ptr<monero_tx_wallet>> monero_wallet_full::get_txs(const monero_tx_query& query, std::vector<std::string>& missing_tx_hashes) const {
MTRACE("get_txs(query)");
// copy query
std::shared_ptr<monero_tx_query> query_sp = std::make_shared<monero_tx_query>(query); // convert to shared pointer
std::shared_ptr<monero_tx_query> _query = query_sp->copy(query_sp, std::make_shared<monero_tx_query>()); // deep copy
// // log query
// if (_query->m_block != boost::none) std::cout << "Tx query's rooted at [block]: " << _query->m_block.get()->serialize() << std::endl;
// else std::cout << "Tx _query: " << _query->serialize() << std::endl;
// temporarily disable transfer and output queries in order to collect all tx context
boost::optional<std::shared_ptr<monero_transfer_query>> transfer_query = _query->m_transfer_query;
boost::optional<std::shared_ptr<monero_output_query>> input_query = _query->m_input_query;
boost::optional<std::shared_ptr<monero_output_query>> output_query = _query->m_output_query;
_query->m_transfer_query = boost::none;
_query->m_input_query = boost::none;
_query->m_output_query = boost::none;
// fetch all transfers that meet tx query
std::shared_ptr<monero_transfer_query> temp_transfer_query = std::make_shared<monero_transfer_query>();
temp_transfer_query->m_tx_query = decontextualize(_query->copy(_query, std::make_shared<monero_tx_query>()));
temp_transfer_query->m_tx_query.get()->m_transfer_query = temp_transfer_query;
std::vector<std::shared_ptr<monero_transfer>> transfers = get_transfers_aux(*temp_transfer_query);
// collect unique txs from transfers while retaining order
std::vector<std::shared_ptr<monero_tx_wallet>> txs = std::vector<std::shared_ptr<monero_tx_wallet>>();
std::unordered_set<std::shared_ptr<monero_tx_wallet>> txsSet;
for (const std::shared_ptr<monero_transfer>& transfer : transfers) {
if (txsSet.find(transfer->m_tx) == txsSet.end()) {
txs.push_back(transfer->m_tx);
txsSet.insert(transfer->m_tx);
}
}
// cache types into maps for merging and lookup
std::map<std::string, std::shared_ptr<monero_tx_wallet>> tx_map;
std::map<uint64_t, std::shared_ptr<monero_block>> block_map;
for (const std::shared_ptr<monero_tx_wallet>& tx : txs) {
merge_tx(tx, tx_map, block_map);
}
// fetch and merge outputs if requested
if ((_query->m_include_outputs != boost::none && *_query->m_include_outputs) || output_query != boost::none) {
std::shared_ptr<monero_output_query> temp_output_query = std::make_shared<monero_output_query>();
temp_output_query->m_tx_query = decontextualize(_query->copy(_query, std::make_shared<monero_tx_query>()));
temp_output_query->m_tx_query.get()->m_output_query = temp_output_query;
std::vector<std::shared_ptr<monero_output_wallet>> outputs = get_outputs_aux(*temp_output_query);
// merge output txs one time while retaining order
std::unordered_set<std::shared_ptr<monero_tx_wallet>> output_txs;
for (const std::shared_ptr<monero_output_wallet>& output : outputs) {
std::shared_ptr<monero_tx_wallet> tx = std::static_pointer_cast<monero_tx_wallet>(output->m_tx);
if (output_txs.find(tx) == output_txs.end()) {
merge_tx(tx, tx_map, block_map);
output_txs.insert(tx);
}
}
}
// restore transfer and output queries
_query->m_transfer_query = transfer_query;
_query->m_input_query = input_query;
_query->m_output_query = output_query;
// filter txs that don't meet transfer query
std::vector<std::shared_ptr<monero_tx_wallet>> queried_txs;
std::vector<std::shared_ptr<monero_tx_wallet>>::iterator tx_iter = txs.begin();
while (tx_iter != txs.end()) {
std::shared_ptr<monero_tx_wallet> tx = *tx_iter;
if (_query->meets_criteria(tx.get())) {
queried_txs.push_back(tx);
tx_iter++;
} else {
tx_map.erase(tx->m_hash.get());
tx_iter = txs.erase(tx_iter);
if (tx->m_block != boost::none) tx->m_block.get()->m_txs.erase(std::remove(tx->m_block.get()->m_txs.begin(), tx->m_block.get()->m_txs.end(), tx), tx->m_block.get()->m_txs.end()); // TODO, no way to use tx_iter?
}
}
txs = queried_txs;
// special case: re-fetch txs if inconsistency caused by needing to make multiple wallet calls // TODO monero-project: offer wallet.get_txs(...)
for (const std::shared_ptr<monero_tx_wallet>& tx : txs) {
if (*tx->m_is_confirmed && tx->m_block == boost::none) {
std::cout << "WARNING: Inconsistency detected building txs from multiple wallet2 calls, re-fetching" << std::endl;
return get_txs(*_query, missing_tx_hashes);
}
}
// if tx hashes requested, order txs and collect missing hashes
if (!_query->m_hashes.empty()) {
txs.clear();
for (const std::string& tx_hash : _query->m_hashes) {
std::map<std::string, std::shared_ptr<monero_tx_wallet>>::const_iterator tx_iter = tx_map.find(tx_hash);
if (tx_iter != tx_map.end()) txs.push_back(tx_iter->second);
else missing_tx_hashes.push_back(tx_hash);
}
}
return txs;
}
std::vector<std::shared_ptr<monero_transfer>> monero_wallet_full::get_transfers(const monero_transfer_query& query) const {
// // log query
// if (query.m_tx_query != boost::none) {
// if ((*query.m_tx_query)->m_block == boost::none) std::cout << "Transfer query's tx query rooted at [tx]:" << (*query.m_tx_query)->serialize() << std::endl;
// else std::cout << "Transfer query's tx query rooted at [block]: " << (*(*query.m_tx_query)->m_block)->serialize() << std::endl;
// } else std::cout << "Transfer query: " << query.serialize() << std::endl;
// get transfers directly if query does not require tx context (e.g. other transfers, outputs)
if (!is_contextual(query)) return get_transfers_aux(query);
// otherwise get txs with full models to fulfill query
std::vector<std::shared_ptr<monero_transfer>> transfers;
for (const std::shared_ptr<monero_tx_wallet>& tx : get_txs(*(query.m_tx_query.get()))) {
for (const std::shared_ptr<monero_transfer>& transfer : tx->filter_transfers(query)) { // collect queried transfers, erase if excluded
transfers.push_back(transfer);
}
}
return transfers;
}
std::vector<std::shared_ptr<monero_output_wallet>> monero_wallet_full::get_outputs(const monero_output_query& query) const {
// // log query
// if (query.m_tx_query != boost::none) {
// if ((*query.m_tx_query)->m_block == boost::none) std::cout << "Output query's tx query rooted at [tx]:" << (*query.m_tx_query)->serialize() << std::endl;
// else std::cout << "Output query's tx query rooted at [block]: " << (*(*query.m_tx_query)->m_block)->serialize() << std::endl;
// } else std::cout << "Output query: " << query.serialize() << std::endl;
// get outputs directly if query does not require tx context (e.g. other outputs, transfers)
if (!is_contextual(query)) return get_outputs_aux(query);
// otherwise get txs with full models to fulfill query
std::vector<std::shared_ptr<monero_output_wallet>> outputs;
for (const std::shared_ptr<monero_tx_wallet>& tx : get_txs(*(query.m_tx_query.get()))) {
for (const std::shared_ptr<monero_output_wallet>& output : tx->filter_outputs_wallet(query)) { // collect queried outputs, erase if excluded
outputs.push_back(output);
}
}
return outputs;
}
std::string monero_wallet_full::export_outputs(bool all) const {
return epee::string_tools::buff_to_hex_nodelimer(m_w2->export_outputs_to_str(all));
}
int monero_wallet_full::import_outputs(const std::string& outputs_hex) {
// validate and parse hex data
cryptonote::blobdata blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(outputs_hex, blob)) {
throw std::runtime_error("Failed to parse hex");
}
// import hex and return result
return m_w2->import_outputs_from_str(blob);
}
std::vector<std::shared_ptr<monero_key_image>> monero_wallet_full::export_key_images(bool all) const {
MTRACE("monero_wallet_full::export_key_images()");
// build key images from wallet2 types
std::vector<std::shared_ptr<monero_key_image>> key_images;
std::pair<uint64_t, std::vector<std::pair<crypto::key_image, crypto::signature>>> ski = m_w2->export_key_images(all);
for (uint64_t n = 0; n < ski.second.size(); ++n) {
std::shared_ptr<monero_key_image> key_image = std::make_shared<monero_key_image>();
key_images.push_back(key_image);
key_image->m_hex = epee::string_tools::pod_to_hex(ski.second[n].first);
key_image->m_signature = epee::string_tools::pod_to_hex(ski.second[n].second);
}
return key_images;
}
std::shared_ptr<monero_key_image_import_result> monero_wallet_full::import_key_images(const std::vector<std::shared_ptr<monero_key_image>>& key_images) {
MTRACE("monero_wallet_full::import_key_images()");
// validate and prepare key images for wallet2
std::vector<std::pair<crypto::key_image, crypto::signature>> ski;
ski.resize(key_images.size());
for (uint64_t n = 0; n < ski.size(); ++n) {
if (!epee::string_tools::hex_to_pod(key_images[n]->m_hex.get(), ski[n].first)) {
throw std::runtime_error("failed to parse key image");
}
if (!epee::string_tools::hex_to_pod(key_images[n]->m_signature.get(), ski[n].second)) {
throw std::runtime_error("failed to parse signature");
}
}
// import key images
uint64_t spent = 0, unspent = 0;
uint64_t height = m_w2->import_key_images(ski, 0, spent, unspent, is_connected_to_daemon()); // TODO: use offset? refer to wallet_rpc_server::on_import_key_images() req.offset
// translate results
std::shared_ptr<monero_key_image_import_result> result = std::make_shared<monero_key_image_import_result>();
result->m_height = height;
result->m_spent_amount = spent;
result->m_unspent_amount = unspent;
return result;
}
void monero_wallet_full::freeze_output(const std::string& key_image) {
if (key_image.empty()) throw std::runtime_error("Must specify key image to freeze");
crypto::key_image ki;
if (!epee::string_tools::hex_to_pod(key_image, ki)) throw new std::runtime_error("failed to parse key imge");
m_w2->freeze(ki);
}
void monero_wallet_full::thaw_output(const std::string& key_image) {
if (key_image.empty()) throw std::runtime_error("Must specify key image to thaw");
crypto::key_image ki;
if (!epee::string_tools::hex_to_pod(key_image, ki)) throw new std::runtime_error("failed to parse key imge");
m_w2->thaw(ki);
}
bool monero_wallet_full::is_output_frozen(const std::string& key_image) {
if (key_image.empty()) throw std::runtime_error("Must specify key image to check if frozen");
crypto::key_image ki;
if (!epee::string_tools::hex_to_pod(key_image, ki)) throw new std::runtime_error("failed to parse key imge");
return m_w2->frozen(ki);
}
std::vector<std::shared_ptr<monero_tx_wallet>> monero_wallet_full::create_txs(const monero_tx_config& config) {
MTRACE("monero_wallet_full::create_txs");
//std::cout << "monero_tx_config: " << config.serialize() << std::endl;
// validate config
if (config.m_account_index == boost::none) throw std::runtime_error("Must specify account index to send from");
// prepare parameters for wallet rpc's validate_transfer()
std::string payment_id = config.m_payment_id == boost::none ? std::string("") : config.m_payment_id.get();
std::list<tools::wallet_rpc::transfer_destination> tr_destinations;
for (const std::shared_ptr<monero_destination>& destination : config.get_normalized_destinations()) {
tools::wallet_rpc::transfer_destination tr_destination;
if (destination->m_amount == boost::none) throw std::runtime_error("Destination amount not defined");
if (destination->m_address == boost::none) throw std::runtime_error("Destination address not defined");
tr_destination.amount = destination->m_amount.get();
tr_destination.address = destination->m_address.get();
tr_destinations.push_back(tr_destination);
}
// validate the requested txs and populate dsts & extra
std::vector<cryptonote::tx_destination_entry> dsts;
std::vector<uint8_t> extra;
epee::json_rpc::error err;
if (!validate_transfer(m_w2.get(), tr_destinations, payment_id, dsts, extra, true, err)) {
throw std::runtime_error(err.message);
}
// prepare parameters for wallet2's create_transactions_2()
uint64_t mixin = m_w2->adjust_mixin(0); // get mixin for call to 'create_transactions_2'
uint32_t priority = m_w2->adjust_priority(config.m_priority == boost::none ? 0 : config.m_priority.get());
uint64_t unlock_height = config.m_unlock_height == boost::none ? 0 : config.m_unlock_height.get();
uint32_t account_index = config.m_account_index.get();
std::set<uint32_t> subaddress_indices;
for (const uint32_t& subaddress_idx : config.m_subaddress_indices) subaddress_indices.insert(subaddress_idx);
// prepare transactions
std::vector<wallet2::pending_tx> ptx_vector = m_w2->create_transactions_2(dsts, mixin, unlock_height, priority, extra, account_index, subaddress_indices);
if (ptx_vector.empty()) throw std::runtime_error("No transaction created");
// check if request cannot be fulfilled due to splitting
if (config.m_can_split != boost::none && config.m_can_split.get() == false && ptx_vector.size() != 1) {
throw std::runtime_error("Transaction would be too large. Try create_txs()");
}
// config for fill_response()
bool get_tx_keys = true;
bool get_tx_hex = true;
bool get_tx_metadata = true;
bool relay = config.m_relay != boost::none && config.m_relay.get();
if (config.m_relay != boost::none && config.m_relay.get() == true && is_multisig()) throw std::runtime_error("Cannot relay multisig transaction until co-signed");
// commit txs (if relaying) and get response using wallet rpc's fill_response()
std::list<std::string> tx_keys;
std::list<uint64_t> tx_amounts;
std::list<uint64_t> tx_fees;
std::list<uint64_t> tx_weights;
std::string multisig_tx_hex;
std::string unsigned_tx_hex;
std::list<std::string> tx_hashes;
std::list<std::string> tx_blobs;
std::list<std::string> tx_metadatas;
std::list<key_image_list> input_key_images_list;
if (!fill_response(m_w2.get(), ptx_vector, get_tx_keys, tx_keys, tx_amounts, tx_fees, tx_weights, multisig_tx_hex, unsigned_tx_hex, !relay, tx_hashes, get_tx_hex, tx_blobs, get_tx_metadata, tx_metadatas, input_key_images_list, err)) {
throw std::runtime_error("need to handle error filling response!"); // TODO
}
// build sent txs from results // TODO: break this into separate utility function
std::vector<std::shared_ptr<monero_tx_wallet>> txs;
auto tx_hashes_iter = tx_hashes.begin();
auto tx_keys_iter = tx_keys.begin();
auto tx_amounts_iter = tx_amounts.begin();
auto tx_fees_iter = tx_fees.begin();
auto tx_weights_iter = tx_weights.begin();
auto tx_blobs_iter = tx_blobs.begin();
auto tx_metadatas_iter = tx_metadatas.begin();
auto input_key_images_list_iter = input_key_images_list.begin();
while (tx_fees_iter != tx_fees.end()) {
// init tx with outgoing transfer from filled values
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
txs.push_back(tx);
tx->m_hash = *tx_hashes_iter;
tx->m_key = *tx_keys_iter;
tx->m_fee = *tx_fees_iter;
tx->m_weight = *tx_weights_iter;
tx->m_full_hex = *tx_blobs_iter;
tx->m_metadata = *tx_metadatas_iter;
std::shared_ptr<monero_outgoing_transfer> out_transfer = std::make_shared<monero_outgoing_transfer>();
tx->m_outgoing_transfer = out_transfer;
out_transfer->m_amount = *tx_amounts_iter;
// init inputs with key images
std::list<std::string> input_key_images = (*input_key_images_list_iter).key_images;
for (const std::string& input_key_image : input_key_images) {
std::shared_ptr<monero_output_wallet> input = std::make_shared<monero_output_wallet>();
input->m_tx = tx;
tx->m_inputs.push_back(input);
input->m_key_image = std::make_shared<monero_key_image>();
input->m_key_image.get()->m_hex = input_key_image;
}
// init other known fields
tx->m_is_outgoing = true;
tx->m_payment_id = config.m_payment_id;
tx->m_is_confirmed = false;
tx->m_is_miner_tx = false;
tx->m_is_failed = false; // TODO: test and handle if true
tx->m_relay = config.m_relay != boost::none && config.m_relay.get();
tx->m_is_relayed = tx->m_relay.get();
tx->m_in_tx_pool = tx->m_relay.get();
if (!tx->m_is_failed.get() && tx->m_is_relayed.get()) tx->m_is_double_spend_seen = false; // TODO: test and handle if true
tx->m_num_confirmations = 0;
tx->m_ring_size = monero_utils::RING_SIZE;
tx->m_unlock_height = config.m_unlock_height == boost::none ? 0 : config.m_unlock_height.get();
tx->m_is_locked = true;
if (tx->m_is_relayed.get()) tx->m_last_relayed_timestamp = static_cast<uint64_t>(time(NULL)); // set last relayed timestamp to current time iff relayed // TODO monero-project: this should be encapsulated in wallet2
out_transfer->m_account_index = config.m_account_index;
if (config.m_subaddress_indices.size() == 1) out_transfer->m_subaddress_indices.push_back(config.m_subaddress_indices[0]); // subaddress index is known iff 1 requested // TODO: get all known subaddress indices here
// iterate to next element
tx_keys_iter++;
tx_amounts_iter++;
tx_fees_iter++;
tx_hashes_iter++;
tx_blobs_iter++;
tx_metadatas_iter++;
input_key_images_list_iter++;
}
// copy destinations if single tx
if (txs.size() == 1) txs[0]->m_outgoing_transfer.get()->m_destinations = config.get_normalized_destinations();
// build tx set
std::shared_ptr<monero_tx_set> tx_set = std::make_shared<monero_tx_set>();
tx_set->m_txs = txs;
for (int i = 0; i < txs.size(); i++) txs[i]->m_tx_set = tx_set;
if (!multisig_tx_hex.empty()) tx_set->m_multisig_tx_hex = multisig_tx_hex;
if (!unsigned_tx_hex.empty()) tx_set->m_unsigned_tx_hex = unsigned_tx_hex;
// notify listeners of spent funds
if (relay) m_w2_listener->on_spend_txs(txs);
return txs;
}
std::vector<std::shared_ptr<monero_tx_wallet>> monero_wallet_full::sweep_unlocked(const monero_tx_config& config) {
// validate config
std::vector<std::shared_ptr<monero_destination>> destinations = config.get_normalized_destinations();
if (destinations.size() != 1) throw std::runtime_error("Must specify exactly one destination to sweep to");
if (destinations[0]->m_address == boost::none) throw std::runtime_error("Must specify destination address to sweep to");
if (destinations[0]->m_amount != boost::none) throw std::runtime_error("Cannot specify amount to sweep");
if (config.m_account_index == boost::none && config.m_subaddress_indices.size() != 0) throw std::runtime_error("Must specify account index if subaddress indices are specified");
// determine account and subaddress indices to sweep; default to all with unlocked balance if not specified
std::map<uint32_t, std::vector<uint32_t>> indices;
if (config.m_account_index != boost::none) {
if (config.m_subaddress_indices.size() != 0) {
indices[config.m_account_index.get()] = config.m_subaddress_indices;
} else {
std::vector<uint32_t> subaddress_indices;
for (const monero_subaddress& subaddress : monero_wallet::get_subaddresses(config.m_account_index.get())) {
if (subaddress.m_unlocked_balance.get() > 0) subaddress_indices.push_back(subaddress.m_index.get());
}
indices[config.m_account_index.get()] = subaddress_indices;
}
} else {
std::vector<monero_account> accounts = monero_wallet::get_accounts(true);
for (const monero_account& account : accounts) {
if (account.m_unlocked_balance.get() > 0) {
std::vector<uint32_t> subaddress_indices;
for (const monero_subaddress& subaddress : account.m_subaddresses) {
if (subaddress.m_unlocked_balance.get() > 0) subaddress_indices.push_back(subaddress.m_index.get());
}
indices[account.m_index.get()] = subaddress_indices;
}
}
}
// sweep from each account and collect resulting txs
std::vector<std::shared_ptr<monero_tx_wallet>> txs;
for (std::pair<uint32_t, std::vector<uint32_t>> subaddress_indices_pair : indices) {
// copy and modify the original config
monero_tx_config copy = config.copy();
copy.m_account_index = subaddress_indices_pair.first;
copy.m_sweep_each_subaddress = false;
// sweep all subaddresses together // TODO monero-project: can this reveal outputs belong to the same wallet?
if (copy.m_sweep_each_subaddress == boost::none || copy.m_sweep_each_subaddress.get() != true) {
copy.m_subaddress_indices = subaddress_indices_pair.second;
std::vector<std::shared_ptr<monero_tx_wallet>> account_txs = sweep_account(copy);
txs.insert(std::end(txs), std::begin(account_txs), std::end(account_txs));
}
// otherwise sweep each subaddress individually
else {
for (uint32_t subaddress_index : subaddress_indices_pair.second) {
std::vector<uint32_t> subaddress_indices;
subaddress_indices.push_back(subaddress_index);
copy.m_subaddress_indices = subaddress_indices;
std::vector<std::shared_ptr<monero_tx_wallet>> account_txs = sweep_account(copy);
txs.insert(std::end(txs), std::begin(account_txs), std::end(account_txs));
}
}
}
// notify listeners of spent funds
if (config.m_relay != boost::none && config.m_relay.get()) m_w2_listener->on_spend_txs(txs);
return txs;
}
std::vector<std::shared_ptr<monero_tx_wallet>> monero_wallet_full::sweep_account(const monero_tx_config& config) {
// validate config
std::vector<std::shared_ptr<monero_destination>> destinations = config.get_normalized_destinations();
if (config.m_account_index == boost::none) throw std::runtime_error("Must specify account index to sweep from");
if (destinations.size() != 1 || destinations[0]->m_address == boost::none || destinations[0]->m_address.get().empty()) throw std::runtime_error("Must provide exactly one destination address to sweep output to");
if (destinations[0]->m_amount != boost::none) throw std::runtime_error("Cannot specify destination amount to sweep");
if (config.m_key_image != boost::none) throw std::runtime_error("Cannot define key image in sweep_account(); use sweep_output() to sweep an output by its key image");
if (config.m_sweep_each_subaddress != boost::none && config.m_sweep_each_subaddress.get() == true) throw std::runtime_error("Cannot sweep each subaddress individually with sweep_account");
// validate the transfer requested and populate dsts & extra
std::list<wallet_rpc::transfer_destination> destination;
destination.push_back(wallet_rpc::transfer_destination());
destination.back().amount = 0;
destination.back().address = destinations[0]->m_address.get();
std::string payment_id = config.m_payment_id == boost::none ? std::string("") : config.m_payment_id.get();
std::vector<cryptonote::tx_destination_entry> dsts;
std::vector<uint8_t> extra;
epee::json_rpc::error err;
if (!validate_transfer(m_w2.get(), destination, payment_id, dsts, extra, true, err)) {
throw std::runtime_error("Failed to validate sweep_account transfer request");
}
// TODO monero-project: this is default `outputs` in COMMAND_RPC_SWEEP_ALL which is not documented
uint64_t num_outputs = 1;
// prepare parameters for wallet2's create_transactions_all()
uint64_t below_amount = config.m_below_amount == boost::none ? 0 : config.m_below_amount.get();
uint64_t mixin = m_w2->adjust_mixin(0);
uint32_t priority = m_w2->adjust_priority(config.m_priority == boost::none ? 0 : config.m_priority.get());
uint64_t unlock_height = config.m_unlock_height == boost::none ? 0 : config.m_unlock_height.get();
uint32_t account_index = config.m_account_index.get();
std::set<uint32_t> subaddress_indices;
for (const uint32_t& subaddress_idx : config.m_subaddress_indices) subaddress_indices.insert(subaddress_idx);
// prepare transactions
std::vector<wallet2::pending_tx> ptx_vector = m_w2->create_transactions_all(below_amount, dsts[0].addr, dsts[0].is_subaddress, num_outputs, mixin, unlock_height, priority, extra, account_index, subaddress_indices);
// config for fill_response()
bool get_tx_keys = true;
bool get_tx_hex = true;
bool get_tx_metadata = true;
bool relay = config.m_relay != boost::none && config.m_relay.get();
// commit txs (if relaying) and get response using wallet rpc's fill_response()
std::list<std::string> tx_keys;
std::list<uint64_t> tx_amounts;
std::list<uint64_t> tx_fees;
std::list<uint64_t> tx_weights;
std::string multisig_tx_hex;
std::string unsigned_tx_hex;
std::list<std::string> tx_hashes;
std::list<std::string> tx_blobs;
std::list<std::string> tx_metadatas;
std::list<key_image_list> input_key_images_list;
if (!fill_response(m_w2.get(), ptx_vector, get_tx_keys, tx_keys, tx_amounts, tx_fees, tx_weights, multisig_tx_hex, unsigned_tx_hex, !relay, tx_hashes, get_tx_hex, tx_blobs, get_tx_metadata, tx_metadatas, input_key_images_list, err)) {
throw std::runtime_error("need to handle error filling response!"); // TODO
}
// build sent txs from results // TODO: break this into separate utility function
std::vector<std::shared_ptr<monero_tx_wallet>> txs;
auto tx_hashes_iter = tx_hashes.begin();
auto tx_keys_iter = tx_keys.begin();
auto tx_amounts_iter = tx_amounts.begin();
auto tx_fees_iter = tx_fees.begin();
auto tx_weights_iter = tx_weights.begin();
auto tx_blobs_iter = tx_blobs.begin();
auto tx_metadatas_iter = tx_metadatas.begin();
auto input_key_images_list_iter = input_key_images_list.begin();
while (tx_fees_iter != tx_fees.end()) {
// init tx with outgoing transfer from filled values
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
txs.push_back(tx);
tx->m_hash = *tx_hashes_iter;
tx->m_is_locked = true;
tx->m_is_outgoing = true;
tx->m_key = *tx_keys_iter;
tx->m_fee = *tx_fees_iter;
tx->m_weight = *tx_weights_iter;
tx->m_full_hex = *tx_blobs_iter;
tx->m_metadata = *tx_metadatas_iter;
std::shared_ptr<monero_outgoing_transfer> out_transfer = std::make_shared<monero_outgoing_transfer>();
tx->m_outgoing_transfer = out_transfer;
out_transfer->m_amount = *tx_amounts_iter;
std::shared_ptr<monero_destination> destination = std::make_shared<monero_destination>(destinations[0]->m_address.get(), out_transfer->m_amount.get());
out_transfer->m_destinations.push_back(destination);
// init inputs with key images
std::list<std::string> input_key_images = (*input_key_images_list_iter).key_images;
for (const std::string& input_key_image : input_key_images) {
std::shared_ptr<monero_output_wallet> input = std::make_shared<monero_output_wallet>();
input->m_tx = tx;
tx->m_inputs.push_back(input);
input->m_key_image = std::make_shared<monero_key_image>();
input->m_key_image.get()->m_hex = input_key_image;
}
// init other known fields
tx->m_payment_id = config.m_payment_id;
tx->m_is_confirmed = false;
tx->m_is_miner_tx = false;
tx->m_is_failed = false; // TODO: test and handle if true
tx->m_relay = relay;
tx->m_is_relayed = relay;
tx->m_in_tx_pool = relay;
if (!tx->m_is_failed.get() && tx->m_is_relayed.get()) tx->m_is_double_spend_seen = false; // TODO: test and handle if true
tx->m_num_confirmations = 0;
tx->m_ring_size = monero_utils::RING_SIZE;
tx->m_unlock_height = config.m_unlock_height == boost::none ? 0 : config.m_unlock_height.get();
if (tx->m_is_relayed.get()) tx->m_last_relayed_timestamp = static_cast<uint64_t>(time(NULL)); // set last relayed timestamp to current time iff relayed // TODO monero-project: this should be encapsulated in wallet2
out_transfer->m_account_index = config.m_account_index;
if (config.m_subaddress_indices.size() == 1) out_transfer->m_subaddress_indices.push_back(config.m_subaddress_indices[0]); // subaddress index is known iff 1 requested // TODO: get all known subaddress indices here
// iterate to next element
tx_keys_iter++;
tx_amounts_iter++;
tx_fees_iter++;
tx_hashes_iter++;
tx_blobs_iter++;
tx_metadatas_iter++;
input_key_images_list_iter++;
}
// link tx set and return
std::shared_ptr<monero_tx_set> tx_set = std::make_shared<monero_tx_set>();
tx_set->m_txs = txs;
for (int i = 0; i < txs.size(); i++) txs[i]->m_tx_set = tx_set;
if (!multisig_tx_hex.empty()) tx_set->m_multisig_tx_hex = multisig_tx_hex;
if (!unsigned_tx_hex.empty()) tx_set->m_unsigned_tx_hex = unsigned_tx_hex;
return txs;
}
std::shared_ptr<monero_tx_wallet> monero_wallet_full::sweep_output(const monero_tx_config& config) {
MTRACE("sweep_output()");
//MTRACE("monero_tx_config: " << config.serialize());
// validate input config
std::vector<std::shared_ptr<monero_destination>> destinations = config.get_normalized_destinations();
if (config.m_key_image == boost::none || config.m_key_image.get().empty()) throw std::runtime_error("Must provide key image of output to sweep");
if (destinations.size() != 1 || destinations[0]->m_address == boost::none || destinations[0]->m_address.get().empty()) throw std::runtime_error("Must provide exactly one destination address to sweep output to");
// validate the transfer queried and populate dsts & extra
std::string m_payment_id = config.m_payment_id == boost::none ? std::string("") : config.m_payment_id.get();
std::vector<cryptonote::tx_destination_entry> dsts;
std::vector<uint8_t> extra;
std::list<wallet_rpc::transfer_destination> destination;
destination.push_back(wallet_rpc::transfer_destination());
destination.back().amount = 0;
destination.back().address = destinations[0]->m_address.get();
epee::json_rpc::error err;
if (!validate_transfer(m_w2.get(), destination, m_payment_id, dsts, extra, true, err)) {
//throw std::runtime_error(er); // TODO
throw std::runtime_error("Handle validate_transfer error!");
}
// validate key image
crypto::key_image ki;
if (!epee::string_tools::hex_to_pod(config.m_key_image.get(), ki)) {
throw std::runtime_error("failed to parse key image");
}
// create transaction
uint64_t mixin = m_w2->adjust_mixin(0);
uint32_t priority = m_w2->adjust_priority(config.m_priority == boost::none ? 0 : config.m_priority.get());
uint64_t unlock_height = config.m_unlock_height == boost::none ? 0 : config.m_unlock_height.get();
std::vector<wallet2::pending_tx> ptx_vector = m_w2->create_transactions_single(ki, dsts[0].addr, dsts[0].is_subaddress, 1, mixin, unlock_height, priority, extra);
// validate created transaction
if (ptx_vector.empty()) throw std::runtime_error("No outputs found");
if (ptx_vector.size() > 1) throw std::runtime_error("Multiple transactions are created, which is not supposed to happen");
const wallet2::pending_tx &ptx = ptx_vector[0];
if (ptx.selected_transfers.size() > 1) throw std::runtime_error("The transaction uses multiple inputs, which is not supposed to happen");
// config for fill_response()
bool get_tx_keys = true;
bool get_tx_hex = true;
bool get_tx_metadata = true;
bool relay = config.m_relay != boost::none && config.m_relay.get();
// commit txs (if relaying) and get response using wallet rpc's fill_response()
std::list<std::string> tx_keys;
std::list<uint64_t> tx_amounts;
std::list<uint64_t> tx_fees;
std::list<uint64_t> tx_weights;
std::string multisig_tx_hex;
std::string unsigned_tx_hex;
std::list<std::string> tx_hashes;
std::list<std::string> tx_blobs;
std::list<std::string> tx_metadatas;
std::list<key_image_list> input_key_images_list;
if (!fill_response(m_w2.get(), ptx_vector, get_tx_keys, tx_keys, tx_amounts, tx_fees, tx_weights, multisig_tx_hex, unsigned_tx_hex, !relay, tx_hashes, get_tx_hex, tx_blobs, get_tx_metadata, tx_metadatas, input_key_images_list, err)) {
throw std::runtime_error("need to handle error filling response!"); // TODO: return err message
}
// build sent txs from results // TODO: use common utility with create_txs() to avoid code duplication
std::vector<std::shared_ptr<monero_tx_wallet>> txs;
auto tx_hashes_iter = tx_hashes.begin();
auto tx_keys_iter = tx_keys.begin();
auto tx_amounts_iter = tx_amounts.begin();
auto tx_fees_iter = tx_fees.begin();
auto tx_weights_iter = tx_weights.begin();
auto tx_blobs_iter = tx_blobs.begin();
auto tx_metadatas_iter = tx_metadatas.begin();
auto input_key_images_list_iter = input_key_images_list.begin();
while (tx_fees_iter != tx_fees.end()) {
// init tx with outgoing transfer from filled values
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
txs.push_back(tx);
tx->m_hash = *tx_hashes_iter;
tx->m_is_outgoing = true;
tx->m_key = *tx_keys_iter;
tx->m_fee = *tx_fees_iter;
tx->m_weight = *tx_weights_iter;
tx->m_full_hex = *tx_blobs_iter;
tx->m_metadata = *tx_metadatas_iter;
std::shared_ptr<monero_outgoing_transfer> out_transfer = std::make_shared<monero_outgoing_transfer>();
tx->m_outgoing_transfer = out_transfer;
out_transfer->m_amount = *tx_amounts_iter;
// init inputs with key images
std::list<std::string> input_key_images = (*input_key_images_list_iter).key_images;
for (const std::string& input_key_image : input_key_images) {
std::shared_ptr<monero_output_wallet> input = std::make_shared<monero_output_wallet>();
input->m_tx = tx;
tx->m_inputs.push_back(input);
input->m_key_image = std::make_shared<monero_key_image>();
input->m_key_image.get()->m_hex = input_key_image;
}
// init other known fields
tx->m_payment_id = config.m_payment_id;
tx->m_is_confirmed = false;
tx->m_is_miner_tx = false;
tx->m_is_failed = false; // TODO: test and handle if true
tx->m_relay = relay;
tx->m_is_relayed = relay;
tx->m_in_tx_pool = relay;
if (!tx->m_is_failed.get() && tx->m_is_relayed.get()) tx->m_is_double_spend_seen = false; // TODO: test and handle if true
tx->m_num_confirmations = 0;
tx->m_ring_size = monero_utils::RING_SIZE;
tx->m_unlock_height = config.m_unlock_height == boost::none ? 0 : config.m_unlock_height.get();
tx->m_is_locked = true;
if (tx->m_is_relayed.get()) tx->m_last_relayed_timestamp = static_cast<uint64_t>(time(NULL)); // set last relayed timestamp to current time iff relayed // TODO monero-project: this should be encapsulated in wallet2
out_transfer->m_account_index = config.m_account_index;
if (config.m_subaddress_indices.size() == 1) out_transfer->m_subaddress_indices.push_back(config.m_subaddress_indices[0]); // subaddress index is known iff 1 requested // TODO: get all known subaddress indices here
out_transfer->m_destinations = destinations;
out_transfer->m_destinations[0]->m_amount = *tx_amounts_iter;
// iterate to next element
tx_keys_iter++;
tx_amounts_iter++;
tx_fees_iter++;
tx_hashes_iter++;
tx_blobs_iter++;
tx_metadatas_iter++;
input_key_images_list_iter++;
}
// validate one transaction
if (txs.size() != 1) throw std::runtime_error("Expected 1 transaction but was " + boost::lexical_cast<std::string>(txs.size()));
// link tx set
std::shared_ptr<monero_tx_set> tx_set = std::make_shared<monero_tx_set>();
tx_set->m_txs = txs;
txs[0]->m_tx_set = tx_set;
if (!multisig_tx_hex.empty()) tx_set->m_multisig_tx_hex = multisig_tx_hex;
if (!unsigned_tx_hex.empty()) tx_set->m_unsigned_tx_hex = unsigned_tx_hex;
// notify listeners of spent funds
if (relay) m_w2_listener->on_spend_txs(txs);
return txs[0];
}
std::vector<std::shared_ptr<monero_tx_wallet>> monero_wallet_full::sweep_dust(bool relay) {
MTRACE("monero_wallet_full::sweep_dust()");
// create transaction to fill
std::vector<wallet2::pending_tx> ptx_vector = m_w2->create_unmixable_sweep_transactions();
// config for fill_response
bool get_tx_keys = true;
bool get_tx_hex = true;
bool get_tx_metadata = true;
// commit txs (if relaying) and get response using wallet rpc's fill_response()
std::list<std::string> tx_keys;
std::list<uint64_t> tx_amounts;
std::list<uint64_t> tx_fees;
std::list<uint64_t> tx_weights;
std::string multisig_tx_hex;
std::string unsigned_tx_hex;
std::list<std::string> tx_hashes;
std::list<std::string> tx_blobs;
std::list<std::string> tx_metadatas;
std::list<key_image_list> input_key_images_list;
epee::json_rpc::error er;
if (!fill_response(m_w2.get(), ptx_vector, get_tx_keys, tx_keys, tx_amounts, tx_fees, tx_weights, multisig_tx_hex, unsigned_tx_hex, !relay, tx_hashes, get_tx_hex, tx_blobs, get_tx_metadata, tx_metadatas, input_key_images_list, er)) {
throw std::runtime_error("need to handle error filling response!"); // TODO: return err message
}
// build sent txs from results // TODO: use common utility with create_txs() to avoid code duplication
std::vector<std::shared_ptr<monero_tx_wallet>> txs;
auto tx_hashes_iter = tx_hashes.begin();
auto tx_keys_iter = tx_keys.begin();
auto tx_amounts_iter = tx_amounts.begin();
auto tx_fees_iter = tx_fees.begin();
auto tx_weights_iter = tx_weights.begin();
auto tx_blobs_iter = tx_blobs.begin();
auto tx_metadatas_iter = tx_metadatas.begin();
auto input_key_images_list_iter = input_key_images_list.begin();
while (tx_fees_iter != tx_fees.end()) {
// init tx with outgoing transfer from filled values
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
txs.push_back(tx);
tx->m_hash = *tx_hashes_iter;
tx->m_is_outgoing = true;
tx->m_key = *tx_keys_iter;
tx->m_fee = *tx_fees_iter;
tx->m_weight = *tx_weights_iter;
tx->m_full_hex = *tx_blobs_iter;
tx->m_metadata = *tx_metadatas_iter;
std::shared_ptr<monero_outgoing_transfer> out_transfer = std::make_shared<monero_outgoing_transfer>();
tx->m_outgoing_transfer = out_transfer;
out_transfer->m_amount = *tx_amounts_iter;
// init inputs with key images
std::list<std::string> input_key_images = (*input_key_images_list_iter).key_images;
for (const std::string& input_key_image : input_key_images) {
std::shared_ptr<monero_output_wallet> input = std::make_shared<monero_output_wallet>();
input->m_tx = tx;
tx->m_inputs.push_back(input);
input->m_key_image = std::make_shared<monero_key_image>();
input->m_key_image.get()->m_hex = input_key_image;
}
// init other known fields
tx->m_is_confirmed = false;
tx->m_is_miner_tx = false;
tx->m_is_failed = false; // TODO: test and handle if true
tx->m_relay = relay;
tx->m_is_relayed = relay;
tx->m_in_tx_pool = relay;
if (!tx->m_is_failed.get() && tx->m_is_relayed.get()) tx->m_is_double_spend_seen = false; // TODO: test and handle if true
tx->m_num_confirmations = 0;
tx->m_ring_size = monero_utils::RING_SIZE;
tx->m_unlock_height = 0;
if (tx->m_is_relayed.get()) tx->m_last_relayed_timestamp = static_cast<uint64_t>(time(NULL)); // set last relayed timestamp to current time iff relayed // TODO monero-project: this should be encapsulated in wallet2
out_transfer->m_destinations[0]->m_amount = *tx_amounts_iter;
// iterate to next element
tx_keys_iter++;
tx_amounts_iter++;
tx_fees_iter++;
tx_hashes_iter++;
tx_blobs_iter++;
tx_metadatas_iter++;
input_key_images_list_iter++;
}
// link tx set
std::shared_ptr<monero_tx_set> tx_set = std::make_shared<monero_tx_set>();
tx_set->m_txs = txs;
for (int i = 0; i < txs.size(); i++) txs[i]->m_tx_set = tx_set;
if (!multisig_tx_hex.empty()) tx_set->m_multisig_tx_hex = multisig_tx_hex;
if (!unsigned_tx_hex.empty()) tx_set->m_unsigned_tx_hex = unsigned_tx_hex;
// notify listeners of spent funds
if (relay) m_w2_listener->on_spend_txs(txs);
return txs;
}
std::vector<std::string> monero_wallet_full::relay_txs(const std::vector<std::string>& tx_metadatas) {
MTRACE("relay_txs()");
// relay each metadata as a tx
std::vector<std::string> tx_hashes;
for (const auto& txMetadata : tx_metadatas) {
// parse tx metadata hex
cryptonote::blobdata blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(txMetadata, blob)) {
throw std::runtime_error("Failed to parse hex");
}
// deserialize tx
bool loaded = false;
tools::wallet2::pending_tx ptx;
try {
binary_archive<false> ar{epee::strspan<std::uint8_t>(blob)};
if (::serialization::serialize(ar, ptx)) loaded = true;
} catch (...) {}
if (!loaded) {
try {
std::istringstream iss(blob);
boost::archive::portable_binary_iarchive ar(iss);
ar >> ptx;
loaded = true;
} catch (...) {}
}
if (!loaded) throw std::runtime_error("Failed to parse tx metadata");
// commit tx
try {
m_w2->commit_tx(ptx);
} catch (const std::exception& e) {
throw std::runtime_error("Failed to commit tx");
}
// collect resulting hash
tx_hashes.push_back(epee::string_tools::pod_to_hex(cryptonote::get_transaction_hash(ptx.tx)));
}
// notify listeners of spent funds
m_w2_listener->on_spend_tx_hashes(tx_hashes);
// return relayed tx hashes
return tx_hashes;
}
monero_tx_set monero_wallet_full::describe_tx_set(const monero_tx_set& tx_set) {
// get unsigned and multisig tx sets
std::string unsigned_tx_hex = tx_set.m_unsigned_tx_hex == boost::none ? "" : tx_set.m_unsigned_tx_hex.get();
std::string multisig_tx_hex = tx_set.m_multisig_tx_hex == boost::none ? "" : tx_set.m_multisig_tx_hex.get();
// validate request
if (m_w2->key_on_device()) throw std::runtime_error("command not supported by HW wallet");
if (m_w2->watch_only()) throw std::runtime_error("command not supported by view-only wallet");
if (unsigned_tx_hex.empty() && multisig_tx_hex.empty()) throw std::runtime_error("no txset provided");
std::vector <wallet2::tx_construction_data> tx_constructions;
if (!unsigned_tx_hex.empty()) {
try {
tools::wallet2::unsigned_tx_set exported_txs;
cryptonote::blobdata blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(unsigned_tx_hex, blob)) throw std::runtime_error("Failed to parse hex.");
if (!m_w2->parse_unsigned_tx_from_str(blob, exported_txs)) throw std::runtime_error("cannot load unsigned_txset");
tx_constructions = exported_txs.txes;
}
catch (const std::exception &e) {
throw std::runtime_error("failed to parse unsigned transfers: " + std::string(e.what()));
}
} else if (!multisig_tx_hex.empty()) {
try {
tools::wallet2::multisig_tx_set exported_txs;
cryptonote::blobdata blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(multisig_tx_hex, blob)) throw std::runtime_error("Failed to parse hex.");
if (!m_w2->parse_multisig_tx_from_str(blob, exported_txs)) throw std::runtime_error("cannot load multisig_txset");
for (uint64_t n = 0; n < exported_txs.m_ptx.size(); ++n) {
tx_constructions.push_back(exported_txs.m_ptx[n].construction_data);
}
}
catch (const std::exception &e) {
throw std::runtime_error("failed to parse multisig transfers: " + std::string(e.what()));
}
}
std::vector<tools::wallet2::pending_tx> ptx; // TODO wallet_rpc_server: unused variable
try {
// gather info for each tx
std::vector<std::shared_ptr<monero_tx_wallet>> txs;
std::unordered_map<cryptonote::account_public_address, std::pair<std::string, uint64_t>> dests;
int first_known_non_zero_change_index = -1;
for (int64_t n = 0; n < tx_constructions.size(); ++n)
{
// pre-initialize tx
std::shared_ptr<monero_tx_wallet> tx = std::make_shared<monero_tx_wallet>();
tx->m_is_outgoing = true;
tx->m_input_sum = 0;
tx->m_output_sum = 0;
tx->m_change_amount = 0;
tx->m_num_dummy_outputs = 0;
tx->m_ring_size = std::numeric_limits<uint32_t>::max(); // smaller ring sizes will overwrite
const tools::wallet2::tx_construction_data &cd = tx_constructions[n];
std::vector<cryptonote::tx_extra_field> tx_extra_fields;
bool has_encrypted_payment_id = false;
crypto::hash8 payment_id8 = crypto::null_hash8;
if (cryptonote::parse_tx_extra(cd.extra, tx_extra_fields))
{
cryptonote::tx_extra_nonce extra_nonce;
if (find_tx_extra_field_by_type(tx_extra_fields, extra_nonce))
{
crypto::hash payment_id;
if(cryptonote::get_encrypted_payment_id_from_tx_extra_nonce(extra_nonce.nonce, payment_id8))
{
if (payment_id8 != crypto::null_hash8)
{
tx->m_payment_id = epee::string_tools::pod_to_hex(payment_id8);
has_encrypted_payment_id = true;
}
}
else if (cryptonote::get_payment_id_from_tx_extra_nonce(extra_nonce.nonce, payment_id))
{
tx->m_payment_id = epee::string_tools::pod_to_hex(payment_id);
}
}
}
for (uint64_t s = 0; s < cd.sources.size(); ++s)
{
tx->m_input_sum = tx->m_input_sum.get() + cd.sources[s].amount;
uint64_t ring_size = cd.sources[s].outputs.size();
if (ring_size < tx->m_ring_size.get())
tx->m_ring_size = ring_size;
}
for (uint64_t d = 0; d < cd.splitted_dsts.size(); ++d)
{
const cryptonote::tx_destination_entry &entry = cd.splitted_dsts[d];
std::string address = cryptonote::get_account_address_as_str(m_w2->nettype(), entry.is_subaddress, entry.addr);
if (has_encrypted_payment_id && !entry.is_subaddress && address != entry.original)
address = cryptonote::get_account_integrated_address_as_str(m_w2->nettype(), entry.addr, payment_id8);
auto i = dests.find(entry.addr);
if (i == dests.end())
dests.insert(std::make_pair(entry.addr, std::make_pair(address, entry.amount)));
else
i->second.second += entry.amount;
tx->m_output_sum = tx->m_output_sum.get() + entry.amount;
}
if (cd.change_dts.amount > 0)
{
auto it = dests.find(cd.change_dts.addr);
if (it == dests.end()) throw std::runtime_error("Claimed change does not go to a paid address");
if (it->second.second < cd.change_dts.amount) throw std::runtime_error("Claimed change is larger than payment to the change address");
if (cd.change_dts.amount > 0)
{
if (first_known_non_zero_change_index == -1)
first_known_non_zero_change_index = n;
const tools::wallet2::tx_construction_data &cdn = tx_constructions[first_known_non_zero_change_index];
if (memcmp(&cd.change_dts.addr, &cdn.change_dts.addr, sizeof(cd.change_dts.addr))) throw std::runtime_error("Change goes to more than one address");
}
tx->m_change_amount = tx->m_change_amount.get() + cd.change_dts.amount;
it->second.second -= cd.change_dts.amount;
if (it->second.second == 0)
dests.erase(cd.change_dts.addr);
}
tx->m_outgoing_transfer = std::make_shared<monero_outgoing_transfer>();
uint64_t n_dummy_outputs = 0;
for (auto i = dests.begin(); i != dests.end(); )
{
if (i->second.second > 0)
{
std::shared_ptr<monero_destination> destination = std::make_shared<monero_destination>();
destination->m_address = i->second.first;
destination->m_amount = i->second.second;
tx->m_outgoing_transfer.get()->m_destinations.push_back(destination);
}
else
tx->m_num_dummy_outputs = tx->m_num_dummy_outputs.get() + 1;
++i;
}
if (tx->m_change_amount.get() > 0)
{
const tools::wallet2::tx_construction_data &cd0 = tx_constructions[0];
tx->m_change_address = get_account_address_as_str(m_w2->nettype(), cd0.subaddr_account > 0, cd0.change_dts.addr);
}
tx->m_fee = tx->m_input_sum.get() - tx->m_output_sum.get();
tx->m_unlock_height = cd.unlock_time;
tx->m_extra_hex = epee::to_hex::string({cd.extra.data(), cd.extra.size()});
txs.push_back(tx);
}
// build and return tx set
monero_tx_set tx_set;
tx_set.m_txs = txs;
return tx_set;
}
catch (const std::exception &e)
{
throw std::runtime_error("failed to parse unsigned transfers");
}
}
// implementation based on monero-project wallet_rpc_server.cpp::on_sign_transfer()
std::string monero_wallet_full::sign_txs(const std::string& unsigned_tx_hex) {
if (m_w2->key_on_device()) throw std::runtime_error("command not supported by HW wallet");
if (m_w2->watch_only()) throw std::runtime_error("command not supported by view-only wallet");
cryptonote::blobdata blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(unsigned_tx_hex, blob)) throw std::runtime_error("Failed to parse hex.");
tools::wallet2::unsigned_tx_set exported_txs;
if(!m_w2->parse_unsigned_tx_from_str(blob, exported_txs)) throw std::runtime_error("cannot load unsigned_txset");
std::vector<tools::wallet2::pending_tx> ptxs;
try {
tools::wallet2::signed_tx_set signed_txs;
std::string ciphertext = m_w2->sign_tx_dump_to_str(exported_txs, ptxs, signed_txs);
if (ciphertext.empty()) throw std::runtime_error("Failed to sign unsigned tx");
return epee::string_tools::buff_to_hex_nodelimer(ciphertext);
} catch (const std::exception &e) {
throw std::runtime_error(std::string("Failed to sign unsigned tx: ") + e.what());
}
}
// implementation based on monero-project wallet_rpc_server.cpp::on_submit_transfer()
std::vector<std::string> monero_wallet_full::submit_txs(const std::string& signed_tx_hex) {
if (m_w2->key_on_device()) throw std::runtime_error("command not supported by HW wallet");
cryptonote::blobdata blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(signed_tx_hex, blob)) throw std::runtime_error("Failed to parse hex.");
std::vector<tools::wallet2::pending_tx> ptx_vector;
try {
if (!m_w2->parse_tx_from_str(blob, ptx_vector, NULL)) throw std::runtime_error("Failed to parse signed tx data.");
} catch (const std::exception &e) {
throw std::runtime_error(std::string("Failed to parse signed tx: ") + e.what());
}
try {
std::vector<std::string> tx_hashes;
for (auto &ptx: ptx_vector) {
m_w2->commit_tx(ptx);
tx_hashes.push_back(epee::string_tools::pod_to_hex(cryptonote::get_transaction_hash(ptx.tx)));
}
m_w2_listener->on_spend_tx_hashes(tx_hashes); // notify listeners of spent funds
return tx_hashes;
} catch (const std::exception &e) {
throw std::runtime_error(std::string("Failed to submit signed tx: ") + e.what());
}
}
std::string monero_wallet_full::sign_message(const std::string& msg, monero_message_signature_type signature_type, uint32_t account_idx, uint32_t subaddress_idx) const {
cryptonote::subaddress_index index = {account_idx, subaddress_idx};
tools::wallet2::message_signature_type_t signature_type_w2 = signature_type == monero_message_signature_type::SIGN_WITH_SPEND_KEY ? tools::wallet2::message_signature_type_t::sign_with_spend_key : tools::wallet2::message_signature_type_t::sign_with_view_key;
return m_w2->sign(msg, signature_type_w2, index);
}
monero_message_signature_result monero_wallet_full::verify_message(const std::string& msg, const std::string& address, const std::string& signature) const {
// validate and parse address or url
cryptonote::address_parse_info info;
std::string err = "Invalid address";
if (!get_account_address_from_str_or_url(info, m_w2->nettype(), address,
[&err](const std::string &url, const std::vector<std::string> &addresses, bool dnssec_valid)->std::string {
if (!dnssec_valid) {
err = std::string("Invalid DNSSEC for ") + url;
return {};
}
if (addresses.empty()) {
err = std::string("No Monero address found at ") + url;
return {};
}
return addresses[0];
}))
{
throw std::runtime_error(err);
}
// verify message
tools:wallet2::message_signature_result_t result_w2 = m_w2->verify(msg, info.address, signature);
// translate result
monero_message_signature_result result;
result.m_is_good = result_w2.valid;
result.m_is_old = result_w2.old;
result.m_signature_type = result_w2.type == tools::wallet2::message_signature_type_t::sign_with_spend_key ? monero_message_signature_type::SIGN_WITH_SPEND_KEY : monero_message_signature_type::SIGN_WITH_VIEW_KEY;
result.m_version = result_w2.version;
return result;
}
std::string monero_wallet_full::get_tx_key(const std::string& tx_hash) const {
MTRACE("monero_wallet_full::get_tx_key()");
// validate and parse tx hash
crypto::hash _tx_hash;
if (!epee::string_tools::hex_to_pod(tx_hash, _tx_hash)) {
throw std::runtime_error("TX hash has invalid format");
}
// get tx key and additional keys
crypto::secret_key _tx_key;
std::vector<crypto::secret_key> additional_tx_keys;
if (!m_w2->get_tx_key(_tx_hash, _tx_key, additional_tx_keys)) {
throw std::runtime_error("No tx secret key is stored for this tx");
}
// build and return tx key with additional keys
epee::wipeable_string s;
s += epee::to_hex::wipeable_string(_tx_key);
for (uint64_t i = 0; i < additional_tx_keys.size(); ++i) {
s += epee::to_hex::wipeable_string(additional_tx_keys[i]);
}
return std::string(s.data(), s.size());
}
std::shared_ptr<monero_check_tx> monero_wallet_full::check_tx_key(const std::string& tx_hash, const std::string& tx_key, const std::string& address) const {
MTRACE("monero_wallet_full::check_tx_key()");
// validate and parse tx hash
crypto::hash _tx_hash;
if (!epee::string_tools::hex_to_pod(tx_hash, _tx_hash)) {
throw std::runtime_error("TX hash has invalid format");
}
// validate and parse tx key
epee::wipeable_string tx_key_str = tx_key;
if (tx_key_str.size() < 64 || tx_key_str.size() % 64) {
throw std::runtime_error("Tx key has invalid format");
}
const char *data = tx_key_str.data();
crypto::secret_key _tx_key;
if (!epee::wipeable_string(data, 64).hex_to_pod(unwrap(unwrap(_tx_key)))) {
throw std::runtime_error("Tx key has invalid format");
}
// get additional keys
uint64_t offset = 64;
std::vector<crypto::secret_key> additional_tx_keys;
while (offset < tx_key_str.size()) {
additional_tx_keys.resize(additional_tx_keys.size() + 1);
if (!epee::wipeable_string(data + offset, 64).hex_to_pod(unwrap(unwrap(additional_tx_keys.back())))) {
throw std::runtime_error("Tx key has invalid format");
}
offset += 64;
}
// validate and parse address
cryptonote::address_parse_info info;
if (!get_account_address_from_str(info, m_w2->nettype(), address)) {
throw std::runtime_error("Invalid address");
}
// initialize and return tx check using wallet2
uint64_t received_amount;
bool in_tx_pool;
uint64_t num_confirmations;
m_w2->check_tx_key(_tx_hash, _tx_key, additional_tx_keys, info.address, received_amount, in_tx_pool, num_confirmations);
std::shared_ptr<monero_check_tx> check_tx = std::make_shared<monero_check_tx>();
check_tx->m_is_good = true; // check is good if we get this far
check_tx->m_received_amount = received_amount;
check_tx->m_in_tx_pool = in_tx_pool;
check_tx->m_num_confirmations = num_confirmations;
return check_tx;
}
std::string monero_wallet_full::get_tx_proof(const std::string& tx_hash, const std::string& address, const std::string& message) const {
// validate and parse tx hash
crypto::hash _tx_hash;
if (!epee::string_tools::hex_to_pod(tx_hash, _tx_hash)) {
throw std::runtime_error("TX hash has invalid format");
}
// validate and parse address
cryptonote::address_parse_info info;
if (!get_account_address_from_str(info, m_w2->nettype(), address)) {
throw std::runtime_error("Invalid address");
}
// get tx proof
return m_w2->get_tx_proof(_tx_hash, info.address, info.is_subaddress, message);
}
std::shared_ptr<monero_check_tx> monero_wallet_full::check_tx_proof(const std::string& tx_hash, const std::string& address, const std::string& message, const std::string& signature) const {
MTRACE("monero_wallet_full::check_tx_proof()");
// validate and parse tx hash
crypto::hash _tx_hash;
if (!epee::string_tools::hex_to_pod(tx_hash, _tx_hash)) {
throw std::runtime_error("TX hash has invalid format");
}
// validate and parse address
cryptonote::address_parse_info info;
if (!get_account_address_from_str(info, m_w2->nettype(), address)) {
throw std::runtime_error("Invalid address");
}
// validate signature
if (signature.empty()) throw std::runtime_error("Must provide signature to check tx proof");
// initialize and return tx check using wallet2
std::shared_ptr<monero_check_tx> check_tx = std::make_shared<monero_check_tx>();
uint64_t received_amount;
bool in_tx_pool;
uint64_t num_confirmations;
check_tx->m_is_good = m_w2->check_tx_proof(_tx_hash, info.address, info.is_subaddress, message, signature, received_amount, in_tx_pool, num_confirmations);
if (check_tx->m_is_good) {
check_tx->m_received_amount = received_amount;
check_tx->m_in_tx_pool = in_tx_pool;
check_tx->m_num_confirmations = num_confirmations;
}
return check_tx;
}
std::string monero_wallet_full::get_spend_proof(const std::string& tx_hash, const std::string& message) const {
MTRACE("monero_wallet_full::get_spend_proof()");
// validate and parse tx hash
crypto::hash _tx_hash;
if (!epee::string_tools::hex_to_pod(tx_hash, _tx_hash)) {
throw std::runtime_error("TX hash has invalid format");
}
// return spend proof signature
return m_w2->get_spend_proof(_tx_hash, message);
}
bool monero_wallet_full::check_spend_proof(const std::string& tx_hash, const std::string& message, const std::string& signature) const {
MTRACE("monero_wallet_full::check_spend_proof()");
// validate and parse tx hash
crypto::hash _tx_hash;
if (!epee::string_tools::hex_to_pod(tx_hash, _tx_hash)) {
throw std::runtime_error("TX hash has invalid format");
}
// validate signature
if (signature.empty()) throw std::runtime_error("Must provide signature to check spend proof");
// check spend proof
return m_w2->check_spend_proof(_tx_hash, message, signature);
}
std::string monero_wallet_full::get_reserve_proof_wallet(const std::string& message) const {
MTRACE("monero_wallet_full::get_reserve_proof_wallet()");
boost::optional<std::pair<uint32_t, uint64_t>> account_minreserve;
return m_w2->get_reserve_proof(account_minreserve, message);
}
std::string monero_wallet_full::get_reserve_proof_account(uint32_t account_idx, uint64_t amount, const std::string& message) const {
MTRACE("monero_wallet_full::get_reserve_proof_account()");
boost::optional<std::pair<uint32_t, uint64_t>> account_minreserve;
if (account_idx >= m_w2->get_num_subaddress_accounts()) throw std::runtime_error("Account index is out of bound");
account_minreserve = std::make_pair(account_idx, amount);
return m_w2->get_reserve_proof(account_minreserve, message);
}
std::shared_ptr<monero_check_reserve> monero_wallet_full::check_reserve_proof(const std::string& address, const std::string& message, const std::string& signature) const {
MTRACE("monero_wallet_full::check_reserve_proof()");
// validate and parse input
cryptonote::address_parse_info info;
if (!get_account_address_from_str(info, m_w2->nettype(), address)) throw std::runtime_error("Invalid address");
if (info.is_subaddress) throw std::runtime_error("Address must not be a subaddress");
if (signature.empty()) throw std::runtime_error("Must provide signature to check reserve proof");
// initialize check reserve using wallet2
std::shared_ptr<monero_check_reserve> reserve_check = std::make_shared<monero_check_reserve>();
uint64_t total_amount;
uint64_t unconfirmed_spent_amount;
reserve_check->m_is_good = m_w2->check_reserve_proof(info.address, message, signature, total_amount, unconfirmed_spent_amount);
if (reserve_check->m_is_good) {
reserve_check->m_total_amount = total_amount;
reserve_check->m_unconfirmed_spent_amount = unconfirmed_spent_amount;
}
return reserve_check;
}
std::string monero_wallet_full::get_tx_note(const std::string& tx_hash) const {
MTRACE("monero_wallet_full::get_tx_note()");
cryptonote::blobdata tx_blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(tx_hash, tx_blob) || tx_blob.size() != sizeof(crypto::hash)) {
throw std::runtime_error("TX hash has invalid format");
}
crypto::hash _tx_hash = *reinterpret_cast<const crypto::hash*>(tx_blob.data());
return m_w2->get_tx_note(_tx_hash);
}
std::vector<std::string> monero_wallet_full::get_tx_notes(const std::vector<std::string>& tx_hashes) const {
MTRACE("monero_wallet_full::get_tx_notes()");
std::vector<std::string> notes;
for (const auto& tx_hash : tx_hashes) notes.push_back(get_tx_note(tx_hash));
return notes;
}
void monero_wallet_full::set_tx_note(const std::string& tx_hash, const std::string& note) {
MTRACE("monero_wallet_full::set_tx_note()");
cryptonote::blobdata tx_blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(tx_hash, tx_blob) || tx_blob.size() != sizeof(crypto::hash)) {
throw std::runtime_error("TX hash has invalid format");
}
crypto::hash _tx_hash = *reinterpret_cast<const crypto::hash*>(tx_blob.data());
m_w2->set_tx_note(_tx_hash, note);
}
void monero_wallet_full::set_tx_notes(const std::vector<std::string>& tx_hashes, const std::vector<std::string>& notes) {
MTRACE("monero_wallet_full::set_tx_notes()");
if (tx_hashes.size() != notes.size()) throw std::runtime_error("Different amount of txids and notes");
for (int i = 0; i < tx_hashes.size(); i++) {
set_tx_note(tx_hashes[i], notes[i]);
}
}
std::vector<monero_address_book_entry> monero_wallet_full::get_address_book_entries(const std::vector<uint64_t>& indices) const {
MTRACE("monero_wallet_full::get_address_book_entries()");
// get wallet2 address book entries
const auto w2_entries = m_w2->get_address_book();
// build entries from wallet2 types
std::vector<monero_address_book_entry> entries;
if (indices.empty()) {
uint64_t idx = 0;
for (const auto &w2_entry: w2_entries) {
std::string address;
if (w2_entry.m_has_payment_id) address = cryptonote::get_account_integrated_address_as_str(m_w2->nettype(), w2_entry.m_address, w2_entry.m_payment_id);
else address = get_account_address_as_str(m_w2->nettype(), w2_entry.m_is_subaddress, w2_entry.m_address);
monero_address_book_entry entry(idx++, address, w2_entry.m_description);
entries.push_back(entry);
}
} else {
for (uint64_t idx : indices) {
if (idx >= w2_entries.size()) throw std::runtime_error("Index out of range: " + std::to_string(idx));
const auto &w2_entry = w2_entries[idx];
std::string address;
if (w2_entry.m_has_payment_id) address = cryptonote::get_account_integrated_address_as_str(m_w2->nettype(), w2_entry.m_address, w2_entry.m_payment_id);
else address = get_account_address_as_str(m_w2->nettype(), w2_entry.m_is_subaddress, w2_entry.m_address);
monero_address_book_entry entry(idx++, address, w2_entry.m_description);
entries.push_back(entry);
}
}
return entries;
}
uint64_t monero_wallet_full::add_address_book_entry(const std::string& address, const std::string& description) {
MTRACE("add_address_book_entry()");
cryptonote::address_parse_info info;
epee::json_rpc::error er;
if(!get_account_address_from_str_or_url(info, m_w2->nettype(), address,
[&er](const std::string &url, const std::vector<std::string> &addresses, bool dnssec_valid)->std::string {
if (!dnssec_valid) throw std::runtime_error(std::string("Invalid DNSSEC for ") + url);
if (addresses.empty()) throw std::runtime_error(std::string("No Monero address found at ") + url);
return addresses[0];
}))
{
throw std::runtime_error(std::string("Invalid address: ") + address);
}
if (!m_w2->add_address_book_row(info.address, info.has_payment_id ? &info.payment_id : NULL, description, info.is_subaddress)) throw std::runtime_error("Failed to add address book entry");
return m_w2->get_address_book().size() - 1;
}
void monero_wallet_full::edit_address_book_entry(uint64_t index, bool set_address, const std::string& address, bool set_description, const std::string& description) {
MTRACE("edit_address_book_entry()");
const auto ab = m_w2->get_address_book();
if (index >= ab.size()) throw std::runtime_error("Index out of range: " + std::to_string(index));
tools::wallet2::address_book_row entry = ab[index];
cryptonote::address_parse_info info;
epee::json_rpc::error er;
if (set_address) {
er.message = "";
if(!get_account_address_from_str_or_url(info, m_w2->nettype(), address,
[&er](const std::string &url, const std::vector<std::string> &addresses, bool dnssec_valid)->std::string {
if (!dnssec_valid) throw std::runtime_error(std::string("Invalid DNSSEC for ") + url);
if (addresses.empty()) throw std::runtime_error(std::string("No Monero address found at ") + url);
return addresses[0];
}))
{
throw std::runtime_error("Invalid address: " + address);
}
entry.m_address = info.address;
entry.m_is_subaddress = info.is_subaddress;
if (info.has_payment_id) entry.m_payment_id = info.payment_id;
}
if (set_description) entry.m_description = description;
if (!m_w2->set_address_book_row(index, entry.m_address, set_address && entry.m_has_payment_id ? &entry.m_payment_id : NULL, entry.m_description, entry.m_is_subaddress)) {
throw std::runtime_error("Failed to edit address book entry");
}
}
void monero_wallet_full::delete_address_book_entry(uint64_t index) {
const auto w2_entries = m_w2->get_address_book();
if (index >= w2_entries.size()) throw std::runtime_error("Index out of range: " + std::to_string(index));
if (!m_w2->delete_address_book_row(index)) throw std::runtime_error("Failed to delete address book entry");
}
std::string monero_wallet_full::get_payment_uri(const monero_tx_config& config) const {
MTRACE("get_payment_uri()");
// validate config
std::vector<std::shared_ptr<monero_destination>> destinations = config.get_normalized_destinations();
if (destinations.size() != 1) throw std::runtime_error("Cannot make URI from supplied parameters: must provide exactly one destination to send funds");
if (destinations.at(0)->m_address == boost::none) throw std::runtime_error("Cannot make URI from supplied parameters: must provide destination address");
if (destinations.at(0)->m_amount == boost::none) throw std::runtime_error("Cannot make URI from supplied parameters: must provide destination amount");
// prepare wallet2 params
std::string address = destinations.at(0)->m_address.get();
std::string payment_id = config.m_payment_id == boost::none ? "" : config.m_payment_id.get();
uint64_t amount = destinations.at(0)->m_amount.get();
std::string note = config.m_note == boost::none ? "" : config.m_note.get();
std::string m_recipient_name = config.m_recipient_name == boost::none ? "" : config.m_recipient_name.get();
// make uri using wallet2
std::string error;
std::string uri = m_w2->make_uri(address, payment_id, amount, note, m_recipient_name, error);
if (uri.empty()) throw std::runtime_error("Cannot make URI from supplied parameters: " + error);
return uri;
}
std::shared_ptr<monero_tx_config> monero_wallet_full::parse_payment_uri(const std::string& uri) const {
MTRACE("parse_payment_uri(" << uri << ")");
// decode uri to parameters
std::string address;
std::string payment_id;
uint64_t amount = 0;
std::string note;
std::string m_recipient_name;
std::vector<std::string> unknown_parameters;
std::string error;
if (!m_w2->parse_uri(uri, address, payment_id, amount, note, m_recipient_name, unknown_parameters, error)) {
throw std::runtime_error("Error parsing URI: " + error);
}
// initialize config
std::shared_ptr<monero_tx_config> config = std::make_shared<monero_tx_config>();
std::shared_ptr<monero_destination> destination = std::make_shared<monero_destination>();
config->m_destinations.push_back(destination);
if (!address.empty()) destination->m_address = address;
destination->m_amount = amount;
if (!payment_id.empty()) config->m_payment_id = payment_id;
if (!note.empty()) config->m_note = note;
if (!m_recipient_name.empty()) config->m_recipient_name = m_recipient_name;
if (!unknown_parameters.empty()) MWARNING("WARNING in monero_wallet_full::parse_payment_uri: URI contains unknown parameters which are discarded"); // TODO: return unknown parameters?
return config;
}
bool monero_wallet_full::get_attribute(const std::string& key, std::string& value) const {
return m_w2->get_attribute(key, value);
}
void monero_wallet_full::set_attribute(const std::string& key, const std::string& val) {
m_w2->set_attribute(key, val);
}
void monero_wallet_full::start_mining(boost::optional<uint64_t> num_threads, boost::optional<bool> background_mining, boost::optional<bool> ignore_battery) {
MTRACE("start_mining()");
// only mine on trusted daemon
if (!m_w2->is_trusted_daemon()) throw std::runtime_error("This command requires a trusted daemon.");
// set defaults
if (num_threads == boost::none || num_threads.get() == 0) num_threads = 1; // TODO: how to autodetect optimal number of threads which daemon supports?
if (background_mining == boost::none) background_mining = false;
if (ignore_battery == boost::none) ignore_battery = false;
// validate num threads
uint64_t max_mining_threads_count = (std::max)(tools::get_max_concurrency(), static_cast<unsigned>(2));
if (num_threads.get() < 1 || max_mining_threads_count < num_threads.get()) {
throw std::runtime_error("The specified number of threads is inappropriate.");
}
// start mining on daemon
cryptonote::COMMAND_RPC_START_MINING::request daemon_req = AUTO_VAL_INIT(daemon_req);
daemon_req.miner_address = m_w2->get_account().get_public_address_str(m_w2->nettype());
daemon_req.threads_count = num_threads.get();
daemon_req.do_background_mining = background_mining.get();
daemon_req.ignore_battery = ignore_battery.get();
cryptonote::COMMAND_RPC_START_MINING::response daemon_res;
bool r = m_w2->invoke_http_json("/start_mining", daemon_req, daemon_res);
if (!r || daemon_res.status != CORE_RPC_STATUS_OK) {
throw std::runtime_error("Couldn't start mining due to unknown error.");
}
}
void monero_wallet_full::stop_mining() {
MTRACE("stop_mining()");
cryptonote::COMMAND_RPC_STOP_MINING::request daemon_req;
cryptonote::COMMAND_RPC_STOP_MINING::response daemon_res;
bool r = m_w2->invoke_http_json("/stop_mining", daemon_req, daemon_res);
if (!r || daemon_res.status != CORE_RPC_STATUS_OK) {
throw std::runtime_error("Couldn't stop mining due to unknown error.");
}
}
uint64_t monero_wallet_full::wait_for_next_block() {
// use mutex and condition variable to wait for block
boost::mutex temp;
boost::condition_variable cv;
// create listener which notifies condition variable when block is added
struct block_notifier : monero_wallet_listener {
boost::mutex* temp;
boost::condition_variable* cv;
uint64_t last_height;
block_notifier(boost::mutex* temp, boost::condition_variable* cv) { this->temp = temp; this->cv = cv; }
void on_new_block(uint64_t height) {
last_height = height;
cv->notify_one();
}
} block_listener(&temp, &cv);
// register the listener
add_listener(block_listener);
// wait until condition variable is notified
boost::mutex::scoped_lock lock(temp);
cv.wait(lock);
// unregister the listener
remove_listener(block_listener);
// return last height
return block_listener.last_height;
}
bool monero_wallet_full::is_multisig_import_needed() const {
return m_w2->multisig() && m_w2->has_multisig_partial_key_images();
}
monero_multisig_info monero_wallet_full::get_multisig_info() const {
monero_multisig_info info;
info.m_is_multisig = m_w2->multisig(&info.m_is_ready, &info.m_threshold, &info.m_num_participants);
return info;
}
std::string monero_wallet_full::prepare_multisig() {
if (m_w2->multisig()) throw std::runtime_error("This wallet is already multisig");
if (m_w2->watch_only()) throw std::runtime_error("This wallet is view-only and cannot be made multisig");
m_w2->enable_multisig(true);
return m_w2->get_multisig_first_kex_msg();
}
std::string monero_wallet_full::make_multisig(const std::vector<std::string>& multisig_hexes, int threshold, const std::string& password) {
if (m_w2->multisig()) throw std::runtime_error("This wallet is already multisig");
if (m_w2->watch_only()) throw std::runtime_error("This wallet is view-only and cannot be made multisig");
boost::lock_guard<boost::mutex> guarg(m_sync_mutex); // do not refresh while making multisig
return m_w2->make_multisig(epee::wipeable_string(password), multisig_hexes, threshold);
}
monero_multisig_init_result monero_wallet_full::exchange_multisig_keys(const std::vector<std::string>& multisig_hexes, const std::string& password) {
// validate state and args
bool ready;
uint32_t threshold, total;
if (!m_w2->multisig(&ready, &threshold, &total)) throw std::runtime_error("This wallet is not multisig");
if (ready) throw std::runtime_error("This wallet is multisig, and already finalized");
if (multisig_hexes.size() + 1 < total) throw std::runtime_error("Needs multisig info from more participants");
// do not refresh while exchanging multisig keys
boost::lock_guard<boost::mutex> guarg(m_sync_mutex);
// import peer multisig keys and get multisig hex to be shared next round
std::string multisig_hex = m_w2->exchange_multisig_keys(epee::wipeable_string(password), multisig_hexes);
// build and return exchange result
monero_multisig_init_result result;
m_w2->multisig(&ready);
result.m_multisig_hex = multisig_hex;
if (ready) result.m_address = m_w2->get_account().get_public_address_str(m_w2->nettype());
return result;
}
std::string monero_wallet_full::export_multisig_hex() {
bool ready;
if (!m_w2->multisig(&ready)) throw std::runtime_error("This wallet is not multisig");
if (!ready) throw std::runtime_error("This wallet is multisig, but not yet finalized");
return epee::string_tools::buff_to_hex_nodelimer(m_w2->export_multisig());
}
int monero_wallet_full::import_multisig_hex(const std::vector<std::string>& multisig_hexes) {
// validate state and args
bool ready;
uint32_t threshold, total;
if (!m_w2->multisig(&ready, &threshold, &total)) throw std::runtime_error("This wallet is not multisig");
if (!ready) throw std::runtime_error("This wallet is multisig, but not yet finalized");
if (multisig_hexes.size() < threshold - 1) throw std::runtime_error("Needs multisig export info from more participants");
// validate and parse each peer multisig hex
std::vector<cryptonote::blobdata> multisig_blobs;
multisig_blobs.resize(multisig_hexes.size());
for (uint64_t n = 0; n < multisig_hexes.size(); ++n) {
if (!epee::string_tools::parse_hexstr_to_binbuff(multisig_hexes[n], multisig_blobs[n])) {
throw std::runtime_error("Failed to parse hex");
}
}
// import peer multisig hex
int num_outputs = m_w2->import_multisig(multisig_blobs);
// if daemon is trusted, rescan spent
if (is_daemon_trusted()) rescan_spent();
// return the number of outputs signed by the given multisig hex
return num_outputs;
}
monero_multisig_sign_result monero_wallet_full::sign_multisig_tx_hex(const std::string& multisig_tx_hex) {
// validate state and args
bool ready;
uint32_t threshold, total;
if (!m_w2->multisig(&ready, &threshold, &total)) throw std::runtime_error("This wallet is not multisig");
if (!ready) throw std::runtime_error("This wallet is multisig, but not yet finalized");
// validate and parse multisig tx hex as blob
cryptonote::blobdata multisig_tx_blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(multisig_tx_hex, multisig_tx_blob)) {
throw std::runtime_error("Failed to parse hex");
}
// validate and parse blob as multisig_tx_set
tools::wallet2::multisig_tx_set ms_tx_set;
if (!m_w2->load_multisig_tx(multisig_tx_blob, ms_tx_set, NULL)) {
throw std::runtime_error("Failed to parse multisig tx data");
}
// sign multisig txs
bool success = false;
std::vector<crypto::hash> tx_hashes;
try {
success = m_w2->sign_multisig_tx(ms_tx_set, tx_hashes);
} catch (const std::exception& e) {
std::string msg = std::string("Failed to sign multisig tx: ") + e.what();
throw std::runtime_error(msg);
}
if (!success) throw std::runtime_error("Failed to sign multisig tx");
// save multisig txs
std::string signed_multisig_tx_hex = epee::string_tools::buff_to_hex_nodelimer(m_w2->save_multisig_tx(ms_tx_set));
// build sign result
monero_multisig_sign_result result;
result.m_signed_multisig_tx_hex = signed_multisig_tx_hex;
for (const crypto::hash& tx_hash : tx_hashes) {
result.m_tx_hashes.push_back(epee::string_tools::pod_to_hex(tx_hash));
}
return result;
}
std::vector<std::string> monero_wallet_full::submit_multisig_tx_hex(const std::string& signed_multisig_tx_hex) {
// validate state and args
bool ready;
uint32_t threshold, total;
if (!m_w2->multisig(&ready, &threshold, &total)) throw std::runtime_error("This wallet is not multisig");
if (!ready) throw std::runtime_error("This wallet is multisig, but not yet finalized");
// validate signed multisig tx hex as blob
cryptonote::blobdata signed_multisig_tx_blob;
if (!epee::string_tools::parse_hexstr_to_binbuff(signed_multisig_tx_hex, signed_multisig_tx_blob)) {
throw std::runtime_error("Failed to parse hex");
}
// validate and parse blob as multisig_tx_set
tools::wallet2::multisig_tx_set signed_multisig_tx_set;
if (!m_w2->load_multisig_tx(signed_multisig_tx_blob, signed_multisig_tx_set, NULL)) {
throw std::runtime_error("Failed to parse multisig tx data");
}
// ensure sufficient number of participants have signed
if (signed_multisig_tx_set.m_signers.size() < threshold) {
throw std::runtime_error("Not enough signers signed this transaction");
}
// commit the transactions
std::vector<std::string> tx_hashes;
try {
for (auto& pending_tx : signed_multisig_tx_set.m_ptx) {
m_w2->commit_tx(pending_tx);
tx_hashes.push_back(epee::string_tools::pod_to_hex(cryptonote::get_transaction_hash(pending_tx.tx)));
}
} catch (const std::exception& e) {
std::string msg = std::string("Failed to submit multisig tx: ") + e.what();
throw std::runtime_error(msg);
}
// return the resulting tx hashes
return tx_hashes;
}
void monero_wallet_full::change_password(const std::string& old_password, const std::string& new_password) {
MTRACE("change_password(" << "***" << ", ***)");
#if !defined(__EMSCRIPTEN__) // TODO: wallet2 verify_password loads from disk so password is verified in js for wasm
if (!m_w2->verify_password(old_password)) throw std::runtime_error("Invalid original password.");
#endif
m_w2->change_password(m_w2->get_wallet_file(), old_password, new_password);
}
void monero_wallet_full::move_to(const std::string& path, const std::string& password) {
MTRACE("move_to(" << path << ", ***)");
m_w2->store_to(path, password);
}
void monero_wallet_full::save() {
MTRACE("save()");
m_w2->store();
}
std::string monero_wallet_full::get_keys_file_buffer(const epee::wipeable_string& password, bool view_only) const {
boost::optional<wallet2::keys_file_data> keys_file_data = m_w2->get_keys_file_data(password, view_only);
std::string buf;
::serialization::dump_binary(keys_file_data.get(), buf);
return buf;
}
std::string monero_wallet_full::get_cache_file_buffer(const epee::wipeable_string& password) const {
boost::optional<wallet2::cache_file_data> cache_file_data = m_w2->get_cache_file_data(password);
std::string buf;
::serialization::dump_binary(cache_file_data.get(), buf);
return buf;
}
void monero_wallet_full::close(bool save) {
MTRACE("close()");
stop_syncing(); // prevent sync thread from starting again
if (save) this->save();
if (m_sync_loop_running) {
m_sync_cv.notify_one();
std::this_thread::sleep_for(std::chrono::milliseconds(1)); // TODO: in emscripten, m_sync_cv.notify_one() returns without waiting, so sleep; bug in emscripten upstream llvm?
m_syncing_thread.join();
}
m_w2->stop();
m_w2->deinit();
m_w2->callback(nullptr); // unregister listener after sync
m_w2_listener.reset(); // wait for queued notifications
}
// ------------------------------- PRIVATE HELPERS ----------------------------
void monero_wallet_full::init_common() {
MTRACE("monero_wallet_full.cpp init_common()");
// emscripten config
#if defined(__EMSCRIPTEN__)
m_w2->enable_dns(false);
// single-threaded if emscripten pthreads not defined
#if !defined(__EMSCRIPTEN_PTHREADS__)
tools::set_max_concurrency(1); // TODO: single-threaded emscripten tools::get_max_concurrency() correctly returns 1 on Safari but 8 on Chrome which fails in common/threadpool constructor
#endif
#endif
// initialize internal state
m_w2_listener = std::unique_ptr<wallet2_listener>(new wallet2_listener(*this, *m_w2));
if (get_daemon_connection() == boost::none) m_is_connected = false;
m_is_synced = false;
m_rescan_on_sync = false;
m_syncing_enabled = false;
m_sync_loop_running = false;
}
std::vector<std::shared_ptr<monero_transfer>> monero_wallet_full::get_transfers_aux(const monero_transfer_query& query) const {
MTRACE("monero_wallet_full::get_transfers(query)");
// // log query
// if (query.m_tx_query != boost::none) {
// if ((*query.m_tx_query)->m_block == boost::none) std::cout << "Transfer query's tx query rooted at [tx]:" << (*query.m_tx_query)->serialize() << std::endl;
// else std::cout << "Transfer query's tx query rooted at [block]: " << (*(*query.m_tx_query)->m_block)->serialize() << std::endl;
// } else std::cout << "Transfer query: " << query.serialize() << std::endl;
// copy and normalize query
std::shared_ptr<monero_transfer_query> _query;
if (query.m_tx_query == boost::none) {
std::shared_ptr<monero_transfer_query> query_ptr = std::make_shared<monero_transfer_query>(query); // convert to shared pointer for copy // TODO: does this copy unecessarily? copy constructor is not defined
_query = query_ptr->copy(query_ptr, std::make_shared<monero_transfer_query>());
_query->m_tx_query = std::make_shared<monero_tx_query>();
_query->m_tx_query.get()->m_transfer_query = _query;
} else {
std::shared_ptr<monero_tx_query> tx_query = query.m_tx_query.get()->copy(query.m_tx_query.get(), std::make_shared<monero_tx_query>());
_query = tx_query->m_transfer_query.get();
}
std::shared_ptr<monero_tx_query> tx_query = _query->m_tx_query.get();
// build parameters for m_w2->get_payments()
uint64_t min_height = tx_query->m_min_height == boost::none ? 0 : *tx_query->m_min_height;
uint64_t max_height = tx_query->m_max_height == boost::none ? CRYPTONOTE_MAX_BLOCK_NUMBER : std::min((uint64_t) CRYPTONOTE_MAX_BLOCK_NUMBER, *tx_query->m_max_height);
if (min_height > 0) min_height--; // TODO monero-project: wallet2::get_payments() m_min_height is exclusive, so manually offset to match intended range (issues #5751, #5598)
boost::optional<uint32_t> account_index = boost::none;
if (_query->m_account_index != boost::none) account_index = *_query->m_account_index;
std::set<uint32_t> subaddress_indices;
for (int i = 0; i < _query->m_subaddress_indices.size(); i++) {
subaddress_indices.insert(_query->m_subaddress_indices[i]);
}
// translate from monero_tx_query to in, out, pending, pool, failed terminology used by monero-wallet-rpc
bool can_be_confirmed = !bool_equals(false, tx_query->m_is_confirmed) && !bool_equals(true, tx_query->m_in_tx_pool) && !bool_equals(true, tx_query->m_is_failed) && !bool_equals(false, tx_query->m_is_relayed);
bool can_be_in_tx_pool = !bool_equals(true, tx_query->m_is_confirmed) && !bool_equals(false, tx_query->m_in_tx_pool) && !bool_equals(true, tx_query->m_is_failed) && tx_query->get_height() == boost::none && tx_query->m_min_height == boost::none && !bool_equals(false, tx_query->m_is_locked);
bool can_be_incoming = !bool_equals(false, _query->m_is_incoming) && !bool_equals(true, _query->is_outgoing()) && !bool_equals(true, _query->m_has_destinations);
bool can_be_outgoing = !bool_equals(false, _query->is_outgoing()) && !bool_equals(true, _query->m_is_incoming);
bool is_in = can_be_incoming && can_be_confirmed;
bool is_out = can_be_outgoing && can_be_confirmed;
bool is_pending = can_be_outgoing && can_be_in_tx_pool;
bool is_pool = can_be_incoming && can_be_in_tx_pool;
bool is_failed = !bool_equals(false, tx_query->m_is_failed) && !bool_equals(true, tx_query->m_is_confirmed) && !bool_equals(true, tx_query->m_in_tx_pool) && !bool_equals(false, tx_query->m_is_locked);
// check if fetching pool txs contradicted by configuration
if (tx_query->m_in_tx_pool != boost::none && tx_query->m_in_tx_pool.get() && !can_be_in_tx_pool) {
throw std::runtime_error("Cannot fetch pool transactions because it contradicts configuration");
}
// cache unique txs and blocks
uint64_t height = get_height();
std::map<std::string, std::shared_ptr<monero_tx_wallet>> tx_map;
std::map<uint64_t, std::shared_ptr<monero_block>> block_map;
// get unconfirmed or failed outgoing transfers
if (is_pending || is_failed) {
std::list<std::pair<crypto::hash, tools::wallet2::unconfirmed_transfer_details>> upayments;
m_w2->get_unconfirmed_payments_out(upayments, account_index, subaddress_indices);
for (std::list<std::pair<crypto::hash, tools::wallet2::unconfirmed_transfer_details>>::const_iterator i = upayments.begin(); i != upayments.end(); ++i) {
if (!tx_query->m_hashes.empty() && std::find(tx_query->m_hashes.begin(), tx_query->m_hashes.end(), epee::string_tools::pod_to_hex(i->first)) == tx_query->m_hashes.end()) continue; // skip if hash filtered
std::shared_ptr<monero_tx_wallet> tx = build_tx_with_outgoing_transfer_unconfirmed(*m_w2, i->first, i->second);
if (tx_query->m_is_failed != boost::none && tx_query->m_is_failed.get() != tx->m_is_failed.get()) continue; // skip if failure filtered
merge_tx(tx, tx_map, block_map);
}
}
// get unconfirmed incoming transfers
if (is_pool) {
// update pool state TODO monero-project: this should be encapsulated in wallet when unconfirmed transfers queried
std::vector<std::tuple<cryptonote::transaction, crypto::hash, bool>> process_txs;
m_w2->update_pool_state(process_txs);
if (!process_txs.empty()) m_w2->process_pool_state(process_txs);
std::list<std::pair<crypto::hash, tools::wallet2::pool_payment_details>> payments;
m_w2->get_unconfirmed_payments(payments, account_index, subaddress_indices);
for (std::list<std::pair<crypto::hash, tools::wallet2::pool_payment_details>>::const_iterator i = payments.begin(); i != payments.end(); ++i) {
if (!tx_query->m_hashes.empty() && std::find(tx_query->m_hashes.begin(), tx_query->m_hashes.end(), epee::string_tools::pod_to_hex(i->second.m_pd.m_tx_hash)) == tx_query->m_hashes.end()) continue; // skip if hash filtered
std::shared_ptr<monero_tx_wallet> tx = build_tx_with_incoming_transfer_unconfirmed(*m_w2, height, i->first, i->second);
merge_tx(tx, tx_map, block_map);
}
}
// get confirmed incoming transfers
if (is_in) {
std::list<std::pair<crypto::hash, tools::wallet2::payment_details>> payments;
m_w2->get_payments(payments, min_height, max_height, account_index, subaddress_indices);
for (std::list<std::pair<crypto::hash, tools::wallet2::payment_details>>::const_iterator i = payments.begin(); i != payments.end(); ++i) {
if (!tx_query->m_hashes.empty() && std::find(tx_query->m_hashes.begin(), tx_query->m_hashes.end(), epee::string_tools::pod_to_hex(i->second.m_tx_hash)) == tx_query->m_hashes.end()) continue; // skip if hash filtered
std::shared_ptr<monero_tx_wallet> tx = build_tx_with_incoming_transfer(*m_w2, height, i->first, i->second);
merge_tx(tx, tx_map, block_map);
}
}
// get confirmed outgoing transfers
if (is_out) {
std::list<std::pair<crypto::hash, tools::wallet2::confirmed_transfer_details>> payments;
m_w2->get_payments_out(payments, min_height, max_height, account_index, subaddress_indices);
for (std::list<std::pair<crypto::hash, tools::wallet2::confirmed_transfer_details>>::const_iterator i = payments.begin(); i != payments.end(); ++i) {
if (!tx_query->m_hashes.empty() && std::find(tx_query->m_hashes.begin(), tx_query->m_hashes.end(), epee::string_tools::pod_to_hex(i->first)) == tx_query->m_hashes.end()) continue; // skip if hash filtered
std::shared_ptr<monero_tx_wallet> tx = build_tx_with_outgoing_transfer(*m_w2, height, i->first, i->second);
merge_tx(tx, tx_map, block_map);
}
}
// sort txs by block height
std::vector<std::shared_ptr<monero_tx_wallet>> txs;
for (std::map<std::string, std::shared_ptr<monero_tx_wallet>>::const_iterator tx_iter = tx_map.begin(); tx_iter != tx_map.end(); tx_iter++) {
txs.push_back(tx_iter->second);
}
sort(txs.begin(), txs.end(), tx_height_less_than);
// filter and return transfers
std::vector<std::shared_ptr<monero_transfer>> transfers;
for (const std::shared_ptr<monero_tx_wallet>& tx : txs) {
// tx is not incoming/outgoing unless already set
if (tx->m_is_incoming == boost::none) tx->m_is_incoming = false;
if (tx->m_is_outgoing == boost::none) tx->m_is_outgoing = false;
// sort incoming transfers
sort(tx->m_incoming_transfers.begin(), tx->m_incoming_transfers.end(), incoming_transfer_before);
// collect queried transfers, erase if excluded
for (const std::shared_ptr<monero_transfer>& transfer : tx->filter_transfers(*_query)) transfers.push_back(transfer);
// remove excluded txs from block
if (tx->m_block != boost::none && tx->m_outgoing_transfer == boost::none && tx->m_incoming_transfers.empty()) {
tx->m_block.get()->m_txs.erase(std::remove(tx->m_block.get()->m_txs.begin(), tx->m_block.get()->m_txs.end(), tx), tx->m_block.get()->m_txs.end()); // TODO, no way to use const_iterator?
}
}
MTRACE("monero_wallet_full.cpp get_transfers() returning " << transfers.size() << " transfers");
return transfers;
}
std::vector<std::shared_ptr<monero_output_wallet>> monero_wallet_full::get_outputs_aux(const monero_output_query& query) const {
MTRACE("monero_wallet_full::get_outputs_aux(query)");
// // log query
// if (query.m_tx_query != boost::none) {
// if ((*query.m_tx_query)->m_block == boost::none) std::cout << "Output query's tx query rooted at [tx]:" << (*query.m_tx_query)->serialize() << std::endl;
// else std::cout << "Output query's tx query rooted at [block]: " << (*(*query.m_tx_query)->m_block)->serialize() << std::endl;
// } else std::cout << "Output query: " << query.serialize() << std::endl;
// copy and normalize query
std::shared_ptr<monero_output_query> _query;
if (query.m_tx_query == boost::none) {
std::shared_ptr<monero_output_query> query_ptr = std::make_shared<monero_output_query>(query); // convert to shared pointer for copy
_query = query_ptr->copy(query_ptr, std::make_shared<monero_output_query>());
} else {
std::shared_ptr<monero_tx_query> tx_query = query.m_tx_query.get()->copy(query.m_tx_query.get(), std::make_shared<monero_tx_query>());
if (query.m_tx_query.get()->m_output_query != boost::none && query.m_tx_query.get()->m_output_query.get().get() == &query) {
_query = tx_query->m_output_query.get();
} else {
if (query.m_tx_query.get()->m_output_query != boost::none) throw std::runtime_error("Output query's tx query must be a circular reference or null");
std::shared_ptr<monero_output_query> query_ptr = std::make_shared<monero_output_query>(query); // convert query to shared pointer for copy
_query = query_ptr->copy(query_ptr, std::make_shared<monero_output_query>());
_query->m_tx_query = tx_query;
}
}
if (_query->m_tx_query == boost::none) _query->m_tx_query = std::make_shared<monero_tx_query>();
// get output data from wallet2
tools::wallet2::transfer_container outputs_w2;
m_w2->get_transfers(outputs_w2);
// cache unique txs and blocks
std::map<std::string, std::shared_ptr<monero_tx_wallet>> tx_map;
std::map<uint64_t, std::shared_ptr<monero_block>> block_map;
for (const auto& output_w2 : outputs_w2) {
// TODO: skip tx building if m_w2 output excluded by indices, etc
std::shared_ptr<monero_tx_wallet> tx = build_tx_with_vout(*m_w2, output_w2);
merge_tx(tx, tx_map, block_map);
}
// sort txs by block height
std::vector<std::shared_ptr<monero_tx_wallet>> txs ;
for (std::map<std::string, std::shared_ptr<monero_tx_wallet>>::const_iterator tx_iter = tx_map.begin(); tx_iter != tx_map.end(); tx_iter++) {
txs.push_back(tx_iter->second);
}
sort(txs.begin(), txs.end(), tx_height_less_than);
// filter and return outputs
std::vector<std::shared_ptr<monero_output_wallet>> outputs;
for (const std::shared_ptr<monero_tx_wallet>& tx : txs) {
// sort outputs
sort(tx->m_outputs.begin(), tx->m_outputs.end(), vout_before);
// collect queried outputs, erase if excluded
for (const std::shared_ptr<monero_output_wallet>& output : tx->filter_outputs_wallet(*_query)) outputs.push_back(output);
// remove txs without outputs
if (tx->m_outputs.empty() && tx->m_block != boost::none) tx->m_block.get()->m_txs.erase(std::remove(tx->m_block.get()->m_txs.begin(), tx->m_block.get()->m_txs.end(), tx), tx->m_block.get()->m_txs.end()); // TODO, no way to use const_iterator?
}
return outputs;
}
// private helper to initialize subaddresses using transfer details
std::vector<monero_subaddress> monero_wallet_full::get_subaddresses_aux(const uint32_t account_idx, const std::vector<uint32_t>& subaddress_indices, const std::vector<tools::wallet2::transfer_details>& transfers) const {
std::vector<monero_subaddress> subaddresses;
// get balances per subaddress as maps
std::map<uint32_t, uint64_t> balance_per_subaddress = m_w2->balance_per_subaddress(account_idx, STRICT_);
std::map<uint32_t, std::pair<uint64_t, std::pair<uint64_t, uint64_t>>> unlocked_balance_per_subaddress = m_w2->unlocked_balance_per_subaddress(account_idx, STRICT_);
// get all indices if no indices given
std::vector<uint32_t> subaddress_indices_req;
if (subaddress_indices.empty()) {
for (uint32_t subaddress_idx = 0; subaddress_idx < m_w2->get_num_subaddresses(account_idx); subaddress_idx++) {
subaddress_indices_req.push_back(subaddress_idx);
}
} else {
subaddress_indices_req = subaddress_indices;
}
// initialize subaddresses at indices
for (uint32_t subaddressIndicesIdx = 0; subaddressIndicesIdx < subaddress_indices_req.size(); subaddressIndicesIdx++) {
monero_subaddress subaddress;
subaddress.m_account_index = account_idx;
uint32_t subaddress_idx = subaddress_indices_req.at(subaddressIndicesIdx);
subaddress.m_index = subaddress_idx;
subaddress.m_address = get_address(account_idx, subaddress_idx);
subaddress.m_label = m_w2->get_subaddress_label({account_idx, subaddress_idx});
auto iter1 = balance_per_subaddress.find(subaddress_idx);
subaddress.m_balance = iter1 == balance_per_subaddress.end() ? 0 : iter1->second;
auto iter2 = unlocked_balance_per_subaddress.find(subaddress_idx);
subaddress.m_unlocked_balance = iter2 == unlocked_balance_per_subaddress.end() ? 0 : iter2->second.first;
cryptonote::subaddress_index index = {account_idx, subaddress_idx};
subaddress.m_num_unspent_outputs = count_if(transfers.begin(), transfers.end(), [&](const tools::wallet2::transfer_details& td) { return !td.m_spent && td.m_subaddr_index == index; });
subaddress.m_is_used = find_if(transfers.begin(), transfers.end(), [&](const tools::wallet2::transfer_details& td) { return td.m_subaddr_index == index; }) != transfers.end();
subaddress.m_num_blocks_to_unlock = iter1 == balance_per_subaddress.end() ? 0 : iter2->second.second.first;
subaddresses.push_back(subaddress);
}
return subaddresses;
}
void monero_wallet_full::run_sync_loop() {
if (m_sync_loop_running) return; // only run one loop at a time
m_sync_loop_running = true;
// start sync loop thread
// TODO: use global threadpool, background sync wasm wallet in c++ thread
m_syncing_thread = boost::thread([this]() {
// sync while enabled
while (m_syncing_enabled) {
auto start = std::chrono::system_clock::now();
try { lock_and_sync(); }
catch (std::exception const& e) { std::cout << "monero_wallet_full failed to background synchronize: " << e.what() << std::endl; }
catch (...) { std::cout << "monero_wallet_full failed to background synchronize" << std::endl; }
// only wait if syncing still enabled
if (m_syncing_enabled) {
boost::mutex::scoped_lock lock(m_syncing_mutex);
boost::posix_time::milliseconds wait_for_ms(m_syncing_interval.load());
boost::posix_time::milliseconds elapsed_time = boost::posix_time::milliseconds(std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::system_clock::now() - start).count());
m_sync_cv.timed_wait(lock, elapsed_time > wait_for_ms ? boost::posix_time::milliseconds(0) : wait_for_ms - elapsed_time); // target regular sync period by accounting for sync time
}
}
m_sync_loop_running = false;
});
}
monero_sync_result monero_wallet_full::lock_and_sync(boost::optional<uint64_t> start_height) {
bool rescan = m_rescan_on_sync.exchange(false);
boost::lock_guard<boost::mutex> guarg(m_sync_mutex); // synchronize sync() and syncAsync()
monero_sync_result result;
result.m_num_blocks_fetched = 0;
result.m_received_money = false;
do {
// skip if daemon is not connected or synced
if (m_is_connected && is_daemon_synced()) {
// rescan blockchain if requested
if (rescan) m_w2->rescan_blockchain(false);
// sync wallet
result = sync_aux(start_height);
}
} while (!rescan && (rescan = m_rescan_on_sync.exchange(false))); // repeat if not rescanned and rescan was requested
return result;
}
monero_sync_result monero_wallet_full::sync_aux(boost::optional<uint64_t> start_height) {
MTRACE("sync_aux()");
// determine sync start height
uint64_t sync_start_height = start_height == boost::none ? std::max(get_height(), get_restore_height()) : *start_height;
if (sync_start_height < get_restore_height()) set_restore_height(sync_start_height); // TODO monero-project: start height processed > requested start height unless sync height manually set
// notify listeners of sync start
m_w2_listener->on_sync_start(sync_start_height);
monero_sync_result result;
// attempt to refresh wallet2 which may throw exception
try {
m_w2->refresh(m_w2->is_trusted_daemon(), sync_start_height, result.m_num_blocks_fetched, result.m_received_money, true);
if (!m_is_synced) m_is_synced = true;
m_w2_listener->update_listening(); // cannot unregister during sync which would segfault
} catch (std::exception& e) {
m_w2_listener->on_sync_end(); // signal end of sync to reset listener's start and end heights
throw;
}
// find and save rings
m_w2->find_and_save_rings(false);
// notify listeners of sync end and check for updated funds
m_w2_listener->on_sync_end();
return result;
}
}