VoxelEngine/src/io/engine_paths.cpp
2025-07-18 22:57:38 +03:00

356 lines
11 KiB
C++

#include "engine_paths.hpp"
#include <algorithm>
#include <array>
#include <sstream>
#include <stack>
#include "typedefs.hpp"
#include "util/stringutil.hpp"
#include <utility>
#include "io/devices/StdfsDevice.hpp"
#include "io/devices/ZipFileDevice.hpp"
#include "world/files/WorldFiles.hpp"
#include "debug/Logger.hpp"
#include <chrono>
#include "maths/util.hpp"
template<int n>
static std::string generate_random_base64() {
auto now = std::chrono::high_resolution_clock::now();
auto seed = now.time_since_epoch().count();
util::PseudoRandom random(seed); // fixme: replace with safe random
ubyte bytes[n];
random.rand(bytes, n);
return util::base64_urlsafe_encode(bytes, n);
}
namespace fs = std::filesystem;
static debug::Logger logger("engine-paths");
static inline io::path SCREENSHOTS_FOLDER = "user:screenshots";
static inline io::path CONTENT_FOLDER = "user:content";
static inline io::path WORLDS_FOLDER = "user:worlds";
void EnginePaths::prepare() {
io::set_device("res", std::make_shared<io::StdfsDevice>(resourcesFolder, false));
io::set_device("user", std::make_shared<io::StdfsDevice>(userFilesFolder));
if (!io::is_directory("res:")) {
throw std::runtime_error(
resourcesFolder.string() + " is not a directory"
);
}
logger.info() << "resources folder: " << fs::canonical(resourcesFolder).u8string();
logger.info() << "user files folder: " << fs::canonical(userFilesFolder).u8string();
logger.info() << "project folder: " << fs::canonical(projectFolder).u8string();
if (!io::is_directory(CONTENT_FOLDER)) {
io::create_directories(CONTENT_FOLDER);
}
io::create_subdevice("core", "res", "");
io::create_subdevice("export", "user", "export");
io::create_subdevice("config", "user", "config");
}
const std::filesystem::path& EnginePaths::getUserFilesFolder() const {
return userFilesFolder;
}
const std::filesystem::path& EnginePaths::getResourcesFolder() const {
return resourcesFolder;
}
io::path EnginePaths::getNewScreenshotFile(const std::string& ext) {
auto folder = SCREENSHOTS_FOLDER;
if (!io::is_directory(folder)) {
io::create_directories(folder);
}
auto t = std::time(nullptr);
auto tm = *std::localtime(&t);
const char* format = "%Y-%m-%d_%H-%M-%S";
std::stringstream ss;
ss << std::put_time(&tm, format);
std::string datetimestr = ss.str();
auto file = folder / ("screenshot-" + datetimestr + "." + ext);
uint index = 0;
while (io::exists(file)) {
file = folder / ("screenshot-" + datetimestr + "-" +
std::to_string(index) + "." + ext);
index++;
}
return file;
}
io::path EnginePaths::getWorldsFolder() const {
return WORLDS_FOLDER;
}
io::path EnginePaths::getCurrentWorldFolder() {
return currentWorldFolder;
}
io::path EnginePaths::getWorldFolderByName(const std::string& name) {
return getWorldsFolder() / name;
}
std::vector<io::path> EnginePaths::scanForWorlds() const {
std::vector<io::path> folders;
auto folder = getWorldsFolder();
if (!io::is_directory(folder)) return folders;
for (const auto& worldFolder : io::directory_iterator(folder)) {
if (!io::is_directory(worldFolder)) {
continue;
}
auto worldFile = worldFolder / WorldFiles::WORLD_FILE;
if (!io::is_regular_file(worldFile)) {
continue;
}
folders.push_back(worldFolder);
}
std::sort(
folders.begin(),
folders.end(),
[](io::path a, io::path b) {
a = a / WorldFiles::WORLD_FILE;
b = b / WorldFiles::WORLD_FILE;
return fs::last_write_time(io::resolve(a)) >
fs::last_write_time(io::resolve(b));
}
);
return folders;
}
void EnginePaths::setUserFilesFolder(std::filesystem::path folder) {
this->userFilesFolder = std::move(folder);
}
void EnginePaths::setResourcesFolder(std::filesystem::path folder) {
this->resourcesFolder = std::move(folder);
}
void EnginePaths::setScriptFolder(std::filesystem::path folder) {
io::set_device("script", std::make_shared<io::StdfsDevice>(folder));
this->scriptFolder = std::move(folder);
}
void EnginePaths::setProjectFolder(std::filesystem::path folder) {
io::set_device("project", std::make_shared<io::StdfsDevice>(folder));
this->projectFolder = std::move(folder);
}
void EnginePaths::setCurrentWorldFolder(io::path folder) {
if (folder.empty()) {
io::remove_device("world");
} else {
io::create_subdevice("world", "user", folder);
}
this->currentWorldFolder = std::move(folder);
}
std::string EnginePaths::mount(const io::path& file) {
if (file.extension() == ".zip") {
auto stream = io::read(file);
auto device = std::make_unique<io::ZipFileDevice>(
std::move(stream), [file]() { return io::read(file); }
);
std::string name;
do {
name = std::string("M.") + generate_random_base64<6>();
} while (std::find(mounted.begin(), mounted.end(), name) != mounted.end());
io::set_device(name, std::move(device));
mounted.push_back(name);
return name;
}
throw std::runtime_error("unable to mount " + file.string());
}
void EnginePaths::unmount(const std::string& name) {
const auto& found = std::find(mounted.begin(), mounted.end(), name);
if (found == mounted.end()) {
throw std::runtime_error(name + " is not mounted");
}
io::remove_device(name);
mounted.erase(found);
}
std::string EnginePaths::createWriteableDevice(const std::string& name) {
const auto& found = writeables.find(name);
if (found != writeables.end()) {
return found->second;
}
io::path folder;
for (const auto& point : entryPoints) {
if (point.name == name) {
folder = point.path;
break;
}
}
if (name == "core") {
folder = "res:";
}
if (folder.emptyOrInvalid()) {
throw std::runtime_error("pack not found");
}
auto entryPoint = std::string("W.") + generate_random_base64<6>();
io::create_subdevice(entryPoint, folder.entryPoint(), folder.pathPart());
writeables[name] = entryPoint;
return entryPoint;
}
void EnginePaths::cleanup() {
// Remove previous content entry-points
for (const auto& [id, _] : entryPoints) {
io::remove_device(id);
}
for (const auto& [_, entryPoint] : writeables) {
io::remove_device(entryPoint);
}
for (const auto& entryPoint : mounted) {
io::remove_device(entryPoint);
}
entryPoints.clear();
writeables.clear();
}
void EnginePaths::setEntryPoints(std::vector<PathsRoot> entryPoints) {
cleanup();
// Create sub-devices
for (const auto& point : entryPoints) {
auto parent = point.path.entryPoint();
io::create_subdevice(point.name, parent, point.path);
}
this->entryPoints = std::move(entryPoints);
}
std::tuple<std::string, std::string> EnginePaths::parsePath(std::string_view path) {
size_t separator = path.find(':');
if (separator == std::string::npos) {
return {"", std::string(path)};
}
auto prefix = std::string(path.substr(0, separator));
auto filename = std::string(path.substr(separator + 1));
return {prefix, filename};
}
ResPaths::ResPaths(std::vector<PathsRoot> roots)
: roots(std::move(roots)) {
}
io::path ResPaths::find(const std::string& filename) const {
for (int i = roots.size() - 1; i >= 0; i--) {
auto& root = roots[i];
auto file = root.path / filename;
if (io::exists(file)) {
return file;
}
}
return io::path("res:") / filename;
}
std::string ResPaths::findRaw(const std::string& filename) const {
for (int i = roots.size() - 1; i >= 0; i--) {
auto& root = roots[i];
if (io::exists(root.path / filename)) {
return root.name + ":" + filename;
}
}
throw std::runtime_error("could not to find file " + util::quote(filename));
}
std::vector<std::string> ResPaths::listdirRaw(const std::string& folderName) const {
std::vector<std::string> entries;
for (int i = roots.size() - 1; i >= 0; i--) {
auto& root = roots[i];
auto folder = root.path / folderName;
if (!io::is_directory(folder)) continue;
for (const auto& file : io::directory_iterator(folder)) {
entries.emplace_back(root.name + ":" + folderName + "/" + file.name());
}
}
return entries;
}
std::vector<io::path> ResPaths::listdir(
const std::string& folderName
) const {
std::vector<io::path> entries;
for (int i = roots.size() - 1; i >= 0; i--) {
auto& root = roots[i];
io::path folder = root.path / folderName;
if (!io::is_directory(folder)) continue;
for (const auto& entry : io::directory_iterator(folder)) {
entries.push_back(entry);
}
}
return entries;
}
dv::value ResPaths::readCombinedList(const std::string& filename) const {
dv::value list = dv::list();
for (const auto& root : roots) {
auto path = root.path / filename;
if (!io::exists(path)) {
continue;
}
try {
auto value = io::read_object(path);
if (!value.isList()) {
logger.warning() << "reading combined list " << root.name << ":"
<< filename << " is not a list (skipped)";
continue;
}
for (const auto& elem : value) {
list.add(elem);
}
} catch (const std::runtime_error& err) {
logger.warning() << "reading combined list " << root.name << ":"
<< filename << ": " << err.what();
}
}
return list;
}
dv::value ResPaths::readCombinedObject(const std::string& filename, bool deep) const {
dv::value object = dv::object();
for (const auto& root : roots) {
auto path = root.path / filename;
if (!io::exists(path)) {
continue;
}
try {
auto value = io::read_object(path);
if (!value.isObject()) {
logger.warning()
<< "reading combined object " << root.name << ": "
<< filename << " is not an object (skipped)";
}
object.merge(std::move(value), deep);
} catch (const std::runtime_error& err) {
logger.warning() << "reading combined object " << root.name << ":"
<< filename << ": " << err.what();
}
}
return object;
}
std::vector<io::path> ResPaths::collectRoots() {
std::vector<io::path> collected;
collected.reserve(roots.size());
for (const auto& root : roots) {
collected.emplace_back(root.path);
}
return collected;
}