Hello, I have a problem with this updater. I heard it’s supposed to be simple, but at some point, I think I did something wrong.
I already had file checking, downloading, etc., working normally on the new client, but it didn’t read the "mods" folder where I had added new content. This caused the new client not to download the modules from "mods." I started experimenting in various ways, and finally, it stopped downloading/checking for updates altogether.
i have Windows 10, TFS 1.4.2 / OTCv8
On server (ovh): (all --enscrypt) (C:\Users\Administrator\Desktop\UniServerZ\www\updates)

Init.lua in data.zip:
C:\Users\Administrator\Desktop\UniServerZ\www\updates\updater.php
In my client.exe is same init.lua
Downloading etc was already working, but it wasn't downloading "mods" and I messed something up through trial and error. I would like to ask for support
website updates/updater.php
Updater API is running. Please send a POST request with valid JSON data.
When the uptader was working and downloading files, I had the following message:
i had in logs in game "ERROR: Unable to find module 'game_tasks' required by 'game_interface'" < game_task was in 'mods".
game_interface tried to look for it in "modules". I didn't want to move to "modules" because I have several things in "mods"
I already had file checking, downloading, etc., working normally on the new client, but it didn’t read the "mods" folder where I had added new content. This caused the new client not to download the modules from "mods." I started experimenting in various ways, and finally, it stopped downloading/checking for updates altogether.
i have Windows 10, TFS 1.4.2 / OTCv8
On server (ovh): (all --enscrypt) (C:\Users\Administrator\Desktop\UniServerZ\www\updates)

