#include <QFile>
#include <QMessageBox>
#include <FileSystem.h>
#include <updater/GoUpdate.h>
#include "UpdateController.h"
#include <QApplication>
#include <thread>
#include <chrono>
#include <LocalPeer.h>

#include "BuildConfig.h"


// from <sys/stat.h>
#ifndef S_IRUSR
#define __S_IREAD 0400         /* Read by owner.  */
#define __S_IWRITE 0200        /* Write by owner.  */
#define __S_IEXEC 0100         /* Execute by owner.  */
#define S_IRUSR __S_IREAD      /* Read by owner.  */
#define S_IWUSR __S_IWRITE     /* Write by owner.  */
#define S_IXUSR __S_IEXEC      /* Execute by owner.  */

#define S_IRGRP (S_IRUSR >> 3) /* Read by group.  */
#define S_IWGRP (S_IWUSR >> 3) /* Write by group.  */
#define S_IXGRP (S_IXUSR >> 3) /* Execute by group.  */

#define S_IROTH (S_IRGRP >> 3) /* Read by others.  */
#define S_IWOTH (S_IWGRP >> 3) /* Write by others.  */
#define S_IXOTH (S_IXGRP >> 3) /* Execute by others.  */
#endif
static QFile::Permissions unixModeToPermissions(const int mode)
{
    QFile::Permissions perms;

    if (mode & S_IRUSR)
    {
        perms |= QFile::ReadUser;
    }
    if (mode & S_IWUSR)
    {
        perms |= QFile::WriteUser;
    }
    if (mode & S_IXUSR)
    {
        perms |= QFile::ExeUser;
    }

    if (mode & S_IRGRP)
    {
        perms |= QFile::ReadGroup;
    }
    if (mode & S_IWGRP)
    {
        perms |= QFile::WriteGroup;
    }
    if (mode & S_IXGRP)
    {
        perms |= QFile::ExeGroup;
    }

    if (mode & S_IROTH)
    {
        perms |= QFile::ReadOther;
    }
    if (mode & S_IWOTH)
    {
        perms |= QFile::WriteOther;
    }
    if (mode & S_IXOTH)
    {
        perms |= QFile::ExeOther;
    }
    return perms;
}

static const QLatin1String liveCheckFile("live.check");

UpdateController::UpdateController(QWidget * parent, const QString& root, const QString updateFilesDir, GoUpdate::OperationList operations)
{
    m_parent = parent;
    m_root = root;
    m_updateFilesDir = updateFilesDir;
    m_operations = operations;
}


