mirror of
https://github.com/feather-wallet/feather.git
synced 2025-01-05 10:29:27 +00:00
250 lines
7.9 KiB
C++
250 lines
7.9 KiB
C++
// SPDX-License-Identifier: BSD-3-Clause
|
|
// SPDX-FileCopyrightText: 2020-2023 The Monero Project
|
|
|
|
#include <QResource>
|
|
#include <QApplication>
|
|
#include <QtCore>
|
|
#include <QtGui>
|
|
#include <singleapplication.h>
|
|
|
|
#include "config-feather.h"
|
|
#include "constants.h"
|
|
#include "MainWindow.h"
|
|
#include "utils/EventFilter.h"
|
|
#include "utils/os/Prestium.h"
|
|
#include "WindowManager.h"
|
|
#include "config.h"
|
|
|
|
#if defined(Q_OS_LINUX) && defined(STACK_TRACE)
|
|
#define BOOST_STACKTRACE_LINK
|
|
#include <signal.h>
|
|
#include <boost/stacktrace.hpp>
|
|
#include <fstream>
|
|
#endif
|
|
|
|
#include <QObject>
|
|
|
|
#if defined(Q_OS_WIN)
|
|
#include <windows.h>
|
|
#include <vfw.h>
|
|
#endif
|
|
|
|
#if defined(Q_OS_LINUX) && defined(STATIC)
|
|
Q_IMPORT_PLUGIN(QXcbIntegrationPlugin)
|
|
Q_IMPORT_PLUGIN(QComposePlatformInputContextPlugin) // Needed for dead keys on Linux
|
|
#endif
|
|
|
|
#if defined(Q_OS_LINUX) && defined(STACK_TRACE)
|
|
void signal_handler(int signum) {
|
|
::signal(signum, SIG_DFL);
|
|
std::stringstream keyStream;
|
|
keyStream << boost::stacktrace::stacktrace();
|
|
std::cout << keyStream.str();
|
|
|
|
// Write stack trace to disk
|
|
if (conf()->get(Config::writeStackTraceToDisk).toBool()) {
|
|
QString crashLogPath{Config::defaultConfigDir().path() + "/crash_report.txt"};
|
|
std::ofstream out(crashLogPath.toStdString());
|
|
out << QString("Version: %1-%2\n").arg(FEATHER_VERSION, FEATHER_COMMIT).toStdString();
|
|
out << QString("OS: %1\n").arg(QSysInfo::prettyProductName()).toStdString();
|
|
out << keyStream.str();
|
|
out.close();
|
|
}
|
|
|
|
// Make a last ditch attempt to restart the application
|
|
QProcess::startDetached(qApp->arguments()[0], qApp->arguments());
|
|
|
|
::raise(SIGABRT);
|
|
}
|
|
#endif
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
Q_INIT_RESOURCE(assets);
|
|
|
|
#if defined(Q_OS_LINUX) && defined(STACK_TRACE)
|
|
::signal(SIGSEGV, &signal_handler);
|
|
::signal(SIGABRT, &signal_handler);
|
|
#endif
|
|
|
|
#if defined(HAS_TOR_BIN)
|
|
Q_INIT_RESOURCE(assets_tor);
|
|
#endif
|
|
|
|
#ifdef _WIN32
|
|
if (AttachConsole(ATTACH_PARENT_PROCESS)) {
|
|
freopen("CONOUT$", "w", stdout);
|
|
freopen("CONOUT$", "w", stderr);
|
|
}
|
|
#endif
|
|
|
|
// #TODO: Fix Windows dark mode handling Qt 6.5 beta1
|
|
// darkmode=2 (default) causes some text to become unreadable
|
|
#if defined(Q_OS_WIN)
|
|
qputenv("QT_QPA_PLATFORM", "windows:darkmode=1");
|
|
#endif
|
|
|
|
QStringList argv_;
|
|
for(int i = 0; i != argc; i++){
|
|
argv_ << QString::fromStdString(argv[i]);
|
|
}
|
|
|
|
QCommandLineParser parser;
|
|
parser.setApplicationDescription("feather");
|
|
parser.addHelpOption();
|
|
parser.addVersionOption();
|
|
|
|
QCommandLineOption useLocalTorOption(QStringList() << "use-local-tor", "Use system wide installed Tor instead of the bundled.");
|
|
parser.addOption(useLocalTorOption);
|
|
|
|
QCommandLineOption quietModeOption(QStringList() << "quiet", "Limit console output");
|
|
parser.addOption(quietModeOption);
|
|
|
|
QCommandLineOption stagenetOption(QStringList() << "stagenet", "Stagenet is for development purposes only.");
|
|
parser.addOption(stagenetOption);
|
|
|
|
QCommandLineOption testnetOption(QStringList() << "testnet", "Testnet is for development purposes only.");
|
|
parser.addOption(testnetOption);
|
|
|
|
bool parsed = parser.parse(argv_);
|
|
if (!parsed) {
|
|
qCritical() << parser.errorText();
|
|
exit(1);
|
|
}
|
|
|
|
bool stagenet = parser.isSet(stagenetOption);
|
|
bool testnet = parser.isSet(testnetOption);
|
|
bool quiet = parser.isSet(quietModeOption);
|
|
|
|
// Setup networkType
|
|
if (stagenet)
|
|
constants::networkType = NetworkType::STAGENET;
|
|
else if (testnet)
|
|
constants::networkType = NetworkType::TESTNET;
|
|
else
|
|
constants::networkType = NetworkType::MAINNET;
|
|
|
|
// Setup QApplication
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
|
|
QApplication::setAttribute(Qt::AA_DisableWindowContextHelpButton);
|
|
#endif
|
|
QApplication::setDesktopSettingsAware(true); // use system font
|
|
QApplication::setApplicationVersion(FEATHER_VERSION);
|
|
|
|
#if defined(Q_OS_LINUX)
|
|
// PassThrough results in muddy text
|
|
QApplication::setHighDpiScaleFactorRoundingPolicy(Qt::HighDpiScaleFactorRoundingPolicy::Round);
|
|
#endif
|
|
|
|
SingleApplication app(argc, argv);
|
|
|
|
QApplication::setQuitOnLastWindowClosed(false);
|
|
QApplication::setApplicationName("FeatherWallet");
|
|
|
|
|
|
// Setup config directories
|
|
QString configDir = Config::defaultConfigDir().path();
|
|
QString config_dir_tor = QString("%1/%2").arg(configDir, "tor");
|
|
QString config_dir_tordata = QString("%1/%2").arg(configDir, "tor/data");
|
|
QStringList createDirs({configDir, config_dir_tor, config_dir_tordata});
|
|
for (const auto &d: createDirs) {
|
|
if (!Utils::dirExists(d)) {
|
|
qDebug() << QString("Creating directory: %1").arg(d);
|
|
if (!QDir().mkpath(d)) {
|
|
qCritical() << "Could not create directory " << d;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Setup logging
|
|
QString logPath = QString("%1/libwallet.log").arg(configDir);
|
|
Monero::Utils::onStartup();
|
|
Monero::Wallet::init("", "feather", logPath.toStdString(), true);
|
|
|
|
bool logLevelFromEnv;
|
|
int logLevel = qEnvironmentVariableIntValue("MONERO_LOG_LEVEL", &logLevelFromEnv);
|
|
if (logLevelFromEnv) {
|
|
conf()->set(Config::logLevel, logLevel);
|
|
} else {
|
|
logLevel = conf()->get(Config::logLevel).toInt();
|
|
}
|
|
|
|
if (parser.isSet("quiet") || conf()->get(Config::disableLogging).toBool()) {
|
|
qWarning() << "Logging is disabled";
|
|
WalletManager::instance()->setLogLevel(-1);
|
|
}
|
|
else if (logLevel >= 0 && logLevel <= Monero::WalletManagerFactory::LogLevel_Max) {
|
|
Monero::WalletManagerFactory::setLogLevel(logLevel);
|
|
}
|
|
|
|
// Setup wallet directory
|
|
QString walletDir = conf()->get(Config::walletDirectory).toString();
|
|
if (walletDir.isEmpty() || Utils::isPortableMode()) {
|
|
walletDir = Utils::defaultWalletDir();
|
|
conf()->set(Config::walletDirectory, walletDir);
|
|
}
|
|
if (!QDir().mkpath(walletDir))
|
|
qCritical() << "Unable to create dir: " << walletDir;
|
|
|
|
// Prestium initial config
|
|
if (conf()->get(Config::firstRun).toBool() && Prestium::detect()) {
|
|
conf()->set(Config::proxy, Config::Proxy::i2p);
|
|
conf()->set(Config::socks5Port, Prestium::i2pPort());
|
|
conf()->set(Config::hideUpdateNotifications, true);
|
|
}
|
|
|
|
if (parser.isSet("use-local-tor"))
|
|
conf()->set(Config::useLocalTor, true);
|
|
|
|
parser.process(app); // Parse again for --help and --version
|
|
|
|
if (!quiet) {
|
|
QMap<QString, QString> info;
|
|
info["Qt"] = QT_VERSION_STR;
|
|
info["Feather"] = FEATHER_VERSION;
|
|
if (stagenet) info["Mode"] = "Stagenet";
|
|
else if (testnet) info["Mode"] = "Testnet";
|
|
else info["Mode"] = "Mainnet";
|
|
info["SSL"] = QSslSocket::sslLibraryVersionString();
|
|
info["SSL build"] = QSslSocket::sslLibraryBuildVersionString();
|
|
#if defined(TOR_VERSION)
|
|
info["Tor version"] = TOR_VERSION;
|
|
#else
|
|
info["Tor version"] = "Not bundled";
|
|
#endif
|
|
for (const QString &k: info.keys()) {
|
|
qWarning().nospace().noquote() << QString("%1: %2").arg(k, info[k]);
|
|
}
|
|
}
|
|
|
|
#if defined(Q_OS_MAC)
|
|
// For some odd reason, if we don't do this, QPushButton's
|
|
// need to be clicked *twice* in order to fire ?!
|
|
QFont fontDef = QApplication::font();
|
|
fontDef.setPointSize(fontDef.pointSize() + 1);
|
|
QApplication::setFont(fontDef);
|
|
#endif
|
|
|
|
qInstallMessageHandler(Utils::applicationLogHandler);
|
|
qRegisterMetaType<QVector<QString>>();
|
|
qRegisterMetaType<TxProofResult>("TxProofResult");
|
|
qRegisterMetaType<QPair<bool, bool>>();
|
|
|
|
EventFilter filter;
|
|
app.installEventFilter(&filter);
|
|
|
|
auto *pool = QThreadPool::globalInstance();
|
|
if (pool->maxThreadCount() < 8) {
|
|
pool->setMaxThreadCount(8);
|
|
}
|
|
|
|
auto wm = windowManager();
|
|
wm->setEventFilter(&filter);
|
|
|
|
QObject::connect(&app, &SingleApplication::instanceStarted, [&wm]() {
|
|
wm->raise();
|
|
});
|
|
|
|
return QApplication::exec();
|
|
}
|