/* Copyright (c) 2016, Stanislaw Halik <sthalik@misaki.pl>

 * Permission to use, copy, modify, and/or distribute this
 * software for any purpose with or without fee is hereby granted,
 * provided that the above copyright notice and this permission
 * notice appear in all copies.
 */

#include <cstdlib>

#include "migration.hpp"

#include "options/options.hpp"

#include <QString>
#include <QSettings>
#include <QChar>

#include <QDebug>

#include <memory>

using namespace options::globals;

// individual migrations are run in the UI thread. they can be interactive if necessary.

namespace migrations::detail {

static std::vector<mptr> migration_list;
static std::vector<mfun> migration_thunks;

void migrator::register_migration(mptr const& m)
{
    const QString date = m->unique_date();

    for (mptr const& m2 : migration_list)
        if (m2->unique_date() == date)
            std::abort();

    if (date.size() != 11)
        abort();

    if (date[8] != QChar('_'))
        abort();

    const QString year = date.left(4);
    const QString month = date.mid(4, 2);
    const QString day = date.mid(6, 2);
    const QString serial = date.mid(9, 2);

    bool ok = true;

    if (year < "2016")
        abort();

    const int month_ = to_int(month, ok), day_ = to_int(day, ok);

    (void) to_int(year, ok);
    (void) to_int(serial, ok);

    if (!ok)
        abort();

    if (month_ < 1 || month_ > 12)
        abort();

    if (day_ < 1 || day_ > 31)
        abort();

    migration_list.push_back(m);
}

void migrator::eval_thunks()
{
    for (auto& fun : migration_thunks)
    {
        mptr m = fun();
        register_migration(m);
    }
    if (!migration_thunks.empty())
        sort_migrations();
    migration_thunks.clear();
}

void migrator::add_migration_thunk(mfun& thunk)
{
    migration_thunks.push_back(thunk);
}

std::vector<mptr>& migrator::migrations()
{
    eval_thunks();
    return migration_list;
}

void migrator::sort_migrations()
{
    std::sort(migration_list.begin(), migration_list.end(),
              [](const mptr x, const mptr y) {
        return x->unique_date() < y->unique_date();
    });
}

QString migrator::last_migration_time()
{
    QString ret;

    with_settings_object([&](QSettings& s) {
        s.beginGroup("migrations");
        ret = s.value("last-migration-at", "19700101_00").toString();
        s.endGroup();
    });

    return ret;
}

QString migrator::time_after_migrations()
{
    const std::vector<mptr>& list = migrations();

    if (list.empty())
        return QStringLiteral("19700101_00");

    QString ret = list[list.size() - 1]->unique_date();
    ret += QStringLiteral("~");

    return ret;
}

void migrator::set_last_migration_time(const QString& val)
{
    with_settings_object([&](QSettings& s) {
        s.beginGroup("migrations");
        const QString old_value = s.value("last-migration-at", "").toString();
        if (val != old_value)
        {
            s.setValue("last-migration-at", val);
            mark_ini_modified();
        }
        s.endGroup();
    });
}

int migrator::to_int(const QString& str, bool& ok)
{
    bool tmp = false;
    const int ret = int(str.toUInt(&tmp));
    ok &= tmp;
    return ret;
}

std::vector<QString> migrator::run()
{
    std::vector<QString> done;

    const QString last_migration = last_migration_time();

    with_settings_object([&](QSettings&) {
        for (mptr const& m : migrations())
        {
            const QString date = m->unique_date();

            if (date <= last_migration)
                continue;

            if (m->should_run())
            {
                const QByteArray name = m->name().toUtf8();
                const QByteArray date = m->unique_date().toUtf8();
                qDebug() << "migrate:" << date.constData() << name.constData();
                m->run();
                done.push_back(m->name());
            }
        }
        mark_config_as_not_needing_migration();
    });

    return done;
}

} // ns migrations::detail

namespace migrations {

migration::migration() = default;
migration::~migration() = default;

} // ns migrations

std::vector<QString> run_migrations()
{
    return migrations::detail::migrator::run();
}

void mark_config_as_not_needing_migration()
{
    using m = migrations::detail::migrator;

    m::mark_config_as_not_needing_migration();
}

void migrations::detail::migrator::mark_config_as_not_needing_migration()
{
    set_last_migration_time(time_after_migrations());
}