1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-26 16:12:54 +08:00

Merge pull request #16520 from peppy/refresh-after-migrate

Improve realm migration performance further
This commit is contained in:
Dan Balasescu 2022-01-20 14:38:23 +09:00 committed by GitHub
commit 64fe7d8dd3
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -38,7 +38,7 @@ namespace osu.Game.Database
public void Run() public void Run()
{ {
using (var ef = efContextFactory.GetForWrite()) using (var ef = efContextFactory.Get())
{ {
migrateSettings(ef); migrateSettings(ef);
migrateSkins(ef); migrateSkins(ef);
@ -46,16 +46,19 @@ namespace osu.Game.Database
migrateScores(ef); migrateScores(ef);
} }
Logger.Log("Refreshing realm...", LoggingTarget.Database);
realmContextFactory.Refresh();
// Delete the database permanently. // Delete the database permanently.
// Will cause future startups to not attempt migration. // Will cause future startups to not attempt migration.
Logger.Log("Migration successful, deleting EF database", LoggingTarget.Database); Logger.Log("Migration successful, deleting EF database", LoggingTarget.Database);
efContextFactory.ResetDatabase(); efContextFactory.ResetDatabase();
} }
private void migrateBeatmaps(DatabaseWriteUsage ef) private void migrateBeatmaps(OsuDbContext ef)
{ {
// can be removed 20220730. // can be removed 20220730.
var existingBeatmapSets = ef.Context.EFBeatmapSetInfo var existingBeatmapSets = ef.EFBeatmapSetInfo
.Include(s => s.Beatmaps).ThenInclude(b => b.RulesetInfo) .Include(s => s.Beatmaps).ThenInclude(b => b.RulesetInfo)
.Include(s => s.Beatmaps).ThenInclude(b => b.Metadata) .Include(s => s.Beatmaps).ThenInclude(b => b.Metadata)
.Include(s => s.Beatmaps).ThenInclude(b => b.BaseDifficulty) .Include(s => s.Beatmaps).ThenInclude(b => b.BaseDifficulty)
@ -87,10 +90,20 @@ namespace osu.Game.Database
} }
else else
{ {
using (var transaction = realm.BeginWrite()) var transaction = realm.BeginWrite();
int written = 0;
try
{ {
foreach (var beatmapSet in existingBeatmapSets) foreach (var beatmapSet in existingBeatmapSets)
{ {
if (++written % 1000 == 0)
{
transaction.Commit();
transaction = realm.BeginWrite();
Logger.Log($"Migrated {written}/{count} beatmaps...", LoggingTarget.Database);
}
var realmBeatmapSet = new BeatmapSetInfo var realmBeatmapSet = new BeatmapSetInfo
{ {
OnlineID = beatmapSet.OnlineID ?? -1, OnlineID = beatmapSet.OnlineID ?? -1,
@ -105,7 +118,10 @@ namespace osu.Game.Database
foreach (var beatmap in beatmapSet.Beatmaps) foreach (var beatmap in beatmapSet.Beatmaps)
{ {
var realmBeatmap = new BeatmapInfo var ruleset = realm.Find<RulesetInfo>(beatmap.RulesetInfo.ShortName);
var metadata = getBestMetadata(beatmap.Metadata, beatmapSet.Metadata);
var realmBeatmap = new BeatmapInfo(ruleset, new BeatmapDifficulty(beatmap.BaseDifficulty), metadata)
{ {
DifficultyName = beatmap.DifficultyName, DifficultyName = beatmap.DifficultyName,
Status = beatmap.Status, Status = beatmap.Status,
@ -131,9 +147,6 @@ namespace osu.Game.Database
CountdownOffset = beatmap.CountdownOffset, CountdownOffset = beatmap.CountdownOffset,
MaxCombo = beatmap.MaxCombo, MaxCombo = beatmap.MaxCombo,
Bookmarks = beatmap.Bookmarks, Bookmarks = beatmap.Bookmarks,
Ruleset = realm.Find<RulesetInfo>(beatmap.RulesetInfo.ShortName),
Difficulty = new BeatmapDifficulty(beatmap.BaseDifficulty),
Metadata = getBestMetadata(beatmap.Metadata, beatmapSet.Metadata),
BeatmapSet = realmBeatmapSet, BeatmapSet = realmBeatmapSet,
}; };
@ -142,10 +155,13 @@ namespace osu.Game.Database
realm.Add(realmBeatmapSet); realm.Add(realmBeatmapSet);
} }
transaction.Commit();
Logger.Log($"Successfully migrated {count} beatmaps to realm", LoggingTarget.Database);
} }
finally
{
transaction.Commit();
}
Logger.Log($"Successfully migrated {count} beatmaps to realm", LoggingTarget.Database);
} }
} }
} }
@ -160,7 +176,7 @@ namespace osu.Game.Database
TitleUnicode = metadata.TitleUnicode, TitleUnicode = metadata.TitleUnicode,
Artist = metadata.Artist, Artist = metadata.Artist,
ArtistUnicode = metadata.ArtistUnicode, ArtistUnicode = metadata.ArtistUnicode,
Author = new RealmUser Author =
{ {
OnlineID = metadata.Author.Id, OnlineID = metadata.Author.Id,
Username = metadata.Author.Username, Username = metadata.Author.Username,
@ -173,10 +189,10 @@ namespace osu.Game.Database
}; };
} }
private void migrateScores(DatabaseWriteUsage db) private void migrateScores(OsuDbContext db)
{ {
// can be removed 20220730. // can be removed 20220730.
var existingScores = db.Context.ScoreInfo var existingScores = db.ScoreInfo
.Include(s => s.Ruleset) .Include(s => s.Ruleset)
.Include(s => s.BeatmapInfo) .Include(s => s.BeatmapInfo)
.Include(s => s.Files) .Include(s => s.Files)
@ -206,26 +222,41 @@ namespace osu.Game.Database
} }
else else
{ {
using (var transaction = realm.BeginWrite()) var transaction = realm.BeginWrite();
int written = 0;
try
{ {
foreach (var score in existingScores) foreach (var score in existingScores)
{ {
var realmScore = new ScoreInfo if (++written % 1000 == 0)
{
transaction.Commit();
transaction = realm.BeginWrite();
Logger.Log($"Migrated {written}/{count} scores...", LoggingTarget.Database);
}
var beatmap = realm.All<BeatmapInfo>().First(b => b.Hash == score.BeatmapInfo.Hash);
var ruleset = realm.Find<RulesetInfo>(score.Ruleset.ShortName);
var user = new RealmUser
{
OnlineID = score.User.OnlineID,
Username = score.User.Username
};
var realmScore = new ScoreInfo(beatmap, ruleset, user)
{ {
Hash = score.Hash, Hash = score.Hash,
DeletePending = score.DeletePending, DeletePending = score.DeletePending,
OnlineID = score.OnlineID ?? -1, OnlineID = score.OnlineID ?? -1,
ModsJson = score.ModsJson, ModsJson = score.ModsJson,
StatisticsJson = score.StatisticsJson, StatisticsJson = score.StatisticsJson,
User = score.User,
TotalScore = score.TotalScore, TotalScore = score.TotalScore,
MaxCombo = score.MaxCombo, MaxCombo = score.MaxCombo,
Accuracy = score.Accuracy, Accuracy = score.Accuracy,
HasReplay = ((IScoreInfo)score).HasReplay, HasReplay = ((IScoreInfo)score).HasReplay,
Date = score.Date, Date = score.Date,
PP = score.PP, PP = score.PP,
BeatmapInfo = realm.All<BeatmapInfo>().First(b => b.Hash == score.BeatmapInfo.Hash),
Ruleset = realm.Find<RulesetInfo>(score.Ruleset.ShortName),
Rank = score.Rank, Rank = score.Rank,
HitEvents = score.HitEvents, HitEvents = score.HitEvents,
Passed = score.Passed, Passed = score.Passed,
@ -240,18 +271,21 @@ namespace osu.Game.Database
realm.Add(realmScore); realm.Add(realmScore);
} }
transaction.Commit();
Logger.Log($"Successfully migrated {count} scores to realm", LoggingTarget.Database);
} }
finally
{
transaction.Commit();
}
Logger.Log($"Successfully migrated {count} scores to realm", LoggingTarget.Database);
} }
} }
} }
private void migrateSkins(DatabaseWriteUsage db) private void migrateSkins(OsuDbContext db)
{ {
// can be removed 20220530. // can be removed 20220530.
var existingSkins = db.Context.SkinInfo var existingSkins = db.SkinInfo
.Include(s => s.Files) .Include(s => s.Files)
.ThenInclude(f => f.FileInfo) .ThenInclude(f => f.FileInfo)
.ToList(); .ToList();
@ -322,10 +356,10 @@ namespace osu.Game.Database
} }
} }
private void migrateSettings(DatabaseWriteUsage db) private void migrateSettings(OsuDbContext db)
{ {
// migrate ruleset settings. can be removed 20220315. // migrate ruleset settings. can be removed 20220315.
var existingSettings = db.Context.DatabasedSetting.ToList(); var existingSettings = db.DatabasedSetting.ToList();
// previous entries in EF are removed post migration. // previous entries in EF are removed post migration.
if (!existingSettings.Any()) if (!existingSettings.Any())