Compare commits

...

54 Commits

Author SHA1 Message Date
c93f896bc5 fix profile issue 2022-03-24 08:40:42 -05:00
ccc8316a2f fix minimap image for live radar 2022-03-24 08:40:20 -05:00
497c15a6a8 update stats to use new meta service 2022-03-23 13:54:42 -05:00
7be096e0b6 add vpn whitelist command 2022-03-23 13:34:04 -05:00
20858991e1 move live radar js into own file 2022-03-23 12:52:11 -05:00
85d44b0eb0 fix issue with multi line output freezing console 2022-03-23 12:09:40 -05:00
51ef67ae9c add BroadcastAsync 2022-03-23 11:43:20 -05:00
63b04be4c7 add tell async and update SharedLibraryCore version 2022-03-23 11:38:09 -05:00
36eb45bb2e mark old meta service as obsolete 2022-03-23 11:31:53 -05:00
04a4dcf153 implement metaservice v2 2022-03-23 08:43:57 -05:00
b46b1eb5e7 fix update on report to penalize flagged users 2022-03-23 08:22:23 -05:00
287635fa36 update integration gsc 2022-03-12 13:41:10 -06:00
f567a03fa7 implement team tracking via game interface (EFClient.Team and EFClient.TeamName) 2022-03-12 13:38:33 -06:00
1b6d8107ae Add T6 Weapon Name Parser Config (#236)
Add T6 Weapon Name Parser Config
2022-03-08 12:08:16 -06:00
1e8f06f3a3 Fix iw3 gamestring typo (#234)
RDP -> RPD
2022-03-08 12:08:04 -06:00
064879fead Add info api for #231 2022-03-08 12:06:46 -06:00
e32e97b9e6 fix issue with loading stats config #237 2022-03-08 11:24:59 -06:00
42313b7816 update action on report to use level enum string 2022-03-07 20:00:05 -06:00
9f4d06c265 refactor some game interface plugin approach 2022-03-07 19:59:34 -06:00
acf66da4ca tweak cod rcon connection and fix max health for hide integration command 2022-03-05 13:13:00 -06:00
59ca399045 more cod rcon tweaks 2022-03-03 08:54:17 -06:00
ef70496546 hopefully fix some issues with rcon socket 2022-03-02 18:21:08 -06:00
e6e56d8d14 add back helper methods without cancellation token for plugins 2022-03-02 08:29:15 -06:00
55b0caf900 tweak game interface values again 2022-03-02 08:28:41 -06:00
a4c3f9c2d1 update delete obsolete plugin migration 2022-03-01 12:47:35 -06:00
241aa0a5f6 tweak rcon timeout for script calls 2022-03-01 12:46:01 -06:00
ec0f59cdb1 add set spectator command for game interface 2022-03-01 12:45:39 -06:00
59d69bd22b add cancellation token for rcon connection to allow more granular control 2022-02-28 20:44:30 -06:00
e9c8ead829 simplify level update so we don't have to worry about linked account levels 2022-02-28 15:20:46 -06:00
58d48a211e make sure iw4madmin exits when selecting "no" to continue with failed server connections 2022-02-28 15:16:30 -06:00
edf8e03b04 don't refresh scoreboard on every page. though I fixed this already... 2022-02-27 21:35:16 -06:00
de2e804b84 improve meta filter menu on profile 2022-02-25 21:09:57 -06:00
b087d4c8de unescape utf characters when saving configs 2022-02-25 09:44:28 -06:00
bd6c0dd5be fix issue with tempban not displaying properly 2022-02-25 08:22:40 -06:00
4ace476242 mark permission changed as sensitive 2022-02-23 16:26:46 -06:00
bb7215dbb6 increment shared library references 2022-02-23 15:57:44 -06:00
88bd47f3ae add search ip shortcut on profile 2022-02-23 15:47:17 -06:00
39a1066c74 add permission level changed meta 2022-02-23 12:47:00 -06:00
18f3c59b9b allow search client exact with quotes 2022-02-23 09:32:59 -06:00
0d88b6293f add create/update times to penalty identifiers 2022-02-23 09:02:01 -06:00
a6b56ceded tweak for integration 2022-02-22 17:10:33 -06:00
78ef977268 simplify ban process with new system 2022-02-22 17:09:50 -06:00
d527a86911 improve mag command matching of maps and gametypes 2022-02-22 08:38:02 -06:00
2e531c4a50 validate game interface commands to ensure it's enabled before trying to execute 2022-02-18 10:15:11 -06:00
45059fcfd9 change mask command alias to not conflict with game interface hide 2022-02-18 10:04:48 -06:00
482cd9c339 Merge branch 'release/pre' of https://github.com/RaidMax/IW4M-Admin into release/pre 2022-02-15 20:23:30 -06:00
51667159a2 fix validation errors freezing initialization 2022-02-15 20:23:16 -06:00
ea18a286b2 improve error output when configuration is invalid 2022-02-15 20:16:21 -06:00
9a6d7c6a20 game interface improvements 2022-02-15 20:05:50 -06:00
adcb75319c Changed .NET 6 Direct Download... (#229)
Changed the .NET 6 Direct Download link to the hosting bundle as it includes both the .NET 6 runtime and the ASP .NET 6 runtime which are both needed.
2022-02-15 09:56:31 -06:00
037fac5786 game interface improvements 2022-02-13 21:38:40 -06:00
f4b892d8f4 improve network log support 2022-02-13 16:50:09 -06:00
3640d1df54 small updates for game interface 2022-02-12 21:54:21 -06:00
f3c6b10a35 add network game log reader ex: net.tcp://ip:port 2022-02-11 15:33:05 -06:00
111 changed files with 19864 additions and 1620 deletions

View File

@ -59,7 +59,6 @@ namespace IW4MAdmin.Application
readonly PenaltyService PenaltySvc; readonly PenaltyService PenaltySvc;
public IConfigurationHandler<ApplicationConfiguration> ConfigHandler; public IConfigurationHandler<ApplicationConfiguration> ConfigHandler;
readonly IPageList PageList; readonly IPageList PageList;
private readonly IMetaService _metaService;
private readonly TimeSpan _throttleTimeout = new TimeSpan(0, 1, 0); private readonly TimeSpan _throttleTimeout = new TimeSpan(0, 1, 0);
private readonly CancellationTokenSource _tokenSource; private readonly CancellationTokenSource _tokenSource;
private readonly Dictionary<string, Task<IList>> _operationLookup = new Dictionary<string, Task<IList>>(); private readonly Dictionary<string, Task<IList>> _operationLookup = new Dictionary<string, Task<IList>>();
@ -81,7 +80,7 @@ namespace IW4MAdmin.Application
ITranslationLookup translationLookup, IConfigurationHandler<CommandConfiguration> commandConfiguration, ITranslationLookup translationLookup, IConfigurationHandler<CommandConfiguration> commandConfiguration,
IConfigurationHandler<ApplicationConfiguration> appConfigHandler, IGameServerInstanceFactory serverInstanceFactory, IConfigurationHandler<ApplicationConfiguration> appConfigHandler, IGameServerInstanceFactory serverInstanceFactory,
IEnumerable<IPlugin> plugins, IParserRegexFactory parserRegexFactory, IEnumerable<IRegisterEvent> customParserEvents, IEnumerable<IPlugin> plugins, IParserRegexFactory parserRegexFactory, IEnumerable<IRegisterEvent> customParserEvents,
IEventHandler eventHandler, IScriptCommandFactory scriptCommandFactory, IDatabaseContextFactory contextFactory, IMetaService metaService, IEventHandler eventHandler, IScriptCommandFactory scriptCommandFactory, IDatabaseContextFactory contextFactory,
IMetaRegistration metaRegistration, IScriptPluginServiceResolver scriptPluginServiceResolver, ClientService clientService, IServiceProvider serviceProvider, IMetaRegistration metaRegistration, IScriptPluginServiceResolver scriptPluginServiceResolver, ClientService clientService, IServiceProvider serviceProvider,
ChangeHistoryService changeHistoryService, ApplicationConfiguration appConfig, PenaltyService penaltyService) ChangeHistoryService changeHistoryService, ApplicationConfiguration appConfig, PenaltyService penaltyService)
{ {
@ -97,7 +96,6 @@ namespace IW4MAdmin.Application
AdditionalRConParsers = new List<IRConParser>() { new BaseRConParser(serviceProvider.GetRequiredService<ILogger<BaseRConParser>>(), parserRegexFactory) }; AdditionalRConParsers = new List<IRConParser>() { new BaseRConParser(serviceProvider.GetRequiredService<ILogger<BaseRConParser>>(), parserRegexFactory) };
TokenAuthenticator = new TokenAuthentication(); TokenAuthenticator = new TokenAuthentication();
_logger = logger; _logger = logger;
_metaService = metaService;
_tokenSource = new CancellationTokenSource(); _tokenSource = new CancellationTokenSource();
_commands = commands.ToList(); _commands = commands.ToList();
_translationLookup = translationLookup; _translationLookup = translationLookup;
@ -412,7 +410,7 @@ namespace IW4MAdmin.Application
if (!validationResult.IsValid) if (!validationResult.IsValid)
{ {
throw new ConfigurationException(_translationLookup["MANAGER_CONFIGURATION_ERROR"]) throw new ConfigurationException("Could not validate configuration")
{ {
Errors = validationResult.Errors.Select(_error => _error.ErrorMessage).ToArray(), Errors = validationResult.Errors.Select(_error => _error.ErrorMessage).ToArray(),
ConfigurationFileName = ConfigHandler.FileName ConfigurationFileName = ConfigHandler.FileName

View File

@ -0,0 +1,64 @@
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Data.Models;
using Data.Models.Client;
using Microsoft.Extensions.Logging;
using SharedLibraryCore;
using SharedLibraryCore.Commands;
using SharedLibraryCore.Configuration;
using SharedLibraryCore.Interfaces;
namespace IW4MAdmin.Application.Commands.ClientTags
{
public class AddClientTagCommand : Command
{
private readonly IMetaServiceV2 _metaService;
public AddClientTagCommand(ILogger<AddClientTagCommand> commandLogger, CommandConfiguration config,
ITranslationLookup layout, IMetaServiceV2 metaService) :
base(config, layout)
{
Name = "addclienttag";
Description = layout["COMMANDS_ADD_CLIENT_TAG_DESC"];
Alias = "act";
Permission = EFClient.Permission.Owner;
RequiresTarget = false;
Arguments = new[]
{
new CommandArgument
{
Name = _translationLookup["COMMANDS_ARGUMENT_TAG"],
Required = true
}
};
_metaService = metaService;
logger = commandLogger;
}
public override async Task ExecuteAsync(GameEvent gameEvent)
{
var existingTags = await _metaService.GetPersistentMetaValue<List<TagMeta>>(EFMeta.ClientTagNameV2) ??
new List<TagMeta>();
var tagName = gameEvent.Data.Trim();
if (existingTags.Any(tag => tag.TagName == tagName))
{
logger.LogWarning("Tag with name {TagName} already exists", tagName);
return;
}
existingTags.Add(new TagMeta
{
Id = (existingTags.LastOrDefault()?.TagId ?? 0) + 1,
Value = tagName
});
await _metaService.SetPersistentMetaValue(EFMeta.ClientTagNameV2, existingTags,
gameEvent.Owner.Manager.CancellationToken);
gameEvent.Origin.Tell(_translationLookup["COMMANDS_ADD_CLIENT_TAG_SUCCESS"].FormatExt(gameEvent.Data));
}
}
}

View File

@ -1,17 +1,19 @@
using System.Linq; using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks; using System.Threading.Tasks;
using Data.Models; using Data.Models;
using Data.Models.Client; using Data.Models.Client;
using SharedLibraryCore;
using SharedLibraryCore.Configuration; using SharedLibraryCore.Configuration;
using SharedLibraryCore.Interfaces; using SharedLibraryCore.Interfaces;
namespace SharedLibraryCore.Commands namespace IW4MAdmin.Application.Commands.ClientTags
{ {
public class ListClientTags : Command public class ListClientTags : Command
{ {
private readonly IMetaService _metaService; private readonly IMetaServiceV2 _metaService;
public ListClientTags(CommandConfiguration config, ITranslationLookup layout, IMetaService metaService) : base( public ListClientTags(CommandConfiguration config, ITranslationLookup layout, IMetaServiceV2 metaService) : base(
config, layout) config, layout)
{ {
Name = "listclienttags"; Name = "listclienttags";
@ -19,14 +21,18 @@ namespace SharedLibraryCore.Commands
Alias = "lct"; Alias = "lct";
Permission = EFClient.Permission.Owner; Permission = EFClient.Permission.Owner;
RequiresTarget = false; RequiresTarget = false;
_metaService = metaService; _metaService = metaService;
} }
public override async Task ExecuteAsync(GameEvent gameEvent) public override async Task ExecuteAsync(GameEvent gameEvent)
{ {
var tags = await _metaService.GetPersistentMeta(EFMeta.ClientTagName); var tags = await _metaService.GetPersistentMetaValue<List<TagMeta>>(EFMeta.ClientTagNameV2);
gameEvent.Origin.Tell(tags.Select(tag => tag.Value));
if (tags is not null)
{
await gameEvent.Origin.TellAsync(tags.Select(tag => tag.TagName),
gameEvent.Owner.Manager.CancellationToken);
}
} }
} }
} }

View File

@ -1,16 +1,20 @@
using System.Threading.Tasks; using System.Collections.Generic;
using System.Threading.Tasks;
using System.Linq;
using Data.Models; using Data.Models;
using Data.Models.Client; using Data.Models.Client;
using SharedLibraryCore;
using SharedLibraryCore.Commands;
using SharedLibraryCore.Configuration; using SharedLibraryCore.Configuration;
using SharedLibraryCore.Interfaces; using SharedLibraryCore.Interfaces;
namespace SharedLibraryCore.Commands namespace IW4MAdmin.Application.Commands.ClientTags
{ {
public class RemoveClientTag : Command public class RemoveClientTag : Command
{ {
private readonly IMetaService _metaService; private readonly IMetaServiceV2 _metaService;
public RemoveClientTag(CommandConfiguration config, ITranslationLookup layout, IMetaService metaService) : base( public RemoveClientTag(CommandConfiguration config, ITranslationLookup layout, IMetaServiceV2 metaService) : base(
config, layout) config, layout)
{ {
Name = "removeclienttag"; Name = "removeclienttag";
@ -32,8 +36,13 @@ namespace SharedLibraryCore.Commands
public override async Task ExecuteAsync(GameEvent gameEvent) public override async Task ExecuteAsync(GameEvent gameEvent)
{ {
await _metaService.RemovePersistentMeta(EFMeta.ClientTagName, gameEvent.Data); var existingMeta = await _metaService.GetPersistentMetaValue<List<TagMeta>>(EFMeta.ClientTagNameV2,
gameEvent.Owner.Manager.CancellationToken);
existingMeta = existingMeta.Where(meta => meta.TagName != gameEvent.Data.Trim()).ToList();
await _metaService.SetPersistentMetaValue(EFMeta.ClientTagNameV2, existingMeta,
gameEvent.Owner.Manager.CancellationToken);
gameEvent.Origin.Tell(_translationLookup["COMMANDS_REMOVE_CLIENT_TAG_SUCCESS"].FormatExt(gameEvent.Data)); gameEvent.Origin.Tell(_translationLookup["COMMANDS_REMOVE_CLIENT_TAG_SUCCESS"].FormatExt(gameEvent.Data));
} }
} }
} }

View File

@ -1,18 +1,22 @@
using System.Linq; using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks; using System.Threading.Tasks;
using Data.Models; using Data.Models;
using Data.Models.Client; using Data.Models.Client;
using SharedLibraryCore;
using SharedLibraryCore.Commands;
using SharedLibraryCore.Configuration; using SharedLibraryCore.Configuration;
using SharedLibraryCore.Dtos;
using SharedLibraryCore.Interfaces; using SharedLibraryCore.Interfaces;
namespace SharedLibraryCore.Commands namespace IW4MAdmin.Application.Commands.ClientTags
{ {
public class SetClientTagCommand : Command public class SetClientTagCommand : Command
{ {
private readonly IMetaService _metaService; private readonly IMetaServiceV2 _metaService;
public SetClientTagCommand(CommandConfiguration config, ITranslationLookup layout, IMetaService metaService) : public SetClientTagCommand(CommandConfiguration config, ITranslationLookup layout, IMetaServiceV2 metaService) :
base(config, layout) base(config, layout)
{ {
Name = "setclienttag"; Name = "setclienttag";
@ -34,8 +38,10 @@ namespace SharedLibraryCore.Commands
public override async Task ExecuteAsync(GameEvent gameEvent) public override async Task ExecuteAsync(GameEvent gameEvent)
{ {
var availableTags = await _metaService.GetPersistentMeta(EFMeta.ClientTagName); var token = gameEvent.Owner.Manager.CancellationToken;
var matchingTag = availableTags.FirstOrDefault(tag => tag.Value == gameEvent.Data);
var availableTags = await _metaService.GetPersistentMetaValue<List<LookupValue<string>>>(EFMeta.ClientTagNameV2, token);
var matchingTag = availableTags.FirstOrDefault(tag => tag.Value == gameEvent.Data.Trim());
if (matchingTag == null) if (matchingTag == null)
{ {
@ -44,8 +50,9 @@ namespace SharedLibraryCore.Commands
} }
gameEvent.Target.Tag = matchingTag.Value; gameEvent.Target.Tag = matchingTag.Value;
await _metaService.AddPersistentMeta(EFMeta.ClientTag, string.Empty, gameEvent.Target, matchingTag); await _metaService.SetPersistentMetaForLookupKey(EFMeta.ClientTagV2, EFMeta.ClientTagNameV2, matchingTag.Id,
gameEvent.Target.ClientId, token);
gameEvent.Origin.Tell(_translationLookup["COMMANDS_SET_CLIENT_TAG_SUCCESS"].FormatExt(matchingTag.Value)); gameEvent.Origin.Tell(_translationLookup["COMMANDS_SET_CLIENT_TAG_SUCCESS"].FormatExt(matchingTag.Value));
} }
} }
} }

View File

@ -0,0 +1,13 @@
using System.Text.Json.Serialization;
using SharedLibraryCore.Interfaces;
namespace IW4MAdmin.Application.Commands.ClientTags;
public class TagMeta : ILookupValue<string>
{
[JsonIgnore] public int TagId => Id;
[JsonIgnore] public string TagName => Value;
public int Id { get; set; }
public string Value { get; set; }
}

View File

@ -1,17 +1,18 @@
using System.Threading.Tasks; using System.Threading.Tasks;
using Data.Models; using Data.Models;
using Data.Models.Client; using Data.Models.Client;
using SharedLibraryCore;
using SharedLibraryCore.Commands;
using SharedLibraryCore.Configuration; using SharedLibraryCore.Configuration;
using SharedLibraryCore.Interfaces; using SharedLibraryCore.Interfaces;
namespace SharedLibraryCore.Commands namespace IW4MAdmin.Application.Commands.ClientTags
{ {
public class UnsetClientTagCommand : Command public class UnsetClientTagCommand : Command
{ {
private readonly IMetaService _metaService; private readonly IMetaServiceV2 _metaService;
public UnsetClientTagCommand(CommandConfiguration config, ITranslationLookup layout, IMetaServiceV2 metaService) :
public UnsetClientTagCommand(CommandConfiguration config, ITranslationLookup layout, IMetaService metaService) :
base(config, layout) base(config, layout)
{ {
Name = "unsetclienttag"; Name = "unsetclienttag";
@ -34,8 +35,9 @@ namespace SharedLibraryCore.Commands
public override async Task ExecuteAsync(GameEvent gameEvent) public override async Task ExecuteAsync(GameEvent gameEvent)
{ {
gameEvent.Target.Tag = null; gameEvent.Target.Tag = null;
await _metaService.RemovePersistentMeta(EFMeta.ClientTag, gameEvent.Target); await _metaService.RemovePersistentMeta(EFMeta.ClientTagV2, gameEvent.Target.ClientId,
gameEvent.Owner.Manager.CancellationToken);
gameEvent.Origin.Tell(_translationLookup["COMMANDS_UNSET_CLIENT_TAG_SUCCESS"]); gameEvent.Origin.Tell(_translationLookup["COMMANDS_UNSET_CLIENT_TAG_SUCCESS"]);
} }
} }
} }

View File

@ -29,9 +29,9 @@ namespace IW4MAdmin.Application.Commands
$"[(Color::Accent){client.ClientPermission.Name}(Color::White){(string.IsNullOrEmpty(client.Tag) ? "" : $" {client.Tag}")}(Color::White)][(Color::Yellow)#{client.ClientNumber}(Color::White)] {client.Name}") $"[(Color::Accent){client.ClientPermission.Name}(Color::White){(string.IsNullOrEmpty(client.Tag) ? "" : $" {client.Tag}")}(Color::White)][(Color::Yellow)#{client.ClientNumber}(Color::White)] {client.Name}")
.ToArray(); .ToArray();
gameEvent.Origin.Tell(clientList); gameEvent.Origin.TellAsync(clientList, gameEvent.Owner.Manager.CancellationToken);
return Task.CompletedTask; return Task.CompletedTask;
} }
} }
} }

View File

@ -54,20 +54,8 @@ namespace IW4MAdmin.Application.Commands
return; return;
} }
string map; var map = match.Groups[1].Length > 0 ? match.Groups[1].ToString() : match.Groups[2].ToString();
string gametype; var gametype = match.Groups[3].Length > 0 ? match.Groups[3].ToString() : match.Groups[4].ToString();
if (match.Groups.Count > 3)
{
map = match.Groups[2].ToString();
gametype = match.Groups[4].ToString();
}
else
{
map = match.Groups[1].ToString();
gametype = match.Groups[3].ToString();
}
var matchingMaps = gameEvent.Owner.FindMap(map); var matchingMaps = gameEvent.Owner.FindMap(map);
var matchingGametypes = _defaultSettings.FindGametype(gametype, gameEvent.Owner.GameName); var matchingGametypes = _defaultSettings.FindGametype(gametype, gameEvent.Owner.GameName);
@ -104,7 +92,7 @@ namespace IW4MAdmin.Application.Commands
_logger.LogDebug("Changing map to {Map} and gametype {Gametype}", map, gametype); _logger.LogDebug("Changing map to {Map} and gametype {Gametype}", map, gametype);
await gameEvent.Owner.SetDvarAsync("g_gametype", gametype); await gameEvent.Owner.SetDvarAsync("g_gametype", gametype, gameEvent.Owner.Manager.CancellationToken);
gameEvent.Owner.Broadcast(_translationLookup["COMMANDS_MAP_SUCCESS"].FormatExt(map)); gameEvent.Owner.Broadcast(_translationLookup["COMMANDS_MAP_SUCCESS"].FormatExt(map));
await Task.Delay(gameEvent.Owner.Manager.GetApplicationSettings().Configuration().MapChangeDelaySeconds); await Task.Delay(gameEvent.Owner.Manager.GetApplicationSettings().Configuration().MapChangeDelaySeconds);

View File

@ -2021,7 +2021,7 @@
"barrett": "Barrett .50cal", "barrett": "Barrett .50cal",
"mp44": "MP44", "mp44": "MP44",
"remington700": "R700", "remington700": "R700",
"rpd": "RDP", "rpd": "RPD",
"saw": " M249 SAW", "saw": " M249 SAW",
"usp": "USP .45", "usp": "USP .45",
"winchester1200": "W1200", "winchester1200": "W1200",
@ -2083,6 +2083,126 @@
"type99rifle": "Arisaka", "type99rifle": "Arisaka",
"mosinrifle": "Mosin-Nagant", "mosinrifle": "Mosin-Nagant",
"ptrs41": "PTRS-41" "ptrs41": "PTRS-41"
},
"T6" : {
"mp7": "MP7",
"pdw57": "PDW-57",
"vector": "Vector K10",
"insas": "MSMC",
"qcw05": "Chicom CQB",
"evoskorpion": "Skorpion EVO",
"peacekeeper": "Peacekeeper",
"tar21": "MTAR",
"type95": "Type 25",
"sig556": "SWAT-556",
"sa58": "FAL-OSW",
"hk416": "M27",
"scar": "SCAR-H",
"saritch": "SMR",
"xm8": "M8A1",
"an94": "AN-94",
"870mcs": "Remington-870 MCS",
"saiga12": "S12",
"ksg": "KSG",
"srm1216": "M1216",
"mk48": "MK 48",
"qbb95": "QBB LSW",
"lsat": "LSAT",
"hamr": "HAMR",
"svu": "SVU-AS",
"dsr50": "DSR 50",
"ballista": "Ballista",
"as50": "XPR-50",
"fiveseven": "Five-Seven",
"fnp45": "TAC-45",
"beretta93r": "B23R",
"judge": "Executioner",
"kard": "KAP-40",
"smaw": "SMAW",
"fhj18": "FHJ-18 AA",
"usrpg": "RPG",
"riotshield": "Assault Shield",
"crossbow": "Crossbow",
"knife_ballistic": "Ballistic Knife",
"knife_held": "Knife",
"knife": "Knife",
"frag_grenade": "Grenade",
"hatchet": "Combat Axe",
"sticky_grenade": "Semtex",
"satchel_charge": "C4",
"bouncingbetty": "Bouncing Betty",
"claymore": "Claymore",
"smoke_center": "Smoke Grenade",
"concussion_grenade": "Concussion",
"emp_grenade": "EMP Grenade",
"sensor_grenade": "Sensor Grenade",
"flash_grenade": "Flashbang",
"proximity_grenade": "Shock Charge",
"pda_hack": "Black Hat",
"trophy_system": "Trophy System",
"tactical_insertion": "Tactical Insertion",
"acog": "ACOG",
"stalker": "Stock",
"swayreduc": "Ballistics CPU",
"ir": "Dual Band",
"dw": "Dual Wield",
"extclip": "Extended Clip",
"halo": "EOTech",
"dualclip": "Fast Mag",
"fmj": "FMJ",
"grip": "Fore Grip",
"gl": "Grenade Launcher",
"dualoptic": "Hybrid Optic",
"is": "Iron Sights",
"steadyaim": "Laser Sight",
"extbarrel": "Long Barrel",
"mms": "MMS",
"fastads": "Quickdraw",
"rf": "Rapid Fire",
"reflex": "Reflex Sight",
"sf": "Select Fire",
"silencer": "Suppressor",
"tacknife": "Tactical Knife",
"stackfire": "Tri-Bolt",
"rangefinder": "Target Finder",
"vzoom": "Variable Zoom",
"spyplane": "UAV",
"rcbomb": "RC-XD",
"missile_drone": "Hunter Killer",
"supplydrop": "Care Package",
"counteruav": "Counter-UAV",
"microwave_turret": "Guardian",
"remote_missile": "Hellstorm Missile",
"planemortar": "Lightning Strike",
"auto_turret": "Sentry Gun",
"minigun": "Death Machine",
"m32": "War Machine",
"qrdrone": "Dragonfire",
"ai_tank_drop": "AGR",
"comlink": "Stealth Chopper",
"spyplane_direction": "Orbital VSAT",
"helicopter_guard": "Escort Drone",
"emp": "EMP",
"straferun": "Warthog",
"remote_mortar": "Lodestar",
"player_gunner": "VTOL Warship",
"dogs": "K9 Unit",
"missile_swarm": "Swarm"
} }
} }
} }

View File

@ -49,6 +49,13 @@ namespace IW4MAdmin.Application.EventParsers
Configuration.Join.AddMapping(ParserRegex.GroupType.OriginNetworkId, 2); Configuration.Join.AddMapping(ParserRegex.GroupType.OriginNetworkId, 2);
Configuration.Join.AddMapping(ParserRegex.GroupType.OriginClientNumber, 3); Configuration.Join.AddMapping(ParserRegex.GroupType.OriginClientNumber, 3);
Configuration.Join.AddMapping(ParserRegex.GroupType.OriginName, 4); Configuration.Join.AddMapping(ParserRegex.GroupType.OriginName, 4);
Configuration.JoinTeam.Pattern = @"^(JT);(-?[A-Fa-f0-9_]{1,32}|bot[0-9]+|0);([0-9]+);(\w+);(.+)$";
Configuration.JoinTeam.AddMapping(ParserRegex.GroupType.EventType, 1);
Configuration.JoinTeam.AddMapping(ParserRegex.GroupType.OriginNetworkId, 2);
Configuration.JoinTeam.AddMapping(ParserRegex.GroupType.OriginClientNumber, 3);
Configuration.JoinTeam.AddMapping(ParserRegex.GroupType.OriginTeam, 4);
Configuration.JoinTeam.AddMapping(ParserRegex.GroupType.OriginName, 5);
Configuration.Damage.Pattern = @"^(D);(-?[A-Fa-f0-9_]{1,32}|bot[0-9]+|0);(-?[0-9]+);(axis|allies|world|none)?;([^;]{1,32});(-?[A-Fa-f0-9_]{1,32}|bot[0-9]+|0)?;(-?[0-9]+);(axis|allies|world|none)?;([^;]{1,32})?;((?:[0-9]+|[a-z]+|_|\+)+);([0-9]+);((?:[A-Z]|_)+);((?:[a-z]|_)+)$"; Configuration.Damage.Pattern = @"^(D);(-?[A-Fa-f0-9_]{1,32}|bot[0-9]+|0);(-?[0-9]+);(axis|allies|world|none)?;([^;]{1,32});(-?[A-Fa-f0-9_]{1,32}|bot[0-9]+|0)?;(-?[0-9]+);(axis|allies|world|none)?;([^;]{1,32})?;((?:[0-9]+|[a-z]+|_|\+)+);([0-9]+);((?:[A-Z]|_)+);((?:[a-z]|_)+)$";
Configuration.Damage.AddMapping(ParserRegex.GroupType.EventType, 1); Configuration.Damage.AddMapping(ParserRegex.GroupType.EventType, 1);
@ -100,6 +107,7 @@ namespace IW4MAdmin.Application.EventParsers
{"K", GameEvent.EventType.Kill}, {"K", GameEvent.EventType.Kill},
{"D", GameEvent.EventType.Damage}, {"D", GameEvent.EventType.Damage},
{"J", GameEvent.EventType.PreConnect}, {"J", GameEvent.EventType.PreConnect},
{"JT", GameEvent.EventType.JoinTeam },
{"Q", GameEvent.EventType.PreDisconnect}, {"Q", GameEvent.EventType.PreDisconnect},
}; };
} }
@ -322,6 +330,42 @@ namespace IW4MAdmin.Application.EventParsers
} }
} }
if (eventType == GameEvent.EventType.JoinTeam)
{
var match = Configuration.JoinTeam.PatternMatcher.Match(logLine);
if (match.Success)
{
var originIdString = match.Values[Configuration.JoinTeam.GroupMapping[ParserRegex.GroupType.OriginNetworkId]];
var originName = match.Values[Configuration.JoinTeam.GroupMapping[ParserRegex.GroupType.OriginName]];
var team = match.Values[Configuration.JoinTeam.GroupMapping[ParserRegex.GroupType.OriginTeam]];
var networkId = originIdString.IsBotGuid() ?
originName.GenerateGuidFromString() :
originIdString.ConvertGuidToLong(Configuration.GuidNumberStyle);
return new GameEvent
{
Type = GameEvent.EventType.JoinTeam,
Data = logLine,
Origin = new EFClient
{
CurrentAlias = new EFAlias
{
Name = match.Values[Configuration.JoinTeam.GroupMapping[ParserRegex.GroupType.OriginName]].TrimNewLine(),
},
NetworkId = networkId,
ClientNumber = Convert.ToInt32(match.Values[Configuration.JoinTeam.GroupMapping[ParserRegex.GroupType.OriginClientNumber]]),
State = EFClient.ClientState.Connected,
},
Extra = team,
RequiredEntity = GameEvent.EventRequiredEntity.None,
GameTime = gameTime,
Source = GameEvent.EventSource.Log
};
}
}
if (eventType == GameEvent.EventType.PreDisconnect) if (eventType == GameEvent.EventType.PreDisconnect)
{ {
var match = Configuration.Quit.PatternMatcher.Match(logLine); var match = Configuration.Quit.PatternMatcher.Match(logLine);

View File

@ -8,11 +8,12 @@ namespace IW4MAdmin.Application.EventParsers
/// generic implementation of the IEventParserConfiguration /// generic implementation of the IEventParserConfiguration
/// allows script plugins to generate dynamic configurations /// allows script plugins to generate dynamic configurations
/// </summary> /// </summary>
sealed internal class DynamicEventParserConfiguration : IEventParserConfiguration internal sealed class DynamicEventParserConfiguration : IEventParserConfiguration
{ {
public string GameDirectory { get; set; } public string GameDirectory { get; set; }
public ParserRegex Say { get; set; } public ParserRegex Say { get; set; }
public ParserRegex Join { get; set; } public ParserRegex Join { get; set; }
public ParserRegex JoinTeam { get; set; }
public ParserRegex Quit { get; set; } public ParserRegex Quit { get; set; }
public ParserRegex Kill { get; set; } public ParserRegex Kill { get; set; }
public ParserRegex Damage { get; set; } public ParserRegex Damage { get; set; }
@ -26,6 +27,7 @@ namespace IW4MAdmin.Application.EventParsers
{ {
Say = parserRegexFactory.CreateParserRegex(); Say = parserRegexFactory.CreateParserRegex();
Join = parserRegexFactory.CreateParserRegex(); Join = parserRegexFactory.CreateParserRegex();
JoinTeam = parserRegexFactory.CreateParserRegex();
Quit = parserRegexFactory.CreateParserRegex(); Quit = parserRegexFactory.CreateParserRegex();
Kill = parserRegexFactory.CreateParserRegex(); Kill = parserRegexFactory.CreateParserRegex();
Damage = parserRegexFactory.CreateParserRegex(); Damage = parserRegexFactory.CreateParserRegex();

View File

@ -18,14 +18,22 @@ namespace IW4MAdmin.Application.Factories
public IGameLogReader CreateGameLogReader(Uri[] logUris, IEventParser eventParser) public IGameLogReader CreateGameLogReader(Uri[] logUris, IEventParser eventParser)
{ {
var baseUri = logUris[0]; var baseUri = logUris[0];
if (baseUri.Scheme == Uri.UriSchemeHttp) if (baseUri.Scheme == Uri.UriSchemeHttp || baseUri.Scheme == Uri.UriSchemeHttps)
{ {
return new GameLogReaderHttp(logUris, eventParser, _serviceProvider.GetRequiredService<ILogger<GameLogReaderHttp>>()); return new GameLogReaderHttp(logUris, eventParser,
_serviceProvider.GetRequiredService<ILogger<GameLogReaderHttp>>());
} }
else if (baseUri.Scheme == Uri.UriSchemeFile) if (baseUri.Scheme == Uri.UriSchemeFile)
{ {
return new GameLogReader(baseUri.LocalPath, eventParser, _serviceProvider.GetRequiredService<ILogger<GameLogReader>>()); return new GameLogReader(baseUri.LocalPath, eventParser,
_serviceProvider.GetRequiredService<ILogger<GameLogReader>>());
}
if (baseUri.Scheme == Uri.UriSchemeNetTcp)
{
return new NetworkGameLogReader(logUris, eventParser,
_serviceProvider.GetRequiredService<ILogger<NetworkGameLogReader>>());
} }
throw new NotImplementedException($"No log reader implemented for Uri scheme \"{baseUri.Scheme}\""); throw new NotImplementedException($"No log reader implemented for Uri scheme \"{baseUri.Scheme}\"");

View File

@ -14,7 +14,7 @@ namespace IW4MAdmin.Application.Factories
internal class GameServerInstanceFactory : IGameServerInstanceFactory internal class GameServerInstanceFactory : IGameServerInstanceFactory
{ {
private readonly ITranslationLookup _translationLookup; private readonly ITranslationLookup _translationLookup;
private readonly IMetaService _metaService; private readonly IMetaServiceV2 _metaService;
private readonly IServiceProvider _serviceProvider; private readonly IServiceProvider _serviceProvider;
/// <summary> /// <summary>
@ -23,7 +23,7 @@ namespace IW4MAdmin.Application.Factories
/// <param name="translationLookup"></param> /// <param name="translationLookup"></param>
/// <param name="rconConnectionFactory"></param> /// <param name="rconConnectionFactory"></param>
public GameServerInstanceFactory(ITranslationLookup translationLookup, public GameServerInstanceFactory(ITranslationLookup translationLookup,
IMetaService metaService, IMetaServiceV2 metaService,
IServiceProvider serviceProvider) IServiceProvider serviceProvider)
{ {
_translationLookup = translationLookup; _translationLookup = translationLookup;
@ -45,4 +45,4 @@ namespace IW4MAdmin.Application.Factories
_serviceProvider.GetRequiredService<ILookupCache<EFServer>>()); _serviceProvider.GetRequiredService<ILookupCache<EFServer>>());
} }
} }
} }

View File

@ -21,7 +21,7 @@ namespace IW4MAdmin.Application.IO
{ {
_reader = gameLogReaderFactory.CreateGameLogReader(gameLogUris, server.EventParser); _reader = gameLogReaderFactory.CreateGameLogReader(gameLogUris, server.EventParser);
_server = server; _server = server;
_ignoreBots = server?.Manager.GetApplicationSettings().Configuration().IgnoreBots ?? false; _ignoreBots = server.Manager.GetApplicationSettings().Configuration()?.IgnoreBots ?? false;
_logger = logger; _logger = logger;
} }
@ -69,7 +69,7 @@ namespace IW4MAdmin.Application.IO
return; return;
} }
var events = await _reader.ReadEventsFromLog(fileDiff, previousFileSize); var events = await _reader.ReadEventsFromLog(fileDiff, previousFileSize, _server);
foreach (var gameEvent in events) foreach (var gameEvent in events)
{ {

View File

@ -28,7 +28,7 @@ namespace IW4MAdmin.Application.IO
_logger = logger; _logger = logger;
} }
public async Task<IEnumerable<GameEvent>> ReadEventsFromLog(long fileSizeDiff, long startPosition) public async Task<IEnumerable<GameEvent>> ReadEventsFromLog(long fileSizeDiff, long startPosition, Server server = null)
{ {
// allocate the bytes for the new log lines // allocate the bytes for the new log lines
List<string> logLines = new List<string>(); List<string> logLines = new List<string>();

View File

@ -34,7 +34,7 @@ namespace IW4MAdmin.Application.IO
public int UpdateInterval => 500; public int UpdateInterval => 500;
public async Task<IEnumerable<GameEvent>> ReadEventsFromLog(long fileSizeDiff, long startPosition) public async Task<IEnumerable<GameEvent>> ReadEventsFromLog(long fileSizeDiff, long startPosition, Server server = null)
{ {
var events = new List<GameEvent>(); var events = new List<GameEvent>();
var response = await _logServerApi.Log(_safeLogPath, lastKey); var response = await _logServerApi.Log(_safeLogPath, lastKey);

View File

@ -0,0 +1,163 @@
using SharedLibraryCore;
using SharedLibraryCore.Interfaces;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Threading;
using System.Threading.Tasks;
using Integrations.Cod;
using Microsoft.Extensions.Logging;
using Serilog.Context;
using ILogger = Microsoft.Extensions.Logging.ILogger;
namespace IW4MAdmin.Application.IO
{
/// <summary>
/// provides capability of reading log files over udp
/// </summary>
class NetworkGameLogReader : IGameLogReader
{
private readonly IEventParser _eventParser;
private readonly ILogger _logger;
private readonly Uri _uri;
private static readonly NetworkLogState State = new();
private bool _stateRegistered;
private CancellationToken _token;
public NetworkGameLogReader(IReadOnlyList<Uri> uris, IEventParser parser, ILogger<NetworkGameLogReader> logger)
{
_eventParser = parser;
_uri = uris[0];
_logger = logger;
}
public long Length => -1;
public int UpdateInterval => 150;
public Task<IEnumerable<GameEvent>> ReadEventsFromLog(long fileSizeDiff, long startPosition,
Server server = null)
{
// todo: other games might support this
var serverEndpoint = (server?.RemoteConnection as CodRConConnection)?.Endpoint;
if (serverEndpoint is null)
{
return Task.FromResult(Enumerable.Empty<GameEvent>());
}
if (!_stateRegistered && !State.EndPointExists(serverEndpoint))
{
try
{
var client = State.RegisterEndpoint(serverEndpoint, BuildLocalEndpoint()).Client;
_stateRegistered = true;
_token = server.Manager.CancellationToken;
if (client == null)
{
using (LogContext.PushProperty("Server", server.ToString()))
{
_logger.LogInformation("Not registering {Name} socket because it is already bound",
nameof(NetworkGameLogReader));
}
return Task.FromResult(Enumerable.Empty<GameEvent>());
}
Task.Run(async () => await ReadNetworkData(client, _token), _token);
}
catch (Exception ex)
{
_logger.LogError(ex, "Could not register {Name} endpoint {Endpoint}",
nameof(NetworkGameLogReader), _uri);
throw;
}
}
var events = new List<GameEvent>();
foreach (var logData in State.GetServerLogData(serverEndpoint)
.Select(log => Utilities.EncodingType.GetString(log)))
{
if (string.IsNullOrWhiteSpace(logData))
{
return Task.FromResult(Enumerable.Empty<GameEvent>());
}
var lines = logData
.Split('\n')
.Where(line => line.Length > 0 && !line.Contains('ÿ'));
foreach (var eventLine in lines)
{
try
{
// this trim end should hopefully fix the nasty runaway regex
var gameEvent = _eventParser.GenerateGameEvent(eventLine.TrimEnd('\r'));
events.Add(gameEvent);
}
catch (Exception ex)
{
_logger.LogError(ex, "Could not properly parse event line from http {EventLine}",
eventLine);
}
}
}
return Task.FromResult((IEnumerable<GameEvent>)events);
}
private async Task ReadNetworkData(UdpClient client, CancellationToken token)
{
while (!token.IsCancellationRequested)
{
// get more data
IPEndPoint remoteEndpoint = null;
byte[] bufferedData = null;
if (client == null)
{
// we already have a socket listening on this port for data, so we don't need to run another thread
break;
}
try
{
var result = await client.ReceiveAsync(_token);
remoteEndpoint = result.RemoteEndPoint;
bufferedData = result.Buffer;
}
catch (OperationCanceledException)
{
_logger.LogDebug("Stopping network log receive");
}
catch (Exception ex)
{
_logger.LogError(ex, "Could not receive lines for {LogReader}", nameof(NetworkGameLogReader));
}
if (bufferedData != null)
{
State.QueueServerLogData(remoteEndpoint, bufferedData);
}
}
}
private IPEndPoint BuildLocalEndpoint()
{
try
{
return new IPEndPoint(Dns.GetHostAddresses(_uri.Host).First(), _uri.Port);
}
catch (Exception ex)
{
_logger.LogError(ex, "Could not setup {LogReader} endpoint", nameof(NetworkGameLogReader));
throw;
}
}
}
}

View File

@ -0,0 +1,138 @@
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Threading;
namespace IW4MAdmin.Application.IO;
public class NetworkLogState : Dictionary<IPEndPoint, UdpClientState>
{
public UdpClientState RegisterEndpoint(IPEndPoint serverEndpoint, IPEndPoint localEndpoint)
{
try
{
lock (this)
{
if (!ContainsKey(serverEndpoint))
{
Add(serverEndpoint, new UdpClientState { Client = new UdpClient(localEndpoint) });
}
}
}
catch (SocketException ex) when (ex.SocketErrorCode == SocketError.AddressAlreadyInUse)
{
lock (this)
{
// we don't add the udp client because it already exists (listening to multiple servers from one socket)
Add(serverEndpoint, new UdpClientState());
}
}
return this[serverEndpoint];
}
public List<byte[]> GetServerLogData(IPEndPoint serverEndpoint)
{
try
{
var state = this[serverEndpoint];
if (state == null)
{
return new List<byte[]>();
}
// it's possible that we could be trying to read and write to the queue simultaneously so we need to wait
this[serverEndpoint].OnAction.Wait();
var data = new List<byte[]>();
while (this[serverEndpoint].AvailableLogData.Count > 0)
{
data.Add(this[serverEndpoint].AvailableLogData.Dequeue());
}
return data;
}
finally
{
if (this[serverEndpoint].OnAction.CurrentCount == 0)
{
this[serverEndpoint].OnAction.Release(1);
}
}
}
public void QueueServerLogData(IPEndPoint serverEndpoint, byte[] data)
{
var endpoint = Keys.FirstOrDefault(key =>
Equals(key.Address, serverEndpoint.Address) && key.Port == serverEndpoint.Port);
try
{
if (endpoint == null)
{
return;
}
// currently our expected start and end characters
var startsWithPrefix = StartsWith(data, "ÿÿÿÿprint\n");
var endsWithDelimiter = data[^1] == '\n';
// we have the data we expected
if (!startsWithPrefix || !endsWithDelimiter)
{
return;
}
// it's possible that we could be trying to read and write to the queue simultaneously so we need to wait
this[endpoint].OnAction.Wait();
this[endpoint].AvailableLogData.Enqueue(data);
}
finally
{
if (endpoint != null && this[endpoint].OnAction.CurrentCount == 0)
{
this[endpoint].OnAction.Release(1);
}
}
}
public bool EndPointExists(IPEndPoint serverEndpoint)
{
lock (this)
{
return ContainsKey(serverEndpoint);
}
}
private static bool StartsWith(byte[] sourceArray, string match)
{
if (sourceArray is null)
{
return false;
}
if (match.Length > sourceArray.Length)
{
return false;
}
return !match.Where((t, i) => sourceArray[i] != (byte)t).Any();
}
}
public class UdpClientState
{
public UdpClient Client { get; set; }
public Queue<byte[]> AvailableLogData { get; } = new();
public SemaphoreSlim OnAction { get; } = new(1, 1);
~UdpClientState()
{
OnAction.Dispose();
Client?.Dispose();
}
}

View File

@ -35,7 +35,7 @@ namespace IW4MAdmin
private static readonly SharedLibraryCore.Localization.TranslationLookup loc = Utilities.CurrentLocalization.LocalizationIndex; private static readonly SharedLibraryCore.Localization.TranslationLookup loc = Utilities.CurrentLocalization.LocalizationIndex;
public GameLogEventDetection LogEvent; public GameLogEventDetection LogEvent;
private readonly ITranslationLookup _translationLookup; private readonly ITranslationLookup _translationLookup;
private readonly IMetaService _metaService; private readonly IMetaServiceV2 _metaService;
private const int REPORT_FLAG_COUNT = 4; private const int REPORT_FLAG_COUNT = 4;
private long lastGameTime = 0; private long lastGameTime = 0;
@ -49,7 +49,7 @@ namespace IW4MAdmin
ServerConfiguration serverConfiguration, ServerConfiguration serverConfiguration,
CommandConfiguration commandConfiguration, CommandConfiguration commandConfiguration,
ITranslationLookup lookup, ITranslationLookup lookup,
IMetaService metaService, IMetaServiceV2 metaService,
IServiceProvider serviceProvider, IServiceProvider serviceProvider,
IClientNoticeMessageFormatter messageFormatter, IClientNoticeMessageFormatter messageFormatter,
ILookupCache<EFServer> serverCache) : base(serviceProvider.GetRequiredService<ILogger<Server>>(), ILookupCache<EFServer> serverCache) : base(serviceProvider.GetRequiredService<ILogger<Server>>(),
@ -236,7 +236,7 @@ namespace IW4MAdmin
private async Task CreatePluginTask(IPlugin plugin, GameEvent gameEvent) private async Task CreatePluginTask(IPlugin plugin, GameEvent gameEvent)
{ {
// we don't want to run the events on parser plugins // we don't want to run the events on parser plugins
if (plugin is ScriptPlugin scriptPlugin && scriptPlugin.IsParser) if (plugin is ScriptPlugin { IsParser: true })
{ {
return; return;
} }
@ -248,6 +248,11 @@ namespace IW4MAdmin
{ {
await plugin.OnEventAsync(gameEvent, this).WithWaitCancellation(tokenSource.Token); await plugin.OnEventAsync(gameEvent, this).WithWaitCancellation(tokenSource.Token);
} }
catch (OperationCanceledException)
{
ServerLogger.LogWarning("Timed out executing event {EventType} for {Plugin}", gameEvent.Type,
plugin.Name);
}
catch (Exception ex) catch (Exception ex)
{ {
Console.WriteLine(loc["SERVER_PLUGIN_ERROR"].FormatExt(plugin.Name, ex.GetType().Name)); Console.WriteLine(loc["SERVER_PLUGIN_ERROR"].FormatExt(plugin.Name, ex.GetType().Name));
@ -290,7 +295,7 @@ namespace IW4MAdmin
} }
} }
if (E.Type == GameEvent.EventType.ConnectionLost) else if (E.Type == GameEvent.EventType.ConnectionLost)
{ {
var exception = E.Extra as Exception; var exception = E.Extra as Exception;
ServerLogger.LogError(exception, ServerLogger.LogError(exception,
@ -304,7 +309,7 @@ namespace IW4MAdmin
Throttled = true; Throttled = true;
} }
if (E.Type == GameEvent.EventType.ConnectionRestored) else if (E.Type == GameEvent.EventType.ConnectionRestored)
{ {
ServerLogger.LogInformation( ServerLogger.LogInformation(
"Connection restored with {server}", ToString()); "Connection restored with {server}", ToString());
@ -316,13 +321,13 @@ namespace IW4MAdmin
if (!string.IsNullOrEmpty(CustomSayName)) if (!string.IsNullOrEmpty(CustomSayName))
{ {
await this.SetDvarAsync("sv_sayname", CustomSayName); await this.SetDvarAsync("sv_sayname", CustomSayName, Manager.CancellationToken);
} }
Throttled = false; Throttled = false;
} }
if (E.Type == GameEvent.EventType.ChangePermission) else if (E.Type == GameEvent.EventType.ChangePermission)
{ {
var newPermission = (Permission) E.Extra; var newPermission = (Permission) E.Extra;
ServerLogger.LogInformation("{origin} is setting {target} to permission level {newPermission}", ServerLogger.LogInformation("{origin} is setting {target} to permission level {newPermission}",
@ -345,7 +350,8 @@ namespace IW4MAdmin
Time = DateTime.UtcNow Time = DateTime.UtcNow
}); });
var clientTag = await _metaService.GetPersistentMeta(EFMeta.ClientTag, E.Origin); var clientTag = await _metaService.GetPersistentMetaByLookup(EFMeta.ClientTagV2,
EFMeta.ClientTagNameV2, E.Origin.ClientId, Manager.CancellationToken);
if (clientTag?.LinkedMeta != null) if (clientTag?.LinkedMeta != null)
{ {
@ -477,7 +483,7 @@ namespace IW4MAdmin
else if (E.Type == GameEvent.EventType.Unflag) else if (E.Type == GameEvent.EventType.Unflag)
{ {
var unflagPenalty = new EFPenalty() var unflagPenalty = new EFPenalty
{ {
Type = EFPenalty.PenaltyType.Unflag, Type = EFPenalty.PenaltyType.Unflag,
Expires = DateTime.UtcNow, Expires = DateTime.UtcNow,
@ -489,7 +495,8 @@ namespace IW4MAdmin
}; };
E.Target.SetLevel(Permission.User, E.Origin); E.Target.SetLevel(Permission.User, E.Origin);
await Manager.GetPenaltyService().RemoveActivePenalties(E.Target.AliasLinkId); await Manager.GetPenaltyService().RemoveActivePenalties(E.Target.AliasLinkId, E.Target.NetworkId,
E.Target.CurrentAlias?.IPAddress);
await Manager.GetPenaltyService().Create(unflagPenalty); await Manager.GetPenaltyService().Create(unflagPenalty);
} }
@ -562,8 +569,10 @@ namespace IW4MAdmin
Time = DateTime.UtcNow Time = DateTime.UtcNow
}); });
await _metaService.AddPersistentMeta("LastMapPlayed", CurrentMap.Alias, E.Origin); await _metaService.SetPersistentMeta("LastMapPlayed", CurrentMap.Alias, E.Origin.ClientId,
await _metaService.AddPersistentMeta("LastServerPlayed", E.Owner.Hostname, E.Origin); Manager.CancellationToken);
await _metaService.SetPersistentMeta("LastServerPlayed", E.Owner.Hostname, E.Origin.ClientId,
Manager.CancellationToken);
} }
else if (E.Type == GameEvent.EventType.PreDisconnect) else if (E.Type == GameEvent.EventType.PreDisconnect)
@ -614,7 +623,7 @@ namespace IW4MAdmin
await OnClientUpdate(E.Origin); await OnClientUpdate(E.Origin);
} }
if (E.Type == GameEvent.EventType.Say) else if (E.Type == GameEvent.EventType.Say)
{ {
if (E.Data?.Length > 0) if (E.Data?.Length > 0)
{ {
@ -644,7 +653,7 @@ namespace IW4MAdmin
} }
} }
if (E.Type == GameEvent.EventType.MapChange) else if (E.Type == GameEvent.EventType.MapChange)
{ {
ServerLogger.LogInformation("New map loaded - {clientCount} active players", ClientNum); ServerLogger.LogInformation("New map loaded - {clientCount} active players", ClientNum);
@ -685,7 +694,7 @@ namespace IW4MAdmin
} }
} }
if (E.Type == GameEvent.EventType.MapEnd) else if (E.Type == GameEvent.EventType.MapEnd)
{ {
ServerLogger.LogInformation("Game ending..."); ServerLogger.LogInformation("Game ending...");
@ -695,18 +704,23 @@ namespace IW4MAdmin
} }
} }
if (E.Type == GameEvent.EventType.Tell) else if (E.Type == GameEvent.EventType.Tell)
{ {
await Tell(E.Message, E.Target); await Tell(E.Message, E.Target);
} }
if (E.Type == GameEvent.EventType.Broadcast) else if (E.Type == GameEvent.EventType.Broadcast)
{ {
if (!Utilities.IsDevelopment && E.Data != null) // hides broadcast when in development mode if (!Utilities.IsDevelopment && E.Data != null) // hides broadcast when in development mode
{ {
await E.Owner.ExecuteCommandAsync(E.Data); await E.Owner.ExecuteCommandAsync(E.Data);
} }
} }
else if (E.Type == GameEvent.EventType.JoinTeam)
{
E.Origin.UpdateTeam(E.Extra as string);
}
lock (ChatHistory) lock (ChatHistory)
{ {
@ -777,7 +791,7 @@ namespace IW4MAdmin
async Task<List<EFClient>[]> PollPlayersAsync() async Task<List<EFClient>[]> PollPlayersAsync()
{ {
var currentClients = GetClientsAsList(); var currentClients = GetClientsAsList();
var statusResponse = (await this.GetStatusAsync()); var statusResponse = await this.GetStatusAsync(Manager.CancellationToken);
var polledClients = statusResponse.Clients.AsEnumerable(); var polledClients = statusResponse.Clients.AsEnumerable();
if (Manager.GetApplicationSettings().Configuration().IgnoreBots) if (Manager.GetApplicationSettings().Configuration().IgnoreBots)
@ -1103,7 +1117,7 @@ namespace IW4MAdmin
RemoteConnection = RConConnectionFactory.CreateConnection(ResolvedIpEndPoint, Password, RconParser.RConEngine); RemoteConnection = RConConnectionFactory.CreateConnection(ResolvedIpEndPoint, Password, RconParser.RConEngine);
RemoteConnection.SetConfiguration(RconParser); RemoteConnection.SetConfiguration(RconParser);
var version = await this.GetMappedDvarValueOrDefaultAsync<string>("version"); var version = await this.GetMappedDvarValueOrDefaultAsync<string>("version", token: Manager.CancellationToken);
Version = version.Value; Version = version.Value;
GameName = Utilities.GetGame(version.Value ?? RconParser.Version); GameName = Utilities.GetGame(version.Value ?? RconParser.Version);
@ -1120,7 +1134,7 @@ namespace IW4MAdmin
Version = RconParser.Version; Version = RconParser.Version;
} }
var svRunning = await this.GetMappedDvarValueOrDefaultAsync<string>("sv_running"); var svRunning = await this.GetMappedDvarValueOrDefaultAsync<string>("sv_running", token: Manager.CancellationToken);
if (!string.IsNullOrEmpty(svRunning.Value) && svRunning.Value != "1") if (!string.IsNullOrEmpty(svRunning.Value) && svRunning.Value != "1")
{ {
@ -1129,27 +1143,28 @@ namespace IW4MAdmin
var infoResponse = RconParser.Configuration.CommandPrefixes.RConGetInfo != null ? await this.GetInfoAsync() : null; var infoResponse = RconParser.Configuration.CommandPrefixes.RConGetInfo != null ? await this.GetInfoAsync() : null;
string hostname = (await this.GetMappedDvarValueOrDefaultAsync<string>("sv_hostname", "hostname", infoResponse)).Value; var hostname = (await this.GetMappedDvarValueOrDefaultAsync<string>("sv_hostname", "hostname", infoResponse, token: Manager.CancellationToken)).Value;
string mapname = (await this.GetMappedDvarValueOrDefaultAsync<string>("mapname", infoResponse: infoResponse)).Value; var mapname = (await this.GetMappedDvarValueOrDefaultAsync<string>("mapname", infoResponse: infoResponse, token: Manager.CancellationToken)).Value;
int maxplayers = (await this.GetMappedDvarValueOrDefaultAsync<int>("sv_maxclients", infoResponse: infoResponse)).Value; var maxplayers = (await this.GetMappedDvarValueOrDefaultAsync<int>("sv_maxclients", infoResponse: infoResponse, token: Manager.CancellationToken)).Value;
string gametype = (await this.GetMappedDvarValueOrDefaultAsync<string>("g_gametype", "gametype", infoResponse)).Value; var gametype = (await this.GetMappedDvarValueOrDefaultAsync<string>("g_gametype", "gametype", infoResponse, token: Manager.CancellationToken)).Value;
var basepath = await this.GetMappedDvarValueOrDefaultAsync<string>("fs_basepath"); var basepath = await this.GetMappedDvarValueOrDefaultAsync<string>("fs_basepath", token: Manager.CancellationToken);
var basegame = await this.GetMappedDvarValueOrDefaultAsync<string>("fs_basegame"); var basegame = await this.GetMappedDvarValueOrDefaultAsync<string>("fs_basegame", token: Manager.CancellationToken);
var homepath = await this.GetMappedDvarValueOrDefaultAsync<string>("fs_homepath"); var homepath = await this.GetMappedDvarValueOrDefaultAsync<string>("fs_homepath", token: Manager.CancellationToken);
var game = (await this.GetMappedDvarValueOrDefaultAsync<string>("fs_game", infoResponse: infoResponse)); var game = await this.GetMappedDvarValueOrDefaultAsync<string>("fs_game", infoResponse: infoResponse, token: Manager.CancellationToken);
var logfile = await this.GetMappedDvarValueOrDefaultAsync<string>("g_log"); var logfile = await this.GetMappedDvarValueOrDefaultAsync<string>("g_log", token: Manager.CancellationToken);
var logsync = await this.GetMappedDvarValueOrDefaultAsync<int>("g_logsync"); var logsync = await this.GetMappedDvarValueOrDefaultAsync<int>("g_logsync", token: Manager.CancellationToken);
var ip = await this.GetMappedDvarValueOrDefaultAsync<string>("net_ip"); var ip = await this.GetMappedDvarValueOrDefaultAsync<string>("net_ip", token: Manager.CancellationToken);
var gamePassword = await this.GetMappedDvarValueOrDefaultAsync("g_password", overrideDefault: ""); var gamePassword = await this.GetMappedDvarValueOrDefaultAsync("g_password", overrideDefault: "", token: Manager.CancellationToken);
if (Manager.GetApplicationSettings().Configuration().EnableCustomSayName) if (Manager.GetApplicationSettings().Configuration().EnableCustomSayName)
{ {
await this.SetDvarAsync("sv_sayname", Manager.GetApplicationSettings().Configuration().CustomSayName); await this.SetDvarAsync("sv_sayname", Manager.GetApplicationSettings().Configuration().CustomSayName,
Manager.CancellationToken);
} }
try try
{ {
var website = await this.GetMappedDvarValueOrDefaultAsync<string>("_website"); var website = await this.GetMappedDvarValueOrDefaultAsync<string>("_website", token: Manager.CancellationToken);
// this occurs for games that don't give us anything back when // this occurs for games that don't give us anything back when
// the dvar is not set // the dvar is not set
@ -1195,14 +1210,14 @@ namespace IW4MAdmin
if (logsync.Value == 0) if (logsync.Value == 0)
{ {
await this.SetDvarAsync("g_logsync", 2); // set to 2 for continous in other games, clamps to 1 for IW4 await this.SetDvarAsync("g_logsync", 2, Manager.CancellationToken); // set to 2 for continous in other games, clamps to 1 for IW4
needsRestart = true; needsRestart = true;
} }
if (string.IsNullOrWhiteSpace(logfile.Value)) if (string.IsNullOrWhiteSpace(logfile.Value))
{ {
logfile.Value = "games_mp.log"; logfile.Value = "games_mp.log";
await this.SetDvarAsync("g_log", logfile.Value); await this.SetDvarAsync("g_log", logfile.Value, Manager.CancellationToken);
needsRestart = true; needsRestart = true;
} }
@ -1214,7 +1229,7 @@ namespace IW4MAdmin
} }
// this DVAR isn't set until the a map is loaded // this DVAR isn't set until the a map is loaded
await this.SetDvarAsync("logfile", 2); await this.SetDvarAsync("logfile", 2, Manager.CancellationToken);
} }
CustomCallback = await ScriptLoaded(); CustomCallback = await ScriptLoaded();
@ -1439,7 +1454,6 @@ namespace IW4MAdmin
Offender = targetClient, Offender = targetClient,
Offense = reason, Offense = reason,
Punisher = originClient, Punisher = originClient,
Link = targetClient.AliasLink,
IsEvadedOffense = isEvade IsEvadedOffense = isEvade
}; };
@ -1474,7 +1488,8 @@ namespace IW4MAdmin
ServerLogger.LogDebug("Creating unban penalty for {targetClient}", targetClient.ToString()); ServerLogger.LogDebug("Creating unban penalty for {targetClient}", targetClient.ToString());
targetClient.SetLevel(Permission.User, originClient); targetClient.SetLevel(Permission.User, originClient);
await Manager.GetPenaltyService().RemoveActivePenalties(targetClient.AliasLink.AliasLinkId); await Manager.GetPenaltyService().RemoveActivePenalties(targetClient.AliasLink.AliasLinkId,
targetClient.NetworkId, targetClient.CurrentAlias?.IPAddress);
await Manager.GetPenaltyService().Create(unbanPenalty); await Manager.GetPenaltyService().Create(unbanPenalty);
} }

View File

@ -18,6 +18,7 @@ using SharedLibraryCore.Repositories;
using SharedLibraryCore.Services; using SharedLibraryCore.Services;
using Stats.Dtos; using Stats.Dtos;
using System; using System;
using System.IO;
using System.Linq; using System.Linq;
using System.Net.Http; using System.Net.Http;
using System.Text; using System.Text;
@ -112,10 +113,11 @@ namespace IW4MAdmin.Application
var tasks = new[] var tasks = new[]
{ {
versionChecker.CheckVersion(), versionChecker.CheckVersion(),
_serverManager.Init(),
_applicationTask _applicationTask
}; };
await _serverManager.Init();
await Task.WhenAll(tasks); await Task.WhenAll(tasks);
} }
@ -138,11 +140,8 @@ namespace IW4MAdmin.Application
if (e is ConfigurationException configException) if (e is ConfigurationException configException)
{ {
if (translationLookup != null) Console.WriteLine("{{fileName}} contains an error."
{ .FormatExt(Path.GetFileName(configException.ConfigurationFileName)));
Console.WriteLine(translationLookup[configException.Message]
.FormatExt(configException.ConfigurationFileName));
}
foreach (var error in configException.Errors) foreach (var error in configException.Errors)
{ {
@ -154,6 +153,8 @@ namespace IW4MAdmin.Application
{ {
Console.WriteLine(e.Message); Console.WriteLine(e.Message);
} }
_serverManager?.Stop();
Console.WriteLine(exitMessage); Console.WriteLine(exitMessage);
await Console.In.ReadAsync(new char[1], 0, 1); await Console.In.ReadAsync(new char[1], 0, 1);
@ -277,7 +278,7 @@ namespace IW4MAdmin.Application
// register the native commands // register the native commands
foreach (var commandType in typeof(SharedLibraryCore.Commands.QuitCommand).Assembly.GetTypes() foreach (var commandType in typeof(SharedLibraryCore.Commands.QuitCommand).Assembly.GetTypes()
.Concat(typeof(Program).Assembly.GetTypes().Where(type => type.Namespace == "IW4MAdmin.Application.Commands")) .Concat(typeof(Program).Assembly.GetTypes().Where(type => type.Namespace?.StartsWith("IW4MAdmin.Application.Commands") ?? false))
.Where(command => command.BaseType == typeof(Command))) .Where(command => command.BaseType == typeof(Command)))
{ {
defaultLogger.LogDebug("Registered native command type {Name}", commandType.Name); defaultLogger.LogDebug("Registered native command type {Name}", commandType.Name);
@ -347,7 +348,7 @@ namespace IW4MAdmin.Application
await defaultConfigHandler.BuildAsync(); await defaultConfigHandler.BuildAsync();
var commandConfigHandler = new BaseConfigurationHandler<CommandConfiguration>("CommandConfiguration"); var commandConfigHandler = new BaseConfigurationHandler<CommandConfiguration>("CommandConfiguration");
await commandConfigHandler.BuildAsync(); await commandConfigHandler.BuildAsync();
var statsCommandHandler = new BaseConfigurationHandler<StatsConfiguration>(); var statsCommandHandler = new BaseConfigurationHandler<StatsConfiguration>("StatsPluginSettings");
await statsCommandHandler.BuildAsync(); await statsCommandHandler.BuildAsync();
var defaultConfig = defaultConfigHandler.Configuration(); var defaultConfig = defaultConfigHandler.Configuration();
var appConfig = appConfigHandler.Configuration(); var appConfig = appConfigHandler.Configuration();
@ -404,7 +405,10 @@ namespace IW4MAdmin.Application
.AddSingleton<IScriptCommandFactory, ScriptCommandFactory>() .AddSingleton<IScriptCommandFactory, ScriptCommandFactory>()
.AddSingleton<IAuditInformationRepository, AuditInformationRepository>() .AddSingleton<IAuditInformationRepository, AuditInformationRepository>()
.AddSingleton<IEntityService<EFClient>, ClientService>() .AddSingleton<IEntityService<EFClient>, ClientService>()
#pragma warning disable CS0618
.AddSingleton<IMetaService, MetaService>() .AddSingleton<IMetaService, MetaService>()
#pragma warning restore CS0618
.AddSingleton<IMetaServiceV2, MetaServiceV2>()
.AddSingleton<ClientService>() .AddSingleton<ClientService>()
.AddSingleton<PenaltyService>() .AddSingleton<PenaltyService>()
.AddSingleton<ChangeHistoryService>() .AddSingleton<ChangeHistoryService>()
@ -418,6 +422,7 @@ namespace IW4MAdmin.Application
UpdatedAliasResourceQueryHelper>() UpdatedAliasResourceQueryHelper>()
.AddSingleton<IResourceQueryHelper<ChatSearchQuery, MessageResponse>, ChatResourceQueryHelper>() .AddSingleton<IResourceQueryHelper<ChatSearchQuery, MessageResponse>, ChatResourceQueryHelper>()
.AddSingleton<IResourceQueryHelper<ClientPaginationRequest, ConnectionHistoryResponse>, ConnectionsResourceQueryHelper>() .AddSingleton<IResourceQueryHelper<ClientPaginationRequest, ConnectionHistoryResponse>, ConnectionsResourceQueryHelper>()
.AddSingleton<IResourceQueryHelper<ClientPaginationRequest, PermissionLevelChangedResponse>, PermissionLevelChangedResourceQueryHelper>()
.AddTransient<IParserPatternMatcher, ParserPatternMatcher>() .AddTransient<IParserPatternMatcher, ParserPatternMatcher>()
.AddSingleton<IRemoteAssemblyHandler, RemoteAssemblyHandler>() .AddSingleton<IRemoteAssemblyHandler, RemoteAssemblyHandler>()
.AddSingleton<IMasterCommunication, MasterCommunication>() .AddSingleton<IMasterCommunication, MasterCommunication>()

View File

@ -5,6 +5,7 @@ using SharedLibraryCore.Interfaces;
using SharedLibraryCore.QueryHelper; using SharedLibraryCore.QueryHelper;
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging;
using ILogger = Microsoft.Extensions.Logging.ILogger; using ILogger = Microsoft.Extensions.Logging.ILogger;
@ -15,19 +16,28 @@ namespace IW4MAdmin.Application.Meta
{ {
private readonly ILogger _logger; private readonly ILogger _logger;
private ITranslationLookup _transLookup; private ITranslationLookup _transLookup;
private readonly IMetaService _metaService; private readonly IMetaServiceV2 _metaService;
private readonly IEntityService<EFClient> _clientEntityService; private readonly IEntityService<EFClient> _clientEntityService;
private readonly IResourceQueryHelper<ClientPaginationRequest, ReceivedPenaltyResponse> _receivedPenaltyHelper; private readonly IResourceQueryHelper<ClientPaginationRequest, ReceivedPenaltyResponse> _receivedPenaltyHelper;
private readonly IResourceQueryHelper<ClientPaginationRequest, AdministeredPenaltyResponse> _administeredPenaltyHelper;
private readonly IResourceQueryHelper<ClientPaginationRequest, AdministeredPenaltyResponse>
_administeredPenaltyHelper;
private readonly IResourceQueryHelper<ClientPaginationRequest, UpdatedAliasResponse> _updatedAliasHelper; private readonly IResourceQueryHelper<ClientPaginationRequest, UpdatedAliasResponse> _updatedAliasHelper;
private readonly IResourceQueryHelper<ClientPaginationRequest, ConnectionHistoryResponse> private readonly IResourceQueryHelper<ClientPaginationRequest, ConnectionHistoryResponse>
_connectionHistoryHelper; _connectionHistoryHelper;
public MetaRegistration(ILogger<MetaRegistration> logger, IMetaService metaService, ITranslationLookup transLookup, IEntityService<EFClient> clientEntityService, private readonly IResourceQueryHelper<ClientPaginationRequest, PermissionLevelChangedResponse>
_permissionLevelHelper;
public MetaRegistration(ILogger<MetaRegistration> logger, IMetaServiceV2 metaService,
ITranslationLookup transLookup, IEntityService<EFClient> clientEntityService,
IResourceQueryHelper<ClientPaginationRequest, ReceivedPenaltyResponse> receivedPenaltyHelper, IResourceQueryHelper<ClientPaginationRequest, ReceivedPenaltyResponse> receivedPenaltyHelper,
IResourceQueryHelper<ClientPaginationRequest, AdministeredPenaltyResponse> administeredPenaltyHelper, IResourceQueryHelper<ClientPaginationRequest, AdministeredPenaltyResponse> administeredPenaltyHelper,
IResourceQueryHelper<ClientPaginationRequest, UpdatedAliasResponse> updatedAliasHelper, IResourceQueryHelper<ClientPaginationRequest, UpdatedAliasResponse> updatedAliasHelper,
IResourceQueryHelper<ClientPaginationRequest, ConnectionHistoryResponse> connectionHistoryHelper) IResourceQueryHelper<ClientPaginationRequest, ConnectionHistoryResponse> connectionHistoryHelper,
IResourceQueryHelper<ClientPaginationRequest, PermissionLevelChangedResponse> permissionLevelHelper)
{ {
_logger = logger; _logger = logger;
_transLookup = transLookup; _transLookup = transLookup;
@ -37,21 +47,31 @@ namespace IW4MAdmin.Application.Meta
_administeredPenaltyHelper = administeredPenaltyHelper; _administeredPenaltyHelper = administeredPenaltyHelper;
_updatedAliasHelper = updatedAliasHelper; _updatedAliasHelper = updatedAliasHelper;
_connectionHistoryHelper = connectionHistoryHelper; _connectionHistoryHelper = connectionHistoryHelper;
_permissionLevelHelper = permissionLevelHelper;
} }
public void Register() public void Register()
{ {
_metaService.AddRuntimeMeta<ClientPaginationRequest, InformationResponse>(MetaType.Information, GetProfileMeta); _metaService.AddRuntimeMeta<ClientPaginationRequest, InformationResponse>(MetaType.Information,
_metaService.AddRuntimeMeta<ClientPaginationRequest, ReceivedPenaltyResponse>(MetaType.ReceivedPenalty, GetReceivedPenaltiesMeta); GetProfileMeta);
_metaService.AddRuntimeMeta<ClientPaginationRequest, AdministeredPenaltyResponse>(MetaType.Penalized, GetAdministeredPenaltiesMeta); _metaService.AddRuntimeMeta<ClientPaginationRequest, ReceivedPenaltyResponse>(MetaType.ReceivedPenalty,
_metaService.AddRuntimeMeta<ClientPaginationRequest, UpdatedAliasResponse>(MetaType.AliasUpdate, GetUpdatedAliasMeta); GetReceivedPenaltiesMeta);
_metaService.AddRuntimeMeta<ClientPaginationRequest, ConnectionHistoryResponse>(MetaType.ConnectionHistory, GetConnectionHistoryMeta); _metaService.AddRuntimeMeta<ClientPaginationRequest, AdministeredPenaltyResponse>(MetaType.Penalized,
GetAdministeredPenaltiesMeta);
_metaService.AddRuntimeMeta<ClientPaginationRequest, UpdatedAliasResponse>(MetaType.AliasUpdate,
GetUpdatedAliasMeta);
_metaService.AddRuntimeMeta<ClientPaginationRequest, ConnectionHistoryResponse>(MetaType.ConnectionHistory,
GetConnectionHistoryMeta);
_metaService.AddRuntimeMeta<ClientPaginationRequest, PermissionLevelChangedResponse>(
MetaType.PermissionLevel, GetPermissionLevelMeta);
} }
private async Task<IEnumerable<InformationResponse>> GetProfileMeta(ClientPaginationRequest request) private async Task<IEnumerable<InformationResponse>> GetProfileMeta(ClientPaginationRequest request,
CancellationToken cancellationToken = default)
{ {
var metaList = new List<InformationResponse>(); var metaList = new List<InformationResponse>();
var lastMapMeta = await _metaService.GetPersistentMeta("LastMapPlayed", new EFClient() { ClientId = request.ClientId }); var lastMapMeta =
await _metaService.GetPersistentMeta("LastMapPlayed", request.ClientId, cancellationToken);
if (lastMapMeta != null) if (lastMapMeta != null)
{ {
@ -68,7 +88,8 @@ namespace IW4MAdmin.Application.Meta
}); });
} }
var lastServerMeta = await _metaService.GetPersistentMeta("LastServerPlayed", new EFClient() { ClientId = request.ClientId }); var lastServerMeta =
await _metaService.GetPersistentMeta("LastServerPlayed", request.ClientId, cancellationToken);
if (lastServerMeta != null) if (lastServerMeta != null)
{ {
@ -89,7 +110,7 @@ namespace IW4MAdmin.Application.Meta
if (client == null) if (client == null)
{ {
_logger.LogWarning("No client found with id {clientId} when generating profile meta", request.ClientId); _logger.LogWarning("No client found with id {ClientId} when generating profile meta", request.ClientId);
return metaList; return metaList;
} }
@ -130,7 +151,8 @@ namespace IW4MAdmin.Application.Meta
{ {
ClientId = client.ClientId, ClientId = client.ClientId,
Key = Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_CONNECTIONS"], Key = Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_CONNECTIONS"],
Value = client.Connections.ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)), Value = client.Connections.ToString("#,##0",
new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)),
ShouldDisplay = true, ShouldDisplay = true,
Column = 1, Column = 1,
Order = 3, Order = 3,
@ -141,7 +163,9 @@ namespace IW4MAdmin.Application.Meta
{ {
ClientId = client.ClientId, ClientId = client.ClientId,
Key = Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_MASKED"], Key = Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_MASKED"],
Value = client.Masked ? Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_TRUE"] : Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_FALSE"], Value = client.Masked
? Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_TRUE"]
: Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_FALSE"],
IsSensitive = true, IsSensitive = true,
Column = 1, Column = 1,
Order = 4, Order = 4,
@ -151,28 +175,39 @@ namespace IW4MAdmin.Application.Meta
return metaList; return metaList;
} }
private async Task<IEnumerable<ReceivedPenaltyResponse>> GetReceivedPenaltiesMeta(ClientPaginationRequest request) private async Task<IEnumerable<ReceivedPenaltyResponse>> GetReceivedPenaltiesMeta(
ClientPaginationRequest request, CancellationToken token = default)
{ {
var penalties = await _receivedPenaltyHelper.QueryResource(request); var penalties = await _receivedPenaltyHelper.QueryResource(request);
return penalties.Results; return penalties.Results;
} }
private async Task<IEnumerable<AdministeredPenaltyResponse>> GetAdministeredPenaltiesMeta(ClientPaginationRequest request) private async Task<IEnumerable<AdministeredPenaltyResponse>> GetAdministeredPenaltiesMeta(
ClientPaginationRequest request, CancellationToken token = default)
{ {
var penalties = await _administeredPenaltyHelper.QueryResource(request); var penalties = await _administeredPenaltyHelper.QueryResource(request);
return penalties.Results; return penalties.Results;
} }
private async Task<IEnumerable<UpdatedAliasResponse>> GetUpdatedAliasMeta(ClientPaginationRequest request) private async Task<IEnumerable<UpdatedAliasResponse>> GetUpdatedAliasMeta(ClientPaginationRequest request,
CancellationToken token = default)
{ {
var aliases = await _updatedAliasHelper.QueryResource(request); var aliases = await _updatedAliasHelper.QueryResource(request);
return aliases.Results; return aliases.Results;
} }
private async Task<IEnumerable<ConnectionHistoryResponse>> GetConnectionHistoryMeta(ClientPaginationRequest request) private async Task<IEnumerable<ConnectionHistoryResponse>> GetConnectionHistoryMeta(
ClientPaginationRequest request, CancellationToken token = default)
{ {
var connections = await _connectionHistoryHelper.QueryResource(request); var connections = await _connectionHistoryHelper.QueryResource(request);
return connections.Results; return connections.Results;
} }
private async Task<IEnumerable<PermissionLevelChangedResponse>> GetPermissionLevelMeta(
ClientPaginationRequest request, CancellationToken token = default)
{
var permissionChanges = await _permissionLevelHelper.QueryResource(request);
return permissionChanges.Results;
}
} }
} }

View File

@ -0,0 +1,51 @@
using System.Linq;
using System.Threading.Tasks;
using Data.Abstractions;
using Data.Models;
using Microsoft.EntityFrameworkCore;
using SharedLibraryCore.Dtos.Meta.Responses;
using SharedLibraryCore.Helpers;
using SharedLibraryCore.Interfaces;
using SharedLibraryCore.QueryHelper;
namespace IW4MAdmin.Application.Meta;
public class
PermissionLevelChangedResourceQueryHelper : IResourceQueryHelper<ClientPaginationRequest,
PermissionLevelChangedResponse>
{
private readonly IDatabaseContextFactory _contextFactory;
public PermissionLevelChangedResourceQueryHelper(IDatabaseContextFactory contextFactory)
{
_contextFactory = contextFactory;
}
public async Task<ResourceQueryHelperResult<PermissionLevelChangedResponse>> QueryResource(
ClientPaginationRequest query)
{
await using var context = _contextFactory.CreateContext();
var auditEntries = context.EFChangeHistory.Where(change => change.TargetEntityId == query.ClientId)
.Where(change => change.TypeOfChange == EFChangeHistory.ChangeType.Permission);
var audits = from change in auditEntries
join client in context.Clients
on change.OriginEntityId equals client.ClientId
select new PermissionLevelChangedResponse
{
ChangedById = change.OriginEntityId,
ChangedByName = client.CurrentAlias.Name,
PreviousPermissionLevelValue = change.PreviousValue,
CurrentPermissionLevelValue = change.CurrentValue,
When = change.TimeChanged,
ClientId = change.TargetEntityId,
IsSensitive = true
};
return new ResourceQueryHelperResult<PermissionLevelChangedResponse>
{
Results = await audits.Skip(query.Offset).Take(query.Count).ToListAsync()
};
}
}

View File

@ -61,7 +61,7 @@ namespace IW4MAdmin.Application.Meta
iqIpLinkedPenalties = ctx.Penalties.AsNoTracking() iqIpLinkedPenalties = ctx.Penalties.AsNoTracking()
.Where(penalty => .Where(penalty =>
linkedPenaltyType.Contains(penalty.Type) && aliasedIds.Contains(penalty.LinkId)); linkedPenaltyType.Contains(penalty.Type) && aliasedIds.Contains(penalty.LinkId ?? -1));
} }
var iqAllPenalties = iqPenalties; var iqAllPenalties = iqPenalties;

View File

@ -88,7 +88,7 @@ namespace IW4MAdmin.Application.Migration
public static void RemoveObsoletePlugins20210322() public static void RemoveObsoletePlugins20210322()
{ {
var files = new[] {"StatsWeb.dll", "StatsWeb.Views.dll"}; var files = new[] {"StatsWeb.dll", "StatsWeb.Views.dll", "IW4ScriptCommands.dll"};
foreach (var file in files) foreach (var file in files)
{ {

View File

@ -27,6 +27,7 @@ namespace IW4MAdmin.Application.Misc
_serializerOptions = new JsonSerializerOptions _serializerOptions = new JsonSerializerOptions
{ {
WriteIndented = true, WriteIndented = true,
Encoder = System.Text.Encodings.Web.JavaScriptEncoder.UnsafeRelaxedJsonEscaping
}; };
_serializerOptions.Converters.Add(new JsonStringEnumConverter()); _serializerOptions.Converters.Add(new JsonStringEnumConverter());
_onSaving = new SemaphoreSlim(1, 1); _onSaving = new SemaphoreSlim(1, 1);
@ -59,7 +60,7 @@ namespace IW4MAdmin.Application.Misc
catch (Exception e) catch (Exception e)
{ {
throw new ConfigurationException("MANAGER_CONFIGURATION_ERROR") throw new ConfigurationException("Could not load configuration")
{ {
Errors = new[] { e.Message }, Errors = new[] { e.Message },
ConfigurationFileName = FileName ConfigurationFileName = FileName

View File

@ -18,6 +18,7 @@ namespace IW4MAdmin.Application.Misc
/// implementation of IMetaService /// implementation of IMetaService
/// used to add and retrieve runtime and persistent meta /// used to add and retrieve runtime and persistent meta
/// </summary> /// </summary>
[Obsolete("Use MetaServiceV2")]
public class MetaService : IMetaService public class MetaService : IMetaService
{ {
private readonly IDictionary<MetaType, List<dynamic>> _metaActions; private readonly IDictionary<MetaType, List<dynamic>> _metaActions;

View File

@ -0,0 +1,500 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text.Json;
using System.Threading;
using System.Threading.Tasks;
using Data.Abstractions;
using Data.Models;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Logging;
using SharedLibraryCore.Dtos;
using SharedLibraryCore.Interfaces;
using SharedLibraryCore.QueryHelper;
using ILogger = Microsoft.Extensions.Logging.ILogger;
namespace IW4MAdmin.Application.Misc;
public class MetaServiceV2 : IMetaServiceV2
{
private readonly IDictionary<MetaType, List<dynamic>> _metaActions;
private readonly IDatabaseContextFactory _contextFactory;
private readonly ILogger _logger;
public MetaServiceV2(ILogger<MetaServiceV2> logger, IDatabaseContextFactory contextFactory)
{
_logger = logger;
_metaActions = new Dictionary<MetaType, List<dynamic>>();
_contextFactory = contextFactory;
}
public async Task SetPersistentMeta(string metaKey, string metaValue, int clientId,
CancellationToken token = default)
{
if (!ValidArgs(metaKey, clientId))
{
return;
}
await using var context = _contextFactory.CreateContext();
var existingMeta = await context.EFMeta
.Where(meta => meta.Key == metaKey)
.Where(meta => meta.ClientId == clientId)
.FirstOrDefaultAsync(token);
if (existingMeta != null)
{
_logger.LogDebug("Updating existing meta with key {Key} and id {Id}", existingMeta.Key,
existingMeta.MetaId);
existingMeta.Value = metaValue;
existingMeta.Updated = DateTime.UtcNow;
}
else
{
_logger.LogDebug("Adding new meta with key {Key}", metaKey);
context.EFMeta.Add(new EFMeta
{
ClientId = clientId,
Created = DateTime.UtcNow,
Key = metaKey,
Value = metaValue,
});
}
await context.SaveChangesAsync(token);
}
public async Task SetPersistentMetaValue<T>(string metaKey, T metaValue, int clientId,
CancellationToken token = default) where T : class
{
if (!ValidArgs(metaKey, clientId))
{
return;
}
string serializedValue;
try
{
serializedValue = JsonSerializer.Serialize(metaValue);
}
catch (Exception ex)
{
_logger.LogError(ex, "Could not serialize meta with key {Key}", metaKey);
return;
}
await SetPersistentMeta(metaKey, serializedValue, clientId, token);
}
public async Task SetPersistentMetaForLookupKey(string metaKey, string lookupKey, int lookupId, int clientId,
CancellationToken token = default)
{
if (!ValidArgs(metaKey, clientId))
{
return;
}
await using var context = _contextFactory.CreateContext();
var lookupMeta = await context.EFMeta.FirstOrDefaultAsync(meta => meta.Key == lookupKey, token);
if (lookupMeta is null)
{
_logger.LogWarning("No lookup meta exists for metaKey {MetaKey} and lookupKey {LookupKey}", metaKey,
lookupKey);
return;
}
var lookupValues = JsonSerializer.Deserialize<List<LookupValue<string>>>(lookupMeta.Value);
if (lookupValues is null)
{
return;
}
var foundLookup = lookupValues.FirstOrDefault(value => value.Id == lookupId);
if (foundLookup is null)
{
_logger.LogWarning("No lookup meta found for provided lookup id {MetaKey}, {LookupKey}, {LookupId}",
metaKey, lookupKey, lookupId);
return;
}
_logger.LogDebug("Setting meta for lookup {MetaKey}, {LookupKey}, {LookupId}",
metaKey, lookupKey, lookupId);
await SetPersistentMeta(metaKey, lookupId.ToString(), clientId, token);
}
public async Task IncrementPersistentMeta(string metaKey, int incrementAmount, int clientId,
CancellationToken token = default)
{
if (!ValidArgs(metaKey, clientId))
{
return;
}
var existingMeta = await GetPersistentMeta(metaKey, clientId, token);
if (!long.TryParse(existingMeta?.Value ?? "0", out var existingValue))
{
existingValue = 0;
}
var newValue = existingValue + incrementAmount;
await SetPersistentMeta(metaKey, newValue.ToString(), clientId, token);
}
public async Task DecrementPersistentMeta(string metaKey, int decrementAmount, int clientId,
CancellationToken token = default)
{
await IncrementPersistentMeta(metaKey, -decrementAmount, clientId, token);
}
public async Task<EFMeta> GetPersistentMeta(string metaKey, int clientId, CancellationToken token = default)
{
if (!ValidArgs(metaKey, clientId))
{
return null;
}
await using var ctx = _contextFactory.CreateContext(enableTracking: false);
return await ctx.EFMeta
.Where(meta => meta.Key == metaKey)
.Where(meta => meta.ClientId == clientId)
.Select(meta => new EFMeta
{
MetaId = meta.MetaId,
Key = meta.Key,
ClientId = meta.ClientId,
Value = meta.Value,
})
.FirstOrDefaultAsync(token);
}
public async Task<T> GetPersistentMetaValue<T>(string metaKey, int clientId, CancellationToken token = default)
where T : class
{
var meta = await GetPersistentMeta(metaKey, clientId, token);
if (meta is null)
{
return default;
}
try
{
return JsonSerializer.Deserialize<T>(meta.Value);
}
catch (Exception ex)
{
_logger.LogError(ex, "Could not deserialize meta with key {Key} and value {Value}", metaKey, meta.Value);
return default;
}
}
public async Task<EFMeta> GetPersistentMetaByLookup(string metaKey, string lookupKey, int clientId,
CancellationToken token = default)
{
await using var context = _contextFactory.CreateContext();
var metaValue = await GetPersistentMeta(metaKey, clientId, token);
if (metaValue is null)
{
_logger.LogWarning("No meta exists for key {Key}, clientId {ClientId}", metaKey, clientId);
return default;
}
var lookupMeta = await context.EFMeta.FirstOrDefaultAsync(meta => meta.Key == lookupKey, token);
if (lookupMeta is null)
{
_logger.LogWarning("No lookup meta exists for metaKey {MetaKey} and lookupKey {LookupKey}", metaKey,
lookupKey);
return default;
}
var lookupId = int.Parse(metaValue.Value);
var lookupValues = JsonSerializer.Deserialize<List<LookupValue<string>>>(lookupMeta.Value);
if (lookupValues is null)
{
return default;
}
var foundLookup = lookupValues.FirstOrDefault(value => value.Id == lookupId);
if (foundLookup is not null)
{
return new EFMeta
{
Created = metaValue.Created,
Updated = metaValue.Updated,
Extra = metaValue.Extra,
MetaId = metaValue.MetaId,
Value = foundLookup.Value
};
}
_logger.LogWarning("No lookup meta found for provided lookup id {MetaKey}, {LookupKey}, {LookupId}",
metaKey, lookupKey, lookupId);
return default;
}
public async Task RemovePersistentMeta(string metaKey, int clientId, CancellationToken token = default)
{
if (!ValidArgs(metaKey, clientId))
{
return;
}
await using var context = _contextFactory.CreateContext();
var existingMeta = await context.EFMeta
.FirstOrDefaultAsync(meta => meta.Key == metaKey && meta.ClientId == clientId, token);
if (existingMeta == null)
{
_logger.LogDebug("No meta with key {Key} found for client id {Id}", metaKey, clientId);
return;
}
_logger.LogDebug("Removing meta for key {Key} with id {Id}", metaKey, existingMeta.MetaId);
context.EFMeta.Remove(existingMeta);
await context.SaveChangesAsync(token);
}
public async Task SetPersistentMeta(string metaKey, string metaValue, CancellationToken token = default)
{
if (string.IsNullOrWhiteSpace(metaKey))
{
_logger.LogWarning("Cannot save meta with no key");
return;
}
await using var ctx = _contextFactory.CreateContext();
var existingMeta = await ctx.EFMeta
.Where(meta => meta.Key == metaKey)
.Where(meta => meta.ClientId == null)
.FirstOrDefaultAsync(token);
if (existingMeta is not null)
{
_logger.LogDebug("Updating existing meta with key {Key} and id {Id}", existingMeta.Key,
existingMeta.MetaId);
existingMeta.Value = metaValue;
existingMeta.Updated = DateTime.UtcNow;
await ctx.SaveChangesAsync(token);
}
else
{
_logger.LogDebug("Adding new meta with key {Key}", metaKey);
ctx.EFMeta.Add(new EFMeta
{
Created = DateTime.UtcNow,
Key = metaKey,
Value = metaValue
});
await ctx.SaveChangesAsync(token);
}
}
public async Task SetPersistentMetaValue<T>(string metaKey, T metaValue, CancellationToken token = default)
where T : class
{
if (string.IsNullOrWhiteSpace(metaKey))
{
_logger.LogWarning("Meta key is null, not setting");
return;
}
if (metaValue is null)
{
_logger.LogWarning("Meta value is null, not setting");
return;
}
string serializedMeta;
try
{
serializedMeta = JsonSerializer.Serialize(metaValue);
}
catch (Exception ex)
{
_logger.LogError(ex, "Could not serialize meta with {Key} and value {Value}", metaKey, metaValue);
return;
}
await SetPersistentMeta(metaKey, serializedMeta, token);
}
public async Task<EFMeta> GetPersistentMeta(string metaKey, CancellationToken token = default)
{
if (string.IsNullOrWhiteSpace(metaKey))
{
return null;
}
await using var context = _contextFactory.CreateContext(false);
return await context.EFMeta.FirstOrDefaultAsync(meta => meta.Key == metaKey, token);
}
public async Task<T> GetPersistentMetaValue<T>(string metaKey, CancellationToken token = default) where T : class
{
if (string.IsNullOrWhiteSpace(metaKey))
{
return default;
}
var meta = await GetPersistentMeta(metaKey, token);
if (meta is null)
{
return default;
}
try
{
return JsonSerializer.Deserialize<T>(meta.Value);
}
catch (Exception ex)
{
_logger.LogError(ex, "Could not serialize meta with key {Key} and value {Value}", metaKey, meta.Value);
return default;
}
}
public async Task RemovePersistentMeta(string metaKey, CancellationToken token = default)
{
if (string.IsNullOrWhiteSpace(metaKey))
{
return;
}
await using var context = _contextFactory.CreateContext(enableTracking: false);
var existingMeta = await context.EFMeta
.Where(meta => meta.Key == metaKey)
.Where(meta => meta.ClientId == null)
.FirstOrDefaultAsync(token);
if (existingMeta != null)
{
_logger.LogDebug("Removing meta for key {Key} with id {Id}", metaKey, existingMeta.MetaId);
context.Remove(existingMeta);
await context.SaveChangesAsync(token);
}
}
public void AddRuntimeMeta<T, TReturnType>(MetaType metaKey,
Func<T, CancellationToken, Task<IEnumerable<TReturnType>>> metaAction)
where T : PaginationRequest where TReturnType : IClientMeta
{
if (!_metaActions.ContainsKey(metaKey))
{
_metaActions.Add(metaKey, new List<dynamic> { metaAction });
}
else
{
_metaActions[metaKey].Add(metaAction);
}
}
public async Task<IEnumerable<IClientMeta>> GetRuntimeMeta(ClientPaginationRequest request, CancellationToken token = default)
{
var metas = await Task.WhenAll(_metaActions.Where(kvp => kvp.Key != MetaType.Information)
.Select(async kvp => await kvp.Value[0](request, token)));
return metas.SelectMany(m => (IEnumerable<IClientMeta>)m)
.OrderByDescending(m => m.When)
.Take(request.Count)
.ToList();
}
public async Task<IEnumerable<T>> GetRuntimeMeta<T>(ClientPaginationRequest request, MetaType metaType, CancellationToken token = default)
where T : IClientMeta
{
if (metaType == MetaType.Information)
{
var allMeta = new List<T>();
var completedMeta = await Task.WhenAll(_metaActions[metaType].Select(async individualMetaRegistration =>
(IEnumerable<T>)await individualMetaRegistration(request, token)));
allMeta.AddRange(completedMeta.SelectMany(meta => meta));
return ProcessInformationMeta(allMeta);
}
var meta = await _metaActions[metaType][0](request) as IEnumerable<T>;
return meta;
}
private static IEnumerable<T> ProcessInformationMeta<T>(IEnumerable<T> meta) where T : IClientMeta
{
var metaList = meta.ToList();
var metaWithColumn = metaList
.Where(m => m.Column != null)
.ToList();
var columnGrouping = metaWithColumn
.GroupBy(m => m.Column)
.ToList();
var metaToSort = metaList.Except(metaWithColumn).ToList();
var table = columnGrouping.Select(metaItem => new List<T>(metaItem)).ToList();
while (metaToSort.Count > 0)
{
var sortingMeta = metaToSort.First();
int IndexOfSmallestColumn()
{
var index = 0;
var smallestColumnSize = int.MaxValue;
for (var i = 0; i < table.Count; i++)
{
if (table[i].Count >= smallestColumnSize)
{
continue;
}
smallestColumnSize = table[i].Count;
index = i;
}
return index;
}
var columnIndex = IndexOfSmallestColumn();
sortingMeta.Column = columnIndex;
sortingMeta.Order = columnGrouping
.First(group => group.Key == columnIndex)
.Count();
table[columnIndex].Add(sortingMeta);
metaToSort.Remove(sortingMeta);
}
return metaList;
}
private static bool ValidArgs(string key, int clientId) => !string.IsNullOrWhiteSpace(key) && clientId > 0;
}

View File

@ -223,47 +223,49 @@ namespace IW4MAdmin.Application.Misc
public async Task OnEventAsync(GameEvent gameEvent, Server server) public async Task OnEventAsync(GameEvent gameEvent, Server server)
{ {
if (_successfullyLoaded) if (!_successfullyLoaded)
{ {
try return;
{ }
await _onProcessing.WaitAsync();
_scriptEngine.SetValue("_gameEvent", gameEvent);
_scriptEngine.SetValue("_server", server);
_scriptEngine.SetValue("_IW4MAdminClient", Utilities.IW4MAdminClient(server));
_scriptEngine.Evaluate("plugin.onEventAsync(_gameEvent, _server)");
}
catch (JavaScriptException ex)
{
using (LogContext.PushProperty("Server", server.ToString()))
{
_logger.LogError(ex,
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin} with event type {EventType} {@LocationInfo}",
nameof(OnEventAsync), Path.GetFileName(_fileName), gameEvent.Type, ex.Location);
}
throw new PluginException("An error occured while executing action for script plugin"); try
{
await _onProcessing.WaitAsync();
_scriptEngine.SetValue("_gameEvent", gameEvent);
_scriptEngine.SetValue("_server", server);
_scriptEngine.SetValue("_IW4MAdminClient", Utilities.IW4MAdminClient(server));
_scriptEngine.Evaluate("plugin.onEventAsync(_gameEvent, _server)");
}
catch (JavaScriptException ex)
{
using (LogContext.PushProperty("Server", server.ToString()))
{
_logger.LogError(ex,
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin} with event type {EventType} {@LocationInfo}",
nameof(OnEventAsync), Path.GetFileName(_fileName), gameEvent.Type, ex.Location);
} }
catch (Exception ex) throw new PluginException("An error occured while executing action for script plugin");
{ }
using (LogContext.PushProperty("Server", server.ToString()))
{
_logger.LogError(ex,
"Encountered error while running {MethodName} for script plugin {Plugin} with event type {EventType}",
nameof(OnEventAsync), _fileName, gameEvent.Type);
}
throw new PluginException("An error occured while executing action for script plugin"); catch (Exception ex)
{
using (LogContext.PushProperty("Server", server.ToString()))
{
_logger.LogError(ex,
"Encountered error while running {MethodName} for script plugin {Plugin} with event type {EventType}",
nameof(OnEventAsync), _fileName, gameEvent.Type);
} }
finally throw new PluginException("An error occured while executing action for script plugin");
}
finally
{
if (_onProcessing.CurrentCount == 0)
{ {
if (_onProcessing.CurrentCount == 0) _onProcessing.Release(1);
{
_onProcessing.Release(1);
}
} }
} }
} }
@ -274,6 +276,8 @@ namespace IW4MAdmin.Application.Misc
{ {
_logger.LogDebug("OnLoad executing for {Name}", Name); _logger.LogDebug("OnLoad executing for {Name}", Name);
_scriptEngine.SetValue("_manager", manager); _scriptEngine.SetValue("_manager", manager);
_scriptEngine.SetValue("getDvar", GetDvarAsync);
_scriptEngine.SetValue("setDvar", SetDvarAsync);
_scriptEngine.Evaluate("plugin.onLoadAsync(_manager)"); _scriptEngine.Evaluate("plugin.onLoadAsync(_manager)");
return Task.CompletedTask; return Task.CompletedTask;
@ -449,6 +453,85 @@ namespace IW4MAdmin.Application.Misc
return commandList; return commandList;
} }
private void GetDvarAsync(Server server, string dvarName, Delegate onCompleted)
{
Task.Run<Task>(async () =>
{
var tokenSource = new CancellationTokenSource();
tokenSource.CancelAfter(TimeSpan.FromSeconds(5));
string result = null;
var success = true;
try
{
result = (await server.GetDvarAsync<string>(dvarName, token: tokenSource.Token)).Value;
}
catch
{
success = false;
}
await _onProcessing.WaitAsync();
try
{
onCompleted.DynamicInvoke(JsValue.Undefined,
new[]
{
JsValue.FromObject(_scriptEngine, server),
JsValue.FromObject(_scriptEngine, dvarName),
JsValue.FromObject(_scriptEngine, result),
JsValue.FromObject(_scriptEngine, success),
});
}
finally
{
if (_onProcessing.CurrentCount == 0)
{
_onProcessing.Release();
}
}
});
}
private void SetDvarAsync(Server server, string dvarName, string dvarValue, Delegate onCompleted)
{
Task.Run<Task>(async () =>
{
var tokenSource = new CancellationTokenSource();
tokenSource.CancelAfter(TimeSpan.FromSeconds(5));
var success = true;
try
{
await server.SetDvarAsync(dvarName, dvarValue, tokenSource.Token);
}
catch
{
success = false;
}
await _onProcessing.WaitAsync();
try
{
onCompleted.DynamicInvoke(JsValue.Undefined,
new[]
{
JsValue.FromObject(_scriptEngine, server),
JsValue.FromObject(_scriptEngine, dvarName),
JsValue.FromObject(_scriptEngine, dvarValue),
JsValue.FromObject(_scriptEngine, success)
});
}
finally
{
if (_onProcessing.CurrentCount == 0)
{
_onProcessing.Release();
}
}
});
}
} }
public class PermissionLevelToStringConverter : IObjectConverter public class PermissionLevelToStringConverter : IObjectConverter

View File

@ -1,6 +1,7 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Globalization; using System.Globalization;
using System.Linq; using System.Linq;
using System.Text.Json;
using System.Threading.Tasks; using System.Threading.Tasks;
using IW4MAdmin.Application.Configuration; using IW4MAdmin.Application.Configuration;
using Jint; using Jint;
@ -84,9 +85,9 @@ namespace IW4MAdmin.Application.Misc
var item = _config[_pluginName][key]; var item = _config[_pluginName][key];
if (item is JArray array) if (item is JsonElement { ValueKind: JsonValueKind.Array } jElem)
{ {
item = array.ToObject<List<dynamic>>(); item = jElem.Deserialize<List<dynamic>>();
} }
return JsValue.FromObject(_scriptEngine, item); return JsValue.FromObject(_scriptEngine, item);

View File

@ -7,6 +7,7 @@ using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq; using System.Linq;
using System.Text.RegularExpressions; using System.Text.RegularExpressions;
using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using Data.Models; using Data.Models;
using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging;
@ -77,19 +78,19 @@ namespace IW4MAdmin.Application.RConParsers
public string RConEngine { get; set; } = "COD"; public string RConEngine { get; set; } = "COD";
public bool IsOneLog { get; set; } public bool IsOneLog { get; set; }
public async Task<string[]> ExecuteCommandAsync(IRConConnection connection, string command) public async Task<string[]> ExecuteCommandAsync(IRConConnection connection, string command, CancellationToken token = default)
{ {
var response = await connection.SendQueryAsync(StaticHelpers.QueryType.COMMAND, command); var response = await connection.SendQueryAsync(StaticHelpers.QueryType.COMMAND, command, token);
return response.Where(item => item != Configuration.CommandPrefixes.RConResponse).ToArray(); return response.Where(item => item != Configuration.CommandPrefixes.RConResponse).ToArray();
} }
public async Task<Dvar<T>> GetDvarAsync<T>(IRConConnection connection, string dvarName, T fallbackValue = default) public async Task<Dvar<T>> GetDvarAsync<T>(IRConConnection connection, string dvarName, T fallbackValue = default, CancellationToken token = default)
{ {
string[] lineSplit; string[] lineSplit;
try try
{ {
lineSplit = await connection.SendQueryAsync(StaticHelpers.QueryType.GET_DVAR, dvarName); lineSplit = await connection.SendQueryAsync(StaticHelpers.QueryType.GET_DVAR, dvarName, token);
} }
catch catch
{ {
@ -98,10 +99,10 @@ namespace IW4MAdmin.Application.RConParsers
throw; throw;
} }
lineSplit = new string[0]; lineSplit = Array.Empty<string>();
} }
string response = string.Join('\n', lineSplit).TrimEnd('\0'); var response = string.Join('\n', lineSplit).TrimEnd('\0');
var match = Regex.Match(response, Configuration.Dvar.Pattern); var match = Regex.Match(response, Configuration.Dvar.Pattern);
if (response.Contains("Unknown command") || if (response.Contains("Unknown command") ||
@ -109,7 +110,7 @@ namespace IW4MAdmin.Application.RConParsers
{ {
if (fallbackValue != null) if (fallbackValue != null)
{ {
return new Dvar<T>() return new Dvar<T>
{ {
Name = dvarName, Name = dvarName,
Value = fallbackValue Value = fallbackValue
@ -119,17 +120,17 @@ namespace IW4MAdmin.Application.RConParsers
throw new DvarException(Utilities.CurrentLocalization.LocalizationIndex["SERVER_ERROR_DVAR"].FormatExt(dvarName)); throw new DvarException(Utilities.CurrentLocalization.LocalizationIndex["SERVER_ERROR_DVAR"].FormatExt(dvarName));
} }
string value = match.Groups[Configuration.Dvar.GroupMapping[ParserRegex.GroupType.RConDvarValue]].Value; var value = match.Groups[Configuration.Dvar.GroupMapping[ParserRegex.GroupType.RConDvarValue]].Value;
string defaultValue = match.Groups[Configuration.Dvar.GroupMapping[ParserRegex.GroupType.RConDvarDefaultValue]].Value; var defaultValue = match.Groups[Configuration.Dvar.GroupMapping[ParserRegex.GroupType.RConDvarDefaultValue]].Value;
string latchedValue = match.Groups[Configuration.Dvar.GroupMapping[ParserRegex.GroupType.RConDvarLatchedValue]].Value; var latchedValue = match.Groups[Configuration.Dvar.GroupMapping[ParserRegex.GroupType.RConDvarLatchedValue]].Value;
string removeTrailingColorCode(string input) => Regex.Replace(input, @"\^7$", ""); string RemoveTrailingColorCode(string input) => Regex.Replace(input, @"\^7$", "");
value = removeTrailingColorCode(value); value = RemoveTrailingColorCode(value);
defaultValue = removeTrailingColorCode(defaultValue); defaultValue = RemoveTrailingColorCode(defaultValue);
latchedValue = removeTrailingColorCode(latchedValue); latchedValue = RemoveTrailingColorCode(latchedValue);
return new Dvar<T>() return new Dvar<T>
{ {
Name = dvarName, Name = dvarName,
Value = string.IsNullOrEmpty(value) ? default : (T)Convert.ChangeType(value, typeof(T)), Value = string.IsNullOrEmpty(value) ? default : (T)Convert.ChangeType(value, typeof(T)),
@ -139,10 +140,12 @@ namespace IW4MAdmin.Application.RConParsers
}; };
} }
public virtual async Task<IStatusResponse> GetStatusAsync(IRConConnection connection) public virtual async Task<IStatusResponse> GetStatusAsync(IRConConnection connection, CancellationToken token = default)
{ {
var response = await connection.SendQueryAsync(StaticHelpers.QueryType.COMMAND_STATUS); var response = await connection.SendQueryAsync(StaticHelpers.QueryType.COMMAND_STATUS, "status", token);
_logger.LogDebug("Status Response {response}", string.Join(Environment.NewLine, response));
_logger.LogDebug("Status Response {Response}", string.Join(Environment.NewLine, response));
return new StatusResponse return new StatusResponse
{ {
Clients = ClientsFromStatus(response).ToArray(), Clients = ClientsFromStatus(response).ToArray(),
@ -183,13 +186,13 @@ namespace IW4MAdmin.Application.RConParsers
return (T)Convert.ChangeType(value, typeof(T)); return (T)Convert.ChangeType(value, typeof(T));
} }
public async Task<bool> SetDvarAsync(IRConConnection connection, string dvarName, object dvarValue) public async Task<bool> SetDvarAsync(IRConConnection connection, string dvarName, object dvarValue, CancellationToken token = default)
{ {
string dvarString = (dvarValue is string str) var dvarString = (dvarValue is string str)
? $"{dvarName} \"{str}\"" ? $"{dvarName} \"{str}\""
: $"{dvarName} {dvarValue}"; : $"{dvarName} {dvarValue}";
return (await connection.SendQueryAsync(StaticHelpers.QueryType.SET_DVAR, dvarString)).Length > 0; return (await connection.SendQueryAsync(StaticHelpers.QueryType.SET_DVAR, dvarString, token)).Length > 0;
} }
private List<EFClient> ClientsFromStatus(string[] Status) private List<EFClient> ClientsFromStatus(string[] Status)

View File

@ -18,6 +18,7 @@ namespace Data.Context
public DbSet<EFAlias> Aliases { get; set; } public DbSet<EFAlias> Aliases { get; set; }
public DbSet<EFAliasLink> AliasLinks { get; set; } public DbSet<EFAliasLink> AliasLinks { get; set; }
public DbSet<EFPenalty> Penalties { get; set; } public DbSet<EFPenalty> Penalties { get; set; }
public DbSet<EFPenaltyIdentifier> PenaltyIdentifiers { get; set; }
public DbSet<EFMeta> EFMeta { get; set; } public DbSet<EFMeta> EFMeta { get; set; }
public DbSet<EFChangeHistory> EFChangeHistory { get; set; } public DbSet<EFChangeHistory> EFChangeHistory { get; set; }
@ -130,6 +131,12 @@ namespace Data.Context
.OnDelete(DeleteBehavior.SetNull); .OnDelete(DeleteBehavior.SetNull);
}); });
modelBuilder.Entity<EFPenaltyIdentifier>(ent =>
{
ent.HasIndex(identifiers => identifiers.NetworkId);
ent.HasIndex(identifiers => identifiers.IPv4Address);
});
modelBuilder.Entity<EFClientConnectionHistory>(ent => ent.HasIndex(history => history.CreatedDateTime)); modelBuilder.Entity<EFClientConnectionHistory>(ent => ent.HasIndex(history => history.CreatedDateTime));
// force full name for database conversion // force full name for database conversion
@ -137,6 +144,7 @@ namespace Data.Context
modelBuilder.Entity<EFAlias>().ToTable("EFAlias"); modelBuilder.Entity<EFAlias>().ToTable("EFAlias");
modelBuilder.Entity<EFAliasLink>().ToTable("EFAliasLinks"); modelBuilder.Entity<EFAliasLink>().ToTable("EFAliasLinks");
modelBuilder.Entity<EFPenalty>().ToTable("EFPenalties"); modelBuilder.Entity<EFPenalty>().ToTable("EFPenalties");
modelBuilder.Entity<EFPenaltyIdentifier>().ToTable("EFPenaltyIdentifiers");
modelBuilder.Entity<EFServerSnapshot>().ToTable(nameof(EFServerSnapshot)); modelBuilder.Entity<EFServerSnapshot>().ToTable(nameof(EFServerSnapshot));
modelBuilder.Entity<EFClientConnectionHistory>().ToTable(nameof(EFClientConnectionHistory)); modelBuilder.Entity<EFClientConnectionHistory>().ToTable(nameof(EFClientConnectionHistory));

View File

@ -24,7 +24,8 @@ namespace Data.MigrationContext
{ {
if (MigrationExtensions.IsMigration) if (MigrationExtensions.IsMigration)
{ {
optionsBuilder.UseMySql(ServerVersion.AutoDetect("Server=127.0.0.1;Database=IW4MAdmin_Migration;Uid=root;Pwd=password;")) var connectionString = "Server=127.0.0.1;Database=IW4MAdmin_Migration;Uid=root;Pwd=password;";
optionsBuilder.UseMySql(connectionString, ServerVersion.AutoDetect(connectionString))
.EnableDetailedErrors() .EnableDetailedErrors()
.EnableSensitiveDataLogging(); .EnableSensitiveDataLogging();
} }

View File

@ -23,6 +23,7 @@ namespace Data.MigrationContext
{ {
if (MigrationExtensions.IsMigration) if (MigrationExtensions.IsMigration)
{ {
AppContext.SetSwitch("Npgsql.EnableLegacyTimestampBehavior", true);
optionsBuilder.UseNpgsql( optionsBuilder.UseNpgsql(
"Host=127.0.0.1;Database=IW4MAdmin_Migration;Username=postgres;Password=password;", "Host=127.0.0.1;Database=IW4MAdmin_Migration;Username=postgres;Password=password;",
options => options.SetPostgresVersion(new Version("12.9"))) options => options.SetPostgresVersion(new Version("12.9")))

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,250 @@
using Microsoft.EntityFrameworkCore.Metadata;
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace Data.Migrations.MySql
{
public partial class AddEFPenaltyIdentifier : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.CreateTable(
name: "EFPenaltyIdentifiers",
columns: table => new
{
PenaltyIdentifierId = table.Column<int>(type: "int", nullable: false)
.Annotation("MySql:ValueGenerationStrategy", MySqlValueGenerationStrategy.IdentityColumn),
IPv4Address = table.Column<int>(type: "int", nullable: true),
NetworkId = table.Column<long>(type: "bigint", nullable: false),
PenaltyId = table.Column<int>(type: "int", nullable: false),
Active = table.Column<bool>(type: "tinyint(1)", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_EFPenaltyIdentifiers", x => x.PenaltyIdentifierId);
table.ForeignKey(
name: "FK_EFPenaltyIdentifiers_EFPenalties_PenaltyId",
column: x => x.PenaltyId,
principalTable: "EFPenalties",
principalColumn: "PenaltyId",
onDelete: ReferentialAction.Cascade);
})
.Annotation("MySql:CharSet", "utf8mb4");
migrationBuilder.CreateIndex(
name: "IX_EFPenaltyIdentifiers_IPv4Address",
table: "EFPenaltyIdentifiers",
column: "IPv4Address");
migrationBuilder.CreateIndex(
name: "IX_EFPenaltyIdentifiers_NetworkId",
table: "EFPenaltyIdentifiers",
column: "NetworkId");
migrationBuilder.CreateIndex(
name: "IX_EFPenaltyIdentifiers_PenaltyId",
table: "EFPenaltyIdentifiers",
column: "PenaltyId");
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropTable(
name: "EFPenaltyIdentifiers");
migrationBuilder.AlterColumn<string>(
name: "Message",
table: "InboxMessages",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "Name",
table: "EFWeaponAttachments",
type: "longtext CHARACTER SET utf8mb4",
nullable: false,
oldClrType: typeof(string),
oldType: "longtext")
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "HostName",
table: "EFServers",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "EndPoint",
table: "EFServers",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "Offense",
table: "EFPenalties",
type: "longtext CHARACTER SET utf8mb4",
nullable: false,
oldClrType: typeof(string),
oldType: "longtext")
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "AutomatedOffense",
table: "EFPenalties",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "Value",
table: "EFMeta",
type: "longtext CHARACTER SET utf8mb4",
nullable: false,
oldClrType: typeof(string),
oldType: "longtext")
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "Extra",
table: "EFMeta",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "Name",
table: "EFMeansOfDeath",
type: "longtext CHARACTER SET utf8mb4",
nullable: false,
oldClrType: typeof(string),
oldType: "longtext")
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "Name",
table: "EFMaps",
type: "longtext CHARACTER SET utf8mb4",
nullable: false,
oldClrType: typeof(string),
oldType: "longtext")
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "PasswordSalt",
table: "EFClients",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "Password",
table: "EFClients",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "Message",
table: "EFClientMessages",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "WeaponReference",
table: "EFClientKills",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "PreviousValue",
table: "EFChangeHistory",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "CurrentValue",
table: "EFChangeHistory",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "WeaponReference",
table: "EFACSnapshot",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
migrationBuilder.AlterColumn<string>(
name: "HitLocationReference",
table: "EFACSnapshot",
type: "longtext CHARACTER SET utf8mb4",
nullable: true,
oldClrType: typeof(string),
oldType: "longtext",
oldNullable: true)
.Annotation("MySql:CharSet", "utf8mb4")
.OldAnnotation("MySql:CharSet", "utf8mb4");
}
}
}

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,56 @@
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace Data.Migrations.MySql
{
public partial class MakeEFPenaltyLinkIdNullable : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties");
migrationBuilder.AlterColumn<int>(
name: "LinkId",
table: "EFPenalties",
type: "int",
nullable: true,
oldClrType: typeof(int),
oldType: "int");
migrationBuilder.AddForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties",
column: "LinkId",
principalTable: "EFAliasLinks",
principalColumn: "AliasLinkId");
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties");
migrationBuilder.AlterColumn<int>(
name: "LinkId",
table: "EFPenalties",
type: "int",
nullable: false,
defaultValue: 0,
oldClrType: typeof(int),
oldType: "int",
oldNullable: true);
migrationBuilder.AddForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties",
column: "LinkId",
principalTable: "EFAliasLinks",
principalColumn: "AliasLinkId",
onDelete: ReferentialAction.Cascade);
}
}
}

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,48 @@
using System;
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace Data.Migrations.MySql
{
public partial class AddAuditFieldsToEFPenaltyIdentifier : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "Active",
table: "EFPenaltyIdentifiers");
migrationBuilder.AddColumn<DateTime>(
name: "CreatedDateTime",
table: "EFPenaltyIdentifiers",
type: "datetime(6)",
nullable: false,
defaultValue: new DateTime(1, 1, 1, 0, 0, 0, 0, DateTimeKind.Unspecified));
migrationBuilder.AddColumn<DateTime>(
name: "UpdatedDateTime",
table: "EFPenaltyIdentifiers",
type: "datetime(6)",
nullable: true);
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "CreatedDateTime",
table: "EFPenaltyIdentifiers");
migrationBuilder.DropColumn(
name: "UpdatedDateTime",
table: "EFPenaltyIdentifiers");
migrationBuilder.AddColumn<bool>(
name: "Active",
table: "EFPenaltyIdentifiers",
type: "tinyint(1)",
nullable: false,
defaultValue: false);
}
}
}

View File

@ -5,6 +5,8 @@ using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure; using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion; using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
#nullable disable
namespace Data.Migrations.MySql namespace Data.Migrations.MySql
{ {
[DbContext(typeof(MySqlDatabaseContext))] [DbContext(typeof(MySqlDatabaseContext))]
@ -14,7 +16,7 @@ namespace Data.Migrations.MySql
{ {
#pragma warning disable 612, 618 #pragma warning disable 612, 618
modelBuilder modelBuilder
.HasAnnotation("ProductVersion", "3.1.10") .HasAnnotation("ProductVersion", "6.0.1")
.HasAnnotation("Relational:MaxIdentifierLength", 64); .HasAnnotation("Relational:MaxIdentifierLength", 64);
modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b => modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b =>
@ -38,7 +40,7 @@ namespace Data.Migrations.MySql
b.HasIndex("Vector3Id"); b.HasIndex("Vector3Id");
b.ToTable("EFACSnapshotVector3"); b.ToTable("EFACSnapshotVector3", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClient", b => modelBuilder.Entity("Data.Models.Client.EFClient", b =>
@ -75,10 +77,10 @@ namespace Data.Migrations.MySql
.HasColumnType("bigint"); .HasColumnType("bigint");
b.Property<string>("Password") b.Property<string>("Password")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<string>("PasswordSalt") b.Property<string>("PasswordSalt")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<int>("TotalConnectionTime") b.Property<int>("TotalConnectionTime")
.HasColumnType("int"); .HasColumnType("int");
@ -92,7 +94,7 @@ namespace Data.Migrations.MySql
b.HasIndex("NetworkId") b.HasIndex("NetworkId")
.IsUnique(); .IsUnique();
b.ToTable("EFClients"); b.ToTable("EFClients", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b => modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b =>
@ -124,7 +126,7 @@ namespace Data.Migrations.MySql
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFClientConnectionHistory"); b.ToTable("EFClientConnectionHistory", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClientKill", b => modelBuilder.Entity("Data.Models.Client.EFClientKill", b =>
@ -179,7 +181,7 @@ namespace Data.Migrations.MySql
.HasColumnType("int"); .HasColumnType("int");
b.Property<string>("WeaponReference") b.Property<string>("WeaponReference")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<DateTime>("When") b.Property<DateTime>("When")
.HasColumnType("datetime(6)"); .HasColumnType("datetime(6)");
@ -198,7 +200,7 @@ namespace Data.Migrations.MySql
b.HasIndex("ViewAnglesVector3Id"); b.HasIndex("ViewAnglesVector3Id");
b.ToTable("EFClientKills"); b.ToTable("EFClientKills", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClientMessage", b => modelBuilder.Entity("Data.Models.Client.EFClientMessage", b =>
@ -214,7 +216,7 @@ namespace Data.Migrations.MySql
.HasColumnType("int"); .HasColumnType("int");
b.Property<string>("Message") b.Property<string>("Message")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<bool>("SentIngame") b.Property<bool>("SentIngame")
.HasColumnType("tinyint(1)"); .HasColumnType("tinyint(1)");
@ -233,7 +235,7 @@ namespace Data.Migrations.MySql
b.HasIndex("TimeSent"); b.HasIndex("TimeSent");
b.ToTable("EFClientMessages"); b.ToTable("EFClientMessages", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b => modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b =>
@ -273,7 +275,7 @@ namespace Data.Migrations.MySql
.HasColumnType("int"); .HasColumnType("int");
b.Property<string>("HitLocationReference") b.Property<string>("HitLocationReference")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<int>("HitOriginId") b.Property<int>("HitOriginId")
.HasColumnType("int"); .HasColumnType("int");
@ -321,7 +323,7 @@ namespace Data.Migrations.MySql
.HasColumnType("int"); .HasColumnType("int");
b.Property<string>("WeaponReference") b.Property<string>("WeaponReference")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<DateTime>("When") b.Property<DateTime>("When")
.HasColumnType("datetime(6)"); .HasColumnType("datetime(6)");
@ -340,7 +342,7 @@ namespace Data.Migrations.MySql
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFACSnapshot"); b.ToTable("EFACSnapshot", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b =>
@ -414,7 +416,7 @@ namespace Data.Migrations.MySql
b.HasIndex("WeaponId"); b.HasIndex("WeaponId");
b.ToTable("EFClientHitStatistics"); b.ToTable("EFClientHitStatistics", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b =>
@ -459,7 +461,7 @@ namespace Data.Migrations.MySql
b.HasIndex("ZScore"); b.HasIndex("ZScore");
b.ToTable("EFClientRankingHistory"); b.ToTable("EFClientRankingHistory", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b =>
@ -478,7 +480,7 @@ namespace Data.Migrations.MySql
b.HasIndex("ClientId"); b.HasIndex("ClientId");
b.ToTable("EFClientRatingHistory"); b.ToTable("EFClientRatingHistory", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b =>
@ -536,7 +538,7 @@ namespace Data.Migrations.MySql
b.HasIndex("ClientId", "TimePlayed", "ZScore"); b.HasIndex("ClientId", "TimePlayed", "ZScore");
b.ToTable("EFClientStatistics"); b.ToTable("EFClientStatistics", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b => modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b =>
@ -549,12 +551,12 @@ namespace Data.Migrations.MySql
.HasColumnType("tinyint(1)"); .HasColumnType("tinyint(1)");
b.Property<int>("EFClientStatisticsClientId") b.Property<int>("EFClientStatisticsClientId")
.HasColumnName("EFClientStatisticsClientId") .HasColumnType("int")
.HasColumnType("int"); .HasColumnName("EFClientStatisticsClientId");
b.Property<long>("EFClientStatisticsServerId") b.Property<long>("EFClientStatisticsServerId")
.HasColumnName("EFClientStatisticsServerId") .HasColumnType("bigint")
.HasColumnType("bigint"); .HasColumnName("EFClientStatisticsServerId");
b.Property<int>("HitCount") b.Property<int>("HitCount")
.HasColumnType("int"); .HasColumnType("int");
@ -574,7 +576,7 @@ namespace Data.Migrations.MySql
b.HasIndex("EFClientStatisticsClientId", "EFClientStatisticsServerId"); b.HasIndex("EFClientStatisticsClientId", "EFClientStatisticsServerId");
b.ToTable("EFHitLocationCounts"); b.ToTable("EFHitLocationCounts", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b => modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b =>
@ -617,7 +619,7 @@ namespace Data.Migrations.MySql
b.HasIndex("When", "ServerId", "Performance", "ActivityAmount"); b.HasIndex("When", "ServerId", "Performance", "ActivityAmount");
b.ToTable("EFRating"); b.ToTable("EFRating", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFHitLocation", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFHitLocation", b =>
@ -634,7 +636,7 @@ namespace Data.Migrations.MySql
b.Property<string>("Name") b.Property<string>("Name")
.IsRequired() .IsRequired()
.HasColumnType("varchar(255) CHARACTER SET utf8mb4"); .HasColumnType("varchar(255)");
b.Property<DateTime?>("UpdatedDateTime") b.Property<DateTime?>("UpdatedDateTime")
.HasColumnType("datetime(6)"); .HasColumnType("datetime(6)");
@ -643,7 +645,7 @@ namespace Data.Migrations.MySql
b.HasIndex("Name"); b.HasIndex("Name");
b.ToTable("EFHitLocations"); b.ToTable("EFHitLocations", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMap", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMap", b =>
@ -660,14 +662,14 @@ namespace Data.Migrations.MySql
b.Property<string>("Name") b.Property<string>("Name")
.IsRequired() .IsRequired()
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<DateTime?>("UpdatedDateTime") b.Property<DateTime?>("UpdatedDateTime")
.HasColumnType("datetime(6)"); .HasColumnType("datetime(6)");
b.HasKey("MapId"); b.HasKey("MapId");
b.ToTable("EFMaps"); b.ToTable("EFMaps", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMeansOfDeath", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMeansOfDeath", b =>
@ -684,14 +686,14 @@ namespace Data.Migrations.MySql
b.Property<string>("Name") b.Property<string>("Name")
.IsRequired() .IsRequired()
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<DateTime?>("UpdatedDateTime") b.Property<DateTime?>("UpdatedDateTime")
.HasColumnType("datetime(6)"); .HasColumnType("datetime(6)");
b.HasKey("MeansOfDeathId"); b.HasKey("MeansOfDeathId");
b.ToTable("EFMeansOfDeath"); b.ToTable("EFMeansOfDeath", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeapon", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeapon", b =>
@ -708,7 +710,7 @@ namespace Data.Migrations.MySql
b.Property<string>("Name") b.Property<string>("Name")
.IsRequired() .IsRequired()
.HasColumnType("varchar(255) CHARACTER SET utf8mb4"); .HasColumnType("varchar(255)");
b.Property<DateTime?>("UpdatedDateTime") b.Property<DateTime?>("UpdatedDateTime")
.HasColumnType("datetime(6)"); .HasColumnType("datetime(6)");
@ -717,7 +719,7 @@ namespace Data.Migrations.MySql
b.HasIndex("Name"); b.HasIndex("Name");
b.ToTable("EFWeapons"); b.ToTable("EFWeapons", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachment", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachment", b =>
@ -734,14 +736,14 @@ namespace Data.Migrations.MySql
b.Property<string>("Name") b.Property<string>("Name")
.IsRequired() .IsRequired()
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<DateTime?>("UpdatedDateTime") b.Property<DateTime?>("UpdatedDateTime")
.HasColumnType("datetime(6)"); .HasColumnType("datetime(6)");
b.HasKey("WeaponAttachmentId"); b.HasKey("WeaponAttachmentId");
b.ToTable("EFWeaponAttachments"); b.ToTable("EFWeaponAttachments", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b =>
@ -776,7 +778,7 @@ namespace Data.Migrations.MySql
b.HasIndex("Attachment3Id"); b.HasIndex("Attachment3Id");
b.ToTable("EFWeaponAttachmentCombos"); b.ToTable("EFWeaponAttachmentCombos", (string)null);
}); });
modelBuilder.Entity("Data.Models.EFAlias", b => modelBuilder.Entity("Data.Models.EFAlias", b =>
@ -799,12 +801,12 @@ namespace Data.Migrations.MySql
b.Property<string>("Name") b.Property<string>("Name")
.IsRequired() .IsRequired()
.HasColumnType("varchar(24) CHARACTER SET utf8mb4") .HasMaxLength(24)
.HasMaxLength(24); .HasColumnType("varchar(24)");
b.Property<string>("SearchableName") b.Property<string>("SearchableName")
.HasColumnType("varchar(24) CHARACTER SET utf8mb4") .HasMaxLength(24)
.HasMaxLength(24); .HasColumnType("varchar(24)");
b.HasKey("AliasId"); b.HasKey("AliasId");
@ -818,7 +820,7 @@ namespace Data.Migrations.MySql
b.HasIndex("Name", "IPAddress"); b.HasIndex("Name", "IPAddress");
b.ToTable("EFAlias"); b.ToTable("EFAlias", (string)null);
}); });
modelBuilder.Entity("Data.Models.EFAliasLink", b => modelBuilder.Entity("Data.Models.EFAliasLink", b =>
@ -832,7 +834,7 @@ namespace Data.Migrations.MySql
b.HasKey("AliasLinkId"); b.HasKey("AliasLinkId");
b.ToTable("EFAliasLinks"); b.ToTable("EFAliasLinks", (string)null);
}); });
modelBuilder.Entity("Data.Models.EFChangeHistory", b => modelBuilder.Entity("Data.Models.EFChangeHistory", b =>
@ -845,11 +847,11 @@ namespace Data.Migrations.MySql
.HasColumnType("tinyint(1)"); .HasColumnType("tinyint(1)");
b.Property<string>("Comment") b.Property<string>("Comment")
.HasColumnType("varchar(128) CHARACTER SET utf8mb4") .HasMaxLength(128)
.HasMaxLength(128); .HasColumnType("varchar(128)");
b.Property<string>("CurrentValue") b.Property<string>("CurrentValue")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<int?>("ImpersonationEntityId") b.Property<int?>("ImpersonationEntityId")
.HasColumnType("int"); .HasColumnType("int");
@ -858,7 +860,7 @@ namespace Data.Migrations.MySql
.HasColumnType("int"); .HasColumnType("int");
b.Property<string>("PreviousValue") b.Property<string>("PreviousValue")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<int>("TargetEntityId") b.Property<int>("TargetEntityId")
.HasColumnType("int"); .HasColumnType("int");
@ -890,12 +892,12 @@ namespace Data.Migrations.MySql
.HasColumnType("datetime(6)"); .HasColumnType("datetime(6)");
b.Property<string>("Extra") b.Property<string>("Extra")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<string>("Key") b.Property<string>("Key")
.IsRequired() .IsRequired()
.HasColumnType("varchar(32) CHARACTER SET utf8mb4") .HasMaxLength(32)
.HasMaxLength(32); .HasColumnType("varchar(32)");
b.Property<int?>("LinkedMetaId") b.Property<int?>("LinkedMetaId")
.HasColumnType("int"); .HasColumnType("int");
@ -905,7 +907,7 @@ namespace Data.Migrations.MySql
b.Property<string>("Value") b.Property<string>("Value")
.IsRequired() .IsRequired()
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.HasKey("MetaId"); b.HasKey("MetaId");
@ -928,7 +930,7 @@ namespace Data.Migrations.MySql
.HasColumnType("tinyint(1)"); .HasColumnType("tinyint(1)");
b.Property<string>("AutomatedOffense") b.Property<string>("AutomatedOffense")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<DateTime?>("Expires") b.Property<DateTime?>("Expires")
.HasColumnType("datetime(6)"); .HasColumnType("datetime(6)");
@ -936,7 +938,7 @@ namespace Data.Migrations.MySql
b.Property<bool>("IsEvadedOffense") b.Property<bool>("IsEvadedOffense")
.HasColumnType("tinyint(1)"); .HasColumnType("tinyint(1)");
b.Property<int>("LinkId") b.Property<int?>("LinkId")
.HasColumnType("int"); .HasColumnType("int");
b.Property<int>("OffenderId") b.Property<int>("OffenderId")
@ -944,7 +946,7 @@ namespace Data.Migrations.MySql
b.Property<string>("Offense") b.Property<string>("Offense")
.IsRequired() .IsRequired()
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<int>("PunisherId") b.Property<int>("PunisherId")
.HasColumnType("int"); .HasColumnType("int");
@ -963,7 +965,39 @@ namespace Data.Migrations.MySql
b.HasIndex("PunisherId"); b.HasIndex("PunisherId");
b.ToTable("EFPenalties"); b.ToTable("EFPenalties", (string)null);
});
modelBuilder.Entity("Data.Models.EFPenaltyIdentifier", b =>
{
b.Property<int>("PenaltyIdentifierId")
.ValueGeneratedOnAdd()
.HasColumnType("int");
b.Property<DateTime>("CreatedDateTime")
.HasColumnType("datetime(6)");
b.Property<int?>("IPv4Address")
.HasColumnType("int");
b.Property<long>("NetworkId")
.HasColumnType("bigint");
b.Property<int>("PenaltyId")
.HasColumnType("int");
b.Property<DateTime?>("UpdatedDateTime")
.HasColumnType("datetime(6)");
b.HasKey("PenaltyIdentifierId");
b.HasIndex("IPv4Address");
b.HasIndex("NetworkId");
b.HasIndex("PenaltyId");
b.ToTable("EFPenaltyIdentifiers", (string)null);
}); });
modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b => modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b =>
@ -982,7 +1016,7 @@ namespace Data.Migrations.MySql
.HasColumnType("tinyint(1)"); .HasColumnType("tinyint(1)");
b.Property<string>("Message") b.Property<string>("Message")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<long?>("ServerId") b.Property<long?>("ServerId")
.HasColumnType("bigint"); .HasColumnType("bigint");
@ -1013,13 +1047,13 @@ namespace Data.Migrations.MySql
.HasColumnType("tinyint(1)"); .HasColumnType("tinyint(1)");
b.Property<string>("EndPoint") b.Property<string>("EndPoint")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<int?>("GameName") b.Property<int?>("GameName")
.HasColumnType("int"); .HasColumnType("int");
b.Property<string>("HostName") b.Property<string>("HostName")
.HasColumnType("longtext CHARACTER SET utf8mb4"); .HasColumnType("longtext");
b.Property<bool>("IsPasswordProtected") b.Property<bool>("IsPasswordProtected")
.HasColumnType("tinyint(1)"); .HasColumnType("tinyint(1)");
@ -1029,7 +1063,7 @@ namespace Data.Migrations.MySql
b.HasKey("ServerId"); b.HasKey("ServerId");
b.ToTable("EFServers"); b.ToTable("EFServers", (string)null);
}); });
modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b => modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b =>
@ -1062,7 +1096,7 @@ namespace Data.Migrations.MySql
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFServerSnapshot"); b.ToTable("EFServerSnapshot", (string)null);
}); });
modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b => modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b =>
@ -1087,7 +1121,7 @@ namespace Data.Migrations.MySql
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFServerStatistics"); b.ToTable("EFServerStatistics", (string)null);
}); });
modelBuilder.Entity("Data.Models.Vector3", b => modelBuilder.Entity("Data.Models.Vector3", b =>
@ -1107,7 +1141,7 @@ namespace Data.Migrations.MySql
b.HasKey("Vector3Id"); b.HasKey("Vector3Id");
b.ToTable("Vector3"); b.ToTable("Vector3", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b => modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b =>
@ -1123,6 +1157,10 @@ namespace Data.Migrations.MySql
.HasForeignKey("Vector3Id") .HasForeignKey("Vector3Id")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Snapshot");
b.Navigation("Vector");
}); });
modelBuilder.Entity("Data.Models.Client.EFClient", b => modelBuilder.Entity("Data.Models.Client.EFClient", b =>
@ -1138,6 +1176,10 @@ namespace Data.Migrations.MySql
.HasForeignKey("CurrentAliasId") .HasForeignKey("CurrentAliasId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("AliasLink");
b.Navigation("CurrentAlias");
}); });
modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b => modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b =>
@ -1153,6 +1195,10 @@ namespace Data.Migrations.MySql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.EFClientKill", b => modelBuilder.Entity("Data.Models.Client.EFClientKill", b =>
@ -1186,6 +1232,18 @@ namespace Data.Migrations.MySql
b.HasOne("Data.Models.Vector3", "ViewAngles") b.HasOne("Data.Models.Vector3", "ViewAngles")
.WithMany() .WithMany()
.HasForeignKey("ViewAnglesVector3Id"); .HasForeignKey("ViewAnglesVector3Id");
b.Navigation("Attacker");
b.Navigation("DeathOrigin");
b.Navigation("KillOrigin");
b.Navigation("Server");
b.Navigation("Victim");
b.Navigation("ViewAngles");
}); });
modelBuilder.Entity("Data.Models.Client.EFClientMessage", b => modelBuilder.Entity("Data.Models.Client.EFClientMessage", b =>
@ -1201,6 +1259,10 @@ namespace Data.Migrations.MySql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b => modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b =>
@ -1238,6 +1300,18 @@ namespace Data.Migrations.MySql
b.HasOne("Data.Models.Server.EFServer", "Server") b.HasOne("Data.Models.Server.EFServer", "Server")
.WithMany() .WithMany()
.HasForeignKey("ServerId"); .HasForeignKey("ServerId");
b.Navigation("Client");
b.Navigation("CurrentViewAngle");
b.Navigation("HitDestination");
b.Navigation("HitOrigin");
b.Navigation("LastStrainAngle");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b =>
@ -1267,6 +1341,18 @@ namespace Data.Migrations.MySql
b.HasOne("Data.Models.Client.Stats.Reference.EFWeapon", "Weapon") b.HasOne("Data.Models.Client.Stats.Reference.EFWeapon", "Weapon")
.WithMany() .WithMany()
.HasForeignKey("WeaponId"); .HasForeignKey("WeaponId");
b.Navigation("Client");
b.Navigation("HitLocation");
b.Navigation("MeansOfDeath");
b.Navigation("Server");
b.Navigation("Weapon");
b.Navigation("WeaponAttachmentCombo");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b =>
@ -1280,6 +1366,10 @@ namespace Data.Migrations.MySql
b.HasOne("Data.Models.Server.EFServer", "Server") b.HasOne("Data.Models.Server.EFServer", "Server")
.WithMany() .WithMany()
.HasForeignKey("ServerId"); .HasForeignKey("ServerId");
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b =>
@ -1289,6 +1379,8 @@ namespace Data.Migrations.MySql
.HasForeignKey("ClientId") .HasForeignKey("ClientId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b =>
@ -1304,6 +1396,10 @@ namespace Data.Migrations.MySql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b => modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b =>
@ -1325,6 +1421,10 @@ namespace Data.Migrations.MySql
.HasForeignKey("EFClientStatisticsClientId", "EFClientStatisticsServerId") .HasForeignKey("EFClientStatisticsClientId", "EFClientStatisticsServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b => modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b =>
@ -1338,6 +1438,10 @@ namespace Data.Migrations.MySql
b.HasOne("Data.Models.Server.EFServer", "Server") b.HasOne("Data.Models.Server.EFServer", "Server")
.WithMany() .WithMany()
.HasForeignKey("ServerId"); .HasForeignKey("ServerId");
b.Navigation("RatingHistory");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b =>
@ -1355,6 +1459,12 @@ namespace Data.Migrations.MySql
b.HasOne("Data.Models.Client.Stats.Reference.EFWeaponAttachment", "Attachment3") b.HasOne("Data.Models.Client.Stats.Reference.EFWeaponAttachment", "Attachment3")
.WithMany() .WithMany()
.HasForeignKey("Attachment3Id"); .HasForeignKey("Attachment3Id");
b.Navigation("Attachment1");
b.Navigation("Attachment2");
b.Navigation("Attachment3");
}); });
modelBuilder.Entity("Data.Models.EFAlias", b => modelBuilder.Entity("Data.Models.EFAlias", b =>
@ -1364,6 +1474,8 @@ namespace Data.Migrations.MySql
.HasForeignKey("LinkId") .HasForeignKey("LinkId")
.OnDelete(DeleteBehavior.Restrict) .OnDelete(DeleteBehavior.Restrict)
.IsRequired(); .IsRequired();
b.Navigation("Link");
}); });
modelBuilder.Entity("Data.Models.EFMeta", b => modelBuilder.Entity("Data.Models.EFMeta", b =>
@ -1376,15 +1488,17 @@ namespace Data.Migrations.MySql
.WithMany() .WithMany()
.HasForeignKey("LinkedMetaId") .HasForeignKey("LinkedMetaId")
.OnDelete(DeleteBehavior.SetNull); .OnDelete(DeleteBehavior.SetNull);
b.Navigation("Client");
b.Navigation("LinkedMeta");
}); });
modelBuilder.Entity("Data.Models.EFPenalty", b => modelBuilder.Entity("Data.Models.EFPenalty", b =>
{ {
b.HasOne("Data.Models.EFAliasLink", "Link") b.HasOne("Data.Models.EFAliasLink", "Link")
.WithMany("ReceivedPenalties") .WithMany("ReceivedPenalties")
.HasForeignKey("LinkId") .HasForeignKey("LinkId");
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("Data.Models.Client.EFClient", "Offender") b.HasOne("Data.Models.Client.EFClient", "Offender")
.WithMany("ReceivedPenalties") .WithMany("ReceivedPenalties")
@ -1397,6 +1511,23 @@ namespace Data.Migrations.MySql
.HasForeignKey("PunisherId") .HasForeignKey("PunisherId")
.OnDelete(DeleteBehavior.Restrict) .OnDelete(DeleteBehavior.Restrict)
.IsRequired(); .IsRequired();
b.Navigation("Link");
b.Navigation("Offender");
b.Navigation("Punisher");
});
modelBuilder.Entity("Data.Models.EFPenaltyIdentifier", b =>
{
b.HasOne("Data.Models.EFPenalty", "Penalty")
.WithMany()
.HasForeignKey("PenaltyId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Penalty");
}); });
modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b => modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b =>
@ -1416,6 +1547,12 @@ namespace Data.Migrations.MySql
.HasForeignKey("SourceClientId") .HasForeignKey("SourceClientId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("DestinationClient");
b.Navigation("Server");
b.Navigation("SourceClient");
}); });
modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b => modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b =>
@ -1431,6 +1568,10 @@ namespace Data.Migrations.MySql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Map");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b => modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b =>
@ -1440,6 +1581,39 @@ namespace Data.Migrations.MySql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Server");
});
modelBuilder.Entity("Data.Models.Client.EFClient", b =>
{
b.Navigation("AdministeredPenalties");
b.Navigation("Meta");
b.Navigation("ReceivedPenalties");
});
modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b =>
{
b.Navigation("PredictedViewAngles");
});
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b =>
{
b.Navigation("Ratings");
});
modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b =>
{
b.Navigation("HitLocations");
});
modelBuilder.Entity("Data.Models.EFAliasLink", b =>
{
b.Navigation("Children");
b.Navigation("ReceivedPenalties");
}); });
#pragma warning restore 612, 618 #pragma warning restore 612, 618
} }

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,56 @@
using Microsoft.EntityFrameworkCore.Migrations;
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
#nullable disable
namespace Data.Migrations.Postgresql
{
public partial class AddEFPenaltyIdentifier : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.CreateTable(
name: "EFPenaltyIdentifiers",
columns: table => new
{
PenaltyIdentifierId = table.Column<int>(type: "integer", nullable: false)
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
IPv4Address = table.Column<int>(type: "integer", nullable: true),
NetworkId = table.Column<long>(type: "bigint", nullable: false),
PenaltyId = table.Column<int>(type: "integer", nullable: false),
Active = table.Column<bool>(type: "boolean", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_EFPenaltyIdentifiers", x => x.PenaltyIdentifierId);
table.ForeignKey(
name: "FK_EFPenaltyIdentifiers_EFPenalties_PenaltyId",
column: x => x.PenaltyId,
principalTable: "EFPenalties",
principalColumn: "PenaltyId",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateIndex(
name: "IX_EFPenaltyIdentifiers_IPv4Address",
table: "EFPenaltyIdentifiers",
column: "IPv4Address");
migrationBuilder.CreateIndex(
name: "IX_EFPenaltyIdentifiers_NetworkId",
table: "EFPenaltyIdentifiers",
column: "NetworkId");
migrationBuilder.CreateIndex(
name: "IX_EFPenaltyIdentifiers_PenaltyId",
table: "EFPenaltyIdentifiers",
column: "PenaltyId");
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropTable(
name: "EFPenaltyIdentifiers");
}
}
}

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,56 @@
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace Data.Migrations.Postgresql
{
public partial class MakeEFPenaltyLinkIdNullable : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties");
migrationBuilder.AlterColumn<int>(
name: "LinkId",
table: "EFPenalties",
type: "integer",
nullable: true,
oldClrType: typeof(int),
oldType: "integer");
migrationBuilder.AddForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties",
column: "LinkId",
principalTable: "EFAliasLinks",
principalColumn: "AliasLinkId");
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties");
migrationBuilder.AlterColumn<int>(
name: "LinkId",
table: "EFPenalties",
type: "integer",
nullable: false,
defaultValue: 0,
oldClrType: typeof(int),
oldType: "integer",
oldNullable: true);
migrationBuilder.AddForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties",
column: "LinkId",
principalTable: "EFAliasLinks",
principalColumn: "AliasLinkId",
onDelete: ReferentialAction.Cascade);
}
}
}

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,48 @@
using System;
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace Data.Migrations.Postgresql
{
public partial class AddAuditFieldsToEFPenaltyIdentifier : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "Active",
table: "EFPenaltyIdentifiers");
migrationBuilder.AddColumn<DateTime>(
name: "CreatedDateTime",
table: "EFPenaltyIdentifiers",
type: "timestamp without time zone",
nullable: false,
defaultValue: new DateTime(1, 1, 1, 0, 0, 0, 0, DateTimeKind.Unspecified));
migrationBuilder.AddColumn<DateTime>(
name: "UpdatedDateTime",
table: "EFPenaltyIdentifiers",
type: "timestamp without time zone",
nullable: true);
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "CreatedDateTime",
table: "EFPenaltyIdentifiers");
migrationBuilder.DropColumn(
name: "UpdatedDateTime",
table: "EFPenaltyIdentifiers");
migrationBuilder.AddColumn<bool>(
name: "Active",
table: "EFPenaltyIdentifiers",
type: "boolean",
nullable: false,
defaultValue: false);
}
}
}

View File

@ -6,6 +6,8 @@ using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion; using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
#nullable disable
namespace Data.Migrations.Postgresql namespace Data.Migrations.Postgresql
{ {
[DbContext(typeof(PostgresqlDatabaseContext))] [DbContext(typeof(PostgresqlDatabaseContext))]
@ -15,16 +17,18 @@ namespace Data.Migrations.Postgresql
{ {
#pragma warning disable 612, 618 #pragma warning disable 612, 618
modelBuilder modelBuilder
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn) .HasAnnotation("ProductVersion", "6.0.1")
.HasAnnotation("ProductVersion", "3.1.10")
.HasAnnotation("Relational:MaxIdentifierLength", 63); .HasAnnotation("Relational:MaxIdentifierLength", 63);
NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b => modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b =>
{ {
b.Property<int>("ACSnapshotVector3Id") b.Property<int>("ACSnapshotVector3Id")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("ACSnapshotVector3Id"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -41,15 +45,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("Vector3Id"); b.HasIndex("Vector3Id");
b.ToTable("EFACSnapshotVector3"); b.ToTable("EFACSnapshotVector3", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClient", b => modelBuilder.Entity("Data.Models.Client.EFClient", b =>
{ {
b.Property<int>("ClientId") b.Property<int>("ClientId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("ClientId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -96,15 +101,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("NetworkId") b.HasIndex("NetworkId")
.IsUnique(); .IsUnique();
b.ToTable("EFClients"); b.ToTable("EFClients", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b => modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b =>
{ {
b.Property<long>("ClientConnectionId") b.Property<long>("ClientConnectionId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("bigint") .HasColumnType("bigint");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<long>("ClientConnectionId"));
b.Property<int>("ClientId") b.Property<int>("ClientId")
.HasColumnType("integer"); .HasColumnType("integer");
@ -129,15 +135,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFClientConnectionHistory"); b.ToTable("EFClientConnectionHistory", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClientKill", b => modelBuilder.Entity("Data.Models.Client.EFClientKill", b =>
{ {
b.Property<long>("KillId") b.Property<long>("KillId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("bigint") .HasColumnType("bigint");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<long>("KillId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -204,15 +211,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("ViewAnglesVector3Id"); b.HasIndex("ViewAnglesVector3Id");
b.ToTable("EFClientKills"); b.ToTable("EFClientKills", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClientMessage", b => modelBuilder.Entity("Data.Models.Client.EFClientMessage", b =>
{ {
b.Property<long>("MessageId") b.Property<long>("MessageId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("bigint") .HasColumnType("bigint");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<long>("MessageId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -240,15 +248,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("TimeSent"); b.HasIndex("TimeSent");
b.ToTable("EFClientMessages"); b.ToTable("EFClientMessages", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b => modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b =>
{ {
b.Property<int>("SnapshotId") b.Property<int>("SnapshotId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("SnapshotId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -348,15 +357,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFACSnapshot"); b.ToTable("EFACSnapshot", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b =>
{ {
b.Property<int>("ClientHitStatisticId") b.Property<int>("ClientHitStatisticId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("ClientHitStatisticId"));
b.Property<int>("ClientId") b.Property<int>("ClientId")
.HasColumnType("integer"); .HasColumnType("integer");
@ -423,15 +433,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("WeaponId"); b.HasIndex("WeaponId");
b.ToTable("EFClientHitStatistics"); b.ToTable("EFClientHitStatistics", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b =>
{ {
b.Property<long>("ClientRankingHistoryId") b.Property<long>("ClientRankingHistoryId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("bigint") .HasColumnType("bigint");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<long>("ClientRankingHistoryId"));
b.Property<int>("ClientId") b.Property<int>("ClientId")
.HasColumnType("integer"); .HasColumnType("integer");
@ -469,15 +480,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("ZScore"); b.HasIndex("ZScore");
b.ToTable("EFClientRankingHistory"); b.ToTable("EFClientRankingHistory", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b =>
{ {
b.Property<int>("RatingHistoryId") b.Property<int>("RatingHistoryId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("RatingHistoryId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -489,7 +501,7 @@ namespace Data.Migrations.Postgresql
b.HasIndex("ClientId"); b.HasIndex("ClientId");
b.ToTable("EFClientRatingHistory"); b.ToTable("EFClientRatingHistory", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b =>
@ -547,26 +559,27 @@ namespace Data.Migrations.Postgresql
b.HasIndex("ClientId", "TimePlayed", "ZScore"); b.HasIndex("ClientId", "TimePlayed", "ZScore");
b.ToTable("EFClientStatistics"); b.ToTable("EFClientStatistics", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b => modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b =>
{ {
b.Property<int>("HitLocationCountId") b.Property<int>("HitLocationCountId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("HitLocationCountId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
b.Property<int>("EFClientStatisticsClientId") b.Property<int>("EFClientStatisticsClientId")
.HasColumnName("EFClientStatisticsClientId") .HasColumnType("integer")
.HasColumnType("integer"); .HasColumnName("EFClientStatisticsClientId");
b.Property<long>("EFClientStatisticsServerId") b.Property<long>("EFClientStatisticsServerId")
.HasColumnName("EFClientStatisticsServerId") .HasColumnType("bigint")
.HasColumnType("bigint"); .HasColumnName("EFClientStatisticsServerId");
b.Property<int>("HitCount") b.Property<int>("HitCount")
.HasColumnType("integer"); .HasColumnType("integer");
@ -586,15 +599,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("EFClientStatisticsClientId", "EFClientStatisticsServerId"); b.HasIndex("EFClientStatisticsClientId", "EFClientStatisticsServerId");
b.ToTable("EFHitLocationCounts"); b.ToTable("EFHitLocationCounts", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b => modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b =>
{ {
b.Property<int>("RatingId") b.Property<int>("RatingId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("RatingId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -630,15 +644,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("When", "ServerId", "Performance", "ActivityAmount"); b.HasIndex("When", "ServerId", "Performance", "ActivityAmount");
b.ToTable("EFRating"); b.ToTable("EFRating", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFHitLocation", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFHitLocation", b =>
{ {
b.Property<int>("HitLocationId") b.Property<int>("HitLocationId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("HitLocationId"));
b.Property<DateTime>("CreatedDateTime") b.Property<DateTime>("CreatedDateTime")
.HasColumnType("timestamp without time zone"); .HasColumnType("timestamp without time zone");
@ -657,15 +672,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("Name"); b.HasIndex("Name");
b.ToTable("EFHitLocations"); b.ToTable("EFHitLocations", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMap", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMap", b =>
{ {
b.Property<int>("MapId") b.Property<int>("MapId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("MapId"));
b.Property<DateTime>("CreatedDateTime") b.Property<DateTime>("CreatedDateTime")
.HasColumnType("timestamp without time zone"); .HasColumnType("timestamp without time zone");
@ -682,15 +698,16 @@ namespace Data.Migrations.Postgresql
b.HasKey("MapId"); b.HasKey("MapId");
b.ToTable("EFMaps"); b.ToTable("EFMaps", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMeansOfDeath", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMeansOfDeath", b =>
{ {
b.Property<int>("MeansOfDeathId") b.Property<int>("MeansOfDeathId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("MeansOfDeathId"));
b.Property<DateTime>("CreatedDateTime") b.Property<DateTime>("CreatedDateTime")
.HasColumnType("timestamp without time zone"); .HasColumnType("timestamp without time zone");
@ -707,15 +724,16 @@ namespace Data.Migrations.Postgresql
b.HasKey("MeansOfDeathId"); b.HasKey("MeansOfDeathId");
b.ToTable("EFMeansOfDeath"); b.ToTable("EFMeansOfDeath", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeapon", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeapon", b =>
{ {
b.Property<int>("WeaponId") b.Property<int>("WeaponId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("WeaponId"));
b.Property<DateTime>("CreatedDateTime") b.Property<DateTime>("CreatedDateTime")
.HasColumnType("timestamp without time zone"); .HasColumnType("timestamp without time zone");
@ -734,15 +752,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("Name"); b.HasIndex("Name");
b.ToTable("EFWeapons"); b.ToTable("EFWeapons", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachment", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachment", b =>
{ {
b.Property<int>("WeaponAttachmentId") b.Property<int>("WeaponAttachmentId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("WeaponAttachmentId"));
b.Property<DateTime>("CreatedDateTime") b.Property<DateTime>("CreatedDateTime")
.HasColumnType("timestamp without time zone"); .HasColumnType("timestamp without time zone");
@ -759,15 +778,16 @@ namespace Data.Migrations.Postgresql
b.HasKey("WeaponAttachmentId"); b.HasKey("WeaponAttachmentId");
b.ToTable("EFWeaponAttachments"); b.ToTable("EFWeaponAttachments", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b =>
{ {
b.Property<int>("WeaponAttachmentComboId") b.Property<int>("WeaponAttachmentComboId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("WeaponAttachmentComboId"));
b.Property<int>("Attachment1Id") b.Property<int>("Attachment1Id")
.HasColumnType("integer"); .HasColumnType("integer");
@ -795,15 +815,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("Attachment3Id"); b.HasIndex("Attachment3Id");
b.ToTable("EFWeaponAttachmentCombos"); b.ToTable("EFWeaponAttachmentCombos", (string)null);
}); });
modelBuilder.Entity("Data.Models.EFAlias", b => modelBuilder.Entity("Data.Models.EFAlias", b =>
{ {
b.Property<int>("AliasId") b.Property<int>("AliasId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("AliasId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -819,12 +840,12 @@ namespace Data.Migrations.Postgresql
b.Property<string>("Name") b.Property<string>("Name")
.IsRequired() .IsRequired()
.HasColumnType("character varying(24)") .HasMaxLength(24)
.HasMaxLength(24); .HasColumnType("character varying(24)");
b.Property<string>("SearchableName") b.Property<string>("SearchableName")
.HasColumnType("character varying(24)") .HasMaxLength(24)
.HasMaxLength(24); .HasColumnType("character varying(24)");
b.HasKey("AliasId"); b.HasKey("AliasId");
@ -838,37 +859,39 @@ namespace Data.Migrations.Postgresql
b.HasIndex("Name", "IPAddress"); b.HasIndex("Name", "IPAddress");
b.ToTable("EFAlias"); b.ToTable("EFAlias", (string)null);
}); });
modelBuilder.Entity("Data.Models.EFAliasLink", b => modelBuilder.Entity("Data.Models.EFAliasLink", b =>
{ {
b.Property<int>("AliasLinkId") b.Property<int>("AliasLinkId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("AliasLinkId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
b.HasKey("AliasLinkId"); b.HasKey("AliasLinkId");
b.ToTable("EFAliasLinks"); b.ToTable("EFAliasLinks", (string)null);
}); });
modelBuilder.Entity("Data.Models.EFChangeHistory", b => modelBuilder.Entity("Data.Models.EFChangeHistory", b =>
{ {
b.Property<int>("ChangeHistoryId") b.Property<int>("ChangeHistoryId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("ChangeHistoryId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
b.Property<string>("Comment") b.Property<string>("Comment")
.HasColumnType("character varying(128)") .HasMaxLength(128)
.HasMaxLength(128); .HasColumnType("character varying(128)");
b.Property<string>("CurrentValue") b.Property<string>("CurrentValue")
.HasColumnType("text"); .HasColumnType("text");
@ -900,8 +923,9 @@ namespace Data.Migrations.Postgresql
{ {
b.Property<int>("MetaId") b.Property<int>("MetaId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("MetaId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -917,8 +941,8 @@ namespace Data.Migrations.Postgresql
b.Property<string>("Key") b.Property<string>("Key")
.IsRequired() .IsRequired()
.HasColumnType("character varying(32)") .HasMaxLength(32)
.HasMaxLength(32); .HasColumnType("character varying(32)");
b.Property<int?>("LinkedMetaId") b.Property<int?>("LinkedMetaId")
.HasColumnType("integer"); .HasColumnType("integer");
@ -945,8 +969,9 @@ namespace Data.Migrations.Postgresql
{ {
b.Property<int>("PenaltyId") b.Property<int>("PenaltyId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("PenaltyId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -960,7 +985,7 @@ namespace Data.Migrations.Postgresql
b.Property<bool>("IsEvadedOffense") b.Property<bool>("IsEvadedOffense")
.HasColumnType("boolean"); .HasColumnType("boolean");
b.Property<int>("LinkId") b.Property<int?>("LinkId")
.HasColumnType("integer"); .HasColumnType("integer");
b.Property<int>("OffenderId") b.Property<int>("OffenderId")
@ -987,15 +1012,50 @@ namespace Data.Migrations.Postgresql
b.HasIndex("PunisherId"); b.HasIndex("PunisherId");
b.ToTable("EFPenalties"); b.ToTable("EFPenalties", (string)null);
});
modelBuilder.Entity("Data.Models.EFPenaltyIdentifier", b =>
{
b.Property<int>("PenaltyIdentifierId")
.ValueGeneratedOnAdd()
.HasColumnType("integer");
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("PenaltyIdentifierId"));
b.Property<DateTime>("CreatedDateTime")
.HasColumnType("timestamp without time zone");
b.Property<int?>("IPv4Address")
.HasColumnType("integer");
b.Property<long>("NetworkId")
.HasColumnType("bigint");
b.Property<int>("PenaltyId")
.HasColumnType("integer");
b.Property<DateTime?>("UpdatedDateTime")
.HasColumnType("timestamp without time zone");
b.HasKey("PenaltyIdentifierId");
b.HasIndex("IPv4Address");
b.HasIndex("NetworkId");
b.HasIndex("PenaltyId");
b.ToTable("EFPenaltyIdentifiers", (string)null);
}); });
modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b => modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b =>
{ {
b.Property<int>("InboxMessageId") b.Property<int>("InboxMessageId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("InboxMessageId"));
b.Property<DateTime>("CreatedDateTime") b.Property<DateTime>("CreatedDateTime")
.HasColumnType("timestamp without time zone"); .HasColumnType("timestamp without time zone");
@ -1054,15 +1114,16 @@ namespace Data.Migrations.Postgresql
b.HasKey("ServerId"); b.HasKey("ServerId");
b.ToTable("EFServers"); b.ToTable("EFServers", (string)null);
}); });
modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b => modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b =>
{ {
b.Property<long>("ServerSnapshotId") b.Property<long>("ServerSnapshotId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("bigint") .HasColumnType("bigint");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<long>("ServerSnapshotId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -1088,15 +1149,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFServerSnapshot"); b.ToTable("EFServerSnapshot", (string)null);
}); });
modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b => modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b =>
{ {
b.Property<int>("StatisticId") b.Property<int>("StatisticId")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("StatisticId"));
b.Property<bool>("Active") b.Property<bool>("Active")
.HasColumnType("boolean"); .HasColumnType("boolean");
@ -1114,15 +1176,16 @@ namespace Data.Migrations.Postgresql
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFServerStatistics"); b.ToTable("EFServerStatistics", (string)null);
}); });
modelBuilder.Entity("Data.Models.Vector3", b => modelBuilder.Entity("Data.Models.Vector3", b =>
{ {
b.Property<int>("Vector3Id") b.Property<int>("Vector3Id")
.ValueGeneratedOnAdd() .ValueGeneratedOnAdd()
.HasColumnType("integer") .HasColumnType("integer");
.HasAnnotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.SerialColumn);
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Vector3Id"));
b.Property<float>("X") b.Property<float>("X")
.HasColumnType("real"); .HasColumnType("real");
@ -1135,7 +1198,7 @@ namespace Data.Migrations.Postgresql
b.HasKey("Vector3Id"); b.HasKey("Vector3Id");
b.ToTable("Vector3"); b.ToTable("Vector3", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b => modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b =>
@ -1151,6 +1214,10 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("Vector3Id") .HasForeignKey("Vector3Id")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Snapshot");
b.Navigation("Vector");
}); });
modelBuilder.Entity("Data.Models.Client.EFClient", b => modelBuilder.Entity("Data.Models.Client.EFClient", b =>
@ -1166,6 +1233,10 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("CurrentAliasId") .HasForeignKey("CurrentAliasId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("AliasLink");
b.Navigation("CurrentAlias");
}); });
modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b => modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b =>
@ -1181,6 +1252,10 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.EFClientKill", b => modelBuilder.Entity("Data.Models.Client.EFClientKill", b =>
@ -1214,6 +1289,18 @@ namespace Data.Migrations.Postgresql
b.HasOne("Data.Models.Vector3", "ViewAngles") b.HasOne("Data.Models.Vector3", "ViewAngles")
.WithMany() .WithMany()
.HasForeignKey("ViewAnglesVector3Id"); .HasForeignKey("ViewAnglesVector3Id");
b.Navigation("Attacker");
b.Navigation("DeathOrigin");
b.Navigation("KillOrigin");
b.Navigation("Server");
b.Navigation("Victim");
b.Navigation("ViewAngles");
}); });
modelBuilder.Entity("Data.Models.Client.EFClientMessage", b => modelBuilder.Entity("Data.Models.Client.EFClientMessage", b =>
@ -1229,6 +1316,10 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b => modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b =>
@ -1266,6 +1357,18 @@ namespace Data.Migrations.Postgresql
b.HasOne("Data.Models.Server.EFServer", "Server") b.HasOne("Data.Models.Server.EFServer", "Server")
.WithMany() .WithMany()
.HasForeignKey("ServerId"); .HasForeignKey("ServerId");
b.Navigation("Client");
b.Navigation("CurrentViewAngle");
b.Navigation("HitDestination");
b.Navigation("HitOrigin");
b.Navigation("LastStrainAngle");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b =>
@ -1295,6 +1398,18 @@ namespace Data.Migrations.Postgresql
b.HasOne("Data.Models.Client.Stats.Reference.EFWeapon", "Weapon") b.HasOne("Data.Models.Client.Stats.Reference.EFWeapon", "Weapon")
.WithMany() .WithMany()
.HasForeignKey("WeaponId"); .HasForeignKey("WeaponId");
b.Navigation("Client");
b.Navigation("HitLocation");
b.Navigation("MeansOfDeath");
b.Navigation("Server");
b.Navigation("Weapon");
b.Navigation("WeaponAttachmentCombo");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b =>
@ -1308,6 +1423,10 @@ namespace Data.Migrations.Postgresql
b.HasOne("Data.Models.Server.EFServer", "Server") b.HasOne("Data.Models.Server.EFServer", "Server")
.WithMany() .WithMany()
.HasForeignKey("ServerId"); .HasForeignKey("ServerId");
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b =>
@ -1317,6 +1436,8 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("ClientId") .HasForeignKey("ClientId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b =>
@ -1332,6 +1453,10 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b => modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b =>
@ -1353,6 +1478,10 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("EFClientStatisticsClientId", "EFClientStatisticsServerId") .HasForeignKey("EFClientStatisticsClientId", "EFClientStatisticsServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b => modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b =>
@ -1366,6 +1495,10 @@ namespace Data.Migrations.Postgresql
b.HasOne("Data.Models.Server.EFServer", "Server") b.HasOne("Data.Models.Server.EFServer", "Server")
.WithMany() .WithMany()
.HasForeignKey("ServerId"); .HasForeignKey("ServerId");
b.Navigation("RatingHistory");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b =>
@ -1383,6 +1516,12 @@ namespace Data.Migrations.Postgresql
b.HasOne("Data.Models.Client.Stats.Reference.EFWeaponAttachment", "Attachment3") b.HasOne("Data.Models.Client.Stats.Reference.EFWeaponAttachment", "Attachment3")
.WithMany() .WithMany()
.HasForeignKey("Attachment3Id"); .HasForeignKey("Attachment3Id");
b.Navigation("Attachment1");
b.Navigation("Attachment2");
b.Navigation("Attachment3");
}); });
modelBuilder.Entity("Data.Models.EFAlias", b => modelBuilder.Entity("Data.Models.EFAlias", b =>
@ -1392,6 +1531,8 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("LinkId") .HasForeignKey("LinkId")
.OnDelete(DeleteBehavior.Restrict) .OnDelete(DeleteBehavior.Restrict)
.IsRequired(); .IsRequired();
b.Navigation("Link");
}); });
modelBuilder.Entity("Data.Models.EFMeta", b => modelBuilder.Entity("Data.Models.EFMeta", b =>
@ -1404,15 +1545,17 @@ namespace Data.Migrations.Postgresql
.WithMany() .WithMany()
.HasForeignKey("LinkedMetaId") .HasForeignKey("LinkedMetaId")
.OnDelete(DeleteBehavior.SetNull); .OnDelete(DeleteBehavior.SetNull);
b.Navigation("Client");
b.Navigation("LinkedMeta");
}); });
modelBuilder.Entity("Data.Models.EFPenalty", b => modelBuilder.Entity("Data.Models.EFPenalty", b =>
{ {
b.HasOne("Data.Models.EFAliasLink", "Link") b.HasOne("Data.Models.EFAliasLink", "Link")
.WithMany("ReceivedPenalties") .WithMany("ReceivedPenalties")
.HasForeignKey("LinkId") .HasForeignKey("LinkId");
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("Data.Models.Client.EFClient", "Offender") b.HasOne("Data.Models.Client.EFClient", "Offender")
.WithMany("ReceivedPenalties") .WithMany("ReceivedPenalties")
@ -1425,6 +1568,23 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("PunisherId") .HasForeignKey("PunisherId")
.OnDelete(DeleteBehavior.Restrict) .OnDelete(DeleteBehavior.Restrict)
.IsRequired(); .IsRequired();
b.Navigation("Link");
b.Navigation("Offender");
b.Navigation("Punisher");
});
modelBuilder.Entity("Data.Models.EFPenaltyIdentifier", b =>
{
b.HasOne("Data.Models.EFPenalty", "Penalty")
.WithMany()
.HasForeignKey("PenaltyId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Penalty");
}); });
modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b => modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b =>
@ -1444,6 +1604,12 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("SourceClientId") .HasForeignKey("SourceClientId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("DestinationClient");
b.Navigation("Server");
b.Navigation("SourceClient");
}); });
modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b => modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b =>
@ -1459,6 +1625,10 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Map");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b => modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b =>
@ -1468,6 +1638,39 @@ namespace Data.Migrations.Postgresql
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Server");
});
modelBuilder.Entity("Data.Models.Client.EFClient", b =>
{
b.Navigation("AdministeredPenalties");
b.Navigation("Meta");
b.Navigation("ReceivedPenalties");
});
modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b =>
{
b.Navigation("PredictedViewAngles");
});
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b =>
{
b.Navigation("Ratings");
});
modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b =>
{
b.Navigation("HitLocations");
});
modelBuilder.Entity("Data.Models.EFAliasLink", b =>
{
b.Navigation("Children");
b.Navigation("ReceivedPenalties");
}); });
#pragma warning restore 612, 618 #pragma warning restore 612, 618
} }

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,55 @@
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace Data.Migrations.Sqlite
{
public partial class AddEFPenaltyIdentifier : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.CreateTable(
name: "EFPenaltyIdentifiers",
columns: table => new
{
PenaltyIdentifierId = table.Column<int>(type: "INTEGER", nullable: false)
.Annotation("Sqlite:Autoincrement", true),
IPv4Address = table.Column<int>(type: "INTEGER", nullable: true),
NetworkId = table.Column<long>(type: "INTEGER", nullable: false),
PenaltyId = table.Column<int>(type: "INTEGER", nullable: false),
Active = table.Column<bool>(type: "INTEGER", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_EFPenaltyIdentifiers", x => x.PenaltyIdentifierId);
table.ForeignKey(
name: "FK_EFPenaltyIdentifiers_EFPenalties_PenaltyId",
column: x => x.PenaltyId,
principalTable: "EFPenalties",
principalColumn: "PenaltyId",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateIndex(
name: "IX_EFPenaltyIdentifiers_IPv4Address",
table: "EFPenaltyIdentifiers",
column: "IPv4Address");
migrationBuilder.CreateIndex(
name: "IX_EFPenaltyIdentifiers_NetworkId",
table: "EFPenaltyIdentifiers",
column: "NetworkId");
migrationBuilder.CreateIndex(
name: "IX_EFPenaltyIdentifiers_PenaltyId",
table: "EFPenaltyIdentifiers",
column: "PenaltyId");
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropTable(
name: "EFPenaltyIdentifiers");
}
}
}

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,56 @@
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace Data.Migrations.Sqlite
{
public partial class MakeEFPenaltyLinkIdNullable : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties");
migrationBuilder.AlterColumn<int>(
name: "LinkId",
table: "EFPenalties",
type: "INTEGER",
nullable: true,
oldClrType: typeof(int),
oldType: "INTEGER");
migrationBuilder.AddForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties",
column: "LinkId",
principalTable: "EFAliasLinks",
principalColumn: "AliasLinkId");
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties");
migrationBuilder.AlterColumn<int>(
name: "LinkId",
table: "EFPenalties",
type: "INTEGER",
nullable: false,
defaultValue: 0,
oldClrType: typeof(int),
oldType: "INTEGER",
oldNullable: true);
migrationBuilder.AddForeignKey(
name: "FK_EFPenalties_EFAliasLinks_LinkId",
table: "EFPenalties",
column: "LinkId",
principalTable: "EFAliasLinks",
principalColumn: "AliasLinkId",
onDelete: ReferentialAction.Cascade);
}
}
}

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,48 @@
using System;
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace Data.Migrations.Sqlite
{
public partial class AddAuditFieldsToEFPenaltyIdentifier : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "Active",
table: "EFPenaltyIdentifiers");
migrationBuilder.AddColumn<DateTime>(
name: "CreatedDateTime",
table: "EFPenaltyIdentifiers",
type: "TEXT",
nullable: false,
defaultValue: DateTime.UtcNow);
migrationBuilder.AddColumn<DateTime>(
name: "UpdatedDateTime",
table: "EFPenaltyIdentifiers",
type: "TEXT",
nullable: true);
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropColumn(
name: "CreatedDateTime",
table: "EFPenaltyIdentifiers");
migrationBuilder.DropColumn(
name: "UpdatedDateTime",
table: "EFPenaltyIdentifiers");
migrationBuilder.AddColumn<bool>(
name: "Active",
table: "EFPenaltyIdentifiers",
type: "INTEGER",
nullable: false,
defaultValue: false);
}
}
}

View File

@ -5,6 +5,8 @@ using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure; using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion; using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
#nullable disable
namespace Data.Migrations.Sqlite namespace Data.Migrations.Sqlite
{ {
[DbContext(typeof(SqliteDatabaseContext))] [DbContext(typeof(SqliteDatabaseContext))]
@ -13,8 +15,7 @@ namespace Data.Migrations.Sqlite
protected override void BuildModel(ModelBuilder modelBuilder) protected override void BuildModel(ModelBuilder modelBuilder)
{ {
#pragma warning disable 612, 618 #pragma warning disable 612, 618
modelBuilder modelBuilder.HasAnnotation("ProductVersion", "6.0.1");
.HasAnnotation("ProductVersion", "3.1.10");
modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b => modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b =>
{ {
@ -37,7 +38,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("Vector3Id"); b.HasIndex("Vector3Id");
b.ToTable("EFACSnapshotVector3"); b.ToTable("EFACSnapshotVector3", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClient", b => modelBuilder.Entity("Data.Models.Client.EFClient", b =>
@ -91,7 +92,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("NetworkId") b.HasIndex("NetworkId")
.IsUnique(); .IsUnique();
b.ToTable("EFClients"); b.ToTable("EFClients", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b => modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b =>
@ -123,7 +124,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFClientConnectionHistory"); b.ToTable("EFClientConnectionHistory", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClientKill", b => modelBuilder.Entity("Data.Models.Client.EFClientKill", b =>
@ -197,7 +198,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("ViewAnglesVector3Id"); b.HasIndex("ViewAnglesVector3Id");
b.ToTable("EFClientKills"); b.ToTable("EFClientKills", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFClientMessage", b => modelBuilder.Entity("Data.Models.Client.EFClientMessage", b =>
@ -232,7 +233,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("TimeSent"); b.HasIndex("TimeSent");
b.ToTable("EFClientMessages"); b.ToTable("EFClientMessages", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b => modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b =>
@ -339,7 +340,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFACSnapshot"); b.ToTable("EFACSnapshot", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b =>
@ -413,7 +414,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("WeaponId"); b.HasIndex("WeaponId");
b.ToTable("EFClientHitStatistics"); b.ToTable("EFClientHitStatistics", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b =>
@ -458,7 +459,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("ZScore"); b.HasIndex("ZScore");
b.ToTable("EFClientRankingHistory"); b.ToTable("EFClientRankingHistory", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b =>
@ -477,7 +478,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("ClientId"); b.HasIndex("ClientId");
b.ToTable("EFClientRatingHistory"); b.ToTable("EFClientRatingHistory", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b =>
@ -535,7 +536,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("ClientId", "TimePlayed", "ZScore"); b.HasIndex("ClientId", "TimePlayed", "ZScore");
b.ToTable("EFClientStatistics"); b.ToTable("EFClientStatistics", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b => modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b =>
@ -548,12 +549,12 @@ namespace Data.Migrations.Sqlite
.HasColumnType("INTEGER"); .HasColumnType("INTEGER");
b.Property<int>("EFClientStatisticsClientId") b.Property<int>("EFClientStatisticsClientId")
.HasColumnName("EFClientStatisticsClientId") .HasColumnType("INTEGER")
.HasColumnType("INTEGER"); .HasColumnName("EFClientStatisticsClientId");
b.Property<long>("EFClientStatisticsServerId") b.Property<long>("EFClientStatisticsServerId")
.HasColumnName("EFClientStatisticsServerId") .HasColumnType("INTEGER")
.HasColumnType("INTEGER"); .HasColumnName("EFClientStatisticsServerId");
b.Property<int>("HitCount") b.Property<int>("HitCount")
.HasColumnType("INTEGER"); .HasColumnType("INTEGER");
@ -573,7 +574,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("EFClientStatisticsClientId", "EFClientStatisticsServerId"); b.HasIndex("EFClientStatisticsClientId", "EFClientStatisticsServerId");
b.ToTable("EFHitLocationCounts"); b.ToTable("EFHitLocationCounts", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b => modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b =>
@ -616,7 +617,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("When", "ServerId", "Performance", "ActivityAmount"); b.HasIndex("When", "ServerId", "Performance", "ActivityAmount");
b.ToTable("EFRating"); b.ToTable("EFRating", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFHitLocation", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFHitLocation", b =>
@ -642,7 +643,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("Name"); b.HasIndex("Name");
b.ToTable("EFHitLocations"); b.ToTable("EFHitLocations", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMap", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMap", b =>
@ -666,7 +667,7 @@ namespace Data.Migrations.Sqlite
b.HasKey("MapId"); b.HasKey("MapId");
b.ToTable("EFMaps"); b.ToTable("EFMaps", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMeansOfDeath", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFMeansOfDeath", b =>
@ -690,7 +691,7 @@ namespace Data.Migrations.Sqlite
b.HasKey("MeansOfDeathId"); b.HasKey("MeansOfDeathId");
b.ToTable("EFMeansOfDeath"); b.ToTable("EFMeansOfDeath", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeapon", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeapon", b =>
@ -716,7 +717,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("Name"); b.HasIndex("Name");
b.ToTable("EFWeapons"); b.ToTable("EFWeapons", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachment", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachment", b =>
@ -740,7 +741,7 @@ namespace Data.Migrations.Sqlite
b.HasKey("WeaponAttachmentId"); b.HasKey("WeaponAttachmentId");
b.ToTable("EFWeaponAttachments"); b.ToTable("EFWeaponAttachments", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b =>
@ -775,7 +776,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("Attachment3Id"); b.HasIndex("Attachment3Id");
b.ToTable("EFWeaponAttachmentCombos"); b.ToTable("EFWeaponAttachmentCombos", (string)null);
}); });
modelBuilder.Entity("Data.Models.EFAlias", b => modelBuilder.Entity("Data.Models.EFAlias", b =>
@ -798,12 +799,12 @@ namespace Data.Migrations.Sqlite
b.Property<string>("Name") b.Property<string>("Name")
.IsRequired() .IsRequired()
.HasColumnType("TEXT") .HasMaxLength(24)
.HasMaxLength(24); .HasColumnType("TEXT");
b.Property<string>("SearchableName") b.Property<string>("SearchableName")
.HasColumnType("TEXT") .HasMaxLength(24)
.HasMaxLength(24); .HasColumnType("TEXT");
b.HasKey("AliasId"); b.HasKey("AliasId");
@ -817,7 +818,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("Name", "IPAddress"); b.HasIndex("Name", "IPAddress");
b.ToTable("EFAlias"); b.ToTable("EFAlias", (string)null);
}); });
modelBuilder.Entity("Data.Models.EFAliasLink", b => modelBuilder.Entity("Data.Models.EFAliasLink", b =>
@ -831,7 +832,7 @@ namespace Data.Migrations.Sqlite
b.HasKey("AliasLinkId"); b.HasKey("AliasLinkId");
b.ToTable("EFAliasLinks"); b.ToTable("EFAliasLinks", (string)null);
}); });
modelBuilder.Entity("Data.Models.EFChangeHistory", b => modelBuilder.Entity("Data.Models.EFChangeHistory", b =>
@ -844,8 +845,8 @@ namespace Data.Migrations.Sqlite
.HasColumnType("INTEGER"); .HasColumnType("INTEGER");
b.Property<string>("Comment") b.Property<string>("Comment")
.HasColumnType("TEXT") .HasMaxLength(128)
.HasMaxLength(128); .HasColumnType("TEXT");
b.Property<string>("CurrentValue") b.Property<string>("CurrentValue")
.HasColumnType("TEXT"); .HasColumnType("TEXT");
@ -893,8 +894,8 @@ namespace Data.Migrations.Sqlite
b.Property<string>("Key") b.Property<string>("Key")
.IsRequired() .IsRequired()
.HasColumnType("TEXT") .HasMaxLength(32)
.HasMaxLength(32); .HasColumnType("TEXT");
b.Property<int?>("LinkedMetaId") b.Property<int?>("LinkedMetaId")
.HasColumnType("INTEGER"); .HasColumnType("INTEGER");
@ -935,7 +936,7 @@ namespace Data.Migrations.Sqlite
b.Property<bool>("IsEvadedOffense") b.Property<bool>("IsEvadedOffense")
.HasColumnType("INTEGER"); .HasColumnType("INTEGER");
b.Property<int>("LinkId") b.Property<int?>("LinkId")
.HasColumnType("INTEGER"); .HasColumnType("INTEGER");
b.Property<int>("OffenderId") b.Property<int>("OffenderId")
@ -962,7 +963,39 @@ namespace Data.Migrations.Sqlite
b.HasIndex("PunisherId"); b.HasIndex("PunisherId");
b.ToTable("EFPenalties"); b.ToTable("EFPenalties", (string)null);
});
modelBuilder.Entity("Data.Models.EFPenaltyIdentifier", b =>
{
b.Property<int>("PenaltyIdentifierId")
.ValueGeneratedOnAdd()
.HasColumnType("INTEGER");
b.Property<DateTime>("CreatedDateTime")
.HasColumnType("TEXT");
b.Property<int?>("IPv4Address")
.HasColumnType("INTEGER");
b.Property<long>("NetworkId")
.HasColumnType("INTEGER");
b.Property<int>("PenaltyId")
.HasColumnType("INTEGER");
b.Property<DateTime?>("UpdatedDateTime")
.HasColumnType("TEXT");
b.HasKey("PenaltyIdentifierId");
b.HasIndex("IPv4Address");
b.HasIndex("NetworkId");
b.HasIndex("PenaltyId");
b.ToTable("EFPenaltyIdentifiers", (string)null);
}); });
modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b => modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b =>
@ -1028,7 +1061,7 @@ namespace Data.Migrations.Sqlite
b.HasKey("ServerId"); b.HasKey("ServerId");
b.ToTable("EFServers"); b.ToTable("EFServers", (string)null);
}); });
modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b => modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b =>
@ -1061,7 +1094,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFServerSnapshot"); b.ToTable("EFServerSnapshot", (string)null);
}); });
modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b => modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b =>
@ -1086,7 +1119,7 @@ namespace Data.Migrations.Sqlite
b.HasIndex("ServerId"); b.HasIndex("ServerId");
b.ToTable("EFServerStatistics"); b.ToTable("EFServerStatistics", (string)null);
}); });
modelBuilder.Entity("Data.Models.Vector3", b => modelBuilder.Entity("Data.Models.Vector3", b =>
@ -1106,7 +1139,7 @@ namespace Data.Migrations.Sqlite
b.HasKey("Vector3Id"); b.HasKey("Vector3Id");
b.ToTable("Vector3"); b.ToTable("Vector3", (string)null);
}); });
modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b => modelBuilder.Entity("Data.Models.Client.EFACSnapshotVector3", b =>
@ -1122,6 +1155,10 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("Vector3Id") .HasForeignKey("Vector3Id")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Snapshot");
b.Navigation("Vector");
}); });
modelBuilder.Entity("Data.Models.Client.EFClient", b => modelBuilder.Entity("Data.Models.Client.EFClient", b =>
@ -1137,6 +1174,10 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("CurrentAliasId") .HasForeignKey("CurrentAliasId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("AliasLink");
b.Navigation("CurrentAlias");
}); });
modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b => modelBuilder.Entity("Data.Models.Client.EFClientConnectionHistory", b =>
@ -1152,6 +1193,10 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.EFClientKill", b => modelBuilder.Entity("Data.Models.Client.EFClientKill", b =>
@ -1185,6 +1230,18 @@ namespace Data.Migrations.Sqlite
b.HasOne("Data.Models.Vector3", "ViewAngles") b.HasOne("Data.Models.Vector3", "ViewAngles")
.WithMany() .WithMany()
.HasForeignKey("ViewAnglesVector3Id"); .HasForeignKey("ViewAnglesVector3Id");
b.Navigation("Attacker");
b.Navigation("DeathOrigin");
b.Navigation("KillOrigin");
b.Navigation("Server");
b.Navigation("Victim");
b.Navigation("ViewAngles");
}); });
modelBuilder.Entity("Data.Models.Client.EFClientMessage", b => modelBuilder.Entity("Data.Models.Client.EFClientMessage", b =>
@ -1200,6 +1257,10 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b => modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b =>
@ -1237,6 +1298,18 @@ namespace Data.Migrations.Sqlite
b.HasOne("Data.Models.Server.EFServer", "Server") b.HasOne("Data.Models.Server.EFServer", "Server")
.WithMany() .WithMany()
.HasForeignKey("ServerId"); .HasForeignKey("ServerId");
b.Navigation("Client");
b.Navigation("CurrentViewAngle");
b.Navigation("HitDestination");
b.Navigation("HitOrigin");
b.Navigation("LastStrainAngle");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientHitStatistic", b =>
@ -1266,6 +1339,18 @@ namespace Data.Migrations.Sqlite
b.HasOne("Data.Models.Client.Stats.Reference.EFWeapon", "Weapon") b.HasOne("Data.Models.Client.Stats.Reference.EFWeapon", "Weapon")
.WithMany() .WithMany()
.HasForeignKey("WeaponId"); .HasForeignKey("WeaponId");
b.Navigation("Client");
b.Navigation("HitLocation");
b.Navigation("MeansOfDeath");
b.Navigation("Server");
b.Navigation("Weapon");
b.Navigation("WeaponAttachmentCombo");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRankingHistory", b =>
@ -1279,6 +1364,10 @@ namespace Data.Migrations.Sqlite
b.HasOne("Data.Models.Server.EFServer", "Server") b.HasOne("Data.Models.Server.EFServer", "Server")
.WithMany() .WithMany()
.HasForeignKey("ServerId"); .HasForeignKey("ServerId");
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b =>
@ -1288,6 +1377,8 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("ClientId") .HasForeignKey("ClientId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b => modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b =>
@ -1303,6 +1394,10 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b => modelBuilder.Entity("Data.Models.Client.Stats.EFHitLocationCount", b =>
@ -1324,6 +1419,10 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("EFClientStatisticsClientId", "EFClientStatisticsServerId") .HasForeignKey("EFClientStatisticsClientId", "EFClientStatisticsServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Client");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b => modelBuilder.Entity("Data.Models.Client.Stats.EFRating", b =>
@ -1337,6 +1436,10 @@ namespace Data.Migrations.Sqlite
b.HasOne("Data.Models.Server.EFServer", "Server") b.HasOne("Data.Models.Server.EFServer", "Server")
.WithMany() .WithMany()
.HasForeignKey("ServerId"); .HasForeignKey("ServerId");
b.Navigation("RatingHistory");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b => modelBuilder.Entity("Data.Models.Client.Stats.Reference.EFWeaponAttachmentCombo", b =>
@ -1354,6 +1457,12 @@ namespace Data.Migrations.Sqlite
b.HasOne("Data.Models.Client.Stats.Reference.EFWeaponAttachment", "Attachment3") b.HasOne("Data.Models.Client.Stats.Reference.EFWeaponAttachment", "Attachment3")
.WithMany() .WithMany()
.HasForeignKey("Attachment3Id"); .HasForeignKey("Attachment3Id");
b.Navigation("Attachment1");
b.Navigation("Attachment2");
b.Navigation("Attachment3");
}); });
modelBuilder.Entity("Data.Models.EFAlias", b => modelBuilder.Entity("Data.Models.EFAlias", b =>
@ -1363,6 +1472,8 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("LinkId") .HasForeignKey("LinkId")
.OnDelete(DeleteBehavior.Restrict) .OnDelete(DeleteBehavior.Restrict)
.IsRequired(); .IsRequired();
b.Navigation("Link");
}); });
modelBuilder.Entity("Data.Models.EFMeta", b => modelBuilder.Entity("Data.Models.EFMeta", b =>
@ -1375,15 +1486,17 @@ namespace Data.Migrations.Sqlite
.WithMany() .WithMany()
.HasForeignKey("LinkedMetaId") .HasForeignKey("LinkedMetaId")
.OnDelete(DeleteBehavior.SetNull); .OnDelete(DeleteBehavior.SetNull);
b.Navigation("Client");
b.Navigation("LinkedMeta");
}); });
modelBuilder.Entity("Data.Models.EFPenalty", b => modelBuilder.Entity("Data.Models.EFPenalty", b =>
{ {
b.HasOne("Data.Models.EFAliasLink", "Link") b.HasOne("Data.Models.EFAliasLink", "Link")
.WithMany("ReceivedPenalties") .WithMany("ReceivedPenalties")
.HasForeignKey("LinkId") .HasForeignKey("LinkId");
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("Data.Models.Client.EFClient", "Offender") b.HasOne("Data.Models.Client.EFClient", "Offender")
.WithMany("ReceivedPenalties") .WithMany("ReceivedPenalties")
@ -1396,6 +1509,23 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("PunisherId") .HasForeignKey("PunisherId")
.OnDelete(DeleteBehavior.Restrict) .OnDelete(DeleteBehavior.Restrict)
.IsRequired(); .IsRequired();
b.Navigation("Link");
b.Navigation("Offender");
b.Navigation("Punisher");
});
modelBuilder.Entity("Data.Models.EFPenaltyIdentifier", b =>
{
b.HasOne("Data.Models.EFPenalty", "Penalty")
.WithMany()
.HasForeignKey("PenaltyId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Penalty");
}); });
modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b => modelBuilder.Entity("Data.Models.Misc.EFInboxMessage", b =>
@ -1415,6 +1545,12 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("SourceClientId") .HasForeignKey("SourceClientId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("DestinationClient");
b.Navigation("Server");
b.Navigation("SourceClient");
}); });
modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b => modelBuilder.Entity("Data.Models.Server.EFServerSnapshot", b =>
@ -1430,6 +1566,10 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Map");
b.Navigation("Server");
}); });
modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b => modelBuilder.Entity("Data.Models.Server.EFServerStatistics", b =>
@ -1439,6 +1579,39 @@ namespace Data.Migrations.Sqlite
.HasForeignKey("ServerId") .HasForeignKey("ServerId")
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
b.Navigation("Server");
});
modelBuilder.Entity("Data.Models.Client.EFClient", b =>
{
b.Navigation("AdministeredPenalties");
b.Navigation("Meta");
b.Navigation("ReceivedPenalties");
});
modelBuilder.Entity("Data.Models.Client.Stats.EFACSnapshot", b =>
{
b.Navigation("PredictedViewAngles");
});
modelBuilder.Entity("Data.Models.Client.Stats.EFClientRatingHistory", b =>
{
b.Navigation("Ratings");
});
modelBuilder.Entity("Data.Models.Client.Stats.EFClientStatistics", b =>
{
b.Navigation("HitLocations");
});
modelBuilder.Entity("Data.Models.EFAliasLink", b =>
{
b.Navigation("Children");
b.Navigation("ReceivedPenalties");
}); });
#pragma warning restore 612, 618 #pragma warning restore 612, 618
} }

View File

@ -10,4 +10,4 @@ namespace Stats.Models
public DateTime CreatedDateTime { get; set; } = DateTime.UtcNow; public DateTime CreatedDateTime { get; set; } = DateTime.UtcNow;
public DateTime? UpdatedDateTime { get; set; } public DateTime? UpdatedDateTime { get; set; }
} }
} }

View File

@ -10,8 +10,8 @@ namespace Data.Models
/// </summary> /// </summary>
public class EFMeta : SharedEntity public class EFMeta : SharedEntity
{ {
public const string ClientTagName = nameof(ClientTagName); public const string ClientTagNameV2 = nameof(ClientTagNameV2);
public const string ClientTag = nameof(ClientTag); public const string ClientTagV2 = nameof(ClientTagV2);
[Key] [Key]
public int MetaId { get; set; } public int MetaId { get; set; }

View File

@ -23,8 +23,7 @@ namespace Data.Models
[Key] [Key]
public int PenaltyId { get; set; } public int PenaltyId { get; set; }
[Required] public int? LinkId { get; set; }
public int LinkId { get; set; }
[ForeignKey("LinkId")] [ForeignKey("LinkId")]
public virtual EFAliasLink Link { get; set; } public virtual EFAliasLink Link { get; set; }
[Required] [Required]

View File

@ -0,0 +1,22 @@
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using Stats.Models;
namespace Data.Models;
public class EFPenaltyIdentifier : AuditFields
{
[Key]
public int PenaltyIdentifierId { get; set; }
public int? IPv4Address { get; set; }
[Required]
public long NetworkId { get; set; }
[Required]
public int PenaltyId { get; set; }
[ForeignKey(nameof(PenaltyId))]
public EFPenalty Penalty { get; set; }
}

View File

@ -33,7 +33,12 @@ init()
level.eventCallbacks[level.eventTypes.clientDataReceived] = ::OnClientDataReceived; level.eventCallbacks[level.eventTypes.clientDataReceived] = ::OnClientDataReceived;
level.eventCallbacks[level.eventTypes.executeCommandRequested] = ::OnExecuteCommand; level.eventCallbacks[level.eventTypes.executeCommandRequested] = ::OnExecuteCommand;
level.eventCallbacks[level.eventTypes.setClientDataCompleted] = ::OnSetClientDataCompleted; level.eventCallbacks[level.eventTypes.setClientDataCompleted] = ::OnSetClientDataCompleted;
if ( GetDvarInt( "sv_iw4madmin_integration_enabled" ) != 1 )
{
return;
}
// start long running tasks // start long running tasks
level thread MonitorClientEvents(); level thread MonitorClientEvents();
level thread MonitorBus(); level thread MonitorBus();
@ -60,7 +65,15 @@ OnPlayerConnect()
} }
player thread OnPlayerSpawned(); player thread OnPlayerSpawned();
player thread OnPlayerJoinedTeam();
player thread OnPlayerJoinedSpectators();
player thread PlayerTrackingOnInterval(); player thread PlayerTrackingOnInterval();
// only toggle if it's enabled
if ( IsDefined( level.nightModeEnabled ) && level.nightModeEnabled )
{
player ToggleNightMode();
}
} }
} }
@ -86,6 +99,30 @@ OnPlayerDisconnect()
} }
} }
OnPlayerJoinedTeam()
{
self endon( "disconnect" );
for( ;; )
{
self waittill( "joined_team" );
// join spec and join team occur at the same moment - out of order logging would be problematic
wait( 0.25 );
LogPrint( GenerateJoinTeamString( false ) );
}
}
OnPlayerJoinedSpectators()
{
self endon( "disconnect" );
for( ;; )
{
self waittill( "joined_spectators" );
LogPrint( GenerateJoinTeamString( true ) );
}
}
OnGameEnded() OnGameEnded()
{ {
level endon ( "disconnect" ); level endon ( "disconnect" );
@ -94,7 +131,7 @@ OnGameEnded()
{ {
level waittill( "game_ended" ); level waittill( "game_ended" );
// note: you can run data code here but it's possible for // note: you can run data code here but it's possible for
// data to get trucated, so we will try a timer based approach for now // data to get truncated, so we will try a timer based approach for now
} }
} }
@ -117,6 +154,11 @@ DisplayWelcomeData()
PlayerConnectEvents() PlayerConnectEvents()
{ {
self endon( "disconnect" ); self endon( "disconnect" );
if ( IsDefined( self.isHidden ) && self.isHidden )
{
self HideImpl();
}
clientData = self.pers[level.clientDataKey]; clientData = self.pers[level.clientDataKey];
@ -199,6 +241,37 @@ DecrementClientMeta( metaKey, decrementValue, clientId )
SetClientMeta( metaKey, decrementValue, clientId, "decrement" ); SetClientMeta( metaKey, decrementValue, clientId, "decrement" );
} }
GenerateJoinTeamString( isSpectator )
{
team = self.team;
if ( IsDefined( self.joining_team ) )
{
team = self.joining_team;
}
else
{
if ( isSpectator || !IsDefined( team ) )
{
team = "spectator";
}
}
guid = self GetXuid();
if ( guid == "0" )
{
guid = self.guid;
}
if ( !IsDefined( guid ) || guid == "0" )
{
guid = "undefined";
}
return "JT;" + guid + ";" + self getEntityNumber() + ";" + team + ";" + self.name + "\n";
}
SetClientMeta( metaKey, metaValue, clientId, direction ) SetClientMeta( metaKey, metaValue, clientId, direction )
{ {
data = "key=" + metaKey + "|value=" + metaValue; data = "key=" + metaKey + "|value=" + metaValue;
@ -228,15 +301,21 @@ SaveTrackingMetrics()
{ {
if ( level.iw4adminIntegrationDebug == 1 ) if ( level.iw4adminIntegrationDebug == 1 )
{ {
self IPrintLn( "Saving tracking metrics for " + self.persistentClientId ); IPrintLn( "Saving tracking metrics for " + self.persistentClientId );
}
if ( !IsDefined( self.lastShotCount ) )
{
self.lastShotCount = 0;
} }
currentShotCount = self getPlayerStat( "mostshotsfired" ); currentShotCount = self getPlayerStat( "mostshotsfired" );
change = currentShotCount - self.lastShotCount; change = currentShotCount - self.lastShotCount;
self.lastShotCount = currentShotCount;
if ( level.iw4adminIntegrationDebug == 1 ) if ( level.iw4adminIntegrationDebug == 1 )
{ {
self IPrintLn( "Total Shots Fired increased by " + change ); IPrintLn( "Total Shots Fired increased by " + change );
} }
if ( !IsDefined( change ) ) if ( !IsDefined( change ) )
@ -251,7 +330,6 @@ SaveTrackingMetrics()
IncrementClientMeta( "TotalShotsFired", change, self.persistentClientId ); IncrementClientMeta( "TotalShotsFired", change, self.persistentClientId );
self.lastShotCount = currentShotCount;
} }
BuildEventRequest( responseExpected, eventType, eventSubtype, entOrId, data ) BuildEventRequest( responseExpected, eventType, eventSubtype, entOrId, data )
@ -391,18 +469,22 @@ NotifyClientEventTimeout( eventType )
NotifyClientEvent( eventInfo ) NotifyClientEvent( eventInfo )
{ {
client = getPlayerFromClientNum( int( eventInfo[3] ) ); origin = getPlayerFromClientNum( int( eventInfo[3] ) );
target = getPlayerFromClientNum( int( eventInfo[4] ) );
event = spawnstruct(); event = spawnstruct();
event.type = eventInfo[1]; event.type = eventInfo[1];
event.subtype = eventInfo[2]; event.subtype = eventInfo[2];
event.data = eventInfo[4]; event.data = eventInfo[5];
event.origin = origin;
event.target = target;
if ( level.iw4adminIntegrationDebug == 1 ) if ( level.iw4adminIntegrationDebug == 1 )
{ {
IPrintLn(event.data); IPrintLn( "NotifyClientEvent->" + event.data );
} }
client = event.origin;
client.event = event; client.event = event;
level notify( level.eventTypes.localClientEvent, client ); level notify( level.eventTypes.localClientEvent, client );
@ -452,33 +534,59 @@ OnClientDataReceived( event )
OnExecuteCommand( event ) OnExecuteCommand( event )
{ {
data = ParseDataString( event.data ); data = ParseDataString( event.data );
response = "";
switch ( event.subtype ) switch ( event.subtype )
{ {
case "GiveWeapon": case "GiveWeapon":
self GiveWeaponImpl( data ); response = event.target GiveWeaponImpl( data );
break; break;
case "TakeWeapons": case "TakeWeapons":
self TakeWeaponsImpl(); response = event.target TakeWeaponsImpl();
break; break;
case "SwitchTeams": case "SwitchTeams":
self TeamSwitchImpl(); response = event.target TeamSwitchImpl();
break; break;
case "Hide": case "Hide":
self HideImpl(); response = self HideImpl();
break; break;
case "Unhide": case "Unhide":
self UnhideImpl(); response = self UnhideImpl();
break; break;
case "Alert": case "Alert":
self AlertImpl( data ); response = event.target AlertImpl( data );
break; break;
case "Goto":
if ( IsDefined( event.target ) )
{
response = self GotoPlayerImpl( event.target );
}
else
{
response = self GotoImpl( data );
}
break;
case "Kill":
response = event.target KillImpl();
break;
case "NightMode":
NightModeImpl();
break;
case "SetSpectator":
response = event.target SetSpectatorImpl();
break;
}
// send back the response to the origin, but only if they're not the target
if ( response != "" && IsPlayer( event.origin ) && event.origin != event.target )
{
event.origin IPrintLnBold( response );
} }
} }
OnSetClientDataCompleted( event ) OnSetClientDataCompleted( event )
{ {
// IW4MAdmin let us know it persisted (success or fail) // IW4MAdmin let us know it persisted (success or fail)
if ( level.iw4adminIntegrationDebug == 1 ) if ( level.iw4adminIntegrationDebug == 1 )
{ {
IPrintLn( "Set Client Data -> subtype = " + event.subType + " status = " + event.data["status"] ); IPrintLn( "Set Client Data -> subtype = " + event.subType + " status = " + event.data["status"] );
@ -491,54 +599,201 @@ OnSetClientDataCompleted( event )
GiveWeaponImpl( data ) GiveWeaponImpl( data )
{ {
if ( IsAlive( self ) ) if ( !IsAlive( self ) )
{ {
self IPrintLnBold( "You have been given a new weapon" ); return self.name + "^7 is not alive";
self GiveWeapon( data["weaponName"] );
self SwitchToWeapon( data["weaponName"] );
} }
self IPrintLnBold( "You have been given a new weapon" );
self GiveWeapon( data["weaponName"] );
self SwitchToWeapon( data["weaponName"] );
return self.name + "^7 has been given ^5" + data["weaponName"];
} }
TakeWeaponsImpl() TakeWeaponsImpl()
{ {
if ( IsAlive( self ) ) if ( !IsAlive( self ) )
{ {
self TakeAllWeapons(); return self.name + "^7 is not alive";
self IPrintLnBold( "All your weapons have been taken" );
} }
self TakeAllWeapons();
self IPrintLnBold( "All your weapons have been taken" );
return "Took weapons from " + self.name;
} }
TeamSwitchImpl() TeamSwitchImpl()
{ {
if ( !IsAlive( self ) )
{
return self.name + "^7 is not alive";
}
team = level.allies;
if ( self.team == "allies" ) if ( self.team == "allies" )
{ {
self [[level.axis]](); team = level.axis;
}
else
{
self [[level.allies]]();
} }
self IPrintLnBold( "You are being team switched" );
wait( 2 );
self [[team]]();
return self.name + "^7 switched to " + self.team;
} }
HideImpl() HideImpl()
{ {
if ( IsAlive( self ) ) if ( !IsAlive( self ) )
{ {
self Hide(); self IPrintLnBold( "You are not alive" );
self IPrintLnBold( "You are now hidden" ); return;
} }
self SetClientDvar( "sv_cheats", 1 );
self SetClientDvar( "cg_thirdperson", 1 );
self SetClientDvar( "sv_cheats", 0 );
if ( !IsDefined( self.savedHealth ) || self.health < 1000 )
{
self.savedHealth = self.health;
self.savedMaxHealth = self.maxhealth;
}
self.maxhealth = 99999;
self.health = 99999;
self.isHidden = true;
self Hide();
self IPrintLnBold( "You are now ^5hidden ^7from other players" );
} }
UnhideImpl() UnhideImpl()
{ {
if ( IsAlive( self ) ) if ( !IsAlive( self ) )
{ {
self Show(); self IPrintLnBold( "You are not alive" );
self IPrintLnBold( "You are now visible" ); return;
} }
if ( !IsDefined( self.isHidden ) || !self.isHidden )
{
self IPrintLnBold( "You are not hidden" );
return;
}
self SetClientDvar( "sv_cheats", 1 );
self SetClientDvar( "cg_thirdperson", 0 );
self SetClientDvar( "sv_cheats", 0 );
self.health = self.savedHealth;
self.maxhealth = self.savedMaxHealth;
self.isHidden = false;
self Show();
self IPrintLnBold( "You are now ^5visible ^7to other players" );
} }
AlertImpl( data ) AlertImpl( data )
{ {
self thread maps\mp\gametypes\_hud_message::oldNotifyMessage( data["alertType"], data["message"], "compass_waypoint_target", ( 1, 0, 0 ), "ui_mp_nukebomb_timer", 7.5 ); self thread maps\mp\gametypes\_hud_message::oldNotifyMessage( data["alertType"], data["message"], "compass_waypoint_target", ( 1, 0, 0 ), "ui_mp_nukebomb_timer", 7.5 );
return "Sent alert to " + self.name;
}
GotoImpl( data )
{
if ( !IsAlive( self ) )
{
self IPrintLnBold( "You are not alive" );
return;
}
position = ( int(data["x"]), int(data["y"]), int(data["z"]) );
self SetOrigin( position );
self IPrintLnBold( "Moved to " + "("+ position[0] + "," + position[1] + "," + position[2] + ")" );
}
GotoPlayerImpl( target )
{
if ( !IsAlive( target ) )
{
self IPrintLnBold( target.name + " is not alive" );
return;
}
self SetOrigin( target GetOrigin() );
self IPrintLnBold( "Moved to " + target.name );
}
KillImpl()
{
if ( !IsAlive( self ) )
{
return self.name + " is not alive";
}
self Suicide();
self IPrintLnBold( "You were killed by " + self.name );
return "You killed " + self.name;
}
NightModeImpl()
{
if ( !IsDefined ( level.nightModeEnabled ) )
{
level.nightModeEnabled = true;
}
else
{
level.nightModeEnabled = !level.nightModeEnabled;
}
message = "^5NightMode ^7is disabled";
if ( level.nightModeEnabled )
{
message = "^5NightMode ^7is enabled";
}
IPrintLnBold( message );
foreach( player in level.players )
{
player ToggleNightMode();
}
}
ToggleNightMode()
{
colorMap = 1;
fxDraw = 1;
if ( IsDefined( level.nightModeEnabled ) && level.nightModeEnabled )
{
colorMap = 0;
fxDraw = 0;
}
self SetClientDvar( "sv_cheats", 1 );
self SetClientDvar( "r_colorMap", colorMap );
self SetClientDvar( "fx_draw", fxDraw );
self SetClientDvar( "sv_cheats", 0 );
}
SetSpectatorImpl()
{
if ( self.pers["team"] == "spectator" )
{
return self.name + " is already spectating";
}
self [[level.spectator]]();
self IPrintLnBold( "You have been moved to spectator" );
return self.name + " has been moved to spectator";
} }

View File

@ -13,6 +13,7 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution
version.txt = version.txt version.txt = version.txt
DeploymentFiles\UpdateIW4MAdmin.ps1 = DeploymentFiles\UpdateIW4MAdmin.ps1 DeploymentFiles\UpdateIW4MAdmin.ps1 = DeploymentFiles\UpdateIW4MAdmin.ps1
DeploymentFiles\UpdateIW4MAdmin.sh = DeploymentFiles\UpdateIW4MAdmin.sh DeploymentFiles\UpdateIW4MAdmin.sh = DeploymentFiles\UpdateIW4MAdmin.sh
GameFiles\IW4x\userraw\scripts\_integration.gsc = GameFiles\IW4x\userraw\scripts\_integration.gsc
EndProjectSection EndProjectSection
EndProject EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "SharedLibraryCore", "SharedLibraryCore\SharedLibraryCore.csproj", "{AA0541A2-8D51-4AD9-B0AC-3D1F5B162481}" Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "SharedLibraryCore", "SharedLibraryCore\SharedLibraryCore.csproj", "{AA0541A2-8D51-4AD9-B0AC-3D1F5B162481}"

View File

@ -23,12 +23,12 @@ namespace Integrations.Cod
/// </summary> /// </summary>
public class CodRConConnection : IRConConnection public class CodRConConnection : IRConConnection
{ {
static readonly ConcurrentDictionary<EndPoint, ConnectionState> ActiveQueries = new ConcurrentDictionary<EndPoint, ConnectionState>(); private static readonly ConcurrentDictionary<EndPoint, ConnectionState> ActiveQueries = new();
public IPEndPoint Endpoint { get; } public IPEndPoint Endpoint { get; }
public string RConPassword { get; } public string RConPassword { get; }
private IRConParser parser; private IRConParser _parser;
private IRConParserConfiguration config; private IRConParserConfiguration _config;
private readonly ILogger _log; private readonly ILogger _log;
private readonly Encoding _gameEncoding; private readonly Encoding _gameEncoding;
private readonly int _retryAttempts; private readonly int _retryAttempts;
@ -44,73 +44,117 @@ namespace Integrations.Cod
public void SetConfiguration(IRConParser parser) public void SetConfiguration(IRConParser parser)
{ {
this.parser = parser; _parser = parser;
config = parser.Configuration; _config = parser.Configuration;
} }
public async Task<string[]> SendQueryAsync(StaticHelpers.QueryType type, string parameters = "") public async Task<string[]> SendQueryAsync(StaticHelpers.QueryType type, string parameters = "",
CancellationToken token = default)
{ {
if (!ActiveQueries.ContainsKey(this.Endpoint)) try
{ {
ActiveQueries.TryAdd(this.Endpoint, new ConnectionState()); return await SendQueryAsyncInternal(type, parameters, token);
}
catch (Exception ex)
{
using (LogContext.PushProperty("Server", Endpoint.ToString()))
{
_log.LogWarning(ex, "Could not complete RCon request");
}
throw;
}
finally
{
if (ActiveQueries[Endpoint].OnComplete.CurrentCount == 0)
{
ActiveQueries[Endpoint].OnComplete.Release();
}
}
}
private async Task<string[]> SendQueryAsyncInternal(StaticHelpers.QueryType type, string parameters = "", CancellationToken token = default)
{
if (!ActiveQueries.ContainsKey(Endpoint))
{
ActiveQueries.TryAdd(Endpoint, new ConnectionState());
} }
var connectionState = ActiveQueries[this.Endpoint]; var connectionState = ActiveQueries[Endpoint];
_log.LogDebug("Waiting for semaphore to be released [{endpoint}]", Endpoint); _log.LogDebug("Waiting for semaphore to be released [{Endpoint}]", Endpoint);
// enter the semaphore so only one query is sent at a time per server. // enter the semaphore so only one query is sent at a time per server.
await connectionState.OnComplete.WaitAsync(); try
{
await connectionState.OnComplete.WaitAsync(token);
}
catch (OperationCanceledException)
{
throw new RConException("Timed out waiting for access to rcon socket");
}
var timeSinceLastQuery = (DateTime.Now - connectionState.LastQuery).TotalMilliseconds; var timeSinceLastQuery = (DateTime.Now - connectionState.LastQuery).TotalMilliseconds;
if (timeSinceLastQuery < config.FloodProtectInterval) if (timeSinceLastQuery < _config.FloodProtectInterval)
{ {
await Task.Delay(config.FloodProtectInterval - (int)timeSinceLastQuery); try
{
await Task.Delay(_config.FloodProtectInterval - (int)timeSinceLastQuery, token);
}
catch (OperationCanceledException)
{
throw new RConException("Timed out waiting for flood protect to expire");
}
} }
connectionState.LastQuery = DateTime.Now; _log.LogDebug("Semaphore has been released [{Endpoint}]", Endpoint);
_log.LogDebug("Query {@QueryInfo}", new { endpoint=Endpoint.ToString(), type, parameters });
_log.LogDebug("Semaphore has been released [{endpoint}]", Endpoint);
_log.LogDebug("Query {@queryInfo}", new { endpoint=Endpoint.ToString(), type, parameters });
byte[] payload = null; byte[] payload = null;
bool waitForResponse = config.WaitForResponse; var waitForResponse = _config.WaitForResponse;
string convertEncoding(string text) string ConvertEncoding(string text)
{ {
byte[] convertedBytes = Utilities.EncodingType.GetBytes(text); var convertedBytes = Utilities.EncodingType.GetBytes(text);
return _gameEncoding.GetString(convertedBytes); return _gameEncoding.GetString(convertedBytes);
} }
try try
{ {
string convertedRConPassword = convertEncoding(RConPassword); var convertedRConPassword = ConvertEncoding(RConPassword);
string convertedParameters = convertEncoding(parameters); var convertedParameters = ConvertEncoding(parameters);
switch (type) switch (type)
{ {
case StaticHelpers.QueryType.GET_DVAR: case StaticHelpers.QueryType.GET_DVAR:
waitForResponse |= true; waitForResponse |= true;
payload = string.Format(config.CommandPrefixes.RConGetDvar, convertedRConPassword, convertedParameters + '\0').Select(Convert.ToByte).ToArray(); payload = string
.Format(_config.CommandPrefixes.RConGetDvar, convertedRConPassword,
convertedParameters + '\0').Select(Convert.ToByte).ToArray();
break; break;
case StaticHelpers.QueryType.SET_DVAR: case StaticHelpers.QueryType.SET_DVAR:
payload = string.Format(config.CommandPrefixes.RConSetDvar, convertedRConPassword, convertedParameters + '\0').Select(Convert.ToByte).ToArray(); payload = string
.Format(_config.CommandPrefixes.RConSetDvar, convertedRConPassword,
convertedParameters + '\0').Select(Convert.ToByte).ToArray();
break; break;
case StaticHelpers.QueryType.COMMAND: case StaticHelpers.QueryType.COMMAND:
payload = string.Format(config.CommandPrefixes.RConCommand, convertedRConPassword, convertedParameters + '\0').Select(Convert.ToByte).ToArray(); payload = string
.Format(_config.CommandPrefixes.RConCommand, convertedRConPassword,
convertedParameters + '\0').Select(Convert.ToByte).ToArray();
break; break;
case StaticHelpers.QueryType.GET_STATUS: case StaticHelpers.QueryType.GET_STATUS:
waitForResponse |= true; waitForResponse |= true;
payload = (config.CommandPrefixes.RConGetStatus + '\0').Select(Convert.ToByte).ToArray(); payload = (_config.CommandPrefixes.RConGetStatus + '\0').Select(Convert.ToByte).ToArray();
break; break;
case StaticHelpers.QueryType.GET_INFO: case StaticHelpers.QueryType.GET_INFO:
waitForResponse |= true; waitForResponse |= true;
payload = (config.CommandPrefixes.RConGetInfo + '\0').Select(Convert.ToByte).ToArray(); payload = (_config.CommandPrefixes.RConGetInfo + '\0').Select(Convert.ToByte).ToArray();
break; break;
case StaticHelpers.QueryType.COMMAND_STATUS: case StaticHelpers.QueryType.COMMAND_STATUS:
waitForResponse |= true; waitForResponse |= true;
payload = string.Format(config.CommandPrefixes.RConCommand, convertedRConPassword, "status\0").Select(Convert.ToByte).ToArray(); payload = string.Format(_config.CommandPrefixes.RConCommand, convertedRConPassword, "status\0")
.Select(Convert.ToByte).ToArray();
break; break;
} }
} }
@ -119,14 +163,14 @@ namespace Integrations.Cod
// e.g: emoji -> windows-1252 // e.g: emoji -> windows-1252
catch (OverflowException ex) catch (OverflowException ex)
{ {
connectionState.OnComplete.Release(1);
using (LogContext.PushProperty("Server", Endpoint.ToString())) using (LogContext.PushProperty("Server", Endpoint.ToString()))
{ {
_log.LogError(ex, "Could not convert RCon data payload to desired encoding {encoding} {params}", _log.LogError(ex, "Could not convert RCon data payload to desired encoding {Encoding} {Params}",
_gameEncoding.EncodingName, parameters); _gameEncoding.EncodingName, parameters);
} }
throw new RConException($"Invalid character encountered when converting encodings"); throw new RConException("Invalid character encountered when converting encodings");
} }
byte[][] response = null; byte[][] response = null;
@ -137,34 +181,66 @@ namespace Integrations.Cod
using (LogContext.PushProperty("Server", Endpoint.ToString())) using (LogContext.PushProperty("Server", Endpoint.ToString()))
{ {
_log.LogInformation( _log.LogInformation(
"Retrying RCon message ({connectionAttempts}/{allowedConnectionFailures} attempts) with parameters {payload}", "Retrying RCon message ({ConnectionAttempts}/{AllowedConnectionFailures} attempts) with parameters {Payload}",
connectionState.ConnectionAttempts, connectionState.ConnectionAttempts,
_retryAttempts, parameters); _retryAttempts, parameters);
} }
} }
using (var socket = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp)
{
DontFragment = false,
Ttl = 100,
ExclusiveAddressUse = true,
})
{
connectionState.SendEventArgs.UserToken = socket;
connectionState.ConnectionAttempts++;
await connectionState.OnSentData.WaitAsync();
await connectionState.OnReceivedData.WaitAsync();
connectionState.BytesReadPerSegment.Clear();
bool exceptionCaught = false;
_log.LogDebug("Sending {payloadLength} bytes to [{endpoint}] ({connectionAttempts}/{allowedConnectionFailures})", using (var socket = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp)
payload.Length, Endpoint, connectionState.ConnectionAttempts, _retryAttempts); {
DontFragment = false,
Ttl = 100,
ExclusiveAddressUse = true,
})
{
// wait for send to be ready
try
{
await connectionState.OnSentData.WaitAsync(token);
}
catch (OperationCanceledException)
{
throw new RConException("Timed out waiting for access to RCon send socket");
}
// wait for receive to be ready
try
{
await connectionState.OnReceivedData.WaitAsync(token);
}
catch (OperationCanceledException)
{
throw new RConException("Timed out waiting for access to RCon receive socket");
}
finally
{
if (connectionState.OnSentData.CurrentCount == 0)
{
connectionState.OnSentData.Release();
}
}
connectionState.SendEventArgs.UserToken = new ConnectionUserToken
{
Socket = socket,
CancellationToken = token
};
connectionState.ConnectionAttempts++;
connectionState.BytesReadPerSegment.Clear();
_log.LogDebug(
"Sending {PayloadLength} bytes to [{Endpoint}] ({ConnectionAttempts}/{AllowedConnectionFailures})",
payload.Length, Endpoint, connectionState.ConnectionAttempts, _retryAttempts);
try try
{ {
connectionState.LastQuery = DateTime.Now;
response = await SendPayloadAsync(payload, waitForResponse, parser.OverrideTimeoutForCommand(parameters)); response = await SendPayloadAsync(payload, waitForResponse,
_parser.OverrideTimeoutForCommand(parameters), token);
if ((response.Length == 0 || response[0].Length == 0) && waitForResponse) if ((response?.Length == 0 || response[0].Length == 0) && waitForResponse)
{ {
throw new RConException("Expected response but got 0 bytes back"); throw new RConException("Expected response but got 0 bytes back");
} }
@ -172,56 +248,73 @@ namespace Integrations.Cod
connectionState.ConnectionAttempts = 0; connectionState.ConnectionAttempts = 0;
} }
catch (OperationCanceledException)
{
// if we timed out due to the cancellation token,
// we don't want to count that as an attempt
connectionState.ConnectionAttempts = 0;
}
catch catch
{ {
// we want to retry with a delay // we want to retry with a delay
if (connectionState.ConnectionAttempts < _retryAttempts) if (connectionState.ConnectionAttempts < _retryAttempts)
{ {
exceptionCaught = true; try
await Task.Delay(StaticHelpers.SocketTimeout(connectionState.ConnectionAttempts)); {
await Task.Delay(StaticHelpers.SocketTimeout(connectionState.ConnectionAttempts), token);
}
catch (OperationCanceledException)
{
return Array.Empty<string>();
}
goto retrySend; goto retrySend;
} }
using (LogContext.PushProperty("Server", Endpoint.ToString())) using (LogContext.PushProperty("Server", Endpoint.ToString()))
{ {
_log.LogWarning( _log.LogWarning(
"Made {connectionAttempts} attempts to send RCon data to server, but received no response", "Made {ConnectionAttempts} attempts to send RCon data to server, but received no response",
connectionState.ConnectionAttempts); connectionState.ConnectionAttempts);
} }
connectionState.ConnectionAttempts = 0; connectionState.ConnectionAttempts = 0;
throw new NetworkException("Reached maximum retry attempts to send RCon data to server"); throw new NetworkException("Reached maximum retry attempts to send RCon data to server");
} }
finally finally
{ {
// we don't want to release if we're going to retry the query try
if (connectionState.OnComplete.CurrentCount == 0 && !exceptionCaught)
{ {
connectionState.OnComplete.Release(1); if (connectionState.OnSentData.CurrentCount == 0)
} {
connectionState.OnSentData.Release();
}
if (connectionState.OnSentData.CurrentCount == 0) if (connectionState.OnReceivedData.CurrentCount == 0)
{ {
connectionState.OnSentData.Release(); connectionState.OnReceivedData.Release();
}
} }
catch
if (connectionState.OnReceivedData.CurrentCount == 0)
{ {
connectionState.OnReceivedData.Release(); // ignored because we can have the socket operation cancelled (which releases the semaphore) but
// this thread is not notified because it's an event
} }
} }
} }
// at this point we can run in parallel and the next request can start because we have our data
if (response.Length == 0) if (response.Length == 0)
{ {
_log.LogDebug("Received empty response for RCon request {@query}", new { endpoint=Endpoint.ToString(), type, parameters }); _log.LogDebug("Received empty response for RCon request {@Query}",
return new string[0]; new { endpoint = Endpoint.ToString(), type, parameters });
return Array.Empty<string>();
} }
string responseString = type == StaticHelpers.QueryType.COMMAND_STATUS ? var responseString = type == StaticHelpers.QueryType.COMMAND_STATUS ?
ReassembleSegmentedStatus(response) : RecombineMessages(response); ReassembleSegmentedStatus(response) : RecombineMessages(response);
// note: not all games respond if the pasword is wrong or not set // note: not all games respond if the password is wrong or not set
if (responseString.Contains("Invalid password") || responseString.Contains("rconpassword")) if (responseString.Contains("Invalid password") || responseString.Contains("rconpassword"))
{ {
throw new RConException(Utilities.CurrentLocalization.LocalizationIndex["SERVER_ERROR_RCON_INVALID"]); throw new RConException(Utilities.CurrentLocalization.LocalizationIndex["SERVER_ERROR_RCON_INVALID"]);
@ -232,26 +325,26 @@ namespace Integrations.Cod
throw new RConException(Utilities.CurrentLocalization.LocalizationIndex["SERVER_ERROR_RCON_NOTSET"]); throw new RConException(Utilities.CurrentLocalization.LocalizationIndex["SERVER_ERROR_RCON_NOTSET"]);
} }
if (responseString.Contains(config.ServerNotRunningResponse)) if (responseString.Contains(_config.ServerNotRunningResponse))
{ {
throw new ServerException(Utilities.CurrentLocalization.LocalizationIndex["SERVER_ERROR_NOT_RUNNING"].FormatExt(Endpoint.ToString())); throw new ServerException(Utilities.CurrentLocalization.LocalizationIndex["SERVER_ERROR_NOT_RUNNING"].FormatExt(Endpoint.ToString()));
} }
string responseHeaderMatch = Regex.Match(responseString, config.CommandPrefixes.RConResponse).Value; var responseHeaderMatch = Regex.Match(responseString, _config.CommandPrefixes.RConResponse).Value;
string[] headerSplit = responseString.Split(type == StaticHelpers.QueryType.GET_INFO ? config.CommandPrefixes.RconGetInfoResponseHeader : responseHeaderMatch); var headerSplit = responseString.Split(type == StaticHelpers.QueryType.GET_INFO ? _config.CommandPrefixes.RconGetInfoResponseHeader : responseHeaderMatch);
if (headerSplit.Length != 2) if (headerSplit.Length != 2)
{ {
using (LogContext.PushProperty("Server", Endpoint.ToString())) using (LogContext.PushProperty("Server", Endpoint.ToString()))
{ {
_log.LogWarning("Invalid response header from server. Expected {expected}, but got {response}", _log.LogWarning("Invalid response header from server. Expected {Expected}, but got {Response}",
config.CommandPrefixes.RConResponse, headerSplit.FirstOrDefault()); _config.CommandPrefixes.RConResponse, headerSplit.FirstOrDefault());
} }
throw new RConException("Unexpected response header from server"); throw new RConException("Unexpected response header from server");
} }
string[] splitResponse = headerSplit.Last().Split(new char[] { '\n' }, StringSplitOptions.RemoveEmptyEntries); var splitResponse = headerSplit.Last().Split(new[] { '\n' }, StringSplitOptions.RemoveEmptyEntries);
return splitResponse; return splitResponse;
} }
@ -261,14 +354,14 @@ namespace Integrations.Cod
/// </summary> /// </summary>
/// <param name="segments">array of segmented byte arrays</param> /// <param name="segments">array of segmented byte arrays</param>
/// <returns></returns> /// <returns></returns>
public string ReassembleSegmentedStatus(byte[][] segments) private string ReassembleSegmentedStatus(IEnumerable<byte[]> segments)
{ {
var splitStatusStrings = new List<string>(); var splitStatusStrings = new List<string>();
foreach (byte[] segment in segments) foreach (var segment in segments)
{ {
string responseString = _gameEncoding.GetString(segment, 0, segment.Length); var responseString = _gameEncoding.GetString(segment, 0, segment.Length);
var statusHeaderMatch = config.StatusHeader.PatternMatcher.Match(responseString); var statusHeaderMatch = _config.StatusHeader.PatternMatcher.Match(responseString);
if (statusHeaderMatch.Success) if (statusHeaderMatch.Success)
{ {
splitStatusStrings.Insert(0, responseString.TrimEnd('\0')); splitStatusStrings.Insert(0, responseString.TrimEnd('\0'));
@ -276,7 +369,7 @@ namespace Integrations.Cod
else else
{ {
splitStatusStrings.Add(responseString.Replace(config.CommandPrefixes.RConResponse, "").TrimEnd('\0')); splitStatusStrings.Add(responseString.Replace(_config.CommandPrefixes.RConResponse, "").TrimEnd('\0'));
} }
} }
@ -288,34 +381,37 @@ namespace Integrations.Cod
/// </summary> /// </summary>
/// <param name="payload"></param> /// <param name="payload"></param>
/// <returns></returns> /// <returns></returns>
private string RecombineMessages(byte[][] payload) private string RecombineMessages(IReadOnlyList<byte[]> payload)
{ {
if (payload.Length == 1) if (payload.Count == 1)
{ {
return _gameEncoding.GetString(payload[0]).TrimEnd('\n') + '\n'; return _gameEncoding.GetString(payload[0]).TrimEnd('\n') + '\n';
} }
else var builder = new StringBuilder();
for (var i = 0; i < payload.Count; i++)
{ {
var builder = new StringBuilder(); var message = _gameEncoding.GetString(payload[i]).TrimEnd('\n') + '\n';
for (int i = 0; i < payload.Length; i++) if (i > 0)
{ {
string message = _gameEncoding.GetString(payload[i]).TrimEnd('\n') + '\n'; message = message.Replace(_config.CommandPrefixes.RConResponse, "");
if (i > 0)
{
message = message.Replace(config.CommandPrefixes.RConResponse, "");
}
builder.Append(message);
} }
builder.Append('\n'); builder.Append(message);
return builder.ToString();
} }
builder.Append('\n');
return builder.ToString();
} }
private async Task<byte[][]> SendPayloadAsync(byte[] payload, bool waitForResponse, TimeSpan overrideTimeout) private async Task<byte[][]> SendPayloadAsync(byte[] payload, bool waitForResponse, TimeSpan overrideTimeout,
CancellationToken token = default)
{ {
var connectionState = ActiveQueries[this.Endpoint]; var connectionState = ActiveQueries[Endpoint];
var rconSocket = (Socket)connectionState.SendEventArgs.UserToken; var rconSocket = ((ConnectionUserToken)connectionState.SendEventArgs.UserToken)?.Socket;
if (rconSocket is null)
{
throw new InvalidOperationException("State is not valid for socket operation");
}
if (connectionState.ReceiveEventArgs.RemoteEndPoint == null && if (connectionState.ReceiveEventArgs.RemoteEndPoint == null &&
connectionState.SendEventArgs.RemoteEndPoint == null) connectionState.SendEventArgs.RemoteEndPoint == null)
@ -323,8 +419,8 @@ namespace Integrations.Cod
// setup the event handlers only once because we're reusing the event args // setup the event handlers only once because we're reusing the event args
connectionState.SendEventArgs.Completed += OnDataSent; connectionState.SendEventArgs.Completed += OnDataSent;
connectionState.ReceiveEventArgs.Completed += OnDataReceived; connectionState.ReceiveEventArgs.Completed += OnDataReceived;
connectionState.SendEventArgs.RemoteEndPoint = this.Endpoint; connectionState.SendEventArgs.RemoteEndPoint = Endpoint;
connectionState.ReceiveEventArgs.RemoteEndPoint = this.Endpoint; connectionState.ReceiveEventArgs.RemoteEndPoint = Endpoint;
connectionState.ReceiveEventArgs.DisconnectReuseSocket = true; connectionState.ReceiveEventArgs.DisconnectReuseSocket = true;
connectionState.SendEventArgs.DisconnectReuseSocket = true; connectionState.SendEventArgs.DisconnectReuseSocket = true;
} }
@ -332,20 +428,22 @@ namespace Integrations.Cod
connectionState.SendEventArgs.SetBuffer(payload); connectionState.SendEventArgs.SetBuffer(payload);
// send the data to the server // send the data to the server
bool sendDataPending = rconSocket.SendToAsync(connectionState.SendEventArgs); var sendDataPending = rconSocket.SendToAsync(connectionState.SendEventArgs);
if (sendDataPending) if (sendDataPending)
{ {
// the send has not been completed asynchronously // the send has not been completed asynchronously
// this really shouldn't ever happen because it's UDP // this really shouldn't ever happen because it's UDP
var complete = await connectionState.OnSentData.WaitAsync(StaticHelpers.SocketTimeout(4), token);
if(!await connectionState.OnSentData.WaitAsync(StaticHelpers.SocketTimeout(1))) if (!complete)
{ {
using(LogContext.PushProperty("Server", Endpoint.ToString())) using(LogContext.PushProperty("Server", Endpoint.ToString()))
{ {
_log.LogWarning("Socket timed out while sending RCon data on attempt {attempt}", _log.LogWarning("Socket timed out while sending RCon data on attempt {Attempt}",
connectionState.ConnectionAttempts); connectionState.ConnectionAttempts);
} }
rconSocket.Close(); rconSocket.Close();
throw new NetworkException("Timed out sending RCon data", rconSocket); throw new NetworkException("Timed out sending RCon data", rconSocket);
} }
@ -359,24 +457,36 @@ namespace Integrations.Cod
connectionState.ReceiveEventArgs.SetBuffer(connectionState.ReceiveBuffer); connectionState.ReceiveEventArgs.SetBuffer(connectionState.ReceiveBuffer);
// get our response back // get our response back
bool receiveDataPending = rconSocket.ReceiveFromAsync(connectionState.ReceiveEventArgs); var receiveDataPending = rconSocket.ReceiveFromAsync(connectionState.ReceiveEventArgs);
if (receiveDataPending) if (receiveDataPending)
{ {
_log.LogDebug("Waiting to asynchronously receive data on attempt #{connectionAttempts}", connectionState.ConnectionAttempts); _log.LogDebug("Waiting to asynchronously receive data on attempt #{ConnectionAttempts}", connectionState.ConnectionAttempts);
if (!await connectionState.OnReceivedData.WaitAsync(
new[] var completed = false;
{
StaticHelpers.SocketTimeout(connectionState.ConnectionAttempts), try
overrideTimeout {
}.Max())) completed = await connectionState.OnReceivedData.WaitAsync(
new[]
{
StaticHelpers.SocketTimeout(connectionState.ConnectionAttempts),
overrideTimeout
}.Max(), token);
}
catch (OperationCanceledException)
{
// ignored
}
if (!completed)
{ {
if (connectionState.ConnectionAttempts > 1) // this reduces some spam for unstable connections if (connectionState.ConnectionAttempts > 1) // this reduces some spam for unstable connections
{ {
using (LogContext.PushProperty("Server", Endpoint.ToString())) using (LogContext.PushProperty("Server", Endpoint.ToString()))
{ {
_log.LogWarning( _log.LogWarning(
"Socket timed out while waiting for RCon response on attempt {attempt} with timeout delay of {timeout}", "Socket timed out while waiting for RCon response on attempt {Attempt} with timeout delay of {Timeout}",
connectionState.ConnectionAttempts, connectionState.ConnectionAttempts,
StaticHelpers.SocketTimeout(connectionState.ConnectionAttempts)); StaticHelpers.SocketTimeout(connectionState.ConnectionAttempts));
} }
@ -388,16 +498,15 @@ namespace Integrations.Cod
} }
rconSocket.Close(); rconSocket.Close();
return GetResponseData(connectionState); return GetResponseData(connectionState);
} }
private byte[][] GetResponseData(ConnectionState connectionState) private static byte[][] GetResponseData(ConnectionState connectionState)
{ {
var responseList = new List<byte[]>(); var responseList = new List<byte[]>();
int totalBytesRead = 0; var totalBytesRead = 0;
foreach (int bytesRead in connectionState.BytesReadPerSegment) foreach (var bytesRead in connectionState.BytesReadPerSegment)
{ {
responseList.Add(connectionState.ReceiveBuffer responseList.Add(connectionState.ReceiveBuffer
.Skip(totalBytesRead) .Skip(totalBytesRead)
@ -412,36 +521,57 @@ namespace Integrations.Cod
private void OnDataReceived(object sender, SocketAsyncEventArgs e) private void OnDataReceived(object sender, SocketAsyncEventArgs e)
{ {
_log.LogDebug("Read {bytesTransferred} bytes from {endpoint}", e.BytesTransferred, e.RemoteEndPoint); _log.LogDebug("Read {BytesTransferred} bytes from {Endpoint}", e.BytesTransferred, e.RemoteEndPoint?.ToString());
// this occurs when we close the socket // this occurs when we close the socket
if (e.BytesTransferred == 0) if (e.BytesTransferred == 0)
{ {
_log.LogDebug("No bytes were transmitted so the connection was probably closed"); _log.LogDebug("No bytes were transmitted so the connection was probably closed");
var semaphore = ActiveQueries[this.Endpoint].OnReceivedData; var semaphore = ActiveQueries[Endpoint].OnReceivedData;
if (semaphore.CurrentCount == 0)
{
semaphore.Release();
}
return;
}
if (sender is not Socket sock) try
{
var semaphore = ActiveQueries[this.Endpoint].OnReceivedData;
if (semaphore.CurrentCount == 0)
{ {
semaphore.Release(); if (semaphore.CurrentCount == 0)
{
semaphore.Release();
}
} }
catch
{
// ignored because we can have the socket operation cancelled (which releases the semaphore) but
// this thread is not notified because it's an event
}
return; return;
} }
var state = ActiveQueries[this.Endpoint]; var state = ActiveQueries[Endpoint];
state.BytesReadPerSegment.Add(e.BytesTransferred); var cancellationRequested = ((ConnectionUserToken)e.UserToken)?.CancellationToken.IsCancellationRequested ??
false;
if (sender is not Socket sock || cancellationRequested)
{
var semaphore = ActiveQueries[Endpoint].OnReceivedData;
try
{
if (semaphore.CurrentCount == 0)
{
semaphore.Release();
}
}
catch
{
// ignored because we can have the socket operation cancelled (which releases the semaphore) but
// this thread is not notified because it's an event
}
return;
}
state.BytesReadPerSegment.Add(e.BytesTransferred);
// I don't even want to know why this works for getting more data from Cod4x // I don't even want to know why this works for getting more data from Cod4x
// but I'm leaving it in here as long as it doesn't break anything. // but I'm leaving it in here as long as it doesn't break anything.
// it's very stupid... // it's very stupid...
@ -450,32 +580,33 @@ namespace Integrations.Cod
try try
{ {
var totalBytesTransferred = e.BytesTransferred; var totalBytesTransferred = e.BytesTransferred;
_log.LogDebug("{total} total bytes transferred with {available} bytes remaining", totalBytesTransferred, _log.LogDebug("{Total} total bytes transferred with {Available} bytes remaining", totalBytesTransferred,
sock.Available); sock.Available);
// we still have available data so the payload was segmented // we still have available data so the payload was segmented
while (sock.Available > 0) while (sock.Available > 0)
{ {
_log.LogDebug("{available} more bytes to be read", sock.Available); _log.LogDebug("{Available} more bytes to be read", sock.Available);
var bufferSpaceAvailable = sock.Available + totalBytesTransferred - state.ReceiveBuffer.Length; var bufferSpaceAvailable = sock.Available + totalBytesTransferred - state.ReceiveBuffer.Length;
if (bufferSpaceAvailable >= 0) if (bufferSpaceAvailable >= 0)
{ {
_log.LogWarning( _log.LogWarning(
"Not enough buffer space to store incoming data {bytesNeeded} additional bytes required", "Not enough buffer space to store incoming data {BytesNeeded} additional bytes required",
bufferSpaceAvailable); bufferSpaceAvailable);
continue; continue;
} }
state.ReceiveEventArgs.SetBuffer(state.ReceiveBuffer, totalBytesTransferred, sock.Available); state.ReceiveEventArgs.SetBuffer(state.ReceiveBuffer, totalBytesTransferred, sock.Available);
if (sock.ReceiveAsync(state.ReceiveEventArgs)) if (sock.ReceiveAsync(state.ReceiveEventArgs))
{ {
_log.LogDebug("Remaining bytes are async"); _log.LogDebug("Remaining bytes are async");
continue; continue;
} }
_log.LogDebug("Read {bytesTransferred} synchronous bytes from {endpoint}", _log.LogDebug("Read {BytesTransferred} synchronous bytes from {Endpoint}",
state.ReceiveEventArgs.BytesTransferred, e.RemoteEndPoint); state.ReceiveEventArgs.BytesTransferred, e.RemoteEndPoint?.ToString());
// we need to increment this here because the callback isn't executed if there's no pending IO // we need to increment this here because the callback isn't executed if there's no pending IO
state.BytesReadPerSegment.Add(state.ReceiveEventArgs.BytesTransferred); state.BytesReadPerSegment.Add(state.ReceiveEventArgs.BytesTransferred);
totalBytesTransferred += state.ReceiveEventArgs.BytesTransferred; totalBytesTransferred += state.ReceiveEventArgs.BytesTransferred;
@ -489,22 +620,38 @@ namespace Integrations.Cod
finally finally
{ {
var semaphore = ActiveQueries[this.Endpoint].OnReceivedData; var semaphore = ActiveQueries[Endpoint].OnReceivedData;
if (semaphore.CurrentCount == 0) try
{ {
semaphore.Release(); if (semaphore.CurrentCount == 0)
{
semaphore.Release();
}
}
catch
{
// ignored because we can have the socket operation cancelled (which releases the semaphore) but
// this thread is not notified because it's an event
} }
} }
} }
private void OnDataSent(object sender, SocketAsyncEventArgs e) private void OnDataSent(object sender, SocketAsyncEventArgs e)
{ {
_log.LogDebug("Sent {byteCount} bytes to {endpoint}", e.Buffer?.Length, e.ConnectSocket?.RemoteEndPoint); _log.LogDebug("Sent {ByteCount} bytes to {Endpoint}", e.Buffer?.Length, e.ConnectSocket?.RemoteEndPoint?.ToString());
var semaphore = ActiveQueries[this.Endpoint].OnSentData; var semaphore = ActiveQueries[Endpoint].OnSentData;
if (semaphore.CurrentCount == 0) try
{ {
semaphore.Release(); if (semaphore.CurrentCount == 0)
{
semaphore.Release();
}
}
catch
{
// ignored because we can have the socket operation cancelled (which releases the semaphore) but
// this thread is not notified because it's an event
} }
} }
} }

View File

@ -24,9 +24,15 @@ namespace Integrations.Cod
public readonly SemaphoreSlim OnSentData = new(1, 1); public readonly SemaphoreSlim OnSentData = new(1, 1);
public readonly SemaphoreSlim OnReceivedData = new (1, 1); public readonly SemaphoreSlim OnReceivedData = new (1, 1);
public List<int> BytesReadPerSegment { get; set; } = new List<int>(); public List<int> BytesReadPerSegment { get; set; } = new();
public SocketAsyncEventArgs SendEventArgs { get; set; } = new SocketAsyncEventArgs(); public SocketAsyncEventArgs SendEventArgs { get; set; } = new();
public SocketAsyncEventArgs ReceiveEventArgs { get; set; } = new SocketAsyncEventArgs(); public SocketAsyncEventArgs ReceiveEventArgs { get; set; } = new();
public DateTime LastQuery { get; set; } = DateTime.Now; public DateTime LastQuery { get; set; } = DateTime.Now;
} }
internal class ConnectionUserToken
{
public Socket Socket { get; set; }
public CancellationToken CancellationToken { get; set; }
}
} }

View File

@ -48,12 +48,12 @@ namespace Integrations.Source
_activeQuery.Dispose(); _activeQuery.Dispose();
} }
public async Task<string[]> SendQueryAsync(StaticHelpers.QueryType type, string parameters = "") public async Task<string[]> SendQueryAsync(StaticHelpers.QueryType type, string parameters = "", CancellationToken token = default)
{ {
try try
{ {
await _activeQuery.WaitAsync(); await _activeQuery.WaitAsync(token);
await WaitForAvailable(); await WaitForAvailable(token);
if (_needNewSocket) if (_needNewSocket)
{ {
@ -66,7 +66,7 @@ namespace Integrations.Source
// ignored // ignored
} }
await Task.Delay(ConnectionTimeout); await Task.Delay(ConnectionTimeout, token);
_rconClient = _rconClientFactory.CreateClient(_ipEndPoint); _rconClient = _rconClientFactory.CreateClient(_ipEndPoint);
_authenticated = false; _authenticated = false;
_needNewSocket = false; _needNewSocket = false;
@ -147,12 +147,12 @@ namespace Integrations.Source
} }
} }
private async Task WaitForAvailable() private async Task WaitForAvailable(CancellationToken token)
{ {
var diff = DateTime.Now - _lastQuery; var diff = DateTime.Now - _lastQuery;
if (diff < FloodDelay) if (diff < FloodDelay)
{ {
await Task.Delay(FloodDelay - diff); await Task.Delay(FloodDelay - diff, token);
} }
} }

View File

@ -10,7 +10,7 @@
<ItemGroup> <ItemGroup>
<PackageReference Include="Microsoft.SyndicationFeed.ReaderWriter" Version="1.0.2" /> <PackageReference Include="Microsoft.SyndicationFeed.ReaderWriter" Version="1.0.2" />
<PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.1.28.1" PrivateAssets="All" /> <PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.3.23.1" PrivateAssets="All" />
</ItemGroup> </ItemGroup>
<Target Name="PostBuild" AfterTargets="PostBuildEvent"> <Target Name="PostBuild" AfterTargets="PostBuildEvent">

View File

@ -23,7 +23,7 @@
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.1.28.1" PrivateAssets="All" /> <PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.3.23.1" PrivateAssets="All" />
</ItemGroup> </ItemGroup>
<Target Name="PostBuild" AfterTargets="PostBuildEvent"> <Target Name="PostBuild" AfterTargets="PostBuildEvent">

View File

@ -41,430 +41,13 @@
<!-- images used by canvas --> <!-- images used by canvas -->
<img class="hide" id="hud_death" src="~/images/radar/death.png" /> <img class="hide" id="hud_death" src="~/images/radar/death.png" />
@section scripts { @section scripts {
<environment include="Development">
<script defer="defer"> <script type="text/javascript" src="~/js/liveradar.js" defer="defer"></script>
const textOffset = 15; </environment>
let previousRadarData = undefined;
let newRadarData = undefined; <script type="text/javascript">
const radarDataUrl = '@Url.Action("Data", "Radar", new { serverId = ViewBag.ActiveServerId })';
/************************ const mapDataUrl = '@Url.Action("Map", "Radar", new { serverId = ViewBag.ActiveServerId })';
* IW4 *
* **********************/
const weapons = {};
weapons["ak47"] = "ak47";
weapons["ak47classic"] = "icon_ak47_classic";
weapons["ak74u"] = "akd74u";
weapons["m16"] = "m16a4";
weapons["m4"] = "m4carbine";
weapons["fn2000"] = "fn2000";
weapons["masada"] = "masada";
weapons["famas"] = "famas";
weapons["fal"] = "fnfal";
weapons["scar"] = "scar_h";
weapons["tavor"] = "tavor";
weapons["mp5k"] = "mp5k";
weapons["uzi"] = "mini_uzi";
weapons["p90"] = "p90";
weapons["kriss"] = "kriss";
weapons["ump45"] = "ump45";
weapons["rpd"] = "rpd";
weapons["sa80"] = "sa80_lmg";
weapons["mg4"] = "mg4";
weapons["m240"] = "m240";
weapons["aug"] = "steyr";
weapons["barrett"] = "barrett50cal";
weapons["wa2000"] = "wa2000";
weapons["m21"] = "m14ebr";
weapons["cheytac"] = "cheytac";
weapons["dragunov"] = "dragunovsvd";
weapons["beretta"] = "m9beretta";
weapons["usp"] = "usp_45";
weapons["deserteagle"] = "desert_eagle";
weapons["deserteaglegold"] = "desert_eagle_gold";
weapons["desert"]
weapons["coltanaconda"] = "colt_anaconda";
weapons["tmp"] = "mp9";
weapons["glock"] = "glock";
weapons["beretta393"] = "beretta393";
weapons["pp2000"] = "pp2000";
weapons["ranger"] = "sawed_off";
weapons["model1887"] = "model1887";
weapons["striker"] = "striker";
weapons["aa12"] = "aa12";
weapons["m1014"] = "benelli_m4";
weapons["spas12"] = "spas12";
weapons["m79"] = "m79";
weapons["rpg"] = "rpg";
weapons["at4"] = "at4";
weapons["stinger"] = "stinger";
weapons["javelin"] = "javelin";
weapons["m40a3"] = "m40a3";
weapons["none"] = "neutral";
weapons["riotshield"] = "riot_shield";
weapons["peacekeeper"] = "peacekeeper";
function drawCircle(context, x, y, color) {
context.beginPath();
context.arc(x, y, 6 * stateInfo.imageScaler, 0, 2 * Math.PI, false);
context.fillStyle = color;
context.fill();
context.lineWidth = 0.5;
context.strokeStyle = 'rgba(255, 255, 255, 0.5)';
context.closePath();
context.stroke();
}
function drawLine(context, x1, y1, x2, y2, color) {
context.beginPath();
context.lineWidth = '3';
context.moveTo(x1, y1);
context.lineTo(x2, y2);
context.closePath();
context.stroke();
}
function drawTriangle(context, v1, v2, v3, color) {
context.beginPath();
context.moveTo(v1.x, v1.y);
context.lineTo(v2.x, v2.y);
context.lineTo(v3.x, v3.y);
context.closePath();
context.fillStyle = color;
context.fill();
}
function drawText(context, x, y, text, size, fillColor, strokeColor, alignment = 'left') {
context.beginPath();
context.save();
context.font = `bold ${Math.max(12, size * stateInfo.imageScaler)}px courier new`;
context.fillStyle = fillColor;
context.shadowColor = strokeColor;
context.shadowBlur = 4;
context.textAlign = alignment;
context.fillText(text, x, y);
context.restore();
context.closePath();
}
function drawImage(context, imgSelector, x, y, alpha = 1) {
context.save();
context.globalAlpha = alpha;
context.drawImage(document.getElementById(imgSelector), x - (15 * stateInfo.imageScaler), y - (15 * stateInfo.imageScaler), 32 * stateInfo.imageScaler, 32 * stateInfo.imageScaler);
context.globalAlpha = 1;
context.restore();
}
function checkCanvasSize(canvas, context, minimap, map) {
let width = Math.round(minimap.width());
if (Math.round(context.canvas.width) != width) {
canvas.width(width);
canvas.height(width);
context.canvas.height = width;
context.canvas.width = context.canvas.height;
}
stateInfo.imageScaler = (stateInfo.canvas.width() / 1024)
stateInfo.mapScalerX = (((stateInfo.mapInfo.right * stateInfo.imageScaler) - (stateInfo.mapInfo.left * stateInfo.imageScaler)) / stateInfo.mapInfo.width);
stateInfo.mapScalerY = (((stateInfo.mapInfo.bottom * stateInfo.imageScaler) - (stateInfo.mapInfo.top * stateInfo.imageScaler)) / stateInfo.mapInfo.height);
stateInfo.mapScaler = (stateInfo.mapScalerX + stateInfo.mapScalerY) / 2
stateInfo.forwardDistance = 500.0;
stateInfo.fovWidth = 40;
}
function calculateViewPosition(x, y, distance) {
let nx = Math.cos(x) * Math.cos(y);
let ny = Math.sin(x) * Math.cos(y);
let nz = Math.sin(360.0 - y);
return { x: (nx * distance) * stateInfo.mapScaler, y: (ny * distance) * stateInfo.mapScaler, z: (nz * distance) * stateInfo.mapScaler };
}
function lerp(start, end, complete) {
return (1 - complete) * start + complete * end;
}
function easeLerp(start, end, t) {
let t2 = (1 - Math.cos(t * Math.PI)) / 2;
return (start * (1-t2) + end * t2);
}
function fixRollAngles(oldAngles, newAngles) {
let newX = newAngles.x;
let newY = newAngles.y;
let angleDifferenceX = (oldAngles.x - newAngles.x);
if (angleDifferenceX > Math.PI) {
newX = oldAngles.x + (Math.PI * 2) - angleDifferenceX;
}
else if (Math.abs(newAngles.x - oldAngles.x) > Math.PI) {
newX = newAngles.x - (Math.PI * 2);
}
let angleDifferenceY = (oldAngles.y - newAngles.y);
if (angleDifferenceY > Math.PI) {
newY = oldAngles.y + (Math.PI * 2) - angleDifferenceY;
}
else if (Math.abs(newAngles.y - oldAngles.y) > Math.PI) {
newY = newAngles.y - (Math.PI * 2);
}
return { x: newX, y: newY };
}
function toRadians(deg) {
return deg * Math.PI / 180.0;
}
function rotate(cx, cy, x, y, angle) {
var radians = (Math.PI / 180) * angle,
cos = Math.cos(radians),
sin = Math.sin(radians),
nx = (cos * (x - cx)) + (sin * (y - cy)) + cx,
ny = (cos * (y - cy)) - (sin * (x - cx)) + cy;
return {
x: nx,
y: ny
};
}
function weaponImageForWeapon(weapon) {
let name = weapon.split('_')[0];
if (weapons[name] == undefined) {
console.log(name);
name = "none";
}
return `../images/radar/hud_weapons/hud_${weapons[name]}.png`;
}
function updatePlayerData() {
$('.player-data-left').html('');
$('.player-data-right').html('');
$.each(newRadarData, function (index, player) {
if (player == null) {
return;
}
let column = index % 2 == 0 ? $('.player-data-left') : $('.player-data-right');
column.append(`<div class="progress" style="height: 1.5rem; background-color: transparent;">
<div style="position: absolute; font-size: 1rem; left: 1.5rem;">${player.name}</div>
<div class="progress-bar bg-success" role="progressbar" style="min-width: 0px; width: ${player.health}%" aria-valuenow="${player.health}" aria-valuemin="0" aria-valuemax="100"></div>
<div class="progress-bar bg-danger" role="progressbar" style="min-width: 0px; border-right: 0px; width: ${100 - player.health}%" aria-valuenow="${100 - player.health}" aria-valuemin="0" aria-valuemax="100"></div>
</div>
<div class="d-flex flex-row flex-wrap p-2 mb-4 bg-dark border-bottom">
<div style="width: 3rem; height: 1.5rem; background-image:url(${weaponImageForWeapon(player.weapon)}); background-size: 3rem 1.5rem;" class="mr-auto text-left">
</div>
<div class="player-stat-icon" style="background-image:url('/images/radar/kills.png')"></div>
<div class="pr-2">${player.kills}</div>
<div class="player-stat-icon" style="background-image:url('/images/radar/death.png')"></div>
<div class="pr-3">${player.deaths}</div>
<span class="align-self-center oi oi-target pr-1"></span>
<div class="pr-3 ">${player.deaths == 0 ? player.kills.toFixed(2) : (player.kills / player.deaths).toFixed(2)}</div>
<span class="align-self-center oi oi-graph pr-1"></span>
<div>${ player.playTime == 0 ? '&mdash;' : Math.round(player.score / (player.playTime / 60))}</div>
</div>`);
});
$('.player-data-left').delay(1000).animate({opacity: 1}, 500);
$('.player-data-right').delay(1000).animate({opacity: 1}, 500);
}
const stateInfo = {
canvas: $('#map_canvas'),
ctx: $('#map_canvas')[0].getContext('2d'),
updateFrequency: 750,
updateFrameTimeDeviation: 0,
forwardDistance: undefined,
fovWidth: undefined,
mapInfo: undefined,
mapScaler: undefined,
deathIcons: {},
deathIconTime: 4000
};
function updateRadarData() {
$.getJSON('@Url.Action("Data", "Radar", new { serverId = ViewBag.ActiveServerId })', function (_radarItem) {
newRadarData = _radarItem;
});
$.getJSON('@Url.Action("Map", "Radar", new { serverId = ViewBag.ActiveServerId })', function (_map) {
stateInfo.mapInfo = _map
});
$.each(newRadarData, function (index, value) {
if (previousRadarData != undefined && index < previousRadarData.length) {
let previous = previousRadarData[index];
// this happens when the player has first joined and we haven't gotten two snapshots yet
if (value == null) {
return;
}
if (previous == null) {
previous = value;
}
// we don't want to treat a disconnected player snapshot as the previous
else if (previous.guid == value.guid) {
value.previous = previous;
}
// we haven't gotten a new item, it's just the old one again
if (previous.id === value.id) {
value.animationTime = previous.animationTime;
value.previous = value;
}
// they died between this snapshot and last so we wanna setup the death icon
if (!value.isAlive && previous.isAlive) {
stateInfo.deathIcons[value.guid] = {
animationTime: now,
location: value.location
};
}
// they respawned between this snapshot and last so we don't want to show wherever the were specating from
else if (value.isAlive && !previous.isAlive) {
value.previous = value;
}
}});
// we switch out the items to
previousRadarData = newRadarData;
$('#map_name').html(stateInfo.mapInfo.alias);
$('#map_list').css('background-image', `url(../images/radar/minimaps/compass_map_${stateInfo.mapInfo.name}@('@')2x.jpg)`);
checkCanvasSize(stateInfo.canvas, stateInfo.ctx, $('#map_list'), stateInfo.mapInfo);
updatePlayerData();
}
function updateMap() {
let ctx = stateInfo.ctx;
ctx.clearRect(0, 0, ctx.canvas.width, ctx.canvas.height);
now = performance.now();
$.each(previousRadarData, function (index, value) {
if (value == null) {
return;
}
if (value.previous == null) {
value.previous = value;
}
// this indicates we got a new snapshot to work with so we set the time based off the previous
// frame deviation to have minimal interpolation skipping
if (value.animationTime === undefined) {
value.animationTime = now - stateInfo.updateFrameTimeDeviation;
}
if (!value.isAlive) {
return;
}
const elapsedFrameTime = now - value.animationTime;
const completionPercent = elapsedFrameTime / stateInfo.updateFrequency;
// certain maps like estate have an off center axis of origin, so we need to account for that
let rotatedPreviousLocation = rotate(stateInfo.mapInfo.centerX, stateInfo.mapInfo.centerY, value.previous.location.x, value.previous.location.y, stateInfo.mapInfo.rotation);
let rotatedCurrentLocation = rotate(stateInfo.mapInfo.centerX, stateInfo.mapInfo.centerY, value.location.x, value.location.y, stateInfo.mapInfo.rotation);
const startX = ((stateInfo.mapInfo.maxLeft - rotatedPreviousLocation.y) * stateInfo.mapScaler) + (stateInfo.mapInfo.left * stateInfo.imageScaler);
const startY = ((stateInfo.mapInfo.maxTop - rotatedPreviousLocation.x) * stateInfo.mapScalerY) + (stateInfo.mapInfo.top * stateInfo.imageScaler);
const endX = ((stateInfo.mapInfo.maxLeft - rotatedCurrentLocation.y) * stateInfo.mapScaler) + (stateInfo.mapInfo.left * stateInfo.imageScaler);
const endY = ((stateInfo.mapInfo.maxTop - rotatedCurrentLocation.x) * stateInfo.mapScalerY) + (stateInfo.mapInfo.top * stateInfo.imageScaler);
let teamColor = value.team == 'allies' ? 'rgb(0, 122, 204, 1)' : 'rgb(255, 69, 69)';
let fovColor = value.team == 'allies' ? 'rgba(0, 122, 204, 0.2)' : 'rgba(255, 69, 69, 0.2)';
// this takes care of moving past the roll-over point of yaw/pitch (ie 360->0)
const rollAngleFix = fixRollAngles(value.previous.radianAngles, value.radianAngles);
const radianLerpX = lerp(value.previous.radianAngles.x, rollAngleFix.x, completionPercent);
const radianLerpY = lerp(value.previous.radianAngles.y, rollAngleFix.y, completionPercent);
// this is some jankiness to get the fov to point the right direction
let firstVertex = calculateViewPosition(toRadians(stateInfo.mapInfo.rotation + stateInfo.mapInfo.viewPositionRotation - 90) - radianLerpX + toRadians(stateInfo.fovWidth), radianLerpY, stateInfo.forwardDistance);
let secondVertex = calculateViewPosition(toRadians(stateInfo.mapInfo.rotation + stateInfo.mapInfo.viewPositionRotation - 90) - radianLerpX - toRadians(stateInfo.fovWidth), radianLerpY, stateInfo.forwardDistance);
let currentX = lerp(startX, endX, completionPercent);
let currentY = lerp(startY, endY, completionPercent);
// we need to calculate the distance from the center of the map so we can scale if necessary
let centerX = ((stateInfo.mapInfo.maxLeft - stateInfo.mapInfo.centerY) * stateInfo.mapScaler) + (stateInfo.mapInfo.left * stateInfo.imageScaler);
let centerY = ((stateInfo.mapInfo.maxTop - stateInfo.mapInfo.centerX) * stateInfo.mapScaler) + (stateInfo.mapInfo.top * stateInfo.imageScaler);
// reuse lerp to scale the pixel to map ratio
currentX = lerp(centerX, currentX, stateInfo.mapInfo.scaler);
currentY = lerp(centerY, currentY, stateInfo.mapInfo.scaler);
drawCircle(ctx, currentX, currentY, teamColor);
drawTriangle(ctx,
{ x: currentX, y: currentY },
{ x: currentX + firstVertex.x, y: currentY + firstVertex.y },
{ x: currentX + secondVertex.x, y: currentY + secondVertex.y },
fovColor);
drawText(ctx, currentX, currentY - (textOffset * stateInfo.imageScaler), value.name, 16, 'white', teamColor, 'center')
});
const completedIcons = [];
for (let key in stateInfo.deathIcons) {
const icon = stateInfo.deathIcons[key];
const x = ((stateInfo.mapInfo.maxLeft - icon.location.y) * stateInfo.mapScaler) + (stateInfo.mapInfo.left * stateInfo.imageScaler);
const y = ((stateInfo.mapInfo.maxTop - icon.location.x) * stateInfo.mapScaler) + (stateInfo.mapInfo.top * stateInfo.imageScaler);
const elapsedFrameTime = now - icon.animationTime;
const completionPercent = elapsedFrameTime / stateInfo.deathIconTime;
const opacity = easeLerp(1, 0, completionPercent);
drawImage(stateInfo.ctx, 'hud_death', x, y, opacity);
if (completionPercent >= 1) {
completedIcons.push(key);
}
}
for (let i = 0; i < completedIcons.length; i++) {
delete stateInfo.deathIcons[completedIcons[i]];
}
window.requestAnimationFrame(updateMap);
}
$(document).ready(function () {
$.getJSON('@Url.Action("Map", "Radar", new { serverId = ViewBag.ActiveServerId })', function (_map) {
stateInfo.mapInfo = _map;
updateRadarData();
setInterval(updateRadarData, stateInfo.updateFrequency);
window.requestAnimationFrame(updateMap);
});
})
</script> </script>
} }

View File

@ -19,7 +19,7 @@
</PropertyGroup> </PropertyGroup>
<ItemGroup> <ItemGroup>
<PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.1.28.1" PrivateAssets="All" /> <PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.3.23.1" PrivateAssets="All" />
</ItemGroup> </ItemGroup>
<Target Name="PostBuild" AfterTargets="PostBuildEvent"> <Target Name="PostBuild" AfterTargets="PostBuildEvent">

View File

@ -16,7 +16,7 @@
</PropertyGroup> </PropertyGroup>
<ItemGroup> <ItemGroup>
<PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.1.28.1" PrivateAssets="All" /> <PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.3.23.1" PrivateAssets="All" />
</ItemGroup> </ItemGroup>
<Target Name="PostBuild" AfterTargets="PostBuildEvent"> <Target Name="PostBuild" AfterTargets="PostBuildEvent">

View File

@ -1,13 +1,12 @@
let plugin = { let plugin = {
author: 'RaidMax', author: 'RaidMax',
version: 1.0, version: 1.1,
name: 'Action on Report', name: 'Action on Report',
enabled: false, // indicates if the plugin is enabled enabled: false, // indicates if the plugin is enabled
reportAction: 'TempBan', // can be TempBan or Ban reportAction: 'TempBan', // can be TempBan or Ban
maxReportCount: 5, // how many reports before action is taken maxReportCount: 5, // how many reports before action is taken
tempBanDurationMinutes: 60, // how long to temporarily ban the player tempBanDurationMinutes: 60, // how long to temporarily ban the player
eventTypes: { 'report': 103 }, eventTypes: { 'report': 103 },
permissionTypes: { 'trusted': 2 },
onEventAsync: function (gameEvent, server) { onEventAsync: function (gameEvent, server) {
if (!this.enabled) { if (!this.enabled) {
@ -15,7 +14,7 @@ let plugin = {
} }
if (gameEvent.Type === this.eventTypes['report']) { if (gameEvent.Type === this.eventTypes['report']) {
if (!gameEvent.Target.IsIngame || gameEvent.Target.Level >= this.permissionTypes['trusted']) { if (!gameEvent.Target.IsIngame || (gameEvent.Target.Level !== 'User' && gameEvent.Target.Level !== 'Flagged')) {
server.Logger.WriteInfo(`Ignoring report for client (id) ${gameEvent.Target.ClientId} because they are privileged or not ingame`); server.Logger.WriteInfo(`Ignoring report for client (id) ${gameEvent.Target.ClientId} because they are privileged or not ingame`);
return; return;
} }

View File

@ -1,28 +1,27 @@
const eventTypes = { const servers = {};
1: 'start', // a server started being monitored
6: 'disconnect', // a client detected a leaving the game
9: 'preconnect', // client detected as joining via log or status
101: 'warn' // client was warned
};
const servers = {};
const inDvar = 'sv_iw4madmin_in'; const inDvar = 'sv_iw4madmin_in';
const outDvar = 'sv_iw4madmin_out'; const outDvar = 'sv_iw4madmin_out';
const pollRate = 750; const pollRate = 900;
const enableCheckTimeout = 10000;
let logger = {}; let logger = {};
const maxQueuedMessages = 25;
let plugin = { let plugin = {
author: 'RaidMax', author: 'RaidMax',
version: 1.0, version: 1.1,
name: 'Game Interface', name: 'Game Interface',
onEventAsync: (gameEvent, server) => { onEventAsync: (gameEvent, server) => {
const eventType = eventTypes[gameEvent.Type];
if (servers[server.EndPoint] != null && !servers[server.EndPoint].enabled) { if (servers[server.EndPoint] != null && !servers[server.EndPoint].enabled) {
return; return;
} }
const eventType = String(gameEvent.TypeName).toLowerCase();
if (eventType === undefined) {
return;
}
switch (eventType) { switch (eventType) {
case 'start': case 'start':
const enabled = initialize(server); const enabled = initialize(server);
@ -33,7 +32,7 @@ let plugin = {
break; break;
case 'preconnect': case 'preconnect':
// when the plugin is reloaded after the servers are started // when the plugin is reloaded after the servers are started
if (servers[server.EndPoint] == null) { if (servers[server.EndPoint] === undefined || servers[server.EndPoint] == null) {
const enabled = initialize(server); const enabled = initialize(server);
if (!enabled) { if (!enabled) {
@ -73,118 +72,99 @@ let plugin = {
}; };
let commands = [{ let commands = [{
// required
name: 'giveweapon', name: 'giveweapon',
// required
description: 'gives specified weapon', description: 'gives specified weapon',
// required
alias: 'gw', alias: 'gw',
// required
permission: 'SeniorAdmin', permission: 'SeniorAdmin',
// optional (defaults to false) targetRequired: true,
targetRequired: false,
// optional
arguments: [{ arguments: [{
name: 'weapon name', name: 'player',
required: true required: true
}], },
{
name: 'weapon name',
required: true
}],
supportedGames: ['IW4'], supportedGames: ['IW4'],
// required
execute: (gameEvent) => { execute: (gameEvent) => {
sendScriptCommand(gameEvent.Owner, 'GiveWeapon', gameEvent.Origin, {weaponName: gameEvent.Data}); if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
return;
}
sendScriptCommand(gameEvent.Owner, 'GiveWeapon', gameEvent.Origin, gameEvent.Target, {weaponName: gameEvent.Data});
} }
}, },
{ {
// required
name: 'takeweapons', name: 'takeweapons',
// required description: 'take all weapons from specified player',
description: 'take all weapons from specifies player',
// required
alias: 'tw', alias: 'tw',
// required
permission: 'SeniorAdmin', permission: 'SeniorAdmin',
// optional (defaults to false)
targetRequired: true, targetRequired: true,
// optional
arguments: [],
supportedGames: ['IW4'],
// required
execute: (gameEvent) => {
sendScriptCommand(gameEvent.Owner, 'TakeWeapons', gameEvent.Target, undefined);
}
},
{
// required
name: 'switchteam',
// required
description: 'switches specified player to the opposite team',
// required
alias: 'st',
// required
permission: 'Administrator',
// optional (defaults to false)
targetRequired: true,
// optional
arguments: [{ arguments: [{
name: 'player', name: 'player',
required: true required: true
}], }],
supportedGames: ['IW4'], supportedGames: ['IW4'],
// required
execute: (gameEvent) => { execute: (gameEvent) => {
sendScriptCommand(gameEvent.Owner, 'SwitchTeams', gameEvent.Target, undefined); if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
return;
}
sendScriptCommand(gameEvent.Owner, 'TakeWeapons', gameEvent.Origin, gameEvent.Target, undefined);
} }
}, },
{ {
// required name: 'switchteam',
name: 'hide', description: 'switches specified player to the opposite team',
// required alias: 'st',
description: 'hide yourself', permission: 'Administrator',
// required targetRequired: true,
alias: 'hi', arguments: [{
// required name: 'player',
permission: 'SeniorAdmin', required: true
// optional (defaults to false) }],
targetRequired: false, supportedGames: ['IW4'],
// optional execute: (gameEvent) => {
arguments: [], if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
supportedGames: ['IW4'], return;
// required }
execute: (gameEvent) => { sendScriptCommand(gameEvent.Owner, 'SwitchTeams', gameEvent.Origin, gameEvent.Target, undefined);
sendScriptCommand(gameEvent.Owner, 'Hide', gameEvent.Origin, undefined); }
} },
}, {
{ name: 'hide',
// required description: 'hide yourself ingame',
name: 'unhide', alias: 'hi',
// required permission: 'SeniorAdmin',
description: 'unhide yourself', targetRequired: false,
// required arguments: [],
alias: 'unh', supportedGames: ['IW4'],
// required execute: (gameEvent) => {
permission: 'SeniorAdmin', if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
// optional (defaults to false) return;
targetRequired: false, }
// optional sendScriptCommand(gameEvent.Owner, 'Hide', gameEvent.Origin, gameEvent.Origin, undefined);
arguments: [], }
supportedGames: ['IW4'], },
// required {
execute: (gameEvent) => { name: 'unhide',
sendScriptCommand(gameEvent.Owner, 'Unhide', gameEvent.Origin, undefined); description: 'unhide yourself ingame',
} alias: 'unh',
}, permission: 'SeniorAdmin',
{ targetRequired: false,
// required arguments: [],
name: 'alert', supportedGames: ['IW4'],
// required execute: (gameEvent) => {
description: 'alert a player', if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
// required return;
alias: 'alr', }
// required sendScriptCommand(gameEvent.Owner, 'Unhide', gameEvent.Origin, gameEvent.Origin, undefined);
permission: 'SeniorAdmin', }
// optional (defaults to false) },
{
name: 'alert',
description: 'alert a player',
alias: 'alr',
permission: 'SeniorAdmin',
targetRequired: true, targetRequired: true,
// optional
arguments: [{ arguments: [{
name: 'player', name: 'player',
required: true required: true
@ -194,80 +174,156 @@ let commands = [{
required: true required: true
}], }],
supportedGames: ['IW4'], supportedGames: ['IW4'],
// required
execute: (gameEvent) => { execute: (gameEvent) => {
sendScriptCommand(gameEvent.Owner, 'Alert', gameEvent.Target, { if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
return;
}
sendScriptCommand(gameEvent.Owner, 'Alert', gameEvent.Origin, gameEvent.Target, {
alertType: 'Alert', alertType: 'Alert',
message: gameEvent.Data message: gameEvent.Data
}); });
} }
},
{
name: 'gotoplayer',
description: 'teleport to a player',
alias: 'g2p',
permission: 'SeniorAdmin',
targetRequired: true,
arguments: [{
name: 'player',
required: true
}],
supportedGames: ['IW4'],
execute: (gameEvent) => {
if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
return;
}
sendScriptCommand(gameEvent.Owner, 'Goto', gameEvent.Origin, gameEvent.Target, undefined);
}
},
{
name: 'goto',
description: 'teleport to a position',
alias: 'g2',
permission: 'SeniorAdmin',
targetRequired: false,
arguments: [{
name: 'x',
required: true
},
{
name: 'y',
required: true
},
{
name: 'z',
required: true
}],
supportedGames: ['IW4'],
execute: (gameEvent) => {
if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
return;
}
const args = String(gameEvent.Data).split(' ');
sendScriptCommand(gameEvent.Owner, 'Goto', gameEvent.Origin, gameEvent.Target, {
x: args[0],
y: args[1],
z: args[2]
});
}
},
{
name: 'kill',
description: 'kill a player',
alias: 'kpl',
permission: 'SeniorAdmin',
targetRequired: true,
arguments: [{
name: 'player',
required: true
}],
supportedGames: ['IW4'],
execute: (gameEvent) => {
if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
return;
}
sendScriptCommand(gameEvent.Owner, 'Kill', gameEvent.Origin, gameEvent.Target, undefined);
}
},
{
name: 'nightmode',
description: 'sets server into nightmode',
alias: 'nitem',
permission: 'SeniorAdmin',
targetRequired: false,
arguments: [],
supportedGames: ['IW4'],
execute: (gameEvent) => {
if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
return;
}
sendScriptCommand(gameEvent.Owner, 'NightMode', gameEvent.Origin, undefined, undefined);
}
},
{
name: 'setspectator',
description: 'sets a player as spectator',
alias: 'spec',
permission: 'Administrator',
targetRequired: true,
arguments: [{
name: 'player',
required: true
}],
supportedGames: ['IW4'],
execute: (gameEvent) => {
if (!validateEnabled(gameEvent.Owner, gameEvent.Origin)) {
return;
}
sendScriptCommand(gameEvent.Owner, 'SetSpectator', gameEvent.Origin, gameEvent.Target, undefined);
}
}]; }];
const sendScriptCommand = (server, command, target, data) => { const sendScriptCommand = (server, command, origin, target, data) => {
const state = servers[server.EndPoint]; const state = servers[server.EndPoint];
if (state === undefined || !state.enabled) { if (state === undefined || !state.enabled) {
return; return;
} }
sendEvent(server, false, 'ExecuteCommandRequested', command, target, data); sendEvent(server, false, 'ExecuteCommandRequested', command, origin, target, data);
} }
const sendEvent = (server, responseExpected, event, subtype, client, data) => { const sendEvent = (server, responseExpected, event, subtype, origin, target, data) => {
const logger = _serviceResolver.ResolveService('ILogger'); const logger = _serviceResolver.ResolveService('ILogger');
const state = servers[server.EndPoint];
let pendingOut = true; if (state.queuedMessages.length >= maxQueuedMessages) {
let pendingCheckCount = 0; logger.WriteWarning('Too many queued messages so we are skipping');
const start = new Date(); return;
while (pendingOut && pendingCheckCount <= 30) {
try {
const out = server.GetServerDvar(outDvar);
pendingOut = !(out == null || out === '' || out === 'null');
} catch (error) {
logger.WriteError(`Could not check server output dvar for IO status ${error}`);
}
if (pendingOut) {
logger.WriteDebug('Waiting for event bus to be cleared');
System.Threading.Tasks.Task.Delay(1000).Wait();
}
pendingCheckCount++;
} }
if (pendingOut) { let targetClientNumber = -1;
logger.WriteWarning(`Reached maximum attempts waiting for output to be available for ${server.EndPoint}`) if (target != null) {
targetClientNumber = target.ClientNumber;
} }
const output = `${responseExpected ? '1' : '0'};${event};${subtype};${client.ClientNumber};${buildDataString(data)}`; const output = `${responseExpected ? '1' : '0'};${event};${subtype};${origin.ClientNumber};${targetClientNumber};${buildDataString(data)}`;
logger.WriteDebug(`Sending output to server ${output}`); logger.WriteDebug(`Queuing output for server ${output}`);
try { state.queuedMessages.push(output);
server.SetServerDvar(outDvar, output);
logger.WriteDebug(`SendEvent took ${(new Date() - start) / 1000}ms`);
} catch (error) {
logger.WriteError(`Could not set server output dvar ${error}`);
}
}; };
const parseEvent = (input) => {
if (input === undefined) {
return {};
}
const eventInfo = input.split(';');
return {
eventType: eventInfo[1],
subType: eventInfo[2],
clientNumber: eventInfo[3],
data: eventInfo.length > 4 ? parseDataString(eventInfo[4]) : undefined
}
}
const initialize = (server) => { const initialize = (server) => {
const logger = _serviceResolver.ResolveService('ILogger'); const logger = _serviceResolver.ResolveService('ILogger');
servers[server.EndPoint] = {
enabled: false
}
let enabled = false; let enabled = false;
try { try {
enabled = server.GetServerDvar('sv_iw4madmin_integration_enabled') === '1'; enabled = server.GetServerDvar('sv_iw4madmin_integration_enabled', enableCheckTimeout) === '1';
} catch (error) { } catch (error) {
logger.WriteError(`Could not get integration status of ${server.EndPoint} - ${error}`); logger.WriteError(`Could not get integration status of ${server.EndPoint} - ${error}`);
} }
@ -275,9 +331,6 @@ const initialize = (server) => {
logger.WriteInfo(`GSC Integration enabled = ${enabled}`); logger.WriteInfo(`GSC Integration enabled = ${enabled}`);
if (!enabled) { if (!enabled) {
servers[server.EndPoint] = {
enabled: false
}
return false; return false;
} }
@ -288,33 +341,38 @@ const initialize = (server) => {
// necessary to prevent multi-threaded access to the JS context // necessary to prevent multi-threaded access to the JS context
timer.SetDependency(_lock); timer.SetDependency(_lock);
servers[server.EndPoint] = { servers[server.EndPoint].timer = timer;
timer: timer, servers[server.EndPoint].enabled = true;
enabled: true servers[server.EndPoint].waitingOnInput = false;
} servers[server.EndPoint].waitingOnOutput = false;
servers[server.EndPoint].queuedMessages = [];
try { setDvar(server, inDvar, '', onSetDvar);
server.SetServerDvar(inDvar, ''); setDvar(server, outDvar, '', onSetDvar);
server.SetServerDvar(outDvar, '');
} catch (error) {
logger.WriteError(`Could set default values bus dvars for ${server.EndPoint} - ${error}`);
}
return true; return true;
}; }
const pollForEvents = server => { function onReceivedDvar(server, dvarName, dvarValue, success) {
const logger = _serviceResolver.ResolveService('ILogger'); const logger = _serviceResolver.ResolveService('ILogger');
logger.WriteDebug(`Received ${dvarName}=${dvarValue} success=${success}`);
let input; let input = dvarValue;
try { const state = servers[server.EndPoint];
input = server.GetServerDvar(inDvar);
} catch (error) { if (state.waitingOnOutput && dvarName === outDvar && isEmpty(dvarValue)) {
logger.WriteError(`Could not get input bus value for ${server.EndPoint} - ${error}`); logger.WriteDebug('Setting out bus to read to send');
return; // reset our flag letting use the out bus is open
state.waitingOnOutput = !success;
} }
if (input === undefined || input === null || input === 'null') { if (state.waitingOnInput && dvarName === inDvar) {
logger.WriteDebug('Setting in bus to ready to receive');
// we've received the data so now we can mark it as ready for more
state.waitingOnInput = false;
}
if (isEmpty(input)) {
input = ''; input = '';
} }
@ -334,7 +392,7 @@ const pollForEvents = server => {
if (event.subType === 'Meta') { if (event.subType === 'Meta') {
const metaService = _serviceResolver.ResolveService('IMetaService'); const metaService = _serviceResolver.ResolveService('IMetaService');
const meta = metaService.GetPersistentMeta(event.data, client).GetAwaiter().GetResult().Value; const meta = metaService.GetPersistentMeta(event.data, client).GetAwaiter().GetResult();
data[event.data] = meta === null ? '' : meta.Value; data[event.data] = meta === null ? '' : meta.Value;
} else { } else {
data = { data = {
@ -344,18 +402,18 @@ const pollForEvents = server => {
}; };
} }
sendEvent(server, false, 'ClientDataReceived', event.subType, client, data); sendEvent(server, false, 'ClientDataReceived', event.subType, client, undefined, data);
} else { } else {
logger.WriteWarning(`Could not find client slot ${event.clientNumber} when processing ${event.eventType}`); logger.WriteWarning(`Could not find client slot ${event.clientNumber} when processing ${event.eventType}`);
sendEvent(server, false, 'ClientDataReceived', 'Fail', {ClientNumber: event.clientNumber}, undefined); sendEvent(server, false, 'ClientDataReceived', 'Fail', event.clientNumber, undefined, {ClientNumber: event.clientNumber});
} }
} }
if (event.eventType === 'SetClientDataRequested') { if (event.eventType === 'SetClientDataRequested') {
let client = server.GetClientByNumber(event.clientNumber); let client = server.GetClientByNumber(event.clientNumber);
let clientId; let clientId;
if (client != null){ if (client != null) {
clientId = client.ClientId; clientId = client.ClientId;
} else { } else {
clientId = parseInt(event.data.clientId); clientId = parseInt(event.data.clientId);
@ -365,7 +423,7 @@ const pollForEvents = server => {
if (clientId == null) { if (clientId == null) {
logger.WriteWarning(`Could not find client slot ${event.clientNumber} when processing ${event.eventType}`); logger.WriteWarning(`Could not find client slot ${event.clientNumber} when processing ${event.eventType}`);
sendEvent(server, false, 'SetClientDataCompleted', 'Meta', {ClientNumber: event.clientNumber}, {status: 'Fail'}); sendEvent(server, false, 'SetClientDataCompleted', 'Meta', {ClientNumber: event.clientNumber}, undefined, {status: 'Fail'});
} else { } else {
if (event.subType === 'Meta') { if (event.subType === 'Meta') {
const metaService = _serviceResolver.ResolveService('IMetaService'); const metaService = _serviceResolver.ResolveService('IMetaService');
@ -378,25 +436,80 @@ const pollForEvents = server => {
} else { } else {
metaService.SetPersistentMeta(event.data['key'], event.data['value'], clientId).GetAwaiter().GetResult(); metaService.SetPersistentMeta(event.data['key'], event.data['value'], clientId).GetAwaiter().GetResult();
} }
sendEvent(server, false, 'SetClientDataCompleted', 'Meta', {ClientNumber: event.clientNumber}, {status: 'Complete'}); sendEvent(server, false, 'SetClientDataCompleted', 'Meta', {ClientNumber: event.clientNumber}, undefined, {status: 'Complete'});
} catch (error) { } catch (error) {
sendEvent(server, false, 'SetClientDataCompleted', 'Meta', {ClientNumber: event.clientNumber}, {status: 'Fail'}); sendEvent(server, false, 'SetClientDataCompleted', 'Meta', {ClientNumber: event.clientNumber}, undefined, {status: 'Fail'});
} }
} }
} }
} }
try { setDvar(server, inDvar, '', onSetDvar);
server.SetServerDvar(inDvar, ''); } else if (server.ClientNum === 0) {
} catch (error) {
logger.WriteError(`Could not reset in bus value for ${server.EndPoint} - ${error}`);
}
}
else if (server.ClientNum === 0) {
servers[server.EndPoint].timer.Stop(); servers[server.EndPoint].timer.Stop();
} }
} }
function onSetDvar(server, dvarName, dvarValue, success) {
const logger = _serviceResolver.ResolveService('ILogger');
logger.WriteDebug(`Completed set of dvar ${dvarName}=${dvarValue}, success=${success}`);
const state = servers[server.EndPoint];
if (dvarName === inDvar && success && isEmpty(dvarValue)) {
logger.WriteDebug('In bus is ready for new data');
// reset our flag letting use the in bus is ready for more data
state.waitingOnInput = false;
}
}
const pollForEvents = server => {
const state = servers[server.EndPoint];
if (state === null || !state.enabled) {
return;
}
if (server.Throttled) {
return;
}
if (!state.waitingOnInput) {
state.waitingOnInput = true;
getDvar(server, inDvar, onReceivedDvar);
}
if (!state.waitingOnOutput) {
if (state.queuedMessages.length === 0) {
logger.WriteDebug('No messages in queue');
return;``
}
state.waitingOnOutput = true;
const nextMessage = state.queuedMessages.splice(0, 1);
setDvar(server, outDvar, nextMessage, onSetDvar);
}
if (state.waitingOnOutput) {
getDvar(server, outDvar, onReceivedDvar);
}
}
const parseEvent = (input) => {
if (input === undefined) {
return {};
}
const eventInfo = input.split(';');
return {
eventType: eventInfo[1],
subType: eventInfo[2],
clientNumber: eventInfo[3],
data: eventInfo.length > 4 ? parseDataString(eventInfo[4]) : undefined
}
}
const buildDataString = data => { const buildDataString = data => {
if (data === undefined) { if (data === undefined) {
return ''; return '';
@ -428,3 +541,15 @@ const parseDataString = data => {
return dict.length === 0 ? data : dict; return dict.length === 0 ? data : dict;
} }
const validateEnabled = (server, origin) => {
const enabled = servers[server.EndPoint] != null && servers[server.EndPoint].enabled;
if (!enabled) {
origin.Tell('Game interface is not enabled on this server');
}
return enabled;
}
function isEmpty(value) {
return value == null || false || value === '' || value === 'null';
}

View File

@ -1,14 +1,31 @@
var plugin = { const commands = [{
author: 'RaidMax', name: "whitelistvpn",
version: 1.2, description: "whitelists a player's client id from VPN detection",
name: 'VPN Detection Plugin', alias: "wv",
permission: "SeniorAdmin",
targetRequired: true,
arguments: [{
name: "players",
required: true
}],
execute: (gameEvent) => {
plugin.vpnExceptionIds.push(gameEvent.Target.ClientId);
plugin.configHandler.SetValue('vpnExceptionIds', plugin.vpnExceptionIds);
gameEvent.Origin.Tell(`Successfully whitelisted ${gameEvent.Target.Name}`);
}
}];
const plugin = {
author: 'RaidMax',
version: 1.3,
name: 'VPN Detection Plugin',
manager: null, manager: null,
logger: null, logger: null,
vpnExceptionIds: [], vpnExceptionIds: [],
checkForVpn: function (origin) { checkForVpn: function (origin) {
var exempt = false; let exempt = false;
// prevent players that are exempt from being kicked // prevent players that are exempt from being kicked
this.vpnExceptionIds.forEach(function (id) { this.vpnExceptionIds.forEach(function (id) {
if (id === origin.ClientId) { if (id === origin.ClientId) {
@ -18,40 +35,41 @@ var plugin = {
}); });
if (exempt) { if (exempt) {
this.logger.WriteInfo(`${origin} is whitelisted, so we are not checking VPN status`);
return; return;
} }
var usingVPN = false; let usingVPN = false;
try { try {
var cl = new System.Net.Http.HttpClient(); const cl = new System.Net.Http.HttpClient();
var re = cl.GetAsync('https://api.xdefcon.com/proxy/check/?ip=' + origin.IPAddressString).Result; const re = cl.GetAsync(`https://api.xdefcon.com/proxy/check/?ip=${origin.IPAddressString}`).Result;
var userAgent = 'IW4MAdmin-' + this.manager.GetApplicationSettings().Configuration().Id; const userAgent = `IW4MAdmin-${this.manager.GetApplicationSettings().Configuration().Id}`;
cl.DefaultRequestHeaders.Add('User-Agent', userAgent); cl.DefaultRequestHeaders.Add('User-Agent', userAgent);
var co = re.Content; const co = re.Content;
var parsedJSON = JSON.parse(co.ReadAsStringAsync().Result); const parsedJSON = JSON.parse(co.ReadAsStringAsync().Result);
co.Dispose(); co.Dispose();
re.Dispose(); re.Dispose();
cl.Dispose(); cl.Dispose();
usingVPN = parsedJSON.success && parsedJSON.proxy; usingVPN = parsedJSON.success && parsedJSON.proxy;
} catch (e) { } catch (e) {
this.logger.WriteWarning('There was a problem checking client IP for VPN ' + e.message); this.logger.WriteWarning(`There was a problem checking client IP for VPN ${e.message}`);
} }
if (usingVPN) { if (usingVPN) {
this.logger.WriteInfo(origin + ' is using a VPN (' + origin.IPAddressString + ')'); this.logger.WriteInfo(origin + ' is using a VPN (' + origin.IPAddressString + ')');
var contactUrl = this.manager.GetApplicationSettings().Configuration().ContactUri; const contactUrl = this.manager.GetApplicationSettings().Configuration().ContactUri;
var additionalInfo = ''; let additionalInfo = '';
if (contactUrl) { if (contactUrl) {
additionalInfo = _localization.LocalizationIndex["SERVER_KICK_VPNS_NOTALLOWED_INFO"] + ' ' + contactUrl; additionalInfo = _localization.LocalizationIndex['SERVER_KICK_VPNS_NOTALLOWED_INFO'] + ' ' + contactUrl;
} }
origin.Kick(_localization.LocalizationIndex["SERVER_KICK_VPNS_NOTALLOWED"] + ' ' + additionalInfo, _IW4MAdminClient); origin.Kick(_localization.LocalizationIndex['SERVER_KICK_VPNS_NOTALLOWED'] + ' ' + additionalInfo, _IW4MAdminClient);
} }
}, },
onEventAsync: function (gameEvent, server) { onEventAsync: function (gameEvent, server) {
// join event // join event
if (gameEvent.Type === 4) { if (gameEvent.TypeName === 'Join') {
this.checkForVpn(gameEvent.Origin); this.checkForVpn(gameEvent.Origin);
} }
}, },
@ -59,6 +77,10 @@ var plugin = {
onLoadAsync: function (manager) { onLoadAsync: function (manager) {
this.manager = manager; this.manager = manager;
this.logger = manager.GetLogger(0); this.logger = manager.GetLogger(0);
this.configHandler = _configHandler;
this.configHandler.GetValue('vpnExceptionIds').forEach(element => this.vpnExceptionIds.push(element));
this.logger.WriteInfo(`Loaded ${this.vpnExceptionIds.length} ids into whitelist`);
}, },
onUnloadAsync: function () { onUnloadAsync: function () {
@ -66,4 +88,4 @@ var plugin = {
onTickAsync: function (server) { onTickAsync: function (server) {
} }
}; };

View File

@ -29,23 +29,21 @@ namespace IW4MAdmin.Plugins.Stats.Commands
_contextFactory = contextFactory; _contextFactory = contextFactory;
} }
public override async Task ExecuteAsync(GameEvent E) public override async Task ExecuteAsync(GameEvent gameEvent)
{ {
var mostKills = await GetMostKills(StatManager.GetIdForServer(E.Owner), Plugin.Config.Configuration(), var mostKills = await GetMostKills(StatManager.GetIdForServer(gameEvent.Owner), Plugin.Config.Configuration(),
_contextFactory, _translationLookup); _contextFactory, _translationLookup);
if (!E.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix)) if (!gameEvent.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix))
{ {
foreach (var stat in mostKills) await gameEvent.Origin.TellAsync(mostKills, gameEvent.Owner.Manager.CancellationToken);
{
E.Origin.Tell(stat);
}
} }
else else
{ {
foreach (var stat in mostKills) foreach (var stat in mostKills)
{ {
E.Owner.Broadcast(stat); await gameEvent.Owner.Broadcast(stat).WaitAsync(Utilities.DefaultCommandTimeout,
gameEvent.Owner.Manager.CancellationToken);
} }
} }
} }

View File

@ -70,22 +70,16 @@ namespace IW4MAdmin.Plugins.Stats.Commands
_contextFactory = contextFactory; _contextFactory = contextFactory;
} }
public override async Task ExecuteAsync(GameEvent E) public override async Task ExecuteAsync(GameEvent gameEvent)
{ {
var topStats = await GetMostPlayed(E.Owner, _translationLookup, _contextFactory); var topStats = await GetMostPlayed(gameEvent.Owner, _translationLookup, _contextFactory);
if (!E.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix)) if (!gameEvent.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix))
{ {
foreach (var stat in topStats) await gameEvent.Origin.TellAsync(topStats, gameEvent.Owner.Manager.CancellationToken);
{
E.Origin.Tell(stat);
}
} }
else else
{ {
foreach (var stat in topStats) gameEvent.Owner.Broadcast(topStats);
{
E.Owner.Broadcast(stat);
}
} }
} }
} }

View File

@ -52,23 +52,21 @@ namespace IW4MAdmin.Plugins.Stats.Commands
_config = config; _config = config;
} }
public override async Task ExecuteAsync(GameEvent E) public override async Task ExecuteAsync(GameEvent gameEvent)
{ {
var topStats = await GetTopStats(E.Owner, _translationLookup); var topStats = await GetTopStats(gameEvent.Owner, _translationLookup);
if (!E.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix)) if (!gameEvent.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix))
{ {
foreach (var stat in topStats) await gameEvent.Origin.TellAsync(topStats, gameEvent.Owner.Manager.CancellationToken);
{
E.Origin.Tell(stat);
}
} }
else else
{ {
foreach (var stat in topStats) foreach (var stat in topStats)
{ {
E.Owner.Broadcast(stat); await gameEvent.Owner.Broadcast(stat).WaitAsync(Utilities.DefaultCommandTimeout,
gameEvent.Owner.Manager.CancellationToken);
} }
} }
} }
} }
} }

View File

@ -18,28 +18,27 @@ namespace Stats.Config
public int MostKillsClientLimit { get; set; } = 5; public int MostKillsClientLimit { get; set; } = 5;
public bool EnableAdvancedMetrics { get; set; } = true; public bool EnableAdvancedMetrics { get; set; } = true;
public WeaponNameParserConfiguration[] WeaponNameParserConfigurations { get; set; } = new[] public WeaponNameParserConfiguration[] WeaponNameParserConfigurations { get; set; } = {
{ new()
new WeaponNameParserConfiguration()
{ {
Game = Server.Game.IW3, Game = Server.Game.IW3,
WeaponSuffix = "mp", WeaponSuffix = "mp",
Delimiters = new[] {'_'} Delimiters = new[] {'_'}
}, },
new WeaponNameParserConfiguration() new()
{ {
Game = Server.Game.IW4, Game = Server.Game.IW4,
WeaponSuffix = "mp", WeaponSuffix = "mp",
Delimiters = new[] {'_'} Delimiters = new[] {'_'}
}, },
new WeaponNameParserConfiguration() new()
{ {
Game = Server.Game.IW5, Game = Server.Game.IW5,
WeaponSuffix = "mp", WeaponSuffix = "mp",
WeaponPrefix = "iw5", WeaponPrefix = "iw5",
Delimiters = new[] {'_'} Delimiters = new[] {'_'}
}, },
new WeaponNameParserConfiguration() new()
{ {
Game = Server.Game.T6, Game = Server.Game.T6,
WeaponSuffix = "mp", WeaponSuffix = "mp",
@ -48,7 +47,7 @@ namespace Stats.Config
}; };
[Obsolete] public IDictionary<long, DetectionType[]> ServerDetectionTypes { get; set; } [Obsolete] public IDictionary<long, DetectionType[]> ServerDetectionTypes { get; set; }
public AnticheatConfiguration AnticheatConfiguration { get; set; } = new AnticheatConfiguration(); public AnticheatConfiguration AnticheatConfiguration { get; set; } = new();
#pragma warning disable CS0612 // Type or member is obsolete #pragma warning disable CS0612 // Type or member is obsolete
public void ApplyMigration() public void ApplyMigration()
@ -77,22 +76,22 @@ namespace Stats.Config
Utilities.CurrentLocalization.LocalizationIndex["PLUGIN_STATS_SETUP_ENABLEAC"].PromptBool(); Utilities.CurrentLocalization.LocalizationIndex["PLUGIN_STATS_SETUP_ENABLEAC"].PromptBool();
KillstreakMessages = new List<StreakMessageConfiguration> KillstreakMessages = new List<StreakMessageConfiguration>
{ {
new StreakMessageConfiguration new()
{ {
Count = -1, Count = -1,
Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_STREAK_MESSAGE_SUICIDE"] Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_STREAK_MESSAGE_SUICIDE"]
}, },
new StreakMessageConfiguration new()
{ {
Count = 5, Count = 5,
Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_STREAK_MESSAGE_5"] Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_STREAK_MESSAGE_5"]
}, },
new StreakMessageConfiguration new()
{ {
Count = 10, Count = 10,
Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_STREAK_MESSAGE_10"] Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_STREAK_MESSAGE_10"]
}, },
new StreakMessageConfiguration new()
{ {
Count = 25, Count = 25,
Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_STREAK_MESSAGE_25"] Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_STREAK_MESSAGE_25"]
@ -101,12 +100,12 @@ namespace Stats.Config
DeathstreakMessages = new List<StreakMessageConfiguration>() DeathstreakMessages = new List<StreakMessageConfiguration>()
{ {
new StreakMessageConfiguration() new()
{ {
Count = 5, Count = 5,
Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_DEATH_STREAK_MESSAGE_5"] Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_DEATH_STREAK_MESSAGE_5"]
}, },
new StreakMessageConfiguration() new()
{ {
Count = 10, Count = 10,
Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_DEATH_STREAK_MESSAGE_10"] Message = Utilities.CurrentLocalization.LocalizationIndex["STATS_DEATH_STREAK_MESSAGE_10"]
@ -119,4 +118,4 @@ namespace Stats.Config
return this; return this;
} }
} }
} }

View File

@ -9,6 +9,7 @@ using Stats.Dtos;
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq; using System.Linq;
using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using Data.Abstractions; using Data.Abstractions;
using Data.Models.Client.Stats; using Data.Models.Client.Stats;
@ -35,7 +36,7 @@ namespace IW4MAdmin.Plugins.Stats
private readonly IDatabaseContextFactory _databaseContextFactory; private readonly IDatabaseContextFactory _databaseContextFactory;
private readonly ITranslationLookup _translationLookup; private readonly ITranslationLookup _translationLookup;
private readonly IMetaService _metaService; private readonly IMetaServiceV2 _metaService;
private readonly IResourceQueryHelper<ChatSearchQuery, MessageResponse> _chatQueryHelper; private readonly IResourceQueryHelper<ChatSearchQuery, MessageResponse> _chatQueryHelper;
private readonly ILogger<StatManager> _managerLogger; private readonly ILogger<StatManager> _managerLogger;
private readonly ILogger<Plugin> _logger; private readonly ILogger<Plugin> _logger;
@ -43,7 +44,7 @@ namespace IW4MAdmin.Plugins.Stats
private readonly IServerDistributionCalculator _serverDistributionCalculator; private readonly IServerDistributionCalculator _serverDistributionCalculator;
public Plugin(ILogger<Plugin> logger, IConfigurationHandlerFactory configurationHandlerFactory, IDatabaseContextFactory databaseContextFactory, public Plugin(ILogger<Plugin> logger, IConfigurationHandlerFactory configurationHandlerFactory, IDatabaseContextFactory databaseContextFactory,
ITranslationLookup translationLookup, IMetaService metaService, IResourceQueryHelper<ChatSearchQuery, MessageResponse> chatQueryHelper, ILogger<StatManager> managerLogger, ITranslationLookup translationLookup, IMetaServiceV2 metaService, IResourceQueryHelper<ChatSearchQuery, MessageResponse> chatQueryHelper, ILogger<StatManager> managerLogger,
IEnumerable<IClientStatisticCalculator> statCalculators, IServerDistributionCalculator serverDistributionCalculator) IEnumerable<IClientStatisticCalculator> statCalculators, IServerDistributionCalculator serverDistributionCalculator)
{ {
Config = configurationHandlerFactory.GetConfigurationHandler<StatsConfiguration>("StatsPluginSettings"); Config = configurationHandlerFactory.GetConfigurationHandler<StatsConfiguration>("StatsPluginSettings");
@ -57,52 +58,52 @@ namespace IW4MAdmin.Plugins.Stats
_serverDistributionCalculator = serverDistributionCalculator; _serverDistributionCalculator = serverDistributionCalculator;
} }
public async Task OnEventAsync(GameEvent E, Server S) public async Task OnEventAsync(GameEvent gameEvent, Server server)
{ {
switch (E.Type) switch (gameEvent.Type)
{ {
case GameEvent.EventType.Start: case GameEvent.EventType.Start:
Manager.AddServer(S); Manager.AddServer(server);
break; break;
case GameEvent.EventType.Disconnect: case GameEvent.EventType.Disconnect:
await Manager.RemovePlayer(E.Origin); await Manager.RemovePlayer(gameEvent.Origin);
break; break;
case GameEvent.EventType.Say: case GameEvent.EventType.Say:
if (!string.IsNullOrEmpty(E.Data) && if (!string.IsNullOrEmpty(gameEvent.Data) &&
E.Origin.ClientId > 1) gameEvent.Origin.ClientId > 1)
{ {
await Manager.AddMessageAsync(E.Origin.ClientId, StatManager.GetIdForServer(S), true, E.Data); await Manager.AddMessageAsync(gameEvent.Origin.ClientId, StatManager.GetIdForServer(server), true, gameEvent.Data);
} }
break; break;
case GameEvent.EventType.MapChange: case GameEvent.EventType.MapChange:
Manager.SetTeamBased(StatManager.GetIdForServer(S), S.Gametype != "dm"); Manager.SetTeamBased(StatManager.GetIdForServer(server), server.Gametype != "dm");
Manager.ResetKillstreaks(S); Manager.ResetKillstreaks(server);
await Manager.Sync(S); await Manager.Sync(server);
break; break;
case GameEvent.EventType.MapEnd: case GameEvent.EventType.MapEnd:
Manager.ResetKillstreaks(S); Manager.ResetKillstreaks(server);
await Manager.Sync(S); await Manager.Sync(server);
break; break;
case GameEvent.EventType.Command: case GameEvent.EventType.Command:
var shouldPersist = !string.IsNullOrEmpty(E.Data) && var shouldPersist = !string.IsNullOrEmpty(gameEvent.Data) &&
E.Extra?.GetType().Name == "SayCommand"; gameEvent.Extra?.GetType().Name == "SayCommand";
if (shouldPersist) if (shouldPersist)
{ {
await Manager.AddMessageAsync(E.Origin.ClientId, StatManager.GetIdForServer(S), false, E.Data); await Manager.AddMessageAsync(gameEvent.Origin.ClientId, StatManager.GetIdForServer(server), false, gameEvent.Data);
} }
break; break;
case GameEvent.EventType.ScriptKill: case GameEvent.EventType.ScriptKill:
string[] killInfo = (E.Data != null) ? E.Data.Split(';') : new string[0]; var killInfo = (gameEvent.Data != null) ? gameEvent.Data.Split(';') : Array.Empty<string>();
if ((S.CustomCallback || ShouldOverrideAnticheatSetting(S)) && killInfo.Length >= 18 && !ShouldIgnoreEvent(E.Origin, E.Target)) if ((server.CustomCallback || ShouldOverrideAnticheatSetting(server)) && killInfo.Length >= 18 && !ShouldIgnoreEvent(gameEvent.Origin, gameEvent.Target))
{ {
// this treats "world" damage as self damage // this treats "world" damage as self damage
if (IsWorldDamage(E.Origin)) if (IsWorldDamage(gameEvent.Origin))
{ {
E.Origin = E.Target; gameEvent.Origin = gameEvent.Target;
} }
await EnsureClientsAdded(E.Origin, E.Target); await EnsureClientsAdded(gameEvent.Origin, gameEvent.Target);
await Manager.AddScriptHit(false, E.Time, E.Origin, E.Target, StatManager.GetIdForServer(S), S.CurrentMap.Name, killInfo[7], killInfo[8], await Manager.AddScriptHit(false, gameEvent.Time, gameEvent.Origin, gameEvent.Target, StatManager.GetIdForServer(server), server.CurrentMap.Name, killInfo[7], killInfo[8],
killInfo[5], killInfo[6], killInfo[3], killInfo[4], killInfo[9], killInfo[10], killInfo[11], killInfo[12], killInfo[13], killInfo[14], killInfo[15], killInfo[16], killInfo[17]); killInfo[5], killInfo[6], killInfo[3], killInfo[4], killInfo[9], killInfo[10], killInfo[11], killInfo[12], killInfo[13], killInfo[14], killInfo[15], killInfo[16], killInfo[17]);
} }
@ -112,42 +113,42 @@ namespace IW4MAdmin.Plugins.Stats
} }
break; break;
case GameEvent.EventType.Kill: case GameEvent.EventType.Kill:
if (!ShouldIgnoreEvent(E.Origin, E.Target)) if (!ShouldIgnoreEvent(gameEvent.Origin, gameEvent.Target))
{ {
// this treats "world" damage as self damage // this treats "world" damage as self damage
if (IsWorldDamage(E.Origin)) if (IsWorldDamage(gameEvent.Origin))
{ {
E.Origin = E.Target; gameEvent.Origin = gameEvent.Target;
} }
await EnsureClientsAdded(E.Origin, E.Target); await EnsureClientsAdded(gameEvent.Origin, gameEvent.Target);
await Manager.AddStandardKill(E.Origin, E.Target); await Manager.AddStandardKill(gameEvent.Origin, gameEvent.Target);
} }
break; break;
case GameEvent.EventType.Damage: case GameEvent.EventType.Damage:
if (!ShouldIgnoreEvent(E.Origin, E.Target)) if (!ShouldIgnoreEvent(gameEvent.Origin, gameEvent.Target))
{ {
// this treats "world" damage as self damage // this treats "world" damage as self damage
if (IsWorldDamage(E.Origin)) if (IsWorldDamage(gameEvent.Origin))
{ {
E.Origin = E.Target; gameEvent.Origin = gameEvent.Target;
} }
Manager.AddDamageEvent(E.Data, E.Origin.ClientId, E.Target.ClientId, StatManager.GetIdForServer(S)); Manager.AddDamageEvent(gameEvent.Data, gameEvent.Origin.ClientId, gameEvent.Target.ClientId, StatManager.GetIdForServer(server));
} }
break; break;
case GameEvent.EventType.ScriptDamage: case GameEvent.EventType.ScriptDamage:
killInfo = (E.Data != null) ? E.Data.Split(';') : new string[0]; killInfo = (gameEvent.Data != null) ? gameEvent.Data.Split(';') : new string[0];
if ((S.CustomCallback || ShouldOverrideAnticheatSetting(S)) && killInfo.Length >= 18 && !ShouldIgnoreEvent(E.Origin, E.Target)) if ((server.CustomCallback || ShouldOverrideAnticheatSetting(server)) && killInfo.Length >= 18 && !ShouldIgnoreEvent(gameEvent.Origin, gameEvent.Target))
{ {
// this treats "world" damage as self damage // this treats "world" damage as self damage
if (IsWorldDamage(E.Origin)) if (IsWorldDamage(gameEvent.Origin))
{ {
E.Origin = E.Target; gameEvent.Origin = gameEvent.Target;
} }
await EnsureClientsAdded(E.Origin, E.Target); await EnsureClientsAdded(gameEvent.Origin, gameEvent.Target);
await Manager.AddScriptHit(true, E.Time, E.Origin, E.Target, StatManager.GetIdForServer(S), S.CurrentMap.Name, killInfo[7], killInfo[8], await Manager.AddScriptHit(true, gameEvent.Time, gameEvent.Origin, gameEvent.Target, StatManager.GetIdForServer(server), server.CurrentMap.Name, killInfo[7], killInfo[8],
killInfo[5], killInfo[6], killInfo[3], killInfo[4], killInfo[9], killInfo[10], killInfo[11], killInfo[12], killInfo[13], killInfo[14], killInfo[15], killInfo[16], killInfo[17]); killInfo[5], killInfo[6], killInfo[3], killInfo[4], killInfo[9], killInfo[10], killInfo[11], killInfo[12], killInfo[13], killInfo[14], killInfo[15], killInfo[16], killInfo[17]);
} }
@ -165,7 +166,7 @@ namespace IW4MAdmin.Plugins.Stats
foreach (var calculator in _statCalculators) foreach (var calculator in _statCalculators)
{ {
await calculator.CalculateForEvent(E); await calculator.CalculateForEvent(gameEvent);
} }
} }
@ -188,23 +189,22 @@ namespace IW4MAdmin.Plugins.Stats
"/Stats/TopPlayersAsync"); "/Stats/TopPlayersAsync");
// meta data info // meta data info
async Task<IEnumerable<InformationResponse>> getStats(ClientPaginationRequest request) async Task<IEnumerable<InformationResponse>> GetStats(ClientPaginationRequest request, CancellationToken token = default)
{ {
IList<EFClientStatistics> clientStats;
await using var ctx = _databaseContextFactory.CreateContext(enableTracking: false); await using var ctx = _databaseContextFactory.CreateContext(enableTracking: false);
clientStats = await ctx.Set<EFClientStatistics>().Where(c => c.ClientId == request.ClientId).ToListAsync(); IList<EFClientStatistics> clientStats = await ctx.Set<EFClientStatistics>().Where(c => c.ClientId == request.ClientId).ToListAsync(token);
int kills = clientStats.Sum(c => c.Kills); var kills = clientStats.Sum(c => c.Kills);
int deaths = clientStats.Sum(c => c.Deaths); var deaths = clientStats.Sum(c => c.Deaths);
double kdr = Math.Round(kills / (double)deaths, 2); var kdr = Math.Round(kills / (double)deaths, 2);
var validPerformanceValues = clientStats.Where(c => c.Performance > 0); var validPerformanceValues = clientStats.Where(c => c.Performance > 0).ToList();
int performancePlayTime = validPerformanceValues.Sum(s => s.TimePlayed); var performancePlayTime = validPerformanceValues.Sum(s => s.TimePlayed);
double performance = Math.Round(validPerformanceValues.Sum(c => c.Performance * c.TimePlayed / performancePlayTime), 2); var performance = Math.Round(validPerformanceValues.Sum(c => c.Performance * c.TimePlayed / performancePlayTime), 2);
double spm = Math.Round(clientStats.Sum(c => c.SPM) / clientStats.Where(c => c.SPM > 0).Count(), 1); var spm = Math.Round(clientStats.Sum(c => c.SPM) / clientStats.Count(c => c.SPM > 0), 1);
return new List<InformationResponse>() return new List<InformationResponse>
{ {
new InformationResponse() new InformationResponse
{ {
Key = Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_RANKING"], Key = Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_CLIENT_META_RANKING"],
Value = "#" + (await Manager.GetClientOverallRanking(request.ClientId)).ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)), Value = "#" + (await Manager.GetClientOverallRanking(request.ClientId)).ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)),
@ -212,7 +212,7 @@ namespace IW4MAdmin.Plugins.Stats
Order = 0, Order = 0,
Type = MetaType.Information Type = MetaType.Information
}, },
new InformationResponse() new InformationResponse
{ {
Key = Utilities.CurrentLocalization.LocalizationIndex["PLUGINS_STATS_TEXT_KILLS"], Key = Utilities.CurrentLocalization.LocalizationIndex["PLUGINS_STATS_TEXT_KILLS"],
Value = kills.ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)), Value = kills.ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)),
@ -220,7 +220,7 @@ namespace IW4MAdmin.Plugins.Stats
Order = 1, Order = 1,
Type = MetaType.Information Type = MetaType.Information
}, },
new InformationResponse() new InformationResponse
{ {
Key = Utilities.CurrentLocalization.LocalizationIndex["PLUGINS_STATS_TEXT_DEATHS"], Key = Utilities.CurrentLocalization.LocalizationIndex["PLUGINS_STATS_TEXT_DEATHS"],
Value = deaths.ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)), Value = deaths.ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)),
@ -228,7 +228,7 @@ namespace IW4MAdmin.Plugins.Stats
Order = 2, Order = 2,
Type = MetaType.Information Type = MetaType.Information
}, },
new InformationResponse() new InformationResponse
{ {
Key = Utilities.CurrentLocalization.LocalizationIndex["PLUGINS_STATS_TEXT_KDR"], Key = Utilities.CurrentLocalization.LocalizationIndex["PLUGINS_STATS_TEXT_KDR"],
Value = kdr.ToString(new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)), Value = kdr.ToString(new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)),
@ -236,7 +236,7 @@ namespace IW4MAdmin.Plugins.Stats
Order = 3, Order = 3,
Type = MetaType.Information Type = MetaType.Information
}, },
new InformationResponse() new InformationResponse
{ {
Key = Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_PROFILE_PERFORMANCE"], Key = Utilities.CurrentLocalization.LocalizationIndex["WEBFRONT_PROFILE_PERFORMANCE"],
Value = performance.ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)), Value = performance.ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)),
@ -244,7 +244,7 @@ namespace IW4MAdmin.Plugins.Stats
Order = 4, Order = 4,
Type = MetaType.Information Type = MetaType.Information
}, },
new InformationResponse() new InformationResponse
{ {
Key = Utilities.CurrentLocalization.LocalizationIndex["PLUGINS_STATS_META_SPM"], Key = Utilities.CurrentLocalization.LocalizationIndex["PLUGINS_STATS_META_SPM"],
Value = spm.ToString(new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)), Value = spm.ToString(new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)),
@ -255,15 +255,13 @@ namespace IW4MAdmin.Plugins.Stats
}; };
} }
async Task<IEnumerable<InformationResponse>> getAnticheatInfo(ClientPaginationRequest request) async Task<IEnumerable<InformationResponse>> GetAnticheatInfo(ClientPaginationRequest request, CancellationToken token = default)
{ {
IList<EFClientStatistics> clientStats; await using var context = _databaseContextFactory.CreateContext(enableTracking: false);
IList<EFClientStatistics> clientStats = await context.Set<EFClientStatistics>()
await using var ctx = _databaseContextFactory.CreateContext(enableTracking: false);
clientStats = await ctx.Set<EFClientStatistics>()
.Include(c => c.HitLocations) .Include(c => c.HitLocations)
.Where(c => c.ClientId == request.ClientId) .Where(c => c.ClientId == request.ClientId)
.ToListAsync(); .ToListAsync(token);
double headRatio = 0; double headRatio = 0;
double chestRatio = 0; double chestRatio = 0;
@ -271,9 +269,9 @@ namespace IW4MAdmin.Plugins.Stats
double chestAbdomenRatio = 0; double chestAbdomenRatio = 0;
double hitOffsetAverage = 0; double hitOffsetAverage = 0;
double averageSnapValue = 0; double averageSnapValue = 0;
double maxStrain = clientStats.Count(c => c.MaxStrain > 0) == 0 ? 0 : clientStats.Max(cs => cs.MaxStrain); var maxStrain = !clientStats.Any(c => c.MaxStrain > 0) ? 0 : clientStats.Max(cs => cs.MaxStrain);
if (clientStats.Where(cs => cs.HitLocations.Count > 0).FirstOrDefault() != null) if (clientStats.Any(cs => cs.HitLocations.Count > 0))
{ {
chestRatio = Math.Round((clientStats.Where(c => c.HitLocations.Count > 0).Sum(c => chestRatio = Math.Round((clientStats.Where(c => c.HitLocations.Count > 0).Sum(c =>
c.HitLocations.First(hl => hl.Location == (int)IW4Info.HitLocation.torso_upper).HitCount) / c.HitLocations.First(hl => hl.Location == (int)IW4Info.HitLocation.torso_upper).HitCount) /
@ -291,12 +289,12 @@ namespace IW4MAdmin.Plugins.Stats
(double)clientStats.Where(c => c.HitLocations.Count > 0) (double)clientStats.Where(c => c.HitLocations.Count > 0)
.Sum(c => c.HitLocations.Where(hl => hl.Location != (int)IW4Info.HitLocation.none).Sum(f => f.HitCount))) * 100.0, 0); .Sum(c => c.HitLocations.Where(hl => hl.Location != (int)IW4Info.HitLocation.none).Sum(f => f.HitCount))) * 100.0, 0);
var validOffsets = clientStats.Where(c => c.HitLocations.Count(hl => hl.HitCount > 0) > 0).SelectMany(hl => hl.HitLocations); var validOffsets = clientStats.Where(c => c.HitLocations.Count(hl => hl.HitCount > 0) > 0).SelectMany(hl => hl.HitLocations).ToList();
hitOffsetAverage = validOffsets.Sum(o => o.HitCount * o.HitOffsetAverage) / (double)validOffsets.Sum(o => o.HitCount); hitOffsetAverage = validOffsets.Sum(o => o.HitCount * o.HitOffsetAverage) / (double)validOffsets.Sum(o => o.HitCount);
averageSnapValue = clientStats.Any(_stats => _stats.AverageSnapValue > 0) ? clientStats.Where(_stats => _stats.AverageSnapValue > 0).Average(_stat => _stat.AverageSnapValue) : 0; averageSnapValue = clientStats.Any(_stats => _stats.AverageSnapValue > 0) ? clientStats.Where(_stats => _stats.AverageSnapValue > 0).Average(_stat => _stat.AverageSnapValue) : 0;
} }
return new List<InformationResponse>() return new List<InformationResponse>
{ {
new InformationResponse() new InformationResponse()
{ {
@ -372,9 +370,9 @@ namespace IW4MAdmin.Plugins.Stats
}; };
} }
async Task<IEnumerable<MessageResponse>> getMessages(ClientPaginationRequest request) async Task<IEnumerable<MessageResponse>> GetMessages(ClientPaginationRequest request, CancellationToken token = default)
{ {
var query = new ChatSearchQuery() var query = new ChatSearchQuery
{ {
ClientId = request.ClientId, ClientId = request.ClientId,
Before = request.Before, Before = request.Before,
@ -388,49 +386,49 @@ namespace IW4MAdmin.Plugins.Stats
if (Config.Configuration().AnticheatConfiguration.Enable) if (Config.Configuration().AnticheatConfiguration.Enable)
{ {
_metaService.AddRuntimeMeta<ClientPaginationRequest, InformationResponse>(MetaType.Information, getAnticheatInfo); _metaService.AddRuntimeMeta<ClientPaginationRequest, InformationResponse>(MetaType.Information, GetAnticheatInfo);
} }
_metaService.AddRuntimeMeta<ClientPaginationRequest, InformationResponse>(MetaType.Information, getStats); _metaService.AddRuntimeMeta<ClientPaginationRequest, InformationResponse>(MetaType.Information, GetStats);
_metaService.AddRuntimeMeta<ClientPaginationRequest, MessageResponse>(MetaType.ChatMessage, getMessages); _metaService.AddRuntimeMeta<ClientPaginationRequest, MessageResponse>(MetaType.ChatMessage, GetMessages);
async Task<string> totalKills(Server server) async Task<string> TotalKills(Server server)
{ {
await using var context = _databaseContextFactory.CreateContext(false); await using var context = _databaseContextFactory.CreateContext(false);
long kills = await context.Set<EFServerStatistics>().Where(s => s.Active).SumAsync(s => s.TotalKills); var kills = await context.Set<EFServerStatistics>().Where(s => s.Active).SumAsync(s => s.TotalKills);
return kills.ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)); return kills.ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName));
} }
async Task<string> totalPlayTime(Server server) async Task<string> TotalPlayTime(Server server)
{ {
await using var context = _databaseContextFactory.CreateContext(false); await using var context = _databaseContextFactory.CreateContext(false);
long playTime = await context.Set<EFServerStatistics>().Where(s => s.Active).SumAsync(s => s.TotalPlayTime); var playTime = await context.Set<EFServerStatistics>().Where(s => s.Active).SumAsync(s => s.TotalPlayTime);
return (playTime / 3600.0).ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName)); return (playTime / 3600.0).ToString("#,##0", new System.Globalization.CultureInfo(Utilities.CurrentLocalization.LocalizationName));
} }
async Task<string> topStats(Server s) async Task<string> TopStats(Server s)
{ {
// todo: this needs to needs to be updated when we DI the lookup // todo: this needs to needs to be updated when we DI the lookup
return string.Join(Environment.NewLine, await Commands.TopStats.GetTopStats(s, Utilities.CurrentLocalization.LocalizationIndex)); return string.Join(Environment.NewLine, await Commands.TopStats.GetTopStats(s, Utilities.CurrentLocalization.LocalizationIndex));
} }
async Task<string> mostPlayed(Server s) async Task<string> MostPlayed(Server s)
{ {
// todo: this needs to needs to be updated when we DI the lookup // todo: this needs to needs to be updated when we DI the lookup
return string.Join(Environment.NewLine, await Commands.MostPlayedCommand.GetMostPlayed(s, Utilities.CurrentLocalization.LocalizationIndex, _databaseContextFactory)); return string.Join(Environment.NewLine, await Commands.MostPlayedCommand.GetMostPlayed(s, Utilities.CurrentLocalization.LocalizationIndex, _databaseContextFactory));
} }
async Task<string> mostKills(Server gameServer) async Task<string> MostKills(Server gameServer)
{ {
return string.Join(Environment.NewLine, return string.Join(Environment.NewLine,
await Commands.MostKillsCommand.GetMostKills(StatManager.GetIdForServer(gameServer), Config.Configuration(), _databaseContextFactory, _translationLookup)); await Commands.MostKillsCommand.GetMostKills(StatManager.GetIdForServer(gameServer), Config.Configuration(), _databaseContextFactory, _translationLookup));
} }
manager.GetMessageTokens().Add(new MessageToken("TOTALKILLS", totalKills)); manager.GetMessageTokens().Add(new MessageToken("TOTALKILLS", TotalKills));
manager.GetMessageTokens().Add(new MessageToken("TOTALPLAYTIME", totalPlayTime)); manager.GetMessageTokens().Add(new MessageToken("TOTALPLAYTIME", TotalPlayTime));
manager.GetMessageTokens().Add(new MessageToken("TOPSTATS", topStats)); manager.GetMessageTokens().Add(new MessageToken("TOPSTATS", TopStats));
manager.GetMessageTokens().Add(new MessageToken("MOSTPLAYED", mostPlayed)); manager.GetMessageTokens().Add(new MessageToken("MOSTPLAYED", MostPlayed));
manager.GetMessageTokens().Add(new MessageToken("MOSTKILLS", mostKills)); manager.GetMessageTokens().Add(new MessageToken("MOSTKILLS", MostKills));
if (Config.Configuration().EnableAdvancedMetrics) if (Config.Configuration().EnableAdvancedMetrics)
{ {
@ -445,7 +443,7 @@ namespace IW4MAdmin.Plugins.Stats
await _serverDistributionCalculator.Initialize(); await _serverDistributionCalculator.Initialize();
} }
public Task OnTickAsync(Server S) public Task OnTickAsync(Server server)
{ {
return Task.CompletedTask; return Task.CompletedTask;
} }

View File

@ -17,7 +17,7 @@
</PropertyGroup> </PropertyGroup>
<ItemGroup> <ItemGroup>
<PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.1.28.1" PrivateAssets="All" /> <PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.3.23.1" PrivateAssets="All" />
</ItemGroup> </ItemGroup>
<Target Name="PostBuild" AfterTargets="PostBuildEvent"> <Target Name="PostBuild" AfterTargets="PostBuildEvent">

View File

@ -20,7 +20,7 @@
</Target> </Target>
<ItemGroup> <ItemGroup>
<PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.1.28.1" PrivateAssets="All" /> <PackageReference Include="RaidMax.IW4MAdmin.SharedLibraryCore" Version="2022.3.23.1" PrivateAssets="All" />
</ItemGroup> </ItemGroup>
</Project> </Project>

View File

@ -14,7 +14,7 @@ Latest binary builds are always available at:
**IW4MAdmin** requires minimal effort to get up and running. **IW4MAdmin** requires minimal effort to get up and running.
### Prerequisites ### Prerequisites
* [.NET Core 6.0.x Runtime](https://www.microsoft.com/net/download) * [.NET Core 6.0.x Runtime](https://www.microsoft.com/net/download)
* [Direct Download (Windows)](https://dotnet.microsoft.com/en-us/download/dotnet/thank-you/runtime-aspnetcore-6.0.1-windows-x64-installer) * [Direct Download (Windows)](https://dotnet.microsoft.com/en-us/download/dotnet/thank-you/runtime-aspnetcore-6.0.1-windows-hosting-bundle-installer)
* [Package Installation Instructions (Linux)](https://docs.microsoft.com/en-us/dotnet/core/install/linux) * [Package Installation Instructions (Linux)](https://docs.microsoft.com/en-us/dotnet/core/install/linux)
### Installation ### Installation
1. Install .NET Core Runtime 1. Install .NET Core Runtime

View File

@ -1,39 +0,0 @@
using System.Threading.Tasks;
using Data.Models;
using Data.Models.Client;
using SharedLibraryCore.Configuration;
using SharedLibraryCore.Interfaces;
namespace SharedLibraryCore.Commands
{
public class AddClientTagCommand : Command
{
private readonly IMetaService _metaService;
public AddClientTagCommand(CommandConfiguration config, ITranslationLookup layout, IMetaService metaService) :
base(config, layout)
{
Name = "addclienttag";
Description = layout["COMMANDS_ADD_CLIENT_TAG_DESC"];
Alias = "act";
Permission = EFClient.Permission.Owner;
RequiresTarget = false;
Arguments = new[]
{
new CommandArgument
{
Name = _translationLookup["COMMANDS_ARGUMENT_TAG"],
Required = true
}
};
_metaService = metaService;
}
public override async Task ExecuteAsync(GameEvent gameEvent)
{
await _metaService.AddPersistentMeta(EFMeta.ClientTagName, gameEvent.Data);
gameEvent.Origin.Tell(_translationLookup["COMMANDS_ADD_CLIENT_TAG_SUCCESS"].FormatExt(gameEvent.Data));
}
}
}

View File

@ -378,29 +378,31 @@ namespace SharedLibraryCore.Commands
}; };
} }
public override async Task ExecuteAsync(GameEvent E) public override async Task ExecuteAsync(GameEvent gameEvent)
{ {
// todo: don't do the lookup here // todo: don't do the lookup here
var penalties = await E.Owner.Manager.GetPenaltyService().GetActivePenaltiesAsync(E.Target.AliasLinkId, E.Target.CurrentAliasId); var penalties = await gameEvent.Owner.Manager.GetPenaltyService().GetActivePenaltiesAsync(gameEvent.Target.AliasLinkId,
gameEvent.Target.CurrentAliasId, gameEvent.Target.NetworkId, gameEvent.Target.CurrentAlias.IPAddress);
if (penalties if (penalties
.FirstOrDefault(p => .FirstOrDefault(p =>
p.Type == EFPenalty.PenaltyType.Ban || p.Type == EFPenalty.PenaltyType.TempBan) != null) p.Type == EFPenalty.PenaltyType.Ban || p.Type == EFPenalty.PenaltyType.TempBan) != null)
{ {
switch ((await E.Target.Unban(E.Data, E.Origin) switch ((await gameEvent.Target.Unban(gameEvent.Data, gameEvent.Origin)
.WaitAsync(Utilities.DefaultCommandTimeout, E.Owner.Manager.CancellationToken)).FailReason) .WaitAsync(Utilities.DefaultCommandTimeout, gameEvent.Owner.Manager.CancellationToken)).FailReason)
{ {
case GameEvent.EventFailReason.None: case GameEvent.EventFailReason.None:
E.Origin.Tell(_translationLookup["COMMANDS_UNBAN_SUCCESS"].FormatExt(E.Target)); gameEvent.Origin.Tell(_translationLookup["COMMANDS_UNBAN_SUCCESS"].FormatExt(gameEvent.Target));
break; break;
default: default:
E.Origin.Tell(_translationLookup["SERVER_ERROR_COMMAND_INGAME"]); gameEvent.Origin.Tell(_translationLookup["SERVER_ERROR_COMMAND_INGAME"]);
break; break;
} }
} }
else else
{ {
E.Origin.Tell(_translationLookup["COMMANDS_UNBAN_FAIL"].FormatExt(E.Target)); gameEvent.Origin.Tell(_translationLookup["COMMANDS_UNBAN_FAIL"].FormatExt(gameEvent.Target));
} }
} }
} }
@ -707,37 +709,35 @@ namespace SharedLibraryCore.Commands
RequiresTarget = false; RequiresTarget = false;
} }
public override Task ExecuteAsync(GameEvent E) public override async Task ExecuteAsync(GameEvent gameEvent)
{ {
if (E.Owner.Manager.GetApplicationSettings().Configuration().GlobalRules?.Length < 1 && if (gameEvent.Owner.Manager.GetApplicationSettings().Configuration().GlobalRules?.Length < 1 &&
E.Owner.ServerConfig.Rules?.Length < 1) gameEvent.Owner.ServerConfig.Rules?.Length < 1)
{ {
var _ = E.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix) var _ = gameEvent.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix)
? E.Owner.Broadcast(_translationLookup["COMMANDS_RULES_NONE"]) ? gameEvent.Owner.Broadcast(_translationLookup["COMMANDS_RULES_NONE"])
: E.Origin.Tell(_translationLookup["COMMANDS_RULES_NONE"]); : gameEvent.Origin.Tell(_translationLookup["COMMANDS_RULES_NONE"]);
} }
else else
{ {
var rules = new List<string>(); var rules = new List<string>();
rules.AddRange(E.Owner.Manager.GetApplicationSettings().Configuration().GlobalRules); rules.AddRange(gameEvent.Owner.Manager.GetApplicationSettings().Configuration().GlobalRules);
if (E.Owner.ServerConfig.Rules != null) if (gameEvent.Owner.ServerConfig.Rules != null)
{ {
rules.AddRange(E.Owner.ServerConfig.Rules); rules.AddRange(gameEvent.Owner.ServerConfig.Rules);
} }
var ruleFomat = rules.Select(r => $"- {r}"); var ruleFormat = rules.Select(r => $"- {r}");
if (E.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix)) if (gameEvent.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix))
{ {
E.Owner.Broadcast(ruleFomat); gameEvent.Owner.Broadcast(ruleFormat);
} }
else else
{ {
E.Origin.Tell(ruleFomat); await gameEvent.Origin.TellAsync(ruleFormat, gameEvent.Owner.Manager.CancellationToken);
} }
} }
return Task.CompletedTask;
} }
} }
@ -850,7 +850,7 @@ namespace SharedLibraryCore.Commands
{ {
Name = "mask"; Name = "mask";
Description = _translationLookup["COMMANDS_MASK_DESC"]; Description = _translationLookup["COMMANDS_MASK_DESC"];
Alias = "hide"; Alias = "ma";
Permission = Permission.Moderator; Permission = Permission.Moderator;
RequiresTarget = false; RequiresTarget = false;
} }
@ -898,7 +898,7 @@ namespace SharedLibraryCore.Commands
public override async Task ExecuteAsync(GameEvent E) public override async Task ExecuteAsync(GameEvent E)
{ {
var existingPenalties = await E.Owner.Manager.GetPenaltyService() var existingPenalties = await E.Owner.Manager.GetPenaltyService()
.GetActivePenaltiesAsync(E.Target.AliasLinkId, E.Target.CurrentAliasId, E.Target.IPAddress); .GetActivePenaltiesAsync(E.Target.AliasLinkId, E.Target.CurrentAliasId, E.Target.NetworkId, E.Target.IPAddress);
var penalty = existingPenalties.FirstOrDefault(b => b.Type > EFPenalty.PenaltyType.Kick); var penalty = existingPenalties.FirstOrDefault(b => b.Type > EFPenalty.PenaltyType.Kick);
if (penalty == null) if (penalty == null)
@ -1135,10 +1135,10 @@ namespace SharedLibraryCore.Commands
/// </summary> /// </summary>
public class SetGravatarCommand : Command public class SetGravatarCommand : Command
{ {
private readonly IMetaService _metaService; private readonly IMetaServiceV2 _metaService;
public SetGravatarCommand(CommandConfiguration config, ITranslationLookup translationLookup, public SetGravatarCommand(CommandConfiguration config, ITranslationLookup translationLookup,
IMetaService metaService) : base(config, translationLookup) IMetaServiceV2 metaService) : base(config, translationLookup)
{ {
Name = "setgravatar"; Name = "setgravatar";
Description = _translationLookup["COMMANDS_GRAVATAR_DESC"]; Description = _translationLookup["COMMANDS_GRAVATAR_DESC"];
@ -1164,7 +1164,8 @@ namespace SharedLibraryCore.Commands
var gravatarEmail = string.Concat(md5 var gravatarEmail = string.Concat(md5
.ComputeHash(E.Data.ToLower().Select(d => Convert.ToByte(d)).ToArray()) .ComputeHash(E.Data.ToLower().Select(d => Convert.ToByte(d)).ToArray())
.Select(h => h.ToString("x2"))); .Select(h => h.ToString("x2")));
await _metaService.AddPersistentMeta("GravatarEmail", gravatarEmail, E.Origin); await _metaService.SetPersistentMeta("GravatarEmail", gravatarEmail, E.Origin.ClientId,
E.Owner.Manager.CancellationToken);
} }
E.Origin.Tell(_translationLookup["COMMANDS_GRAVATAR_SUCCESS_NEW"]); E.Origin.Tell(_translationLookup["COMMANDS_GRAVATAR_SUCCESS_NEW"]);
@ -1188,7 +1189,7 @@ namespace SharedLibraryCore.Commands
public static async Task<string> GetNextMap(Server s, ITranslationLookup lookup) public static async Task<string> GetNextMap(Server s, ITranslationLookup lookup)
{ {
var mapRotation = (await s.GetDvarAsync<string>("sv_mapRotation")).Value?.ToLower() ?? ""; var mapRotation = (await s.GetDvarAsync<string>("sv_mapRotation", token: s.Manager.CancellationToken)).Value?.ToLower() ?? "";
var regexMatches = Regex.Matches(mapRotation, var regexMatches = Regex.Matches(mapRotation,
@"((?:gametype|exec) +(?:([a-z]{1,4})(?:.cfg)?))? *map ([a-z|_|\d]+)", RegexOptions.IgnoreCase) @"((?:gametype|exec) +(?:([a-z]{1,4})(?:.cfg)?))? *map ([a-z|_|\d]+)", RegexOptions.IgnoreCase)
.ToList(); .ToList();

View File

@ -0,0 +1,9 @@
using SharedLibraryCore.Interfaces;
namespace SharedLibraryCore.Dtos;
public class LookupValue<TObject> : ILookupValue<TObject>
{
public int Id { get; set; }
public TObject Value { get; set; }
}

View File

@ -0,0 +1,20 @@
using System;
using Data.Models.Client;
namespace SharedLibraryCore.Dtos.Meta.Responses;
public class PermissionLevelChangedResponse : BaseMetaResponse
{
public EFClient.Permission PreviousPermissionLevel =>
(EFClient.Permission)Enum.Parse(typeof(EFClient.Permission),
PreviousPermissionLevelValue ?? EFClient.Permission.User.ToString());
public string PreviousPermissionLevelValue { get; set; }
public EFClient.Permission CurrentPermissionLevel => (EFClient.Permission)Enum.Parse(typeof(EFClient.Permission),
CurrentPermissionLevelValue ?? EFClient.Permission.User.ToString());
public string CurrentPermissionLevelValue { get; set; }
public int ChangedById { get; set; }
public string ChangedByName { get; set; }
}

View File

@ -257,6 +257,7 @@ namespace SharedLibraryCore
public EFClient Target; public EFClient Target;
public EventType Type; public EventType Type;
public string TypeName => Type.ToString();
public GameEvent() public GameEvent()
{ {
@ -334,6 +335,14 @@ namespace SharedLibraryCore
processed = true; processed = true;
} }
finally
{
if (_eventFinishedWaiter.CurrentCount == 0)
{
_eventFinishedWaiter.Release();
}
}
if (!processed) if (!processed)
{ {
using (LogContext.PushProperty("Server", Owner?.ToString())) using (LogContext.PushProperty("Server", Owner?.ToString()))

View File

@ -20,6 +20,7 @@ namespace SharedLibraryCore.Interfaces
public enum MetaType public enum MetaType
{ {
All = -1,
Other, Other,
Information, Information,
AliasUpdate, AliasUpdate,
@ -27,6 +28,7 @@ namespace SharedLibraryCore.Interfaces
Penalized, Penalized,
ReceivedPenalty, ReceivedPenalty,
QuickMessage, QuickMessage,
ConnectionHistory ConnectionHistory,
PermissionLevel,
} }
} }

View File

@ -18,6 +18,11 @@ namespace SharedLibraryCore.Interfaces
/// stores the regex information for a join event printed in the game log /// stores the regex information for a join event printed in the game log
/// </summary> /// </summary>
ParserRegex Join { get; set; } ParserRegex Join { get; set; }
/// <summary>
/// stores the regex information for a join team event printed in the game log
/// </summary>
ParserRegex JoinTeam { get; set; }
/// <summary> /// <summary>
/// stores the regex information for a quit event printed in the game log /// stores the regex information for a quit event printed in the game log
@ -59,4 +64,4 @@ namespace SharedLibraryCore.Interfaces
/// </summary> /// </summary>
NumberStyles GuidNumberStyle { get; set; } NumberStyles GuidNumberStyle { get; set; }
} }
} }

View File

@ -23,7 +23,8 @@ namespace SharedLibraryCore.Interfaces
/// </summary> /// </summary>
/// <param name="fileSizeDiff"></param> /// <param name="fileSizeDiff"></param>
/// <param name="startPosition"></param> /// <param name="startPosition"></param>
/// <param name="server"></param>
/// <returns></returns> /// <returns></returns>
Task<IEnumerable<GameEvent>> ReadEventsFromLog(long fileSizeDiff, long startPosition); Task<IEnumerable<GameEvent>> ReadEventsFromLog(long fileSizeDiff, long startPosition, Server server);
} }
} }

View File

@ -0,0 +1,7 @@
namespace SharedLibraryCore.Interfaces;
public interface ILookupValue<TObject>
{
int Id { get; }
TObject Value { get; }
}

View File

@ -8,6 +8,7 @@ using SharedLibraryCore.QueryHelper;
namespace SharedLibraryCore.Interfaces namespace SharedLibraryCore.Interfaces
{ {
[Obsolete("Use IMetaServiceV2")]
public interface IMetaService public interface IMetaService
{ {
/// <summary> /// <summary>

View File

@ -0,0 +1,194 @@
using System;
using System.Collections.Generic;
using System.Threading;
using System.Threading.Tasks;
using Data.Models;
using SharedLibraryCore.Dtos;
using SharedLibraryCore.QueryHelper;
namespace SharedLibraryCore.Interfaces;
public interface IMetaServiceV2
{
#region PER_CLIENT
/// <summary>
/// adds or updates meta key and value to the database as simple string
/// </summary>
/// <param name="metaKey">key of meta data</param>
/// <param name="metaValue">value of the meta data</param>
/// <param name="clientId">id of the client to save the meta for</param>
/// <param name="token"></param>
/// <returns></returns>
Task SetPersistentMeta(string metaKey, string metaValue, int clientId, CancellationToken token = default);
/// <summary>
/// add or update meta key and value to the database as serialized object type
/// </summary>
/// <param name="metaKey"></param>
/// <param name="metaValue"></param>
/// <param name="clientId"></param>
/// <param name="token"></param>
/// <typeparam name="T">type of object being serialized</typeparam>
/// <returns></returns>
Task SetPersistentMetaValue<T>(string metaKey, T metaValue, int clientId, CancellationToken token = default)
where T : class;
/// <summary>
/// Sets meta key to a linked lookup key and id
/// </summary>
/// <param name="metaKey">Key for the client meta</param>
/// <param name="lookupKey">Key of the global lookup meta</param>
/// <param name="lookupId">Id in the list of lookup values</param>
/// <param name="clientId">id of the client</param>
/// <param name="token"></param>
/// <typeparam name="T"></typeparam>
/// <returns></returns>
Task SetPersistentMetaForLookupKey(string metaKey, string lookupKey, int lookupId, int clientId,
CancellationToken token = default);
/// <summary>
/// increments meta value and persists to the database
/// <remarks>if the meta value does not already exist it will be set to the increment amount</remarks>
/// <remarks>the assumption is made that the existing value is <see cref="int"/></remarks>
/// </summary>
/// <param name="metaKey">key of meta data</param>
/// <param name="incrementAmount">value to increment by</param>
/// <param name="clientId">id of the client to save the meta for</param>
/// <param name="token"></param>
/// <returns></returns>
Task IncrementPersistentMeta(string metaKey, int incrementAmount, int clientId,
CancellationToken token = default);
/// <summary>
/// decrements meta value and persists to the database
/// <remarks>if the meta value does not already exist it will be set to the decrement amount</remarks>
/// <remarks>the assumption is made that the existing value is <see cref="int"/></remarks>
/// </summary>
/// <param name="metaKey">key of meta data</param>
/// <param name="decrementAmount">value to increment by</param>
/// <param name="clientId">id of the client to save the meta for</param>
/// <param name="token"></param>
/// <returns></returns>
Task DecrementPersistentMeta(string metaKey, int decrementAmount, int clientId,
CancellationToken token = default);
/// <summary>
/// retrieves meta entry
/// </summary>
/// <param name="metaKey"></param>
/// <param name="clientId"></param>
/// <param name="token"></param>
/// <returns></returns>
Task<EFMeta> GetPersistentMeta(string metaKey, int clientId, CancellationToken token = default);
/// <summary>
/// retrieves meta value as deserialized object
/// </summary>
/// <param name="metaKey"></param>
/// <param name="clientId"></param>
/// <param name="token"></param>
/// <typeparam name="T">object type to deserialize into</typeparam>
/// <returns></returns>
Task<T> GetPersistentMetaValue<T>(string metaKey, int clientId, CancellationToken token = default)
where T : class;
/// <summary>
/// retrieves meta entry by with associated lookup value as string
/// </summary>
/// <param name="metaKey"></param>
/// <param name="lookupKey"></param>
/// <param name="clientId"></param>
/// <param name="token"></param>
/// <typeparam name="T"></typeparam>
/// <returns></returns>
Task<EFMeta> GetPersistentMetaByLookup(string metaKey, string lookupKey, int clientId,
CancellationToken token = default);
/// <summary>
/// removes meta key with given value
/// </summary>
/// <param name="metaKey">key of meta data</param>
/// <param name="clientId">client to delete the meta for</param>
/// <param name="token"></param>
/// <returns></returns>
Task RemovePersistentMeta(string metaKey, int clientId, CancellationToken token = default);
#endregion
#region GLOBAL
/// <summary>
/// adds or updates meta key and value to the database
/// </summary>
/// <param name="metaKey">key of meta data</param>
/// <param name="metaValue">value of the meta data</param>
/// <param name="token"></param>
/// <returns></returns>
Task SetPersistentMeta(string metaKey, string metaValue, CancellationToken token = default);
/// <summary>
/// serializes and sets (create or update) meta key and value
/// </summary>
/// <param name="metaKey"></param>
/// <param name="metaValue"></param>
/// <param name="token"></param>
/// <typeparam name="T"></typeparam>
/// <returns></returns>
Task SetPersistentMetaValue<T>(string metaKey, T metaValue, CancellationToken token = default) where T : class;
/// <summary>
/// removes meta key with given value
/// </summary>
/// <param name="metaKey">key of the meta data</param>
/// <param name="metaValue">value of the meta data</param>
/// <param name="token"></param>
/// <returns></returns>
Task RemovePersistentMeta(string metaKey, CancellationToken token = default);
/// <summary>
/// retrieves collection of meta for given key
/// </summary>
/// <param name="metaKey">key to retrieve values for</param>
/// <param name="token"></param>
/// <returns></returns>
Task<EFMeta> GetPersistentMeta(string metaKey, CancellationToken token = default);
/// <summary>
/// returns value of meta key if it exists
/// </summary>
/// <param name="metaKey"></param>
/// <param name="token"></param>
/// <typeparam name="T"></typeparam>
/// <returns></returns>
Task<T> GetPersistentMetaValue<T>(string metaKey, CancellationToken token = default) where T : class;
#endregion
/// <summary>
/// adds a meta task to the runtime meta list
/// </summary>
/// <param name="metaKey">type of meta</param>
/// <param name="metaAction">action to perform</param>
void AddRuntimeMeta<T, TReturn>(MetaType metaKey,
Func<T, CancellationToken, Task<IEnumerable<TReturn>>> metaAction)
where TReturn : IClientMeta where T : PaginationRequest;
/// <summary>
/// retrieves all the runtime meta information for given client idea
/// </summary>
/// <param name="request">request information</param>
/// <param name="token"></param>
/// <returns></returns>
Task<IEnumerable<IClientMeta>> GetRuntimeMeta(ClientPaginationRequest request, CancellationToken token = default);
/// <summary>
/// retrieves all the runtime of provided type
/// </summary>
/// <param name="request">>request information</param>
/// <param name="metaType">type of meta to retreive</param>
/// <param name="token"></param>
/// <returns></returns>
Task<IEnumerable<T>> GetRuntimeMeta<T>(ClientPaginationRequest request, MetaType metaType, CancellationToken token = default)
where T : IClientMeta;
}

View File

@ -1,4 +1,5 @@
using System.Threading.Tasks; using System.Threading;
using System.Threading.Tasks;
using SharedLibraryCore.RCon; using SharedLibraryCore.RCon;
namespace SharedLibraryCore.Interfaces namespace SharedLibraryCore.Interfaces
@ -14,7 +15,7 @@ namespace SharedLibraryCore.Interfaces
/// <param name="type">type of RCon query to perform</param> /// <param name="type">type of RCon query to perform</param>
/// <param name="parameters">optional parameter list</param> /// <param name="parameters">optional parameter list</param>
/// <returns></returns> /// <returns></returns>
Task<string[]> SendQueryAsync(StaticHelpers.QueryType type, string parameters = ""); Task<string[]> SendQueryAsync(StaticHelpers.QueryType type, string parameters = "", CancellationToken token = default);
/// <summary> /// <summary>
/// sets the rcon parser /// sets the rcon parser
@ -22,4 +23,4 @@ namespace SharedLibraryCore.Interfaces
/// <param name="config">parser</param> /// <param name="config">parser</param>
void SetConfiguration(IRConParser config); void SetConfiguration(IRConParser config);
} }
} }

View File

@ -1,4 +1,5 @@
using System; using System;
using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using static SharedLibraryCore.Server; using static SharedLibraryCore.Server;
@ -52,7 +53,7 @@ namespace SharedLibraryCore.Interfaces
/// <param name="dvarName">name of DVAR</param> /// <param name="dvarName">name of DVAR</param>
/// <param name="fallbackValue">default value to return if dvar retrieval fails</param> /// <param name="fallbackValue">default value to return if dvar retrieval fails</param>
/// <returns></returns> /// <returns></returns>
Task<Dvar<T>> GetDvarAsync<T>(IRConConnection connection, string dvarName, T fallbackValue = default); Task<Dvar<T>> GetDvarAsync<T>(IRConConnection connection, string dvarName, T fallbackValue = default, CancellationToken token = default);
/// <summary> /// <summary>
/// set value of DVAR by name /// set value of DVAR by name
@ -61,7 +62,7 @@ namespace SharedLibraryCore.Interfaces
/// <param name="dvarName">name of DVAR to set</param> /// <param name="dvarName">name of DVAR to set</param>
/// <param name="dvarValue">value to set DVAR to</param> /// <param name="dvarValue">value to set DVAR to</param>
/// <returns></returns> /// <returns></returns>
Task<bool> SetDvarAsync(IRConConnection connection, string dvarName, object dvarValue); Task<bool> SetDvarAsync(IRConConnection connection, string dvarName, object dvarValue, CancellationToken token = default);
/// <summary> /// <summary>
/// executes a console command on the server /// executes a console command on the server
@ -69,8 +70,8 @@ namespace SharedLibraryCore.Interfaces
/// <param name="connection">RCon connection to use</param> /// <param name="connection">RCon connection to use</param>
/// <param name="command">console command to execute</param> /// <param name="command">console command to execute</param>
/// <returns></returns> /// <returns></returns>
Task<string[]> ExecuteCommandAsync(IRConConnection connection, string command); Task<string[]> ExecuteCommandAsync(IRConConnection connection, string command, CancellationToken token = default);
/// <summary> /// <summary>
/// get the list of connected clients from status response /// get the list of connected clients from status response
/// </summary> /// </summary>
@ -78,7 +79,7 @@ namespace SharedLibraryCore.Interfaces
/// <returns> /// <returns>
/// <see cref="IStatusResponse" /> /// <see cref="IStatusResponse" />
/// </returns> /// </returns>
Task<IStatusResponse> GetStatusAsync(IRConConnection connection); Task<IStatusResponse> GetStatusAsync(IRConConnection connection, CancellationToken token = default);
/// <summary> /// <summary>
/// retrieves the value of given dvar key if it exists in the override dict /// retrieves the value of given dvar key if it exists in the override dict
@ -103,4 +104,4 @@ namespace SharedLibraryCore.Interfaces
/// <returns></returns> /// <returns></returns>
TimeSpan OverrideTimeoutForCommand(string command); TimeSpan OverrideTimeoutForCommand(string command);
} }
} }

View File

@ -39,6 +39,14 @@ namespace SharedLibraryCore.Database.Models
Disconnecting Disconnecting
} }
public enum TeamType
{
Unknown,
Spectator,
Allies,
Axis
}
[NotMapped] private readonly SemaphoreSlim _processingEvent; [NotMapped] private readonly SemaphoreSlim _processingEvent;
public EFClient() public EFClient()
@ -101,6 +109,9 @@ namespace SharedLibraryCore.Database.Models
[NotMapped] public string GuidString => NetworkId.ToString("x"); [NotMapped] public string GuidString => NetworkId.ToString("x");
[NotMapped] public ClientState State { get; set; } [NotMapped] public ClientState State { get; set; }
[NotMapped] public TeamType Team { get; set; }
[NotMapped] public string TeamName { get; set; }
[NotMapped] [NotMapped]
// this is kinda dirty, but I need localizable level names // this is kinda dirty, but I need localizable level names
@ -113,8 +124,8 @@ namespace SharedLibraryCore.Database.Models
[NotMapped] [NotMapped]
public string Tag public string Tag
{ {
get => GetAdditionalProperty<string>(EFMeta.ClientTag); get => GetAdditionalProperty<string>(EFMeta.ClientTagV2);
set => SetAdditionalProperty(EFMeta.ClientTag, value); set => SetAdditionalProperty(EFMeta.ClientTagV2, value);
} }
[NotMapped] [NotMapped]
@ -166,6 +177,7 @@ namespace SharedLibraryCore.Database.Models
return e; return e;
} }
[Obsolete("Use TellAsync")]
public void Tell(IEnumerable<string> messages) public void Tell(IEnumerable<string> messages)
{ {
foreach (var message in messages) foreach (var message in messages)
@ -176,6 +188,19 @@ namespace SharedLibraryCore.Database.Models
} }
} }
public async Task TellAsync(IEnumerable<string> messages, CancellationToken token = default)
{
foreach (var message in messages)
{
if (token.IsCancellationRequested)
{
return;
}
await Tell(message).WaitAsync(Utilities.DefaultCommandTimeout, token);
}
}
/// <summary> /// <summary>
/// warn a client with given reason /// warn a client with given reason
/// </summary> /// </summary>
@ -653,7 +678,7 @@ namespace SharedLibraryCore.Database.Models
// we want to get any penalties that are tied to their IP or AliasLink (but not necessarily their GUID) // we want to get any penalties that are tied to their IP or AliasLink (but not necessarily their GUID)
var activePenalties = await CurrentServer.Manager.GetPenaltyService() var activePenalties = await CurrentServer.Manager.GetPenaltyService()
.GetActivePenaltiesAsync(AliasLinkId, CurrentAliasId, ipAddress); .GetActivePenaltiesAsync(AliasLinkId, CurrentAliasId, NetworkId, ipAddress);
var banPenalty = activePenalties.FirstOrDefault(_penalty => _penalty.Type == EFPenalty.PenaltyType.Ban); var banPenalty = activePenalties.FirstOrDefault(_penalty => _penalty.Type == EFPenalty.PenaltyType.Ban);
var tempbanPenalty = var tempbanPenalty =
activePenalties.FirstOrDefault(_penalty => _penalty.Type == EFPenalty.PenaltyType.TempBan); activePenalties.FirstOrDefault(_penalty => _penalty.Type == EFPenalty.PenaltyType.TempBan);
@ -710,6 +735,17 @@ namespace SharedLibraryCore.Database.Models
return true; return true;
} }
public void UpdateTeam(string newTeam)
{
if (string.IsNullOrEmpty(newTeam))
{
return;
}
Team = Enum.TryParse(newTeam, true, out TeamType team) ? team : TeamType.Unknown;
TeamName = newTeam;
}
public async Task Lock() public async Task Lock()
{ {
var result = await _processingEvent.WaitAsync(Utilities.DefaultCommandTimeout); var result = await _processingEvent.WaitAsync(Utilities.DefaultCommandTimeout);

View File

@ -1,5 +1,4 @@
using System; using System;
using System.Globalization;
namespace SharedLibraryCore.RCon namespace SharedLibraryCore.RCon
{ {
@ -48,16 +47,6 @@ namespace SharedLibraryCore.RCon
COMMAND_STATUS COMMAND_STATUS
} }
/// <summary>
/// line seperator char included in response from the server
/// </summary>
public static char SeperatorChar = (char)int.Parse("0a", NumberStyles.AllowHexSpecifier);
/// <summary>
/// interval in milliseconds to wait before sending the next RCon request
/// </summary>
public static readonly int FloodProtectionInterval = 750;
/// <summary> /// <summary>
/// timeout in seconds to wait for a socket send or receive before giving up /// timeout in seconds to wait for a socket send or receive before giving up
/// </summary> /// </summary>
@ -73,4 +62,4 @@ namespace SharedLibraryCore.RCon
}; };
} }
} }
} }

View File

@ -238,6 +238,7 @@ namespace SharedLibraryCore
return e; return e;
} }
[Obsolete("Use BroadcastAsync")]
public void Broadcast(IEnumerable<string> messages, EFClient sender = null) public void Broadcast(IEnumerable<string> messages, EFClient sender = null)
{ {
foreach (var message in messages) foreach (var message in messages)
@ -248,6 +249,19 @@ namespace SharedLibraryCore
} }
} }
public async Task BroadcastAsync(IEnumerable<string> messages, EFClient sender = null,
CancellationToken token = default)
{
foreach (var message in messages)
{
if (Manager.CancellationToken.IsCancellationRequested)
{
return;
}
await Broadcast(message, sender).WaitAsync(Utilities.DefaultCommandTimeout, Manager.CancellationToken);
}
}
/// <summary> /// <summary>
/// Send a message to a particular players /// Send a message to a particular players
@ -377,7 +391,7 @@ namespace SharedLibraryCore
{ {
try try
{ {
return (await this.GetDvarAsync("sv_customcallbacks", "0")).Value == "1"; return (await this.GetDvarAsync("sv_customcallbacks", "0", Manager.CancellationToken)).Value == "1";
} }
catch (DvarException) catch (DvarException)
@ -388,19 +402,48 @@ namespace SharedLibraryCore
public abstract Task<long> GetIdForServer(Server server = null); public abstract Task<long> GetIdForServer(Server server = null);
public string[] ExecuteServerCommand(string command) public string[] ExecuteServerCommand(string command, int timeoutMs = 1000)
{ {
return this.ExecuteCommandAsync(command).GetAwaiter().GetResult(); var tokenSource = new CancellationTokenSource();
tokenSource.CancelAfter(TimeSpan.FromSeconds(timeoutMs));
try
{
return this.ExecuteCommandAsync(command, tokenSource.Token).GetAwaiter().GetResult();
}
catch
{
return null;
}
} }
public string GetServerDvar(string dvarName) public string GetServerDvar(string dvarName, int timeoutMs = 1000)
{ {
return this.GetDvarAsync<string>(dvarName).GetAwaiter().GetResult()?.Value; var tokenSource = new CancellationTokenSource();
tokenSource.CancelAfter(TimeSpan.FromSeconds(timeoutMs));
try
{
return this.GetDvarAsync<string>(dvarName, token: tokenSource.Token).GetAwaiter().GetResult().Value;
}
catch
{
return null;
}
} }
public void SetServerDvar(string dvarName, string dvarValue) public bool SetServerDvar(string dvarName, string dvarValue, int timeoutMs = 1000)
{ {
this.SetDvarAsync(dvarName, dvarValue).GetAwaiter().GetResult(); var tokenSource = new CancellationTokenSource();
tokenSource.CancelAfter(TimeSpan.FromSeconds(timeoutMs));
try
{
this.SetDvarAsync(dvarName, dvarValue, tokenSource.Token).GetAwaiter().GetResult();
return true;
}
catch
{
return false;
}
} }
public EFClient GetClientByNumber(int clientNumber) => public EFClient GetClientByNumber(int clientNumber) =>

View File

@ -2,6 +2,7 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Globalization; using System.Globalization;
using System.Linq; using System.Linq;
using System.Text.RegularExpressions;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using Data.Abstractions; using Data.Abstractions;
@ -464,7 +465,7 @@ namespace SharedLibraryCore.Services
// link2 is deleted // link2 is deleted
// link2 penalties are orphaned // link2 penalties are orphaned
await context.Penalties await context.Penalties
.Where(_penalty => completeAliasLinkIds.Contains(_penalty.LinkId)) .Where(_penalty => completeAliasLinkIds.Contains(_penalty.LinkId ?? -1))
.ForEachAsync(_penalty => _penalty.LinkId = newAliasLink.AliasLinkId); .ForEachAsync(_penalty => _penalty.LinkId = newAliasLink.AliasLinkId);
entity.AliasLink = newAliasLink; entity.AliasLink = newAliasLink;
@ -596,51 +597,16 @@ namespace SharedLibraryCore.Services
/// <returns></returns> /// <returns></returns>
public virtual async Task UpdateLevel(Permission newPermission, EFClient temporalClient, EFClient origin) public virtual async Task UpdateLevel(Permission newPermission, EFClient temporalClient, EFClient origin)
{ {
await using var ctx = _contextFactory.CreateContext(); await using var context = _contextFactory.CreateContext();
var entity = await ctx.Clients var entity = await context.Clients
.Where(_client => _client.ClientId == temporalClient.ClientId) .Where(client => client.ClientId == temporalClient.ClientId)
.FirstAsync(); .FirstAsync();
var oldPermission = entity.Level; _logger.LogInformation("Updating {ClientId} from {OldPermission} to {NewPermission} ",
temporalClient.ClientId, entity.Level, newPermission);
entity.Level = newPermission; entity.Level = newPermission;
await ctx.SaveChangesAsync(); await context.SaveChangesAsync();
using (LogContext.PushProperty("Server", temporalClient?.CurrentServer?.ToString()))
{
_logger.LogInformation("Updated {clientId} to {newPermission}", temporalClient.ClientId, newPermission);
var linkedPermissionSet = new[] { Permission.Banned, Permission.Flagged };
// if their permission level has been changed to level that needs to be updated on all accounts
if (linkedPermissionSet.Contains(newPermission) || linkedPermissionSet.Contains(oldPermission))
{
//get all clients that have the same linkId
var iqMatchingClients = ctx.Clients
.Where(_client => _client.AliasLinkId == entity.AliasLinkId);
var iqLinkClients = new List<Data.Models.Client.EFClient>().AsQueryable();
if (!_appConfig.EnableImplicitAccountLinking)
{
var linkIds = await ctx.Aliases.Where(alias =>
alias.IPAddress != null && alias.IPAddress == temporalClient.IPAddress)
.Select(alias => alias.LinkId)
.ToListAsync();
iqLinkClients = ctx.Clients.Where(client => linkIds.Contains(client.AliasLinkId));
}
// this updates the level for all the clients with the same LinkId
// only if their new level is flagged or banned
await iqMatchingClients.Union(iqLinkClients).ForEachAsync(_client =>
{
_client.Level = newPermission;
_logger.LogInformation("Updated linked {clientId} to {newPermission}", _client.ClientId,
newPermission);
});
await ctx.SaveChangesAsync();
}
}
temporalClient.Level = newPermission; temporalClient.Level = newPermission;
} }
@ -839,13 +805,14 @@ namespace SharedLibraryCore.Services
} }
catch catch
{ {
// ignored
} }
var ipAddress = trimmedIdentifier.ConvertToIP(); var ipAddress = trimmedIdentifier.ConvertToIP();
var iqLinkIds = context.Aliases.Where(_alias => _alias.Active); var iqLinkIds = context.Aliases.Where(_alias => _alias.Active);
// we want to query for the IP ADdress // we want to query for the IP Address
if (ipAddress != null) if (ipAddress != null)
{ {
iqLinkIds = iqLinkIds.Where(_alias => _alias.IPAddress == ipAddress); iqLinkIds = iqLinkIds.Where(_alias => _alias.IPAddress == ipAddress);
@ -866,12 +833,22 @@ namespace SharedLibraryCore.Services
var iqClients = context.Clients var iqClients = context.Clients
.Where(_client => _client.Active); .Where(_client => _client.Active);
var match = Regex.Match(trimmedIdentifier ?? "", "\"(.+)\"");
iqClients = iqClients.Where(_client => networkId == _client.NetworkId || if (match.Success)
linkIds.Contains(_client.AliasLinkId) {
|| !_appConfig.EnableImplicitAccountLinking && iqClients = iqClients.Where(client =>
_client.CurrentAlias.IPAddress != null && client.CurrentAlias.SearchableName.ToLower().Equals(match.Groups[1].ToString().ToLower()));
_client.CurrentAlias.IPAddress == ipAddress); }
else
{
iqClients = iqClients.Where(_client => networkId == _client.NetworkId ||
linkIds.Contains(_client.AliasLinkId)
|| !_appConfig.EnableImplicitAccountLinking &&
_client.CurrentAlias.IPAddress != null &&
_client.CurrentAlias.IPAddress == ipAddress);
}
// we want to project our results // we want to project our results
var iqClientProjection = iqClients.OrderByDescending(_client => _client.LastConnection) var iqClientProjection = iqClients.OrderByDescending(_client => _client.LastConnection)

View File

@ -31,7 +31,7 @@ namespace SharedLibraryCore.Services
Active = true, Active = true,
OffenderId = newEntity.Offender.ClientId, OffenderId = newEntity.Offender.ClientId,
PunisherId = newEntity.Punisher.ClientId, PunisherId = newEntity.Punisher.ClientId,
LinkId = newEntity.Link.AliasLinkId, LinkId = newEntity.Link?.AliasLinkId,
Type = newEntity.Type, Type = newEntity.Type,
Expires = newEntity.Expires, Expires = newEntity.Expires,
Offense = newEntity.Offense, Offense = newEntity.Offense,
@ -40,6 +40,18 @@ namespace SharedLibraryCore.Services
newEntity.Punisher.AdministeredPenalties?.FirstOrDefault()?.AutomatedOffense, newEntity.Punisher.AdministeredPenalties?.FirstOrDefault()?.AutomatedOffense,
IsEvadedOffense = newEntity.IsEvadedOffense IsEvadedOffense = newEntity.IsEvadedOffense
}; };
if (LinkedPenalties.Contains(newEntity.Type))
{
var penaltyIdentifiers = new EFPenaltyIdentifier
{
Penalty = penalty,
NetworkId = newEntity.Offender.NetworkId,
IPv4Address = newEntity.Offender.CurrentAlias.IPAddress
};
context.PenaltyIdentifiers.Add(penaltyIdentifiers);
}
context.Penalties.Add(penalty); context.Penalties.Add(penalty);
await context.SaveChangesAsync(); await context.SaveChangesAsync();
@ -103,64 +115,25 @@ namespace SharedLibraryCore.Services
return await iqPenalties.ToListAsync(); return await iqPenalties.ToListAsync();
} }
/// <summary> private static readonly EFPenalty.PenaltyType[] LinkedPenalties =
/// retrieves penalty information for meta service { EFPenalty.PenaltyType.Ban, EFPenalty.PenaltyType.Flag, EFPenalty.PenaltyType.TempBan };
/// </summary>
/// <param name="clientId">database id of the client</param> private static readonly Expression<Func<EFPenalty, bool>> Filter = p =>
/// <param name="count">how many items to retrieve</param> LinkedPenalties.Contains(p.Type) && p.Active && (p.Expires == null || p.Expires > DateTime.UtcNow);
/// <param name="offset">not used</param>
/// <param name="startAt">retreive penalties older than this</param> private static readonly Expression<Func<EFPenaltyIdentifier, bool>> FilterById = pi =>
/// <returns></returns> LinkedPenalties.Contains(pi.Penalty.Type) && pi.Penalty.Active &&
public async Task<IList<PenaltyInfo>> GetClientPenaltyForMetaAsync(int clientId, int count, int offset, (pi.Penalty.Expires == null || pi.Penalty.Expires > DateTime.UtcNow);
DateTime? startAt)
public async Task<List<EFPenalty>> GetActivePenaltiesAsync(int linkId, int currentAliasId, long networkId,
int? ip = null)
{ {
var linkedPenaltyType = Utilities.LinkedPenaltyTypes(); var penaltiesByIdentifier = await GetActivePenaltiesByIdentifier(ip, networkId);
await using var context = _contextFactory.CreateContext(false); if (penaltiesByIdentifier.Any())
var linkId = await context.Clients.AsNoTracking() {
.Where(_penalty => _penalty.ClientId == clientId) return penaltiesByIdentifier;
.Select(_penalty => _penalty.AliasLinkId) }
.FirstOrDefaultAsync();
var iqPenalties = context.Penalties.AsNoTracking()
.Where(_penalty => _penalty.OffenderId == clientId || _penalty.PunisherId == clientId ||
linkedPenaltyType.Contains(_penalty.Type) && _penalty.LinkId == linkId)
.Where(_penalty => _penalty.When <= startAt)
.OrderByDescending(_penalty => _penalty.When)
.Skip(offset)
.Take(count)
.Select(_penalty => new PenaltyInfo
{
Id = _penalty.PenaltyId,
Offense = _penalty.Offense,
AutomatedOffense = _penalty.AutomatedOffense,
OffenderId = _penalty.OffenderId,
OffenderName = _penalty.Offender.CurrentAlias.Name,
PunisherId = _penalty.PunisherId,
PunisherName = _penalty.Punisher.CurrentAlias.Name,
PunisherLevel = _penalty.Punisher.Level,
PenaltyType = _penalty.Type,
Expires = _penalty.Expires,
TimePunished = _penalty.When,
IsEvade = _penalty.IsEvadedOffense
});
return await iqPenalties.Distinct().ToListAsync();
}
public async Task<List<EFPenalty>> GetActivePenaltiesAsync(int linkId, int currentAliasId, int? ip = null,
bool includePunisherName = false)
{
var now = DateTime.UtcNow;
Expression<Func<EFPenalty, bool>> filter = p => new[]
{
EFPenalty.PenaltyType.TempBan,
EFPenalty.PenaltyType.Ban,
EFPenalty.PenaltyType.Flag
}.Contains(p.Type) &&
p.Active &&
(p.Expires == null || p.Expires > now);
await using var context = _contextFactory.CreateContext(false); await using var context = _contextFactory.CreateContext(false);
@ -171,12 +144,13 @@ namespace SharedLibraryCore.Services
iqIpPenalties = context.Aliases iqIpPenalties = context.Aliases
.Where(a => a.IPAddress != null && a.IPAddress == ip) .Where(a => a.IPAddress != null && a.IPAddress == ip)
.SelectMany(a => a.Link.ReceivedPenalties) .SelectMany(a => a.Link.ReceivedPenalties)
.Where(filter); .Where(Filter);
} }
else else
{ {
var usedIps = await context.Aliases.AsNoTracking() var usedIps = await context.Aliases.AsNoTracking()
.Where(alias => (alias.LinkId == linkId || alias.AliasId == currentAliasId) && alias.IPAddress != null) .Where(alias =>
(alias.LinkId == linkId || alias.AliasId == currentAliasId) && alias.IPAddress != null)
.Select(alias => alias.IPAddress).ToListAsync(); .Select(alias => alias.IPAddress).ToListAsync();
var aliasedIds = await context.Aliases.AsNoTracking().Where(alias => usedIps.Contains(alias.IPAddress)) var aliasedIds = await context.Aliases.AsNoTracking().Where(alias => usedIps.Contains(alias.IPAddress))
@ -184,8 +158,8 @@ namespace SharedLibraryCore.Services
.ToListAsync(); .ToListAsync();
iqIpPenalties = context.Penalties.AsNoTracking() iqIpPenalties = context.Penalties.AsNoTracking()
.Where(penalty => aliasedIds.Contains(penalty.LinkId) || penalty.LinkId == linkId) .Where(penalty => aliasedIds.Contains(penalty.LinkId ?? -1) || penalty.LinkId == linkId)
.Where(filter); .Where(Filter);
} }
var activeIpPenalties = await iqIpPenalties.ToListAsync(); var activeIpPenalties = await iqIpPenalties.ToListAsync();
@ -195,11 +169,35 @@ namespace SharedLibraryCore.Services
return activePenalties.OrderByDescending(p => p.When).ToList(); return activePenalties.OrderByDescending(p => p.When).ToList();
} }
public virtual async Task RemoveActivePenalties(int aliasLinkId, int? ipAddress = null) public async Task<List<EFPenalty>> GetActivePenaltiesByIdentifier(int? ip, long networkId)
{
await using var context = _contextFactory.CreateContext(false);
var activePenaltiesIds = context.PenaltyIdentifiers.Where(identifier =>
identifier.IPv4Address != null && identifier.IPv4Address == ip || identifier.NetworkId == networkId)
.Where(FilterById);
return await activePenaltiesIds.Select(ids => ids.Penalty).ToListAsync();
}
public virtual async Task RemoveActivePenalties(int aliasLinkId, long networkId, int? ipAddress = null)
{ {
await using var context = _contextFactory.CreateContext(); await using var context = _contextFactory.CreateContext();
var now = DateTime.UtcNow; var now = DateTime.UtcNow;
var activePenalties = await GetActivePenaltiesByIdentifier(ipAddress, networkId);
if (activePenalties.Any())
{
var ids = activePenalties.Select(penalty => penalty.PenaltyId);
await context.Penalties.Where(penalty => ids.Contains(penalty.PenaltyId))
.ForEachAsync(penalty =>
{
penalty.Active = false;
penalty.Expires = now;
});
await context.SaveChangesAsync();
return;
}
var penaltiesByLink = context.Penalties var penaltiesByLink = context.Penalties
.Where(p => p.LinkId == aliasLinkId) .Where(p => p.LinkId == aliasLinkId)
.Where(p => p.Expires > now || p.Expires == null); .Where(p => p.Expires > now || p.Expires == null);

View File

@ -4,7 +4,7 @@
<OutputType>Library</OutputType> <OutputType>Library</OutputType>
<TargetFramework>net6.0</TargetFramework> <TargetFramework>net6.0</TargetFramework>
<PackageId>RaidMax.IW4MAdmin.SharedLibraryCore</PackageId> <PackageId>RaidMax.IW4MAdmin.SharedLibraryCore</PackageId>
<Version>2022.01.28.1</Version> <Version>2022.3.23.1</Version>
<Authors>RaidMax</Authors> <Authors>RaidMax</Authors>
<Company>Forever None</Company> <Company>Forever None</Company>
<Configurations>Debug;Release;Prerelease</Configurations> <Configurations>Debug;Release;Prerelease</Configurations>
@ -19,7 +19,9 @@
<IsPackable>true</IsPackable> <IsPackable>true</IsPackable>
<PackageLicenseExpression>MIT</PackageLicenseExpression> <PackageLicenseExpression>MIT</PackageLicenseExpression>
<Description>Shared Library for IW4MAdmin</Description> <Description>Shared Library for IW4MAdmin</Description>
<PackageVersion>2022.01.28.1</PackageVersion> <PackageVersion>2022.3.23.1</PackageVersion>
<GenerateDocumentationFile>true</GenerateDocumentationFile>
<NoWarn>$(NoWarn);1591</NoWarn>
</PropertyGroup> </PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Prerelease|AnyCPU'"> <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Prerelease|AnyCPU'">
@ -27,6 +29,10 @@
<DebugSymbols>true</DebugSymbols> <DebugSymbols>true</DebugSymbols>
</PropertyGroup> </PropertyGroup>
<PropertyGroup>
<DocumentationFile>bin\$(Configuration)\$(TargetFramework)\$(AssemblyName).xml</DocumentationFile>
</PropertyGroup>
<ItemGroup> <ItemGroup>
<PackageReference Include="FluentValidation" Version="10.3.6" /> <PackageReference Include="FluentValidation" Version="10.3.6" />
<PackageReference Include="Humanizer.Core" Version="2.13.14" /> <PackageReference Include="Humanizer.Core" Version="2.13.14" />

View File

@ -42,7 +42,8 @@ namespace SharedLibraryCore
#endif #endif
public static Encoding EncodingType; public static Encoding EncodingType;
public static Layout CurrentLocalization = new Layout(new Dictionary<string, string>()); public static Layout CurrentLocalization = new Layout(new Dictionary<string, string>());
public static TimeSpan DefaultCommandTimeout { get; set; } = new TimeSpan(0, 0, 25);
public static TimeSpan DefaultCommandTimeout { get; set; } = new(0, 0, Utilities.IsDevelopment ? 360 : 25);
public static char[] DirectorySeparatorChars = { '\\', '/' }; public static char[] DirectorySeparatorChars = { '\\', '/' };
public static char CommandPrefix { get; set; } = '!'; public static char CommandPrefix { get; set; } = '!';
@ -723,14 +724,21 @@ namespace SharedLibraryCore
.Replace('/', '_'); .Replace('/', '_');
} }
public static Task<Dvar<T>> GetDvarAsync<T>(this Server server, string dvarName, T fallbackValue = default) public static async Task<Dvar<T>> GetDvarAsync<T>(this Server server, string dvarName,
T fallbackValue = default, CancellationToken token = default)
{ {
return server.RconParser.GetDvarAsync(server.RemoteConnection, dvarName, fallbackValue); return await server.RconParser.GetDvarAsync(server.RemoteConnection, dvarName, fallbackValue, token);
}
public static async Task<Dvar<T>> GetDvarAsync<T>(this Server server, string dvarName,
T fallbackValue = default)
{
return await GetDvarAsync(server, dvarName, fallbackValue, default);
} }
public static async Task<Dvar<T>> GetMappedDvarValueOrDefaultAsync<T>(this Server server, string dvarName, public static async Task<Dvar<T>> GetMappedDvarValueOrDefaultAsync<T>(this Server server, string dvarName,
string infoResponseName = null, IDictionary<string, string> infoResponse = null, string infoResponseName = null, IDictionary<string, string> infoResponse = null,
T overrideDefault = default) T overrideDefault = default, CancellationToken token = default)
{ {
// todo: unit test this // todo: unit test this
var mappedKey = server.RconParser.GetOverrideDvarName(dvarName); var mappedKey = server.RconParser.GetOverrideDvarName(dvarName);
@ -749,22 +757,32 @@ namespace SharedLibraryCore
}; };
} }
return await server.GetDvarAsync(mappedKey, defaultValue); return await server.GetDvarAsync(mappedKey, defaultValue, token: token);
} }
public static Task SetDvarAsync(this Server server, string dvarName, object dvarValue) public static async Task SetDvarAsync(this Server server, string dvarName, object dvarValue, CancellationToken token = default)
{ {
return server.RconParser.SetDvarAsync(server.RemoteConnection, dvarName, dvarValue); await server.RconParser.SetDvarAsync(server.RemoteConnection, dvarName, dvarValue, token);
}
public static async Task SetDvarAsync(this Server server, string dvarName, object dvarValue)
{
await SetDvarAsync(server, dvarName, dvarValue, default);
} }
public static async Task<string[]> ExecuteCommandAsync(this Server server, string commandName, CancellationToken token = default)
{
return await server.RconParser.ExecuteCommandAsync(server.RemoteConnection, commandName, token);
}
public static async Task<string[]> ExecuteCommandAsync(this Server server, string commandName) public static async Task<string[]> ExecuteCommandAsync(this Server server, string commandName)
{ {
return await server.RconParser.ExecuteCommandAsync(server.RemoteConnection, commandName); return await ExecuteCommandAsync(server, commandName, default);
} }
public static Task<IStatusResponse> GetStatusAsync(this Server server) public static Task<IStatusResponse> GetStatusAsync(this Server server, CancellationToken token)
{ {
return server.RconParser.GetStatusAsync(server.RemoteConnection); return server.RconParser.GetStatusAsync(server.RemoteConnection, token);
} }
/// <summary> /// <summary>
@ -956,6 +974,19 @@ namespace SharedLibraryCore
} }
} }
public static async Task<T> WithWaitCancellation<T>(this Task<T> task,
CancellationToken cancellationToken)
{
var completedTask = await Task.WhenAny(task, Task.Delay(Timeout.Infinite, cancellationToken));
if (completedTask == task)
{
return await task;
}
cancellationToken.ThrowIfCancellationRequested();
throw new InvalidOperationException("Infinite delay task completed.");
}
public static async Task<T> WithTimeout<T>(this Task<T> task, TimeSpan timeout) public static async Task<T> WithTimeout<T>(this Task<T> task, TimeSpan timeout)
{ {
await Task.WhenAny(task, Task.Delay(timeout)); await Task.WhenAny(task, Task.Delay(timeout));

Some files were not shown because too many files have changed in this diff Show More