mirror of
https://github.com/btcpayserver/btcpayserver.git
synced 2024-11-20 02:28:31 +01:00
339 lines
12 KiB
C#
339 lines
12 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Security.Claims;
|
|
using System.Threading.Tasks;
|
|
using BTCPayServer.Data;
|
|
using BTCPayServer.Filters;
|
|
using BTCPayServer.Models;
|
|
using BTCPayServer.Models.PaymentRequestViewModels;
|
|
using BTCPayServer.PaymentRequest;
|
|
using BTCPayServer.Payments;
|
|
using BTCPayServer.Payments.Lightning;
|
|
using BTCPayServer.Rating;
|
|
using BTCPayServer.Security;
|
|
using BTCPayServer.Services.Invoices;
|
|
using BTCPayServer.Services.PaymentRequests;
|
|
using BTCPayServer.Services.Rates;
|
|
using BTCPayServer.Services.Stores;
|
|
using Ganss.XSS;
|
|
using Microsoft.AspNetCore.Authorization;
|
|
using Microsoft.AspNetCore.Http.Extensions;
|
|
using Microsoft.AspNetCore.Identity;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.AspNetCore.Mvc.Rendering;
|
|
using NBitpayClient;
|
|
|
|
namespace BTCPayServer.Controllers
|
|
{
|
|
[Route("payment-requests")]
|
|
[Authorize(AuthenticationSchemes = Policies.CookieAuthentication)]
|
|
public class PaymentRequestController : Controller
|
|
{
|
|
private readonly InvoiceController _InvoiceController;
|
|
private readonly UserManager<ApplicationUser> _UserManager;
|
|
private readonly StoreRepository _StoreRepository;
|
|
private readonly RateFetcher _RateFetcher;
|
|
private readonly BTCPayNetworkProvider _BtcPayNetworkProvider;
|
|
private readonly PaymentRequestRepository _PaymentRequestRepository;
|
|
private readonly PaymentRequestService _PaymentRequestService;
|
|
private readonly EventAggregator _EventAggregator;
|
|
private readonly CurrencyNameTable _Currencies;
|
|
private readonly HtmlSanitizer _htmlSanitizer;
|
|
|
|
public PaymentRequestController(
|
|
InvoiceController invoiceController,
|
|
UserManager<ApplicationUser> userManager,
|
|
StoreRepository storeRepository,
|
|
RateFetcher rateFetcher,
|
|
BTCPayNetworkProvider btcPayNetworkProvider,
|
|
PaymentRequestRepository paymentRequestRepository,
|
|
PaymentRequestService paymentRequestService,
|
|
EventAggregator eventAggregator,
|
|
CurrencyNameTable currencies,
|
|
HtmlSanitizer htmlSanitizer)
|
|
{
|
|
_InvoiceController = invoiceController;
|
|
_UserManager = userManager;
|
|
_StoreRepository = storeRepository;
|
|
_RateFetcher = rateFetcher;
|
|
_BtcPayNetworkProvider = btcPayNetworkProvider;
|
|
_PaymentRequestRepository = paymentRequestRepository;
|
|
_PaymentRequestService = paymentRequestService;
|
|
_EventAggregator = eventAggregator;
|
|
_Currencies = currencies;
|
|
_htmlSanitizer = htmlSanitizer;
|
|
}
|
|
|
|
[HttpGet]
|
|
[Route("")]
|
|
[BitpayAPIConstraint(false)]
|
|
public async Task<IActionResult> GetPaymentRequests(int skip = 0, int count = 50, string statusMessage = null)
|
|
{
|
|
var result = await _PaymentRequestRepository.FindPaymentRequests(new PaymentRequestQuery()
|
|
{
|
|
UserId = GetUserId(), Skip = skip, Count = count
|
|
});
|
|
return View(new ListPaymentRequestsViewModel()
|
|
{
|
|
Skip = skip,
|
|
StatusMessage = statusMessage,
|
|
Count = count,
|
|
Total = result.Total,
|
|
Items = result.Items.Select(data => new ViewPaymentRequestViewModel(data)).ToList()
|
|
});
|
|
}
|
|
|
|
[HttpGet]
|
|
[Route("edit/{id?}")]
|
|
public async Task<IActionResult> EditPaymentRequest(string id, string statusMessage = null)
|
|
{
|
|
SelectList stores = null;
|
|
var data = await _PaymentRequestRepository.FindPaymentRequest(id, GetUserId());
|
|
if (data == null && !string.IsNullOrEmpty(id))
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
if (data != null && data.Status != PaymentRequestData.PaymentRequestStatus.Creating)
|
|
{
|
|
return RedirectToAction("ViewPaymentRequest", new
|
|
{
|
|
id
|
|
});
|
|
}
|
|
stores = new SelectList(await _StoreRepository.GetStoresByUserId(GetUserId()), nameof(StoreData.Id),
|
|
nameof(StoreData.StoreName), data?.StoreDataId);
|
|
if (!stores.Any())
|
|
{
|
|
return RedirectToAction("GetPaymentRequests",
|
|
new
|
|
{
|
|
StatusMessage = "Error: You need to create at least one store before creating a payment request"
|
|
});
|
|
}
|
|
|
|
return View(new UpdatePaymentRequestViewModel(data)
|
|
{
|
|
Stores = stores,
|
|
StatusMessage = statusMessage
|
|
});
|
|
}
|
|
|
|
[HttpPost]
|
|
[Route("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 != null && data.Status != PaymentRequestData.PaymentRequestStatus.Creating)
|
|
{
|
|
return RedirectToAction("ViewPaymentRequest", new
|
|
{
|
|
id
|
|
});
|
|
}
|
|
|
|
if (!ModelState.IsValid)
|
|
{
|
|
viewModel.Stores = new SelectList(await _StoreRepository.GetStoresByUserId(GetUserId()),
|
|
nameof(StoreData.Id),
|
|
nameof(StoreData.StoreName), data?.StoreDataId);
|
|
|
|
return View(viewModel);
|
|
}
|
|
|
|
if (data == null)
|
|
{
|
|
data = new PaymentRequestData();
|
|
}
|
|
|
|
data.StoreDataId = viewModel.StoreId;
|
|
var blob = data.GetBlob();
|
|
|
|
blob.Title = viewModel.Title;
|
|
blob.Email = viewModel.Email;
|
|
blob.Description = _htmlSanitizer.Sanitize(viewModel.Description);
|
|
blob.Amount = viewModel.Amount;
|
|
blob.ExpiryDate = viewModel.ExpiryDate;
|
|
blob.Currency = viewModel.Currency;
|
|
blob.EmbeddedCSS = viewModel.EmbeddedCSS;
|
|
blob.CustomCSSLink = viewModel.CustomCSSLink;
|
|
blob.AllowCustomPaymentAmounts = viewModel.AllowCustomPaymentAmounts;
|
|
|
|
data.SetBlob(blob);
|
|
data.Status = viewModel.Action == "publish" ? PaymentRequestData.PaymentRequestStatus.Pending : PaymentRequestData.PaymentRequestStatus.Creating;
|
|
data = await _PaymentRequestRepository.CreateOrUpdatePaymentRequest(data);
|
|
_EventAggregator.Publish(new PaymentRequestUpdated()
|
|
{
|
|
Data = data,
|
|
PaymentRequestId = data.Id
|
|
});
|
|
return RedirectToAction("EditPaymentRequest", new {id = data.Id, StatusMessage = "Saved"});
|
|
}
|
|
|
|
[HttpGet]
|
|
[Route("{id}/remove")]
|
|
[BitpayAPIConstraint(false)]
|
|
public async Task<IActionResult> RemovePaymentRequestPrompt(string id)
|
|
{
|
|
var data = await _PaymentRequestRepository.FindPaymentRequest(id, GetUserId());
|
|
if (data == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
var blob = data.GetBlob();
|
|
return View("Confirm", new ConfirmModel()
|
|
{
|
|
Title = $"Remove Payment Request",
|
|
Description = $"Are you sure to remove access to remove payment request '{blob.Title}' ?",
|
|
Action = "Delete"
|
|
});
|
|
}
|
|
|
|
[HttpPost]
|
|
[Route("{id}/remove")]
|
|
[BitpayAPIConstraint(false)]
|
|
public async Task<IActionResult> RemovePaymentRequest(string id)
|
|
{
|
|
var result = await _PaymentRequestRepository.RemovePaymentRequest(id, GetUserId());
|
|
if (result)
|
|
{
|
|
return RedirectToAction("GetPaymentRequests",
|
|
new {StatusMessage = "Payment request successfully removed"});
|
|
}
|
|
else
|
|
{
|
|
return RedirectToAction("GetPaymentRequests",
|
|
new
|
|
{
|
|
StatusMessage =
|
|
"Payment request could not be removed. Any request that has generated invoices cannot be removed."
|
|
});
|
|
}
|
|
}
|
|
|
|
[HttpGet]
|
|
[Route("{id}")]
|
|
[AllowAnonymous]
|
|
public async Task<IActionResult> ViewPaymentRequest(string id)
|
|
{
|
|
var result = await _PaymentRequestService.GetPaymentRequest(id, GetUserId());
|
|
if (result == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
return View(result);
|
|
}
|
|
|
|
[HttpGet]
|
|
[Route("{id}/pay")]
|
|
[AllowAnonymous]
|
|
public async Task<IActionResult> PayPaymentRequest(string id, bool redirectToInvoice = true, decimal? amount = null)
|
|
{
|
|
var result = ((await ViewPaymentRequest(id)) as ViewResult)?.Model as ViewPaymentRequestViewModel;
|
|
if (result == null)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
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 statusesAllowedToDisplay = new List<InvoiceStatus>()
|
|
{
|
|
InvoiceStatus.New
|
|
};
|
|
var validInvoice = result.Invoices.FirstOrDefault(invoice =>
|
|
Enum.TryParse<InvoiceStatus>(invoice.Status, true, out var status) &&
|
|
statusesAllowedToDisplay.Contains(status));
|
|
|
|
if (validInvoice != null)
|
|
{
|
|
if (redirectToInvoice)
|
|
{
|
|
return RedirectToAction("Checkout", "Invoice", new {Id = validInvoice.Id});
|
|
}
|
|
|
|
return Ok(validInvoice.Id);
|
|
}
|
|
|
|
if (result.AllowCustomPaymentAmounts && amount != null)
|
|
{
|
|
var invoiceAmount = result.AmountDue < amount ? result.AmountDue : amount;
|
|
|
|
return await CreateInvoiceForPaymentRequest(id, redirectToInvoice, result, invoiceAmount);
|
|
}
|
|
|
|
|
|
return await CreateInvoiceForPaymentRequest(id, redirectToInvoice, result);
|
|
}
|
|
|
|
private async Task<IActionResult> CreateInvoiceForPaymentRequest(string id,
|
|
bool redirectToInvoice,
|
|
ViewPaymentRequestViewModel result,
|
|
decimal? amount = null)
|
|
{
|
|
var pr = await _PaymentRequestRepository.FindPaymentRequest(id, null);
|
|
var blob = pr.GetBlob();
|
|
var store = pr.StoreData;
|
|
store.AdditionalClaims.Add(new Claim(Policies.CanCreateInvoice.Key, store.Id));
|
|
try
|
|
{
|
|
var newInvoiceId = (await _InvoiceController.CreateInvoiceCore(new Invoice()
|
|
{
|
|
OrderId = $"{PaymentRequestRepository.GetOrderIdForPaymentRequest(id)}",
|
|
Currency = blob.Currency,
|
|
Price = amount.GetValueOrDefault(result.AmountDue) ,
|
|
FullNotifications = true,
|
|
BuyerEmail = result.Email,
|
|
RedirectURL = Request.GetDisplayUrl().Replace("/pay", "", StringComparison.InvariantCulture),
|
|
}, store, HttpContext.Request.GetAbsoluteRoot())).Data.Id;
|
|
|
|
if (redirectToInvoice)
|
|
{
|
|
return RedirectToAction("Checkout", "Invoice", new {Id = newInvoiceId});
|
|
}
|
|
|
|
return Ok(newInvoiceId);
|
|
}
|
|
catch (BitpayHttpException e)
|
|
{
|
|
return BadRequest(e.Message);
|
|
}
|
|
}
|
|
|
|
private string GetUserId()
|
|
{
|
|
return _UserManager.GetUserId(User);
|
|
}
|
|
}
|
|
}
|