using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net.Http; using System.Security.Claims; using System.Text; using System.Threading; using System.Threading.Tasks; using BTCPayServer.Configuration; using BTCPayServer.HostedServices; using BTCPayServer.Hosting; using BTCPayServer.Rating; using BTCPayServer.Services; using BTCPayServer.Services.Invoices; using BTCPayServer.Services.Rates; using BTCPayServer.Services.Stores; using BTCPayServer.Tests.Logging; using BTCPayServer.Tests.Mocks; using Microsoft.AspNetCore.Hosting; using Microsoft.AspNetCore.Hosting.Server.Features; using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Mvc; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.DependencyInjection.Extensions; using Microsoft.Extensions.Logging; using NBitcoin; using NBXplorer; using AuthenticationSchemes = BTCPayServer.Security.AuthenticationSchemes; namespace BTCPayServer.Tests { public enum TestDatabases { Postgres, MySQL, } public class BTCPayServerTester : IDisposable { private readonly string _Directory; public BTCPayServerTester(string scope) { this._Directory = scope ?? throw new ArgumentNullException(nameof(scope)); } public Uri NBXplorerUri { get; set; } public Uri LTCNBXplorerUri { get; set; } public Uri LBTCNBXplorerUri { get; set; } public Uri ServerUri { get; set; } public string MySQL { get; set; } public string Postgres { get; set; } IWebHost _Host; public int Port { get; set; } public TestDatabases TestDatabase { get; set; } public bool MockRates { get; set; } = true; public string SocksEndpoint { get; set; } public HashSet Chains { get; set; } = new HashSet() { "BTC" }; public bool UseLightning { get; set; } public bool AllowAdminRegistration { get; set; } = true; public bool DisableRegistration { get; set; } = false; public async Task StartAsync() { if (!Directory.Exists(_Directory)) Directory.CreateDirectory(_Directory); string chain = NBXplorerDefaultSettings.GetFolderName(NetworkType.Regtest); string chainDirectory = Path.Combine(_Directory, chain); if (!Directory.Exists(chainDirectory)) Directory.CreateDirectory(chainDirectory); StringBuilder config = new StringBuilder(); config.AppendLine($"{chain.ToLowerInvariant()}=1"); if (InContainer) { config.AppendLine($"bind=0.0.0.0"); } config.AppendLine($"port={Port}"); config.AppendLine($"chains={string.Join(',', Chains)}"); if (Chains.Contains("BTC", StringComparer.OrdinalIgnoreCase)) { config.AppendLine($"btc.explorer.url={NBXplorerUri.AbsoluteUri}"); config.AppendLine($"btc.explorer.cookiefile=0"); } if (UseLightning) { config.AppendLine($"btc.lightning={IntegratedLightning.AbsoluteUri}"); var localLndBackupFile = Path.Combine(_Directory, "walletunlock.json"); File.Copy(TestUtils.GetTestDataFullPath("LndSeedBackup/walletunlock.json"), localLndBackupFile, true); config.AppendLine($"btc.external.lndseedbackup={localLndBackupFile}"); } if (Chains.Contains("LTC", StringComparer.OrdinalIgnoreCase)) { config.AppendLine($"ltc.explorer.url={LTCNBXplorerUri.AbsoluteUri}"); config.AppendLine($"ltc.explorer.cookiefile=0"); } if (Chains.Contains("LBTC", StringComparer.OrdinalIgnoreCase)) { config.AppendLine($"lbtc.explorer.url={LBTCNBXplorerUri.AbsoluteUri}"); config.AppendLine($"lbtc.explorer.cookiefile=0"); } if (AllowAdminRegistration) config.AppendLine("allow-admin-registration=1"); config.AppendLine($"torrcfile={TestUtils.GetTestDataFullPath("Tor/torrc")}"); config.AppendLine($"socksendpoint={SocksEndpoint}"); config.AppendLine($"debuglog=debug.log"); if (!string.IsNullOrEmpty(SSHPassword) && string.IsNullOrEmpty(SSHKeyFile)) config.AppendLine($"sshpassword={SSHPassword}"); if (!string.IsNullOrEmpty(SSHKeyFile)) config.AppendLine($"sshkeyfile={SSHKeyFile}"); if (!string.IsNullOrEmpty(SSHConnection)) config.AppendLine($"sshconnection={SSHConnection}"); if (TestDatabase == TestDatabases.MySQL && !String.IsNullOrEmpty(MySQL)) config.AppendLine($"mysql=" + MySQL); else if (!String.IsNullOrEmpty(Postgres)) config.AppendLine($"postgres=" + Postgres); var confPath = Path.Combine(chainDirectory, "settings.config"); await File.WriteAllTextAsync(confPath, config.ToString()); ServerUri = new Uri("http://" + HostName + ":" + Port + "/"); HttpClient = new HttpClient(); HttpClient.BaseAddress = ServerUri; Environment.SetEnvironmentVariable("ASPNETCORE_ENVIRONMENT", "Development"); var conf = new DefaultConfiguration() { Logger = Logs.LogProvider.CreateLogger("Console") }.CreateConfiguration(new[] { "--datadir", _Directory, "--conf", confPath, "--disable-registration", DisableRegistration ? "true" : "false" }); _Host = new WebHostBuilder() .UseConfiguration(conf) .UseContentRoot(FindBTCPayServerDirectory()) .UseWebRoot(Path.Combine(FindBTCPayServerDirectory(), "wwwroot")) .ConfigureServices(s => { s.AddLogging(l => { l.AddFilter("System.Net.Http.HttpClient", LogLevel.Critical); l.SetMinimumLevel(LogLevel.Information) .AddFilter("Microsoft", LogLevel.Error) .AddFilter("Hangfire", LogLevel.Error) .AddProvider(Logs.LogProvider); }); }) .ConfigureServices(services => { services.TryAddSingleton(new BTCPayServer.Services.Fees.FixedFeeProvider(new FeeRate(100L, 1))); }) .UseKestrel() .UseStartup() .Build(); await _Host.StartWithTasksAsync(); var urls = _Host.ServerFeatures.Get().Addresses; foreach (var url in urls) { Logs.Tester.LogInformation("Listening on " + url); } Logs.Tester.LogInformation("Server URI " + ServerUri); InvoiceRepository = (InvoiceRepository)_Host.Services.GetService(typeof(InvoiceRepository)); StoreRepository = (StoreRepository)_Host.Services.GetService(typeof(StoreRepository)); Networks = (BTCPayNetworkProvider)_Host.Services.GetService(typeof(BTCPayNetworkProvider)); if (MockRates) { var rateProvider = (RateProviderFactory)_Host.Services.GetService(typeof(RateProviderFactory)); rateProvider.Providers.Clear(); coinAverageMock = new MockRateProvider(); coinAverageMock.ExchangeRates.Add(new PairRate(CurrencyPair.Parse("BTC_USD"), new BidAsk(5000m))); coinAverageMock.ExchangeRates.Add(new PairRate(CurrencyPair.Parse("BTC_CAD"), new BidAsk(4500m))); coinAverageMock.ExchangeRates.Add(new PairRate(CurrencyPair.Parse("BTC_LTC"), new BidAsk(162m))); coinAverageMock.ExchangeRates.Add(new PairRate(CurrencyPair.Parse("LTC_USD"), new BidAsk(500m))); rateProvider.Providers.Add("coingecko", coinAverageMock); var bitflyerMock = new MockRateProvider(); bitflyerMock.ExchangeRates.Add(new PairRate(CurrencyPair.Parse("BTC_JPY"), new BidAsk(700000m))); rateProvider.Providers.Add("bitflyer", bitflyerMock); var quadrigacx = new MockRateProvider(); quadrigacx.ExchangeRates.Add(new PairRate(CurrencyPair.Parse("BTC_CAD"), new BidAsk(6000m))); rateProvider.Providers.Add("quadrigacx", quadrigacx); var bittrex = new MockRateProvider(); bittrex.ExchangeRates.Add(new PairRate(CurrencyPair.Parse("DOGE_BTC"), new BidAsk(0.004m))); rateProvider.Providers.Add("bittrex", bittrex); var bitfinex = new MockRateProvider(); bitfinex.ExchangeRates.Add(new PairRate(CurrencyPair.Parse("UST_BTC"), new BidAsk(0.000136m))); rateProvider.Providers.Add("bitfinex", bitfinex); var bitpay = new MockRateProvider(); bitpay.ExchangeRates.Add(new PairRate(CurrencyPair.Parse("ETB_BTC"), new BidAsk(0.1m))); rateProvider.Providers.Add("bitpay", bitpay); } Logs.Tester.LogInformation("Waiting site is operational..."); await WaitSiteIsOperational(); Logs.Tester.LogInformation("Site is now operational"); } MockRateProvider coinAverageMock; private async Task WaitSiteIsOperational() { _ = HttpClient.GetAsync("/").ConfigureAwait(false); using (var cts = new CancellationTokenSource(20_000)) { var synching = WaitIsFullySynched(cts.Token); await Task.WhenAll(synching).ConfigureAwait(false); } // Opportunistic call to wake up view compilation in debug mode, we don't need to await. } private async Task WaitIsFullySynched(CancellationToken cancellationToken) { var dashBoard = GetService(); while (!dashBoard.IsFullySynched()) { await Task.Delay(10, cancellationToken).ConfigureAwait(false); } } private string FindBTCPayServerDirectory() { var solutionDirectory = TestUtils.TryGetSolutionDirectoryInfo(Directory.GetCurrentDirectory()); return Path.Combine(solutionDirectory.FullName, "BTCPayServer"); } public HttpClient HttpClient { get; set; } public string HostName { get; internal set; } public InvoiceRepository InvoiceRepository { get; private set; } public StoreRepository StoreRepository { get; private set; } public BTCPayNetworkProvider Networks { get; private set; } public Uri IntegratedLightning { get; internal set; } public bool InContainer { get; internal set; } public T GetService() { return _Host.Services.GetRequiredService(); } public IServiceProvider ServiceProvider => _Host.Services; public string SSHPassword { get; internal set; } public string SSHKeyFile { get; internal set; } public string SSHConnection { get; set; } public T GetController(string userId = null, string storeId = null, bool isAdmin = false) where T : Controller { var context = new DefaultHttpContext(); context.Request.Host = new HostString("127.0.0.1", Port); context.Request.Scheme = "http"; context.Request.Protocol = "http"; if (userId != null) { List claims = new List(); claims.Add(new Claim(ClaimTypes.NameIdentifier, userId)); if (isAdmin) claims.Add(new Claim(ClaimTypes.Role, Roles.ServerAdmin)); context.User = new ClaimsPrincipal(new ClaimsIdentity(claims.ToArray(), AuthenticationSchemes.Cookie)); } if (storeId != null) { context.SetStoreData(GetService().FindStore(storeId, userId).GetAwaiter().GetResult()); } var scope = (IServiceScopeFactory)_Host.Services.GetService(typeof(IServiceScopeFactory)); var provider = scope.CreateScope().ServiceProvider; context.RequestServices = provider; var httpAccessor = provider.GetRequiredService(); httpAccessor.HttpContext = context; var controller = (T)ActivatorUtilities.CreateInstance(provider, typeof(T)); controller.Url = new UrlHelperMock(new Uri($"http://{HostName}:{Port}/")); controller.ControllerContext = new ControllerContext() { HttpContext = context }; return controller; } public void Dispose() { if (_Host != null) _Host.Dispose(); } public void ChangeRate(string pair, BidAsk bidAsk) { var p = CurrencyPair.Parse(pair); var index = coinAverageMock.ExchangeRates.FindIndex(o => o.CurrencyPair == p); coinAverageMock.ExchangeRates[index] = new PairRate(p, bidAsk); } } }