// SPDX-License-Identifier: BSD-3-Clause // SPDX-FileCopyrightText: 2020-2023 The Monero Project #include #include #include #include #include #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 #include #include #endif #include #if defined(Q_OS_WIN) #include #include #endif #if defined(Q_OS_LINUX) && defined(STATIC) Q_IMPORT_PLUGIN(QXcbIntegrationPlugin) #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 (config()->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) { config()->set(Config::logLevel, logLevel); } else { logLevel = config()->get(Config::logLevel).toInt(); } if (parser.isSet("quiet") || config()->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 = config()->get(Config::walletDirectory).toString(); if (walletDir.isEmpty() || Utils::isPortableMode()) { walletDir = Utils::defaultWalletDir(); config()->set(Config::walletDirectory, walletDir); } if (!QDir().mkpath(walletDir)) qCritical() << "Unable to create dir: " << walletDir; // Prestium initial config if (config()->get(Config::firstRun).toBool() && Prestium::detect()) { config()->set(Config::proxy, Config::Proxy::i2p); config()->set(Config::socks5Port, Prestium::i2pPort()); } if (parser.isSet("use-local-tor")) config()->set(Config::useLocalTor, true); parser.process(app); // Parse again for --help and --version if (!quiet) { QMap 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>(); qRegisterMetaType("TxProofResult"); qRegisterMetaType>(); EventFilter filter; app.installEventFilter(&filter); auto *pool = QThreadPool::globalInstance(); if (pool->maxThreadCount() < 8) { pool->setMaxThreadCount(8); } WindowManager windowManager(QCoreApplication::instance(), &filter); QObject::connect(&app, &SingleApplication::instanceStarted, [&windowManager]() { windowManager.raise(); }); return QApplication::exec(); }