#!/bin/sh # ============================================================ # PHP-Debug-Modus konfigurieren (per Umgebungsvariable) # ============================================================ PHP_DISPLAY_ERRORS="${PHP_DISPLAY_ERRORS:-false}" PHP_INI_DIR=$(find /etc/php -type d -name "conf.d" -path "*/apache2/*" | head -1) if [ "$PHP_DISPLAY_ERRORS" = "true" ]; then echo "==> PHP-Debug-Modus AKTIVIERT (PHP_DISPLAY_ERRORS=true)" cat > "$PHP_INI_DIR/99-adminserv-debug.ini" < PHP-Debug-Modus deaktiviert" cat > "$PHP_INI_DIR/99-adminserv-debug.ini" < Erster Start erkannt: Kopiere AdminServ-Dateien ins Volume..." cp -a "$DEFAULT_CONTROLPANEL"/* "$ADMINSERV_DIR/" chmod -R 777 "$ADMINSERV_DIR/logs/" chmod 666 "$ADMINSERV_DIR/config/adminlevel.cfg.php" chmod 666 "$ADMINSERV_DIR/config/servers.cfg.php" chmod 666 "$ADMINSERV_DIR/config/adminserv.cfg.php" chown -R www-data:www-data "$ADMINSERV_DIR/" # AdminServ-Server-Eintrag automatisch konfigurieren XMLRPC_PORT="${SERVER_XMLRPC_PORT:-5000}" # Servernamen fuer PHP-Single-Quotes escapen SAFE_NAME=$(printf '%s' "${SERVER_NAME:-Trackmania Server}" | sed "s/'/\\\\'/g") # ds_pw: Passwort fuer DisplayServ (Serverstatusanzeige auf der Login-Seite) DS_PW=$(printf '%s' "${SERVER_USER_PASSWORD:-User}" | sed "s/'/\\\\'/g") cat > "$ADMINSERV_DIR/config/servers.cfg.php" < array( 'address' => '127.0.0.1', 'port' => ${XMLRPC_PORT}, 'mapsbasepath' => '', 'matchsettings' => 'MatchSettings/', 'adminlevel' => array('SuperAdmin' => 'all', 'Admin' => 'all', 'User' => 'all'), 'ds_pw' => '${DS_PW}' ), ); } ?> EOPHP chmod 666 "$ADMINSERV_DIR/config/servers.cfg.php" chown www-data:www-data "$ADMINSERV_DIR/config/servers.cfg.php" echo " AdminServ-Server-Eintrag automatisch konfiguriert (Port: ${XMLRPC_PORT})." echo " AdminServ-Dateien erfolgreich kopiert." # ============================================================ # RemoteCP: Automatische Konfiguration # ============================================================ REMOTECP_DIR="$ADMINSERV_DIR/remotecp" if [ -d "$REMOTECP_DIR" ]; then echo "==> Konfiguriere RemoteCP..." # DB-Konfiguration aus Umgebungsvariablen REMOTECP_DB_HOST="${REMOTECP_DB_HOST:-}" REMOTECP_DB_NAME="${REMOTECP_DB_NAME:-remotecp}" REMOTECP_DB_USER="${REMOTECP_DB_USER:-remotecp}" REMOTECP_DB_PASSWORD="${REMOTECP_DB_PASSWORD:-}" if [ -n "$REMOTECP_DB_HOST" ] && [ -n "$REMOTECP_DB_PASSWORD" ]; then DB_ENABLED="true" DB_DSN="mysql:dbname=${REMOTECP_DB_NAME};host=${REMOTECP_DB_HOST}" else DB_ENABLED="false" DB_DSN="mysql:dbname=remotecp;host=localhost" echo " HINWEIS: Keine DB-Zugangsdaten gesetzt (REMOTECP_DB_HOST/REMOTECP_DB_PASSWORD)." echo " RemoteCP wird ohne Datenbank konfiguriert. Manuelle Einrichtung moeglich." fi # servers.xml: Serververbindung und Datenbank automatisch konfigurieren SA_PW=$(printf '%s' "${SERVER_SA_PASSWORD:-SuperAdmin}" | sed 's/&/\&/g; s//\>/g') SAFE_RCP_NAME=$(printf '%s' "${SERVER_NAME:-Trackmania Server}" | sed 's/&/\&/g; s//\>/g') SAFE_DB_DSN=$(printf '%s' "$DB_DSN" | sed 's/&/\&/g; s//\>/g') SAFE_DB_USER=$(printf '%s' "$REMOTECP_DB_USER" | sed 's/&/\&/g; s//\>/g') SAFE_DB_PW=$(printf '%s' "$REMOTECP_DB_PASSWORD" | sed 's/&/\&/g; s//\>/g') cat > "$REMOTECP_DIR/xml/servers.xml" < 1 ${SAFE_RCP_NAME} 127.0.0.1 ${XMLRPC_PORT} ${SA_PW} 000000 localhost username password /GameData/Tracks/ ${SAFE_DB_DSN} ${SAFE_DB_USER} ${SAFE_DB_PW} guestlist.txt blacklist.txt EORCPSERV # admins.xml: Admin-Zugang aus SuperAdmin-Passwort konfigurieren RCP_PW="${SERVER_SA_PASSWORD:-SuperAdmin}" RCP_PW_MD5=$(printf '%s' "$RCP_PW" | md5sum | cut -d' ' -f1) cat > "$REMOTECP_DIR/xml/admins.xml" < L1 true rcplive 5b8e508f6f4a95bc581a37243d88f07e false false en G1 true Guest adb831a7fdd83dd1e2a309ce7591dff8 false false en 1 true SuperAdmin ${RCP_PW_MD5} false false de EORCPADM # ============================================================ # RemoteCP: Datenbank-Initialisierung # ============================================================ if [ "$DB_ENABLED" = "true" ]; then echo " Warte auf MariaDB (${REMOTECP_DB_HOST})..." DB_READY=false for i in $(seq 1 30); do if mysql -h "$REMOTECP_DB_HOST" -u "$REMOTECP_DB_USER" -p"$REMOTECP_DB_PASSWORD" "$REMOTECP_DB_NAME" -e "SELECT 1" > /dev/null 2>&1; then echo " MariaDB erreichbar." DB_READY=true break fi echo " Versuch $i/30 - MariaDB noch nicht bereit, warte 3s..." sleep 3 done if [ "$DB_READY" = "true" ]; then echo " Importiere RemoteCP-Datenbankschema..." for sqlfile in "$REMOTECP_DIR"/plugins/*/mysql_*.sql "$REMOTECP_DIR"/live/*/mysql_*.sql; do if [ -f "$sqlfile" ]; then echo " -> $(basename "$sqlfile")" mysql -h "$REMOTECP_DB_HOST" -u "$REMOTECP_DB_USER" -p"$REMOTECP_DB_PASSWORD" "$REMOTECP_DB_NAME" < "$sqlfile" fi done # Installer-Markierung setzen (ueberspringt den Web-Installer) echo "installed" > "$REMOTECP_DIR/cache/installed" chown www-data:www-data "$REMOTECP_DIR/cache/installed" echo " RemoteCP-Datenbank erfolgreich initialisiert." else echo " WARNUNG: MariaDB nicht erreichbar nach 90s!" echo " RemoteCP-Datenbank muss manuell eingerichtet werden." echo " Installer: http:///remotecp/index.php?page=install" fi fi # Berechtigungen fuer RemoteCP setzen chmod -R 777 "$REMOTECP_DIR/cache" chmod -R 777 "$REMOTECP_DIR/xml" chown -R www-data:www-data "$REMOTECP_DIR/" echo " RemoteCP-Konfiguration abgeschlossen (Port: ${XMLRPC_PORT}, User: SuperAdmin)." fi else echo "==> Vorhandene AdminServ-Daten gefunden. Keine Aenderungen." fi # ============================================================ # XAseco: First-Run-Logik # ============================================================ # Beim ersten Start (leeres Volume) werden die XAseco-Dateien # aus dem Default-Template ins Volume kopiert und die # Konfiguration aus den Umgebungsvariablen angewendet. # ============================================================ XASECO_DIR="/opt/tmserver/xaseco" DEFAULT_XASECO="/opt/tmserver/default-xaseco" XASECO_ENABLED="${XASECO_ENABLED:-true}" if [ "$XASECO_ENABLED" = "true" ]; then if [ ! -f "$XASECO_DIR/aseco.php" ]; then echo "==> Erster Start erkannt: Kopiere XAseco-Dateien ins Volume..." cp -a "$DEFAULT_XASECO"/* "$XASECO_DIR/" XMLRPC_PORT="${SERVER_XMLRPC_PORT:-5000}" XASECO_ADMIN="${XASECO_MASTERADMIN_LOGIN:-}" SA_PW_XASECO=$(printf '%s' "${SERVER_SA_PASSWORD:-SuperAdmin}" | sed 's/&/\&/g; s//\>/g') # --- config.xml: MasterAdmin und TMServer-Verbindung konfigurieren --- if [ -n "$XASECO_ADMIN" ]; then SAFE_ADMIN=$(printf '%s' "$XASECO_ADMIN" | sed 's/&/\&/g; s//\>/g') # MasterAdmin in die masteradmins-Liste einfuegen (nach -Tag) sed -i "//a\\ ${SAFE_ADMIN} " "$XASECO_DIR/config.xml" echo " config.xml: MasterAdmin '${XASECO_ADMIN}' gesetzt." else echo " HINWEIS: XASECO_MASTERADMIN_LOGIN nicht gesetzt." echo " Bitte manuell in xaseco/config.xml eintragen!" fi sed -i "s|YOUR_SUPERADMIN_PASSWORD|${SA_PW_XASECO}|" "$XASECO_DIR/config.xml" sed -i "s|5000|${XMLRPC_PORT}|" "$XASECO_DIR/config.xml" echo " config.xml: TMServer-Verbindung konfiguriert (Port: ${XMLRPC_PORT})." # --- adminops.xml: Admin-Login eintragen --- if [ -n "$XASECO_ADMIN" ]; then SAFE_ADMIN=$(printf '%s' "$XASECO_ADMIN" | sed 's/&/\&/g; s//\>/g') sed -i "s|||" "$XASECO_DIR/adminops.xml" # Admin in die admins-Liste einfuegen sed -i "//a\\ ${SAFE_ADMIN} " "$XASECO_DIR/adminops.xml" echo " adminops.xml: Admin '${XASECO_ADMIN}' eingetragen." fi # --- localdatabase.xml: MySQL-Verbindung konfigurieren --- XASECO_DB_HOST="${XASECO_DB_HOST:-mariadb}" XASECO_DB_NAME="${XASECO_DB_NAME:-xaseco}" XASECO_DB_USER="${XASECO_DB_USER:-xaseco}" XASECO_DB_PASSWORD="${XASECO_DB_PASSWORD:-}" if [ -n "$XASECO_DB_PASSWORD" ]; then SAFE_DB_HOST=$(printf '%s' "$XASECO_DB_HOST" | sed 's/&/\&/g; s//\>/g') SAFE_DB_USER=$(printf '%s' "$XASECO_DB_USER" | sed 's/&/\&/g; s//\>/g') SAFE_DB_PW=$(printf '%s' "$XASECO_DB_PASSWORD" | sed 's/&/\&/g; s//\>/g') SAFE_DB_NAME=$(printf '%s' "$XASECO_DB_NAME" | sed 's/&/\&/g; s//\>/g') sed -i "s|localhost|${SAFE_DB_HOST}|" "$XASECO_DIR/localdatabase.xml" sed -i "s|YOUR_MYSQL_LOGIN|${SAFE_DB_USER}|" "$XASECO_DIR/localdatabase.xml" sed -i "s|YOUR_MYSQL_PASSWORD|${SAFE_DB_PW}|" "$XASECO_DIR/localdatabase.xml" sed -i "s|aseco|${SAFE_DB_NAME}|" "$XASECO_DIR/localdatabase.xml" echo " localdatabase.xml: MySQL-Verbindung konfiguriert (Host: ${XASECO_DB_HOST}, DB: ${XASECO_DB_NAME})." else echo " WARNUNG: XASECO_DB_PASSWORD nicht gesetzt!" echo " XAseco-Datenbank muss manuell in xaseco/localdatabase.xml konfiguriert werden." fi # --- dedimania.xml: Server-Account konfigurieren --- DEDI_LOGIN="${SERVER_LOGIN:-}" DEDI_PASSWORD="${SERVER_LOGIN_PASSWORD:-}" DEDI_NATION="${XASECO_DEDIMANIA_NATION:-DEU}" if [ -n "$DEDI_LOGIN" ] && [ -n "$DEDI_PASSWORD" ]; then SAFE_DEDI_LOGIN=$(printf '%s' "$DEDI_LOGIN" | sed 's/&/\&/g; s//\>/g') SAFE_DEDI_PW=$(printf '%s' "$DEDI_PASSWORD" | sed 's/&/\&/g; s//\>/g') SAFE_DEDI_NATION=$(printf '%s' "$DEDI_NATION" | sed 's/&/\&/g; s//\>/g') sed -i "s|YOUR_SERVER_LOGIN|${SAFE_DEDI_LOGIN}|" "$XASECO_DIR/dedimania.xml" sed -i "s|YOUR_SERVER_PASSWORD|${SAFE_DEDI_PW}|" "$XASECO_DIR/dedimania.xml" sed -i "s|YOUR_SERVER_NATION|${SAFE_DEDI_NATION}|" "$XASECO_DIR/dedimania.xml" echo " dedimania.xml: Server-Account konfiguriert (Nation: ${DEDI_NATION})." else echo " HINWEIS: SERVER_LOGIN/SERVER_LOGIN_PASSWORD nicht gesetzt." echo " Dedimania wird ohne Account konfiguriert." fi # --- XAseco-Datenbank: Schema importieren --- if [ -n "$XASECO_DB_PASSWORD" ]; then echo " Warte auf MariaDB (${XASECO_DB_HOST}) fuer XAseco-DB..." DB_READY=false for i in $(seq 1 30); do if mysql -h "$XASECO_DB_HOST" -u "$XASECO_DB_USER" -p"$XASECO_DB_PASSWORD" "$XASECO_DB_NAME" -e "SELECT 1" > /dev/null 2>&1; then echo " MariaDB erreichbar." DB_READY=true break fi echo " Versuch $i/30 - MariaDB noch nicht bereit, warte 3s..." sleep 3 done if [ "$DB_READY" = "true" ]; then echo " Importiere XAseco-Datenbankschema..." for sqlfile in "$XASECO_DIR"/localdb/*.sql; do if [ -f "$sqlfile" ]; then echo " -> $(basename "$sqlfile")" mysql -h "$XASECO_DB_HOST" -u "$XASECO_DB_USER" -p"$XASECO_DB_PASSWORD" "$XASECO_DB_NAME" < "$sqlfile" fi done echo " XAseco-Datenbank erfolgreich initialisiert." else echo " WARNUNG: MariaDB nicht erreichbar nach 90s!" echo " XAseco-Datenbankschema muss manuell importiert werden." fi fi echo " XAseco-Konfiguration abgeschlossen." else echo "==> Vorhandene XAseco-Daten gefunden. Keine Aenderungen." fi fi # ============================================================ # XAseco: TeamSpeak3-Plugin Gateway aktualisieren (fuer bestehende Volumes) # ============================================================ # Das Original-TS3-Gateway ist nicht mehr verfuegbar. Die eigene # teamspeak3.xml mit dem Ersatz-Gateway wird in das Volume kopiert, # falls sie fehlt oder noch das alte (nicht mehr erreichbare) Gateway # referenziert. Gleichzeitig wird das Plugin reaktiviert, falls es # in einer frueheren Version auskommentiert wurde. # ============================================================ XASECO_DIR_TS3="/opt/tmserver/xaseco" TS3_XML="$XASECO_DIR_TS3/teamspeak3.xml" TS3_DEFAULT="/opt/tmserver/default-xaseco/teamspeak3.xml" TS3_PLUGINS_XML="$XASECO_DIR_TS3/plugins.xml" # teamspeak3.xml aktualisieren: Kopieren wenn fehlend oder veraltet if [ -f "$TS3_DEFAULT" ]; then if [ ! -f "$TS3_XML" ]; then echo "==> TeamSpeak3-Gateway: teamspeak3.xml fehlt, kopiere aus Template..." cp "$TS3_DEFAULT" "$TS3_XML" echo " teamspeak3.xml erfolgreich kopiert." elif ! diff -q "$TS3_DEFAULT" "$TS3_XML" > /dev/null 2>&1; then echo "==> TeamSpeak3-Gateway: teamspeak3.xml wird aktualisiert..." cp "$TS3_DEFAULT" "$TS3_XML" echo " teamspeak3.xml erfolgreich aktualisiert." fi fi # TS3-Plugin reaktivieren, falls es auskommentiert ist if [ -f "$TS3_PLUGINS_XML" ] && grep -q '' "$TS3_PLUGINS_XML"; then echo "==> TeamSpeak3-Plugin: Reaktiviere auskommentiertes Plugin..." sed -i 's||plugin.teamspeak3.php|' "$TS3_PLUGINS_XML" echo " TeamSpeak3-Plugin erfolgreich aktiviert." fi # ============================================================ # RemoteCP: PHP-Warnungen in Plugins fixen (fuer bestehende Volumes) # ============================================================ # RemoteCP nutzt bare constants (pt_custom, pt_points, ...), die in # PHP 7.2+ Warnungen ausloesen. Die gepatchte Datei aus dem Image # wird in das Volume kopiert, falls die alte Version noch vorhanden ist. # ============================================================ CUSTOMPOINTS_FILE="/var/www/html/remotecp/plugins/CustomPoints/index.php" CUSTOMPOINTS_DEFAULT="/opt/tmserver/default-controlpanel/remotecp/plugins/CustomPoints/index.php" if [ -f "$CUSTOMPOINTS_FILE" ] && ! grep -q 'defined.*pt_custom' "$CUSTOMPOINTS_FILE"; then echo "==> Patche CustomPoints-Plugin (PHP-Warnungen beheben)..." cp "$CUSTOMPOINTS_DEFAULT" "$CUSTOMPOINTS_FILE" chown www-data:www-data "$CUSTOMPOINTS_FILE" echo " CustomPoints-Plugin erfolgreich gepatcht." fi # ============================================================ # AdminServ: MatchSettings-Bugfixes fuer bestehende Volumes # ============================================================ # 1) get_matchset_mapimport.php: Berechnet den relativen Pfad aus dem # absoluten Dropdown-Pfad statt den URL-Parameter 'd' zu verwenden. # Ohne Fix wird z.B. "MatchSettings/" statt "Challenges/Downloaded/" # als Praefix in die MatchSettings-Datei geschrieben. # 2) maps-creatematchset.php: Ueberspringt GetModeScriptInfo fuer # TmForever (Methode existiert nur in ManiaPlanet/TM2, Fehler -506). # ============================================================ ADMINSERV_MAPIMPORT="/var/www/html/resources/ajax/get_matchset_mapimport.php" ADMINSERV_MAPIMPORT_DEFAULT="/opt/tmserver/default-controlpanel/resources/ajax/get_matchset_mapimport.php" if [ -f "$ADMINSERV_MAPIMPORT" ] && ! grep -q 'relativePath' "$ADMINSERV_MAPIMPORT"; then echo "==> Patche AdminServ: MatchSettings Map-Import (Pfad-Fix)..." cp "$ADMINSERV_MAPIMPORT_DEFAULT" "$ADMINSERV_MAPIMPORT" chown www-data:www-data "$ADMINSERV_MAPIMPORT" echo " get_matchset_mapimport.php erfolgreich gepatcht." fi ADMINSERV_CREATEMATCHSET="/var/www/html/resources/process/maps-creatematchset.php" ADMINSERV_CREATEMATCHSET_DEFAULT="/opt/tmserver/default-controlpanel/resources/process/maps-creatematchset.php" if [ -f "$ADMINSERV_CREATEMATCHSET" ] && grep -q "query('GetModeScriptInfo')" "$ADMINSERV_CREATEMATCHSET" && ! grep -q "SERVER_VERSION_NAME != 'TmForever'" "$ADMINSERV_CREATEMATCHSET"; then echo "==> Patche AdminServ: GetModeScriptInfo-Fix fuer TmForever..." cp "$ADMINSERV_CREATEMATCHSET_DEFAULT" "$ADMINSERV_CREATEMATCHSET" chown www-data:www-data "$ADMINSERV_CREATEMATCHSET" echo " maps-creatematchset.php erfolgreich gepatcht." fi # ============================================================ # RemoteCP: Mods-Plugin settings.xml aktualisieren (fuer bestehende Volumes) # ============================================================ # Die vorkonfigurierte Skin-Liste aus dem Image wird in das Volume # kopiert, falls die alte Standard-settings.xml noch vorhanden ist # (erkennbar am Beispiel-Eintrag "blacksunonline.com"). # ============================================================ MODS_SETTINGS_FILE="/var/www/html/remotecp/plugins/Mods/settings.xml" MODS_SETTINGS_DEFAULT="/opt/tmserver/default-controlpanel/remotecp/plugins/Mods/settings.xml" if [ -f "$MODS_SETTINGS_FILE" ] && grep -q 'blacksunonline.com' "$MODS_SETTINGS_FILE"; then echo "==> Aktualisiere RemoteCP Mods-Plugin (Skin-Liste von techniverse.net)..." cp "$MODS_SETTINGS_DEFAULT" "$MODS_SETTINGS_FILE" chown www-data:www-data "$MODS_SETTINGS_FILE" echo " Mods/settings.xml erfolgreich aktualisiert." fi echo "Starting apache server" service apache2 start CONFIG="/opt/tmserver/GameData/Config/dedicated_cfg.txt" GAME_SETTINGS="/opt/tmserver/GameData/Tracks/MatchSettings/custom_game_settings.txt" GAMEDATA_DIR="/opt/tmserver/GameData" DEFAULT_GAMEDATA="/opt/tmserver/default-gamedata" # ============================================================ # Persistente GameData: First-Run-Logik # ============================================================ # Beim ersten Start (leeres Volume) wird das gesamte GameData- # Verzeichnis aus dem Default-Template ins Volume kopiert und # die Umgebungsvariablen auf die Config angewendet. # Bei weiteren Starts wird die vorhandene Konfiguration beibehalten, # damit manuelle Aenderungen nicht ueberschrieben werden. # Mit FORCE_CONFIG_UPDATE=true kann ein erneutes Anwenden erzwungen werden. # ============================================================ FORCE_CONFIG_UPDATE="${FORCE_CONFIG_UPDATE:-false}" if [ ! -f "$CONFIG" ]; then echo "==> Erster Start erkannt: Kopiere Default-GameData ins Volume..." cp -a "$DEFAULT_GAMEDATA"/* "$GAMEDATA_DIR/" chmod -R 777 "$GAMEDATA_DIR/Config/" mkdir -p "$GAMEDATA_DIR/Config/AdminServ/ServerOptions" chown -R www-data:www-data "$GAMEDATA_DIR/Config/AdminServ" # Tracks-Verzeichnis fuer AdminServ beschreibbar machen (Maps-Upload/Download) chown -R www-data:www-data "$GAMEDATA_DIR/Tracks/" chmod -R 755 "$GAMEDATA_DIR/Tracks/" APPLY_ENV=true elif [ "$FORCE_CONFIG_UPDATE" = "true" ]; then echo "==> FORCE_CONFIG_UPDATE ist aktiv: Umgebungsvariablen werden erneut angewendet..." echo " ACHTUNG: Manuelle Aenderungen an den betroffenen Feldern werden ueberschrieben!" # Template neu kopieren, damit alle Platzhalter vorhanden sind cp "$DEFAULT_GAMEDATA/Config/dedicated_cfg.txt" "$CONFIG" cp "$DEFAULT_GAMEDATA/Tracks/MatchSettings/custom_game_settings.txt" "$GAME_SETTINGS" APPLY_ENV=true else echo "==> Vorhandene Konfiguration gefunden. Umgebungsvariablen werden NICHT angewendet." echo " Zum erneuten Anwenden: FORCE_CONFIG_UPDATE=true setzen." APPLY_ENV=false fi # ============================================================ # Platzhalter in dedicated_cfg.txt durch Umgebungsvariablen ersetzen # ============================================================ if [ "$APPLY_ENV" = "true" ]; then echo "Ersetze Platzhalter in dedicated_cfg.txt mit Umgebungsvariablen..." # Authentifizierung sed -i "s|%%SERVER_SA_PASSWORD%%|${SERVER_SA_PASSWORD}|g" "$CONFIG" sed -i "s|%%SERVER_ADM_PASSWORD%%|${SERVER_ADM_PASSWORD}|g" "$CONFIG" sed -i "s|%%SERVER_USER_PASSWORD%%|${SERVER_USER_PASSWORD}|g" "$CONFIG" # Masterserver-Account sed -i "s|%%SERVER_LOGIN%%|${SERVER_LOGIN}|g" "$CONFIG" sed -i "s|%%SERVER_LOGIN_PASSWORD%%|${SERVER_LOGIN_PASSWORD}|g" "$CONFIG" sed -i "s|%%SERVER_VALIDATION_KEY%%|${SERVER_VALIDATION_KEY}|g" "$CONFIG" # Server-Optionen sed -i "s|%%SERVER_NAME%%|${SERVER_NAME}|g" "$CONFIG" sed -i "s|%%SERVER_DESC%%|${SERVER_DESC}|g" "$CONFIG" sed -i "s|%%SERVER_HIDE%%|${SERVER_HIDE}|g" "$CONFIG" sed -i "s|%%SERVER_MAX_PLAYERS%%|${SERVER_MAX_PLAYERS}|g" "$CONFIG" sed -i "s|%%SERVER_PASSWORD%%|${SERVER_PASSWORD}|g" "$CONFIG" sed -i "s|%%SERVER_MAX_SPECTATORS%%|${SERVER_MAX_SPECTATORS}|g" "$CONFIG" sed -i "s|%%SERVER_SPEC_PASSWORD%%|${SERVER_SPEC_PASSWORD}|g" "$CONFIG" sed -i "s|%%SERVER_LADDER_MODE%%|${SERVER_LADDER_MODE}|g" "$CONFIG" sed -i "s|%%SERVER_LADDER_LIMIT_MAX%%|${SERVER_LADDER_LIMIT_MAX:-60000}|g" "$CONFIG" # Netzwerk sed -i "s|%%SERVER_PORT%%|${SERVER_PORT}|g" "$CONFIG" sed -i "s|%%SERVER_P2P_PORT%%|${SERVER_P2P_PORT}|g" "$CONFIG" sed -i "s|%%SERVER_XMLRPC_PORT%%|${SERVER_XMLRPC_PORT}|g" "$CONFIG" sed -i "s|%%SERVER_UPLOAD_RATE%%|${SERVER_UPLOAD_RATE}|g" "$CONFIG" sed -i "s|%%SERVER_DOWNLOAD_RATE%%|${SERVER_DOWNLOAD_RATE}|g" "$CONFIG" # Spieleinstellungen (MatchSettings) sed -i "s|[^<]*|${ALLWARMUPDURATION:-0}|" "$GAME_SETTINGS" echo "Platzhalter erfolgreich ersetzt." fi # ============================================================ # AdminServ ServerOptions: Exportierte Einstellungen anwenden # ============================================================ # Falls ein AdminServ-Export in GameData/Config/AdminServ/ServerOptions/ # vorhanden ist, werden die darin enthaltenen Werte (z.B. Servername, # Beschreibung, Spielerzahl) in die dedicated_cfg.txt uebernommen. # So bleiben Aenderungen, die ueber AdminServ vorgenommen und exportiert # wurden, auch nach einem Container-Neustart erhalten. # ============================================================ ADMINSERV_OPTIONS_DIR="$GAMEDATA_DIR/Config/AdminServ/ServerOptions" if [ -d "$ADMINSERV_OPTIONS_DIR" ]; then LATEST_EXPORT=$(ls -t "$ADMINSERV_OPTIONS_DIR"/*.txt "$ADMINSERV_OPTIONS_DIR"/*.xml 2>/dev/null | head -1) if [ -n "$LATEST_EXPORT" ] && [ -f "$LATEST_EXPORT" ]; then echo "==> AdminServ ServerOptions-Export gefunden: $(basename "$LATEST_EXPORT")" echo " Uebernehme exportierte Einstellungen in dedicated_cfg.txt..." php -r ' $xmlFile = $argv[1]; $cfgFile = $argv[2]; // AdminServ-Export parsen $dom = new DOMDocument(); if (!@$dom->load($xmlFile)) { echo " WARNUNG: AdminServ-Export konnte nicht gelesen werden.\n"; exit(0); } $root = $dom->documentElement; $exportValues = []; foreach ($root->childNodes as $node) { if ($node->nodeType === XML_ELEMENT_NODE) { $exportValues[$node->nodeName] = $node->nodeValue; } } // Mapping: AdminServ-XML-Feld => dedicated_cfg.txt-Feld $mapping = [ "Name" => "name", "Comment" => "comment", "HideServer" => "hide_server", "NextMaxPlayers" => "max_players", "Password" => "password", "PasswordForSpectator" => "password_spectator", "NextMaxSpectators" => "max_spectators", "NextLadderMode" => "ladder_mode", "NextCallVoteTimeOut" => "callvote_timeout", "CallVoteRatio" => "callvote_ratio", "AllowChallengeDownload" => "allow_challenge_download", "AutoSaveReplays" => "autosave_replays", "IsP2PUpload" => "enable_p2p_upload", "IsP2PDownload" => "enable_p2p_download", ]; // Bool-Felder: 1/0 => True/False (dedicated_cfg.txt-Format) $boolFields = [ "allow_challenge_download", "autosave_replays", "enable_p2p_upload", "enable_p2p_download", ]; // Ladder-Modus: 0 => inactive, 1 => forced $ladderMap = ["0" => "inactive", "1" => "forced"]; // Zu ersetzende Werte aufbauen $replacements = []; foreach ($mapping as $xmlField => $cfgField) { if (isset($exportValues[$xmlField])) { $value = $exportValues[$xmlField]; if (in_array($cfgField, $boolFields)) { $value = ($value == "1" || strtolower($value) === "true") ? "True" : "False"; } if ($cfgField === "ladder_mode" && isset($ladderMap[$value])) { $value = $ladderMap[$value]; } $replacements[$cfgField] = $value; } } if (empty($replacements)) { echo " Keine anwendbaren Einstellungen im Export gefunden.\n"; exit(0); } // dedicated_cfg.txt zeilenweise verarbeiten // Nur Tags innerhalb von werden ersetzt, // damit und in unangetastet bleiben. $lines = file($cfgFile); $inServerOptions = false; $updated = 0; foreach ($lines as $i => $line) { if (strpos($line, "") !== false) { $inServerOptions = true; } if (strpos($line, "") !== false) { $inServerOptions = false; } if ($inServerOptions) { foreach ($replacements as $field => $value) { $pattern = "/(<" . preg_quote($field, "/") . ">)[^<]*(<\/" . preg_quote($field, "/") . ">)/"; if (preg_match($pattern, $line)) { $safeValue = htmlspecialchars($value, ENT_XML1 | ENT_QUOTES, "UTF-8"); // $ und \ im Replacement escapen, damit preg_replace // sie nicht als Backreferences interpretiert (wichtig // fuer TM-Farbcodes wie $03F, $z, $s etc.) $escapedValue = str_replace(["\\", "$"], ["\\\\", "\\$"], $safeValue); $lines[$i] = preg_replace($pattern, "\${1}" . $escapedValue . "\${2}", $line, 1); echo " " . $field . " => " . $value . "\n"; $updated++; unset($replacements[$field]); break; } } } } if ($updated > 0) { file_put_contents($cfgFile, implode("", $lines)); echo " " . $updated . " Einstellung(en) aus AdminServ-Export uebernommen.\n"; } ' "$LATEST_EXPORT" "$CONFIG" fi else echo "==> Kein AdminServ ServerOptions-Verzeichnis gefunden. Ueberspringe Import." fi # ============================================================ # MatchSettings: Neueste Datei automatisch ermitteln # ============================================================ # Ueber die Umgebungsvariable MATCHSETTINGS_FILE kann gesteuert werden, # welche MatchSettings-Datei beim Serverstart geladen wird: # - "auto" (Standard): Die neueste .txt-Datei im MatchSettings-Ordner # wird automatisch anhand des Aenderungsdatums ermittelt. # - "": Eine bestimmte Datei wird direkt verwendet. # Fallback: custom_game_settings.txt (Standard-MatchSettings aus dem Image). # ============================================================ MATCHSETTINGS_DIR="$GAMEDATA_DIR/Tracks/MatchSettings" MATCHSETTINGS_FILE_ENV="${MATCHSETTINGS_FILE:-auto}" if [ "$MATCHSETTINGS_FILE_ENV" = "auto" ]; then echo "==> MatchSettings: Automatische Erkennung (MATCHSETTINGS_FILE=auto)..." # Neueste .txt-Datei im MatchSettings-Ordner anhand des Aenderungsdatums ermitteln NEWEST_MS=$(ls -t "$MATCHSETTINGS_DIR"/*.txt 2>/dev/null | head -1) if [ -n "$NEWEST_MS" ] && [ -f "$NEWEST_MS" ]; then MS_FILENAME=$(basename "$NEWEST_MS") GAME_SETTINGS_PATH="MatchSettings/${MS_FILENAME}" echo " Neueste MatchSettings gefunden: ${MS_FILENAME}" echo " Aenderungsdatum: $(stat -c '%y' "$NEWEST_MS" 2>/dev/null || ls -la "$NEWEST_MS" | awk '{print $6, $7, $8}')" else GAME_SETTINGS_PATH="MatchSettings/custom_game_settings.txt" echo " Keine .txt-Dateien in ${MATCHSETTINGS_DIR} gefunden." echo " Fallback: ${GAME_SETTINGS_PATH}" fi else # Explizit angegebene Datei verwenden if [ -f "$MATCHSETTINGS_DIR/$MATCHSETTINGS_FILE_ENV" ]; then GAME_SETTINGS_PATH="MatchSettings/${MATCHSETTINGS_FILE_ENV}" echo "==> MatchSettings: Verwende explizit gesetzte Datei: ${MATCHSETTINGS_FILE_ENV}" else echo "==> WARNUNG: Angegebene MatchSettings-Datei nicht gefunden: ${MATCHSETTINGS_FILE_ENV}" echo " Vorhandene Dateien in ${MATCHSETTINGS_DIR}:" ls -la "$MATCHSETTINGS_DIR"/*.txt 2>/dev/null || echo " (keine .txt-Dateien vorhanden)" GAME_SETTINGS_PATH="MatchSettings/custom_game_settings.txt" echo " Fallback: ${GAME_SETTINGS_PATH}" fi fi echo " Aktive MatchSettings: ${GAME_SETTINGS_PATH}" # Bestimme Server-Modus (Standard: internet) SERVER_MODE="${SERVER_MODE:-internet}" if [ "$SERVER_MODE" = "internet" ]; then echo "Configuring Internet-Dedicated mode" if [ -z "$SERVER_LOGIN" ] || [ -z "$SERVER_VALIDATION_KEY" ]; then echo "ERROR: SERVER_LOGIN and SERVER_VALIDATION_KEY are required for Internet-Dedicated mode." echo "Set SERVER_MODE=lan to start in LAN mode, or provide the required variables." exit 1 fi LAUNCH_MODE="/internet" else echo "Configuring LAN-Dedicated mode" LAUNCH_MODE="/lan" fi echo "Server config dedicated_cfg.txt is" cat "$CONFIG" echo "Launching Server in ${SERVER_MODE} mode (MatchSettings: ${GAME_SETTINGS_PATH})" ./TrackmaniaServer /dedicated_cfg=dedicated_cfg.txt /game_settings=${GAME_SETTINGS_PATH} /nodaemon ${LAUNCH_MODE} & TM_PID=$! echo "TrackmaniaServer gestartet (PID: ${TM_PID})" # ============================================================ # Warte auf XMLRPC-Verfuegbarkeit # ============================================================ # Sowohl XAseco als auch Forced Mods benoetigen eine aktive # XMLRPC-Verbindung zum TrackmaniaServer. # ============================================================ echo "==> Warte auf TrackmaniaServer XMLRPC..." XMLRPC_PORT="${SERVER_XMLRPC_PORT:-5000}" XMLRPC_READY=false for i in $(seq 1 30); do if php -r "@fsockopen('127.0.0.1', ${XMLRPC_PORT}, \$e, \$m, 2) ? exit(0) : exit(1);" 2>/dev/null; then XMLRPC_READY=true break fi sleep 2 done if [ "$XMLRPC_READY" = "true" ]; then echo " XMLRPC-Port ${XMLRPC_PORT} erreichbar." else echo " WARNUNG: XMLRPC-Port ${XMLRPC_PORT} nicht erreichbar nach 60s!" fi # ============================================================ # Forced Mods (Skins) per XMLRPC setzen # ============================================================ # Wenn FORCE_MOD_*-Variablen gesetzt sind, werden die # entsprechenden Mods per SetForcedMods-XMLRPC-Aufruf beim # Serverstart automatisch aktiviert. Dies funktioniert bei # jedem Containerstart und ist unabhaengig von FORCE_CONFIG_UPDATE. # ============================================================ # Pruefen, ob mindestens ein Mod gesetzt ist (Variable fuer spaeter) HAS_MODS=false if [ "$XMLRPC_READY" = "true" ]; then for ENV_NAME in FORCE_MOD_STADIUM FORCE_MOD_ISLAND FORCE_MOD_BAY FORCE_MOD_COAST FORCE_MOD_SPEED FORCE_MOD_ALPINE FORCE_MOD_RALLY; do eval "MOD_VAL=\${$ENV_NAME:-}" if [ -n "$MOD_VAL" ]; then HAS_MODS=true break fi done fi # ============================================================ # XAseco starten (nach TrackmaniaServer, VOR Forced Mods) # ============================================================ # XAseco muss zuerst starten und sich initialisieren, damit # es die Forced Mods nicht ueberschreibt oder zuruecksetzt. # ============================================================ if [ "$XMLRPC_READY" = "true" ]; then if [ "${XASECO_ENABLED:-true}" = "true" ] && [ -f "/opt/tmserver/xaseco/aseco.php" ]; then echo "==> Starte XAseco..." cd /opt/tmserver/xaseco php aseco.php TMN >aseco.log 2>&1 & XASECO_PID=$! echo " XAseco gestartet (PID: ${XASECO_PID})" cd /opt/tmserver elif [ "${XASECO_ENABLED:-true}" != "true" ]; then echo "==> XAseco ist deaktiviert (XASECO_ENABLED=${XASECO_ENABLED})." fi else echo " WARNUNG: XMLRPC nicht erreichbar - XAseco und Forced Mods wurden NICHT gestartet." fi # ============================================================ # Forced Mods (Skins) per XMLRPC setzen # ============================================================ # Wird NACH XAseco-Start ausgefuehrt, damit XAseco die Mods # nicht bei seiner Initialisierung zuruecksetzt. # ============================================================ if [ "$XMLRPC_READY" = "true" ] && [ "$HAS_MODS" = "true" ]; then # Warten, damit XAseco und der TM-Server sich vollstaendig initialisieren echo "==> Forced Mods: Warte 10 Sekunden auf vollstaendige Server-Initialisierung..." sleep 10 echo "==> Forced Mods: Setze Mods per XMLRPC..." # JSON-Array der Mods aufbauen MODS_JSON="[" MODS_FIRST=true for PAIR in "FORCE_MOD_STADIUM:Stadium" "FORCE_MOD_ISLAND:Island" "FORCE_MOD_BAY:Bay" "FORCE_MOD_COAST:Coast" "FORCE_MOD_SPEED:Speed" "FORCE_MOD_ALPINE:Alpine" "FORCE_MOD_RALLY:Rally"; do VAR_NAME="${PAIR%%:*}" ENV_NAME="${PAIR##*:}" eval "MOD_URL=\${$VAR_NAME:-}" if [ -n "$MOD_URL" ]; then if [ "$MODS_FIRST" = "true" ]; then MODS_FIRST=false else MODS_JSON="${MODS_JSON}," fi # URL fuer JSON escapen (Backslash und Anfuehrungszeichen) SAFE_URL=$(printf '%s' "$MOD_URL" | sed 's/\\/\\\\/g; s/"/\\"/g') MODS_JSON="${MODS_JSON}{\"Env\":\"${ENV_NAME}\",\"Url\":\"${SAFE_URL}\"}" echo " ${ENV_NAME} => ${MOD_URL}" fi done MODS_JSON="${MODS_JSON}]" # GBXRemote2-Protokoll: Authenticate + SetForcedMods SA_PW_MODS="${SERVER_SA_PASSWORD:-SuperAdmin}" php -r ' $port = (int)$argv[1]; $password = $argv[2]; $modsJson = $argv[3]; $mods = json_decode($modsJson, true); if (empty($mods)) { echo " Keine Mods zu setzen.\n"; exit(0); } // GBXRemote2: Verbindung herstellen $fp = @fsockopen("127.0.0.1", $port, $errno, $errstr, 5); if (!$fp) { echo " FEHLER: Verbindung zu XMLRPC fehlgeschlagen ($errno: $errstr).\n"; exit(1); } stream_set_timeout($fp, 10); // Handshake lesen (4 Bytes Laenge + Protokollstring) $data = fread($fp, 4); if (strlen($data) < 4) { echo " FEHLER: Handshake fehlgeschlagen.\n"; fclose($fp); exit(1); } $info = unpack("Vsize", $data); $handshake = fread($fp, $info["size"]); if (strpos($handshake, "GBXRemote") === false) { echo " FEHLER: Kein GBXRemote-Protokoll.\n"; fclose($fp); exit(1); } echo " Protokoll: $handshake\n"; $reqhandle = 0x80000001; // XML-RPC-Wert kodieren function encodeVal($v) { if (is_bool($v)) return "" . ($v ? "1" : "0") . ""; if (is_int($v)) return "" . $v . ""; if (is_string($v)) return "" . htmlspecialchars($v, ENT_XML1) . ""; if (is_array($v)) { if (array_keys($v) !== range(0, count($v) - 1)) { $x = ""; foreach ($v as $k => $val) $x .= "" . $k . "" . encodeVal($val) . ""; return $x . ""; } else { $x = ""; foreach ($v as $val) $x .= encodeVal($val); return $x . ""; } } return "" . htmlspecialchars((string)$v, ENT_XML1) . ""; } // Ein einzelnes Paket vom Server lesen (Header + Body) function readPacket($fp) { $header = ""; while (strlen($header) < 8) { $chunk = @fread($fp, 8 - strlen($header)); if ($chunk === false || strlen($chunk) === 0) return false; $header .= $chunk; } $info = unpack("Vsize/Vhandle", $header); $size = $info["size"]; $handle = $info["handle"]; if ($size > 4194304 || $size == 0) return false; $body = ""; $remaining = $size; while ($remaining > 0) { $chunk = @fread($fp, min($remaining, 8192)); if ($chunk === false || strlen($chunk) === 0) break; $body .= $chunk; $remaining -= strlen($chunk); } return ["handle" => $handle, "body" => $body]; } // XMLRPC-Request senden und Antwort lesen // Callbacks (handle < 0x80000000) werden uebersprungen function gbxQuery($fp, &$reqhandle, $method, $params) { $xml = "" . "" . $method . ""; foreach ($params as $p) $xml .= "" . encodeVal($p) . ""; $xml .= ""; $myHandle = $reqhandle++; $packet = pack("VV", strlen($xml), $myHandle) . $xml; $written = @fwrite($fp, $packet); if ($written === false || $written === 0) { echo " FEHLER: Konnte Request nicht senden ($method).\n"; return false; } // Auf Antwort warten, Callbacks ueberspringen for ($attempt = 0; $attempt < 30; $attempt++) { $pkt = readPacket($fp); if ($pkt === false) { echo " FEHLER: Keine Antwort fuer $method.\n"; return false; } // Callback? (Handle < 0x80000000) -> ueberspringen if ($pkt["handle"] < 0x80000000) { continue; } // Response gefunden return $pkt["body"]; } echo " FEHLER: Zu viele Callbacks, keine Antwort fuer $method.\n"; return false; } // 1. Authenticate echo " Authentifiziere als SuperAdmin...\n"; $resp = gbxQuery($fp, $reqhandle, "Authenticate", ["SuperAdmin", $password]); if ($resp === false || strpos($resp, "1") === false) { echo " FEHLER: Authentifizierung fehlgeschlagen.\n"; if ($resp) echo " Antwort: " . substr(trim($resp), 0, 500) . "\n"; fclose($fp); exit(1); } echo " Authentifizierung erfolgreich.\n"; // 2. EnableCallbacks deaktivieren (weniger Rauschen) gbxQuery($fp, $reqhandle, "EnableCallbacks", [false]); // 3. SetForcedMods(override=true, mods=[{Env, Url}, ...]) // Debug: Zeige das XML das wir senden $setXml = "" . "SetForcedMods" . "" . encodeVal(true) . "" . "" . encodeVal($mods) . "" . ""; echo " Debug SetForcedMods-XML:\n " . $setXml . "\n"; echo " Sende SetForcedMods (" . count($mods) . " Mod(s))...\n"; $resp = gbxQuery($fp, $reqhandle, "SetForcedMods", [true, $mods]); echo " SetForcedMods-Antwort: " . trim($resp) . "\n"; if ($resp !== false && strpos($resp, "1") !== false) { echo " SetForcedMods: OK\n"; } else { echo " FEHLER: SetForcedMods fehlgeschlagen.\n"; } // 4. GetForcedMods zur Verifikation (vollstaendige Antwort) echo " Verifiziere mit GetForcedMods...\n"; $resp = gbxQuery($fp, $reqhandle, "GetForcedMods", []); echo " GetForcedMods-Antwort:\n " . trim($resp) . "\n"; fclose($fp); ' "$XMLRPC_PORT" "$SA_PW_MODS" "$MODS_JSON" elif [ "$XMLRPC_READY" = "true" ]; then echo "==> Forced Mods: Keine FORCE_MOD_*-Variablen gesetzt. Ueberspringe." fi # Auf TrackmaniaServer warten (Hauptprozess) wait $TM_PID