// Copyright (c) .NET Foundation. All rights reserved. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. using System; using System.Collections.Generic; using System.Net.Http; using Microsoft.AspNetCore.Server.IntegrationTesting; using Microsoft.AspNetCore.Server.IntegrationTesting.IIS; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Testing; namespace Microsoft.AspNetCore.Server.IISIntegration.FunctionalTests { public class OutOfProcessTestSiteFixture : IISTestSiteFixture { public OutOfProcessTestSiteFixture() : base(Configure) { } private static void Configure(IISDeploymentParameters deploymentParameters) { deploymentParameters.ApplicationPath = Helpers.GetOutOfProcessTestSitesPath(); deploymentParameters.HostingModel = HostingModel.OutOfProcess; } } public class OutOfProcessV1TestSiteFixture : IISTestSiteFixture { public OutOfProcessV1TestSiteFixture() : base(Configure) { } private static void Configure(IISDeploymentParameters deploymentParameters) { deploymentParameters.ApplicationPath = Helpers.GetOutOfProcessTestSitesPath(); deploymentParameters.HostingModel = HostingModel.OutOfProcess; deploymentParameters.AncmVersion = AncmVersion.AspNetCoreModule; } } public class IISTestSiteFixture : IDisposable { private ApplicationDeployer _deployer; private ILoggerFactory _loggerFactory; private ForwardingProvider _forwardingProvider; private IISDeploymentResult _deploymentResult; private readonly Action _configure; public IISTestSiteFixture() : this(_ => { }) { } public IISTestSiteFixture(Action configure) { var logging = AssemblyTestLog.ForAssembly(typeof(IISTestSiteFixture).Assembly); _loggerFactory = logging.CreateLoggerFactory(null, nameof(IISTestSiteFixture)); _forwardingProvider = new ForwardingProvider(); _loggerFactory.AddProvider(_forwardingProvider); _configure = configure; } public HttpClient Client => DeploymentResult.HttpClient; public IISDeploymentResult DeploymentResult { get { EnsureInitialized(); return _deploymentResult; } } public TestConnection CreateTestConnection() { return new TestConnection(Client.BaseAddress.Port); } public void Dispose() { _deployer?.Dispose(); } public void Attach(LoggedTest test) { if (_forwardingProvider.LoggerFactory != null) { throw new InvalidOperationException("Test instance is already attached to this fixture"); } _forwardingProvider.LoggerFactory = test.LoggerFactory; } public void Detach(LoggedTest test) { if (_forwardingProvider.LoggerFactory != test.LoggerFactory) { throw new InvalidOperationException("Different test is attached to this fixture"); } _forwardingProvider.LoggerFactory = null; } private void EnsureInitialized() { if (_deployer != null) { return; } var deploymentParameters = new IISDeploymentParameters(Helpers.GetInProcessTestSitesPath(), DeployerSelector.ServerType, RuntimeFlavor.CoreClr, RuntimeArchitecture.x64) { TargetFramework = Tfm.NetCoreApp22, AncmVersion = AncmVersion.AspNetCoreModuleV2, HostingModel = HostingModel.InProcess, PublishApplicationBeforeDeployment = true, }; _configure(deploymentParameters); _deployer = IISApplicationDeployerFactory.Create(deploymentParameters, _loggerFactory); _deploymentResult = (IISDeploymentResult)_deployer.DeployAsync().Result; } private class ForwardingProvider : ILoggerProvider { private readonly List _loggers = new List(); private ILoggerFactory _loggerFactory; public ILoggerFactory LoggerFactory { get => _loggerFactory; set { lock (_loggers) { _loggerFactory = value; foreach (var logger in _loggers) { logger.Logger = _loggerFactory?.CreateLogger("FIXTURE:" + logger.Name); } } } } public void Dispose() { lock (_loggers) { _loggers.Clear(); } } public ILogger CreateLogger(string categoryName) { lock (_loggers) { var logger = new ForwardingLogger(categoryName); _loggers.Add(logger); return logger; } } } internal class ForwardingLogger : ILogger { public ForwardingLogger(string name) { Name = name; } public ILogger Logger { get; set; } public string Name { get; set; } public void Log(LogLevel logLevel, EventId eventId, TState state, Exception exception, Func formatter) { Logger?.Log(logLevel, eventId, state, exception, formatter); } public bool IsEnabled(LogLevel logLevel) => Logger?.IsEnabled(logLevel) == true; public IDisposable BeginScope(TState state) => Logger?.BeginScope(state); } } }