2021-11-15 06:25:59 +01:00
|
|
|
#nullable enable
|
2020-07-22 13:58:41 +02:00
|
|
|
using System;
|
2021-10-06 11:19:34 +02:00
|
|
|
using System.Collections.Generic;
|
2020-07-22 13:58:41 +02:00
|
|
|
using System.Linq;
|
2022-11-28 09:53:08 +01:00
|
|
|
using System.Threading;
|
2020-07-22 13:58:41 +02:00
|
|
|
using System.Threading.Tasks;
|
2020-11-17 13:46:23 +01:00
|
|
|
using BTCPayServer.Abstractions.Constants;
|
2022-02-24 09:00:44 +01:00
|
|
|
using BTCPayServer.Abstractions.Extensions;
|
2020-07-22 13:58:41 +02:00
|
|
|
using BTCPayServer.Client;
|
2020-07-24 08:13:21 +02:00
|
|
|
using BTCPayServer.Client.Models;
|
2022-11-28 09:53:08 +01:00
|
|
|
using BTCPayServer.Data;
|
|
|
|
using BTCPayServer.HostedServices;
|
2020-07-22 13:58:41 +02:00
|
|
|
using BTCPayServer.Payments;
|
2023-01-06 14:18:07 +01:00
|
|
|
using BTCPayServer.Rating;
|
2023-06-16 03:47:58 +02:00
|
|
|
using BTCPayServer.Security.Greenfield;
|
2020-12-10 15:34:50 +01:00
|
|
|
using BTCPayServer.Services;
|
2020-07-22 13:58:41 +02:00
|
|
|
using BTCPayServer.Services.Invoices;
|
2022-11-28 09:53:08 +01:00
|
|
|
using BTCPayServer.Services.Rates;
|
2020-07-22 13:58:41 +02:00
|
|
|
using Microsoft.AspNetCore.Authorization;
|
|
|
|
using Microsoft.AspNetCore.Cors;
|
2022-07-06 15:17:33 +02:00
|
|
|
using Microsoft.AspNetCore.Http;
|
2020-07-22 13:58:41 +02:00
|
|
|
using Microsoft.AspNetCore.Mvc;
|
2020-12-10 15:34:50 +01:00
|
|
|
using Microsoft.AspNetCore.Routing;
|
2020-07-22 13:58:41 +02:00
|
|
|
using NBitcoin;
|
|
|
|
using CreateInvoiceRequest = BTCPayServer.Client.Models.CreateInvoiceRequest;
|
|
|
|
using InvoiceData = BTCPayServer.Client.Models.InvoiceData;
|
|
|
|
|
2022-01-14 05:05:23 +01:00
|
|
|
namespace BTCPayServer.Controllers.Greenfield
|
2020-07-22 13:58:41 +02:00
|
|
|
{
|
|
|
|
[ApiController]
|
|
|
|
[Authorize(AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
|
|
|
[EnableCors(CorsPolicies.All)]
|
2022-01-07 04:17:59 +01:00
|
|
|
public class GreenfieldInvoiceController : Controller
|
2020-07-22 13:58:41 +02:00
|
|
|
{
|
2022-01-07 04:32:00 +01:00
|
|
|
private readonly UIInvoiceController _invoiceController;
|
2020-07-22 13:58:41 +02:00
|
|
|
private readonly InvoiceRepository _invoiceRepository;
|
2020-12-10 15:34:50 +01:00
|
|
|
private readonly LinkGenerator _linkGenerator;
|
2022-11-28 09:53:08 +01:00
|
|
|
private readonly CurrencyNameTable _currencyNameTable;
|
|
|
|
private readonly BTCPayNetworkProvider _networkProvider;
|
|
|
|
private readonly PullPaymentHostedService _pullPaymentService;
|
|
|
|
private readonly RateFetcher _rateProvider;
|
2022-12-08 05:16:18 +01:00
|
|
|
private readonly InvoiceActivator _invoiceActivator;
|
2022-11-28 09:53:08 +01:00
|
|
|
private readonly ApplicationDbContextFactory _dbContextFactory;
|
2023-12-14 11:15:36 +01:00
|
|
|
private readonly IAuthorizationService _authorizationService;
|
2020-07-22 13:58:41 +02:00
|
|
|
|
2020-12-10 15:34:50 +01:00
|
|
|
public LanguageService LanguageService { get; }
|
|
|
|
|
2022-01-07 04:32:00 +01:00
|
|
|
public GreenfieldInvoiceController(UIInvoiceController invoiceController, InvoiceRepository invoiceRepository,
|
2021-04-07 06:08:42 +02:00
|
|
|
LinkGenerator linkGenerator, LanguageService languageService, BTCPayNetworkProvider btcPayNetworkProvider,
|
2022-12-08 05:16:18 +01:00
|
|
|
CurrencyNameTable currencyNameTable, RateFetcher rateProvider,
|
|
|
|
InvoiceActivator invoiceActivator,
|
2023-12-14 11:15:36 +01:00
|
|
|
PullPaymentHostedService pullPaymentService,
|
|
|
|
ApplicationDbContextFactory dbContextFactory,
|
|
|
|
IAuthorizationService authorizationService)
|
2020-07-22 13:58:41 +02:00
|
|
|
{
|
|
|
|
_invoiceController = invoiceController;
|
|
|
|
_invoiceRepository = invoiceRepository;
|
2020-12-10 15:34:50 +01:00
|
|
|
_linkGenerator = linkGenerator;
|
2022-11-28 09:53:08 +01:00
|
|
|
_currencyNameTable = currencyNameTable;
|
2022-12-08 05:16:18 +01:00
|
|
|
_networkProvider = btcPayNetworkProvider;
|
2022-11-28 09:53:08 +01:00
|
|
|
_rateProvider = rateProvider;
|
2022-12-08 05:16:18 +01:00
|
|
|
_invoiceActivator = invoiceActivator;
|
2022-11-28 09:53:08 +01:00
|
|
|
_pullPaymentService = pullPaymentService;
|
|
|
|
_dbContextFactory = dbContextFactory;
|
2023-12-14 11:15:36 +01:00
|
|
|
_authorizationService = authorizationService;
|
2020-12-10 15:34:50 +01:00
|
|
|
LanguageService = languageService;
|
2020-07-22 13:58:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
[Authorize(Policy = Policies.CanViewInvoices,
|
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
2020-07-24 12:46:46 +02:00
|
|
|
[HttpGet("~/api/v1/stores/{storeId}/invoices")]
|
2021-11-15 06:25:59 +01:00
|
|
|
public async Task<IActionResult> GetInvoices(string storeId, [FromQuery] string[]? orderId = null, [FromQuery] string[]? status = null,
|
2021-04-26 05:37:56 +02:00
|
|
|
[FromQuery]
|
|
|
|
[ModelBinder(typeof(ModelBinders.DateTimeOffsetModelBinder))]
|
|
|
|
DateTimeOffset? startDate = null,
|
2021-12-31 08:59:02 +01:00
|
|
|
[FromQuery]
|
2021-04-26 05:37:56 +02:00
|
|
|
[ModelBinder(typeof(ModelBinders.DateTimeOffsetModelBinder))]
|
2021-07-14 16:32:20 +02:00
|
|
|
DateTimeOffset? endDate = null,
|
2021-11-15 06:25:59 +01:00
|
|
|
[FromQuery] string? textSearch = null,
|
2021-10-31 11:47:12 +01:00
|
|
|
[FromQuery] bool includeArchived = false,
|
|
|
|
[FromQuery] int? skip = null,
|
|
|
|
[FromQuery] int? take = null
|
|
|
|
)
|
2020-07-22 13:58:41 +02:00
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return StoreNotFound();
|
2020-07-22 13:58:41 +02:00
|
|
|
}
|
2021-04-26 05:37:56 +02:00
|
|
|
if (startDate is DateTimeOffset s &&
|
|
|
|
endDate is DateTimeOffset e &&
|
|
|
|
s > e)
|
|
|
|
{
|
|
|
|
this.ModelState.AddModelError(nameof(startDate), "startDate should not be above endDate");
|
|
|
|
this.ModelState.AddModelError(nameof(endDate), "endDate should not be below startDate");
|
|
|
|
}
|
2020-07-22 13:58:41 +02:00
|
|
|
|
2021-04-26 05:37:56 +02:00
|
|
|
if (!ModelState.IsValid)
|
|
|
|
return this.CreateValidationError(ModelState);
|
2020-07-24 08:13:21 +02:00
|
|
|
var invoices =
|
|
|
|
await _invoiceRepository.GetInvoices(new InvoiceQuery()
|
|
|
|
{
|
2021-10-31 11:47:12 +01:00
|
|
|
Skip = skip,
|
|
|
|
Take = take,
|
2021-12-31 08:59:02 +01:00
|
|
|
StoreId = new[] { store.Id },
|
2021-04-26 04:32:44 +02:00
|
|
|
IncludeArchived = includeArchived,
|
2021-04-26 05:37:56 +02:00
|
|
|
StartDate = startDate,
|
|
|
|
EndDate = endDate,
|
2021-04-26 04:32:44 +02:00
|
|
|
OrderId = orderId,
|
2021-07-14 16:32:20 +02:00
|
|
|
Status = status,
|
|
|
|
TextSearch = textSearch
|
2020-07-24 08:13:21 +02:00
|
|
|
});
|
2020-07-22 13:58:41 +02:00
|
|
|
|
|
|
|
return Ok(invoices.Select(ToModel));
|
|
|
|
}
|
2021-12-31 08:59:02 +01:00
|
|
|
|
2020-07-22 13:58:41 +02:00
|
|
|
[Authorize(Policy = Policies.CanViewInvoices,
|
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
|
|
|
[HttpGet("~/api/v1/stores/{storeId}/invoices/{invoiceId}")]
|
|
|
|
public async Task<IActionResult> GetInvoice(string storeId, string invoiceId)
|
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-07-22 13:58:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var invoice = await _invoiceRepository.GetInvoice(invoiceId, true);
|
2021-03-06 05:25:40 +01:00
|
|
|
if (invoice?.StoreId != store.Id)
|
2020-07-22 13:58:41 +02:00
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-07-22 13:58:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return Ok(ToModel(invoice));
|
|
|
|
}
|
|
|
|
|
2021-07-10 17:30:01 +02:00
|
|
|
[Authorize(Policy = Policies.CanModifyInvoices,
|
2020-07-22 13:58:41 +02:00
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
|
|
|
[HttpDelete("~/api/v1/stores/{storeId}/invoices/{invoiceId}")]
|
|
|
|
public async Task<IActionResult> ArchiveInvoice(string storeId, string invoiceId)
|
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
|
|
|
}
|
|
|
|
var invoice = await _invoiceRepository.GetInvoice(invoiceId, true);
|
|
|
|
if (invoice?.StoreId != store.Id)
|
|
|
|
{
|
|
|
|
return InvoiceNotFound();
|
2020-07-22 13:58:41 +02:00
|
|
|
}
|
2020-07-24 08:13:21 +02:00
|
|
|
await _invoiceRepository.ToggleInvoiceArchival(invoiceId, true, storeId);
|
2020-07-22 13:58:41 +02:00
|
|
|
return Ok();
|
|
|
|
}
|
|
|
|
|
2021-07-10 17:30:01 +02:00
|
|
|
[Authorize(Policy = Policies.CanModifyInvoices,
|
2020-12-12 07:15:34 +01:00
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
|
|
|
[HttpPut("~/api/v1/stores/{storeId}/invoices/{invoiceId}")]
|
|
|
|
public async Task<IActionResult> UpdateInvoice(string storeId, string invoiceId, UpdateInvoiceRequest request)
|
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-12-12 07:15:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var result = await _invoiceRepository.UpdateInvoiceMetadata(invoiceId, storeId, request.Metadata);
|
|
|
|
if (result != null)
|
|
|
|
{
|
|
|
|
return Ok(ToModel(result));
|
|
|
|
}
|
|
|
|
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-12-12 07:15:34 +01:00
|
|
|
}
|
|
|
|
|
2020-07-22 13:58:41 +02:00
|
|
|
[Authorize(Policy = Policies.CanCreateInvoice,
|
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
|
|
|
[HttpPost("~/api/v1/stores/{storeId}/invoices")]
|
|
|
|
public async Task<IActionResult> CreateInvoice(string storeId, CreateInvoiceRequest request)
|
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return StoreNotFound();
|
2020-07-22 13:58:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (request.Amount < 0.0m)
|
|
|
|
{
|
|
|
|
ModelState.AddModelError(nameof(request.Amount), "The amount should be 0 or more.");
|
|
|
|
}
|
2023-06-16 03:47:58 +02:00
|
|
|
if (request.Amount > GreenfieldConstants.MaxAmount)
|
|
|
|
{
|
|
|
|
ModelState.AddModelError(nameof(request.Amount), $"The amount should less than {GreenfieldConstants.MaxAmount}.");
|
|
|
|
}
|
2023-05-04 15:14:15 +02:00
|
|
|
request.Checkout ??= new CreateInvoiceRequest.CheckoutOptions();
|
2020-07-22 13:58:41 +02:00
|
|
|
if (request.Checkout.PaymentMethods?.Any() is true)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < request.Checkout.PaymentMethods.Length; i++)
|
|
|
|
{
|
|
|
|
if (!PaymentMethodId.TryParse(request.Checkout.PaymentMethods[i], out _))
|
|
|
|
{
|
|
|
|
request.AddModelError(invoiceRequest => invoiceRequest.Checkout.PaymentMethods[i],
|
|
|
|
"Invalid payment method", this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-26 07:01:39 +02:00
|
|
|
if (request.Checkout.Expiration != null && request.Checkout.Expiration < TimeSpan.FromSeconds(30.0))
|
2020-07-22 13:58:41 +02:00
|
|
|
{
|
2020-08-26 07:01:39 +02:00
|
|
|
request.AddModelError(invoiceRequest => invoiceRequest.Checkout.Expiration,
|
|
|
|
"Expiration time must be at least 30 seconds", this);
|
2020-07-22 13:58:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (request.Checkout.PaymentTolerance != null &&
|
|
|
|
(request.Checkout.PaymentTolerance < 0 || request.Checkout.PaymentTolerance > 100))
|
|
|
|
{
|
|
|
|
request.AddModelError(invoiceRequest => invoiceRequest.Checkout.PaymentTolerance,
|
|
|
|
"PaymentTolerance can only be between 0 and 100 percent", this);
|
|
|
|
}
|
|
|
|
|
2020-12-10 15:34:50 +01:00
|
|
|
if (request.Checkout.DefaultLanguage != null)
|
|
|
|
{
|
2021-07-27 08:17:56 +02:00
|
|
|
var lang = LanguageService.FindLanguage(request.Checkout.DefaultLanguage);
|
2020-12-10 15:34:50 +01:00
|
|
|
if (lang == null)
|
|
|
|
{
|
|
|
|
request.AddModelError(invoiceRequest => invoiceRequest.Checkout.DefaultLanguage,
|
|
|
|
"The requested defaultLang does not exists, Browse the ~/misc/lang page of your BTCPay Server instance to see the list of supported languages.", this);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Ensure this is good case
|
|
|
|
request.Checkout.DefaultLanguage = lang.Code;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-22 13:58:41 +02:00
|
|
|
if (!ModelState.IsValid)
|
|
|
|
return this.CreateValidationError(ModelState);
|
2023-05-04 15:14:15 +02:00
|
|
|
|
2020-07-24 12:46:46 +02:00
|
|
|
try
|
|
|
|
{
|
2020-08-26 07:01:39 +02:00
|
|
|
var invoice = await _invoiceController.CreateInvoiceCoreRaw(request, store,
|
2021-11-04 10:03:34 +01:00
|
|
|
Request.GetAbsoluteRoot());
|
2020-07-24 12:46:46 +02:00
|
|
|
return Ok(ToModel(invoice));
|
|
|
|
}
|
|
|
|
catch (BitpayHttpException e)
|
|
|
|
{
|
2020-07-27 10:43:35 +02:00
|
|
|
return this.CreateAPIError(null, e.Message);
|
2020-07-24 12:46:46 +02:00
|
|
|
}
|
2020-07-22 13:58:41 +02:00
|
|
|
}
|
|
|
|
|
2021-07-10 17:30:01 +02:00
|
|
|
[Authorize(Policy = Policies.CanModifyInvoices,
|
2020-07-24 08:13:21 +02:00
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
2020-07-27 10:43:35 +02:00
|
|
|
[HttpPost("~/api/v1/stores/{storeId}/invoices/{invoiceId}/status")]
|
|
|
|
public async Task<IActionResult> MarkInvoiceStatus(string storeId, string invoiceId,
|
|
|
|
MarkInvoiceStatusRequest request)
|
2020-07-24 08:13:21 +02:00
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-07-24 08:13:21 +02:00
|
|
|
}
|
|
|
|
|
2020-07-24 09:40:37 +02:00
|
|
|
var invoice = await _invoiceRepository.GetInvoice(invoiceId, true);
|
|
|
|
if (invoice.StoreId != store.Id)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-07-24 09:40:37 +02:00
|
|
|
}
|
|
|
|
|
2020-07-27 10:43:35 +02:00
|
|
|
if (!await _invoiceRepository.MarkInvoiceStatus(invoice.Id, request.Status))
|
2020-07-24 09:40:37 +02:00
|
|
|
{
|
2020-07-27 10:43:35 +02:00
|
|
|
ModelState.AddModelError(nameof(request.Status),
|
2021-02-23 13:18:16 +01:00
|
|
|
"Status can only be marked to invalid or settled within certain conditions.");
|
2020-07-24 09:40:37 +02:00
|
|
|
}
|
|
|
|
|
2020-07-27 10:43:35 +02:00
|
|
|
if (!ModelState.IsValid)
|
|
|
|
return this.CreateValidationError(ModelState);
|
|
|
|
|
|
|
|
return await GetInvoice(storeId, invoiceId);
|
|
|
|
}
|
|
|
|
|
2021-07-10 17:30:01 +02:00
|
|
|
[Authorize(Policy = Policies.CanModifyInvoices,
|
2020-07-27 10:43:35 +02:00
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
|
|
|
[HttpPost("~/api/v1/stores/{storeId}/invoices/{invoiceId}/unarchive")]
|
|
|
|
public async Task<IActionResult> UnarchiveInvoice(string storeId, string invoiceId)
|
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-07-27 10:43:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var invoice = await _invoiceRepository.GetInvoice(invoiceId, true);
|
|
|
|
if (invoice.StoreId != store.Id)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-07-27 10:43:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!invoice.Archived)
|
|
|
|
{
|
|
|
|
return this.CreateAPIError("already-unarchived", "Invoice is already unarchived");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!ModelState.IsValid)
|
|
|
|
return this.CreateValidationError(ModelState);
|
|
|
|
|
|
|
|
await _invoiceRepository.ToggleInvoiceArchival(invoiceId, false, storeId);
|
|
|
|
return await GetInvoice(storeId, invoiceId);
|
|
|
|
}
|
2021-12-31 08:59:02 +01:00
|
|
|
|
2020-09-10 13:54:36 +02:00
|
|
|
[Authorize(Policy = Policies.CanViewInvoices,
|
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
|
|
|
[HttpGet("~/api/v1/stores/{storeId}/invoices/{invoiceId}/payment-methods")]
|
2021-05-14 09:16:19 +02:00
|
|
|
public async Task<IActionResult> GetInvoicePaymentMethods(string storeId, string invoiceId, bool onlyAccountedPayments = true)
|
2020-09-10 13:54:36 +02:00
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-09-10 13:54:36 +02:00
|
|
|
}
|
2020-07-27 10:43:35 +02:00
|
|
|
|
2020-09-10 13:54:36 +02:00
|
|
|
var invoice = await _invoiceRepository.GetInvoice(invoiceId, true);
|
2020-12-17 06:43:43 +01:00
|
|
|
if (invoice?.StoreId != store.Id)
|
2020-09-10 13:54:36 +02:00
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2020-09-10 13:54:36 +02:00
|
|
|
}
|
2020-07-27 10:43:35 +02:00
|
|
|
|
2021-05-14 09:16:19 +02:00
|
|
|
return Ok(ToPaymentMethodModels(invoice, onlyAccountedPayments));
|
2020-09-10 13:54:36 +02:00
|
|
|
}
|
2021-04-07 06:08:42 +02:00
|
|
|
|
|
|
|
[Authorize(Policy = Policies.CanViewInvoices,
|
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
|
|
|
[HttpPost("~/api/v1/stores/{storeId}/invoices/{invoiceId}/payment-methods/{paymentMethod}/activate")]
|
|
|
|
public async Task<IActionResult> ActivateInvoicePaymentMethod(string storeId, string invoiceId, string paymentMethod)
|
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2021-04-07 06:08:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var invoice = await _invoiceRepository.GetInvoice(invoiceId, true);
|
|
|
|
if (invoice?.StoreId != store.Id)
|
|
|
|
{
|
2021-04-08 08:57:01 +02:00
|
|
|
return InvoiceNotFound();
|
2021-04-07 06:08:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (PaymentMethodId.TryParse(paymentMethod, out var paymentMethodId))
|
|
|
|
{
|
2022-12-08 05:16:18 +01:00
|
|
|
await _invoiceActivator.ActivateInvoicePaymentMethod(paymentMethodId, invoice, store);
|
2021-04-07 06:08:42 +02:00
|
|
|
return Ok();
|
|
|
|
}
|
2021-04-08 08:57:01 +02:00
|
|
|
ModelState.AddModelError(nameof(paymentMethod), "Invalid payment method");
|
|
|
|
return this.CreateValidationError(ModelState);
|
2021-04-07 06:08:42 +02:00
|
|
|
}
|
2021-04-08 08:57:01 +02:00
|
|
|
|
2023-12-14 11:15:36 +01:00
|
|
|
[Authorize(Policy = Policies.CanCreateNonApprovedPullPayments,
|
2022-11-28 09:53:08 +01:00
|
|
|
AuthenticationSchemes = AuthenticationSchemes.Greenfield)]
|
2022-11-28 12:58:18 +01:00
|
|
|
[HttpPost("~/api/v1/stores/{storeId}/invoices/{invoiceId}/refund")]
|
2022-11-28 09:53:08 +01:00
|
|
|
public async Task<IActionResult> RefundInvoice(
|
|
|
|
string storeId,
|
|
|
|
string invoiceId,
|
|
|
|
RefundInvoiceRequest request,
|
|
|
|
CancellationToken cancellationToken = default
|
|
|
|
)
|
|
|
|
{
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
|
|
{
|
|
|
|
return StoreNotFound();
|
|
|
|
}
|
|
|
|
|
|
|
|
var invoice = await _invoiceRepository.GetInvoice(invoiceId, true);
|
|
|
|
if (invoice == null)
|
|
|
|
{
|
|
|
|
return InvoiceNotFound();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (invoice.StoreId != store.Id)
|
|
|
|
{
|
|
|
|
return InvoiceNotFound();
|
|
|
|
}
|
|
|
|
if (!invoice.GetInvoiceState().CanRefund())
|
|
|
|
{
|
|
|
|
return this.CreateAPIError("non-refundable", "Cannot refund this invoice");
|
|
|
|
}
|
2022-11-28 12:58:18 +01:00
|
|
|
PaymentMethod? invoicePaymentMethod = null;
|
|
|
|
PaymentMethodId? paymentMethodId = null;
|
|
|
|
if (request.PaymentMethod is not null && PaymentMethodId.TryParse(request.PaymentMethod, out paymentMethodId))
|
2022-11-28 09:53:08 +01:00
|
|
|
{
|
2022-11-28 12:58:18 +01:00
|
|
|
invoicePaymentMethod = invoice.GetPaymentMethods().SingleOrDefault(method => method.GetId() == paymentMethodId);
|
|
|
|
}
|
|
|
|
if (invoicePaymentMethod is null)
|
|
|
|
{
|
2023-05-11 10:33:33 +02:00
|
|
|
ModelState.AddModelError(nameof(request.PaymentMethod), "Please select one of the payment methods which were available for the original invoice");
|
2022-11-28 09:53:08 +01:00
|
|
|
}
|
2022-11-28 12:58:18 +01:00
|
|
|
if (request.RefundVariant is null)
|
2023-05-11 10:33:33 +02:00
|
|
|
ModelState.AddModelError(nameof(request.RefundVariant), "`refundVariant` is mandatory");
|
2022-11-28 12:58:18 +01:00
|
|
|
if (!ModelState.IsValid || invoicePaymentMethod is null || paymentMethodId is null)
|
|
|
|
return this.CreateValidationError(ModelState);
|
2022-11-28 09:53:08 +01:00
|
|
|
|
2023-05-11 10:33:33 +02:00
|
|
|
var accounting = invoicePaymentMethod.Calculate();
|
2023-07-19 11:47:32 +02:00
|
|
|
var cryptoPaid = accounting.Paid;
|
2022-11-28 09:53:08 +01:00
|
|
|
var cdCurrency = _currencyNameTable.GetCurrencyData(invoice.Currency, true);
|
|
|
|
var paidCurrency = Math.Round(cryptoPaid * invoicePaymentMethod.Rate, cdCurrency.Divisibility);
|
|
|
|
var rateResult = await _rateProvider.FetchRate(
|
|
|
|
new CurrencyPair(paymentMethodId.CryptoCode, invoice.Currency),
|
|
|
|
store.GetStoreBlob().GetRateRules(_networkProvider),
|
|
|
|
cancellationToken
|
|
|
|
);
|
2023-05-11 10:33:33 +02:00
|
|
|
var cryptoCode = invoicePaymentMethod.GetId().CryptoCode;
|
2022-11-28 09:53:08 +01:00
|
|
|
var paymentMethodDivisibility = _currencyNameTable.GetCurrencyData(paymentMethodId.CryptoCode, false)?.Divisibility ?? 8;
|
2023-05-11 10:33:33 +02:00
|
|
|
var paidAmount = cryptoPaid.RoundToSignificant(paymentMethodDivisibility);
|
|
|
|
var createPullPayment = new CreatePullPayment
|
2022-11-28 09:53:08 +01:00
|
|
|
{
|
|
|
|
BOLT11Expiration = store.GetStoreBlob().RefundBOLT11Expiration,
|
|
|
|
Name = request.Name ?? $"Refund {invoice.Id}",
|
|
|
|
Description = request.Description,
|
|
|
|
StoreId = storeId,
|
|
|
|
PaymentMethodIds = new[] { paymentMethodId },
|
|
|
|
};
|
|
|
|
|
2022-11-28 12:58:18 +01:00
|
|
|
if (request.RefundVariant != RefundVariant.Custom)
|
|
|
|
{
|
|
|
|
if (request.CustomAmount is not null)
|
2023-05-11 10:33:33 +02:00
|
|
|
ModelState.AddModelError(nameof(request.CustomAmount), "CustomAmount should only be set if the refundVariant is Custom");
|
2022-11-28 12:58:18 +01:00
|
|
|
if (request.CustomCurrency is not null)
|
2023-05-11 10:33:33 +02:00
|
|
|
ModelState.AddModelError(nameof(request.CustomCurrency), "CustomCurrency should only be set if the refundVariant is Custom");
|
|
|
|
}
|
|
|
|
if (request.SubtractPercentage is < 0 or > 100)
|
|
|
|
{
|
|
|
|
ModelState.AddModelError(nameof(request.SubtractPercentage), "Percentage must be a numeric value between 0 and 100");
|
|
|
|
}
|
|
|
|
if (!ModelState.IsValid)
|
|
|
|
{
|
|
|
|
return this.CreateValidationError(ModelState);
|
2022-11-28 12:58:18 +01:00
|
|
|
}
|
|
|
|
|
2023-05-11 10:33:33 +02:00
|
|
|
var appliedDivisibility = paymentMethodDivisibility;
|
2022-11-28 09:53:08 +01:00
|
|
|
switch (request.RefundVariant)
|
|
|
|
{
|
|
|
|
case RefundVariant.RateThen:
|
2023-05-11 10:33:33 +02:00
|
|
|
createPullPayment.Currency = cryptoCode;
|
|
|
|
createPullPayment.Amount = paidAmount;
|
2022-11-28 09:53:08 +01:00
|
|
|
createPullPayment.AutoApproveClaims = true;
|
|
|
|
break;
|
2023-01-06 14:18:07 +01:00
|
|
|
|
2022-11-28 09:53:08 +01:00
|
|
|
case RefundVariant.CurrentRate:
|
2023-05-11 10:33:33 +02:00
|
|
|
createPullPayment.Currency = cryptoCode;
|
|
|
|
createPullPayment.Amount = Math.Round(paidCurrency / rateResult.BidAsk.Bid, appliedDivisibility);
|
2022-11-28 09:53:08 +01:00
|
|
|
createPullPayment.AutoApproveClaims = true;
|
|
|
|
break;
|
2023-01-06 14:18:07 +01:00
|
|
|
|
2022-11-28 09:53:08 +01:00
|
|
|
case RefundVariant.Fiat:
|
2023-05-11 10:33:33 +02:00
|
|
|
appliedDivisibility = cdCurrency.Divisibility;
|
2022-11-28 09:53:08 +01:00
|
|
|
createPullPayment.Currency = invoice.Currency;
|
|
|
|
createPullPayment.Amount = paidCurrency;
|
|
|
|
createPullPayment.AutoApproveClaims = false;
|
|
|
|
break;
|
2023-01-06 14:18:07 +01:00
|
|
|
|
2023-05-11 10:33:33 +02:00
|
|
|
case RefundVariant.OverpaidAmount:
|
|
|
|
if (invoice.ExceptionStatus != InvoiceExceptionStatus.PaidOver)
|
|
|
|
{
|
|
|
|
ModelState.AddModelError(nameof(request.RefundVariant), "Invoice is not overpaid");
|
|
|
|
}
|
|
|
|
if (!ModelState.IsValid)
|
|
|
|
{
|
|
|
|
return this.CreateValidationError(ModelState);
|
|
|
|
}
|
|
|
|
|
2023-07-19 11:47:32 +02:00
|
|
|
var dueAmount = accounting.TotalDue;
|
2023-05-11 10:33:33 +02:00
|
|
|
createPullPayment.Currency = cryptoCode;
|
|
|
|
createPullPayment.Amount = Math.Round(paidAmount - dueAmount, appliedDivisibility);
|
|
|
|
createPullPayment.AutoApproveClaims = true;
|
|
|
|
break;
|
|
|
|
|
2022-11-28 09:53:08 +01:00
|
|
|
case RefundVariant.Custom:
|
2023-01-06 14:18:07 +01:00
|
|
|
if (request.CustomAmount is null || (request.CustomAmount is decimal v && v <= 0))
|
|
|
|
{
|
2023-05-11 10:33:33 +02:00
|
|
|
ModelState.AddModelError(nameof(request.CustomAmount), "Amount must be greater than 0");
|
2022-11-28 09:53:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
string.IsNullOrEmpty(request.CustomCurrency) ||
|
|
|
|
_currencyNameTable.GetCurrencyData(request.CustomCurrency, false) == null
|
|
|
|
)
|
|
|
|
{
|
|
|
|
ModelState.AddModelError(nameof(request.CustomCurrency), "Invalid currency");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rateResult.BidAsk is null)
|
|
|
|
{
|
|
|
|
ModelState.AddModelError(nameof(request.RefundVariant),
|
|
|
|
$"Impossible to fetch rate: {rateResult.EvaluatedRule}");
|
|
|
|
}
|
|
|
|
|
2022-11-28 12:58:18 +01:00
|
|
|
if (!ModelState.IsValid || request.CustomAmount is null)
|
2022-11-28 09:53:08 +01:00
|
|
|
{
|
|
|
|
return this.CreateValidationError(ModelState);
|
|
|
|
}
|
|
|
|
|
|
|
|
createPullPayment.Currency = request.CustomCurrency;
|
2022-11-28 12:58:18 +01:00
|
|
|
createPullPayment.Amount = request.CustomAmount.Value;
|
2022-11-28 09:53:08 +01:00
|
|
|
createPullPayment.AutoApproveClaims = paymentMethodId.CryptoCode == request.CustomCurrency;
|
|
|
|
break;
|
2023-01-06 14:18:07 +01:00
|
|
|
|
2022-11-28 09:53:08 +01:00
|
|
|
default:
|
|
|
|
ModelState.AddModelError(nameof(request.RefundVariant), "Please select a valid refund option");
|
|
|
|
return this.CreateValidationError(ModelState);
|
|
|
|
}
|
2023-05-11 10:33:33 +02:00
|
|
|
|
|
|
|
// reduce by percentage
|
|
|
|
if (request.SubtractPercentage is > 0 and <= 100)
|
|
|
|
{
|
|
|
|
var reduceByAmount = createPullPayment.Amount * (request.SubtractPercentage / 100);
|
|
|
|
createPullPayment.Amount = Math.Round(createPullPayment.Amount - reduceByAmount, appliedDivisibility);
|
|
|
|
}
|
2022-11-28 09:53:08 +01:00
|
|
|
|
2023-12-14 11:15:36 +01:00
|
|
|
createPullPayment.AutoApproveClaims = createPullPayment.AutoApproveClaims && (await _authorizationService.AuthorizeAsync(User, createPullPayment.StoreId ,Policies.CanCreatePullPayments)).Succeeded;
|
2022-11-28 09:53:08 +01:00
|
|
|
var ppId = await _pullPaymentService.CreatePullPayment(createPullPayment);
|
2023-01-06 14:18:07 +01:00
|
|
|
|
2022-11-28 09:53:08 +01:00
|
|
|
await using var ctx = _dbContextFactory.CreateContext();
|
2023-11-21 04:52:40 +01:00
|
|
|
|
2022-11-28 09:53:08 +01:00
|
|
|
ctx.Refunds.Add(new RefundData
|
|
|
|
{
|
|
|
|
InvoiceDataId = invoice.Id,
|
|
|
|
PullPaymentDataId = ppId
|
|
|
|
});
|
|
|
|
await ctx.SaveChangesAsync(cancellationToken);
|
|
|
|
|
|
|
|
var pp = await _pullPaymentService.GetPullPayment(ppId, false);
|
|
|
|
return this.Ok(CreatePullPaymentData(pp));
|
|
|
|
}
|
|
|
|
|
|
|
|
private Client.Models.PullPaymentData CreatePullPaymentData(Data.PullPaymentData pp)
|
|
|
|
{
|
|
|
|
var ppBlob = pp.GetBlob();
|
|
|
|
return new BTCPayServer.Client.Models.PullPaymentData()
|
|
|
|
{
|
|
|
|
Id = pp.Id,
|
|
|
|
StartsAt = pp.StartDate,
|
|
|
|
ExpiresAt = pp.EndDate,
|
|
|
|
Amount = ppBlob.Limit,
|
|
|
|
Name = ppBlob.Name,
|
|
|
|
Description = ppBlob.Description,
|
|
|
|
Currency = ppBlob.Currency,
|
|
|
|
Period = ppBlob.Period,
|
|
|
|
Archived = pp.Archived,
|
|
|
|
AutoApproveClaims = ppBlob.AutoApproveClaims,
|
|
|
|
BOLT11Expiration = ppBlob.BOLT11Expiration,
|
|
|
|
ViewLink = _linkGenerator.GetUriByAction(
|
|
|
|
nameof(UIPullPaymentController.ViewPullPayment),
|
|
|
|
"UIPullPayment",
|
|
|
|
new { pullPaymentId = pp.Id },
|
|
|
|
Request.Scheme,
|
|
|
|
Request.Host,
|
|
|
|
Request.PathBase)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-04-08 08:57:01 +02:00
|
|
|
private IActionResult InvoiceNotFound()
|
|
|
|
{
|
|
|
|
return this.CreateAPIError(404, "invoice-not-found", "The invoice was not found");
|
|
|
|
}
|
|
|
|
private IActionResult StoreNotFound()
|
|
|
|
{
|
|
|
|
return this.CreateAPIError(404, "store-not-found", "The store was not found");
|
|
|
|
}
|
|
|
|
|
2021-05-14 09:16:19 +02:00
|
|
|
private InvoicePaymentMethodDataModel[] ToPaymentMethodModels(InvoiceEntity entity, bool includeAccountedPaymentOnly)
|
2020-09-10 13:54:36 +02:00
|
|
|
{
|
|
|
|
return entity.GetPaymentMethods().Select(
|
|
|
|
method =>
|
|
|
|
{
|
|
|
|
var accounting = method.Calculate();
|
|
|
|
var details = method.GetPaymentMethodDetails();
|
2021-05-14 09:16:19 +02:00
|
|
|
var payments = method.ParentEntity.GetPayments(includeAccountedPaymentOnly).Where(paymentEntity =>
|
2020-09-10 13:54:36 +02:00
|
|
|
paymentEntity.GetPaymentMethodId() == method.GetId());
|
|
|
|
|
2023-01-13 09:29:41 +01:00
|
|
|
return new InvoicePaymentMethodDataModel
|
2020-09-10 13:54:36 +02:00
|
|
|
{
|
2021-04-07 06:08:42 +02:00
|
|
|
Activated = details.Activated,
|
2020-09-10 13:54:36 +02:00
|
|
|
PaymentMethod = method.GetId().ToStringNormalized(),
|
2021-11-15 06:25:59 +01:00
|
|
|
CryptoCode = method.GetId().CryptoCode,
|
2020-09-10 13:54:36 +02:00
|
|
|
Destination = details.GetPaymentDestination(),
|
|
|
|
Rate = method.Rate,
|
2023-07-19 11:47:32 +02:00
|
|
|
Due = accounting.DueUncapped,
|
|
|
|
TotalPaid = accounting.Paid,
|
|
|
|
PaymentMethodPaid = accounting.CryptoPaid,
|
|
|
|
Amount = accounting.TotalDue,
|
|
|
|
NetworkFee = accounting.NetworkFee,
|
2020-09-10 13:54:36 +02:00
|
|
|
PaymentLink =
|
2023-04-10 09:38:49 +02:00
|
|
|
method.GetId().PaymentType.GetPaymentLink(method.Network, entity, details, accounting.Due,
|
2020-09-10 13:54:36 +02:00
|
|
|
Request.GetAbsoluteRoot()),
|
2022-02-07 09:39:48 +01:00
|
|
|
Payments = payments.Select(paymentEntity => ToPaymentModel(entity, paymentEntity)).ToList(),
|
|
|
|
AdditionalData = details.GetAdditionalData()
|
2020-09-10 13:54:36 +02:00
|
|
|
};
|
|
|
|
}).ToArray();
|
|
|
|
}
|
2021-12-31 08:59:02 +01:00
|
|
|
|
2021-10-05 11:10:41 +02:00
|
|
|
public static InvoicePaymentMethodDataModel.Payment ToPaymentModel(InvoiceEntity entity, PaymentEntity paymentEntity)
|
|
|
|
{
|
|
|
|
var data = paymentEntity.GetCryptoPaymentData();
|
|
|
|
return new InvoicePaymentMethodDataModel.Payment()
|
|
|
|
{
|
|
|
|
Destination = data.GetDestination(),
|
|
|
|
Id = data.GetPaymentId(),
|
|
|
|
Status = !paymentEntity.Accounted
|
|
|
|
? InvoicePaymentMethodDataModel.Payment.PaymentStatus.Invalid
|
|
|
|
: data.PaymentConfirmed(paymentEntity, entity.SpeedPolicy) || data.PaymentCompleted(paymentEntity)
|
|
|
|
? InvoicePaymentMethodDataModel.Payment.PaymentStatus.Settled
|
|
|
|
: InvoicePaymentMethodDataModel.Payment.PaymentStatus.Processing,
|
|
|
|
Fee = paymentEntity.NetworkFee,
|
|
|
|
Value = data.GetValue(),
|
|
|
|
ReceivedDate = paymentEntity.ReceivedTime.DateTime
|
|
|
|
};
|
|
|
|
}
|
2022-07-06 15:17:33 +02:00
|
|
|
|
2020-07-24 09:40:37 +02:00
|
|
|
private InvoiceData ToModel(InvoiceEntity entity)
|
2022-07-06 15:17:33 +02:00
|
|
|
{
|
|
|
|
return ToModel(entity, _linkGenerator, Request);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static InvoiceData ToModel(InvoiceEntity entity, LinkGenerator linkGenerator, HttpRequest? request)
|
2020-07-22 13:58:41 +02:00
|
|
|
{
|
2021-10-06 11:19:34 +02:00
|
|
|
var statuses = new List<InvoiceStatus>();
|
|
|
|
var state = entity.GetInvoiceState();
|
|
|
|
if (state.CanMarkComplete())
|
|
|
|
{
|
|
|
|
statuses.Add(InvoiceStatus.Settled);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state.CanMarkInvalid())
|
|
|
|
{
|
|
|
|
statuses.Add(InvoiceStatus.Invalid);
|
|
|
|
}
|
2020-07-22 13:58:41 +02:00
|
|
|
return new InvoiceData()
|
|
|
|
{
|
2021-06-24 16:15:51 +02:00
|
|
|
StoreId = entity.StoreId,
|
2020-08-26 14:31:08 +02:00
|
|
|
ExpirationTime = entity.ExpirationTime,
|
|
|
|
MonitoringExpiration = entity.MonitoringExpiration,
|
|
|
|
CreatedTime = entity.InvoiceTime,
|
2020-08-25 07:33:00 +02:00
|
|
|
Amount = entity.Price,
|
2021-08-03 10:03:00 +02:00
|
|
|
Type = entity.Type,
|
2020-07-22 13:58:41 +02:00
|
|
|
Id = entity.Id,
|
2023-01-06 14:18:07 +01:00
|
|
|
CheckoutLink = request is null ? null : linkGenerator.CheckoutLink(entity.Id, request.Scheme, request.Host, request.PathBase),
|
2020-11-23 07:57:05 +01:00
|
|
|
Status = entity.Status.ToModernStatus(),
|
2020-07-27 10:43:35 +02:00
|
|
|
AdditionalStatus = entity.ExceptionStatus,
|
2020-08-25 07:33:00 +02:00
|
|
|
Currency = entity.Currency,
|
2021-11-01 07:53:33 +01:00
|
|
|
Archived = entity.Archived,
|
2020-08-25 07:33:00 +02:00
|
|
|
Metadata = entity.Metadata.ToJObject(),
|
2021-10-06 11:19:34 +02:00
|
|
|
AvailableStatusesForManualMarking = statuses.ToArray(),
|
2020-07-22 13:58:41 +02:00
|
|
|
Checkout = new CreateInvoiceRequest.CheckoutOptions()
|
|
|
|
{
|
2020-08-26 07:01:39 +02:00
|
|
|
Expiration = entity.ExpirationTime - entity.InvoiceTime,
|
|
|
|
Monitoring = entity.MonitoringExpiration - entity.ExpirationTime,
|
2020-07-22 13:58:41 +02:00
|
|
|
PaymentTolerance = entity.PaymentTolerance,
|
|
|
|
PaymentMethods =
|
2020-08-26 14:24:37 +02:00
|
|
|
entity.GetPaymentMethods().Select(method => method.GetId().ToStringNormalized()).ToArray(),
|
2021-08-30 00:54:54 +02:00
|
|
|
DefaultPaymentMethod = entity.DefaultPaymentMethod,
|
2020-12-10 15:34:50 +01:00
|
|
|
SpeedPolicy = entity.SpeedPolicy,
|
2021-03-02 03:50:01 +01:00
|
|
|
DefaultLanguage = entity.DefaultLanguage,
|
|
|
|
RedirectAutomatically = entity.RedirectAutomatically,
|
2021-10-27 16:32:56 +02:00
|
|
|
RequiresRefundEmail = entity.RequiresRefundEmail,
|
2022-11-02 10:21:33 +01:00
|
|
|
CheckoutType = entity.CheckoutType,
|
2021-03-02 03:50:01 +01:00
|
|
|
RedirectURL = entity.RedirectURLTemplate
|
2022-07-06 14:14:55 +02:00
|
|
|
},
|
|
|
|
Receipt = entity.ReceiptOptions
|
2020-07-22 13:58:41 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|