2021-10-11 04:11:02 +02:00
|
|
|
using System;
|
2022-11-02 10:21:33 +01:00
|
|
|
using System.Globalization;
|
2021-10-11 04:11:02 +02:00
|
|
|
using System.Linq;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using BTCPayServer.Data;
|
|
|
|
using BTCPayServer.Filters;
|
2022-11-24 00:53:32 +01:00
|
|
|
using BTCPayServer.Lightning;
|
2021-10-11 04:11:02 +02:00
|
|
|
using BTCPayServer.Payments;
|
2021-12-31 08:59:02 +01:00
|
|
|
using BTCPayServer.Services;
|
2021-10-11 04:11:02 +02:00
|
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
using NBitcoin;
|
|
|
|
|
|
|
|
namespace BTCPayServer.Controllers
|
|
|
|
{
|
2022-01-07 04:32:00 +01:00
|
|
|
public partial class UIInvoiceController
|
2021-10-11 04:11:02 +02:00
|
|
|
{
|
2021-10-11 11:01:32 +02:00
|
|
|
public class FakePaymentRequest
|
|
|
|
{
|
|
|
|
public Decimal Amount { get; set; }
|
2021-11-11 10:31:15 +01:00
|
|
|
public string CryptoCode { get; set; } = "BTC";
|
2022-11-24 00:53:32 +01:00
|
|
|
public string PaymentMethodId { get; set; } = "BTC";
|
2021-10-11 11:01:32 +02:00
|
|
|
}
|
2021-11-29 07:23:56 +01:00
|
|
|
|
|
|
|
public class MineBlocksRequest
|
|
|
|
{
|
2021-11-29 11:44:56 +01:00
|
|
|
public int BlockCount { get; set; } = 1;
|
2021-11-29 07:23:56 +01:00
|
|
|
public string CryptoCode { get; set; } = "BTC";
|
|
|
|
}
|
2021-12-31 08:59:02 +01:00
|
|
|
|
2022-11-02 10:21:33 +01:00
|
|
|
[HttpPost("i/{invoiceId}/test-payment")]
|
2021-10-11 05:32:09 +02:00
|
|
|
[CheatModeRoute]
|
|
|
|
public async Task<IActionResult> TestPayment(string invoiceId, FakePaymentRequest request, [FromServices] Cheater cheater)
|
2021-10-11 04:11:02 +02:00
|
|
|
{
|
|
|
|
var invoice = await _InvoiceRepository.GetInvoice(invoiceId);
|
|
|
|
var store = await _StoreRepository.FindStore(invoice.StoreId);
|
2022-11-02 10:21:33 +01:00
|
|
|
var isSats = request.CryptoCode.ToUpper(CultureInfo.InvariantCulture) == "SATS";
|
|
|
|
var cryptoCode = isSats ? "BTC" : request.CryptoCode;
|
|
|
|
var amount = new Money(request.Amount, isSats ? MoneyUnit.Satoshi : MoneyUnit.BTC);
|
2022-11-24 00:53:32 +01:00
|
|
|
var network = _NetworkProvider.GetNetwork<BTCPayNetwork>(cryptoCode).NBitcoinNetwork;
|
2023-01-06 14:18:07 +01:00
|
|
|
var paymentMethodId = new[] { store.GetDefaultPaymentId() }
|
2022-11-24 00:53:32 +01:00
|
|
|
.Concat(store.GetEnabledPaymentIds(_NetworkProvider))
|
|
|
|
.FirstOrDefault(p => p?.ToString() == request.PaymentMethodId);
|
2023-01-06 14:18:07 +01:00
|
|
|
|
2021-10-11 04:11:02 +02:00
|
|
|
try
|
|
|
|
{
|
|
|
|
var paymentMethod = invoice.GetPaymentMethod(paymentMethodId);
|
2022-11-24 00:53:32 +01:00
|
|
|
var destination = paymentMethod?.GetPaymentMethodDetails().GetPaymentDestination();
|
2023-01-06 14:18:07 +01:00
|
|
|
|
2022-11-24 00:53:32 +01:00
|
|
|
switch (paymentMethod?.GetId().PaymentType)
|
2021-10-11 11:01:32 +02:00
|
|
|
{
|
2022-11-24 00:53:32 +01:00
|
|
|
case BitcoinPaymentType:
|
|
|
|
var address = BitcoinAddress.Create(destination, network);
|
|
|
|
var txid = (await cheater.CashCow.SendToAddressAsync(address, amount)).ToString();
|
2023-01-06 14:18:07 +01:00
|
|
|
|
2022-11-24 00:53:32 +01:00
|
|
|
return Ok(new
|
|
|
|
{
|
|
|
|
Txid = txid,
|
2023-07-19 11:47:32 +02:00
|
|
|
AmountRemaining = paymentMethod.Calculate().Due - amount.ToDecimal(MoneyUnit.BTC),
|
2023-01-06 14:18:07 +01:00
|
|
|
SuccessMessage = $"Created transaction {txid}"
|
2022-11-24 00:53:32 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
case LightningPaymentType:
|
|
|
|
// requires the channels to be set up using the BTCPayServer.Tests/docker-lightning-channel-setup.sh script
|
|
|
|
LightningConnectionString.TryParse(Environment.GetEnvironmentVariable("BTCPAY_BTCEXTERNALLNDREST"), false, out var lnConnection);
|
|
|
|
var lnClient = LightningClientFactory.CreateClient(lnConnection, network);
|
|
|
|
var lnAmount = new LightMoney(amount.Satoshi, LightMoneyUnit.Satoshi);
|
|
|
|
var response = await lnClient.Pay(destination, new PayInvoiceParams { Amount = lnAmount });
|
|
|
|
|
|
|
|
if (response.Result == PayResult.Ok)
|
|
|
|
{
|
|
|
|
var bolt11 = BOLT11PaymentRequest.Parse(destination, network);
|
|
|
|
var paymentHash = bolt11.PaymentHash?.ToString();
|
2023-07-19 11:47:32 +02:00
|
|
|
var paid = response.Details.TotalAmount.ToDecimal(LightMoneyUnit.BTC);
|
2022-11-24 00:53:32 +01:00
|
|
|
return Ok(new
|
|
|
|
{
|
|
|
|
Txid = paymentHash,
|
2023-07-19 11:47:32 +02:00
|
|
|
AmountRemaining = paymentMethod.Calculate().TotalDue - paid,
|
2023-01-06 14:18:07 +01:00
|
|
|
SuccessMessage = $"Sent payment {paymentHash}"
|
2022-11-24 00:53:32 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
return UnprocessableEntity(new
|
|
|
|
{
|
2023-03-27 12:12:11 +02:00
|
|
|
ErrorMessage = response.ErrorDetail
|
2022-11-24 00:53:32 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
default:
|
|
|
|
return UnprocessableEntity(new
|
|
|
|
{
|
2023-03-27 12:12:11 +02:00
|
|
|
ErrorMessage = $"Payment method {paymentMethodId} is not supported"
|
2022-11-24 00:53:32 +01:00
|
|
|
});
|
|
|
|
}
|
2023-01-06 14:18:07 +01:00
|
|
|
|
2021-10-11 04:11:02 +02:00
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
2021-10-11 11:01:32 +02:00
|
|
|
return BadRequest(new
|
|
|
|
{
|
2023-03-27 12:12:11 +02:00
|
|
|
ErrorMessage = e.Message
|
2021-10-11 11:01:32 +02:00
|
|
|
});
|
2021-10-11 04:11:02 +02:00
|
|
|
}
|
|
|
|
}
|
2021-12-31 08:59:02 +01:00
|
|
|
|
2022-11-02 10:21:33 +01:00
|
|
|
[HttpPost("i/{invoiceId}/mine-blocks")]
|
2021-11-29 07:23:56 +01:00
|
|
|
[CheatModeRoute]
|
2021-12-10 12:31:04 +01:00
|
|
|
public IActionResult MineBlock(string invoiceId, MineBlocksRequest request, [FromServices] Cheater cheater)
|
2021-11-29 07:23:56 +01:00
|
|
|
{
|
2021-11-29 11:44:56 +01:00
|
|
|
var blockRewardBitcoinAddress = cheater.CashCow.GetNewAddress();
|
2021-11-29 07:23:56 +01:00
|
|
|
try
|
|
|
|
{
|
2021-12-31 08:59:02 +01:00
|
|
|
if (request.BlockCount > 0)
|
2021-11-29 07:23:56 +01:00
|
|
|
{
|
2021-11-29 11:44:56 +01:00
|
|
|
cheater.CashCow.GenerateToAddress(request.BlockCount, blockRewardBitcoinAddress);
|
2022-11-24 00:53:32 +01:00
|
|
|
return Ok(new { SuccessMessage = $"Mined {request.BlockCount} block{(request.BlockCount == 1 ? "" : "s")} " });
|
2021-11-29 07:23:56 +01:00
|
|
|
}
|
2022-11-24 00:53:32 +01:00
|
|
|
return BadRequest(new { ErrorMessage = "Number of blocks should be at least 1" });
|
2021-11-29 07:23:56 +01:00
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
2022-11-24 00:53:32 +01:00
|
|
|
return BadRequest(new { ErrorMessage = e.Message });
|
2021-11-29 07:23:56 +01:00
|
|
|
}
|
|
|
|
}
|
2021-10-11 04:11:02 +02:00
|
|
|
|
2022-11-02 10:21:33 +01:00
|
|
|
[HttpPost("i/{invoiceId}/expire")]
|
2021-10-11 05:32:09 +02:00
|
|
|
[CheatModeRoute]
|
2022-11-24 00:53:32 +01:00
|
|
|
public async Task<IActionResult> Expire(string invoiceId, int seconds, [FromServices] Cheater cheater)
|
2021-10-11 04:11:02 +02:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2022-11-24 00:53:32 +01:00
|
|
|
await cheater.UpdateInvoiceExpiry(invoiceId, TimeSpan.FromSeconds(seconds));
|
|
|
|
return Ok(new { SuccessMessage = $"Invoice set to expire in {seconds} seconds." });
|
2021-10-11 04:11:02 +02:00
|
|
|
}
|
|
|
|
catch (Exception e)
|
|
|
|
{
|
2021-10-11 10:58:01 +02:00
|
|
|
return BadRequest(new { ErrorMessage = e.Message });
|
2021-10-11 04:11:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|