#include <filesystem>
#include <stdexcept>
#include <whereami.h>

#include "Asset.h"
#include "api/Config.h"

using namespace crepe;
using namespace std;

Asset::Asset(const string & src) : src(find_asset(src)) { }
Asset::Asset(const char * src) : src(find_asset(src)) { }

const string & Asset::get_path() const noexcept { return this->src; }

string Asset::find_asset(const string & src) const {
	auto & cfg = Config::get_instance();
	auto & root_pattern = cfg.asset.root_pattern;

	// if root_pattern is empty, find_asset must return all paths as-is
	if (root_pattern.empty()) return src;

	// absolute paths do not need to be resolved, only canonicalized
	filesystem::path path = src;
	if (path.is_absolute())
		return filesystem::canonical(path);

	// find directory matching root_pattern
	filesystem::path root = this->whereami();
	while (1) {
		if (filesystem::exists(root / root_pattern))
			break;
		if (!root.has_parent_path())
			throw runtime_error(format("Asset: Cannot find root pattern ({})", root_pattern));
		root = root.parent_path();
	}

	// join path to root (base directory) and canonicalize
	return filesystem::canonical(root / path);
}

string Asset::whereami() const noexcept {
	string path;
	size_t path_length = wai_getExecutablePath(NULL, 0, NULL);
	path.resize(path_length + 1); // wai writes null byte
	wai_getExecutablePath(path.data(), path_length, NULL);
	path.resize(path_length);
	return path;
}

bool Asset::operator==(const Asset & other) const noexcept {
	return this->src == other.src;
}

size_t std::hash<const Asset>::operator()(const Asset & asset) const noexcept {
	return std::hash<string>{}(asset.get_path());
};