mirror of
https://github.com/ppy/osu.git
synced 2025-01-22 15:52:55 +08:00
545c04aaf5
Resolves an issue I've been able to locally reproduce on windows. Basically, the `File.Copy` would begin while realm was blocking. The "restore" operation is posted to the `SynchronizationContext` to run on next update call, but in the mean time the copy would begin, causing a conflict of interest. Very dangerous. Only really noticeable on windows.
388 lines
14 KiB
C#
388 lines
14 KiB
C#
// 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.
|
|
|
|
#nullable disable
|
|
|
|
using System;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Runtime.CompilerServices;
|
|
using NUnit.Framework;
|
|
using osu.Framework.Allocation;
|
|
using osu.Framework.Configuration;
|
|
using osu.Framework.Platform;
|
|
using osu.Framework.Testing;
|
|
using osu.Game.Configuration;
|
|
using osu.Game.IO;
|
|
|
|
namespace osu.Game.Tests.NonVisual
|
|
{
|
|
[TestFixture]
|
|
public class CustomDataDirectoryTest : ImportTest
|
|
{
|
|
[Test]
|
|
public void TestDefaultDirectory()
|
|
{
|
|
using (var host = new CustomTestHeadlessGameHost())
|
|
{
|
|
try
|
|
{
|
|
string defaultStorageLocation = getDefaultLocationFor(host);
|
|
|
|
var osu = LoadOsuIntoHost(host);
|
|
var storage = osu.Dependencies.Get<Storage>();
|
|
|
|
Assert.That(storage.GetFullPath("."), Is.EqualTo(defaultStorageLocation));
|
|
}
|
|
finally
|
|
{
|
|
host.Exit();
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void TestCustomDirectory()
|
|
{
|
|
string customPath = prepareCustomPath();
|
|
|
|
using (var host = new CustomTestHeadlessGameHost())
|
|
{
|
|
using (var storageConfig = new StorageConfigManager(host.InitialStorage))
|
|
storageConfig.SetValue(StorageConfig.FullPath, customPath);
|
|
|
|
try
|
|
{
|
|
var osu = LoadOsuIntoHost(host);
|
|
|
|
// switch to DI'd storage
|
|
var storage = osu.Dependencies.Get<Storage>();
|
|
|
|
Assert.That(storage.GetFullPath("."), Is.EqualTo(customPath));
|
|
}
|
|
finally
|
|
{
|
|
host.Exit();
|
|
cleanupPath(customPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void TestSubDirectoryLookup()
|
|
{
|
|
string customPath = prepareCustomPath();
|
|
|
|
using (var host = new CustomTestHeadlessGameHost())
|
|
{
|
|
using (var storageConfig = new StorageConfigManager(host.InitialStorage))
|
|
storageConfig.SetValue(StorageConfig.FullPath, customPath);
|
|
|
|
try
|
|
{
|
|
var osu = LoadOsuIntoHost(host);
|
|
|
|
// switch to DI'd storage
|
|
var storage = osu.Dependencies.Get<Storage>();
|
|
|
|
string actualTestFile = Path.Combine(customPath, "rulesets", "test");
|
|
|
|
File.WriteAllText(actualTestFile, "test");
|
|
|
|
var rulesetStorage = storage.GetStorageForDirectory("rulesets");
|
|
string lookupPath = rulesetStorage.GetFiles(".").Single();
|
|
|
|
Assert.That(lookupPath, Is.EqualTo("test"));
|
|
}
|
|
finally
|
|
{
|
|
host.Exit();
|
|
cleanupPath(customPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void TestMigration()
|
|
{
|
|
string customPath = prepareCustomPath();
|
|
|
|
using (var host = new CustomTestHeadlessGameHost())
|
|
{
|
|
try
|
|
{
|
|
string defaultStorageLocation = getDefaultLocationFor(host);
|
|
|
|
var osu = LoadOsuIntoHost(host);
|
|
var storage = osu.Dependencies.Get<Storage>();
|
|
var osuStorage = storage as MigratableStorage;
|
|
|
|
// Store the current storage's path. We'll need to refer to this for assertions in the original directory after the migration completes.
|
|
string originalDirectory = storage.GetFullPath(".");
|
|
|
|
// ensure we perform a save
|
|
host.Dependencies.Get<FrameworkConfigManager>().Save();
|
|
|
|
// ensure we "use" cache
|
|
host.Storage.GetStorageForDirectory("cache");
|
|
|
|
// for testing nested files are not ignored (only top level)
|
|
host.Storage.GetStorageForDirectory("test-nested").GetStorageForDirectory("cache");
|
|
|
|
Assert.That(storage.GetFullPath("."), Is.EqualTo(defaultStorageLocation));
|
|
|
|
osu.Migrate(customPath);
|
|
|
|
Assert.That(storage.GetFullPath("."), Is.EqualTo(customPath));
|
|
|
|
// ensure cache was not moved
|
|
Assert.That(Directory.Exists(Path.Combine(originalDirectory, "cache")));
|
|
|
|
// ensure nested cache was moved
|
|
Assert.That(!Directory.Exists(Path.Combine(originalDirectory, "test-nested", "cache")));
|
|
Assert.That(storage.ExistsDirectory(Path.Combine("test-nested", "cache")));
|
|
|
|
Assert.That(osuStorage, Is.Not.Null);
|
|
|
|
// In the following tests, realm files are ignored as
|
|
// - in the case of checking the source, interacting with the pipe files (.realm.note) may
|
|
// lead to unexpected behaviour.
|
|
// - in the case of checking the destination, the files may have already been recreated by the game
|
|
// as part of the standard migration flow.
|
|
|
|
foreach (string file in osuStorage.IgnoreFiles)
|
|
{
|
|
if (!file.Contains(".realm", StringComparison.Ordinal))
|
|
{
|
|
Assert.That(File.Exists(Path.Combine(originalDirectory, file)));
|
|
Assert.That(storage.Exists(file), Is.False, () => $"{file} exists in destination when it was expected to be ignored");
|
|
}
|
|
}
|
|
|
|
foreach (string dir in osuStorage.IgnoreDirectories)
|
|
{
|
|
if (!dir.Contains(".realm", StringComparison.Ordinal))
|
|
{
|
|
Assert.That(Directory.Exists(Path.Combine(originalDirectory, dir)));
|
|
Assert.That(storage.Exists(dir), Is.False, () => $"{dir} exists in destination when it was expected to be ignored");
|
|
}
|
|
}
|
|
|
|
Assert.That(new StreamReader(Path.Combine(originalDirectory, "storage.ini")).ReadToEnd().Contains($"FullPath = {customPath}"));
|
|
}
|
|
finally
|
|
{
|
|
host.Exit();
|
|
cleanupPath(customPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void TestMigrationBetweenTwoTargets()
|
|
{
|
|
string customPath = prepareCustomPath();
|
|
string customPath2 = prepareCustomPath();
|
|
|
|
using (var host = new CustomTestHeadlessGameHost())
|
|
{
|
|
try
|
|
{
|
|
var osu = LoadOsuIntoHost(host);
|
|
|
|
Assert.DoesNotThrow(() => osu.Migrate(customPath));
|
|
Assert.That(File.Exists(Path.Combine(customPath, OsuGameBase.CLIENT_DATABASE_FILENAME)));
|
|
|
|
Assert.DoesNotThrow(() => osu.Migrate(customPath2));
|
|
Assert.That(File.Exists(Path.Combine(customPath2, OsuGameBase.CLIENT_DATABASE_FILENAME)));
|
|
|
|
// some files may have been left behind for whatever reason, but that's not what we're testing here.
|
|
cleanupPath(customPath);
|
|
|
|
Assert.DoesNotThrow(() => osu.Migrate(customPath));
|
|
Assert.That(File.Exists(Path.Combine(customPath, OsuGameBase.CLIENT_DATABASE_FILENAME)));
|
|
}
|
|
finally
|
|
{
|
|
host.Exit();
|
|
cleanupPath(customPath);
|
|
cleanupPath(customPath2);
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void TestMigrationToSameTargetFails()
|
|
{
|
|
string customPath = prepareCustomPath();
|
|
|
|
using (var host = new CustomTestHeadlessGameHost())
|
|
{
|
|
try
|
|
{
|
|
var osu = LoadOsuIntoHost(host);
|
|
|
|
Assert.DoesNotThrow(() => osu.Migrate(customPath));
|
|
Assert.Throws<ArgumentException>(() => osu.Migrate(customPath));
|
|
}
|
|
finally
|
|
{
|
|
host.Exit();
|
|
cleanupPath(customPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void TestMigrationFailsOnExistingData()
|
|
{
|
|
string customPath = prepareCustomPath();
|
|
string customPath2 = prepareCustomPath();
|
|
|
|
using (var host = new CustomTestHeadlessGameHost())
|
|
{
|
|
try
|
|
{
|
|
var osu = LoadOsuIntoHost(host);
|
|
|
|
var storage = osu.Dependencies.Get<Storage>();
|
|
var osuStorage = storage as OsuStorage;
|
|
|
|
string originalDirectory = storage.GetFullPath(".");
|
|
|
|
Assert.DoesNotThrow(() => osu.Migrate(customPath));
|
|
Assert.That(File.Exists(Path.Combine(customPath, OsuGameBase.CLIENT_DATABASE_FILENAME)));
|
|
|
|
Directory.CreateDirectory(customPath2);
|
|
File.WriteAllText(Path.Combine(customPath2, OsuGameBase.CLIENT_DATABASE_FILENAME), "I am a text");
|
|
|
|
// Fails because file already exists.
|
|
Assert.Throws<ArgumentException>(() => osu.Migrate(customPath2));
|
|
|
|
osuStorage?.ChangeDataPath(customPath2);
|
|
|
|
Assert.That(osuStorage?.CustomStoragePath, Is.EqualTo(customPath2));
|
|
Assert.That(new StreamReader(Path.Combine(originalDirectory, "storage.ini")).ReadToEnd().Contains($"FullPath = {customPath2}"));
|
|
}
|
|
finally
|
|
{
|
|
host.Exit();
|
|
cleanupPath(customPath);
|
|
cleanupPath(customPath2);
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void TestMigrationToNestedTargetFails()
|
|
{
|
|
string customPath = prepareCustomPath();
|
|
|
|
using (var host = new CustomTestHeadlessGameHost())
|
|
{
|
|
try
|
|
{
|
|
var osu = LoadOsuIntoHost(host);
|
|
|
|
Assert.DoesNotThrow(() => osu.Migrate(customPath));
|
|
|
|
string subFolder = Path.Combine(customPath, "sub");
|
|
|
|
if (Directory.Exists(subFolder))
|
|
Directory.Delete(subFolder, true);
|
|
|
|
Directory.CreateDirectory(subFolder);
|
|
|
|
Assert.Throws<ArgumentException>(() => osu.Migrate(subFolder));
|
|
}
|
|
finally
|
|
{
|
|
host.Exit();
|
|
cleanupPath(customPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void TestMigrationToSeeminglyNestedTarget()
|
|
{
|
|
string customPath = prepareCustomPath();
|
|
|
|
using (var host = new CustomTestHeadlessGameHost())
|
|
{
|
|
try
|
|
{
|
|
var osu = LoadOsuIntoHost(host);
|
|
|
|
Assert.DoesNotThrow(() => osu.Migrate(customPath));
|
|
|
|
string seeminglySubFolder = customPath + "sub";
|
|
|
|
if (Directory.Exists(seeminglySubFolder))
|
|
Directory.Delete(seeminglySubFolder, true);
|
|
|
|
Directory.CreateDirectory(seeminglySubFolder);
|
|
|
|
osu.Migrate(seeminglySubFolder);
|
|
}
|
|
finally
|
|
{
|
|
host.Exit();
|
|
cleanupPath(customPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
private static string getDefaultLocationFor(CustomTestHeadlessGameHost host)
|
|
{
|
|
string path = Path.Combine(TestRunHeadlessGameHost.TemporaryTestDirectory, host.Name);
|
|
|
|
if (Directory.Exists(path))
|
|
Directory.Delete(path, true);
|
|
|
|
return path;
|
|
}
|
|
|
|
private static string prepareCustomPath() => Path.Combine(TestRunHeadlessGameHost.TemporaryTestDirectory, $"custom-path-{Guid.NewGuid()}");
|
|
|
|
private static void cleanupPath(string path)
|
|
{
|
|
try
|
|
{
|
|
if (Directory.Exists(path))
|
|
Directory.Delete(path, true);
|
|
}
|
|
catch
|
|
{
|
|
}
|
|
}
|
|
|
|
public class CustomTestHeadlessGameHost : CleanRunHeadlessGameHost
|
|
{
|
|
public Storage InitialStorage { get; }
|
|
|
|
public CustomTestHeadlessGameHost([CallerMemberName] string callingMethodName = @"")
|
|
: base(callingMethodName: callingMethodName)
|
|
{
|
|
string defaultStorageLocation = getDefaultLocationFor(this);
|
|
|
|
InitialStorage = new DesktopStorage(defaultStorageLocation, this);
|
|
InitialStorage.DeleteDirectory(string.Empty);
|
|
}
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
{
|
|
base.Dispose(isDisposing);
|
|
|
|
try
|
|
{
|
|
// the storage may have changed from the initial location.
|
|
// this handles cleanup of the initial location.
|
|
InitialStorage.DeleteDirectory(string.Empty);
|
|
}
|
|
catch { }
|
|
}
|
|
}
|
|
}
|
|
}
|