mirror of
https://github.com/MindWorkAI/AI-Studio.git
synced 2026-02-12 03:41:38 +00:00
Added Rust failure detection to the .NET server (#637)
Some checks are pending
Build and Release / Prepare & create release (push) Blocked by required conditions
Build and Release / Read metadata (push) Waiting to run
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-apple-darwin, osx-arm64, macos-latest, aarch64-apple-darwin, dmg updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-pc-windows-msvc.exe, win-arm64, windows-latest, aarch64-pc-windows-msvc, nsis updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-unknown-linux-gnu, linux-arm64, ubuntu-22.04-arm, aarch64-unknown-linux-gnu, appimage deb updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-apple-darwin, osx-x64, macos-latest, x86_64-apple-darwin, dmg updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-pc-windows-msvc.exe, win-x64, windows-latest, x86_64-pc-windows-msvc, nsis updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-unknown-linux-gnu, linux-x64, ubuntu-22.04, x86_64-unknown-linux-gnu, appimage deb updater) (push) Blocked by required conditions
Build and Release / Publish release (push) Blocked by required conditions
Some checks are pending
Build and Release / Prepare & create release (push) Blocked by required conditions
Build and Release / Read metadata (push) Waiting to run
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-apple-darwin, osx-arm64, macos-latest, aarch64-apple-darwin, dmg updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-pc-windows-msvc.exe, win-arm64, windows-latest, aarch64-pc-windows-msvc, nsis updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-unknown-linux-gnu, linux-arm64, ubuntu-22.04-arm, aarch64-unknown-linux-gnu, appimage deb updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-apple-darwin, osx-x64, macos-latest, x86_64-apple-darwin, dmg updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-pc-windows-msvc.exe, win-x64, windows-latest, x86_64-pc-windows-msvc, nsis updater) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-unknown-linux-gnu, linux-x64, ubuntu-22.04, x86_64-unknown-linux-gnu, appimage deb updater) (push) Blocked by required conditions
Build and Release / Publish release (push) Blocked by required conditions
This commit is contained in:
parent
3763b1e4a4
commit
530b5f6cf8
@ -134,6 +134,7 @@ internal sealed class Program
|
|||||||
builder.Services.AddHostedService<TemporaryChatService>();
|
builder.Services.AddHostedService<TemporaryChatService>();
|
||||||
builder.Services.AddHostedService<EnterpriseEnvironmentService>();
|
builder.Services.AddHostedService<EnterpriseEnvironmentService>();
|
||||||
builder.Services.AddHostedService<GlobalShortcutService>();
|
builder.Services.AddHostedService<GlobalShortcutService>();
|
||||||
|
builder.Services.AddHostedService<RustAvailabilityMonitorService>();
|
||||||
|
|
||||||
// ReSharper disable AccessToDisposedClosure
|
// ReSharper disable AccessToDisposedClosure
|
||||||
builder.Services.AddHostedService<RustService>(_ => rust);
|
builder.Services.AddHostedService<RustService>(_ => rust);
|
||||||
|
|||||||
@ -15,6 +15,7 @@ public enum Event
|
|||||||
SHOW_WARNING,
|
SHOW_WARNING,
|
||||||
SHOW_SUCCESS,
|
SHOW_SUCCESS,
|
||||||
TAURI_EVENT_RECEIVED,
|
TAURI_EVENT_RECEIVED,
|
||||||
|
RUST_SERVICE_UNAVAILABLE,
|
||||||
|
|
||||||
// Update events:
|
// Update events:
|
||||||
USER_SEARCH_FOR_UPDATE,
|
USER_SEARCH_FOR_UPDATE,
|
||||||
|
|||||||
@ -42,6 +42,7 @@ public sealed class EnterpriseEnvironmentService(ILogger<EnterpriseEnvironmentSe
|
|||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
logger.LogError(e, "Failed to fetch the enterprise remove configuration ID from the Rust service.");
|
logger.LogError(e, "Failed to fetch the enterprise remove configuration ID from the Rust service.");
|
||||||
|
await MessageBus.INSTANCE.SendMessage(null, Event.RUST_SERVICE_UNAVAILABLE, "EnterpriseEnvRemoveConfigId failed");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -60,6 +61,7 @@ public sealed class EnterpriseEnvironmentService(ILogger<EnterpriseEnvironmentSe
|
|||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
logger.LogError(e, "Failed to fetch the enterprise configuration server URL from the Rust service.");
|
logger.LogError(e, "Failed to fetch the enterprise configuration server URL from the Rust service.");
|
||||||
|
await MessageBus.INSTANCE.SendMessage(null, Event.RUST_SERVICE_UNAVAILABLE, "EnterpriseEnvConfigServerUrl failed");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -71,6 +73,7 @@ public sealed class EnterpriseEnvironmentService(ILogger<EnterpriseEnvironmentSe
|
|||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
logger.LogError(e, "Failed to fetch the enterprise configuration ID from the Rust service.");
|
logger.LogError(e, "Failed to fetch the enterprise configuration ID from the Rust service.");
|
||||||
|
await MessageBus.INSTANCE.SendMessage(null, Event.RUST_SERVICE_UNAVAILABLE, "EnterpriseEnvConfigId failed");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -0,0 +1,111 @@
|
|||||||
|
using Microsoft.AspNetCore.Components;
|
||||||
|
|
||||||
|
namespace AIStudio.Tools.Services;
|
||||||
|
|
||||||
|
public sealed class RustAvailabilityMonitorService : BackgroundService, IMessageBusReceiver
|
||||||
|
{
|
||||||
|
private const int UNAVAILABLE_EVENT_THRESHOLD = 2;
|
||||||
|
|
||||||
|
private readonly ILogger<RustAvailabilityMonitorService> logger;
|
||||||
|
private readonly MessageBus messageBus;
|
||||||
|
private readonly RustService rustService;
|
||||||
|
private readonly IHostApplicationLifetime appLifetime;
|
||||||
|
|
||||||
|
private int rustUnavailableCount;
|
||||||
|
private int availabilityCheckTriggered;
|
||||||
|
|
||||||
|
// To prevent multiple shutdown triggers. We use int instead of bool for Interlocked operations.
|
||||||
|
private int shutdownTriggered;
|
||||||
|
|
||||||
|
public RustAvailabilityMonitorService(
|
||||||
|
ILogger<RustAvailabilityMonitorService> logger,
|
||||||
|
MessageBus messageBus,
|
||||||
|
RustService rustService,
|
||||||
|
IHostApplicationLifetime appLifetime)
|
||||||
|
{
|
||||||
|
this.logger = logger;
|
||||||
|
this.messageBus = messageBus;
|
||||||
|
this.rustService = rustService;
|
||||||
|
this.appLifetime = appLifetime;
|
||||||
|
|
||||||
|
this.messageBus.RegisterComponent(this);
|
||||||
|
this.ApplyFilters([], [Event.RUST_SERVICE_UNAVAILABLE]);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
||||||
|
{
|
||||||
|
this.logger.LogInformation("The Rust availability monitor service was initialized.");
|
||||||
|
await Task.Delay(Timeout.InfiniteTimeSpan, stoppingToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
public override async Task StopAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
this.messageBus.Unregister(this);
|
||||||
|
await base.StopAsync(cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Task ProcessMessage<T>(ComponentBase? sendingComponent, Event triggeredEvent, T? data)
|
||||||
|
{
|
||||||
|
if (triggeredEvent is not Event.RUST_SERVICE_UNAVAILABLE)
|
||||||
|
return Task.CompletedTask;
|
||||||
|
|
||||||
|
var reason = data switch
|
||||||
|
{
|
||||||
|
string s when !string.IsNullOrWhiteSpace(s) => s,
|
||||||
|
_ => "unknown reason",
|
||||||
|
};
|
||||||
|
|
||||||
|
// Thread-safe incrementation of the unavailable count and check against the threshold:
|
||||||
|
var numEvents = Interlocked.Increment(ref this.rustUnavailableCount);
|
||||||
|
|
||||||
|
// On the first event, trigger some Rust availability checks to confirm.
|
||||||
|
// Just fire and forget - we don't need to await this here.
|
||||||
|
if (numEvents == 1 && Interlocked.Exchange(ref this.availabilityCheckTriggered, 1) == 0)
|
||||||
|
{
|
||||||
|
//
|
||||||
|
// This is also useful to speed up the detection of Rust availability issues,
|
||||||
|
// as it triggers two immediate checks instead of waiting for the next scheduled check.
|
||||||
|
// Scheduled checks are typically every few minutes, which might be too long to wait
|
||||||
|
// in case of critical Rust service failures.
|
||||||
|
//
|
||||||
|
// On the other hand, we cannot kill the .NET server on the first failure, as it might
|
||||||
|
// be a transient issue.
|
||||||
|
//
|
||||||
|
|
||||||
|
_ = this.VerifyRustAvailability();
|
||||||
|
_ = this.VerifyRustAvailability();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (numEvents <= UNAVAILABLE_EVENT_THRESHOLD)
|
||||||
|
{
|
||||||
|
this.logger.LogWarning("Rust service unavailable (num repeats={NumRepeats}, threshold={Threshold}). Reason = '{Reason}'. Waiting for more occurrences before shutting down the server.", numEvents, UNAVAILABLE_EVENT_THRESHOLD, reason);
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure shutdown is only triggered once:
|
||||||
|
if (Interlocked.Exchange(ref this.shutdownTriggered, 1) != 0)
|
||||||
|
return Task.CompletedTask;
|
||||||
|
|
||||||
|
this.logger.LogError("Rust service unavailable (num repeats={NumRepeats}, threshold={Threshold}). Reason = '{Reason}'. Shutting down the server.", numEvents, UNAVAILABLE_EVENT_THRESHOLD, reason);
|
||||||
|
this.appLifetime.StopApplication();
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Task<TResult?> ProcessMessageWithResult<TPayload, TResult>(ComponentBase? sendingComponent, Event triggeredEvent, TPayload? data)
|
||||||
|
{
|
||||||
|
return Task.FromResult<TResult?>(default);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task VerifyRustAvailability()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await this.rustService.ReadUserLanguage();
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
this.logger.LogWarning(e, "Rust availability check failed.");
|
||||||
|
await this.messageBus.SendMessage(null, Event.RUST_SERVICE_UNAVAILABLE, "Rust availability check failed");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -62,6 +62,7 @@ public partial class RustService
|
|||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
this.logger!.LogError("Error while streaming Tauri events: {Message}", e.Message);
|
this.logger!.LogError("Error while streaming Tauri events: {Message}", e.Message);
|
||||||
|
await this.ReportRustServiceUnavailable("Tauri event stream error");
|
||||||
await Task.Delay(TimeSpan.FromSeconds(3), stopToken);
|
await Task.Delay(TimeSpan.FromSeconds(3), stopToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -32,6 +32,12 @@ public sealed partial class RustService
|
|||||||
}
|
}
|
||||||
catch
|
catch
|
||||||
{
|
{
|
||||||
|
//
|
||||||
|
// We don't expect this to ever happen because the HTTP client cannot raise exceptions in fire-and-forget mode.
|
||||||
|
// This is because we don't await the task, so any exceptions thrown during the HTTP request are not propagated
|
||||||
|
// back to the caller.
|
||||||
|
//
|
||||||
|
|
||||||
Console.WriteLine("Failed to send log event to Rust service.");
|
Console.WriteLine("Failed to send log event to Rust service.");
|
||||||
// Ignore errors to avoid log loops
|
// Ignore errors to avoid log loops
|
||||||
}
|
}
|
||||||
|
|||||||
@ -69,6 +69,8 @@ public sealed partial class RustService : BackgroundService
|
|||||||
this.encryptor = encryptionService;
|
this.encryptor = encryptionService;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Task ReportRustServiceUnavailable(string reason) => MessageBus.INSTANCE.SendMessage(null, Event.RUST_SERVICE_UNAVAILABLE, reason);
|
||||||
|
|
||||||
#region Overrides of BackgroundService
|
#region Overrides of BackgroundService
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@ -3,6 +3,7 @@
|
|||||||
- Added the current date and time to the system prompt for better context in conversations. Thanks Peer `peerschuett` for the contribution.
|
- Added the current date and time to the system prompt for better context in conversations. Thanks Peer `peerschuett` for the contribution.
|
||||||
- Added the ability to control the voice recording with transcription (in preview) by using a system-wide shortcut. The shortcut can be configured in the application settings or by using a configuration plugin. Thus, a uniform shortcut can be defined for an entire organization.
|
- Added the ability to control the voice recording with transcription (in preview) by using a system-wide shortcut. The shortcut can be configured in the application settings or by using a configuration plugin. Thus, a uniform shortcut can be defined for an entire organization.
|
||||||
- Added error handling for the context window overflow, which can occur with huge file attachments in chats or the document analysis assistant.
|
- Added error handling for the context window overflow, which can occur with huge file attachments in chats or the document analysis assistant.
|
||||||
|
- Added Rust failure detection to the .NET server. This is helpful in order to handle critical failures and shut down the application gracefully.
|
||||||
- Improved the error handling for model loading in provider dialogs (LLMs, embeddings, transcriptions).
|
- Improved the error handling for model loading in provider dialogs (LLMs, embeddings, transcriptions).
|
||||||
- Improved the microphone handling (transcription preview) so that all sound effects and the voice recording are processed without interruption.
|
- Improved the microphone handling (transcription preview) so that all sound effects and the voice recording are processed without interruption.
|
||||||
- Improved the handling of self-hosted providers in the configuration dialogs (LLMs, embeddings, and transcriptions) when the host cannot provide a list of models.
|
- Improved the handling of self-hosted providers in the configuration dialogs (LLMs, embeddings, and transcriptions) when the host cannot provide a list of models.
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user