2022-06-08 03:48:13 +00:00
|
|
|
using System.Text;
|
2023-10-20 21:40:35 +00:00
|
|
|
using Amazon.Runtime;
|
|
|
|
using Amazon.S3;
|
|
|
|
using Amazon.S3.Model;
|
2022-06-08 03:48:13 +00:00
|
|
|
using FluentFTP.Helpers;
|
|
|
|
using Microsoft.AspNetCore.Authorization;
|
|
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
using Microsoft.Extensions.Logging;
|
2022-06-08 13:12:44 +00:00
|
|
|
using Microsoft.Net.Http.Headers;
|
2022-06-08 03:48:13 +00:00
|
|
|
using Wabbajack.BuildServer;
|
|
|
|
using Wabbajack.Downloaders;
|
|
|
|
using Wabbajack.Downloaders.Interfaces;
|
|
|
|
using Wabbajack.DTOs;
|
|
|
|
using Wabbajack.DTOs.DownloadStates;
|
|
|
|
using Wabbajack.Hashing.xxHash64;
|
2023-10-20 21:40:35 +00:00
|
|
|
using Wabbajack.Paths;
|
2022-06-08 03:48:13 +00:00
|
|
|
using Wabbajack.Paths.IO;
|
2023-10-20 21:40:35 +00:00
|
|
|
using Wabbajack.RateLimiter;
|
2022-06-08 03:48:13 +00:00
|
|
|
using Wabbajack.VFS;
|
|
|
|
|
|
|
|
namespace Wabbajack.Server.Controllers;
|
|
|
|
|
|
|
|
[ApiController]
|
|
|
|
[Route("/proxy")]
|
|
|
|
public class Proxy : ControllerBase
|
|
|
|
{
|
|
|
|
private readonly ILogger<Proxy> _logger;
|
|
|
|
private readonly DownloadDispatcher _dispatcher;
|
|
|
|
private readonly TemporaryFileManager _tempFileManager;
|
|
|
|
private readonly AppSettings _appSettings;
|
|
|
|
private readonly FileHashCache _hashCache;
|
2023-10-20 21:40:35 +00:00
|
|
|
private readonly IAmazonS3 _s3;
|
|
|
|
private readonly string _bucket;
|
|
|
|
|
|
|
|
private string _redirectUrl = "https://proxy.wabbajack.org/";
|
|
|
|
private readonly IResource<DownloadDispatcher> _resource;
|
|
|
|
|
|
|
|
public Proxy(ILogger<Proxy> logger, DownloadDispatcher dispatcher, TemporaryFileManager tempFileManager,
|
|
|
|
FileHashCache hashCache, AppSettings appSettings, IAmazonS3 s3, IResource<DownloadDispatcher> resource)
|
2022-06-08 03:48:13 +00:00
|
|
|
{
|
|
|
|
_logger = logger;
|
|
|
|
_dispatcher = dispatcher;
|
|
|
|
_tempFileManager = tempFileManager;
|
|
|
|
_appSettings = appSettings;
|
|
|
|
_hashCache = hashCache;
|
2023-10-20 21:40:35 +00:00
|
|
|
_s3 = s3;
|
|
|
|
_bucket = _appSettings.S3.ProxyFilesBucket;
|
|
|
|
_resource = resource;
|
2022-06-08 03:48:13 +00:00
|
|
|
}
|
2022-06-09 13:37:08 +00:00
|
|
|
|
|
|
|
[HttpHead]
|
|
|
|
public async Task<IActionResult> ProxyHead(CancellationToken token, [FromQuery] Uri uri, [FromQuery] string? name,
|
|
|
|
[FromQuery] string? hash)
|
|
|
|
{
|
|
|
|
var cacheName = (await Encoding.UTF8.GetBytes(uri.ToString()).Hash()).ToHex();
|
2023-10-20 22:04:47 +00:00
|
|
|
return new RedirectResult(_redirectUrl + cacheName);
|
2022-06-09 13:37:08 +00:00
|
|
|
}
|
|
|
|
|
2022-06-08 03:48:13 +00:00
|
|
|
[HttpGet]
|
|
|
|
public async Task<IActionResult> ProxyGet(CancellationToken token, [FromQuery] Uri uri, [FromQuery] string? name, [FromQuery] string? hash)
|
|
|
|
{
|
2023-10-20 21:40:35 +00:00
|
|
|
|
|
|
|
Hash hashResult = default;
|
2022-06-08 03:48:13 +00:00
|
|
|
var shouldMatch = hash != null ? Hash.FromHex(hash) : default;
|
|
|
|
|
|
|
|
_logger.LogInformation("Got proxy request for {Uri}", uri);
|
|
|
|
var state = _dispatcher.Parse(uri);
|
|
|
|
var cacheName = (await Encoding.UTF8.GetBytes(uri.ToString()).Hash()).ToHex();
|
2023-10-20 21:40:35 +00:00
|
|
|
var cacheFile = await GetCacheEntry(cacheName);
|
2022-06-08 03:48:13 +00:00
|
|
|
|
|
|
|
if (state == null)
|
|
|
|
{
|
|
|
|
return BadRequest(new {Type = "Could not get state from Uri", Uri = uri.ToString()});
|
|
|
|
}
|
|
|
|
|
|
|
|
var archive = new Archive
|
|
|
|
{
|
|
|
|
Name = name ?? "",
|
|
|
|
State = state,
|
|
|
|
Hash = shouldMatch
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
var downloader = _dispatcher.Downloader(archive);
|
|
|
|
if (downloader is not IProxyable)
|
|
|
|
{
|
|
|
|
return BadRequest(new {Type = "Downloader is not IProxyable", Downloader = downloader.GetType().FullName});
|
|
|
|
}
|
|
|
|
|
2023-10-20 21:40:35 +00:00
|
|
|
if (cacheFile != null && (DateTime.UtcNow - cacheFile.LastModified) > TimeSpan.FromHours(4))
|
2022-06-08 03:48:13 +00:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
var verify = await _dispatcher.Verify(archive, token);
|
|
|
|
if (verify)
|
2023-10-20 21:40:35 +00:00
|
|
|
await TouchCacheEntry(cacheName);
|
2022-06-08 03:48:13 +00:00
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
2023-10-20 21:40:35 +00:00
|
|
|
_logger.LogInformation(ex, "When trying to verify cached file ({Hash}) {Url}",
|
|
|
|
cacheFile.Hash, uri);
|
|
|
|
await TouchCacheEntry(cacheName);
|
2022-06-08 03:48:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-20 21:40:35 +00:00
|
|
|
if (cacheFile != null && (DateTime.Now - cacheFile.LastModified) > TimeSpan.FromHours(24))
|
2022-06-08 03:48:13 +00:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2023-10-20 21:40:35 +00:00
|
|
|
await DeleteCacheEntry(cacheName);
|
2022-06-08 03:48:13 +00:00
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
_logger.LogError(ex, "When trying to delete expired file");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-20 21:40:35 +00:00
|
|
|
var redirectUrl = _redirectUrl + cacheName + "?response-content-disposition=attachment;filename=" + (name ?? "unknown");
|
|
|
|
if (cacheFile != null)
|
2022-06-08 03:48:13 +00:00
|
|
|
{
|
|
|
|
if (hash != default)
|
|
|
|
{
|
2023-10-20 21:40:35 +00:00
|
|
|
if (cacheFile.Hash != shouldMatch)
|
2022-06-08 03:48:13 +00:00
|
|
|
return BadRequest(new {Type = "Unmatching Hashes", Expected = shouldMatch.ToHex(), Found = hashResult.ToHex()});
|
|
|
|
}
|
2023-10-20 21:40:35 +00:00
|
|
|
return new RedirectResult(redirectUrl);
|
2022-06-08 03:48:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_logger.LogInformation("Downloading proxy request for {Uri}", uri);
|
|
|
|
|
|
|
|
var tempFile = _tempFileManager.CreateFile(deleteOnDispose:false);
|
|
|
|
|
2022-06-08 13:12:44 +00:00
|
|
|
var proxyDownloader = _dispatcher.Downloader(archive) as IProxyable;
|
2022-06-08 03:48:13 +00:00
|
|
|
|
2023-10-20 21:40:35 +00:00
|
|
|
using var job = await _resource.Begin("Downloading file", 0, token);
|
|
|
|
hashResult = await proxyDownloader!.Download(archive, tempFile.Path, job, token);
|
|
|
|
|
|
|
|
|
|
|
|
if (hash != default && hashResult != shouldMatch)
|
|
|
|
{
|
|
|
|
if (tempFile.Path.FileExists())
|
|
|
|
tempFile.Path.Delete();
|
2023-10-20 22:17:15 +00:00
|
|
|
return NotFound();
|
2022-06-08 03:48:13 +00:00
|
|
|
}
|
2023-10-20 21:40:35 +00:00
|
|
|
|
|
|
|
await PutCacheEntry(tempFile.Path, cacheName, hashResult);
|
2022-06-08 03:48:13 +00:00
|
|
|
|
2023-10-20 21:40:35 +00:00
|
|
|
_logger.LogInformation("Returning proxy request for {Uri}", uri);
|
|
|
|
return new RedirectResult(redirectUrl);
|
|
|
|
}
|
2022-06-08 13:12:44 +00:00
|
|
|
|
2023-10-20 21:40:35 +00:00
|
|
|
private async Task<CacheStatus?> GetCacheEntry(string name)
|
|
|
|
{
|
2023-10-20 22:04:47 +00:00
|
|
|
GetObjectMetadataResponse info;
|
|
|
|
try
|
2023-10-20 21:40:35 +00:00
|
|
|
{
|
2023-10-20 22:04:47 +00:00
|
|
|
info = await _s3.GetObjectMetadataAsync(new GetObjectMetadataRequest()
|
|
|
|
{
|
|
|
|
BucketName = _bucket,
|
|
|
|
Key = name,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
catch (Exception _)
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2023-10-20 21:40:35 +00:00
|
|
|
if (info.HttpStatusCode == System.Net.HttpStatusCode.NotFound)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (info.Metadata["WJ-Hash"] == null)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (!Hash.TryGetFromHex(info.Metadata["WJ-Hash"], out var hash))
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return new CacheStatus
|
|
|
|
{
|
|
|
|
LastModified = info.LastModified,
|
|
|
|
Size = info.ContentLength,
|
|
|
|
Hash = hash
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
private async Task TouchCacheEntry(string name)
|
|
|
|
{
|
|
|
|
await _s3.CopyObjectAsync(new CopyObjectRequest()
|
|
|
|
{
|
|
|
|
SourceBucket = _bucket,
|
|
|
|
DestinationBucket = _bucket,
|
|
|
|
SourceKey = name,
|
|
|
|
DestinationKey = name,
|
|
|
|
MetadataDirective = S3MetadataDirective.REPLACE,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private async Task PutCacheEntry(AbsolutePath path, string name, Hash hash)
|
|
|
|
{
|
|
|
|
var obj = new PutObjectRequest
|
|
|
|
{
|
|
|
|
BucketName = _bucket,
|
|
|
|
Key = name,
|
|
|
|
FilePath = path.ToString(),
|
|
|
|
ContentType = "application/octet-stream",
|
|
|
|
DisablePayloadSigning = true
|
|
|
|
};
|
|
|
|
obj.Metadata.Add("WJ-Hash", hash.ToHex());
|
|
|
|
await _s3.PutObjectAsync(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
private async Task DeleteCacheEntry(string name)
|
|
|
|
{
|
|
|
|
await _s3.DeleteObjectAsync(new DeleteObjectRequest
|
|
|
|
{
|
|
|
|
BucketName = _bucket,
|
|
|
|
Key = name
|
|
|
|
});
|
|
|
|
}
|
2022-06-08 03:48:13 +00:00
|
|
|
|
2023-10-20 21:40:35 +00:00
|
|
|
record CacheStatus
|
|
|
|
{
|
|
|
|
public DateTime LastModified { get; init; }
|
|
|
|
public long Size { get; init; }
|
|
|
|
|
|
|
|
public Hash Hash { get; init; }
|
2022-06-08 03:48:13 +00:00
|
|
|
}
|
|
|
|
}
|