2020-07-13 22:10:05 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Security.Claims;
|
2020-08-06 22:40:07 +00:00
|
|
|
|
using System.Text;
|
2021-09-27 12:42:46 +00:00
|
|
|
|
using System.Threading;
|
2020-07-13 22:10:05 +00:00
|
|
|
|
using System.Threading.Tasks;
|
2020-06-16 22:21:01 +00:00
|
|
|
|
using Microsoft.AspNetCore.Authorization;
|
2020-05-20 03:25:41 +00:00
|
|
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
|
using Microsoft.Extensions.Logging;
|
2021-09-27 12:42:46 +00:00
|
|
|
|
using Wabbajack.Downloaders;
|
|
|
|
|
using Wabbajack.DTOs.JsonConverters;
|
|
|
|
|
using Wabbajack.DTOs.ServerResponses;
|
|
|
|
|
using Wabbajack.Hashing.xxHash64;
|
2020-05-20 03:25:41 +00:00
|
|
|
|
using Wabbajack.Server.DataLayer;
|
2020-08-05 20:53:19 +00:00
|
|
|
|
using Wabbajack.Server.DTOs;
|
2020-05-20 03:25:41 +00:00
|
|
|
|
using Wabbajack.Server.Services;
|
2021-09-27 12:42:46 +00:00
|
|
|
|
using Wabbajack.Server.TokenProviders;
|
2020-05-20 03:25:41 +00:00
|
|
|
|
|
|
|
|
|
namespace Wabbajack.BuildServer.Controllers
|
|
|
|
|
{
|
|
|
|
|
[ApiController]
|
|
|
|
|
public class ModUpgrade : ControllerBase
|
|
|
|
|
{
|
|
|
|
|
private ILogger<ModUpgrade> _logger;
|
|
|
|
|
private SqlService _sql;
|
|
|
|
|
private AppSettings _settings;
|
2020-06-06 21:44:30 +00:00
|
|
|
|
private QuickSync _quickSync;
|
2021-09-27 12:42:46 +00:00
|
|
|
|
private readonly DTOSerializer _dtos;
|
|
|
|
|
private readonly DownloadDispatcher _dispatcher;
|
|
|
|
|
private readonly IFtpSiteCredentials _ftpSite;
|
2020-05-20 03:25:41 +00:00
|
|
|
|
|
2021-09-27 12:42:46 +00:00
|
|
|
|
public ModUpgrade(ILogger<ModUpgrade> logger, SqlService sql, DiscordWebHook discord, QuickSync quickSync, AppSettings settings, DTOSerializer dtos,
|
|
|
|
|
DownloadDispatcher dispatcher, IFtpSiteCredentials ftp)
|
2020-05-20 03:25:41 +00:00
|
|
|
|
{
|
|
|
|
|
_logger = logger;
|
|
|
|
|
_sql = sql;
|
|
|
|
|
_settings = settings;
|
2020-06-06 21:44:30 +00:00
|
|
|
|
_quickSync = quickSync;
|
2021-09-27 12:42:46 +00:00
|
|
|
|
_ftpSite = ftp;
|
|
|
|
|
_dtos = dtos;
|
|
|
|
|
_dispatcher = dispatcher;
|
2020-05-20 03:25:41 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[HttpPost]
|
2020-06-16 22:21:01 +00:00
|
|
|
|
[Authorize(Roles = "User")]
|
2020-05-20 03:25:41 +00:00
|
|
|
|
[Route("/mod_upgrade")]
|
2021-09-27 12:42:46 +00:00
|
|
|
|
public async Task<IActionResult> PostModUpgrade(CancellationToken token)
|
2020-05-20 03:25:41 +00:00
|
|
|
|
{
|
2020-07-13 22:10:05 +00:00
|
|
|
|
var isAuthor = User.Claims.Any(c => c.Type == ClaimTypes.Role && c.Value == "Author");
|
2021-09-27 12:42:46 +00:00
|
|
|
|
var request = await _dtos.DeserializeAsync<ModUpgradeRequest>(Request.Body);
|
2020-07-13 22:10:05 +00:00
|
|
|
|
if (!isAuthor)
|
2020-05-20 03:25:41 +00:00
|
|
|
|
{
|
2021-09-27 12:42:46 +00:00
|
|
|
|
var srcDownload = await _sql.GetArchiveDownload(request!.OldArchive.State.PrimaryKeyString,
|
2020-07-13 22:10:05 +00:00
|
|
|
|
request.OldArchive.Hash, request.OldArchive.Size);
|
|
|
|
|
var destDownload = await _sql.GetArchiveDownload(request.NewArchive.State.PrimaryKeyString,
|
|
|
|
|
request.NewArchive.Hash, request.NewArchive.Size);
|
|
|
|
|
|
|
|
|
|
if (srcDownload == default || destDownload == default ||
|
|
|
|
|
await _sql.FindPatch(srcDownload.Id, destDownload.Id) == default)
|
|
|
|
|
{
|
2021-09-27 12:42:46 +00:00
|
|
|
|
if (!await _dispatcher.IsAllowed(request, token))
|
2020-07-13 22:10:05 +00:00
|
|
|
|
{
|
|
|
|
|
_logger.Log(LogLevel.Information,
|
|
|
|
|
$"Upgrade requested from {request.OldArchive.Hash} to {request.NewArchive.Hash} rejected as upgrade is invalid");
|
|
|
|
|
return BadRequest("Invalid mod upgrade");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (_settings.ValidateModUpgrades && !await _sql.HashIsInAModlist(request.OldArchive.Hash))
|
|
|
|
|
{
|
|
|
|
|
_logger.Log(LogLevel.Information,
|
|
|
|
|
$"Upgrade requested from {request.OldArchive.Hash} to {request.NewArchive.Hash} rejected as src hash is not in a curated modlist");
|
|
|
|
|
return BadRequest("Hash is not in a recent modlist");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-20 03:25:41 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-07-13 22:10:05 +00:00
|
|
|
|
try
|
|
|
|
|
{
|
2021-09-27 12:42:46 +00:00
|
|
|
|
if (await _dispatcher.Verify(request!.OldArchive, token))
|
2020-07-13 22:10:05 +00:00
|
|
|
|
{
|
2020-08-12 04:25:12 +00:00
|
|
|
|
//_logger.LogInformation(
|
|
|
|
|
// $"Refusing to upgrade ({request.OldArchive.State.PrimaryKeyString}), old archive is valid");
|
2020-07-13 22:10:05 +00:00
|
|
|
|
return NotFound("File is Valid");
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-10-01 03:50:09 +00:00
|
|
|
|
catch (Exception)
|
2020-06-05 20:53:44 +00:00
|
|
|
|
{
|
2020-08-12 04:25:12 +00:00
|
|
|
|
//_logger.LogInformation(
|
|
|
|
|
// $"Refusing to upgrade ({request.OldArchive.State.PrimaryKeyString}), due to upgrade failure");
|
2020-07-19 23:09:59 +00:00
|
|
|
|
return NotFound("File is Valid");
|
2020-06-05 20:53:44 +00:00
|
|
|
|
}
|
2020-07-13 22:10:05 +00:00
|
|
|
|
|
2020-05-20 03:25:41 +00:00
|
|
|
|
var oldDownload = await _sql.GetOrEnqueueArchive(request.OldArchive);
|
2020-08-06 22:40:07 +00:00
|
|
|
|
|
|
|
|
|
if (await _sql.IsNoPatch(oldDownload.Archive.Hash))
|
|
|
|
|
{
|
|
|
|
|
return BadRequest("File has NoPatch attached");
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-20 03:25:41 +00:00
|
|
|
|
var newDownload = await _sql.GetOrEnqueueArchive(request.NewArchive);
|
|
|
|
|
|
|
|
|
|
var patch = await _sql.FindOrEnqueuePatch(oldDownload.Id, newDownload.Id);
|
|
|
|
|
if (patch.Finished.HasValue)
|
|
|
|
|
{
|
|
|
|
|
if (patch.PatchSize != 0)
|
|
|
|
|
{
|
2020-08-12 04:25:12 +00:00
|
|
|
|
//_logger.Log(LogLevel.Information, $"Upgrade requested from {oldDownload.Archive.Hash} to {newDownload.Archive.Hash} patch Found");
|
2021-09-27 12:42:46 +00:00
|
|
|
|
var host = (await _ftpSite.Get())[StorageSpace.Patches].Username == "wabbajacktest" ? "test-files" : "patches";
|
2020-06-06 21:44:30 +00:00
|
|
|
|
await _sql.MarkPatchUsage(oldDownload.Id, newDownload.Id);
|
2020-05-20 03:25:41 +00:00
|
|
|
|
return
|
|
|
|
|
Ok(
|
2021-03-11 02:28:28 +00:00
|
|
|
|
$"https://{host}.wabbajack.org/{request.OldArchive.Hash.ToHex()}_{request.NewArchive.Hash.ToHex()}");
|
2020-05-20 03:25:41 +00:00
|
|
|
|
}
|
2020-08-12 04:25:12 +00:00
|
|
|
|
//_logger.Log(LogLevel.Information, $"Upgrade requested from {oldDownload.Archive.Hash} to {newDownload.Archive.Hash} patch found but was failed");
|
2020-05-20 03:25:41 +00:00
|
|
|
|
|
|
|
|
|
return NotFound("Patch creation failed");
|
|
|
|
|
}
|
2020-06-06 21:44:30 +00:00
|
|
|
|
|
|
|
|
|
if (!newDownload.DownloadFinished.HasValue)
|
|
|
|
|
{
|
|
|
|
|
await _quickSync.Notify<ArchiveDownloader>();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
await _quickSync.Notify<PatchBuilder>();
|
|
|
|
|
}
|
|
|
|
|
|
2020-08-12 04:25:12 +00:00
|
|
|
|
//_logger.Log(LogLevel.Information, $"Upgrade requested from {oldDownload.Archive.Hash} to {newDownload.Archive.Hash} patch found is processing");
|
2020-05-20 03:25:41 +00:00
|
|
|
|
// Still processing
|
|
|
|
|
return Accepted();
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-13 22:10:05 +00:00
|
|
|
|
[HttpGet]
|
|
|
|
|
[Authorize(Roles = "User")]
|
|
|
|
|
[Route("/mod_upgrade/find/{hashAsHex}")]
|
|
|
|
|
public async Task<IActionResult> FindUpgrade(string hashAsHex)
|
|
|
|
|
{
|
|
|
|
|
var hash = Hash.FromHex(hashAsHex);
|
|
|
|
|
|
|
|
|
|
var patches = await _sql.PatchesForSource(hash);
|
2021-09-27 12:42:46 +00:00
|
|
|
|
return Ok(_dtos.Serialize(patches.Select(p => p.Dest.Archive).ToArray()));
|
2020-07-13 22:10:05 +00:00
|
|
|
|
}
|
2020-08-06 22:40:07 +00:00
|
|
|
|
|
|
|
|
|
[HttpGet]
|
|
|
|
|
[Authorize(Roles = "Author")]
|
|
|
|
|
[Route("/mod_upgrade/no_patch/{hashAsHex}/{rationaleAsHex}")]
|
|
|
|
|
public async Task<IActionResult> PurgePatch(string hashAsHex, string rationaleAsHex)
|
|
|
|
|
{
|
|
|
|
|
var hash = Hash.FromHex(hashAsHex);
|
|
|
|
|
var rationale = Encoding.UTF8.GetString(rationaleAsHex.FromHex());
|
|
|
|
|
await _sql.PurgePatch(hash, rationale);
|
|
|
|
|
return Ok("Purged");
|
|
|
|
|
}
|
2020-08-08 03:40:03 +00:00
|
|
|
|
|
|
|
|
|
[HttpGet]
|
|
|
|
|
[Authorize(Roles = "User")]
|
|
|
|
|
[Route("/mirror/{hashAsHex}")]
|
|
|
|
|
public async Task<IActionResult> HaveHash(string hashAsHex)
|
|
|
|
|
{
|
|
|
|
|
var result = await _sql.HaveMirror(Hash.FromHex(hashAsHex));
|
2021-09-27 12:42:46 +00:00
|
|
|
|
if (result) return Ok($"https://{(await _ftpSite.Get())[StorageSpace.Mirrors].Username}.b-cdn.net/{hashAsHex}");
|
2020-08-08 03:40:03 +00:00
|
|
|
|
return NotFound("Not Mirrored");
|
|
|
|
|
}
|
2020-07-13 22:10:05 +00:00
|
|
|
|
|
|
|
|
|
|
2020-05-20 03:25:41 +00:00
|
|
|
|
}
|
|
|
|
|
}
|