2using System.Collections.Generic;
7using System.Threading.Tasks;
9using HotChocolate.Execution;
11using Microsoft.Extensions.DependencyInjection;
12using Microsoft.Extensions.Hosting;
13using Microsoft.Extensions.Logging;
14using Microsoft.Extensions.Options;
35#if WATCHDOG_FREE_RESTART
44 internal IHost? Host {
get;
private set; }
119 public async ValueTask
Run(CancellationToken cancellationToken)
123 using (var fsWatcher = updateDirectory !=
null ?
new FileSystemWatcher(updateDirectory) :
null)
125 if (fsWatcher !=
null)
132 fsWatcher.EnableRaisingEvents =
true;
139 logger = Host.Services.GetRequiredService<ILogger<Server>>();
142 using (cancellationToken.Register(() =>
logger.LogInformation(
"Server termination requested!")))
147 var generalConfigurationOptions = Host.Services.GetRequiredService<IOptions<GeneralConfiguration>>();
155 catch (OperationCanceledException ex)
157 logger.LogDebug(ex,
"Server run cancelled!");
182 ArgumentNullException.ThrowIfNull(updateExecutor);
183 ArgumentNullException.ThrowIfNull(newVersion);
188 throw new InvalidOperationException(
"Tried to start update when server was initialized without an updatePath set!");
190 var
logger = this.logger!;
191 logger.LogTrace(
"Begin ApplyUpdate...");
193 CancellationToken criticalCancellationToken;
198 logger.LogDebug(
"Aborted update due to concurrency conflict!");
203 throw new InvalidOperationException(
"Tried to update a non-running Server!");
209 async Task RunUpdate()
211 var updateExecutedSuccessfully =
false;
214 updateExecutedSuccessfully = await updateExecutor.
ExecuteUpdate(
updatePath, criticalCancellationToken, criticalCancellationToken);
216 catch (OperationCanceledException ex)
218 logger.LogDebug(ex,
"Update cancelled!");
223 logger.LogError(ex,
"Update errored!");
227 if (updateExecutedSuccessfully)
229 logger.LogTrace(
"Update complete!");
234 logger.LogTrace(
"Stopping host due to termination request...");
239 logger.LogTrace(
"Update failed!");
251 ArgumentNullException.ThrowIfNull(handler);
255 var
logger = this.logger!;
259 logger.LogTrace(
"Registering restart handler {handlerImplementationName}...", handler);
270 logger.LogWarning(
"Restart handler {handlerImplementationName} register after a shutdown had begun!", handler);
283 if (exception !=
null)
287 return ValueTask.CompletedTask;
298 var internalConfigurationOptions = services.GetRequiredService<IOptions<InternalConfiguration>>();
299 var apiDumpPath = internalConfigurationOptions.Value.DumpGraphQLApiPath;
300 if (String.IsNullOrWhiteSpace(apiDumpPath))
303 logger!.LogInformation(
"Dumping GraphQL API spec to {path} and exiting...", apiDumpPath);
306 var resolver = services.GetRequiredService<IRequestExecutorResolver>();
307 var executor = await resolver.GetRequestExecutorAsync(cancellationToken: cancellationToken);
308 var sdl = executor.Schema.Print();
310 var ioManager = services.GetRequiredService<
IIOManager>();
311 await ioManager.
WriteAllBytes(apiDumpPath, Encoding.UTF8.GetBytes(sdl), cancellationToken);
322 throw new InvalidOperationException(
"Server restarts are not supported");
325 throw new InvalidOperationException(
"Tried to control a non-running Server!");
337 if (otherException !=
null)
351 async ValueTask
RestartImpl(Version? newVersion,
Exception? exception,
bool requireWatchdog,
bool completeAsap)
356 bool isGracefulShutdown = !requireWatchdog && exception ==
null;
357 var
logger = this.logger!;
359 "Begin {restartType}...",
362 ?
"semi-graceful shutdown"
363 :
"graceful shutdown"
370 logger.LogTrace(
"Aborted restart due to concurrency conflict!");
378 if (exception ==
null)
380 var giveHandlersTimeToWaitAround = isGracefulShutdown && !completeAsap;
381 logger.LogInformation(
"Stopping server...");
382 using var cts =
new CancellationTokenSource(
383 TimeSpan.FromMinutes(
384 giveHandlersTimeToWaitAround
387 var cancellationToken = cts.Token;
390 ValueTask eventsTask;
395 x => x.HandleRestart(newVersion, giveHandlersTimeToWaitAround, cancellationToken))
398 logger.LogTrace(
"Joining restart handlers...");
401 catch (OperationCanceledException ex)
403 if (isGracefulShutdown)
404 logger.LogWarning(ex,
"Graceful shutdown timeout hit! Existing DreamDaemon processes will be terminated!");
408 "Restart timeout hit! Existing DreamDaemon processes will be lost and must be killed manually before being restarted with TGS!");
412 logger.LogError(e,
"Restart handlers error!");
426 logger?.LogTrace(
"FileSystemWatcher triggered.");
429 if (eventArgs.FullPath == Path.GetFullPath(
updatePath!) && File.Exists(eventArgs.FullPath))
431 logger?.LogInformation(
"Host watchdog appears to be requesting server termination!");
443 logger?.LogInformation(
"An update is in progress, we will wait for that to complete...");
453 logger!.LogDebug(
"Stopping host...");
Extension methods for the ValueTask and ValueTask<TResult> classes.
static async ValueTask WhenAll(IEnumerable< ValueTask > tasks)
Fully await a given list of tasks .
General configuration options.
uint ShutdownTimeoutMinutes
The timeout minutes for gracefully stopping the server.
uint RestartTimeoutMinutes
The timeout minutes for restarting the server.
bool WatchdogPresent
true if live updates are supported, false. TryStartUpdate(IServerUpdateExecutor, Version) and Restart...
IRestartRegistration RegisterForRestart(IRestartHandler handler)
Register a given handler to run before stopping the server for a restart.A new IRestartRegistration ...
ValueTask Die(Exception? exception)
Kill the server with a fatal exception.A Task representing the running operation.
bool UpdateInProgress
Whether or not the server is currently updating.
async ValueTask Run(CancellationToken cancellationToken)
Runs the IServer.A ValueTask representing the running operation.
bool TryStartUpdate(IServerUpdateExecutor updateExecutor, Version newVersion)
Attempt to update with a given updateExecutor .true if the update started successfully,...
Task? updateTask
The Task that is used for asynchronously updating the server.
async ValueTask RestartImpl(Version? newVersion, Exception? exception, bool requireWatchdog, bool completeAsap)
Implements Restart().
void CheckSanity(bool checkWatchdog)
Throws an InvalidOperationException if the IServerControl cannot be used.
ValueTask GracefulShutdown(bool detach)
Gracefully shutsdown the Host.A ValueTask representing the running operation.
readonly List< IRestartHandler > restartHandlers
The IRestartHandlers to run when the Server restarts.
CancellationTokenSource? cancellationTokenSource
The cancellationTokenSource for the Server.
ValueTask Restart()
Restarts the Host.A ValueTask representing the running operation.
bool terminateIfUpdateFails
If there is an update in progress and this flag is set, it should stop the server immediately if it f...
readonly? string updatePath
The absolute path to install updates to.
ILogger< Server >? logger
The ILogger for the Server.
void WatchForShutdownFileCreation(object sender, FileSystemEventArgs eventArgs)
Event handler for the updatePath's FileSystemWatcher. Triggers shutdown if requested by host watchdog...
readonly IHostBuilder hostBuilder
The IHostBuilder for the Server.
Exception? propagatedException
The Exception to propagate when the server terminates.
GeneralConfiguration? generalConfiguration
The GeneralConfiguration for the Server.
readonly object restartLock
lock object for certain restart related operations.
bool shutdownInProgress
If the server is being shut down or restarted.
async ValueTask< bool > DumpGraphQLSchemaIfRequested(IServiceProvider services, CancellationToken cancellationToken)
Checks if InternalConfiguration.DumpGraphQLApiPath is set and dumps the GraphQL API Schema to it if s...
Server(IHostBuilder hostBuilder, string? updatePath)
Initializes a new instance of the Server class.
bool RestartRequested
If the IServer should restart.
void StopServerImmediate()
Fires off the cancellationTokenSource without any checks, shutting down everything.
void CheckExceptionPropagation(Exception? otherException)
Re-throw propagatedException if it exists.
Runs a given disposeAction on Dispose.
Handler for server restarts.
Represents the lifetime of a IRestartHandler registration.
Represents a service that may take an updated Host assembly and run it, stopping the current assembly...
Executes server update operations.
ValueTask< bool > ExecuteUpdate(string updatePath, CancellationToken cancellationToken, CancellationToken criticalCancellationToken)
Executes a pending server update by extracting the new server to a given updatePath .
Interface for using filesystems.
ValueTask WriteAllBytes(string path, byte[] contents, CancellationToken cancellationToken)
Writes some contents to a file at path overwriting previous content.