From e07ca73ff08624754658cdd54639cba49f2815d4 Mon Sep 17 00:00:00 2001
From: AL <26797547+Al12rs@users.noreply.github.com>
Date: Mon, 1 Jul 2024 13:57:31 +0200
Subject: [PATCH 1/4] Move downloads out of Temp and clean up after cancel and
complete.
---
.../DownloadService.cs | 33 +++++++--
.../DownloadSettings.cs | 67 +++++++++++++++++++
.../Interfaces/IDownloadService.cs | 5 ++
.../Interfaces/IDownloadTask.cs | 11 +--
.../Services.cs | 2 +
.../Tasks/ADownloadTask.cs | 54 ++++++++++-----
.../AdvancedHttpDownloader.cs | 10 +--
.../DTOs/DownloadState.cs | 4 +-
.../DownloadStatusDesignViewModel.cs | 1 -
.../DownloadServiceDataStoreTests.cs | 31 +++++----
.../DownloadServiceTests.cs | 9 +--
.../AdvancedHttpDownloaderTests.cs | 2 +-
12 files changed, 174 insertions(+), 55 deletions(-)
create mode 100644 src/Networking/NexusMods.Networking.Downloaders/DownloadSettings.cs
diff --git a/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs b/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
index 7d8f75c9cf..27467ceb04 100644
--- a/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
+++ b/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
@@ -1,8 +1,8 @@
using System.Collections.ObjectModel;
+using System.ComponentModel;
using DynamicData;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
-using NexusMods.Abstractions.MnemonicDB.Attributes.Extensions;
using NexusMods.Abstractions.NexusWebApi.Types;
using NexusMods.MnemonicDB.Abstractions;
using NexusMods.Networking.Downloaders.Interfaces;
@@ -14,8 +14,7 @@
using Microsoft.Extensions.Hosting;
using NexusMods.Abstractions.Activities;
using NexusMods.Abstractions.IO;
-using NexusMods.MnemonicDB.Abstractions.DatomIterators;
-using NexusMods.MnemonicDB.Abstractions.Query;
+using NexusMods.Abstractions.Settings;
using NexusMods.Paths;
namespace NexusMods.Networking.Downloaders;
@@ -25,6 +24,9 @@ public class DownloadService : IDownloadService, IDisposable, IHostedService
{
///
public ReadOnlyObservableCollection Downloads => _downloadsCollection;
+
+ ///
+ public AbsolutePath OngoingDownloadsDirectory => _downloadDirectory;
private ReadOnlyObservableCollection _downloadsCollection = ReadOnlyObservableCollection.Empty;
private readonly SourceCache _downloads = new(t => t.PersistentState.Id);
@@ -34,14 +36,26 @@ public class DownloadService : IDownloadService, IDisposable, IHostedService
private bool _isDisposed;
private readonly CompositeDisposable _disposables;
private readonly IFileStore _fileStore;
+ private AbsolutePath _downloadDirectory;
- public DownloadService(ILogger logger, IServiceProvider provider, IFileStore fileStore, IConnection conn)
+ public DownloadService(
+ ILogger logger,
+ IServiceProvider provider,
+ IFileStore fileStore,
+ IConnection conn,
+ IFileSystem fs,
+ ISettingsManager settingsManager)
{
_logger = logger;
_provider = provider;
_conn = conn;
_disposables = new CompositeDisposable();
_fileStore = fileStore;
+ _downloadDirectory = settingsManager.Get().OngoingDownloadLocation.ToPath(fs);
+ if (!_downloadDirectory.DirectoryExists())
+ {
+ _downloadDirectory.CreateDirectory();
+ }
}
internal IEnumerable GetItemsToResume()
@@ -209,8 +223,17 @@ public async Task StopAsync(CancellationToken cancellationToken)
// TODO(Al12rs): should Suspend() instead, but only after moving ongoing dl files outside Temp folder,
// that is otherwise cleaned up on application close, causing exceptions due to file in use,
// on top of discarding all the progress.
- .Select(dl => dl.Cancel());
+ .Select(dl => dl.Suspend());
await Task.WhenAll(suspendingTasks);
}
+
+ ///
+ /// Set a custom downloadDirectory, for tests only.
+ /// Directory should already exist.
+ [EditorBrowsable(EditorBrowsableState.Never)]
+ internal void SetDownloadDirectory(AbsolutePath path)
+ {
+ _downloadDirectory = path;
+ }
}
diff --git a/src/Networking/NexusMods.Networking.Downloaders/DownloadSettings.cs b/src/Networking/NexusMods.Networking.Downloaders/DownloadSettings.cs
new file mode 100644
index 0000000000..bcb878941c
--- /dev/null
+++ b/src/Networking/NexusMods.Networking.Downloaders/DownloadSettings.cs
@@ -0,0 +1,67 @@
+using JetBrains.Annotations;
+using Microsoft.Extensions.DependencyInjection;
+using NexusMods.Abstractions.Settings;
+using NexusMods.Paths;
+
+namespace NexusMods.Networking.Downloaders;
+
+///
+/// Settings for downloads.
+///
+[PublicAPI]
+public class DownloadSettings : ISettings
+{
+
+ ///
+ /// Base directory where files generated during ongoing download operations are located.
+ ///
+ ///
+ /// Should not be placed in Temp or similar directories,
+ /// as download files may need to persist across application and system restarts.
+ ///
+ public ConfigurablePath OngoingDownloadLocation { get; set; }
+
+
+ ///
+ public static ISettingsBuilder Configure(ISettingsBuilder settingsBuilder)
+ {
+ return settingsBuilder
+ .ConfigureDefault(CreateDefault)
+ .ConfigureStorageBackend(builder => builder.UseJson());
+ }
+
+ ///
+ /// Create default value.
+ ///
+ public static DownloadSettings CreateDefault(IServiceProvider serviceProvider)
+ {
+ var os = serviceProvider.GetRequiredService().OS;
+
+ return new DownloadSettings
+ {
+ OngoingDownloadLocation = GetStandardDownloadPath(os),
+ };
+ }
+
+ private static ConfigurablePath GetStandardDownloadPath(IOSInformation os)
+ {
+ var basePath = os.MatchPlatform(
+ onWindows: () => KnownPath.LocalApplicationDataDirectory,
+ onLinux: () => KnownPath.XDG_DATA_HOME,
+ onOSX: () => KnownPath.LocalApplicationDataDirectory
+ );
+ // NOTE: OSX ".App" is apparently special, using _ instead of . to prevent weirdness
+ var baseDirectoryName = os.IsOSX ? "NexusMods_App/" : "NexusMods.App/";
+ var downloadsSubPath = baseDirectoryName + "Downloads/Ongoing";
+
+ return new ConfigurablePath(basePath, downloadsSubPath);
+ }
+
+ ///
+ /// Absolute path to the standard downloads' folder.
+ ///
+ public static AbsolutePath GetStandardDownloadsFolder(IFileSystem fs)
+ {
+ return GetStandardDownloadPath(fs.OS).ToPath(fs);
+ }
+}
diff --git a/src/Networking/NexusMods.Networking.Downloaders/Interfaces/IDownloadService.cs b/src/Networking/NexusMods.Networking.Downloaders/Interfaces/IDownloadService.cs
index 7349b21573..be6a6ae835 100644
--- a/src/Networking/NexusMods.Networking.Downloaders/Interfaces/IDownloadService.cs
+++ b/src/Networking/NexusMods.Networking.Downloaders/Interfaces/IDownloadService.cs
@@ -20,6 +20,11 @@ public interface IDownloadService
///
ReadOnlyObservableCollection Downloads { get; }
+ ///
+ /// The base directory for ongoing downloads
+ ///
+ AbsolutePath OngoingDownloadsDirectory { get; }
+
///
/// Adds a task that will download from a NXM link.
///
diff --git a/src/Networking/NexusMods.Networking.Downloaders/Interfaces/IDownloadTask.cs b/src/Networking/NexusMods.Networking.Downloaders/Interfaces/IDownloadTask.cs
index 3a48c19854..1e0eb352cc 100644
--- a/src/Networking/NexusMods.Networking.Downloaders/Interfaces/IDownloadTask.cs
+++ b/src/Networking/NexusMods.Networking.Downloaders/Interfaces/IDownloadTask.cs
@@ -16,9 +16,9 @@ public interface IDownloadTask
DownloaderState.ReadOnly PersistentState { get; }
///
- /// The download location of the task.
+ /// Path of the ongoing download file
///
- public AbsolutePath DownloadLocation { get; }
+ public AbsolutePath DownloadPath { get; }
///
/// Calculates the download speed of the current job.
@@ -93,12 +93,7 @@ public enum DownloadTaskStatus : byte
Downloading,
///
- /// The mod is being archived (and possibly installed) to a loadout.
- ///
- Installing,
-
- ///
- /// The task has ran to completion.
+ /// The task has run to completion.
///
Completed,
diff --git a/src/Networking/NexusMods.Networking.Downloaders/Services.cs b/src/Networking/NexusMods.Networking.Downloaders/Services.cs
index ba79dc2c5e..bad08dcc64 100644
--- a/src/Networking/NexusMods.Networking.Downloaders/Services.cs
+++ b/src/Networking/NexusMods.Networking.Downloaders/Services.cs
@@ -1,4 +1,5 @@
using Microsoft.Extensions.DependencyInjection;
+using NexusMods.Abstractions.Settings;
using NexusMods.MnemonicDB.Abstractions;
using NexusMods.Networking.Downloaders.Interfaces;
using NexusMods.Networking.Downloaders.Tasks;
@@ -18,6 +19,7 @@ public static class Services
public static IServiceCollection AddDownloaders(this IServiceCollection services)
{
return services.AddSingleton()
+ .AddSettings()
.AddHostedService(sp=> sp.GetRequiredService())
.AddSingleton(sp=> sp.GetRequiredService())
.AddTransient()
diff --git a/src/Networking/NexusMods.Networking.Downloaders/Tasks/ADownloadTask.cs b/src/Networking/NexusMods.Networking.Downloaders/Tasks/ADownloadTask.cs
index e6dd8e0fdf..ba47ff2e1b 100644
--- a/src/Networking/NexusMods.Networking.Downloaders/Tasks/ADownloadTask.cs
+++ b/src/Networking/NexusMods.Networking.Downloaders/Tasks/ADownloadTask.cs
@@ -1,18 +1,13 @@
-using System.ComponentModel;
-using System.Runtime.CompilerServices;
-using DynamicData.Kernel;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
using NexusMods.Abstractions.Activities;
using NexusMods.Abstractions.FileStore;
-using NexusMods.Abstractions.FileStore.Downloads;
using NexusMods.Abstractions.HttpDownloader;
-using NexusMods.Abstractions.IO;
-using NexusMods.DataModel;
using NexusMods.MnemonicDB.Abstractions;
using NexusMods.Networking.Downloaders.Interfaces;
using NexusMods.Networking.Downloaders.Tasks.State;
using NexusMods.Paths;
+using NexusMods.Paths.Utilities;
using ReactiveUI;
using ReactiveUI.Fody.Helpers;
@@ -30,33 +25,33 @@ public abstract class ADownloadTask : ReactiveObject, IDownloadTask
///
protected HttpDownloaderState? TransientState = null!;
protected ILogger Logger;
- protected TemporaryFileManager TemporaryFileManager;
protected HttpClient HttpClient;
protected IHttpDownloader HttpDownloader;
protected CancellationTokenSource CancellationTokenSource;
- protected TemporaryPath _downloadLocation = default!;
+ protected AbsolutePath _downloadPath = default!;
protected IFileSystem FileSystem;
protected IFileOriginRegistry FileOriginRegistry;
private DownloaderState.ReadOnly _persistentState;
-
+ private IDownloadService _downloadService;
+
protected ADownloadTask(IServiceProvider provider)
{
Connection = provider.GetRequiredService();
Logger = provider.GetRequiredService>();
- TemporaryFileManager = provider.GetRequiredService();
HttpClient = provider.GetRequiredService();
HttpDownloader = provider.GetRequiredService();
CancellationTokenSource = new CancellationTokenSource();
ActivityFactory = provider.GetRequiredService();
FileSystem = provider.GetRequiredService();
FileOriginRegistry = provider.GetRequiredService();
+ _downloadService = provider.GetRequiredService();
}
public void Init(DownloaderState.ReadOnly state)
{
PersistentState = state;
Downloaded = state.Downloaded;
- _downloadLocation = new TemporaryPath(FileSystem, FileSystem.FromUnsanitizedFullPath(state.DownloadPath), false);
+ _downloadPath = FileSystem.FromUnsanitizedFullPath(state.DownloadPath);
}
///
@@ -73,13 +68,19 @@ public void RefreshState()
///
protected EntityId Create(ITransaction tx)
{
- _downloadLocation = TemporaryFileManager.CreateFile();
+ // Add a subfolder for the download task
+ var guid = Guid.NewGuid().ToString();
+ var downloadSubfolder = _downloadService.OngoingDownloadsDirectory.Combine(guid);
+ downloadSubfolder.CreateDirectory();
+
+ _downloadPath = downloadSubfolder.Combine(guid).AppendExtension(KnownExtensions.Tmp);
+
var state = new DownloaderState.New(tx)
{
FriendlyName = "",
Status = DownloadTaskStatus.Idle,
Downloaded = Size.Zero,
- DownloadPath = DownloadLocation.ToString(),
+ DownloadPath = DownloadPath.ToString(),
};
return state.Id;
}
@@ -151,7 +152,7 @@ protected async Task MarkComplete()
[Reactive]
public DownloaderState.ReadOnly PersistentState { get; protected set; }
- public AbsolutePath DownloadLocation => _downloadLocation;
+ public AbsolutePath DownloadPath => _downloadPath;
[Reactive] public Bandwidth Bandwidth { get; set; } = Bandwidth.From(0);
@@ -173,6 +174,10 @@ public async Task Cancel()
try { await CancellationTokenSource.CancelAsync(); }
catch (Exception) { /* ignored */ }
await SetStatus(DownloadTaskStatus.Cancelled);
+
+ // Cleanup the download directory (this could actually still be in use by the downloader,
+ // since CancelAsync is not guaranteed to wait for exception handlers to finish handling the cancellation)
+ CleanupDownloadFiles();
}
///
@@ -193,24 +198,39 @@ public async Task Resume()
await SetStatus(DownloadTaskStatus.Downloading);
TransientState = new HttpDownloaderState
{
- Activity = ActivityFactory.Create(IHttpDownloader.Group, "Downloading {FileName}", DownloadLocation),
+ Activity = ActivityFactory.Create(IHttpDownloader.Group, "Downloading {FileName}", DownloadPath),
};
_ = StartActivityUpdater();
Logger.LogDebug("Dispatching download task for {Name}", PersistentState.FriendlyName);
- await Download(DownloadLocation, CancellationTokenSource.Token);
+ try
+ {
+ await Download(DownloadPath, CancellationTokenSource.Token);
+ }
+ catch (OperationCanceledException e)
+ {
+ return;
+ }
+
UpdateActivity();
await SetStatus(DownloadTaskStatus.Analyzing);
Logger.LogInformation("Finished download of {Name} starting analysis", PersistentState.FriendlyName);
await AnalyzeFile();
await MarkComplete();
+ CleanupDownloadFiles();
+ }
+
+ // Delete the download task subfolder and all files within it.
+ private void CleanupDownloadFiles()
+ {
+ DownloadPath.Parent.DeleteDirectory(recursive: true);
}
private async Task AnalyzeFile()
{
try
{
- await FileOriginRegistry.RegisterDownload(DownloadLocation, PersistentState.Id, PersistentState.FriendlyName);
+ await FileOriginRegistry.RegisterDownload(DownloadPath, PersistentState.Id, PersistentState.FriendlyName);
}
catch (Exception ex)
{
diff --git a/src/Networking/NexusMods.Networking.HttpDownloader/AdvancedHttpDownloader.cs b/src/Networking/NexusMods.Networking.HttpDownloader/AdvancedHttpDownloader.cs
index bd7c6a3f1a..2e2ccc6dc9 100644
--- a/src/Networking/NexusMods.Networking.HttpDownloader/AdvancedHttpDownloader.cs
+++ b/src/Networking/NexusMods.Networking.HttpDownloader/AdvancedHttpDownloader.cs
@@ -163,12 +163,12 @@ private async Task DownloadWithoutResume(IReadOnlyList
private static async Task FinalizeDownload(DownloadState state, CancellationToken cancel)
{
- var tempPath = state.TempFilePath;
+ var progressPath = state.ProgressFilePath;
if (state.HasIncompleteChunk) return Hash.Zero;
state.StateFilePath.Delete();
- File.Move(tempPath.ToString(), state.Destination.ToString(), true);
+ File.Move(progressPath.ToString(), state.Destination.ToString(), true);
return await state.Destination.XxHash64Async(token: cancel);
}
@@ -177,8 +177,8 @@ private static async Task FinalizeDownload(DownloadState state, Cancellati
private async Task FileWriterTask(DownloadState state, ChannelReader writes,
IActivitySource job, CancellationToken cancel)
{
- var tempPath = state.TempFilePath;
- await using var file = tempPath.Open(FileMode.OpenOrCreate, FileAccess.Write);
+ var progressPath = state.ProgressFilePath;
+ await using var file = progressPath.Open(FileMode.OpenOrCreate, FileAccess.Write);
file.SetLength((long)(ulong)state.TotalSize);
while (true)
@@ -471,7 +471,7 @@ private async Task InitiateState(AbsolutePath destination, Size s
{
DownloadState? state = null;
var stateFilePath = DownloadState.GetStateFilePath(destination);
- if (stateFilePath.FileExists && DownloadState.GetTempFilePath(destination).FileExists)
+ if (stateFilePath.FileExists && DownloadState.GetProgressFilePath(destination).FileExists)
{
_logger.LogInformation("Resuming prior download {FilePath}", destination);
state = await DeserializeDownloadState(stateFilePath, cancel);
diff --git a/src/Networking/NexusMods.Networking.HttpDownloader/DTOs/DownloadState.cs b/src/Networking/NexusMods.Networking.HttpDownloader/DTOs/DownloadState.cs
index b07bacb61c..ea00781f80 100644
--- a/src/Networking/NexusMods.Networking.HttpDownloader/DTOs/DownloadState.cs
+++ b/src/Networking/NexusMods.Networking.HttpDownloader/DTOs/DownloadState.cs
@@ -69,7 +69,7 @@ public Size CompletedSize
/// The file where the download is stored while it is in progress
///
[JsonIgnore]
- public AbsolutePath TempFilePath => GetTempFilePath(Destination);
+ public AbsolutePath ProgressFilePath => GetProgressFilePath(Destination);
///
/// Based on the destination, get the path to the state file
@@ -83,7 +83,7 @@ public Size CompletedSize
///
///
///
- public static AbsolutePath GetTempFilePath(AbsolutePath destination) => destination.ReplaceExtension(new Extension(".downloading"));
+ public static AbsolutePath GetProgressFilePath(AbsolutePath destination) => destination.ReplaceExtension(new Extension(".downloading"));
///
diff --git a/src/NexusMods.App.UI/Controls/DownloadGrid/Columns/DownloadStatus/DownloadStatusDesignViewModel.cs b/src/NexusMods.App.UI/Controls/DownloadGrid/Columns/DownloadStatus/DownloadStatusDesignViewModel.cs
index 6d73af8c31..15c86378b8 100644
--- a/src/NexusMods.App.UI/Controls/DownloadGrid/Columns/DownloadStatus/DownloadStatusDesignViewModel.cs
+++ b/src/NexusMods.App.UI/Controls/DownloadGrid/Columns/DownloadStatus/DownloadStatusDesignViewModel.cs
@@ -71,7 +71,6 @@ public static string FormatStatus(DownloadTaskStatus state, long totalBytes, lon
DownloadTaskStatus.Idle => Language.DownloadStatusDesignViewModel_FormatStatus_Queued,
DownloadTaskStatus.Paused => Language.DownloadStatusDesignViewModel_FormatStatus_Paused,
DownloadTaskStatus.Downloading => Language.DownloadStatusDesignViewModel_FormatStatus_Downloading,
- DownloadTaskStatus.Installing => Language.DownloadStatusDesignViewModel_FormatStatus_Installing,
DownloadTaskStatus.Completed => Language.DownloadStatusDesignViewModel_FormatStatus_Complete,
DownloadTaskStatus.Analyzing => Language.DownloadStatusDesignViewModel_FormatStatus_Analyzing,
DownloadTaskStatus.Cancelled => Language.DownloadStatusDesignViewModel_FormatStatus_Cancelled,
diff --git a/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceDataStoreTests.cs b/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceDataStoreTests.cs
index 1461abe546..e405f8a312 100644
--- a/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceDataStoreTests.cs
+++ b/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceDataStoreTests.cs
@@ -7,14 +7,21 @@
namespace NexusMods.Networking.Downloaders.Tests;
-public class DownloadServiceDataStoreTests(
- DownloadService downloadService,
- IServiceProvider serviceProvider,
- LocalHttpServer server,
- TemporaryFileManager temporaryFileManager,
- IHttpDownloader httpDownloader)
- : AGameTest(serviceProvider)
+public class DownloadServiceDataStoreTests : AGameTest
{
+ private readonly DownloadService _downloadService;
+ private readonly LocalHttpServer _server;
+
+ public DownloadServiceDataStoreTests(DownloadService downloadService,
+ IServiceProvider serviceProvider,
+ LocalHttpServer server,
+ TemporaryFileManager temporaryFileManager,
+ IHttpDownloader httpDownloader) : base(serviceProvider)
+ {
+ _downloadService = downloadService;
+ _server = server;
+ _downloadService.SetDownloadDirectory(temporaryFileManager.CreateFolder(prefix: nameof(DownloadServiceDataStoreTests)));
+ }
// Create a new instance of the DownloadService
@@ -22,7 +29,7 @@ public class DownloadServiceDataStoreTests(
public async Task WhenComplete_StaysPersistedInDataStore()
{
var currentCount = GetTaskCountIncludingCompleted();
- await downloadService.AddTask(new Uri($"{server.Prefix}Resources/RootedAtGameFolder/-Skyrim 202X 9.0 - Architecture-2347-9-0-1664994366.zip"));
+ await _downloadService.AddTask(new Uri($"{_server.Prefix}Resources/RootedAtGameFolder/-Skyrim 202X 9.0 - Architecture-2347-9-0-1664994366.zip"));
var newCount = GetTaskCountIncludingCompleted();
newCount.Should().Be(currentCount + 1);
}
@@ -34,8 +41,8 @@ public async Task WhenStarted_IsPersistedInDataStore()
// Should be persisted into datastore on start, because
var currentCount = GetTasks().Count();
- var url = new Uri($"{server.Prefix}Resources/RootedAtGameFolder/-Skyrim 202X 9.0 - Architecture-2347-9-0-1664994366.zip");
- await downloadService.AddTask(url);
+ var url = new Uri($"{_server.Prefix}Resources/RootedAtGameFolder/-Skyrim 202X 9.0 - Architecture-2347-9-0-1664994366.zip");
+ await _downloadService.AddTask(url);
var newCount = GetTasks().Count();
newCount.Should().Be(currentCount + 1);
@@ -43,13 +50,13 @@ public async Task WhenStarted_IsPersistedInDataStore()
private IEnumerable GetTasks()
{
- return downloadService.Downloads
+ return _downloadService.Downloads
.ToList();
}
private int GetTaskCountIncludingCompleted()
{
- return downloadService.Downloads.Count;
+ return _downloadService.Downloads.Count;
}
}
diff --git a/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceTests.cs b/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceTests.cs
index cb8e033a4a..4cc06b4065 100644
--- a/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceTests.cs
+++ b/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceTests.cs
@@ -20,6 +20,7 @@ public DownloadServiceTests(DownloadService downloadService,
_httpServer = httpServer;
_downloadService = downloadService;
_temporaryFileManager = temporaryFileManager;
+ _downloadService.SetDownloadDirectory(_temporaryFileManager.CreateFolder(prefix:nameof(DownloadServiceTests)));
}
[Fact]
@@ -52,8 +53,8 @@ public async Task AddTask_Uri_ShouldDownload()
DownloadTaskStatus.Downloading,
DownloadTaskStatus.Completed);
- task.DownloadLocation.FileExists.Should().BeTrue();
- (await task.DownloadLocation.ReadAllTextAsync()).Should().Be("Hello, World!");
+ // File is deleted after Analyzing and repacking
+ task.DownloadPath.FileExists.Should().BeFalse();
task.Downloaded.Value.Should().BeGreaterThan(0);
}
@@ -107,8 +108,8 @@ public async Task CanSuspendDownloads()
DownloadTaskStatus.Downloading,
DownloadTaskStatus.Completed);
- task.DownloadLocation.FileExists.Should().BeTrue();
- (await task.DownloadLocation.ReadAllTextAsync()).Should().Be("Suspended Test");
+ // File is deleted after Analyzing and repacking
+ task.DownloadPath.FileExists.Should().BeFalse();
task.Downloaded.Value.Should().BeGreaterThan(0);
}
diff --git a/tests/Networking/NexusMods.Networking.HttpDownloader.Tests/AdvancedHttpDownloaderTests.cs b/tests/Networking/NexusMods.Networking.HttpDownloader.Tests/AdvancedHttpDownloaderTests.cs
index 987dc13a23..68ea59f931 100644
--- a/tests/Networking/NexusMods.Networking.HttpDownloader.Tests/AdvancedHttpDownloaderTests.cs
+++ b/tests/Networking/NexusMods.Networking.HttpDownloader.Tests/AdvancedHttpDownloaderTests.cs
@@ -70,7 +70,7 @@ public async Task CanResumeDownload()
var downloadTask = _httpDownloader.DownloadAsync(sources,path, token: tokenSource.Token);
var progressFile = DownloadState.GetStateFilePath(path.Path);
- var downloadingFile = DownloadState.GetTempFilePath(path.Path);
+ var downloadingFile = DownloadState.GetProgressFilePath(path.Path);
while (!progressFile.FileExists)
await Task.Delay(10);
From 0534f37087ec48b5e7590ba29dd7d8587587dd33 Mon Sep 17 00:00:00 2001
From: AL <26797547+Al12rs@users.noreply.github.com>
Date: Mon, 1 Jul 2024 16:37:11 +0200
Subject: [PATCH 2/4] Remove outdated comment
---
.../NexusMods.Networking.Downloaders/DownloadService.cs | 3 ---
1 file changed, 3 deletions(-)
diff --git a/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs b/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
index 27467ceb04..adab5efab3 100644
--- a/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
+++ b/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
@@ -220,9 +220,6 @@ public async Task StopAsync(CancellationToken cancellationToken)
{
var suspendingTasks = _downloads.Items
.Where(dl => dl.PersistentState.Status == DownloadTaskStatus.Downloading)
- // TODO(Al12rs): should Suspend() instead, but only after moving ongoing dl files outside Temp folder,
- // that is otherwise cleaned up on application close, causing exceptions due to file in use,
- // on top of discarding all the progress.
.Select(dl => dl.Suspend());
await Task.WhenAll(suspendingTasks);
From 7dd99e83c332edb56ea19c82a973203a76141dfe Mon Sep 17 00:00:00 2001
From: AL <26797547+Al12rs@users.noreply.github.com>
Date: Mon, 1 Jul 2024 16:39:15 +0200
Subject: [PATCH 3/4] Remove backing field from property
---
.../DownloadService.cs | 11 +++++------
1 file changed, 5 insertions(+), 6 deletions(-)
diff --git a/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs b/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
index adab5efab3..452f11e70d 100644
--- a/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
+++ b/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
@@ -26,7 +26,7 @@ public class DownloadService : IDownloadService, IDisposable, IHostedService
public ReadOnlyObservableCollection Downloads => _downloadsCollection;
///
- public AbsolutePath OngoingDownloadsDirectory => _downloadDirectory;
+ public AbsolutePath OngoingDownloadsDirectory { get; private set; }
private ReadOnlyObservableCollection _downloadsCollection = ReadOnlyObservableCollection.Empty;
private readonly SourceCache _downloads = new(t => t.PersistentState.Id);
@@ -36,7 +36,6 @@ public class DownloadService : IDownloadService, IDisposable, IHostedService
private bool _isDisposed;
private readonly CompositeDisposable _disposables;
private readonly IFileStore _fileStore;
- private AbsolutePath _downloadDirectory;
public DownloadService(
ILogger logger,
@@ -51,10 +50,10 @@ public DownloadService(
_conn = conn;
_disposables = new CompositeDisposable();
_fileStore = fileStore;
- _downloadDirectory = settingsManager.Get().OngoingDownloadLocation.ToPath(fs);
- if (!_downloadDirectory.DirectoryExists())
+ OngoingDownloadsDirectory = settingsManager.Get().OngoingDownloadLocation.ToPath(fs);
+ if (!OngoingDownloadsDirectory.DirectoryExists())
{
- _downloadDirectory.CreateDirectory();
+ OngoingDownloadsDirectory.CreateDirectory();
}
}
@@ -231,6 +230,6 @@ public async Task StopAsync(CancellationToken cancellationToken)
[EditorBrowsable(EditorBrowsableState.Never)]
internal void SetDownloadDirectory(AbsolutePath path)
{
- _downloadDirectory = path;
+ OngoingDownloadsDirectory = path;
}
}
From 5feae1f774f1f36485e18ea0a3b69545afab308a Mon Sep 17 00:00:00 2001
From: AL <26797547+Al12rs@users.noreply.github.com>
Date: Tue, 2 Jul 2024 10:56:22 +0200
Subject: [PATCH 4/4] Use single setting override location for tests, to avoid
potential concurrency problems of switching base folder of the same service.
---
.../NexusMods.Networking.Downloaders/DownloadService.cs | 8 --------
.../NexusMods.Networking.Downloaders/DownloadSettings.cs | 2 +-
.../DownloadServiceDataStoreTests.cs | 7 +------
.../DownloadServiceTests.cs | 3 ---
.../NexusMods.Networking.Downloaders.Tests/Startup.cs | 9 +++++++++
5 files changed, 11 insertions(+), 18 deletions(-)
diff --git a/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs b/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
index 452f11e70d..52d413dbb8 100644
--- a/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
+++ b/src/Networking/NexusMods.Networking.Downloaders/DownloadService.cs
@@ -224,12 +224,4 @@ public async Task StopAsync(CancellationToken cancellationToken)
await Task.WhenAll(suspendingTasks);
}
- ///
- /// Set a custom downloadDirectory, for tests only.
- /// Directory should already exist.
- [EditorBrowsable(EditorBrowsableState.Never)]
- internal void SetDownloadDirectory(AbsolutePath path)
- {
- OngoingDownloadsDirectory = path;
- }
}
diff --git a/src/Networking/NexusMods.Networking.Downloaders/DownloadSettings.cs b/src/Networking/NexusMods.Networking.Downloaders/DownloadSettings.cs
index bcb878941c..360dd80416 100644
--- a/src/Networking/NexusMods.Networking.Downloaders/DownloadSettings.cs
+++ b/src/Networking/NexusMods.Networking.Downloaders/DownloadSettings.cs
@@ -9,7 +9,7 @@ namespace NexusMods.Networking.Downloaders;
/// Settings for downloads.
///
[PublicAPI]
-public class DownloadSettings : ISettings
+public record DownloadSettings : ISettings
{
///
diff --git a/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceDataStoreTests.cs b/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceDataStoreTests.cs
index cc54719979..dfda51ff2b 100644
--- a/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceDataStoreTests.cs
+++ b/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceDataStoreTests.cs
@@ -1,9 +1,7 @@
using FluentAssertions;
-using NexusMods.Abstractions.HttpDownloader;
using NexusMods.Games.RedEngine;
using NexusMods.Games.TestFramework;
using NexusMods.Networking.Downloaders.Interfaces;
-using NexusMods.Paths;
namespace NexusMods.Networking.Downloaders.Tests;
@@ -14,13 +12,10 @@ public class DownloadServiceDataStoreTests : AGameTest
public DownloadServiceDataStoreTests(DownloadService downloadService,
IServiceProvider serviceProvider,
- LocalHttpServer server,
- TemporaryFileManager temporaryFileManager,
- IHttpDownloader httpDownloader) : base(serviceProvider)
+ LocalHttpServer server) : base(serviceProvider)
{
_downloadService = downloadService;
_server = server;
- _downloadService.SetDownloadDirectory(temporaryFileManager.CreateFolder(prefix: nameof(DownloadServiceDataStoreTests)));
}
// Create a new instance of the DownloadService
diff --git a/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceTests.cs b/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceTests.cs
index 448b706841..999cd2ab65 100644
--- a/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceTests.cs
+++ b/tests/Networking/NexusMods.Networking.Downloaders.Tests/DownloadServiceTests.cs
@@ -10,7 +10,6 @@ public class DownloadServiceTests
// For the uninitiated with xUnit: This is initialized before every test.
private readonly DownloadService _downloadService;
private readonly LocalHttpServer _httpServer;
- private readonly TemporaryFileManager _temporaryFileManager;
private IReadOnlyCollection _downloadTasks;
public DownloadServiceTests(DownloadService downloadService,
@@ -18,8 +17,6 @@ public DownloadServiceTests(DownloadService downloadService,
{
_httpServer = httpServer;
_downloadService = downloadService;
- _temporaryFileManager = temporaryFileManager;
- _downloadService.SetDownloadDirectory(_temporaryFileManager.CreateFolder(prefix:nameof(DownloadServiceTests)));
}
[Fact]
diff --git a/tests/Networking/NexusMods.Networking.Downloaders.Tests/Startup.cs b/tests/Networking/NexusMods.Networking.Downloaders.Tests/Startup.cs
index a5d414c227..a4c97f8c8a 100644
--- a/tests/Networking/NexusMods.Networking.Downloaders.Tests/Startup.cs
+++ b/tests/Networking/NexusMods.Networking.Downloaders.Tests/Startup.cs
@@ -1,11 +1,13 @@
using Microsoft.Extensions.DependencyInjection;
using NexusMods.Abstractions.GuidedInstallers;
using NexusMods.Abstractions.Serialization;
+using NexusMods.Abstractions.Settings;
using NexusMods.App.BuildInfo;
using NexusMods.Games.FOMOD;
using NexusMods.Games.Generic;
using NexusMods.Games.RedEngine;
using NexusMods.Games.TestFramework;
+using NexusMods.Paths;
using NexusMods.StandardGameLocators.TestHelpers;
namespace NexusMods.Networking.Downloaders.Tests;
@@ -14,6 +16,9 @@ public class Startup
{
public void ConfigureServices(IServiceCollection services)
{
+ const KnownPath baseKnownPath = KnownPath.EntryDirectory;
+ var baseDirectory = $"NexusMods.Downloaders.Tests-{Guid.NewGuid()}";
+
services
.AddSingleton()
.AddDefaultServicesForTesting()
@@ -24,6 +29,10 @@ public void ConfigureServices(IServiceCollection services)
.AddSerializationAbstractions()
.AddFomod()
.AddDownloaders()
+ .OverrideSettingsForTests(settings => settings with
+ {
+ OngoingDownloadLocation = new ConfigurablePath(baseKnownPath, $"{baseDirectory}/Downloads/Ongoing"),
+ })
.AddSingleton()
.Validate();
}