// 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.Collections.Generic; using System.IO; using System.Linq; using System.Threading.Tasks; using osu.Framework.Logging; using osu.Framework.Platform; using osu.Game.Collections; using osu.Game.IO.Legacy; using osu.Game.Overlays.Notifications; namespace osu.Game.Database { public class LegacyCollectionImporter { public Action? PostNotification { protected get; set; } private readonly RealmAccess realm; private const string database_name = "collection.db"; public LegacyCollectionImporter(RealmAccess realm) { this.realm = realm; } public Task GetAvailableCount(Storage storage) { if (!storage.Exists(database_name)) return Task.FromResult(0); return Task.Run(() => { using (var stream = storage.GetStream(database_name)) return readCollections(stream).Count; }); } /// /// This is a temporary method and will likely be replaced by a full-fledged (and more correctly placed) migration process in the future. /// public Task ImportFromStorage(Storage storage) { if (!storage.Exists(database_name)) { // This handles situations like when the user does not have a collections.db file Logger.Log($"No {database_name} available in osu!stable installation", LoggingTarget.Information, LogLevel.Error); return Task.CompletedTask; } return Task.Run(async () => { using (var stream = storage.GetStream(database_name)) await Import(stream).ConfigureAwait(false); }); } public async Task Import(Stream stream) { var notification = new ProgressNotification { State = ProgressNotificationState.Active, Text = "Collections import is initialising..." }; PostNotification?.Invoke(notification); var importedCollections = readCollections(stream, notification); await importCollections(importedCollections).ConfigureAwait(false); notification.CompletionText = $"Imported {importedCollections.Count} collections"; notification.State = ProgressNotificationState.Completed; } private Task importCollections(List newCollections) { var tcs = new TaskCompletionSource(); try { realm.Write(r => { foreach (var collection in newCollections) { var existing = r.All().FirstOrDefault(c => c.Name == collection.Name); if (existing != null) { foreach (string newBeatmap in existing.BeatmapMD5Hashes) { if (!existing.BeatmapMD5Hashes.Contains(newBeatmap)) existing.BeatmapMD5Hashes.Add(newBeatmap); } } else r.Add(collection); } }); tcs.SetResult(true); } catch (Exception e) { Logger.Error(e, "Failed to import collection."); tcs.SetException(e); } return tcs.Task; } private List readCollections(Stream stream, ProgressNotification? notification = null) { if (notification != null) { notification.Text = "Reading collections..."; notification.Progress = 0; } var result = new List(); try { using (var sr = new SerializationReader(stream)) { sr.ReadInt32(); // Version int collectionCount = sr.ReadInt32(); result.Capacity = collectionCount; for (int i = 0; i < collectionCount; i++) { if (notification?.CancellationToken.IsCancellationRequested == true) return result; var collection = new BeatmapCollection(sr.ReadString()); int mapCount = sr.ReadInt32(); for (int j = 0; j < mapCount; j++) { if (notification?.CancellationToken.IsCancellationRequested == true) return result; string checksum = sr.ReadString(); collection.BeatmapMD5Hashes.Add(checksum); } if (notification != null) { notification.Text = $"Imported {i + 1} of {collectionCount} collections"; notification.Progress = (float)(i + 1) / collectionCount; } result.Add(collection); } } } catch (Exception e) { Logger.Error(e, "Failed to read collection database."); } return result; } } }