mirror of
https://github.com/btcpayserver/btcpayserver.git
synced 2025-01-18 21:32:27 +01:00
433 lines
22 KiB
C#
433 lines
22 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using BTCPayServer.Abstractions.Constants;
|
|
using BTCPayServer.Client;
|
|
using BTCPayServer.Data;
|
|
using BTCPayServer.Hwi;
|
|
using BTCPayServer.ModelBinders;
|
|
using BTCPayServer.Payments;
|
|
using BTCPayServer.Payments.Bitcoin;
|
|
using BTCPayServer.Services.Invoices;
|
|
using Microsoft.AspNetCore.Authorization;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using NBitcoin;
|
|
using NBXplorer.DerivationStrategy;
|
|
using Newtonsoft.Json.Linq;
|
|
|
|
namespace BTCPayServer.Controllers
|
|
{
|
|
[Route("vault")]
|
|
[Authorize(AuthenticationSchemes = AuthenticationSchemes.Cookie, Policy = Policies.CanModifyStoreSettings)]
|
|
public class UIVaultController : Controller
|
|
{
|
|
private readonly PaymentMethodHandlerDictionary _handlers;
|
|
private readonly IAuthorizationService _authorizationService;
|
|
|
|
public UIVaultController(PaymentMethodHandlerDictionary handlers, IAuthorizationService authorizationService)
|
|
{
|
|
_handlers = handlers;
|
|
_authorizationService = authorizationService;
|
|
}
|
|
|
|
[Route("{cryptoCode}/xpub")]
|
|
[Route("wallets/{walletId}/xpub")]
|
|
public async Task<IActionResult> VaultBridgeConnection(string cryptoCode = null,
|
|
[ModelBinder(typeof(WalletIdModelBinder))]
|
|
WalletId walletId = null)
|
|
{
|
|
if (!HttpContext.WebSockets.IsWebSocketRequest)
|
|
return NotFound();
|
|
cryptoCode = cryptoCode ?? walletId.CryptoCode;
|
|
bool versionChecked = false;
|
|
using (var cts = new CancellationTokenSource(TimeSpan.FromMinutes(10)))
|
|
{
|
|
var cancellationToken = cts.Token;
|
|
if (!_handlers.TryGetValue(PaymentTypes.CHAIN.GetPaymentMethodId(cryptoCode), out var h) || h is not IHasNetwork { Network: var network })
|
|
return NotFound();
|
|
var websocket = await HttpContext.WebSockets.AcceptWebSocketAsync();
|
|
var vaultClient = new VaultClient(websocket);
|
|
var hwi = new Hwi.HwiClient(network.NBitcoinNetwork)
|
|
{
|
|
Transport = new VaultHWITransport(vaultClient)
|
|
};
|
|
Hwi.HwiDeviceClient device = null;
|
|
HwiEnumerateEntry deviceEntry = null;
|
|
HDFingerprint? fingerprint = null;
|
|
string password = null;
|
|
var websocketHelper = new WebSocketHelper(websocket);
|
|
async Task FetchFingerprint()
|
|
{
|
|
fingerprint = (await device.GetXPubAsync(new KeyPath("44'"), cancellationToken)).ExtPubKey.ParentFingerprint;
|
|
device = new HwiDeviceClient(hwi, DeviceSelectors.FromFingerprint(fingerprint.Value), deviceEntry.Model, fingerprint) { Password = password };
|
|
}
|
|
async Task<bool> RequireDeviceUnlocking()
|
|
{
|
|
if (deviceEntry == null)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"need-device\"}", cancellationToken);
|
|
return true;
|
|
}
|
|
if (deviceEntry.Code is HwiErrorCode.DeviceNotInitialized)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"need-initialized\"}", cancellationToken);
|
|
return true;
|
|
}
|
|
if (deviceEntry.Code is HwiErrorCode.DeviceNotReady)
|
|
{
|
|
if (IsTrezorT(deviceEntry))
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"need-passphrase-on-device\"}", cancellationToken);
|
|
return true;
|
|
}
|
|
else if (deviceEntry.NeedsPinSent is true)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"need-pin\"}", cancellationToken);
|
|
return true;
|
|
}
|
|
else if (deviceEntry.NeedsPassphraseSent is true && password is null)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"need-passphrase\"}", cancellationToken);
|
|
return true;
|
|
}
|
|
}
|
|
if (IsTrezorOne(deviceEntry) && password is null)
|
|
{
|
|
fingerprint = null; // There will be a new fingerprint
|
|
device = new HwiDeviceClient(hwi, DeviceSelectors.FromDeviceType("trezor", deviceEntry.Path), deviceEntry.Model, null);
|
|
await websocketHelper.Send("{ \"error\": \"need-passphrase\"}", cancellationToken);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
JObject o = null;
|
|
try
|
|
{
|
|
while (true)
|
|
{
|
|
var command = await websocketHelper.NextMessageAsync(cancellationToken);
|
|
switch (command)
|
|
{
|
|
case "set-passphrase":
|
|
device.Password = await websocketHelper.NextMessageAsync(cancellationToken);
|
|
password = device.Password;
|
|
break;
|
|
case "ask-sign":
|
|
if (await RequireDeviceUnlocking())
|
|
{
|
|
continue;
|
|
}
|
|
if (walletId == null)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"invalid-walletId\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
if (fingerprint is null)
|
|
{
|
|
await FetchFingerprint();
|
|
}
|
|
await websocketHelper.Send("{ \"info\": \"ready\"}", cancellationToken);
|
|
o = JObject.Parse(await websocketHelper.NextMessageAsync(cancellationToken));
|
|
var authorization = await _authorizationService.AuthorizeAsync(User, Policies.CanModifyStoreSettings);
|
|
if (!authorization.Succeeded)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"not-authorized\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
var psbt = PSBT.Parse(o["psbt"].Value<string>(), network.NBitcoinNetwork);
|
|
var derivationSettings = GetDerivationSchemeSettings(walletId);
|
|
derivationSettings.RebaseKeyPaths(psbt);
|
|
|
|
// we ensure that the device fingerprint is part of the derivation settings
|
|
if (derivationSettings.AccountKeySettings.All(a => a.RootFingerprint != fingerprint))
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"wrong-wallet\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
|
|
// otherwise, let the device check if it can sign anything
|
|
var signableInputs = psbt.Inputs
|
|
.SelectMany(i => i.HDKeyPaths)
|
|
.Where(i => i.Value.MasterFingerprint == fingerprint)
|
|
.ToArray();
|
|
if (signableInputs.Length > 0)
|
|
{
|
|
var actualPubKey = (await device.GetXPubAsync(signableInputs[0].Value.KeyPath)).GetPublicKey();
|
|
if (actualPubKey != signableInputs[0].Key)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"wrong-keypath\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
|
|
if (derivationSettings.IsMultiSigOnServer)
|
|
{
|
|
var alreadySigned = psbt.Inputs.Any(a =>
|
|
a.PartialSigs.Any(a => a.Key == actualPubKey));
|
|
if (alreadySigned)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"already-signed-psbt\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
try
|
|
{
|
|
psbt = await device.SignPSBTAsync(psbt, cancellationToken);
|
|
}
|
|
catch (HwiException)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"user-reject\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
o = new JObject();
|
|
o.Add("psbt", psbt.ToBase64());
|
|
await websocketHelper.Send(o.ToString(), cancellationToken);
|
|
break;
|
|
case "display-address":
|
|
if (await RequireDeviceUnlocking())
|
|
{
|
|
continue;
|
|
}
|
|
var k = RootedKeyPath.Parse(await websocketHelper.NextMessageAsync(cancellationToken));
|
|
await device.DisplayAddressAsync(GetScriptPubKeyType(k), k.KeyPath, cancellationToken);
|
|
await websocketHelper.Send("{ \"info\": \"ok\"}", cancellationToken);
|
|
break;
|
|
case "ask-pin":
|
|
if (device == null)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"need-device\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
try
|
|
{
|
|
await device.PromptPinAsync(cancellationToken);
|
|
}
|
|
catch (HwiException ex) when (ex.ErrorCode == HwiErrorCode.DeviceAlreadyUnlocked)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"device-already-unlocked\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
await websocketHelper.Send("{ \"info\": \"prompted, please input the pin\"}", cancellationToken);
|
|
var pin = int.Parse(await websocketHelper.NextMessageAsync(cancellationToken), CultureInfo.InvariantCulture);
|
|
if (await device.SendPinAsync(pin, cancellationToken))
|
|
{
|
|
goto askdevice;
|
|
}
|
|
else
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"incorrect-pin\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
case "ask-xpub":
|
|
if (await RequireDeviceUnlocking())
|
|
{
|
|
continue;
|
|
}
|
|
await websocketHelper.Send("{ \"info\": \"ok\"}", cancellationToken);
|
|
var askedXpub = JObject.Parse(await websocketHelper.NextMessageAsync(cancellationToken));
|
|
var addressType = askedXpub["addressType"].Value<string>();
|
|
var accountNumber = askedXpub["accountNumber"].Value<int>();
|
|
JObject result = new JObject();
|
|
var factory = network.NBXplorerNetwork.DerivationStrategyFactory;
|
|
if (fingerprint is null)
|
|
{
|
|
await FetchFingerprint();
|
|
}
|
|
result["fingerprint"] = fingerprint.Value.ToString();
|
|
|
|
DerivationStrategyBase strategy = null;
|
|
|
|
KeyPath keyPath = (addressType switch
|
|
{
|
|
"taproot" => new KeyPath("86'"),
|
|
"segwit" => new KeyPath("84'"),
|
|
"segwitWrapped" => new KeyPath("49'"),
|
|
"legacy" => new KeyPath("44'"),
|
|
_ => null
|
|
})?.Derive(network.CoinType).Derive(accountNumber, true);
|
|
if (keyPath is null)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"invalid-addresstype\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
BitcoinExtPubKey xpub = await device.GetXPubAsync(keyPath);
|
|
if (!network.NBitcoinNetwork.Consensus.SupportSegwit && addressType != "legacy")
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"segwit-notsupported\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
|
|
if (!network.NBitcoinNetwork.Consensus.SupportTaproot && addressType == "taproot")
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"taproot-notsupported\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
if (addressType == "taproot")
|
|
{
|
|
strategy = factory.CreateDirectDerivationStrategy(xpub, new DerivationStrategyOptions()
|
|
{
|
|
ScriptPubKeyType = ScriptPubKeyType.TaprootBIP86
|
|
});
|
|
}
|
|
else if (addressType == "segwit")
|
|
{
|
|
strategy = factory.CreateDirectDerivationStrategy(xpub, new DerivationStrategyOptions()
|
|
{
|
|
ScriptPubKeyType = ScriptPubKeyType.Segwit
|
|
});
|
|
}
|
|
else if (addressType == "segwitWrapped")
|
|
{
|
|
strategy = factory.CreateDirectDerivationStrategy(xpub, new DerivationStrategyOptions()
|
|
{
|
|
ScriptPubKeyType = ScriptPubKeyType.SegwitP2SH
|
|
});
|
|
}
|
|
else if (addressType == "legacy")
|
|
{
|
|
strategy = factory.CreateDirectDerivationStrategy(xpub, new DerivationStrategyOptions()
|
|
{
|
|
ScriptPubKeyType = ScriptPubKeyType.Legacy
|
|
});
|
|
}
|
|
|
|
result.Add(new JProperty("strategy", strategy.ToString()));
|
|
result.Add(new JProperty("accountKey", xpub.ToString()));
|
|
result.Add(new JProperty("keyPath", keyPath.ToString()));
|
|
await websocketHelper.Send(result.ToString(), cancellationToken);
|
|
break;
|
|
case "ask-passphrase":
|
|
if (command == "ask-passphrase")
|
|
{
|
|
if (deviceEntry == null)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"need-device\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
// The make the trezor T ask for password
|
|
await device.GetXPubAsync(new KeyPath("44'"), cancellationToken);
|
|
}
|
|
goto askdevice;
|
|
case "ask-device":
|
|
askdevice:
|
|
if (!versionChecked)
|
|
{
|
|
var version = await hwi.GetVersionAsync(cancellationToken);
|
|
if (version.Major < 2)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"vault-outdated\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
versionChecked = true;
|
|
}
|
|
password = null;
|
|
deviceEntry = null;
|
|
device = null;
|
|
var entries = (await hwi.EnumerateEntriesAsync(cancellationToken)).ToList();
|
|
deviceEntry = entries.FirstOrDefault();
|
|
if (deviceEntry == null)
|
|
{
|
|
await websocketHelper.Send("{ \"error\": \"no-device\"}", cancellationToken);
|
|
continue;
|
|
}
|
|
var model = deviceEntry.Model ?? "Unsupported hardware wallet, try to update BTCPay Server Vault";
|
|
device = new HwiDeviceClient(hwi, deviceEntry.DeviceSelector, model, deviceEntry.Fingerprint);
|
|
fingerprint = device.Fingerprint;
|
|
JObject json = new JObject();
|
|
json.Add("model", model);
|
|
await websocketHelper.Send(json.ToString(), cancellationToken);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
catch (FormatException ex)
|
|
{
|
|
JObject obj = new JObject();
|
|
obj.Add("error", "invalid-network");
|
|
obj.Add("details", ex.ToString());
|
|
try
|
|
{
|
|
await websocketHelper.Send(obj.ToString(), cancellationToken);
|
|
}
|
|
catch { }
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
JObject obj = new JObject();
|
|
obj.Add("error", "unknown-error");
|
|
obj.Add("message", ex.Message);
|
|
obj.Add("details", ex.ToString());
|
|
try
|
|
{
|
|
await websocketHelper.Send(obj.ToString(), cancellationToken);
|
|
}
|
|
catch { }
|
|
}
|
|
finally
|
|
{
|
|
await websocketHelper.DisposeAsync(cancellationToken);
|
|
}
|
|
}
|
|
return new EmptyResult();
|
|
}
|
|
|
|
private ScriptPubKeyType GetScriptPubKeyType(RootedKeyPath keyPath)
|
|
{
|
|
var path = keyPath.KeyPath.ToString();
|
|
if (path.StartsWith("86'", StringComparison.OrdinalIgnoreCase))
|
|
return ScriptPubKeyType.TaprootBIP86;
|
|
if (path.StartsWith("84'", StringComparison.OrdinalIgnoreCase))
|
|
return ScriptPubKeyType.Segwit;
|
|
if (path.StartsWith("49'", StringComparison.OrdinalIgnoreCase))
|
|
return ScriptPubKeyType.SegwitP2SH;
|
|
if (path.StartsWith("44'", StringComparison.OrdinalIgnoreCase))
|
|
return ScriptPubKeyType.Legacy;
|
|
throw new NotSupportedException("Unsupported keypath");
|
|
}
|
|
|
|
private bool SameSelector(DeviceSelector a, DeviceSelector b)
|
|
{
|
|
var aargs = new List<string>();
|
|
a.AddArgs(aargs);
|
|
var bargs = new List<string>();
|
|
b.AddArgs(bargs);
|
|
if (aargs.Count != bargs.Count)
|
|
return false;
|
|
for (int i = 0; i < aargs.Count; i++)
|
|
{
|
|
if (aargs[i] != bargs[i])
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
private static bool IsTrezorT(HwiEnumerateEntry deviceEntry)
|
|
{
|
|
return deviceEntry.Model.Contains("Trezor_T", StringComparison.OrdinalIgnoreCase);
|
|
}
|
|
private static bool IsTrezorOne(HwiEnumerateEntry deviceEntry)
|
|
{
|
|
return deviceEntry.Model.Contains("trezor_1", StringComparison.OrdinalIgnoreCase);
|
|
}
|
|
|
|
public StoreData CurrentStore
|
|
{
|
|
get
|
|
{
|
|
return HttpContext.GetStoreData();
|
|
}
|
|
}
|
|
|
|
private DerivationSchemeSettings GetDerivationSchemeSettings(WalletId walletId)
|
|
{
|
|
var pmi = Payments.PaymentTypes.CHAIN.GetPaymentMethodId(walletId.CryptoCode);
|
|
return CurrentStore.GetPaymentMethodConfig<DerivationSchemeSettings>(pmi, _handlers);
|
|
}
|
|
}
|
|
}
|