using System; using System.Collections.Generic; using Microsoft.AspNetCore.Diagnostics; using Microsoft.AspNetCore.Mvc; using SharedLibraryCore; using SharedLibraryCore.Dtos; using SharedLibraryCore.Interfaces; using System.Linq; using System.Threading; using System.Threading.Tasks; using Data.Models; using Microsoft.Extensions.Logging; using ILogger = Microsoft.Extensions.Logging.ILogger; namespace WebfrontCore.Controllers { public class HomeController : BaseController { private readonly ITranslationLookup _translationLookup; private readonly ILogger _logger; private readonly IServerDataViewer _serverDataViewer; private readonly ILookup _pluginTypeNames; public HomeController(ILogger logger, IManager manager, ITranslationLookup translationLookup, IServerDataViewer serverDataViewer, IEnumerable v1Plugins, IEnumerable v2Plugins) : base(manager) { _logger = logger; _translationLookup = translationLookup; _serverDataViewer = serverDataViewer; _pluginTypeNames = v1Plugins.Select(plugin => (plugin.GetType(), plugin.Name)) .Concat(v2Plugins.Select(plugin => (plugin.GetType(), plugin.Name))) .ToLookup(selector => selector.Item1, selector => selector.Name); } public async Task Index(Reference.Game? game = null, CancellationToken cancellationToken = default) { ViewBag.Description = Localization["WEBFRONT_DESCRIPTION_HOME"]; ViewBag.Title = Localization["WEBFRONT_HOME_TITLE"]; ViewBag.Keywords = Localization["WEBFRONT_KEWORDS_HOME"]; var servers = Manager.GetServers().Where(server => game is null || server.GameName == (Server.Game?)game) .ToList(); var (clientCount, time) = await _serverDataViewer.MaxConcurrentClientsAsync(gameCode: game, token: cancellationToken); var (count, recentCount) = await _serverDataViewer.ClientCountsAsync(gameCode: game, token: cancellationToken); var model = new IW4MAdminInfo { TotalAvailableClientSlots = servers.Sum(server => server.MaxClients), TotalOccupiedClientSlots = servers.SelectMany(server => server.GetClientsAsList()).Count(), TotalClientCount = count, RecentClientCount = recentCount, MaxConcurrentClients = clientCount ?? 0, MaxConcurrentClientsTime = time ?? DateTime.UtcNow, Game = game, ActiveServerGames = Manager.GetServers().Select(server => (Reference.Game)server.GameName).Distinct() .ToArray() }; return View(model); } public IActionResult Error() { var exceptionFeature = HttpContext.Features.Get(); _logger.LogError("[Webfront] {path} {message} {@exception}", exceptionFeature.Path, exceptionFeature.Error.Message, exceptionFeature.Error); ViewBag.Description = Localization["WEBFRONT_ERROR_DESC"]; ViewBag.Title = Localization["WEBFRONT_ERROR_TITLE"]; return View(exceptionFeature.Error); } public IActionResult ResponseStatusCode(int? statusCode = null) { return View(statusCode); } public IActionResult Help() { ViewBag.IsFluid = true; ViewBag.Title = Localization["WEBFRONT_NAV_HELP"]; ViewBag.CommandPrefix = Manager.GetApplicationSettings().Configuration().CommandPrefix; // we don't need to the name of the shared library assembly var commands = Manager.GetCommands() .Where(command => command.Permission <= Client.Level) .OrderByDescending(command => command.Permission) .GroupBy(command => { if (command.GetType().Name == "ScriptCommand") { return _translationLookup["WEBFRONT_HELP_SCRIPT_PLUGIN"]; } var assemblyName = command.GetType().Assembly.GetName().Name; if (assemblyName is "IW4MAdmin" or "SharedLibraryCore") { return _translationLookup["WEBFRONT_HELP_COMMAND_NATIVE"]; } var pluginType = command.GetType().Assembly.GetTypes() .FirstOrDefault(type => typeof(IPlugin).IsAssignableFrom(type) || typeof(IPluginV2).IsAssignableFrom(type)); return _pluginTypeNames[pluginType].FirstOrDefault() ?? _translationLookup["WEBFRONT_HELP_COMMAND_NATIVE"]; }) .Select(group => (group.Key, group.AsEnumerable())); return View(commands); } } }