monero-gui/src/libwalletqt/WalletManager.cpp

398 lines
13 KiB
C++
Raw Normal View History

2016-02-23 15:59:26 +00:00
#include "WalletManager.h"
#include "Wallet.h"
#include "wallet/api/wallet2_api.h"
#include "zxcvbn-c/zxcvbn.h"
2017-01-23 04:15:49 +00:00
#include "QRCodeImageProvider.h"
#include <QFile>
#include <QFileInfo>
#include <QDir>
#include <QDebug>
#include <QUrl>
#include <QtConcurrent/QtConcurrent>
#include <QMutex>
#include <QMutexLocker>
2017-05-03 12:37:33 +00:00
#include <QString>
WalletManager * WalletManager::m_instance = nullptr;
WalletManager *WalletManager::instance()
{
if (!m_instance) {
m_instance = new WalletManager;
}
return m_instance;
}
Wallet *WalletManager::createWallet(const QString &path, const QString &password,
2018-03-05 16:19:45 +00:00
const QString &language, NetworkType::Type nettype)
{
QMutexLocker locker(&m_mutex);
if (m_currentWallet) {
qDebug() << "Closing open m_currentWallet" << m_currentWallet;
delete m_currentWallet;
}
2016-12-13 18:51:07 +00:00
Monero::Wallet * w = m_pimpl->createWallet(path.toStdString(), password.toStdString(),
2018-03-05 16:19:45 +00:00
language.toStdString(), static_cast<Monero::NetworkType>(nettype));
m_currentWallet = new Wallet(w);
return m_currentWallet;
2016-02-29 14:39:39 +00:00
}
2018-03-05 16:19:45 +00:00
Wallet *WalletManager::openWallet(const QString &path, const QString &password, NetworkType::Type nettype)
2016-02-29 14:39:39 +00:00
{
QMutexLocker locker(&m_mutex);
if (m_currentWallet) {
qDebug() << "Closing open m_currentWallet" << m_currentWallet;
delete m_currentWallet;
}
2018-03-05 16:19:45 +00:00
qDebug("%s: opening wallet at %s, nettype = %d ",
__PRETTY_FUNCTION__, qPrintable(path), nettype);
2016-02-29 14:39:39 +00:00
2018-03-05 16:19:45 +00:00
Monero::Wallet * w = m_pimpl->openWallet(path.toStdString(), password.toStdString(), static_cast<Monero::NetworkType>(nettype));
qDebug("%s: opened wallet: %s, status: %d", __PRETTY_FUNCTION__, w->address(0, 0).c_str(), w->status());
m_currentWallet = new Wallet(w);
// move wallet to the GUI thread. Otherwise it wont be emitting signals
if (m_currentWallet->thread() != qApp->thread()) {
m_currentWallet->moveToThread(qApp->thread());
}
return m_currentWallet;
}
2018-03-05 16:19:45 +00:00
void WalletManager::openWalletAsync(const QString &path, const QString &password, NetworkType::Type nettype)
{
QFuture<Wallet*> future = QtConcurrent::run(this, &WalletManager::openWallet,
2018-03-05 16:19:45 +00:00
path, password, nettype);
QFutureWatcher<Wallet*> * watcher = new QFutureWatcher<Wallet*>();
2017-01-30 17:22:59 +00:00
connect(watcher, &QFutureWatcher<Wallet*>::finished,
this, [this, watcher]() {
QFuture<Wallet*> future = watcher->future();
watcher->deleteLater();
emit walletOpened(future.result());
});
2017-01-30 17:22:59 +00:00
watcher->setFuture(future);
}
2018-03-05 16:19:45 +00:00
Wallet *WalletManager::recoveryWallet(const QString &path, const QString &memo, NetworkType::Type nettype, quint64 restoreHeight)
{
QMutexLocker locker(&m_mutex);
if (m_currentWallet) {
qDebug() << "Closing open m_currentWallet" << m_currentWallet;
delete m_currentWallet;
}
2018-03-05 16:19:45 +00:00
Monero::Wallet * w = m_pimpl->recoveryWallet(path.toStdString(), memo.toStdString(), static_cast<Monero::NetworkType>(nettype), restoreHeight);
m_currentWallet = new Wallet(w);
return m_currentWallet;
2016-02-29 14:39:39 +00:00
}
2018-03-05 16:19:45 +00:00
Wallet *WalletManager::createWalletFromKeys(const QString &path, const QString &language, NetworkType::Type nettype,
2017-01-26 20:49:32 +00:00
const QString &address, const QString &viewkey, const QString &spendkey,
quint64 restoreHeight)
{
QMutexLocker locker(&m_mutex);
if (m_currentWallet) {
qDebug() << "Closing open m_currentWallet" << m_currentWallet;
delete m_currentWallet;
m_currentWallet = NULL;
}
2018-03-05 16:19:45 +00:00
Monero::Wallet * w = m_pimpl->createWalletFromKeys(path.toStdString(), language.toStdString(), static_cast<Monero::NetworkType>(nettype), restoreHeight,
2017-01-26 20:49:32 +00:00
address.toStdString(), viewkey.toStdString(), spendkey.toStdString());
m_currentWallet = new Wallet(w);
return m_currentWallet;
}
Wallet *WalletManager::createWalletFromDevice(const QString &path, const QString &password, NetworkType::Type nettype,
const QString &deviceName, quint64 restoreHeight, const QString &subaddressLookahead)
{
QMutexLocker locker(&m_mutex);
if (m_currentWallet) {
qDebug() << "Closing open m_currentWallet" << m_currentWallet;
delete m_currentWallet;
m_currentWallet = NULL;
}
Monero::Wallet * w = m_pimpl->createWalletFromDevice(path.toStdString(), password.toStdString(), static_cast<Monero::NetworkType>(nettype),
deviceName.toStdString(), restoreHeight, subaddressLookahead.toStdString());
m_currentWallet = new Wallet(w);
return m_currentWallet;
}
QString WalletManager::closeWallet()
2016-02-29 14:39:39 +00:00
{
QMutexLocker locker(&m_mutex);
QString result;
if (m_currentWallet) {
result = m_currentWallet->address(0, 0);
delete m_currentWallet;
} else {
qCritical() << "Trying to close non existing wallet " << m_currentWallet;
result = "0";
}
return result;
}
void WalletManager::closeWalletAsync()
{
QFuture<QString> future = QtConcurrent::run(this, &WalletManager::closeWallet);
QFutureWatcher<QString> * watcher = new QFutureWatcher<QString>();
connect(watcher, &QFutureWatcher<QString>::finished,
this, [this, watcher]() {
QFuture<QString> future = watcher->future();
watcher->deleteLater();
emit walletClosed(future.result());
});
2017-01-30 17:22:59 +00:00
watcher->setFuture(future);
2016-02-29 14:39:39 +00:00
}
bool WalletManager::walletExists(const QString &path) const
2016-02-29 14:39:39 +00:00
{
return m_pimpl->walletExists(path.toStdString());
2016-02-29 14:39:39 +00:00
}
QStringList WalletManager::findWallets(const QString &path)
2016-02-29 14:39:39 +00:00
{
std::vector<std::string> found_wallets = m_pimpl->findWallets(path.toStdString());
QStringList result;
for (const auto &w : found_wallets) {
result.append(QString::fromStdString(w));
}
return result;
2016-02-29 14:39:39 +00:00
}
QString WalletManager::errorString() const
{
return tr("Unknown error");
}
2016-02-23 15:59:26 +00:00
bool WalletManager::moveWallet(const QString &src, const QString &dst)
2016-02-23 15:59:26 +00:00
{
return true;
}
2016-02-23 15:59:26 +00:00
QString WalletManager::walletLanguage(const QString &locale)
{
return "English";
}
2016-08-23 13:07:52 +00:00
quint64 WalletManager::maximumAllowedAmount() const
{
2016-12-13 18:51:07 +00:00
return Monero::Wallet::maximumAllowedAmount();
2016-08-23 13:07:52 +00:00
}
QString WalletManager::maximumAllowedAmountAsSting() const
{
return WalletManager::displayAmount(WalletManager::maximumAllowedAmount());
}
2016-08-23 13:07:52 +00:00
QString WalletManager::displayAmount(quint64 amount) const
2016-06-08 10:53:24 +00:00
{
2016-12-13 18:51:07 +00:00
return QString::fromStdString(Monero::Wallet::displayAmount(amount));
2016-06-08 10:53:24 +00:00
}
2016-08-23 13:07:52 +00:00
quint64 WalletManager::amountFromString(const QString &amount) const
{
2016-12-13 18:51:07 +00:00
return Monero::Wallet::amountFromString(amount.toStdString());
}
2016-08-23 13:07:52 +00:00
quint64 WalletManager::amountFromDouble(double amount) const
{
2016-12-13 18:51:07 +00:00
return Monero::Wallet::amountFromDouble(amount);
}
bool WalletManager::paymentIdValid(const QString &payment_id) const
{
2016-12-13 18:51:07 +00:00
return Monero::Wallet::paymentIdValid(payment_id.toStdString());
}
2018-03-05 16:19:45 +00:00
bool WalletManager::addressValid(const QString &address, NetworkType::Type nettype) const
{
2018-03-05 16:19:45 +00:00
return Monero::Wallet::addressValid(address.toStdString(), static_cast<Monero::NetworkType>(nettype));
}
2018-03-05 16:19:45 +00:00
bool WalletManager::keyValid(const QString &key, const QString &address, bool isViewKey, NetworkType::Type nettype) const
2017-01-26 20:49:32 +00:00
{
std::string error;
2018-03-05 16:19:45 +00:00
if(!Monero::Wallet::keyValid(key.toStdString(), address.toStdString(), isViewKey, static_cast<Monero::NetworkType>(nettype), error)){
2017-01-26 20:49:32 +00:00
qDebug() << QString::fromStdString(error);
return false;
}
return true;
}
2018-03-05 16:19:45 +00:00
QString WalletManager::paymentIdFromAddress(const QString &address, NetworkType::Type nettype) const
{
2018-03-05 16:19:45 +00:00
return QString::fromStdString(Monero::Wallet::paymentIdFromAddress(address.toStdString(), static_cast<Monero::NetworkType>(nettype)));
}
void WalletManager::setDaemonAddress(const QString &address)
{
m_pimpl->setDaemonAddress(address.toStdString());
}
bool WalletManager::connected() const
{
return m_pimpl->connected();
}
quint64 WalletManager::networkDifficulty() const
{
return m_pimpl->networkDifficulty();
}
quint64 WalletManager::blockchainHeight() const
{
return m_pimpl->blockchainHeight();
}
quint64 WalletManager::blockchainTargetHeight() const
{
return m_pimpl->blockchainTargetHeight();
}
double WalletManager::miningHashRate() const
{
return m_pimpl->miningHashRate();
}
bool WalletManager::isMining() const
{
if(!m_currentWallet->connected())
return false;
return m_pimpl->isMining();
}
bool WalletManager::startMining(const QString &address, quint32 threads, bool backgroundMining, bool ignoreBattery)
{
if(threads == 0)
threads = 1;
return m_pimpl->startMining(address.toStdString(), threads, backgroundMining, ignoreBattery);
}
bool WalletManager::stopMining()
{
return m_pimpl->stopMining();
}
2017-08-06 14:47:11 +00:00
bool WalletManager::localDaemonSynced() const
{
return blockchainHeight() > 1 && blockchainHeight() >= blockchainTargetHeight();
}
bool WalletManager::isDaemonLocal(const QString &daemon_address) const
{
return Monero::Utils::isAddressLocal(daemon_address.toStdString());
}
2016-12-17 15:57:42 +00:00
QString WalletManager::resolveOpenAlias(const QString &address) const
{
bool dnssec_valid = false;
std::string res = m_pimpl->resolveOpenAlias(address.toStdString(), dnssec_valid);
res = std::string(dnssec_valid ? "true" : "false") + "|" + res;
return QString::fromStdString(res);
}
2017-01-11 03:16:21 +00:00
bool WalletManager::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)
{
if (m_currentWallet)
return m_currentWallet->parse_uri(uri, address, payment_id, amount, tx_description, recipient_name, unknown_parameters, error);
return false;
}
2016-12-17 15:57:42 +00:00
2016-08-23 07:32:01 +00:00
void WalletManager::setLogLevel(int logLevel)
{
2016-12-13 18:51:07 +00:00
Monero::WalletManagerFactory::setLogLevel(logLevel);
2016-08-23 07:32:01 +00:00
}
void WalletManager::setLogCategories(const QString &categories)
{
Monero::WalletManagerFactory::setLogCategories(categories.toStdString());
}
QString WalletManager::urlToLocalPath(const QUrl &url) const
{
return QDir::toNativeSeparators(url.toLocalFile());
}
QUrl WalletManager::localPathToUrl(const QString &path) const
{
return QUrl::fromLocalFile(path);
}
#ifndef DISABLE_PASS_STRENGTH_METER
double WalletManager::getPasswordStrength(const QString &password) const
{
static const char *local_dict[] = {
"monero", "fluffypony", NULL
};
if (!ZxcvbnInit("zxcvbn.dict")) {
fprintf(stderr, "Failed to open zxcvbn.dict\n");
return 0.0;
}
double e = ZxcvbnMatch(password.toStdString().c_str(), local_dict, NULL);
ZxcvbnUnInit();
return e;
}
#endif
2017-01-23 04:15:49 +00:00
bool WalletManager::saveQrCode(const QString &code, const QString &path) const
{
QSize size;
// 240 <=> mainLayout.qrCodeSize (Receive.qml)
return QRCodeImageProvider::genQrImage(code, &size).scaled(size.expandedTo(QSize(240, 240)), Qt::KeepAspectRatio).save(path, "PNG", 100);
}
2017-08-05 22:10:59 +00:00
void WalletManager::checkUpdatesAsync(const QString &software, const QString &subdir) const
{
QFuture<QString> future = QtConcurrent::run(this, &WalletManager::checkUpdates,
software, subdir);
QFutureWatcher<QString> * watcher = new QFutureWatcher<QString>();
connect(watcher, &QFutureWatcher<Wallet*>::finished,
this, [this, watcher]() {
QFuture<QString> future = watcher->future();
watcher->deleteLater();
qDebug() << "Checking for updates - done";
emit checkUpdatesComplete(future.result());
});
watcher->setFuture(future);
}
2017-02-19 10:38:03 +00:00
QString WalletManager::checkUpdates(const QString &software, const QString &subdir) const
{
2017-08-05 22:10:59 +00:00
qDebug() << "Checking for updates";
2017-02-19 10:38:03 +00:00
const std::tuple<bool, std::string, std::string, std::string, std::string> result = Monero::WalletManager::checkUpdates(software.toStdString(), subdir.toStdString());
if (!std::get<0>(result))
return QString("");
return QString::fromStdString(std::get<1>(result) + "|" + std::get<2>(result) + "|" + std::get<3>(result) + "|" + std::get<4>(result));
}
2017-05-03 12:37:33 +00:00
bool WalletManager::clearWalletCache(const QString &wallet_path) const
{
QString fileName = wallet_path;
// Make sure wallet file is not .keys
fileName.replace(".keys","");
QFile walletCache(fileName);
QString suffix = ".old_cache";
QString newFileName = fileName + suffix;
// create unique file name
for (int i = 1; QFile::exists(newFileName); i++) {
newFileName = QString("%1%2.%3").arg(fileName).arg(suffix).arg(i);
}
return walletCache.rename(newFileName);
}
WalletManager::WalletManager(QObject *parent) : QObject(parent)
{
2016-12-13 18:51:07 +00:00
m_pimpl = Monero::WalletManagerFactory::getWalletManager();
2016-02-23 15:59:26 +00:00
}