Files
hcs/Hcs.WebApp/BackgroundServices/ResultGetService.cs

143 lines
5.3 KiB
C#

using Hcs.Broker;
using Hcs.Broker.Logger;
using Hcs.Broker.MessageCapturer;
using Hcs.WebApp.BackgroundServices.ResultGetters;
using Hcs.WebApp.Config;
using Hcs.WebApp.Data.Hcs;
using Hcs.WebApp.Services;
using Hcs.WebApp.Utils;
namespace Hcs.WebApp.BackgroundServices
{
public class ResultGetService(
ResultGetState state,
ResultGetterFactory resultGetterFactory,
IServiceScopeFactory scopeFactory) : BackgroundService
{
private class GetState
{
public int attempt;
public int timer;
public bool done;
}
private const int ITERATION_TIME = 10000;
private const int SLEEP_TIME = 30000;
private readonly ResultGetState state = state;
private readonly ResultGetterFactory resultGetterFactory = resultGetterFactory;
private readonly IServiceScopeFactory scopeFactory = scopeFactory;
private readonly List<(Operation operation, GetState state)> entries = [];
private Broker.Logger.ILogger logger;
private IMessageCapturer messageCapturer;
private IClient client;
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
{
await InitializeStateAsync();
InitializeClient();
while (!stoppingToken.IsCancellationRequested)
{
while (state.TryDequeueOperation(out var operation))
{
state.SetProcessingOperation(operation);
if (stoppingToken.IsCancellationRequested) return;
entries.Add(new (operation, new GetState()));
}
if (entries.Count > 0)
{
await Task.Delay(ITERATION_TIME, stoppingToken);
var scope = scopeFactory.CreateScope();
foreach (var entry in entries)
{
entry.state.timer += ITERATION_TIME;
var endedAt = DateTime.MinValue;
var failureReason = string.Empty;
var send = entry.state.attempt switch
{
0 => entry.state.timer >= 10000,
1 => entry.state.timer >= 60000,
2 => entry.state.timer >= 300000,
3 => entry.state.timer >= 900000,
_ => entry.state.timer >= 1800000,
};
if (send)
{
try
{
var resultGetter = resultGetterFactory.CreateResultGetter(scope, client, entry.operation);
var response = await resultGetter.GetAsync();
if (response.success)
{
entry.state.done = true;
endedAt = response.endedAt;
}
}
catch (Exception e)
{
var headquartersService = scope.ServiceProvider.GetRequiredService<HeadquartersService>();
endedAt = DateTime.UtcNow;
failureReason = e.CombineMessages();
await headquartersService.SetOperationEndedWithFailAsync(entry.operation.Id, endedAt, failureReason);
entry.state.done = true;
}
entry.state.attempt++;
entry.state.timer = 0;
}
if (entry.state.done)
{
state.UnsetProcessingOperation(entry.operation);
state.InvokeOnOperationEnded(entry.operation.Id, entry.operation.CampaignId, endedAt, failureReason);
}
}
entries.RemoveAll(x => x.state.done);
}
else
{
await Task.Delay(SLEEP_TIME, stoppingToken);
}
}
}
private async Task InitializeStateAsync()
{
using var scope = scopeFactory.CreateScope();
var headquartersService = scope.ServiceProvider.GetRequiredService<HeadquartersService>();
var operations = await headquartersService.GetResultWaitingOperationsAsync();
foreach (var operation in operations)
{
state.EnqueueOperation(operation);
}
state.Ready = true;
}
private void InitializeClient()
{
logger = new ActionLogger();
messageCapturer = new FileMessageCapturer("incoming", logger);
using var scope = scopeFactory.CreateScope();
var configuration = scope.ServiceProvider.GetRequiredService<IConfiguration>();
var config = configuration.GetSection("BrokerConfig").Get<BrokerConfig>()!;
var clientProvider = scope.ServiceProvider.GetRequiredService<IClientProvider>();
client = clientProvider.CreateClient(config, logger, messageCapturer);
}
}
}