486 lines
15 KiB

/*==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;
}