#include #include "loader/component_loader.hpp" #include "console.hpp" #include "fastfiles.hpp" #include "dvars.hpp" #include "scheduler.hpp" #include "updater.hpp" #include "version.hpp" #include "game/game.hpp" #include "game/dvars.hpp" #include "game/ui_scripting/execution.hpp" #include #include #include #include #include #include #include #define MASTER "https://master.fed0001.xyz/h1-mod/" #define FILES_PATH "files.json" #define FILES_PATH_DEV "files-dev.json" #define DATA_PATH "data/" #define DATA_PATH_DEV "data-dev/" #define ERR_UPDATE_CHECK_FAIL "Failed to check for updates:\n%s" #define ERR_UPDATE_CHECK_FAIL_BAD_RESPONSE "Bad response" #define ERR_DOWNLOAD_FAIL "Failed to download file %s:\n%s" #define ERR_WRITE_FAIL "Failed to write file " #define BINARY_NAME "h1-mod.exe" namespace updater { namespace { game::dvar_t* cl_auto_update; bool has_tried_update = false; struct status { bool done; bool success; }; struct file_data { std::string name; std::string data; }; struct update_data_t { bool restart_required{}; bool cancelled{}; status check{}; status download{}; std::string error{}; std::string current_file{}; std::vector required_files{}; std::vector garbage_files{}; }; // remove this at some point std::vector old_data_files = { {"./data"}, {"./cdata"}, }; utils::concurrency::container update_data; std::string select(const std::string& main, const std::string& develop) { if (GIT_BRANCH == "develop"s) { return develop; } return main; } std::string load_binary_name() { utils::nt::library self; return self.get_name(); } std::string get_binary_name() { static const auto name = load_binary_name(); return name; } void notify(const std::string& name) { scheduler::once([=]() { ui_scripting::notify(name, {}); }, scheduler::pipeline::lui); } void set_update_check_status(bool done, bool success, const std::string& error = {}) { update_data.access([done, success, error](update_data_t& data_) { data_.check.done = done; data_.check.success = success; data_.error = error; notify("update_check_done"); }); } void set_update_download_status(bool done, bool success, const std::string& error = {}) { update_data.access([done, success, error](update_data_t& data_) { data_.download.done = done; data_.download.success = success; data_.error = error; notify("update_done"); }); } bool check_file(const std::string& name, const std::string& sha) { std::string data; if (get_binary_name() == name) { if (!utils::io::read_file(name, &data)) { return false; } } else { const auto appdata_folder = utils::properties::get_appdata_path(); const auto path = (appdata_folder / name).generic_string(); if (!utils::io::read_file(path, &data)) { return false; } } if (utils::cryptography::sha1::compute(data, true) != sha) { return false; } return true; } std::string get_time_str() { return utils::string::va("%i", uint32_t(time(nullptr))); } std::optional download_file(const std::string& name) { return utils::http::get_data(MASTER + select(DATA_PATH, DATA_PATH_DEV) + name + "?" + get_time_str()); } bool has_old_data_files() { bool has = false; for (const auto& file : old_data_files) { if (utils::io::directory_exists(file)) { has = true; } } return has; } void delete_old_data_files() { for (const auto& file : old_data_files) { std::filesystem::remove_all(file); } } bool is_update_cancelled() { return update_data.access([](update_data_t& data_) { return data_.cancelled; }); } bool write_file(const std::string& name, const std::string& data) { if (get_binary_name() == name && utils::io::file_exists(name) && !utils::io::move_file(name, name + ".old")) { return false; } if (get_binary_name() == name) { return utils::io::write_file(name, data); } else { const auto appdata_folder = utils::properties::get_appdata_path(); const auto path = (appdata_folder / name).generic_string(); return utils::io::write_file(path, data); } } void delete_old_file() { utils::io::remove_file(get_binary_name() + ".old"); } void reset_data() { update_data.access([](update_data_t& data_) { data_ = {}; }); } std::vector find_garbage_files(const std::vector& update_files) { std::vector garbage_files{}; const auto appdata_folder = utils::properties::get_appdata_path(); const auto path = (appdata_folder / CLIENT_DATA_FOLDER).generic_string(); if (!utils::io::directory_exists(path)) { return {}; } const auto current_files = utils::io::list_files_recursively(path); for (const auto& file : current_files) { bool found = false; for (const auto& update_file : update_files) { const auto update_file_ = (appdata_folder / update_file).generic_string(); const auto path_a = std::filesystem::path(file); const auto path_b = std::filesystem::path(update_file_); const auto is_directory = utils::io::directory_exists(file); const auto compare = path_a.compare(path_b); if ((is_directory && compare == -1) || compare == 0) { found = true; break; } } if (!found) { console::debug("[Updater] Found extra file %s\n", file.data()); if (file.ends_with(".ff")) { update_data.access([](update_data_t& data_) { data_.restart_required = true; }); } garbage_files.push_back(file); } } return garbage_files; } std::string get_mode_flag() { if (game::environment::is_mp()) { return "-multiplayer"; } if (game::environment::is_sp()) { return "-singleplayer"; } return {}; } std::string curl_error(CURLcode code) { const auto str_error = curl_easy_strerror(code); return utils::string::va("%s (%i)", str_error, code); } } void relaunch() { const auto mode = game::environment::is_mp() ? "-multiplayer" : "-singleplayer"; utils::nt::relaunch_self(mode); utils::nt::terminate(); } void set_has_tried_update(bool tried) { has_tried_update = tried; } bool get_has_tried_update() { return has_tried_update; } bool auto_updates_enabled() { return cl_auto_update->current.enabled; } bool is_update_check_done() { return update_data.access([](update_data_t& data_) { return data_.check.done; }); } bool is_update_download_done() { return update_data.access([](update_data_t& data_) { return data_.download.done; }); } bool get_update_check_status() { return update_data.access([](update_data_t& data_) { return data_.check.success; }); } bool get_update_download_status() { return update_data.access([](update_data_t& data_) { return data_.download.success; }); } bool is_update_available() { return update_data.access([](update_data_t& data_) { return data_.required_files.size() > 0 || data_.garbage_files.size() > 0 || has_old_data_files(); }); } bool is_restart_required() { return update_data.access([](update_data_t& data_) { return data_.restart_required; }); } std::string get_last_error() { return update_data.access([](update_data_t& data_) { return data_.error; }); } std::string get_current_file() { return update_data.access([](update_data_t& data_) { return data_.current_file; }); } void cancel_update() { console::debug("[Updater] Cancelling update\n"); return update_data.access([](update_data_t& data_) { data_.cancelled = true; }); } void start_update_check() { cancel_update(); reset_data(); console::debug("[Updater] starting update check\n"); scheduler::once([]() { const auto files_data = utils::http::get_data(MASTER + select(FILES_PATH, FILES_PATH_DEV) + "?" + get_time_str()); if (is_update_cancelled()) { reset_data(); return; } if (!files_data.has_value()) { set_update_check_status(true, false, utils::string::va(ERR_UPDATE_CHECK_FAIL, "Unknown error")); return; } const auto& value = files_data.value(); if (value.code != CURLE_OK) { const auto error = curl_error(value.code); set_update_check_status(true, false, utils::string::va(ERR_UPDATE_CHECK_FAIL, error.data())); return; } rapidjson::Document j; j.Parse(value.buffer.data()); if (!j.IsArray()) { set_update_check_status(true, false, ERR_UPDATE_CHECK_FAIL_BAD_RESPONSE); return; } std::vector required_files; std::vector update_files; const auto files = j.GetArray(); for (const auto& file : files) { if (!file.IsArray() || file.Size() != 3 || !file[0].IsString() || !file[2].IsString()) { continue; } const auto name = file[0].GetString(); const auto sha = file[2].GetString(); update_files.push_back(name); if (!check_file(name, sha)) { if (get_binary_name() == name) { update_data.access([](update_data_t& data_) { data_.restart_required = true; }); } std::string name_ = name; if (name_.ends_with(".ff")) { update_data.access([](update_data_t& data_) { data_.restart_required = true; }); } console::debug("[Updater] need file %s\n", name); required_files.push_back(name); } } const auto garbage_files = find_garbage_files(update_files); update_data.access([&](update_data_t& data_) { data_.check.done = true; data_.check.success = true; data_.required_files = required_files; data_.garbage_files = garbage_files; }); notify("update_check_done"); }, scheduler::pipeline::async); } void start_update_download() { console::debug("[Updater] starting update download\n"); if (!is_update_check_done() || !get_update_check_status() || is_update_cancelled()) { return; } delete_old_data_files(); const auto garbage_files = update_data.access>([](update_data_t& data_) { return data_.garbage_files; }); update_data.access([](update_data_t& data_) { if (data_.restart_required) { fastfiles::close_fastfile_handles(); } }); for (const auto& file : garbage_files) { try { std::filesystem::remove_all(file); } catch (...) { console::error("Failed to delete %s\n", file.data()); } } scheduler::once([]() { const auto required_files = update_data.access>([](update_data_t& data_) { return data_.required_files; }); std::vector downloads; for (const auto& file : required_files) { update_data.access([file](update_data_t& data_) { data_.current_file = file; }); console::debug("[Updater] downloading file %s\n", file.data()); const auto data = download_file(file); if (is_update_cancelled()) { reset_data(); return; } if (!data.has_value()) { set_update_download_status(true, false, utils::string::va(ERR_DOWNLOAD_FAIL, file.data(), "Unknown error")); return; } const auto& value = data.value(); if (value.code != CURLE_OK) { const auto error = curl_error(value.code); set_update_download_status(true, false, utils::string::va(ERR_DOWNLOAD_FAIL, file.data(), error.data())); return; } downloads.push_back({file, value.buffer}); } for (const auto& download : downloads) { if (!write_file(download.name, download.data)) { set_update_download_status(true, false, ERR_WRITE_FAIL + download.name); return; } } set_update_download_status(true, true); }, scheduler::pipeline::async); } class component final : public component_interface { public: void post_unpack() override { delete_old_file(); cl_auto_update = dvars::register_bool("cg_auto_update", true, game::DVAR_FLAG_SAVED, "Automatically check for updates"); } }; } REGISTER_COMPONENT(updater::component)