osu/osu.Game.Tests/Database/RealmLiveTests.cs

297 lines
11 KiB
C#
Raw Normal View History

2021-09-30 14:46:16 +00: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;
using System.Diagnostics;
using System.Linq;
using System.Threading.Tasks;
using NUnit.Framework;
using osu.Framework.Extensions;
using osu.Framework.Testing;
using osu.Game.Beatmaps;
2021-09-30 14:46:16 +00:00
using osu.Game.Database;
using Realms;
#nullable enable
namespace osu.Game.Tests.Database
{
public class RealmLiveTests : RealmTest
{
2021-10-01 05:30:27 +00:00
[Test]
2021-11-26 05:38:39 +00:00
public void TestLiveEquality()
2021-10-01 05:30:27 +00:00
{
RunTestWithRealm((realm, _) =>
2021-10-01 05:30:27 +00:00
{
ILive<BeatmapInfo> beatmap = realm.Run(r => r.Write(_ => r.Add(new BeatmapInfo(CreateRuleset(), new BeatmapDifficulty(), new BeatmapMetadata()))).ToLive(realm));
2021-10-01 05:30:27 +00:00
ILive<BeatmapInfo> beatmap2 = realm.Run(r => r.All<BeatmapInfo>().First().ToLive(realm));
2021-10-01 05:30:27 +00:00
2021-11-26 05:38:39 +00:00
Assert.AreEqual(beatmap, beatmap2);
2021-10-01 05:30:27 +00:00
});
}
[Test]
public void TestAccessAfterStorageMigrate()
{
RunTestWithRealm((realm, storage) =>
{
var beatmap = new BeatmapInfo(CreateRuleset(), new BeatmapDifficulty(), new BeatmapMetadata());
ILive<BeatmapInfo>? liveBeatmap = null;
realm.Run(r =>
{
r.Write(_ => r.Add(beatmap));
liveBeatmap = beatmap.ToLive(realm);
});
using (realm.BlockAllOperations())
{
// recycle realm before migrating
}
using (var migratedStorage = new TemporaryNativeStorage("realm-test-migration-target"))
{
migratedStorage.DeleteDirectory(string.Empty);
storage.Migrate(migratedStorage);
Assert.IsFalse(liveBeatmap?.PerformRead(l => l.Hidden));
}
});
}
[Test]
public void TestAccessAfterAttach()
{
RunTestWithRealm((realm, _) =>
{
var beatmap = new BeatmapInfo(CreateRuleset(), new BeatmapDifficulty(), new BeatmapMetadata());
var liveBeatmap = beatmap.ToLive(realm);
realm.Run(r => r.Write(_ => r.Add(beatmap)));
Assert.IsFalse(liveBeatmap.PerformRead(l => l.Hidden));
});
}
[Test]
public void TestAccessNonManaged()
{
var beatmap = new BeatmapInfo(CreateRuleset(), new BeatmapDifficulty(), new BeatmapMetadata());
var liveBeatmap = beatmap.ToLiveUnmanaged();
Assert.IsFalse(beatmap.Hidden);
Assert.IsFalse(liveBeatmap.Value.Hidden);
Assert.IsFalse(liveBeatmap.PerformRead(l => l.Hidden));
Assert.Throws<InvalidOperationException>(() => liveBeatmap.PerformWrite(l => l.Hidden = true));
Assert.IsFalse(beatmap.Hidden);
Assert.IsFalse(liveBeatmap.Value.Hidden);
Assert.IsFalse(liveBeatmap.PerformRead(l => l.Hidden));
}
2021-09-30 14:46:16 +00:00
[Test]
2021-11-30 02:55:13 +00:00
public void TestScopedReadWithoutContext()
2021-09-30 14:46:16 +00:00
{
RunTestWithRealm((realm, _) =>
2021-09-30 14:46:16 +00:00
{
ILive<BeatmapInfo>? liveBeatmap = null;
2021-09-30 14:46:16 +00:00
Task.Factory.StartNew(() =>
{
realm.Run(threadContext =>
2021-09-30 14:46:16 +00:00
{
var beatmap = threadContext.Write(r => r.Add(new BeatmapInfo(CreateRuleset(), new BeatmapDifficulty(), new BeatmapMetadata())));
2021-09-30 14:46:16 +00:00
liveBeatmap = beatmap.ToLive(realm);
});
}, TaskCreationOptions.LongRunning | TaskCreationOptions.HideScheduler).WaitSafely();
2021-09-30 14:46:16 +00:00
Debug.Assert(liveBeatmap != null);
Task.Factory.StartNew(() =>
{
2021-11-30 02:55:13 +00:00
liveBeatmap.PerformRead(beatmap =>
2021-09-30 14:46:16 +00:00
{
2021-11-30 02:55:13 +00:00
Assert.IsTrue(beatmap.IsValid);
Assert.IsFalse(beatmap.Hidden);
});
}, TaskCreationOptions.LongRunning | TaskCreationOptions.HideScheduler).WaitSafely();
2021-09-30 14:46:16 +00:00
});
}
[Test]
2021-11-30 02:55:13 +00:00
public void TestScopedWriteWithoutContext()
2021-09-30 14:46:16 +00:00
{
RunTestWithRealm((realm, _) =>
2021-09-30 14:46:16 +00:00
{
ILive<BeatmapInfo>? liveBeatmap = null;
2021-09-30 14:46:16 +00:00
Task.Factory.StartNew(() =>
{
realm.Run(threadContext =>
2021-09-30 14:46:16 +00:00
{
var beatmap = threadContext.Write(r => r.Add(new BeatmapInfo(CreateRuleset(), new BeatmapDifficulty(), new BeatmapMetadata())));
2021-09-30 14:46:16 +00:00
liveBeatmap = beatmap.ToLive(realm);
});
}, TaskCreationOptions.LongRunning | TaskCreationOptions.HideScheduler).WaitSafely();
2021-09-30 14:46:16 +00:00
Debug.Assert(liveBeatmap != null);
Task.Factory.StartNew(() =>
{
2021-11-30 02:55:13 +00:00
liveBeatmap.PerformWrite(beatmap => { beatmap.Hidden = true; });
liveBeatmap.PerformRead(beatmap => { Assert.IsTrue(beatmap.Hidden); });
}, TaskCreationOptions.LongRunning | TaskCreationOptions.HideScheduler).WaitSafely();
2021-09-30 14:46:16 +00:00
});
}
[Test]
2021-11-30 02:55:13 +00:00
public void TestValueAccessNonManaged()
{
RunTestWithRealm((realm, _) =>
2021-11-30 02:55:13 +00:00
{
var beatmap = new BeatmapInfo(CreateRuleset(), new BeatmapDifficulty(), new BeatmapMetadata());
var liveBeatmap = beatmap.ToLive(realm);
2021-11-30 02:55:13 +00:00
Assert.DoesNotThrow(() =>
{
var __ = liveBeatmap.Value;
});
});
}
[Test]
public void TestValueAccessWithOpenContextFails()
2021-09-30 14:46:16 +00:00
{
RunTestWithRealm((realm, _) =>
2021-09-30 14:46:16 +00:00
{
ILive<BeatmapInfo>? liveBeatmap = null;
2021-09-30 14:46:16 +00:00
Task.Factory.StartNew(() =>
{
realm.Run(threadContext =>
2021-09-30 14:46:16 +00:00
{
var beatmap = threadContext.Write(r => r.Add(new BeatmapInfo(CreateRuleset(), new BeatmapDifficulty(), new BeatmapMetadata())));
2021-09-30 14:46:16 +00:00
liveBeatmap = beatmap.ToLive(realm);
});
}, TaskCreationOptions.LongRunning | TaskCreationOptions.HideScheduler).WaitSafely();
2021-09-30 14:46:16 +00:00
Debug.Assert(liveBeatmap != null);
Task.Factory.StartNew(() =>
{
2021-11-30 02:55:13 +00:00
// Can't be used, without a valid context.
Assert.Throws<InvalidOperationException>(() =>
{
var __ = liveBeatmap.Value;
});
// Can't be used, even from within a valid context.
realm.Run(threadContext =>
2021-11-30 02:55:13 +00:00
{
Assert.Throws<InvalidOperationException>(() =>
{
var __ = liveBeatmap.Value;
});
});
}, TaskCreationOptions.LongRunning | TaskCreationOptions.HideScheduler).WaitSafely();
2021-09-30 14:46:16 +00:00
});
}
[Test]
public void TestValueAccessWithoutOpenContextFails()
{
RunTestWithRealm((realm, _) =>
2021-09-30 14:46:16 +00:00
{
ILive<BeatmapInfo>? liveBeatmap = null;
2021-09-30 14:46:16 +00:00
Task.Factory.StartNew(() =>
{
realm.Run(threadContext =>
2021-09-30 14:46:16 +00:00
{
var beatmap = threadContext.Write(r => r.Add(new BeatmapInfo(CreateRuleset(), new BeatmapDifficulty(), new BeatmapMetadata())));
2021-09-30 14:46:16 +00:00
liveBeatmap = beatmap.ToLive(realm);
});
}, TaskCreationOptions.LongRunning | TaskCreationOptions.HideScheduler).WaitSafely();
2021-09-30 14:46:16 +00:00
Debug.Assert(liveBeatmap != null);
Task.Factory.StartNew(() =>
{
Assert.Throws<InvalidOperationException>(() =>
{
var unused = liveBeatmap.Value;
});
}, TaskCreationOptions.LongRunning | TaskCreationOptions.HideScheduler).WaitSafely();
2021-09-30 14:46:16 +00:00
});
}
[Test]
public void TestLiveAssumptions()
{
RunTestWithRealm((realm, _) =>
2021-09-30 14:46:16 +00:00
{
int changesTriggered = 0;
realm.RegisterCustomSubscription(outerRealm =>
2021-09-30 14:46:16 +00:00
{
outerRealm.All<BeatmapInfo>().QueryAsyncWithNotifications(gotChange);
ILive<BeatmapInfo>? liveBeatmap = null;
2021-09-30 14:46:16 +00:00
Task.Factory.StartNew(() =>
{
realm.Run(innerRealm =>
2021-09-30 14:46:16 +00:00
{
var ruleset = CreateRuleset();
var beatmap = innerRealm.Write(r => r.Add(new BeatmapInfo(ruleset, new BeatmapDifficulty(), new BeatmapMetadata())));
2021-09-30 14:46:16 +00:00
// add a second beatmap to ensure that a full refresh occurs below.
// not just a refresh from the resolved Live.
innerRealm.Write(r => r.Add(new BeatmapInfo(ruleset, new BeatmapDifficulty(), new BeatmapMetadata())));
2021-09-30 14:46:16 +00:00
liveBeatmap = beatmap.ToLive(realm);
});
}, TaskCreationOptions.LongRunning | TaskCreationOptions.HideScheduler).WaitSafely();
2021-09-30 14:46:16 +00:00
Debug.Assert(liveBeatmap != null);
// not yet seen by main context
Assert.AreEqual(0, outerRealm.All<BeatmapInfo>().Count());
2021-09-30 14:46:16 +00:00
Assert.AreEqual(0, changesTriggered);
liveBeatmap.PerformRead(resolved =>
{
// retrieval causes an implicit refresh. even changes that aren't related to the retrieval are fired at this point.
// ReSharper disable once AccessToDisposedClosure
Assert.AreEqual(2, outerRealm.All<BeatmapInfo>().Count());
Assert.AreEqual(1, changesTriggered);
2021-09-30 14:46:16 +00:00
// can access properties without a crash.
Assert.IsFalse(resolved.Hidden);
2021-09-30 14:46:16 +00:00
// ReSharper disable once AccessToDisposedClosure
outerRealm.Write(r =>
{
// can use with the main context.
r.Remove(resolved);
});
2021-09-30 14:46:16 +00:00
});
return null;
});
2021-09-30 14:46:16 +00:00
void gotChange(IRealmCollection<BeatmapInfo> sender, ChangeSet changes, Exception error)
2021-09-30 14:46:16 +00:00
{
changesTriggered++;
}
});
}
}
}