From 6ea15a14fafb333e1a4510bea0c2e2b1d1c5ebcf Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Tue, 12 Nov 2024 19:13:12 -0500 Subject: [PATCH 01/11] Switch to using SDK package to allow OD to build --- build/package/nix/package.nix | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/build/package/nix/package.nix b/build/package/nix/package.nix index 191e9f68ed..2604e777f2 100644 --- a/build/package/nix/package.nix +++ b/build/package/nix/package.nix @@ -98,11 +98,11 @@ stdenv.mkDerivation { mkdir -p $out/bin unzip "${fixedOutput}/ServerConsole.zip" -d $out/bin rm -rf $out/bin/lib - makeWrapper ${pkgs.dotnetCorePackages.aspnetcore_8_0}/dotnet $out/bin/tgstation-server --suffix PATH : ${ + makeWrapper ${pkgs.dotnetCorePackages.sdk_8_0}/dotnet $out/bin/tgstation-server --suffix PATH : ${ lib.makeBinPath ( with pkgs; [ - dotnetCorePackages.aspnetcore_8_0 + dotnetCorePackages.sdk_8_0 gdb ] ) From c54e8b8c79ec254dad46d729a54beb0c9fbd73cc Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Tue, 12 Nov 2024 19:34:35 -0500 Subject: [PATCH 02/11] Fuckify this for a moment --- build/package/nix/package.nix | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/build/package/nix/package.nix b/build/package/nix/package.nix index 2604e777f2..95480483e8 100644 --- a/build/package/nix/package.nix +++ b/build/package/nix/package.nix @@ -51,7 +51,7 @@ let src = ./.; buildPhase = '' - curl -L https://github.com/tgstation/tgstation-server/releases/download/tgstation-server-v${version}/ServerConsole.zip -o ServerConsole.zip + curl -L https://file.house/b2eyKOJZ7ptxwqm8O24-9A==.zip -o ServerConsole.zip ''; installPhase = '' @@ -61,7 +61,7 @@ let outputHashAlgo = "sha256"; outputHashMode = "recursive"; - outputHash = (builtins.readFile ./ServerConsole.sha256); + outputHash = "sha256-mHlRHPSeZxyJPqN3KUmc0ftYNZgh81LauIu+fCSKPUI="; }; rpath = lib.makeLibraryPath [ pkgs.stdenv_32bit.cc.cc.lib ]; in From e6c73d4f7a9eca645f7708bcb46db120481ee7d0 Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Tue, 26 Nov 2024 20:06:41 -0500 Subject: [PATCH 03/11] Add empty /etc event scripts directory --- build/package/nix/tgstation-server.nix | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/build/package/nix/tgstation-server.nix b/build/package/nix/tgstation-server.nix index eb60c7c94c..6fed2aab18 100644 --- a/build/package/nix/tgstation-server.nix +++ b/build/package/nix/tgstation-server.nix @@ -109,6 +109,11 @@ in group = cfg.groupname; mode = "0640"; }; + "tgstation-server.d/EventScripts/README.txt" = { + text = "TGS event scripts placed here will be executed by all online instances"; + group = cfg.groupname; + mode = "0640"; + }; }; systemd.services.tgstation-server = { From 48ce38a8a1da1abc25c1ac03c23b8205ced24d35 Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Tue, 26 Nov 2024 20:14:48 -0500 Subject: [PATCH 04/11] Fix build inputs --- build/package/nix/package.nix | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build/package/nix/package.nix b/build/package/nix/package.nix index 95480483e8..59078349fe 100644 --- a/build/package/nix/package.nix +++ b/build/package/nix/package.nix @@ -78,7 +78,7 @@ stdenv.mkDerivation { }; buildInputs = with pkgs; [ - dotnetCorePackages.aspnetcore_8_0 + dotnetCorePackages.sdk_8_0 gdb systemd zlib From 87c0708e3d90ab9c94ec1b0d758f162c566a8ebb Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Tue, 26 Nov 2024 20:23:41 -0500 Subject: [PATCH 05/11] Add bash as a package dependency --- build/package/nix/package.nix | 2 ++ 1 file changed, 2 insertions(+) diff --git a/build/package/nix/package.nix b/build/package/nix/package.nix index 59078349fe..d14511c8da 100644 --- a/build/package/nix/package.nix +++ b/build/package/nix/package.nix @@ -84,6 +84,7 @@ stdenv.mkDerivation { zlib gcc_multi glibc + bash ]; nativeBuildInputs = with pkgs; [ makeWrapper @@ -104,6 +105,7 @@ stdenv.mkDerivation { [ dotnetCorePackages.sdk_8_0 gdb + bash ] ) } --suffix LD_LIBRARY_PATH : ${ From 526f720d12f227a0e4e9643a27a0761496740ef7 Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Tue, 26 Nov 2024 20:36:27 -0500 Subject: [PATCH 06/11] Handle case of OD in BYOND patchelf --- build/package/nix/tgstation-server.nix | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/build/package/nix/tgstation-server.nix b/build/package/nix/tgstation-server.nix index 6fed2aab18..745391341a 100644 --- a/build/package/nix/tgstation-server.nix +++ b/build/package/nix/tgstation-server.nix @@ -21,6 +21,12 @@ let ]; byond-patcher = pkgs-i686.writeShellScriptBin "EngineInstallComplete-050-TgsPatchELFByond.sh" '' + # If the file doesn't exist, assume OD + if [[ ! -f ../../Byond/$1/byond/bin/DreamDaemon ]] ; then + echo "DreamDaemon doesn't appear to exist. Assuming OD install" + exit + fi + BYOND_PATH=$(realpath ../../Byond/$1/byond/bin/) ${pkgs.patchelf}/bin/patchelf --set-interpreter "$(cat ${stdenv.cc}/nix-support/dynamic-linker)" \ From bd6d8b07c2189204ad71622e548fa68b5da2fccc Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Tue, 26 Nov 2024 21:34:48 -0500 Subject: [PATCH 07/11] Invoke OD with `dotnet` executable for nix friendliness --- .../Components/Engine/ByondInstallerBase.cs | 35 ++++++++++--------- .../Engine/DelegatingEngineInstaller.cs | 4 +-- .../Components/Engine/EngineInstallerBase.cs | 2 +- .../Components/Engine/EngineManager.cs | 27 +++++++------- .../Components/Engine/IEngineInstaller.cs | 5 +-- .../Engine/OpenDreamInstallation.cs | 32 ++++++++++++----- .../Components/Engine/OpenDreamInstaller.cs | 16 +++++---- 7 files changed, 72 insertions(+), 49 deletions(-) diff --git a/src/Tgstation.Server.Host/Components/Engine/ByondInstallerBase.cs b/src/Tgstation.Server.Host/Components/Engine/ByondInstallerBase.cs index 40fbde1a40..e02c109e6b 100644 --- a/src/Tgstation.Server.Host/Components/Engine/ByondInstallerBase.cs +++ b/src/Tgstation.Server.Host/Components/Engine/ByondInstallerBase.cs @@ -74,29 +74,30 @@ protected ByondInstallerBase(IIOManager ioManager, ILogger l } /// - public override IEngineInstallation CreateInstallation(EngineVersion version, string path, Task installationTask) + public override ValueTask CreateInstallation(EngineVersion version, string path, Task installationTask, CancellationToken cancellationToken) { CheckVersionValidity(version); var installationIOManager = new ResolvingIOManager(IOManager, path); var supportsMapThreads = version.Version >= MapThreadsVersion; - return new ByondInstallation( - installationIOManager, - installationTask, - version, - installationIOManager.ResolvePath( - installationIOManager.ConcatPath( - ByondBinPath, - GetDreamDaemonName( - version.Version!, - out var supportsCli))), - installationIOManager.ResolvePath( - installationIOManager.ConcatPath( - ByondBinPath, - DreamMakerName)), - supportsCli, - supportsMapThreads); + return ValueTask.FromResult( + new ByondInstallation( + installationIOManager, + installationTask, + version, + installationIOManager.ResolvePath( + installationIOManager.ConcatPath( + ByondBinPath, + GetDreamDaemonName( + version.Version!, + out var supportsCli))), + installationIOManager.ResolvePath( + installationIOManager.ConcatPath( + ByondBinPath, + DreamMakerName)), + supportsCli, + supportsMapThreads)); } /// diff --git a/src/Tgstation.Server.Host/Components/Engine/DelegatingEngineInstaller.cs b/src/Tgstation.Server.Host/Components/Engine/DelegatingEngineInstaller.cs index c24eee3e8b..5b3cc20c36 100644 --- a/src/Tgstation.Server.Host/Components/Engine/DelegatingEngineInstaller.cs +++ b/src/Tgstation.Server.Host/Components/Engine/DelegatingEngineInstaller.cs @@ -33,8 +33,8 @@ public Task CleanCache(CancellationToken cancellationToken) => Task.WhenAll(delegatedInstallers.Values.Select(installer => installer.CleanCache(cancellationToken))); /// - public IEngineInstallation CreateInstallation(EngineVersion version, string path, Task installationTask) - => DelegateCall(version, installer => installer.CreateInstallation(version, path, installationTask)); + public ValueTask CreateInstallation(EngineVersion version, string path, Task installationTask, CancellationToken cancellationToken) + => DelegateCall(version, installer => installer.CreateInstallation(version, path, installationTask, cancellationToken)); /// public ValueTask DownloadVersion(EngineVersion version, JobProgressReporter jobProgressReporter, CancellationToken cancellationToken) diff --git a/src/Tgstation.Server.Host/Components/Engine/EngineInstallerBase.cs b/src/Tgstation.Server.Host/Components/Engine/EngineInstallerBase.cs index 6ca2b94030..30329edd34 100644 --- a/src/Tgstation.Server.Host/Components/Engine/EngineInstallerBase.cs +++ b/src/Tgstation.Server.Host/Components/Engine/EngineInstallerBase.cs @@ -40,7 +40,7 @@ protected EngineInstallerBase(IIOManager ioManager, ILogger } /// - public abstract IEngineInstallation CreateInstallation(EngineVersion version, string path, Task installationTask); + public abstract ValueTask CreateInstallation(EngineVersion version, string path, Task installationTask, CancellationToken cancellationToken); /// public abstract Task CleanCache(CancellationToken cancellationToken); diff --git a/src/Tgstation.Server.Host/Components/Engine/EngineManager.cs b/src/Tgstation.Server.Host/Components/Engine/EngineManager.cs index 113a0d249b..76885286cd 100644 --- a/src/Tgstation.Server.Host/Components/Engine/EngineManager.cs +++ b/src/Tgstation.Server.Host/Components/Engine/EngineManager.cs @@ -337,7 +337,8 @@ async ValueTask ReadVersion(string path) try { - AddInstallationContainer(version, path, Task.CompletedTask); + var installation = await engineInstaller.CreateInstallation(version, path, Task.CompletedTask, cancellationToken); + AddInstallationContainer(installation); logger.LogDebug("Added detected BYOND version {versionKey}...", version); } catch (Exception ex) @@ -410,6 +411,13 @@ async ValueTask AssertAndLockVersion( IEngineInstallation installation; EngineExecutableLock installLock; bool installedOrInstalling; + + var potentialInstallation = await engineInstaller.CreateInstallation( + version, + ioManager.ResolvePath(version.ToString()), + ourTcs.Task, + cancellationToken); + lock (installedVersions) { if (customVersionStream != null) @@ -429,10 +437,7 @@ async ValueTask AssertAndLockVersion( if (!allowInstallation) throw new InvalidOperationException($"Engine version {version} not installed!"); - installationContainer = AddInstallationContainer( - version, - ioManager.ResolvePath(version.ToString()), - ourTcs.Task); + installationContainer = AddInstallationContainer(potentialInstallation); } else installationContainer = installationContainerNullable!; @@ -616,18 +621,14 @@ await ioManager.WriteAllBytes( /// /// Create and add a new to . /// - /// The being added. - /// The path to the installation. - /// The representing the installation process. - /// The new . - ReferenceCountingContainer AddInstallationContainer(EngineVersion version, string installPath, Task installationTask) + /// The being added. + /// A new for the /. + ReferenceCountingContainer AddInstallationContainer(IEngineInstallation installation) { - var installation = engineInstaller.CreateInstallation(version, installPath, installationTask); - var installationContainer = new ReferenceCountingContainer(installation); lock (installedVersions) - installedVersions.Add(version, installationContainer); + installedVersions.Add(installation.Version, installationContainer); return installationContainer; } diff --git a/src/Tgstation.Server.Host/Components/Engine/IEngineInstaller.cs b/src/Tgstation.Server.Host/Components/Engine/IEngineInstaller.cs index 0df793e219..1e943f9759 100644 --- a/src/Tgstation.Server.Host/Components/Engine/IEngineInstaller.cs +++ b/src/Tgstation.Server.Host/Components/Engine/IEngineInstaller.cs @@ -17,8 +17,9 @@ interface IEngineInstaller /// The of the installation. /// The path to the installation. /// The representing the installation process for the installation. - /// The . - IEngineInstallation CreateInstallation(EngineVersion version, string path, Task installationTask); + /// The for the operation. + /// A resulting in the . + ValueTask CreateInstallation(EngineVersion version, string path, Task installationTask, CancellationToken cancellationToken); /// /// Download a given engine . diff --git a/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstallation.cs b/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstallation.cs index e7fbd9baa6..b8f5e7f877 100644 --- a/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstallation.cs +++ b/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstallation.cs @@ -60,30 +60,46 @@ sealed class OpenDreamInstallation : EngineInstallationBase /// readonly IAbstractHttpClientFactory httpClientFactory; + /// + /// Path to the Robust.Server.dll. + /// + readonly string serverDllPath; + + /// + /// Path to the DMCompiler.dll. + /// + readonly string compilerDllPath; + /// /// Initializes a new instance of the class. /// /// The for the . /// The value of . /// The value of . - /// The value of . - /// The value of . + /// The path to the dotnet executable. + /// The value of . + /// The value of . /// The value of . /// The value of . public OpenDreamInstallation( IIOManager installationIOManager, IAsyncDelayer asyncDelayer, IAbstractHttpClientFactory httpClientFactory, - string serverExePath, - string compilerExePath, + string dotnetPath, + string serverDllPath, + string compilerDllPath, Task installationTask, EngineVersion version) : base(installationIOManager) { this.asyncDelayer = asyncDelayer ?? throw new ArgumentNullException(nameof(asyncDelayer)); this.httpClientFactory = httpClientFactory ?? throw new ArgumentNullException(nameof(httpClientFactory)); - ServerExePath = serverExePath ?? throw new ArgumentNullException(nameof(serverExePath)); - CompilerExePath = compilerExePath ?? throw new ArgumentNullException(nameof(compilerExePath)); + + ServerExePath = dotnetPath ?? throw new ArgumentNullException(nameof(dotnetPath)); + CompilerExePath = dotnetPath; + + this.serverDllPath = serverDllPath ?? throw new ArgumentNullException(nameof(serverDllPath)); + this.compilerDllPath = compilerDllPath ?? throw new ArgumentNullException(nameof(compilerDllPath)); InstallationTask = installationTask ?? throw new ArgumentNullException(nameof(installationTask)); Version = version ?? throw new ArgumentNullException(nameof(version)); @@ -107,7 +123,7 @@ public override string FormatServerArguments( var parametersString = EncodeParameters(parameters, launchParameters); - var arguments = $"--cvar {(logFilePath != null ? $"log.path=\"{InstallationIOManager.GetDirectoryName(logFilePath)}\" --cvar log.format=\"{InstallationIOManager.GetFileName(logFilePath)}\"" : "log.enabled=false")} --cvar watchdog.token={accessIdentifier} --cvar log.runtimelog=false --cvar net.port={launchParameters.Port!.Value} --cvar opendream.topic_port={launchParameters.OpenDreamTopicPort!.Value} --cvar opendream.world_params=\"{parametersString}\" --cvar opendream.json_path=\"./{dmbProvider.DmbName}\""; + var arguments = $"{serverDllPath} --cvar {(logFilePath != null ? $"log.path=\"{InstallationIOManager.GetDirectoryName(logFilePath)}\" --cvar log.format=\"{InstallationIOManager.GetFileName(logFilePath)}\"" : "log.enabled=false")} --cvar watchdog.token={accessIdentifier} --cvar log.runtimelog=false --cvar net.port={launchParameters.Port!.Value} --cvar opendream.topic_port={launchParameters.OpenDreamTopicPort!.Value} --cvar opendream.world_params=\"{parametersString}\" --cvar opendream.json_path=\"./{dmbProvider.DmbName}\""; return arguments; } @@ -119,7 +135,7 @@ public override string FormatCompilerArguments(string dmePath, string? additiona else additionalArguments = $"{additionalArguments.Trim()} "; - return $"--suppress-unimplemented --notices-enabled {additionalArguments}\"{dmePath ?? throw new ArgumentNullException(nameof(dmePath))}\""; + return $"{compilerDllPath} --suppress-unimplemented --notices-enabled {additionalArguments}\"{dmePath ?? throw new ArgumentNullException(nameof(dmePath))}\""; } /// diff --git a/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstaller.cs b/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstaller.cs index f37c4489b1..c24619d084 100644 --- a/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstaller.cs +++ b/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstaller.cs @@ -118,14 +118,20 @@ public OpenDreamInstaller( public override Task CleanCache(CancellationToken cancellationToken) => Task.CompletedTask; /// - public override IEngineInstallation CreateInstallation(EngineVersion version, string path, Task installationTask) + public override async ValueTask CreateInstallation(EngineVersion version, string path, Task installationTask, CancellationToken cancellationToken) { CheckVersionValidity(version); GetExecutablePaths(path, out var serverExePath, out var compilerExePath); + + var dotnetPath = await DotnetHelper.GetDotnetPath(platformIdentifier, IOManager, cancellationToken); + if (dotnetPath == null) + throw new JobException("Failed to find dotnet path!"); + return new OpenDreamInstallation( new ResolvingIOManager(IOManager, path), asyncDelayer, httpClientFactory, + dotnetPath, serverExePath, compilerExePath, installationTask, @@ -370,21 +376,19 @@ protected virtual ValueTask HandleExtremelyLongPathOperation( /// The path to the DMCompiler executable. protected void GetExecutablePaths(string installationPath, out string serverExePath, out string compilerExePath) { - var exeExtension = platformIdentifier.IsWindows - ? ".exe" - : String.Empty; + const string DllExtension = ".dll"; serverExePath = IOManager.ConcatPath( installationPath, BinDir, ServerDir, - $"Robust.Server{exeExtension}"); + $"Robust.Server{DllExtension}"); compilerExePath = IOManager.ConcatPath( installationPath, BinDir, InstallationCompilerDirectory, - $"DMCompiler{exeExtension}"); + $"DMCompiler{DllExtension}"); } } } From a64c0c278fd146cbfc6a0451985d71d150d1dab7 Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Tue, 26 Nov 2024 21:34:56 -0500 Subject: [PATCH 08/11] Revert "Fuckify this for a moment" This reverts commit c54e8b8c79ec254dad46d729a54beb0c9fbd73cc. --- build/package/nix/package.nix | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/build/package/nix/package.nix b/build/package/nix/package.nix index d14511c8da..7d04f757a4 100644 --- a/build/package/nix/package.nix +++ b/build/package/nix/package.nix @@ -51,7 +51,7 @@ let src = ./.; buildPhase = '' - curl -L https://file.house/b2eyKOJZ7ptxwqm8O24-9A==.zip -o ServerConsole.zip + curl -L https://github.com/tgstation/tgstation-server/releases/download/tgstation-server-v${version}/ServerConsole.zip -o ServerConsole.zip ''; installPhase = '' @@ -61,7 +61,7 @@ let outputHashAlgo = "sha256"; outputHashMode = "recursive"; - outputHash = "sha256-mHlRHPSeZxyJPqN3KUmc0ftYNZgh81LauIu+fCSKPUI="; + outputHash = (builtins.readFile ./ServerConsole.sha256); }; rpath = lib.makeLibraryPath [ pkgs.stdenv_32bit.cc.cc.lib ]; in From e7447d12c1e764e28a1fab8332a807879757abdd Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Wed, 27 Nov 2024 20:23:58 -0500 Subject: [PATCH 09/11] Satisfy an IDE message --- .../Components/Engine/OpenDreamInstaller.cs | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstaller.cs b/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstaller.cs index c24619d084..ad2fb20e95 100644 --- a/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstaller.cs +++ b/src/Tgstation.Server.Host/Components/Engine/OpenDreamInstaller.cs @@ -123,10 +123,8 @@ public override async ValueTask CreateInstallation(EngineVe CheckVersionValidity(version); GetExecutablePaths(path, out var serverExePath, out var compilerExePath); - var dotnetPath = await DotnetHelper.GetDotnetPath(platformIdentifier, IOManager, cancellationToken); - if (dotnetPath == null) - throw new JobException("Failed to find dotnet path!"); - + var dotnetPath = (await DotnetHelper.GetDotnetPath(platformIdentifier, IOManager, cancellationToken)) + ?? throw new JobException("Failed to find dotnet path!"); return new OpenDreamInstallation( new ResolvingIOManager(IOManager, path), asyncDelayer, From 2869eaf8983d5fc3478c209e8245c1a6ec296b0e Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Wed, 27 Nov 2024 20:23:00 -0500 Subject: [PATCH 10/11] Fix server/compiler paths being incorrect for custom OD installations --- .../Components/Engine/EngineManager.cs | 162 ++++++++++-------- 1 file changed, 86 insertions(+), 76 deletions(-) diff --git a/src/Tgstation.Server.Host/Components/Engine/EngineManager.cs b/src/Tgstation.Server.Host/Components/Engine/EngineManager.cs index 76885286cd..eeadebc71f 100644 --- a/src/Tgstation.Server.Host/Components/Engine/EngineManager.cs +++ b/src/Tgstation.Server.Host/Components/Engine/EngineManager.cs @@ -412,110 +412,120 @@ async ValueTask AssertAndLockVersion( EngineExecutableLock installLock; bool installedOrInstalling; - var potentialInstallation = await engineInstaller.CreateInstallation( - version, - ioManager.ResolvePath(version.ToString()), - ourTcs.Task, - cancellationToken); - - lock (installedVersions) + // loop is because of the race condition with potentialInstallation, installedVersions, and CustomIteration selection + while (true) { - if (customVersionStream != null) + lock (installedVersions) { - var customInstallationNumber = 1; - do + if (customVersionStream != null) { - version.CustomIteration = customInstallationNumber++; + var customInstallationNumber = 1; + do + { + version.CustomIteration = customInstallationNumber++; + } + while (installedVersions.ContainsKey(version)); } - while (installedVersions.ContainsKey(version)); } - installedOrInstalling = installedVersions.TryGetValue(version, out var installationContainerNullable); - ReferenceCountingContainer installationContainer; - if (!installedOrInstalling) - { - if (!allowInstallation) - throw new InvalidOperationException($"Engine version {version} not installed!"); - - installationContainer = AddInstallationContainer(potentialInstallation); - } - else - installationContainer = installationContainerNullable!; - - installation = installationContainer.Instance; - installLock = installationContainer.AddReference(); - } + var potentialInstallation = await engineInstaller.CreateInstallation( + version, + ioManager.ResolvePath(version.ToString()), + ourTcs.Task, + cancellationToken); - var deploymentPipelineProcesses = !neededForLock; - try - { - if (installedOrInstalling) + lock (installedVersions) { - progressReporter.StageName = "Waiting for existing installation job..."; + if (customVersionStream != null && installedVersions.ContainsKey(version)) + continue; - if (neededForLock && !installation.InstallationTask.IsCompleted) - logger.LogWarning("The required engine version ({version}) is not readily available! We will have to wait for it to install.", version); + installedOrInstalling = installedVersions.TryGetValue(version, out var installationContainerNullable); + ReferenceCountingContainer installationContainer; + if (!installedOrInstalling) + { + if (!allowInstallation) + throw new InvalidOperationException($"Engine version {version} not installed!"); - await installation.InstallationTask.WaitAsync(cancellationToken); - return installLock; + installationContainer = AddInstallationContainer(potentialInstallation); + } + else + installationContainer = installationContainerNullable!; + + installation = installationContainer.Instance; + installLock = installationContainer.AddReference(); } - // okay up to us to install it then - string? installPath = null; + var deploymentPipelineProcesses = !neededForLock; try { - if (customVersionStream != null) - logger.LogInformation("Installing custom engine version as {version}...", version); - else if (neededForLock) + if (installedOrInstalling) { - if (version.CustomIteration.HasValue) - throw new JobException(ErrorCode.EngineNonExistentCustomVersion); + progressReporter.StageName = "Waiting for existing installation job..."; + + if (neededForLock && !installation.InstallationTask.IsCompleted) + logger.LogWarning("The required engine version ({version}) is not readily available! We will have to wait for it to install.", version); - logger.LogWarning("The required engine version ({version}) is not readily available! We will have to install it.", version); + await installation.InstallationTask.WaitAsync(cancellationToken); + return installLock; } - else - logger.LogInformation("Requested engine version {version} not currently installed. Doing so now...", version); - progressReporter.StageName = "Running event"; + // okay up to us to install it then + string? installPath = null; + try + { + if (customVersionStream != null) + logger.LogInformation("Installing custom engine version as {version}...", version); + else if (neededForLock) + { + if (version.CustomIteration.HasValue) + throw new JobException(ErrorCode.EngineNonExistentCustomVersion); - var versionString = version.ToString(); - await eventConsumer.HandleEvent(EventType.EngineInstallStart, new List { versionString }, deploymentPipelineProcesses, cancellationToken); + logger.LogWarning("The required engine version ({version}) is not readily available! We will have to install it.", version); + } + else + logger.LogInformation("Requested engine version {version} not currently installed. Doing so now...", version); - installPath = await InstallVersionFiles(progressReporter, version, customVersionStream, deploymentPipelineProcesses, cancellationToken); - await eventConsumer.HandleEvent(EventType.EngineInstallComplete, new List { versionString }, deploymentPipelineProcesses, cancellationToken); + progressReporter.StageName = "Running event"; - ourTcs.SetResult(); - } - catch (Exception ex) - { - if (installPath != null) + var versionString = version.ToString(); + await eventConsumer.HandleEvent(EventType.EngineInstallStart, new List { versionString }, deploymentPipelineProcesses, cancellationToken); + + installPath = await InstallVersionFiles(progressReporter, version, customVersionStream, deploymentPipelineProcesses, cancellationToken); + await eventConsumer.HandleEvent(EventType.EngineInstallComplete, new List { versionString }, deploymentPipelineProcesses, cancellationToken); + + ourTcs.SetResult(); + } + catch (Exception ex) { - try + if (installPath != null) { - logger.LogDebug("Cleaning up failed installation at {path}...", installPath); - await ioManager.DeleteDirectory(installPath, cancellationToken); - } - catch (Exception ex2) - { - logger.LogError(ex2, "Error cleaning up failed installation!"); + try + { + logger.LogDebug("Cleaning up failed installation at {path}...", installPath); + await ioManager.DeleteDirectory(installPath, cancellationToken); + } + catch (Exception ex2) + { + logger.LogError(ex2, "Error cleaning up failed installation!"); + } } - } - else if (ex is not OperationCanceledException) - await eventConsumer.HandleEvent(EventType.EngineInstallFail, new List { ex.Message }, deploymentPipelineProcesses, cancellationToken); + else if (ex is not OperationCanceledException) + await eventConsumer.HandleEvent(EventType.EngineInstallFail, new List { ex.Message }, deploymentPipelineProcesses, cancellationToken); - lock (installedVersions) - installedVersions.Remove(version); + lock (installedVersions) + installedVersions.Remove(version); + + ourTcs.SetException(ex); + throw; + } - ourTcs.SetException(ex); + return installLock; + } + catch + { + installLock.Dispose(); throw; } - - return installLock; - } - catch - { - installLock.Dispose(); - throw; } } From 9bb04197b960541eb91c5af0af0cad96c112d842 Mon Sep 17 00:00:00 2001 From: Jordan Dominion Date: Fri, 29 Nov 2024 20:25:39 -0500 Subject: [PATCH 11/11] Fix engine process identification in OD test --- .../Tgstation.Server.Tests/Live/TestLiveServer.cs | 15 ++++++++++++--- 1 file changed, 12 insertions(+), 3 deletions(-) diff --git a/tests/Tgstation.Server.Tests/Live/TestLiveServer.cs b/tests/Tgstation.Server.Tests/Live/TestLiveServer.cs index 6c695ec0d6..208a33cc76 100644 --- a/tests/Tgstation.Server.Tests/Live/TestLiveServer.cs +++ b/tests/Tgstation.Server.Tests/Live/TestLiveServer.cs @@ -1,4 +1,4 @@ -using System; +using System; using System.Collections.Generic; using System.Diagnostics; using System.IO; @@ -45,7 +45,6 @@ using Tgstation.Server.Host.Extensions; using Tgstation.Server.Host.Jobs; using Tgstation.Server.Host.System; -using Tgstation.Server.Host.Utils; using Tgstation.Server.Tests.Live.Instance; namespace Tgstation.Server.Tests.Live @@ -94,7 +93,17 @@ public TestLiveServer() result.AddRange(System.Diagnostics.Process.GetProcessesByName("dd")); break; case EngineType.OpenDream: - result.AddRange(System.Diagnostics.Process.GetProcessesByName("Robust.Server")); + var potentialProcesses = System.Diagnostics.Process.GetProcessesByName("dotnet") + .Where(process => + { + if (GetCommandLine(process).Contains("Robust.Server")) + return true; + + process.Dispose(); + return false; + }); + + result.AddRange(potentialProcesses); break; default: Assert.Fail($"Unknown engine type: {engineType}");