using System; using System.Collections.Generic; using System.Linq; using System.Net; using System.Net.Http; using System.Security.Claims; using System.Threading.Tasks; using FluentFTP.Helpers; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Mvc; using Microsoft.Extensions.Logging; using Nettle; using Wabbajack.Common; using Wabbajack.DTOs.GitHub; using Wabbajack.DTOs.JsonConverters; using Wabbajack.Networking.GitHub; using Wabbajack.Paths.IO; using Wabbajack.Server.DataLayer; using Wabbajack.Server.Services; namespace Wabbajack.BuildServer.Controllers; [Authorize(Roles = "Author")] [Route("/author_controls")] public class AuthorControls : ControllerBase { private readonly HttpClient _client; private readonly DTOSerializer _dtos; private readonly Client _gitHubClient; private readonly QuickSync _quickSync; private readonly AppSettings _settings; private readonly ILogger _logger; private readonly SqlService _sql; public AuthorControls(ILogger logger, SqlService sql, QuickSync quickSync, HttpClient client, AppSettings settings, DTOSerializer dtos, Client gitHubClient) { _logger = logger; _sql = sql; _quickSync = quickSync; _client = client; _settings = settings; _dtos = dtos; _gitHubClient = gitHubClient; } [Route("login/{authorKey}")] [AllowAnonymous] public async Task Login(string authorKey) { Response.Cookies.Append(ApiKeyAuthenticationHandler.ApiKeyHeaderName, authorKey); return Redirect($"{_settings.WabbajackBuildServerUri}author_controls/home"); } [Route("lists")] [HttpGet] public async Task AuthorLists() { var user = User.FindFirstValue(ClaimTypes.Name); List lists = new(); foreach (var file in Enum.GetValues()) lists.AddRange((await _gitHubClient.GetData(file)).Lists.Where(l => l.Maintainers.Contains(user)) .Select(lst => lst.Links.MachineURL)); return Ok(lists); } [Route("lists/download_metadata")] [HttpPost] public async Task PostDownloadMetadata() { var user = User.FindFirstValue(ClaimTypes.Name); var data = await _dtos.DeserializeAsync(Request.Body); try { await _gitHubClient.UpdateList(user, data); await _quickSync.Notify(); } catch (Exception ex) { _logger.LogError(ex, "During posting of download_metadata"); return BadRequest(ex); } return Ok(data); } private static async Task HomePageTemplate(object o) { var data = await KnownFolders.EntryPoint.Combine(@"Controllers\Templates\AuthorControls.html") .ReadAllTextAsync(); var func = NettleEngine.GetCompiler().Compile(data); return func(o); } [Route("home")] [Authorize("")] public async Task HomePage() { var user = User.FindFirstValue(ClaimTypes.Name); var files = (await _sql.AllAuthoredFiles()) .Where(af => af.Author == user) .Select(af => new { Size = af.Size.FileSizeToString(), OriginalSize = af.Size, Name = af.OriginalFileName, MangledName = af.MungedName, UploadedDate = af.LastTouched }) .OrderBy(f => f.Name) .ThenBy(f => f.UploadedDate) .ToList(); var result = HomePageTemplate(new { User = user, TotalUsage = files.Select(f => f.OriginalSize).Sum().ToFileSizeString(), WabbajackFiles = files.Where(f => f.Name.EndsWith(Ext.Wabbajack.ToString())), OtherFiles = files.Where(f => !f.Name.EndsWith(Ext.Wabbajack.ToString())) }); return new ContentResult { ContentType = "text/html", StatusCode = (int) HttpStatusCode.OK, Content = await result }; } }