#include "WalletManager.h" #include "Wallet.h" #include "wallet/api/wallet2_api.h" #include "zxcvbn-c/zxcvbn.h" #include "QRCodeImageProvider.h" #include #include #include #include #include #include #include #include #include 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, const QString &language, NetworkType::Type nettype, quint64 kdfRounds) { QMutexLocker locker(&m_mutex); if (m_currentWallet) { qDebug() << "Closing open m_currentWallet" << m_currentWallet; delete m_currentWallet; } Monero::Wallet * w = m_pimpl->createWallet(path.toStdString(), password.toStdString(), language.toStdString(), static_cast(nettype), kdfRounds); m_currentWallet = new Wallet(w); return m_currentWallet; } Wallet *WalletManager::openWallet(const QString &path, const QString &password, NetworkType::Type nettype, quint64 kdfRounds) { QMutexLocker locker(&m_mutex); if (m_currentWallet) { qDebug() << "Closing open m_currentWallet" << m_currentWallet; delete m_currentWallet; } qDebug("%s: opening wallet at %s, nettype = %d ", __PRETTY_FUNCTION__, qPrintable(path), nettype); Monero::Wallet * w = m_pimpl->openWallet(path.toStdString(), password.toStdString(), static_cast(nettype), kdfRounds); 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; } void WalletManager::openWalletAsync(const QString &path, const QString &password, NetworkType::Type nettype, quint64 kdfRounds) { QFuture future = QtConcurrent::run(this, &WalletManager::openWallet, path, password, nettype, kdfRounds); QFutureWatcher * watcher = new QFutureWatcher(); connect(watcher, &QFutureWatcher::finished, this, [this, watcher]() { QFuture future = watcher->future(); watcher->deleteLater(); emit walletOpened(future.result()); }); watcher->setFuture(future); } Wallet *WalletManager::recoveryWallet(const QString &path, const QString &memo, NetworkType::Type nettype, quint64 restoreHeight, quint64 kdfRounds) { QMutexLocker locker(&m_mutex); if (m_currentWallet) { qDebug() << "Closing open m_currentWallet" << m_currentWallet; delete m_currentWallet; } Monero::Wallet * w = m_pimpl->recoveryWallet(path.toStdString(), "", memo.toStdString(), static_cast(nettype), restoreHeight, kdfRounds); m_currentWallet = new Wallet(w); return m_currentWallet; } Wallet *WalletManager::createWalletFromKeys(const QString &path, const QString &language, NetworkType::Type nettype, const QString &address, const QString &viewkey, const QString &spendkey, quint64 restoreHeight, quint64 kdfRounds) { 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->createWalletFromKeys(path.toStdString(), "", language.toStdString(), static_cast(nettype), restoreHeight, address.toStdString(), viewkey.toStdString(), spendkey.toStdString(), kdfRounds); 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(nettype), deviceName.toStdString(), restoreHeight, subaddressLookahead.toStdString()); m_currentWallet = new Wallet(w); return m_currentWallet; } QString WalletManager::closeWallet() { 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 future = QtConcurrent::run(this, &WalletManager::closeWallet); QFutureWatcher * watcher = new QFutureWatcher(); connect(watcher, &QFutureWatcher::finished, this, [this, watcher]() { QFuture future = watcher->future(); watcher->deleteLater(); emit walletClosed(future.result()); }); watcher->setFuture(future); } bool WalletManager::walletExists(const QString &path) const { return m_pimpl->walletExists(path.toStdString()); } QStringList WalletManager::findWallets(const QString &path) { std::vector found_wallets = m_pimpl->findWallets(path.toStdString()); QStringList result; for (const auto &w : found_wallets) { result.append(QString::fromStdString(w)); } return result; } QString WalletManager::errorString() const { return tr("Unknown error"); } bool WalletManager::moveWallet(const QString &src, const QString &dst) { return true; } QString WalletManager::walletLanguage(const QString &locale) { return "English"; } quint64 WalletManager::maximumAllowedAmount() const { return Monero::Wallet::maximumAllowedAmount(); } QString WalletManager::maximumAllowedAmountAsSting() const { return WalletManager::displayAmount(WalletManager::maximumAllowedAmount()); } QString WalletManager::displayAmount(quint64 amount) const { return QString::fromStdString(Monero::Wallet::displayAmount(amount)); } quint64 WalletManager::amountFromString(const QString &amount) const { return Monero::Wallet::amountFromString(amount.toStdString()); } quint64 WalletManager::amountFromDouble(double amount) const { return Monero::Wallet::amountFromDouble(amount); } bool WalletManager::paymentIdValid(const QString &payment_id) const { return Monero::Wallet::paymentIdValid(payment_id.toStdString()); } bool WalletManager::addressValid(const QString &address, NetworkType::Type nettype) const { return Monero::Wallet::addressValid(address.toStdString(), static_cast(nettype)); } bool WalletManager::keyValid(const QString &key, const QString &address, bool isViewKey, NetworkType::Type nettype) const { std::string error; if(!Monero::Wallet::keyValid(key.toStdString(), address.toStdString(), isViewKey, static_cast(nettype), error)){ qDebug() << QString::fromStdString(error); return false; } return true; } QString WalletManager::paymentIdFromAddress(const QString &address, NetworkType::Type nettype) const { return QString::fromStdString(Monero::Wallet::paymentIdFromAddress(address.toStdString(), static_cast(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(); } bool WalletManager::localDaemonSynced() const { return blockchainHeight() > 1 && blockchainHeight() >= blockchainTargetHeight(); } bool WalletManager::isDaemonLocal(const QString &daemon_address) const { return Monero::Utils::isAddressLocal(daemon_address.toStdString()); } 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); } bool WalletManager::parse_uri(const QString &uri, QString &address, QString &payment_id, uint64_t &amount, QString &tx_description, QString &recipient_name, QVector &unknown_parameters, QString &error) const { if (m_currentWallet) return m_currentWallet->parse_uri(uri, address, payment_id, amount, tx_description, recipient_name, unknown_parameters, error); return false; } QVariantMap WalletManager::parse_uri_to_object(const QString &uri) const { QString address; QString payment_id; uint64_t amount; QString tx_description; QString recipient_name; QVector unknown_parameters; QString error; QVariantMap result; if (this->parse_uri(uri, address, payment_id, amount, tx_description, recipient_name, unknown_parameters, error)) { result.insert("address", address); result.insert("payment_id", payment_id); result.insert("amount", this->displayAmount(amount)); result.insert("tx_description", tx_description); result.insert("recipient_name", recipient_name); } else { result.insert("error", error); } return result; } void WalletManager::setLogLevel(int logLevel) { Monero::WalletManagerFactory::setLogLevel(logLevel); } 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 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); } void WalletManager::checkUpdatesAsync(const QString &software, const QString &subdir) const { QFuture future = QtConcurrent::run(this, &WalletManager::checkUpdates, software, subdir); QFutureWatcher * watcher = new QFutureWatcher(); connect(watcher, &QFutureWatcher::finished, this, [this, watcher]() { QFuture future = watcher->future(); watcher->deleteLater(); qDebug() << "Checking for updates - done"; emit checkUpdatesComplete(future.result()); }); watcher->setFuture(future); } QString WalletManager::checkUpdates(const QString &software, const QString &subdir) const { qDebug() << "Checking for updates"; const std::tuple 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)); } 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) { m_pimpl = Monero::WalletManagerFactory::getWalletManager(); }