/* 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.
 */

#pragma once

#include <QString>
#include "export.hpp"

#include <memory>
#include <vector>
#include <functional>

namespace migrations {

struct migration;
class registrator;

namespace detail {
    using mptr = std::shared_ptr<migration>;
    using mfun = std::function<mptr ()>;

    struct migrator
    {
        static std::vector<QString> run();
        static void add_migration_thunk(std::function<mptr()>& thunk);
        static void mark_config_as_not_needing_migration();

    private:
        static void sort_migrations();

        static void register_migration(const mptr& m);
        static std::vector<mptr>& migrations();

        static void eval_thunks();

        static QString last_migration_time();
        static QString time_after_migrations();

        static void set_last_migration_time(const QString& val);
        static int to_int(const QString& str, bool& ok);
    };

    template<typename t>
    struct registrator final
    {
        registrator()
        {
            mfun f { [] { return std::shared_ptr<migration>(new t); } };

            migrator::add_migration_thunk(f);
        }
    };
}

#ifndef __COUNTER__
#   error "oops, need __COUNTER__ extension for preprocessor"
#endif

#define MIGRATE_EXPAND2(x) x
#define MIGRATE_EXPAND1(x) MIGRATE_EXPAND2(x)
#define MIGRATE_EXPANDED2(type, ctr) \
    static ::migrations::detail::registrator<type> opentrack_migration_registrator_ ## ctr
#define MIGRATE_EXPANDED1(type, ctr) \
    MIGRATE_EXPANDED2(type, ctr)

#define OPENTRACK_MIGRATION(type) \
    MIGRATE_EXPANDED1(type, MIGRATE_EXPAND1(__COUNTER__))

#ifdef Q_CREATOR_RUN
#   pragma clang diagnostic ignored "-Wweak-vtables"
#endif

struct migration
{
    migration() = default;
    migration(const migration&) = delete;
    migration& operator=(const migration&) = delete;

    inline virtual ~migration();
    virtual QString unique_date() const = 0;
    virtual QString name() const = 0;
    virtual bool should_run() const = 0;
    virtual void run() = 0;
};

inline migration::~migration() {}

} // ns migrations

OTR_MIGRATION_EXPORT std::vector<QString> run_migrations();
OTR_MIGRATION_EXPORT void mark_config_as_not_needing_migration();