2020-05-09 05:12:51 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2020-11-19 22:56:30 +00:00
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Net;
|
2021-02-17 05:46:05 +00:00
|
|
|
|
using System.Reflection;
|
|
|
|
|
using System.Text;
|
2020-05-09 05:12:51 +00:00
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
|
using Microsoft.Extensions.Logging;
|
2020-11-19 22:56:30 +00:00
|
|
|
|
using Nettle;
|
2021-02-17 05:46:05 +00:00
|
|
|
|
using Wabbajack.Common;
|
2020-05-09 05:12:51 +00:00
|
|
|
|
using Wabbajack.Common.StatusFeed;
|
|
|
|
|
using Wabbajack.Server;
|
|
|
|
|
using Wabbajack.Server.DataLayer;
|
|
|
|
|
using Wabbajack.Server.DTOs;
|
2020-11-19 22:56:30 +00:00
|
|
|
|
using Wabbajack.Server.Services;
|
2020-05-09 05:12:51 +00:00
|
|
|
|
|
|
|
|
|
namespace Wabbajack.BuildServer.Controllers
|
|
|
|
|
{
|
|
|
|
|
[Route("/heartbeat")]
|
|
|
|
|
public class Heartbeat : ControllerBase
|
|
|
|
|
{
|
|
|
|
|
static Heartbeat()
|
|
|
|
|
{
|
|
|
|
|
_startTime = DateTime.Now;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
private static DateTime _startTime;
|
2020-11-19 22:56:30 +00:00
|
|
|
|
|
|
|
|
|
private QuickSync _quickSync;
|
|
|
|
|
private ListValidator _listValidator;
|
2020-05-09 05:12:51 +00:00
|
|
|
|
|
2020-11-19 22:56:30 +00:00
|
|
|
|
|
|
|
|
|
public Heartbeat(ILogger<Heartbeat> logger, SqlService sql, GlobalInformation globalInformation, QuickSync quickSync, ListValidator listValidator)
|
2020-05-09 05:12:51 +00:00
|
|
|
|
{
|
|
|
|
|
_globalInformation = globalInformation;
|
|
|
|
|
_sql = sql;
|
|
|
|
|
_logger = logger;
|
2020-11-19 22:56:30 +00:00
|
|
|
|
_quickSync = quickSync;
|
|
|
|
|
_listValidator = listValidator;
|
2020-05-09 05:12:51 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private const int MAX_LOG_SIZE = 128;
|
2021-06-24 23:01:03 +00:00
|
|
|
|
private static List<string> Log = new();
|
2020-05-09 05:12:51 +00:00
|
|
|
|
private GlobalInformation _globalInformation;
|
|
|
|
|
private SqlService _sql;
|
|
|
|
|
private ILogger<Heartbeat> _logger;
|
|
|
|
|
|
|
|
|
|
public static void AddToLog(IStatusMessage msg)
|
|
|
|
|
{
|
|
|
|
|
lock (Log)
|
|
|
|
|
{
|
|
|
|
|
Log.Add(msg.ToString());
|
|
|
|
|
if (Log.Count > MAX_LOG_SIZE)
|
|
|
|
|
Log.RemoveAt(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[HttpGet]
|
|
|
|
|
public async Task<IActionResult> GetHeartbeat()
|
|
|
|
|
{
|
|
|
|
|
return Ok(new HeartbeatResult
|
|
|
|
|
{
|
|
|
|
|
Uptime = DateTime.Now - _startTime,
|
|
|
|
|
LastNexusUpdate = _globalInformation.TimeSinceLastNexusSync,
|
|
|
|
|
});
|
|
|
|
|
}
|
2020-11-19 22:56:30 +00:00
|
|
|
|
|
|
|
|
|
private static readonly Func<object, string> HandleGetReport = NettleEngine.GetCompiler().Compile(@"
|
|
|
|
|
<html><body>
|
|
|
|
|
<h2>Server Status</h2>
|
|
|
|
|
|
|
|
|
|
<h3>Service Overview ({{services.Length}}):</h3>
|
|
|
|
|
<ul>
|
|
|
|
|
{{each $.services }}
|
|
|
|
|
{{if $.IsLate}}
|
2020-12-31 06:44:42 +00:00
|
|
|
|
<li><a href='/heartbeat/report/services/{{$.Name}}.html'><b>{{$.Name}} - {{$.Time}} - {{$.MaxTime}}</b></a></li>
|
2020-11-19 22:56:30 +00:00
|
|
|
|
{{else}}
|
2020-12-31 06:44:42 +00:00
|
|
|
|
<li><a href='/heartbeat/report/services/{{$.Name}}.html'>{{$.Name}} - {{$.Time}} - {{$.MaxTime}}</a></li>
|
2020-11-19 22:56:30 +00:00
|
|
|
|
{{/if}}
|
|
|
|
|
{{/each}}
|
|
|
|
|
</ul>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<h3>Lists ({{lists.Length}}):</h3>
|
|
|
|
|
<ul>
|
|
|
|
|
{{each $.lists }}
|
2021-06-25 03:56:02 +00:00
|
|
|
|
<li><a href='/lists/status/{{$.Name}}.html'>{{$.Name}}</a> - {{$.Time}} {{$.FailMessage}}</li>
|
2020-11-19 22:56:30 +00:00
|
|
|
|
{{/each}}
|
|
|
|
|
</ul>
|
|
|
|
|
</body></html>
|
|
|
|
|
");
|
|
|
|
|
|
|
|
|
|
[HttpGet("report")]
|
|
|
|
|
public async Task<ContentResult> Report()
|
|
|
|
|
{
|
|
|
|
|
var response = HandleGetReport(new
|
|
|
|
|
{
|
|
|
|
|
services = (await _quickSync.Report())
|
2020-11-20 05:00:07 +00:00
|
|
|
|
.Select(s => new {Name = s.Key.Name, Time = s.Value.LastRunTime, MaxTime = s.Value.Delay, IsLate = s.Value.LastRunTime > s.Value.Delay})
|
2020-11-19 22:56:30 +00:00
|
|
|
|
.OrderBy(s => s.Name)
|
|
|
|
|
.ToArray(),
|
2021-06-25 03:56:02 +00:00
|
|
|
|
lists = _listValidator.ValidationInfo.Select(s => new
|
|
|
|
|
{
|
|
|
|
|
Name = s.Key,
|
|
|
|
|
Time = s.Value.ValidationTime,
|
|
|
|
|
FailMessage = s.Value.Detailed.HasFailures ? "Failed" : ""
|
|
|
|
|
})
|
2020-11-19 22:56:30 +00:00
|
|
|
|
.OrderBy(l => l.Name)
|
|
|
|
|
.ToArray()
|
|
|
|
|
});
|
|
|
|
|
return new ContentResult
|
|
|
|
|
{
|
|
|
|
|
ContentType = "text/html",
|
|
|
|
|
StatusCode = (int) HttpStatusCode.OK,
|
|
|
|
|
Content = response
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
}
|
2020-12-31 06:44:42 +00:00
|
|
|
|
|
|
|
|
|
private static readonly Func<object, string> HandleGetServiceReport = NettleEngine.GetCompiler().Compile(@"
|
|
|
|
|
<html><body>
|
|
|
|
|
<h2>Service Status: {{Name}} {{TimeSinceLastRun}}</h2>
|
|
|
|
|
|
|
|
|
|
<h3>Service Overview ({{ActiveWorkQueue.Length}}):</h3>
|
|
|
|
|
<ul>
|
|
|
|
|
{{each $.ActiveWorkQueue }}
|
|
|
|
|
<li>{{$.Name}} {{$.Time}}</li>
|
|
|
|
|
{{/each}}
|
|
|
|
|
</ul>
|
2020-05-09 05:12:51 +00:00
|
|
|
|
|
2020-12-31 06:44:42 +00:00
|
|
|
|
</body></html>
|
|
|
|
|
");
|
2020-05-09 05:12:51 +00:00
|
|
|
|
|
2020-12-31 06:44:42 +00:00
|
|
|
|
[HttpGet("report/services/{serviceName}.html")]
|
|
|
|
|
public async Task<ContentResult> ReportServiceStatus(string serviceName)
|
|
|
|
|
{
|
|
|
|
|
var services = await _quickSync.Report();
|
|
|
|
|
var info = services.First(kvp => kvp.Key.Name == serviceName);
|
2020-05-09 05:12:51 +00:00
|
|
|
|
|
2020-12-31 06:44:42 +00:00
|
|
|
|
var response = HandleGetServiceReport(new
|
|
|
|
|
{
|
|
|
|
|
Name = info.Key.Name,
|
|
|
|
|
TimeSinceLastRun = DateTime.UtcNow - info.Value.LastRunTime,
|
|
|
|
|
ActiveWorkQueue = info.Value.ActiveWork.Select(p => new
|
|
|
|
|
{
|
|
|
|
|
Name = p.Item1,
|
|
|
|
|
Time = DateTime.UtcNow - p.Item2
|
|
|
|
|
}).OrderByDescending(kp => kp.Time)
|
|
|
|
|
.ToArray()
|
|
|
|
|
|
|
|
|
|
});
|
|
|
|
|
return new ContentResult
|
|
|
|
|
{
|
|
|
|
|
ContentType = "text/html",
|
|
|
|
|
StatusCode = (int) HttpStatusCode.OK,
|
|
|
|
|
Content = response
|
|
|
|
|
};
|
|
|
|
|
}
|
2020-05-09 05:12:51 +00:00
|
|
|
|
}
|
|
|
|
|
}
|