using System; using System.Collections.Generic; using System.Linq; using System.Net; using System.Threading.Tasks; using Microsoft.AspNetCore.Mvc; using Microsoft.Extensions.Logging; using Nettle; using Wabbajack.BuildServer.Model.Models; using Wabbajack.BuildServer.Models; using Wabbajack.Common; using Wabbajack.Lib.Downloaders; using Wabbajack.Lib.ModListRegistry; namespace Wabbajack.BuildServer.Controllers { [ApiController] [Route("/lists")] public class ListValidation : AControllerBase { public ListValidation(ILogger logger, SqlService sql) : base(logger, sql) { } public async Task> GetSummaries() { var data = await SQL.GetValidationData(); using var queue = new WorkQueue(); var results = data.ModLists.PMap(queue, list => { var archives = list.ModList.Archives.Select(archive => { switch (archive.State) { case NexusDownloader.State nexusState when data.NexusFiles.Contains(( nexusState.Game.MetaData().NexusGameId, nexusState.ModID, nexusState.FileID)): return (archive, true); case NexusDownloader.State nexusState: return (archive, false); case ManualDownloader.State _: return (archive, true); default: { if (data.ArchiveStatus.TryGetValue((archive.State.PrimaryKeyString, archive.Hash), out var isValid)) { return (archive, isValid); } return (archive, false); } } }).ToList(); var failedCount = archives.Count(f => !f.Item2); var passCount = archives.Count(f => f.Item2); var summary = new ModListSummary { Checked = DateTime.UtcNow, Failed = failedCount, MachineURL = list.Metadata.Links.MachineURL, Name = list.Metadata.Title, Passed = passCount }; var detailed = new DetailedStatus { Name = list.Metadata.Title, Checked = DateTime.UtcNow, DownloadMetaData = list.Metadata.DownloadMetadata, HasFailures = failedCount > 0, MachineName = list.Metadata.Links.MachineURL, Archives = archives.Select(a => new DetailedStatusItem { Archive = a.archive, IsFailing = !a.Item2 }).ToList() }; return (summary, detailed); }); return await results; } [HttpGet] [Route("status.json")] public async Task> HandleGetLists() { return (await GetSummaries()).Select(d => d.Summary); } private static readonly Func HandleGetRssFeedTemplate = NettleEngine.GetCompiler().Compile(@" {{lst.Name}} - Broken Mods http://build.wabbajack.org/status/{{lst.Name}}.html These are mods that are broken and need updating {{ each $.failed }} {{$.Archive.Name}} {{$.Archive.Hash}} {{$.Archive.State.PrimaryKeyString}} {{$.Archive.Name}} {{/each}} "); [HttpGet] [Route("status/{Name}/broken.rss")] public async Task HandleGetRSSFeed(string Name) { var lst = await DetailedStatus(Name); var response = HandleGetRssFeedTemplate(new { lst, failed = lst.Archives.Where(a => a.IsFailing).ToList(), passed = lst.Archives.Where(a => !a.IsFailing).ToList() }); return new ContentResult { ContentType = "application/rss+xml", StatusCode = (int) HttpStatusCode.OK, Content = response }; } private static readonly Func HandleGetListTemplate = NettleEngine.GetCompiler().Compile(@"

{{lst.Name}} - {{lst.Checked}} - {{ago}}min ago

Failed ({{failed.Count}}):

    {{each $.failed }}
  • {{$.Archive.Name}}
  • {{/each}}

Passed ({{passed.Count}}):

    {{each $.passed }}
  • {{$.Archive.Name}}
  • {{/each}}
"); [HttpGet] [Route("status/{Name}.html")] public async Task HandleGetListHtml(string Name) { var lst = await DetailedStatus(Name); var response = HandleGetListTemplate(new { lst, ago = (DateTime.UtcNow - lst.Checked).TotalMinutes, failed = lst.Archives.Where(a => a.IsFailing).ToList(), passed = lst.Archives.Where(a => !a.IsFailing).ToList() }); return new ContentResult { ContentType = "text/html", StatusCode = (int) HttpStatusCode.OK, Content = response }; } [HttpGet] [Route("status/{Name}.json")] public async Task HandleGetListJson(string Name) { return Ok((await DetailedStatus(Name)).ToJson()); } private async Task DetailedStatus(string Name) { return (await GetSummaries()) .Select(d => d.Detailed) .FirstOrDefault(d => d.MachineName == Name); } } }