void UpdateController::installUpdates()
{
    qint64 pid = -1;
    QStringList args;
    bool started = false;

    qDebug() << "Installing updates.";
#ifdef Q_OS_WIN
    QString finishCmd = QApplication::applicationFilePath();
#elif defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
    QString finishCmd = FS::PathCombine(m_root, BuildConfig.LAUNCHER_NAME);
#elif defined Q_OS_MAC
    QString finishCmd = QApplication::applicationFilePath();
#else
#error Unsupported operating system.
#endif

    QString backupPath = FS::PathCombine(m_root, "update", "backup");
    QDir origin(m_root);

    // clean up the backup folder. it should be empty before we start
    if(!FS::deletePath(backupPath))
    {
        qWarning() << "couldn't remove previous backup folder" << backupPath;
    }
    // and it should exist.
    if(!FS::ensureFolderPathExists(backupPath))
    {
        qWarning() << "couldn't create folder" << backupPath;
        return;
    }

    bool useXPHack = false;
    QString exePath;
    QString exeOrigin;
    QString exeBackup;

    // perform the update operations
    for(auto op: m_operations)
    {
        switch(op.type)
        {
            // replace = move original out to backup, if it exists, move the new file in its place
            case GoUpdate::Operation::OP_REPLACE:
            {
#ifdef Q_OS_WIN32
                QString windowsExeName = BuildConfig.LAUNCHER_NAME + ".exe";
                // hack for people renaming the .exe because ... reasons :)
                if(op.destination == windowsExeName)
                {
                    op.destination = QFileInfo(QApplication::applicationFilePath()).fileName();
                }
#endif
                QFileInfo destination (FS::PathCombine(m_root, op.destination));
                if(destination.exists())
                {
                    QString backupName = op.destination;
                    backupName.replace('/', '_');
                    QString backupFilePath = FS::PathCombine(backupPath, backupName);
                    if(!QFile::rename(destination.absoluteFilePath(), backupFilePath))
                    {
                        qWarning() << "Couldn't move:" << destination.absoluteFilePath() << "to" << backupFilePath;
                        m_failedOperationType = Replace;
                        m_failedFile = op.destination;
                        fail();
                        return;
                    }
                    BackupEntry be;
                    be.original = destination.absoluteFilePath();
                    be.backup = backupFilePath;
                    be.update = op.source;
                    m_replace_backups.append(be);
                }
                // make sure the folder we are putting this into exists
                if(!FS::ensureFilePathExists(destination.absoluteFilePath()))
                {
                    qWarning() << "REPLACE: Couldn't create folder:" << destination.absoluteFilePath();
                    m_failedOperationType = Replace;
                    m_failedFile = op.destination;
                    fail();
                    return;
                }
                // now move the new file in
                if(!QFile::rename(op.source, destination.absoluteFilePath()))
                {
                    qWarning() << "REPLACE: Couldn't move:" << op.source << "to" << destination.absoluteFilePath();
                    m_failedOperationType = Replace;
                    m_failedFile = op.destination;
                    fail();
                    return;
                }
                QFile::setPermissions(destination.absoluteFilePath(), unixModeToPermissions(op.destinationMode));
            }
            break;
            // delete = move original to backup
            case GoUpdate::Operation::OP_DELETE:
            {
                QString destFilePath = FS::PathCombine(m_root, op.destination);
                if(QFile::exists(destFilePath))
                {
                    QString backupName = op.destination;
                    backupName.replace('/', '_');
                    QString trashFilePath = FS::PathCombine(backupPath, backupName);

                    if(!QFile::rename(destFilePath, trashFilePath))
                    {
                        qWarning() << "DELETE: Couldn't move:" << op.destination << "to" << trashFilePath;
                        m_failedFile = op.destination;
                        m_failedOperationType = Delete;
                        fail();
                        return;
                    }
                    BackupEntry be;
                    be.original = destFilePath;
                    be.backup = trashFilePath;
                    m_delete_backups.append(be);
                }
            }
            break;
        }
    }

    // try to start the new binary
    args = qApp->arguments();
    args.removeFirst();

    // on old Windows, do insane things... no error checking here, this is just to have something.
    if(useXPHack)
    {
        QString script;
        auto nativePath = QDir::toNativeSeparators(exePath);
        auto nativeOriginPath = QDir::toNativeSeparators(exeOrigin);
        auto nativeBackupPath = QDir::toNativeSeparators(exeBackup);

        // so we write this vbscript thing...
        QTextStream out(&script);
        out << "WScript.Sleep 1000\n";
        out << "Set fso=CreateObject(\"Scripting.FileSystemObject\")\n";
        out << "Set shell=CreateObject(\"WScript.Shell\")\n";
        out << "fso.MoveFile \"" << nativePath << "\", \"" << nativeBackupPath << "\"\n";
        out << "fso.MoveFile \"" << nativeOriginPath << "\", \"" << nativePath << "\"\n";
        out << "shell.Run \"" << nativePath << "\"\n";

        QString scriptPath = FS::PathCombine(m_root, "update", "update.vbs");

        // we save it
        QFile scriptFile(scriptPath);
        scriptFile.open(QIODevice::WriteOnly);
        scriptFile.write(script.toLocal8Bit().replace("\n", "\r\n"));
        scriptFile.close();

        // we run it
        started = QProcess::startDetached("wscript", {scriptPath}, m_root);

        // and we quit. conscious thought.
        qApp->quit();
        return;
    }
    bool doLiveCheck = true;
    bool startFailed = false;

    // remove live check file, if any
    if(QFile::exists(liveCheckFile))
    {
        if(!QFile::remove(liveCheckFile))
        {
            qWarning() << "Couldn't remove the" << liveCheckFile << "file! We will proceed without :(";
            doLiveCheck = false;
        }
    }

    if(doLiveCheck)
    {
        if(!args.contains("--alive"))
        {
            args.append("--alive");
        }
    }

    // FIXME: reparse args and construct a safe variant from scratch. This is a workaround for GH-1874:
    QStringList realargs;
    int skip = 0;
    for(auto & arg: args)
    {
        if(skip)
        {
            skip--;
            continue;
        }
        if(arg == "-l")
        {
            skip = 1;
            continue;
        }
        realargs.append(arg);
    }

    // start the updated application
    started = QProcess::startDetached(finishCmd, realargs, QDir::currentPath(), &pid);
    // much dumber check - just find out if the call
    if(!started || pid == -1)
    {
        qWarning() << "Couldn't start new process properly!";
        startFailed = true;
    }
    if(!startFailed && doLiveCheck)
    {
        int attempts = 0;
        while(attempts < 10)
        {
            attempts++;
            QString key;
            std::this_thread::sleep_for(std::chrono::milliseconds(250));
            if(!QFile::exists(liveCheckFile))
            {
                qWarning() << "Couldn't find the" << liveCheckFile << "file!";
                startFailed = true;
                continue;
            }
            try
            {
                key = QString::fromUtf8(FS::read(liveCheckFile));
                auto id = ApplicationId::fromRawString(key);
                LocalPeer peer(nullptr, id);
                if(peer.isClient())
                {
                    startFailed = false;
                    qDebug() << "Found process started with key " << key;
                    break;
                }
                else
                {
                    startFailed = true;
                    qDebug() << "Process started with key " << key << "apparently died or is not reponding...";
                    break;
                }
            }
            catch (const Exception &e)
            {
                qWarning() << "Couldn't read the" << liveCheckFile << "file!";
                startFailed = true;
                continue;
            }
        }
    }
    if(startFailed)
    {
        m_failedOperationType = Start;
        fail();
        return;
    }
    else
    {
        origin.rmdir(m_updateFilesDir);
        qApp->quit();
        return;
    }
}

