2018-07-26 15:32:24 +02:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Globalization;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Net.WebSockets;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using BTCPayServer.Data;
|
|
|
|
|
using BTCPayServer.HostedServices;
|
|
|
|
|
using BTCPayServer.ModelBinders;
|
|
|
|
|
using BTCPayServer.Models;
|
|
|
|
|
using BTCPayServer.Models.WalletViewModels;
|
|
|
|
|
using BTCPayServer.Security;
|
|
|
|
|
using BTCPayServer.Services;
|
2018-07-26 16:23:28 +02:00
|
|
|
|
using BTCPayServer.Services.Rates;
|
2018-07-26 15:32:24 +02:00
|
|
|
|
using BTCPayServer.Services.Stores;
|
|
|
|
|
using BTCPayServer.Services.Wallets;
|
|
|
|
|
using LedgerWallet;
|
|
|
|
|
using Microsoft.AspNetCore.Authorization;
|
|
|
|
|
using Microsoft.AspNetCore.Http;
|
|
|
|
|
using Microsoft.AspNetCore.Identity;
|
|
|
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
|
using Microsoft.Extensions.Options;
|
|
|
|
|
using NBitcoin;
|
|
|
|
|
using NBXplorer.DerivationStrategy;
|
2018-10-26 16:07:39 +02:00
|
|
|
|
using NBXplorer.Models;
|
2018-07-26 15:32:24 +02:00
|
|
|
|
using Newtonsoft.Json;
|
|
|
|
|
using static BTCPayServer.Controllers.StoresController;
|
|
|
|
|
|
|
|
|
|
namespace BTCPayServer.Controllers
|
|
|
|
|
{
|
|
|
|
|
[Route("wallets")]
|
|
|
|
|
[Authorize(AuthenticationSchemes = Policies.CookieAuthentication)]
|
|
|
|
|
[AutoValidateAntiforgeryToken]
|
2018-10-09 16:48:14 +02:00
|
|
|
|
public partial class WalletsController : Controller
|
2018-07-26 15:32:24 +02:00
|
|
|
|
{
|
2018-10-09 16:48:14 +02:00
|
|
|
|
public StoreRepository Repository { get; }
|
|
|
|
|
public BTCPayNetworkProvider NetworkProvider { get; }
|
|
|
|
|
public ExplorerClientProvider ExplorerClientProvider { get; }
|
|
|
|
|
|
2018-07-26 15:32:24 +02:00
|
|
|
|
private readonly UserManager<ApplicationUser> _userManager;
|
|
|
|
|
private readonly IOptions<MvcJsonOptions> _mvcJsonOptions;
|
|
|
|
|
private readonly NBXplorerDashboard _dashboard;
|
2018-10-09 16:48:14 +02:00
|
|
|
|
|
2018-07-26 15:32:24 +02:00
|
|
|
|
private readonly IFeeProviderFactory _feeRateProvider;
|
|
|
|
|
private readonly BTCPayWalletProvider _walletProvider;
|
2018-10-09 16:48:14 +02:00
|
|
|
|
public RateFetcher RateFetcher { get; }
|
2018-10-31 16:19:25 +01:00
|
|
|
|
[TempData]
|
|
|
|
|
public string StatusMessage { get; set; }
|
|
|
|
|
|
2018-07-26 16:23:28 +02:00
|
|
|
|
CurrencyNameTable _currencyTable;
|
2018-07-26 15:32:24 +02:00
|
|
|
|
public WalletsController(StoreRepository repo,
|
2018-07-26 16:23:28 +02:00
|
|
|
|
CurrencyNameTable currencyTable,
|
2018-07-26 15:32:24 +02:00
|
|
|
|
BTCPayNetworkProvider networkProvider,
|
|
|
|
|
UserManager<ApplicationUser> userManager,
|
|
|
|
|
IOptions<MvcJsonOptions> mvcJsonOptions,
|
|
|
|
|
NBXplorerDashboard dashboard,
|
2018-08-22 09:53:40 +02:00
|
|
|
|
RateFetcher rateProvider,
|
2018-07-26 15:32:24 +02:00
|
|
|
|
ExplorerClientProvider explorerProvider,
|
|
|
|
|
IFeeProviderFactory feeRateProvider,
|
|
|
|
|
BTCPayWalletProvider walletProvider)
|
|
|
|
|
{
|
2018-07-26 16:23:28 +02:00
|
|
|
|
_currencyTable = currencyTable;
|
2018-10-09 16:48:14 +02:00
|
|
|
|
Repository = repo;
|
|
|
|
|
RateFetcher = rateProvider;
|
|
|
|
|
NetworkProvider = networkProvider;
|
2018-07-26 15:32:24 +02:00
|
|
|
|
_userManager = userManager;
|
|
|
|
|
_mvcJsonOptions = mvcJsonOptions;
|
|
|
|
|
_dashboard = dashboard;
|
2018-10-09 16:48:14 +02:00
|
|
|
|
ExplorerClientProvider = explorerProvider;
|
2018-07-26 15:32:24 +02:00
|
|
|
|
_feeRateProvider = feeRateProvider;
|
|
|
|
|
_walletProvider = walletProvider;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<IActionResult> ListWallets()
|
|
|
|
|
{
|
|
|
|
|
var wallets = new ListWalletsViewModel();
|
2018-10-09 16:48:14 +02:00
|
|
|
|
var stores = await Repository.GetStoresByUserId(GetUserId());
|
2018-07-26 15:32:24 +02:00
|
|
|
|
|
|
|
|
|
var onChainWallets = stores
|
2018-10-09 16:48:14 +02:00
|
|
|
|
.SelectMany(s => s.GetSupportedPaymentMethods(NetworkProvider)
|
2019-05-08 16:39:11 +02:00
|
|
|
|
.OfType<DerivationSchemeSettings>()
|
2018-07-26 15:32:24 +02:00
|
|
|
|
.Select(d => ((Wallet: _walletProvider.GetWallet(d.Network),
|
2019-05-08 16:39:11 +02:00
|
|
|
|
DerivationStrategy: d.AccountDerivation,
|
2018-07-26 15:32:24 +02:00
|
|
|
|
Network: d.Network)))
|
|
|
|
|
.Where(_ => _.Wallet != null)
|
|
|
|
|
.Select(_ => (Wallet: _.Wallet,
|
|
|
|
|
Store: s,
|
|
|
|
|
Balance: GetBalanceString(_.Wallet, _.DerivationStrategy),
|
|
|
|
|
DerivationStrategy: _.DerivationStrategy,
|
|
|
|
|
Network: _.Network)))
|
|
|
|
|
.ToList();
|
|
|
|
|
|
|
|
|
|
foreach (var wallet in onChainWallets)
|
|
|
|
|
{
|
|
|
|
|
ListWalletsViewModel.WalletViewModel walletVm = new ListWalletsViewModel.WalletViewModel();
|
|
|
|
|
wallets.Wallets.Add(walletVm);
|
|
|
|
|
walletVm.Balance = await wallet.Balance + " " + wallet.Wallet.Network.CryptoCode;
|
|
|
|
|
if (!wallet.Store.HasClaim(Policies.CanModifyStoreSettings.Key))
|
|
|
|
|
{
|
|
|
|
|
walletVm.Balance = "";
|
|
|
|
|
}
|
|
|
|
|
walletVm.CryptoCode = wallet.Network.CryptoCode;
|
|
|
|
|
walletVm.StoreId = wallet.Store.Id;
|
|
|
|
|
walletVm.Id = new WalletId(wallet.Store.Id, wallet.Network.CryptoCode);
|
|
|
|
|
walletVm.StoreName = wallet.Store.StoreName;
|
|
|
|
|
walletVm.IsOwner = wallet.Store.HasClaim(Policies.CanModifyStoreSettings.Key);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return View(wallets);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("{walletId}")]
|
2018-07-26 17:08:07 +02:00
|
|
|
|
public async Task<IActionResult> WalletTransactions(
|
2018-07-26 15:32:24 +02:00
|
|
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
|
|
|
|
WalletId walletId)
|
|
|
|
|
{
|
2018-10-09 16:48:14 +02:00
|
|
|
|
var store = await Repository.FindStore(walletId.StoreId, GetUserId());
|
2019-05-08 16:39:11 +02:00
|
|
|
|
DerivationSchemeSettings paymentMethod = GetPaymentMethod(walletId, store);
|
2018-07-26 17:08:07 +02:00
|
|
|
|
if (paymentMethod == null)
|
2018-07-26 15:32:24 +02:00
|
|
|
|
return NotFound();
|
|
|
|
|
|
2018-07-26 17:08:07 +02:00
|
|
|
|
var wallet = _walletProvider.GetWallet(paymentMethod.Network);
|
2019-05-08 16:39:11 +02:00
|
|
|
|
var transactions = await wallet.FetchTransactions(paymentMethod.AccountDerivation);
|
2018-07-26 17:08:07 +02:00
|
|
|
|
|
|
|
|
|
var model = new ListTransactionsViewModel();
|
2018-10-09 16:48:14 +02:00
|
|
|
|
foreach (var tx in transactions.UnconfirmedTransactions.Transactions.Concat(transactions.ConfirmedTransactions.Transactions))
|
2018-07-26 17:08:07 +02:00
|
|
|
|
{
|
|
|
|
|
var vm = new ListTransactionsViewModel.TransactionViewModel();
|
|
|
|
|
model.Transactions.Add(vm);
|
|
|
|
|
vm.Id = tx.TransactionId.ToString();
|
|
|
|
|
vm.Link = string.Format(CultureInfo.InvariantCulture, paymentMethod.Network.BlockExplorerLink, vm.Id);
|
|
|
|
|
vm.Timestamp = tx.Timestamp;
|
|
|
|
|
vm.Positive = tx.BalanceChange >= Money.Zero;
|
|
|
|
|
vm.Balance = tx.BalanceChange.ToString();
|
2018-11-05 09:26:49 +01:00
|
|
|
|
vm.IsConfirmed = tx.Confirmations != 0;
|
2018-07-26 17:08:07 +02:00
|
|
|
|
}
|
2018-07-27 05:03:56 +02:00
|
|
|
|
model.Transactions = model.Transactions.OrderByDescending(t => t.Timestamp).ToList();
|
2018-07-26 17:08:07 +02:00
|
|
|
|
return View(model);
|
|
|
|
|
}
|
2018-07-26 15:32:24 +02:00
|
|
|
|
|
2018-07-26 17:08:07 +02:00
|
|
|
|
|
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("{walletId}/send")]
|
|
|
|
|
public async Task<IActionResult> WalletSend(
|
|
|
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
2019-05-08 05:34:33 +02:00
|
|
|
|
WalletId walletId, string defaultDestination = null, string defaultAmount = null)
|
2018-07-26 17:08:07 +02:00
|
|
|
|
{
|
|
|
|
|
if (walletId?.StoreId == null)
|
|
|
|
|
return NotFound();
|
2018-10-09 16:48:14 +02:00
|
|
|
|
var store = await Repository.FindStore(walletId.StoreId, GetUserId());
|
2019-05-08 16:39:11 +02:00
|
|
|
|
DerivationSchemeSettings paymentMethod = GetPaymentMethod(walletId, store);
|
2018-07-26 15:32:24 +02:00
|
|
|
|
if (paymentMethod == null)
|
|
|
|
|
return NotFound();
|
2018-10-31 16:19:25 +01:00
|
|
|
|
var network = this.NetworkProvider.GetNetwork(walletId?.CryptoCode);
|
|
|
|
|
if (network == null)
|
|
|
|
|
return NotFound();
|
2018-07-26 16:23:28 +02:00
|
|
|
|
var storeData = store.GetStoreBlob();
|
2018-10-09 16:48:14 +02:00
|
|
|
|
var rateRules = store.GetStoreBlob().GetRateRules(NetworkProvider);
|
2018-08-01 11:38:46 +02:00
|
|
|
|
rateRules.Spread = 0.0m;
|
2018-07-26 18:17:43 +02:00
|
|
|
|
var currencyPair = new Rating.CurrencyPair(paymentMethod.PaymentId.CryptoCode, GetCurrencyCode(storeData.DefaultLang) ?? "USD");
|
2018-10-31 16:19:25 +01:00
|
|
|
|
WalletSendModel model = new WalletSendModel()
|
2018-10-09 16:48:14 +02:00
|
|
|
|
{
|
2018-10-31 16:19:25 +01:00
|
|
|
|
Destination = defaultDestination,
|
|
|
|
|
CryptoCode = walletId.CryptoCode
|
2018-10-09 16:48:14 +02:00
|
|
|
|
};
|
2018-10-31 16:19:25 +01:00
|
|
|
|
if (double.TryParse(defaultAmount, out var amount))
|
|
|
|
|
model.Amount = (decimal)amount;
|
|
|
|
|
|
|
|
|
|
var feeProvider = _feeRateProvider.CreateFeeProvider(network);
|
|
|
|
|
var recommendedFees = feeProvider.GetFeeRateAsync();
|
2019-05-08 16:39:11 +02:00
|
|
|
|
var balance = _walletProvider.GetWallet(network).GetBalance(paymentMethod.AccountDerivation);
|
2018-10-31 16:19:25 +01:00
|
|
|
|
model.CurrentBalance = (await balance).ToDecimal(MoneyUnit.BTC);
|
|
|
|
|
model.RecommendedSatoshiPerByte = (int)(await recommendedFees).GetFee(1).Satoshi;
|
|
|
|
|
model.FeeSatoshiPerByte = model.RecommendedSatoshiPerByte;
|
2019-05-08 08:24:20 +02:00
|
|
|
|
model.SupportRBF = network.SupportRBF;
|
2018-07-26 16:23:28 +02:00
|
|
|
|
using (CancellationTokenSource cts = new CancellationTokenSource())
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
cts.CancelAfter(TimeSpan.FromSeconds(5));
|
2019-03-05 09:09:17 +01:00
|
|
|
|
var result = await RateFetcher.FetchRate(currencyPair, rateRules, cts.Token).WithCancellation(cts.Token);
|
2018-07-27 11:04:41 +02:00
|
|
|
|
if (result.BidAsk != null)
|
2018-07-26 16:23:28 +02:00
|
|
|
|
{
|
2018-07-27 11:04:41 +02:00
|
|
|
|
model.Rate = result.BidAsk.Center;
|
2018-07-26 16:23:28 +02:00
|
|
|
|
model.Divisibility = _currencyTable.GetNumberFormatInfo(currencyPair.Right, true).CurrencyDecimalDigits;
|
|
|
|
|
model.Fiat = currencyPair.Right;
|
|
|
|
|
}
|
2018-07-26 17:32:09 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
model.RateError = $"{result.EvaluatedRule} ({string.Join(", ", result.Errors.OfType<object>().ToArray())})";
|
|
|
|
|
}
|
2018-07-26 16:23:28 +02:00
|
|
|
|
}
|
2018-10-09 16:48:14 +02:00
|
|
|
|
catch (Exception ex) { model.RateError = ex.Message; }
|
2018-07-26 16:23:28 +02:00
|
|
|
|
}
|
2018-07-26 15:32:24 +02:00
|
|
|
|
return View(model);
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-31 16:19:25 +01:00
|
|
|
|
[HttpPost]
|
|
|
|
|
[Route("{walletId}/send")]
|
|
|
|
|
public async Task<IActionResult> WalletSend(
|
|
|
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
2019-05-08 07:39:37 +02:00
|
|
|
|
WalletId walletId, WalletSendModel vm, string command = null, CancellationToken cancellation = default)
|
2018-10-31 16:19:25 +01:00
|
|
|
|
{
|
|
|
|
|
if (walletId?.StoreId == null)
|
|
|
|
|
return NotFound();
|
|
|
|
|
var store = await Repository.FindStore(walletId.StoreId, GetUserId());
|
|
|
|
|
if (store == null)
|
|
|
|
|
return NotFound();
|
|
|
|
|
var network = this.NetworkProvider.GetNetwork(walletId?.CryptoCode);
|
|
|
|
|
if (network == null)
|
|
|
|
|
return NotFound();
|
2019-05-08 08:24:20 +02:00
|
|
|
|
vm.SupportRBF = network.SupportRBF;
|
2018-10-31 16:19:25 +01:00
|
|
|
|
var destination = ParseDestination(vm.Destination, network.NBitcoinNetwork);
|
|
|
|
|
if (destination == null)
|
|
|
|
|
ModelState.AddModelError(nameof(vm.Destination), "Invalid address");
|
|
|
|
|
|
|
|
|
|
if (vm.Amount.HasValue)
|
|
|
|
|
{
|
|
|
|
|
if (vm.CurrentBalance == vm.Amount.Value && !vm.SubstractFees)
|
|
|
|
|
ModelState.AddModelError(nameof(vm.Amount), "You are sending all your balance to the same destination, you should substract the fees");
|
|
|
|
|
if (vm.CurrentBalance < vm.Amount.Value)
|
|
|
|
|
ModelState.AddModelError(nameof(vm.Amount), "You are sending more than what you own");
|
|
|
|
|
}
|
|
|
|
|
if (!ModelState.IsValid)
|
|
|
|
|
return View(vm);
|
|
|
|
|
|
2019-05-11 13:02:32 +02:00
|
|
|
|
var storeData = (await Repository.FindStore(walletId.StoreId, GetUserId()));
|
|
|
|
|
var derivationScheme = GetPaymentMethod(walletId, storeData);
|
|
|
|
|
var psbt = await CreatePSBT(network, derivationScheme, vm, cancellation);
|
|
|
|
|
|
2019-05-08 07:39:37 +02:00
|
|
|
|
if (command == "ledger")
|
|
|
|
|
{
|
2019-05-11 13:26:31 +02:00
|
|
|
|
return ViewWalletSendLedger(psbt.PSBT, psbt.ChangeAddress);
|
2019-05-08 07:39:37 +02:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2019-05-08 16:58:25 +02:00
|
|
|
|
try
|
2019-05-11 13:26:31 +02:00
|
|
|
|
{
|
2019-05-10 17:29:29 +02:00
|
|
|
|
if (command == "analyze-psbt")
|
|
|
|
|
return View(nameof(WalletPSBT), new WalletPSBTViewModel()
|
|
|
|
|
{
|
|
|
|
|
Decoded = psbt.PSBT.ToString(),
|
|
|
|
|
PSBT = psbt.PSBT.ToBase64()
|
|
|
|
|
});
|
2019-05-11 13:26:31 +02:00
|
|
|
|
return FilePSBT(psbt.PSBT, $"Send-{vm.Amount.Value}-{network.CryptoCode}-to-{destination[0].ToString()}.psbt");
|
2019-05-08 16:58:25 +02:00
|
|
|
|
}
|
|
|
|
|
catch (NBXplorerException ex)
|
|
|
|
|
{
|
|
|
|
|
ModelState.AddModelError(nameof(vm.Amount), ex.Error.Message);
|
|
|
|
|
return View(vm);
|
|
|
|
|
}
|
|
|
|
|
catch (NotSupportedException)
|
|
|
|
|
{
|
|
|
|
|
ModelState.AddModelError(nameof(vm.Destination), "You need to update your version of NBXplorer");
|
|
|
|
|
return View(vm);
|
|
|
|
|
}
|
2019-05-08 07:39:37 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-11 13:26:31 +02:00
|
|
|
|
private IActionResult FilePSBT(PSBT psbt, string fileName)
|
|
|
|
|
{
|
|
|
|
|
return File(psbt.ToBytes(), "application/octet-stream", fileName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private ViewResult ViewWalletSendLedger(PSBT psbt, BitcoinAddress hintChange = null)
|
|
|
|
|
{
|
|
|
|
|
return View("WalletSendLedger", new WalletSendLedgerModel()
|
|
|
|
|
{
|
|
|
|
|
PSBT = psbt.ToBase64(),
|
|
|
|
|
HintChange = hintChange?.ToString(),
|
|
|
|
|
WebsocketPath = this.Url.Action(nameof(LedgerConnection)),
|
|
|
|
|
SuccessPath = this.Url.Action(nameof(WalletSendLedgerSuccess))
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-09 12:34:45 +02:00
|
|
|
|
[NonAction]
|
2019-05-11 13:02:32 +02:00
|
|
|
|
public async Task<CreatePSBTResponse> CreatePSBT(BTCPayNetwork network, DerivationSchemeSettings derivationSettings, WalletSendModel sendModel, CancellationToken cancellationToken)
|
2019-05-08 07:39:37 +02:00
|
|
|
|
{
|
|
|
|
|
var nbx = ExplorerClientProvider.GetExplorerClient(network);
|
|
|
|
|
CreatePSBTRequest psbtRequest = new CreatePSBTRequest();
|
|
|
|
|
CreatePSBTDestination psbtDestination = new CreatePSBTDestination();
|
|
|
|
|
psbtRequest.Destinations.Add(psbtDestination);
|
2019-05-08 08:24:20 +02:00
|
|
|
|
if (network.SupportRBF)
|
|
|
|
|
{
|
|
|
|
|
psbtRequest.RBF = !sendModel.DisableRBF;
|
|
|
|
|
}
|
2019-05-08 07:39:37 +02:00
|
|
|
|
psbtDestination.Destination = BitcoinAddress.Create(sendModel.Destination, network.NBitcoinNetwork);
|
2019-05-11 13:02:32 +02:00
|
|
|
|
psbtDestination.Amount = Money.Coins(sendModel.Amount.Value);
|
2019-05-08 07:39:37 +02:00
|
|
|
|
psbtRequest.FeePreference = new FeePreference();
|
|
|
|
|
psbtRequest.FeePreference.ExplicitFeeRate = new FeeRate(Money.Satoshis(sendModel.FeeSatoshiPerByte), 1);
|
|
|
|
|
if (sendModel.NoChange)
|
|
|
|
|
{
|
|
|
|
|
psbtRequest.ExplicitChangeAddress = psbtDestination.Destination;
|
|
|
|
|
}
|
|
|
|
|
psbtDestination.SubstractFees = sendModel.SubstractFees;
|
2019-05-08 16:54:34 +02:00
|
|
|
|
if (derivationSettings.AccountKeyPath != null && derivationSettings.AccountKeyPath.Indexes.Length != 0)
|
|
|
|
|
{
|
2019-05-10 12:30:10 +02:00
|
|
|
|
psbtRequest.RebaseKeyPaths = new List<PSBTRebaseKeyRules>()
|
2019-05-08 16:54:34 +02:00
|
|
|
|
{
|
2019-05-10 12:30:10 +02:00
|
|
|
|
new PSBTRebaseKeyRules()
|
2019-05-08 16:54:34 +02:00
|
|
|
|
{
|
2019-05-10 12:30:10 +02:00
|
|
|
|
AccountKeyPath = derivationSettings.AccountKeyPath,
|
|
|
|
|
AccountKey = derivationSettings.AccountKey,
|
|
|
|
|
MasterFingerprint = derivationSettings.RootFingerprint is HDFingerprint fp ? fp : default
|
2019-05-08 16:54:34 +02:00
|
|
|
|
}
|
2019-05-10 12:30:10 +02:00
|
|
|
|
};
|
2019-05-08 16:54:34 +02:00
|
|
|
|
}
|
2019-05-10 12:30:10 +02:00
|
|
|
|
var psbt = (await nbx.CreatePSBTAsync(derivationSettings.AccountDerivation, psbtRequest, cancellationToken));
|
|
|
|
|
if (psbt == null)
|
|
|
|
|
throw new NotSupportedException("You need to update your version of NBXplorer");
|
2019-05-08 07:39:37 +02:00
|
|
|
|
return psbt;
|
2018-10-31 16:19:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private IDestination[] ParseDestination(string destination, Network network)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
2018-11-01 04:54:25 +01:00
|
|
|
|
destination = destination?.Trim();
|
2018-10-31 16:19:25 +01:00
|
|
|
|
return new IDestination[] { BitcoinAddress.Create(destination, network) };
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-10 17:29:29 +02:00
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("{walletId}/psbt")]
|
|
|
|
|
public IActionResult WalletPSBT()
|
|
|
|
|
{
|
|
|
|
|
return View(new WalletPSBTViewModel());
|
|
|
|
|
}
|
|
|
|
|
[HttpPost]
|
|
|
|
|
[Route("{walletId}/psbt")]
|
|
|
|
|
public IActionResult WalletPSBT(
|
|
|
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
|
|
|
|
WalletId walletId,
|
|
|
|
|
WalletPSBTViewModel vm)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
if (!string.IsNullOrEmpty(vm.PSBT))
|
|
|
|
|
vm.Decoded = PSBT.Parse(vm.PSBT, NetworkProvider.GetNetwork(walletId.CryptoCode).NBitcoinNetwork).ToString();
|
|
|
|
|
}
|
|
|
|
|
catch (FormatException ex)
|
|
|
|
|
{
|
|
|
|
|
ModelState.AddModelError(nameof(vm.PSBT), ex.Message);
|
|
|
|
|
}
|
|
|
|
|
return View(vm);
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-11 13:26:31 +02:00
|
|
|
|
[HttpPost]
|
|
|
|
|
[Route("{walletId}/psbt/sign")]
|
|
|
|
|
public IActionResult WalletPSBTSign(
|
|
|
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
|
|
|
|
WalletId walletId,
|
|
|
|
|
WalletPSBTViewModel vm,
|
|
|
|
|
string command = null
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
var psbt = PSBT.Parse(vm.PSBT, NetworkProvider.GetNetwork(walletId.CryptoCode).NBitcoinNetwork);
|
|
|
|
|
if (command == "ledger")
|
|
|
|
|
{
|
|
|
|
|
return ViewWalletSendLedger(psbt);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return FilePSBT(psbt, "psbt-export.psbt");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-26 16:07:39 +02:00
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("{walletId}/rescan")]
|
|
|
|
|
public async Task<IActionResult> WalletRescan(
|
|
|
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
|
|
|
|
WalletId walletId)
|
|
|
|
|
{
|
|
|
|
|
if (walletId?.StoreId == null)
|
|
|
|
|
return NotFound();
|
|
|
|
|
var store = await Repository.FindStore(walletId.StoreId, GetUserId());
|
2019-05-08 16:39:11 +02:00
|
|
|
|
DerivationSchemeSettings paymentMethod = GetPaymentMethod(walletId, store);
|
2018-10-26 16:07:39 +02:00
|
|
|
|
if (paymentMethod == null)
|
|
|
|
|
return NotFound();
|
|
|
|
|
|
|
|
|
|
var vm = new RescanWalletModel();
|
2018-11-04 14:46:27 +01:00
|
|
|
|
vm.IsFullySync = _dashboard.IsFullySynched(walletId.CryptoCode, out var unused);
|
2018-10-26 16:07:39 +02:00
|
|
|
|
vm.IsServerAdmin = User.Claims.Any(c => c.Type == Policies.CanModifyServerSettings.Key && c.Value == "true");
|
|
|
|
|
vm.IsSupportedByCurrency = _dashboard.Get(walletId.CryptoCode)?.Status?.BitcoinStatus?.Capabilities?.CanScanTxoutSet == true;
|
|
|
|
|
var explorer = ExplorerClientProvider.GetExplorerClient(walletId.CryptoCode);
|
2019-05-08 16:39:11 +02:00
|
|
|
|
var scanProgress = await explorer.GetScanUTXOSetInformationAsync(paymentMethod.AccountDerivation);
|
2018-10-31 16:19:25 +01:00
|
|
|
|
if (scanProgress != null)
|
2018-10-26 16:07:39 +02:00
|
|
|
|
{
|
|
|
|
|
vm.PreviousError = scanProgress.Error;
|
|
|
|
|
if (scanProgress.Status == ScanUTXOStatus.Queued || scanProgress.Status == ScanUTXOStatus.Pending)
|
|
|
|
|
{
|
|
|
|
|
if (scanProgress.Progress == null)
|
|
|
|
|
{
|
|
|
|
|
vm.Progress = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
vm.Progress = scanProgress.Progress.OverallProgress;
|
|
|
|
|
vm.RemainingTime = TimeSpan.FromSeconds(scanProgress.Progress.RemainingSeconds).PrettyPrint();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (scanProgress.Status == ScanUTXOStatus.Complete)
|
|
|
|
|
{
|
|
|
|
|
vm.LastSuccess = scanProgress.Progress;
|
|
|
|
|
vm.TimeOfScan = (scanProgress.Progress.CompletedAt.Value - scanProgress.Progress.StartedAt).PrettyPrint();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return View(vm);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[HttpPost]
|
|
|
|
|
[Route("{walletId}/rescan")]
|
2018-10-27 15:51:09 +02:00
|
|
|
|
[Authorize(Policy = Policies.CanModifyServerSettings.Key)]
|
2018-10-26 16:07:39 +02:00
|
|
|
|
public async Task<IActionResult> WalletRescan(
|
|
|
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
|
|
|
|
WalletId walletId, RescanWalletModel vm)
|
|
|
|
|
{
|
|
|
|
|
if (walletId?.StoreId == null)
|
|
|
|
|
return NotFound();
|
|
|
|
|
var store = await Repository.FindStore(walletId.StoreId, GetUserId());
|
2019-05-08 16:39:11 +02:00
|
|
|
|
DerivationSchemeSettings paymentMethod = GetPaymentMethod(walletId, store);
|
2018-10-26 16:07:39 +02:00
|
|
|
|
if (paymentMethod == null)
|
|
|
|
|
return NotFound();
|
|
|
|
|
var explorer = ExplorerClientProvider.GetExplorerClient(walletId.CryptoCode);
|
|
|
|
|
try
|
|
|
|
|
{
|
2019-05-08 16:39:11 +02:00
|
|
|
|
await explorer.ScanUTXOSetAsync(paymentMethod.AccountDerivation, vm.BatchSize, vm.GapLimit, vm.StartingIndex);
|
2018-10-26 16:07:39 +02:00
|
|
|
|
}
|
|
|
|
|
catch (NBXplorerException ex) when (ex.Error.Code == "scanutxoset-in-progress")
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
return RedirectToAction();
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-26 18:17:43 +02:00
|
|
|
|
private string GetCurrencyCode(string defaultLang)
|
|
|
|
|
{
|
|
|
|
|
if (defaultLang == null)
|
|
|
|
|
return null;
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var ri = new RegionInfo(defaultLang);
|
|
|
|
|
return ri.ISOCurrencySymbol;
|
|
|
|
|
}
|
2018-10-09 16:48:14 +02:00
|
|
|
|
catch (ArgumentException) { }
|
2018-07-26 18:17:43 +02:00
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-08 16:39:11 +02:00
|
|
|
|
private DerivationSchemeSettings GetPaymentMethod(WalletId walletId, StoreData store)
|
2018-07-26 17:08:07 +02:00
|
|
|
|
{
|
|
|
|
|
if (store == null || !store.HasClaim(Policies.CanModifyStoreSettings.Key))
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
var paymentMethod = store
|
2018-10-09 16:48:14 +02:00
|
|
|
|
.GetSupportedPaymentMethods(NetworkProvider)
|
2019-05-08 16:39:11 +02:00
|
|
|
|
.OfType<DerivationSchemeSettings>()
|
2018-07-26 17:08:07 +02:00
|
|
|
|
.FirstOrDefault(p => p.PaymentId.PaymentType == Payments.PaymentTypes.BTCLike && p.PaymentId.CryptoCode == walletId.CryptoCode);
|
|
|
|
|
return paymentMethod;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-26 15:32:24 +02:00
|
|
|
|
private static async Task<string> GetBalanceString(BTCPayWallet wallet, DerivationStrategyBase derivationStrategy)
|
|
|
|
|
{
|
|
|
|
|
using (CancellationTokenSource cts = new CancellationTokenSource(TimeSpan.FromSeconds(10)))
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
return (await wallet.GetBalance(derivationStrategy, cts.Token)).ToString();
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
return "--";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetUserId()
|
|
|
|
|
{
|
|
|
|
|
return _userManager.GetUserId(User);
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-31 16:19:25 +01:00
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("{walletId}/send/ledger/success")]
|
|
|
|
|
public IActionResult WalletSendLedgerSuccess(
|
|
|
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
|
|
|
|
WalletId walletId,
|
|
|
|
|
string txid)
|
2018-07-26 15:32:24 +02:00
|
|
|
|
{
|
2018-10-31 16:19:25 +01:00
|
|
|
|
StatusMessage = $"Transaction broadcasted ({txid})";
|
|
|
|
|
return RedirectToAction(nameof(this.WalletTransactions), new { walletId = walletId.ToString() });
|
2018-07-26 15:32:24 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[HttpGet]
|
2018-10-31 16:19:25 +01:00
|
|
|
|
[Route("{walletId}/send/ledger/ws")]
|
2018-07-26 15:32:24 +02:00
|
|
|
|
public async Task<IActionResult> LedgerConnection(
|
2018-10-31 16:19:25 +01:00
|
|
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
|
|
|
|
WalletId walletId,
|
2018-07-26 15:32:24 +02:00
|
|
|
|
string command,
|
|
|
|
|
// getinfo
|
|
|
|
|
// getxpub
|
|
|
|
|
int account = 0,
|
|
|
|
|
// sendtoaddress
|
2019-05-11 13:02:32 +02:00
|
|
|
|
string psbt = null,
|
|
|
|
|
string hintChange = null
|
2018-07-26 15:32:24 +02:00
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
if (!HttpContext.WebSockets.IsWebSocketRequest)
|
|
|
|
|
return NotFound();
|
2018-10-31 16:19:25 +01:00
|
|
|
|
|
2019-05-11 13:02:32 +02:00
|
|
|
|
var network = NetworkProvider.GetNetwork(walletId.CryptoCode);
|
|
|
|
|
if (network == null)
|
|
|
|
|
throw new FormatException("Invalid value for crypto code");
|
2018-12-26 07:04:11 +01:00
|
|
|
|
var storeData = (await Repository.FindStore(walletId.StoreId, GetUserId()));
|
2019-05-08 16:39:11 +02:00
|
|
|
|
var derivationSettings = GetPaymentMethod(walletId, storeData);
|
2018-10-31 16:19:25 +01:00
|
|
|
|
|
2018-07-26 15:32:24 +02:00
|
|
|
|
var webSocket = await HttpContext.WebSockets.AcceptWebSocketAsync();
|
|
|
|
|
|
|
|
|
|
using (var normalOperationTimeout = new CancellationTokenSource())
|
|
|
|
|
using (var signTimeout = new CancellationTokenSource())
|
|
|
|
|
{
|
|
|
|
|
normalOperationTimeout.CancelAfter(TimeSpan.FromMinutes(30));
|
2019-05-10 07:36:25 +02:00
|
|
|
|
var hw = new LedgerHardwareWalletService(webSocket);
|
2019-05-08 07:39:37 +02:00
|
|
|
|
var model = new WalletSendLedgerModel();
|
2018-07-26 15:32:24 +02:00
|
|
|
|
object result = null;
|
|
|
|
|
try
|
|
|
|
|
{
|
2018-10-31 16:19:25 +01:00
|
|
|
|
if (command == "test")
|
|
|
|
|
{
|
|
|
|
|
result = await hw.Test(normalOperationTimeout.Token);
|
2018-07-26 15:32:24 +02:00
|
|
|
|
}
|
|
|
|
|
if (command == "sendtoaddress")
|
|
|
|
|
{
|
|
|
|
|
if (!_dashboard.IsFullySynched(network.CryptoCode, out var summary))
|
|
|
|
|
throw new Exception($"{network.CryptoCode}: not started or fully synched");
|
2019-05-02 11:56:01 +02:00
|
|
|
|
|
2019-05-09 18:05:37 +02:00
|
|
|
|
// Some deployment does not have the AccountKeyPath set, let's fix this...
|
|
|
|
|
if (derivationSettings.AccountKeyPath == null)
|
2018-07-26 15:32:24 +02:00
|
|
|
|
{
|
2018-12-26 07:10:00 +01:00
|
|
|
|
// If the saved wallet key path is not present or incorrect, let's scan the wallet to see if it can sign strategy
|
2019-05-10 03:55:10 +02:00
|
|
|
|
var foundKeyPath = await hw.FindKeyPathFromDerivation(network,
|
|
|
|
|
derivationSettings.AccountDerivation,
|
2019-05-10 03:48:30 +02:00
|
|
|
|
normalOperationTimeout.Token);
|
|
|
|
|
derivationSettings.AccountKeyPath = foundKeyPath ?? throw new HardwareWalletException($"This store is not configured to use this ledger");
|
2019-05-08 16:39:11 +02:00
|
|
|
|
storeData.SetSupportedPaymentMethod(derivationSettings);
|
2018-12-26 07:04:11 +01:00
|
|
|
|
await Repository.UpdateStore(storeData);
|
|
|
|
|
}
|
2019-05-10 03:48:30 +02:00
|
|
|
|
// If it has already the AccountKeyPath, we did not looked up for it, so we need to check if we are on the right ledger
|
2019-05-09 18:05:37 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// Checking if ledger is right with the RootFingerprint is faster as it does not need to make a query to the parent xpub,
|
|
|
|
|
// but some deployment does not have it, so let's use AccountKeyPath instead
|
|
|
|
|
if (derivationSettings.RootFingerprint == null)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
var actualPubKey = await hw.GetExtPubKey(network, derivationSettings.AccountKeyPath, normalOperationTimeout.Token);
|
|
|
|
|
if (!derivationSettings.AccountDerivation.GetExtPubKeys().Any(p => p.GetPublicKey() == actualPubKey.GetPublicKey()))
|
|
|
|
|
throw new HardwareWalletException($"This store is not configured to use this ledger");
|
|
|
|
|
}
|
|
|
|
|
// We have the root fingerprint, we can check the root from it
|
|
|
|
|
else
|
|
|
|
|
{
|
2019-05-10 03:48:30 +02:00
|
|
|
|
var actualPubKey = await hw.GetPubKey(network, new KeyPath(), normalOperationTimeout.Token);
|
|
|
|
|
if (actualPubKey.GetHDFingerPrint() != derivationSettings.RootFingerprint.Value)
|
2019-05-09 18:05:37 +02:00
|
|
|
|
throw new HardwareWalletException($"This store is not configured to use this ledger");
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-01-15 15:50:45 +01:00
|
|
|
|
|
2019-05-09 18:05:37 +02:00
|
|
|
|
// Some deployment does not have the RootFingerprint set, let's fix this...
|
|
|
|
|
if (derivationSettings.RootFingerprint == null)
|
|
|
|
|
{
|
2019-05-10 03:48:30 +02:00
|
|
|
|
derivationSettings.RootFingerprint = (await hw.GetPubKey(network, new KeyPath(), normalOperationTimeout.Token)).GetHDFingerPrint();
|
2019-05-09 18:05:37 +02:00
|
|
|
|
storeData.SetSupportedPaymentMethod(derivationSettings);
|
|
|
|
|
await Repository.UpdateStore(storeData);
|
|
|
|
|
}
|
2018-07-26 15:32:24 +02:00
|
|
|
|
|
2019-05-11 13:02:32 +02:00
|
|
|
|
var psbtResponse = new CreatePSBTResponse()
|
|
|
|
|
{
|
|
|
|
|
PSBT = PSBT.Parse(psbt, network.NBitcoinNetwork),
|
|
|
|
|
ChangeAddress = string.IsNullOrEmpty(hintChange) ? null : BitcoinAddress.Create(hintChange, network.NBitcoinNetwork)
|
|
|
|
|
};
|
2018-07-26 15:32:24 +02:00
|
|
|
|
signTimeout.CancelAfter(TimeSpan.FromMinutes(5));
|
2019-05-11 13:02:32 +02:00
|
|
|
|
psbtResponse.PSBT = await hw.SignTransactionAsync(psbtResponse.PSBT, psbtResponse.ChangeAddress?.ScriptPubKey, signTimeout.Token);
|
|
|
|
|
if(!psbtResponse.PSBT.TryFinalize(out var errors))
|
2018-07-26 15:32:24 +02:00
|
|
|
|
{
|
2019-05-02 11:56:01 +02:00
|
|
|
|
throw new Exception($"Error while finalizing the transaction ({new PSBTException(errors).ToString()})");
|
|
|
|
|
}
|
2019-05-11 13:02:32 +02:00
|
|
|
|
var transaction = psbtResponse.PSBT.ExtractTransaction();
|
2018-07-26 15:32:24 +02:00
|
|
|
|
try
|
|
|
|
|
{
|
2019-05-08 07:39:37 +02:00
|
|
|
|
var broadcastResult = await ExplorerClientProvider.GetExplorerClient(network).BroadcastAsync(transaction);
|
2019-05-02 11:56:01 +02:00
|
|
|
|
if (!broadcastResult.Success)
|
2018-07-26 15:32:24 +02:00
|
|
|
|
{
|
2019-05-02 11:56:01 +02:00
|
|
|
|
throw new Exception($"RPC Error while broadcasting: {broadcastResult.RPCCode} {broadcastResult.RPCCodeMessage} {broadcastResult.RPCMessage}");
|
2018-07-26 15:32:24 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
throw new Exception("Error while broadcasting: " + ex.Message);
|
|
|
|
|
}
|
2019-05-02 11:56:01 +02:00
|
|
|
|
var wallet = _walletProvider.GetWallet(network);
|
2019-05-08 16:54:34 +02:00
|
|
|
|
wallet.InvalidateCache(derivationSettings.AccountDerivation);
|
2018-07-26 15:32:24 +02:00
|
|
|
|
result = new SendToAddressResult() { TransactionId = transaction.GetHash().ToString() };
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (OperationCanceledException)
|
|
|
|
|
{ result = new LedgerTestResult() { Success = false, Error = "Timeout" }; }
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{ result = new LedgerTestResult() { Success = false, Error = ex.Message }; }
|
|
|
|
|
finally { hw.Dispose(); }
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
if (result != null)
|
|
|
|
|
{
|
|
|
|
|
UTF8Encoding UTF8NOBOM = new UTF8Encoding(false);
|
|
|
|
|
var bytes = UTF8NOBOM.GetBytes(JsonConvert.SerializeObject(result, _mvcJsonOptions.Value.SerializerSettings));
|
|
|
|
|
await webSocket.SendAsync(new ArraySegment<byte>(bytes), WebSocketMessageType.Text, true, new CancellationTokenSource(2000).Token);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch { }
|
|
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
await webSocket.CloseSocket();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return new EmptyResult();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public class GetInfoResult
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class SendToAddressResult
|
|
|
|
|
{
|
|
|
|
|
public string TransactionId { get; set; }
|
|
|
|
|
}
|
|
|
|
|
}
|