mirror of
https://github.com/btcpayserver/btcpayserver.git
synced 2025-01-19 05:33:31 +01:00
180 lines
7.2 KiB
C#
180 lines
7.2 KiB
C#
#nullable enable
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using BTCPayServer.Abstractions.Contracts;
|
|
using BTCPayServer.Client.Models;
|
|
using BTCPayServer.Data;
|
|
using BTCPayServer.HostedServices;
|
|
using BTCPayServer.Payments;
|
|
using BTCPayServer.Payouts;
|
|
using BTCPayServer.Services.Invoices;
|
|
using BTCPayServer.Services.Stores;
|
|
using Microsoft.AspNetCore.Mvc.ModelBinding;
|
|
using Microsoft.Extensions.Logging;
|
|
using PayoutData = BTCPayServer.Data.PayoutData;
|
|
using PayoutProcessorData = BTCPayServer.Data.PayoutProcessorData;
|
|
|
|
namespace BTCPayServer.PayoutProcessors;
|
|
|
|
public class AutomatedPayoutConstants
|
|
{
|
|
public const double MinIntervalMinutes = 1.0;
|
|
public const double DefaultIntervalMinutes = 60.0;
|
|
public const double MaxIntervalMinutes = 24 * 60; //1 day
|
|
public static void ValidateInterval(ModelStateDictionary modelState, TimeSpan timeSpan, string parameterName)
|
|
{
|
|
if (timeSpan < TimeSpan.FromMinutes(AutomatedPayoutConstants.MinIntervalMinutes))
|
|
{
|
|
modelState.AddModelError(parameterName, $"The minimum interval is {MinIntervalMinutes * 60} seconds");
|
|
}
|
|
if (timeSpan > TimeSpan.FromMinutes(AutomatedPayoutConstants.MaxIntervalMinutes))
|
|
{
|
|
modelState.AddModelError(parameterName, $"The maximum interval is {MaxIntervalMinutes * 60} seconds");
|
|
}
|
|
}
|
|
}
|
|
public abstract class BaseAutomatedPayoutProcessor<T> : BaseAsyncService where T : AutomatedPayoutBlob, new()
|
|
{
|
|
protected readonly StoreRepository _storeRepository;
|
|
protected readonly PayoutProcessorData PayoutProcessorSettings;
|
|
protected readonly ApplicationDbContextFactory _applicationDbContextFactory;
|
|
private readonly PaymentMethodHandlerDictionary _paymentHandlers;
|
|
protected readonly PayoutMethodId PayoutMethodId;
|
|
protected readonly PaymentMethodId PaymentMethodId;
|
|
private readonly IPluginHookService _pluginHookService;
|
|
protected readonly EventAggregator _eventAggregator;
|
|
|
|
protected BaseAutomatedPayoutProcessor(
|
|
PaymentMethodId paymentMethodId,
|
|
ILoggerFactory logger,
|
|
StoreRepository storeRepository,
|
|
PayoutProcessorData payoutProcessorSettings,
|
|
ApplicationDbContextFactory applicationDbContextFactory,
|
|
PaymentMethodHandlerDictionary paymentHandlers,
|
|
IPluginHookService pluginHookService,
|
|
EventAggregator eventAggregator) : base(logger.CreateLogger($"{payoutProcessorSettings.Processor}:{payoutProcessorSettings.StoreId}:{payoutProcessorSettings.PayoutMethodId}"))
|
|
{
|
|
PaymentMethodId = paymentMethodId;
|
|
_storeRepository = storeRepository;
|
|
PayoutProcessorSettings = payoutProcessorSettings;
|
|
PayoutMethodId = PayoutProcessorSettings.GetPayoutMethodId();
|
|
_applicationDbContextFactory = applicationDbContextFactory;
|
|
_paymentHandlers = paymentHandlers;
|
|
_pluginHookService = pluginHookService;
|
|
_eventAggregator = eventAggregator;
|
|
this.NoLogsOnExit = true;
|
|
}
|
|
|
|
internal override Task[] InitializeTasks()
|
|
{
|
|
_subscription = _eventAggregator.SubscribeAsync<PayoutEvent>(OnPayoutEvent);
|
|
return new[] { CreateLoopTask(Act) };
|
|
}
|
|
|
|
|
|
public override Task StopAsync(CancellationToken cancellationToken)
|
|
{
|
|
_subscription?.Dispose();
|
|
return base.StopAsync(cancellationToken);
|
|
}
|
|
|
|
private Task OnPayoutEvent(PayoutEvent arg)
|
|
{
|
|
if (arg.Type == PayoutEvent.PayoutEventType.Approved &&
|
|
PayoutProcessorSettings.StoreId == arg.Payout.StoreDataId &&
|
|
arg.Payout.GetPayoutMethodId() == PayoutMethodId &&
|
|
GetBlob(PayoutProcessorSettings).ProcessNewPayoutsInstantly)
|
|
{
|
|
SkipInterval();
|
|
}
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
protected virtual Task Process(object paymentMethodConfig, List<PayoutData> payouts) =>
|
|
throw new NotImplementedException();
|
|
|
|
protected virtual async Task<bool> ProcessShouldSave(object paymentMethodConfig, List<PayoutData> payouts)
|
|
{
|
|
await Process(paymentMethodConfig, payouts);
|
|
return true;
|
|
}
|
|
|
|
private async Task Act()
|
|
{
|
|
var store = await _storeRepository.FindStore(PayoutProcessorSettings.StoreId);
|
|
var paymentMethod = store?.GetPaymentMethodConfig(PaymentMethodId, _paymentHandlers, true);
|
|
|
|
var blob = GetBlob(PayoutProcessorSettings);
|
|
if (paymentMethod is not null)
|
|
{
|
|
await using var context = _applicationDbContextFactory.CreateContext();
|
|
var payouts = await PullPaymentHostedService.GetPayouts(
|
|
new PullPaymentHostedService.PayoutQuery()
|
|
{
|
|
States = new[] { PayoutState.AwaitingPayment },
|
|
PayoutMethods = new[] { PayoutProcessorSettings.PayoutMethodId },
|
|
Processor = PayoutProcessorSettings.Processor,
|
|
Stores = new[] {PayoutProcessorSettings.StoreId}
|
|
}, context, CancellationToken);
|
|
|
|
await _pluginHookService.ApplyAction("before-automated-payout-processing",
|
|
new BeforePayoutActionData(store, PayoutProcessorSettings, payouts));
|
|
if (payouts.Any())
|
|
{
|
|
Logs.PayServer.LogInformation(
|
|
$"{payouts.Count} found to process. Starting (and after will sleep for {blob.Interval})");
|
|
if (await ProcessShouldSave(paymentMethod, payouts))
|
|
{
|
|
await context.SaveChangesAsync();
|
|
|
|
foreach (var payoutData in payouts.Where(payoutData => payoutData.State != PayoutState.AwaitingPayment))
|
|
{
|
|
_eventAggregator.Publish(new PayoutEvent(PayoutEvent.PayoutEventType.Updated, payoutData));
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// Allow plugins do to something after automatic payout processing
|
|
await _pluginHookService.ApplyAction("after-automated-payout-processing",
|
|
new AfterPayoutActionData(store, PayoutProcessorSettings, payouts));
|
|
}
|
|
|
|
// Clip interval
|
|
if (blob.Interval < TimeSpan.FromMinutes(AutomatedPayoutConstants.MinIntervalMinutes))
|
|
blob.Interval = TimeSpan.FromMinutes(AutomatedPayoutConstants.MinIntervalMinutes);
|
|
if (blob.Interval > TimeSpan.FromMinutes(AutomatedPayoutConstants.MaxIntervalMinutes))
|
|
blob.Interval = TimeSpan.FromMinutes(AutomatedPayoutConstants.MaxIntervalMinutes);
|
|
try
|
|
{
|
|
using var cts = CancellationTokenSource.CreateLinkedTokenSource(CancellationToken, _timerCTs.Token);
|
|
await Task.Delay(blob.Interval, cts.Token);
|
|
}
|
|
catch (TaskCanceledException)
|
|
{
|
|
}
|
|
}
|
|
|
|
private CancellationTokenSource _timerCTs = new CancellationTokenSource();
|
|
private IEventAggregatorSubscription? _subscription;
|
|
|
|
private readonly object _intervalLock = new object();
|
|
|
|
public void SkipInterval()
|
|
{
|
|
lock (_intervalLock)
|
|
{
|
|
_timerCTs.Cancel();
|
|
_timerCTs = new CancellationTokenSource();
|
|
}
|
|
}
|
|
|
|
public static T GetBlob(PayoutProcessorData payoutProcesserSettings)
|
|
{
|
|
return payoutProcesserSettings.HasTypedBlob<T>().GetBlob() ?? new T();
|
|
}
|
|
}
|