2019-01-24 08:43:03 +00:00
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
2018-04-13 09:19:50 +00:00
using System ;
using System.Collections.Generic ;
using System.IO ;
using System.Linq ;
2018-08-31 09:28:53 +00:00
using System.Threading.Tasks ;
2018-07-18 03:58:28 +00:00
using JetBrains.Annotations ;
2018-04-13 09:19:50 +00:00
using Microsoft.EntityFrameworkCore ;
2018-11-28 10:16:05 +00:00
using osu.Framework.Extensions ;
2018-06-05 02:28:51 +00:00
using osu.Framework.IO.File ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Logging ;
using osu.Framework.Platform ;
using osu.Game.IO ;
using osu.Game.IO.Archives ;
using osu.Game.IPC ;
using osu.Game.Overlays.Notifications ;
using osu.Game.Utils ;
using SharpCompress.Common ;
using FileInfo = osu . Game . IO . FileInfo ;
namespace osu.Game.Database
{
/// <summary>
/// Encapsulates a model store class to give it import functionality.
/// Adds cross-functionality with <see cref="FileStore"/> to give access to the central file store for the provided model.
/// </summary>
/// <typeparam name="TModel">The model type.</typeparam>
/// <typeparam name="TFileModel">The associated file join type.</typeparam>
public abstract class ArchiveModelManager < TModel , TFileModel > : ICanAcceptFiles
where TModel : class , IHasFiles < TFileModel > , IHasPrimaryKey , ISoftDelete
where TFileModel : INamedFileInfo , new ( )
{
2018-11-29 09:07:51 +00:00
public delegate void ItemAddedDelegate ( TModel model , bool existing , bool silent ) ;
2018-11-28 11:19:21 +00:00
2018-04-13 09:19:50 +00:00
/// <summary>
/// Set an endpoint for notifications to be posted to.
/// </summary>
public Action < Notification > PostNotification { protected get ; set ; }
/// <summary>
/// Fired when a new <see cref="TModel"/> becomes available in the database.
/// This is not guaranteed to run on the update thread.
/// </summary>
2018-11-28 11:19:21 +00:00
public event ItemAddedDelegate ItemAdded ;
2018-04-13 09:19:50 +00:00
/// <summary>
/// Fired when a <see cref="TModel"/> is removed from the database.
/// This is not guaranteed to run on the update thread.
/// </summary>
public event Action < TModel > ItemRemoved ;
public virtual string [ ] HandledExtensions = > new [ ] { ".zip" } ;
protected readonly FileStore Files ;
protected readonly IDatabaseContextFactory ContextFactory ;
protected readonly MutableDatabaseBackedStore < TModel > ModelStore ;
// ReSharper disable once NotAccessedField.Local (we should keep a reference to this so it is not finalised)
private ArchiveImportIPCChannel ipc ;
2018-09-09 13:37:15 +00:00
private readonly List < Action > queuedEvents = new List < Action > ( ) ;
2018-05-28 10:56:27 +00:00
2018-05-28 12:45:05 +00:00
/// <summary>
/// Allows delaying of outwards events until an operation is confirmed (at a database level).
/// </summary>
2018-05-28 10:56:27 +00:00
private bool delayingEvents ;
2018-05-28 12:45:05 +00:00
/// <summary>
/// Begin delaying outwards events.
/// </summary>
private void delayEvents ( ) = > delayingEvents = true ;
2018-05-28 10:56:27 +00:00
2018-05-28 12:45:05 +00:00
/// <summary>
/// Flush delayed events and disable delaying.
/// </summary>
/// <param name="perform">Whether the flushed events should be performed.</param>
2018-05-28 10:56:27 +00:00
private void flushEvents ( bool perform )
{
2018-09-09 13:37:15 +00:00
Action [ ] events ;
lock ( queuedEvents )
{
events = queuedEvents . ToArray ( ) ;
queuedEvents . Clear ( ) ;
}
2018-05-28 10:56:27 +00:00
if ( perform )
{
2018-09-09 13:37:15 +00:00
foreach ( var a in events )
2018-05-28 10:56:27 +00:00
a . Invoke ( ) ;
}
delayingEvents = false ;
}
private void handleEvent ( Action a )
{
if ( delayingEvents )
2018-09-21 02:50:36 +00:00
lock ( queuedEvents )
queuedEvents . Add ( a ) ;
2018-05-28 10:56:27 +00:00
else
a . Invoke ( ) ;
}
2018-04-13 09:19:50 +00:00
protected ArchiveModelManager ( Storage storage , IDatabaseContextFactory contextFactory , MutableDatabaseBackedStore < TModel > modelStore , IIpcHost importHost = null )
{
ContextFactory = contextFactory ;
ModelStore = modelStore ;
2018-11-29 09:07:51 +00:00
ModelStore . ItemAdded + = ( item , silent ) = > handleEvent ( ( ) = > ItemAdded ? . Invoke ( item , false , silent ) ) ;
2018-05-28 10:56:27 +00:00
ModelStore . ItemRemoved + = s = > handleEvent ( ( ) = > ItemRemoved ? . Invoke ( s ) ) ;
2018-04-13 09:19:50 +00:00
Files = new FileStore ( contextFactory , storage ) ;
if ( importHost ! = null )
ipc = new ArchiveImportIPCChannel ( importHost , this ) ;
ModelStore . Cleanup ( ) ;
}
/// <summary>
/// Import one or more <see cref="TModel"/> items from filesystem <paramref name="paths"/>.
/// This will post notifications tracking progress.
/// </summary>
/// <param name="paths">One or more archive locations on disk.</param>
public void Import ( params string [ ] paths )
{
var notification = new ProgressNotification
{
Text = "Import is initialising..." ,
Progress = 0 ,
State = ProgressNotificationState . Active ,
} ;
PostNotification ? . Invoke ( notification ) ;
List < TModel > imported = new List < TModel > ( ) ;
int current = 0 ;
foreach ( string path in paths )
{
if ( notification . State = = ProgressNotificationState . Cancelled )
// user requested abort
return ;
try
{
notification . Text = $"Importing ({++current} of {paths.Length})\n{Path.GetFileName(path)}" ;
2018-12-18 19:49:53 +00:00
2019-01-29 09:34:10 +00:00
imported . Add ( Import ( path ) ) ;
2018-04-13 09:19:50 +00:00
notification . Progress = ( float ) current / paths . Length ;
}
catch ( Exception e )
{
e = e . InnerException ? ? e ;
Logger . Error ( e , $@"Could not import ({Path.GetFileName(path)})" ) ;
}
}
2018-09-07 09:14:23 +00:00
if ( imported . Count = = 0 )
{
notification . Text = "Import failed!" ;
notification . State = ProgressNotificationState . Cancelled ;
}
else
{
notification . CompletionText = $"Imported {current} {typeof(TModel).Name.Replace(" Info ", " ").ToLower()}s!" ;
2018-09-07 09:18:03 +00:00
notification . CompletionClickAction + = ( ) = >
{
if ( imported . Count > 0 )
PresentCompletedImport ( imported ) ;
return true ;
} ;
2018-09-07 09:14:23 +00:00
notification . State = ProgressNotificationState . Completed ;
}
2018-04-13 09:19:50 +00:00
}
2019-01-29 09:34:10 +00:00
/// <summary>
/// Import one <see cref="TModel"/> from the filesystem and delete the file on success.
/// </summary>
2019-01-29 14:04:48 +00:00
/// <param name="path">The archive location on disk.</param>
2019-01-29 09:34:10 +00:00
/// <returns>The imported model, if successful.</returns>
public TModel Import ( string path )
{
TModel import ;
using ( ArchiveReader reader = getReaderFrom ( path ) )
import = Import ( reader ) ;
// We may or may not want to delete the file depending on where it is stored.
// e.g. reconstructing/repairing database with items from default storage.
// Also, not always a single file, i.e. for LegacyFilesystemReader
// TODO: Add a check to prevent files from storage to be deleted.
try
{
if ( import ! = null & & File . Exists ( path ) )
File . Delete ( path ) ;
}
catch ( Exception e )
{
Logger . Error ( e , $@"Could not delete original file after import ({Path.GetFileName(path)})" ) ;
}
return import ;
}
2018-09-07 07:30:11 +00:00
protected virtual void PresentCompletedImport ( IEnumerable < TModel > imported )
{
}
2018-04-13 09:19:50 +00:00
/// <summary>
/// Import an item from an <see cref="ArchiveReader"/>.
/// </summary>
/// <param name="archive">The archive to be imported.</param>
public TModel Import ( ArchiveReader archive )
{
2018-07-18 03:58:28 +00:00
try
{
2018-08-24 08:57:39 +00:00
var model = CreateModel ( archive ) ;
2018-11-28 10:16:05 +00:00
if ( model = = null ) return null ;
2018-11-30 06:09:15 +00:00
model . Hash = computeHash ( archive ) ;
2018-11-28 10:16:05 +00:00
2018-11-29 09:07:51 +00:00
return Import ( model , false , archive ) ;
2018-07-18 03:58:28 +00:00
}
catch ( Exception e )
{
Logger . Error ( e , $"Model creation of {archive.Name} failed." , LoggingTarget . Database ) ;
return null ;
}
}
2018-11-28 10:16:05 +00:00
/// <summary>
/// Any file extensions which should be included in hash creation.
/// Generally should include all file types which determine the file's uniqueness.
/// Large files should be avoided if possible.
/// </summary>
protected abstract string [ ] HashableFileTypes { get ; }
/// <summary>
2018-11-30 06:09:15 +00:00
/// Create a SHA-2 hash from the provided archive based on file content of all files matching <see cref="HashableFileTypes"/>.
2018-11-28 10:16:05 +00:00
/// </summary>
2018-11-30 06:09:15 +00:00
private string computeHash ( ArchiveReader reader )
2018-11-28 10:16:05 +00:00
{
// for now, concatenate all .osu files in the set to create a unique hash.
MemoryStream hashable = new MemoryStream ( ) ;
2018-11-30 06:09:15 +00:00
foreach ( string file in reader . Filenames . Where ( f = > HashableFileTypes . Any ( f . EndsWith ) ) )
2018-11-28 10:16:05 +00:00
using ( Stream s = reader . GetStream ( file ) )
s . CopyTo ( hashable ) ;
return hashable . ComputeSHA2Hash ( ) ;
}
2018-07-18 03:58:28 +00:00
/// <summary>
/// Import an item from a <see cref="TModel"/>.
/// </summary>
/// <param name="item">The model to be imported.</param>
2018-11-29 09:07:51 +00:00
/// <param name="silent">Whether the user should be notified fo the import.</param>
2018-07-18 03:58:28 +00:00
/// <param name="archive">An optional archive to use for model population.</param>
2018-11-29 09:07:51 +00:00
public TModel Import ( TModel item , bool silent = false , ArchiveReader archive = null )
2018-07-18 03:58:28 +00:00
{
2018-05-28 12:45:05 +00:00
delayEvents ( ) ;
2018-05-28 10:56:27 +00:00
try
2018-04-13 09:19:50 +00:00
{
2018-08-17 04:50:27 +00:00
Logger . Log ( $"Importing {item}..." , LoggingTarget . Database ) ;
2018-05-28 10:56:27 +00:00
using ( var write = ContextFactory . GetForWrite ( ) ) // used to share a context for full import. keep in mind this will block all writes.
{
2018-05-29 04:48:14 +00:00
try
{
if ( ! write . IsTransactionLeader ) throw new InvalidOperationException ( $"Ensure there is no parent transaction so errors can correctly be handled by {this}" ) ;
2018-04-13 09:19:50 +00:00
2018-05-29 04:48:14 +00:00
var existing = CheckForExisting ( item ) ;
2018-04-13 09:19:50 +00:00
2018-05-29 07:14:09 +00:00
if ( existing ! = null )
{
2018-11-28 10:01:22 +00:00
Undelete ( existing ) ;
2018-07-18 03:58:28 +00:00
Logger . Log ( $"Found existing {typeof(TModel)} for {item} (ID {existing.ID}). Skipping import." , LoggingTarget . Database ) ;
2018-11-29 09:07:51 +00:00
handleEvent ( ( ) = > ItemAdded ? . Invoke ( existing , true , silent ) ) ;
2018-05-29 07:14:09 +00:00
return existing ;
}
2018-04-13 09:19:50 +00:00
2018-07-18 03:58:28 +00:00
if ( archive ! = null )
item . Files = createFileInfos ( archive , Files ) ;
2018-04-13 09:19:50 +00:00
2018-05-29 04:48:14 +00:00
Populate ( item , archive ) ;
2018-04-13 09:19:50 +00:00
2018-05-29 04:48:14 +00:00
// import to store
2018-11-29 09:07:51 +00:00
ModelStore . Add ( item , silent ) ;
2018-05-29 04:48:14 +00:00
}
catch ( Exception e )
{
write . Errors . Add ( e ) ;
throw ;
}
2018-05-28 10:56:27 +00:00
}
2018-05-29 07:14:09 +00:00
2018-07-18 03:58:28 +00:00
Logger . Log ( $"Import of {item} successfully completed!" , LoggingTarget . Database ) ;
2018-04-13 09:19:50 +00:00
}
2018-05-29 09:37:45 +00:00
catch ( Exception e )
2018-05-28 10:56:27 +00:00
{
2018-07-18 03:58:28 +00:00
Logger . Error ( e , $"Import of {item} failed and has been rolled back." , LoggingTarget . Database ) ;
2018-05-28 10:56:27 +00:00
item = null ;
}
2018-05-29 10:43:52 +00:00
finally
{
// we only want to flush events after we've confirmed the write context didn't have any errors.
flushEvents ( item ! = null ) ;
}
2018-05-28 10:56:27 +00:00
return item ;
2018-04-13 09:19:50 +00:00
}
/// <summary>
/// Perform an update of the specified item.
/// TODO: Support file changes.
/// </summary>
/// <param name="item">The item to update.</param>
public void Update ( TModel item ) = > ModelStore . Update ( item ) ;
/// <summary>
/// Delete an item from the manager.
/// Is a no-op for already deleted items.
/// </summary>
/// <param name="item">The item to delete.</param>
2018-09-21 03:21:27 +00:00
/// <returns>false if no operation was performed</returns>
public bool Delete ( TModel item )
2018-04-13 09:19:50 +00:00
{
2018-05-30 04:43:43 +00:00
using ( ContextFactory . GetForWrite ( ) )
2018-04-13 09:19:50 +00:00
{
// re-fetch the model on the import context.
2018-09-21 00:01:04 +00:00
var foundModel = queryModel ( ) . Include ( s = > s . Files ) . ThenInclude ( f = > f . FileInfo ) . FirstOrDefault ( s = > s . ID = = item . ID ) ;
2018-04-13 09:19:50 +00:00
2018-09-21 03:21:27 +00:00
if ( foundModel = = null | | foundModel . DeletePending ) return false ;
2018-04-13 09:19:50 +00:00
if ( ModelStore . Delete ( foundModel ) )
Files . Dereference ( foundModel . Files . Select ( f = > f . FileInfo ) . ToArray ( ) ) ;
2018-09-21 03:21:27 +00:00
return true ;
2018-04-13 09:19:50 +00:00
}
}
/// <summary>
/// Delete multiple items.
/// This will post notifications tracking progress.
/// </summary>
public void Delete ( List < TModel > items )
{
if ( items . Count = = 0 ) return ;
var notification = new ProgressNotification
{
Progress = 0 ,
2018-08-31 09:32:21 +00:00
CompletionText = $"Deleted all {typeof(TModel).Name.Replace(" Info ", " ").ToLower()}s!" ,
2018-04-13 09:19:50 +00:00
State = ProgressNotificationState . Active ,
} ;
PostNotification ? . Invoke ( notification ) ;
int i = 0 ;
using ( ContextFactory . GetForWrite ( ) )
{
foreach ( var b in items )
{
if ( notification . State = = ProgressNotificationState . Cancelled )
// user requested abort
return ;
notification . Text = $"Deleting ({++i} of {items.Count})" ;
Delete ( b ) ;
notification . Progress = ( float ) i / items . Count ;
}
}
notification . State = ProgressNotificationState . Completed ;
}
/// <summary>
/// Restore multiple items that were previously deleted.
/// This will post notifications tracking progress.
/// </summary>
public void Undelete ( List < TModel > items )
{
if ( ! items . Any ( ) ) return ;
var notification = new ProgressNotification
{
CompletionText = "Restored all deleted items!" ,
Progress = 0 ,
State = ProgressNotificationState . Active ,
} ;
PostNotification ? . Invoke ( notification ) ;
int i = 0 ;
using ( ContextFactory . GetForWrite ( ) )
{
foreach ( var item in items )
{
if ( notification . State = = ProgressNotificationState . Cancelled )
// user requested abort
return ;
notification . Text = $"Restoring ({++i} of {items.Count})" ;
Undelete ( item ) ;
notification . Progress = ( float ) i / items . Count ;
}
}
notification . State = ProgressNotificationState . Completed ;
}
/// <summary>
/// Restore an item that was previously deleted. Is a no-op if the item is not in a deleted state, or has its protected flag set.
/// </summary>
/// <param name="item">The item to restore</param>
public void Undelete ( TModel item )
{
using ( var usage = ContextFactory . GetForWrite ( ) )
{
usage . Context . ChangeTracker . AutoDetectChangesEnabled = false ;
if ( ! ModelStore . Undelete ( item ) ) return ;
Files . Reference ( item . Files . Select ( f = > f . FileInfo ) . ToArray ( ) ) ;
usage . Context . ChangeTracker . AutoDetectChangesEnabled = true ;
}
}
/// <summary>
/// Create all required <see cref="FileInfo"/>s for the provided archive, adding them to the global file store.
/// </summary>
private List < TFileModel > createFileInfos ( ArchiveReader reader , FileStore files )
{
var fileInfos = new List < TFileModel > ( ) ;
// import files to manager
foreach ( string file in reader . Filenames )
using ( Stream s = reader . GetStream ( file ) )
fileInfos . Add ( new TFileModel
{
2018-10-07 17:15:42 +00:00
Filename = FileSafety . PathStandardise ( file ) ,
2018-04-13 09:19:50 +00:00
FileInfo = files . Add ( s )
} ) ;
return fileInfos ;
}
2018-08-31 09:28:53 +00:00
#region osu - stable import
/// <summary>
/// Set a storage with access to an osu-stable install for import purposes.
/// </summary>
public Func < Storage > GetStableStorage { private get ; set ; }
/// <summary>
/// Denotes whether an osu-stable installation is present to perform automated imports from.
/// </summary>
public bool StableInstallationAvailable = > GetStableStorage ? . Invoke ( ) ! = null ;
/// <summary>
/// The relative path from osu-stable's data directory to import items from.
/// </summary>
protected virtual string ImportFromStablePath = > null ;
/// <summary>
/// This is a temporary method and will likely be replaced by a full-fledged (and more correctly placed) migration process in the future.
/// </summary>
public Task ImportFromStableAsync ( )
{
var stable = GetStableStorage ? . Invoke ( ) ;
if ( stable = = null )
{
Logger . Log ( "No osu!stable installation available!" , LoggingTarget . Information , LogLevel . Error ) ;
return Task . CompletedTask ;
}
2018-09-18 01:05:28 +00:00
if ( ! stable . ExistsDirectory ( ImportFromStablePath ) )
2018-09-15 13:53:59 +00:00
{
2018-09-18 01:05:28 +00:00
// This handles situations like when the user does not have a Skins folder
2018-09-21 02:50:36 +00:00
Logger . Log ( $"No {ImportFromStablePath} folder available in osu!stable installation" , LoggingTarget . Information , LogLevel . Error ) ;
2018-09-18 01:05:28 +00:00
return Task . CompletedTask ;
}
2018-08-31 09:28:53 +00:00
return Task . Factory . StartNew ( ( ) = > Import ( stable . GetDirectories ( ImportFromStablePath ) . Select ( f = > stable . GetFullPath ( f ) ) . ToArray ( ) ) , TaskCreationOptions . LongRunning ) ;
}
#endregion
2018-04-13 09:19:50 +00:00
/// <summary>
/// Create a barebones model from the provided archive.
/// Actual expensive population should be done in <see cref="Populate"/>; this should just prepare for duplicate checking.
/// </summary>
/// <param name="archive">The archive to create the model for.</param>
2018-08-25 05:51:42 +00:00
/// <returns>A model populated with minimal information. Returning a null will abort importing silently.</returns>
2018-04-13 09:19:50 +00:00
protected abstract TModel CreateModel ( ArchiveReader archive ) ;
/// <summary>
/// Populate the provided model completely from the given archive.
/// After this method, the model should be in a state ready to commit to a store.
/// </summary>
/// <param name="model">The model to populate.</param>
2018-07-18 03:58:28 +00:00
/// <param name="archive">The archive to use as a reference for population. May be null.</param>
protected virtual void Populate ( TModel model , [ CanBeNull ] ArchiveReader archive )
2018-04-13 09:19:50 +00:00
{
}
2018-11-28 10:01:22 +00:00
/// <summary>
/// Check whether an existing model already exists for a new import item.
/// </summary>
/// <param name="model">The new model proposed for import. Note that <see cref="Populate"/> has not yet been run on this model.</param>
/// <returns>An existing model which matches the criteria to skip importing, else null.</returns>
2018-11-30 09:40:06 +00:00
protected virtual TModel CheckForExisting ( TModel model ) = > model . Hash = = null ? null : ModelStore . ConsumableItems . FirstOrDefault ( b = > b . Hash = = model . Hash ) ;
2018-04-13 09:19:50 +00:00
private DbSet < TModel > queryModel ( ) = > ContextFactory . Get ( ) . Set < TModel > ( ) ;
/// <summary>
/// Creates an <see cref="ArchiveReader"/> from a valid storage path.
/// </summary>
/// <param name="path">A file or folder path resolving the archive content.</param>
/// <returns>A reader giving access to the archive's content.</returns>
private ArchiveReader getReaderFrom ( string path )
{
if ( ZipUtils . IsZipArchive ( path ) )
2018-08-15 06:49:55 +00:00
return new ZipArchiveReader ( File . Open ( path , FileMode . Open , FileAccess . Read , FileShare . Read ) , Path . GetFileName ( path ) ) ;
2018-04-13 09:19:50 +00:00
if ( Directory . Exists ( path ) )
2018-11-28 06:13:27 +00:00
return new LegacyDirectoryArchiveReader ( path ) ;
2018-11-28 07:13:16 +00:00
if ( File . Exists ( path ) )
return new LegacyFileArchiveReader ( path ) ;
2018-04-13 09:19:50 +00:00
throw new InvalidFormatException ( $"{path} is not a valid archive" ) ;
}
}
}