void UpdateController::fail()
{
    qWarning() << "Update failed!";

    QString msg;
    bool doRollback = false;
    QString failTitle = QObject::tr("Update failed!");
    QString rollFailTitle = QObject::tr("Rollback failed!");
    switch (m_failedOperationType)
    {
        case Replace:
        {
            msg = QObject::tr(
                "Couldn't replace file %1. Changes will be reverted.\n"
                "See the %2 log file for details."
            ).arg(m_failedFile, BuildConfig.LAUNCHER_NAME);
            doRollback = true;
            QMessageBox::critical(m_parent, failTitle, msg);
            break;
        }
        case Delete:
        {
            msg = QObject::tr(
                "Couldn't remove file %1. Changes will be reverted.\n"
                "See the %2 log file for details."
            ).arg(m_failedFile, BuildConfig.LAUNCHER_NAME);
            doRollback = true;
            QMessageBox::critical(m_parent, failTitle, msg);
            break;
        }
        case Start:
        {
            msg = QObject::tr("The new version didn't start or is too old and doesn't respond to startup checks.\n"
                "\n"
                "Roll back to previous version?");
            auto result = QMessageBox::critical(
                m_parent,
                failTitle,
                msg,
                QMessageBox::Yes | QMessageBox::No,
                QMessageBox::Yes
            );
            doRollback = (result == QMessageBox::Yes);
            break;
        }
        case Nothing:
        default:
            return;
    }
    if(doRollback)
    {
        auto rollbackOK = rollback();
        if(!rollbackOK)
        {
            msg = QObject::tr("The rollback failed too.\n"
                "You will have to repair %1 manually.\n"
                "Please let us know why and how this happened.").arg(BuildConfig.LAUNCHER_NAME);
            QMessageBox::critical(m_parent, rollFailTitle, msg);
            qApp->quit();
        }
    }
    else
    {
        qApp->quit();
    }
}

bool UpdateController::rollback()
{
    bool revertOK = true;
    // if the above failed, roll back changes
    for(auto backup:m_replace_backups)
    {
        qWarning() << "restoring" << backup.original << "from" << backup.backup;
        if(!QFile::rename(backup.original, backup.update))
        {
            revertOK = false;
            qWarning() << "moving new" << backup.original << "back to" << backup.update << "failed!";
            continue;
        }

        if(!QFile::rename(backup.backup, backup.original))
        {
            revertOK = false;
            qWarning() << "restoring" << backup.original << "failed!";
        }
    }
    for(auto backup:m_delete_backups)
    {
        qWarning() << "restoring" << backup.original << "from" << backup.backup;
        if(!QFile::rename(backup.backup, backup.original))
        {
            revertOK = false;
            qWarning() << "restoring" << backup.original << "failed!";
        }
    }
    return revertOK;
}