2020-06-29 04:44:35 +02:00
|
|
|
using System;
|
2017-09-13 08:47:34 +02:00
|
|
|
using System.Collections.Generic;
|
2020-06-28 10:55:27 +02:00
|
|
|
using System.Globalization;
|
2017-09-13 08:47:34 +02:00
|
|
|
using System.IO;
|
2020-06-28 10:55:27 +02:00
|
|
|
using System.Linq;
|
2017-10-12 09:33:53 +02:00
|
|
|
using System.Net.Http;
|
2020-06-28 10:55:27 +02:00
|
|
|
using System.Threading.Tasks;
|
2023-11-29 10:51:40 +01:00
|
|
|
using BTCPayServer.Hosting;
|
2018-08-30 04:50:39 +02:00
|
|
|
using BTCPayServer.Lightning;
|
2021-03-02 03:11:58 +01:00
|
|
|
using BTCPayServer.Lightning.CLightning;
|
|
|
|
using BTCPayServer.Payments.Lightning;
|
2020-06-28 10:55:27 +02:00
|
|
|
using BTCPayServer.Tests.Lnd;
|
2019-03-27 07:53:06 +01:00
|
|
|
using BTCPayServer.Tests.Logging;
|
2023-11-29 10:51:40 +01:00
|
|
|
using Microsoft.Extensions.Configuration.Memory;
|
|
|
|
using Microsoft.Extensions.Configuration;
|
2021-11-22 09:16:08 +01:00
|
|
|
using Microsoft.Extensions.Logging;
|
2020-06-28 10:55:27 +02:00
|
|
|
using NBitcoin;
|
|
|
|
using NBitcoin.RPC;
|
|
|
|
using NBitpayClient;
|
|
|
|
using NBXplorer;
|
2024-07-24 13:16:20 +02:00
|
|
|
using BTCPayServer.Abstractions.Contracts;
|
|
|
|
using System.Diagnostics.Metrics;
|
2017-09-13 08:47:34 +02:00
|
|
|
|
|
|
|
namespace BTCPayServer.Tests
|
|
|
|
{
|
2017-10-27 10:53:04 +02:00
|
|
|
public class ServerTester : IDisposable
|
|
|
|
{
|
2024-08-28 11:52:08 +02:00
|
|
|
public const string DefaultConnectionString = "User ID=postgres;Include Error Detail=true;Host=127.0.0.1;Port=39372;Database=btcpayserver";
|
2020-11-13 08:28:15 +01:00
|
|
|
public List<IDisposable> Resources = new List<IDisposable>();
|
2020-06-29 05:07:48 +02:00
|
|
|
readonly string _Directory;
|
2021-11-22 09:16:08 +01:00
|
|
|
|
|
|
|
public ILoggerProvider LoggerProvider { get; }
|
|
|
|
|
|
|
|
internal ILog TestLogs;
|
2023-11-29 10:51:40 +01:00
|
|
|
public ServerTester(string scope, bool newDb, ILog testLogs, ILoggerProvider loggerProvider, BTCPayNetworkProvider networkProvider)
|
2017-10-27 10:53:04 +02:00
|
|
|
{
|
2021-11-22 09:16:08 +01:00
|
|
|
LoggerProvider = loggerProvider;
|
|
|
|
this.TestLogs = testLogs;
|
2017-10-27 10:53:04 +02:00
|
|
|
_Directory = scope;
|
|
|
|
if (Directory.Exists(_Directory))
|
|
|
|
Utils.DeleteDirectory(_Directory);
|
|
|
|
if (!Directory.Exists(_Directory))
|
|
|
|
Directory.CreateDirectory(_Directory);
|
|
|
|
|
2024-01-18 09:16:57 +01:00
|
|
|
_NetworkProvider = networkProvider;
|
2019-05-29 11:43:50 +02:00
|
|
|
ExplorerNode = new RPCClient(RPCCredentialString.Parse(GetEnvironment("TESTS_BTCRPCCONNECTION", "server=http://127.0.0.1:43782;ceiwHEbqWI83:DwubwWsoo3")), NetworkProvider.GetNetwork<BTCPayNetwork>("BTC").NBitcoinNetwork);
|
2019-05-03 05:59:11 +02:00
|
|
|
ExplorerNode.ScanRPCCapabilities();
|
2018-01-11 09:29:48 +01:00
|
|
|
|
2019-05-29 11:43:50 +02:00
|
|
|
ExplorerClient = new ExplorerClient(NetworkProvider.GetNetwork<BTCPayNetwork>("BTC").NBXplorerNetwork, new Uri(GetEnvironment("TESTS_BTCNBXPLORERURL", "http://127.0.0.1:32838/")));
|
2018-05-25 17:44:59 +02:00
|
|
|
|
2021-11-22 09:16:08 +01:00
|
|
|
PayTester = new BTCPayServerTester(TestLogs, LoggerProvider, Path.Combine(_Directory, "pay"))
|
2017-10-27 10:53:04 +02:00
|
|
|
{
|
|
|
|
NBXplorerUri = ExplorerClient.Address,
|
2020-02-29 06:15:06 +01:00
|
|
|
// TODO: The fact that we use same conn string as development database can cause huge problems with tests
|
|
|
|
// since in dev we already can have some users / stores registered, while on CI database is being initalized
|
|
|
|
// for the first time and first registered user gets admin status by default
|
2024-08-28 11:52:08 +02:00
|
|
|
Postgres = GetEnvironment("TESTS_POSTGRES", DefaultConnectionString),
|
2022-07-05 07:39:50 +02:00
|
|
|
ExplorerPostgres = GetEnvironment("TESTS_EXPLORER_POSTGRES", "User ID=postgres;Include Error Detail=true;Host=127.0.0.1;Port=39372;Database=nbxplorer"),
|
2017-10-27 10:53:04 +02:00
|
|
|
};
|
2020-03-18 15:10:15 +01:00
|
|
|
if (newDb)
|
|
|
|
{
|
|
|
|
var r = RandomUtils.GetUInt32();
|
|
|
|
PayTester.Postgres = PayTester.Postgres.Replace("btcpayserver", $"btcpayserver{r}");
|
2023-01-30 01:46:12 +01:00
|
|
|
TestLogs.LogInformation($"Database used: btcpayserver{r}");
|
2020-03-18 15:10:15 +01:00
|
|
|
}
|
2018-02-17 05:18:16 +01:00
|
|
|
PayTester.Port = int.Parse(GetEnvironment("TESTS_PORT", Utils.FreeTcpPort().ToString(CultureInfo.InvariantCulture)), CultureInfo.InvariantCulture);
|
2017-10-27 10:53:04 +02:00
|
|
|
PayTester.HostName = GetEnvironment("TESTS_HOSTNAME", "127.0.0.1");
|
2018-03-20 16:31:19 +01:00
|
|
|
PayTester.InContainer = bool.Parse(GetEnvironment("TESTS_INCONTAINER", "false"));
|
2019-09-06 09:51:49 +02:00
|
|
|
|
|
|
|
PayTester.SSHPassword = GetEnvironment("TESTS_SSHPASSWORD", "opD3i2282D");
|
|
|
|
PayTester.SSHKeyFile = GetEnvironment("TESTS_SSHKEYFILE", "");
|
|
|
|
PayTester.SSHConnection = GetEnvironment("TESTS_SSHCONNECTION", "root@127.0.0.1:21622");
|
2020-04-08 15:40:41 +02:00
|
|
|
PayTester.SocksEndpoint = GetEnvironment("TESTS_SOCKSENDPOINT", "localhost:9050");
|
2018-04-15 14:18:51 +02:00
|
|
|
}
|
2024-07-24 13:16:20 +02:00
|
|
|
|
|
|
|
public void ActivateLangs()
|
|
|
|
{
|
|
|
|
TestLogs.LogInformation("Activating Langs...");
|
|
|
|
var dir = TestUtils.GetTestDataFullPath("Langs");
|
|
|
|
var langdir = Path.Combine(PayTester._Directory, "Langs");
|
|
|
|
Directory.CreateDirectory(langdir);
|
|
|
|
foreach (var file in Directory.GetFiles(dir))
|
|
|
|
File.Copy(file, Path.Combine(langdir, Path.GetFileName(file)));
|
|
|
|
}
|
|
|
|
|
2024-08-30 01:34:23 +02:00
|
|
|
|
2019-12-24 10:11:21 +01:00
|
|
|
public void ActivateLTC()
|
|
|
|
{
|
|
|
|
LTCExplorerNode = new RPCClient(RPCCredentialString.Parse(GetEnvironment("TESTS_LTCRPCCONNECTION", "server=http://127.0.0.1:43783;ceiwHEbqWI83:DwubwWsoo3")), NetworkProvider.GetNetwork<BTCPayNetwork>("LTC").NBitcoinNetwork);
|
|
|
|
LTCExplorerClient = new ExplorerClient(NetworkProvider.GetNetwork<BTCPayNetwork>("LTC").NBXplorerNetwork, new Uri(GetEnvironment("TESTS_LTCNBXPLORERURL", "http://127.0.0.1:32838/")));
|
|
|
|
PayTester.Chains.Add("LTC");
|
|
|
|
PayTester.LTCNBXplorerUri = LTCExplorerClient.Address;
|
|
|
|
}
|
|
|
|
public void ActivateLBTC()
|
|
|
|
{
|
|
|
|
LBTCExplorerNode = new RPCClient(RPCCredentialString.Parse(GetEnvironment("TESTS_LBTCRPCCONNECTION", "server=http://127.0.0.1:19332;liquid:liquid")), NetworkProvider.GetNetwork<BTCPayNetwork>("LBTC").NBitcoinNetwork);
|
|
|
|
LBTCExplorerClient = new ExplorerClient(NetworkProvider.GetNetwork<BTCPayNetwork>("LBTC").NBXplorerNetwork, new Uri(GetEnvironment("TESTS_LBTCNBXPLORERURL", "http://127.0.0.1:32838/")));
|
|
|
|
PayTester.Chains.Add("LBTC");
|
|
|
|
PayTester.LBTCNBXplorerUri = LBTCExplorerClient.Address;
|
|
|
|
}
|
2020-07-28 11:30:23 +02:00
|
|
|
|
2019-12-24 10:11:21 +01:00
|
|
|
public void ActivateLightning()
|
2021-03-02 03:11:58 +01:00
|
|
|
{
|
2023-05-16 02:17:21 +02:00
|
|
|
ActivateLightning(LightningConnectionType.CLightning);
|
2021-03-02 03:11:58 +01:00
|
|
|
}
|
2023-11-21 10:55:02 +01:00
|
|
|
public void ActivateLightning(string internalNode)
|
2019-12-24 10:11:21 +01:00
|
|
|
{
|
|
|
|
var btc = NetworkProvider.GetNetwork<BTCPayNetwork>("BTC").NBitcoinNetwork;
|
2023-11-21 10:55:02 +01:00
|
|
|
var factory = new LightningClientFactory(btc);
|
|
|
|
CustomerLightningD = factory.Create(GetEnvironment("TEST_CUSTOMERLIGHTNINGD", "type=clightning;server=tcp://127.0.0.1:30992/"));
|
|
|
|
MerchantLightningD = factory.Create(GetEnvironment("TEST_MERCHANTLIGHTNINGD", "type=clightning;server=tcp://127.0.0.1:30993/"));
|
2020-08-16 15:09:10 +02:00
|
|
|
MerchantCharge = new ChargeTester(this, "TEST_MERCHANTCHARGE", "type=charge;server=http://127.0.0.1:54938/;api-token=foiewnccewuify;allowinsecure=true", "merchant_lightningd", btc);
|
2021-02-01 07:55:13 +01:00
|
|
|
MerchantLnd = new LndMockTester(this, "TEST_MERCHANTLND", "http://lnd:lnd@127.0.0.1:35531/", "merchant_lnd", btc);
|
2019-12-24 10:11:21 +01:00
|
|
|
PayTester.UseLightning = true;
|
2021-03-02 03:11:58 +01:00
|
|
|
PayTester.IntegratedLightning = GetLightningConnectionString(internalNode, true);
|
|
|
|
}
|
2023-11-28 15:20:03 +01:00
|
|
|
public string GetLightningConnectionString(string connectionType, bool isMerchant)
|
2021-03-02 03:11:58 +01:00
|
|
|
{
|
|
|
|
string connectionString = null;
|
|
|
|
if (connectionType is null)
|
2024-04-04 09:31:04 +02:00
|
|
|
return LightningPaymentMethodConfig.InternalNode;
|
2023-05-16 02:17:21 +02:00
|
|
|
if (connectionType == LightningConnectionType.CLightning)
|
2021-03-02 03:11:58 +01:00
|
|
|
{
|
|
|
|
if (isMerchant)
|
|
|
|
connectionString = "type=clightning;server=" +
|
|
|
|
((CLightningClient)MerchantLightningD).Address.AbsoluteUri;
|
|
|
|
else
|
|
|
|
connectionString = "type=clightning;server=" +
|
|
|
|
((CLightningClient)CustomerLightningD).Address.AbsoluteUri;
|
|
|
|
}
|
|
|
|
else if (connectionType == LightningConnectionType.LndREST)
|
|
|
|
{
|
|
|
|
if (isMerchant)
|
|
|
|
connectionString = $"type=lnd-rest;server={MerchantLnd.Swagger.BaseUrl};allowinsecure=true";
|
|
|
|
else
|
|
|
|
throw new NotSupportedException();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
throw new NotSupportedException(connectionType.ToString());
|
|
|
|
return connectionString;
|
2019-12-24 10:11:21 +01:00
|
|
|
}
|
2019-12-24 08:20:44 +01:00
|
|
|
|
2018-04-15 14:18:51 +02:00
|
|
|
public bool Dockerized
|
|
|
|
{
|
|
|
|
get; set;
|
|
|
|
}
|
|
|
|
|
2019-10-07 09:04:25 +02:00
|
|
|
public Task StartAsync()
|
2018-04-15 14:18:51 +02:00
|
|
|
{
|
2019-10-07 09:04:25 +02:00
|
|
|
return PayTester.StartAsync();
|
2017-12-13 07:49:19 +01:00
|
|
|
}
|
|
|
|
|
2018-03-17 07:45:44 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Connect a customer LN node to the merchant LN node
|
|
|
|
/// </summary>
|
|
|
|
/// <returns></returns>
|
2019-03-27 07:53:06 +01:00
|
|
|
public async Task EnsureChannelsSetup()
|
2017-12-13 07:49:19 +01:00
|
|
|
{
|
2021-11-22 09:16:08 +01:00
|
|
|
TestLogs.LogInformation("Connecting channels");
|
|
|
|
BTCPayServer.Lightning.Tests.ConnectChannels.Logs = LoggerProvider.CreateLogger("Connect channels");
|
2019-03-27 07:53:06 +01:00
|
|
|
await BTCPayServer.Lightning.Tests.ConnectChannels.ConnectAll(ExplorerNode, GetLightningSenderClients(), GetLightningDestClients()).ConfigureAwait(false);
|
2021-11-22 09:16:08 +01:00
|
|
|
TestLogs.LogInformation("Channels connected");
|
2018-03-17 07:45:44 +01:00
|
|
|
}
|
|
|
|
|
2018-08-30 04:50:39 +02:00
|
|
|
private IEnumerable<ILightningClient> GetLightningSenderClients()
|
2018-03-17 07:45:44 +01:00
|
|
|
{
|
2018-08-30 04:50:39 +02:00
|
|
|
yield return CustomerLightningD;
|
2018-07-01 14:41:06 +02:00
|
|
|
}
|
|
|
|
|
2018-08-30 04:50:39 +02:00
|
|
|
private IEnumerable<ILightningClient> GetLightningDestClients()
|
2018-07-01 14:41:06 +02:00
|
|
|
{
|
2018-08-30 04:50:39 +02:00
|
|
|
yield return MerchantLightningD;
|
|
|
|
yield return MerchantLnd.Client;
|
2018-02-25 16:48:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public void SendLightningPayment(Invoice invoice)
|
|
|
|
{
|
|
|
|
SendLightningPaymentAsync(invoice).GetAwaiter().GetResult();
|
|
|
|
}
|
|
|
|
|
2022-05-18 07:57:36 +02:00
|
|
|
public async Task<PayResponse> SendLightningPaymentAsync(Invoice invoice)
|
2018-02-25 16:48:12 +01:00
|
|
|
{
|
2023-04-25 01:51:38 +02:00
|
|
|
var bolt11 = invoice.CryptoInfo.Where(o => o.PaymentUrls?.BOLT11 != null).First().PaymentUrls.BOLT11;
|
2018-02-25 16:48:12 +01:00
|
|
|
bolt11 = bolt11.Replace("lightning:", "", StringComparison.OrdinalIgnoreCase);
|
2022-05-18 07:57:36 +02:00
|
|
|
return await CustomerLightningD.Pay(bolt11);
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
|
|
|
|
2020-09-24 12:18:19 +02:00
|
|
|
public async Task<T> WaitForEvent<T>(Func<Task> action, Func<T, bool> correctEvent = null)
|
2020-03-05 19:04:08 +01:00
|
|
|
{
|
2020-03-29 17:28:22 +02:00
|
|
|
var tcs = new TaskCompletionSource<T>(TaskCreationOptions.RunContinuationsAsynchronously);
|
2020-03-05 19:04:08 +01:00
|
|
|
var sub = PayTester.GetService<EventAggregator>().Subscribe<T>(evt =>
|
|
|
|
{
|
2020-11-13 08:28:15 +01:00
|
|
|
if (correctEvent is null)
|
2020-09-24 12:18:19 +02:00
|
|
|
tcs.TrySetResult(evt);
|
|
|
|
else if (correctEvent(evt))
|
|
|
|
{
|
|
|
|
tcs.TrySetResult(evt);
|
|
|
|
}
|
2020-03-05 19:04:08 +01:00
|
|
|
});
|
2023-04-27 03:59:19 +02:00
|
|
|
if (action != null)
|
|
|
|
await action.Invoke();
|
2020-03-29 17:28:22 +02:00
|
|
|
var result = await tcs.Task;
|
2020-03-05 19:04:08 +01:00
|
|
|
sub.Dispose();
|
2020-03-29 17:28:22 +02:00
|
|
|
return result;
|
2020-03-05 19:04:08 +01:00
|
|
|
}
|
|
|
|
|
2018-08-30 04:50:39 +02:00
|
|
|
public ILightningClient CustomerLightningD { get; set; }
|
2018-06-15 22:02:40 +02:00
|
|
|
|
2018-08-30 04:50:39 +02:00
|
|
|
public ILightningClient MerchantLightningD { get; private set; }
|
2018-02-23 07:21:42 +01:00
|
|
|
public ChargeTester MerchantCharge { get; private set; }
|
2018-05-25 17:44:59 +02:00
|
|
|
public LndMockTester MerchantLnd { get; set; }
|
2017-12-13 07:49:19 +01:00
|
|
|
|
|
|
|
internal string GetEnvironment(string variable, string defaultValue)
|
2017-10-27 10:53:04 +02:00
|
|
|
{
|
|
|
|
var var = Environment.GetEnvironmentVariable(variable);
|
|
|
|
return String.IsNullOrEmpty(var) ? defaultValue : var;
|
|
|
|
}
|
|
|
|
|
|
|
|
public TestAccount NewAccount()
|
|
|
|
{
|
|
|
|
return new TestAccount(this);
|
|
|
|
}
|
2024-01-18 09:16:57 +01:00
|
|
|
BTCPayNetworkProvider _NetworkProvider;
|
|
|
|
public BTCPayNetworkProvider NetworkProvider
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
return PayTester?.Networks ?? _NetworkProvider;
|
|
|
|
}
|
|
|
|
}
|
2017-10-27 10:53:04 +02:00
|
|
|
public RPCClient ExplorerNode
|
|
|
|
{
|
|
|
|
get; set;
|
|
|
|
}
|
2024-08-30 01:34:23 +02:00
|
|
|
|
2018-01-11 09:29:48 +01:00
|
|
|
public RPCClient LTCExplorerNode
|
|
|
|
{
|
|
|
|
get; set;
|
|
|
|
}
|
2020-06-28 10:55:27 +02:00
|
|
|
|
2019-12-24 08:20:44 +01:00
|
|
|
public RPCClient LBTCExplorerNode { get; set; }
|
2020-07-29 12:11:54 +02:00
|
|
|
public ExplorerClient LTCExplorerClient { get; set; }
|
|
|
|
public ExplorerClient LBTCExplorerClient { get; set; }
|
2018-01-11 09:29:48 +01:00
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
public ExplorerClient ExplorerClient
|
|
|
|
{
|
|
|
|
get; set;
|
|
|
|
}
|
|
|
|
|
|
|
|
public BTCPayServerTester PayTester
|
|
|
|
{
|
|
|
|
get; set;
|
|
|
|
}
|
2019-01-04 16:37:09 +01:00
|
|
|
|
2018-07-19 12:31:17 +02:00
|
|
|
public List<string> Stores { get; internal set; } = new List<string>();
|
2023-03-08 13:36:51 +01:00
|
|
|
public bool DeleteStore { get; set; } = true;
|
2023-04-27 05:48:47 +02:00
|
|
|
public BTCPayNetworkBase DefaultNetwork => NetworkProvider.DefaultNetwork;
|
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
public void Dispose()
|
|
|
|
{
|
2020-11-13 08:28:15 +01:00
|
|
|
foreach (var r in this.Resources)
|
|
|
|
r.Dispose();
|
2021-11-22 09:16:08 +01:00
|
|
|
TestLogs.LogInformation("Disposing the BTCPayTester...");
|
2023-03-08 13:36:51 +01:00
|
|
|
if (DeleteStore)
|
2018-07-19 12:31:17 +02:00
|
|
|
{
|
2023-03-08 13:36:51 +01:00
|
|
|
foreach (var store in Stores)
|
|
|
|
{
|
|
|
|
Xunit.Assert.True(PayTester.StoreRepository.DeleteStore(store).GetAwaiter().GetResult());
|
|
|
|
}
|
2018-07-19 12:31:17 +02:00
|
|
|
}
|
2017-10-27 10:53:04 +02:00
|
|
|
if (PayTester != null)
|
|
|
|
PayTester.Dispose();
|
2021-11-22 09:16:08 +01:00
|
|
|
TestLogs.LogInformation("BTCPayTester disposed");
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
|
|
|
}
|
2017-09-13 08:47:34 +02:00
|
|
|
}
|