Skip to content

Commit

Permalink
update to game interface/integration for persistent stat data
Browse files Browse the repository at this point in the history
  • Loading branch information
RaidMax committed Jul 13, 2022
1 parent 28d56da commit eaac12f
Show file tree
Hide file tree
Showing 8 changed files with 102 additions and 13 deletions.
19 changes: 18 additions & 1 deletion Application/IW4MServer.cs
Original file line number Diff line number Diff line change
Expand Up @@ -373,7 +373,7 @@ protected override async Task<bool> ProcessEvent(GameEvent E)
var clientTag = await _metaService.GetPersistentMetaByLookup(EFMeta.ClientTagV2,
EFMeta.ClientTagNameV2, E.Origin.ClientId, Manager.CancellationToken);

if (clientTag.Value != null)
if (clientTag?.Value != null)
{
E.Origin.Tag = clientTag.Value;
}
Expand Down Expand Up @@ -768,6 +768,23 @@ await Manager.GetPenaltyService().RemoveActivePenalties(E.Target.AliasLinkId, E.
{
E.Origin.UpdateTeam(E.Extra as string);
}

else if (E.Type == GameEvent.EventType.MetaUpdated)
{
if (E.Extra is "PersistentStatClientId" && int.TryParse(E.Data, out var persistentClientId))
{
var penalties = await Manager.GetPenaltyService().GetActivePenaltiesByClientId(persistentClientId);
var banPenalty = penalties.FirstOrDefault(penalty => penalty.Type == EFPenalty.PenaltyType.Ban);

if (banPenalty is not null && E.Origin.Level != Permission.Banned)
{
ServerLogger.LogInformation(
"Banning {Client} as they have have provided a persistent clientId of {PersistentClientId}, which is banned",
E.Origin.ToString(), persistentClientId);
E.Origin.Ban(loc["SERVER_BAN_EVADE"].FormatExt(persistentClientId), Utilities.IW4MAdminClient(this), true);
}
}
}

lock (ChatHistory)
{
Expand Down
6 changes: 6 additions & 0 deletions Application/Misc/EventPublisher.cs
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ public class EventPublisher : IEventPublisher
{
public event EventHandler<GameEvent> OnClientDisconnect;
public event EventHandler<GameEvent> OnClientConnect;
public event EventHandler<GameEvent> OnClientMetaUpdated;

private readonly ILogger _logger;

Expand All @@ -33,6 +34,11 @@ public void Publish(GameEvent gameEvent)
{
OnClientDisconnect?.Invoke(this, gameEvent);
}

if (gameEvent.Type == GameEvent.EventType.MetaUpdated)
{
OnClientMetaUpdated?.Invoke(this, gameEvent);
}
}

catch (Exception ex)
Expand Down
27 changes: 26 additions & 1 deletion Application/Misc/MetaServiceV2.cs
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,10 @@
using Data.Abstractions;
using Data.Models;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
using SharedLibraryCore;
using SharedLibraryCore.Database.Models;
using SharedLibraryCore.Dtos;
using SharedLibraryCore.Interfaces;
using SharedLibraryCore.QueryHelper;
Expand All @@ -19,13 +22,15 @@ public class MetaServiceV2 : IMetaServiceV2
{
private readonly IDictionary<MetaType, List<dynamic>> _metaActions;
private readonly IDatabaseContextFactory _contextFactory;
private readonly IServiceProvider _serviceProvider;
private readonly ILogger _logger;

public MetaServiceV2(ILogger<MetaServiceV2> logger, IDatabaseContextFactory contextFactory)
public MetaServiceV2(ILogger<MetaServiceV2> logger, IDatabaseContextFactory contextFactory, IServiceProvider serviceProvider)
{
_logger = logger;
_metaActions = new Dictionary<MetaType, List<dynamic>>();
_contextFactory = contextFactory;
_serviceProvider = serviceProvider;
}

public async Task SetPersistentMeta(string metaKey, string metaValue, int clientId,
Expand Down Expand Up @@ -64,6 +69,26 @@ public async Task SetPersistentMeta(string metaKey, string metaValue, int client
}

await context.SaveChangesAsync(token);


var manager = _serviceProvider.GetRequiredService<IManager>();
var matchingClient = manager.GetActiveClients().FirstOrDefault(client => client.ClientId == clientId);
var server = matchingClient?.CurrentServer ?? manager.GetServers().FirstOrDefault();

if (server is not null)
{
manager.AddEvent(new GameEvent
{
Type = GameEvent.EventType.MetaUpdated,
Origin = matchingClient ?? new EFClient
{
ClientId = clientId
},
Data = metaValue,
Extra = metaKey,
Owner = server
});
}
}

public async Task SetPersistentMetaValue<T>(string metaKey, T metaValue, int clientId,
Expand Down
3 changes: 2 additions & 1 deletion Application/Misc/ScriptPlugin.cs
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,8 @@ public async Task Initialize(IManager manager, IScriptCommandFactory scriptComma
typeof(System.Net.Http.HttpClient).Assembly,
typeof(EFClient).Assembly,
typeof(Utilities).Assembly,
typeof(Encoding).Assembly
typeof(Encoding).Assembly,
typeof(CancellationTokenSource).Assembly
})
.CatchClrExceptions()
.AddObjectConverter(new PermissionLevelToStringConverter()));
Expand Down
25 changes: 23 additions & 2 deletions GameFiles/_integration.gsc
Original file line number Diff line number Diff line change
Expand Up @@ -53,8 +53,6 @@ init()
level thread OnPlayerConnect();
}



//////////////////////////////////
// Client Methods
//////////////////////////////////
Expand Down Expand Up @@ -167,6 +165,28 @@ DisplayWelcomeData()
self IPrintLnBold( "You were last seen ^5" + clientData.lastConnection );
}

