2026-02-03 13:32:17 +00:00
|
|
|
namespace AIStudio.Tools.Services;
|
|
|
|
|
|
|
|
|
|
public sealed partial class RustService
|
|
|
|
|
{
|
2026-06-02 15:22:59 +00:00
|
|
|
public async Task<TDatabaseInfo> GetDatabaseInfo<TDatabaseInfo>(
|
|
|
|
|
string databaseName,
|
|
|
|
|
string infoPath,
|
|
|
|
|
Func<string, TDatabaseInfo> unavailableFactory,
|
|
|
|
|
CancellationToken cancellationToken = default)
|
2026-02-03 13:32:17 +00:00
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
2026-05-19 06:24:22 +00:00
|
|
|
using var cts = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken);
|
|
|
|
|
cts.CancelAfter(TimeSpan.FromSeconds(45));
|
2026-06-02 15:22:59 +00:00
|
|
|
|
|
|
|
|
var databaseInfo = await this.http.GetFromJsonAsync<TDatabaseInfo>(infoPath, this.jsonRustSerializerOptions, cts.Token);
|
|
|
|
|
return databaseInfo ?? unavailableFactory("The database information response was empty.");
|
2026-05-19 06:24:22 +00:00
|
|
|
}
|
|
|
|
|
catch (OperationCanceledException) when (cancellationToken.IsCancellationRequested)
|
|
|
|
|
{
|
|
|
|
|
if(this.logger is not null)
|
2026-06-02 15:22:59 +00:00
|
|
|
this.logger.LogWarning("Fetching {DatabaseName} info from Rust service was cancelled by caller.", databaseName);
|
2026-05-19 06:24:22 +00:00
|
|
|
else
|
2026-06-02 15:22:59 +00:00
|
|
|
Console.WriteLine($"Fetching {databaseName} info from Rust service was cancelled by caller.");
|
|
|
|
|
|
|
|
|
|
return unavailableFactory("Operation cancelled by caller.");
|
2026-02-03 13:32:17 +00:00
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
if(this.logger is not null)
|
2026-06-02 15:22:59 +00:00
|
|
|
this.logger.LogError(e, "Error while fetching {DatabaseName} info from Rust service.", databaseName);
|
2026-02-03 13:32:17 +00:00
|
|
|
else
|
2026-06-02 15:22:59 +00:00
|
|
|
Console.WriteLine($"Error while fetching {databaseName} info from Rust service: '{e}'.");
|
|
|
|
|
|
|
|
|
|
return unavailableFactory(e.Message);
|
2026-02-03 13:32:17 +00:00
|
|
|
}
|
|
|
|
|
}
|
2026-06-02 15:22:59 +00:00
|
|
|
|
|
|
|
|
public async Task ExecuteDatabaseOperation<TRequest>(string databaseName, string path, TRequest request, CancellationToken cancellationToken = default)
|
|
|
|
|
{
|
|
|
|
|
using var cts = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken);
|
|
|
|
|
cts.CancelAfter(TimeSpan.FromMinutes(5));
|
|
|
|
|
|
|
|
|
|
using var response = await this.http.PostAsJsonAsync(path, request, this.jsonRustSerializerOptions, cts.Token);
|
|
|
|
|
response.EnsureSuccessStatusCode();
|
|
|
|
|
|
|
|
|
|
var operation = await response.Content.ReadFromJsonAsync<DatabaseOperationResponse>(this.jsonRustSerializerOptions, cts.Token);
|
|
|
|
|
if (operation is not { Success: true })
|
|
|
|
|
throw new InvalidOperationException(operation?.Issue ?? $"The {databaseName} operation failed.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private sealed record DatabaseOperationResponse(bool Success, string Issue);
|
|
|
|
|
}
|