2using System.Collections.Generic;
3using System.Diagnostics;
5using System.Net.Http.Headers;
9using System.Threading.Tasks;
11using Microsoft.Extensions.Logging;
91 Task installationTask,
93 : base(installationIOManager)
98 ServerExePath = dotnetPath ??
throw new ArgumentNullException(nameof(dotnetPath));
103 InstallationTask = installationTask ??
throw new ArgumentNullException(nameof(installationTask));
104 Version = version ??
throw new ArgumentNullException(nameof(version));
107 throw new ArgumentException($
"Invalid EngineType: {version.Engine.Value}", nameof(version));
113 IReadOnlyDictionary<string, string> parameters,
117 ArgumentNullException.ThrowIfNull(dmbProvider);
118 ArgumentNullException.ThrowIfNull(parameters);
119 ArgumentNullException.ThrowIfNull(launchParameters);
122 throw new ArgumentException($
"parameters must have \"{DMApiConstants.ParamAccessIdentifier}\" set!", nameof(parameters));
126 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}\"";
133 if (String.IsNullOrWhiteSpace(additionalArguments))
134 additionalArguments = String.Empty;
136 additionalArguments = $
"{additionalArguments.Trim()} ";
138 return $
"{compilerDllPath} --suppress-unimplemented --notices-enabled {additionalArguments}\"{dmePath ?? throw new ArgumentNullException(nameof(dmePath))}\"";
145 string accessIdentifier,
147 CancellationToken cancellationToken)
149 ArgumentNullException.ThrowIfNull(logger);
151 const int MaximumTerminationSeconds = 5;
153 logger.LogTrace(
"Attempting Robust.Server graceful exit (Timeout: {seconds}s)...", MaximumTerminationSeconds);
154 var timeout = asyncDelayer.Delay(TimeSpan.FromSeconds(MaximumTerminationSeconds), cancellationToken).AsTask();
156 if (lifetime.IsCompleted)
157 logger.LogTrace(
"Robust.Server already exited");
159 var stopwatch = Stopwatch.StartNew();
162 using var httpClient = httpClientFactory.CreateClient();
163 using var request =
new HttpRequestMessage();
164 request.Headers.Add(
"WatchdogToken", accessIdentifier);
165 request.RequestUri =
new Uri($
"http://localhost:{port}/shutdown");
166 request.Content =
new StringContent(
167 "{\"Reason\":\"TGS session termination\"}",
169 new MediaTypeHeaderValue(MediaTypeNames.Application.Json));
170 request.Method = HttpMethod.Post;
172 var responseTask = httpClient.SendAsync(request, HttpCompletionOption.ResponseHeadersRead, cancellationToken);
175 await Task.WhenAny(timeout, lifetime, responseTask);
176 if (responseTask.IsCompleted)
178 using var response = await responseTask;
179 if (response.IsSuccessStatusCode)
181 logger.LogDebug(
"Robust.Server responded to the shutdown command successfully ({requestMs}ms). Waiting for exit...", stopwatch.ElapsedMilliseconds);
182 await Task.WhenAny(timeout, lifetime);
186 if (!lifetime.IsCompleted)
187 logger.LogWarning(
"Robust.Server graceful exit timed out!");
189 catch (
Exception ex) when (ex is not OperationCanceledException)
191 logger.LogDebug(ex,
"Unable to send graceful exit request to Robust.Server watchdog API!");
194 if (lifetime.IsCompleted)
196 logger.LogTrace(
"Robust.Server exited without termination");
202 logger.LogTrace(
"Robust.Server graceful shutdown attempt took {totalMs}ms", stopwatch.ElapsedMilliseconds);
205 await base.StopServerProcess(logger, process, accessIdentifier, port, cancellationToken);
Information about an engine installation.
EngineType? Engine
The EngineType.
Launch settings for DreamDaemon.
static string EncodeParameters(IReadOnlyDictionary< string, string > parameters, DreamDaemonLaunchParameters launchParameters)
Encode given parameters for passing as world.params on the command line.
Implementation of IEngineInstallation for EngineType.OpenDream.
override string CompilerExePath
The full path to the dm/DreamMaker executable.
readonly IAbstractHttpClientFactory httpClientFactory
The IAbstractHttpClientFactory for the OpenDreamInstallation.
readonly IAsyncDelayer asyncDelayer
The IAsyncDelayer for the OpenDreamInstallation.
readonly string compilerDllPath
Path to the DMCompiler.dll.
OpenDreamInstallation(IIOManager installationIOManager, IAsyncDelayer asyncDelayer, IAbstractHttpClientFactory httpClientFactory, string dotnetPath, string serverDllPath, string compilerDllPath, Task installationTask, EngineVersion version)
Initializes a new instance of the OpenDreamInstallation class.
override bool PromptsForNetworkAccess
If ServerExePath may create network prompts.
override async ValueTask StopServerProcess(ILogger logger, IProcess process, string accessIdentifier, ushort port, CancellationToken cancellationToken)
Kills a given engine server process .A ValueTask representing the running operation.
override string FormatCompilerArguments(string dmePath, string? additionalArguments)
Return the command line arguments for compiling a given dmePath if compilation is necessary....
override string FormatServerArguments(IDmbProvider dmbProvider, IReadOnlyDictionary< string, string > parameters, DreamDaemonLaunchParameters launchParameters, string? logFilePath)
Return the command line arguments for launching with given launchParameters .The formatted arguments ...
override bool HasStandardOutput
If ServerExePath supports being run as a command-line application and outputs log information to be c...
override EngineVersion Version
The EngineVersion of the IEngineInstallation.
override string ServerExePath
The full path to the game server executable.
override bool PreferFileLogging
If HasStandardOutput is set, this indicates that the engine server has good file logging that should ...
override bool UseDotnetDump
If dotnet-dump should be used to create process dumps for this installation.
override Task InstallationTask
The Task that completes when the BYOND version finished installing.
readonly string serverDllPath
Path to the Robust.Server.dll.
Constants used for communication with the DMAPI.
const string ParamAccessIdentifier
Identifies the DMApiParameters.AccessIdentifier for the session.
Provides absolute paths to the latest compiled .dmbs.
Interface for using filesystems.
Task< int?> Lifetime
The Task<TResult> resulting in the exit code of the process or null if the process was detached.
Abstraction over a global::System.Diagnostics.Process.
For waiting asynchronously.
EngineType
The type of engine the codebase is using.