/*==LICENSE==* CyanWorlds.com Engine - MMOG client, server and tools Copyright (C) 2011 Cyan Worlds, Inc. This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. Additional permissions under GNU GPL version 3 section 7 If you modify this Program, or any covered work, by linking or combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK (or a modified version of those libraries), containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the licensors of this Program grant you additional permission to convey the resulting work. Corresponding Source for a non-source form of such a combination shall include the source code for the parts of OpenSSL and IJG JPEG Library used as well as that of the covered work. You can contact Cyan Worlds, Inc. by email legal@cyan.com or by snail mail at: Cyan Worlds, Inc. 14617 N Newport Hwy Mead, WA 99021 *==LICENSE==*/ #include "HeadSpin.h" #include "plClientLauncher.h" #include "plFileSystem.h" #include "plProduct.h" #include "hsThread.h" #include "hsTimer.h" #include "plCmdParser.h" #include "pnUtils/pnUtils.h" #include "pnAsyncCore/pnAsyncCore.h" #include "plNetGameLib/plNetGameLib.h" #include "plStatusLog/plStatusLog.h" #include "pfPatcher/plManifests.h" #include "pfPatcher/pfPatcher.h" #include "pfConsoleCore/pfConsoleEngine.h" PF_CONSOLE_LINK_FILE(Core) #include <algorithm> #include <curl/curl.h> #include <deque> plClientLauncher::ErrorFunc s_errorProc = nullptr; // don't even ask, cause I'm not happy about this. const int kNetTransTimeout = 5 * 60 * 1000; // 5m const int kShardStatusUpdateTime = 5; // 5s const int kAsyncCoreShutdownTime = 2 * 1000; // 2s const int kNetCoreUpdateSleepTime = 10; // 10ms // =================================================== class plShardStatus : public hsThread { double fLastUpdate; volatile bool fRunning; hsEvent fUpdateEvent; char fCurlError[CURL_ERROR_SIZE]; public: plClientLauncher::StatusFunc fShardFunc; plShardStatus() : fRunning(true), fLastUpdate(0) { } virtual hsError Run(); void Shutdown(); void Update(); }; static size_t ICurlCallback(void* buffer, size_t size, size_t nmemb, void* thread) { static char status[256]; strncpy(status, (const char *)buffer, std::min<size_t>(size * nmemb, arrsize(status))); status[arrsize(status) - 1] = 0; static_cast<plShardStatus*>(thread)->fShardFunc(status); return size * nmemb; } hsError plShardStatus::Run() { { plString url = GetServerStatusUrl(); // initialize CURL std::unique_ptr<CURL, std::function<void(CURL*)>> curl(curl_easy_init(), curl_easy_cleanup); curl_easy_setopt(curl.get(), CURLOPT_ERRORBUFFER, fCurlError); curl_easy_setopt(curl.get(), CURLOPT_USERAGENT, "UruClient/1.0"); curl_easy_setopt(curl.get(), CURLOPT_URL, url.c_str()); curl_easy_setopt(curl.get(), CURLOPT_WRITEDATA, this); curl_easy_setopt(curl.get(), CURLOPT_WRITEFUNCTION, ICurlCallback); // we want to go ahead and run once fUpdateEvent.Signal(); // loop until we die! do { fUpdateEvent.Wait(); if (!fRunning) break; if (!url.IsEmpty() && curl_easy_perform(curl.get())) fShardFunc(fCurlError); fLastUpdate = hsTimer::GetSysSeconds(); } while (fRunning); } return hsOK; } void plShardStatus::Shutdown() { fRunning = false; fUpdateEvent.Signal(); } void plShardStatus::Update() { double now = hsTimer::GetSysSeconds(); if ((now - fLastUpdate) >= kShardStatusUpdateTime) fUpdateEvent.Signal(); } // =================================================== class plRedistUpdater : public hsThread { bool fSuccess; public: plClientLauncher* fParent; plClientLauncher::InstallRedistFunc fInstallProc; std::deque<plFileName> fRedistQueue; plRedistUpdater() : fSuccess(true) { } ~plRedistUpdater() { // If anything is left in the deque, it was not installed. // We should unlink them so the next launch will redownload and install them. std::for_each(fRedistQueue.begin(), fRedistQueue.end(), [] (const plFileName& file) { plFileSystem::Unlink(file); } ); } virtual void OnQuit() { // If we succeeded, then we should launch the game client... if (fSuccess) fParent->LaunchClient(); } virtual hsError Run() { while (!fRedistQueue.empty()) { if (fInstallProc(fRedistQueue.back())) fRedistQueue.pop_back(); else { s_errorProc(kNetErrInternalError, fRedistQueue.back().AsString()); fSuccess = false; return hsFail; } } return hsOK; } virtual void Start() { if (fRedistQueue.empty()) OnQuit(); else hsThread::Start(); } }; // =================================================== plClientLauncher::plClientLauncher() : fFlags(0), fServerIni("server.ini"), fPatcherFactory(nullptr), fClientExecutable(plManifest::ClientExecutable()), fStatusThread(new plShardStatus()), fInstallerThread(new plRedistUpdater()) { pfPatcher::GetLog()->AddLine(plProduct::ProductString().c_str()); } plClientLauncher::~plClientLauncher() { } // =================================================== plString plClientLauncher::GetAppArgs() const { // If -Repair was specified, there are no args for the next call... if (hsCheckBits(fFlags, kRepairGame)) { return ""; } plStringStream ss; ss << "-ServerIni="; ss << fServerIni.AsString(); // optional args if (hsCheckBits(fFlags, kClientImage)) ss << " -Image"; if (hsCheckBits(fFlags, kPatchOnly)) ss << " -PatchOnly"; if (hsCheckBits(fFlags, kSkipLoginDialog)) ss << " -SkipLoginDialog"; return ss.GetString(); } void plClientLauncher::IOnPatchComplete(ENetError result, const plString& msg) { if (IS_NET_SUCCESS(result)) { // a couple of options // 1. we self-patched and didn't update anything. patch the main client. // 2. we self-patched and did things and stuff... re-run myself. // 3. we patched the client... run it. if (!hsCheckBits(fFlags, kHaveSelfPatched) && (fClientExecutable == plManifest::ClientExecutable())) { // case 1 hsSetBits(fFlags, kHaveSelfPatched); PatchClient(); } else { // cases 2 & 3 -- update any redistributables, then launch the client. fInstallerThread->fParent = this; fInstallerThread->Start(); } } else if (s_errorProc) s_errorProc(result, msg); } bool plClientLauncher::IApproveDownload(const plFileName& file) { // So, for a repair, what we want to do is quite simple. // That is: download everything that is NOT in the root directory. plFileName path = file.StripFileName(); return !path.AsString().IsEmpty(); } void plClientLauncher::LaunchClient() const { if (fStatusFunc) fStatusFunc("Launching..."); fLaunchClientFunc(fClientExecutable, GetAppArgs()); } void plClientLauncher::PatchClient() { if (fStatusFunc) { if (hsCheckBits(fFlags, kGameDataOnly)) fStatusFunc("Verifying game data..."); else fStatusFunc("Checking for updates..."); } hsAssert(fPatcherFactory, "why is the patcher factory nil?"); pfPatcher* patcher = fPatcherFactory(); patcher->OnCompletion(std::bind(&plClientLauncher::IOnPatchComplete, this, std::placeholders::_1, std::placeholders::_2)); patcher->OnSelfPatch([&](const plFileName& file) { fClientExecutable = file; }); patcher->OnRedistUpdate([&](const plFileName& file) { fInstallerThread->fRedistQueue.push_back(file); }); // If this is a repair, we need to approve the downloads... if (hsCheckBits(fFlags, kGameDataOnly)) patcher->OnFileDownloadDesired(std::bind(&plClientLauncher::IApproveDownload, this, std::placeholders::_1)); // Let's get 'er done. if (hsCheckBits(fFlags, kHaveSelfPatched)) { if (hsCheckBits(fFlags, kClientImage)) patcher->RequestManifest(plManifest::ClientImageManifest()); else patcher->RequestManifest(plManifest::ClientManifest()); } else patcher->RequestManifest(plManifest::PatcherManifest()); patcher->Start(); } bool plClientLauncher::CompleteSelfPatch(std::function<void(void)> waitProc) const { if (hsCheckBits(fFlags, kHaveSelfPatched)) return false; plString myExe = plFileSystem::GetCurrentAppPath().GetFileName(); if (myExe.CompareI(plManifest::PatcherExecutable().AsString()) != 0) { waitProc(); // so now we need to unlink the old patcher, and move ME into that fool's place... // then we can continue on our merry way! if (!plFileSystem::Unlink(plManifest::PatcherExecutable())) { hsMessageBox("Failed to delete old patcher executable!", "Error", hsMessageBoxNormal, hsMessageBoxIconError); return true; } if (!plFileSystem::Move(plFileSystem::GetCurrentAppPath(), plManifest::PatcherExecutable())) { hsMessageBox("Failed to move patcher executable!", "Error", hsMessageBoxNormal, hsMessageBoxIconError); return true; } // Now, execute the new patcher... fLaunchClientFunc(plManifest::PatcherExecutable(), GetAppArgs() + " -NoSelfPatch"); return true; } return false; } // =================================================== static void IGotFileServIPs(ENetError result, void* param, const plString& addr) { plClientLauncher* launcher = static_cast<plClientLauncher*>(param); NetCliGateKeeperDisconnect(); if (IS_NET_SUCCESS(result)) { // bah... why do I even bother plString eapSucks[] = { addr }; NetCliFileStartConnect(eapSucks, 1, true); // Who knows if we will actually connect. So let's start updating. launcher->PatchClient(); } else if (s_errorProc) s_errorProc(result, "Failed to get FileServ addresses"); } static void IEapSucksErrorProc(ENetProtocol protocol, ENetError error) { if (s_errorProc) { plString msg = plFormat("Protocol: {}", NetProtocolToString(protocol)); s_errorProc(error, msg); } } void plClientLauncher::InitializeNetCore() { // initialize shard status hsTimer::SetRealTime(true); fStatusThread->Start(); // init eap... AsyncCoreInitialize(); NetClientInitialize(); NetClientSetErrorHandler(IEapSucksErrorProc); NetClientSetTransTimeoutMs(kNetTransTimeout); // Gotta grab the filesrvs from the gate const plString* addrs; uint32_t num = GetGateKeeperSrvHostnames(addrs); NetCliGateKeeperStartConnect(addrs, num); NetCliGateKeeperFileSrvIpAddressRequest(IGotFileServIPs, this, true); } // =================================================== void plClientLauncher::PumpNetCore() const { // this ain't net core, but it needs to be pumped :( hsTimer::IncSysSeconds(); // pump eap NetClientUpdate(); // pump shard status fStatusThread->Update(); // don't nom all the CPU... kthx hsSleep::Sleep(kNetCoreUpdateSleepTime); } void plClientLauncher::ShutdownNetCore() const { // shutdown shard status fStatusThread->Shutdown(); // unhook the neterr callback at this point because all transactions // will fail when we call NetClientDestroy s_errorProc = nullptr; // shutdown eap NetCliGateKeeperDisconnect(); NetCliFileDisconnect(); NetClientDestroy(); // shutdown eap (part deux) AsyncCoreDestroy(kAsyncCoreShutdownTime); } // =================================================== bool plClientLauncher::LoadServerIni() const { PF_CONSOLE_INITIALIZE(Core); pfConsoleEngine console; return console.ExecuteFile(fServerIni); } void plClientLauncher::ParseArguments() { #define APPLY_FLAG(arg, flag) \ if (cmdParser.GetBool(arg)) \ fFlags |= flag; enum { kArgServerIni, kArgNoSelfPatch, kArgImage, kArgRepairGame, kArgPatchOnly, kArgSkipLoginDialog }; const plCmdArgDef cmdLineArgs[] = { { kCmdArgFlagged | kCmdTypeString, "ServerIni", kArgServerIni }, { kCmdArgFlagged | kCmdTypeBool, "NoSelfPatch", kArgNoSelfPatch }, { kCmdArgFlagged | kCmdTypeBool, "Image", kArgImage }, { kCmdArgFlagged | kCmdTypeBool, "Repair", kArgRepairGame }, { kCmdArgFlagged | kCmdTypeBool, "PatchOnly", kArgPatchOnly }, { kCmdArgFlagged | kCmdTypeBool, "SkipLoginDialog", kArgSkipLoginDialog } }; std::vector<plString> args; args.reserve(__argc); for (size_t i = 0; i < __argc; i++) { args.push_back(plString::FromUtf8(__argv[i])); } plCmdParser cmdParser(cmdLineArgs, arrsize(cmdLineArgs)); cmdParser.Parse(args); // cache 'em if (cmdParser.IsSpecified(kArgServerIni)) fServerIni = cmdParser.GetString(kArgServerIni); APPLY_FLAG(kArgNoSelfPatch, kHaveSelfPatched); APPLY_FLAG(kArgImage, kClientImage); APPLY_FLAG(kArgRepairGame, kRepairGame); APPLY_FLAG(kArgPatchOnly, kPatchOnly); APPLY_FLAG(kArgSkipLoginDialog, kSkipLoginDialog); // last chance setup if (hsCheckBits(fFlags, kPatchOnly)) fClientExecutable = ""; else if (hsCheckBits(fFlags, kRepairGame)) fClientExecutable = plManifest::PatcherExecutable(); #undef APPLY_FLAG } void plClientLauncher::SetErrorProc(ErrorFunc proc) { s_errorProc = proc; } void plClientLauncher::SetInstallerProc(InstallRedistFunc proc) { fInstallerThread->fInstallProc = proc; } void plClientLauncher::SetShardProc(StatusFunc proc) { fStatusThread->fShardFunc = proc; }