1
0
mirror of https://github.com/ppy/osu.git synced 2024-09-24 11:27:23 +08:00
osu-lazer/osu.Game/IO/FileStore.cs

119 lines
3.6 KiB
C#
Raw Normal View History

// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using System;
using System.IO;
using System.Linq;
2017-10-05 03:52:12 +08:00
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
{
/// <summary>
/// Handles the Store and retrieval of Files/FileSets to the database backing
/// </summary>
public class FileStore : DatabaseBackedStore
{
public readonly IResourceStore<byte[]> Store;
public Storage Storage => base.Storage;
public FileStore(Func<OsuDbContext> getContext, Storage storage) : base(getContext, storage.GetStorageForDirectory(@"files"))
{
Store = new StorageBackedResourceStore(Storage);
}
protected override void Prepare(bool reset = false)
{
if (reset)
{
if (Storage.ExistsDirectory(string.Empty))
Storage.DeleteDirectory(string.Empty);
2017-10-17 14:50:42 +08:00
GetContext().Database.ExecuteSqlCommand("DELETE FROM FileInfo");
}
}
public FileInfo Add(Stream data, bool reference = true)
{
2017-10-17 14:50:42 +08:00
var context = GetContext();
string hash = data.ComputeSHA2Hash();
2017-10-17 14:50:42 +08:00
var existing = context.FileInfo.FirstOrDefault(f => f.Hash == hash);
var info = existing ?? new FileInfo { Hash = hash };
string path = 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 (reference || existing == null)
Reference(info);
return info;
}
2017-10-17 14:50:42 +08:00
public void Reference(params FileInfo[] files) => reference(GetContext(), files);
private void reference(OsuDbContext context, FileInfo[] files)
{
2017-10-16 12:09:10 +08:00
foreach (var f in files.GroupBy(f => f.ID))
{
2017-10-17 14:50:42 +08:00
var refetch = context.Find<FileInfo>(f.First().ID) ?? f.First();
2017-10-16 12:09:10 +08:00
refetch.ReferenceCount += f.Count();
2017-10-17 14:50:42 +08:00
context.FileInfo.Update(refetch);
2017-10-16 12:09:10 +08:00
}
2017-10-17 14:50:42 +08:00
context.SaveChanges();
}
2017-10-17 14:50:42 +08:00
public void Dereference(params FileInfo[] files) => dereference(GetContext(), files);
private void dereference(OsuDbContext context, FileInfo[] files)
{
2017-10-16 12:09:10 +08:00
foreach (var f in files.GroupBy(f => f.ID))
{
var refetch = context.FileInfo.Find(f.Key);
2017-10-17 09:25:18 +08:00
refetch.ReferenceCount -= f.Count();
context.FileInfo.Update(refetch);
2017-10-16 12:09:10 +08:00
}
2017-10-05 03:52:12 +08:00
2017-10-17 14:50:42 +08:00
context.SaveChanges();
}
public override void Cleanup()
{
2017-10-17 14:50:42 +08:00
var context = GetContext();
foreach (var f in context.FileInfo.Where(f => f.ReferenceCount < 1))
{
2017-10-05 03:52:12 +08:00
try
{
Storage.Delete(f.StoragePath);
2017-10-17 14:50:42 +08:00
context.FileInfo.Remove(f);
}
2017-10-05 03:52:12 +08:00
catch (Exception e)
{
Logger.Error(e, $@"Could not delete beatmap {f}");
}
}
2017-10-17 09:25:18 +08:00
2017-10-17 14:50:42 +08:00
context.SaveChanges();
}
}
2017-10-05 03:52:12 +08:00
}