#include "server.hpp" #define CRYPTOPP_ENABLE_NAMESPACE_WEAK 1 #include #include #include #include #include #include #include namespace fs = std::filesystem; UpdateServer::UpdateServer() { update_files(); Post("/files", [this](const httplib::Request& req, httplib::Response& res) { Json::Reader reader; Json::Value j; if (!reader.parse(req.body, j)) { SPDLOG_WARN("Invalid JSON request: {}", req.body); res.status = 400; return; } if (!j.isObject() || !j.isMember("arch") || !j.isMember("file")) { SPDLOG_WARN("Invalid JSON request: {}", req.body); res.status = 400; return; } auto arch = j["arch"].asString(); auto file_name = j["file"].asString(); SPDLOG_INFO("{} requested download {}/{}", req.remote_addr, arch, file_name); if (!is_file_exists(arch, file_name)) { SPDLOG_WARN("File {}/{} not exists", arch, file_name); res.status = 400; return; } auto& file_data = m_files[arch][file_name].data; res.set_content(file_data.data(), file_data.size(), "application/octet-stream"); res.status = 200; }); Post("/get_hash", [this](const httplib::Request& req, httplib::Response& res) { Json::Reader reader; Json::Value j; if (!reader.parse(req.body, j)) { SPDLOG_WARN("Invalid JSON request: {}", req.body); res.status = 400; return; } if (!j.isObject() || !j.isMember("arch") || !j.isMember("file")) { SPDLOG_WARN("Invalid JSON request: {}", req.body); res.status = 400; return; } auto arch = j["arch"].asString(); auto file_name = j["file"].asString(); SPDLOG_INFO("{} requested {}/{} hash", req.remote_addr, arch, file_name); if (!is_file_exists(arch, file_name)) { SPDLOG_WARN("File {}/{} not exists", arch, file_name); res.status = 400; return; } auto& hash = m_files[arch][file_name].hash; res.set_content(hash, "text/plain"); res.status = 200; }); Post("/reload", [this](const httplib::Request& req, httplib::Response& res) { SPDLOG_INFO("{} requested reload", req.remote_addr); if (req.body == m_reload_key) { update_files(); res.status = 200; return; } SPDLOG_WARN("Invalid reload key: {}", req.body); res.status = 400; }); } UpdateServer& UpdateServer::start(int port, const std::string& reload_key) { m_reload_key = reload_key; m_server_thread = std::thread([this, port] { SPDLOG_INFO("Update server on port {}, reload key: {}", port, m_reload_key); listen("0.0.0.0", port); }); return *this; } void UpdateServer::join() { m_server_thread.join(); } UpdateServer& UpdateServer::Instance() { static UpdateServer instance; return instance; } void UpdateServer::update_files() { m_files.clear(); fs::path assets_path = fs::current_path() / "assets"; if (!exists(assets_path)) { return; } for (auto& arch : fs::directory_iterator(assets_path)) { if (!fs::is_directory(arch)) { continue; } for (auto& file : fs::directory_iterator(arch)) { if (!fs::is_regular_file(file)) { continue; } auto arch_name = arch.path().filename().string(); auto file_name = file.path().filename().string(); auto size = file.file_size(); FileData data(size); std::ifstream f(file.path().string(), std::ios::binary); f.read(data.data(), size); auto md5 = hash_file(data); SPDLOG_INFO("Load {}/{}, hash: {}", arch_name, file_name, md5); m_files[arch_name][file_name] = { .data = std::move(data), .hash = std::move(md5) }; } } } UpdateServer::FileHash UpdateServer::hash_file(const FileData& data) { FileHash hash; CryptoPP::Weak::MD5 md5; CryptoPP::StringSource s( reinterpret_cast(data.data()), data.size(), true, new CryptoPP::HashFilter(md5, new CryptoPP::HexEncoder( new CryptoPP::StringSink(hash) ) ) ); return hash; } bool UpdateServer::is_file_exists(const std::string& arch, const std::string& file) const { if (!m_files.contains(arch)) { return false; } auto& files = m_files.at(arch); return files.contains(file); }