Restructure solution layout by module

This commit is contained in:
master
2025-10-28 15:10:40 +02:00
parent 95daa159c4
commit d870da18ce
4103 changed files with 192899 additions and 187024 deletions

View File

@@ -0,0 +1,85 @@
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options;
using StellaOps.Scanner.Cache.Abstractions;
namespace StellaOps.Scanner.Cache.Maintenance;
public sealed class ScannerCacheMaintenanceService : BackgroundService
{
private readonly ILayerCacheStore _layerCache;
private readonly IFileContentAddressableStore _fileCas;
private readonly IOptions<ScannerCacheOptions> _options;
private readonly ILogger<ScannerCacheMaintenanceService> _logger;
private readonly TimeProvider _timeProvider;
public ScannerCacheMaintenanceService(
ILayerCacheStore layerCache,
IFileContentAddressableStore fileCas,
IOptions<ScannerCacheOptions> options,
ILogger<ScannerCacheMaintenanceService> logger,
TimeProvider? timeProvider = null)
{
_layerCache = layerCache ?? throw new ArgumentNullException(nameof(layerCache));
_fileCas = fileCas ?? throw new ArgumentNullException(nameof(fileCas));
_options = options ?? throw new ArgumentNullException(nameof(options));
_logger = logger ?? throw new ArgumentNullException(nameof(logger));
_timeProvider = timeProvider ?? TimeProvider.System;
}
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
{
var settings = _options.Value;
if (!settings.Enabled)
{
_logger.LogInformation("Scanner cache disabled; maintenance loop will not start.");
return;
}
if (!settings.EnableAutoEviction)
{
_logger.LogInformation("Scanner cache automatic eviction disabled by configuration.");
return;
}
var interval = settings.MaintenanceInterval > TimeSpan.Zero
? settings.MaintenanceInterval
: TimeSpan.FromMinutes(15);
_logger.LogInformation("Scanner cache maintenance loop started with interval {Interval}", interval);
await RunMaintenanceAsync(stoppingToken).ConfigureAwait(false);
using var timer = new PeriodicTimer(interval, _timeProvider);
while (await timer.WaitForNextTickAsync(stoppingToken).ConfigureAwait(false))
{
await RunMaintenanceAsync(stoppingToken).ConfigureAwait(false);
}
}
private async Task RunMaintenanceAsync(CancellationToken cancellationToken)
{
try
{
var layerExpired = await _layerCache.EvictExpiredAsync(cancellationToken).ConfigureAwait(false);
var layerCompacted = await _layerCache.CompactAsync(cancellationToken).ConfigureAwait(false);
var casExpired = await _fileCas.EvictExpiredAsync(cancellationToken).ConfigureAwait(false);
var casCompacted = await _fileCas.CompactAsync(cancellationToken).ConfigureAwait(false);
_logger.LogDebug(
"Scanner cache maintenance tick complete (layers expired={LayersExpired}, layers compacted={LayersCompacted}, cas expired={CasExpired}, cas compacted={CasCompacted})",
layerExpired,
layerCompacted,
casExpired,
casCompacted);
}
catch (OperationCanceledException) when (cancellationToken.IsCancellationRequested)
{
// Shutting down; ignore.
}
catch (Exception ex)
{
_logger.LogError(ex, "Scanner cache maintenance tick failed");
}
}
}