// Copyright (c) 2007-2017 ppy Pty Ltd . // Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE using System; using System.IO; using System.Linq; using Microsoft.EntityFrameworkCore; using osu.Framework.Extensions; using osu.Framework.IO.Stores; using osu.Framework.Logging; using osu.Framework.Platform; using osu.Game.Database; namespace osu.Game.IO { /// /// Handles the Store and retrieval of Files/FileSets to the database backing /// public class FileStore : DatabaseBackedStore { private const string prefix = "files"; public readonly ResourceStore Store; public FileStore(OsuDbContext connection, Storage storage) : base(connection, storage) { Store = new NamespacedResourceStore(new StorageBackedResourceStore(storage), prefix); } protected override Type[] ValidTypes => new[] { typeof(FileInfo), }; protected override void Prepare(bool reset = false) { if (reset) { // in earlier versions we stored beatmaps as solid archives, but not any more. if (Storage.ExistsDirectory("beatmaps")) Storage.DeleteDirectory("beatmaps"); if (Storage.ExistsDirectory(prefix)) Storage.DeleteDirectory(prefix); Connection.Database.ExecuteSqlCommand("DELETE FROM FileInfo"); } } protected override void StartupTasks() { base.StartupTasks(); deletePending(); } public FileInfo Add(Stream data, bool reference = true) { string hash = data.ComputeSHA2Hash(); var existing = Connection.FileInfo.FirstOrDefault(f => f.Hash == hash); var info = existing ?? new FileInfo { Hash = hash }; string path = Path.Combine(prefix, info.StoragePath); // we may be re-adding a file to fix missing store entries. if (!Storage.Exists(path)) { data.Seek(0, SeekOrigin.Begin); using (var output = Storage.GetStream(path, FileAccess.Write)) data.CopyTo(output); data.Seek(0, SeekOrigin.Begin); } if (existing == null) Connection.FileInfo.Add(info); if (reference || existing == null) Reference(info); Connection.SaveChanges(); return info; } public void Reference(params FileInfo[] files) { var incrementedFiles = files.GroupBy(f => f.Id).Select(f => { var accurateRefCount = Connection.Find(f.First().Id); accurateRefCount.ReferenceCount += f.Count(); return accurateRefCount; }); //Connection.FileInfo.UpdateRange(incrementedFiles); Connection.SaveChanges(); } public void Dereference(params FileInfo[] files) { var incrementedFiles = files.GroupBy(f => f.Id).Select(f => { var accurateRefCount = Connection.Find(f.First().Id); accurateRefCount.ReferenceCount -= f.Count(); return accurateRefCount; }); //Connection.FileInfo.UpdateRange(incrementedFiles); Connection.SaveChanges(); } private void deletePending() { foreach (var f in Connection.FileInfo.Where(f => f.ReferenceCount < 1)) { try { Storage.Delete(Path.Combine(prefix, f.StoragePath)); Connection.FileInfo.Remove(f); } catch (Exception e) { Logger.Error(e, $@"Could not delete beatmap {f}"); } } } } }