1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-28 07:23:14 +08:00

Add support for starting with a fresh realm database if the existing one is not usable

The most common scenario is switching between schema versions when
testing. This should alleviate the manual overhead of that for the
majority of cases.

For users, this will show a notification on startup if their database
was purged, similar to what we had with EF.
This commit is contained in:
Dean Herbert 2022-01-18 16:05:12 +09:00
parent 3429fd8768
commit 246a4a4bfe

View File

@ -105,8 +105,20 @@ namespace osu.Game.Database
if (!Filename.EndsWith(realm_extension, StringComparison.Ordinal)) if (!Filename.EndsWith(realm_extension, StringComparison.Ordinal))
Filename += realm_extension; Filename += realm_extension;
// This method triggers the first `CreateContext` call, which will implicitly run realm migrations and bring the schema up-to-date. try
cleanupPendingDeletions(); {
// This method triggers the first `CreateContext` call, which will implicitly run realm migrations and bring the schema up-to-date.
cleanupPendingDeletions();
}
catch (Exception e)
{
Logger.Error(e, "Realm startup failed with unrecoverable error; starting with a fresh database. A backup of your database has been made.");
CreateBackup($"{Filename.Replace(realm_extension, string.Empty)}_{DateTimeOffset.UtcNow.ToUnixTimeSeconds()}_corrupt{realm_extension}");
storage.Delete(Filename);
cleanupPendingDeletions();
}
} }
private void cleanupPendingDeletions() private void cleanupPendingDeletions()
@ -396,14 +408,23 @@ namespace osu.Game.Database
const int sleep_length = 200; const int sleep_length = 200;
int timeout = 5000; int timeout = 5000;
// see https://github.com/realm/realm-dotnet/discussions/2657 try
while (!Compact())
{ {
Thread.Sleep(sleep_length); // see https://github.com/realm/realm-dotnet/discussions/2657
timeout -= sleep_length; while (!Compact())
{
Thread.Sleep(sleep_length);
timeout -= sleep_length;
if (timeout < 0) if (timeout < 0)
throw new TimeoutException(@"Took too long to acquire lock"); throw new TimeoutException(@"Took too long to acquire lock");
}
}
catch (Exception e)
{
// Compact may fail if the realm is in a bad state.
// We still want to continue with the blocking operation, though.
Logger.Log($"Realm compact failed with error {e}", LoggingTarget.Database);
} }
} }
catch catch