#include "pluginmanager.h"

#include <regex>
#include "plugins.h"
#include "config/profile.h"
#include "core/convar/concommand.h"

namespace fs = std::filesystem;

PluginManager* g_pPluginManager;

const std::vector<Plugin>& PluginManager::GetLoadedPlugins() const
{
	return this->plugins;
}

const std::optional<Plugin> PluginManager::GetPlugin(HMODULE handle) const
{
	for (const Plugin& plugin : GetLoadedPlugins())
		if (plugin.m_handle == handle)
			return plugin;
	return std::nullopt;
}

void PluginManager::LoadPlugin(fs::path path, bool reloaded)
{
	Plugin plugin = Plugin(path.string());

	if (!plugin.IsValid())
	{
		NS::log::PLUGINSYS->warn("Unloading invalid plugin '{}'", path.string());
		plugin.Unload();
		return;
	}

	plugins.push_back(plugin);
	plugin.Init(reloaded);
}

inline void FindPlugins(fs::path pluginPath, std::vector<fs::path>& paths)
{
	// ensure dirs exist
	if (!fs::exists(pluginPath) || !fs::is_directory(pluginPath))
	{
		return;
	}

	for (const fs::directory_entry& entry : fs::directory_iterator(pluginPath))
	{
		if (fs::is_regular_file(entry) && entry.path().extension() == ".dll")
			paths.emplace_back(entry.path());
	}
}

bool PluginManager::LoadPlugins(bool reloaded)
{
	if (strstr(GetCommandLineA(), "-noplugins") != NULL)
	{
		NS::log::PLUGINSYS->warn("-noplugins detected; skipping loading plugins");
		return false;
	}

	fs::create_directories(GetThunderstoreModFolderPath());

	std::vector<fs::path> paths;

	pluginPath = GetNorthstarPrefix() + "\\plugins";

	fs::path libPath = fs::absolute(pluginPath + "\\lib");
	if (fs::exists(libPath) && fs::is_directory(libPath))
		AddDllDirectory(libPath.wstring().c_str());

	FindPlugins(pluginPath, paths);

	// Special case for Thunderstore mods dir
	std::filesystem::directory_iterator thunderstoreModsDir = fs::directory_iterator(GetThunderstoreModFolderPath());
	// Set up regex for `AUTHOR-MOD-VERSION` pattern
	std::regex pattern(R"(.*\\([a-zA-Z0-9_]+)-([a-zA-Z0-9_]+)-(\d+\.\d+\.\d+))");
	for (fs::directory_entry dir : thunderstoreModsDir)
	{
		fs::path pluginsDir = dir.path() / "plugins";
		// Use regex to match `AUTHOR-MOD-VERSION` pattern
		if (!std::regex_match(dir.path().string(), pattern))
		{
			spdlog::warn("The following directory did not match 'AUTHOR-MOD-VERSION': {}", dir.path().string());
			continue; // skip loading package that doesn't match
		}

		fs::path libDir = fs::absolute(pluginsDir / "lib");
		if (fs::exists(libDir) && fs::is_directory(libDir))
			AddDllDirectory(libDir.wstring().c_str());

		FindPlugins(pluginsDir, paths);
	}

	if (paths.empty())
	{
		NS::log::PLUGINSYS->warn("Could not find any plugins. Skipped loading plugins");
		return false;
	}

	for (fs::path path : paths)
	{
		LoadPlugin(path, reloaded);
	}

	InformAllPluginsInitialized();

	return true;
}

void PluginManager::ReloadPlugins()
{
	for (const Plugin& plugin : this->GetLoadedPlugins())
	{
		plugin.Unload();
	}

	this->plugins.clear();
	this->LoadPlugins(true);
}

void PluginManager::RemovePlugin(HMODULE handle)
{
	for (size_t i = 0; i < plugins.size(); i++)
	{
		Plugin* plugin = &plugins[i];
		if (plugin->m_handle == handle)
		{
			plugins.erase(plugins.begin() + i);
			return;
		}
	}
}

void PluginManager::InformAllPluginsInitialized() const
{
	for (const Plugin& plugin : GetLoadedPlugins())
	{
		plugin.Finalize();
	}
}

void PluginManager::InformSqvmCreated(CSquirrelVM* sqvm) const
{
	for (const Plugin& plugin : GetLoadedPlugins())
	{
		plugin.OnSqvmCreated(sqvm);
	}
}

void PluginManager::InformSqvmDestroying(CSquirrelVM* sqvm) const
{
	for (const Plugin& plugin : GetLoadedPlugins())
	{
		plugin.OnSqvmDestroying(sqvm);
	}
}

void PluginManager::InformDllLoad(HMODULE module, fs::path path) const
{
	std::string fn = path.filename().string(); // without this the string gets freed immediately lmao
	const char* filename = fn.c_str();
	for (const Plugin& plugin : GetLoadedPlugins())
	{
		plugin.OnLibraryLoaded(module, filename);
	}
}

void PluginManager::RunFrame() const
{
	for (const Plugin& plugin : GetLoadedPlugins())
	{
		plugin.RunFrame();
	}
}

void ConCommand_reload_plugins(const CCommand& args)
{
	g_pPluginManager->ReloadPlugins();
}

ON_DLL_LOAD_RELIESON("engine.dll", PluginManager, ConCommand, (CModule module))
{
	RegisterConCommand("reload_plugins", ConCommand_reload_plugins, "reloads plugins", FCVAR_NONE);
}