// SPDX-FileCopyrightText: 2023 Joshua Goins // SPDX-License-Identifier: GPL-3.0-or-later #include "gameinstaller.h" #include #include #include #include #include #include #include #include "account.h" #include "assetupdater.h" #include "astra_log.h" #include "benchmarkinstaller.h" #include "compatibilitytoolinstaller.h" #include "gamerunner.h" #include "launchercore.h" #include "sapphirelogin.h" #include "squareenixlogin.h" #include "utility.h" using namespace Qt::StringLiterals; LauncherCore::LauncherCore() : QObject() { m_settings = new LauncherSettings(this); m_mgr = new QNetworkAccessManager(this); m_sapphireLogin = new SapphireLogin(*this, this); m_squareEnixLogin = new SquareEnixLogin(*this, this); m_profileManager = new ProfileManager(*this, this); m_accountManager = new AccountManager(*this, this); m_runner = new GameRunner(*this, this); m_profileManager->load(); m_accountManager->load(); // restore profile -> account connections for (auto profile : m_profileManager->profiles()) { if (auto account = m_accountManager->getByUuid(profile->accountUuid())) { profile->setAccount(account); } } // set default profile, if found if (auto profile = m_profileManager->getProfileByUUID(m_settings->currentProfile())) { setCurrentProfile(profile); } m_loadingFinished = true; Q_EMIT loadingFinished(); } void LauncherCore::initializeSteam() { m_steamApi = new SteamAPI(this); m_steamApi->setLauncherMode(true); } void LauncherCore::login(Profile *profile, const QString &username, const QString &password, const QString &oneTimePassword) { Q_ASSERT(profile != nullptr); auto loginInformation = new LoginInformation(this); loginInformation->profile = profile; // Benchmark never has to login, of course if (!profile->isBenchmark()) { loginInformation->username = username; loginInformation->password = password; loginInformation->oneTimePassword = oneTimePassword; if (profile->account()->rememberPassword()) { profile->account()->setPassword(password); } } beginLogin(*loginInformation); } bool LauncherCore::autoLogin(Profile *profile) { Q_ASSERT(profile != nullptr); QString otp; if (profile->account()->useOTP()) { if (!profile->account()->rememberOTP()) { Q_EMIT loginError(i18n("This account does not have an OTP secret set, but requires it for login.")); return false; } otp = profile->account()->getOTP(); if (otp.isEmpty()) { Q_EMIT loginError(i18n("Failed to generate OTP, review the stored secret.")); return false; } } login(profile, profile->account()->name(), profile->account()->getPassword(), otp); return true; } GameInstaller *LauncherCore::createInstaller(Profile *profile) { Q_ASSERT(profile != nullptr); return new GameInstaller(*this, *profile, this); } GameInstaller *LauncherCore::createInstallerFromExisting(Profile *profile, const QString &filePath) { Q_ASSERT(profile != nullptr); return new GameInstaller(*this, *profile, filePath, this); } CompatibilityToolInstaller *LauncherCore::createCompatInstaller() { return new CompatibilityToolInstaller(*this, this); } BenchmarkInstaller *LauncherCore::createBenchmarkInstaller(Profile *profile) { Q_ASSERT(profile != nullptr); return new BenchmarkInstaller(*this, *profile, this); } BenchmarkInstaller *LauncherCore::createBenchmarkInstallerFromExisting(Profile *profile, const QString &filePath) { Q_ASSERT(profile != nullptr); return new BenchmarkInstaller(*this, *profile, filePath, this); } void LauncherCore::clearAvatarCache() { const QString cacheLocation = QStandardPaths::standardLocations(QStandardPaths::CacheLocation)[0] + QStringLiteral("/avatars"); if (QDir(cacheLocation).exists()) { QDir(cacheLocation).removeRecursively(); } } void LauncherCore::refreshNews() { fetchNews(); } void LauncherCore::refreshLogoImage() { const QDir cacheDir = QStandardPaths::standardLocations(QStandardPaths::StandardLocation::CacheLocation).last(); const QDir logoDir = cacheDir.absoluteFilePath(QStringLiteral("logos")); if (!logoDir.exists()) { QDir().mkpath(logoDir.absolutePath()); } const auto saveTexture = [](GameData *data, const QString &path, const QString &name) { if (QFile::exists(name)) { return; } auto file = physis_gamedata_extract_file(data, path.toStdString().c_str()); if (file.data != nullptr) { auto tex = physis_texture_parse(file); QImage image(tex.rgba, tex.width, tex.height, QImage::Format_RGBA8888); image.save(name); } }; // TODO: this finds the first profile that has a valid image, but this could probably be cached per-profile for (int i = 0; i < m_profileManager->numProfiles(); i++) { auto profile = m_profileManager->getProfile(i); if (profile->isGameInstalled() && profile->gameData()) { // A Realm Reborn saveTexture(profile->gameData(), QStringLiteral("ui/uld/Title_Logo.tex"), logoDir.absoluteFilePath(QStringLiteral("ffxiv.png"))); for (int j = 0; j < profile->numInstalledExpansions(); j++) { const int expansionNumber = 100 * (j + 3); // logo number starts at 300 for ex1 saveTexture(profile->gameData(), QStringLiteral("ui/uld/Title_Logo%1_hr1.tex").arg(expansionNumber), logoDir.absoluteFilePath(QStringLiteral("ex%1.png").arg(j + 1))); } } } QList imageFiles; // TODO: sort QDirIterator it(logoDir.absolutePath()); while (it.hasNext()) { const QFileInfo logoFile(it.next()); if (logoFile.completeSuffix() != QStringLiteral("png")) { continue; } imageFiles.push_back(logoFile.absoluteFilePath()); } if (!imageFiles.isEmpty()) { m_cachedLogoImage = imageFiles.last(); Q_EMIT cachedLogoImageChanged(); } } Profile *LauncherCore::currentProfile() const { return m_profileManager->getProfile(m_currentProfileIndex); } void LauncherCore::setCurrentProfile(Profile *profile) { Q_ASSERT(profile != nullptr); const int newIndex = m_profileManager->getProfileIndex(profile->uuid()); if (newIndex != m_currentProfileIndex) { m_currentProfileIndex = newIndex; m_settings->setCurrentProfile(profile->uuid()); m_settings->config()->save(); Q_EMIT currentProfileChanged(); } } [[nodiscard]] QString LauncherCore::autoLoginProfileName() const { return m_settings->config()->autoLoginProfile(); } [[nodiscard]] Profile *LauncherCore::autoLoginProfile() const { if (m_settings->config()->autoLoginProfile().isEmpty()) { return nullptr; } return m_profileManager->getProfileByUUID(m_settings->config()->autoLoginProfile()); } void LauncherCore::setAutoLoginProfile(Profile *profile) { if (profile != nullptr) { auto uuid = profile->uuid(); if (uuid != m_settings->config()->autoLoginProfile()) { m_settings->config()->setAutoLoginProfile(uuid); } } else { m_settings->config()->setAutoLoginProfile({}); } m_settings->config()->save(); Q_EMIT autoLoginProfileChanged(); } void LauncherCore::buildRequest(const Profile &settings, QNetworkRequest &request) { Utility::setSSL(request); if (settings.account()->license() == Account::GameLicense::macOS) { request.setHeader(QNetworkRequest::UserAgentHeader, QByteArrayLiteral("macSQEXAuthor/2.0.0(MacOSX; ja-jp)")); } else { request.setHeader(QNetworkRequest::UserAgentHeader, QStringLiteral("SQEXAuthor/2.0.0(Windows 6.2; ja-jp; %1)").arg(QString::fromUtf8(QSysInfo::bootUniqueId()))); } request.setRawHeader(QByteArrayLiteral("Accept"), QByteArrayLiteral("image/gif, image/jpeg, image/pjpeg, application/x-ms-application, application/xaml+xml, " "application/x-ms-xbap, */*")); request.setRawHeader(QByteArrayLiteral("Accept-Encoding"), QByteArrayLiteral("gzip, deflate")); request.setRawHeader(QByteArrayLiteral("Accept-Language"), QByteArrayLiteral("en-us")); } void LauncherCore::setupIgnoreSSL(QNetworkReply *reply) { Q_ASSERT(reply != nullptr); if (m_settings->preferredProtocol() == QStringLiteral("http")) { connect(reply, &QNetworkReply::sslErrors, this, [reply](const QList &errors) { reply->ignoreSslErrors(errors); }); } } bool LauncherCore::isLoadingFinished() const { return m_loadingFinished; } bool LauncherCore::isSteam() const { return m_steamApi != nullptr; } bool LauncherCore::isSteamDeck() const { return qEnvironmentVariable("SteamDeck") == QStringLiteral("1"); } bool LauncherCore::isWindows() { #if defined(Q_OS_WIN) return true; #else return false; #endif } bool LauncherCore::needsCompatibilityTool() { return !isWindows(); } bool LauncherCore::isPatching() const { return m_isPatching; } QNetworkAccessManager *LauncherCore::mgr() { return m_mgr; } LauncherSettings *LauncherCore::settings() { return m_settings; } ProfileManager *LauncherCore::profileManager() { return m_profileManager; } AccountManager *LauncherCore::accountManager() { return m_accountManager; } Headline *LauncherCore::headline() const { return m_headline; } QString LauncherCore::cachedLogoImage() const { return m_cachedLogoImage; } QCoro::Task<> LauncherCore::beginLogin(LoginInformation &info) { // Hmm, I don't think we're set up for this yet? if (!info.profile->isBenchmark()) { info.profile->account()->updateConfig(); } auto assetUpdater = new AssetUpdater(*info.profile, *this, this); if (co_await assetUpdater->update()) { std::optional auth; if (!info.profile->isBenchmark()) { if (info.profile->account()->isSapphire()) { auth = co_await m_sapphireLogin->login(info.profile->account()->lobbyUrl(), info); } else { auth = co_await m_squareEnixLogin->login(&info); } } // If we expect an auth ticket, don't continue if missing if (!info.profile->isBenchmark() && auth == std::nullopt) { co_return; } Q_EMIT stageChanged(i18n("Launching game...")); if (isSteam()) { m_steamApi->setLauncherMode(false); } m_runner->beginGameExecutable(*info.profile, auth); } assetUpdater->deleteLater(); } QCoro::Task<> LauncherCore::fetchNews() { qInfo(ASTRA_LOG) << "Fetching news..."; QUrlQuery query; query.addQueryItem(QStringLiteral("lang"), QStringLiteral("en-us")); query.addQueryItem(QStringLiteral("media"), QStringLiteral("pcapp")); QUrl url; url.setScheme(m_settings->preferredProtocol()); url.setHost(QStringLiteral("frontier.%1").arg(m_settings->squareEnixServer())); url.setPath(QStringLiteral("/news/headline.json")); url.setQuery(query); QNetworkRequest request(QUrl(QStringLiteral("%1&%2").arg(url.toString(), QString::number(QDateTime::currentMSecsSinceEpoch())))); request.setRawHeader(QByteArrayLiteral("Accept"), QByteArrayLiteral("application/json, text/plain, */*")); request.setRawHeader(QByteArrayLiteral("Origin"), QByteArrayLiteral("https://launcher.finalfantasyxiv.com")); request.setRawHeader(QByteArrayLiteral("Referer"), QStringLiteral("https://launcher.finalfantasyxiv.com/v600/index.html?rc_lang=%1&time=%2") .arg(QStringLiteral("en-us"), QDateTime::currentDateTimeUtc().toString(QStringLiteral("yyyy-MM-dd-HH"))) .toUtf8()); Utility::printRequest(QStringLiteral("GET"), request); auto reply = mgr()->get(request); co_await reply; auto document = QJsonDocument::fromJson(reply->readAll()); auto headline = new Headline(this); if (document.isEmpty()) { headline->failedToLoad = true; } const auto parseNews = [](QJsonObject object) -> News { News news; news.date = QDateTime::fromString(object["date"_L1].toString(), Qt::DateFormat::ISODate); news.id = object["id"_L1].toString(); news.tag = object["tag"_L1].toString(); news.title = object["title"_L1].toString(); if (object["url"_L1].toString().isEmpty()) { news.url = QUrl(QStringLiteral("https://na.finalfantasyxiv.com/lodestone/news/detail/%1").arg(news.id)); } else { news.url = QUrl(object["url"_L1].toString()); } return news; }; for (const auto bannerObject : document.object()["banner"_L1].toArray()) { headline->banners.push_back( {.link = QUrl(bannerObject.toObject()["link"_L1].toString()), .bannerImage = QUrl(bannerObject.toObject()["lsb_banner"_L1].toString())}); } for (const auto newsObject : document.object()["news"_L1].toArray()) { headline->news.push_back(parseNews(newsObject.toObject())); } for (const auto pinnedObject : document.object()["pinned"_L1].toArray()) { headline->pinned.push_back(parseNews(pinnedObject.toObject())); } for (const auto pinnedObject : document.object()["topics"_L1].toArray()) { headline->topics.push_back(parseNews(pinnedObject.toObject())); } m_headline = headline; Q_EMIT newsChanged(); } #include "moc_launchercore.cpp"