wabbajack/Wabbajack.Server/Controllers/AuthoredFiles.cs

195 lines
7.3 KiB
C#
Raw Permalink Normal View History

2020-05-09 22:16:16 +00:00
using System;
using System.IO;
using System.IO.Compression;
2021-11-27 18:31:35 +00:00
using System.Linq;
2020-05-09 22:16:16 +00:00
using System.Net;
using System.Security.Claims;
2021-09-27 12:42:46 +00:00
using System.Threading;
2020-05-09 22:16:16 +00:00
using System.Threading.Tasks;
using FluentFTP;
2020-06-03 20:16:41 +00:00
using Microsoft.AspNetCore.Authorization;
2020-05-09 22:16:16 +00:00
using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Logging;
2021-11-27 18:31:35 +00:00
using Microsoft.Extensions.Primitives;
2020-05-10 01:35:42 +00:00
using Nettle;
2020-05-09 22:16:16 +00:00
using Wabbajack.Common;
2021-09-27 12:42:46 +00:00
using Wabbajack.DTOs.CDN;
using Wabbajack.DTOs.JsonConverters;
using Wabbajack.Hashing.xxHash64;
2021-11-27 18:31:35 +00:00
using Wabbajack.Server.DataModels;
2020-05-09 22:16:16 +00:00
using Wabbajack.Server.DTOs;
2021-12-20 16:22:40 +00:00
using Wabbajack.Server.Extensions;
using Wabbajack.Server.Services;
2020-05-09 22:16:16 +00:00
2021-10-23 16:51:17 +00:00
namespace Wabbajack.BuildServer.Controllers;
[Authorize(Roles = "Author")]
[Route("/authored_files")]
public class AuthoredFiles : ControllerBase
2020-05-09 22:16:16 +00:00
{
2021-12-20 16:31:10 +00:00
private static readonly Func<object, string> HandleGetListTemplate = NettleEngine.GetCompiler().Compile(@"
2021-10-23 16:51:17 +00:00
<html><body>
<table>
{{each $.files }}
2021-11-27 18:31:35 +00:00
<tr>
2021-12-20 16:27:46 +00:00
<td><a href='https://authored-files.wabbajack.org/{{$.Definition.MungedName}}'>{{$.Definition.OriginalFileName}}</a></td>
2021-11-27 18:31:35 +00:00
<td>{{$.HumanSize}}</td>
<td>{{$.Definition.Author}}</td>
<td>{{$.Updated}}</td>
2021-12-20 16:27:46 +00:00
<td><a href='/authored_files/direct_link/{{$.Definition.MungedName}}'>(Slow) HTTP Direct Link</a></td>
2021-11-27 18:31:35 +00:00
</tr>
2021-10-23 16:51:17 +00:00
{{/each}}
</table>
</body></html>
");
private readonly DTOSerializer _dtos;
private readonly DiscordWebHook _discord;
private readonly ILogger<AuthoredFiles> _logger;
private readonly AppSettings _settings;
2021-11-27 18:31:35 +00:00
private readonly AuthorFiles _authoredFiles;
2021-10-23 16:51:17 +00:00
2021-11-27 18:31:35 +00:00
public AuthoredFiles(ILogger<AuthoredFiles> logger, AuthorFiles authorFiles, AppSettings settings, DiscordWebHook discord,
DTOSerializer dtos)
2021-10-23 16:51:17 +00:00
{
_logger = logger;
_settings = settings;
_discord = discord;
_dtos = dtos;
2021-11-27 18:31:35 +00:00
_authoredFiles = authorFiles;
2021-10-23 16:51:17 +00:00
}
2021-11-27 18:31:35 +00:00
2021-10-23 16:51:17 +00:00
[HttpPut]
[Route("{serverAssignedUniqueId}/part/{index}")]
public async Task<IActionResult> UploadFilePart(CancellationToken token, string serverAssignedUniqueId, long index)
2020-05-09 22:16:16 +00:00
{
2021-10-23 16:51:17 +00:00
var user = User.FindFirstValue(ClaimTypes.Name);
2021-11-27 18:31:35 +00:00
var definition = await _authoredFiles.ReadDefinitionForServerId(serverAssignedUniqueId);
2021-10-23 16:51:17 +00:00
if (definition.Author != user)
return Forbid("File Id does not match authorized user");
_logger.Log(LogLevel.Information,
$"Uploading File part {definition.OriginalFileName} - ({index} / {definition.Parts.Length})");
2021-11-27 18:31:35 +00:00
2021-10-23 16:51:17 +00:00
var part = definition.Parts[index];
await using var ms = new MemoryStream();
await Request.Body.CopyToLimitAsync(ms, (int) part.Size, token);
ms.Position = 0;
if (ms.Length != part.Size)
return BadRequest($"Couldn't read enough data for part {part.Size} vs {ms.Length}");
var hash = await ms.Hash(token);
if (hash != part.Hash)
return BadRequest(
$"Hashes don't match for index {index}. Sizes ({ms.Length} vs {part.Size}). Hashes ({hash} vs {part.Hash}");
ms.Position = 0;
2021-11-27 18:31:35 +00:00
await using var partStream = await _authoredFiles.CreatePart(definition.MungedName, (int)index);
await ms.CopyToAsync(partStream, token);
2021-10-23 16:51:17 +00:00
return Ok(part.Hash.ToBase64());
}
[HttpPut]
[Route("create")]
public async Task<IActionResult> CreateUpload()
{
var user = User.FindFirstValue(ClaimTypes.Name);
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
var definition = (await _dtos.DeserializeAsync<FileDefinition>(Request.Body))!;
2020-05-09 22:16:16 +00:00
2021-10-23 16:51:17 +00:00
_logger.Log(LogLevel.Information, "Creating File upload {originalFileName}", definition.OriginalFileName);
2021-11-27 18:31:35 +00:00
definition.ServerAssignedUniqueId = Guid.NewGuid().ToString();
definition.Author = user;
await _authoredFiles.WriteDefinition(definition);
2021-10-23 16:51:17 +00:00
await _discord.Send(Channel.Ham,
new DiscordMessage
{
Content =
$"{user} has started uploading {definition.OriginalFileName} ({definition.Size.ToFileSizeString()})"
});
return Ok(definition.ServerAssignedUniqueId);
}
[HttpPut]
[Route("{serverAssignedUniqueId}/finish")]
public async Task<IActionResult> CreateUpload(string serverAssignedUniqueId)
{
var user = User.FindFirstValue(ClaimTypes.Name);
2021-11-27 18:31:35 +00:00
var definition = await _authoredFiles.ReadDefinitionForServerId(serverAssignedUniqueId);
2021-10-23 16:51:17 +00:00
if (definition.Author != user)
return Forbid("File Id does not match authorized user");
_logger.Log(LogLevel.Information, $"Finalizing file upload {definition.OriginalFileName}");
await _discord.Send(Channel.Ham,
new DiscordMessage
2020-05-09 22:16:16 +00:00
{
2021-10-23 16:51:17 +00:00
Content =
$"{user} has finished uploading {definition.OriginalFileName} ({definition.Size.ToFileSizeString()})"
});
2020-05-09 22:16:16 +00:00
2021-10-23 16:51:17 +00:00
var host = _settings.TestMode ? "test-files" : "authored-files";
return Ok($"https://{host}.wabbajack.org/{definition.MungedName}");
}
2020-05-09 22:16:16 +00:00
2021-10-23 16:51:17 +00:00
[HttpDelete]
[Route("{serverAssignedUniqueId}")]
public async Task<IActionResult> DeleteUpload(string serverAssignedUniqueId)
{
var user = User.FindFirstValue(ClaimTypes.Name);
2021-11-27 18:31:35 +00:00
var definition = (await _authoredFiles.AllAuthoredFiles())
.First(f => f.Definition.ServerAssignedUniqueId == serverAssignedUniqueId)
.Definition;
2021-10-23 16:51:17 +00:00
if (definition.Author != user)
return Forbid("File Id does not match authorized user");
await _discord.Send(Channel.Ham,
new DiscordMessage
{
Content =
$"{user} is deleting {definition.MungedName}, {definition.Size.ToFileSizeString()} to be freed"
});
_logger.Log(LogLevel.Information, $"Deleting upload {definition.OriginalFileName}");
2020-05-09 22:16:16 +00:00
2021-11-27 18:31:35 +00:00
await _authoredFiles.DeleteFile(definition);
2021-10-23 16:51:17 +00:00
return Ok();
}
[HttpGet]
[AllowAnonymous]
[Route("")]
public async Task<ContentResult> UploadedFilesGet()
{
2021-11-27 18:31:35 +00:00
var files = await _authoredFiles.AllAuthoredFiles();
var response = HandleGetListTemplate(new {files = files.OrderByDescending(f => f.Updated).ToArray()});
2021-10-23 16:51:17 +00:00
return new ContentResult
2020-05-10 01:35:42 +00:00
{
2021-10-23 16:51:17 +00:00
ContentType = "text/html",
StatusCode = (int) HttpStatusCode.OK,
Content = response
};
2020-05-09 22:16:16 +00:00
}
2021-11-27 18:31:35 +00:00
[HttpGet]
[AllowAnonymous]
[Route("direct_link/{mungedName}")]
public async Task DirectLink(string mungedName)
{
2021-12-20 16:22:40 +00:00
mungedName = _authoredFiles.DecodeName(mungedName);
2021-11-27 18:31:35 +00:00
var definition = await _authoredFiles.ReadDefinition(mungedName);
Response.Headers.ContentDisposition =
new StringValues($"attachment; filename={definition.OriginalFileName}");
Response.Headers.ContentType = new StringValues("application/octet-stream");
Response.Headers.ContentLength = definition.Size;
Response.Headers.ETag = definition.MungedName + "_direct";
2021-11-27 18:31:35 +00:00
foreach (var part in definition.Parts)
{
await using var partStream = await _authoredFiles.StreamForPart(mungedName, (int)part.Index);
await partStream.CopyToAsync(Response.Body);
}
}
2021-10-23 16:51:17 +00:00
}