IW4M-Admin/WebfrontCore/Controllers/Client/ClientController.cs

283 lines
12 KiB
C#
Raw Normal View History

2018-02-21 20:29:23 -05:00
using Microsoft.AspNetCore.Mvc;
2018-04-08 02:44:42 -04:00
using SharedLibraryCore;
using SharedLibraryCore.Database.Models;
2018-04-08 02:44:42 -04:00
using SharedLibraryCore.Dtos;
using SharedLibraryCore.Dtos.Meta.Responses;
move all the deployment setup into 2.4 pr (#85) * don't run build commands in release * fix test file * Set up CI with Azure Pipelines [skip ci] * Include fonts and fix automessage hidden command * more project changes * migration from bower to libman * more lib man changes * project update for sneaky commands * add missing canvas.js dep update projects not to have stupid extra dlls include in previous * update pipeline file * update post publish script and pipeline definition * fix broken yaml * move encoding conversion to seperate script * remove extra uneeded rank icons remove garbage language files being created remove frontend lib when done * fix publish script path * grab localizations through powershell * fix broken batch :shrug: * actually fixed * only include runtime compilation in debug mode for webfront * don't deploy un minified css use full jquery version * add step to download the scss for open iconic change the font path * update mkdir for iconic path * don't include old iconic css * correct font path for real now * copy script plugins * lots of changes for deployment * build the projects * use projectdir instead of solution dir * nerf script commands plugin fix live radar left over command * actually kill script command post build * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * fix the font file copy (I think) * maybe fix delete folder issue * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines
2019-12-02 16:52:36 -05:00
using SharedLibraryCore.Interfaces;
using SharedLibraryCore.QueryHelper;
2018-02-21 20:29:23 -05:00
using System;
using System.Collections.Generic;
using System.Linq;
2022-03-23 09:43:57 -04:00
using System.Threading;
2018-02-21 20:29:23 -05:00
using System.Threading.Tasks;
using Data.Models;
2022-07-20 11:32:26 -04:00
using SharedLibraryCore.Services;
using Stats.Config;
using WebfrontCore.Permissions;
2023-01-23 17:38:16 -05:00
using WebfrontCore.QueryHelpers.Models;
using WebfrontCore.ViewComponents;
2018-02-21 20:29:23 -05:00
namespace WebfrontCore.Controllers
{
2018-03-06 02:22:19 -05:00
public class ClientController : BaseController
2018-02-21 20:29:23 -05:00
{
2022-03-23 09:43:57 -04:00
private readonly IMetaServiceV2 _metaService;
2022-01-28 18:28:49 -05:00
private readonly StatsConfiguration _config;
2022-04-19 19:43:58 -04:00
private readonly IGeoLocationService _geoLocationService;
2022-07-20 11:32:26 -04:00
private readonly ClientService _clientService;
private readonly IInteractionRegistration _interactionRegistration;
2023-01-23 17:38:16 -05:00
private readonly IResourceQueryHelper<ClientResourceRequest, ClientResourceResponse> _clientResourceHelper;
move all the deployment setup into 2.4 pr (#85) * don't run build commands in release * fix test file * Set up CI with Azure Pipelines [skip ci] * Include fonts and fix automessage hidden command * more project changes * migration from bower to libman * more lib man changes * project update for sneaky commands * add missing canvas.js dep update projects not to have stupid extra dlls include in previous * update pipeline file * update post publish script and pipeline definition * fix broken yaml * move encoding conversion to seperate script * remove extra uneeded rank icons remove garbage language files being created remove frontend lib when done * fix publish script path * grab localizations through powershell * fix broken batch :shrug: * actually fixed * only include runtime compilation in debug mode for webfront * don't deploy un minified css use full jquery version * add step to download the scss for open iconic change the font path * update mkdir for iconic path * don't include old iconic css * correct font path for real now * copy script plugins * lots of changes for deployment * build the projects * use projectdir instead of solution dir * nerf script commands plugin fix live radar left over command * actually kill script command post build * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * fix the font file copy (I think) * maybe fix delete folder issue * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines
2019-12-02 16:52:36 -05:00
2022-04-19 19:43:58 -04:00
public ClientController(IManager manager, IMetaServiceV2 metaService, StatsConfiguration config,
2023-01-23 17:38:16 -05:00
IGeoLocationService geoLocationService, ClientService clientService,
IInteractionRegistration interactionRegistration,
IResourceQueryHelper<ClientResourceRequest, ClientResourceResponse> clientResourceHelper) : base(manager)
{
_metaService = metaService;
2022-01-28 18:28:49 -05:00
_config = config;
2022-04-19 19:43:58 -04:00
_geoLocationService = geoLocationService;
2022-07-20 11:32:26 -04:00
_clientService = clientService;
_interactionRegistration = interactionRegistration;
2023-01-23 17:38:16 -05:00
_clientResourceHelper = clientResourceHelper;
move all the deployment setup into 2.4 pr (#85) * don't run build commands in release * fix test file * Set up CI with Azure Pipelines [skip ci] * Include fonts and fix automessage hidden command * more project changes * migration from bower to libman * more lib man changes * project update for sneaky commands * add missing canvas.js dep update projects not to have stupid extra dlls include in previous * update pipeline file * update post publish script and pipeline definition * fix broken yaml * move encoding conversion to seperate script * remove extra uneeded rank icons remove garbage language files being created remove frontend lib when done * fix publish script path * grab localizations through powershell * fix broken batch :shrug: * actually fixed * only include runtime compilation in debug mode for webfront * don't deploy un minified css use full jquery version * add step to download the scss for open iconic change the font path * update mkdir for iconic path * don't include old iconic css * correct font path for real now * copy script plugins * lots of changes for deployment * build the projects * use projectdir instead of solution dir * nerf script commands plugin fix live radar left over command * actually kill script command post build * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * fix the font file copy (I think) * maybe fix delete folder issue * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines * Update azure-pipelines.yml for Azure Pipelines
2019-12-02 16:52:36 -05:00
}
2022-04-19 19:43:58 -04:00
[Obsolete]
public IActionResult ProfileAsync(int id, MetaType? metaFilterType,
CancellationToken token = default) => RedirectToAction("Profile", "Client", new
{ id, metaFilterType });
2022-04-19 19:43:58 -04:00
public async Task<IActionResult> Profile(int id, MetaType? metaFilterType, CancellationToken token = default)
2018-02-21 20:29:23 -05:00
{
2018-03-06 02:22:19 -05:00
var client = await Manager.GetClientService().Get(id);
if (client == null)
{
return NotFound();
}
2018-04-08 02:44:42 -04:00
2022-02-22 18:09:50 -05:00
var activePenalties = await Manager.GetPenaltyService().GetActivePenaltiesAsync(client.AliasLinkId,
client.CurrentAliasId, client.NetworkId, client.GameName, client.IPAddress);
2022-02-01 19:20:29 -05:00
var persistentMetaTask = new[]
{
2022-04-19 19:43:58 -04:00
_metaService.GetPersistentMetaByLookup(EFMeta.ClientTagV2, EFMeta.ClientTagNameV2, client.ClientId,
token),
2022-07-20 11:32:26 -04:00
_metaService.GetPersistentMeta("GravatarEmail", client.ClientId, token),
2022-02-01 19:20:29 -05:00
};
var persistentMeta = await Task.WhenAll(persistentMetaTask);
var tag = persistentMeta[0];
var gravatar = persistentMeta[1];
2022-07-20 11:32:26 -04:00
var note = await _metaService.GetPersistentMetaValue<ClientNoteMetaResponse>("ClientNotes", client.ClientId,
token);
2022-02-01 19:20:29 -05:00
2022-03-23 09:43:57 -04:00
if (tag?.Value != null)
{
2022-03-23 09:43:57 -04:00
client.SetAdditionalProperty(EFMeta.ClientTagV2, tag.Value);
}
2022-07-20 12:39:46 -04:00
if (!string.IsNullOrWhiteSpace(note?.Note))
2022-07-20 11:32:26 -04:00
{
note.OriginEntityName = await _clientService.GetClientNameById(note.OriginEntityId);
}
var interactions =
await _interactionRegistration.GetInteractions("Webfront::Profile", id, client.GameName, token);
// even though we haven't set their level to "banned" yet
// (ie they haven't reconnected with the infringing player identifier)
// we want to show them as banned as to not confuse people.
var hasActiveBan = activePenalties.Any(penalty => penalty.Type == EFPenalty.PenaltyType.Ban);
if (hasActiveBan)
{
client.Level = Data.Models.Client.EFClient.Permission.Banned;
}
var displayLevelInt = (int)client.Level;
var displayLevel = client.Level.ToLocalizedLevelName();
// if a linked ban has been revoked but they haven't reconnected, we should not show them as still banned
var shouldHideBanLevel = !hasActiveBan && client.Level == Data.Models.Client.EFClient.Permission.Banned;
if (!Authorized && client.Level.ShouldHideLevel() || shouldHideBanLevel)
{
displayLevelInt = (int)Data.Models.Client.EFClient.Permission.User;
displayLevel = Data.Models.Client.EFClient.Permission.User.ToLocalizedLevelName();
}
displayLevel = string.IsNullOrEmpty(client.Tag) ? displayLevel : $"{displayLevel} ({client.Tag})";
2022-04-19 19:43:58 -04:00
var ingameClient = Manager.GetActiveClients().FirstOrDefault(c => c.ClientId == client.ClientId);
2022-02-01 19:20:29 -05:00
var clientDto = new PlayerInfo
2018-02-21 20:29:23 -05:00
{
Name = client.Name,
Game = client.GameName,
Level = displayLevel,
LevelInt = displayLevelInt,
2018-02-21 20:29:23 -05:00
ClientId = client.ClientId,
2022-04-19 19:43:58 -04:00
IPAddress = PermissionsSet.HasPermission(WebfrontEntity.ClientIPAddress, WebfrontPermission.Read)
? client.IPAddressString
: null,
2018-02-21 20:29:23 -05:00
NetworkId = client.NetworkId,
Meta = new List<InformationResponse>(),
2018-02-21 20:29:23 -05:00
Aliases = client.AliasLink.Children
2022-04-19 19:43:58 -04:00
.Select(alias => (alias.Name, alias.DateAdded))
.GroupBy(alias => alias.Name.StripColors())
// we want the longest "duplicate" name
2022-04-19 19:43:58 -04:00
.Select(grp => grp.OrderByDescending(item => item.Name.Length).First())
.Distinct()
2018-02-21 20:29:23 -05:00
.ToList(),
2022-04-19 19:43:58 -04:00
IPs = PermissionsSet.HasPermission(WebfrontEntity.ClientIPAddress, WebfrontPermission.Read)
? client.AliasLink.Children
.Select(alias => (alias.IPAddress.ConvertIPtoString(), alias.DateAdded))
.GroupBy(alias => alias.Item1)
.Select(grp => grp.OrderByDescending(item => item.DateAdded).First())
.Distinct()
.ToList()
: new List<(string, DateTime)>(),
HasActivePenalty = activePenalties.Any(penalty => penalty.Type != EFPenalty.PenaltyType.Flag),
Online = ingameClient != null,
TimeOnline = (DateTime.UtcNow - client.LastConnection).HumanizeForCurrentCulture(),
LinkedAccounts = client.LinkedAccounts,
2022-04-19 19:43:58 -04:00
MetaFilterType = metaFilterType,
ConnectProtocolUrl = ingameClient?.CurrentServer.EventParser.URLProtocolFormat.FormatExt(
ingameClient.CurrentServer.ResolvedIpEndPoint.Address.IsInternal()
? Program.Manager.ExternalIPAddress
: ingameClient.CurrentServer.ListenAddress,
ingameClient.CurrentServer.ListenPort),
2022-04-19 19:43:58 -04:00
CurrentServerName = ingameClient?.CurrentServer?.Hostname,
2022-07-20 11:32:26 -04:00
GeoLocationInfo = await _geoLocationService.Locate(client.IPAddressString),
NoteMeta = string.IsNullOrWhiteSpace(note?.Note) ? null: note,
Interactions = interactions.ToList()
2018-02-21 20:29:23 -05:00
};
2018-03-09 03:01:12 -05:00
var meta = await _metaService.GetRuntimeMeta<InformationResponse>(new ClientPaginationRequest
{
ClientId = client.ClientId,
Before = DateTime.UtcNow
}, MetaType.Information);
if (gravatar != null)
{
clientDto.Meta.Add(new InformationResponse()
{
Key = "GravatarEmail",
Type = MetaType.Other,
Value = gravatar.Value
});
}
// Reducing the enum value for Temp/Mute so bans appear in client banner first
clientDto.ActivePenalty = activePenalties.MaxBy(penalty => penalty.Type switch
{
EFPenalty.PenaltyType.TempMute => 0,
EFPenalty.PenaltyType.Mute => 1,
_ => (int)penalty.Type
});
clientDto.Meta.AddRange(Authorized ? meta : meta.Where(m => !m.IsSensitive));
2022-02-01 19:20:29 -05:00
var strippedName = clientDto.Name.StripColors();
2022-07-05 13:02:43 -04:00
ViewBag.Title = $"{strippedName} | {Localization["WEBFRONT_CLIENT_PROFILE_TITLE"]}";
ViewBag.Description = Localization["WEBFRONT_PROFILE_DESCRIPTION"].FormatExt(strippedName);
2022-01-28 18:28:49 -05:00
ViewBag.UseNewStats = _config?.EnableAdvancedMetrics ?? true;
2018-02-21 20:29:23 -05:00
return View("Profile/Index", clientDto);
}
2022-04-19 19:43:58 -04:00
public async Task<IActionResult> Privileged()
2018-02-21 20:29:23 -05:00
{
if (Manager.GetApplicationSettings().Configuration().EnablePrivilegedUserPrivacy && !Authorized)
{
return RedirectToAction("Index", "Home");
}
2022-04-19 19:43:58 -04:00
2018-03-06 02:22:19 -05:00
var admins = (await Manager.GetClientService().GetPrivilegedClients())
.OrderByDescending(_client => _client.Level)
.ThenBy(_client => _client.Name);
var adminsDict = new Dictionary<EFClient.Permission, IList<ClientInfo>>();
2018-02-21 20:29:23 -05:00
foreach (var admin in admins)
{
if (!adminsDict.ContainsKey(admin.Level))
{
2018-02-21 20:29:23 -05:00
adminsDict.Add(admin.Level, new List<ClientInfo>());
}
2018-02-21 20:29:23 -05:00
adminsDict[admin.Level].Add(new ClientInfo
2018-02-21 20:29:23 -05:00
{
Name = admin.Name,
2022-04-19 19:43:58 -04:00
ClientId = admin.ClientId,
LastConnection = admin.LastConnection,
IsMasked = admin.Masked,
Game = admin.GameName
2018-02-21 20:29:23 -05:00
});
}
ViewBag.Title = Localization["WEBFRONT_CLIENT_PRIVILEGED_TITLE"];
ViewBag.Description = Localization["WEBFRONT_DESCRIPTION_PRIVILEGED"];
ViewBag.Keywords = Localization["WEBFRONT_KEYWORDS_PRIVILEGED"];
2018-02-21 20:29:23 -05:00
return View("Privileged/Index", adminsDict);
}
2022-04-19 19:43:58 -04:00
public async Task<IActionResult> Find(string clientName)
2018-02-21 20:29:23 -05:00
{
if (string.IsNullOrWhiteSpace(clientName))
{
return StatusCode(400);
}
2019-04-25 22:05:35 -04:00
var clientsDto = await Manager.GetClientService().FindClientsByIdentifier(clientName);
foreach (var client in clientsDto)
{
if (!Authorized && ((Data.Models.Client.EFClient.Permission)client.LevelInt).ShouldHideLevel())
{
client.LevelInt = (int)Data.Models.Client.EFClient.Permission.User;
client.Level = Data.Models.Client.EFClient.Permission.User.ToLocalizedLevelName();
}
}
2018-02-21 20:29:23 -05:00
2022-04-19 19:43:58 -04:00
ViewBag.SearchTerm = clientName;
ViewBag.ResultCount = clientsDto.Count;
2022-07-05 13:42:17 -04:00
ViewBag.Title = Localization["WEBFRONT_SEARCH_RESULTS_TITLE"];
2022-06-16 15:02:44 -04:00
2018-02-21 20:29:23 -05:00
return View("Find/Index", clientsDto);
}
2023-01-23 17:38:16 -05:00
public async Task<IActionResult> AdvancedFind(ClientResourceRequest request)
{
ViewBag.Title = Localization["WEBFRONT_SEARCH_RESULTS_TITLE"];
ViewBag.ClientResourceRequest = request;
2023-09-13 23:50:37 -04:00
request.RequesterPermission = Client.Level;
2023-01-23 17:38:16 -05:00
var response = await _clientResourceHelper.QueryResource(request);
return request.Offset > 0
? PartialView("Find/_AdvancedFindList", response.Results)
: View("Find/AdvancedFind", response.Results);
}
2022-04-19 19:43:58 -04:00
public IActionResult Meta(int id, int count, int offset, long? startAt, MetaType? metaFilterType,
CancellationToken token)
{
var request = new ClientPaginationRequest
{
ClientId = id,
Count = count,
Offset = offset,
Before = DateTime.FromFileTimeUtc(startAt ?? DateTime.UtcNow.ToFileTimeUtc())
};
return ViewComponent(typeof(ProfileMetaListViewComponent), new
{
clientId = request.ClientId,
count = request.Count,
offset = request.Offset,
startAt = request.Before,
metaType = metaFilterType,
token
});
}
2018-02-21 20:29:23 -05:00
}
}