2020-05-09 03:56:06 +00:00
|
|
|
|
using System;
|
2020-06-15 04:05:00 +00:00
|
|
|
|
using System.Collections.Generic;
|
2020-05-09 13:04:38 +00:00
|
|
|
|
using System.Globalization;
|
2021-02-17 05:46:05 +00:00
|
|
|
|
using System.IO;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
using System.Linq;
|
2020-06-15 04:05:00 +00:00
|
|
|
|
using System.Net;
|
2021-02-17 05:46:05 +00:00
|
|
|
|
using System.Reflection;
|
|
|
|
|
using System.Text;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
using System.Threading.Tasks;
|
2021-02-17 05:46:05 +00:00
|
|
|
|
using Microsoft.AspNetCore.Http;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
|
using Microsoft.Extensions.Logging;
|
2020-06-15 04:05:00 +00:00
|
|
|
|
using Nettle;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
using Wabbajack.Common;
|
2020-07-07 12:29:05 +00:00
|
|
|
|
using Wabbajack.Server;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
using Wabbajack.Server.DataLayer;
|
|
|
|
|
using Wabbajack.Server.DTOs;
|
2021-02-18 05:44:54 +00:00
|
|
|
|
using Wabbajack.Server.Services;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
using WebSocketSharp;
|
|
|
|
|
using LogLevel = Microsoft.Extensions.Logging.LogLevel;
|
|
|
|
|
|
|
|
|
|
namespace Wabbajack.BuildServer.Controllers
|
|
|
|
|
{
|
|
|
|
|
[ApiController]
|
|
|
|
|
[Route("/metrics")]
|
|
|
|
|
public class MetricsController : ControllerBase
|
|
|
|
|
{
|
|
|
|
|
private SqlService _sql;
|
|
|
|
|
private ILogger<MetricsController> _logger;
|
2021-02-18 05:44:54 +00:00
|
|
|
|
private MetricsKeyCache _keyCache;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
|
2021-02-18 05:44:54 +00:00
|
|
|
|
public MetricsController(ILogger<MetricsController> logger, SqlService sql, MetricsKeyCache keyCache)
|
2020-05-09 03:56:06 +00:00
|
|
|
|
{
|
|
|
|
|
_sql = sql;
|
|
|
|
|
_logger = logger;
|
2021-02-18 05:44:54 +00:00
|
|
|
|
_keyCache = keyCache;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[HttpGet]
|
2020-05-09 13:04:38 +00:00
|
|
|
|
[Route("{subject}/{value}")]
|
|
|
|
|
public async Task<Result> LogMetricAsync(string subject, string value)
|
2020-05-09 03:56:06 +00:00
|
|
|
|
{
|
|
|
|
|
var date = DateTime.UtcNow;
|
2021-02-18 05:44:54 +00:00
|
|
|
|
var metricsKey = Request.Headers[Consts.MetricsKeyHeader].FirstOrDefault();
|
|
|
|
|
if (metricsKey != null)
|
|
|
|
|
await _keyCache.AddKey(metricsKey);
|
2021-02-17 05:46:05 +00:00
|
|
|
|
|
|
|
|
|
// Used in tests
|
2021-02-18 05:44:54 +00:00
|
|
|
|
if (value == "Default" || value == "untitled" || subject == "failed_download" || Guid.TryParse(value, out _))
|
2021-02-17 05:46:05 +00:00
|
|
|
|
return new Result { Timestamp = date};
|
|
|
|
|
|
2021-02-18 05:44:54 +00:00
|
|
|
|
await Log(date, subject, value, metricsKey);
|
2020-05-09 03:56:06 +00:00
|
|
|
|
return new Result { Timestamp = date};
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-09 13:04:38 +00:00
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("report/{subject}")]
|
2020-08-26 04:03:43 +00:00
|
|
|
|
[ResponseCache(Duration = 60 * 60)]
|
2020-05-09 13:04:38 +00:00
|
|
|
|
public async Task<IActionResult> MetricsReport(string subject)
|
|
|
|
|
{
|
2020-11-20 05:00:07 +00:00
|
|
|
|
var metrics = (await _sql.MetricsReport(subject)).ToList();
|
|
|
|
|
var labels = metrics.GroupBy(m => m.Date)
|
|
|
|
|
.OrderBy(m => m.Key)
|
|
|
|
|
.Select(m => m.Key)
|
|
|
|
|
.ToArray();
|
|
|
|
|
var labelStrings = labels.Select(l => l.ToString("MM-dd-yyy")).ToList();
|
|
|
|
|
var results = metrics
|
2020-05-09 13:04:38 +00:00
|
|
|
|
.GroupBy(m => m.Subject)
|
2020-11-20 05:00:07 +00:00
|
|
|
|
.Select(g =>
|
2020-05-09 13:04:38 +00:00
|
|
|
|
{
|
2020-11-20 05:00:07 +00:00
|
|
|
|
var indexed = g.ToDictionary(m => m.Date, m => m.Count);
|
|
|
|
|
return new MetricResult
|
|
|
|
|
{
|
|
|
|
|
SeriesName = g.Key,
|
|
|
|
|
Labels = labelStrings,
|
|
|
|
|
Values = labels.Select(l => indexed.TryGetValue(l, out var found) ? found : 0).ToList()
|
|
|
|
|
};
|
2020-05-09 13:04:38 +00:00
|
|
|
|
});
|
|
|
|
|
return Ok(results.ToList());
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-07 12:29:05 +00:00
|
|
|
|
[HttpGet]
|
2020-07-07 20:17:49 +00:00
|
|
|
|
[Route("badge/{name}/total_installs_badge.json")]
|
|
|
|
|
public async Task<IActionResult> TotalInstallsBadge(string name)
|
2020-07-07 12:29:05 +00:00
|
|
|
|
{
|
2020-07-07 20:17:49 +00:00
|
|
|
|
var results = await _sql.TotalInstalls(name);
|
2020-07-07 12:29:05 +00:00
|
|
|
|
|
|
|
|
|
Response.ContentType = "application/json";
|
2020-07-07 20:17:49 +00:00
|
|
|
|
|
|
|
|
|
return Ok(results == 0
|
|
|
|
|
? new Badge($"Modlist {name} not found!", "Error") {color = "red"}
|
2021-02-17 05:46:05 +00:00
|
|
|
|
: new Badge("Installations: ", "____") {color = "green"});
|
2020-07-07 12:29:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-07-07 20:17:49 +00:00
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("badge/{name}/unique_installs_badge.json")]
|
|
|
|
|
public async Task<IActionResult> UniqueInstallsBadge(string name)
|
|
|
|
|
{
|
|
|
|
|
var results = await _sql.UniqueInstalls(name);
|
|
|
|
|
|
|
|
|
|
Response.ContentType = "application/json";
|
|
|
|
|
|
|
|
|
|
return Ok(results == 0
|
|
|
|
|
? new Badge($"Modlist {name} not found!", "Error") {color = "red"}
|
2021-02-17 05:46:05 +00:00
|
|
|
|
: new Badge("Installations: ", "____"){color = "green"}) ;
|
2020-07-07 20:17:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-06-15 04:05:00 +00:00
|
|
|
|
private static readonly Func<object, string> ReportTemplate = NettleEngine.GetCompiler().Compile(@"
|
|
|
|
|
<html><body>
|
|
|
|
|
<h2>Tar Report for {{$.key}}</h2>
|
|
|
|
|
<h3>Ban Status: {{$.status}}</h3>
|
|
|
|
|
<table>
|
|
|
|
|
{{each $.log }}
|
|
|
|
|
<tr>
|
|
|
|
|
<td>{{$.Timestamp}}</td>
|
|
|
|
|
<td>{{$.Path}}</td>
|
|
|
|
|
<td>{{$.Key}}</td>
|
|
|
|
|
</tr>
|
|
|
|
|
{{/each}}
|
|
|
|
|
</table>
|
|
|
|
|
</body></html>
|
|
|
|
|
");
|
|
|
|
|
|
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("tarlog/{key}")]
|
|
|
|
|
public async Task<IActionResult> TarLog(string key)
|
|
|
|
|
{
|
|
|
|
|
var isTarKey = await _sql.IsTarKey(key);
|
|
|
|
|
|
|
|
|
|
List<(DateTime, string, string)> report = new List<(DateTime, string, string)>();
|
|
|
|
|
|
|
|
|
|
if (isTarKey) report = await _sql.FullTarReport(key);
|
|
|
|
|
|
|
|
|
|
var response = ReportTemplate(new
|
|
|
|
|
{
|
|
|
|
|
key = key,
|
|
|
|
|
status = isTarKey ? "BANNED" : "NOT BANNED",
|
|
|
|
|
log = report.Select(entry => new
|
|
|
|
|
{
|
|
|
|
|
Timestamp = entry.Item1,
|
|
|
|
|
Path = entry.Item2,
|
|
|
|
|
Key = entry.Item3
|
|
|
|
|
}).ToList()
|
|
|
|
|
});
|
|
|
|
|
return new ContentResult
|
|
|
|
|
{
|
|
|
|
|
ContentType = "text/html",
|
|
|
|
|
StatusCode = (int) HttpStatusCode.OK,
|
|
|
|
|
Content = response
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-09 03:56:06 +00:00
|
|
|
|
private async Task Log(DateTime timestamp, string action, string subject, string metricsKey = null)
|
|
|
|
|
{
|
2020-08-12 04:25:12 +00:00
|
|
|
|
//_logger.Log(LogLevel.Information, $"Log - {timestamp} {action} {subject} {metricsKey}");
|
2020-05-09 03:56:06 +00:00
|
|
|
|
await _sql.IngestMetric(new Metric
|
|
|
|
|
{
|
|
|
|
|
Timestamp = timestamp, Action = action, Subject = subject, MetricsKey = metricsKey
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class Result
|
|
|
|
|
{
|
|
|
|
|
public DateTime Timestamp { get; set; }
|
|
|
|
|
}
|
2021-02-17 05:46:05 +00:00
|
|
|
|
|
|
|
|
|
class TotalListTemplateData
|
|
|
|
|
{
|
|
|
|
|
public string Title { get; set; }
|
|
|
|
|
public long Total { get; set; }
|
|
|
|
|
public Item[] Items { get; set; }
|
|
|
|
|
|
|
|
|
|
public class Item
|
|
|
|
|
{
|
|
|
|
|
public long Count { get; set; }
|
|
|
|
|
public string Title { get; set; }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static Func<object, string> _totalListTemplate;
|
|
|
|
|
|
2021-02-18 05:44:54 +00:00
|
|
|
|
|
2021-02-17 05:46:05 +00:00
|
|
|
|
private static Func<object, string> TotalListTemplate
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
if (_totalListTemplate == null)
|
|
|
|
|
{
|
|
|
|
|
var resource = Assembly.GetExecutingAssembly()
|
|
|
|
|
.GetManifestResourceStream("Wabbajack.Server.Controllers.Templates.TotalListTemplate.html")!
|
|
|
|
|
.ReadAll();
|
|
|
|
|
_totalListTemplate = NettleEngine.GetCompiler().Compile(Encoding.UTF8.GetString(resource));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return _totalListTemplate;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-02-18 05:44:54 +00:00
|
|
|
|
|
|
|
|
|
|
2021-02-17 05:46:05 +00:00
|
|
|
|
|
|
|
|
|
[HttpGet("total_installs.html")]
|
|
|
|
|
[ResponseCache(Duration = 60 * 60)]
|
|
|
|
|
public async Task<ContentResult> TotalInstalls()
|
|
|
|
|
{
|
|
|
|
|
var data = await _sql.GetTotalInstalls();
|
|
|
|
|
var result = TotalListTemplate(new TotalListTemplateData
|
|
|
|
|
{
|
|
|
|
|
Title = "Total Installs",
|
|
|
|
|
Total = data.Sum(d => d.Item2),
|
|
|
|
|
Items = data.Select(d => new TotalListTemplateData.Item {Title = d.Item1, Count = d.Item2})
|
|
|
|
|
.ToArray()
|
|
|
|
|
});
|
|
|
|
|
return new ContentResult {
|
|
|
|
|
ContentType = "text/html",
|
|
|
|
|
StatusCode = (int)HttpStatusCode.OK,
|
|
|
|
|
Content = result};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[HttpGet("total_unique_installs.html")]
|
|
|
|
|
[ResponseCache(Duration = 60 * 60)]
|
|
|
|
|
public async Task<ContentResult> TotalUniqueInstalls()
|
|
|
|
|
{
|
|
|
|
|
var data = await _sql.GetTotalUniqueInstalls();
|
|
|
|
|
var result = TotalListTemplate(new TotalListTemplateData
|
|
|
|
|
{
|
|
|
|
|
Title = "Total Unique Installs",
|
|
|
|
|
Total = data.Sum(d => d.Item2),
|
|
|
|
|
Items = data.Select(d => new TotalListTemplateData.Item {Title = d.Item1, Count = d.Item2})
|
|
|
|
|
.ToArray()
|
|
|
|
|
});
|
|
|
|
|
return new ContentResult {
|
|
|
|
|
ContentType = "text/html",
|
|
|
|
|
StatusCode = (int)HttpStatusCode.OK,
|
|
|
|
|
Content = result};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[HttpGet("dump.json")]
|
|
|
|
|
public async Task<IActionResult> DataDump()
|
|
|
|
|
{
|
|
|
|
|
return Ok(await _sql.MetricsDump().ToArrayAsync());
|
|
|
|
|
}
|
2021-02-18 05:44:54 +00:00
|
|
|
|
|
2020-05-09 03:56:06 +00:00
|
|
|
|
}
|
|
|
|
|
}
|