// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.IO; using System.Linq; using System.Threading; using Microsoft.EntityFrameworkCore.Storage; using osu.Framework.Logging; using osu.Framework.Platform; using osu.Framework.Statistics; namespace osu.Game.Database { public class DatabaseContextFactory : IDatabaseContextFactory { private readonly Storage storage; public const string DATABASE_NAME = @"client.db"; private ThreadLocal threadContexts; private readonly object writeLock = new object(); private bool currentWriteDidWrite; private bool currentWriteDidError; private int currentWriteUsages; private IDbContextTransaction currentWriteTransaction; public DatabaseContextFactory(Storage storage) { this.storage = storage; recycleThreadContexts(); } private static readonly GlobalStatistic reads = GlobalStatistics.Get("Database", "Get (Read)"); private static readonly GlobalStatistic writes = GlobalStatistics.Get("Database", "Get (Write)"); private static readonly GlobalStatistic commits = GlobalStatistics.Get("Database", "Commits"); private static readonly GlobalStatistic rollbacks = GlobalStatistics.Get("Database", "Rollbacks"); /// /// Get a context for the current thread for read-only usage. /// If a is in progress, the existing write-safe context will be returned. /// public OsuDbContext Get() { reads.Value++; return threadContexts.Value; } /// /// Request a context for write usage. Can be consumed in a nested fashion (and will return the same underlying context). /// This method may block if a write is already active on a different thread. /// /// Whether to start a transaction for this write. /// A usage containing a usable context. public DatabaseWriteUsage GetForWrite(bool withTransaction = true) { writes.Value++; Monitor.Enter(writeLock); OsuDbContext context; try { if (currentWriteTransaction == null && withTransaction) { // this mitigates the fact that changes on tracked entities will not be rolled back with the transaction by ensuring write operations are always executed in isolated contexts. // if this results in sub-optimal efficiency, we may need to look into removing Database-level transactions in favour of running SaveChanges where we currently commit the transaction. if (threadContexts.IsValueCreated) recycleThreadContexts(); context = threadContexts.Value; currentWriteTransaction = context.Database.BeginTransaction(); } else { // we want to try-catch the retrieval of the context because it could throw an error (in CreateContext). context = threadContexts.Value; } } catch { // retrieval of a context could trigger a fatal error. Monitor.Exit(writeLock); throw; } Interlocked.Increment(ref currentWriteUsages); return new DatabaseWriteUsage(context, usageCompleted) { IsTransactionLeader = currentWriteTransaction != null && currentWriteUsages == 1 }; } private void usageCompleted(DatabaseWriteUsage usage) { int usages = Interlocked.Decrement(ref currentWriteUsages); try { currentWriteDidWrite |= usage.PerformedWrite; currentWriteDidError |= usage.Errors.Any(); if (usages == 0) { if (currentWriteDidError) { rollbacks.Value++; currentWriteTransaction?.Rollback(); } else { commits.Value++; currentWriteTransaction?.Commit(); } if (currentWriteDidWrite || currentWriteDidError) { // explicitly dispose to ensure any outstanding flushes happen as soon as possible (and underlying resources are purged). usage.Context.Dispose(); // once all writes are complete, we want to refresh thread-specific contexts to make sure they don't have stale local caches. recycleThreadContexts(); } currentWriteTransaction = null; currentWriteDidWrite = false; currentWriteDidError = false; } } finally { Monitor.Exit(writeLock); } } private void recycleThreadContexts() { // Contexts for other threads are not disposed as they may be in use elsewhere. Instead, fresh contexts are exposed // for other threads to use, and we rely on the finalizer inside OsuDbContext to handle their previous contexts threadContexts?.Value.Dispose(); threadContexts = new ThreadLocal(CreateContext, true); } protected virtual OsuDbContext CreateContext() => new OsuDbContext(CreateDatabaseConnectionString(DATABASE_NAME, storage)) { Database = { AutoTransactionsEnabled = false } }; public void CreateBackup(string backupFilename) { Logger.Log($"Creating full EF database backup at {backupFilename}", LoggingTarget.Database); using (var source = storage.GetStream(DATABASE_NAME, mode: FileMode.Open)) using (var destination = storage.GetStream(backupFilename, FileAccess.Write, FileMode.CreateNew)) source.CopyTo(destination); } public void ResetDatabase() { lock (writeLock) { recycleThreadContexts(); try { int attempts = 10; // Retry logic taken from MigratableStorage.AttemptOperation. while (true) { try { storage.Delete(DATABASE_NAME); return; } catch (Exception) { if (attempts-- == 0) throw; } Thread.Sleep(250); } } catch { // for now we are not sure why file handles are kept open by EF, but this is generally only used in testing } } } public void FlushConnections() { if (threadContexts != null) { foreach (var context in threadContexts.Values) context.Dispose(); } recycleThreadContexts(); } public static string CreateDatabaseConnectionString(string filename, Storage storage) => string.Concat("Data Source=", storage.GetFullPath($@"{filename}", true)); private readonly ManualResetEventSlim migrationComplete = new ManualResetEventSlim(); public void SetMigrationCompletion() => migrationComplete.Set(); public void WaitForMigrationCompletion() => migrationComplete.Wait(); } }