2021-01-07 13:07:36 +08:00
|
|
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
|
|
|
// See the LICENCE file in the repository root for full licence text.
|
|
|
|
|
|
|
|
using System.Threading;
|
2021-01-12 13:36:35 +08:00
|
|
|
using osu.Framework.Logging;
|
2021-01-07 13:07:36 +08:00
|
|
|
using osu.Framework.Platform;
|
|
|
|
using osu.Framework.Statistics;
|
|
|
|
using Realms;
|
|
|
|
|
|
|
|
namespace osu.Game.Database
|
|
|
|
{
|
|
|
|
public class RealmContextFactory : IRealmFactory
|
|
|
|
{
|
|
|
|
private readonly Storage storage;
|
|
|
|
|
|
|
|
private const string database_name = @"client";
|
|
|
|
|
2021-01-12 13:25:07 +08:00
|
|
|
private const int schema_version = 5;
|
|
|
|
|
2021-01-12 13:36:35 +08:00
|
|
|
private readonly ThreadLocal<Realm> threadContexts;
|
2021-01-07 13:07:36 +08:00
|
|
|
|
2021-01-12 13:36:35 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Lock object which is held for the duration of a write operation (via <see cref="GetForWrite"/>).
|
|
|
|
/// </summary>
|
2021-01-07 13:07:36 +08:00
|
|
|
private readonly object writeLock = new object();
|
|
|
|
|
|
|
|
private ThreadLocal<bool> refreshCompleted = new ThreadLocal<bool>();
|
|
|
|
|
|
|
|
private bool rollbackRequired;
|
|
|
|
|
|
|
|
private int currentWriteUsages;
|
|
|
|
|
|
|
|
private Transaction currentWriteTransaction;
|
|
|
|
|
2021-01-11 17:57:56 +08:00
|
|
|
public RealmContextFactory(Storage storage)
|
2021-01-07 13:07:36 +08:00
|
|
|
{
|
|
|
|
this.storage = storage;
|
2021-01-11 17:57:56 +08:00
|
|
|
|
2021-01-12 13:36:35 +08:00
|
|
|
threadContexts = new ThreadLocal<Realm>(createContext, true);
|
2021-01-07 14:41:29 +08:00
|
|
|
|
2021-01-12 13:42:43 +08:00
|
|
|
// creating a context will ensure our schema is up-to-date and migrated.
|
|
|
|
var realm = Get();
|
|
|
|
Logger.Log($"Opened realm \"{realm.Config.DatabasePath}\" at version {realm.Config.SchemaVersion}");
|
2021-01-07 14:41:29 +08:00
|
|
|
}
|
|
|
|
|
2021-01-12 12:51:37 +08:00
|
|
|
private void onMigration(Migration migration, ulong lastSchemaVersion)
|
2021-01-07 14:41:29 +08:00
|
|
|
{
|
2021-01-07 13:07:36 +08:00
|
|
|
}
|
|
|
|
|
2021-01-07 14:01:41 +08:00
|
|
|
private static readonly GlobalStatistic<int> reads = GlobalStatistics.Get<int>("Realm", "Get (Read)");
|
|
|
|
private static readonly GlobalStatistic<int> writes = GlobalStatistics.Get<int>("Realm", "Get (Write)");
|
|
|
|
private static readonly GlobalStatistic<int> commits = GlobalStatistics.Get<int>("Realm", "Commits");
|
|
|
|
private static readonly GlobalStatistic<int> rollbacks = GlobalStatistics.Get<int>("Realm", "Rollbacks");
|
2021-01-12 13:25:22 +08:00
|
|
|
private static readonly GlobalStatistic<int> contexts_open = GlobalStatistics.Get<int>("Realm", "Contexts (Open)");
|
|
|
|
private static readonly GlobalStatistic<int> contexts_created = GlobalStatistics.Get<int>("Realm", "Contexts (Created)");
|
2021-01-07 13:07:36 +08:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Get a context for the current thread for read-only usage.
|
|
|
|
/// If a <see cref="RealmWriteUsage"/> is in progress, the existing write-safe context will be returned.
|
|
|
|
/// </summary>
|
|
|
|
public Realm Get()
|
|
|
|
{
|
|
|
|
reads.Value++;
|
|
|
|
return getContextForCurrentThread();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// 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.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns>A usage containing a usable context.</returns>
|
|
|
|
public RealmWriteUsage GetForWrite()
|
|
|
|
{
|
|
|
|
writes.Value++;
|
|
|
|
Monitor.Enter(writeLock);
|
|
|
|
Realm context;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
context = getContextForCurrentThread();
|
|
|
|
|
2021-01-07 14:51:29 +08:00
|
|
|
currentWriteTransaction ??= context.BeginWrite();
|
2021-01-07 13:07:36 +08:00
|
|
|
}
|
|
|
|
catch
|
|
|
|
{
|
|
|
|
// retrieval of a context could trigger a fatal error.
|
|
|
|
Monitor.Exit(writeLock);
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
|
|
|
|
Interlocked.Increment(ref currentWriteUsages);
|
|
|
|
|
|
|
|
return new RealmWriteUsage(context, usageCompleted) { IsTransactionLeader = currentWriteTransaction != null && currentWriteUsages == 1 };
|
|
|
|
}
|
|
|
|
|
|
|
|
private Realm getContextForCurrentThread()
|
|
|
|
{
|
|
|
|
var context = threadContexts.Value;
|
2021-01-12 13:25:22 +08:00
|
|
|
|
2021-01-07 13:07:36 +08:00
|
|
|
if (context?.IsClosed != false)
|
2021-01-12 13:36:35 +08:00
|
|
|
threadContexts.Value = context = createContext();
|
2021-01-07 13:07:36 +08:00
|
|
|
|
2021-01-12 13:25:22 +08:00
|
|
|
contexts_open.Value = threadContexts.Values.Count;
|
|
|
|
|
2021-01-07 13:07:36 +08:00
|
|
|
if (!refreshCompleted.Value)
|
|
|
|
{
|
2021-01-12 13:25:22 +08:00
|
|
|
// to keep things simple, realm refreshes are currently performed per thread context at the point of retrieval.
|
|
|
|
// in the future this should likely be run as part of the update loop for the main (update thread) context.
|
2021-01-07 13:07:36 +08:00
|
|
|
context.Refresh();
|
|
|
|
refreshCompleted.Value = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return context;
|
|
|
|
}
|
|
|
|
|
2021-01-12 13:36:35 +08:00
|
|
|
private Realm createContext()
|
|
|
|
{
|
|
|
|
contexts_created.Value++;
|
|
|
|
|
|
|
|
return Realm.GetInstance(new RealmConfiguration(storage.GetFullPath($"{database_name}.realm", true))
|
|
|
|
{
|
|
|
|
SchemaVersion = schema_version,
|
|
|
|
MigrationCallback = onMigration,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-01-07 13:07:36 +08:00
|
|
|
private void usageCompleted(RealmWriteUsage usage)
|
|
|
|
{
|
|
|
|
int usages = Interlocked.Decrement(ref currentWriteUsages);
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
rollbackRequired |= usage.RollbackRequired;
|
|
|
|
|
|
|
|
if (usages == 0)
|
|
|
|
{
|
|
|
|
if (rollbackRequired)
|
|
|
|
{
|
|
|
|
rollbacks.Value++;
|
|
|
|
currentWriteTransaction?.Rollback();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
commits.Value++;
|
|
|
|
currentWriteTransaction?.Commit();
|
|
|
|
}
|
|
|
|
|
|
|
|
currentWriteTransaction = null;
|
|
|
|
rollbackRequired = false;
|
|
|
|
|
|
|
|
refreshCompleted = new ThreadLocal<bool>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
finally
|
|
|
|
{
|
|
|
|
Monitor.Exit(writeLock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|