Compare commits
218 Commits
2021.11.23
...
2022.06.02
Author | SHA1 | Date | |
---|---|---|---|
a38789adb9 | |||
e459b2fcde | |||
26853a0005 | |||
ee14306db9 | |||
169105e849 | |||
7c10e0e3de | |||
2f7eb07e39 | |||
0f9f4f597b | |||
880f9333d9 | |||
31da5d352e | |||
83a469cae3 | |||
1f13f9122c | |||
dd8c4f438f | |||
2230036d45 | |||
1700b7da91 | |||
fab97ccad4 | |||
0bf0d033f7 | |||
2bbabcb9e8 | |||
1995dbd080 | |||
a3b94b50e3 | |||
bc38b36e4a | |||
e346aa037e | |||
389c687420 | |||
074e36413e | |||
104d9bdc4c | |||
c51d28937b | |||
ffa8a46feb | |||
91c46dbdd4 | |||
ff0d22c142 | |||
3ad4aa2196 | |||
d462892467 | |||
cabedb6f0b | |||
5b7f5160b2 | |||
0a8e415af8 | |||
7b3ddd58c6 | |||
ed1032415e | |||
35b43e7438 | |||
284c2e9726 | |||
fd049edb3f | |||
4884abee76 | |||
1df76b6ac3 | |||
4c42a1d511 | |||
27635a6dd3 | |||
0175425708 | |||
5e12bf60b5 | |||
2f10ca8599 | |||
62ec18309e | |||
87361bf3d7 | |||
dc45136077 | |||
21b0a7998d | |||
20b8f0b99a | |||
314ff96e71 | |||
d7c4f5452c | |||
3cb50635e5 | |||
4fbe0ee0ed | |||
4023ca37d4 | |||
425ec2621d | |||
15c3ca53e2 | |||
6097ca504c | |||
70cd01eafb | |||
a2d5e37c6f | |||
19bd47d0f4 | |||
dc97956bc3 | |||
89fdc00f9b | |||
039a05d9ad | |||
25fb5fdc14 | |||
1e67f6e86c | |||
7dbdf87728 | |||
180a4911bc | |||
31123d9a33 | |||
3cf0f54ceb | |||
eafd7cb530 | |||
770785e979 | |||
92d713d188 | |||
34e531ef8d | |||
724992ef33 | |||
557cc1614f | |||
f90cdbef16 | |||
a863f78678 | |||
c93f896bc5 | |||
ccc8316a2f | |||
497c15a6a8 | |||
7be096e0b6 | |||
20858991e1 | |||
85d44b0eb0 | |||
51ef67ae9c | |||
63b04be4c7 | |||
36eb45bb2e | |||
04a4dcf153 | |||
b46b1eb5e7 | |||
287635fa36 | |||
f567a03fa7 | |||
1b6d8107ae | |||
1e8f06f3a3 | |||
064879fead | |||
e32e97b9e6 | |||
42313b7816 | |||
9f4d06c265 | |||
acf66da4ca | |||
59ca399045 | |||
ef70496546 | |||
e6e56d8d14 | |||
55b0caf900 | |||
a4c3f9c2d1 | |||
241aa0a5f6 | |||
ec0f59cdb1 | |||
59d69bd22b | |||
e9c8ead829 | |||
58d48a211e | |||
edf8e03b04 | |||
de2e804b84 | |||
b087d4c8de | |||
bd6c0dd5be | |||
4ace476242 | |||
bb7215dbb6 | |||
88bd47f3ae | |||
39a1066c74 | |||
18f3c59b9b | |||
0d88b6293f | |||
a6b56ceded | |||
78ef977268 | |||
d527a86911 | |||
2e531c4a50 | |||
45059fcfd9 | |||
482cd9c339 | |||
51667159a2 | |||
ea18a286b2 | |||
9a6d7c6a20 | |||
adcb75319c | |||
037fac5786 | |||
f4b892d8f4 | |||
3640d1df54 | |||
f3c6b10a35 | |||
4dec284b31 | |||
c9cf7be341 | |||
aa6ae0ab8d | |||
12dfd8c558 | |||
07f675eadc | |||
576d7015fa | |||
b1a1aae6c0 | |||
a0f4ceccfe | |||
b7a76cc4a2 | |||
261da918c7 | |||
2ed5e00bcb | |||
6ca94f8da8 | |||
3b532cf1f7 | |||
40966ed74d | |||
45eacabc28 | |||
0b02b7627a | |||
fc3a24ca17 | |||
209cb6cdd0 | |||
cfd4296f5c | |||
b275fbaced | |||
b2a3625288 | |||
0d3e2cb0bc | |||
505a2c4c2d | |||
8730a3fab8 | |||
3539101a40 | |||
7ccdee7d1b | |||
f4b160b735 | |||
73036dc1c7 | |||
6cfcce23cc | |||
8649b0efe9 | |||
f554536b95 | |||
11efc039b5 | |||
916ea4163b | |||
0bed1c728a | |||
7171b3753e | |||
a602e8caed | |||
e4cb3abb20 | |||
686b297d32 | |||
fb11bf54a6 | |||
11d2b0da90 | |||
8bd0337168 | |||
74b565ebae | |||
2b467d6ef9 | |||
e90355307d | |||
d3962989b5 | |||
16831aaccb | |||
032753236b | |||
7fcb2202bd | |||
7910fc73a3 | |||
a8d581eab7 | |||
bd27977b1e | |||
092ca5f9bd | |||
3f0b1b892a | |||
c713fdacb0 | |||
f5854f8d03 | |||
67be4f8e7f | |||
9baad44ab4 | |||
76f5933074 | |||
4cce336fb9 | |||
5d12ff471b | |||
5d7ac7498f | |||
15cb114c15 | |||
e739c91b52 | |||
17c9944eef | |||
4a89744ee9 | |||
66010a2fa2 | |||
ce3119425f | |||
307ff3ddeb | |||
7f2fa390c7 | |||
a88b30562c | |||
08bcd23cbc | |||
072571d341 | |||
35e42516f1 | |||
2210ccea68 | |||
08b93fcc10 | |||
ab05b45016 | |||
825dd6f382 | |||
f99fdac4b0 | |||
f7897763e3 | |||
5b95cdaca8 | |||
c4e0c4c36a | |||
31d0dfc7d3 | |||
8f52714fb7 | |||
e4153e0c2f | |||
8d0c48614f |
1
.gitignore
vendored
1
.gitignore
vendored
@ -224,7 +224,6 @@ bootstrap-custom.min.css
|
|||||||
bootstrap-custom.css
|
bootstrap-custom.css
|
||||||
**/Master/static
|
**/Master/static
|
||||||
**/Master/dev_env
|
**/Master/dev_env
|
||||||
/WebfrontCore/Views/Plugins/*
|
|
||||||
/WebfrontCore/wwwroot/**/dds
|
/WebfrontCore/wwwroot/**/dds
|
||||||
/WebfrontCore/wwwroot/images/radar/*
|
/WebfrontCore/wwwroot/images/radar/*
|
||||||
|
|
||||||
|
@ -22,7 +22,7 @@ namespace IW4MAdmin.Application.API.Master
|
|||||||
public int Uptime { get; set; }
|
public int Uptime { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Specifices the version of the instance
|
/// Specifies the version of the instance
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[JsonProperty("version")]
|
[JsonProperty("version")]
|
||||||
[JsonConverter(typeof(BuildNumberJsonConverter))]
|
[JsonConverter(typeof(BuildNumberJsonConverter))]
|
||||||
@ -33,5 +33,11 @@ namespace IW4MAdmin.Application.API.Master
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
[JsonProperty("servers")]
|
[JsonProperty("servers")]
|
||||||
public List<ApiServer> Servers { get; set; }
|
public List<ApiServer> Servers { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Url IW4MAdmin is listening on
|
||||||
|
/// </summary>
|
||||||
|
[JsonProperty("webfront_url")]
|
||||||
|
public string WebfrontUrl { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<OutputType>Exe</OutputType>
|
<OutputType>Exe</OutputType>
|
||||||
<TargetFramework>netcoreapp3.1</TargetFramework>
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
<MvcRazorExcludeRefAssembliesFromPublish>false</MvcRazorExcludeRefAssembliesFromPublish>
|
<MvcRazorExcludeRefAssembliesFromPublish>false</MvcRazorExcludeRefAssembliesFromPublish>
|
||||||
<PackageId>RaidMax.IW4MAdmin.Application</PackageId>
|
<PackageId>RaidMax.IW4MAdmin.Application</PackageId>
|
||||||
<Version>2020.0.0.0</Version>
|
<Version>2020.0.0.0</Version>
|
||||||
@ -24,14 +24,15 @@
|
|||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Jint" Version="3.0.0-beta-1632" />
|
<PackageReference Include="Jint" Version="3.0.0-beta-2038" />
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="3.1.10">
|
<PackageReference Include="MaxMind.GeoIP2" Version="5.1.0" />
|
||||||
|
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="6.0.1">
|
||||||
<PrivateAssets>all</PrivateAssets>
|
<PrivateAssets>all</PrivateAssets>
|
||||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||||
</PackageReference>
|
</PackageReference>
|
||||||
<PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="3.1.10" />
|
<PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="6.0.0" />
|
||||||
<PackageReference Include="RestEase" Version="1.5.1" />
|
<PackageReference Include="RestEase" Version="1.5.5" />
|
||||||
<PackageReference Include="System.Text.Encoding.CodePages" Version="5.0.0" />
|
<PackageReference Include="System.Text.Encoding.CodePages" Version="6.0.0" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
@ -63,6 +64,9 @@
|
|||||||
<None Update="Configuration\LoggingConfiguration.json">
|
<None Update="Configuration\LoggingConfiguration.json">
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
</None>
|
</None>
|
||||||
|
<None Update="Resources\GeoLite2-Country.mmdb">
|
||||||
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<Target Name="PreBuild" BeforeTargets="PreBuildEvent">
|
<Target Name="PreBuild" BeforeTargets="PreBuildEvent">
|
||||||
|
@ -15,6 +15,7 @@ using System;
|
|||||||
using System.Collections;
|
using System.Collections;
|
||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Collections.Immutable;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
@ -26,6 +27,7 @@ using IW4MAdmin.Application.Migration;
|
|||||||
using Microsoft.Extensions.DependencyInjection;
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using Serilog.Context;
|
using Serilog.Context;
|
||||||
|
using SharedLibraryCore.Formatting;
|
||||||
using static SharedLibraryCore.GameEvent;
|
using static SharedLibraryCore.GameEvent;
|
||||||
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
||||||
using ObsoleteLogger = SharedLibraryCore.Interfaces.ILogger;
|
using ObsoleteLogger = SharedLibraryCore.Interfaces.ILogger;
|
||||||
@ -57,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>>();
|
||||||
@ -79,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)
|
||||||
{
|
{
|
||||||
@ -95,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;
|
||||||
@ -217,6 +217,8 @@ namespace IW4MAdmin.Application
|
|||||||
return _commands;
|
return _commands;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public IReadOnlyList<IManagerCommand> Commands => _commands.ToImmutableList();
|
||||||
|
|
||||||
public async Task UpdateServerStates()
|
public async Task UpdateServerStates()
|
||||||
{
|
{
|
||||||
// store the server hash code and task for it
|
// store the server hash code and task for it
|
||||||
@ -232,13 +234,6 @@ namespace IW4MAdmin.Application
|
|||||||
.Select(ut => ut.Key)
|
.Select(ut => ut.Key)
|
||||||
.ToList();
|
.ToList();
|
||||||
|
|
||||||
// this is to prevent the log reader from starting before the initial
|
|
||||||
// query of players on the server
|
|
||||||
if (serverTasksToRemove.Count > 0)
|
|
||||||
{
|
|
||||||
IsInitialized = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// remove the update tasks as they have completed
|
// remove the update tasks as they have completed
|
||||||
foreach (var serverId in serverTasksToRemove.Where(serverId => runningUpdateTasks.ContainsKey(serverId)))
|
foreach (var serverId in serverTasksToRemove.Where(serverId => runningUpdateTasks.ContainsKey(serverId)))
|
||||||
{
|
{
|
||||||
@ -351,9 +346,9 @@ namespace IW4MAdmin.Application
|
|||||||
// copy over default config if it doesn't exist
|
// copy over default config if it doesn't exist
|
||||||
if (!_appConfig.Servers?.Any() ?? true)
|
if (!_appConfig.Servers?.Any() ?? true)
|
||||||
{
|
{
|
||||||
var defaultConfig = new BaseConfigurationHandler<DefaultSettings>("DefaultSettings").Configuration();
|
var defaultHandler = new BaseConfigurationHandler<DefaultSettings>("DefaultSettings");
|
||||||
//ConfigHandler.Set((ApplicationConfiguration)new ApplicationConfiguration().Generate());
|
await defaultHandler.BuildAsync();
|
||||||
//var newConfig = ConfigHandler.Configuration();
|
var defaultConfig = defaultHandler.Configuration();
|
||||||
|
|
||||||
_appConfig.AutoMessages = defaultConfig.AutoMessages;
|
_appConfig.AutoMessages = defaultConfig.AutoMessages;
|
||||||
_appConfig.GlobalRules = defaultConfig.GlobalRules;
|
_appConfig.GlobalRules = defaultConfig.GlobalRules;
|
||||||
@ -415,7 +410,7 @@ namespace IW4MAdmin.Application
|
|||||||
|
|
||||||
if (!validationResult.IsValid)
|
if (!validationResult.IsValid)
|
||||||
{
|
{
|
||||||
throw new ConfigurationException("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
|
||||||
@ -452,6 +447,17 @@ namespace IW4MAdmin.Application
|
|||||||
Encoding.RegisterProvider(CodePagesEncodingProvider.Instance);
|
Encoding.RegisterProvider(CodePagesEncodingProvider.Instance);
|
||||||
Utilities.EncodingType = Encoding.GetEncoding(!string.IsNullOrEmpty(_appConfig.CustomParserEncoding) ? _appConfig.CustomParserEncoding : "windows-1252");
|
Utilities.EncodingType = Encoding.GetEncoding(!string.IsNullOrEmpty(_appConfig.CustomParserEncoding) ? _appConfig.CustomParserEncoding : "windows-1252");
|
||||||
|
|
||||||
|
foreach (var parser in AdditionalRConParsers)
|
||||||
|
{
|
||||||
|
if (!parser.Configuration.ColorCodeMapping.ContainsKey(ColorCodes.Accent.ToString()))
|
||||||
|
{
|
||||||
|
parser.Configuration.ColorCodeMapping.Add(ColorCodes.Accent.ToString(),
|
||||||
|
parser.Configuration.ColorCodeMapping.TryGetValue(_appConfig.IngameAccentColorKey, out var colorCode)
|
||||||
|
? colorCode
|
||||||
|
: "");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region COMMANDS
|
#region COMMANDS
|
||||||
@ -477,13 +483,17 @@ namespace IW4MAdmin.Application
|
|||||||
|
|
||||||
// this is because I want to store the command prefix in IW4MAdminSettings, but can't easily
|
// this is because I want to store the command prefix in IW4MAdminSettings, but can't easily
|
||||||
// inject it to all the places that need it
|
// inject it to all the places that need it
|
||||||
cmdConfig.CommandPrefix = _appConfig.CommandPrefix;
|
cmdConfig.CommandPrefix = _appConfig?.CommandPrefix ?? "!";
|
||||||
cmdConfig.BroadcastCommandPrefix = _appConfig.BroadcastCommandPrefix;
|
cmdConfig.BroadcastCommandPrefix = _appConfig?.BroadcastCommandPrefix ?? "@";
|
||||||
|
|
||||||
foreach (var cmd in commandsToAddToConfig)
|
foreach (var cmd in commandsToAddToConfig)
|
||||||
{
|
{
|
||||||
|
if (cmdConfig.Commands.ContainsKey(cmd.CommandConfigNameForType()))
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
cmdConfig.Commands.Add(cmd.CommandConfigNameForType(),
|
cmdConfig.Commands.Add(cmd.CommandConfigNameForType(),
|
||||||
new CommandProperties()
|
new CommandProperties
|
||||||
{
|
{
|
||||||
Name = cmd.Name,
|
Name = cmd.Name,
|
||||||
Alias = cmd.Alias,
|
Alias = cmd.Alias,
|
||||||
@ -511,6 +521,7 @@ namespace IW4MAdmin.Application
|
|||||||
|
|
||||||
Console.WriteLine(_translationLookup["MANAGER_COMMUNICATION_INFO"]);
|
Console.WriteLine(_translationLookup["MANAGER_COMMUNICATION_INFO"]);
|
||||||
await InitializeServers();
|
await InitializeServers();
|
||||||
|
IsInitialized = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task InitializeServers()
|
private async Task InitializeServers()
|
||||||
@ -532,7 +543,7 @@ namespace IW4MAdmin.Application
|
|||||||
|
|
||||||
_servers.Add(ServerInstance);
|
_servers.Add(ServerInstance);
|
||||||
Console.WriteLine(Utilities.CurrentLocalization.LocalizationIndex["MANAGER_MONITORING_TEXT"].FormatExt(ServerInstance.Hostname.StripColors()));
|
Console.WriteLine(Utilities.CurrentLocalization.LocalizationIndex["MANAGER_MONITORING_TEXT"].FormatExt(ServerInstance.Hostname.StripColors()));
|
||||||
_logger.LogInformation("Finishing initialization and now monitoring [{server}]", ServerInstance.Hostname, ServerInstance.ToString());
|
_logger.LogInformation("Finishing initialization and now monitoring [{Server}]", ServerInstance.Hostname);
|
||||||
}
|
}
|
||||||
|
|
||||||
// add the start event for this server
|
// add the start event for this server
|
||||||
@ -576,16 +587,29 @@ namespace IW4MAdmin.Application
|
|||||||
|
|
||||||
public async Task Start() => await UpdateServerStates();
|
public async Task Start() => await UpdateServerStates();
|
||||||
|
|
||||||
public void Stop()
|
public async Task Stop()
|
||||||
{
|
{
|
||||||
|
foreach (var plugin in Plugins)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await plugin.OnUnloadAsync().WithTimeout(Utilities.DefaultCommandTimeout);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Could not cleanly unload plugin {PluginName}", plugin.Name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
_tokenSource.Cancel();
|
_tokenSource.Cancel();
|
||||||
|
|
||||||
IsRunning = false;
|
IsRunning = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Restart()
|
public void Restart()
|
||||||
{
|
{
|
||||||
IsRestartRequested = true;
|
IsRestartRequested = true;
|
||||||
Stop();
|
Stop().GetAwaiter().GetResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
[Obsolete]
|
[Obsolete]
|
||||||
@ -605,6 +629,11 @@ namespace IW4MAdmin.Application
|
|||||||
return _servers.SelectMany(s => s.Clients).ToList().Where(p => p != null).ToList();
|
return _servers.SelectMany(s => s.Clients).ToList().Where(p => p != null).ToList();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public EFClient FindActiveClient(EFClient client) =>client.ClientNumber < 0 ?
|
||||||
|
GetActiveClients()
|
||||||
|
.FirstOrDefault(c => c.NetworkId == client.NetworkId) ?? client :
|
||||||
|
client;
|
||||||
|
|
||||||
public ClientService GetClientService()
|
public ClientService GetClientService()
|
||||||
{
|
{
|
||||||
return ClientSvc;
|
return ClientSvc;
|
||||||
|
@ -14,3 +14,4 @@ if not exist "%TargetDir%Plugins" (
|
|||||||
)
|
)
|
||||||
|
|
||||||
xcopy /y "%SolutionDir%Build\Plugins" "%TargetDir%Plugins\"
|
xcopy /y "%SolutionDir%Build\Plugins" "%TargetDir%Plugins\"
|
||||||
|
del "%TargetDir%Plugins\SQLite*"
|
||||||
|
@ -23,6 +23,7 @@ echo setting up default folders
|
|||||||
if not exist "%PublishDir%\Configuration" md "%PublishDir%\Configuration"
|
if not exist "%PublishDir%\Configuration" md "%PublishDir%\Configuration"
|
||||||
move "%PublishDir%\DefaultSettings.json" "%PublishDir%\Configuration\"
|
move "%PublishDir%\DefaultSettings.json" "%PublishDir%\Configuration\"
|
||||||
if not exist "%PublishDir%\Lib\" md "%PublishDir%\Lib\"
|
if not exist "%PublishDir%\Lib\" md "%PublishDir%\Lib\"
|
||||||
|
del "%PublishDir%\Microsoft.CodeAnalysis*.dll" /F /Q
|
||||||
move "%PublishDir%\*.dll" "%PublishDir%\Lib\"
|
move "%PublishDir%\*.dll" "%PublishDir%\Lib\"
|
||||||
move "%PublishDir%\*.json" "%PublishDir%\Lib\"
|
move "%PublishDir%\*.json" "%PublishDir%\Lib\"
|
||||||
move "%PublishDir%\runtimes" "%PublishDir%\Lib\runtimes"
|
move "%PublishDir%\runtimes" "%PublishDir%\Lib\runtimes"
|
||||||
@ -30,16 +31,37 @@ move "%PublishDir%\ru" "%PublishDir%\Lib\ru"
|
|||||||
move "%PublishDir%\de" "%PublishDir%\Lib\de"
|
move "%PublishDir%\de" "%PublishDir%\Lib\de"
|
||||||
move "%PublishDir%\pt" "%PublishDir%\Lib\pt"
|
move "%PublishDir%\pt" "%PublishDir%\Lib\pt"
|
||||||
move "%PublishDir%\es" "%PublishDir%\Lib\es"
|
move "%PublishDir%\es" "%PublishDir%\Lib\es"
|
||||||
|
rmdir /Q /S "%PublishDir%\cs"
|
||||||
|
rmdir /Q /S "%PublishDir%\fr"
|
||||||
|
rmdir /Q /S "%PublishDir%\it"
|
||||||
|
rmdir /Q /S "%PublishDir%\ja"
|
||||||
|
rmdir /Q /S "%PublishDir%\ko"
|
||||||
|
rmdir /Q /S "%PublishDir%\pl"
|
||||||
|
rmdir /Q /S "%PublishDir%\pt-BR"
|
||||||
|
rmdir /Q /S "%PublishDir%\tr"
|
||||||
|
rmdir /Q /S "%PublishDir%\zh-Hans"
|
||||||
|
rmdir /Q /S "%PublishDir%\zh-Hant"
|
||||||
if exist "%PublishDir%\refs" move "%PublishDir%\refs" "%PublishDir%\Lib\refs"
|
if exist "%PublishDir%\refs" move "%PublishDir%\refs" "%PublishDir%\Lib\refs"
|
||||||
|
|
||||||
echo making start scripts
|
echo making start scripts
|
||||||
@(echo @echo off && echo @title IW4MAdmin && echo set DOTNET_CLI_TELEMETRY_OPTOUT=1 && echo dotnet Lib\IW4MAdmin.dll && echo pause) > "%PublishDir%\StartIW4MAdmin.cmd"
|
@(echo @echo off && echo @title IW4MAdmin && echo set DOTNET_CLI_TELEMETRY_OPTOUT=1 && echo dotnet Lib\IW4MAdmin.dll && echo pause) > "%PublishDir%\StartIW4MAdmin.cmd"
|
||||||
@(echo #!/bin/bash&& echo export DOTNET_CLI_TELEMETRY_OPTOUT=1&& echo dotnet Lib/IW4MAdmin.dll) > "%PublishDir%\StartIW4MAdmin.sh"
|
@(echo #!/bin/bash&& echo export DOTNET_CLI_TELEMETRY_OPTOUT=1&& echo dotnet Lib/IW4MAdmin.dll) > "%PublishDir%\StartIW4MAdmin.sh"
|
||||||
|
|
||||||
|
echo copying update scripts
|
||||||
|
copy "%SourceDir%\DeploymentFiles\UpdateIW4MAdmin.ps1" "%PublishDir%\UpdateIW4MAdmin.ps1"
|
||||||
|
copy "%SourceDir%\DeploymentFiles\UpdateIW4MAdmin.sh" "%PublishDir%\UpdateIW4MAdmin.sh"
|
||||||
|
|
||||||
echo moving front-end library dependencies
|
echo moving front-end library dependencies
|
||||||
if not exist "%PublishDir%\wwwroot\font" mkdir "%PublishDir%\wwwroot\font"
|
if not exist "%PublishDir%\wwwroot\font" mkdir "%PublishDir%\wwwroot\font"
|
||||||
move "WebfrontCore\wwwroot\lib\open-iconic\font\fonts\*.*" "%PublishDir%\wwwroot\font\"
|
move "WebfrontCore\wwwroot\lib\open-iconic\font\fonts\*.*" "%PublishDir%\wwwroot\font\"
|
||||||
if exist "%PublishDir%\wwwroot\lib" rd /s /q "%PublishDir%\wwwroot\lib"
|
if exist "%PublishDir%\wwwroot\lib" rd /s /q "%PublishDir%\wwwroot\lib"
|
||||||
|
if not exist "%PublishDir%\wwwroot\css" mkdir "%PublishDir%\wwwroot\css"
|
||||||
|
move "WebfrontCore\wwwroot\css\global.min.css" "%PublishDir%\wwwroot\css\global.min.css"
|
||||||
|
if not exist "%PublishDir%\wwwroot\js" mkdir "%PublishDir%\wwwroot\js"
|
||||||
|
move "%SourceDir%\WebfrontCore\wwwroot\js\global.min.js" "%PublishDir%\wwwroot\js\global.min.js"
|
||||||
|
if not exist "%PublishDir%\wwwroot\images" mkdir "%PublishDir%\wwwroot\images"
|
||||||
|
xcopy "%SourceDir%\WebfrontCore\wwwroot\images" "%PublishDir%\wwwroot\images" /E /H /C /I
|
||||||
|
|
||||||
|
|
||||||
echo setting permissions...
|
echo setting permissions...
|
||||||
cacls "%PublishDir%" /t /e /p Everyone:F
|
cacls "%PublishDir%" /t /e /p Everyone:F
|
64
Application/Commands/ClientTags/AddClientTagCommand.cs
Normal file
64
Application/Commands/ClientTags/AddClientTagCommand.cs
Normal 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));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
38
Application/Commands/ClientTags/ListClientTags.cs
Normal file
38
Application/Commands/ClientTags/ListClientTags.cs
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands.ClientTags
|
||||||
|
{
|
||||||
|
public class ListClientTags : Command
|
||||||
|
{
|
||||||
|
private readonly IMetaServiceV2 _metaService;
|
||||||
|
|
||||||
|
public ListClientTags(CommandConfiguration config, ITranslationLookup layout, IMetaServiceV2 metaService) : base(
|
||||||
|
config, layout)
|
||||||
|
{
|
||||||
|
Name = "listclienttags";
|
||||||
|
Description = layout["COMMANDS_LIST_CLIENT_TAGS_DESC"];
|
||||||
|
Alias = "lct";
|
||||||
|
Permission = EFClient.Permission.Owner;
|
||||||
|
RequiresTarget = false;
|
||||||
|
_metaService = metaService;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override async Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
var tags = await _metaService.GetPersistentMetaValue<List<TagMeta>>(EFMeta.ClientTagNameV2);
|
||||||
|
|
||||||
|
if (tags is not null)
|
||||||
|
{
|
||||||
|
await gameEvent.Origin.TellAsync(tags.Select(tag => tag.TagName),
|
||||||
|
gameEvent.Owner.Manager.CancellationToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,16 +1,21 @@
|
|||||||
using SharedLibraryCore.Configuration;
|
using System.Collections.Generic;
|
||||||
using SharedLibraryCore.Database.Models;
|
|
||||||
using SharedLibraryCore.Interfaces;
|
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using System.Linq;
|
||||||
using Data.Models;
|
using Data.Models;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
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(config, layout)
|
public RemoveClientTag(CommandConfiguration config, ITranslationLookup layout, IMetaServiceV2 metaService) : base(
|
||||||
|
config, layout)
|
||||||
{
|
{
|
||||||
Name = "removeclienttag";
|
Name = "removeclienttag";
|
||||||
Description = layout["COMMANDS_REMOVE_CLIENT_TAG_DESC"];
|
Description = layout["COMMANDS_REMOVE_CLIENT_TAG_DESC"];
|
||||||
@ -19,7 +24,7 @@ namespace SharedLibraryCore.Commands
|
|||||||
RequiresTarget = false;
|
RequiresTarget = false;
|
||||||
Arguments = new[]
|
Arguments = new[]
|
||||||
{
|
{
|
||||||
new CommandArgument()
|
new CommandArgument
|
||||||
{
|
{
|
||||||
Name = _translationLookup["COMMANDS_ARGUMENT_TAG"],
|
Name = _translationLookup["COMMANDS_ARGUMENT_TAG"],
|
||||||
Required = true
|
Required = true
|
||||||
@ -31,7 +36,12 @@ 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));
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,18 +1,23 @@
|
|||||||
using SharedLibraryCore.Configuration;
|
using System.Collections.Generic;
|
||||||
using SharedLibraryCore.Database.Models;
|
|
||||||
using SharedLibraryCore.Interfaces;
|
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Data.Models;
|
using Data.Models;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Dtos;
|
||||||
|
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) : base(config, layout)
|
public SetClientTagCommand(CommandConfiguration config, ITranslationLookup layout, IMetaServiceV2 metaService) :
|
||||||
|
base(config, layout)
|
||||||
{
|
{
|
||||||
Name = "setclienttag";
|
Name = "setclienttag";
|
||||||
Description = layout["COMMANDS_SET_CLIENT_TAG_DESC"];
|
Description = layout["COMMANDS_SET_CLIENT_TAG_DESC"];
|
||||||
@ -21,7 +26,7 @@ namespace SharedLibraryCore.Commands
|
|||||||
RequiresTarget = true;
|
RequiresTarget = true;
|
||||||
Arguments = new[]
|
Arguments = new[]
|
||||||
{
|
{
|
||||||
new CommandArgument()
|
new CommandArgument
|
||||||
{
|
{
|
||||||
Name = _translationLookup["COMMANDS_ARGUMENT_TAG"],
|
Name = _translationLookup["COMMANDS_ARGUMENT_TAG"],
|
||||||
Required = true
|
Required = true
|
||||||
@ -33,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)
|
||||||
{
|
{
|
||||||
@ -43,7 +50,8 @@ 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));
|
||||||
}
|
}
|
||||||
}
|
}
|
13
Application/Commands/ClientTags/TagMeta.cs
Normal file
13
Application/Commands/ClientTags/TagMeta.cs
Normal 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; }
|
||||||
|
}
|
@ -1,18 +1,19 @@
|
|||||||
using SharedLibraryCore.Configuration;
|
using System.Threading.Tasks;
|
||||||
using SharedLibraryCore.Database.Models;
|
|
||||||
using SharedLibraryCore.Interfaces;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Data.Models;
|
using Data.Models;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
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";
|
||||||
Description = layout["COMMANDS_UNSET_CLIENT_TAG_DESC"];
|
Description = layout["COMMANDS_UNSET_CLIENT_TAG_DESC"];
|
||||||
@ -21,7 +22,7 @@ namespace SharedLibraryCore.Commands
|
|||||||
RequiresTarget = true;
|
RequiresTarget = true;
|
||||||
Arguments = new[]
|
Arguments = new[]
|
||||||
{
|
{
|
||||||
new CommandArgument()
|
new CommandArgument
|
||||||
{
|
{
|
||||||
Name = _translationLookup["COMMANDS_ARGUMENT_TAG"],
|
Name = _translationLookup["COMMANDS_ARGUMENT_TAG"],
|
||||||
Required = true
|
Required = true
|
||||||
@ -34,7 +35,8 @@ 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"]);
|
||||||
}
|
}
|
||||||
}
|
}
|
59
Application/Commands/FindPlayerCommand.cs
Normal file
59
Application/Commands/FindPlayerCommand.cs
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
using System;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Finds player by name
|
||||||
|
/// </summary>
|
||||||
|
public class FindPlayerCommand : Command
|
||||||
|
{
|
||||||
|
public FindPlayerCommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config,
|
||||||
|
translationLookup)
|
||||||
|
{
|
||||||
|
Name = "find";
|
||||||
|
Description = _translationLookup["COMMANDS_FIND_DESC"];
|
||||||
|
Alias = "f";
|
||||||
|
Permission = EFClient.Permission.Administrator;
|
||||||
|
RequiresTarget = false;
|
||||||
|
Arguments = new[]
|
||||||
|
{
|
||||||
|
new CommandArgument()
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMANDS_ARGS_PLAYER"],
|
||||||
|
Required = true
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public override async Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
if (gameEvent.Data.Length < 3)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_FIND_MIN"]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var players = await gameEvent.Owner.Manager.GetClientService().FindClientsByIdentifier(gameEvent.Data);
|
||||||
|
|
||||||
|
if (!players.Any())
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_FIND_EMPTY"]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var client in players)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_FIND_FORMAT_V2"].FormatExt(client.Name,
|
||||||
|
client.ClientId, Utilities.ConvertLevelToColor((EFClient.Permission) client.LevelInt, client.Level),
|
||||||
|
client.IPAddress, (DateTime.UtcNow - client.LastConnection).HumanizeForCurrentCulture()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
93
Application/Commands/HelpCommand.cs
Normal file
93
Application/Commands/HelpCommand.cs
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
using System;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Prints help information
|
||||||
|
/// </summary>
|
||||||
|
public class HelpCommand : Command
|
||||||
|
{
|
||||||
|
public HelpCommand(CommandConfiguration config, ITranslationLookup translationLookup) :
|
||||||
|
base(config, translationLookup)
|
||||||
|
{
|
||||||
|
Name = "help";
|
||||||
|
Description = translationLookup["COMMANDS_HELP_DESC"];
|
||||||
|
Alias = "h";
|
||||||
|
Permission = EFClient.Permission.User;
|
||||||
|
RequiresTarget = false;
|
||||||
|
Arguments = new[]
|
||||||
|
{
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = translationLookup["COMMANDS_ARGS_COMMANDS"],
|
||||||
|
Required = false
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
var searchTerm = gameEvent.Data.Trim();
|
||||||
|
var availableCommands = gameEvent.Owner.Manager.Commands.Distinct().Where(command =>
|
||||||
|
command.SupportedGames == null || !command.SupportedGames.Any() ||
|
||||||
|
command.SupportedGames.Contains(gameEvent.Owner.GameName))
|
||||||
|
.Where(command => gameEvent.Origin.Level >= command.Permission);
|
||||||
|
|
||||||
|
if (searchTerm.Length > 2)
|
||||||
|
{
|
||||||
|
var matchingCommand = availableCommands.FirstOrDefault(command =>
|
||||||
|
command.Name.Equals(searchTerm, StringComparison.InvariantCultureIgnoreCase) ||
|
||||||
|
command.Alias.Equals(searchTerm, StringComparison.InvariantCultureIgnoreCase));
|
||||||
|
|
||||||
|
if (matchingCommand != null)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_HELP_SEARCH_RESULT"]
|
||||||
|
.FormatExt(matchingCommand.Name, matchingCommand.Alias));
|
||||||
|
gameEvent.Origin.Tell(matchingCommand.Syntax);
|
||||||
|
}
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_HELP_NOTFOUND"]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var commandStrings = availableCommands.Select((command, index) =>
|
||||||
|
new
|
||||||
|
{
|
||||||
|
response = $" {_translationLookup["COMMANDS_HELP_LIST_FORMAT"].FormatExt(command.Name)} ",
|
||||||
|
index
|
||||||
|
});
|
||||||
|
|
||||||
|
var helpResponse = new StringBuilder();
|
||||||
|
|
||||||
|
foreach (var item in commandStrings)
|
||||||
|
{
|
||||||
|
helpResponse.Append(item.response);
|
||||||
|
if (item.index == 0 || item.index % 4 != 0)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
gameEvent.Origin.Tell(helpResponse.ToString());
|
||||||
|
helpResponse = new StringBuilder();
|
||||||
|
}
|
||||||
|
|
||||||
|
gameEvent.Origin.Tell(helpResponse.ToString());
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_HELP_MOREINFO"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
50
Application/Commands/ListAdminsCommand.cs
Normal file
50
Application/Commands/ListAdminsCommand.cs
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
using System;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Lists all unmasked admins
|
||||||
|
/// </summary>
|
||||||
|
public class ListAdminsCommand : Command
|
||||||
|
{
|
||||||
|
public ListAdminsCommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config,
|
||||||
|
translationLookup)
|
||||||
|
{
|
||||||
|
Name = "admins";
|
||||||
|
Description = _translationLookup["COMMANDS_ADMINS_DESC"];
|
||||||
|
Alias = "a";
|
||||||
|
Permission = EFClient.Permission.User;
|
||||||
|
RequiresTarget = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string OnlineAdmins(Server server, ITranslationLookup lookup)
|
||||||
|
{
|
||||||
|
var onlineAdmins = server.GetClientsAsList()
|
||||||
|
.Where(p => p.Level > EFClient.Permission.Flagged)
|
||||||
|
.Where(p => !p.Masked)
|
||||||
|
.Select(p =>
|
||||||
|
$"[(Color::Yellow){Utilities.ConvertLevelToColor(p.Level, p.ClientPermission.Name)}(Color::White)] {p.Name}")
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
return onlineAdmins.Any() ? string.Join(Environment.NewLine, onlineAdmins) : lookup["COMMANDS_ADMINS_NONE"];
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
foreach (var line in OnlineAdmins(gameEvent.Owner, _translationLookup).Split(Environment.NewLine))
|
||||||
|
{
|
||||||
|
var _ = gameEvent.Message.IsBroadcastCommand(_config.BroadcastCommandPrefix)
|
||||||
|
? gameEvent.Owner.Broadcast(line)
|
||||||
|
: gameEvent.Origin.Tell(line);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
57
Application/Commands/ListAliasesCommand.cs
Normal file
57
Application/Commands/ListAliasesCommand.cs
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Lists alises of specified client
|
||||||
|
/// </summary>
|
||||||
|
public class ListAliasesCommand : Command
|
||||||
|
{
|
||||||
|
public ListAliasesCommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config,
|
||||||
|
translationLookup)
|
||||||
|
{
|
||||||
|
Name = "alias";
|
||||||
|
Description = _translationLookup["COMMANDS_ALIAS_DESC"];
|
||||||
|
Alias = "known";
|
||||||
|
Permission = EFClient.Permission.Moderator;
|
||||||
|
RequiresTarget = true;
|
||||||
|
Arguments = new[]
|
||||||
|
{
|
||||||
|
new CommandArgument()
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMANDS_ARGS_PLAYER"],
|
||||||
|
Required = true,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
var message = new StringBuilder();
|
||||||
|
var names = new List<string>(gameEvent.Target.AliasLink.Children.Select(a => a.Name));
|
||||||
|
var ips = new List<string>(gameEvent.Target.AliasLink.Children.Select(a => a.IPAddress.ConvertIPtoString())
|
||||||
|
.Distinct());
|
||||||
|
|
||||||
|
gameEvent.Origin.Tell($"[(Color::Accent){gameEvent.Target}(Color::White)]");
|
||||||
|
|
||||||
|
message.Append($"{_translationLookup["COMMANDS_ALIAS_ALIASES"]}: ");
|
||||||
|
message.Append(string.Join(" | ", names));
|
||||||
|
gameEvent.Origin.Tell(message.ToString());
|
||||||
|
|
||||||
|
message.Clear();
|
||||||
|
message.Append($"{_translationLookup["COMMANDS_ALIAS_IPS"]}: ");
|
||||||
|
message.Append(string.Join(" | ", ips));
|
||||||
|
gameEvent.Origin.Tell(message.ToString());
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
37
Application/Commands/ListClientsCommand.cs
Normal file
37
Application/Commands/ListClientsCommand.cs
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// List online clients
|
||||||
|
/// </summary>
|
||||||
|
public class ListClientsCommand : Command
|
||||||
|
{
|
||||||
|
public ListClientsCommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config,
|
||||||
|
translationLookup)
|
||||||
|
{
|
||||||
|
Name = "list";
|
||||||
|
Description = _translationLookup["COMMANDS_LIST_DESC"];
|
||||||
|
Alias = "l";
|
||||||
|
Permission = EFClient.Permission.Moderator;
|
||||||
|
RequiresTarget = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
var clientList = gameEvent.Owner.GetClientsAsList()
|
||||||
|
.Select(client =>
|
||||||
|
$"[(Color::Accent){client.ClientPermission.Name}(Color::White){(string.IsNullOrEmpty(client.Tag) ? "" : $" {client.Tag}")}(Color::White)][(Color::Yellow)#{client.ClientNumber}(Color::White)] {client.Name}")
|
||||||
|
.ToArray();
|
||||||
|
|
||||||
|
gameEvent.Origin.TellAsync(clientList, gameEvent.Owner.Manager.CancellationToken);
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
41
Application/Commands/ListPluginsCommand.cs
Normal file
41
Application/Commands/ListPluginsCommand.cs
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Lists the loaded plugins
|
||||||
|
/// </summary>
|
||||||
|
public class ListPluginsCommand : Command
|
||||||
|
{
|
||||||
|
private readonly IEnumerable<IPlugin> _plugins;
|
||||||
|
|
||||||
|
public ListPluginsCommand(CommandConfiguration config, ITranslationLookup translationLookup,
|
||||||
|
IEnumerable<IPlugin> plugins) : base(config, translationLookup)
|
||||||
|
{
|
||||||
|
Name = "plugins";
|
||||||
|
Description = _translationLookup["COMMANDS_PLUGINS_DESC"];
|
||||||
|
Alias = "p";
|
||||||
|
Permission = EFClient.Permission.Administrator;
|
||||||
|
RequiresTarget = false;
|
||||||
|
_plugins = plugins;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_PLUGINS_LOADED"]);
|
||||||
|
foreach (var plugin in _plugins.Where(plugin => !plugin.IsParser))
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_LIST_PLUGINS_FORMAT"]
|
||||||
|
.FormatExt(plugin.Name, plugin.Version, plugin.Author));
|
||||||
|
}
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
59
Application/Commands/ListReportsCommand.cs
Normal file
59
Application/Commands/ListReportsCommand.cs
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Helpers;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// List all reports on the server
|
||||||
|
/// </summary>
|
||||||
|
public class ListReportsCommand : Command
|
||||||
|
{
|
||||||
|
public ListReportsCommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config,
|
||||||
|
translationLookup)
|
||||||
|
{
|
||||||
|
Name = "reports";
|
||||||
|
Description = _translationLookup["COMMANDS_REPORTS_DESC"];
|
||||||
|
Alias = "reps";
|
||||||
|
Permission = EFClient.Permission.Moderator;
|
||||||
|
RequiresTarget = false;
|
||||||
|
Arguments = new[]
|
||||||
|
{
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMANDS_ARGS_CLEAR"],
|
||||||
|
Required = false
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
if (gameEvent.Data != null && gameEvent.Data.ToLower().Contains(_translationLookup["COMMANDS_ARGS_CLEAR"]))
|
||||||
|
{
|
||||||
|
gameEvent.Owner.Reports = new List<Report>();
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_REPORTS_CLEAR_SUCCESS"]);
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gameEvent.Owner.Reports.Count < 1)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_REPORTS_NONE"]);
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var report in gameEvent.Owner.Reports)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(
|
||||||
|
$"(Color::Accent){report.Origin.Name}(Color::White) -> (Color::Red){report.Target.Name}(Color::White): {report.Reason}");
|
||||||
|
}
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
116
Application/Commands/MapAndGameTypeCommand.cs
Normal file
116
Application/Commands/MapAndGameTypeCommand.cs
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
using System.Linq;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using IW4MAdmin.Application.Extensions;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
public class MapAndGameTypeCommand : Command
|
||||||
|
{
|
||||||
|
private const string ArgumentRegexPattern = "(?:\"([^\"]+)\"|([^\\s]+)) (?:\"([^\"]+)\"|([^\\s]+))";
|
||||||
|
private readonly ILogger _logger;
|
||||||
|
private readonly DefaultSettings _defaultSettings;
|
||||||
|
|
||||||
|
public MapAndGameTypeCommand(ILogger<MapAndGameTypeCommand> logger, CommandConfiguration config,
|
||||||
|
DefaultSettings defaultSettings, ITranslationLookup layout) : base(config, layout)
|
||||||
|
{
|
||||||
|
Name = "mapandgametype";
|
||||||
|
Description = _translationLookup["COMMANDS_MAG_DESCRIPTION"];
|
||||||
|
Alias = "mag";
|
||||||
|
Permission = EFClient.Permission.Administrator;
|
||||||
|
RequiresTarget = false;
|
||||||
|
Arguments = new[]
|
||||||
|
{
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMADS_MAG_ARG_1"],
|
||||||
|
Required = true
|
||||||
|
},
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMADS_MAG_ARG_2"],
|
||||||
|
Required = true
|
||||||
|
}
|
||||||
|
};
|
||||||
|
_logger = logger;
|
||||||
|
_defaultSettings = defaultSettings;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override async Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
var match = Regex.Match(gameEvent.Data.Trim(), ArgumentRegexPattern,
|
||||||
|
RegexOptions.Compiled | RegexOptions.IgnoreCase);
|
||||||
|
|
||||||
|
if (!match.Success)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(Syntax);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var map = match.Groups[1].Length > 0 ? match.Groups[1].ToString() : match.Groups[2].ToString();
|
||||||
|
var gametype = match.Groups[3].Length > 0 ? match.Groups[3].ToString() : match.Groups[4].ToString();
|
||||||
|
|
||||||
|
var matchingMaps = gameEvent.Owner.FindMap(map);
|
||||||
|
var matchingGametypes = _defaultSettings.FindGametype(gametype, gameEvent.Owner.GameName);
|
||||||
|
|
||||||
|
if (matchingMaps.Count > 1)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_MAG_MULTIPLE_MAPS"]);
|
||||||
|
|
||||||
|
foreach (var matchingMap in matchingMaps)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(
|
||||||
|
$"[(Color::Yellow){matchingMap.Alias}(Color::White)] [(Color::Yellow){matchingMap.Name}(Color::White)]");
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (matchingGametypes.Count > 1)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_MAG_MULTIPLE_GAMETYPES"]);
|
||||||
|
|
||||||
|
foreach (var matchingGametype in matchingGametypes)
|
||||||
|
{
|
||||||
|
gameEvent.Origin.Tell(
|
||||||
|
$"[(Color::Yellow){matchingGametype.Alias}(Color::White)] [(Color::Yellow){matchingGametype.Name}(Color::White)]");
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
map = matchingMaps.FirstOrDefault()?.Name ?? map;
|
||||||
|
gametype = matchingGametypes.FirstOrDefault()?.Name ?? gametype;
|
||||||
|
var hasMatchingGametype = matchingGametypes.Any();
|
||||||
|
|
||||||
|
_logger.LogDebug("Changing map to {Map} and gametype {Gametype}", map, gametype);
|
||||||
|
|
||||||
|
await gameEvent.Owner.SetDvarAsync("g_gametype", gametype, gameEvent.Owner.Manager.CancellationToken);
|
||||||
|
gameEvent.Owner.Broadcast(_translationLookup["COMMANDS_MAP_SUCCESS"].FormatExt(map));
|
||||||
|
await Task.Delay(gameEvent.Owner.Manager.GetApplicationSettings().Configuration().MapChangeDelaySeconds);
|
||||||
|
|
||||||
|
switch (gameEvent.Owner.GameName)
|
||||||
|
{
|
||||||
|
case Server.Game.IW5:
|
||||||
|
await gameEvent.Owner.ExecuteCommandAsync(
|
||||||
|
$"load_dsr {(hasMatchingGametype ? gametype.ToUpper() + "_default" : gametype)}");
|
||||||
|
await gameEvent.Owner.ExecuteCommandAsync($"map {map}");
|
||||||
|
break;
|
||||||
|
case Server.Game.T6:
|
||||||
|
await gameEvent.Owner.ExecuteCommandAsync($"exec {gametype}.cfg");
|
||||||
|
await gameEvent.Owner.ExecuteCommandAsync($"map {map}");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
await gameEvent.Owner.ExecuteCommandAsync($"map {map}");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
45
Application/Commands/PrivateMessageCommand.cs
Normal file
45
Application/Commands/PrivateMessageCommand.cs
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Sends a private message to another player
|
||||||
|
/// </summary>
|
||||||
|
public class PrivateMessageCommand : Command
|
||||||
|
{
|
||||||
|
public PrivateMessageCommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config, translationLookup)
|
||||||
|
{
|
||||||
|
Name = "privatemessage";
|
||||||
|
Description = _translationLookup["COMMANDS_PM_DESC"];
|
||||||
|
Alias = "pm";
|
||||||
|
Permission = EFClient.Permission.User;
|
||||||
|
RequiresTarget = true;
|
||||||
|
Arguments = new[]
|
||||||
|
{
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMANDS_ARGS_PLAYER"],
|
||||||
|
Required = true
|
||||||
|
},
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMANDS_ARGS_MESSAGE"],
|
||||||
|
Required = true
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
gameEvent.Target.Tell(_translationLookup["COMMANDS_PRIVATE_MESSAGE_FORMAT"].FormatExt(gameEvent.Origin.Name, gameEvent.Data));
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_PRIVATE_MESSAGE_RESULT"]
|
||||||
|
.FormatExt(gameEvent.Target.Name, gameEvent.Data));
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
77
Application/Commands/ReportClientCommand.cs
Normal file
77
Application/Commands/ReportClientCommand.cs
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Report client for given reason
|
||||||
|
/// </summary>
|
||||||
|
public class ReportClientCommand : Command
|
||||||
|
{
|
||||||
|
public ReportClientCommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config,
|
||||||
|
translationLookup)
|
||||||
|
{
|
||||||
|
Name = "report";
|
||||||
|
Description = _translationLookup["COMMANDS_REPORT_DESC"];
|
||||||
|
Alias = "rep";
|
||||||
|
Permission = EFClient.Permission.User;
|
||||||
|
RequiresTarget = true;
|
||||||
|
Arguments = new[]
|
||||||
|
{
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMANDS_ARGS_PLAYER"],
|
||||||
|
Required = true
|
||||||
|
},
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMANDS_ARGS_REASON"],
|
||||||
|
Required = true
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public override async Task ExecuteAsync(GameEvent commandEvent)
|
||||||
|
{
|
||||||
|
if (commandEvent.Data.ToLower().Contains("camp"))
|
||||||
|
{
|
||||||
|
commandEvent.Origin.Tell(_translationLookup["COMMANDS_REPORT_FAIL_CAMP"]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var success = false;
|
||||||
|
|
||||||
|
switch ((await commandEvent.Target.Report(commandEvent.Data, commandEvent.Origin)
|
||||||
|
.WaitAsync(Utilities.DefaultCommandTimeout, commandEvent.Owner.Manager.CancellationToken)).FailReason)
|
||||||
|
{
|
||||||
|
case GameEvent.EventFailReason.None:
|
||||||
|
commandEvent.Origin.Tell(_translationLookup["COMMANDS_REPORT_SUCCESS"]);
|
||||||
|
success = true;
|
||||||
|
break;
|
||||||
|
case GameEvent.EventFailReason.Exception:
|
||||||
|
commandEvent.Origin.Tell(_translationLookup["COMMANDS_REPORT_FAIL_DUPLICATE"]);
|
||||||
|
break;
|
||||||
|
case GameEvent.EventFailReason.Permission:
|
||||||
|
commandEvent.Origin.Tell(_translationLookup["COMMANDS_REPORT_FAIL"]
|
||||||
|
.FormatExt(commandEvent.Target.Name));
|
||||||
|
break;
|
||||||
|
case GameEvent.EventFailReason.Invalid:
|
||||||
|
commandEvent.Origin.Tell(_translationLookup["COMMANDS_REPORT_FAIL_SELF"]);
|
||||||
|
break;
|
||||||
|
case GameEvent.EventFailReason.Throttle:
|
||||||
|
commandEvent.Origin.Tell(_translationLookup["COMMANDS_REPORT_FAIL_TOOMANY"]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
commandEvent.Owner.ToAdmins(
|
||||||
|
$"(Color::Accent){commandEvent.Origin.Name}(Color::White) -> (Color::Red){commandEvent.Target.Name}(Color::White): {commandEvent.Data}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
46
Application/Commands/SayAllCommand.cs
Normal file
46
Application/Commands/SayAllCommand.cs
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Prints out a message to all clients on all servers
|
||||||
|
/// </summary>
|
||||||
|
public class SayAllCommand : Command
|
||||||
|
{
|
||||||
|
public SayAllCommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config,
|
||||||
|
translationLookup)
|
||||||
|
{
|
||||||
|
Name = "sayall";
|
||||||
|
Description = _translationLookup["COMMANDS_SAY_ALL_DESC"];
|
||||||
|
Alias = "sa";
|
||||||
|
Permission = EFClient.Permission.Moderator;
|
||||||
|
RequiresTarget = false;
|
||||||
|
Arguments = new[]
|
||||||
|
{
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMANDS_ARGS_MESSAGE"],
|
||||||
|
Required = true
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
var message = $"(Color::Accent){gameEvent.Origin.Name}(Color::White) - (Color::Red){gameEvent.Data}";
|
||||||
|
|
||||||
|
foreach (var server in gameEvent.Owner.Manager.GetServers())
|
||||||
|
{
|
||||||
|
server.Broadcast(message, gameEvent.Origin);
|
||||||
|
}
|
||||||
|
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_SAY_SUCCESS"]);
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
42
Application/Commands/SayCommand.cs
Normal file
42
Application/Commands/SayCommand.cs
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
using System.Threading.Tasks;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Commands;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
using EFClient = Data.Models.Client.EFClient;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Prints out a message to all clients on the server
|
||||||
|
/// </summary>
|
||||||
|
public class SayCommand : Command
|
||||||
|
{
|
||||||
|
public SayCommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config,
|
||||||
|
translationLookup)
|
||||||
|
{
|
||||||
|
Name = "say";
|
||||||
|
Description = _translationLookup["COMMANDS_SAY_DESC"];
|
||||||
|
Alias = "s";
|
||||||
|
Permission = EFClient.Permission.Moderator;
|
||||||
|
RequiresTarget = false;
|
||||||
|
Arguments = new[]
|
||||||
|
{
|
||||||
|
new CommandArgument
|
||||||
|
{
|
||||||
|
Name = _translationLookup["COMMANDS_ARGS_MESSAGE"],
|
||||||
|
Required = true
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
gameEvent.Owner.Broadcast(
|
||||||
|
_translationLookup["COMMANDS_SAY_FORMAT"].FormatExt(gameEvent.Origin.Name, gameEvent.Data),
|
||||||
|
gameEvent.Origin);
|
||||||
|
gameEvent.Origin.Tell(_translationLookup["COMMANDS_SAY_SUCCESS"]);
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
38
Application/Commands/WhoAmICommand.cs
Normal file
38
Application/Commands/WhoAmICommand.cs
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
using System.Threading.Tasks;
|
||||||
|
using Data.Models.Client;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Commands
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Prints client information
|
||||||
|
/// </summary>
|
||||||
|
public class WhoAmICommand : Command
|
||||||
|
{
|
||||||
|
public WhoAmICommand(CommandConfiguration config, ITranslationLookup translationLookup) : base(config,
|
||||||
|
translationLookup)
|
||||||
|
{
|
||||||
|
Name = "whoami";
|
||||||
|
Description = _translationLookup["COMMANDS_WHO_DESC"];
|
||||||
|
Alias = "who";
|
||||||
|
Permission = EFClient.Permission.User;
|
||||||
|
RequiresTarget = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Task ExecuteAsync(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
var you =
|
||||||
|
"[(Color::Yellow)#{{clientNumber}}(Color::White)] [(Color::Yellow)@{{clientId}}(Color::White)] [{{networkId}}] [{{ip}}] [(Color::Cyan){{level}}(Color::White){{tag}}(Color::White)] {{name}}"
|
||||||
|
.FormatExt(gameEvent.Origin.ClientNumber,
|
||||||
|
gameEvent.Origin.ClientId, gameEvent.Origin.GuidString,
|
||||||
|
gameEvent.Origin.IPAddressString, gameEvent.Origin.ClientPermission.Name,
|
||||||
|
string.IsNullOrEmpty(gameEvent.Origin.Tag) ? "" : $" {gameEvent.Origin.Tag}",
|
||||||
|
gameEvent.Origin.Name);
|
||||||
|
gameEvent.Origin.Tell(you);
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -18,6 +18,13 @@
|
|||||||
"rollingInterval": "Day",
|
"rollingInterval": "Day",
|
||||||
"outputTemplate": "[{Timestamp:yyyy-MM-dd HH:mm:ss.fff} {Server} {Level:u3}] {Message:lj}{NewLine}{Exception}"
|
"outputTemplate": "[{Timestamp:yyyy-MM-dd HH:mm:ss.fff} {Server} {Level:u3}] {Message:lj}{NewLine}{Exception}"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "Console",
|
||||||
|
"Args": {
|
||||||
|
"outputTemplate": "[{Timestamp:yyyy-MM-dd HH:mm:ss.fff} {Server} {Level:u3}] {Message:lj}{NewLine}{Exception}",
|
||||||
|
"RestrictedToMinimumLevel": "Fatal"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"Enrich": [
|
"Enrich": [
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
{
|
{
|
||||||
"AutoMessagePeriod": 60,
|
"AutoMessagePeriod": 60,
|
||||||
"AutoMessages": [
|
"AutoMessages": [
|
||||||
"This server uses ^5IW4M Admin v{{VERSION}} ^7get it at ^5raidmax.org/IW4MAdmin",
|
"This server uses (Color::Accent)IW4M Admin v{{VERSION}} (Color::White)get it at (Color::Accent)raidmax.org/IW4MAdmin",
|
||||||
"^5IW4M Admin ^7sees ^5YOU!",
|
"(Color::Accent)IW4M Admin (Color::White)sees (Color::Accent)YOU!",
|
||||||
"{{TOPSTATS}}",
|
"{{TOPSTATS}}",
|
||||||
"This server has seen a total of ^5{{TOTALPLAYERS}} ^7players!",
|
"This server has seen a total of (Color::Accent){{TOTALPLAYERS}} (Color::White)players!",
|
||||||
"Cheaters are ^1unwelcome ^7 on this server",
|
"Cheaters are (Color::Red)unwelcome (Color::White)on this server",
|
||||||
"Did you know 8/10 people agree with unverified statistics?"
|
"Did you know 8/10 people agree with unverified statistics?"
|
||||||
],
|
],
|
||||||
"GlobalRules": [
|
"GlobalRules": [
|
||||||
@ -68,6 +68,504 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"Gametypes": [
|
||||||
|
{
|
||||||
|
"Game": "IW4",
|
||||||
|
"Gametypes": [
|
||||||
|
{
|
||||||
|
"Name": "arena",
|
||||||
|
"Alias": "Arena"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "ctf",
|
||||||
|
"Alias": "Capture The Flag"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dd",
|
||||||
|
"Alias": "Demolition"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dm",
|
||||||
|
"Alias": "Free For All"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dom",
|
||||||
|
"Alias": "Domination"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "gtnw",
|
||||||
|
"Alias": "Global Thermo-Nuclear War"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "koth",
|
||||||
|
"Alias": "Headquarters"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "oneflag",
|
||||||
|
"Alias": "One-Flag CTF"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sab",
|
||||||
|
"Alias": "Sabotage"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sd",
|
||||||
|
"Alias": "Search & Destroy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "war",
|
||||||
|
"Alias": "Team Deathmatch"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Game": "T4",
|
||||||
|
"Gametypes": [
|
||||||
|
{
|
||||||
|
"Name": "ctf",
|
||||||
|
"Alias": "Capture The Flag"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dom",
|
||||||
|
"Alias": "Domination"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dm",
|
||||||
|
"Alias": "Free For All"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "koth",
|
||||||
|
"Alias": "Headquarters"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "tdm",
|
||||||
|
"Alias": "Team Deathmatch"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sab",
|
||||||
|
"Alias": "Sabotage"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sd",
|
||||||
|
"Alias": "Search & Destroy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "twar",
|
||||||
|
"Alias": "War"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Game": "IW5",
|
||||||
|
"Gametypes": [
|
||||||
|
{
|
||||||
|
"Name": "tdm",
|
||||||
|
"Alias": "Team Deathmatch"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dom",
|
||||||
|
"Alias": "Domination"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "ctf",
|
||||||
|
"Alias": "Capture The Flag"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dd",
|
||||||
|
"Alias": "Demolition"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dz",
|
||||||
|
"Alias": "Drop Zone"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "ffa",
|
||||||
|
"Alias": "Free For All"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "gg",
|
||||||
|
"Alias": "Gun Game"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hq",
|
||||||
|
"Alias": "Headquarters"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "koth",
|
||||||
|
"Alias": "Headquarters"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "inf",
|
||||||
|
"Alias": "Infected"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "jug",
|
||||||
|
"Alias": "Juggernaut"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "kc",
|
||||||
|
"Alias": "Kill Confirmed"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "oic",
|
||||||
|
"Alias": "One In The Chamber"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sab",
|
||||||
|
"Alias": "Sabotage"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sd",
|
||||||
|
"Alias": "Search & Destroy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "tdef",
|
||||||
|
"Alias": "Team Defender"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "tj",
|
||||||
|
"Alias": "Team Juggernaut"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Game": "T5",
|
||||||
|
"Gametypes": [
|
||||||
|
{
|
||||||
|
"Name": "ctf",
|
||||||
|
"Alias": "Capture The Flag"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dem",
|
||||||
|
"Alias": "Demolition"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dom",
|
||||||
|
"Alias": "Domination"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dm",
|
||||||
|
"Alias": "Free For All"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "gun",
|
||||||
|
"Alias": "Gun Game"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hlnd",
|
||||||
|
"Alias": "Sticks & Stones"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "koth",
|
||||||
|
"Alias": "Headquarters"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "oic",
|
||||||
|
"Alias": "One In The Chamber"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sab",
|
||||||
|
"Alias": "Sabotage"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sd",
|
||||||
|
"Alias": "Search & Destroy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "shrp",
|
||||||
|
"Alias": "Sharpshooter"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "tdm",
|
||||||
|
"Alias": "Team Deathmatch"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Game": "IW6",
|
||||||
|
"Gametypes": [
|
||||||
|
{
|
||||||
|
"Name": "blitz",
|
||||||
|
"Alias": "Blitz"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "conf",
|
||||||
|
"Alias": "Kill Confirmed"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "cranked",
|
||||||
|
"Alias": "Cranked"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dm",
|
||||||
|
"Alias": "Free For All"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dom",
|
||||||
|
"Alias": "Domination"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "grind",
|
||||||
|
"Alias": "Grind"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "grnd",
|
||||||
|
"Alias": "Drop Zone"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "gun",
|
||||||
|
"Alias": "Gun Game"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "horde",
|
||||||
|
"Alias": "Safeguard"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "infect",
|
||||||
|
"Alias": "Infected"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sd",
|
||||||
|
"Alias": "Search & Destroy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "siege",
|
||||||
|
"Alias": "Reinforce"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sotf",
|
||||||
|
"Alias": "Hunted"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sotf_ffa",
|
||||||
|
"Alias": "Hunted FFA"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sr",
|
||||||
|
"Alias": "Search & Rescue"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "war",
|
||||||
|
"Alias": "Team Deathmatch"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Game": "T6",
|
||||||
|
"Gametypes": [
|
||||||
|
{
|
||||||
|
"Name": "conf",
|
||||||
|
"Alias": "Kill Confirmed"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "ctf",
|
||||||
|
"Alias": "Capture The Flag"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dem",
|
||||||
|
"Alias": "Demolition"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dm",
|
||||||
|
"Alias": "Free For All"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dom",
|
||||||
|
"Alias": "Domination"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "gun",
|
||||||
|
"Alias": "Gun Game"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hq",
|
||||||
|
"Alias": "Headquarters"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "koth",
|
||||||
|
"Alias": "Hardpoint"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "oic",
|
||||||
|
"Alias": "One In The Chamber"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "oneflag",
|
||||||
|
"Alias": "One-Flag CTF"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sas",
|
||||||
|
"Alias": "Sticks & Stones"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sd",
|
||||||
|
"Alias": "Search & Destroy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "shrp",
|
||||||
|
"Alias": "Sharpshooter"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "tdm",
|
||||||
|
"Alias": "Team Deathmatch"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Game": "T7",
|
||||||
|
"Gametypes": [
|
||||||
|
{
|
||||||
|
"Name": "ball",
|
||||||
|
"Alias": "Uplink"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "clean",
|
||||||
|
"Alias": "Fracture"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "conf",
|
||||||
|
"Alias": "Kill Confirmed"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "ctf",
|
||||||
|
"Alias": "Capture The Flag"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dom",
|
||||||
|
"Alias": "Domination"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dem",
|
||||||
|
"Alias": "Demolition"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dm",
|
||||||
|
"Alias": "Free For All"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "escort",
|
||||||
|
"Alias": "Safeguard"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "gun",
|
||||||
|
"Alias": "Gun Game"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "koth",
|
||||||
|
"Alias": "Hardpoint"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sd",
|
||||||
|
"Alias": "Search & Destroy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "tdm",
|
||||||
|
"Alias": "Team Deathmatch"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_ball",
|
||||||
|
"Alias": "Hardcore Uplink"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_clean",
|
||||||
|
"Alias": "Hardcore Fracture"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_conf",
|
||||||
|
"Alias": "Hardcore Kill Confirmed"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_ctf",
|
||||||
|
"Alias": "Hardcore Capture The Flag"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_dom",
|
||||||
|
"Alias": "Hardcore Domination"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_dem",
|
||||||
|
"Alias": "Hardcore Demolition"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_dm",
|
||||||
|
"Alias": "Hardcore Free For All"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_escort",
|
||||||
|
"Alias": "Hardcore Safeguard"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_gun",
|
||||||
|
"Alias": "Hardcore Gun Game"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_koth",
|
||||||
|
"Alias": "Hardcore Hardpoint"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_sd",
|
||||||
|
"Alias": "Hardcore Search & Destroy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hc_tdm",
|
||||||
|
"Alias": "Hardcore Team Deathmatch"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Game": "SHG1",
|
||||||
|
"Gametypes": [
|
||||||
|
{
|
||||||
|
"Name": "ball",
|
||||||
|
"Alias": "Uplink"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "conf",
|
||||||
|
"Alias": "Kill Confirmed"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "ctf",
|
||||||
|
"Alias": "Capture The Flag"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dom",
|
||||||
|
"Alias": "Domination"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "dm",
|
||||||
|
"Alias": "Free For All"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "gun",
|
||||||
|
"Alias": "Gun Game"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "hp",
|
||||||
|
"Alias": "Hardpoint"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "infect",
|
||||||
|
"Alias": "Infected"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sd",
|
||||||
|
"Alias": "Search & Destroy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "sr",
|
||||||
|
"Alias": "Search & Rescue"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "war",
|
||||||
|
"Alias": "Team Deathmatch"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "twar",
|
||||||
|
"Alias": "Momentum"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
"Maps": [
|
"Maps": [
|
||||||
{
|
{
|
||||||
"Game": "IW3",
|
"Game": "IW3",
|
||||||
@ -703,6 +1201,18 @@
|
|||||||
{
|
{
|
||||||
"Alias": "Highrise",
|
"Alias": "Highrise",
|
||||||
"Name": "mp_highrise"
|
"Name": "mp_highrise"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Alias": "Favela",
|
||||||
|
"Name": "mp_favela"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Alias": "Nuketown",
|
||||||
|
"Name": "mp_nuked"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Alias": "Skidrow",
|
||||||
|
"Name": "mp_nightshift"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@ -1511,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",
|
||||||
@ -1573,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"
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -50,7 +50,14 @@ namespace IW4MAdmin.Application.EventParsers
|
|||||||
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.Damage.Pattern = @"^(D);(-?[A-Fa-f0-9_]{1,32}|bot[0-9]+|0);(-?[0-9]+);(axis|allies|world|none)?;([^;]{1,24});(-?[A-Fa-f0-9_]{1,32}|bot[0-9]+|0)?;(-?[0-9]+);(axis|allies|world|none)?;([^;]{1,24})?;((?:[0-9]+|[a-z]+|_|\+)+);([0-9]+);((?:[A-Z]|_)+);((?:[a-z]|_)+)$";
|
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.AddMapping(ParserRegex.GroupType.EventType, 1);
|
Configuration.Damage.AddMapping(ParserRegex.GroupType.EventType, 1);
|
||||||
Configuration.Damage.AddMapping(ParserRegex.GroupType.TargetNetworkId, 2);
|
Configuration.Damage.AddMapping(ParserRegex.GroupType.TargetNetworkId, 2);
|
||||||
Configuration.Damage.AddMapping(ParserRegex.GroupType.TargetClientNumber, 3);
|
Configuration.Damage.AddMapping(ParserRegex.GroupType.TargetClientNumber, 3);
|
||||||
@ -65,7 +72,7 @@ namespace IW4MAdmin.Application.EventParsers
|
|||||||
Configuration.Damage.AddMapping(ParserRegex.GroupType.MeansOfDeath, 12);
|
Configuration.Damage.AddMapping(ParserRegex.GroupType.MeansOfDeath, 12);
|
||||||
Configuration.Damage.AddMapping(ParserRegex.GroupType.HitLocation, 13);
|
Configuration.Damage.AddMapping(ParserRegex.GroupType.HitLocation, 13);
|
||||||
|
|
||||||
Configuration.Kill.Pattern = @"^(K);(-?[A-Fa-f0-9_]{1,32}|bot[0-9]+|0);(-?[0-9]+);(axis|allies|world|none)?;([^;]{1,24});(-?[A-Fa-f0-9_]{1,32}|bot[0-9]+|0)?;(-?[0-9]+);(axis|allies|world|none)?;([^;]{1,24})?;((?:[0-9]+|[a-z]+|_|\+)+);([0-9]+);((?:[A-Z]|_)+);((?:[a-z]|_)+)$";
|
Configuration.Kill.Pattern = @"^(K);(-?[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.Kill.AddMapping(ParserRegex.GroupType.EventType, 1);
|
Configuration.Kill.AddMapping(ParserRegex.GroupType.EventType, 1);
|
||||||
Configuration.Kill.AddMapping(ParserRegex.GroupType.TargetNetworkId, 2);
|
Configuration.Kill.AddMapping(ParserRegex.GroupType.TargetNetworkId, 2);
|
||||||
Configuration.Kill.AddMapping(ParserRegex.GroupType.TargetClientNumber, 3);
|
Configuration.Kill.AddMapping(ParserRegex.GroupType.TargetClientNumber, 3);
|
||||||
@ -90,7 +97,8 @@ namespace IW4MAdmin.Application.EventParsers
|
|||||||
{Configuration.Say, GameEvent.EventType.Say},
|
{Configuration.Say, GameEvent.EventType.Say},
|
||||||
{Configuration.Kill, GameEvent.EventType.Kill},
|
{Configuration.Kill, GameEvent.EventType.Kill},
|
||||||
{Configuration.MapChange, GameEvent.EventType.MapChange},
|
{Configuration.MapChange, GameEvent.EventType.MapChange},
|
||||||
{Configuration.MapEnd, GameEvent.EventType.MapEnd}
|
{Configuration.MapEnd, GameEvent.EventType.MapEnd},
|
||||||
|
{Configuration.JoinTeam, GameEvent.EventType.JoinTeam}
|
||||||
};
|
};
|
||||||
|
|
||||||
_eventTypeMap = new Dictionary<string, GameEvent.EventType>
|
_eventTypeMap = new Dictionary<string, GameEvent.EventType>
|
||||||
@ -100,7 +108,8 @@ 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},
|
||||||
{"Q", GameEvent.EventType.PreDisconnect},
|
{"JT", GameEvent.EventType.JoinTeam},
|
||||||
|
{"Q", GameEvent.EventType.PreDisconnect}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -322,6 +331,47 @@ 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]];
|
||||||
|
|
||||||
|
if (Configuration.TeamMapping.ContainsKey(team))
|
||||||
|
{
|
||||||
|
team = Configuration.TeamMapping[team].ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
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.Origin,
|
||||||
|
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);
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
using SharedLibraryCore.Interfaces;
|
using System.Collections.Generic;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
using SharedLibraryCore;
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Database.Models;
|
||||||
|
|
||||||
namespace IW4MAdmin.Application.EventParsers
|
namespace IW4MAdmin.Application.EventParsers
|
||||||
{
|
{
|
||||||
@ -8,11 +10,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; }
|
||||||
@ -22,10 +25,13 @@ namespace IW4MAdmin.Application.EventParsers
|
|||||||
public ParserRegex MapEnd { get; set; }
|
public ParserRegex MapEnd { get; set; }
|
||||||
public NumberStyles GuidNumberStyle { get; set; } = NumberStyles.HexNumber;
|
public NumberStyles GuidNumberStyle { get; set; } = NumberStyles.HexNumber;
|
||||||
|
|
||||||
|
public Dictionary<string, EFClient.TeamType> TeamMapping { get; set; } = new();
|
||||||
|
|
||||||
public DynamicEventParserConfiguration(IParserRegexFactory parserRegexFactory)
|
public DynamicEventParserConfiguration(IParserRegexFactory parserRegexFactory)
|
||||||
{
|
{
|
||||||
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();
|
||||||
|
@ -1,6 +1,10 @@
|
|||||||
using IW4MAdmin.Application.Misc;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using IW4MAdmin.Application.Misc;
|
||||||
using SharedLibraryCore.Interfaces;
|
using SharedLibraryCore.Interfaces;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using SharedLibraryCore;
|
||||||
|
using SharedLibraryCore.Configuration;
|
||||||
|
|
||||||
namespace IW4MAdmin.Application.Extensions
|
namespace IW4MAdmin.Application.Extensions
|
||||||
{
|
{
|
||||||
@ -13,9 +17,19 @@ namespace IW4MAdmin.Application.Extensions
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static string CommandConfigNameForType(this IManagerCommand command)
|
public static string CommandConfigNameForType(this IManagerCommand command)
|
||||||
{
|
{
|
||||||
return command.GetType() == typeof(ScriptCommand) ?
|
return command.GetType() == typeof(ScriptCommand)
|
||||||
$"{char.ToUpper(command.Name[0])}{command.Name.Substring(1)}Command" :
|
? $"{char.ToUpper(command.Name[0])}{command.Name.Substring(1)}Command"
|
||||||
command.GetType().Name;
|
: command.GetType().Name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static IList<Map> FindMap(this Server server, string mapName) => server.Maps.Where(map =>
|
||||||
|
map.Name.Equals(mapName, StringComparison.InvariantCultureIgnoreCase) ||
|
||||||
|
map.Alias.Equals(mapName, StringComparison.InvariantCultureIgnoreCase)).ToList();
|
||||||
|
|
||||||
|
public static IList<Gametype> FindGametype(this DefaultSettings settings, string gameType, Server.Game? game = null) =>
|
||||||
|
settings.Gametypes?.Where(gt => game == null || gt.Game == game)
|
||||||
|
.SelectMany(gt => gt.Gametypes).Where(gt =>
|
||||||
|
gt.Alias.Contains(gameType, StringComparison.CurrentCultureIgnoreCase) ||
|
||||||
|
gt.Name.Contains(gameType, StringComparison.CurrentCultureIgnoreCase)).ToList();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -78,8 +78,10 @@ namespace IW4MAdmin.Application.Extensions
|
|||||||
case "mysql":
|
case "mysql":
|
||||||
var appendTimeout = !appConfig.ConnectionString.Contains("default command timeout",
|
var appendTimeout = !appConfig.ConnectionString.Contains("default command timeout",
|
||||||
StringComparison.InvariantCultureIgnoreCase);
|
StringComparison.InvariantCultureIgnoreCase);
|
||||||
|
var connectionString =
|
||||||
|
appConfig.ConnectionString + (appendTimeout ? ";default command timeout=0" : "");
|
||||||
services.AddSingleton(sp => (DbContextOptions) new DbContextOptionsBuilder<MySqlDatabaseContext>()
|
services.AddSingleton(sp => (DbContextOptions) new DbContextOptionsBuilder<MySqlDatabaseContext>()
|
||||||
.UseMySql(appConfig.ConnectionString + (appendTimeout ? ";default command timeout=0" : ""),
|
.UseMySql(connectionString, ServerVersion.AutoDetect(connectionString),
|
||||||
mysqlOptions => mysqlOptions.EnableRetryOnFailure())
|
mysqlOptions => mysqlOptions.EnableRetryOnFailure())
|
||||||
.UseLoggerFactory(sp.GetRequiredService<ILoggerFactory>()).Options);
|
.UseLoggerFactory(sp.GetRequiredService<ILoggerFactory>()).Options);
|
||||||
return services;
|
return services;
|
||||||
@ -92,7 +94,7 @@ namespace IW4MAdmin.Application.Extensions
|
|||||||
postgresqlOptions =>
|
postgresqlOptions =>
|
||||||
{
|
{
|
||||||
postgresqlOptions.EnableRetryOnFailure();
|
postgresqlOptions.EnableRetryOnFailure();
|
||||||
postgresqlOptions.SetPostgresVersion(new Version("9.4"));
|
postgresqlOptions.SetPostgresVersion(new Version("12.9"));
|
||||||
})
|
})
|
||||||
.UseLoggerFactory(sp.GetRequiredService<ILoggerFactory>()).Options);
|
.UseLoggerFactory(sp.GetRequiredService<ILoggerFactory>()).Options);
|
||||||
return services;
|
return services;
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
using IW4MAdmin.Application.Misc;
|
using System.Threading.Tasks;
|
||||||
|
using IW4MAdmin.Application.Misc;
|
||||||
using SharedLibraryCore.Interfaces;
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
namespace IW4MAdmin.Application.Factories
|
namespace IW4MAdmin.Application.Factories
|
||||||
@ -17,7 +18,17 @@ namespace IW4MAdmin.Application.Factories
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public IConfigurationHandler<T> GetConfigurationHandler<T>(string name) where T : IBaseConfiguration
|
public IConfigurationHandler<T> GetConfigurationHandler<T>(string name) where T : IBaseConfiguration
|
||||||
{
|
{
|
||||||
return new BaseConfigurationHandler<T>(name);
|
var handler = new BaseConfigurationHandler<T>(name);
|
||||||
|
handler.BuildAsync().Wait();
|
||||||
|
return handler;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc/>
|
||||||
|
public async Task<IConfigurationHandler<T>> GetConfigurationHandlerAsync<T>(string name) where T : IBaseConfiguration
|
||||||
|
{
|
||||||
|
var handler = new BaseConfigurationHandler<T>(name);
|
||||||
|
await handler.BuildAsync();
|
||||||
|
return handler;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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}\"");
|
||||||
|
@ -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;
|
||||||
|
@ -6,6 +6,7 @@ using SharedLibraryCore.Interfaces;
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
using Data.Models.Client;
|
using Data.Models.Client;
|
||||||
using Microsoft.Extensions.DependencyInjection;
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
@ -30,7 +31,7 @@ namespace IW4MAdmin.Application.Factories
|
|||||||
|
|
||||||
/// <inheritdoc/>
|
/// <inheritdoc/>
|
||||||
public IManagerCommand CreateScriptCommand(string name, string alias, string description, string permission,
|
public IManagerCommand CreateScriptCommand(string name, string alias, string description, string permission,
|
||||||
bool isTargetRequired, IEnumerable<(string, bool)> args, Action<GameEvent> executeAction)
|
bool isTargetRequired, IEnumerable<(string, bool)> args, Func<GameEvent, Task> executeAction, Server.Game[] supportedGames)
|
||||||
{
|
{
|
||||||
var permissionEnum = Enum.Parse<EFClient.Permission>(permission);
|
var permissionEnum = Enum.Parse<EFClient.Permission>(permission);
|
||||||
var argsArray = args.Select(_arg => new CommandArgument
|
var argsArray = args.Select(_arg => new CommandArgument
|
||||||
@ -40,7 +41,7 @@ namespace IW4MAdmin.Application.Factories
|
|||||||
}).ToArray();
|
}).ToArray();
|
||||||
|
|
||||||
return new ScriptCommand(name, alias, description, isTargetRequired, permissionEnum, argsArray, executeAction,
|
return new ScriptCommand(name, alias, description, isTargetRequired, permissionEnum, argsArray, executeAction,
|
||||||
_config, _transLookup, _serviceProvider.GetRequiredService<ILogger<ScriptCommand>>());
|
_config, _transLookup, _serviceProvider.GetRequiredService<ILogger<ScriptCommand>>(), supportedGames);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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)
|
||||||
{
|
{
|
||||||
|
@ -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>();
|
||||||
|
@ -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);
|
||||||
|
163
Application/IO/NetworkGameLogReader.cs
Normal file
163
Application/IO/NetworkGameLogReader.cs
Normal 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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
138
Application/IO/NetworkLogState.cs
Normal file
138
Application/IO/NetworkLogState.cs
Normal 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();
|
||||||
|
}
|
||||||
|
}
|
@ -12,6 +12,7 @@ using System.Collections.Generic;
|
|||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Sockets;
|
||||||
using System.Runtime.InteropServices;
|
using System.Runtime.InteropServices;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
@ -23,6 +24,7 @@ using Serilog.Context;
|
|||||||
using static SharedLibraryCore.Database.Models.EFClient;
|
using static SharedLibraryCore.Database.Models.EFClient;
|
||||||
using Data.Models;
|
using Data.Models;
|
||||||
using Data.Models.Server;
|
using Data.Models.Server;
|
||||||
|
using IW4MAdmin.Application.Commands;
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
using static Data.Models.Client.EFClient;
|
using static Data.Models.Client.EFClient;
|
||||||
|
|
||||||
@ -33,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;
|
||||||
|
|
||||||
@ -47,15 +49,17 @@ 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>>(),
|
||||||
|
#pragma warning disable CS0612
|
||||||
serviceProvider.GetRequiredService<SharedLibraryCore.Interfaces.ILogger>(),
|
serviceProvider.GetRequiredService<SharedLibraryCore.Interfaces.ILogger>(),
|
||||||
|
#pragma warning restore CS0612
|
||||||
serverConfiguration,
|
serverConfiguration,
|
||||||
serviceProvider.GetRequiredService<IManager>(),
|
serviceProvider.GetRequiredService<IManager>(),
|
||||||
serviceProvider.GetRequiredService<IRConConnectionFactory>(),
|
serviceProvider.GetRequiredService<IRConConnectionFactory>(),
|
||||||
serviceProvider.GetRequiredService<IGameLogReaderFactory>())
|
serviceProvider.GetRequiredService<IGameLogReaderFactory>(), serviceProvider)
|
||||||
{
|
{
|
||||||
_translationLookup = lookup;
|
_translationLookup = lookup;
|
||||||
_metaService = metaService;
|
_metaService = metaService;
|
||||||
@ -92,6 +96,8 @@ namespace IW4MAdmin
|
|||||||
client.ClientNumber = clientFromLog.ClientNumber;
|
client.ClientNumber = clientFromLog.ClientNumber;
|
||||||
client.Score = clientFromLog.Score;
|
client.Score = clientFromLog.Score;
|
||||||
client.Ping = clientFromLog.Ping;
|
client.Ping = clientFromLog.Ping;
|
||||||
|
client.Team = clientFromLog.Team;
|
||||||
|
client.TeamName = clientFromLog.TeamName;
|
||||||
client.CurrentServer = this;
|
client.CurrentServer = this;
|
||||||
client.State = ClientState.Connecting;
|
client.State = ClientState.Connecting;
|
||||||
|
|
||||||
@ -232,7 +238,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;
|
||||||
}
|
}
|
||||||
@ -242,11 +248,16 @@ namespace IW4MAdmin
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
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"]);
|
Console.WriteLine(loc["SERVER_PLUGIN_ERROR"].FormatExt(plugin.Name, ex.GetType().Name));
|
||||||
ServerLogger.LogError(ex, "Could not execute {methodName} for plugin {plugin}",
|
ServerLogger.LogError(ex, "Could not execute {methodName} for plugin {plugin}",
|
||||||
nameof(plugin.OnEventAsync), plugin.Name);
|
nameof(plugin.OnEventAsync), plugin.Name);
|
||||||
}
|
}
|
||||||
@ -286,7 +297,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,
|
||||||
@ -300,7 +311,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());
|
||||||
@ -312,13 +323,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}",
|
||||||
@ -341,7 +352,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)
|
||||||
{
|
{
|
||||||
@ -351,7 +363,7 @@ namespace IW4MAdmin
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
var factory = _serviceProvider.GetRequiredService<IDatabaseContextFactory>();
|
var factory = _serviceProvider.GetRequiredService<IDatabaseContextFactory>();
|
||||||
await using var context = factory.CreateContext();
|
await using var context = factory.CreateContext(enableTracking: false);
|
||||||
|
|
||||||
var messageCount = await context.InboxMessages
|
var messageCount = await context.InboxMessages
|
||||||
.CountAsync(msg => msg.DestinationClientId == E.Origin.ClientId && !msg.IsDelivered);
|
.CountAsync(msg => msg.DestinationClientId == E.Origin.ClientId && !msg.IsDelivered);
|
||||||
@ -419,6 +431,7 @@ namespace IW4MAdmin
|
|||||||
Clients[E.Origin.ClientNumber] = E.Origin;
|
Clients[E.Origin.ClientNumber] = E.Origin;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
E.Origin.GameName = (Reference.Game?)GameName;
|
||||||
E.Origin = await OnClientConnected(E.Origin);
|
E.Origin = await OnClientConnected(E.Origin);
|
||||||
E.Target = E.Origin;
|
E.Target = E.Origin;
|
||||||
}
|
}
|
||||||
@ -433,7 +446,7 @@ namespace IW4MAdmin
|
|||||||
|
|
||||||
if (E.Origin.Level > Permission.Moderator)
|
if (E.Origin.Level > Permission.Moderator)
|
||||||
{
|
{
|
||||||
E.Origin.Tell(string.Format(loc["SERVER_REPORT_COUNT"], E.Owner.Reports.Count));
|
E.Origin.Tell(loc["SERVER_REPORT_COUNT_V2"].FormatExt(E.Owner.Reports.Count));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -473,7 +486,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,
|
||||||
@ -485,7 +498,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);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -495,7 +509,8 @@ namespace IW4MAdmin
|
|||||||
{
|
{
|
||||||
Origin = E.Origin,
|
Origin = E.Origin,
|
||||||
Target = E.Target,
|
Target = E.Target,
|
||||||
Reason = E.Data
|
Reason = E.Data,
|
||||||
|
ReportedOn = DateTime.UtcNow
|
||||||
});
|
});
|
||||||
|
|
||||||
var newReport = new EFPenalty()
|
var newReport = new EFPenalty()
|
||||||
@ -558,8 +573,8 @@ 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);
|
await _metaService.SetPersistentMeta("LastServerPlayed", E.Owner.Hostname, E.Origin.ClientId);
|
||||||
}
|
}
|
||||||
|
|
||||||
else if (E.Type == GameEvent.EventType.PreDisconnect)
|
else if (E.Type == GameEvent.EventType.PreDisconnect)
|
||||||
@ -610,7 +625,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)
|
||||||
{
|
{
|
||||||
@ -630,7 +645,7 @@ namespace IW4MAdmin
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ChatHistory.Add(new ChatInfo()
|
ChatHistory.Add(new ChatInfo
|
||||||
{
|
{
|
||||||
Name = E.Origin.Name,
|
Name = E.Origin.Name,
|
||||||
Message = message,
|
Message = message,
|
||||||
@ -640,7 +655,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);
|
||||||
|
|
||||||
@ -681,7 +696,7 @@ namespace IW4MAdmin
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (E.Type == GameEvent.EventType.MapEnd)
|
else if (E.Type == GameEvent.EventType.MapEnd)
|
||||||
{
|
{
|
||||||
ServerLogger.LogInformation("Game ending...");
|
ServerLogger.LogInformation("Game ending...");
|
||||||
|
|
||||||
@ -691,12 +706,12 @@ 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
|
||||||
{
|
{
|
||||||
@ -704,6 +719,11 @@ namespace IW4MAdmin
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
else if (E.Type == GameEvent.EventType.JoinTeam)
|
||||||
|
{
|
||||||
|
E.Origin.UpdateTeam(E.Extra as string);
|
||||||
|
}
|
||||||
|
|
||||||
lock (ChatHistory)
|
lock (ChatHistory)
|
||||||
{
|
{
|
||||||
while (ChatHistory.Count > Math.Ceiling(ClientNum / 2.0))
|
while (ChatHistory.Count > Math.Ceiling(ClientNum / 2.0))
|
||||||
@ -725,11 +745,11 @@ namespace IW4MAdmin
|
|||||||
|
|
||||||
private async Task OnClientUpdate(EFClient origin)
|
private async Task OnClientUpdate(EFClient origin)
|
||||||
{
|
{
|
||||||
var client = GetClientsAsList().FirstOrDefault(_client => _client.Equals(origin));
|
var client = Manager.GetActiveClients().FirstOrDefault(c => c.NetworkId == origin.NetworkId);
|
||||||
|
|
||||||
if (client == null)
|
if (client == null)
|
||||||
{
|
{
|
||||||
ServerLogger.LogWarning("{origin} expected to exist in client list for update, but they do not", origin.ToString());
|
ServerLogger.LogWarning("{Origin} expected to exist in client list for update, but they do not", origin.ToString());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -755,10 +775,10 @@ namespace IW4MAdmin
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
else if ((client.IPAddress != null && client.State == ClientState.Disconnecting) ||
|
else if (client.IPAddress != null && client.State == ClientState.Disconnecting ||
|
||||||
client.Level == Permission.Banned)
|
client.Level == Permission.Banned)
|
||||||
{
|
{
|
||||||
ServerLogger.LogWarning("{client} state is Unknown (probably kicked), but they are still connected. trying to kick again...", origin.ToString());
|
ServerLogger.LogWarning("{Client} state is Unknown (probably kicked), but they are still connected. trying to kick again...", origin.ToString());
|
||||||
await client.CanConnect(client.IPAddress, Manager.GetApplicationSettings().Configuration().EnableImplicitAccountLinking);
|
await client.CanConnect(client.IPAddress, Manager.GetApplicationSettings().Configuration().EnableImplicitAccountLinking);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -772,8 +792,16 @@ namespace IW4MAdmin
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
async Task<List<EFClient>[]> PollPlayersAsync()
|
async Task<List<EFClient>[]> PollPlayersAsync()
|
||||||
{
|
{
|
||||||
|
var tokenSource = new CancellationTokenSource();
|
||||||
|
tokenSource.CancelAfter(TimeSpan.FromSeconds(5));
|
||||||
var currentClients = GetClientsAsList();
|
var currentClients = GetClientsAsList();
|
||||||
var statusResponse = (await this.GetStatusAsync());
|
var statusResponse = await this.GetStatusAsync(tokenSource.Token);
|
||||||
|
|
||||||
|
if (statusResponse is null)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
var polledClients = statusResponse.Clients.AsEnumerable();
|
var polledClients = statusResponse.Clients.AsEnumerable();
|
||||||
|
|
||||||
if (Manager.GetApplicationSettings().Configuration().IgnoreBots)
|
if (Manager.GetApplicationSettings().Configuration().IgnoreBots)
|
||||||
@ -885,16 +913,10 @@ namespace IW4MAdmin
|
|||||||
|
|
||||||
await e.WaitAsync(Utilities.DefaultCommandTimeout, new CancellationTokenRegistration().Token);
|
await e.WaitAsync(Utilities.DefaultCommandTimeout, new CancellationTokenRegistration().Token);
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (var plugin in Manager.Plugins)
|
|
||||||
{
|
|
||||||
await plugin.OnUnloadAsync();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
DateTime start = DateTime.Now;
|
private DateTime _lastMessageSent = DateTime.Now;
|
||||||
DateTime playerCountStart = DateTime.Now;
|
private DateTime _lastPlayerCount = DateTime.Now;
|
||||||
DateTime lastCount = DateTime.Now;
|
|
||||||
|
|
||||||
public override async Task<bool> ProcessUpdatesAsync(CancellationToken cts)
|
public override async Task<bool> ProcessUpdatesAsync(CancellationToken cts)
|
||||||
{
|
{
|
||||||
@ -908,14 +930,21 @@ namespace IW4MAdmin
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (Manager.GetApplicationSettings().Configuration().RConPollRate == int.MaxValue && Utilities.IsDevelopment)
|
if (Manager.GetApplicationSettings().Configuration().RConPollRate == int.MaxValue &&
|
||||||
|
Utilities.IsDevelopment)
|
||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
var polledClients = await PollPlayersAsync();
|
var polledClients = await PollPlayersAsync();
|
||||||
|
|
||||||
foreach (var disconnectingClient in polledClients[1].Where(_client => !_client.IsZombieClient /* ignores "fake" zombie clients */))
|
if (polledClients is null)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var disconnectingClient in polledClients[1]
|
||||||
|
.Where(client => !client.IsZombieClient /* ignores "fake" zombie clients */))
|
||||||
{
|
{
|
||||||
disconnectingClient.CurrentServer = this;
|
disconnectingClient.CurrentServer = this;
|
||||||
var e = new GameEvent()
|
var e = new GameEvent()
|
||||||
@ -931,23 +960,20 @@ namespace IW4MAdmin
|
|||||||
}
|
}
|
||||||
|
|
||||||
// this are our new connecting clients
|
// this are our new connecting clients
|
||||||
foreach (var client in polledClients[0])
|
foreach (var client in polledClients[0].Where(client =>
|
||||||
|
!string.IsNullOrEmpty(client.Name) && (client.Ping != 999 || client.IsBot)))
|
||||||
{
|
{
|
||||||
// note: this prevents players in ZMBI state from being registered with no name
|
|
||||||
if (string.IsNullOrEmpty(client.Name) || (client.Ping == 999 && !client.IsBot))
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
client.CurrentServer = this;
|
client.CurrentServer = this;
|
||||||
var e = new GameEvent()
|
client.GameName = (Reference.Game?)GameName;
|
||||||
|
|
||||||
|
var e = new GameEvent
|
||||||
{
|
{
|
||||||
Type = GameEvent.EventType.PreConnect,
|
Type = GameEvent.EventType.PreConnect,
|
||||||
Origin = client,
|
Origin = client,
|
||||||
Owner = this,
|
Owner = this,
|
||||||
IsBlocking = true,
|
IsBlocking = true,
|
||||||
Extra = client.GetAdditionalProperty<string>("BotGuid"),
|
Extra = client.GetAdditionalProperty<string>("BotGuid"),
|
||||||
Source = GameEvent.EventSource.Status
|
Source = GameEvent.EventSource.Status,
|
||||||
};
|
};
|
||||||
|
|
||||||
Manager.AddEvent(e);
|
Manager.AddEvent(e);
|
||||||
@ -958,19 +984,19 @@ namespace IW4MAdmin
|
|||||||
foreach (var client in polledClients[2])
|
foreach (var client in polledClients[2])
|
||||||
{
|
{
|
||||||
client.CurrentServer = this;
|
client.CurrentServer = this;
|
||||||
var e = new GameEvent()
|
var gameEvent = new GameEvent
|
||||||
{
|
{
|
||||||
Type = GameEvent.EventType.Update,
|
Type = GameEvent.EventType.Update,
|
||||||
Origin = client,
|
Origin = client,
|
||||||
Owner = this
|
Owner = this
|
||||||
};
|
};
|
||||||
|
|
||||||
Manager.AddEvent(e);
|
Manager.AddEvent(gameEvent);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Throttled)
|
if (Throttled)
|
||||||
{
|
{
|
||||||
var _event = new GameEvent()
|
var gameEvent = new GameEvent
|
||||||
{
|
{
|
||||||
Type = GameEvent.EventType.ConnectionRestored,
|
Type = GameEvent.EventType.ConnectionRestored,
|
||||||
Owner = this,
|
Owner = this,
|
||||||
@ -978,65 +1004,52 @@ namespace IW4MAdmin
|
|||||||
Target = Utilities.IW4MAdminClient(this)
|
Target = Utilities.IW4MAdminClient(this)
|
||||||
};
|
};
|
||||||
|
|
||||||
Manager.AddEvent(_event);
|
Manager.AddEvent(gameEvent);
|
||||||
}
|
}
|
||||||
|
|
||||||
LastPoll = DateTime.Now;
|
LastPoll = DateTime.Now;
|
||||||
}
|
}
|
||||||
|
|
||||||
catch (NetworkException e)
|
catch (NetworkException ex)
|
||||||
{
|
{
|
||||||
if (!Throttled)
|
if (Throttled)
|
||||||
{
|
{
|
||||||
var _event = new GameEvent()
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
var gameEvent = new GameEvent
|
||||||
{
|
{
|
||||||
Type = GameEvent.EventType.ConnectionLost,
|
Type = GameEvent.EventType.ConnectionLost,
|
||||||
Owner = this,
|
Owner = this,
|
||||||
Origin = Utilities.IW4MAdminClient(this),
|
Origin = Utilities.IW4MAdminClient(this),
|
||||||
Target = Utilities.IW4MAdminClient(this),
|
Target = Utilities.IW4MAdminClient(this),
|
||||||
Extra = e,
|
Extra = ex,
|
||||||
Data = ConnectionErrors.ToString()
|
Data = ConnectionErrors.ToString()
|
||||||
};
|
};
|
||||||
|
|
||||||
Manager.AddEvent(_event);
|
Manager.AddEvent(gameEvent);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
RunServerCollection();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (DateTime.Now - _lastMessageSent <=
|
||||||
|
TimeSpan.FromSeconds(Manager.GetApplicationSettings().Configuration().AutoMessagePeriod) ||
|
||||||
|
BroadcastMessages.Count <= 0 || ClientNum <= 0)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
LastMessage = DateTime.Now - start;
|
|
||||||
lastCount = DateTime.Now;
|
|
||||||
|
|
||||||
var appConfig = _serviceProvider.GetService<ApplicationConfiguration>();
|
|
||||||
// update the player history
|
|
||||||
if (lastCount - playerCountStart >= appConfig.ServerDataCollectionInterval)
|
|
||||||
{
|
|
||||||
var maxItems = Math.Ceiling(appConfig.MaxClientHistoryTime.TotalMinutes /
|
|
||||||
appConfig.ServerDataCollectionInterval.TotalMinutes);
|
|
||||||
while ( ClientHistory.Count > maxItems)
|
|
||||||
{
|
|
||||||
ClientHistory.Dequeue();
|
|
||||||
}
|
|
||||||
|
|
||||||
ClientHistory.Enqueue(new PlayerHistory(ClientNum));
|
|
||||||
playerCountStart = DateTime.Now;
|
|
||||||
}
|
|
||||||
|
|
||||||
// send out broadcast messages
|
// send out broadcast messages
|
||||||
if (LastMessage.TotalSeconds > Manager.GetApplicationSettings().Configuration().AutoMessagePeriod
|
var messages =
|
||||||
&& BroadcastMessages.Count > 0
|
(await this.ProcessMessageToken(Manager.GetMessageTokens(), BroadcastMessages[NextMessage])).Split(
|
||||||
&& ClientNum > 0)
|
Environment.NewLine);
|
||||||
{
|
await BroadcastAsync(messages, token: Manager.CancellationToken);
|
||||||
string[] messages = (await this.ProcessMessageToken(Manager.GetMessageTokens(), BroadcastMessages[NextMessage])).Split(Environment.NewLine);
|
|
||||||
|
|
||||||
foreach (string message in messages)
|
NextMessage = NextMessage == BroadcastMessages.Count - 1 ? 0 : NextMessage + 1;
|
||||||
{
|
_lastMessageSent = DateTime.Now;
|
||||||
Broadcast(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
NextMessage = NextMessage == (BroadcastMessages.Count - 1) ? 0 : NextMessage + 1;
|
|
||||||
start = DateTime.Now;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -1054,6 +1067,7 @@ namespace IW4MAdmin
|
|||||||
{
|
{
|
||||||
ServerLogger.LogWarning(e, "Undesirable exception occured during processing updates");
|
ServerLogger.LogWarning(e, "Undesirable exception occured during processing updates");
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1063,16 +1077,48 @@ namespace IW4MAdmin
|
|||||||
{
|
{
|
||||||
ServerLogger.LogError(e, "Unexpected exception occured during processing updates");
|
ServerLogger.LogError(e, "Unexpected exception occured during processing updates");
|
||||||
}
|
}
|
||||||
|
|
||||||
Console.WriteLine(loc["SERVER_ERROR_EXCEPTION"].FormatExt($"[{IP}:{Port}]"));
|
Console.WriteLine(loc["SERVER_ERROR_EXCEPTION"].FormatExt($"[{IP}:{Port}]"));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void RunServerCollection()
|
||||||
|
{
|
||||||
|
var appConfig = _serviceProvider.GetService<ApplicationConfiguration>();
|
||||||
|
|
||||||
|
if (DateTime.Now - _lastPlayerCount < appConfig?.ServerDataCollectionInterval)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var maxItems = Math.Ceiling(appConfig.MaxClientHistoryTime.TotalMinutes /
|
||||||
|
appConfig.ServerDataCollectionInterval.TotalMinutes);
|
||||||
|
|
||||||
|
while (ClientHistory.ClientCounts.Count > maxItems)
|
||||||
|
{
|
||||||
|
ClientHistory.ClientCounts.RemoveAt(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
ClientHistory.ClientCounts.Add(new ClientCountSnapshot
|
||||||
|
{
|
||||||
|
ClientCount = ClientNum,
|
||||||
|
ConnectionInterrupted = Throttled,
|
||||||
|
Time = DateTime.UtcNow,
|
||||||
|
Map = CurrentMap.Name
|
||||||
|
});
|
||||||
|
|
||||||
|
_lastPlayerCount = DateTime.Now;
|
||||||
|
}
|
||||||
|
|
||||||
public async Task Initialize()
|
public async Task Initialize()
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
ResolvedIpEndPoint = new IPEndPoint((await Dns.GetHostAddressesAsync(IP)).First(), Port);
|
ResolvedIpEndPoint =
|
||||||
|
new IPEndPoint(
|
||||||
|
(await Dns.GetHostAddressesAsync(IP)).First(address =>
|
||||||
|
address.AddressFamily == AddressFamily.InterNetwork), Port);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@ -1081,10 +1127,14 @@ namespace IW4MAdmin
|
|||||||
}
|
}
|
||||||
|
|
||||||
RconParser = Manager.AdditionalRConParsers
|
RconParser = Manager.AdditionalRConParsers
|
||||||
.FirstOrDefault(_parser => _parser.Version == ServerConfig.RConParserVersion);
|
.FirstOrDefault(parser =>
|
||||||
|
parser.Version == ServerConfig.RConParserVersion ||
|
||||||
|
parser.Name == ServerConfig.RConParserVersion);
|
||||||
|
|
||||||
EventParser = Manager.AdditionalEventParsers
|
EventParser = Manager.AdditionalEventParsers
|
||||||
.FirstOrDefault(_parser => _parser.Version == ServerConfig.EventParserVersion);
|
.FirstOrDefault(parser =>
|
||||||
|
parser.Version == ServerConfig.EventParserVersion ||
|
||||||
|
parser.Name == ServerConfig.RConParserVersion);
|
||||||
|
|
||||||
RconParser ??= Manager.AdditionalRConParsers[0];
|
RconParser ??= Manager.AdditionalRConParsers[0];
|
||||||
EventParser ??= Manager.AdditionalEventParsers[0];
|
EventParser ??= Manager.AdditionalEventParsers[0];
|
||||||
@ -1092,7 +1142,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);
|
||||||
|
|
||||||
@ -1101,7 +1151,7 @@ namespace IW4MAdmin
|
|||||||
GameName = RconParser.GameName;
|
GameName = RconParser.GameName;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (version?.Value?.Length != 0)
|
if (version.Value?.Length != 0)
|
||||||
{
|
{
|
||||||
var matchedRconParser = Manager.AdditionalRConParsers.FirstOrDefault(_parser => _parser.Version == version.Value);
|
var matchedRconParser = Manager.AdditionalRConParsers.FirstOrDefault(_parser => _parser.Version == version.Value);
|
||||||
RconParser.Configuration = matchedRconParser != null ? matchedRconParser.Configuration : RconParser.Configuration;
|
RconParser.Configuration = matchedRconParser != null ? matchedRconParser.Configuration : RconParser.Configuration;
|
||||||
@ -1109,7 +1159,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")
|
||||||
{
|
{
|
||||||
@ -1118,27 +1168,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
|
||||||
@ -1184,14 +1235,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;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1203,7 +1254,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();
|
||||||
@ -1321,12 +1372,9 @@ namespace IW4MAdmin
|
|||||||
public override async Task Warn(string reason, EFClient targetClient, EFClient targetOrigin)
|
public override async Task Warn(string reason, EFClient targetClient, EFClient targetOrigin)
|
||||||
{
|
{
|
||||||
// ensure player gets warned if command not performed on them in game
|
// ensure player gets warned if command not performed on them in game
|
||||||
targetClient = targetClient.ClientNumber < 0 ?
|
var activeClient = Manager.FindActiveClient(targetClient);
|
||||||
Manager.GetActiveClients()
|
|
||||||
.FirstOrDefault(c => c.ClientId == targetClient?.ClientId) ?? targetClient :
|
|
||||||
targetClient;
|
|
||||||
|
|
||||||
var newPenalty = new EFPenalty()
|
var newPenalty = new EFPenalty
|
||||||
{
|
{
|
||||||
Type = EFPenalty.PenaltyType.Warning,
|
Type = EFPenalty.PenaltyType.Warning,
|
||||||
Expires = DateTime.UtcNow,
|
Expires = DateTime.UtcNow,
|
||||||
@ -1336,31 +1384,28 @@ namespace IW4MAdmin
|
|||||||
Link = targetClient.AliasLink
|
Link = targetClient.AliasLink
|
||||||
};
|
};
|
||||||
|
|
||||||
ServerLogger.LogDebug("Creating warn penalty for {targetClient}", targetClient.ToString());
|
ServerLogger.LogDebug("Creating warn penalty for {TargetClient}", targetClient.ToString());
|
||||||
await newPenalty.TryCreatePenalty(Manager.GetPenaltyService(), ServerLogger);
|
await newPenalty.TryCreatePenalty(Manager.GetPenaltyService(), ServerLogger);
|
||||||
|
|
||||||
if (targetClient.IsIngame)
|
if (activeClient.IsIngame)
|
||||||
{
|
{
|
||||||
if (targetClient.Warnings >= 4)
|
if (activeClient.Warnings >= 4)
|
||||||
{
|
{
|
||||||
targetClient.Kick(loc["SERVER_WARNLIMT_REACHED"], Utilities.IW4MAdminClient(this));
|
activeClient.Kick(loc["SERVER_WARNLIMT_REACHED"], Utilities.IW4MAdminClient(this));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// todo: move to translation sheet
|
var message = loc["COMMANDS_WARNING_FORMAT_V2"]
|
||||||
string message = $"^1{loc["SERVER_WARNING"]} ^7[^3{targetClient.Warnings}^7]: ^3{targetClient.Name}^7, {reason}";
|
.FormatExt(activeClient.Warnings, activeClient.Name, reason);
|
||||||
targetClient.CurrentServer.Broadcast(message);
|
activeClient.CurrentServer.Broadcast(message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override async Task Kick(string reason, EFClient targetClient, EFClient originClient, EFPenalty previousPenalty)
|
public override async Task Kick(string reason, EFClient targetClient, EFClient originClient, EFPenalty previousPenalty)
|
||||||
{
|
{
|
||||||
targetClient = targetClient.ClientNumber < 0 ?
|
var activeClient = Manager.FindActiveClient(targetClient);
|
||||||
Manager.GetActiveClients()
|
|
||||||
.FirstOrDefault(c => c.ClientId == targetClient?.ClientId) ?? targetClient :
|
|
||||||
targetClient;
|
|
||||||
|
|
||||||
var newPenalty = new EFPenalty()
|
var newPenalty = new EFPenalty
|
||||||
{
|
{
|
||||||
Type = EFPenalty.PenaltyType.Kick,
|
Type = EFPenalty.PenaltyType.Kick,
|
||||||
Expires = DateTime.UtcNow,
|
Expires = DateTime.UtcNow,
|
||||||
@ -1370,132 +1415,115 @@ namespace IW4MAdmin
|
|||||||
Link = targetClient.AliasLink
|
Link = targetClient.AliasLink
|
||||||
};
|
};
|
||||||
|
|
||||||
ServerLogger.LogDebug("Creating kick penalty for {targetClient}", targetClient.ToString());
|
ServerLogger.LogDebug("Creating kick penalty for {TargetClient}", targetClient.ToString());
|
||||||
await newPenalty.TryCreatePenalty(Manager.GetPenaltyService(), ServerLogger);
|
await newPenalty.TryCreatePenalty(Manager.GetPenaltyService(), ServerLogger);
|
||||||
|
|
||||||
if (targetClient.IsIngame)
|
if (activeClient.IsIngame)
|
||||||
{
|
{
|
||||||
var e = new GameEvent()
|
var gameEvent = new GameEvent
|
||||||
{
|
{
|
||||||
Type = GameEvent.EventType.PreDisconnect,
|
Type = GameEvent.EventType.PreDisconnect,
|
||||||
Origin = targetClient,
|
Origin = activeClient,
|
||||||
Owner = this
|
Owner = this
|
||||||
};
|
};
|
||||||
|
|
||||||
Manager.AddEvent(e);
|
Manager.AddEvent(gameEvent);
|
||||||
|
|
||||||
var temporalClientId = targetClient.GetAdditionalProperty<string>("ConnectionClientId");
|
|
||||||
var parsedClientId = string.IsNullOrEmpty(temporalClientId) ? (int?)null : int.Parse(temporalClientId);
|
|
||||||
var clientNumber = parsedClientId ?? targetClient.ClientNumber;
|
|
||||||
|
|
||||||
var formattedKick = string.Format(RconParser.Configuration.CommandPrefixes.Kick,
|
var formattedKick = string.Format(RconParser.Configuration.CommandPrefixes.Kick,
|
||||||
clientNumber,
|
activeClient.TemporalClientNumber,
|
||||||
_messageFormatter.BuildFormattedMessage(RconParser.Configuration,
|
_messageFormatter.BuildFormattedMessage(RconParser.Configuration,
|
||||||
newPenalty,
|
newPenalty,
|
||||||
previousPenalty));
|
previousPenalty));
|
||||||
await targetClient.CurrentServer.ExecuteCommandAsync(formattedKick);
|
ServerLogger.LogDebug("Executing tempban kick command for {ActiveClient}", activeClient.ToString());
|
||||||
|
await activeClient.CurrentServer.ExecuteCommandAsync(formattedKick);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override async Task TempBan(string Reason, TimeSpan length, EFClient targetClient, EFClient originClient)
|
public override async Task TempBan(string reason, TimeSpan length, EFClient targetClient, EFClient originClient)
|
||||||
{
|
{
|
||||||
// ensure player gets kicked if command not performed on them in the same server
|
// ensure player gets kicked if command not performed on them in the same server
|
||||||
targetClient = targetClient.ClientNumber < 0 ?
|
var activeClient = Manager.FindActiveClient(targetClient);
|
||||||
Manager.GetActiveClients()
|
|
||||||
.FirstOrDefault(c => c.ClientId == targetClient?.ClientId) ?? targetClient :
|
|
||||||
targetClient;
|
|
||||||
|
|
||||||
var newPenalty = new EFPenalty()
|
var newPenalty = new EFPenalty
|
||||||
{
|
{
|
||||||
Type = EFPenalty.PenaltyType.TempBan,
|
Type = EFPenalty.PenaltyType.TempBan,
|
||||||
Expires = DateTime.UtcNow + length,
|
Expires = DateTime.UtcNow + length,
|
||||||
Offender = targetClient,
|
Offender = targetClient,
|
||||||
Offense = Reason,
|
Offense = reason,
|
||||||
Punisher = originClient,
|
Punisher = originClient,
|
||||||
Link = targetClient.AliasLink
|
Link = targetClient.AliasLink
|
||||||
};
|
};
|
||||||
|
|
||||||
ServerLogger.LogDebug("Creating tempban penalty for {targetClient}", targetClient.ToString());
|
ServerLogger.LogDebug("Creating tempban penalty for {TargetClient}", targetClient.ToString());
|
||||||
await newPenalty.TryCreatePenalty(Manager.GetPenaltyService(), ServerLogger);
|
await newPenalty.TryCreatePenalty(Manager.GetPenaltyService(), ServerLogger);
|
||||||
|
|
||||||
if (targetClient.IsIngame)
|
if (activeClient.IsIngame)
|
||||||
{
|
{
|
||||||
var temporalClientId = targetClient.GetAdditionalProperty<string>("ConnectionClientId");
|
|
||||||
var parsedClientId = string.IsNullOrEmpty(temporalClientId) ? (int?)null : int.Parse(temporalClientId);
|
|
||||||
var clientNumber = parsedClientId ?? targetClient.ClientNumber;
|
|
||||||
|
|
||||||
var formattedKick = string.Format(RconParser.Configuration.CommandPrefixes.Kick,
|
var formattedKick = string.Format(RconParser.Configuration.CommandPrefixes.Kick,
|
||||||
clientNumber,
|
activeClient.TemporalClientNumber,
|
||||||
_messageFormatter.BuildFormattedMessage(RconParser.Configuration, newPenalty));
|
_messageFormatter.BuildFormattedMessage(RconParser.Configuration, newPenalty));
|
||||||
ServerLogger.LogDebug("Executing tempban kick command for {targetClient}", targetClient.ToString());
|
ServerLogger.LogDebug("Executing tempban kick command for {ActiveClient}", activeClient.ToString());
|
||||||
await targetClient.CurrentServer.ExecuteCommandAsync(formattedKick);
|
await activeClient.CurrentServer.ExecuteCommandAsync(formattedKick);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override async Task Ban(string reason, EFClient targetClient, EFClient originClient, bool isEvade = false)
|
public override async Task Ban(string reason, EFClient targetClient, EFClient originClient, bool isEvade = false)
|
||||||
{
|
{
|
||||||
// ensure player gets kicked if command not performed on them in the same server
|
// ensure player gets kicked if command not performed on them in the same server
|
||||||
targetClient = targetClient.ClientNumber < 0 ?
|
var activeClient = Manager.FindActiveClient(targetClient);
|
||||||
Manager.GetActiveClients()
|
|
||||||
.FirstOrDefault(c => c.ClientId == targetClient?.ClientId) ?? targetClient :
|
|
||||||
targetClient;
|
|
||||||
|
|
||||||
EFPenalty newPenalty = new EFPenalty()
|
var newPenalty = new EFPenalty
|
||||||
{
|
{
|
||||||
Type = EFPenalty.PenaltyType.Ban,
|
Type = EFPenalty.PenaltyType.Ban,
|
||||||
Expires = null,
|
Expires = null,
|
||||||
Offender = targetClient,
|
Offender = targetClient,
|
||||||
Offense = reason,
|
Offense = reason,
|
||||||
Punisher = originClient,
|
Punisher = originClient,
|
||||||
Link = targetClient.AliasLink,
|
|
||||||
IsEvadedOffense = isEvade
|
IsEvadedOffense = isEvade
|
||||||
};
|
};
|
||||||
|
|
||||||
ServerLogger.LogDebug("Creating ban penalty for {targetClient}", targetClient.ToString());
|
ServerLogger.LogDebug("Creating ban penalty for {TargetClient}", targetClient.ToString());
|
||||||
targetClient.SetLevel(Permission.Banned, originClient);
|
activeClient.SetLevel(Permission.Banned, originClient);
|
||||||
await newPenalty.TryCreatePenalty(Manager.GetPenaltyService(), ServerLogger);
|
await newPenalty.TryCreatePenalty(Manager.GetPenaltyService(), ServerLogger);
|
||||||
|
|
||||||
if (targetClient.IsIngame)
|
if (activeClient.IsIngame)
|
||||||
{
|
{
|
||||||
ServerLogger.LogDebug("Attempting to kicking newly banned client {targetClient}", targetClient.ToString());
|
ServerLogger.LogDebug("Attempting to kicking newly banned client {ActiveClient}", activeClient.ToString());
|
||||||
|
|
||||||
var temporalClientId = targetClient.GetAdditionalProperty<string>("ConnectionClientId");
|
|
||||||
var parsedClientId = string.IsNullOrEmpty(temporalClientId) ? (int?)null : int.Parse(temporalClientId);
|
|
||||||
var clientNumber = parsedClientId ?? targetClient.ClientNumber;
|
|
||||||
|
|
||||||
var formattedString = string.Format(RconParser.Configuration.CommandPrefixes.Kick,
|
var formattedString = string.Format(RconParser.Configuration.CommandPrefixes.Kick,
|
||||||
clientNumber,
|
activeClient.TemporalClientNumber,
|
||||||
_messageFormatter.BuildFormattedMessage(RconParser.Configuration, newPenalty));
|
_messageFormatter.BuildFormattedMessage(RconParser.Configuration, newPenalty));
|
||||||
await targetClient.CurrentServer.ExecuteCommandAsync(formattedString);
|
await activeClient.CurrentServer.ExecuteCommandAsync(formattedString);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override public async Task Unban(string reason, EFClient Target, EFClient Origin)
|
public override async Task Unban(string reason, EFClient targetClient, EFClient originClient)
|
||||||
{
|
{
|
||||||
var unbanPenalty = new EFPenalty()
|
var unbanPenalty = new EFPenalty
|
||||||
{
|
{
|
||||||
Type = EFPenalty.PenaltyType.Unban,
|
Type = EFPenalty.PenaltyType.Unban,
|
||||||
Expires = DateTime.Now,
|
Expires = DateTime.Now,
|
||||||
Offender = Target,
|
Offender = targetClient,
|
||||||
Offense = reason,
|
Offense = reason,
|
||||||
Punisher = Origin,
|
Punisher = originClient,
|
||||||
When = DateTime.UtcNow,
|
When = DateTime.UtcNow,
|
||||||
Active = true,
|
Active = true,
|
||||||
Link = Target.AliasLink
|
Link = targetClient.AliasLink
|
||||||
};
|
};
|
||||||
|
|
||||||
ServerLogger.LogDebug("Creating unban penalty for {targetClient}", Target.ToString());
|
ServerLogger.LogDebug("Creating unban penalty for {targetClient}", targetClient.ToString());
|
||||||
Target.SetLevel(Permission.User, Origin);
|
targetClient.SetLevel(Permission.User, originClient);
|
||||||
await Manager.GetPenaltyService().RemoveActivePenalties(Target.AliasLink.AliasLinkId);
|
await Manager.GetPenaltyService().RemoveActivePenalties(targetClient.AliasLink.AliasLinkId,
|
||||||
|
targetClient.NetworkId, targetClient.CurrentAlias?.IPAddress);
|
||||||
await Manager.GetPenaltyService().Create(unbanPenalty);
|
await Manager.GetPenaltyService().Create(unbanPenalty);
|
||||||
}
|
}
|
||||||
|
|
||||||
override public void InitializeTokens()
|
public override void InitializeTokens()
|
||||||
{
|
{
|
||||||
Manager.GetMessageTokens().Add(new MessageToken("TOTALPLAYERS", (Server s) => Task.Run(async () => (await Manager.GetClientService().GetTotalClientsAsync()).ToString())));
|
Manager.GetMessageTokens().Add(new MessageToken("TOTALPLAYERS", (Server s) => Task.Run(async () => (await Manager.GetClientService().GetTotalClientsAsync()).ToString())));
|
||||||
Manager.GetMessageTokens().Add(new MessageToken("VERSION", (Server s) => Task.FromResult(Application.Program.Version.ToString())));
|
Manager.GetMessageTokens().Add(new MessageToken("VERSION", (Server s) => Task.FromResult(Application.Program.Version.ToString())));
|
||||||
Manager.GetMessageTokens().Add(new MessageToken("NEXTMAP", (Server s) => SharedLibraryCore.Commands.NextMapCommand.GetNextMap(s, _translationLookup)));
|
Manager.GetMessageTokens().Add(new MessageToken("NEXTMAP", (Server s) => SharedLibraryCore.Commands.NextMapCommand.GetNextMap(s, _translationLookup)));
|
||||||
Manager.GetMessageTokens().Add(new MessageToken("ADMINS", (Server s) => Task.FromResult(SharedLibraryCore.Commands.ListAdminsCommand.OnlineAdmins(s, _translationLookup))));
|
Manager.GetMessageTokens().Add(new MessageToken("ADMINS", (Server s) => Task.FromResult(ListAdminsCommand.OnlineAdmins(s, _translationLookup))));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -18,7 +18,9 @@ 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.Text;
|
using System.Text;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
@ -33,16 +35,17 @@ using IW4MAdmin.Plugins.Stats.Client.Abstractions;
|
|||||||
using IW4MAdmin.Plugins.Stats.Client;
|
using IW4MAdmin.Plugins.Stats.Client;
|
||||||
using Stats.Client.Abstractions;
|
using Stats.Client.Abstractions;
|
||||||
using Stats.Client;
|
using Stats.Client;
|
||||||
|
using Stats.Config;
|
||||||
using Stats.Helpers;
|
using Stats.Helpers;
|
||||||
|
|
||||||
namespace IW4MAdmin.Application
|
namespace IW4MAdmin.Application
|
||||||
{
|
{
|
||||||
public class Program
|
public class Program
|
||||||
{
|
{
|
||||||
public static BuildNumber Version { get; private set; } = BuildNumber.Parse(Utilities.GetVersionAsString());
|
public static BuildNumber Version { get; } = BuildNumber.Parse(Utilities.GetVersionAsString());
|
||||||
public static ApplicationManager ServerManager;
|
private static ApplicationManager _serverManager;
|
||||||
private static Task ApplicationTask;
|
private static Task _applicationTask;
|
||||||
private static ServiceProvider serviceProvider;
|
private static ServiceProvider _serviceProvider;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// entrypoint of the application
|
/// entrypoint of the application
|
||||||
@ -55,7 +58,7 @@ namespace IW4MAdmin.Application
|
|||||||
Console.OutputEncoding = Encoding.UTF8;
|
Console.OutputEncoding = Encoding.UTF8;
|
||||||
Console.ForegroundColor = ConsoleColor.Gray;
|
Console.ForegroundColor = ConsoleColor.Gray;
|
||||||
|
|
||||||
Console.CancelKeyPress += new ConsoleCancelEventHandler(OnCancelKey);
|
Console.CancelKeyPress += OnCancelKey;
|
||||||
|
|
||||||
Console.WriteLine("=====================================================");
|
Console.WriteLine("=====================================================");
|
||||||
Console.WriteLine(" IW4MAdmin");
|
Console.WriteLine(" IW4MAdmin");
|
||||||
@ -74,10 +77,14 @@ namespace IW4MAdmin.Application
|
|||||||
/// <param name="e"></param>
|
/// <param name="e"></param>
|
||||||
private static async void OnCancelKey(object sender, ConsoleCancelEventArgs e)
|
private static async void OnCancelKey(object sender, ConsoleCancelEventArgs e)
|
||||||
{
|
{
|
||||||
ServerManager?.Stop();
|
if (_serverManager is not null)
|
||||||
if (ApplicationTask != null)
|
|
||||||
{
|
{
|
||||||
await ApplicationTask;
|
await _serverManager.Stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_applicationTask is not null)
|
||||||
|
{
|
||||||
|
await _applicationTask;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -91,8 +98,7 @@ namespace IW4MAdmin.Application
|
|||||||
ITranslationLookup translationLookup = null;
|
ITranslationLookup translationLookup = null;
|
||||||
var logger = BuildDefaultLogger<Program>(new ApplicationConfiguration());
|
var logger = BuildDefaultLogger<Program>(new ApplicationConfiguration());
|
||||||
Utilities.DefaultLogger = logger;
|
Utilities.DefaultLogger = logger;
|
||||||
IServiceCollection services = null;
|
logger.LogInformation("Begin IW4MAdmin startup. Version is {Version} {@Args}", Version, args);
|
||||||
logger.LogInformation("Begin IW4MAdmin startup. Version is {version} {@args}", Version, args);
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@ -101,22 +107,30 @@ namespace IW4MAdmin.Application
|
|||||||
ConfigurationMigration.CheckDirectories();
|
ConfigurationMigration.CheckDirectories();
|
||||||
ConfigurationMigration.RemoveObsoletePlugins20210322();
|
ConfigurationMigration.RemoveObsoletePlugins20210322();
|
||||||
logger.LogDebug("Configuring services...");
|
logger.LogDebug("Configuring services...");
|
||||||
services = ConfigureServices(args);
|
var services = await ConfigureServices(args);
|
||||||
serviceProvider = services.BuildServiceProvider();
|
_serviceProvider = services.BuildServiceProvider();
|
||||||
var versionChecker = serviceProvider.GetRequiredService<IMasterCommunication>();
|
var versionChecker = _serviceProvider.GetRequiredService<IMasterCommunication>();
|
||||||
ServerManager = (ApplicationManager) serviceProvider.GetRequiredService<IManager>();
|
_serverManager = (ApplicationManager) _serviceProvider.GetRequiredService<IManager>();
|
||||||
translationLookup = serviceProvider.GetRequiredService<ITranslationLookup>();
|
translationLookup = _serviceProvider.GetRequiredService<ITranslationLookup>();
|
||||||
|
|
||||||
await versionChecker.CheckVersion();
|
_applicationTask = RunApplicationTasksAsync(logger, services);
|
||||||
await ServerManager.Init();
|
var tasks = new[]
|
||||||
|
{
|
||||||
|
versionChecker.CheckVersion(),
|
||||||
|
_applicationTask
|
||||||
|
};
|
||||||
|
|
||||||
|
await _serverManager.Init();
|
||||||
|
|
||||||
|
await Task.WhenAll(tasks);
|
||||||
}
|
}
|
||||||
|
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
string failMessage = translationLookup == null
|
var failMessage = translationLookup == null
|
||||||
? "Failed to initialize IW4MAdmin"
|
? "Failed to initialize IW4MAdmin"
|
||||||
: translationLookup["MANAGER_INIT_FAIL"];
|
: translationLookup["MANAGER_INIT_FAIL"];
|
||||||
string exitMessage = translationLookup == null
|
var exitMessage = translationLookup == null
|
||||||
? "Press enter to exit..."
|
? "Press enter to exit..."
|
||||||
: translationLookup["MANAGER_EXIT"];
|
: translationLookup["MANAGER_EXIT"];
|
||||||
|
|
||||||
@ -130,13 +144,10 @@ 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 (string error in configException.Errors)
|
foreach (var error in configException.Errors)
|
||||||
{
|
{
|
||||||
Console.WriteLine(error);
|
Console.WriteLine(error);
|
||||||
}
|
}
|
||||||
@ -147,32 +158,22 @@ namespace IW4MAdmin.Application
|
|||||||
Console.WriteLine(e.Message);
|
Console.WriteLine(e.Message);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (_serverManager is not null)
|
||||||
|
{
|
||||||
|
await _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);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
try
|
if (_serverManager.IsRestartRequested)
|
||||||
{
|
|
||||||
ApplicationTask = RunApplicationTasksAsync(logger, services);
|
|
||||||
await ApplicationTask;
|
|
||||||
}
|
|
||||||
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
logger.LogCritical(e, "Failed to launch IW4MAdmin");
|
|
||||||
string failMessage = translationLookup == null
|
|
||||||
? "Failed to launch IW4MAdmin"
|
|
||||||
: translationLookup["MANAGER_INIT_FAIL"];
|
|
||||||
Console.WriteLine($"{failMessage}: {e.GetExceptionInfo()}");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ServerManager.IsRestartRequested)
|
|
||||||
{
|
{
|
||||||
goto restart;
|
goto restart;
|
||||||
}
|
}
|
||||||
|
|
||||||
serviceProvider.Dispose();
|
await _serviceProvider.DisposeAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -181,24 +182,26 @@ namespace IW4MAdmin.Application
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
private static async Task RunApplicationTasksAsync(ILogger logger, IServiceCollection services)
|
private static async Task RunApplicationTasksAsync(ILogger logger, IServiceCollection services)
|
||||||
{
|
{
|
||||||
var webfrontTask = ServerManager.GetApplicationSettings().Configuration().EnableWebFront
|
var webfrontTask = _serverManager.GetApplicationSettings().Configuration().EnableWebFront
|
||||||
? WebfrontCore.Program.Init(ServerManager, serviceProvider, services, ServerManager.CancellationToken)
|
? WebfrontCore.Program.Init(_serverManager, _serviceProvider, services, _serverManager.CancellationToken)
|
||||||
: Task.CompletedTask;
|
: Task.CompletedTask;
|
||||||
|
|
||||||
var collectionService = serviceProvider.GetRequiredService<IServerDataCollector>();
|
var collectionService = _serviceProvider.GetRequiredService<IServerDataCollector>();
|
||||||
|
|
||||||
// we want to run this one on a manual thread instead of letting the thread pool handle it,
|
// we want to run this one on a manual thread instead of letting the thread pool handle it,
|
||||||
// because we can't exit early from waiting on console input, and it prevents us from restarting
|
// because we can't exit early from waiting on console input, and it prevents us from restarting
|
||||||
var inputThread = new Thread(async () => await ReadConsoleInput(logger));
|
async void ReadInput() => await ReadConsoleInput(logger);
|
||||||
|
|
||||||
|
var inputThread = new Thread(ReadInput);
|
||||||
inputThread.Start();
|
inputThread.Start();
|
||||||
|
|
||||||
var tasks = new[]
|
var tasks = new[]
|
||||||
{
|
{
|
||||||
ServerManager.Start(),
|
|
||||||
webfrontTask,
|
webfrontTask,
|
||||||
serviceProvider.GetRequiredService<IMasterCommunication>()
|
_serverManager.Start(),
|
||||||
.RunUploadStatus(ServerManager.CancellationToken),
|
_serviceProvider.GetRequiredService<IMasterCommunication>()
|
||||||
collectionService.BeginCollectionAsync(cancellationToken: ServerManager.CancellationToken)
|
.RunUploadStatus(_serverManager.CancellationToken),
|
||||||
|
collectionService.BeginCollectionAsync(cancellationToken: _serverManager.CancellationToken)
|
||||||
};
|
};
|
||||||
|
|
||||||
logger.LogDebug("Starting webfront and input tasks");
|
logger.LogDebug("Starting webfront and input tasks");
|
||||||
@ -208,7 +211,6 @@ namespace IW4MAdmin.Application
|
|||||||
Console.WriteLine(Utilities.CurrentLocalization.LocalizationIndex["MANAGER_SHUTDOWN_SUCCESS"]);
|
Console.WriteLine(Utilities.CurrentLocalization.LocalizationIndex["MANAGER_SHUTDOWN_SUCCESS"]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// reads input from the console and executes entered commands on the default server
|
/// reads input from the console and executes entered commands on the default server
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -221,34 +223,43 @@ namespace IW4MAdmin.Application
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
string lastCommand;
|
EFClient origin = null;
|
||||||
var Origin = Utilities.IW4MAdminClient(ServerManager.Servers[0]);
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
while (!ServerManager.CancellationToken.IsCancellationRequested)
|
while (!_serverManager.CancellationToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
lastCommand = await Console.In.ReadLineAsync();
|
if (!_serverManager.IsInitialized)
|
||||||
|
{
|
||||||
|
await Task.Delay(1000);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (lastCommand?.Length > 0)
|
var lastCommand = await Console.In.ReadLineAsync();
|
||||||
|
|
||||||
|
if (lastCommand == null)
|
||||||
{
|
{
|
||||||
if (lastCommand?.Length > 0)
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!lastCommand.Any())
|
||||||
{
|
{
|
||||||
GameEvent E = new GameEvent()
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var gameEvent = new GameEvent
|
||||||
{
|
{
|
||||||
Type = GameEvent.EventType.Command,
|
Type = GameEvent.EventType.Command,
|
||||||
Data = lastCommand,
|
Data = lastCommand,
|
||||||
Origin = Origin,
|
Origin = origin ??= Utilities.IW4MAdminClient(_serverManager.Servers.FirstOrDefault()),
|
||||||
Owner = ServerManager.Servers[0]
|
Owner = _serverManager.Servers[0]
|
||||||
};
|
};
|
||||||
|
|
||||||
ServerManager.AddEvent(E);
|
_serverManager.AddEvent(gameEvent);
|
||||||
await E.WaitAsync(Utilities.DefaultCommandTimeout, ServerManager.CancellationToken);
|
await gameEvent.WaitAsync(Utilities.DefaultCommandTimeout, _serverManager.CancellationToken);
|
||||||
Console.Write('>');
|
Console.Write('>');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (OperationCanceledException)
|
catch (OperationCanceledException)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
@ -274,10 +285,10 @@ 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);
|
||||||
serviceCollection.AddSingleton(typeof(IManagerCommand), commandType);
|
serviceCollection.AddSingleton(typeof(IManagerCommand), commandType);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -285,40 +296,40 @@ namespace IW4MAdmin.Application
|
|||||||
var (plugins, commands, configurations) = pluginImporter.DiscoverAssemblyPluginImplementations();
|
var (plugins, commands, configurations) = pluginImporter.DiscoverAssemblyPluginImplementations();
|
||||||
foreach (var pluginType in plugins)
|
foreach (var pluginType in plugins)
|
||||||
{
|
{
|
||||||
defaultLogger.LogDebug("Registered plugin type {name}", pluginType.FullName);
|
defaultLogger.LogDebug("Registered plugin type {Name}", pluginType.FullName);
|
||||||
serviceCollection.AddSingleton(typeof(IPlugin), pluginType);
|
serviceCollection.AddSingleton(typeof(IPlugin), pluginType);
|
||||||
}
|
}
|
||||||
|
|
||||||
// register the plugin commands
|
// register the plugin commands
|
||||||
foreach (var commandType in commands)
|
foreach (var commandType in commands)
|
||||||
{
|
{
|
||||||
defaultLogger.LogDebug("Registered plugin command type {name}", commandType.FullName);
|
defaultLogger.LogDebug("Registered plugin command type {Name}", commandType.FullName);
|
||||||
serviceCollection.AddSingleton(typeof(IManagerCommand), commandType);
|
serviceCollection.AddSingleton(typeof(IManagerCommand), commandType);
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (var configurationType in configurations)
|
foreach (var configurationType in configurations)
|
||||||
{
|
{
|
||||||
defaultLogger.LogDebug("Registered plugin config type {name}", configurationType.Name);
|
defaultLogger.LogDebug("Registered plugin config type {Name}", configurationType.Name);
|
||||||
var configInstance = (IBaseConfiguration) Activator.CreateInstance(configurationType);
|
var configInstance = (IBaseConfiguration) Activator.CreateInstance(configurationType);
|
||||||
var handlerType = typeof(BaseConfigurationHandler<>).MakeGenericType(configurationType);
|
var handlerType = typeof(BaseConfigurationHandler<>).MakeGenericType(configurationType);
|
||||||
var handlerInstance = Activator.CreateInstance(handlerType, new[] {configInstance.Name()});
|
var handlerInstance = Activator.CreateInstance(handlerType, configInstance.Name());
|
||||||
var genericInterfaceType = typeof(IConfigurationHandler<>).MakeGenericType(configurationType);
|
var genericInterfaceType = typeof(IConfigurationHandler<>).MakeGenericType(configurationType);
|
||||||
|
|
||||||
serviceCollection.AddSingleton(genericInterfaceType, handlerInstance);
|
serviceCollection.AddSingleton(genericInterfaceType, handlerInstance);
|
||||||
}
|
}
|
||||||
|
|
||||||
// register any script plugins
|
// register any script plugins
|
||||||
foreach (var scriptPlugin in pluginImporter.DiscoverScriptPlugins())
|
foreach (var plugin in pluginImporter.DiscoverScriptPlugins())
|
||||||
{
|
{
|
||||||
serviceCollection.AddSingleton(scriptPlugin);
|
serviceCollection.AddSingleton(plugin);
|
||||||
}
|
}
|
||||||
|
|
||||||
// register any eventable types
|
// register any eventable types
|
||||||
foreach (var assemblyType in typeof(Program).Assembly.GetTypes()
|
foreach (var assemblyType in typeof(Program).Assembly.GetTypes()
|
||||||
.Where(_asmType => typeof(IRegisterEvent).IsAssignableFrom(_asmType))
|
.Where(asmType => typeof(IRegisterEvent).IsAssignableFrom(asmType))
|
||||||
.Union(plugins.SelectMany(_asm => _asm.Assembly.GetTypes())
|
.Union(plugins.SelectMany(asm => asm.Assembly.GetTypes())
|
||||||
.Distinct()
|
.Distinct()
|
||||||
.Where(_asmType => typeof(IRegisterEvent).IsAssignableFrom(_asmType))))
|
.Where(asmType => typeof(IRegisterEvent).IsAssignableFrom(asmType))))
|
||||||
{
|
{
|
||||||
var instance = Activator.CreateInstance(assemblyType) as IRegisterEvent;
|
var instance = Activator.CreateInstance(assemblyType) as IRegisterEvent;
|
||||||
serviceCollection.AddSingleton(instance);
|
serviceCollection.AddSingleton(instance);
|
||||||
@ -331,25 +342,39 @@ namespace IW4MAdmin.Application
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Configures the dependency injection services
|
/// Configures the dependency injection services
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private static IServiceCollection ConfigureServices(string[] args)
|
private static async Task<IServiceCollection> ConfigureServices(string[] args)
|
||||||
{
|
{
|
||||||
|
// todo: this is a quick fix
|
||||||
|
AppContext.SetSwitch("Npgsql.EnableLegacyTimestampBehavior", true);
|
||||||
|
|
||||||
// setup the static resources (config/master api/translations)
|
// setup the static resources (config/master api/translations)
|
||||||
var serviceCollection = new ServiceCollection();
|
var serviceCollection = new ServiceCollection();
|
||||||
var appConfigHandler = new BaseConfigurationHandler<ApplicationConfiguration>("IW4MAdminSettings");
|
var appConfigHandler = new BaseConfigurationHandler<ApplicationConfiguration>("IW4MAdminSettings");
|
||||||
|
await appConfigHandler.BuildAsync();
|
||||||
var defaultConfigHandler = new BaseConfigurationHandler<DefaultSettings>("DefaultSettings");
|
var defaultConfigHandler = new BaseConfigurationHandler<DefaultSettings>("DefaultSettings");
|
||||||
|
await defaultConfigHandler.BuildAsync();
|
||||||
|
var commandConfigHandler = new BaseConfigurationHandler<CommandConfiguration>("CommandConfiguration");
|
||||||
|
await commandConfigHandler.BuildAsync();
|
||||||
|
var statsCommandHandler = new BaseConfigurationHandler<StatsConfiguration>("StatsPluginSettings");
|
||||||
|
await statsCommandHandler.BuildAsync();
|
||||||
var defaultConfig = defaultConfigHandler.Configuration();
|
var defaultConfig = defaultConfigHandler.Configuration();
|
||||||
var appConfig = appConfigHandler.Configuration();
|
var appConfig = appConfigHandler.Configuration();
|
||||||
var masterUri = Utilities.IsDevelopment
|
var masterUri = Utilities.IsDevelopment
|
||||||
? new Uri("http://127.0.0.1:8080")
|
? new Uri("http://127.0.0.1:8080")
|
||||||
: appConfig?.MasterUrl ?? new ApplicationConfiguration().MasterUrl;
|
: appConfig?.MasterUrl ?? new ApplicationConfiguration().MasterUrl;
|
||||||
var masterRestClient = RestClient.For<IMasterApi>(masterUri);
|
var httpClient = new HttpClient
|
||||||
|
{
|
||||||
|
BaseAddress = masterUri,
|
||||||
|
Timeout = TimeSpan.FromSeconds(15)
|
||||||
|
};
|
||||||
|
var masterRestClient = RestClient.For<IMasterApi>(httpClient);
|
||||||
var translationLookup = Configure.Initialize(Utilities.DefaultLogger, masterRestClient, appConfig);
|
var translationLookup = Configure.Initialize(Utilities.DefaultLogger, masterRestClient, appConfig);
|
||||||
|
|
||||||
if (appConfig == null)
|
if (appConfig == null)
|
||||||
{
|
{
|
||||||
appConfig = (ApplicationConfiguration) new ApplicationConfiguration().Generate();
|
appConfig = (ApplicationConfiguration) new ApplicationConfiguration().Generate();
|
||||||
appConfigHandler.Set(appConfig);
|
appConfigHandler.Set(appConfig);
|
||||||
appConfigHandler.Save();
|
await appConfigHandler.Save();
|
||||||
}
|
}
|
||||||
|
|
||||||
// register override level names
|
// register override level names
|
||||||
@ -367,15 +392,14 @@ namespace IW4MAdmin.Application
|
|||||||
serviceCollection
|
serviceCollection
|
||||||
.AddBaseLogger(appConfig)
|
.AddBaseLogger(appConfig)
|
||||||
.AddSingleton(defaultConfig)
|
.AddSingleton(defaultConfig)
|
||||||
.AddSingleton<IServiceCollection>(_serviceProvider => serviceCollection)
|
.AddSingleton<IServiceCollection>(serviceCollection)
|
||||||
.AddSingleton<IConfigurationHandler<DefaultSettings>, BaseConfigurationHandler<DefaultSettings>>()
|
.AddSingleton<IConfigurationHandler<DefaultSettings>, BaseConfigurationHandler<DefaultSettings>>()
|
||||||
.AddSingleton((IConfigurationHandler<ApplicationConfiguration>) appConfigHandler)
|
.AddSingleton((IConfigurationHandler<ApplicationConfiguration>) appConfigHandler)
|
||||||
.AddSingleton(
|
.AddSingleton<IConfigurationHandler<CommandConfiguration>>(commandConfigHandler)
|
||||||
new BaseConfigurationHandler<CommandConfiguration>("CommandConfiguration") as
|
|
||||||
IConfigurationHandler<CommandConfiguration>)
|
|
||||||
.AddSingleton(appConfig)
|
.AddSingleton(appConfig)
|
||||||
.AddSingleton(_serviceProvider =>
|
.AddSingleton(statsCommandHandler.Configuration() ?? new StatsConfiguration())
|
||||||
_serviceProvider.GetRequiredService<IConfigurationHandler<CommandConfiguration>>()
|
.AddSingleton(serviceProvider =>
|
||||||
|
serviceProvider.GetRequiredService<IConfigurationHandler<CommandConfiguration>>()
|
||||||
.Configuration() ?? new CommandConfiguration())
|
.Configuration() ?? new CommandConfiguration())
|
||||||
.AddSingleton<IPluginImporter, PluginImporter>()
|
.AddSingleton<IPluginImporter, PluginImporter>()
|
||||||
.AddSingleton<IMiddlewareActionHandler, MiddlewareActionHandler>()
|
.AddSingleton<IMiddlewareActionHandler, MiddlewareActionHandler>()
|
||||||
@ -388,7 +412,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>()
|
||||||
@ -402,11 +429,14 @@ 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>()
|
||||||
.AddSingleton<IManager, ApplicationManager>()
|
.AddSingleton<IManager, ApplicationManager>()
|
||||||
|
#pragma warning disable CS0612
|
||||||
.AddSingleton<SharedLibraryCore.Interfaces.ILogger, Logger>()
|
.AddSingleton<SharedLibraryCore.Interfaces.ILogger, Logger>()
|
||||||
|
#pragma warning restore CS0612
|
||||||
.AddSingleton<IClientNoticeMessageFormatter, ClientNoticeMessageFormatter>()
|
.AddSingleton<IClientNoticeMessageFormatter, ClientNoticeMessageFormatter>()
|
||||||
.AddSingleton<IClientStatisticCalculator, HitCalculator>()
|
.AddSingleton<IClientStatisticCalculator, HitCalculator>()
|
||||||
.AddSingleton<IServerDistributionCalculator, ServerDistributionCalculator>()
|
.AddSingleton<IServerDistributionCalculator, ServerDistributionCalculator>()
|
||||||
@ -417,6 +447,8 @@ namespace IW4MAdmin.Application
|
|||||||
.AddSingleton<IServerDataViewer, ServerDataViewer>()
|
.AddSingleton<IServerDataViewer, ServerDataViewer>()
|
||||||
.AddSingleton<IServerDataCollector, ServerDataCollector>()
|
.AddSingleton<IServerDataCollector, ServerDataCollector>()
|
||||||
.AddSingleton<IEventPublisher, EventPublisher>()
|
.AddSingleton<IEventPublisher, EventPublisher>()
|
||||||
|
.AddSingleton<IGeoLocationService>(new GeoLocationService(Path.Join(".", "Resources", "GeoLite2-Country.mmdb")))
|
||||||
|
.AddTransient<IScriptPluginTimerHelper, ScriptPluginTimerHelper>()
|
||||||
.AddSingleton(translationLookup)
|
.AddSingleton(translationLookup)
|
||||||
.AddDatabaseContextOptions(appConfig);
|
.AddDatabaseContextOptions(appConfig);
|
||||||
|
|
||||||
|
@ -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)
|
||||||
{
|
{
|
||||||
@ -63,12 +83,12 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
Value = lastMapMeta.Value,
|
Value = lastMapMeta.Value,
|
||||||
ShouldDisplay = true,
|
ShouldDisplay = true,
|
||||||
Type = MetaType.Information,
|
Type = MetaType.Information,
|
||||||
Column = 1,
|
|
||||||
Order = 6
|
Order = 6
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
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)
|
||||||
{
|
{
|
||||||
@ -80,8 +100,7 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
Value = lastServerMeta.Value,
|
Value = lastServerMeta.Value,
|
||||||
ShouldDisplay = true,
|
ShouldDisplay = true,
|
||||||
Type = MetaType.Information,
|
Type = MetaType.Information,
|
||||||
Column = 0,
|
Order = 7
|
||||||
Order = 6
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -89,7 +108,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;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -99,8 +118,7 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
Key = _transLookup["WEBFRONT_PROFILE_META_PLAY_TIME"],
|
Key = _transLookup["WEBFRONT_PROFILE_META_PLAY_TIME"],
|
||||||
Value = TimeSpan.FromHours(client.TotalConnectionTime / 3600.0).HumanizeForCurrentCulture(),
|
Value = TimeSpan.FromHours(client.TotalConnectionTime / 3600.0).HumanizeForCurrentCulture(),
|
||||||
ShouldDisplay = true,
|
ShouldDisplay = true,
|
||||||
Column = 1,
|
Order = 8,
|
||||||
Order = 0,
|
|
||||||
Type = MetaType.Information
|
Type = MetaType.Information
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -110,8 +128,7 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
Key = _transLookup["WEBFRONT_PROFILE_META_FIRST_SEEN"],
|
Key = _transLookup["WEBFRONT_PROFILE_META_FIRST_SEEN"],
|
||||||
Value = (DateTime.UtcNow - client.FirstConnection).HumanizeForCurrentCulture(),
|
Value = (DateTime.UtcNow - client.FirstConnection).HumanizeForCurrentCulture(),
|
||||||
ShouldDisplay = true,
|
ShouldDisplay = true,
|
||||||
Column = 1,
|
Order = 9,
|
||||||
Order = 1,
|
|
||||||
Type = MetaType.Information
|
Type = MetaType.Information
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -121,8 +138,7 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
Key = _transLookup["WEBFRONT_PROFILE_META_LAST_SEEN"],
|
Key = _transLookup["WEBFRONT_PROFILE_META_LAST_SEEN"],
|
||||||
Value = (DateTime.UtcNow - client.LastConnection).HumanizeForCurrentCulture(),
|
Value = (DateTime.UtcNow - client.LastConnection).HumanizeForCurrentCulture(),
|
||||||
ShouldDisplay = true,
|
ShouldDisplay = true,
|
||||||
Column = 1,
|
Order = 10,
|
||||||
Order = 2,
|
|
||||||
Type = MetaType.Information
|
Type = MetaType.Information
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -130,10 +146,10 @@ 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,
|
Order = 11,
|
||||||
Order = 3,
|
|
||||||
Type = MetaType.Information
|
Type = MetaType.Information
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -141,38 +157,50 @@ 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,
|
Order = 12,
|
||||||
Order = 4,
|
|
||||||
Type = MetaType.Information
|
Type = MetaType.Information
|
||||||
});
|
});
|
||||||
|
|
||||||
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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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()
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -11,6 +11,7 @@ using SharedLibraryCore.Dtos.Meta.Responses;
|
|||||||
using SharedLibraryCore.Helpers;
|
using SharedLibraryCore.Helpers;
|
||||||
using SharedLibraryCore.Interfaces;
|
using SharedLibraryCore.Interfaces;
|
||||||
using SharedLibraryCore.QueryHelper;
|
using SharedLibraryCore.QueryHelper;
|
||||||
|
using SharedLibraryCore.Services;
|
||||||
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
||||||
|
|
||||||
namespace IW4MAdmin.Application.Meta
|
namespace IW4MAdmin.Application.Meta
|
||||||
@ -19,7 +20,8 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
/// implementation of IResourceQueryHelper
|
/// implementation of IResourceQueryHelper
|
||||||
/// used to pull in penalties applied to a given client id
|
/// used to pull in penalties applied to a given client id
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class ReceivedPenaltyResourceQueryHelper : IResourceQueryHelper<ClientPaginationRequest, ReceivedPenaltyResponse>
|
public class
|
||||||
|
ReceivedPenaltyResourceQueryHelper : IResourceQueryHelper<ClientPaginationRequest, ReceivedPenaltyResponse>
|
||||||
{
|
{
|
||||||
private readonly ILogger _logger;
|
private readonly ILogger _logger;
|
||||||
private readonly IDatabaseContextFactory _contextFactory;
|
private readonly IDatabaseContextFactory _contextFactory;
|
||||||
@ -33,7 +35,8 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
_appConfig = appConfig;
|
_appConfig = appConfig;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<ResourceQueryHelperResult<ReceivedPenaltyResponse>> QueryResource(ClientPaginationRequest query)
|
public async Task<ResourceQueryHelperResult<ReceivedPenaltyResponse>> QueryResource(
|
||||||
|
ClientPaginationRequest query)
|
||||||
{
|
{
|
||||||
var linkedPenaltyType = Utilities.LinkedPenaltyTypes();
|
var linkedPenaltyType = Utilities.LinkedPenaltyTypes();
|
||||||
await using var ctx = _contextFactory.CreateContext(enableTracking: false);
|
await using var ctx = _contextFactory.CreateContext(enableTracking: false);
|
||||||
@ -48,20 +51,27 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
linkedPenaltyType.Contains(_penalty.Type) && _penalty.LinkId == linkId.AliasLinkId);
|
linkedPenaltyType.Contains(_penalty.Type) && _penalty.LinkId == linkId.AliasLinkId);
|
||||||
|
|
||||||
IQueryable<EFPenalty> iqIpLinkedPenalties = null;
|
IQueryable<EFPenalty> iqIpLinkedPenalties = null;
|
||||||
|
IQueryable<EFPenalty> identifierPenalties = null;
|
||||||
|
|
||||||
if (!_appConfig.EnableImplicitAccountLinking)
|
if (!_appConfig.EnableImplicitAccountLinking)
|
||||||
{
|
{
|
||||||
var usedIps = await ctx.Aliases.AsNoTracking()
|
var usedIps = await ctx.Aliases.AsNoTracking()
|
||||||
.Where(alias => (alias.LinkId == linkId.AliasLinkId || alias.AliasId == linkId.CurrentAliasId) && alias.IPAddress != null)
|
.Where(alias =>
|
||||||
|
(alias.LinkId == linkId.AliasLinkId || alias.AliasId == linkId.CurrentAliasId) &&
|
||||||
|
alias.IPAddress != null)
|
||||||
.Select(alias => alias.IPAddress).ToListAsync();
|
.Select(alias => alias.IPAddress).ToListAsync();
|
||||||
|
|
||||||
|
identifierPenalties = ctx.PenaltyIdentifiers.AsNoTracking().Where(identifier =>
|
||||||
|
identifier.IPv4Address != null && usedIps.Contains(identifier.IPv4Address))
|
||||||
|
.Select(id => id.Penalty);
|
||||||
|
|
||||||
var aliasedIds = await ctx.Aliases.AsNoTracking().Where(alias => usedIps.Contains(alias.IPAddress))
|
var aliasedIds = await ctx.Aliases.AsNoTracking().Where(alias => usedIps.Contains(alias.IPAddress))
|
||||||
.Select(alias => alias.LinkId)
|
.Select(alias => alias.LinkId)
|
||||||
.ToListAsync();
|
.ToListAsync();
|
||||||
|
|
||||||
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;
|
||||||
@ -71,6 +81,11 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
iqAllPenalties = iqPenalties.Union(iqIpLinkedPenalties);
|
iqAllPenalties = iqPenalties.Union(iqIpLinkedPenalties);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (identifierPenalties != null)
|
||||||
|
{
|
||||||
|
iqAllPenalties = iqPenalties.Union(identifierPenalties);
|
||||||
|
}
|
||||||
|
|
||||||
var penalties = await iqAllPenalties
|
var penalties = await iqAllPenalties
|
||||||
.Where(_penalty => _penalty.When < query.Before)
|
.Where(_penalty => _penalty.When < query.Before)
|
||||||
.OrderByDescending(_penalty => _penalty.When)
|
.OrderByDescending(_penalty => _penalty.When)
|
||||||
@ -97,7 +112,7 @@ namespace IW4MAdmin.Application.Meta
|
|||||||
{
|
{
|
||||||
// todo: maybe actually count
|
// todo: maybe actually count
|
||||||
RetrievedResultCount = penalties.Count,
|
RetrievedResultCount = penalties.Count,
|
||||||
Results = penalties
|
Results = penalties.Distinct()
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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)
|
||||||
{
|
{
|
||||||
|
12
Application/Misc/AsyncResult.cs
Normal file
12
Application/Misc/AsyncResult.cs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Misc;
|
||||||
|
|
||||||
|
public class AsyncResult : IAsyncResult
|
||||||
|
{
|
||||||
|
public object AsyncState { get; set; }
|
||||||
|
public WaitHandle AsyncWaitHandle { get; set; }
|
||||||
|
public bool CompletedSynchronously { get; set; }
|
||||||
|
public bool IsCompleted { get; set; }
|
||||||
|
}
|
@ -1,10 +1,13 @@
|
|||||||
using Newtonsoft.Json;
|
using SharedLibraryCore;
|
||||||
using SharedLibraryCore;
|
|
||||||
using SharedLibraryCore.Exceptions;
|
using SharedLibraryCore.Exceptions;
|
||||||
using SharedLibraryCore.Interfaces;
|
using SharedLibraryCore.Interfaces;
|
||||||
using System;
|
using System;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
|
using System.Text.Json;
|
||||||
|
using System.Text.Json.Serialization;
|
||||||
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using JsonSerializer = System.Text.Json.JsonSerializer;
|
||||||
|
|
||||||
namespace IW4MAdmin.Application.Misc
|
namespace IW4MAdmin.Application.Misc
|
||||||
{
|
{
|
||||||
@ -14,27 +17,40 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
/// <typeparam name="T">base configuration type</typeparam>
|
/// <typeparam name="T">base configuration type</typeparam>
|
||||||
public class BaseConfigurationHandler<T> : IConfigurationHandler<T> where T : IBaseConfiguration
|
public class BaseConfigurationHandler<T> : IConfigurationHandler<T> where T : IBaseConfiguration
|
||||||
{
|
{
|
||||||
T _configuration;
|
private T _configuration;
|
||||||
|
private readonly SemaphoreSlim _onSaving;
|
||||||
|
private readonly JsonSerializerOptions _serializerOptions;
|
||||||
|
|
||||||
public BaseConfigurationHandler(string fn)
|
|
||||||
|
public BaseConfigurationHandler(string fileName)
|
||||||
{
|
{
|
||||||
FileName = Path.Join(Utilities.OperatingDirectory, "Configuration", $"{fn}.json");
|
_serializerOptions = new JsonSerializerOptions
|
||||||
Build();
|
{
|
||||||
|
WriteIndented = true,
|
||||||
|
Encoder = System.Text.Encodings.Web.JavaScriptEncoder.UnsafeRelaxedJsonEscaping
|
||||||
|
};
|
||||||
|
_serializerOptions.Converters.Add(new JsonStringEnumConverter());
|
||||||
|
_onSaving = new SemaphoreSlim(1, 1);
|
||||||
|
FileName = Path.Join(Utilities.OperatingDirectory, "Configuration", $"{fileName}.json");
|
||||||
}
|
}
|
||||||
|
|
||||||
public BaseConfigurationHandler() : this(typeof(T).Name)
|
public BaseConfigurationHandler() : this(typeof(T).Name)
|
||||||
{
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
~BaseConfigurationHandler()
|
||||||
|
{
|
||||||
|
_onSaving.Dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
public string FileName { get; }
|
public string FileName { get; }
|
||||||
|
|
||||||
public void Build()
|
public async Task BuildAsync()
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var configContent = File.ReadAllText(FileName);
|
await using var fileStream = File.OpenRead(FileName);
|
||||||
_configuration = JsonConvert.DeserializeObject<T>(configContent);
|
_configuration = await JsonSerializer.DeserializeAsync<T>(fileStream, _serializerOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
catch (FileNotFoundException)
|
catch (FileNotFoundException)
|
||||||
@ -44,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
|
||||||
@ -52,16 +68,23 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public Task Save()
|
public async Task Save()
|
||||||
{
|
{
|
||||||
var settings = new JsonSerializerSettings()
|
try
|
||||||
{
|
{
|
||||||
Formatting = Formatting.Indented
|
await _onSaving.WaitAsync();
|
||||||
};
|
|
||||||
settings.Converters.Add(new Newtonsoft.Json.Converters.StringEnumConverter());
|
|
||||||
|
|
||||||
var appConfigJSON = JsonConvert.SerializeObject(_configuration, settings);
|
await using var fileStream = File.Create(FileName);
|
||||||
return File.WriteAllTextAsync(FileName, appConfigJSON);
|
await JsonSerializer.SerializeAsync(fileStream, _configuration, _serializerOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
if (_onSaving.CurrentCount == 0)
|
||||||
|
{
|
||||||
|
_onSaving.Release(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public T Configuration()
|
public T Configuration()
|
||||||
|
13
Application/Misc/GeoLocationResult.cs
Normal file
13
Application/Misc/GeoLocationResult.cs
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Misc;
|
||||||
|
|
||||||
|
public class GeoLocationResult : IGeoLocationResult
|
||||||
|
{
|
||||||
|
public string Country { get; set; }
|
||||||
|
public string CountryCode { get; set; }
|
||||||
|
public string Region { get; set; }
|
||||||
|
public string ASN { get; set; }
|
||||||
|
public string Timezone { get; set; }
|
||||||
|
public string Organization { get; set; }
|
||||||
|
}
|
40
Application/Misc/GeoLocationService.cs
Normal file
40
Application/Misc/GeoLocationService.cs
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
using System;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using MaxMind.GeoIP2;
|
||||||
|
using MaxMind.GeoIP2.Responses;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Misc;
|
||||||
|
|
||||||
|
public class GeoLocationService : IGeoLocationService
|
||||||
|
{
|
||||||
|
private readonly string _sourceAddress;
|
||||||
|
|
||||||
|
public GeoLocationService(string sourceAddress)
|
||||||
|
{
|
||||||
|
_sourceAddress = sourceAddress;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Task<IGeoLocationResult> Locate(string address)
|
||||||
|
{
|
||||||
|
CountryResponse country = null;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
using var reader = new DatabaseReader(_sourceAddress);
|
||||||
|
reader.TryCountry(address, out country);
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
// ignored
|
||||||
|
}
|
||||||
|
|
||||||
|
var response = new GeoLocationResult
|
||||||
|
{
|
||||||
|
Country = country?.Country.Name ?? "Unknown",
|
||||||
|
CountryCode = country?.Country.IsoCode ?? ""
|
||||||
|
};
|
||||||
|
|
||||||
|
return Task.FromResult((IGeoLocationResult)response);
|
||||||
|
}
|
||||||
|
}
|
@ -26,7 +26,8 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
private readonly ApplicationConfiguration _appConfig;
|
private readonly ApplicationConfiguration _appConfig;
|
||||||
private readonly BuildNumber _fallbackVersion = BuildNumber.Parse("99.99.99.99");
|
private readonly BuildNumber _fallbackVersion = BuildNumber.Parse("99.99.99.99");
|
||||||
private readonly int _apiVersion = 1;
|
private readonly int _apiVersion = 1;
|
||||||
private bool firstHeartBeat = true;
|
private bool _firstHeartBeat = true;
|
||||||
|
private static readonly TimeSpan Interval = TimeSpan.FromSeconds(30);
|
||||||
|
|
||||||
public MasterCommunication(ILogger<MasterCommunication> logger, ApplicationConfiguration appConfig, ITranslationLookup translationLookup, IMasterApi apiInstance, IManager manager)
|
public MasterCommunication(ILogger<MasterCommunication> logger, ApplicationConfiguration appConfig, ITranslationLookup translationLookup, IMasterApi apiInstance, IManager manager)
|
||||||
{
|
{
|
||||||
@ -93,53 +94,24 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
|
|
||||||
public async Task RunUploadStatus(CancellationToken token)
|
public async Task RunUploadStatus(CancellationToken token)
|
||||||
{
|
{
|
||||||
// todo: clean up this logic
|
|
||||||
bool connected;
|
|
||||||
|
|
||||||
while (!token.IsCancellationRequested)
|
while (!token.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
|
{
|
||||||
|
if (_manager.IsRunning)
|
||||||
{
|
{
|
||||||
await UploadStatus();
|
await UploadStatus();
|
||||||
}
|
}
|
||||||
|
|
||||||
catch (System.Net.Http.HttpRequestException e)
|
|
||||||
{
|
|
||||||
_logger.LogWarning(e, "Could not send heartbeat");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
catch (AggregateException e)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.LogWarning(e, "Could not send heartbeat");
|
_logger.LogWarning(ex, "Could not send heartbeat");
|
||||||
var exceptions = e.InnerExceptions.Where(ex => ex.GetType() == typeof(ApiException));
|
|
||||||
|
|
||||||
foreach (var ex in exceptions)
|
|
||||||
{
|
|
||||||
if (((ApiException)ex).StatusCode == System.Net.HttpStatusCode.Unauthorized)
|
|
||||||
{
|
|
||||||
connected = false;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
catch (ApiException e)
|
|
||||||
{
|
|
||||||
_logger.LogWarning(e, "Could not send heartbeat");
|
|
||||||
if (e.StatusCode == System.Net.HttpStatusCode.Unauthorized)
|
|
||||||
{
|
|
||||||
connected = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
_logger.LogWarning(e, "Could not send heartbeat");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await Task.Delay(30000, token);
|
await Task.Delay(Interval, token);
|
||||||
}
|
}
|
||||||
|
|
||||||
catch
|
catch
|
||||||
@ -151,7 +123,7 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
|
|
||||||
private async Task UploadStatus()
|
private async Task UploadStatus()
|
||||||
{
|
{
|
||||||
if (firstHeartBeat)
|
if (_firstHeartBeat)
|
||||||
{
|
{
|
||||||
var token = await _apiInstance.Authenticate(new AuthenticationId
|
var token = await _apiInstance.Authenticate(new AuthenticationId
|
||||||
{
|
{
|
||||||
@ -179,12 +151,13 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
Id = s.EndPoint,
|
Id = s.EndPoint,
|
||||||
Port = (short)s.Port,
|
Port = (short)s.Port,
|
||||||
IPAddress = s.IP
|
IPAddress = s.IP
|
||||||
}).ToList()
|
}).ToList(),
|
||||||
|
WebfrontUrl = _appConfig.WebfrontUrl
|
||||||
};
|
};
|
||||||
|
|
||||||
Response<ResultMessage> response = null;
|
Response<ResultMessage> response = null;
|
||||||
|
|
||||||
if (firstHeartBeat)
|
if (_firstHeartBeat)
|
||||||
{
|
{
|
||||||
response = await _apiInstance.AddInstance(instance);
|
response = await _apiInstance.AddInstance(instance);
|
||||||
}
|
}
|
||||||
@ -192,7 +165,7 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
response = await _apiInstance.UpdateInstance(instance.Id, instance);
|
response = await _apiInstance.UpdateInstance(instance.Id, instance);
|
||||||
firstHeartBeat = false;
|
_firstHeartBeat = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (response.ResponseMessage.StatusCode != System.Net.HttpStatusCode.OK)
|
if (response.ResponseMessage.StatusCode != System.Net.HttpStatusCode.OK)
|
||||||
|
@ -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;
|
||||||
@ -68,6 +69,29 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
await ctx.SaveChangesAsync();
|
await ctx.SaveChangesAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task SetPersistentMeta(string metaKey, string metaValue, int clientId)
|
||||||
|
{
|
||||||
|
await AddPersistentMeta(metaKey, metaValue, new EFClient { ClientId = clientId });
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task IncrementPersistentMeta(string metaKey, int incrementAmount, int clientId)
|
||||||
|
{
|
||||||
|
var existingMeta = await GetPersistentMeta(metaKey, new EFClient { ClientId = clientId });
|
||||||
|
|
||||||
|
if (!long.TryParse(existingMeta?.Value ?? "0", out var existingValue))
|
||||||
|
{
|
||||||
|
existingValue = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
var newValue = existingValue + incrementAmount;
|
||||||
|
await SetPersistentMeta(metaKey, newValue.ToString(), clientId);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task DecrementPersistentMeta(string metaKey, int decrementAmount, int clientId)
|
||||||
|
{
|
||||||
|
await IncrementPersistentMeta(metaKey, -decrementAmount, clientId);
|
||||||
|
}
|
||||||
|
|
||||||
public async Task AddPersistentMeta(string metaKey, string metaValue)
|
public async Task AddPersistentMeta(string metaKey, string metaValue)
|
||||||
{
|
{
|
||||||
await using var ctx = _contextFactory.CreateContext();
|
await using var ctx = _contextFactory.CreateContext();
|
||||||
@ -207,42 +231,30 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
|
|
||||||
public async Task<IEnumerable<IClientMeta>> GetRuntimeMeta(ClientPaginationRequest request)
|
public async Task<IEnumerable<IClientMeta>> GetRuntimeMeta(ClientPaginationRequest request)
|
||||||
{
|
{
|
||||||
var meta = new List<IClientMeta>();
|
var metas = await Task.WhenAll(_metaActions.Where(kvp => kvp.Key != MetaType.Information)
|
||||||
|
.Select(async kvp => await kvp.Value[0](request)));
|
||||||
|
|
||||||
foreach (var (type, actions) in _metaActions)
|
return metas.SelectMany(m => (IEnumerable<IClientMeta>)m)
|
||||||
{
|
.OrderByDescending(m => m.When)
|
||||||
// information is not listed chronologically
|
|
||||||
if (type != MetaType.Information)
|
|
||||||
{
|
|
||||||
var metaItems = await actions[0](request);
|
|
||||||
meta.AddRange(metaItems);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return meta.OrderByDescending(_meta => _meta.When)
|
|
||||||
.Take(request.Count)
|
.Take(request.Count)
|
||||||
.ToList();
|
.ToList();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<IEnumerable<T>> GetRuntimeMeta<T>(ClientPaginationRequest request, MetaType metaType) where T : IClientMeta
|
public async Task<IEnumerable<T>> GetRuntimeMeta<T>(ClientPaginationRequest request, MetaType metaType) where T : IClientMeta
|
||||||
{
|
{
|
||||||
IEnumerable<T> meta;
|
|
||||||
if (metaType == MetaType.Information)
|
if (metaType == MetaType.Information)
|
||||||
{
|
{
|
||||||
var allMeta = new List<T>();
|
var allMeta = new List<T>();
|
||||||
|
|
||||||
foreach (var individualMetaRegistration in _metaActions[metaType])
|
var completedMeta = await Task.WhenAll(_metaActions[metaType].Select(async individualMetaRegistration =>
|
||||||
{
|
(IEnumerable<T>)await individualMetaRegistration(request)));
|
||||||
allMeta.AddRange(await individualMetaRegistration(request));
|
|
||||||
}
|
allMeta.AddRange(completedMeta.SelectMany(meta => meta));
|
||||||
|
|
||||||
return ProcessInformationMeta(allMeta);
|
return ProcessInformationMeta(allMeta);
|
||||||
}
|
}
|
||||||
|
|
||||||
else
|
var meta = await _metaActions[metaType][0](request) as IEnumerable<T>;
|
||||||
{
|
|
||||||
meta = await _metaActions[metaType][0](request) as IEnumerable<T>;
|
|
||||||
}
|
|
||||||
|
|
||||||
return meta;
|
return meta;
|
||||||
}
|
}
|
||||||
|
453
Application/Misc/MetaServiceV2.cs
Normal file
453
Application/Misc/MetaServiceV2.cs
Normal file
@ -0,0 +1,453 @@
|
|||||||
|
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.LogDebug("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, token) as IEnumerable<T>;
|
||||||
|
|
||||||
|
return meta;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static IEnumerable<T> ProcessInformationMeta<T>(IEnumerable<T> meta) where T : IClientMeta
|
||||||
|
{
|
||||||
|
return meta;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static bool ValidArgs(string key, int clientId) => !string.IsNullOrWhiteSpace(key) && clientId > 0;
|
||||||
|
}
|
@ -6,6 +6,7 @@ using SharedLibraryCore.Interfaces;
|
|||||||
using System.Linq;
|
using System.Linq;
|
||||||
using SharedLibraryCore;
|
using SharedLibraryCore;
|
||||||
using IW4MAdmin.Application.API.Master;
|
using IW4MAdmin.Application.API.Master;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using SharedLibraryCore.Configuration;
|
using SharedLibraryCore.Configuration;
|
||||||
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
||||||
@ -39,24 +40,23 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public IEnumerable<IPlugin> DiscoverScriptPlugins()
|
public IEnumerable<IPlugin> DiscoverScriptPlugins()
|
||||||
{
|
{
|
||||||
string pluginDir = $"{Utilities.OperatingDirectory}{PLUGIN_DIR}{Path.DirectorySeparatorChar}";
|
var pluginDir = $"{Utilities.OperatingDirectory}{PLUGIN_DIR}{Path.DirectorySeparatorChar}";
|
||||||
|
|
||||||
if (Directory.Exists(pluginDir))
|
if (!Directory.Exists(pluginDir))
|
||||||
{
|
{
|
||||||
var scriptPluginFiles = Directory.GetFiles(pluginDir, "*.js").AsEnumerable().Union(GetRemoteScripts());
|
return Enumerable.Empty<IPlugin>();
|
||||||
|
}
|
||||||
|
|
||||||
_logger.LogDebug("Discovered {count} potential script plugins", scriptPluginFiles.Count());
|
var scriptPluginFiles =
|
||||||
|
Directory.GetFiles(pluginDir, "*.js").AsEnumerable().Union(GetRemoteScripts()).ToList();
|
||||||
|
|
||||||
if (scriptPluginFiles.Count() > 0)
|
_logger.LogDebug("Discovered {count} potential script plugins", scriptPluginFiles.Count);
|
||||||
{
|
|
||||||
foreach (string fileName in scriptPluginFiles)
|
return scriptPluginFiles.Select(fileName =>
|
||||||
{
|
{
|
||||||
_logger.LogDebug("Discovered script plugin {fileName}", fileName);
|
_logger.LogDebug("Discovered script plugin {fileName}", fileName);
|
||||||
var plugin = new ScriptPlugin(_logger, fileName);
|
return new ScriptPlugin(_logger, fileName);
|
||||||
yield return plugin;
|
}).ToList();
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -83,19 +83,47 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
.GroupBy(_assembly => _assembly.FullName).Select(_assembly => _assembly.OrderByDescending(_assembly => _assembly.GetName().Version).First());
|
.GroupBy(_assembly => _assembly.FullName).Select(_assembly => _assembly.OrderByDescending(_assembly => _assembly.GetName().Version).First());
|
||||||
|
|
||||||
pluginTypes = assemblies
|
pluginTypes = assemblies
|
||||||
.SelectMany(_asm => _asm.GetTypes())
|
.SelectMany(_asm =>
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
return _asm.GetTypes();
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
return Enumerable.Empty<Type>();
|
||||||
|
}
|
||||||
|
})
|
||||||
.Where(_assemblyType => _assemblyType.GetInterface(nameof(IPlugin), false) != null);
|
.Where(_assemblyType => _assemblyType.GetInterface(nameof(IPlugin), false) != null);
|
||||||
|
|
||||||
_logger.LogDebug("Discovered {count} plugin implementations", pluginTypes.Count());
|
_logger.LogDebug("Discovered {count} plugin implementations", pluginTypes.Count());
|
||||||
|
|
||||||
commandTypes = assemblies
|
commandTypes = assemblies
|
||||||
.SelectMany(_asm => _asm.GetTypes())
|
.SelectMany(_asm =>{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
return _asm.GetTypes();
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
return Enumerable.Empty<Type>();
|
||||||
|
}
|
||||||
|
})
|
||||||
.Where(_assemblyType => _assemblyType.IsClass && _assemblyType.BaseType == typeof(Command));
|
.Where(_assemblyType => _assemblyType.IsClass && _assemblyType.BaseType == typeof(Command));
|
||||||
|
|
||||||
_logger.LogDebug("Discovered {count} plugin commands", commandTypes.Count());
|
_logger.LogDebug("Discovered {count} plugin commands", commandTypes.Count());
|
||||||
|
|
||||||
configurationTypes = assemblies
|
configurationTypes = assemblies
|
||||||
.SelectMany(asm => asm.GetTypes())
|
.SelectMany(asm => {
|
||||||
|
try
|
||||||
|
{
|
||||||
|
return asm.GetTypes();
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
return Enumerable.Empty<Type>();
|
||||||
|
}
|
||||||
|
})
|
||||||
.Where(asmType =>
|
.Where(asmType =>
|
||||||
asmType.IsClass && asmType.GetInterface(nameof(IBaseConfiguration), false) != null);
|
asmType.IsClass && asmType.GetInterface(nameof(IBaseConfiguration), false) != null);
|
||||||
|
|
||||||
|
@ -6,7 +6,6 @@ using System;
|
|||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Data.Models.Client;
|
using Data.Models.Client;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using static SharedLibraryCore.Database.Models.EFClient;
|
|
||||||
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
||||||
|
|
||||||
namespace IW4MAdmin.Application.Misc
|
namespace IW4MAdmin.Application.Misc
|
||||||
@ -16,14 +15,15 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public class ScriptCommand : Command
|
public class ScriptCommand : Command
|
||||||
{
|
{
|
||||||
private readonly Action<GameEvent> _executeAction;
|
private readonly Func<GameEvent, Task> _executeAction;
|
||||||
private readonly ILogger _logger;
|
private readonly ILogger _logger;
|
||||||
|
|
||||||
public ScriptCommand(string name, string alias, string description, bool isTargetRequired, EFClient.Permission permission,
|
public ScriptCommand(string name, string alias, string description, bool isTargetRequired,
|
||||||
CommandArgument[] args, Action<GameEvent> executeAction, CommandConfiguration config, ITranslationLookup layout, ILogger<ScriptCommand> logger)
|
EFClient.Permission permission,
|
||||||
|
CommandArgument[] args, Func<GameEvent, Task> executeAction, CommandConfiguration config,
|
||||||
|
ITranslationLookup layout, ILogger<ScriptCommand> logger, Server.Game[] supportedGames)
|
||||||
: base(config, layout)
|
: base(config, layout)
|
||||||
{
|
{
|
||||||
|
|
||||||
_executeAction = executeAction;
|
_executeAction = executeAction;
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
Name = name;
|
Name = name;
|
||||||
@ -32,6 +32,7 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
RequiresTarget = isTargetRequired;
|
RequiresTarget = isTargetRequired;
|
||||||
Permission = permission;
|
Permission = permission;
|
||||||
Arguments = args;
|
Arguments = args;
|
||||||
|
SupportedGames = supportedGames;
|
||||||
}
|
}
|
||||||
|
|
||||||
public override async Task ExecuteAsync(GameEvent e)
|
public override async Task ExecuteAsync(GameEvent e)
|
||||||
@ -43,7 +44,7 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await Task.Run(() => _executeAction(e));
|
await _executeAction(e);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
@ -13,6 +13,7 @@ using System.Linq;
|
|||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using Jint.Runtime.Interop;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using Serilog.Context;
|
using Serilog.Context;
|
||||||
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
||||||
@ -36,12 +37,12 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public bool IsParser { get; private set; }
|
public bool IsParser { get; private set; }
|
||||||
|
|
||||||
public FileSystemWatcher Watcher { get; private set; }
|
public FileSystemWatcher Watcher { get; }
|
||||||
|
|
||||||
private Engine _scriptEngine;
|
private Engine _scriptEngine;
|
||||||
private readonly string _fileName;
|
private readonly string _fileName;
|
||||||
private readonly SemaphoreSlim _onProcessing;
|
private readonly SemaphoreSlim _onProcessing = new(1, 1);
|
||||||
private bool successfullyLoaded;
|
private bool _successfullyLoaded;
|
||||||
private readonly List<string> _registeredCommandNames;
|
private readonly List<string> _registeredCommandNames;
|
||||||
private readonly ILogger _logger;
|
private readonly ILogger _logger;
|
||||||
|
|
||||||
@ -49,15 +50,14 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_fileName = filename;
|
_fileName = filename;
|
||||||
Watcher = new FileSystemWatcher()
|
Watcher = new FileSystemWatcher
|
||||||
{
|
{
|
||||||
Path = workingDirectory == null ? $"{Utilities.OperatingDirectory}Plugins{Path.DirectorySeparatorChar}" : workingDirectory,
|
Path = workingDirectory ?? $"{Utilities.OperatingDirectory}Plugins{Path.DirectorySeparatorChar}",
|
||||||
NotifyFilter = NotifyFilters.Size,
|
NotifyFilter = NotifyFilters.Size,
|
||||||
Filter = _fileName.Split(Path.DirectorySeparatorChar).Last()
|
Filter = _fileName.Split(Path.DirectorySeparatorChar).Last()
|
||||||
};
|
};
|
||||||
|
|
||||||
Watcher.EnableRaisingEvents = true;
|
Watcher.EnableRaisingEvents = true;
|
||||||
_onProcessing = new SemaphoreSlim(1, 1);
|
|
||||||
_registeredCommandNames = new List<string>();
|
_registeredCommandNames = new List<string>();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,12 +67,13 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
_onProcessing.Dispose();
|
_onProcessing.Dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task Initialize(IManager manager, IScriptCommandFactory scriptCommandFactory, IScriptPluginServiceResolver serviceResolver)
|
public async Task Initialize(IManager manager, IScriptCommandFactory scriptCommandFactory,
|
||||||
|
IScriptPluginServiceResolver serviceResolver)
|
||||||
|
{
|
||||||
|
try
|
||||||
{
|
{
|
||||||
await _onProcessing.WaitAsync();
|
await _onProcessing.WaitAsync();
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// for some reason we get an event trigger when the file is not finished being modified.
|
// for some reason we get an event trigger when the file is not finished being modified.
|
||||||
// this must have been a change in .NET CORE 3.x
|
// this must have been a change in .NET CORE 3.x
|
||||||
// so if the new file is empty we can't process it yet
|
// so if the new file is empty we can't process it yet
|
||||||
@ -81,26 +82,27 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool firstRun = _scriptEngine == null;
|
var firstRun = _scriptEngine == null;
|
||||||
|
|
||||||
// it's been loaded before so we need to call the unload event
|
// it's been loaded before so we need to call the unload event
|
||||||
if (!firstRun)
|
if (!firstRun)
|
||||||
{
|
{
|
||||||
await OnUnloadAsync();
|
await OnUnloadAsync();
|
||||||
|
|
||||||
foreach (string commandName in _registeredCommandNames)
|
foreach (var commandName in _registeredCommandNames)
|
||||||
{
|
{
|
||||||
_logger.LogDebug("Removing plugin registered command {command}", commandName);
|
_logger.LogDebug("Removing plugin registered command {Command}", commandName);
|
||||||
manager.RemoveCommandByName(commandName);
|
manager.RemoveCommandByName(commandName);
|
||||||
}
|
}
|
||||||
|
|
||||||
_registeredCommandNames.Clear();
|
_registeredCommandNames.Clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
successfullyLoaded = false;
|
_successfullyLoaded = false;
|
||||||
string script;
|
string script;
|
||||||
|
|
||||||
using (var stream = new FileStream(_fileName, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))
|
await using (var stream =
|
||||||
|
new FileStream(_fileName, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))
|
||||||
{
|
{
|
||||||
using (var reader = new StreamReader(stream, Encoding.Default))
|
using (var reader = new StreamReader(stream, Encoding.Default))
|
||||||
{
|
{
|
||||||
@ -116,39 +118,28 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
typeof(Utilities).Assembly,
|
typeof(Utilities).Assembly,
|
||||||
typeof(Encoding).Assembly
|
typeof(Encoding).Assembly
|
||||||
})
|
})
|
||||||
.CatchClrExceptions());
|
.CatchClrExceptions()
|
||||||
|
.AddObjectConverter(new PermissionLevelToStringConverter()));
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
_scriptEngine.Execute(script);
|
_scriptEngine.Execute(script);
|
||||||
}
|
|
||||||
catch (JavaScriptException ex)
|
|
||||||
{
|
|
||||||
|
|
||||||
_logger.LogError(ex,
|
|
||||||
"Encountered JavaScript runtime error while executing {methodName} for script plugin {plugin} at {@locationInfo}",
|
|
||||||
nameof(Initialize), _fileName, ex.Location);
|
|
||||||
throw new PluginException($"A JavaScript parsing error occured while initializing script plugin");
|
|
||||||
}
|
|
||||||
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
|
|
||||||
_logger.LogError(e,
|
|
||||||
"Encountered unexpected error while running {methodName} for script plugin {plugin}",
|
|
||||||
nameof(Initialize), _fileName);
|
|
||||||
throw new PluginException($"An unexpected error occured while initialization script plugin");
|
|
||||||
}
|
|
||||||
|
|
||||||
_scriptEngine.SetValue("_localization", Utilities.CurrentLocalization);
|
_scriptEngine.SetValue("_localization", Utilities.CurrentLocalization);
|
||||||
_scriptEngine.SetValue("_serviceResolver", serviceResolver);
|
_scriptEngine.SetValue("_serviceResolver", serviceResolver);
|
||||||
dynamic pluginObject = _scriptEngine.GetValue("plugin").ToObject();
|
_scriptEngine.SetValue("_lock", _onProcessing);
|
||||||
|
dynamic pluginObject = _scriptEngine.Evaluate("plugin").ToObject();
|
||||||
|
|
||||||
Author = pluginObject.author;
|
Author = pluginObject.author;
|
||||||
Name = pluginObject.name;
|
Name = pluginObject.name;
|
||||||
Version = (float)pluginObject.version;
|
Version = (float)pluginObject.version;
|
||||||
|
|
||||||
var commands = _scriptEngine.GetValue("commands");
|
var commands = JsValue.Undefined;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
commands = _scriptEngine.Evaluate("commands");
|
||||||
|
}
|
||||||
|
catch (JavaScriptException)
|
||||||
|
{
|
||||||
|
// ignore because commands aren't defined;
|
||||||
|
}
|
||||||
|
|
||||||
if (commands != JsValue.Undefined)
|
if (commands != JsValue.Undefined)
|
||||||
{
|
{
|
||||||
@ -156,7 +147,7 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
{
|
{
|
||||||
foreach (var command in GenerateScriptCommands(commands, scriptCommandFactory))
|
foreach (var command in GenerateScriptCommands(commands, scriptCommandFactory))
|
||||||
{
|
{
|
||||||
_logger.LogDebug("Adding plugin registered command {commandName}", command.Name);
|
_logger.LogDebug("Adding plugin registered command {CommandName}", command.Name);
|
||||||
manager.AddAdditionalCommand(command);
|
manager.AddAdditionalCommand(command);
|
||||||
_registeredCommandNames.Add(command.Name);
|
_registeredCommandNames.Add(command.Name);
|
||||||
}
|
}
|
||||||
@ -164,51 +155,110 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
|
|
||||||
catch (RuntimeBinderException e)
|
catch (RuntimeBinderException e)
|
||||||
{
|
{
|
||||||
throw new PluginException($"Not all required fields were found: {e.Message}") { PluginFile = _fileName };
|
throw new PluginException($"Not all required fields were found: {e.Message}")
|
||||||
|
{ PluginFile = _fileName };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_scriptEngine.SetValue("_configHandler", new ScriptPluginConfigurationWrapper(Name, _scriptEngine));
|
|
||||||
await OnLoadAsync(manager);
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (pluginObject.isParser)
|
if (pluginObject.isParser)
|
||||||
{
|
{
|
||||||
|
await OnLoadAsync(manager);
|
||||||
IsParser = true;
|
IsParser = true;
|
||||||
IEventParser eventParser = (IEventParser)_scriptEngine.GetValue("eventParser").ToObject();
|
var eventParser = (IEventParser)_scriptEngine.Evaluate("eventParser").ToObject();
|
||||||
IRConParser rconParser = (IRConParser)_scriptEngine.GetValue("rconParser").ToObject();
|
var rconParser = (IRConParser)_scriptEngine.Evaluate("rconParser").ToObject();
|
||||||
manager.AdditionalEventParsers.Add(eventParser);
|
manager.AdditionalEventParsers.Add(eventParser);
|
||||||
manager.AdditionalRConParsers.Add(rconParser);
|
manager.AdditionalRConParsers.Add(rconParser);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
catch (RuntimeBinderException) { }
|
catch (RuntimeBinderException)
|
||||||
|
{
|
||||||
|
var configWrapper = new ScriptPluginConfigurationWrapper(Name, _scriptEngine);
|
||||||
|
await configWrapper.InitializeAsync();
|
||||||
|
_scriptEngine.SetValue("_configHandler", configWrapper);
|
||||||
|
await OnLoadAsync(manager);
|
||||||
|
}
|
||||||
|
|
||||||
if (!firstRun)
|
if (!firstRun)
|
||||||
{
|
{
|
||||||
await OnLoadAsync(manager);
|
await OnLoadAsync(manager);
|
||||||
}
|
}
|
||||||
|
|
||||||
successfullyLoaded = true;
|
_successfullyLoaded = true;
|
||||||
|
}
|
||||||
|
catch (JavaScriptException ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex,
|
||||||
|
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin} at {@LocationInfo}",
|
||||||
|
nameof(Initialize), Path.GetFileName(_fileName), ex.Location);
|
||||||
|
|
||||||
|
throw new PluginException("An error occured while initializing script plugin");
|
||||||
|
}
|
||||||
|
catch (Exception ex) when (ex.InnerException is JavaScriptException jsEx)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex,
|
||||||
|
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin} initialization {@LocationInfo}",
|
||||||
|
nameof(Initialize), _fileName, jsEx.Location);
|
||||||
|
|
||||||
|
throw new PluginException("An error occured while initializing script plugin");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex,
|
||||||
|
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin}",
|
||||||
|
nameof(OnLoadAsync), Path.GetFileName(_fileName));
|
||||||
|
|
||||||
|
throw new PluginException("An error occured while executing action for script plugin");
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
if (_onProcessing.CurrentCount == 0)
|
||||||
|
{
|
||||||
|
_onProcessing.Release(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task OnEventAsync(GameEvent gameEvent, Server server)
|
||||||
|
{
|
||||||
|
if (!_successfullyLoaded)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
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)
|
catch (JavaScriptException ex)
|
||||||
|
{
|
||||||
|
using (LogContext.PushProperty("Server", server.ToString()))
|
||||||
{
|
{
|
||||||
_logger.LogError(ex,
|
_logger.LogError(ex,
|
||||||
"Encountered JavaScript runtime error while executing {methodName} for script plugin {plugin} initialization {@locationInfo}",
|
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin} with event type {EventType} {@LocationInfo}",
|
||||||
nameof(OnLoadAsync), _fileName, ex.Location);
|
nameof(OnEventAsync), Path.GetFileName(_fileName), gameEvent.Type, ex.Location);
|
||||||
|
}
|
||||||
|
|
||||||
throw new PluginException("An error occured while initializing script plugin");
|
throw new PluginException("An error occured while executing action for script plugin");
|
||||||
}
|
}
|
||||||
|
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
using (LogContext.PushProperty("Server", server.ToString()))
|
||||||
{
|
{
|
||||||
_logger.LogError(ex,
|
_logger.LogError(ex,
|
||||||
"Encountered unexpected error while running {methodName} for script plugin {plugin}",
|
"Encountered error while running {MethodName} for script plugin {Plugin} with event type {EventType}",
|
||||||
nameof(OnLoadAsync), _fileName);
|
nameof(OnEventAsync), _fileName, gameEvent.Type);
|
||||||
|
}
|
||||||
|
|
||||||
throw new PluginException("An unexpected error occured while initializing script plugin");
|
throw new PluginException("An error occured while executing action for script plugin");
|
||||||
}
|
}
|
||||||
|
|
||||||
finally
|
finally
|
||||||
@ -220,73 +270,71 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task OnEventAsync(GameEvent E, Server S)
|
|
||||||
{
|
|
||||||
if (successfullyLoaded)
|
|
||||||
{
|
|
||||||
await _onProcessing.WaitAsync();
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
_scriptEngine.SetValue("_gameEvent", E);
|
|
||||||
_scriptEngine.SetValue("_server", S);
|
|
||||||
_scriptEngine.SetValue("_IW4MAdminClient", Utilities.IW4MAdminClient(S));
|
|
||||||
_scriptEngine.Execute("plugin.onEventAsync(_gameEvent, _server)").GetCompletionValue();
|
|
||||||
}
|
|
||||||
|
|
||||||
catch (JavaScriptException ex)
|
|
||||||
{
|
|
||||||
using (LogContext.PushProperty("Server", S.ToString()))
|
|
||||||
{
|
|
||||||
_logger.LogError(ex,
|
|
||||||
"Encountered JavaScript runtime error while executing {methodName} for script plugin {plugin} with event type {eventType} {@locationInfo}",
|
|
||||||
nameof(OnEventAsync), _fileName, E.Type, ex.Location);
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new PluginException($"An error occured while executing action for script plugin");
|
|
||||||
}
|
|
||||||
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
using (LogContext.PushProperty("Server", S.ToString()))
|
|
||||||
{
|
|
||||||
_logger.LogError(e,
|
|
||||||
"Encountered unexpected error while running {methodName} for script plugin {plugin} with event type {eventType}",
|
|
||||||
nameof(OnEventAsync), _fileName, E.Type);
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new PluginException($"An error occured while executing action for script plugin");
|
|
||||||
}
|
|
||||||
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
if (_onProcessing.CurrentCount == 0)
|
|
||||||
{
|
|
||||||
_onProcessing.Release(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Task OnLoadAsync(IManager manager)
|
public Task OnLoadAsync(IManager manager)
|
||||||
{
|
{
|
||||||
_logger.LogDebug("OnLoad executing for {name}", Name);
|
try
|
||||||
|
{
|
||||||
|
_logger.LogDebug("OnLoad executing for {Name}", Name);
|
||||||
_scriptEngine.SetValue("_manager", manager);
|
_scriptEngine.SetValue("_manager", manager);
|
||||||
return Task.FromResult(_scriptEngine.Execute("plugin.onLoadAsync(_manager)").GetCompletionValue());
|
_scriptEngine.SetValue("getDvar", BeginGetDvar);
|
||||||
|
_scriptEngine.SetValue("setDvar", BeginSetDvar);
|
||||||
|
_scriptEngine.Evaluate("plugin.onLoadAsync(_manager)");
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
catch (JavaScriptException ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex,
|
||||||
|
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin} at {@LocationInfo}",
|
||||||
|
nameof(OnLoadAsync), Path.GetFileName(_fileName), ex.Location);
|
||||||
|
|
||||||
|
throw new PluginException("A runtime error occured while executing action for script plugin");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex,
|
||||||
|
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin}",
|
||||||
|
nameof(OnLoadAsync), Path.GetFileName(_fileName));
|
||||||
|
|
||||||
|
throw new PluginException("An error occured while executing action for script plugin");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public Task OnTickAsync(Server S)
|
public async Task OnTickAsync(Server server)
|
||||||
{
|
{
|
||||||
_scriptEngine.SetValue("_server", S);
|
_scriptEngine.SetValue("_server", server);
|
||||||
return Task.FromResult(_scriptEngine.Execute("plugin.onTickAsync(_server)").GetCompletionValue());
|
await Task.FromResult(_scriptEngine.Evaluate("plugin.onTickAsync(_server)"));
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task OnUnloadAsync()
|
public Task OnUnloadAsync()
|
||||||
{
|
{
|
||||||
if (successfullyLoaded)
|
if (!_successfullyLoaded)
|
||||||
{
|
{
|
||||||
await Task.FromResult(_scriptEngine.Execute("plugin.onUnloadAsync()").GetCompletionValue());
|
return Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
_scriptEngine.Evaluate("plugin.onUnloadAsync()");
|
||||||
|
}
|
||||||
|
catch (JavaScriptException ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex,
|
||||||
|
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin} at {@LocationInfo}",
|
||||||
|
nameof(OnUnloadAsync), Path.GetFileName(_fileName), ex.Location);
|
||||||
|
|
||||||
|
throw new PluginException("A runtime error occured while executing action for script plugin");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex,
|
||||||
|
"Encountered JavaScript runtime error while executing {MethodName} for script plugin {Plugin}",
|
||||||
|
nameof(OnUnloadAsync), Path.GetFileName(_fileName));
|
||||||
|
|
||||||
|
throw new PluginException("An error occured while executing action for script plugin");
|
||||||
|
}
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -295,9 +343,10 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
/// <param name="commands">commands value from jint parser</param>
|
/// <param name="commands">commands value from jint parser</param>
|
||||||
/// <param name="scriptCommandFactory">factory to create the command from</param>
|
/// <param name="scriptCommandFactory">factory to create the command from</param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public IEnumerable<IManagerCommand> GenerateScriptCommands(JsValue commands, IScriptCommandFactory scriptCommandFactory)
|
private IEnumerable<IManagerCommand> GenerateScriptCommands(JsValue commands,
|
||||||
|
IScriptCommandFactory scriptCommandFactory)
|
||||||
{
|
{
|
||||||
List<IManagerCommand> commandList = new List<IManagerCommand>();
|
var commandList = new List<IManagerCommand>();
|
||||||
|
|
||||||
// go through each defined command
|
// go through each defined command
|
||||||
foreach (var command in commands.AsArray())
|
foreach (var command in commands.AsArray())
|
||||||
@ -307,9 +356,10 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
string alias = dynamicCommand.alias;
|
string alias = dynamicCommand.alias;
|
||||||
string description = dynamicCommand.description;
|
string description = dynamicCommand.description;
|
||||||
string permission = dynamicCommand.permission;
|
string permission = dynamicCommand.permission;
|
||||||
bool targetRequired = false;
|
List<Server.Game> supportedGames = null;
|
||||||
|
var targetRequired = false;
|
||||||
|
|
||||||
List<(string, bool)> args = new List<(string, bool)>();
|
var args = new List<(string, bool)>();
|
||||||
dynamic arguments = null;
|
dynamic arguments = null;
|
||||||
|
|
||||||
try
|
try
|
||||||
@ -340,26 +390,152 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void execute(GameEvent e)
|
|
||||||
{
|
|
||||||
_scriptEngine.SetValue("_event", e);
|
|
||||||
var jsEventObject = _scriptEngine.GetValue("_event");
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
dynamicCommand.execute.Target.Invoke(jsEventObject);
|
foreach (var game in dynamicCommand.supportedGames)
|
||||||
|
{
|
||||||
|
supportedGames ??= new List<Server.Game>();
|
||||||
|
supportedGames.Add(Enum.Parse(typeof(Server.Game), game.ToString()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (RuntimeBinderException)
|
||||||
|
{
|
||||||
|
// supported games is optional
|
||||||
|
}
|
||||||
|
|
||||||
|
async Task Execute(GameEvent gameEvent)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await _onProcessing.WaitAsync();
|
||||||
|
|
||||||
|
_scriptEngine.SetValue("_event", gameEvent);
|
||||||
|
var jsEventObject = _scriptEngine.Evaluate("_event");
|
||||||
|
|
||||||
|
dynamicCommand.execute.Target.Invoke(_scriptEngine, jsEventObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
catch (JavaScriptException ex)
|
catch (JavaScriptException ex)
|
||||||
{
|
{
|
||||||
throw new PluginException($"An error occured while executing action for script plugin: {ex.Error} (Line: {ex.Location.Start.Line}, Character: {ex.Location.Start.Column})") { PluginFile = _fileName };
|
using (LogContext.PushProperty("Server", gameEvent.Owner?.ToString()))
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Could not execute command action for {Filename} {@Location}",
|
||||||
|
Path.GetFileName(_fileName), ex.Location);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new PluginException("A runtime error occured while executing action for script plugin");
|
||||||
|
}
|
||||||
|
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
using (LogContext.PushProperty("Server", gameEvent.Owner?.ToString()))
|
||||||
|
{
|
||||||
|
_logger.LogError(ex,
|
||||||
|
"Could not execute command action for script plugin {FileName}",
|
||||||
|
Path.GetFileName(_fileName));
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new PluginException("An error occured while executing action for script plugin");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
if (_onProcessing.CurrentCount == 0)
|
||||||
|
{
|
||||||
|
_onProcessing.Release(1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
commandList.Add(scriptCommandFactory.CreateScriptCommand(name, alias, description, permission, targetRequired, args, execute));
|
commandList.Add(scriptCommandFactory.CreateScriptCommand(name, alias, description, permission,
|
||||||
|
targetRequired, args, Execute, supportedGames?.ToArray()));
|
||||||
}
|
}
|
||||||
|
|
||||||
return commandList;
|
return commandList;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void BeginGetDvar(Server server, string dvarName, Delegate onCompleted)
|
||||||
|
{
|
||||||
|
var tokenSource = new CancellationTokenSource();
|
||||||
|
tokenSource.CancelAfter(TimeSpan.FromSeconds(15));
|
||||||
|
|
||||||
|
server.BeginGetDvar(dvarName, result =>
|
||||||
|
{
|
||||||
|
var shouldRelease = false;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
_onProcessing.Wait(tokenSource.Token);
|
||||||
|
shouldRelease = true;
|
||||||
|
var (success, value) = (ValueTuple<bool, string>)result.AsyncState;
|
||||||
|
|
||||||
|
onCompleted.DynamicInvoke(JsValue.Undefined,
|
||||||
|
new[]
|
||||||
|
{
|
||||||
|
JsValue.FromObject(_scriptEngine, server),
|
||||||
|
JsValue.FromObject(_scriptEngine, dvarName),
|
||||||
|
JsValue.FromObject(_scriptEngine, value),
|
||||||
|
JsValue.FromObject(_scriptEngine, success)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
if (_onProcessing.CurrentCount == 0 && shouldRelease)
|
||||||
|
{
|
||||||
|
_onProcessing.Release();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, tokenSource.Token);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void BeginSetDvar(Server server, string dvarName, string dvarValue, Delegate onCompleted)
|
||||||
|
{
|
||||||
|
var tokenSource = new CancellationTokenSource();
|
||||||
|
tokenSource.CancelAfter(TimeSpan.FromSeconds(15));
|
||||||
|
|
||||||
|
server.BeginSetDvar(dvarName, dvarValue, result =>
|
||||||
|
{
|
||||||
|
var shouldRelease = false;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
_onProcessing.Wait(tokenSource.Token);
|
||||||
|
shouldRelease = true;
|
||||||
|
var success = (bool)result.AsyncState;
|
||||||
|
|
||||||
|
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 && shouldRelease)
|
||||||
|
{
|
||||||
|
_onProcessing.Release();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, tokenSource.Token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class PermissionLevelToStringConverter : IObjectConverter
|
||||||
|
{
|
||||||
|
public bool TryConvert(Engine engine, object value, out JsValue result)
|
||||||
|
{
|
||||||
|
if (value is Data.Models.Client.EFClient.Permission)
|
||||||
|
{
|
||||||
|
result = value.ToString();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
result = JsValue.Null;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
@ -12,19 +13,24 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
public class ScriptPluginConfigurationWrapper
|
public class ScriptPluginConfigurationWrapper
|
||||||
{
|
{
|
||||||
private readonly BaseConfigurationHandler<ScriptPluginConfiguration> _handler;
|
private readonly BaseConfigurationHandler<ScriptPluginConfiguration> _handler;
|
||||||
private readonly ScriptPluginConfiguration _config;
|
private ScriptPluginConfiguration _config;
|
||||||
private readonly string _pluginName;
|
private readonly string _pluginName;
|
||||||
private readonly Engine _scriptEngine;
|
private readonly Engine _scriptEngine;
|
||||||
|
|
||||||
public ScriptPluginConfigurationWrapper(string pluginName, Engine scriptEngine)
|
public ScriptPluginConfigurationWrapper(string pluginName, Engine scriptEngine)
|
||||||
{
|
{
|
||||||
_handler = new BaseConfigurationHandler<ScriptPluginConfiguration>("ScriptPluginSettings");
|
_handler = new BaseConfigurationHandler<ScriptPluginConfiguration>("ScriptPluginSettings");
|
||||||
_config = _handler.Configuration() ??
|
|
||||||
(ScriptPluginConfiguration) new ScriptPluginConfiguration().Generate();
|
|
||||||
_pluginName = pluginName;
|
_pluginName = pluginName;
|
||||||
_scriptEngine = scriptEngine;
|
_scriptEngine = scriptEngine;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task InitializeAsync()
|
||||||
|
{
|
||||||
|
await _handler.BuildAsync();
|
||||||
|
_config = _handler.Configuration() ??
|
||||||
|
(ScriptPluginConfiguration) new ScriptPluginConfiguration().Generate();
|
||||||
|
}
|
||||||
|
|
||||||
private static int? AsInteger(double d)
|
private static int? AsInteger(double d)
|
||||||
{
|
{
|
||||||
return int.TryParse(d.ToString(CultureInfo.InvariantCulture), out var parsed) ? parsed : (int?) null;
|
return int.TryParse(d.ToString(CultureInfo.InvariantCulture), out var parsed) ? parsed : (int?) null;
|
||||||
@ -79,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);
|
||||||
|
159
Application/Misc/ScriptPluginTimerHelper.cs
Normal file
159
Application/Misc/ScriptPluginTimerHelper.cs
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
using Jint.Native;
|
||||||
|
using Jint.Runtime;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using SharedLibraryCore.Interfaces;
|
||||||
|
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
||||||
|
|
||||||
|
namespace IW4MAdmin.Application.Misc;
|
||||||
|
|
||||||
|
public class ScriptPluginTimerHelper : IScriptPluginTimerHelper
|
||||||
|
{
|
||||||
|
private Timer _timer;
|
||||||
|
private Action _actions;
|
||||||
|
private Delegate _jsAction;
|
||||||
|
private string _actionName;
|
||||||
|
private const int DefaultDelay = 0;
|
||||||
|
private const int DefaultInterval = 1000;
|
||||||
|
private readonly ILogger _logger;
|
||||||
|
private readonly ManualResetEventSlim _onRunningTick = new();
|
||||||
|
private SemaphoreSlim _onDependentAction;
|
||||||
|
|
||||||
|
public ScriptPluginTimerHelper(ILogger<ScriptPluginTimerHelper> logger)
|
||||||
|
{
|
||||||
|
_logger = logger;
|
||||||
|
}
|
||||||
|
|
||||||
|
~ScriptPluginTimerHelper()
|
||||||
|
{
|
||||||
|
if (_timer != null)
|
||||||
|
{
|
||||||
|
Stop();
|
||||||
|
}
|
||||||
|
_onRunningTick.Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Start(int delay, int interval)
|
||||||
|
{
|
||||||
|
if (_actions is null)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException("Timer action must be defined before starting");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (delay < 0)
|
||||||
|
{
|
||||||
|
throw new ArgumentException("Timer delay must be >= 0");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (interval < 20)
|
||||||
|
{
|
||||||
|
throw new ArgumentException("Timer interval must be at least 20ms");
|
||||||
|
}
|
||||||
|
|
||||||
|
Stop();
|
||||||
|
|
||||||
|
_logger.LogDebug("Starting script timer...");
|
||||||
|
|
||||||
|
_onRunningTick.Set();
|
||||||
|
_timer ??= new Timer(callback => _actions(), null, delay, interval);
|
||||||
|
IsRunning = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Start(int interval)
|
||||||
|
{
|
||||||
|
Start(DefaultDelay, interval);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Start()
|
||||||
|
{
|
||||||
|
Start(DefaultDelay, DefaultInterval);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Stop()
|
||||||
|
{
|
||||||
|
if (_timer == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogDebug("Stopping script timer...");
|
||||||
|
_timer.Change(Timeout.Infinite, Timeout.Infinite);
|
||||||
|
_timer.Dispose();
|
||||||
|
_timer = null;
|
||||||
|
IsRunning = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnTick(Delegate action, string actionName)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(actionName))
|
||||||
|
{
|
||||||
|
throw new ArgumentException("actionName must be provided", nameof(actionName));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (action is null)
|
||||||
|
{
|
||||||
|
throw new ArgumentException("action must be provided", nameof(action));
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogDebug("Adding new action with name {ActionName}", actionName);
|
||||||
|
|
||||||
|
_jsAction = action;
|
||||||
|
_actionName = actionName;
|
||||||
|
_actions = OnTick;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ReleaseThreads()
|
||||||
|
{
|
||||||
|
_onRunningTick.Set();
|
||||||
|
|
||||||
|
if (_onDependentAction?.CurrentCount != 0)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_onDependentAction?.Release(1);
|
||||||
|
}
|
||||||
|
private void OnTick()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (!_onRunningTick.IsSet)
|
||||||
|
{
|
||||||
|
_logger.LogDebug("Previous {OnTick} is still running, so we are skipping this one",
|
||||||
|
nameof(OnTick));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_onRunningTick.Reset();
|
||||||
|
|
||||||
|
// the js engine is not thread safe so we need to ensure we're not executing OnTick and OnEventAsync simultaneously
|
||||||
|
_onDependentAction?.WaitAsync().Wait();
|
||||||
|
var start = DateTime.Now;
|
||||||
|
_jsAction.DynamicInvoke(JsValue.Undefined, new[] { JsValue.Undefined });
|
||||||
|
_logger.LogDebug("OnTick took {Time}ms", (DateTime.Now - start).TotalMilliseconds);
|
||||||
|
ReleaseThreads();
|
||||||
|
}
|
||||||
|
|
||||||
|
catch (Exception ex) when (ex.InnerException is JavaScriptException jsex)
|
||||||
|
{
|
||||||
|
_logger.LogError(jsex,
|
||||||
|
"Could not execute timer tick for script action {ActionName} [@{LocationInfo}]", _actionName,
|
||||||
|
jsex.Location);
|
||||||
|
ReleaseThreads();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Could not execute timer tick for script action {ActionName}", _actionName);
|
||||||
|
_onRunningTick.Set();
|
||||||
|
ReleaseThreads();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetDependency(SemaphoreSlim dependentSemaphore)
|
||||||
|
{
|
||||||
|
_onDependentAction = dependentSemaphore;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool IsRunning { get; private set; }
|
||||||
|
}
|
@ -94,7 +94,8 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
PeriodBlock = (int) (DateTimeOffset.UtcNow - DateTimeOffset.UnixEpoch).TotalMinutes,
|
PeriodBlock = (int) (DateTimeOffset.UtcNow - DateTimeOffset.UnixEpoch).TotalMinutes,
|
||||||
ServerId = await server.GetIdForServer(),
|
ServerId = await server.GetIdForServer(),
|
||||||
MapId = await GetOrCreateMap(server.CurrentMap.Name, (Reference.Game) server.GameName, token),
|
MapId = await GetOrCreateMap(server.CurrentMap.Name, (Reference.Game) server.GameName, token),
|
||||||
ClientCount = server.ClientNum
|
ClientCount = server.ClientNum,
|
||||||
|
ConnectionInterrupted = server.Throttled,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
|
@ -24,7 +24,7 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
private readonly IDataValueCache<EFServerSnapshot, List<ClientHistoryInfo>> _clientHistoryCache;
|
private readonly IDataValueCache<EFServerSnapshot, List<ClientHistoryInfo>> _clientHistoryCache;
|
||||||
|
|
||||||
private readonly TimeSpan? _cacheTimeSpan =
|
private readonly TimeSpan? _cacheTimeSpan =
|
||||||
Utilities.IsDevelopment ? TimeSpan.FromSeconds(1) : (TimeSpan?) TimeSpan.FromMinutes(1);
|
Utilities.IsDevelopment ? TimeSpan.FromSeconds(30) : (TimeSpan?) TimeSpan.FromMinutes(10);
|
||||||
|
|
||||||
public ServerDataViewer(ILogger<ServerDataViewer> logger, IDataValueCache<EFServerSnapshot, (int?, DateTime?)> snapshotCache,
|
public ServerDataViewer(ILogger<ServerDataViewer> logger, IDataValueCache<EFServerSnapshot, (int?, DateTime?)> snapshotCache,
|
||||||
IDataValueCache<EFClient, (int, int)> serverStatsCache,
|
IDataValueCache<EFClient, (int, int)> serverStatsCache,
|
||||||
@ -36,7 +36,8 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
_clientHistoryCache = clientHistoryCache;
|
_clientHistoryCache = clientHistoryCache;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<(int?, DateTime?)> MaxConcurrentClientsAsync(long? serverId = null, TimeSpan? overPeriod = null,
|
public async Task<(int?, DateTime?)>
|
||||||
|
MaxConcurrentClientsAsync(long? serverId = null, TimeSpan? overPeriod = null,
|
||||||
CancellationToken token = default)
|
CancellationToken token = default)
|
||||||
{
|
{
|
||||||
_snapshotCache.SetCacheItem(async (snapshots, cancellationToken) =>
|
_snapshotCache.SetCacheItem(async (snapshots, cancellationToken) =>
|
||||||
@ -83,7 +84,7 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
_logger.LogDebug("Max concurrent clients since {Start} is {Clients}", oldestEntry, maxClients);
|
_logger.LogDebug("Max concurrent clients since {Start} is {Clients}", oldestEntry, maxClients);
|
||||||
|
|
||||||
return (maxClients, maxClientsTime);
|
return (maxClients, maxClientsTime);
|
||||||
}, nameof(MaxConcurrentClientsAsync), _cacheTimeSpan);
|
}, nameof(MaxConcurrentClientsAsync), _cacheTimeSpan, true);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@ -107,7 +108,7 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
cancellationToken);
|
cancellationToken);
|
||||||
|
|
||||||
return (count, recentCount);
|
return (count, recentCount);
|
||||||
}, nameof(_serverStatsCache), _cacheTimeSpan);
|
}, nameof(_serverStatsCache), _cacheTimeSpan, true);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@ -134,7 +135,9 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
{
|
{
|
||||||
snapshot.ServerId,
|
snapshot.ServerId,
|
||||||
snapshot.CapturedAt,
|
snapshot.CapturedAt,
|
||||||
snapshot.ClientCount
|
snapshot.ClientCount,
|
||||||
|
snapshot.ConnectionInterrupted,
|
||||||
|
MapName = snapshot.Map.Name,
|
||||||
})
|
})
|
||||||
.OrderBy(snapshot => snapshot.CapturedAt)
|
.OrderBy(snapshot => snapshot.CapturedAt)
|
||||||
.ToListAsync(cancellationToken);
|
.ToListAsync(cancellationToken);
|
||||||
@ -142,8 +145,8 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
return history.GroupBy(snapshot => snapshot.ServerId).Select(byServer => new ClientHistoryInfo
|
return history.GroupBy(snapshot => snapshot.ServerId).Select(byServer => new ClientHistoryInfo
|
||||||
{
|
{
|
||||||
ServerId = byServer.Key,
|
ServerId = byServer.Key,
|
||||||
ClientCounts = byServer.Select(snapshot => new ClientCountSnapshot()
|
ClientCounts = byServer.Select(snapshot => new ClientCountSnapshot
|
||||||
{Time = snapshot.CapturedAt, ClientCount = snapshot.ClientCount}).ToList()
|
{ Time = snapshot.CapturedAt, ClientCount = snapshot.ClientCount, ConnectionInterrupted = snapshot.ConnectionInterrupted ?? false, Map = snapshot.MapName}).ToList()
|
||||||
}).ToList();
|
}).ToList();
|
||||||
}, nameof(_clientHistoryCache), TimeSpan.MaxValue);
|
}, nameof(_clientHistoryCache), TimeSpan.MaxValue);
|
||||||
|
|
||||||
|
@ -7,26 +7,26 @@ using System.Text;
|
|||||||
|
|
||||||
namespace IW4MAdmin.Application.Misc
|
namespace IW4MAdmin.Application.Misc
|
||||||
{
|
{
|
||||||
class TokenAuthentication : ITokenAuthentication
|
internal class TokenAuthentication : ITokenAuthentication
|
||||||
{
|
{
|
||||||
private readonly ConcurrentDictionary<long, TokenState> _tokens;
|
private readonly ConcurrentDictionary<long, TokenState> _tokens;
|
||||||
private readonly RNGCryptoServiceProvider _random;
|
private readonly RandomNumberGenerator _random;
|
||||||
private readonly static TimeSpan _timeoutPeriod = new TimeSpan(0, 0, 120);
|
private static readonly TimeSpan TimeoutPeriod = new TimeSpan(0, 0, 120);
|
||||||
private const short TOKEN_LENGTH = 4;
|
private const short TokenLength = 4;
|
||||||
|
|
||||||
public TokenAuthentication()
|
public TokenAuthentication()
|
||||||
{
|
{
|
||||||
_tokens = new ConcurrentDictionary<long, TokenState>();
|
_tokens = new ConcurrentDictionary<long, TokenState>();
|
||||||
_random = new RNGCryptoServiceProvider();
|
_random = RandomNumberGenerator.Create();
|
||||||
}
|
}
|
||||||
|
|
||||||
public bool AuthorizeToken(long networkId, string token)
|
public bool AuthorizeToken(long networkId, string token)
|
||||||
{
|
{
|
||||||
bool authorizeSuccessful = _tokens.ContainsKey(networkId) && _tokens[networkId].Token == token;
|
var authorizeSuccessful = _tokens.ContainsKey(networkId) && _tokens[networkId].Token == token;
|
||||||
|
|
||||||
if (authorizeSuccessful)
|
if (authorizeSuccessful)
|
||||||
{
|
{
|
||||||
_tokens.TryRemove(networkId, out TokenState _);
|
_tokens.TryRemove(networkId, out _);
|
||||||
}
|
}
|
||||||
|
|
||||||
return authorizeSuccessful;
|
return authorizeSuccessful;
|
||||||
@ -34,15 +34,15 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
|
|
||||||
public TokenState GenerateNextToken(long networkId)
|
public TokenState GenerateNextToken(long networkId)
|
||||||
{
|
{
|
||||||
TokenState state = null;
|
TokenState state;
|
||||||
|
|
||||||
if (_tokens.ContainsKey(networkId))
|
if (_tokens.ContainsKey(networkId))
|
||||||
{
|
{
|
||||||
state = _tokens[networkId];
|
state = _tokens[networkId];
|
||||||
|
|
||||||
if ((DateTime.Now - state.RequestTime) > _timeoutPeriod)
|
if ((DateTime.Now - state.RequestTime) > TimeoutPeriod)
|
||||||
{
|
{
|
||||||
_tokens.TryRemove(networkId, out TokenState _);
|
_tokens.TryRemove(networkId, out _);
|
||||||
}
|
}
|
||||||
|
|
||||||
else
|
else
|
||||||
@ -51,11 +51,11 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
state = new TokenState()
|
state = new TokenState
|
||||||
{
|
{
|
||||||
NetworkId = networkId,
|
NetworkId = networkId,
|
||||||
Token = _generateToken(),
|
Token = _generateToken(),
|
||||||
TokenDuration = _timeoutPeriod
|
TokenDuration = TimeoutPeriod
|
||||||
};
|
};
|
||||||
|
|
||||||
_tokens.TryAdd(networkId, state);
|
_tokens.TryAdd(networkId, state);
|
||||||
@ -63,31 +63,31 @@ namespace IW4MAdmin.Application.Misc
|
|||||||
// perform some housekeeping so we don't have built up tokens if they're not ever used
|
// perform some housekeeping so we don't have built up tokens if they're not ever used
|
||||||
foreach (var (key, value) in _tokens)
|
foreach (var (key, value) in _tokens)
|
||||||
{
|
{
|
||||||
if ((DateTime.Now - value.RequestTime) > _timeoutPeriod)
|
if ((DateTime.Now - value.RequestTime) > TimeoutPeriod)
|
||||||
{
|
{
|
||||||
_tokens.TryRemove(key, out TokenState _);
|
_tokens.TryRemove(key, out _);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
|
|
||||||
public string _generateToken()
|
private string _generateToken()
|
||||||
{
|
{
|
||||||
bool validCharacter(char c)
|
bool ValidCharacter(char c)
|
||||||
{
|
{
|
||||||
// this ensure that the characters are 0-9, A-Z, a-z
|
// this ensure that the characters are 0-9, A-Z, a-z
|
||||||
return (c > 47 && c < 58) || (c > 64 && c < 91) || (c > 96 && c < 123);
|
return (c > 47 && c < 58) || (c > 64 && c < 91) || (c > 96 && c < 123);
|
||||||
}
|
}
|
||||||
|
|
||||||
StringBuilder token = new StringBuilder();
|
var token = new StringBuilder();
|
||||||
|
|
||||||
while (token.Length < TOKEN_LENGTH)
|
var charSet = new byte[1];
|
||||||
|
while (token.Length < TokenLength)
|
||||||
{
|
{
|
||||||
byte[] charSet = new byte[1];
|
|
||||||
_random.GetBytes(charSet);
|
_random.GetBytes(charSet);
|
||||||
|
|
||||||
if (validCharacter((char)charSet[0]))
|
if (ValidCharacter((char)charSet[0]))
|
||||||
{
|
{
|
||||||
token.Append((char)charSet[0]);
|
token.Append((char)charSet[0]);
|
||||||
}
|
}
|
||||||
|
@ -7,8 +7,10 @@ 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 IW4MAdmin.Application.Misc;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using static SharedLibraryCore.Server;
|
using static SharedLibraryCore.Server;
|
||||||
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
using ILogger = Microsoft.Extensions.Logging.ILogger;
|
||||||
@ -77,19 +79,20 @@ 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);
|
command = command.FormatMessageForEngine(Configuration?.ColorCodeMapping);
|
||||||
|
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 +101,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 +112,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 +122,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 +142,36 @@ namespace IW4MAdmin.Application.RConParsers
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public virtual async Task<IStatusResponse> GetStatusAsync(IRConConnection connection)
|
public void BeginGetDvar(IRConConnection connection, string dvarName, AsyncCallback callback, CancellationToken token = default)
|
||||||
{
|
{
|
||||||
var response = await connection.SendQueryAsync(StaticHelpers.QueryType.COMMAND_STATUS);
|
GetDvarAsync<string>(connection, dvarName, token: token).ContinueWith(action =>
|
||||||
_logger.LogDebug("Status Response {response}", string.Join(Environment.NewLine, response));
|
{
|
||||||
|
if (action.Exception is null)
|
||||||
|
{
|
||||||
|
callback?.Invoke(new AsyncResult
|
||||||
|
{
|
||||||
|
IsCompleted = true,
|
||||||
|
AsyncState = (true, action.Result.Value)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
callback?.Invoke(new AsyncResult
|
||||||
|
{
|
||||||
|
IsCompleted = true,
|
||||||
|
AsyncState = (false, (string)null)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual async Task<IStatusResponse> GetStatusAsync(IRConConnection connection, CancellationToken token = default)
|
||||||
|
{
|
||||||
|
var response = await connection.SendQueryAsync(StaticHelpers.QueryType.COMMAND_STATUS, "status", token);
|
||||||
|
|
||||||
|
_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 +212,38 @@ 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;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void BeginSetDvar(IRConConnection connection, string dvarName, object dvarValue, AsyncCallback callback,
|
||||||
|
CancellationToken token = default)
|
||||||
|
{
|
||||||
|
SetDvarAsync(connection, dvarName, dvarValue, token).ContinueWith(action =>
|
||||||
|
{
|
||||||
|
if (action.Exception is null)
|
||||||
|
{
|
||||||
|
callback?.Invoke(new AsyncResult
|
||||||
|
{
|
||||||
|
IsCompleted = true,
|
||||||
|
AsyncState = true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
callback?.Invoke(new AsyncResult
|
||||||
|
{
|
||||||
|
IsCompleted = true,
|
||||||
|
AsyncState = false
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, token);
|
||||||
}
|
}
|
||||||
|
|
||||||
private List<EFClient> ClientsFromStatus(string[] Status)
|
private List<EFClient> ClientsFromStatus(string[] Status)
|
||||||
|
@ -3,6 +3,7 @@ using SharedLibraryCore.Interfaces;
|
|||||||
using SharedLibraryCore.RCon;
|
using SharedLibraryCore.RCon;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
|
using SharedLibraryCore.Formatting;
|
||||||
|
|
||||||
namespace IW4MAdmin.Application.RConParsers
|
namespace IW4MAdmin.Application.RConParsers
|
||||||
{
|
{
|
||||||
@ -28,6 +29,25 @@ namespace IW4MAdmin.Application.RConParsers
|
|||||||
public int NoticeMaximumLines { get; set; } = 8;
|
public int NoticeMaximumLines { get; set; } = 8;
|
||||||
public int NoticeMaxCharactersPerLine { get; set; } = 50;
|
public int NoticeMaxCharactersPerLine { get; set; } = 50;
|
||||||
public string NoticeLineSeparator { get; set; } = Environment.NewLine;
|
public string NoticeLineSeparator { get; set; } = Environment.NewLine;
|
||||||
|
public int? DefaultRConPort { get; set; }
|
||||||
|
public string DefaultInstallationDirectoryHint { get; set; }
|
||||||
|
public short FloodProtectInterval { get; set; } = 750;
|
||||||
|
|
||||||
|
public ColorCodeMapping ColorCodeMapping { get; set; } = new ColorCodeMapping
|
||||||
|
{
|
||||||
|
// this is the default mapping (IW4), but can be overridden as needed in the parsers
|
||||||
|
{ColorCodes.Black.ToString(), "^0"},
|
||||||
|
{ColorCodes.Red.ToString(), "^1"},
|
||||||
|
{ColorCodes.Green.ToString(), "^2"},
|
||||||
|
{ColorCodes.Yellow.ToString(), "^3"},
|
||||||
|
{ColorCodes.Blue.ToString(), "^4"},
|
||||||
|
{ColorCodes.Cyan.ToString(), "^5"},
|
||||||
|
{ColorCodes.Pink.ToString(), "^6"},
|
||||||
|
{ColorCodes.White.ToString(), "^7"},
|
||||||
|
{ColorCodes.Map.ToString(), "^8"},
|
||||||
|
{ColorCodes.Grey.ToString(), "^9"},
|
||||||
|
{ColorCodes.Wildcard.ToString(), ":^"},
|
||||||
|
};
|
||||||
|
|
||||||
public DynamicRConParserConfiguration(IParserRegexFactory parserRegexFactory)
|
public DynamicRConParserConfiguration(IParserRegexFactory parserRegexFactory)
|
||||||
{
|
{
|
||||||
|
BIN
Application/Resources/GeoLite2-Country.mmdb
Normal file
BIN
Application/Resources/GeoLite2-Country.mmdb
Normal file
Binary file not shown.
@ -5,9 +5,10 @@ using Microsoft.EntityFrameworkCore;
|
|||||||
|
|
||||||
namespace Data.Abstractions
|
namespace Data.Abstractions
|
||||||
{
|
{
|
||||||
public interface IDataValueCache<T, V> where T : class
|
public interface IDataValueCache<TEntityType, TReturnType> where TEntityType : class
|
||||||
{
|
{
|
||||||
void SetCacheItem(Func<DbSet<T>, CancellationToken, Task<V>> itemGetter, string keyName, TimeSpan? expirationTime = null);
|
void SetCacheItem(Func<DbSet<TEntityType>, CancellationToken, Task<TReturnType>> itemGetter, string keyName,
|
||||||
Task<V> GetCacheItem(string keyName, CancellationToken token = default);
|
TimeSpan? expirationTime = null, bool autoRefresh = false);
|
||||||
|
Task<TReturnType> GetCacheItem(string keyName, CancellationToken token = default);
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -23,7 +23,7 @@ namespace Data.Context
|
|||||||
{
|
{
|
||||||
var link = new EFAliasLink();
|
var link = new EFAliasLink();
|
||||||
|
|
||||||
context.Clients.Add(new EFClient()
|
context.Clients.Add(new EFClient
|
||||||
{
|
{
|
||||||
Active = false,
|
Active = false,
|
||||||
Connections = 0,
|
Connections = 0,
|
||||||
@ -33,7 +33,7 @@ namespace Data.Context
|
|||||||
Masked = true,
|
Masked = true,
|
||||||
NetworkId = 0,
|
NetworkId = 0,
|
||||||
AliasLink = link,
|
AliasLink = link,
|
||||||
CurrentAlias = new EFAlias()
|
CurrentAlias = new EFAlias
|
||||||
{
|
{
|
||||||
Link = link,
|
Link = link,
|
||||||
Active = true,
|
Active = true,
|
||||||
|
@ -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; }
|
||||||
|
|
||||||
@ -118,7 +119,10 @@ namespace Data.Context
|
|||||||
ent.HasIndex(a => a.Name);
|
ent.HasIndex(a => a.Name);
|
||||||
ent.Property(_alias => _alias.SearchableName).HasMaxLength(24);
|
ent.Property(_alias => _alias.SearchableName).HasMaxLength(24);
|
||||||
ent.HasIndex(_alias => _alias.SearchableName);
|
ent.HasIndex(_alias => _alias.SearchableName);
|
||||||
ent.HasIndex(_alias => new {_alias.Name, _alias.IPAddress}).IsUnique();
|
ent.HasIndex(_alias => new {_alias.Name, _alias.IPAddress});
|
||||||
|
ent.Property(alias => alias.SearchableIPAddress)
|
||||||
|
.HasComputedColumnSql(@"((IPAddress & 255) || '.' || ((IPAddress >> 8) & 255)) || '.' || ((IPAddress >> 16) & 255) || '.' || ((IPAddress >> 24) & 255)", stored: true);
|
||||||
|
ent.HasIndex(alias => alias.SearchableIPAddress);
|
||||||
});
|
});
|
||||||
|
|
||||||
modelBuilder.Entity<EFMeta>(ent =>
|
modelBuilder.Entity<EFMeta>(ent =>
|
||||||
@ -130,6 +134,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 +147,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));
|
||||||
|
|
||||||
|
@ -1,30 +1,27 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFramework>netcoreapp3.1</TargetFramework>
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
<Configurations>Debug;Release;Prerelease</Configurations>
|
<Configurations>Debug;Release;Prerelease</Configurations>
|
||||||
<Platforms>AnyCPU</Platforms>
|
<Platforms>AnyCPU</Platforms>
|
||||||
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
|
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
|
||||||
<PackageId>RaidMax.IW4MAdmin.Data</PackageId>
|
<PackageId>RaidMax.IW4MAdmin.Data</PackageId>
|
||||||
<Title>RaidMax.IW4MAdmin.Data</Title>
|
<Title>RaidMax.IW4MAdmin.Data</Title>
|
||||||
<Authors />
|
<Authors />
|
||||||
<PackageVersion>1.0.7</PackageVersion>
|
<PackageVersion>1.2.0</PackageVersion>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="3.1.10" />
|
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="6.0.1" />
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Tools" Version="3.1.10">
|
<PackageReference Include="Microsoft.EntityFrameworkCore.InMemory" Version="6.0.1" />
|
||||||
|
<PackageReference Include="Microsoft.EntityFrameworkCore.Tools" Version="6.0.1">
|
||||||
<PrivateAssets>all</PrivateAssets>
|
<PrivateAssets>all</PrivateAssets>
|
||||||
<IncludeAssets>runtime; build; native; contentfiles</IncludeAssets>
|
<IncludeAssets>runtime; build; native; contentfiles</IncludeAssets>
|
||||||
</PackageReference>
|
</PackageReference>
|
||||||
<PackageReference Include="Npgsql" Version="4.1.7" />
|
<PackageReference Include="Npgsql" Version="6.0.2" />
|
||||||
<PackageReference Include="Npgsql.EntityFrameworkCore.PostgreSQL" Version="3.1.4" />
|
<PackageReference Include="Npgsql.EntityFrameworkCore.PostgreSQL" Version="6.0.2" />
|
||||||
<PackageReference Include="Pomelo.EntityFrameworkCore.MySql" Version="3.2.4" />
|
<PackageReference Include="Pomelo.EntityFrameworkCore.MySql" Version="6.0.1" />
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Sqlite" Version="3.1.10" />
|
<PackageReference Include="Microsoft.EntityFrameworkCore.Sqlite" Version="6.0.1" />
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<ItemGroup Condition="'$(Configuration)'=='Debug'">
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.InMemory" Version="3.1.10" />
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
@ -1,58 +1,84 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Concurrent;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Data.Abstractions;
|
using Data.Abstractions;
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
using Timer = System.Timers.Timer;
|
||||||
|
|
||||||
namespace Data.Helpers
|
namespace Data.Helpers
|
||||||
{
|
{
|
||||||
public class DataValueCache<T, V> : IDataValueCache<T, V> where T : class
|
public class DataValueCache<TEntityType, TReturnType> : IDataValueCache<TEntityType, TReturnType>
|
||||||
|
where TEntityType : class
|
||||||
{
|
{
|
||||||
private readonly ILogger _logger;
|
private readonly ILogger _logger;
|
||||||
private readonly IDatabaseContextFactory _contextFactory;
|
private readonly IDatabaseContextFactory _contextFactory;
|
||||||
private readonly Dictionary<string, CacheState> _cacheStates = new Dictionary<string, CacheState>();
|
|
||||||
private const int DefaultExpireMinutes = 15;
|
|
||||||
|
|
||||||
private class CacheState
|
private readonly ConcurrentDictionary<string, CacheState<TReturnType>> _cacheStates =
|
||||||
|
new ConcurrentDictionary<string, CacheState<TReturnType>>();
|
||||||
|
|
||||||
|
private bool _autoRefresh;
|
||||||
|
private const int DefaultExpireMinutes = 15;
|
||||||
|
private Timer _timer;
|
||||||
|
|
||||||
|
private class CacheState<TCacheType>
|
||||||
{
|
{
|
||||||
public string Key { get; set; }
|
public string Key { get; set; }
|
||||||
public DateTime LastRetrieval { get; set; }
|
public DateTime LastRetrieval { get; set; }
|
||||||
public TimeSpan ExpirationTime { get; set; }
|
public TimeSpan ExpirationTime { get; set; }
|
||||||
public Func<DbSet<T>, CancellationToken, Task<V>> Getter { get; set; }
|
public Func<DbSet<TEntityType>, CancellationToken, Task<TCacheType>> Getter { get; set; }
|
||||||
public V Value { get; set; }
|
public TCacheType Value { get; set; }
|
||||||
|
public bool IsSet { get; set; }
|
||||||
|
|
||||||
public bool IsExpired => ExpirationTime != TimeSpan.MaxValue &&
|
public bool IsExpired => ExpirationTime != TimeSpan.MaxValue &&
|
||||||
(DateTime.Now - LastRetrieval.Add(ExpirationTime)).TotalSeconds > 0;
|
(DateTime.Now - LastRetrieval.Add(ExpirationTime)).TotalSeconds > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public DataValueCache(ILogger<DataValueCache<T, V>> logger, IDatabaseContextFactory contextFactory)
|
public DataValueCache(ILogger<DataValueCache<TEntityType, TReturnType>> logger,
|
||||||
|
IDatabaseContextFactory contextFactory)
|
||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_contextFactory = contextFactory;
|
_contextFactory = contextFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SetCacheItem(Func<DbSet<T>, CancellationToken, Task<V>> getter, string key,
|
~DataValueCache()
|
||||||
TimeSpan? expirationTime = null)
|
{
|
||||||
|
_timer?.Stop();
|
||||||
|
_timer?.Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetCacheItem(Func<DbSet<TEntityType>, CancellationToken, Task<TReturnType>> getter, string key,
|
||||||
|
TimeSpan? expirationTime = null, bool autoRefresh = false)
|
||||||
{
|
{
|
||||||
if (_cacheStates.ContainsKey(key))
|
if (_cacheStates.ContainsKey(key))
|
||||||
{
|
{
|
||||||
_logger.LogDebug("Cache key {key} is already added", key);
|
_logger.LogDebug("Cache key {Key} is already added", key);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var state = new CacheState()
|
var state = new CacheState<TReturnType>
|
||||||
{
|
{
|
||||||
Key = key,
|
Key = key,
|
||||||
Getter = getter,
|
Getter = getter,
|
||||||
ExpirationTime = expirationTime ?? TimeSpan.FromMinutes(DefaultExpireMinutes)
|
ExpirationTime = expirationTime ?? TimeSpan.FromMinutes(DefaultExpireMinutes)
|
||||||
};
|
};
|
||||||
|
|
||||||
_cacheStates.Add(key, state);
|
_autoRefresh = autoRefresh;
|
||||||
|
|
||||||
|
_cacheStates.TryAdd(key, state);
|
||||||
|
|
||||||
|
if (!_autoRefresh || expirationTime == TimeSpan.MaxValue)
|
||||||
|
{
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<V> GetCacheItem(string keyName, CancellationToken cancellationToken = default)
|
_timer = new Timer(state.ExpirationTime.TotalMilliseconds);
|
||||||
|
_timer.Elapsed += async (sender, args) => await RunCacheUpdate(state, CancellationToken.None);
|
||||||
|
_timer.Start();
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<TReturnType> GetCacheItem(string keyName, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
if (!_cacheStates.ContainsKey(keyName))
|
if (!_cacheStates.ContainsKey(keyName))
|
||||||
{
|
{
|
||||||
@ -61,7 +87,9 @@ namespace Data.Helpers
|
|||||||
|
|
||||||
var state = _cacheStates[keyName];
|
var state = _cacheStates[keyName];
|
||||||
|
|
||||||
if (state.IsExpired || state.Value == null)
|
// when auto refresh is off we want to check the expiration and value
|
||||||
|
// when auto refresh is on, we want to only check the value, because it'll be refreshed automatically
|
||||||
|
if ((state.IsExpired || !state.IsSet) && !_autoRefresh || _autoRefresh && !state.IsSet)
|
||||||
{
|
{
|
||||||
await RunCacheUpdate(state, cancellationToken);
|
await RunCacheUpdate(state, cancellationToken);
|
||||||
}
|
}
|
||||||
@ -69,19 +97,21 @@ namespace Data.Helpers
|
|||||||
return state.Value;
|
return state.Value;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task RunCacheUpdate(CacheState state, CancellationToken token)
|
private async Task RunCacheUpdate(CacheState<TReturnType> state, CancellationToken token)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
_logger.LogDebug("Running update for {ClassName} {@State}", GetType().Name, state);
|
||||||
await using var context = _contextFactory.CreateContext(false);
|
await using var context = _contextFactory.CreateContext(false);
|
||||||
var set = context.Set<T>();
|
var set = context.Set<TEntityType>();
|
||||||
var value = await state.Getter(set, token);
|
var value = await state.Getter(set, token);
|
||||||
state.Value = value;
|
state.Value = value;
|
||||||
|
state.IsSet = true;
|
||||||
state.LastRetrieval = DateTime.Now;
|
state.LastRetrieval = DateTime.Now;
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.LogError(ex, "Could not get cached value for {key}", state.Key);
|
_logger.LogError(ex, "Could not get cached value for {Key}", state.Key);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -102,7 +102,7 @@ namespace Data.Helpers
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await using var context = _contextFactory.CreateContext();
|
await using var context = _contextFactory.CreateContext(false);
|
||||||
_cachedItems = await context.Set<T>().ToDictionaryAsync(item => item.Id);
|
_cachedItems = await context.Set<T>().ToDictionaryAsync(item => item.Id);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
|
@ -24,9 +24,10 @@ namespace Data.MigrationContext
|
|||||||
{
|
{
|
||||||
if (MigrationExtensions.IsMigration)
|
if (MigrationExtensions.IsMigration)
|
||||||
{
|
{
|
||||||
optionsBuilder.UseMySql("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;";
|
||||||
.EnableDetailedErrors(true)
|
optionsBuilder.UseMySql(connectionString, ServerVersion.AutoDetect(connectionString))
|
||||||
.EnableSensitiveDataLogging(true);
|
.EnableDetailedErrors()
|
||||||
|
.EnableSensitiveDataLogging();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -23,9 +23,10 @@ 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("9.4")))
|
options => options.SetPostgresVersion(new Version("12.9")))
|
||||||
.EnableDetailedErrors(true)
|
.EnableDetailedErrors(true)
|
||||||
.EnableSensitiveDataLogging(true);
|
.EnableSensitiveDataLogging(true);
|
||||||
}
|
}
|
||||||
|
1449
Data/Migrations/MySql/20220102221729_RemoveUniqueAliasIndexConstraint.Designer.cs
generated
Normal file
1449
Data/Migrations/MySql/20220102221729_RemoveUniqueAliasIndexConstraint.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,32 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
namespace Data.Migrations.MySql
|
||||||
|
{
|
||||||
|
public partial class RemoveUniqueAliasIndexConstraint : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropIndex(
|
||||||
|
name: "IX_EFAlias_Name_IPAddress",
|
||||||
|
table: "EFAlias");
|
||||||
|
|
||||||
|
migrationBuilder.CreateIndex(
|
||||||
|
name: "IX_EFAlias_Name_IPAddress",
|
||||||
|
table: "EFAlias",
|
||||||
|
columns: new[] { "Name", "IPAddress" });
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropIndex(
|
||||||
|
name: "IX_EFAlias_Name_IPAddress",
|
||||||
|
table: "EFAlias");
|
||||||
|
|
||||||
|
migrationBuilder.CreateIndex(
|
||||||
|
name: "IX_EFAlias_Name_IPAddress",
|
||||||
|
table: "EFAlias",
|
||||||
|
columns: new[] { "Name", "IPAddress" },
|
||||||
|
unique: true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1622
Data/Migrations/MySql/20220222215149_AddEFPenaltyIdentifier.Designer.cs
generated
Normal file
1622
Data/Migrations/MySql/20220222215149_AddEFPenaltyIdentifier.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
250
Data/Migrations/MySql/20220222215149_AddEFPenaltyIdentifier.cs
Normal file
250
Data/Migrations/MySql/20220222215149_AddEFPenaltyIdentifier.cs
Normal 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");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1620
Data/Migrations/MySql/20220222230049_MakeEFPenaltyLinkIdNullable.Designer.cs
generated
Normal file
1620
Data/Migrations/MySql/20220222230049_MakeEFPenaltyLinkIdNullable.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1623
Data/Migrations/MySql/20220223150028_AddAuditFieldsToEFPenaltyIdentifier.Designer.cs
generated
Normal file
1623
Data/Migrations/MySql/20220223150028_AddAuditFieldsToEFPenaltyIdentifier.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1626
Data/Migrations/MySql/20220329213440_AddConnectionInterruptedToEFServerSnapshot.Designer.cs
generated
Normal file
1626
Data/Migrations/MySql/20220329213440_AddConnectionInterruptedToEFServerSnapshot.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,25 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace Data.Migrations.MySql
|
||||||
|
{
|
||||||
|
public partial class AddConnectionInterruptedToEFServerSnapshot : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<bool>(
|
||||||
|
name: "ConnectionInterrupted",
|
||||||
|
table: "EFServerSnapshot",
|
||||||
|
type: "tinyint(1)",
|
||||||
|
nullable: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "ConnectionInterrupted",
|
||||||
|
table: "EFServerSnapshot");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1631
Data/Migrations/MySql/20220404151444_AddSearchableIPToEFAlias.Designer.cs
generated
Normal file
1631
Data/Migrations/MySql/20220404151444_AddSearchableIPToEFAlias.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,28 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace Data.Migrations.MySql
|
||||||
|
{
|
||||||
|
public partial class AddSearchableIPToEFAlias : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<string>(
|
||||||
|
name: "SearchableIPAddress",
|
||||||
|
table: "EFAlias",
|
||||||
|
type: "longtext",
|
||||||
|
nullable: true,
|
||||||
|
computedColumnSql: "CONCAT((IPAddress & 255), \".\", ((IPAddress >> 8) & 255), \".\", ((IPAddress >> 16) & 255), \".\", ((IPAddress >> 24) & 255))",
|
||||||
|
stored: true)
|
||||||
|
.Annotation("MySql:CharSet", "utf8mb4");
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "SearchableIPAddress",
|
||||||
|
table: "EFAlias");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1633
Data/Migrations/MySql/20220404192417_AddIndexToSearchableIPToEFAlias.Designer.cs
generated
Normal file
1633
Data/Migrations/MySql/20220404192417_AddIndexToSearchableIPToEFAlias.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,24 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace Data.Migrations.MySql
|
||||||
|
{
|
||||||
|
public partial class AddIndexToSearchableIPToEFAlias : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.CreateIndex(
|
||||||
|
name: "IX_EFAlias_SearchableIPAddress",
|
||||||
|
table: "EFAlias",
|
||||||
|
column: "SearchableIPAddress");
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropIndex(
|
||||||
|
name: "IX_EFAlias_SearchableIPAddress",
|
||||||
|
table: "EFAlias");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1636
Data/Migrations/MySql/20220422202702_AddGameToEFClient.Designer.cs
generated
Normal file
1636
Data/Migrations/MySql/20220422202702_AddGameToEFClient.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
25
Data/Migrations/MySql/20220422202702_AddGameToEFClient.cs
Normal file
25
Data/Migrations/MySql/20220422202702_AddGameToEFClient.cs
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace Data.Migrations.MySql
|
||||||
|
{
|
||||||
|
public partial class AddGameToEFClient : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<int>(
|
||||||
|
name: "GameName",
|
||||||
|
table: "EFClients",
|
||||||
|
type: "int",
|
||||||
|
nullable: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "GameName",
|
||||||
|
table: "EFClients");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -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 =>
|
||||||
@ -62,6 +64,9 @@ namespace Data.Migrations.MySql
|
|||||||
b.Property<DateTime>("FirstConnection")
|
b.Property<DateTime>("FirstConnection")
|
||||||
.HasColumnType("datetime(6)");
|
.HasColumnType("datetime(6)");
|
||||||
|
|
||||||
|
b.Property<int?>("GameName")
|
||||||
|
.HasColumnType("int");
|
||||||
|
|
||||||
b.Property<DateTime>("LastConnection")
|
b.Property<DateTime>("LastConnection")
|
||||||
.HasColumnType("datetime(6)");
|
.HasColumnType("datetime(6)");
|
||||||
|
|
||||||
@ -75,10 +80,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 +97,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 +129,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 +184,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 +203,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 +219,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 +238,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 +278,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 +326,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 +345,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 +419,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 +464,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 +483,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 +541,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 +554,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 +579,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 +622,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 +639,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 +648,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 +665,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 +689,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 +713,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 +722,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 +739,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 +781,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 +804,17 @@ 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>("SearchableIPAddress")
|
||||||
|
.ValueGeneratedOnAddOrUpdate()
|
||||||
|
.HasColumnType("varchar(255)")
|
||||||
|
.HasComputedColumnSql("((IPAddress & 255) || '.' || ((IPAddress >> 8) & 255)) || '.' || ((IPAddress >> 16) & 255) || '.' || ((IPAddress >> 24) & 255)", true);
|
||||||
|
|
||||||
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");
|
||||||
|
|
||||||
@ -814,12 +824,13 @@ namespace Data.Migrations.MySql
|
|||||||
|
|
||||||
b.HasIndex("Name");
|
b.HasIndex("Name");
|
||||||
|
|
||||||
|
b.HasIndex("SearchableIPAddress");
|
||||||
|
|
||||||
b.HasIndex("SearchableName");
|
b.HasIndex("SearchableName");
|
||||||
|
|
||||||
b.HasIndex("Name", "IPAddress")
|
b.HasIndex("Name", "IPAddress");
|
||||||
.IsUnique();
|
|
||||||
|
|
||||||
b.ToTable("EFAlias");
|
b.ToTable("EFAlias", (string)null);
|
||||||
});
|
});
|
||||||
|
|
||||||
modelBuilder.Entity("Data.Models.EFAliasLink", b =>
|
modelBuilder.Entity("Data.Models.EFAliasLink", b =>
|
||||||
@ -833,7 +844,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 =>
|
||||||
@ -846,11 +857,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");
|
||||||
@ -859,7 +870,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");
|
||||||
@ -891,12 +902,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");
|
||||||
@ -906,7 +917,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");
|
||||||
|
|
||||||
@ -929,7 +940,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)");
|
||||||
@ -937,7 +948,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")
|
||||||
@ -945,7 +956,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");
|
||||||
@ -964,7 +975,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 =>
|
||||||
@ -983,7 +1026,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");
|
||||||
@ -1014,13 +1057,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)");
|
||||||
@ -1030,7 +1073,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 =>
|
||||||
@ -1048,6 +1091,9 @@ namespace Data.Migrations.MySql
|
|||||||
b.Property<int>("ClientCount")
|
b.Property<int>("ClientCount")
|
||||||
.HasColumnType("int");
|
.HasColumnType("int");
|
||||||
|
|
||||||
|
b.Property<bool?>("ConnectionInterrupted")
|
||||||
|
.HasColumnType("tinyint(1)");
|
||||||
|
|
||||||
b.Property<int>("MapId")
|
b.Property<int>("MapId")
|
||||||
.HasColumnType("int");
|
.HasColumnType("int");
|
||||||
|
|
||||||
@ -1063,7 +1109,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 =>
|
||||||
@ -1088,7 +1134,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 =>
|
||||||
@ -1108,7 +1154,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 =>
|
||||||
@ -1124,6 +1170,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 =>
|
||||||
@ -1139,6 +1189,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 =>
|
||||||
@ -1154,6 +1208,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 =>
|
||||||
@ -1187,6 +1245,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 =>
|
||||||
@ -1202,6 +1272,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 =>
|
||||||
@ -1239,6 +1313,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 =>
|
||||||
@ -1268,6 +1354,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 =>
|
||||||
@ -1281,6 +1379,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 =>
|
||||||
@ -1290,6 +1392,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 =>
|
||||||
@ -1305,6 +1409,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 =>
|
||||||
@ -1326,6 +1434,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 =>
|
||||||
@ -1339,6 +1451,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 =>
|
||||||
@ -1356,6 +1472,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 =>
|
||||||
@ -1365,6 +1487,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 =>
|
||||||
@ -1377,15 +1501,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")
|
||||||
@ -1398,6 +1524,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 =>
|
||||||
@ -1417,6 +1560,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 =>
|
||||||
@ -1432,6 +1581,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 =>
|
||||||
@ -1441,6 +1594,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
|
||||||
}
|
}
|
||||||
|
1477
Data/Migrations/Postgresql/20220102221801_RemoveUniqueAliasIndexConstraint.Designer.cs
generated
Normal file
1477
Data/Migrations/Postgresql/20220102221801_RemoveUniqueAliasIndexConstraint.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,32 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
namespace Data.Migrations.Postgresql
|
||||||
|
{
|
||||||
|
public partial class RemoveUniqueAliasIndexConstraint : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropIndex(
|
||||||
|
name: "IX_EFAlias_Name_IPAddress",
|
||||||
|
table: "EFAlias");
|
||||||
|
|
||||||
|
migrationBuilder.CreateIndex(
|
||||||
|
name: "IX_EFAlias_Name_IPAddress",
|
||||||
|
table: "EFAlias",
|
||||||
|
columns: new[] { "Name", "IPAddress" });
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropIndex(
|
||||||
|
name: "IX_EFAlias_Name_IPAddress",
|
||||||
|
table: "EFAlias");
|
||||||
|
|
||||||
|
migrationBuilder.CreateIndex(
|
||||||
|
name: "IX_EFAlias_Name_IPAddress",
|
||||||
|
table: "EFAlias",
|
||||||
|
columns: new[] { "Name", "IPAddress" },
|
||||||
|
unique: true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1679
Data/Migrations/Postgresql/20220222222045_AddEFPenaltyIdentifier.Designer.cs
generated
Normal file
1679
Data/Migrations/Postgresql/20220222222045_AddEFPenaltyIdentifier.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -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");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1677
Data/Migrations/Postgresql/20220222230153_MakeEFPenaltyLinkIdNullable.Designer.cs
generated
Normal file
1677
Data/Migrations/Postgresql/20220222230153_MakeEFPenaltyLinkIdNullable.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1680
Data/Migrations/Postgresql/20220223150106_AddAuditFieldsToEFPenaltyIdentifier.Designer.cs
generated
Normal file
1680
Data/Migrations/Postgresql/20220223150106_AddAuditFieldsToEFPenaltyIdentifier.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1683
Data/Migrations/Postgresql/20220329213521_AddConnectionInterruptedToEFServerSnapshot.Designer.cs
generated
Normal file
1683
Data/Migrations/Postgresql/20220329213521_AddConnectionInterruptedToEFServerSnapshot.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,25 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace Data.Migrations.Postgresql
|
||||||
|
{
|
||||||
|
public partial class AddConnectionInterruptedToEFServerSnapshot : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<bool>(
|
||||||
|
name: "ConnectionInterrupted",
|
||||||
|
table: "EFServerSnapshot",
|
||||||
|
type: "boolean",
|
||||||
|
nullable: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "ConnectionInterrupted",
|
||||||
|
table: "EFServerSnapshot");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1688
Data/Migrations/Postgresql/20220404185627_AddSearchableIPToEFAlias.Designer.cs
generated
Normal file
1688
Data/Migrations/Postgresql/20220404185627_AddSearchableIPToEFAlias.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user