2021-12-27 15:37:20 +00:00
|
|
|
using System;
|
|
|
|
using System.Reactive.Subjects;
|
2022-05-17 03:27:23 +00:00
|
|
|
using System.Text.Json;
|
2021-12-27 15:37:20 +00:00
|
|
|
using System.Threading;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
using Wabbajack.Common;
|
|
|
|
using Wabbajack.Downloaders;
|
|
|
|
using Wabbajack.DTOs;
|
2022-05-17 03:27:23 +00:00
|
|
|
using Wabbajack.DTOs.JsonConverters;
|
2021-12-27 15:37:20 +00:00
|
|
|
using Wabbajack.Paths;
|
|
|
|
using Wabbajack.Paths.IO;
|
|
|
|
using Wabbajack.RateLimiter;
|
|
|
|
using Wabbajack.VFS;
|
|
|
|
|
|
|
|
namespace Wabbajack.Services.OSIntegrated.Services;
|
|
|
|
|
|
|
|
public class ModListDownloadMaintainer
|
|
|
|
{
|
|
|
|
private readonly ILogger<ModListDownloadMaintainer> _logger;
|
|
|
|
private readonly Configuration _configuration;
|
|
|
|
private readonly DownloadDispatcher _dispatcher;
|
|
|
|
private readonly FileHashCache _hashCache;
|
|
|
|
private readonly IResource<DownloadDispatcher> _rateLimiter;
|
2021-12-30 00:15:37 +00:00
|
|
|
private int _downloadingCount;
|
2022-05-17 03:27:23 +00:00
|
|
|
private readonly DTOSerializer _dtos;
|
2021-12-27 15:37:20 +00:00
|
|
|
|
|
|
|
public ModListDownloadMaintainer(ILogger<ModListDownloadMaintainer> logger, Configuration configuration,
|
2022-05-17 03:27:23 +00:00
|
|
|
DownloadDispatcher dispatcher, FileHashCache hashCache, DTOSerializer dtos, IResource<DownloadDispatcher> rateLimiter)
|
2021-12-27 15:37:20 +00:00
|
|
|
{
|
|
|
|
_logger = logger;
|
|
|
|
_configuration = configuration;
|
|
|
|
_dispatcher = dispatcher;
|
|
|
|
_hashCache = hashCache;
|
|
|
|
_rateLimiter = rateLimiter;
|
2021-12-30 00:15:37 +00:00
|
|
|
_downloadingCount = 0;
|
2022-05-17 03:27:23 +00:00
|
|
|
_dtos = dtos;
|
2021-12-27 15:37:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public AbsolutePath ModListPath(ModlistMetadata metadata)
|
|
|
|
{
|
2022-03-30 03:39:48 +00:00
|
|
|
return _configuration.ModListsDownloadLocation.Combine(metadata.NamespacedName.Replace("/", "_@@_")).WithExtension(Ext.Wabbajack);
|
2021-12-27 15:37:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public async Task<bool> HaveModList(ModlistMetadata metadata, CancellationToken? token = null)
|
|
|
|
{
|
|
|
|
token ??= CancellationToken.None;
|
|
|
|
var path = ModListPath(metadata);
|
|
|
|
if (!path.FileExists()) return false;
|
|
|
|
|
2021-12-30 00:15:37 +00:00
|
|
|
if (_hashCache.TryGetHashCache(path, out var hash) && hash == metadata.DownloadMetadata!.Hash) return true;
|
|
|
|
if (_downloadingCount > 0) return false;
|
|
|
|
|
2021-12-27 15:37:20 +00:00
|
|
|
return await _hashCache.FileHashCachedAsync(path, token.Value) == metadata.DownloadMetadata!.Hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
public (IObservable<Percent> Progress, Task Task) DownloadModlist(ModlistMetadata metadata, CancellationToken? token = null)
|
|
|
|
{
|
|
|
|
var path = ModListPath(metadata);
|
|
|
|
|
|
|
|
token ??= CancellationToken.None;
|
|
|
|
|
|
|
|
var progress = new Subject<Percent>();
|
|
|
|
progress.OnNext(Percent.Zero);
|
|
|
|
|
|
|
|
var tsk = Task.Run(async () =>
|
|
|
|
{
|
2021-12-27 15:43:10 +00:00
|
|
|
try
|
2021-12-27 15:37:20 +00:00
|
|
|
{
|
2021-12-30 00:15:37 +00:00
|
|
|
Interlocked.Increment(ref _downloadingCount);
|
2022-07-26 04:39:00 +00:00
|
|
|
using var job = await _rateLimiter.Begin($"Downloading {metadata.Title}", metadata.DownloadMetadata!.Size,
|
2021-12-27 15:43:10 +00:00
|
|
|
token.Value);
|
|
|
|
|
|
|
|
job.OnUpdate += (_, pr) => { progress.OnNext(pr.Progress); };
|
|
|
|
|
|
|
|
var hash = await _dispatcher.Download(new Archive()
|
|
|
|
{
|
|
|
|
State = _dispatcher.Parse(new Uri(metadata.Links.Download))!,
|
|
|
|
Size = metadata.DownloadMetadata.Size,
|
|
|
|
Hash = metadata.DownloadMetadata.Hash
|
|
|
|
}, path, job, token.Value);
|
2021-12-27 15:37:20 +00:00
|
|
|
|
2021-12-27 15:43:10 +00:00
|
|
|
_hashCache.FileHashWriteCache(path, hash);
|
2022-05-17 03:27:23 +00:00
|
|
|
await path.WithExtension(Ext.MetaData).WriteAllTextAsync(JsonSerializer.Serialize(metadata));
|
2021-12-27 15:43:10 +00:00
|
|
|
}
|
|
|
|
finally
|
2021-12-27 15:37:20 +00:00
|
|
|
{
|
2021-12-27 15:43:10 +00:00
|
|
|
progress.OnCompleted();
|
2021-12-30 00:15:37 +00:00
|
|
|
Interlocked.Decrement(ref _downloadingCount);
|
2021-12-27 15:43:10 +00:00
|
|
|
}
|
2021-12-27 15:37:20 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
return (progress, tsk);
|
|
|
|
}
|
|
|
|
}
|