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