2020-05-24 13:20:03 +02:00
|
|
|
/*
|
|
|
|
|
* This file is part of the Flowee project
|
|
|
|
|
* Copyright (C) 2020 Tom Zander <tomz@freedommail.ch>
|
|
|
|
|
*
|
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
#include "Wallet.h"
|
2020-10-17 16:38:37 +02:00
|
|
|
#include "Wallet_p.h"
|
2020-11-06 18:16:39 +01:00
|
|
|
#include "FloweePay.h"
|
2020-05-24 13:20:03 +02:00
|
|
|
|
|
|
|
|
#include <primitives/script.h>
|
|
|
|
|
#include <streaming/BufferPool.h>
|
|
|
|
|
#include <streaming/MessageBuilder.h>
|
|
|
|
|
#include <streaming/MessageParser.h>
|
|
|
|
|
#include <base58.h>
|
|
|
|
|
|
2020-06-08 14:37:29 +02:00
|
|
|
#include <QFile>
|
2020-05-24 13:20:03 +02:00
|
|
|
#include <QSet>
|
2020-11-04 21:58:17 +01:00
|
|
|
#include <QTimer>
|
2020-11-05 22:15:40 +01:00
|
|
|
#include <QThread>
|
2020-05-24 13:20:03 +02:00
|
|
|
|
2020-10-16 19:25:52 +02:00
|
|
|
// #define DEBUGUTXO
|
2020-06-08 14:37:29 +02:00
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Fee estimation needs to know how much space is used per input.
|
|
|
|
|
* 32 + 4 for the prevTx
|
|
|
|
|
* Then 72 + 33 bytes for the signature and pub-key (+-1).
|
2020-11-06 18:16:39 +01:00
|
|
|
* Using schnorr we can gain 8 bytes for the signature (not included here).
|
2020-06-08 14:37:29 +02:00
|
|
|
*/
|
|
|
|
|
constexpr int BYTES_PER_OUTPUT = 149;
|
2020-11-06 20:05:42 +01:00
|
|
|
const int MATURATION_AGE = 100; // the amount of blocks a coinbase takes before we can spend it
|
2020-06-08 14:37:29 +02:00
|
|
|
|
2020-10-15 19:18:54 +02:00
|
|
|
// static
|
|
|
|
|
Wallet *Wallet::createWallet(const boost::filesystem::path &basedir, uint16_t segmentId, const QString &name)
|
|
|
|
|
{
|
|
|
|
|
Wallet *wallet = new Wallet();
|
|
|
|
|
wallet->m_basedir = basedir / QString("wallet-%1/").arg(segmentId).toStdString();
|
|
|
|
|
wallet->m_segment.reset(new PrivacySegment(segmentId, wallet));
|
|
|
|
|
if (name.isEmpty())
|
|
|
|
|
wallet->m_name = QString("unnamed-%1").arg(segmentId);
|
|
|
|
|
else
|
|
|
|
|
wallet->m_name = name;
|
|
|
|
|
|
|
|
|
|
return wallet;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Wallet::Wallet()
|
2020-10-16 18:17:56 +02:00
|
|
|
: m_walletChanged(true)
|
2020-10-15 19:18:54 +02:00
|
|
|
{
|
|
|
|
|
}
|
2020-05-24 13:20:03 +02:00
|
|
|
|
|
|
|
|
Wallet::Wallet(const boost::filesystem::path &basedir, uint16_t segmentId)
|
|
|
|
|
: m_segment(new PrivacySegment(segmentId, this)),
|
|
|
|
|
m_basedir(basedir / QString("wallet-%1/").arg(segmentId).toStdString())
|
|
|
|
|
{
|
|
|
|
|
loadSecrets();
|
|
|
|
|
loadWallet();
|
|
|
|
|
rebuildBloom();
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-02 21:49:06 +01:00
|
|
|
Wallet::~Wallet()
|
|
|
|
|
{
|
|
|
|
|
// these return instantly if nothing has to be saved.
|
|
|
|
|
saveSecrets();
|
|
|
|
|
saveWallet();
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-06 20:05:42 +01:00
|
|
|
Wallet::WalletTransaction Wallet::createWalletTransactionFromTx(const Tx &tx, const uint256 &txid) const
|
2020-11-02 21:49:06 +01:00
|
|
|
{
|
|
|
|
|
WalletTransaction wtx;
|
|
|
|
|
wtx.txid = txid;
|
|
|
|
|
|
|
|
|
|
OutputRef prevTx;
|
|
|
|
|
int inputIndex = -1;
|
|
|
|
|
int outputIndex = -1;
|
|
|
|
|
Output output;
|
|
|
|
|
logDebug() << "new tx." << wtx.txid;
|
|
|
|
|
Tx::Iterator iter(tx);
|
|
|
|
|
while (iter.next() != Tx::End) {
|
|
|
|
|
if (iter.tag() == Tx::PrevTxHash) {
|
2020-11-06 20:05:42 +01:00
|
|
|
const uint256 prevTxhash = iter.uint256Data();
|
|
|
|
|
if (++inputIndex == 0)
|
|
|
|
|
wtx.isCoinbase = prevTxhash.IsNull();
|
|
|
|
|
if (!wtx.isCoinbase) {
|
|
|
|
|
auto i = m_txidCash.find(prevTxhash);
|
|
|
|
|
prevTx.setTxIndex((i != m_txidCash.end()) ? i->second : 0);
|
|
|
|
|
if (i != m_txidCash.end())
|
|
|
|
|
logDebug() << " Input:" << inputIndex << "prevTx:" << prevTxhash
|
|
|
|
|
<< Log::Hex << i->second << prevTx.encoded();
|
|
|
|
|
}
|
2020-11-02 21:49:06 +01:00
|
|
|
} else if (iter.tag() == Tx::PrevTxIndex) {
|
|
|
|
|
if (prevTx.txIndex() > 0) { // we know the prevTx
|
|
|
|
|
assert(iter.longData() < 0xFFFF); // it would break our scheme
|
|
|
|
|
prevTx.setOutputIndex(iter.intData());
|
|
|
|
|
|
|
|
|
|
auto utxo = m_unspentOutputs.find(prevTx.encoded());
|
|
|
|
|
if (utxo != m_unspentOutputs.end()) {
|
|
|
|
|
// input is spending one of our UTXOs
|
|
|
|
|
logDebug() << " -> spent UTXO";
|
|
|
|
|
wtx.inputToWTX.insert(std::make_pair(inputIndex, prevTx.encoded()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (iter.tag() == Tx::OutputValue) {
|
|
|
|
|
++outputIndex;
|
|
|
|
|
output.value = iter.longData();
|
|
|
|
|
}
|
|
|
|
|
else if (iter.tag() == Tx::OutputScript) {
|
|
|
|
|
output.walletSecretId = findSecretFor(iter.byteData());
|
|
|
|
|
if (output.walletSecretId > 0) {
|
|
|
|
|
logDebug() << " output"<< outputIndex << "pays to wallet id" << output.walletSecretId;
|
|
|
|
|
wtx.outputs.insert(std::make_pair(outputIndex, output));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return wtx;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::newTransaction(const Tx &tx)
|
|
|
|
|
{
|
2020-11-04 18:59:24 +01:00
|
|
|
int firstNewTransaction;
|
2020-11-02 21:49:06 +01:00
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
2020-11-04 18:59:24 +01:00
|
|
|
firstNewTransaction = m_nextWalletTransactionId;
|
2020-11-02 21:49:06 +01:00
|
|
|
const uint256 txid = tx.createHash();
|
|
|
|
|
if (m_txidCash.find(txid) != m_txidCash.end()) // already known
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
WalletTransaction wtx = createWalletTransactionFromTx(tx, txid);
|
2020-11-06 20:05:42 +01:00
|
|
|
Q_ASSERT(wtx.isCoinbase == false);
|
2020-11-02 21:49:06 +01:00
|
|
|
if (wtx.outputs.empty() && wtx.inputToWTX.empty()) {
|
|
|
|
|
// no connection to our UTXOs
|
|
|
|
|
if (--m_bloomScore < 25)
|
|
|
|
|
rebuildBloom();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
wtx.minedBlockHeight = WalletPriv::Unconfirmed;
|
2020-11-02 21:49:06 +01:00
|
|
|
|
|
|
|
|
// Mark UTXOs locked that this tx spent to avoid double spending them.
|
|
|
|
|
for (auto i = wtx.inputToWTX.begin(); i != wtx.inputToWTX.end(); ++i) {
|
2020-11-03 20:16:36 +01:00
|
|
|
m_autoLockedOutputs.insert(std::make_pair(i->second, m_nextWalletTransactionId));
|
2020-11-02 21:49:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// insert new UTXOs
|
|
|
|
|
for (auto i = wtx.outputs.begin(); i != wtx.outputs.end(); ++i) {
|
|
|
|
|
uint64_t key = m_nextWalletTransactionId;
|
|
|
|
|
key <<= 16;
|
|
|
|
|
key += i->first;
|
|
|
|
|
logDebug() << " inserting output"<< i->first << Log::Hex << i->second.walletSecretId << key;
|
|
|
|
|
m_unspentOutputs.insert(std::make_pair(key, i->second.value));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// and remember the transaction
|
|
|
|
|
m_txidCash.insert(std::make_pair(wtx.txid, m_nextWalletTransactionId));
|
|
|
|
|
m_walletTransactions.insert(std::make_pair(m_nextWalletTransactionId++, wtx));
|
|
|
|
|
m_walletChanged = true;
|
|
|
|
|
|
|
|
|
|
logCritical() << "Wallet" << m_segment->segmentId() << "claims" << tx.createHash() << "[unconfirmed]";
|
|
|
|
|
} // mutex scope
|
|
|
|
|
saveTransaction(tx);
|
2020-11-06 22:15:03 +01:00
|
|
|
recalculateBalance();
|
2020-11-04 18:59:24 +01:00
|
|
|
|
|
|
|
|
emit utxosChanged();
|
|
|
|
|
emit appendedTransactions(firstNewTransaction, 1);
|
2020-11-02 21:49:06 +01:00
|
|
|
}
|
|
|
|
|
|
2020-05-24 13:20:03 +02:00
|
|
|
void Wallet::newTransactions(const BlockHeader &header, int blockHeight, const std::deque<Tx> &blockTransactions)
|
|
|
|
|
{
|
2020-10-17 16:38:37 +02:00
|
|
|
auto transactions = WalletPriv::sortTransactions(blockTransactions);
|
2020-05-24 13:20:03 +02:00
|
|
|
std::deque<Tx> transactionsToSave;
|
2020-11-03 20:16:36 +01:00
|
|
|
std::set<int> ejectedTransactions;
|
2020-05-24 13:20:03 +02:00
|
|
|
int firstNewTransaction;
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
firstNewTransaction = m_nextWalletTransactionId;
|
|
|
|
|
for (auto tx: transactions) {
|
2020-11-02 21:49:06 +01:00
|
|
|
const uint256 txid = tx.createHash();
|
2020-05-24 13:20:03 +02:00
|
|
|
WalletTransaction wtx;
|
2020-11-02 21:49:06 +01:00
|
|
|
|
|
|
|
|
auto oldTx = m_txidCash.find(txid);
|
2020-11-03 20:16:36 +01:00
|
|
|
int walletTransactionId = m_nextWalletTransactionId;
|
2020-11-02 21:49:06 +01:00
|
|
|
if (oldTx == m_txidCash.end()) {
|
|
|
|
|
wtx = createWalletTransactionFromTx(tx, txid);
|
|
|
|
|
if (wtx.outputs.empty() && wtx.inputToWTX.empty()) {
|
|
|
|
|
// no connection to our UTXOs
|
|
|
|
|
if (--m_bloomScore < 25)
|
|
|
|
|
rebuildBloom();
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// we already seen it before.
|
|
|
|
|
wtx = m_walletTransactions.find(oldTx->second)->second;
|
|
|
|
|
// check if the one we saw was unconfirmed or not.
|
|
|
|
|
if (wtx.minedBlockHeight >= 0)
|
|
|
|
|
continue;
|
2020-11-03 20:16:36 +01:00
|
|
|
walletTransactionId = oldTx->second;
|
2020-11-02 21:49:06 +01:00
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
const bool wasUnconfirmed = wtx.minedBlockHeight == WalletPriv::Unconfirmed;
|
2020-05-24 13:20:03 +02:00
|
|
|
wtx.minedBlock = header.createHash();
|
|
|
|
|
wtx.minedBlockHeight = blockHeight;
|
|
|
|
|
|
|
|
|
|
// remove UTXOs this Tx spent
|
|
|
|
|
for (auto i = wtx.inputToWTX.begin(); i != wtx.inputToWTX.end(); ++i) {
|
|
|
|
|
auto iter = m_unspentOutputs.find(i->second);
|
|
|
|
|
assert(iter != m_unspentOutputs.end()); // a double spend? With checked merkle-block? Thats...odd.
|
|
|
|
|
if (iter != m_unspentOutputs.end())
|
|
|
|
|
m_unspentOutputs.erase(iter);
|
|
|
|
|
|
2020-11-02 21:49:06 +01:00
|
|
|
// unlock UTXOs
|
|
|
|
|
auto lockedIter = m_autoLockedOutputs.find(i->second);
|
2020-11-03 20:16:36 +01:00
|
|
|
if (lockedIter != m_autoLockedOutputs.end()) {
|
|
|
|
|
if (lockedIter->second != walletTransactionId) {
|
|
|
|
|
// if this output was locked by another transaction then that means
|
|
|
|
|
// that other transaction was rejected (double spent) since the
|
|
|
|
|
// tx we are now processing made it into a block.
|
|
|
|
|
// remember so we can process ejected transactions below.
|
|
|
|
|
ejectedTransactions.insert(lockedIter->second);
|
|
|
|
|
}
|
2020-11-02 21:49:06 +01:00
|
|
|
m_autoLockedOutputs.erase(lockedIter);
|
2020-11-03 20:16:36 +01:00
|
|
|
}
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-11-02 21:49:06 +01:00
|
|
|
|
|
|
|
|
if (!wasUnconfirmed) { // unconfirmed transactions already had their outputs added
|
|
|
|
|
// insert new UTXOs
|
|
|
|
|
for (auto i = wtx.outputs.begin(); i != wtx.outputs.end(); ++i) {
|
|
|
|
|
uint64_t key = m_nextWalletTransactionId;
|
|
|
|
|
key <<= 16;
|
|
|
|
|
key += i->first;
|
|
|
|
|
logDebug() << " inserting output"<< i->first << Log::Hex << i->second.walletSecretId << key;
|
|
|
|
|
m_unspentOutputs.insert(std::make_pair(key, i->second.value));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// and remember the transaction
|
2020-11-03 20:16:36 +01:00
|
|
|
if (oldTx == m_txidCash.end()) {
|
|
|
|
|
Q_ASSERT(walletTransactionId == m_nextWalletTransactionId);
|
|
|
|
|
m_txidCash.insert(std::make_pair(wtx.txid, m_nextWalletTransactionId));
|
|
|
|
|
m_walletTransactions.insert(std::make_pair(m_nextWalletTransactionId++, wtx));
|
|
|
|
|
transactionsToSave.push_back(tx);
|
|
|
|
|
}
|
2020-11-05 22:15:40 +01:00
|
|
|
else { // update the old one with the new data.
|
|
|
|
|
auto wtxIter = m_walletTransactions.find(oldTx->second);
|
|
|
|
|
assert(wtxIter != m_walletTransactions.end());
|
|
|
|
|
wtxIter->second = wtx;
|
|
|
|
|
}
|
2020-05-24 13:20:03 +02:00
|
|
|
m_walletChanged = true;
|
|
|
|
|
|
|
|
|
|
logCritical() << "Wallet" << m_segment->segmentId() << "claims" << tx.createHash() << "@" << blockHeight;
|
|
|
|
|
}
|
|
|
|
|
assert(m_nextWalletTransactionId - firstNewTransaction == int(transactionsToSave.size()));
|
2020-11-03 20:16:36 +01:00
|
|
|
|
|
|
|
|
// In processing the transactions from a block we might find that one of our
|
|
|
|
|
// unconfirmed transactions ended up being rejected due to double spend.
|
|
|
|
|
// We need to update our internal book keeping and eject the transaction.
|
|
|
|
|
// We actually mark it Rejected (in the blockHeight).
|
|
|
|
|
for (auto ejectedTx : ejectedTransactions) {
|
|
|
|
|
auto tx = m_walletTransactions.find(ejectedTx);
|
|
|
|
|
Q_ASSERT(tx != m_walletTransactions.end());
|
|
|
|
|
logDebug() << "Confirmed transaction(s) in block" << blockHeight <<
|
|
|
|
|
"made invalid transaction:" << ejectedTx << tx->second.txid;
|
|
|
|
|
auto &wtx = tx->second;
|
2020-11-06 18:16:39 +01:00
|
|
|
wtx.minedBlockHeight = WalletPriv::Rejected;
|
2020-11-03 20:16:36 +01:00
|
|
|
// Any outputs we locked need to be unlocked
|
|
|
|
|
for (auto i = wtx.inputToWTX.begin(); i != wtx.inputToWTX.end(); ++i) {
|
|
|
|
|
auto iter = m_autoLockedOutputs.find(i->second);
|
|
|
|
|
if (iter != m_autoLockedOutputs.end() && iter->second == ejectedTx)
|
|
|
|
|
m_autoLockedOutputs.erase(iter);
|
|
|
|
|
}
|
|
|
|
|
// Any UTXOs we created for this rejected Tx need to be removed
|
|
|
|
|
for (auto i = wtx.outputs.begin(); i != wtx.outputs.end(); ++i) {
|
|
|
|
|
uint64_t key = ejectedTx;
|
|
|
|
|
key <<= 16;
|
|
|
|
|
key += i->first;
|
|
|
|
|
auto utxo = m_unspentOutputs.find(key);
|
|
|
|
|
if (utxo != m_unspentOutputs.end())
|
|
|
|
|
m_unspentOutputs.erase(utxo);
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-11-02 21:49:06 +01:00
|
|
|
} // mutex scope
|
2020-11-06 22:15:03 +01:00
|
|
|
|
2020-05-24 13:20:03 +02:00
|
|
|
if (!transactionsToSave.empty()) {
|
|
|
|
|
emit utxosChanged();
|
|
|
|
|
emit appendedTransactions(firstNewTransaction, transactionsToSave.size());
|
2020-11-06 22:15:03 +01:00
|
|
|
for (auto tx : transactionsToSave) { // save the Tx to disk.
|
|
|
|
|
saveTransaction(tx);
|
|
|
|
|
}
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-11-06 22:15:03 +01:00
|
|
|
recalculateBalance();
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
|
|
|
|
|
2020-11-02 21:49:06 +01:00
|
|
|
void Wallet::saveTransaction(const Tx &tx)
|
|
|
|
|
{
|
|
|
|
|
QString dir("%1/%2/");
|
|
|
|
|
dir = dir.arg(QString::fromStdString(m_basedir.string()));
|
|
|
|
|
try {
|
|
|
|
|
const QString txid = QString::fromStdString(tx.createHash().ToString());
|
|
|
|
|
QString localdir = dir.arg(txid.left(2));
|
|
|
|
|
boost::filesystem::create_directories(localdir.toStdString());
|
|
|
|
|
QString filename = txid.mid(2);
|
|
|
|
|
|
|
|
|
|
std::ofstream txSaver;
|
|
|
|
|
txSaver.open((localdir + filename).toStdString());
|
|
|
|
|
txSaver.write(tx.data().begin(), tx.size());
|
|
|
|
|
} catch (const std::exception &e) {
|
|
|
|
|
logFatal() << "Could not store transaction" << e.what();
|
2020-11-04 21:58:17 +01:00
|
|
|
throw;
|
2020-11-02 21:49:06 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-05 22:21:50 +01:00
|
|
|
Tx Wallet::loadTransaction(const uint256 &txid, Streaming::BufferPool &pool)
|
2020-11-04 21:58:17 +01:00
|
|
|
{
|
|
|
|
|
QString path = QString::fromStdString(txid.ToString());
|
|
|
|
|
path.insert(2, '/');
|
|
|
|
|
path = QString::fromStdString(m_basedir.string()) + "/" + path;
|
|
|
|
|
|
|
|
|
|
QFile reader(path);
|
|
|
|
|
if (reader.open(QIODevice::ReadOnly)) {
|
2020-11-05 22:21:50 +01:00
|
|
|
pool.reserve(reader.size());
|
|
|
|
|
reader.read(pool.begin(), reader.size());
|
|
|
|
|
return Tx(pool.commit(reader.size()));
|
2020-11-04 21:58:17 +01:00
|
|
|
}
|
|
|
|
|
// return empty tx
|
|
|
|
|
return Tx();
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-06 20:05:42 +01:00
|
|
|
int Wallet::findSecretFor(const Streaming::ConstBuffer &outputScript) const
|
2020-05-24 13:20:03 +02:00
|
|
|
{
|
|
|
|
|
std::vector<std::vector<uint8_t> > vSolutions;
|
|
|
|
|
Script::TxnOutType whichType;
|
|
|
|
|
if (!Script::solver(outputScript, whichType, vSolutions))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
CKeyID keyID;
|
|
|
|
|
switch (whichType)
|
|
|
|
|
{
|
|
|
|
|
case Script::TX_PUBKEY:
|
|
|
|
|
keyID = CPubKey(vSolutions[0]).GetID();
|
|
|
|
|
break;
|
|
|
|
|
case Script::TX_PUBKEYHASH:
|
|
|
|
|
keyID = CKeyID(uint160(vSolutions[0]));
|
|
|
|
|
break;
|
|
|
|
|
case Script::TX_SCRIPTHASH:
|
|
|
|
|
case Script::TX_MULTISIG:
|
|
|
|
|
// we don't store those in the wallet.
|
|
|
|
|
return -1;
|
|
|
|
|
case Script::TX_NONSTANDARD:
|
|
|
|
|
case Script::TX_NULL_DATA:
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (auto i = m_walletSecrets.begin(); i != m_walletSecrets.end(); ++i) {
|
|
|
|
|
if (i->second.address == keyID) {
|
|
|
|
|
return i->first;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::rebuildBloom()
|
|
|
|
|
{
|
|
|
|
|
auto lock = m_segment->clearFilter();
|
|
|
|
|
for (auto priv : m_walletSecrets) {
|
2020-06-08 14:31:45 +02:00
|
|
|
if (priv.second.initialHeight > 0)
|
|
|
|
|
m_segment->addKeyToFilter(priv.second.address, priv.second.initialHeight);
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
|
|
|
|
for (auto utxo : m_unspentOutputs) {
|
2020-06-08 14:31:45 +02:00
|
|
|
OutputRef ref(utxo.first);
|
|
|
|
|
assert(m_walletTransactions.find(ref.txIndex()) != m_walletTransactions.end());
|
|
|
|
|
auto tx = m_walletTransactions.at(ref.txIndex());
|
|
|
|
|
m_segment->addToFilter(tx.txid, ref.outputIndex());
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
|
|
|
|
m_bloomScore = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-11 10:24:05 +02:00
|
|
|
bool Wallet::isSingleAddressWallet() const
|
|
|
|
|
{
|
|
|
|
|
return m_singleAddressWallet;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::setSingleAddressWallet(bool singleAddressWallet)
|
|
|
|
|
{
|
|
|
|
|
m_singleAddressWallet = singleAddressWallet;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-04 21:58:17 +01:00
|
|
|
void Wallet::broadcastTxFinished(int txIndex, bool success)
|
|
|
|
|
{
|
2020-11-05 22:21:50 +01:00
|
|
|
QMutexLocker locker(&m_lock);
|
2020-11-04 21:58:17 +01:00
|
|
|
for (int i = 0; i < m_broadcastingTransactions.size(); ++i) {
|
|
|
|
|
if (m_broadcastingTransactions.at(i)->txIndex() == txIndex) {
|
|
|
|
|
m_broadcastingTransactions.removeAt(i);
|
2020-11-05 22:21:50 +01:00
|
|
|
|
|
|
|
|
if (!success) {
|
|
|
|
|
auto wtx = m_walletTransactions.find(txIndex);
|
|
|
|
|
if (wtx != m_walletTransactions.end()) {
|
|
|
|
|
logCritical() << "Marking transaction invalid";
|
|
|
|
|
|
2020-11-06 18:16:39 +01:00
|
|
|
wtx->second.minedBlockHeight = WalletPriv::Rejected;
|
2020-11-05 22:21:50 +01:00
|
|
|
}
|
|
|
|
|
}
|
2020-11-04 21:58:17 +01:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-24 13:20:03 +02:00
|
|
|
QString Wallet::name() const
|
|
|
|
|
{
|
|
|
|
|
return m_name;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::setName(const QString &name)
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
m_name = name;
|
|
|
|
|
m_walletChanged = true;
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-08 14:37:29 +02:00
|
|
|
const uint256 &Wallet::txid(Wallet::OutputRef ref) const
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
auto iter = m_walletTransactions.find(ref.txIndex());
|
|
|
|
|
if (m_walletTransactions.end() == iter)
|
|
|
|
|
throw std::runtime_error("Invalid ref");
|
|
|
|
|
return iter->second.txid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Tx::Output Wallet::txOutout(Wallet::OutputRef ref) const
|
|
|
|
|
{
|
|
|
|
|
QString txid;
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
auto iter = m_walletTransactions.find(ref.txIndex());
|
|
|
|
|
if (m_walletTransactions.end() == iter)
|
|
|
|
|
throw std::runtime_error("Invalid ref");
|
|
|
|
|
txid = QString::fromStdString(iter->second.txid.ToString());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString filename("%1/%2/%3");
|
|
|
|
|
filename = filename.arg(QString::fromStdString(m_basedir.string()));
|
|
|
|
|
filename = filename.arg(txid.left(2));
|
|
|
|
|
filename = filename.arg(txid.mid(2));
|
|
|
|
|
|
|
|
|
|
QFile file(filename);
|
|
|
|
|
if (!file.open(QIODevice::ReadOnly))
|
|
|
|
|
throw std::runtime_error("missing data");
|
|
|
|
|
|
|
|
|
|
const int fileSize = file.size();
|
|
|
|
|
Streaming::BufferPool pool(fileSize);
|
|
|
|
|
file.read(pool.begin(), fileSize);
|
|
|
|
|
Tx tx(pool.commit(fileSize));
|
|
|
|
|
return tx.output(ref.outputIndex());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const CKey &Wallet::unlockKey(Wallet::OutputRef ref) const
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
auto iter = m_walletTransactions.find(ref.txIndex());
|
|
|
|
|
if (m_walletTransactions.end() == iter)
|
|
|
|
|
throw std::runtime_error("Invalid ref");
|
|
|
|
|
auto iter2 = iter->second.outputs.find(ref.outputIndex());
|
|
|
|
|
if (iter2 == iter->second.outputs.end())
|
|
|
|
|
throw std::runtime_error("Invalid ref(2)");
|
|
|
|
|
auto iter3 = m_walletSecrets.find(iter2->second.walletSecretId);
|
|
|
|
|
assert(iter3 != m_walletSecrets.end());
|
|
|
|
|
return iter3->second.privKey;
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-11 10:24:05 +02:00
|
|
|
CKeyID Wallet::nextChangeAddress()
|
2020-06-08 14:37:29 +02:00
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
for (auto i = m_walletSecrets.begin(); i != m_walletSecrets.end(); ++i) {
|
2020-06-11 10:24:05 +02:00
|
|
|
if (m_singleAddressWallet)
|
|
|
|
|
return i->second.address; // just return the first then.
|
2020-06-08 14:37:29 +02:00
|
|
|
if (i->second.initialHeight == -1) // is change address.
|
|
|
|
|
return i->second.address;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// no change addresses, lets make some.
|
|
|
|
|
CKeyID answer;
|
|
|
|
|
for (int i = 0; i < 50; ++i) {
|
|
|
|
|
WalletSecret secret;
|
|
|
|
|
secret.privKey.MakeNewKey();
|
|
|
|
|
|
|
|
|
|
const CPubKey pubkey = secret.privKey.GetPubKey();
|
|
|
|
|
secret.address = pubkey.GetID();
|
|
|
|
|
if (i == 0)
|
|
|
|
|
answer = secret.address;
|
|
|
|
|
m_walletSecrets.insert(std::make_pair(m_nextWalletSecretId++, secret));
|
|
|
|
|
}
|
|
|
|
|
m_secretsChanged = true;
|
|
|
|
|
saveSecrets();
|
|
|
|
|
|
|
|
|
|
return answer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
namespace {
|
2020-10-23 19:45:08 +02:00
|
|
|
struct UnspentOutput {
|
|
|
|
|
Wallet::OutputRef outputRef;
|
|
|
|
|
qint64 value = 0; // in satoshis
|
|
|
|
|
int score = 0; // the score gained by using this tx.
|
2020-06-08 14:37:29 +02:00
|
|
|
};
|
|
|
|
|
|
2020-10-23 19:45:08 +02:00
|
|
|
int scoreForSolution(size_t outputCount, int change, size_t unspentOutputCount)
|
|
|
|
|
{
|
|
|
|
|
assert(unspentOutputCount > 0);
|
|
|
|
|
assert(outputCount > 0);
|
|
|
|
|
assert(change > 0);
|
|
|
|
|
|
|
|
|
|
const int resultingOutputCount = unspentOutputCount - outputCount;
|
|
|
|
|
int score = 0;
|
|
|
|
|
// aim to keep our output count between 10 and 15
|
|
|
|
|
if (resultingOutputCount > 10 && resultingOutputCount <= 15)
|
|
|
|
|
score = 1000; // perfection
|
|
|
|
|
else if (resultingOutputCount > 5 && resultingOutputCount < 15)
|
|
|
|
|
score = 250;
|
|
|
|
|
else if (resultingOutputCount < 25 && resultingOutputCount > 10)
|
|
|
|
|
score = 250;
|
|
|
|
|
else if (resultingOutputCount > 25)
|
|
|
|
|
score -= (resultingOutputCount - 25) * 10;
|
|
|
|
|
else
|
|
|
|
|
score -= (5 - resultingOutputCount) * 10; // for the 0 - 5 range
|
|
|
|
|
|
|
|
|
|
// in most cases no modifier is added due to change
|
|
|
|
|
if (change < 100)
|
|
|
|
|
score += 2000; // thats very nice (if over 0, that's for the miner)
|
|
|
|
|
else if (change < 1000) // we would create very small UTXO, not nice.
|
|
|
|
|
score -= 1000;
|
|
|
|
|
else if (change < 5000) // ditto
|
|
|
|
|
score -= 800;
|
|
|
|
|
|
|
|
|
|
return score;
|
|
|
|
|
}
|
2020-06-08 14:37:29 +02:00
|
|
|
}
|
|
|
|
|
|
2020-10-23 19:45:08 +02:00
|
|
|
Wallet::OutputSet Wallet::findInputsFor(qint64 output, int feePerByte, int txSize, int64_t &change) const
|
2020-06-08 14:37:29 +02:00
|
|
|
{
|
|
|
|
|
/*
|
|
|
|
|
* The main selection criterea is the amount of outputs we have afterwards.
|
|
|
|
|
*
|
|
|
|
|
* The goal is to always have between 10 and 15 outputs of varying sizes in
|
|
|
|
|
* our wallet, this makes sure we avoid being without confirmed outputs. Even
|
|
|
|
|
* on medium-heavy usage.
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* As we assume the first items in the list of unspentOutputs are the oldest, all
|
2020-10-23 19:45:08 +02:00
|
|
|
* we need to do is find the combination of inputs that works best.
|
2020-06-08 14:37:29 +02:00
|
|
|
*/
|
|
|
|
|
|
2020-10-23 19:45:08 +02:00
|
|
|
const int currentBlockHeight = FloweePay::instance()->headerChainHeight();
|
2020-06-08 14:37:29 +02:00
|
|
|
|
2020-10-23 19:45:08 +02:00
|
|
|
QList<UnspentOutput> unspentOutputs;
|
|
|
|
|
std::map<uint64_t, size_t> utxosBySize;
|
|
|
|
|
unspentOutputs.reserve(m_unspentOutputs.size());
|
2020-06-08 14:37:29 +02:00
|
|
|
for (auto iter = m_unspentOutputs.begin(); iter != m_unspentOutputs.end(); ++iter) {
|
2020-11-02 21:49:06 +01:00
|
|
|
if (m_autoLockedOutputs.find(iter->first) != m_autoLockedOutputs.end())
|
|
|
|
|
continue;
|
2020-10-23 19:45:08 +02:00
|
|
|
UnspentOutput out;
|
|
|
|
|
out.value = iter->second;
|
|
|
|
|
out.outputRef = OutputRef(iter->first);
|
|
|
|
|
auto wtxIter = m_walletTransactions.find(out.outputRef.txIndex());
|
|
|
|
|
Q_ASSERT(wtxIter != m_walletTransactions.end());
|
2020-06-08 14:37:29 +02:00
|
|
|
|
2020-10-23 19:45:08 +02:00
|
|
|
int h = wtxIter->second.minedBlockHeight;
|
2020-11-06 18:16:39 +01:00
|
|
|
if (h == WalletPriv::Unconfirmed) {
|
2020-10-23 19:45:08 +02:00
|
|
|
out.score = -10; // unconfirmed.
|
2020-11-06 20:05:42 +01:00
|
|
|
} else if (wtxIter->second.isCoinbase
|
|
|
|
|
&& h + MATURATION_AGE >= m_lastBlockHeightSeen) {
|
|
|
|
|
// don't spend an immature coinbase
|
|
|
|
|
continue;
|
2020-10-23 19:45:08 +02:00
|
|
|
} else {
|
|
|
|
|
const int diff = currentBlockHeight - h;
|
|
|
|
|
if (diff > 4024)
|
|
|
|
|
out.score = 50;
|
|
|
|
|
else if (diff > 1008)
|
|
|
|
|
out.score = 30;
|
|
|
|
|
else if (diff > 144)
|
|
|
|
|
out.score = 10;
|
2020-06-08 14:37:29 +02:00
|
|
|
}
|
2020-10-23 19:45:08 +02:00
|
|
|
utxosBySize.insert(std::make_pair(iter->second, unspentOutputs.size()));
|
|
|
|
|
unspentOutputs.push_back(out);
|
|
|
|
|
}
|
2020-06-08 14:37:29 +02:00
|
|
|
|
2020-10-23 19:45:08 +02:00
|
|
|
// First simply walk from oldest to newest until funded
|
|
|
|
|
OutputSet bestSet;
|
|
|
|
|
int bestScore = 0;
|
|
|
|
|
bestSet.fee = txSize * feePerByte;
|
|
|
|
|
for (auto iter = unspentOutputs.begin(); iter != unspentOutputs.end(); ++iter) {
|
|
|
|
|
bestSet.outputs.push_back(OutputRef(iter->outputRef));
|
|
|
|
|
bestSet.totalSats += iter->value;
|
|
|
|
|
bestSet.fee += BYTES_PER_OUTPUT * feePerByte;
|
|
|
|
|
bestScore += iter->score;
|
|
|
|
|
|
2020-10-23 22:34:34 +02:00
|
|
|
if (output != -1 && bestSet.totalSats - bestSet.fee >= output)
|
2020-10-23 19:45:08 +02:00
|
|
|
break;
|
|
|
|
|
}
|
2020-10-23 22:34:34 +02:00
|
|
|
if (output == -1) { // the magic number to return all outputs
|
|
|
|
|
change = 0;
|
|
|
|
|
return bestSet;
|
|
|
|
|
}
|
2020-10-23 19:45:08 +02:00
|
|
|
if (bestSet.totalSats - bestSet.fee < output)
|
2020-10-23 20:20:42 +02:00
|
|
|
return OutputSet();
|
2020-10-23 19:45:08 +02:00
|
|
|
|
|
|
|
|
bestScore += scoreForSolution(bestSet.outputs.size(),
|
|
|
|
|
bestSet.totalSats - bestSet.fee - output, unspentOutputs.size());
|
|
|
|
|
// try a new set.
|
|
|
|
|
OutputSet current;
|
|
|
|
|
int score = 0;
|
|
|
|
|
current.fee = txSize * feePerByte;
|
|
|
|
|
auto iterBySize = utxosBySize.end();
|
|
|
|
|
while (iterBySize != utxosBySize.begin()) {
|
|
|
|
|
--iterBySize;
|
|
|
|
|
const auto &utxo = unspentOutputs.at(iterBySize->second);
|
|
|
|
|
current.outputs.push_back(utxo.outputRef);
|
|
|
|
|
current.totalSats += utxo.value;
|
|
|
|
|
current.fee += BYTES_PER_OUTPUT * feePerByte;
|
|
|
|
|
score += utxo.score;
|
|
|
|
|
|
|
|
|
|
if (current.totalSats - current.fee >= output)
|
2020-06-08 14:37:29 +02:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-23 19:45:08 +02:00
|
|
|
if (current.totalSats - current.fee >= output) {
|
|
|
|
|
score += scoreForSolution(current.outputs.size(),
|
|
|
|
|
current.totalSats - current.fee - output, unspentOutputs.size());
|
|
|
|
|
|
|
|
|
|
// compare with the cost of oldest to newest.
|
|
|
|
|
if (score > bestScore) {
|
|
|
|
|
bestScore = score;
|
|
|
|
|
bestSet = current;
|
2020-06-08 14:37:29 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-23 19:45:08 +02:00
|
|
|
// Last we use random sets.
|
|
|
|
|
for (int setIndex = 0; setIndex < 50; ++setIndex) {
|
|
|
|
|
current = OutputSet();
|
|
|
|
|
score = 0;
|
|
|
|
|
current.fee = txSize * feePerByte;
|
|
|
|
|
auto outputs = unspentOutputs;
|
|
|
|
|
do {
|
|
|
|
|
Q_ASSERT(!outputs.empty());
|
|
|
|
|
const int index = static_cast<int>(rand() % outputs.size());
|
|
|
|
|
Q_ASSERT(outputs.size() > index);
|
|
|
|
|
const auto &out = outputs[index];
|
|
|
|
|
current.outputs.push_back(out.outputRef);
|
|
|
|
|
current.totalSats += out.value;
|
|
|
|
|
current.fee += BYTES_PER_OUTPUT * feePerByte;
|
|
|
|
|
score += out.score;
|
|
|
|
|
|
|
|
|
|
outputs.removeAt(index); // take it.
|
|
|
|
|
} while (current.totalSats - current.fee < output);
|
|
|
|
|
|
|
|
|
|
score += scoreForSolution(current.outputs.size(),
|
|
|
|
|
current.totalSats - current.fee - output, unspentOutputs.size());
|
|
|
|
|
Q_ASSERT(current.totalSats - current.fee >= output);
|
|
|
|
|
if (score > bestScore) {
|
|
|
|
|
bestScore = score;
|
|
|
|
|
bestSet = current;
|
|
|
|
|
}
|
2020-06-08 14:37:29 +02:00
|
|
|
}
|
|
|
|
|
|
2020-10-23 19:45:08 +02:00
|
|
|
change = current.totalSats - current.fee - output;
|
|
|
|
|
return current;
|
2020-06-08 14:37:29 +02:00
|
|
|
}
|
|
|
|
|
|
2020-11-04 21:58:17 +01:00
|
|
|
void Wallet::setLastSynchedBlockHeight(int height)
|
2020-10-17 17:34:40 +02:00
|
|
|
{
|
2020-11-05 22:21:50 +01:00
|
|
|
if (m_lastBlockHeightSeen == height)
|
|
|
|
|
return;
|
|
|
|
|
m_walletChanged = true;
|
|
|
|
|
m_lastBlockHeightSeen = height;
|
2020-10-17 17:34:40 +02:00
|
|
|
emit lastBlockSynchedChanged();
|
2020-11-06 22:15:03 +01:00
|
|
|
|
|
|
|
|
recalculateBalance();
|
2020-11-06 20:05:42 +01:00
|
|
|
emit utxosChanged(); // in case there was an immature coinbase, this updates the balance
|
2020-11-04 21:58:17 +01:00
|
|
|
|
|
|
|
|
if (height == FloweePay::instance()->headerChainHeight()) {
|
2020-11-05 22:21:50 +01:00
|
|
|
// start this in my own thread and free of mutex-locks
|
|
|
|
|
QTimer::singleShot(0, this, SLOT(broadcastUnconfirmed()));
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-11-04 21:58:17 +01:00
|
|
|
|
2020-11-05 22:21:50 +01:00
|
|
|
void Wallet::broadcastUnconfirmed()
|
|
|
|
|
{
|
|
|
|
|
Q_ASSERT(thread() == QThread::currentThread());
|
2020-11-04 21:58:17 +01:00
|
|
|
|
2020-11-05 22:21:50 +01:00
|
|
|
// we are (again) up-to-date.
|
|
|
|
|
// Lets broadcast any transactions that have not yet been confirmed.
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
m_broadcastingTransactions.clear();
|
|
|
|
|
std::unique_ptr<Streaming::BufferPool> pool;
|
2020-11-04 21:58:17 +01:00
|
|
|
|
2020-11-05 22:21:50 +01:00
|
|
|
for (auto iter = m_walletTransactions.begin();
|
|
|
|
|
iter != m_walletTransactions.end(); ++iter) {
|
|
|
|
|
|
2020-11-06 18:16:39 +01:00
|
|
|
if (iter->second.minedBlockHeight == WalletPriv::Unconfirmed) {
|
2020-11-05 22:21:50 +01:00
|
|
|
if (pool.get() == nullptr)
|
|
|
|
|
pool.reset(new Streaming::BufferPool());
|
|
|
|
|
|
|
|
|
|
auto tx = loadTransaction(iter->second.txid, *pool);
|
|
|
|
|
if (tx.data().size() > 64) {
|
|
|
|
|
auto bc = std::make_shared<WalletInfoObject>(this, iter->first, tx);
|
|
|
|
|
bc->moveToThread(thread());
|
|
|
|
|
logDebug() << " broadcasting transaction" << tx.createHash() << tx.size();
|
|
|
|
|
m_broadcastingTransactions.append(bc);
|
|
|
|
|
FloweePay::instance()->p2pNet()->connectionManager().broadcastTransaction(bc);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
logCritical() << "Unconfirmed transaction could not be found on disk!";
|
2020-11-04 21:58:17 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-10-17 17:34:40 +02:00
|
|
|
}
|
|
|
|
|
|
2020-05-24 13:20:03 +02:00
|
|
|
PrivacySegment * Wallet::segment() const
|
|
|
|
|
{
|
|
|
|
|
return m_segment.get();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::createNewPrivateKey(int currentBlockheight)
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
WalletSecret secret;
|
|
|
|
|
secret.privKey.MakeNewKey();
|
|
|
|
|
|
|
|
|
|
const CPubKey pubkey = secret.privKey.GetPubKey();
|
|
|
|
|
secret.address = pubkey.GetID();
|
|
|
|
|
secret.initialHeight = currentBlockheight;
|
|
|
|
|
m_walletSecrets.insert(std::make_pair(m_nextWalletSecretId++, secret));
|
|
|
|
|
m_secretsChanged = true;
|
|
|
|
|
saveSecrets();
|
|
|
|
|
|
2020-06-08 14:31:45 +02:00
|
|
|
m_segment->addKeyToFilter(pubkey.GetID(), currentBlockheight);
|
2020-05-24 13:20:03 +02:00
|
|
|
|
|
|
|
|
rebuildBloom();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Wallet::addPrivateKey(const QString &privKey, int startBlockHeight)
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
CBase58Data encodedData;
|
|
|
|
|
auto bytes = privKey.toLatin1();
|
|
|
|
|
encodedData.SetString(bytes.constData());
|
2020-10-30 18:06:58 +01:00
|
|
|
if (encodedData.isMainnetPrivKey() || encodedData.isTestnetPrivKey()) {
|
2020-05-24 13:20:03 +02:00
|
|
|
WalletSecret secret;
|
|
|
|
|
secret.privKey.Set(encodedData.data().begin(), encodedData.data().begin() + 32,
|
|
|
|
|
encodedData.data().size() > 32 && encodedData.data()[32] == 1);
|
|
|
|
|
|
|
|
|
|
// TODO loop over secrets and avoid adding one privkey twice.
|
|
|
|
|
|
|
|
|
|
const CPubKey pubkey = secret.privKey.GetPubKey();
|
|
|
|
|
secret.address = pubkey.GetID();
|
|
|
|
|
secret.initialHeight = startBlockHeight;
|
|
|
|
|
m_walletSecrets.insert(std::make_pair(m_nextWalletSecretId++, secret));
|
|
|
|
|
m_secretsChanged = true;
|
|
|
|
|
saveSecrets();
|
|
|
|
|
|
2020-06-08 14:31:45 +02:00
|
|
|
m_segment->addKeyToFilter(pubkey.GetID(), startBlockHeight);
|
2020-05-24 13:20:03 +02:00
|
|
|
|
|
|
|
|
rebuildBloom();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
logFatal() << "ERROR. Wallet: added string is not a private key";
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::loadSecrets()
|
|
|
|
|
{
|
|
|
|
|
std::ifstream in((m_basedir / "secrets.dat").string());
|
|
|
|
|
if (!in.is_open())
|
2020-10-15 19:18:54 +02:00
|
|
|
throw std::runtime_error("Missing secrets.dat");
|
2020-05-24 13:20:03 +02:00
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
const auto dataSize = boost::filesystem::file_size(m_basedir / "secrets.dat");
|
|
|
|
|
Streaming::BufferPool pool(dataSize);
|
|
|
|
|
in.read(pool.begin(), dataSize);
|
|
|
|
|
Streaming::MessageParser parser(pool.commit(dataSize));
|
|
|
|
|
WalletSecret secret;
|
|
|
|
|
int index = 0;
|
|
|
|
|
while (parser.next() == Streaming::FoundTag) {
|
2020-11-06 18:16:39 +01:00
|
|
|
if (parser.tag() == WalletPriv::Separator) {
|
2020-05-24 13:20:03 +02:00
|
|
|
if (index > 0 && secret.address.size() > 0) {
|
|
|
|
|
m_walletSecrets.insert(std::make_pair(index, secret));
|
|
|
|
|
m_nextWalletSecretId = std::max(m_nextWalletSecretId, index);
|
|
|
|
|
}
|
|
|
|
|
secret = WalletSecret();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::Index) {
|
2020-05-24 13:20:03 +02:00
|
|
|
index = parser.intData();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::PrivKey) {
|
2020-05-24 13:20:03 +02:00
|
|
|
auto d = parser.unsignedBytesData();
|
|
|
|
|
secret.privKey.Set(d.begin(), d.end(), true);
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::PubKeyHash) {
|
2020-05-24 13:20:03 +02:00
|
|
|
auto d = parser.bytesDataBuffer();
|
|
|
|
|
secret.address = CKeyID(d.begin());
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::HeightCreated) {
|
2020-05-24 13:20:03 +02:00
|
|
|
secret.initialHeight = parser.intData();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::IsSingleAddressWallet) {
|
2020-06-11 10:24:05 +02:00
|
|
|
m_singleAddressWallet = parser.boolData();
|
|
|
|
|
}
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
|
|
|
|
m_secretsChanged = false;
|
|
|
|
|
++m_nextWalletSecretId;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::saveSecrets()
|
|
|
|
|
{
|
|
|
|
|
// mutex already locked
|
|
|
|
|
if (!m_secretsChanged)
|
|
|
|
|
return;
|
|
|
|
|
Streaming::BufferPool pool(m_walletSecrets.size() * 70);
|
|
|
|
|
Streaming::MessageBuilder builder(pool);
|
|
|
|
|
for (const auto &item : m_walletSecrets) {
|
2020-11-06 18:16:39 +01:00
|
|
|
builder.add(WalletPriv::Index, item.first);
|
|
|
|
|
builder.addByteArray(WalletPriv::PrivKey, item.second.privKey.begin(), item.second.privKey.size());
|
|
|
|
|
builder.addByteArray(WalletPriv::PubKeyHash, item.second.address.begin(), item.second.address.size());
|
2020-05-24 13:20:03 +02:00
|
|
|
if (item.second.initialHeight > 0)
|
2020-11-06 18:16:39 +01:00
|
|
|
builder.add(WalletPriv::HeightCreated, item.second.initialHeight);
|
2020-05-24 13:20:03 +02:00
|
|
|
|
2020-11-06 18:16:39 +01:00
|
|
|
builder.add(WalletPriv::Separator, true);
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-06-11 10:24:05 +02:00
|
|
|
if (m_singleAddressWallet)
|
2020-11-06 18:16:39 +01:00
|
|
|
builder.add(WalletPriv::IsSingleAddressWallet, true);
|
2020-05-24 13:20:03 +02:00
|
|
|
auto data = builder.buffer();
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
boost::filesystem::create_directories(m_basedir);
|
|
|
|
|
boost::filesystem::remove(m_basedir / "secrets.dat~");
|
|
|
|
|
|
|
|
|
|
std::ofstream outFile((m_basedir / "secrets.dat~").string());
|
|
|
|
|
outFile.write(data.begin(), data.size());
|
2020-05-27 18:35:37 +02:00
|
|
|
outFile.flush();
|
|
|
|
|
outFile.close();
|
2020-05-24 13:20:03 +02:00
|
|
|
boost::filesystem::rename(m_basedir / "secrets.dat~", m_basedir / "secrets.dat");
|
|
|
|
|
} catch (const std::exception &e) {
|
|
|
|
|
logFatal() << "Failed to save the database. Reason:" << e.what();
|
|
|
|
|
}
|
|
|
|
|
m_secretsChanged = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::loadWallet()
|
|
|
|
|
{
|
|
|
|
|
std::ifstream in((m_basedir / "wallet.dat").string());
|
|
|
|
|
if (!in.is_open())
|
|
|
|
|
return;
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
const auto dataSize = boost::filesystem::file_size(m_basedir / "wallet.dat");
|
|
|
|
|
Streaming::BufferPool pool(dataSize);
|
|
|
|
|
in.read(pool.begin(), dataSize);
|
|
|
|
|
Streaming::MessageParser parser(pool.commit(dataSize));
|
|
|
|
|
WalletTransaction wtx;
|
|
|
|
|
int index = 0;
|
|
|
|
|
int inputIndex = -1;
|
|
|
|
|
int outputIndex = -1;
|
|
|
|
|
int tmp = 0;
|
2020-11-06 18:16:39 +01:00
|
|
|
WalletPriv::InputLockStateEnum inputLock = WalletPriv::Unlocked;
|
2020-11-03 20:16:36 +01:00
|
|
|
int autoLockId = -1;
|
2020-05-24 13:20:03 +02:00
|
|
|
Output output;
|
|
|
|
|
QSet<int> newTx;
|
2020-11-05 22:18:32 +01:00
|
|
|
int highestBlockHeight = 0;
|
2020-05-24 13:20:03 +02:00
|
|
|
while (parser.next() == Streaming::FoundTag) {
|
2020-11-06 18:16:39 +01:00
|
|
|
if (parser.tag() == WalletPriv::Separator) {
|
2020-05-24 13:20:03 +02:00
|
|
|
assert(index > 0);
|
|
|
|
|
assert(m_walletTransactions.find(index) == m_walletTransactions.end());
|
|
|
|
|
assert(!wtx.inputToWTX.empty() || !wtx.outputs.empty());
|
|
|
|
|
m_walletTransactions.insert(std::make_pair(index, wtx));
|
|
|
|
|
m_txidCash.insert(std::make_pair(wtx.txid, index));
|
|
|
|
|
m_nextWalletTransactionId = std::max(m_nextWalletTransactionId, index);
|
|
|
|
|
// insert outputs of new tx.
|
|
|
|
|
for (auto i = wtx.outputs.begin(); i != wtx.outputs.end(); ++i) {
|
2020-06-08 14:31:45 +02:00
|
|
|
OutputRef ref(index, i->first);
|
|
|
|
|
m_unspentOutputs.insert(std::make_pair(ref.encoded(), i->second.value));
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
|
|
|
|
newTx.insert(index);
|
|
|
|
|
|
2020-10-16 19:25:52 +02:00
|
|
|
#ifdef DEBUGUTXO
|
|
|
|
|
logFatal() << "Wallet has tx: " << wtx.txid << "@" << wtx.minedBlockHeight;
|
|
|
|
|
for (auto pair : wtx.outputs) {
|
|
|
|
|
logFatal() << " ++ " << pair.first << pair.second.value << "sat";
|
|
|
|
|
}
|
|
|
|
|
for (auto pair : wtx.inputToWTX) {
|
|
|
|
|
OutputRef ref(pair.second);
|
|
|
|
|
logFatal() << " -- " << pair.first << ref.txIndex() << ref.outputIndex();
|
|
|
|
|
auto w = m_walletTransactions.find(ref.txIndex());
|
|
|
|
|
if (w != m_walletTransactions.end())
|
|
|
|
|
logFatal() << " " << w->second.txid;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2020-05-24 13:20:03 +02:00
|
|
|
wtx = WalletTransaction();
|
|
|
|
|
inputIndex = -1;
|
|
|
|
|
outputIndex = -1;
|
|
|
|
|
output = Output();
|
2020-11-06 20:05:42 +01:00
|
|
|
index = -1;
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::Index) {
|
2020-05-24 13:20:03 +02:00
|
|
|
index = parser.intData();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::TxId) {
|
2020-05-24 13:20:03 +02:00
|
|
|
wtx.txid = parser.uint256Data();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::BlockHash) {
|
2020-05-24 13:20:03 +02:00
|
|
|
wtx.minedBlock = parser.uint256Data();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::BlockHeight) {
|
2020-05-24 13:20:03 +02:00
|
|
|
wtx.minedBlockHeight = parser.intData();
|
2020-11-05 22:18:32 +01:00
|
|
|
highestBlockHeight = std::max(parser.intData(), highestBlockHeight);
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-11-06 20:05:42 +01:00
|
|
|
else if (parser.tag() == WalletPriv::OutputFromCoinbase) {
|
|
|
|
|
wtx.isCoinbase = parser.boolData();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::InputIndex) {
|
2020-05-24 13:20:03 +02:00
|
|
|
inputIndex = parser.intData();
|
2020-11-02 21:49:06 +01:00
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::InputLockState) {
|
|
|
|
|
if (parser.intData() == WalletPriv::AutoLocked || parser.intData() == WalletPriv::UserLocked)
|
|
|
|
|
inputLock = static_cast<WalletPriv::InputLockStateEnum>(parser.intData());
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::InputLockAutoSpender) {
|
2020-11-03 20:16:36 +01:00
|
|
|
autoLockId = parser.intData();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::InputSpendsTx) {
|
2020-05-24 13:20:03 +02:00
|
|
|
tmp = parser.intData();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::InputSpendsOutput) {
|
2020-05-24 13:20:03 +02:00
|
|
|
assert(inputIndex >= 0);
|
|
|
|
|
assert(tmp != 0);
|
2020-06-08 14:31:45 +02:00
|
|
|
assert(parser.longData() < 0xFFFF);
|
|
|
|
|
OutputRef ref(tmp, parser.intData());
|
|
|
|
|
wtx.inputToWTX.insert(std::make_pair(inputIndex, ref.encoded()));
|
2020-11-06 18:16:39 +01:00
|
|
|
if (inputLock == WalletPriv::AutoLocked && autoLockId != -1)
|
2020-11-03 20:16:36 +01:00
|
|
|
m_autoLockedOutputs.insert(std::make_pair(ref.encoded(), autoLockId));
|
2020-05-24 13:20:03 +02:00
|
|
|
tmp = 0;
|
2020-11-06 18:16:39 +01:00
|
|
|
inputLock = WalletPriv::Unlocked;
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::OutputIndex) {
|
2020-05-24 13:20:03 +02:00
|
|
|
outputIndex = parser.intData();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::OutputValue) {
|
2020-05-24 13:20:03 +02:00
|
|
|
output.value = parser.longData();
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::KeyStoreIndex) {
|
2020-05-24 13:20:03 +02:00
|
|
|
assert(outputIndex >= 0);
|
|
|
|
|
assert(output.value > 0);
|
|
|
|
|
output.walletSecretId = parser.intData();
|
|
|
|
|
wtx.outputs.insert(std::make_pair(outputIndex, output));
|
|
|
|
|
outputIndex = -1;
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::WalletName) {
|
2020-05-24 13:20:03 +02:00
|
|
|
assert(parser.isString());
|
|
|
|
|
m_name = QString::fromUtf8(parser.stringData().c_str(), parser.dataLength());
|
|
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
else if (parser.tag() == WalletPriv::LastSynchedBlock) {
|
2020-11-05 22:18:32 +01:00
|
|
|
highestBlockHeight = std::max(parser.intData(), highestBlockHeight);
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// after inserting all outputs during load, now remove all inputs these tx's spent.
|
|
|
|
|
for (auto index : newTx) {
|
|
|
|
|
auto iter = m_walletTransactions.find(index);
|
|
|
|
|
assert(iter != m_walletTransactions.end());
|
|
|
|
|
|
2020-11-06 18:41:12 +01:00
|
|
|
if (iter->second.minedBlockHeight != WalletPriv::Unconfirmed) {
|
|
|
|
|
// remove UTXOs this Tx spent
|
|
|
|
|
for (auto i = iter->second.inputToWTX.begin(); i != iter->second.inputToWTX.end(); ++i) {
|
|
|
|
|
auto utxo = m_unspentOutputs.find(i->second);
|
|
|
|
|
assert(utxo != m_unspentOutputs.end()); // Loading should be done in-order to avoid this.
|
|
|
|
|
if (utxo != m_unspentOutputs.end())
|
|
|
|
|
m_unspentOutputs.erase(utxo);
|
|
|
|
|
}
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
|
|
|
|
}
|
2020-11-06 22:15:03 +01:00
|
|
|
if (highestBlockHeight > 0) {
|
|
|
|
|
m_segment->blockSynched(highestBlockHeight);
|
|
|
|
|
m_segment->blockSynched(highestBlockHeight); // yes, twice.
|
|
|
|
|
} else {
|
|
|
|
|
// otherwise the blockSynced() implicitly calls this.
|
|
|
|
|
recalculateBalance();
|
|
|
|
|
}
|
2020-05-24 13:20:03 +02:00
|
|
|
|
2020-10-16 19:25:52 +02:00
|
|
|
#ifdef DEBUGUTXO
|
|
|
|
|
for (auto output : m_unspentOutputs) {
|
|
|
|
|
OutputRef ref(output.first);
|
|
|
|
|
auto utxo = m_walletTransactions.find(ref.txIndex());
|
|
|
|
|
assert(utxo != m_walletTransactions.end());
|
|
|
|
|
auto out = utxo->second.outputs.find(ref.outputIndex());
|
|
|
|
|
assert(out != utxo->second.outputs.end());
|
|
|
|
|
assert(out->second.value == output.second);
|
|
|
|
|
logFatal() << "Unspent: " << utxo->second.txid << ref.outputIndex() << "\t->" << out->second.value << "sats";
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2020-05-24 13:20:03 +02:00
|
|
|
#ifndef NDEBUG
|
|
|
|
|
// sanity check: the inputs should resolve to transactions in our list.
|
|
|
|
|
for (auto tx : m_walletTransactions) {
|
|
|
|
|
for (auto in : tx.second.inputToWTX) {
|
|
|
|
|
auto key = in.second;
|
|
|
|
|
int outputIndex = key & 0xFFFF;
|
|
|
|
|
key >>= 16;
|
|
|
|
|
assert(m_walletTransactions.find(key) != m_walletTransactions.end());
|
|
|
|
|
auto spendingTx = m_walletTransactions.at(key);
|
|
|
|
|
assert(spendingTx.outputs.find(outputIndex) != spendingTx.outputs.end());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
m_walletChanged = false;
|
|
|
|
|
++m_nextWalletTransactionId;
|
|
|
|
|
emit utxosChanged();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::saveWallet()
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
if (!m_walletChanged)
|
|
|
|
|
return;
|
|
|
|
|
Streaming::BufferPool pool(m_walletTransactions.size() * 110 + m_name.size() * 3 + 100);
|
|
|
|
|
Streaming::MessageBuilder builder(pool);
|
|
|
|
|
for (const auto &item : m_walletTransactions) {
|
2020-11-06 18:16:39 +01:00
|
|
|
builder.add(WalletPriv::Index, item.first);
|
|
|
|
|
builder.add(WalletPriv::TxId, item.second.txid);
|
|
|
|
|
builder.add(WalletPriv::BlockHash, item.second.minedBlock);
|
|
|
|
|
builder.add(WalletPriv::BlockHeight, item.second.minedBlockHeight);
|
2020-11-06 20:05:42 +01:00
|
|
|
if (item.second.isCoinbase)
|
|
|
|
|
builder.add(WalletPriv::OutputFromCoinbase, true);
|
2020-05-24 13:20:03 +02:00
|
|
|
for (auto i = item.second.inputToWTX.begin(); i != item.second.inputToWTX.end(); ++i) {
|
2020-11-06 18:16:39 +01:00
|
|
|
builder.add(WalletPriv::InputIndex, i->first);
|
2020-11-06 22:15:03 +01:00
|
|
|
const OutputRef ref(i->second);
|
|
|
|
|
builder.add(WalletPriv::InputSpendsTx, ref.txIndex());
|
|
|
|
|
builder.add(WalletPriv::InputSpendsOutput, ref.outputIndex());
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
|
|
|
|
for (auto i = item.second.outputs.begin(); i != item.second.outputs.end(); ++i) {
|
2020-11-06 18:16:39 +01:00
|
|
|
builder.add(WalletPriv::OutputIndex, i->first);
|
|
|
|
|
builder.add(WalletPriv::OutputValue, i->second.value);
|
|
|
|
|
builder.add(WalletPriv::KeyStoreIndex, i->second.walletSecretId);
|
2020-11-06 22:15:03 +01:00
|
|
|
|
|
|
|
|
auto lock = m_autoLockedOutputs.find(OutputRef(item.first, i->first).encoded());
|
|
|
|
|
if (lock != m_autoLockedOutputs.end()) {
|
|
|
|
|
builder.add(WalletPriv::InputLockState, WalletPriv::AutoLocked);
|
|
|
|
|
builder.add(WalletPriv::InputLockAutoSpender, lock->second);
|
|
|
|
|
}
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
builder.add(WalletPriv::Separator, true);
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-11-06 18:16:39 +01:00
|
|
|
builder.add(WalletPriv::LastSynchedBlock, m_segment->lastBlockSynched());
|
|
|
|
|
builder.add(WalletPriv::WalletName, m_name.toUtf8().toStdString());
|
2020-05-24 13:20:03 +02:00
|
|
|
|
|
|
|
|
auto data = builder.buffer();
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
boost::filesystem::create_directories(m_basedir);
|
|
|
|
|
boost::filesystem::remove(m_basedir / "wallet.dat~");
|
|
|
|
|
|
|
|
|
|
std::ofstream outFile((m_basedir / "wallet.dat~").string());
|
|
|
|
|
outFile.write(data.begin(), data.size());
|
2020-05-27 18:35:37 +02:00
|
|
|
outFile.flush();
|
|
|
|
|
outFile.close();
|
2020-05-24 13:20:03 +02:00
|
|
|
boost::filesystem::rename(m_basedir / "wallet.dat~", m_basedir / "wallet.dat");
|
|
|
|
|
} catch (const std::exception &e) {
|
|
|
|
|
logFatal() << "Failed to save the database. Reason:" << e.what();
|
|
|
|
|
}
|
|
|
|
|
m_walletChanged = false;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-06 22:15:03 +01:00
|
|
|
|
|
|
|
|
void Wallet::recalculateBalance()
|
2020-05-24 13:20:03 +02:00
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
2020-11-06 22:15:03 +01:00
|
|
|
qint64 balanceConfirmed = 0;
|
|
|
|
|
qint64 balanceImmature = 0;
|
|
|
|
|
qint64 balanceUnconfirmed = 0;
|
2020-05-24 13:20:03 +02:00
|
|
|
for (auto utxo : m_unspentOutputs) {
|
2020-11-06 20:05:42 +01:00
|
|
|
auto wtx = m_walletTransactions.find(OutputRef(utxo.first).txIndex());
|
|
|
|
|
Q_ASSERT(wtx != m_walletTransactions.end());
|
2020-11-06 22:15:03 +01:00
|
|
|
const int h = wtx->second.minedBlockHeight;
|
|
|
|
|
if (h == WalletPriv::Rejected)
|
2020-11-06 20:05:42 +01:00
|
|
|
continue;
|
2020-11-06 22:15:03 +01:00
|
|
|
else if (m_autoLockedOutputs.find(utxo.first) != m_autoLockedOutputs.end())
|
|
|
|
|
continue;
|
|
|
|
|
else if (h == WalletPriv::Unconfirmed)
|
|
|
|
|
balanceUnconfirmed += utxo.second;
|
|
|
|
|
else if (wtx->second.isCoinbase && h + MATURATION_AGE > m_lastBlockHeightSeen)
|
|
|
|
|
balanceImmature += utxo.second;
|
|
|
|
|
else
|
|
|
|
|
balanceConfirmed += utxo.second;
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
2020-11-06 22:15:03 +01:00
|
|
|
if (m_balanceConfirmed == balanceConfirmed
|
|
|
|
|
&& m_balanceImmature == balanceImmature
|
|
|
|
|
&& m_balanceUnconfirmed == balanceUnconfirmed)
|
|
|
|
|
return;
|
|
|
|
|
m_balanceConfirmed = balanceConfirmed;
|
|
|
|
|
m_balanceImmature = balanceImmature;
|
|
|
|
|
m_balanceUnconfirmed = balanceUnconfirmed;
|
|
|
|
|
emit balanceChanged();
|
2020-05-24 13:20:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int Wallet::unspentOutputCount() const
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
return m_unspentOutputs.size();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int Wallet::historicalOutputCount() const
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_lock);
|
|
|
|
|
int count = 0;
|
|
|
|
|
for (auto wtx : m_walletTransactions) {
|
|
|
|
|
count += wtx.second.outputs.size();
|
|
|
|
|
}
|
|
|
|
|
return count;
|
|
|
|
|
}
|