// 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.Linq; using System.Reflection; using Microsoft.AspNet.Mvc.ActionConstraints; using Microsoft.AspNet.Mvc.ApiExplorer; using Microsoft.AspNet.Mvc.ApplicationModels; using Microsoft.AspNet.Mvc.Core; using Microsoft.AspNet.Mvc.MvcServiceCollectionExtensionsTestControllers; using Microsoft.AspNet.Mvc.Razor; using Microsoft.Framework.DependencyInjection; using Microsoft.Framework.OptionsModel; using Xunit; namespace Microsoft.AspNet.Mvc { public class MvcServiceCollectionExtensionsTest { [Fact] public void WithControllersAsServices_AddsTypesToControllerTypeProviderAndServiceCollection() { // Arrange var collection = new ServiceCollection(); var controllerTypes = new[] { typeof(ControllerTypeA).GetTypeInfo(), typeof(TypeBController).GetTypeInfo() }; // Act MvcServiceCollectionExtensions.WithControllersAsServices(collection, controllerTypes); // Assert var services = collection.ToList(); Assert.Equal(4, services.Count); Assert.Equal(typeof(ControllerTypeA), services[0].ServiceType); Assert.Equal(typeof(ControllerTypeA), services[0].ImplementationType); Assert.Equal(ServiceLifetime.Transient, services[0].Lifetime); Assert.Equal(typeof(TypeBController), services[1].ServiceType); Assert.Equal(typeof(TypeBController), services[1].ImplementationType); Assert.Equal(ServiceLifetime.Transient, services[1].Lifetime); Assert.Equal(typeof(IControllerActivator), services[2].ServiceType); Assert.Equal(typeof(ServiceBasedControllerActivator), services[2].ImplementationType); Assert.Equal(ServiceLifetime.Transient, services[2].Lifetime); Assert.Equal(typeof(IControllerTypeProvider), services[3].ServiceType); var typeProvider = Assert.IsType(services[3].ImplementationInstance); Assert.Equal(controllerTypes, typeProvider.ControllerTypes.OrderBy(c => c.Name)); Assert.Equal(ServiceLifetime.Singleton, services[3].Lifetime); } [Fact] public void WithControllersAsServices_ScansControllersFromSpecifiedAssemblies() { // Arrange var collection = new ServiceCollection(); var assemblies = new[] { GetType().Assembly }; var controllerTypes = new[] { typeof(ControllerTypeA), typeof(TypeBController) }; // Act MvcServiceCollectionExtensions.WithControllersAsServices(collection, assemblies); // Assert var services = collection.ToList(); Assert.Equal(4, services.Count); Assert.Equal(typeof(ControllerTypeA), services[0].ServiceType); Assert.Equal(typeof(ControllerTypeA), services[0].ImplementationType); Assert.Equal(ServiceLifetime.Transient, services[0].Lifetime); Assert.Equal(typeof(TypeBController), services[1].ServiceType); Assert.Equal(typeof(TypeBController), services[1].ImplementationType); Assert.Equal(ServiceLifetime.Transient, services[1].Lifetime); Assert.Equal(typeof(IControllerActivator), services[2].ServiceType); Assert.Equal(typeof(ServiceBasedControllerActivator), services[2].ImplementationType); Assert.Equal(ServiceLifetime.Transient, services[2].Lifetime); Assert.Equal(typeof(IControllerTypeProvider), services[3].ServiceType); var typeProvider = Assert.IsType(services[3].ImplementationInstance); Assert.Equal(controllerTypes, typeProvider.ControllerTypes.OrderBy(c => c.Name)); Assert.Equal(ServiceLifetime.Singleton, services[3].Lifetime); } // Some MVC services can be registered multiple times, for example, 'IConfigureOptions' can // be registered by calling 'ConfigureMvc(...)' before the call to 'AddMvc()' in which case the options // configiuration is run in the order they were registered. // For these kind of multi registration service types, we want to make sure that MVC appends its services // to the list i.e it does a 'Add' rather than 'TryAdd' on the ServiceCollection. [Fact] public void MultiRegistrationServiceTypes_AreRegistered_MultipleTimes() { // Arrange var services = new ServiceCollection(); var multiRegistrationServiceTypes = MutliRegistrationServiceTypes; // Act MvcServiceCollectionExtensions.AddMvcServices(services); MvcServiceCollectionExtensions.AddMvcServices(services); // Assert foreach (var serviceType in multiRegistrationServiceTypes) { AssertServiceCountEquals(services, serviceType, 2); } } [Fact] public void SingleRegistrationServiceTypes_AreNotRegistered_MultipleTimes() { // Arrange var services = new ServiceCollection(); var multiRegistrationServiceTypes = MutliRegistrationServiceTypes; // Act MvcServiceCollectionExtensions.AddMvcServices(services); MvcServiceCollectionExtensions.AddMvcServices(services); // Assert var singleRegistrationServiceTypes = services .Where(serviceDescriptor => !multiRegistrationServiceTypes.Contains(serviceDescriptor.ServiceType)) .Select(serviceDescriptor => serviceDescriptor.ServiceType); foreach (var singleRegistrationType in singleRegistrationServiceTypes) { AssertServiceCountEquals(services, singleRegistrationType, 1); } } private IEnumerable MutliRegistrationServiceTypes { get { return new[] { typeof(IConfigureOptions), typeof(IConfigureOptions), typeof(IActionConstraintProvider), typeof(IActionDescriptorProvider), typeof(IActionInvokerProvider), typeof(IFilterProvider), typeof(IApiDescriptionProvider), typeof(IApplicationModelProvider), }; } } private void AssertServiceCountEquals( IServiceCollection services, Type serviceType, int expectedServiceRegistrationCount) { var serviceDescriptors = services.Where(serviceDescriptor => serviceDescriptor.ServiceType == serviceType); var actual = serviceDescriptors.Count(); Assert.True( (expectedServiceRegistrationCount == actual), $"Expected service type '{serviceType}' to be registered {expectedServiceRegistrationCount}" + $" time(s) but was actually registered {actual} time(s)."); } private class CustomActivator : IControllerActivator { public object Create(ActionContext context, Type controllerType) { throw new NotImplementedException(); } } public class CustomTypeProvider : IControllerTypeProvider { public IEnumerable ControllerTypes { get; set; } } } } // These controllers are used to test the UseControllersAsServices implementation // which REQUIRES that they be public top-level classes. To avoid having to stub out the // implementation of this class to test it, they are just top level classes. Don't reuse // these outside this test - find a better way or use nested classes to keep the tests // independent. namespace Microsoft.AspNet.Mvc.MvcServiceCollectionExtensionsTestControllers { public class ControllerTypeA : Controller { } public class TypeBController { } }