|
|
|
@ -2,6 +2,11 @@
|
|
|
|
|
// Copyright (c) 2014-2020, The Monero Project.
|
|
|
|
|
|
|
|
|
|
#include "Wallet.h"
|
|
|
|
|
|
|
|
|
|
#include <chrono>
|
|
|
|
|
#include <stdexcept>
|
|
|
|
|
#include <thread>
|
|
|
|
|
|
|
|
|
|
#include "PendingTransaction.h"
|
|
|
|
|
#include "UnsignedTransaction.h"
|
|
|
|
|
#include "TransactionHistory.h"
|
|
|
|
@ -27,6 +32,8 @@
|
|
|
|
|
#include <QVector>
|
|
|
|
|
#include <QMutexLocker>
|
|
|
|
|
|
|
|
|
|
#include "utils/ScopeGuard.h"
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
static const int DAEMON_BLOCKCHAIN_HEIGHT_CACHE_TTL_SECONDS = 5;
|
|
|
|
|
static const int DAEMON_BLOCKCHAIN_TARGET_HEIGHT_CACHE_TTL_SECONDS = 30;
|
|
|
|
@ -100,6 +107,19 @@ bool Wallet::disconnected() const
|
|
|
|
|
return m_disconnected;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Wallet::refreshing() const
|
|
|
|
|
{
|
|
|
|
|
return m_refreshing;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::refreshingSet(bool value)
|
|
|
|
|
{
|
|
|
|
|
if (m_refreshing.exchange(value) != value)
|
|
|
|
|
{
|
|
|
|
|
emit refreshingChanged();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::setConnectionStatus(ConnectionStatus value)
|
|
|
|
|
{
|
|
|
|
|
if (m_connectionStatus == value)
|
|
|
|
@ -120,6 +140,29 @@ void Wallet::setConnectionStatus(ConnectionStatus value)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString Wallet::getProxyAddress() const
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_proxyMutex);
|
|
|
|
|
return m_proxyAddress;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::setProxyAddress(QString address)
|
|
|
|
|
{
|
|
|
|
|
m_scheduler.run([this, address] {
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_proxyMutex);
|
|
|
|
|
|
|
|
|
|
if (!m_walletImpl->setProxy(address.toStdString()))
|
|
|
|
|
{
|
|
|
|
|
qCritical() << "failed to set proxy" << address;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_proxyAddress = address;
|
|
|
|
|
}
|
|
|
|
|
emit proxyAddressChanged();
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Wallet::synchronized() const
|
|
|
|
|
{
|
|
|
|
|
return m_walletImpl->synchronized();
|
|
|
|
@ -145,20 +188,18 @@ QString Wallet::path() const
|
|
|
|
|
return QDir::toNativeSeparators(QString::fromStdString(m_walletImpl->path()));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//void Wallet::storeAsync(const QVariant &callback, const QString &path /* = "" */)
|
|
|
|
|
//void Wallet::storeAsync(const QJSValue &callback, const QString &path /* = "" */)
|
|
|
|
|
//{
|
|
|
|
|
// const auto future = m_scheduler.run(
|
|
|
|
|
// [this, path] {
|
|
|
|
|
// QMutexLocker locker(&m_storeMutex);
|
|
|
|
|
|
|
|
|
|
// return QVariantList({m_walletImpl->store(path.toStdString())});
|
|
|
|
|
// QMutexLocker locker(&m_asyncMutex);
|
|
|
|
|
//
|
|
|
|
|
// return QJSValueList({m_walletImpl->store(path.toStdString())});
|
|
|
|
|
// },
|
|
|
|
|
// callback);
|
|
|
|
|
// if (!future.first)
|
|
|
|
|
// {
|
|
|
|
|
// QVariant(callback).call(QVariantList({false}));
|
|
|
|
|
// QJSValue(callback).call(QJSValueList({false}));
|
|
|
|
|
// }
|
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
@ -167,7 +208,7 @@ void Wallet::store(const QString &path)
|
|
|
|
|
m_walletImpl->store(path.toStdString());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Wallet::init(const QString &daemonAddress, bool trustedDaemon, quint64 upperTransactionLimit, bool isRecovering, bool isRecoveringFromDevice, quint64 restoreHeight)
|
|
|
|
|
bool Wallet::init(const QString &daemonAddress, bool trustedDaemon, quint64 upperTransactionLimit, bool isRecovering, bool isRecoveringFromDevice, quint64 restoreHeight, const QString& proxyAddress)
|
|
|
|
|
{
|
|
|
|
|
qDebug() << "init non async";
|
|
|
|
|
if (isRecovering){
|
|
|
|
@ -181,7 +222,20 @@ bool Wallet::init(const QString &daemonAddress, bool trustedDaemon, quint64 uppe
|
|
|
|
|
if (isRecovering || isRecoveringFromDevice) {
|
|
|
|
|
m_walletImpl->setRefreshFromBlockHeight(restoreHeight);
|
|
|
|
|
}
|
|
|
|
|
m_walletImpl->init(daemonAddress.toStdString(), upperTransactionLimit, m_daemonUsername.toStdString(), m_daemonPassword.toStdString());
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_proxyMutex);
|
|
|
|
|
|
|
|
|
|
if (!m_walletImpl->init(daemonAddress.toStdString(), upperTransactionLimit, m_daemonUsername.toStdString(), m_daemonPassword.toStdString(), false, false, proxyAddress.toStdString()))
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
m_proxyAddress = proxyAddress;
|
|
|
|
|
}
|
|
|
|
|
emit proxyAddressChanged();
|
|
|
|
|
|
|
|
|
|
setTrustedDaemon(trustedDaemon);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
@ -193,17 +247,24 @@ void Wallet::setDaemonLogin(const QString &daemonUsername, const QString &daemon
|
|
|
|
|
m_daemonPassword = daemonPassword;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::initAsync(const QString &daemonAddress, bool trustedDaemon, quint64 upperTransactionLimit, bool isRecovering, bool isRecoveringFromDevice, quint64 restoreHeight)
|
|
|
|
|
void Wallet::initAsync(
|
|
|
|
|
const QString &daemonAddress,
|
|
|
|
|
bool trustedDaemon /* = false */,
|
|
|
|
|
quint64 upperTransactionLimit /* = 0 */,
|
|
|
|
|
bool isRecovering /* = false */,
|
|
|
|
|
bool isRecoveringFromDevice /* = false */,
|
|
|
|
|
quint64 restoreHeight /* = 0 */,
|
|
|
|
|
const QString &proxyAddress /* = "" */)
|
|
|
|
|
{
|
|
|
|
|
qDebug() << "initAsync: " + daemonAddress;
|
|
|
|
|
const auto future = m_scheduler.run([this, daemonAddress, trustedDaemon, upperTransactionLimit, isRecovering, isRecoveringFromDevice, restoreHeight] {
|
|
|
|
|
bool success = init(daemonAddress, trustedDaemon, upperTransactionLimit, isRecovering, isRecoveringFromDevice, restoreHeight);
|
|
|
|
|
const auto future = m_scheduler.run([this, daemonAddress, trustedDaemon, upperTransactionLimit, isRecovering, isRecoveringFromDevice, restoreHeight, proxyAddress] {
|
|
|
|
|
bool success = init(daemonAddress, trustedDaemon, upperTransactionLimit, isRecovering, isRecoveringFromDevice, restoreHeight, proxyAddress);
|
|
|
|
|
if (success)
|
|
|
|
|
{
|
|
|
|
|
emit walletCreationHeightChanged();
|
|
|
|
|
qDebug() << "init async finished - starting refresh";
|
|
|
|
|
connected(true);
|
|
|
|
|
m_walletImpl->startRefresh();
|
|
|
|
|
startRefresh();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
if (future.first)
|
|
|
|
@ -406,42 +467,37 @@ bool Wallet::importKeyImages(const QString& path)
|
|
|
|
|
return m_walletImpl->importKeyImages(path.toStdString());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Wallet::refresh()
|
|
|
|
|
bool Wallet::refresh(bool historyAndSubaddresses /* = true */)
|
|
|
|
|
{
|
|
|
|
|
refreshingSet(true);
|
|
|
|
|
const auto cleanup = sg::make_scope_guard([this]() noexcept {
|
|
|
|
|
refreshingSet(false);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_asyncMutex);
|
|
|
|
|
|
|
|
|
|
bool result = m_walletImpl->refresh();
|
|
|
|
|
if (historyAndSubaddresses)
|
|
|
|
|
{
|
|
|
|
|
m_history->refresh(currentSubaddressAccount());
|
|
|
|
|
m_subaddress->refresh(currentSubaddressAccount());
|
|
|
|
|
m_coins->refresh(currentSubaddressAccount());
|
|
|
|
|
m_subaddressAccount->getAll();
|
|
|
|
|
}
|
|
|
|
|
if (result)
|
|
|
|
|
emit updated();
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::refreshAsync()
|
|
|
|
|
{
|
|
|
|
|
qDebug() << "refresh async";
|
|
|
|
|
m_walletImpl->refreshAsync();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::setAutoRefreshInterval(int seconds)
|
|
|
|
|
{
|
|
|
|
|
m_walletImpl->setAutoRefreshInterval(seconds);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int Wallet::autoRefreshInterval() const
|
|
|
|
|
{
|
|
|
|
|
return m_walletImpl->autoRefreshInterval();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::startRefresh() const
|
|
|
|
|
void Wallet::startRefresh()
|
|
|
|
|
{
|
|
|
|
|
m_walletImpl->startRefresh();
|
|
|
|
|
m_refreshEnabled = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::pauseRefresh() const
|
|
|
|
|
void Wallet::pauseRefresh()
|
|
|
|
|
{
|
|
|
|
|
m_walletImpl->pauseRefresh();
|
|
|
|
|
m_refreshEnabled = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PendingTransaction *Wallet::createTransaction(const QString &dst_addr, const QString &payment_id,
|
|
|
|
@ -556,7 +612,6 @@ void Wallet::disposeTransaction(UnsignedTransaction *t)
|
|
|
|
|
delete t;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// @ TODO: QJSValue could probably be QVARIANT
|
|
|
|
|
//void Wallet::estimateTransactionFeeAsync(const QString &destination,
|
|
|
|
|
// quint64 amount,
|
|
|
|
|
// PendingTransaction::Priority priority,
|
|
|
|
@ -831,7 +886,6 @@ bool Wallet::verifySignedMessage(const QString &message, const QString &address,
|
|
|
|
|
return m_walletImpl->verifySignedMessage(message.toStdString(), address.toStdString(), signature.toStdString());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Wallet::parse_uri(const QString &uri, QString &address, QString &payment_id, uint64_t &amount, QString &tx_description, QString &recipient_name, QVector<QString> &unknown_parameters, QString &error)
|
|
|
|
|
{
|
|
|
|
|
std::string s_address, s_payment_id, s_tx_description, s_recipient_name, s_error;
|
|
|
|
@ -852,6 +906,8 @@ bool Wallet::parse_uri(const QString &uri, QString &address, QString &payment_id
|
|
|
|
|
|
|
|
|
|
bool Wallet::rescanSpent()
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&m_asyncMutex);
|
|
|
|
|
|
|
|
|
|
return m_walletImpl->rescanSpent();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -1020,6 +1076,8 @@ Wallet::Wallet(Monero::Wallet *w, QObject *parent)
|
|
|
|
|
, m_subaddressAccount(nullptr)
|
|
|
|
|
, m_subaddressAccountModel(nullptr)
|
|
|
|
|
, m_coinsModel(nullptr)
|
|
|
|
|
, m_refreshEnabled(false)
|
|
|
|
|
, m_refreshing(false)
|
|
|
|
|
, m_scheduler(this)
|
|
|
|
|
{
|
|
|
|
|
m_history = new TransactionHistory(m_walletImpl->history(), this);
|
|
|
|
@ -1038,11 +1096,13 @@ Wallet::Wallet(Monero::Wallet *w, QObject *parent)
|
|
|
|
|
m_connectionStatusRunning = false;
|
|
|
|
|
m_daemonUsername = "";
|
|
|
|
|
m_daemonPassword = "";
|
|
|
|
|
|
|
|
|
|
startRefreshThread();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Wallet::~Wallet()
|
|
|
|
|
{
|
|
|
|
|
qDebug() << "~Wallet: Closing wallet";
|
|
|
|
|
qDebug("~Wallet: Closing wallet");
|
|
|
|
|
|
|
|
|
|
m_scheduler.shutdownWaitForFinished();
|
|
|
|
|
|
|
|
|
@ -1072,3 +1132,32 @@ Wallet::~Wallet()
|
|
|
|
|
m_walletListener = NULL;
|
|
|
|
|
qDebug("m_walletImpl deleted");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Wallet::startRefreshThread()
|
|
|
|
|
{
|
|
|
|
|
const auto future = m_scheduler.run([this] {
|
|
|
|
|
static constexpr const size_t refreshIntervalSec = 10;
|
|
|
|
|
static constexpr const size_t intervalResolutionMs = 100;
|
|
|
|
|
|
|
|
|
|
auto last = std::chrono::steady_clock::now();
|
|
|
|
|
while (!m_scheduler.stopping())
|
|
|
|
|
{
|
|
|
|
|
if (m_refreshEnabled)
|
|
|
|
|
{
|
|
|
|
|
const auto now = std::chrono::steady_clock::now();
|
|
|
|
|
const auto elapsed = std::chrono::duration_cast<std::chrono::seconds>(now - last).count();
|
|
|
|
|
if (elapsed >= refreshIntervalSec)
|
|
|
|
|
{
|
|
|
|
|
refresh(false);
|
|
|
|
|
last = std::chrono::steady_clock::now();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(intervalResolutionMs));
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
if (!future.first)
|
|
|
|
|
{
|
|
|
|
|
throw std::runtime_error("failed to start auto refresh thread");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|