Files
Umbraco-CMS/src/Umbraco.Tests/Testing/UmbracoTestBase.cs
2018-11-24 15:38:00 +01:00

426 lines
16 KiB
C#

using System;
using System.IO;
using System.Linq;
using System.Reflection;
using AutoMapper;
using Examine;
using Moq;
using NUnit.Framework;
using Umbraco.Core;
using Umbraco.Core.Cache;
using Umbraco.Core.Components;
using Umbraco.Core.Composing;
using Umbraco.Core.Composing.Composers;
using Umbraco.Core.Configuration;
using Umbraco.Core.Configuration.UmbracoSettings;
using Umbraco.Core.Events;
using Umbraco.Core.IO;
using Umbraco.Core.IO.MediaPathSchemes;
using Umbraco.Core.Logging;
using Umbraco.Core.Logging.Serilog;
using Umbraco.Core.Manifest;
using Umbraco.Core.Models.PublishedContent;
using Umbraco.Core.Persistence;
using Umbraco.Core.Persistence.Mappers;
using Umbraco.Core.Persistence.Repositories.Implement;
using Umbraco.Core.Persistence.SqlSyntax;
using Umbraco.Core.PropertyEditors;
using Umbraco.Core.Scoping;
using Umbraco.Core.Services;
using Umbraco.Core.Services.Implement;
using Umbraco.Core.Strings;
using Umbraco.Tests.TestHelpers;
using Umbraco.Tests.TestHelpers.Stubs;
using Umbraco.Web;
using Umbraco.Web.Services;
using Umbraco.Tests.Testing.Objects.Accessors;
using Umbraco.Web.Actions;
using Umbraco.Web.Composing.Composers;
using Umbraco.Web.ContentApps;
using Current = Umbraco.Core.Composing.Current;
using Umbraco.Web.Routing;
namespace Umbraco.Tests.Testing
{
/// <summary>
/// Provides the top-level base class for all Umbraco integration tests.
/// </summary>
/// <remarks>
/// True unit tests do not need to inherit from this class, but most of Umbraco tests
/// are not true unit tests but integration tests requiring services, databases, etc. This class
/// provides all the necessary environment, through DI. Yes, DI is bad in tests - unit tests.
/// But it is OK in integration tests.
/// </remarks>
public abstract class UmbracoTestBase
{
// this class
// ensures that Current is properly resetted
// ensures that a service container is properly initialized and disposed
// compose the required dependencies according to test options (UmbracoTestAttribute)
//
// everything is virtual (because, why not?)
// starting a test runs like this:
// - SetUp() // when overriding, call base.SetUp() *first* then setup your own stuff
// --- Compose() // when overriding, call base.Commpose() *first* then compose your own stuff
// --- Initialize() // same
// - test runs
// - TearDown() // when overriding, clear you own stuff *then* call base.TearDown()
//
// about attributes
//
// this class defines the SetUp and TearDown methods, with proper attributes, and
// these attributes are *inherited* so classes inheriting from this class should *not*
// add the attributes to SetUp nor TearDown again
//
// this class is *not* marked with the TestFeature attribute because it is *not* a
// test feature, and no test "base" class should be. only actual test feature classes
// should be marked with that attribute.
protected IContainer Container { get; private set; }
protected UmbracoTestAttribute Options { get; private set; }
protected static bool FirstTestInSession = true;
protected bool FirstTestInFixture = true;
internal TestObjects TestObjects { get; private set; }
private static TypeLoader _commonTypeLoader;
private TypeLoader _featureTypeLoader;
#region Accessors
protected ILogger Logger => Container.GetInstance<ILogger>();
protected IProfiler Profiler => Container.GetInstance<IProfiler>();
protected virtual ProfilingLogger ProfilingLogger => Container.GetInstance<ProfilingLogger>();
protected CacheHelper CacheHelper => Container.GetInstance<CacheHelper>();
protected virtual ISqlSyntaxProvider SqlSyntax => Container.GetInstance<ISqlSyntaxProvider>();
protected IMapperCollection Mappers => Container.GetInstance<IMapperCollection>();
#endregion
#region Setup
[SetUp]
public virtual void SetUp()
{
// should not need this if all other tests were clean
// but hey, never know, better avoid garbage-in
Reset();
Container = Current.Container = ContainerFactory.Create();
TestObjects = new TestObjects(Container);
// get/merge the attributes marking the method and/or the classes
Options = TestOptionAttributeBase.GetTestOptions<UmbracoTestAttribute>();
Compose();
Initialize();
}
protected virtual void Compose()
{
ComposeLogging(Options.Logger);
ComposeCacheHelper();
ComposeAutoMapper(Options.AutoMapper);
ComposePluginManager(Options.PluginManager);
ComposeDatabase(Options.Database);
ComposeApplication(Options.WithApplication);
// etc
ComposeWeb();
ComposeWtf();
// not sure really
var composition = new Composition(Container, RuntimeLevel.Run);
Compose(composition);
}
protected virtual void Compose(Composition composition)
{ }
protected virtual void Initialize()
{
InitializeAutoMapper(Options.AutoMapper);
InitializeApplication(Options.WithApplication);
}
#endregion
#region Compose
protected virtual void ComposeLogging(UmbracoTestOptions.Logger option)
{
if (option == UmbracoTestOptions.Logger.Mock)
{
Container.RegisterSingleton(f => Mock.Of<ILogger>());
Container.RegisterSingleton(f => Mock.Of<IProfiler>());
}
else if (option == UmbracoTestOptions.Logger.Serilog)
{
Container.RegisterSingleton<ILogger>(f => new SerilogLogger(new FileInfo(TestHelper.MapPathForTest("~/unit-test.config"))));
Container.RegisterSingleton<IProfiler>(f => new LogProfiler(f.GetInstance<ILogger>()));
}
else if (option == UmbracoTestOptions.Logger.Console)
{
Container.RegisterSingleton<ILogger>(f => new ConsoleLogger());
Container.RegisterSingleton<IProfiler>(f => new LogProfiler(f.GetInstance<ILogger>()));
}
Container.RegisterSingleton(f => new ProfilingLogger(f.GetInstance<ILogger>(), f.GetInstance<IProfiler>()));
}
protected virtual void ComposeWeb()
{
//TODO: Should we 'just' register the WebRuntimeComponent?
// imported from TestWithSettingsBase
// which was inherited by TestWithApplicationBase so pretty much used everywhere
Umbraco.Web.Composing.Current.UmbracoContextAccessor = new TestUmbracoContextAccessor();
// web
Container.Register(_ => Umbraco.Web.Composing.Current.UmbracoContextAccessor);
Container.RegisterSingleton<PublishedRouter>();
Container.RegisterCollectionBuilder<ContentFinderCollectionBuilder>();
Container.Register<IContentLastChanceFinder, TestLastChanceFinder>();
Container.Register<IVariationContextAccessor, TestVariationContextAccessor>();
}
protected virtual void ComposeWtf()
{
// what else?
var runtimeStateMock = new Mock<IRuntimeState>();
runtimeStateMock.Setup(x => x.Level).Returns(RuntimeLevel.Run);
Container.RegisterSingleton(f => runtimeStateMock.Object);
// ah...
Container.RegisterCollectionBuilder<ActionCollectionBuilder>();
Container.RegisterCollectionBuilder<PropertyValueConverterCollectionBuilder>();
Container.RegisterSingleton<IPublishedContentTypeFactory, PublishedContentTypeFactory>();
Container.RegisterSingleton<IMediaPathScheme, OriginalMediaPathScheme>();
// register empty content apps collection
Container.RegisterCollectionBuilder<ContentAppDefinitionCollectionBuilder>();
}
protected virtual void ComposeCacheHelper()
{
Container.RegisterSingleton(f => CacheHelper.Disabled);
Container.RegisterSingleton(f => f.GetInstance<CacheHelper>().RuntimeCache);
}
protected virtual void ComposeAutoMapper(bool configure)
{
if (configure == false) return;
Container.ComposeCoreMappingProfiles();
Container.ComposeWebMappingProfiles();
}
protected virtual void ComposePluginManager(UmbracoTestOptions.PluginManager pluginManager)
{
Container.RegisterSingleton(f =>
{
switch (pluginManager)
{
case UmbracoTestOptions.PluginManager.Default:
return _commonTypeLoader ?? (_commonTypeLoader = CreateCommonPluginManager(f));
case UmbracoTestOptions.PluginManager.PerFixture:
return _featureTypeLoader ?? (_featureTypeLoader = CreatePluginManager(f));
case UmbracoTestOptions.PluginManager.PerTest:
return CreatePluginManager(f);
default:
throw new ArgumentOutOfRangeException(nameof(pluginManager));
}
});
}
protected virtual TypeLoader CreatePluginManager(IContainer f)
{
return CreateCommonPluginManager(f);
}
private static TypeLoader CreateCommonPluginManager(IContainer f)
{
return new TypeLoader(f.GetInstance<CacheHelper>().RuntimeCache, f.GetInstance<IGlobalSettings>(), f.GetInstance<ProfilingLogger>(), false)
{
AssembliesToScan = new[]
{
Assembly.Load("Umbraco.Core"),
Assembly.Load("Umbraco.Web"),
Assembly.Load("Umbraco.Tests")
}
};
}
protected virtual void ComposeDatabase(UmbracoTestOptions.Database option)
{
if (option == UmbracoTestOptions.Database.None) return;
// create the file
// create the schema
}
protected virtual void ComposeApplication(bool withApplication)
{
if (withApplication == false) return;
var umbracoSettings = SettingsForTests.GetDefaultUmbracoSettings();
var globalSettings = SettingsForTests.GetDefaultGlobalSettings();
//apply these globally
SettingsForTests.ConfigureSettings(umbracoSettings);
SettingsForTests.ConfigureSettings(globalSettings);
// default Datalayer/Repositories/SQL/Database/etc...
Container.ComposeRepositories();
// register basic stuff that might need to be there for some container resolvers to work
Container.RegisterSingleton(factory => umbracoSettings);
Container.RegisterSingleton(factory => globalSettings);
Container.RegisterSingleton(factory => umbracoSettings.Content);
Container.RegisterSingleton(factory => umbracoSettings.Templates);
Container.RegisterSingleton(factory => umbracoSettings.WebRouting);
Container.RegisterSingleton<IExamineManager>(factory => ExamineManager.Instance);
// register filesystems
Container.RegisterSingleton(factory => TestObjects.GetFileSystemsMock());
var logger = Mock.Of<ILogger>();
var scheme = Mock.Of<IMediaPathScheme>();
var config = Mock.Of<IContentSection>();
var mediaFileSystem = new MediaFileSystem(Mock.Of<IFileSystem>(), config, scheme, logger);
Container.RegisterSingleton<IMediaFileSystem>(factory => mediaFileSystem);
// no factory (noop)
Container.RegisterSingleton<IPublishedModelFactory, NoopPublishedModelFactory>();
// register application stuff (database factory & context, services...)
Container.RegisterCollectionBuilder<MapperCollectionBuilder>()
.AddCore();
Container.RegisterSingleton<IEventMessagesFactory>(_ => new TransientEventMessagesFactory());
var sqlSyntaxProviders = TestObjects.GetDefaultSqlSyntaxProviders(Logger);
Container.RegisterSingleton<ISqlSyntaxProvider>(_ => sqlSyntaxProviders.OfType<SqlCeSyntaxProvider>().First());
Container.RegisterSingleton<IUmbracoDatabaseFactory>(f => new UmbracoDatabaseFactory(
Constants.System.UmbracoConnectionName,
sqlSyntaxProviders,
Logger,
Mock.Of<IMapperCollection>()));
Container.RegisterSingleton(f => f.TryGetInstance<IUmbracoDatabaseFactory>().SqlContext);
Container.RegisterCollectionBuilder<UrlSegmentProviderCollectionBuilder>(); // empty
Container.RegisterSingleton(factory
=> TestObjects.GetScopeProvider(factory.TryGetInstance<ILogger>(), factory.TryGetInstance<FileSystems>(), factory.TryGetInstance<IUmbracoDatabaseFactory>()));
Container.RegisterSingleton(factory => (IScopeAccessor) factory.GetInstance<IScopeProvider>());
Container.ComposeServices();
// composition root is doing weird things, fix
Container.RegisterSingleton<IApplicationTreeService, ApplicationTreeService>();
Container.RegisterSingleton<ISectionService, SectionService>();
// somehow property editor ends up wanting this
Container.RegisterCollectionBuilder<ManifestValueValidatorCollectionBuilder>();
Container.RegisterSingleton<ManifestParser>();
// note - don't register collections, use builders
Container.RegisterCollectionBuilder<DataEditorCollectionBuilder>();
var temp = Container.GetInstance<DataEditorCollectionBuilder>();
Container.RegisterSingleton<PropertyEditorCollection>();
Container.RegisterSingleton<ParameterEditorCollection>();
}
#endregion
#region Initialize
protected virtual void InitializeAutoMapper(bool configure)
{
if (configure == false) return;
Mapper.Initialize(configuration =>
{
var profiles = Container.GetAllInstances<Profile>();
foreach (var profile in profiles)
configuration.AddProfile(profile);
});
}
protected virtual void InitializeApplication(bool withApplication)
{
if (withApplication == false) return;
TestHelper.InitializeContentDirectories();
}
#endregion
#region TearDown and Reset
[TearDown]
public virtual void TearDown()
{
FirstTestInFixture = false;
FirstTestInSession = false;
Reset();
if (Options.WithApplication)
{
TestHelper.CleanContentDirectories();
TestHelper.CleanUmbracoSettingsConfig();
}
}
protected virtual void Reset()
{
// reset and dispose scopes
// ensures we don't leak an opened database connection
// which would lock eg SqlCe .sdf files
if (Container?.TryGetInstance<IScopeProvider>() is ScopeProvider scopeProvider)
{
Core.Scoping.Scope scope;
while ((scope = scopeProvider.AmbientScope) != null)
{
scope.Reset();
scope.Dispose();
}
}
Current.Reset();
Container?.Dispose();
Container = null;
// reset all other static things that should not be static ;(
UriUtility.ResetAppDomainAppVirtualPath();
SettingsForTests.Reset(); // fixme - should it be optional?
Mapper.Reset();
// clear static events
DocumentRepository.ClearScopeEvents();
MediaRepository.ClearScopeEvents();
MemberRepository.ClearScopeEvents();
ContentTypeService.ClearScopeEvents();
MediaTypeService.ClearScopeEvents();
MemberTypeService.ClearScopeEvents();
}
#endregion
}
}