All checks were successful
Tag and Release Lightless / tag-and-release (push) Successful in 2m27s
2.0.0 Changes: - Reworked shell finder UI with compact or list view with profile tags showing with the listing, allowing moderators to broadcast the syncshell as well to have it be used more. - Reworked user list in syncshell admin screen to have filter visible and moved away from table to its own thing, allowing to copy uid/note/alias when clicking on the name. - Reworked download bars and download box to make it look more modern, removed the jitter around, so it shouldn't vibrate around much. - Chat has been added to the top menu, working in Zone or in Syncshells to be used there. - Paired system has been revamped to make pausing and unpausing faster, and loading people should be faster as well. - Moved to the internal object table to have faster load times for users; people should load in faster - Compactor is running on a multi-threaded level instead of single-threaded; this should increase the speed of compacting files - Nameplate Service has been reworked so it wouldn't use the nameplate handler anymore. - Files can be resized when downloading to reduce load on users if they aren't compressed. (can be toggled to resize all). - Penumbra Collections are now only made when people are visible, reducing the load on boot-up when having many syncshells in your list. - Lightfinder plates have been moved away from using Nameplates, but will use an overlay. - Main UI has been changed a bit with a gradient, and on hover will glow up now. - Reworked Profile UI for Syncshell and Users to be more user-facing with more customizable items. - Reworked Settings UI to look more modern. - Performance should be better due to new systems that would dispose of the collections and better caching of items. Co-authored-by: defnotken <itsdefnotken@gmail.com> Co-authored-by: azyges <aaaaaa@aaa.aaa> Co-authored-by: choco <choco@patat.nl> Co-authored-by: cake <admin@cakeandbanana.nl> Co-authored-by: Minmoose <KennethBohr@outlook.com> Reviewed-on: #92
205 lines
8.0 KiB
C#
205 lines
8.0 KiB
C#
using LightlessSync.LightlessConfiguration;
|
|
using LightlessSync.Utils;
|
|
using Microsoft.Extensions.Hosting;
|
|
using Microsoft.Extensions.Logging;
|
|
using System.Collections.Concurrent;
|
|
using System.Globalization;
|
|
using System.Text;
|
|
|
|
namespace LightlessSync.Services;
|
|
|
|
public sealed class PerformanceCollectorService : IHostedService
|
|
{
|
|
private const string _counterSplit = "=>";
|
|
private readonly ILogger<PerformanceCollectorService> _logger;
|
|
private readonly LightlessConfigService _lightlessConfigService;
|
|
public ConcurrentDictionary<string, RollingList<(TimeOnly, long)>> PerformanceCounters { get; } = new(StringComparer.Ordinal);
|
|
private readonly CancellationTokenSource _periodicLogPruneTaskCts = new();
|
|
|
|
public PerformanceCollectorService(ILogger<PerformanceCollectorService> logger, LightlessConfigService lightlessConfigService)
|
|
{
|
|
_logger = logger;
|
|
_lightlessConfigService = lightlessConfigService;
|
|
}
|
|
|
|
public T LogPerformance<T>(object sender, LightlessInterpolatedStringHandler counterName, Func<T> func, int maxEntries = 10000)
|
|
{
|
|
if (!_lightlessConfigService.Current.LogPerformance) return func.Invoke();
|
|
|
|
var owner = sender.GetType().Name;
|
|
var counter = counterName.BuildMessage();
|
|
var cn = string.Concat(owner, _counterSplit, counter);
|
|
|
|
if (!PerformanceCounters.TryGetValue(cn, out var list))
|
|
list = PerformanceCounters[cn] = new(maxEntries);
|
|
|
|
var dt = DateTime.UtcNow.Ticks;
|
|
try
|
|
{
|
|
return func.Invoke();
|
|
}
|
|
finally
|
|
{
|
|
var elapsed = DateTime.UtcNow.Ticks - dt;
|
|
#if DEBUG
|
|
if (TimeSpan.FromTicks(elapsed) > TimeSpan.FromMilliseconds(10))
|
|
_logger.LogWarning(">10ms spike on {counterName}: {time}", cn, TimeSpan.FromTicks(elapsed));
|
|
#endif
|
|
list.Add((TimeOnly.FromDateTime(DateTime.Now), elapsed));
|
|
}
|
|
}
|
|
|
|
public void LogPerformance(object sender, LightlessInterpolatedStringHandler counterName, Action act, int maxEntries = 10000)
|
|
{
|
|
if (!_lightlessConfigService.Current.LogPerformance) { act.Invoke(); return; }
|
|
|
|
var owner = sender.GetType().Name;
|
|
var counter = counterName.BuildMessage();
|
|
var cn = string.Concat(owner, _counterSplit, counter);
|
|
|
|
if (!PerformanceCounters.TryGetValue(cn, out var list))
|
|
list = PerformanceCounters[cn] = new(maxEntries);
|
|
|
|
var dt = DateTime.UtcNow.Ticks;
|
|
try
|
|
{
|
|
act.Invoke();
|
|
}
|
|
finally
|
|
{
|
|
var elapsed = DateTime.UtcNow.Ticks - dt;
|
|
#if DEBUG
|
|
if (TimeSpan.FromTicks(elapsed) > TimeSpan.FromMilliseconds(10))
|
|
_logger.LogWarning(">10ms spike on {counterName}: {time}", cn, TimeSpan.FromTicks(elapsed));
|
|
#endif
|
|
list.Add((TimeOnly.FromDateTime(DateTime.Now), elapsed));
|
|
}
|
|
}
|
|
|
|
public Task StartAsync(CancellationToken cancellationToken)
|
|
{
|
|
_logger.LogInformation("Starting PerformanceCollectorService");
|
|
_ = Task.Run(PeriodicLogPrune, _periodicLogPruneTaskCts.Token);
|
|
_logger.LogInformation("Started PerformanceCollectorService");
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
public Task StopAsync(CancellationToken cancellationToken)
|
|
{
|
|
_periodicLogPruneTaskCts.Cancel();
|
|
_periodicLogPruneTaskCts.Dispose();
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
internal void PrintPerformanceStats(int limitBySeconds = 0)
|
|
{
|
|
if (!_lightlessConfigService.Current.LogPerformance)
|
|
{
|
|
_logger.LogWarning("Performance counters are disabled");
|
|
}
|
|
|
|
StringBuilder sb = new();
|
|
if (limitBySeconds > 0)
|
|
{
|
|
sb.AppendLine($"Performance Metrics over the past {limitBySeconds} seconds of each counter");
|
|
}
|
|
else
|
|
{
|
|
sb.AppendLine("Performance metrics over total lifetime of each counter");
|
|
}
|
|
var data = PerformanceCounters.ToList();
|
|
var longestCounterName = data.OrderByDescending(d => d.Key.Length).First().Key.Length + 2;
|
|
sb.Append("-Last".PadRight(15, '-'));
|
|
sb.Append('|');
|
|
sb.Append("-Max".PadRight(15, '-'));
|
|
sb.Append('|');
|
|
sb.Append("-Average".PadRight(15, '-'));
|
|
sb.Append('|');
|
|
sb.Append("-Last Update".PadRight(15, '-'));
|
|
sb.Append('|');
|
|
sb.Append("-Entries".PadRight(10, '-'));
|
|
sb.Append('|');
|
|
sb.Append("-Counter Name".PadRight(longestCounterName, '-'));
|
|
sb.AppendLine();
|
|
var orderedData = data.OrderBy(k => k.Key, StringComparer.OrdinalIgnoreCase).ToList();
|
|
var previousCaller = SplitCounterKey(orderedData[0].Key).Owner;
|
|
foreach (var entry in orderedData)
|
|
{
|
|
var newCaller = SplitCounterKey(entry.Key).Owner;
|
|
if (!string.Equals(previousCaller, newCaller, StringComparison.Ordinal))
|
|
{
|
|
DrawSeparator(sb, longestCounterName);
|
|
}
|
|
|
|
var pastEntries = limitBySeconds > 0 ? entry.Value.Where(e => e.Item1.AddMinutes(limitBySeconds / 60.0d) >= TimeOnly.FromDateTime(DateTime.Now)).ToList() : [.. entry.Value];
|
|
|
|
if (pastEntries.Any())
|
|
{
|
|
sb.Append((" " + TimeSpan.FromTicks(pastEntries.LastOrDefault() == default ? 0 : pastEntries[^1].Item2).TotalMilliseconds.ToString("0.00000", CultureInfo.InvariantCulture)).PadRight(15));
|
|
sb.Append('|');
|
|
sb.Append((" " + TimeSpan.FromTicks(pastEntries.Max(m => m.Item2)).TotalMilliseconds.ToString("0.00000", CultureInfo.InvariantCulture)).PadRight(15));
|
|
sb.Append('|');
|
|
sb.Append((" " + TimeSpan.FromTicks((long)pastEntries.Average(m => m.Item2)).TotalMilliseconds.ToString("0.00000", CultureInfo.InvariantCulture)).PadRight(15));
|
|
sb.Append('|');
|
|
sb.Append((" " + (pastEntries.LastOrDefault() == default ? "-" : pastEntries[^1].Item1.ToString("HH:mm:ss.ffff", CultureInfo.InvariantCulture))).PadRight(15, ' '));
|
|
sb.Append('|');
|
|
sb.Append((" " + pastEntries.Count).PadRight(10));
|
|
sb.Append('|');
|
|
sb.Append(' ').Append(entry.Key);
|
|
sb.AppendLine();
|
|
}
|
|
|
|
previousCaller = newCaller;
|
|
}
|
|
|
|
DrawSeparator(sb, longestCounterName);
|
|
|
|
_logger.LogInformation("{perf}", sb.ToString());
|
|
}
|
|
|
|
private static (string Owner, string Counter) SplitCounterKey(string cn)
|
|
{
|
|
var parts = cn.Split(_counterSplit, 2, StringSplitOptions.None);
|
|
return (parts[0], parts.Length > 1 ? parts[1] : string.Empty);
|
|
}
|
|
|
|
private static void DrawSeparator(StringBuilder sb, int longestCounterName)
|
|
{
|
|
sb.Append("".PadRight(15, '-'));
|
|
sb.Append('+');
|
|
sb.Append("".PadRight(15, '-'));
|
|
sb.Append('+');
|
|
sb.Append("".PadRight(15, '-'));
|
|
sb.Append('+');
|
|
sb.Append("".PadRight(15, '-'));
|
|
sb.Append('+');
|
|
sb.Append("".PadRight(10, '-'));
|
|
sb.Append('+');
|
|
sb.Append("".PadRight(longestCounterName, '-'));
|
|
sb.AppendLine();
|
|
}
|
|
|
|
private async Task PeriodicLogPrune()
|
|
{
|
|
while (!_periodicLogPruneTaskCts.Token.IsCancellationRequested)
|
|
{
|
|
await Task.Delay(TimeSpan.FromMinutes(10), _periodicLogPruneTaskCts.Token).ConfigureAwait(false);
|
|
|
|
foreach (var entries in PerformanceCounters.ToList())
|
|
{
|
|
try
|
|
{
|
|
var last = entries.Value.ToList()[^1];
|
|
if (last.Item1.AddMinutes(10) < TimeOnly.FromDateTime(DateTime.Now) && !PerformanceCounters.TryRemove(entries.Key, out _))
|
|
{
|
|
_logger.LogDebug("Could not remove performance counter {counter}", entries.Key);
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
_logger.LogWarning(e, "Error removing performance counter {counter}", entries.Key);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} |