2020-06-16 15:39:20 +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.
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Threading;
|
|
|
|
using osu.Framework.Platform;
|
|
|
|
|
|
|
|
namespace osu.Game.IO
|
|
|
|
{
|
2020-06-22 10:59:38 +00:00
|
|
|
/// <summary>
|
2020-06-22 11:03:24 +00:00
|
|
|
/// A <see cref="WrappedStorage"/> that is migratable to different locations.
|
2020-06-22 10:59:38 +00:00
|
|
|
/// </summary>
|
2020-06-16 15:39:20 +00:00
|
|
|
public abstract class MigratableStorage : WrappedStorage
|
|
|
|
{
|
2020-10-19 06:36:27 +00:00
|
|
|
/// <summary>
|
|
|
|
/// A relative list of directory paths which should not be migrated.
|
|
|
|
/// </summary>
|
2020-06-24 21:01:56 +00:00
|
|
|
public virtual string[] IgnoreDirectories => Array.Empty<string>();
|
2020-10-19 06:36:27 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// A relative list of file paths which should not be migrated.
|
|
|
|
/// </summary>
|
2020-06-24 21:01:56 +00:00
|
|
|
public virtual string[] IgnoreFiles => Array.Empty<string>();
|
2020-06-23 21:58:28 +00:00
|
|
|
|
2020-06-22 10:43:01 +00:00
|
|
|
protected MigratableStorage(Storage storage, string subPath = null)
|
2020-06-16 15:39:20 +00:00
|
|
|
: base(storage, subPath)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-07-01 20:57:16 +00:00
|
|
|
/// <summary>
|
|
|
|
/// A general purpose migration method to move the storage to a different location.
|
|
|
|
/// <param name="newStorage">The target storage of the migration.</param>
|
|
|
|
/// </summary>
|
|
|
|
public virtual void Migrate(Storage newStorage)
|
|
|
|
{
|
|
|
|
var source = new DirectoryInfo(GetFullPath("."));
|
|
|
|
var destination = new DirectoryInfo(newStorage.GetFullPath("."));
|
|
|
|
|
|
|
|
// using Uri is the easiest way to check equality and contains (https://stackoverflow.com/a/7710620)
|
|
|
|
var sourceUri = new Uri(source.FullName + Path.DirectorySeparatorChar);
|
|
|
|
var destinationUri = new Uri(destination.FullName + Path.DirectorySeparatorChar);
|
|
|
|
|
|
|
|
if (sourceUri == destinationUri)
|
2020-07-01 22:32:09 +00:00
|
|
|
throw new ArgumentException("Destination provided is already the current location", destination.FullName);
|
2020-07-01 20:57:16 +00:00
|
|
|
|
|
|
|
if (sourceUri.IsBaseOf(destinationUri))
|
2020-07-01 22:32:09 +00:00
|
|
|
throw new ArgumentException("Destination provided is inside the source", destination.FullName);
|
2020-07-01 20:57:16 +00:00
|
|
|
|
|
|
|
// ensure the new location has no files present, else hard abort
|
|
|
|
if (destination.Exists)
|
|
|
|
{
|
|
|
|
if (destination.GetFiles().Length > 0 || destination.GetDirectories().Length > 0)
|
2020-07-01 22:32:09 +00:00
|
|
|
throw new ArgumentException("Destination provided already has files or directories present", destination.FullName);
|
2020-07-01 20:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CopyRecursive(source, destination);
|
|
|
|
ChangeTargetStorage(newStorage);
|
|
|
|
DeleteRecursive(source);
|
|
|
|
}
|
2020-06-23 21:58:28 +00:00
|
|
|
|
2020-06-22 10:43:01 +00:00
|
|
|
protected void DeleteRecursive(DirectoryInfo target, bool topLevelExcludes = true)
|
2020-06-16 15:39:20 +00:00
|
|
|
{
|
|
|
|
foreach (System.IO.FileInfo fi in target.GetFiles())
|
|
|
|
{
|
2020-06-23 22:37:29 +00:00
|
|
|
if (topLevelExcludes && IgnoreFiles.Contains(fi.Name))
|
2020-06-16 15:39:20 +00:00
|
|
|
continue;
|
|
|
|
|
2020-06-22 10:43:01 +00:00
|
|
|
AttemptOperation(() => fi.Delete());
|
2020-06-16 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach (DirectoryInfo dir in target.GetDirectories())
|
|
|
|
{
|
2020-06-23 22:37:29 +00:00
|
|
|
if (topLevelExcludes && IgnoreDirectories.Contains(dir.Name))
|
2020-06-16 15:39:20 +00:00
|
|
|
continue;
|
|
|
|
|
2020-06-22 10:43:01 +00:00
|
|
|
AttemptOperation(() => dir.Delete(true));
|
2020-06-16 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (target.GetFiles().Length == 0 && target.GetDirectories().Length == 0)
|
2020-06-22 10:43:01 +00:00
|
|
|
AttemptOperation(target.Delete);
|
2020-06-16 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
2020-06-22 10:43:01 +00:00
|
|
|
protected void CopyRecursive(DirectoryInfo source, DirectoryInfo destination, bool topLevelExcludes = true)
|
2020-06-16 15:39:20 +00:00
|
|
|
{
|
|
|
|
// based off example code https://docs.microsoft.com/en-us/dotnet/api/system.io.directoryinfo
|
|
|
|
if (!destination.Exists)
|
2020-06-22 09:56:14 +00:00
|
|
|
Directory.CreateDirectory(destination.FullName);
|
2020-06-16 15:39:20 +00:00
|
|
|
|
|
|
|
foreach (System.IO.FileInfo fi in source.GetFiles())
|
|
|
|
{
|
2020-06-23 22:37:29 +00:00
|
|
|
if (topLevelExcludes && IgnoreFiles.Contains(fi.Name))
|
2020-06-16 15:39:20 +00:00
|
|
|
continue;
|
|
|
|
|
2020-06-22 10:43:01 +00:00
|
|
|
AttemptOperation(() => fi.CopyTo(Path.Combine(destination.FullName, fi.Name), true));
|
2020-06-16 15:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach (DirectoryInfo dir in source.GetDirectories())
|
|
|
|
{
|
2020-06-23 22:37:29 +00:00
|
|
|
if (topLevelExcludes && IgnoreDirectories.Contains(dir.Name))
|
2020-06-16 15:39:20 +00:00
|
|
|
continue;
|
|
|
|
|
2020-06-22 10:43:01 +00:00
|
|
|
CopyRecursive(dir, destination.CreateSubdirectory(dir.Name), false);
|
2020-06-16 15:39:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Attempt an IO operation multiple times and only throw if none of the attempts succeed.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="action">The action to perform.</param>
|
|
|
|
/// <param name="attempts">The number of attempts (250ms wait between each).</param>
|
2020-06-22 10:43:01 +00:00
|
|
|
protected static void AttemptOperation(Action action, int attempts = 10)
|
2020-06-16 15:39:20 +00:00
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
action();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
catch (Exception)
|
|
|
|
{
|
|
|
|
if (attempts-- == 0)
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
|
|
|
|
Thread.Sleep(250);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|