@@ -1,4 +1,4 @@
|
||||
// Copyright 2019-2023 The Hush developers
|
||||
// Copyright 2019-2024 The Hush developers
|
||||
// Released under the GPLv3
|
||||
#include "connection.h"
|
||||
#include "mainwindow.h"
|
||||
@@ -9,6 +9,8 @@
|
||||
#include "controller.h"
|
||||
#include "../lib/silentdragonlitelib.h"
|
||||
#include "precompiled.h"
|
||||
#include <QThreadPool>
|
||||
#include "sdl.h"
|
||||
|
||||
using json = nlohmann::json;
|
||||
|
||||
@@ -32,7 +34,7 @@ ConnectionLoader::ConnectionLoader(MainWindow* main, Controller* rpc)
|
||||
connD->setupUi(d);
|
||||
|
||||
auto theme = Settings::getInstance()->get_theme_name();
|
||||
qDebug() << theme << "theme " << theme << " has loaded";
|
||||
DEBUG("theme " << theme << " has loaded");
|
||||
auto size = QSize(512,512);
|
||||
|
||||
if (theme == "Dark" || theme == "Midnight") {
|
||||
@@ -55,6 +57,7 @@ ConnectionLoader::ConnectionLoader(MainWindow* main, Controller* rpc)
|
||||
|
||||
ConnectionLoader::~ConnectionLoader()
|
||||
{
|
||||
DEBUG("destroying ConnectionLoader");
|
||||
delete isSyncing;
|
||||
delete connD;
|
||||
delete d;
|
||||
@@ -62,6 +65,7 @@ ConnectionLoader::~ConnectionLoader()
|
||||
|
||||
void ConnectionLoader::loadConnection()
|
||||
{
|
||||
DEBUG("calling doAutoConnect");
|
||||
QTimer::singleShot(1, [=]() { this->doAutoConnect(); });
|
||||
if (!Settings::getInstance()->isHeadless())
|
||||
d->exec();
|
||||
@@ -69,7 +73,22 @@ void ConnectionLoader::loadConnection()
|
||||
|
||||
void ConnectionLoader::loadProgress()
|
||||
{
|
||||
QTimer::singleShot(1, [=]() { this->ShowProgress(); });
|
||||
bool failed = false;
|
||||
QTimer::singleShot(1, [=]() mutable {
|
||||
DEBUG("failed=" << failed);
|
||||
// continually retry ShowProgress() until it succeeds
|
||||
// by running without an exception
|
||||
do {
|
||||
try {
|
||||
this->ShowProgress();
|
||||
failed = false;
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("caught exception " << e.what() );
|
||||
failed = true;
|
||||
}
|
||||
} while (failed);
|
||||
|
||||
});
|
||||
if (!Settings::getInstance()->isHeadless())
|
||||
d->exec();
|
||||
}
|
||||
@@ -83,56 +102,62 @@ void ConnectionLoader::ShowProgress()
|
||||
|
||||
auto connection = makeConnection(config);
|
||||
auto me = this;
|
||||
qDebug() << __func__ << ": server=" << config->server << " connection=" << connection << " me=" << me;
|
||||
DEBUG("server=" << config->server << " connection=" << connection << " me=" << me);
|
||||
|
||||
isSyncing = new QAtomicInteger<bool>();
|
||||
isSyncing->store(true);
|
||||
main->logger->write("isSyncing");
|
||||
|
||||
DEBUG("isSyncing");
|
||||
|
||||
// Do a sync after import
|
||||
syncTimer = new QTimer(main);
|
||||
main->logger->write("Beginning sync after import wif");
|
||||
connection->doRPCWithDefaultErrorHandling("sync", "", [=](auto) {
|
||||
DEBUG("Beginning sync after import wif");
|
||||
connection->doRPC("sync", "", [=](auto) {
|
||||
DEBUG("finished syncing");
|
||||
isSyncing->store(false);
|
||||
// Cancel the timer
|
||||
syncTimer->deleteLater();
|
||||
// When sync is done, set the connection
|
||||
this->doRPCSetConnectionShield(connection);
|
||||
}, [=](auto) {
|
||||
DEBUG("sync rpc error! server=" << config->server);
|
||||
});
|
||||
|
||||
|
||||
// While it is syncing, we'll show the status updates while it is alive.
|
||||
QObject::connect(syncTimer, &QTimer::timeout, [=]() {
|
||||
// Check the sync status
|
||||
DEBUG("Check the sync status");
|
||||
if (isSyncing != nullptr && isSyncing->load()) {
|
||||
// Get the sync status
|
||||
|
||||
DEBUG("Get the sync status");
|
||||
try {
|
||||
connection->doRPC("syncstatus", "", [=](json reply) {
|
||||
if (isSyncing != nullptr && reply.find("synced_blocks") != reply.end())
|
||||
|
||||
{
|
||||
qint64 synced = reply["synced_blocks"].get<json::number_unsigned_t>();
|
||||
qint64 total = reply["total_blocks"].get<json::number_unsigned_t>();
|
||||
me->showInformation(
|
||||
"Syncing... " + QString::number(synced) + " / " + QString::number(total)
|
||||
);
|
||||
}
|
||||
},
|
||||
[=](QString err) {
|
||||
qDebug() << "Sync error" << err;
|
||||
});
|
||||
}catch (...)
|
||||
{
|
||||
main->logger->write("catch sync progress reply");
|
||||
}
|
||||
|
||||
}
|
||||
});
|
||||
connection->doRPC("syncstatus", "", [=](json reply) {
|
||||
if (isSyncing != nullptr && reply.find("synced_blocks") != reply.end()) {
|
||||
qint64 synced = reply["synced_blocks"].get<json::number_unsigned_t>();
|
||||
qint64 total = reply["total_blocks"].get<json::number_unsigned_t>();
|
||||
me->showInformation(
|
||||
"Syncing... " + QString::number(synced) + " / " + QString::number(total)
|
||||
);
|
||||
}
|
||||
}, [=](QString err) {
|
||||
DEBUG("Sync error " << err);
|
||||
// We may have gotten "Unexpected compression flag: 60"
|
||||
// or some other error, so let's try another server
|
||||
config->server = Settings::getRandomServer();
|
||||
DEBUG("Changed server to " << config->server );
|
||||
});
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("syncstatus exception: " << e.what() );
|
||||
main->logger->write("catch sync progress reply");
|
||||
|
||||
syncTimer->setInterval(1* 1000);
|
||||
syncTimer->start();
|
||||
main->logger->write("Start sync timer");
|
||||
// rethrow exception so loadProgress can catch
|
||||
// it and retry the entire ShowProgress() function again
|
||||
throw new std::runtime_error(std::string("syncstatus failed"));
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
int interval = 1*1000;
|
||||
syncTimer->setInterval(interval);
|
||||
syncTimer->start();
|
||||
DEBUG("Start sync timer with interval=" << interval);
|
||||
}
|
||||
|
||||
void ConnectionLoader::doAutoConnect()
|
||||
@@ -140,43 +165,53 @@ void ConnectionLoader::doAutoConnect()
|
||||
auto config = std::shared_ptr<ConnectionConfig>(new ConnectionConfig());
|
||||
config->dangerous = false;
|
||||
config->server = Settings::getInstance()->getSettings().server;
|
||||
qDebug() << __func__ << " server=" << config->server;
|
||||
DEBUG(" server=" << config->server);
|
||||
|
||||
// Initialize the library
|
||||
main->logger->write(QObject::tr("Attempting to initialize library with ") + config->server);
|
||||
DEBUG("Attempting to initialize library with "<< config->server);
|
||||
|
||||
// Check to see if there's an existing wallet
|
||||
if (litelib_wallet_exists(Settings::getDefaultChainName().toStdString().c_str())) {
|
||||
qDebug() << __func__ << ": using existing wallet";
|
||||
DEBUG("using existing wallet");
|
||||
main->logger->write(QObject::tr("Using existing wallet."));
|
||||
char* resp = litelib_initialize_existing(
|
||||
config->dangerous,
|
||||
config->server.toStdString().c_str()
|
||||
);
|
||||
QString response = litelib_process_response(resp);
|
||||
|
||||
if (response.toUpper().trimmed() != "OK") {
|
||||
config->server = Settings::getRandomServer();
|
||||
|
||||
resp = litelib_initialize_existing(
|
||||
QString response = "";
|
||||
try {
|
||||
char* resp = litelib_initialize_existing(
|
||||
config->dangerous,
|
||||
config->server.toStdString().c_str()
|
||||
);
|
||||
response = litelib_process_response(resp);
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("caught an exception, ignoring: " << e.what());
|
||||
}
|
||||
|
||||
if (response.toUpper().trimmed() != "OK") {
|
||||
config->server = Settings::getRandomServer();
|
||||
|
||||
try {
|
||||
char* resp = litelib_initialize_existing(
|
||||
config->dangerous,
|
||||
config->server.toStdString().c_str()
|
||||
);
|
||||
response = litelib_process_response(resp);
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("caught an exception, ignoring: " << e.what());
|
||||
}
|
||||
|
||||
if (response.toUpper().trimmed() != "OK") {
|
||||
QString resp = "Error when connecting to " + config->server + ": " + response;
|
||||
showError(resp);
|
||||
return;
|
||||
} else {
|
||||
qDebug() << __func__ << ": Successfully connected to random server: " << config->server << " !!!";
|
||||
DEBUG("Successfully connected to random server: " << config->server << " !!!");
|
||||
}
|
||||
} else {
|
||||
qDebug() << __func__ << ": Successfully connected to " << config->server << " !!!";
|
||||
DEBUG("Successfully connected to " << config->server << " !!!");
|
||||
}
|
||||
|
||||
} else {
|
||||
qDebug() << __func__ << ": no existing wallet";
|
||||
DEBUG("no existing wallet");
|
||||
main->logger->write(QObject::tr("Create/restore wallet."));
|
||||
createOrRestore(config->dangerous, config->server);
|
||||
d->show();
|
||||
@@ -187,61 +222,77 @@ void ConnectionLoader::doAutoConnect()
|
||||
qDebug() << __func__ << ": server=" << config->server
|
||||
<< " connection=" << connection << " me=" << me << endl;
|
||||
|
||||
|
||||
// After the lib is initialized, try to do get info
|
||||
connection->doRPC("info", "", [=](auto reply) {
|
||||
// If success, set the connection
|
||||
main->logger->write("Connection is online.");
|
||||
DEBUG("Connection is online.");
|
||||
connection->setInfo(reply);
|
||||
main->logger->write("getting Connection reply");
|
||||
DEBUG("getting Connection reply");
|
||||
isSyncing = new QAtomicInteger<bool>();
|
||||
isSyncing->store(true);
|
||||
main->logger->write("isSyncing");
|
||||
DEBUG("isSyncing");
|
||||
|
||||
// Do a sync at startup
|
||||
syncTimer = new QTimer(main);
|
||||
main->logger->write("Beginning sync");
|
||||
connection->doRPCWithDefaultErrorHandling("sync", "", [=](auto) {
|
||||
DEBUG("Beginning sync");
|
||||
connection->doRPC("sync", "", [=](auto) {
|
||||
DEBUG("finished syncing");
|
||||
isSyncing->store(false);
|
||||
// Cancel the timer
|
||||
syncTimer->deleteLater();
|
||||
// When sync is done, set the connection
|
||||
this->doRPCSetConnection(connection);
|
||||
}, [=](auto) mutable {
|
||||
DEBUG("sync rpc error! server=" << config->server);
|
||||
// continually retry sync RPC until it succeeds
|
||||
// don't change server each time it fails
|
||||
bool failed = true;
|
||||
do {
|
||||
// config->server = Settings::getRandomServer();
|
||||
// auto connection = makeConnection(config);
|
||||
// DEBUG("changed server to " << config->server);
|
||||
connection->doRPC("sync", "", [=](auto) mutable {
|
||||
DEBUG("sync success with server=" << config->server);
|
||||
failed = false;
|
||||
isSyncing->store(false);
|
||||
// Cancel the timer
|
||||
syncTimer->deleteLater();
|
||||
// When sync is done, set the connection
|
||||
this->doRPCSetConnection(connection);
|
||||
}, [=](auto) {
|
||||
DEBUG("sync failed with server=" << config->server << " . continuing sync loop" );
|
||||
});
|
||||
} while (failed);
|
||||
});
|
||||
|
||||
// While it is syncing, we'll show the status updates while it is alive.
|
||||
QObject::connect(syncTimer, &QTimer::timeout, [=]() {
|
||||
// Check the sync status
|
||||
DEBUG("Check the sync status");
|
||||
if (isSyncing != nullptr && isSyncing->load()) {
|
||||
// Get the sync status
|
||||
|
||||
DEBUG("Getting the sync status");
|
||||
try {
|
||||
connection->doRPC("syncstatus", "", [=](json reply) {
|
||||
if (isSyncing != nullptr && reply.find("synced_blocks") != reply.end())
|
||||
|
||||
{
|
||||
if (isSyncing != nullptr && reply.find("synced_blocks") != reply.end()) {
|
||||
qint64 synced = reply["synced_blocks"].get<json::number_unsigned_t>();
|
||||
qint64 total = reply["total_blocks"].get<json::number_unsigned_t>();
|
||||
me->showInformation(
|
||||
"Syncing... " + QString::number(synced) + " / " + QString::number(total)
|
||||
);
|
||||
}
|
||||
},
|
||||
[=](QString err) {
|
||||
qDebug() << "Sync error" << err;
|
||||
});
|
||||
}catch (...)
|
||||
{
|
||||
main->logger->write("catch sync progress reply");
|
||||
|
||||
}
|
||||
|
||||
},
|
||||
[=](QString err) {
|
||||
DEBUG("syncstatus error" << err);
|
||||
});
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("caught exception from syncstatus: " << e.what());
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
syncTimer->setInterval(1* 1000);
|
||||
int interval = 1*1000;
|
||||
syncTimer->setInterval(interval);
|
||||
syncTimer->start();
|
||||
main->logger->write("Start sync timer");
|
||||
DEBUG("Start sync timer with interval=" << interval);
|
||||
|
||||
}, [=](QString err) {
|
||||
showError(err);
|
||||
@@ -255,13 +306,13 @@ void ConnectionLoader::createOrRestore(bool dangerous, QString server)
|
||||
d->hide();
|
||||
// Create a wizard
|
||||
FirstTimeWizard wizard(dangerous,server);
|
||||
main->logger->write("Start new Wallet with FirstimeWizard");
|
||||
DEBUG("Start new Wallet with FirstimeWizard");
|
||||
wizard.exec();
|
||||
}
|
||||
|
||||
void ConnectionLoader::doRPCSetConnection(Connection* conn)
|
||||
{
|
||||
qDebug() << "Connectionloader finished, setting connection";
|
||||
DEBUG("Connectionloader finished, setting connection");
|
||||
main->logger->write("Connectionloader finished, setting connection");
|
||||
rpc->setConnection(conn);
|
||||
d->accept();
|
||||
@@ -272,17 +323,16 @@ void ConnectionLoader::doRPCSetConnection(Connection* conn)
|
||||
main->logger->write("Path to Wallet.dat : " );
|
||||
qDebug() << __func__ << ": wallet path =" << plaintextWallet;
|
||||
plaintextWallet.remove();
|
||||
|
||||
} catch (...) {
|
||||
qDebug() << "No plaintext wallet found! file=" << plaintextWallet;
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("Caught exception" << e.what() );
|
||||
DEBUG("No plaintext wallet found! file=" << plaintextWallet);
|
||||
main->logger->write("no Plaintext wallet.dat");
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void ConnectionLoader::doRPCSetConnectionShield(Connection* conn)
|
||||
{
|
||||
qDebug() << "Importing finished, setting connection";
|
||||
DEBUG("Importing finished, setting connection");
|
||||
rpc->setConnection(conn);
|
||||
d->accept();
|
||||
main->getRPC()->shield([=] (auto) {});
|
||||
@@ -293,9 +343,10 @@ void ConnectionLoader::doRPCSetConnectionShield(Connection* conn)
|
||||
main->logger->write("Path to Wallet.dat : " );
|
||||
qDebug() << __func__ << ": wallet path =" << plaintextWallet;
|
||||
plaintextWallet.remove();
|
||||
} catch (...) {
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("Caught exception" << e.what() );
|
||||
main->logger->write("no Plaintext wallet.dat");
|
||||
qDebug() << "No plaintext wallet found! file=" << plaintextWallet;
|
||||
DEBUG("No plaintext wallet found! file=" << plaintextWallet);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -346,19 +397,50 @@ QString litelib_process_response(char* resp)
|
||||
************************************************************************************/
|
||||
void Executor::run()
|
||||
{
|
||||
char* resp = litelib_execute(this->cmd.toStdString().c_str(), this->args.toStdString().c_str());
|
||||
QString reply = litelib_process_response(resp);
|
||||
auto parsed = json::parse(
|
||||
reply.toStdString().c_str(),
|
||||
nullptr,
|
||||
false
|
||||
);
|
||||
if (parsed.is_discarded() || parsed.is_null())
|
||||
emit handleError(reply);
|
||||
auto config = std::shared_ptr<ConnectionConfig>(new ConnectionConfig());
|
||||
DEBUG("cmd=" << cmd << " args=" << args << " server=" << config->server);
|
||||
QString response = "";
|
||||
try {
|
||||
char* resp = litelib_execute(this->cmd.toStdString().c_str(), this->args.toStdString().c_str());
|
||||
response = litelib_process_response(resp);
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("ignoring exception: " << e.what() );
|
||||
}
|
||||
|
||||
else
|
||||
//TODO: we can do stricter error checking
|
||||
if (response.isEmpty()) {
|
||||
config->server = Settings::getRandomServer();
|
||||
|
||||
try {
|
||||
char* resp = litelib_initialize_existing(
|
||||
config->dangerous,
|
||||
config->server.toStdString().c_str()
|
||||
);
|
||||
response = litelib_process_response(resp);
|
||||
resp = litelib_execute(this->cmd.toStdString().c_str(), this->args.toStdString().c_str());
|
||||
response = litelib_process_response(resp);
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("server= " << config->server << " gave exception: " << e.what() );
|
||||
emit handleError(response);
|
||||
}
|
||||
}
|
||||
try {
|
||||
auto parsed = json::parse(
|
||||
response.toStdString().c_str(),
|
||||
nullptr,
|
||||
false
|
||||
);
|
||||
|
||||
if (parsed.is_discarded() || parsed.is_null()) {
|
||||
emit handleError(response);
|
||||
} else {
|
||||
emit responseReady(parsed);
|
||||
}
|
||||
} catch (const std::exception& e) {
|
||||
DEBUG("exception when parsing json: " << e.what() );
|
||||
emit handleError(response);
|
||||
}
|
||||
|
||||
emit responseReady(parsed);
|
||||
}
|
||||
|
||||
void Callback::processRPCCallback(json resp)
|
||||
@@ -386,11 +468,12 @@ Connection::Connection(MainWindow* m, std::shared_ptr<ConnectionConfig> conf)
|
||||
|
||||
void Connection::doRPC(const QString cmd, const QString args, const std::function<void(json)>& cb, const std::function<void(QString)>& errCb)
|
||||
{
|
||||
if (shutdownInProgress)
|
||||
// Ignoring RPC because shutdown in progress
|
||||
if (shutdownInProgress) {
|
||||
DEBUG("Ignoring RPC because shutdown in progress");
|
||||
return;
|
||||
}
|
||||
|
||||
qDebug() << __func__ << ": " << cmd;
|
||||
DEBUG("cmd=" << cmd << " args=" << args);
|
||||
|
||||
// Create a runner.
|
||||
auto runner = new Executor(cmd, args);
|
||||
@@ -405,7 +488,7 @@ void Connection::doRPC(const QString cmd, const QString args, const std::functio
|
||||
|
||||
void Connection::doRPCWithDefaultErrorHandling(const QString cmd, const QString args, const std::function<void(json)>& cb)
|
||||
{
|
||||
qDebug() << __func__ << ": " << cmd;
|
||||
DEBUG("cmd=" << cmd << " args=" << args);
|
||||
doRPC(cmd, args, cb, [=] (QString err) {
|
||||
this->showTxError(err);
|
||||
});
|
||||
@@ -413,7 +496,7 @@ void Connection::doRPCWithDefaultErrorHandling(const QString cmd, const QString
|
||||
|
||||
void Connection::doRPCIgnoreError(const QString cmd, const QString args, const std::function<void(json)>& cb)
|
||||
{
|
||||
qDebug() << __func__ << ": " << cmd;
|
||||
DEBUG("cmd=" << cmd << " args=" << args);
|
||||
doRPC(cmd, args, cb, [=] (auto) {
|
||||
// Ignored error handling
|
||||
});
|
||||
@@ -445,5 +528,6 @@ void Connection::showTxError(const QString& error)
|
||||
*/
|
||||
void Connection::shutdown()
|
||||
{
|
||||
DEBUG("shutting down");
|
||||
shutdownInProgress = true;
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user