2019-03-05 21:05:34 +00:00
|
|
|
// Copyright (c) 2014-2019, The Monero Project
|
2016-04-20 10:01:00 +00:00
|
|
|
//
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// Redistribution and use in source and binary forms, with or without modification, are
|
|
|
|
// permitted provided that the following conditions are met:
|
|
|
|
//
|
|
|
|
// 1. Redistributions of source code must retain the above copyright notice, this list of
|
|
|
|
// conditions and the following disclaimer.
|
|
|
|
//
|
|
|
|
// 2. Redistributions in binary form must reproduce the above copyright notice, this 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
|
|
|
|
|
2016-04-20 10:17:27 +00:00
|
|
|
|
2016-04-20 10:01:00 +00:00
|
|
|
#include "transaction_history.h"
|
2016-04-20 10:17:27 +00:00
|
|
|
#include "transaction_info.h"
|
|
|
|
#include "wallet.h"
|
|
|
|
|
|
|
|
#include "crypto/hash.h"
|
|
|
|
#include "wallet/wallet2.h"
|
2016-04-22 10:21:08 +00:00
|
|
|
|
2016-04-20 10:17:27 +00:00
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <list>
|
|
|
|
|
2016-04-22 10:21:08 +00:00
|
|
|
using namespace epee;
|
|
|
|
|
2016-12-13 15:21:38 +00:00
|
|
|
namespace Monero {
|
2016-04-20 10:17:27 +00:00
|
|
|
|
2016-04-22 10:21:08 +00:00
|
|
|
TransactionHistory::~TransactionHistory() {}
|
|
|
|
|
|
|
|
|
2016-04-20 10:17:27 +00:00
|
|
|
TransactionHistoryImpl::TransactionHistoryImpl(WalletImpl *wallet)
|
2016-04-22 10:21:08 +00:00
|
|
|
: m_wallet(wallet)
|
2016-04-20 10:17:27 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
TransactionHistoryImpl::~TransactionHistoryImpl()
|
|
|
|
{
|
2016-11-18 21:42:08 +00:00
|
|
|
for (auto t : m_history)
|
|
|
|
delete t;
|
2016-04-20 10:17:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int TransactionHistoryImpl::count() const
|
|
|
|
{
|
2016-10-06 20:25:43 +00:00
|
|
|
boost::shared_lock<boost::shared_mutex> lock(m_historyMutex);
|
|
|
|
int result = m_history.size();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
TransactionInfo *TransactionHistoryImpl::transaction(int index) const
|
|
|
|
{
|
|
|
|
boost::shared_lock<boost::shared_mutex> lock(m_historyMutex);
|
|
|
|
// sanity check
|
|
|
|
if (index < 0)
|
|
|
|
return nullptr;
|
|
|
|
unsigned index_ = static_cast<unsigned>(index);
|
|
|
|
return index_ < m_history.size() ? m_history[index_] : nullptr;
|
2016-04-20 10:17:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TransactionInfo *TransactionHistoryImpl::transaction(const std::string &id) const
|
|
|
|
{
|
2016-10-06 20:25:43 +00:00
|
|
|
boost::shared_lock<boost::shared_mutex> lock(m_historyMutex);
|
2016-10-04 20:11:19 +00:00
|
|
|
auto itr = std::find_if(m_history.begin(), m_history.end(),
|
|
|
|
[&](const TransactionInfo * ti) {
|
|
|
|
return ti->hash() == id;
|
|
|
|
});
|
|
|
|
return itr != m_history.end() ? *itr : nullptr;
|
2016-04-20 10:17:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<TransactionInfo *> TransactionHistoryImpl::getAll() const
|
|
|
|
{
|
2016-10-06 20:25:43 +00:00
|
|
|
boost::shared_lock<boost::shared_mutex> lock(m_historyMutex);
|
2016-04-22 10:21:08 +00:00
|
|
|
return m_history;
|
2016-04-20 10:17:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TransactionHistoryImpl::refresh()
|
|
|
|
{
|
2016-10-05 16:01:26 +00:00
|
|
|
// multithreaded access:
|
2016-10-06 20:25:43 +00:00
|
|
|
// boost::lock_guard<boost::mutex> guarg(m_historyMutex);
|
|
|
|
// for "write" access, locking exclusively
|
|
|
|
boost::unique_lock<boost::shared_mutex> lock(m_historyMutex);
|
2016-10-05 16:01:26 +00:00
|
|
|
|
2016-04-20 10:17:27 +00:00
|
|
|
// TODO: configurable values;
|
|
|
|
uint64_t min_height = 0;
|
|
|
|
uint64_t max_height = (uint64_t)-1;
|
2017-01-12 17:23:23 +00:00
|
|
|
uint64_t wallet_height = m_wallet->blockChainHeight();
|
2016-04-20 10:17:27 +00:00
|
|
|
|
2016-04-22 10:21:08 +00:00
|
|
|
// delete old transactions;
|
|
|
|
for (auto t : m_history)
|
|
|
|
delete t;
|
2016-04-29 13:26:14 +00:00
|
|
|
m_history.clear();
|
|
|
|
|
2016-04-22 10:21:08 +00:00
|
|
|
// transactions are stored in wallet2:
|
|
|
|
// - confirmed_transfer_details - out transfers
|
|
|
|
// - unconfirmed_transfer_details - pending out transfers
|
|
|
|
// - payment_details - input transfers
|
|
|
|
|
|
|
|
// payments are "input transactions";
|
2016-04-29 13:26:14 +00:00
|
|
|
// one input transaction contains only one transfer. e.g. <transaction_id> - <100XMR>
|
|
|
|
|
|
|
|
std::list<std::pair<crypto::hash, tools::wallet2::payment_details>> in_payments;
|
|
|
|
m_wallet->m_wallet->get_payments(in_payments, min_height, max_height);
|
|
|
|
for (std::list<std::pair<crypto::hash, tools::wallet2::payment_details>>::const_iterator i = in_payments.begin(); i != in_payments.end(); ++i) {
|
2016-04-20 10:17:27 +00:00
|
|
|
const tools::wallet2::payment_details &pd = i->second;
|
|
|
|
std::string payment_id = string_tools::pod_to_hex(i->first);
|
|
|
|
if (payment_id.substr(16).find_first_not_of('0') == std::string::npos)
|
|
|
|
payment_id = payment_id.substr(0,16);
|
2016-04-22 10:21:08 +00:00
|
|
|
TransactionInfoImpl * ti = new TransactionInfoImpl();
|
|
|
|
ti->m_paymentid = payment_id;
|
|
|
|
ti->m_amount = pd.m_amount;
|
|
|
|
ti->m_direction = TransactionInfo::Direction_In;
|
|
|
|
ti->m_hash = string_tools::pod_to_hex(pd.m_tx_hash);
|
|
|
|
ti->m_blockheight = pd.m_block_height;
|
2017-02-19 02:42:10 +00:00
|
|
|
ti->m_subaddrIndex = { pd.m_subaddr_index.minor };
|
|
|
|
ti->m_subaddrAccount = pd.m_subaddr_index.major;
|
|
|
|
ti->m_label = m_wallet->m_wallet->get_subaddress_label(pd.m_subaddr_index);
|
2016-10-04 20:11:19 +00:00
|
|
|
ti->m_timestamp = pd.m_timestamp;
|
2017-08-04 23:53:16 +00:00
|
|
|
ti->m_confirmations = (wallet_height > pd.m_block_height) ? wallet_height - pd.m_block_height : 0;
|
2017-08-03 19:37:45 +00:00
|
|
|
ti->m_unlock_time = pd.m_unlock_time;
|
2016-04-22 10:21:08 +00:00
|
|
|
m_history.push_back(ti);
|
|
|
|
|
2016-04-29 13:26:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// confirmed output transactions
|
|
|
|
// one output transaction may contain more than one money transfer, e.g.
|
|
|
|
// <transaction_id>:
|
|
|
|
// transfer1: 100XMR to <address_1>
|
|
|
|
// transfer2: 50XMR to <address_2>
|
|
|
|
// fee: fee charged per transaction
|
|
|
|
//
|
|
|
|
|
|
|
|
std::list<std::pair<crypto::hash, tools::wallet2::confirmed_transfer_details>> out_payments;
|
|
|
|
m_wallet->m_wallet->get_payments_out(out_payments, min_height, max_height);
|
|
|
|
|
|
|
|
for (std::list<std::pair<crypto::hash, tools::wallet2::confirmed_transfer_details>>::const_iterator i = out_payments.begin();
|
|
|
|
i != out_payments.end(); ++i) {
|
|
|
|
|
|
|
|
const crypto::hash &hash = i->first;
|
|
|
|
const tools::wallet2::confirmed_transfer_details &pd = i->second;
|
|
|
|
|
|
|
|
uint64_t change = pd.m_change == (uint64_t)-1 ? 0 : pd.m_change; // change may not be known
|
2016-11-02 23:11:30 +00:00
|
|
|
uint64_t fee = pd.m_amount_in - pd.m_amount_out;
|
2016-04-29 13:26:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
std::string payment_id = string_tools::pod_to_hex(i->second.m_payment_id);
|
|
|
|
if (payment_id.substr(16).find_first_not_of('0') == std::string::npos)
|
|
|
|
payment_id = payment_id.substr(0,16);
|
|
|
|
|
|
|
|
|
|
|
|
TransactionInfoImpl * ti = new TransactionInfoImpl();
|
|
|
|
ti->m_paymentid = payment_id;
|
|
|
|
ti->m_amount = pd.m_amount_in - change - fee;
|
|
|
|
ti->m_fee = fee;
|
|
|
|
ti->m_direction = TransactionInfo::Direction_Out;
|
|
|
|
ti->m_hash = string_tools::pod_to_hex(hash);
|
|
|
|
ti->m_blockheight = pd.m_block_height;
|
2017-02-19 02:42:10 +00:00
|
|
|
ti->m_subaddrIndex = pd.m_subaddr_indices;
|
|
|
|
ti->m_subaddrAccount = pd.m_subaddr_account;
|
|
|
|
ti->m_label = pd.m_subaddr_indices.size() == 1 ? m_wallet->m_wallet->get_subaddress_label({pd.m_subaddr_account, *pd.m_subaddr_indices.begin()}) : "";
|
2016-10-04 20:11:19 +00:00
|
|
|
ti->m_timestamp = pd.m_timestamp;
|
2017-08-04 23:53:16 +00:00
|
|
|
ti->m_confirmations = (wallet_height > pd.m_block_height) ? wallet_height - pd.m_block_height : 0;
|
2016-04-29 13:26:14 +00:00
|
|
|
|
|
|
|
// single output transaction might contain multiple transfers
|
|
|
|
for (const auto &d: pd.m_dests) {
|
2018-02-16 11:04:04 +00:00
|
|
|
ti->m_transfers.push_back({d.amount, get_account_address_as_str(m_wallet->m_wallet->nettype(), d.is_subaddress, d.addr)});
|
2016-04-29 13:26:14 +00:00
|
|
|
}
|
|
|
|
m_history.push_back(ti);
|
2016-04-20 10:17:27 +00:00
|
|
|
}
|
2016-04-29 13:26:14 +00:00
|
|
|
|
|
|
|
// unconfirmed output transactions
|
2017-01-12 17:23:23 +00:00
|
|
|
std::list<std::pair<crypto::hash, tools::wallet2::unconfirmed_transfer_details>> upayments_out;
|
|
|
|
m_wallet->m_wallet->get_unconfirmed_payments_out(upayments_out);
|
|
|
|
for (std::list<std::pair<crypto::hash, tools::wallet2::unconfirmed_transfer_details>>::const_iterator i = upayments_out.begin(); i != upayments_out.end(); ++i) {
|
2016-04-29 13:26:14 +00:00
|
|
|
const tools::wallet2::unconfirmed_transfer_details &pd = i->second;
|
|
|
|
const crypto::hash &hash = i->first;
|
2016-08-06 18:19:25 +00:00
|
|
|
uint64_t amount = pd.m_amount_in;
|
|
|
|
uint64_t fee = amount - pd.m_amount_out;
|
2016-04-29 13:26:14 +00:00
|
|
|
std::string payment_id = string_tools::pod_to_hex(i->second.m_payment_id);
|
|
|
|
if (payment_id.substr(16).find_first_not_of('0') == std::string::npos)
|
|
|
|
payment_id = payment_id.substr(0,16);
|
|
|
|
bool is_failed = pd.m_state == tools::wallet2::unconfirmed_transfer_details::failed;
|
|
|
|
|
|
|
|
TransactionInfoImpl * ti = new TransactionInfoImpl();
|
|
|
|
ti->m_paymentid = payment_id;
|
2017-02-19 02:42:10 +00:00
|
|
|
ti->m_amount = amount - pd.m_change - fee;
|
2016-04-29 13:26:14 +00:00
|
|
|
ti->m_fee = fee;
|
|
|
|
ti->m_direction = TransactionInfo::Direction_Out;
|
|
|
|
ti->m_failed = is_failed;
|
|
|
|
ti->m_pending = true;
|
|
|
|
ti->m_hash = string_tools::pod_to_hex(hash);
|
2017-02-19 02:42:10 +00:00
|
|
|
ti->m_subaddrIndex = pd.m_subaddr_indices;
|
|
|
|
ti->m_subaddrAccount = pd.m_subaddr_account;
|
|
|
|
ti->m_label = pd.m_subaddr_indices.size() == 1 ? m_wallet->m_wallet->get_subaddress_label({pd.m_subaddr_account, *pd.m_subaddr_indices.begin()}) : "";
|
2016-10-04 20:11:19 +00:00
|
|
|
ti->m_timestamp = pd.m_timestamp;
|
2017-01-12 17:23:23 +00:00
|
|
|
ti->m_confirmations = 0;
|
2016-04-29 13:26:14 +00:00
|
|
|
m_history.push_back(ti);
|
|
|
|
}
|
2017-01-12 17:23:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
// unconfirmed payments (tx pool)
|
2017-09-22 12:57:20 +00:00
|
|
|
std::list<std::pair<crypto::hash, tools::wallet2::pool_payment_details>> upayments;
|
2017-01-12 17:23:23 +00:00
|
|
|
m_wallet->m_wallet->get_unconfirmed_payments(upayments);
|
2017-09-22 12:57:20 +00:00
|
|
|
for (std::list<std::pair<crypto::hash, tools::wallet2::pool_payment_details>>::const_iterator i = upayments.begin(); i != upayments.end(); ++i) {
|
|
|
|
const tools::wallet2::payment_details &pd = i->second.m_pd;
|
2017-01-12 17:23:23 +00:00
|
|
|
std::string payment_id = string_tools::pod_to_hex(i->first);
|
|
|
|
if (payment_id.substr(16).find_first_not_of('0') == std::string::npos)
|
|
|
|
payment_id = payment_id.substr(0,16);
|
|
|
|
TransactionInfoImpl * ti = new TransactionInfoImpl();
|
|
|
|
ti->m_paymentid = payment_id;
|
|
|
|
ti->m_amount = pd.m_amount;
|
|
|
|
ti->m_direction = TransactionInfo::Direction_In;
|
|
|
|
ti->m_hash = string_tools::pod_to_hex(pd.m_tx_hash);
|
|
|
|
ti->m_blockheight = pd.m_block_height;
|
|
|
|
ti->m_pending = true;
|
2017-02-19 02:42:10 +00:00
|
|
|
ti->m_subaddrIndex = { pd.m_subaddr_index.minor };
|
|
|
|
ti->m_subaddrAccount = pd.m_subaddr_index.major;
|
|
|
|
ti->m_label = m_wallet->m_wallet->get_subaddress_label(pd.m_subaddr_index);
|
2017-01-12 17:23:23 +00:00
|
|
|
ti->m_timestamp = pd.m_timestamp;
|
|
|
|
ti->m_confirmations = 0;
|
|
|
|
m_history.push_back(ti);
|
|
|
|
|
|
|
|
LOG_PRINT_L1(__FUNCTION__ << ": Unconfirmed payment found " << pd.m_amount);
|
|
|
|
}
|
|
|
|
|
2016-04-20 10:17:27 +00:00
|
|
|
}
|
2016-04-20 10:01:00 +00:00
|
|
|
|
2016-10-06 21:43:45 +00:00
|
|
|
} // namespace
|
2016-12-13 15:21:38 +00:00
|
|
|
|
|
|
|
namespace Bitmonero = Monero;
|