Init.lua in data.zip:
LUA:
-- CONFIG
APP_NAME = "xxxxx" -- important, change it, it's name for config dir and files in appdata
APP_VERSION = 1341 -- client version for updater and login to identify outdated client
DEFAULT_LAYOUT = "retro" -- on android it's forced to "mobile", check code bellow
-- If you don't use updater or other service, set it to updater = ""
Services = {
website = "http://************", -- currently not used
updater = "http:/xxxxx.pl/updates/updater.php",
stats = "",
crash = "http://************/api/crash.php",
feedback = "http://************/api/feedback.php",
status = ""
}
-- Servers accept http login url, websocket login url or ip:port:version
Servers = {
Localhost = "xxxxx:7171:1098" --- ip, port, protocol version
}
--Server = "ws://************:3000/"
--Server = "ws://127.0.0.1:88/"
--USE_NEW_ENERGAME = true -- uses entergamev2 based on websockets instead of entergame
ALLOW_CUSTOM_SERVERS = false -- if true it shows option ANOTHER on server list
g_app.setName("xxxxx")
-- CONFIG END
-- print first terminal message
g_logger.info(os.date("== application started at %b %d %Y %X"))
g_logger.info(g_app.getName() .. ' ' .. g_app.getVersion() .. ' rev ' .. g_app.getBuildRevision() .. ' (' .. g_app.getBuildCommit() .. ') made by ' .. g_app.getAuthor() .. ' built on ' .. g_app.getBuildDate() .. ' for arch ' .. g_app.getBuildArch())
if not g_resources.directoryExists("/data") then
g_logger.fatal("Data dir doesn't exist.")
end
if not g_resources.directoryExists("/modules") then
g_logger.fatal("Modules dir doesn't exist.")
end
-- settings
g_configs.loadSettings("/config.otml")
-- set layout
local settings = g_configs.getSettings()
local layout = DEFAULT_LAYOUT
if g_app.isMobile() then
layout = "mobile"
elseif settings:exists('layout') then
layout = settings:getValue('layout')
end
g_resources.setLayout(layout)
-- load mods
g_modules.discoverModules()
g_modules.ensureModuleLoaded("corelib")
local function loadModules()
-- libraries modules 0-99
g_modules.autoLoadModules(99)
g_modules.ensureModuleLoaded("gamelib")
-- client modules 100-499
g_modules.autoLoadModules(499)
g_modules.ensureModuleLoaded("client")
-- game modules 500-999
g_modules.autoLoadModules(999)
g_modules.ensureModuleLoaded("game_interface")
-- mods 1000-9999
g_modules.autoLoadModules(9999)
end
-- report crash
if type(Services.crash) == 'string' and Services.crash:len() > 4 and g_modules.getModule("crash_reporter") then
g_modules.ensureModuleLoaded("crash_reporter")
end
-- run updater, must use data.zip
if type(Services.updater) == 'string' and Services.updater:len() > 4
and g_resources.isLoadedFromArchive() and g_modules.getModule("updater") then
g_modules.ensureModuleLoaded("updater")
return Updater.init(loadModules)
end
loadModules()
C:\Users\Administrator\Desktop\UniServerZ\www\updates\updater.php
LUA:
<?php
// CONFIG
$files_dir = "C:/Users/Administrator/Desktop/UniServerZ/www/updates"; // Folder z plikami
$files_url = "http://xxxxxx.pl/updates/"; // Publiczny adres URL
$files_and_dirs = array("data.zip"); // Wskazujemy tylko na data.zip
$checksum_file = "checksums.txt";
$checksum_update_interval = 60; // seconds
$binaries = array(
"WIN32-WGL" => "client.exe",
"WIN32-EGL" => "client.exe",
"WIN32-WGL-GCC" => "client.exe",
"WIN32-EGL-GCC" => "client.exe",
"X11-GLX" => "",
"X11-EGL" => "",
"ANDROID-EGL" => "" // Nie obsługujemy Androida
);
// CONFIG END
function sendError($error) {
http_response_code(400);
echo(json_encode(array("error" => $error)));
die();
}
// Handle GET requests for diagnostics
if ($_SERVER['REQUEST_METHOD'] === 'GET') {
echo "Updater API is running. Please send a POST request with valid JSON data.";
exit;
}
// Read input data
$data = json_decode(file_get_contents("php://input"));
if (!$data) {
error_log("Received invalid data: " . file_get_contents("php://input")); // Log input data for debugging
sendError("Invalid input data");
}
// Extract data from input
$version = $data->version ?? 0;
$build = $data->build ?? "";
$os = $data->os ?? "unknown";
$platform = $data->platform ?? "";
$args = $data->args ?? [];
$binary = $binaries[$platform] ?? "";
// Cache management
$cache = null;
$cache_file = sys_get_temp_dir() . DIRECTORY_SEPARATOR . $checksum_file;
if (file_exists($cache_file) && (filemtime($cache_file) + $checksum_update_interval > time())) {
$cache = json_decode(file_get_contents($cache_file), true);
}
if (!$cache) {
// Update cache
$dir = realpath($files_dir);
if (!$dir) {
sendError("Invalid files directory: $files_dir");
}
$rii = new RecursiveIteratorIterator(new RecursiveDirectoryIterator($dir, FilesystemIterator::SKIP_DOTS));
$cache = array();
foreach ($rii as $file) {
if (!$file->isFile()) continue;
$path = str_replace($dir, '', $file->getPathname());
$path = str_replace(DIRECTORY_SEPARATOR, '/', $path);
$cache[$path] = hash_file("crc32b", $file->getPathname());
}
file_put_contents($cache_file . ".tmp", json_encode($cache));
rename($cache_file . ".tmp", $cache_file);
}
// Build response
$ret = array("url" => $files_url, "files" => array(), "keepFiles" => true);
foreach ($cache as $file => $checksum) {
if (in_array($file, $files_and_dirs)) { // Sprawdzamy tylko pliki w files_and_dirs
$ret["files"][$file] = $checksum;
}
}
// Return JSON response
header('Content-Type: application/json');
echo(json_encode($ret, JSON_PRETTY_PRINT));
In my client.exe is same init.lua
Downloading etc was already working, but it wasn't downloading "mods" and I messed something up through trial and error. I would like to ask for support

Post automatically merged:
website updates/updater.php
Updater API is running. Please send a POST request with valid JSON data.
When the uptader was working and downloading files, I had the following message:
i had in logs in game "ERROR: Unable to find module 'game_tasks' required by 'game_interface'" < game_task was in 'mods".
game_interface tried to look for it in "modules". I didn't want to move to "modules" because I have several things in "mods"
Last edited: