mirror of
https://github.com/btcpayserver/btcpayserver.git
synced 2024-11-20 10:40:29 +01:00
5984edb7f4
* Disallow cancelling payment request when "Allow payee to create invoices in their own denomination" is not enabled close #2802 * Disallow cancelling pending invoice on BE * Update non-JS version of payment request view to disallow cancelling * Update CanCancelPaymentWhenPossible * Fix typo
369 lines
14 KiB
C#
369 lines
14 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using BTCPayServer.Abstractions.Constants;
|
|
using BTCPayServer.Abstractions.Extensions;
|
|
using BTCPayServer.Abstractions.Models;
|
|
using BTCPayServer.Client.Models;
|
|
using BTCPayServer.Data;
|
|
using BTCPayServer.Filters;
|
|
using BTCPayServer.Models.PaymentRequestViewModels;
|
|
using BTCPayServer.PaymentRequest;
|
|
using BTCPayServer.Services.Invoices;
|
|
using BTCPayServer.Services.PaymentRequests;
|
|
using BTCPayServer.Services.Rates;
|
|
using BTCPayServer.Services.Stores;
|
|
using Microsoft.AspNetCore.Authorization;
|
|
using Microsoft.AspNetCore.Identity;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.AspNetCore.Mvc.Rendering;
|
|
using Microsoft.AspNetCore.Routing;
|
|
using Newtonsoft.Json.Linq;
|
|
using PaymentRequestData = BTCPayServer.Data.PaymentRequestData;
|
|
using StoreData = BTCPayServer.Data.StoreData;
|
|
|
|
namespace BTCPayServer.Controllers
|
|
{
|
|
[Route("payment-requests")]
|
|
[Authorize(AuthenticationSchemes = AuthenticationSchemes.Cookie)]
|
|
public class PaymentRequestController : Controller
|
|
{
|
|
private readonly InvoiceController _InvoiceController;
|
|
private readonly UserManager<ApplicationUser> _UserManager;
|
|
private readonly StoreRepository _StoreRepository;
|
|
private readonly PaymentRequestRepository _PaymentRequestRepository;
|
|
private readonly PaymentRequestService _PaymentRequestService;
|
|
private readonly EventAggregator _EventAggregator;
|
|
private readonly CurrencyNameTable _Currencies;
|
|
private readonly InvoiceRepository _InvoiceRepository;
|
|
private readonly LinkGenerator _linkGenerator;
|
|
|
|
public PaymentRequestController(
|
|
InvoiceController invoiceController,
|
|
UserManager<ApplicationUser> userManager,
|
|
StoreRepository storeRepository,
|
|
PaymentRequestRepository paymentRequestRepository,
|
|
PaymentRequestService paymentRequestService,
|
|
EventAggregator eventAggregator,
|
|
CurrencyNameTable currencies,
|
|
InvoiceRepository invoiceRepository,
|
|
LinkGenerator linkGenerator)
|
|
{
|
|
_InvoiceController = invoiceController;
|
|
_UserManager = userManager;
|
|
_StoreRepository = storeRepository;
|
|
_PaymentRequestRepository = paymentRequestRepository;
|
|
_PaymentRequestService = paymentRequestService;
|
|
_EventAggregator = eventAggregator;
|
|
_Currencies = currencies;
|
|
_InvoiceRepository = invoiceRepository;
|
|
_linkGenerator = linkGenerator;
|
|
}
|
|
|
|
[HttpGet("")]
|
|
[BitpayAPIConstraint(false)]
|
|
public async Task<IActionResult> GetPaymentRequests(ListPaymentRequestsViewModel model = null)
|
|
{
|
|
model = this.ParseListQuery(model ?? new ListPaymentRequestsViewModel());
|
|
|
|
var includeArchived = new SearchString(model.SearchTerm).GetFilterBool("includearchived") == true;
|
|
var result = await _PaymentRequestRepository.FindPaymentRequests(new PaymentRequestQuery()
|
|
{
|
|
UserId = GetUserId(),
|
|
Skip = model.Skip,
|
|
Count = model.Count,
|
|
IncludeArchived = includeArchived
|
|
});
|
|
|
|
model.Total = result.Total;
|
|
model.Items = result.Items.Select(data => new ViewPaymentRequestViewModel(data)).ToList();
|
|
return View(model);
|
|
}
|
|
|
|
[HttpGet("edit/{id?}")]
|
|
public async Task<IActionResult> EditPaymentRequest(string id)
|
|
{
|
|
var data = await _PaymentRequestRepository.FindPaymentRequest(id, GetUserId());
|
|
if (data == null && !string.IsNullOrEmpty(id))
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
SelectList stores = new SelectList(await _StoreRepository.GetStoresByUserId(GetUserId()), nameof(StoreData.Id),
|
|
nameof(StoreData.StoreName), data?.StoreDataId);
|
|
if (!stores.Any())
|
|
{
|
|
TempData.SetStatusMessageModel(new StatusMessageModel
|
|
{
|
|
Html =
|
|
$"Error: You need to create at least one store. <a href='{Url.Action("CreateStore", "UserStores")}' class='alert-link'>Create store</a>",
|
|
Severity = StatusMessageModel.StatusSeverity.Error
|
|
});
|
|
return RedirectToAction("GetPaymentRequests");
|
|
}
|
|
|
|
return View(nameof(EditPaymentRequest), new UpdatePaymentRequestViewModel(data) { Stores = stores });
|
|
}
|
|
|
|
[HttpPost("edit/{id?}")]
|
|
public async Task<IActionResult> EditPaymentRequest(string id, UpdatePaymentRequestViewModel viewModel)
|
|
{
|
|
if (string.IsNullOrEmpty(viewModel.Currency) ||
|
|
_Currencies.GetCurrencyData(viewModel.Currency, false) == null)
|
|
ModelState.AddModelError(nameof(viewModel.Currency), "Invalid currency");
|
|
|
|
var data = await _PaymentRequestRepository.FindPaymentRequest(id, GetUserId());
|
|
if (data == null && !string.IsNullOrEmpty(id))
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
if (data?.Archived is true && viewModel.Archived is true)
|
|
{
|
|
ModelState.AddModelError(string.Empty, "You cannot edit an archived payment request.");
|
|
}
|
|
|
|
if (!ModelState.IsValid)
|
|
{
|
|
viewModel.Stores = new SelectList(await _StoreRepository.GetStoresByUserId(GetUserId()),
|
|
nameof(StoreData.Id),
|
|
nameof(StoreData.StoreName), data?.StoreDataId);
|
|
|
|
return View(nameof(EditPaymentRequest), viewModel);
|
|
}
|
|
|
|
if (data == null)
|
|
{
|
|
data = new PaymentRequestData();
|
|
}
|
|
|
|
data.StoreDataId = viewModel.StoreId;
|
|
data.Archived = viewModel.Archived;
|
|
var blob = data.GetBlob();
|
|
|
|
blob.Title = viewModel.Title;
|
|
blob.Email = viewModel.Email;
|
|
blob.Description = viewModel.Description;
|
|
blob.Amount = viewModel.Amount;
|
|
blob.ExpiryDate = viewModel.ExpiryDate?.ToUniversalTime();
|
|
blob.Currency = viewModel.Currency;
|
|
blob.EmbeddedCSS = viewModel.EmbeddedCSS;
|
|
blob.CustomCSSLink = viewModel.CustomCSSLink;
|
|
blob.AllowCustomPaymentAmounts = viewModel.AllowCustomPaymentAmounts;
|
|
|
|
data.SetBlob(blob);
|
|
if (string.IsNullOrEmpty(id))
|
|
{
|
|
data.Created = DateTimeOffset.UtcNow;
|
|
}
|
|
|
|
data = await _PaymentRequestRepository.CreateOrUpdatePaymentRequest(data);
|
|
_EventAggregator.Publish(new PaymentRequestUpdated { Data = data, PaymentRequestId = data.Id, });
|
|
|
|
TempData[WellKnownTempData.SuccessMessage] = "Saved";
|
|
return RedirectToAction(nameof(EditPaymentRequest), new { id = data.Id });
|
|
}
|
|
|
|
[HttpGet("{id}")]
|
|
[AllowAnonymous]
|
|
public async Task<IActionResult> ViewPaymentRequest(string id)
|
|
{
|
|
var result = await _PaymentRequestService.GetPaymentRequest(id, GetUserId());
|
|
if (result == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
result.HubPath = PaymentRequestHub.GetHubPath(Request);
|
|
return View(result);
|
|
}
|
|
|
|
[HttpGet("{id}/pay")]
|
|
[AllowAnonymous]
|
|
public async Task<IActionResult> PayPaymentRequest(string id, bool redirectToInvoice = true,
|
|
decimal? amount = null, CancellationToken cancellationToken = default)
|
|
{
|
|
if (amount.HasValue && amount.Value <= 0)
|
|
{
|
|
return BadRequest("Please provide an amount greater than 0");
|
|
}
|
|
|
|
var result = await _PaymentRequestService.GetPaymentRequest(id, GetUserId());
|
|
if (result == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
if (result.Archived)
|
|
{
|
|
if (redirectToInvoice)
|
|
{
|
|
return RedirectToAction("ViewPaymentRequest", new { Id = id });
|
|
}
|
|
|
|
return BadRequest("Payment Request cannot be paid as it has been archived");
|
|
}
|
|
|
|
result.HubPath = PaymentRequestHub.GetHubPath(Request);
|
|
if (result.AmountDue <= 0)
|
|
{
|
|
if (redirectToInvoice)
|
|
{
|
|
return RedirectToAction("ViewPaymentRequest", new { Id = id });
|
|
}
|
|
|
|
return BadRequest("Payment Request has already been settled.");
|
|
}
|
|
|
|
if (result.ExpiryDate.HasValue && DateTime.Now >= result.ExpiryDate)
|
|
{
|
|
if (redirectToInvoice)
|
|
{
|
|
return RedirectToAction("ViewPaymentRequest", new { Id = id });
|
|
}
|
|
|
|
return BadRequest("Payment Request has expired");
|
|
}
|
|
|
|
var stateAllowedToDisplay = new HashSet<InvoiceState>
|
|
{
|
|
new InvoiceState(InvoiceStatusLegacy.New, InvoiceExceptionStatus.None),
|
|
new InvoiceState(InvoiceStatusLegacy.New, InvoiceExceptionStatus.PaidPartial),
|
|
};
|
|
var currentInvoice = result
|
|
.Invoices
|
|
.FirstOrDefault(invoice => stateAllowedToDisplay.Contains(invoice.State));
|
|
if (currentInvoice != null)
|
|
{
|
|
if (redirectToInvoice)
|
|
{
|
|
return RedirectToAction("Checkout", "Invoice", new { currentInvoice.Id });
|
|
}
|
|
|
|
return Ok(currentInvoice.Id);
|
|
}
|
|
|
|
if (result.AllowCustomPaymentAmounts && amount != null)
|
|
amount = Math.Min(result.AmountDue, amount.Value);
|
|
else
|
|
amount = result.AmountDue;
|
|
|
|
var pr = await _PaymentRequestRepository.FindPaymentRequest(id, null, cancellationToken);
|
|
var blob = pr.GetBlob();
|
|
var store = pr.StoreData;
|
|
try
|
|
{
|
|
var redirectUrl = _linkGenerator.PaymentRequestLink(id, Request.Scheme, Request.Host, Request.PathBase);
|
|
|
|
var invoiceMetadata =
|
|
new InvoiceMetadata
|
|
{
|
|
OrderId = PaymentRequestRepository.GetOrderIdForPaymentRequest(id),
|
|
PaymentRequestId = id,
|
|
BuyerEmail = result.Email
|
|
};
|
|
|
|
var invoiceRequest =
|
|
new CreateInvoiceRequest
|
|
{
|
|
Metadata = invoiceMetadata.ToJObject(),
|
|
Currency = blob.Currency,
|
|
Amount = amount.Value,
|
|
Checkout = {RedirectURL = redirectUrl}
|
|
};
|
|
|
|
var additionalTags = new List<string> {PaymentRequestRepository.GetInternalTag(id)};
|
|
var newInvoice = await _InvoiceController.CreateInvoiceCoreRaw(invoiceRequest,store, "/",additionalTags, cancellationToken);
|
|
|
|
if (redirectToInvoice)
|
|
{
|
|
return RedirectToAction("Checkout", "Invoice", new { newInvoice.Id });
|
|
}
|
|
|
|
return Ok(newInvoice.Id);
|
|
}
|
|
catch (BitpayHttpException e)
|
|
{
|
|
return BadRequest(e.Message);
|
|
}
|
|
}
|
|
|
|
[HttpGet("{id}/cancel")]
|
|
public async Task<IActionResult> CancelUnpaidPendingInvoice(string id, bool redirect = true)
|
|
{
|
|
var result = await _PaymentRequestService.GetPaymentRequest(id, GetUserId());
|
|
if (result == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
if (!result.AllowCustomPaymentAmounts) {
|
|
return BadRequest("Not allowed to cancel this invoice");
|
|
}
|
|
|
|
var invoices = result.Invoices.Where(requestInvoice =>
|
|
requestInvoice.State.Status == InvoiceStatusLegacy.New && !requestInvoice.Payments.Any());
|
|
|
|
if (!invoices.Any())
|
|
{
|
|
return BadRequest("No unpaid pending invoice to cancel");
|
|
}
|
|
|
|
foreach (var invoice in invoices)
|
|
{
|
|
await _InvoiceRepository.MarkInvoiceStatus(invoice.Id, InvoiceStatus.Invalid);
|
|
}
|
|
|
|
if (redirect)
|
|
{
|
|
TempData[WellKnownTempData.SuccessMessage] = "Payment cancelled";
|
|
return RedirectToAction(nameof(ViewPaymentRequest), new { Id = id });
|
|
}
|
|
|
|
return Ok("Payment cancelled");
|
|
}
|
|
|
|
private string GetUserId()
|
|
{
|
|
return _UserManager.GetUserId(User);
|
|
}
|
|
|
|
[HttpGet("{id}/clone")]
|
|
public async Task<IActionResult> ClonePaymentRequest(string id)
|
|
{
|
|
var result = await EditPaymentRequest(id);
|
|
if (result is ViewResult viewResult)
|
|
{
|
|
var model = (UpdatePaymentRequestViewModel)viewResult.Model;
|
|
model.Id = null;
|
|
model.Archived = false;
|
|
model.ExpiryDate = null;
|
|
model.Title = $"Clone of {model.Title}";
|
|
return View("EditPaymentRequest", model);
|
|
}
|
|
|
|
return NotFound();
|
|
}
|
|
|
|
[HttpGet("{id}/archive")]
|
|
public async Task<IActionResult> TogglePaymentRequestArchival(string id)
|
|
{
|
|
var result = await EditPaymentRequest(id);
|
|
if (result is ViewResult viewResult)
|
|
{
|
|
var model = (UpdatePaymentRequestViewModel)viewResult.Model;
|
|
model.Archived = !model.Archived;
|
|
result = await EditPaymentRequest(id, model);
|
|
TempData[WellKnownTempData.SuccessMessage] = model.Archived
|
|
? "The payment request has been archived and will no longer appear in the payment request list by default again."
|
|
: "The payment request has been unarchived and will appear in the payment request list by default.";
|
|
return result;
|
|
}
|
|
|
|
return NotFound();
|
|
}
|
|
}
|
|
}
|