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;
|
2017-09-13 08:47:34 +02:00
|
|
|
using System.Runtime.CompilerServices;
|
2020-06-28 10:55:27 +02:00
|
|
|
using System.Threading.Tasks;
|
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;
|
2020-06-28 10:55:27 +02:00
|
|
|
using NBitcoin;
|
|
|
|
using NBitcoin.RPC;
|
|
|
|
using NBitpayClient;
|
|
|
|
using NBXplorer;
|
2017-09-13 08:47:34 +02:00
|
|
|
|
|
|
|
namespace BTCPayServer.Tests
|
|
|
|
{
|
2017-10-27 10:53:04 +02:00
|
|
|
public class ServerTester : IDisposable
|
|
|
|
{
|
2020-06-28 10:55:27 +02:00
|
|
|
public static ServerTester Create([CallerMemberNameAttribute] string scope = null, bool newDb = false)
|
2017-10-27 10:53:04 +02:00
|
|
|
{
|
2020-03-18 15:10:15 +01:00
|
|
|
return new ServerTester(scope, newDb);
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
|
|
|
|
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;
|
2020-03-18 15:10:15 +01:00
|
|
|
public ServerTester(string scope, bool newDb)
|
2017-10-27 10:53:04 +02:00
|
|
|
{
|
|
|
|
_Directory = scope;
|
|
|
|
if (Directory.Exists(_Directory))
|
|
|
|
Utils.DeleteDirectory(_Directory);
|
|
|
|
if (!Directory.Exists(_Directory))
|
|
|
|
Directory.CreateDirectory(_Directory);
|
|
|
|
|
2021-01-27 06:39:38 +01:00
|
|
|
NetworkProvider = new BTCPayNetworkProvider(ChainName.Regtest);
|
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
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
PayTester = new BTCPayServerTester(Path.Combine(_Directory, "pay"))
|
|
|
|
{
|
|
|
|
NBXplorerUri = ExplorerClient.Address,
|
2018-10-27 16:15:21 +02:00
|
|
|
TestDatabase = Enum.Parse<TestDatabases>(GetEnvironment("TESTS_DB", TestDatabases.Postgres.ToString()), true),
|
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
|
2018-02-26 10:58:02 +01:00
|
|
|
Postgres = GetEnvironment("TESTS_POSTGRES", "User ID=postgres;Host=127.0.0.1;Port=39372;Database=btcpayserver"),
|
2019-12-24 10:11:21 +01:00
|
|
|
MySQL = GetEnvironment("TESTS_MYSQL", "User ID=root;Host=127.0.0.1;Port=33036;Database=btcpayserver")
|
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}");
|
|
|
|
PayTester.MySQL = PayTester.MySQL.Replace("btcpayserver", $"btcpayserver{r}");
|
|
|
|
}
|
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
|
|
|
}
|
2020-07-29 12:11:54 +02:00
|
|
|
#if ALTCOINS
|
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
|
|
|
public void ActivateETH()
|
|
|
|
{
|
|
|
|
PayTester.Chains.Add("ETH");
|
|
|
|
}
|
|
|
|
|
2020-07-29 12:11:54 +02:00
|
|
|
#endif
|
2019-12-24 10:11:21 +01:00
|
|
|
public void ActivateLightning()
|
2021-03-02 03:11:58 +01:00
|
|
|
{
|
|
|
|
ActivateLightning(LightningConnectionType.Charge);
|
|
|
|
}
|
|
|
|
public void ActivateLightning(LightningConnectionType internalNode)
|
2019-12-24 10:11:21 +01:00
|
|
|
{
|
|
|
|
var btc = NetworkProvider.GetNetwork<BTCPayNetwork>("BTC").NBitcoinNetwork;
|
|
|
|
CustomerLightningD = LightningClientFactory.CreateClient(GetEnvironment("TEST_CUSTOMERLIGHTNINGD", "type=clightning;server=tcp://127.0.0.1:30992/"), btc);
|
|
|
|
MerchantLightningD = LightningClientFactory.CreateClient(GetEnvironment("TEST_MERCHANTLIGHTNINGD", "type=clightning;server=tcp://127.0.0.1:30993/"), btc);
|
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);
|
|
|
|
}
|
|
|
|
public string GetLightningConnectionString(LightningConnectionType? connectionType, bool isMerchant)
|
|
|
|
{
|
|
|
|
string connectionString = null;
|
|
|
|
if (connectionType is null)
|
|
|
|
return LightningSupportedPaymentMethod.InternalNode;
|
|
|
|
if (connectionType == LightningConnectionType.Charge)
|
|
|
|
{
|
|
|
|
if (isMerchant)
|
|
|
|
connectionString = $"type=charge;server={MerchantCharge.Client.Uri.AbsoluteUri};allowinsecure=true";
|
|
|
|
else
|
|
|
|
throw new NotSupportedException();
|
|
|
|
}
|
|
|
|
else if (connectionType == LightningConnectionType.CLightning)
|
|
|
|
{
|
|
|
|
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
|
|
|
{
|
2019-03-27 07:53:06 +01:00
|
|
|
Logs.Tester.LogInformation("Connecting channels");
|
2020-04-26 20:53:45 +02:00
|
|
|
BTCPayServer.Lightning.Tests.ConnectChannels.Logs = Logs.LogProvider.CreateLogger("Connect channels");
|
2019-03-27 07:53:06 +01:00
|
|
|
await BTCPayServer.Lightning.Tests.ConnectChannels.ConnectAll(ExplorerNode, GetLightningSenderClients(), GetLightningDestClients()).ConfigureAwait(false);
|
|
|
|
Logs.Tester.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();
|
|
|
|
}
|
|
|
|
|
|
|
|
public async Task SendLightningPaymentAsync(Invoice invoice)
|
|
|
|
{
|
|
|
|
var bolt11 = invoice.CryptoInfo.Where(o => o.PaymentUrls.BOLT11 != null).First().PaymentUrls.BOLT11;
|
|
|
|
bolt11 = bolt11.Replace("lightning:", "", StringComparison.OrdinalIgnoreCase);
|
2018-08-30 04:50:39 +02:00
|
|
|
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
|
|
|
});
|
|
|
|
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);
|
|
|
|
}
|
2018-01-11 14:52:28 +01:00
|
|
|
|
|
|
|
public BTCPayNetworkProvider NetworkProvider { get; private set; }
|
2017-10-27 10:53:04 +02:00
|
|
|
public RPCClient ExplorerNode
|
|
|
|
{
|
|
|
|
get; set;
|
|
|
|
}
|
2020-07-29 12:11:54 +02:00
|
|
|
#if ALTCOINS
|
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; }
|
|
|
|
#endif
|
2018-01-11 09:29:48 +01:00
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
public ExplorerClient ExplorerClient
|
|
|
|
{
|
|
|
|
get; set;
|
|
|
|
}
|
|
|
|
|
2020-06-29 05:07:48 +02:00
|
|
|
readonly HttpClient _Http = new HttpClient();
|
2017-10-27 10:53:04 +02:00
|
|
|
|
|
|
|
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>();
|
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();
|
2019-10-19 07:09:19 +02:00
|
|
|
Logs.Tester.LogInformation("Disposing the BTCPayTester...");
|
2018-08-30 04:50:39 +02:00
|
|
|
foreach (var store in Stores)
|
2018-07-19 12:31:17 +02:00
|
|
|
{
|
|
|
|
Xunit.Assert.True(PayTester.StoreRepository.DeleteStore(store).GetAwaiter().GetResult());
|
|
|
|
}
|
2017-10-27 10:53:04 +02:00
|
|
|
if (PayTester != null)
|
|
|
|
PayTester.Dispose();
|
2019-10-19 07:09:19 +02:00
|
|
|
Logs.Tester.LogInformation("BTCPayTester disposed");
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
|
|
|
}
|
2017-09-13 08:47:34 +02:00
|
|
|
}
|