• There is NO official Otland's Discord server and NO official Otland's server list. The Otland's Staff does not manage any Discord server or server list. Moderators or administrator of any Discord server or server lists have NO connection to the Otland's Staff. Do not get scammed!

OTCv8 uptader Windows

Rycina

Member
Joined
Nov 7, 2024
Messages
47
Reaction score
6
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)

up1.webp

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 :D
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:
"$files_and_dirs <span>=</span> <span>array</span><span>(</span><span>"data.zip"</span><span>)</span><span>;</span> <span>//</span> Wskazujemy tylko na data<span>.</span>zip"

This reads only the data.zip, although I find it easier to dont pack it up and leave the files and folders how it is. Sometimes it wont really work how you want it.

I have the following, so it includes all folders etc.

"$files_and_dirs = array("init.lua", "data", "modules", "layouts", "mods");"

------------
Next to that, thank @Michael Orsino for that, Windows uses \\ instead of /
So your "$files_dir <span>=</span> <span>"C:/Users/Administrator/Desktop/UniServerZ/www/updates"</span><span>;</span> <span>//</span> Folder z plikami
"

Must be:

$files_dir <span>=</span> <span>"C:\\Users\\Administrator\\Desktop\\UniServerZ\\www\\updates"</span><span>;</span> <span>//</span> Folder z plikami
 
"$files_and_dirs <span>=</span> <span>array</span><span>(</span><span>"data.zip"</span><span>)</span><span>;</span> <span>//</span> Wskazujemy tylko na data<span>.</span>zip"

This reads only the data.zip, although I find it easier to dont pack it up and leave the files and folders how it is. Sometimes it wont really work how you want it.

I have the following, so it includes all folders etc.

"$files_and_dirs = array("init.lua", "data", "modules", "layouts", "mods");"

------------
Next to that, thank @Michael Orsino for that, Windows uses \\ instead of /
So your "$files_dir <span>=</span> <span>"C:/Users/Administrator/Desktop/UniServerZ/www/updates"</span><span>;</span> <span>//</span> Folder z plikami
"

Must be:

$files_dir <span>=</span> <span>"C:\\Users\\Administrator\\Desktop\\UniServerZ\\www\\updates"</span><span>;</span> <span>//</span> Folder z plikami
Thank you, but that's probably not it. When I run clienta.exe, I don't even check for updates like before. If I delete something from the main clienta.exe folder, it does not download the fileacc.webp
uptader
$files_dir = "C:\\Users\\Administrator\\Desktop\\UniServerZ\\www\\updates";
$files_and_dirs = array("init.lua", "data", "modules", "layouts", "mods");

doesnt work.

I think I didn't have a path with "\\" before when the uptader worked (but mods didn't find it)



in init.lua


LUA:
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()

This doesn't force it to be data.zip ?
 
Last edited:
up
Post automatically merged:

I managed to run the updater and I see that it is downloading "mods", but at the end of the download I get an error that it must be "zip" even though it downloaded everything
acc2.webp

acc.webp

acc1.webp

Init.lua in client folder:
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 below

-- Services configuration
Services = {
  website = "http://xxxxx.pl", -- not used for now
  updater = "http://xxxxx.pl/updates/updater.php",
  stats = "",
  crash = "",
  feedback = "",
  status = ""
}

-- Servers accept http login url, websocket login url or ip:port:version
Servers = {
  Localhost = "xxxxx.pl:7171:1098"  -- ip, port, protocol version
}

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())

-- Ensure essential directories exist
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

-- Initialize updater if updater URL is set
if type(Services.updater) == 'string' and Services.updater:len() > 4
  and g_modules.getModule("updater") then
  g_modules.ensureModuleLoaded("updater")
  return Updater.init(loadModules)
end

-- Load modules if updater is not used
loadModules()


updater on server:

LUA:
<?php

// CONFIG

$files_dir = "C:/Users/Administrator/Desktop/UniServerZ/www/updates";

$files_url = "http://xxxxx.pl/updates/";

$files_and_dirs = array("init.lua", "data", "modules", "layouts", "mods");

$checksum_file = "checksums.txt";

$checksum_update_interval = 60; // seconds

$binaries = array(

    "WIN32-WGL" => "xxxxxClient.zip",

    "WIN32-EGL" => "xxxxxClient.zip",

    "WIN32-WGL-GCC" => "xxxxxClient.zip",

    "WIN32-EGL-GCC" => "xxxxxClient.zip",

    "X11-GLX" => "xxxxxClient.zip",

    "X11-EGL" => "xxxxxClient.zip",

    "ANDROID-EGL" => "",

    "ANDROID64-EGL" => ""

);

// CONFIG END



function sendError($error) {

    echo json_encode(array("error" => $error));

    die();

}



// Read input data

$data = json_decode(file_get_contents("php://input"));

if (!$data) {

    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 if it does not exist or is outdated

    $dir = realpath($files_dir);

    if (!$dir) {

        sendError("Invalid files directory: $files_dir");

    }



    $rii = new RecursiveIteratorIterator(new RecursiveDirectoryIterator($dir));

    $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());

    }



    // Save cache to a temporary file and rename atomically

    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" => false);



foreach ($cache as $file => $checksum) {

    $base = trim(explode("/", ltrim($file, "/"))[0]);

    if (in_array($base, $files_and_dirs)) {

        $ret["files"][$file] = $checksum;

    }



    if ($base === $binary && !empty($binary)) {

        $ret["binary"] = array("file" => $file, "checksum" => $checksum);

    }

}



// Return JSON response

echo json_encode($ret, JSON_PRETTY_PRINT);

?>


Windows10
 
Last edited:
Did you encrypt the data.zip before running the client?
Did you place the files in the files folder in "C:\Users\Administrator\Desktop\UniServerZ\www\updates"?
 
Did you encrypt the data.zip before running the client?
Did you place the files in the files folder in "C:\Users\Administrator\Desktop\UniServerZ\www\updates"?
As I wrote, I am trying to ensure that the uptader is not in .zip but in folders as on the screen
 
Back
Top