/* Copyright 2013-2019 MultiMC Contributors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#include <QDir>
#include <QString>
#include <QJsonDocument>
#include <QJsonObject>
#include <QJsonArray>
#include <QJsonValue>
#include <quazip.h>
#include <quazipfile.h>

#include "Mod.h"
#include "settings/INIFile.h"
#include <FileSystem.h>
#include <QDebug>

Mod::Mod(const QFileInfo &file)
{
    repath(file);
    m_changedDateTime = file.lastModified();
}

void Mod::repath(const QFileInfo &file)
{
    m_file = file;
    QString name_base = file.fileName();

    m_type = Mod::MOD_UNKNOWN;

    if (m_file.isDir())
    {
        m_type = MOD_FOLDER;
        m_name = name_base;
        m_mmc_id = name_base;
    }
    else if (m_file.isFile())
    {
        if (name_base.endsWith(".disabled"))
        {
            m_enabled = false;
            name_base.chop(9);
        }
        else
        {
            m_enabled = true;
        }
        m_mmc_id = name_base;
        if (name_base.endsWith(".zip") || name_base.endsWith(".jar"))
        {
            m_type = MOD_ZIPFILE;
            name_base.chop(4);
        }
        else if (name_base.endsWith(".litemod"))
        {
            m_type = MOD_LITEMOD;
            name_base.chop(8);
        }
        else
        {
            m_type = MOD_SINGLEFILE;
        }
        m_name = name_base;
    }

    if (m_type == MOD_ZIPFILE)
    {
        QuaZip zip(m_file.filePath());
        if (!zip.open(QuaZip::mdUnzip))
            return;

        QuaZipFile file(&zip);

        if (zip.setCurrentFile("mcmod.info"))
        {
            if (!file.open(QIODevice::ReadOnly))
            {
                zip.close();
                return;
            }

            ReadMCModInfo(file.readAll());
            file.close();
            zip.close();
            return;
        }
        else if (zip.setCurrentFile("forgeversion.properties"))
        {
            if (!file.open(QIODevice::ReadOnly))
            {
                zip.close();
                return;
            }

            ReadForgeInfo(file.readAll());
            file.close();
            zip.close();
            return;
        }

        zip.close();
    }
    else if (m_type == MOD_FOLDER)
    {
        QFileInfo mcmod_info(FS::PathCombine(m_file.filePath(), "mcmod.info"));
        if (mcmod_info.isFile())
        {
            QFile mcmod(mcmod_info.filePath());
            if (!mcmod.open(QIODevice::ReadOnly))
                return;
            auto data = mcmod.readAll();
            if (data.isEmpty() || data.isNull())
                return;
            ReadMCModInfo(data);
        }
    }
    else if (m_type == MOD_LITEMOD)
    {
        QuaZip zip(m_file.filePath());
        if (!zip.open(QuaZip::mdUnzip))
            return;

        QuaZipFile file(&zip);

        if (zip.setCurrentFile("litemod.json"))
        {
            if (!file.open(QIODevice::ReadOnly))
            {
                zip.close();
                return;
            }

            ReadLiteModInfo(file.readAll());
            file.close();
        }
        zip.close();
    }
}

// NEW format
// https://github.com/MinecraftForge/FML/wiki/FML-mod-information-file/6f62b37cea040daf350dc253eae6326dd9c822c3

// OLD format:
// https://github.com/MinecraftForge/FML/wiki/FML-mod-information-file/5bf6a2d05145ec79387acc0d45c958642fb049fc
void Mod::ReadMCModInfo(QByteArray contents)
{
    auto getInfoFromArray = [&](QJsonArray arr)->void
    {
        if (!arr.at(0).isObject())
            return;
        auto firstObj = arr.at(0).toObject();
        m_mod_id = firstObj.value("modid").toString();
        m_name = firstObj.value("name").toString();
        m_version = firstObj.value("version").toString();
        m_homeurl = firstObj.value("url").toString();
        m_updateurl = firstObj.value("updateUrl").toString();
        m_homeurl = m_homeurl.trimmed();
        if(!m_homeurl.isEmpty())
        {
            // fix up url.
            if (!m_homeurl.startsWith("http://") && !m_homeurl.startsWith("https://") &&
                !m_homeurl.startsWith("ftp://"))
            {
                m_homeurl.prepend("http://");
            }
        }
        m_description = firstObj.value("description").toString();
        QJsonArray authors = firstObj.value("authorList").toArray();
        if (authors.size() == 0)
            authors = firstObj.value("authors").toArray();

        if (authors.size() == 0)
            m_authors = "";
        else if (authors.size() >= 1)
        {
            m_authors = authors.at(0).toString();
            for (int i = 1; i < authors.size(); i++)
            {
                m_authors += ", " + authors.at(i).toString();
            }
        }
        m_credits = firstObj.value("credits").toString();
        return;
    }
    ;
    QJsonParseError jsonError;
    QJsonDocument jsonDoc = QJsonDocument::fromJson(contents, &jsonError);
    // this is the very old format that had just the array
    if (jsonDoc.isArray())
    {
        getInfoFromArray(jsonDoc.array());
    }
    else if (jsonDoc.isObject())
    {
        auto val = jsonDoc.object().value("modinfoversion");
        if(val.isUndefined())
            val = jsonDoc.object().value("modListVersion");
        int version = val.toDouble();
        if (version != 2)
        {
            qCritical() << "BAD stuff happened to mod json:";
            qCritical() << contents;
            return;
        }
        auto arrVal = jsonDoc.object().value("modlist");
        if(arrVal.isUndefined())
            arrVal = jsonDoc.object().value("modList");
        if (arrVal.isArray())
        {
            getInfoFromArray(arrVal.toArray());
        }
    }
}

void Mod::ReadForgeInfo(QByteArray contents)
{
    // Read the data
    m_name = "Minecraft Forge";
    m_mod_id = "Forge";
    m_homeurl = "http://www.minecraftforge.net/forum/";
    INIFile ini;
    if (!ini.loadFile(contents))
        return;

    QString major = ini.get("forge.major.number", "0").toString();
    QString minor = ini.get("forge.minor.number", "0").toString();
    QString revision = ini.get("forge.revision.number", "0").toString();
    QString build = ini.get("forge.build.number", "0").toString();

    m_version = major + "." + minor + "." + revision + "." + build;
}

void Mod::ReadLiteModInfo(QByteArray contents)
{
    QJsonParseError jsonError;
    QJsonDocument jsonDoc = QJsonDocument::fromJson(contents, &jsonError);
    auto object = jsonDoc.object();
    if (object.contains("name"))
    {
        m_mod_id = m_name = object.value("name").toString();
    }
    if (object.contains("version"))
    {
        m_version = object.value("version").toString("");
    }
    else
    {
        m_version = object.value("revision").toString("");
    }
    m_mcversion = object.value("mcversion").toString();
    m_authors = object.value("author").toString();
    m_description = object.value("description").toString();
    m_homeurl = object.value("url").toString();
}

bool Mod::replace(Mod &with)
{
    if (!destroy())
        return false;
    bool success = false;
    auto t = with.type();

    if (t == MOD_ZIPFILE || t == MOD_SINGLEFILE || t == MOD_LITEMOD)
    {
        qDebug() << "Copy: " << with.m_file.filePath() << " to " << m_file.filePath();
        success = QFile::copy(with.m_file.filePath(), m_file.filePath());
    }
    if (t == MOD_FOLDER)
    {
        success = FS::copy(with.m_file.filePath(), m_file.path())();
    }
    if (success)
    {
        m_name = with.m_name;
        m_mmc_id = with.m_mmc_id;
        m_mod_id = with.m_mod_id;
        m_version = with.m_version;
        m_mcversion = with.m_mcversion;
        m_description = with.m_description;
        m_authors = with.m_authors;
        m_credits = with.m_credits;
        m_homeurl = with.m_homeurl;
        m_type = with.m_type;
        m_file.refresh();
    }
    return success;
}

bool Mod::destroy()
{
    if (m_type == MOD_FOLDER)
    {
        QDir d(m_file.filePath());
        if (d.removeRecursively())
        {
            m_type = MOD_UNKNOWN;
            return true;
        }
        return false;
    }
    else if (m_type == MOD_SINGLEFILE || m_type == MOD_ZIPFILE || m_type == MOD_LITEMOD)
    {
        QFile f(m_file.filePath());
        if (f.remove())
        {
            m_type = MOD_UNKNOWN;
            return true;
        }
        return false;
    }
    return true;
}

QString Mod::version() const
{
    switch (type())
    {
    case MOD_ZIPFILE:
    case MOD_LITEMOD:
        return m_version;
    case MOD_FOLDER:
        return "Folder";
    case MOD_SINGLEFILE:
        return "File";
    default:
        return "VOID";
    }
}

bool Mod::enable(bool value)
{
    if (m_type == Mod::MOD_UNKNOWN || m_type == Mod::MOD_FOLDER)
        return false;

    if (m_enabled == value)
        return false;

    QString path = m_file.absoluteFilePath();
    if (value)
    {
        QFile foo(path);
        if (!path.endsWith(".disabled"))
            return false;
        path.chop(9);
        if (!foo.rename(path))
            return false;
    }
    else
    {
        QFile foo(path);
        path += ".disabled";
        if (!foo.rename(path))
            return false;
    }
    m_file = QFileInfo(path);
    m_enabled = value;
    return true;
}
bool Mod::operator==(const Mod &other) const
{
    return mmc_id() == other.mmc_id();
}
bool Mod::strongCompare(const Mod &other) const
{
    return mmc_id() == other.mmc_id() && version() == other.version() && type() == other.type();
}