SetPersistentData()
{
storedClientId = self GetPlayerData( "bests", "none" );

if ( storedClientId != 0 )
{
if ( level.iw4adminIntegrationDebug == 1 )
{
IPrintLn( "Uploading persistent client id " + storedClientId );
}

SetClientMeta( "PersistentStatClientId", storedClientId );
}

if ( level.iw4adminIntegrationDebug == 1 )
{
IPrintLn( "Persisting client id " + self.persistentClientId );
}

self SetPlayerData( "bests", "none", int( self.persistentClientId ) );
}

PlayerConnectEvents()
{
self endon( "disconnect" );
Expand Down Expand Up @@ -643,6 +663,7 @@ OnClientDataReceived( event )
self.persistentClientId = event.data["clientId"];

self thread DisplayWelcomeData();
self setPersistentData();
}

OnExecuteCommand( event )
Expand Down
20 changes: 12 additions & 8 deletions Plugins/ScriptPlugins/GameInterface.js
Original file line number Diff line number Diff line change
Expand Up @@ -463,7 +463,11 @@ function onReceivedDvar(server, dvarName, dvarValue, success) {
if (input.length > 0) {
const event = parseEvent(input)

logger.WriteDebug(`Processing input... ${event.eventType} ${event.subType} ${event.data} ${event.clientNumber}`);
logger.WriteDebug(`Processing input... ${event.eventType} ${event.subType} ${event.data.toString()} ${event.clientNumber}`);

const metaService = _serviceResolver.ResolveService('IMetaServiceV2');
const threading = importNamespace('System.Threading');
const token = new threading.CancellationTokenSource().Token;

// todo: refactor to mapping if possible
if (event.eventType === 'ClientDataRequested') {
Expand All @@ -475,8 +479,8 @@ function onReceivedDvar(server, dvarName, dvarValue, success) {
let data = [];

if (event.subType === 'Meta') {
const metaService = _serviceResolver.ResolveService('IMetaService');
const meta = metaService.GetPersistentMeta(event.data, client).GetAwaiter().GetResult();
const metaService = _serviceResolver.ResolveService('IMetaServiceV2');
const meta = metaService.GetPersistentMeta(event.data, client, token).GetAwaiter().GetResult();
data[event.data] = meta === null ? '' : meta.Value;
} else {
data = {
Expand Down Expand Up @@ -510,19 +514,19 @@ function onReceivedDvar(server, dvarName, dvarValue, success) {
sendEvent(server, false, 'SetClientDataCompleted', 'Meta', {ClientNumber: event.clientNumber}, undefined, {status: 'Fail'});
} else {
if (event.subType === 'Meta') {
const metaService = _serviceResolver.ResolveService('IMetaService');
try {
logger.WriteDebug(`Key=${event.data['key']}, Value=${event.data['value']}`);
logger.WriteDebug(`Key=${event.data['key']}, Value=${event.data['value']}, Direction=${event.data['direction']} ${token}`);
if (event.data['direction'] != null) {
event.data['direction'] = 'up'
? metaService.IncrementPersistentMeta(event.data['key'], event.data['value'], clientId).GetAwaiter().GetResult()
: metaService.DecrementPersistentMeta(event.data['key'], event.data['value'], clientId).GetAwaiter().GetResult();
? metaService.IncrementPersistentMeta(event.data['key'], event.data['value'], clientId, token).GetAwaiter().GetResult()
: metaService.DecrementPersistentMeta(event.data['key'], event.data['value'], clientId, token).GetAwaiter().GetResult();
} else {
metaService.SetPersistentMeta(event.data['key'], event.data['value'], clientId).GetAwaiter().GetResult();
metaService.SetPersistentMeta(event.data['key'], event.data['value'], clientId, token).GetAwaiter().GetResult();
}
sendEvent(server, false, 'SetClientDataCompleted', 'Meta', {ClientNumber: event.clientNumber}, undefined, {status: 'Complete'});
} catch (error) {
sendEvent(server, false, 'SetClientDataCompleted', 'Meta', {ClientNumber: event.clientNumber}, undefined, {status: 'Fail'});
logger.WriteError('Could not persist client meta ' + error.toString());
}
}
}
Expand Down
5 changes: 5 additions & 0 deletions SharedLibraryCore/Events/GameEvent.cs
Original file line number Diff line number Diff line change
Expand Up @@ -204,6 +204,11 @@ public enum EventType
/// client logged out of webfront
/// </summary>
Logout = 113,

/// <summary>
/// meta value updated on client
/// </summary>
MetaUpdated = 114,

// events "generated" by IW4MAdmin
/// <summary>
Expand Down
10 changes: 10 additions & 0 deletions SharedLibraryCore/Services/PenaltyService.cs
Original file line number Diff line number Diff line change
Expand Up @@ -193,6 +193,16 @@ public async Task<List<EFPenalty>> GetActivePenaltiesByIdentifier(int? ip, long
return await activePenaltiesIds.Select(ids => ids.Penalty).ToListAsync();
}

public async Task<List<EFPenalty>> GetActivePenaltiesByClientId(int clientId)
{
await using var context = _contextFactory.CreateContext(false);
return await context.PenaltyIdentifiers
.Where(identifier => identifier.Penalty.Offender.ClientId == clientId)
.Select(identifier => identifier.Penalty)
.Where(Filter)
.ToListAsync();
}

public async Task<List<EFPenalty>> ActivePenaltiesByRecentIdentifiers(int linkId)
{
await using var context = _contextFactory.CreateContext(false);
Expand Down

0 comments on commit eaac12f

Please sign in to comment.