mirror of
https://github.com/btcpayserver/btcpayserver.git
synced 2025-01-19 05:33:31 +01:00
d5d0be5824
* Editorconfig: Add space_before_self_closing setting This was a difference between the way dotnet-format and Rider format code. See https://www.jetbrains.com/help/rider/EditorConfig_Index.html * Editorconfig: Keep 4 spaces indentation for Swagger JSON files They are all formatted that way, let's keep it like that. * Apply dotnet-format, mostly white-space related changes
102 lines
3.1 KiB
C#
102 lines
3.1 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Threading;
|
|
using System.Threading.Channels;
|
|
using System.Threading.Tasks;
|
|
using BTCPayServer.Logging;
|
|
using Microsoft.Extensions.Hosting;
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
namespace BTCPayServer.HostedServices
|
|
{
|
|
public class EventHostedServiceBase : IHostedService
|
|
{
|
|
private readonly EventAggregator _EventAggregator;
|
|
|
|
public Logs Logs { get; }
|
|
|
|
public EventAggregator EventAggregator => _EventAggregator;
|
|
|
|
private List<IEventAggregatorSubscription> _Subscriptions;
|
|
private CancellationTokenSource _Cts;
|
|
public CancellationToken CancellationToken => _Cts.Token;
|
|
public EventHostedServiceBase(EventAggregator eventAggregator, Logs logs)
|
|
{
|
|
_EventAggregator = eventAggregator;
|
|
Logs = logs;
|
|
}
|
|
|
|
public EventHostedServiceBase(EventAggregator eventAggregator, ILogger logger)
|
|
{
|
|
_EventAggregator = eventAggregator;
|
|
Logs = new Logs() { PayServer = logger, Events = logger, Configuration = logger };
|
|
}
|
|
|
|
readonly Channel<object> _Events = Channel.CreateUnbounded<object>();
|
|
public async Task ProcessEvents(CancellationToken cancellationToken)
|
|
{
|
|
while (await _Events.Reader.WaitToReadAsync(cancellationToken))
|
|
{
|
|
if (_Events.Reader.TryRead(out var evt))
|
|
{
|
|
try
|
|
{
|
|
await ProcessEvent(evt, cancellationToken);
|
|
}
|
|
catch when (cancellationToken.IsCancellationRequested)
|
|
{
|
|
throw;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Logs.PayServer.LogWarning(ex, $"Unhandled exception in {this.GetType().Name}");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
protected virtual Task ProcessEvent(object evt, CancellationToken cancellationToken)
|
|
{
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
|
|
protected virtual void SubscribeToEvents()
|
|
{
|
|
|
|
}
|
|
|
|
protected void Subscribe<T>()
|
|
{
|
|
_Subscriptions.Add(_EventAggregator.Subscribe<T>(e => _Events.Writer.TryWrite(e)));
|
|
}
|
|
|
|
protected void PushEvent(object obj)
|
|
{
|
|
_Events.Writer.TryWrite(obj);
|
|
}
|
|
|
|
public virtual Task StartAsync(CancellationToken cancellationToken)
|
|
{
|
|
_Subscriptions = new List<IEventAggregatorSubscription>();
|
|
SubscribeToEvents();
|
|
_Cts = new CancellationTokenSource();
|
|
_ProcessingEvents = ProcessEvents(_Cts.Token);
|
|
return Task.CompletedTask;
|
|
}
|
|
Task _ProcessingEvents = Task.CompletedTask;
|
|
|
|
public virtual async Task StopAsync(CancellationToken cancellationToken)
|
|
{
|
|
_Subscriptions?.ForEach(subscription => subscription.Dispose());
|
|
_Cts?.Cancel();
|
|
try
|
|
{
|
|
await _ProcessingEvents;
|
|
}
|
|
catch (OperationCanceledException)
|
|
{ }
|
|
}
|
|
}
|
|
}
|