using System; using System.Collections.Generic; using System.Data.SQLite; using System.Linq; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; using NUnit.Framework; using NzbDrone.Core.Datastore; using NzbDrone.Core.Datastore.Migration.Framework; namespace NzbDrone.Core.Test.Framework { public abstract class DbTest : DbTest where TSubject : class where TModel : ModelBase, new() { private TSubject _subject; protected BasicRepository Storage { get; private set; } protected IList AllStoredModels => Storage.All().ToList(); protected TModel StoredModel => Storage.All().Single(); [SetUp] public void CoreTestSetup() { _subject = null; Storage = Mocker.Resolve>(); } protected TSubject Subject { get { if (_subject == null) { _subject = Mocker.Resolve(); } return _subject; } } } [Category("DbTest")] public abstract class DbTest : CoreTest { private ITestDatabase _db; protected virtual MigrationType MigrationType => MigrationType.Main; protected ITestDatabase Db { get { if (_db == null) { throw new InvalidOperationException("Test object database doesn't exists. Make sure you call WithRealDb() if you intend to use an actual database."); } return _db; } } protected virtual ITestDatabase WithTestDb(MigrationContext migrationContext) { var factory = Mocker.Resolve(); var database = factory.Create(migrationContext); Mocker.SetConstant(database); switch (MigrationType) { case MigrationType.Main: { var mainDb = new MainDatabase(database); Mocker.SetConstant(mainDb); break; } case MigrationType.Log: { var logDb = new LogDatabase(database); Mocker.SetConstant(logDb); break; } default: { throw new ArgumentException("Invalid MigrationType"); } } var testDb = new TestDatabase(database); return testDb; } protected virtual void SetupLogging() { Mocker.SetConstant(NullLoggerProvider.Instance); } protected void SetupContainer() { WithTempAsAppPath(); SetupLogging(); Mocker.SetConstant(Mocker.Resolve()); Mocker.SetConstant(Mocker.Resolve()); SqlBuilderExtensions.LogSql = true; } [SetUp] public virtual void SetupDb() { SetupContainer(); _db = WithTestDb(new MigrationContext(MigrationType)); } [TearDown] public void TearDown() { // Make sure there are no lingering connections. (When this happens it means we haven't disposed something properly) GC.Collect(); GC.WaitForPendingFinalizers(); SQLiteConnection.ClearAllPools(); if (TestFolderInfo != null) { DeleteTempFolder(TestFolderInfo.AppDataFolder); } } } }