think I finished reworking the event system

added http log reading support for debugging remotely
started working on unit test framework
This commit is contained in:
RaidMax 2018-08-28 16:32:59 -05:00
parent 56cb8c50e7
commit bbefd53db4
23 changed files with 543 additions and 296 deletions

View File

@ -140,26 +140,26 @@ namespace IW4MAdmin.Application.EventParsers
} }
} }
if (eventType == "Q") //if (eventType == "Q")
{ //{
var regexMatch = Regex.Match(logLine, @"^(Q;)(.{1,32});([0-9]+);(.*)$"); // var regexMatch = Regex.Match(logLine, @"^(Q;)(.{1,32});([0-9]+);(.*)$");
if (regexMatch.Success) // if (regexMatch.Success)
{ // {
return new GameEvent() // return new GameEvent()
{ // {
Type = GameEvent.EventType.Quit, // Type = GameEvent.EventType.Quit,
Data = logLine, // Data = logLine,
Owner = server, // Owner = server,
Origin = new Player() // Origin = new Player()
{ // {
Name = regexMatch.Groups[4].ToString().StripColors(), // Name = regexMatch.Groups[4].ToString().StripColors(),
NetworkId = regexMatch.Groups[2].ToString().ConvertLong(), // NetworkId = regexMatch.Groups[2].ToString().ConvertLong(),
ClientNumber = Convert.ToInt32(regexMatch.Groups[3].ToString()), // ClientNumber = Convert.ToInt32(regexMatch.Groups[3].ToString()),
State = Player.ClientState.Connecting // State = Player.ClientState.Connecting
} // }
}; // };
} // }
} //}
if (eventType.Contains("ExitLevel")) if (eventType.Contains("ExitLevel"))
{ {

View File

@ -6,11 +6,11 @@ using System.Threading.Tasks;
namespace IW4MAdmin.Application.IO namespace IW4MAdmin.Application.IO
{ {
class GameLogEvent class GameLogEventDetection
{ {
Server Server; Server Server;
long PreviousFileSize; long PreviousFileSize;
GameLogReader Reader; IGameLogReader Reader;
readonly string GameLogFile; readonly string GameLogFile;
class EventState class EventState
@ -19,14 +19,22 @@ namespace IW4MAdmin.Application.IO
public string ServerId { get; set; } public string ServerId { get; set; }
} }
public GameLogEvent(Server server, string gameLogPath, string gameLogName) public GameLogEventDetection(Server server, string gameLogPath, string gameLogName)
{ {
GameLogFile = gameLogPath; GameLogFile = gameLogPath;
Reader = new GameLogReader(gameLogPath, server.EventParser); // todo: abtract this more
if (gameLogPath.StartsWith("http"))
{
Reader = new GameLogReaderHttp(gameLogPath, server.EventParser);
}
else
{
Reader = new GameLogReader(gameLogPath, server.EventParser);
}
Server = server; Server = server;
Task.Run(async () => Task.Run(async () =>
{ {
while (!server.Manager.ShutdownRequested()) while (!server.Manager.ShutdownRequested())
{ {
if ((server.Manager as ApplicationManager).IsInitialized) if ((server.Manager as ApplicationManager).IsInitialized)
@ -44,7 +52,7 @@ namespace IW4MAdmin.Application.IO
private void OnEvent(object state) private void OnEvent(object state)
{ {
long newLength = new FileInfo(GameLogFile).Length; long newLength = Reader.Length;
try try
{ {

View File

@ -7,11 +7,15 @@ using System.Text;
namespace IW4MAdmin.Application.IO namespace IW4MAdmin.Application.IO
{ {
class GameLogReader class GameLogReader : IGameLogReader
{ {
IEventParser Parser; IEventParser Parser;
readonly string LogFile; readonly string LogFile;
public long Length => new FileInfo(LogFile).Length;
public int UpdateInterval => 100;
public GameLogReader(string logFile, IEventParser parser) public GameLogReader(string logFile, IEventParser parser)
{ {
LogFile = logFile; LogFile = logFile;

View File

@ -0,0 +1,84 @@
using SharedLibraryCore;
using SharedLibraryCore.Interfaces;
using System;
using System.Collections.Generic;
using System.IO;
using System.Net;
using System.Net.Http;
using System.Text;
namespace IW4MAdmin.Application.IO
{
/// <summary>
/// provides capibility of reading log files over HTTP
/// </summary>
class GameLogReaderHttp : IGameLogReader
{
readonly IEventParser Parser;
readonly string LogFile;
public GameLogReaderHttp(string logFile, IEventParser parser)
{
LogFile = logFile;
Parser = parser;
}
public long Length
{
get
{
using (var cl = new HttpClient())
{
using (var re = cl.GetAsync($"{LogFile}?length=1").Result)
{
using (var content = re.Content)
{
return Convert.ToInt64(content.ReadAsStringAsync().Result ?? "0");
}
}
}
}
}
public int UpdateInterval => 1000;
public ICollection<GameEvent> EventsFromLog(Server server, long fileSizeDiff, long startPosition)
{
string log;
using (var cl = new HttpClient())
{
using (var re = cl.GetAsync($"{LogFile}?start={fileSizeDiff}").Result)
{
using (var content = re.Content)
{
log = content.ReadAsStringAsync().Result;
}
}
}
List<GameEvent> events = new List<GameEvent>();
// parse each line
foreach (string eventLine in log.Split(Environment.NewLine))
{
if (eventLine.Length > 0)
{
try
{
// todo: catch elsewhere
events.Add(Parser.GetEvent(server, eventLine));
}
catch (Exception e)
{
Program.ServerManager.GetLogger().WriteWarning("Could not properly parse event line");
Program.ServerManager.GetLogger().WriteDebug(e.Message);
Program.ServerManager.GetLogger().WriteDebug(eventLine);
}
}
}
return events;
}
}
}

View File

@ -50,9 +50,6 @@ namespace IW4MAdmin.Application
Localization.Configure.Initialize(ServerManager.GetApplicationSettings().Configuration()?.CustomLocale); Localization.Configure.Initialize(ServerManager.GetApplicationSettings().Configuration()?.CustomLocale);
loc = Utilities.CurrentLocalization.LocalizationIndex; loc = Utilities.CurrentLocalization.LocalizationIndex;
using (var db = new DatabaseContext(ServerManager.GetApplicationSettings().Configuration()?.ConnectionString))
new ContextSeed(db).Seed().Wait();
var api = API.Master.Endpoint.Get(); var api = API.Master.Endpoint.Get();
var version = new API.Master.VersionInfo() var version = new API.Master.VersionInfo()

View File

@ -21,6 +21,7 @@ using Newtonsoft.Json.Linq;
using System.Text; using System.Text;
using IW4MAdmin.Application.API.Master; using IW4MAdmin.Application.API.Master;
using System.Reflection; using System.Reflection;
using SharedLibraryCore.Database;
namespace IW4MAdmin.Application namespace IW4MAdmin.Application
{ {
@ -36,7 +37,7 @@ namespace IW4MAdmin.Application
// define what the delagate function looks like // define what the delagate function looks like
public delegate void OnServerEventEventHandler(object sender, GameEventArgs e); public delegate void OnServerEventEventHandler(object sender, GameEventArgs e);
// expose the event handler so we can execute the events // expose the event handler so we can execute the events
public OnServerEventEventHandler OnServerEvent { get; private set; } public OnServerEventEventHandler OnServerEvent { get; set; }
public DateTime StartTime { get; private set; } public DateTime StartTime { get; private set; }
static ApplicationManager Instance; static ApplicationManager Instance;
@ -46,10 +47,10 @@ namespace IW4MAdmin.Application
ClientService ClientSvc; ClientService ClientSvc;
readonly AliasService AliasSvc; readonly AliasService AliasSvc;
readonly PenaltyService PenaltySvc; readonly PenaltyService PenaltySvc;
BaseConfigurationHandler<ApplicationConfiguration> ConfigHandler; public BaseConfigurationHandler<ApplicationConfiguration> ConfigHandler;
EventApi Api; EventApi Api;
GameEventHandler Handler; GameEventHandler Handler;
ManualResetEventSlim OnEvent; ManualResetEventSlim OnQuit;
readonly IPageList PageList; readonly IPageList PageList;
public class GameEventArgs : System.ComponentModel.AsyncCompletedEventArgs public class GameEventArgs : System.ComponentModel.AsyncCompletedEventArgs
@ -78,7 +79,7 @@ namespace IW4MAdmin.Application
//ServerEventOccurred += Api.OnServerEvent; //ServerEventOccurred += Api.OnServerEvent;
ConfigHandler = new BaseConfigurationHandler<ApplicationConfiguration>("IW4MAdminSettings"); ConfigHandler = new BaseConfigurationHandler<ApplicationConfiguration>("IW4MAdminSettings");
StartTime = DateTime.UtcNow; StartTime = DateTime.UtcNow;
OnEvent = new ManualResetEventSlim(); OnQuit = new ManualResetEventSlim();
PageList = new PageList(); PageList = new PageList();
OnServerEvent += OnServerEventAsync; OnServerEvent += OnServerEventAsync;
} }
@ -110,15 +111,16 @@ namespace IW4MAdmin.Application
await newEvent.Owner.ExecuteEvent(newEvent); await newEvent.Owner.ExecuteEvent(newEvent);
//// todo: this is a hacky mess //// todo: this is a hacky mess
if (newEvent.Origin?.DelayedEvents?.Count > 0 && if (newEvent.Origin?.DelayedEvents.Count > 0 &&
newEvent.Origin?.State == Player.ClientState.Connected) newEvent.Origin?.State == Player.ClientState.Connected)
{ {
var events = newEvent.Origin.DelayedEvents; var events = newEvent.Origin.DelayedEvents;
// add the delayed event to the queue // add the delayed event to the queue
while (events?.Count > 0) while(events.Count > 0)
{ {
var e = events.Dequeue(); var e = events.Dequeue();
e.Origin = newEvent.Origin; e.Origin = newEvent.Origin;
// check if the target was assigned // check if the target was assigned
if (e.Target != null) if (e.Target != null)
@ -133,9 +135,12 @@ namespace IW4MAdmin.Application
continue; continue;
} }
} }
Logger.WriteDebug($"Adding delayed event of type {e.Type} for {e.Origin} back for processing");
this.GetEventHandler().AddEvent(e); this.GetEventHandler().AddEvent(e);
} }
} }
Api.OnServerEvent(this, newEvent);
#if DEBUG #if DEBUG
Logger.WriteDebug("Processed Event"); Logger.WriteDebug("Processed Event");
#endif #endif
@ -248,6 +253,11 @@ namespace IW4MAdmin.Application
Running = true; Running = true;
#region DATABASE #region DATABASE
using (var db = new DatabaseContext(GetApplicationSettings().Configuration()?.ConnectionString))
{
await new ContextSeed(db).Seed();
}
var ipList = (await ClientSvc.Find(c => c.Level > Player.Permission.Trusted)) var ipList = (await ClientSvc.Find(c => c.Level > Player.Permission.Trusted))
.Select(c => new .Select(c => new
{ {
@ -513,8 +523,8 @@ namespace IW4MAdmin.Application
while (Running) while (Running)
{ {
OnEvent.Wait(); OnQuit.Wait();
OnEvent.Reset(); OnQuit.Reset();
} }
_servers.Clear(); _servers.Clear();
} }
@ -558,7 +568,7 @@ namespace IW4MAdmin.Application
public void SetHasEvent() public void SetHasEvent()
{ {
OnEvent.Set(); OnQuit.Set();
} }
public IList<Assembly> GetPluginAssemblies() => SharedLibraryCore.Plugins.PluginImporter.PluginAssemblies; public IList<Assembly> GetPluginAssemblies() => SharedLibraryCore.Plugins.PluginImporter.PluginAssemblies;

View File

@ -27,7 +27,7 @@ namespace IW4MAdmin.Application.RconParsers
public async Task<string[]> ExecuteCommandAsync(Connection connection, string command) public async Task<string[]> ExecuteCommandAsync(Connection connection, string command)
{ {
var response = await connection.SendQueryAsync(StaticHelpers.QueryType.COMMAND, command); var response = await connection.SendQueryAsync(StaticHelpers.QueryType.COMMAND, command);
return response.Skip(1).ToArray(); return response.Skip(1).ToArray();
} }
@ -117,7 +117,6 @@ namespace IW4MAdmin.Application.RconParsers
IsBot = ip == 0, IsBot = ip == 0,
State = Player.ClientState.Connecting State = Player.ClientState.Connecting
}; };
StatusPlayers.Add(P); StatusPlayers.Add(P);
} }
} }

View File

@ -10,7 +10,7 @@ using SharedLibraryCore.Objects;
using SharedLibraryCore.RCon; using SharedLibraryCore.RCon;
using SharedLibraryCore.Exceptions; using SharedLibraryCore.Exceptions;
namespace IW4MAdmin.WApplication.RconParsers namespace IW4MAdmin.Application.RconParsers
{ {
public class IW5MRConParser : IRConParser public class IW5MRConParser : IRConParser
{ {

View File

@ -20,14 +20,13 @@ using IW4MAdmin.Application.RconParsers;
using IW4MAdmin.Application.EventParsers; using IW4MAdmin.Application.EventParsers;
using IW4MAdmin.Application.IO; using IW4MAdmin.Application.IO;
using IW4MAdmin.Application.Core; using IW4MAdmin.Application.Core;
using IW4MAdmin.WApplication.RconParsers;
namespace IW4MAdmin namespace IW4MAdmin
{ {
public class IW4MServer : Server public class IW4MServer : Server
{ {
private static readonly Index loc = Utilities.CurrentLocalization.LocalizationIndex; private static readonly Index loc = Utilities.CurrentLocalization.LocalizationIndex;
private GameLogEvent LogEvent; private GameLogEventDetection LogEvent;
private ClientAuthentication AuthQueue; private ClientAuthentication AuthQueue;
public IW4MServer(IManager mgr, ServerConfiguration cfg) : base(mgr, cfg) public IW4MServer(IManager mgr, ServerConfiguration cfg) : base(mgr, cfg)
@ -56,8 +55,11 @@ namespace IW4MAdmin
public async Task OnPlayerJoined(Player logClient) public async Task OnPlayerJoined(Player logClient)
{ {
if (Players[logClient.ClientNumber] == null || var existingClient = Players[logClient.ClientNumber];
Players[logClient.ClientNumber].NetworkId != logClient.NetworkId)
if (existingClient == null ||
(existingClient.NetworkId != logClient.NetworkId &&
existingClient.State != Player.ClientState.Connected))
{ {
Logger.WriteDebug($"Log detected {logClient} joining"); Logger.WriteDebug($"Log detected {logClient} joining");
Players[logClient.ClientNumber] = logClient; Players[logClient.ClientNumber] = logClient;
@ -68,9 +70,8 @@ namespace IW4MAdmin
override public async Task<bool> AddPlayer(Player polledPlayer) override public async Task<bool> AddPlayer(Player polledPlayer)
{ {
//if ((polledPlayer.Ping == 999 && !polledPlayer.IsBot) || if ((polledPlayer.Ping == 999 && !polledPlayer.IsBot) ||
// polledPlayer.Ping < 1 || polledPlayer.Ping < 1 ||
if (
polledPlayer.ClientNumber < 0) polledPlayer.ClientNumber < 0)
{ {
//Logger.WriteDebug($"Skipping client not in connected state {P}"); //Logger.WriteDebug($"Skipping client not in connected state {P}");
@ -78,7 +79,7 @@ namespace IW4MAdmin
} }
// set this when they are waiting for authentication // set this when they are waiting for authentication
if (Players[polledPlayer.ClientNumber] == null && if (Players[polledPlayer.ClientNumber] == null &&
polledPlayer.State == Player.ClientState.Connecting) polledPlayer.State == Player.ClientState.Connecting)
{ {
Players[polledPlayer.ClientNumber] = polledPlayer; Players[polledPlayer.ClientNumber] = polledPlayer;
@ -186,6 +187,8 @@ namespace IW4MAdmin
player.IsBot = polledPlayer.IsBot; player.IsBot = polledPlayer.IsBot;
player.Score = polledPlayer.Score; player.Score = polledPlayer.Score;
player.CurrentServer = this; player.CurrentServer = this;
player.DelayedEvents = (Players[player.ClientNumber]?.DelayedEvents) ?? new Queue<GameEvent>();
Players[player.ClientNumber] = player; Players[player.ClientNumber] = player;
var activePenalties = await Manager.GetPenaltyService().GetActivePenaltiesAsync(player.AliasLinkId, player.IPAddress); var activePenalties = await Manager.GetPenaltyService().GetActivePenaltiesAsync(player.AliasLinkId, player.IPAddress);
@ -278,7 +281,6 @@ namespace IW4MAdmin
public override async Task ExecuteEvent(GameEvent E) public override async Task ExecuteEvent(GameEvent E)
{ {
bool canExecuteCommand = true; bool canExecuteCommand = true;
Manager.GetEventApi().OnServerEvent(this, E);
await ProcessEvent(E); await ProcessEvent(E);
Command C = null; Command C = null;
@ -387,15 +389,16 @@ namespace IW4MAdmin
} }
} }
if (E.Type == GameEvent.EventType.Connect) else if (E.Type == GameEvent.EventType.Connect)
{ {
E.Origin.State = Player.ClientState.Authenticated; E.Origin.State = Player.ClientState.Authenticated;
// add them to the server // add them to the server
if (!await AddPlayer(E.Origin)) if (!await AddPlayer(E.Origin))
{ {
throw new ServerException("Player didn't pass authorization, so we are discontinuing event"); E.Origin.State = Player.ClientState.Connecting;
throw new ServerException("client didn't pass authorization, so we are discontinuing event");
} }
// hack makes the event propgate with the correct info // hack: makes the event propgate with the correct info
E.Origin = Players[E.Origin.ClientNumber]; E.Origin = Players[E.Origin.ClientNumber];
ChatHistory.Add(new ChatInfo() ChatHistory.Add(new ChatInfo()
@ -416,27 +419,27 @@ namespace IW4MAdmin
else if (E.Type == GameEvent.EventType.Quit) else if (E.Type == GameEvent.EventType.Quit)
{ {
var origin = Players.FirstOrDefault(p => p != null && p.NetworkId == E.Origin.NetworkId); //var origin = Players.FirstOrDefault(p => p != null && p.NetworkId == E.Origin.NetworkId);
if (origin != null && //if (origin != null &&
// we only want to forward the event if they are connected. // // we only want to forward the event if they are connected.
origin.State == Player.ClientState.Connected) // origin.State == Player.ClientState.Connected)
{ //{
var e = new GameEvent() // var e = new GameEvent()
{ // {
Type = GameEvent.EventType.Disconnect, // Type = GameEvent.EventType.Disconnect,
Origin = origin, // Origin = origin,
Owner = this // Owner = this
}; // };
Manager.GetEventHandler().AddEvent(e); // Manager.GetEventHandler().AddEvent(e);
} //}
else if (origin != null && //else if (origin != null &&
origin.State != Player.ClientState.Connected) // origin.State != Player.ClientState.Connected)
{ //{
await RemovePlayer(origin.ClientNumber); // await RemovePlayer(origin.ClientNumber);
} //}
} }
else if (E.Type == GameEvent.EventType.Disconnect) else if (E.Type == GameEvent.EventType.Disconnect)
@ -448,7 +451,13 @@ namespace IW4MAdmin
Time = DateTime.UtcNow Time = DateTime.UtcNow
}); });
var currentState = E.Origin.State;
await RemovePlayer(E.Origin.ClientNumber); await RemovePlayer(E.Origin.ClientNumber);
if (currentState != Player.ClientState.Connected)
{
throw new ServerException("Disconnecting player was not in a connected state");
}
} }
if (E.Type == GameEvent.EventType.Say) if (E.Type == GameEvent.EventType.Say)
@ -555,7 +564,7 @@ namespace IW4MAdmin
#endif #endif
Throttled = false; Throttled = false;
foreach(var client in polledClients) foreach (var client in polledClients)
{ {
// todo: move out somehwere // todo: move out somehwere
var existingClient = Players[client.ClientNumber] ?? client; var existingClient = Players[client.ClientNumber] ?? client;
@ -564,7 +573,7 @@ namespace IW4MAdmin
} }
var disconnectingClients = currentClients.Except(polledClients); var disconnectingClients = currentClients.Except(polledClients);
var connectingClients = polledClients.Except(currentClients); var connectingClients = polledClients.Except(currentClients.Where(c => c.State == Player.ClientState.Connected));
return new List<Player>[] { connectingClients.ToList(), disconnectingClients.ToList() }; return new List<Player>[] { connectingClients.ToList(), disconnectingClients.ToList() };
} }
@ -634,8 +643,8 @@ namespace IW4MAdmin
} }
// wait for all the connect tasks to finish // wait for all the connect tasks to finish
await Task.WhenAll(waiterList.Select(t => Task.Run(() => t.Wait()))); await Task.WhenAll(waiterList.Select(t => Task.Run(() => t.Wait(5000))));
if (ConnectionErrors > 0) if (ConnectionErrors > 0)
{ {
Logger.WriteVerbose($"{loc["MANAGER_CONNECTION_REST"]} {IP}:{Port}"); Logger.WriteVerbose($"{loc["MANAGER_CONNECTION_REST"]} {IP}:{Port}");
@ -806,10 +815,10 @@ namespace IW4MAdmin
CustomCallback = await ScriptLoaded(); CustomCallback = await ScriptLoaded();
string mainPath = EventParser.GetGameDir(); string mainPath = EventParser.GetGameDir();
#if DEBUG #if DEBUG
basepath.Value = @""; basepath.Value = @"D:\";
#endif #endif
string logPath; string logPath;
if (GameName == Game.IW5) if (GameName == Game.IW5 || ServerConfig.ManualLogPath?.Length > 0)
{ {
logPath = ServerConfig.ManualLogPath; logPath = ServerConfig.ManualLogPath;
} }
@ -831,11 +840,13 @@ namespace IW4MAdmin
Logger.WriteError($"{logPath} {loc["SERVER_ERROR_DNE"]}"); Logger.WriteError($"{logPath} {loc["SERVER_ERROR_DNE"]}");
#if !DEBUG #if !DEBUG
throw new ServerException($"{loc["SERVER_ERROR_LOG"]} {logPath}"); throw new ServerException($"{loc["SERVER_ERROR_LOG"]} {logPath}");
#else
LogEvent = new GameLogEventDetection(this, logPath, logfile.Value);
#endif #endif
} }
else else
{ {
LogEvent = new GameLogEvent(this, logPath, logfile.Value); LogEvent = new GameLogEventDetection(this, logPath, logfile.Value);
} }
Logger.WriteInfo($"Log file is {logPath}"); Logger.WriteInfo($"Log file is {logPath}");

View File

@ -625,7 +625,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
int individualClientRanking = await ctx.Set<EFRating>() int individualClientRanking = await ctx.Set<EFRating>()
.Where(c => c.ServerId == clientStats.ServerId) .Where(c => c.ServerId == clientStats.ServerId)
.Where(r => r.RatingHistory.Client.Level != Player.Permission.Banned) .Where(r => r.RatingHistory.Client.Level != Player.Permission.Banned)
.Where(r => r.ActivityAmount > 3600) .Where(r => r.ActivityAmount > Plugin.Config.Configuration().TopPlayersMinPlayTime)
.Where(r => r.RatingHistory.Client.LastConnection > thirtyDaysAgo) .Where(r => r.RatingHistory.Client.LastConnection > thirtyDaysAgo)
.Where(c => c.RatingHistory.ClientId != client.ClientId) .Where(c => c.RatingHistory.ClientId != client.ClientId)
.Where(r => r.Newest) .Where(r => r.Newest)
@ -670,11 +670,16 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
}); });
// weight the overall performance based on play time // weight the overall performance based on play time
var performanceAverage = clientStatsList.Sum(p => (p.Performance * p.TimePlayed)) / clientStatsList.Sum(p => p.TimePlayed); double performanceAverage = clientStatsList.Sum(p => (p.Performance * p.TimePlayed)) / clientStatsList.Sum(p => p.TimePlayed);
if (double.IsNaN(performanceAverage))
{
performanceAverage = clientStatsList.Average(p => p.Performance);
}
int overallClientRanking = await ctx.Set<EFRating>() int overallClientRanking = await ctx.Set<EFRating>()
.Where(r => r.RatingHistory.Client.Level != Player.Permission.Banned) .Where(r => r.RatingHistory.Client.Level != Player.Permission.Banned)
.Where(r => r.ActivityAmount > 3600) .Where(r => r.ActivityAmount > Plugin.Config.Configuration().TopPlayersMinPlayTime)
.Where(r => r.RatingHistory.Client.LastConnection > thirtyDaysAgo) .Where(r => r.RatingHistory.Client.LastConnection > thirtyDaysAgo)
.Where(r => r.RatingHistory.ClientId != client.ClientId) .Where(r => r.RatingHistory.ClientId != client.ClientId)
.Where(r => r.ServerId == null) .Where(r => r.ServerId == null)

View File

@ -0,0 +1,61 @@
using IW4MAdmin.Application;
using SharedLibraryCore.Configuration;
using SharedLibraryCore.Interfaces;
using System;
using System.Collections.Generic;
using System.IO;
using System.Text;
using System.Threading.Tasks;
using Xunit;
namespace Tests
{
public class ManagerFixture : IDisposable
{
public ApplicationManager Manager { get; private set; }
public ManagerFixture()
{
File.WriteAllText("test_mp.log", "TEST_LOG_FILE");
Manager = Program.ServerManager;
var config = new ApplicationConfiguration
{
Servers = new List<ServerConfiguration>()
{
new ServerConfiguration()
{
AutoMessages = new List<string>(),
IPAddress = "127.0.0.1",
Password = "test",
Port = 28963,
Rules = new List<string>(),
ManualLogPath = "https://raidmax.org/IW4MAdmin/getlog.php"
}
},
AutoMessages = new List<string>(),
GlobalRules = new List<string>(),
Maps = new List<MapConfiguration>(),
RConPollRate = 10000
};
Manager.ConfigHandler = new BaseConfigurationHandler<ApplicationConfiguration>("Test.json");
Manager.ConfigHandler.Set(config);
Manager.Init().Wait();
Task.Run(() => Manager.Start());
}
public void Dispose()
{
Manager.Stop();
}
}
[CollectionDefinition("ManagerCollection")]
public class ManagerCollection : ICollectionFixture<ManagerFixture>
{
}
}

View File

@ -0,0 +1,194 @@
using IW4MAdmin.Application;
using SharedLibraryCore;
using SharedLibraryCore.Interfaces;
using SharedLibraryCore.Objects;
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading;
using Xunit;
namespace Tests
{
[Collection("ManagerCollection")]
public class ManagerTests
{
readonly ApplicationManager Manager;
public ManagerTests(ManagerFixture fixture)
{
Manager = fixture.Manager;
}
[Fact]
public void AreCommandNamesUnique()
{
bool test = Manager.GetCommands().Count == Manager.GetCommands().Select(c => c.Name).Distinct().Count();
Assert.True(test, "command names are not unique");
}
[Fact]
public void AreCommandAliasesUnique()
{
var mgr = IW4MAdmin.Application.Program.ServerManager;
bool test = mgr.GetCommands().Count == mgr.GetCommands().Select(c => c.Alias).Distinct().Count();
Assert.True(test, "command aliases are not unique");
}
[Fact]
public void AddAndRemoveClientsViaJoinShouldSucceed()
{
var server = Manager.GetServers().First();
var waiters = new Queue<ManualResetEventSlim>();
int clientStartIndex = 4;
int clientNum = 10;
for (int i = clientStartIndex; i < clientStartIndex + clientNum; i++)
{
var e = new GameEvent()
{
Type = GameEvent.EventType.Join,
Origin = new Player()
{
Name = $"Player{i}",
NetworkId = i,
ClientNumber = i - 1
},
Owner = server
};
server.Manager.GetEventHandler().AddEvent(e);
waiters.Enqueue(e.OnProcessed);
}
while (waiters.Count > 0)
{
waiters.Dequeue().Wait();
}
Assert.True(server.ClientNum == clientNum, $"client num does not match added client num [{server.ClientNum}:{clientNum}]");
for (int i = clientStartIndex; i < clientStartIndex + clientNum; i++)
{
var e = new GameEvent()
{
Type = GameEvent.EventType.Disconnect,
Origin = new Player()
{
Name = $"Player{i}",
NetworkId = i,
ClientNumber = i - 1
},
Owner = server
};
server.Manager.GetEventHandler().AddEvent(e);
waiters.Enqueue(e.OnProcessed);
}
while (waiters.Count > 0)
{
waiters.Dequeue().Wait();
}
Assert.True(server.ClientNum == 0, "there are still clients connected");
}
[Fact]
public void AddAndRemoveClientsViaRconShouldSucceed()
{
var server = Manager.GetServers().First();
var waiters = new Queue<ManualResetEventSlim>();
int clientIndexStart = 1;
int clientNum = 8;
for (int i = clientIndexStart; i < clientNum + clientIndexStart; i++)
{
var e = new GameEvent()
{
Type = GameEvent.EventType.Connect,
Origin = new Player()
{
Name = $"Player{i}",
NetworkId = i,
ClientNumber = i - 1,
IPAddress = i,
Ping = 50,
CurrentServer = server
},
Owner = server,
};
Manager.GetEventHandler().AddEvent(e);
waiters.Enqueue(e.OnProcessed);
}
while (waiters.Count > 0)
{
waiters.Dequeue().Wait();
}
int actualClientNum = server.GetPlayersAsList().Count(p => p.State == Player.ClientState.Connected);
Assert.True(actualClientNum == clientNum, $"client connected states don't match [{actualClientNum}:{clientNum}");
for (int i = clientIndexStart; i < clientNum + clientIndexStart; i++)
{
var e = new GameEvent()
{
Type = GameEvent.EventType.Disconnect,
Origin = new Player()
{
Name = $"Player{i}",
NetworkId = i,
ClientNumber = i - 1,
IPAddress = i,
Ping = 50,
CurrentServer = server
},
Owner = server,
};
Manager.GetEventHandler().AddEvent(e);
waiters.Enqueue(e.OnProcessed);
}
while (waiters.Count > 0)
{
waiters.Dequeue().Wait();
}
actualClientNum = server.ClientNum;
Assert.True(actualClientNum == 0, "there are clients still connected");
}
[Fact]
public void AddClientViaLog()
{
var resetEvent = new ManualResetEventSlim();
resetEvent.Reset();
Manager.OnServerEvent += (sender, eventArgs) =>
{
if (eventArgs.Event.Type == GameEvent.EventType.Join)
{
eventArgs.Event.OnProcessed.Wait();
Assert.True(false);
}
};
File.AppendAllText("test_mp.log", " 2:33 J;224b3d0bc64ab4f9;0;goober");
resetEvent.Wait(5000);
}
}
}

View File

@ -1,192 +0,0 @@
#if DEBUG
using System;
using System.Linq;
using System.Threading.Tasks;
using SharedLibraryCore;
using SharedLibraryCore.Interfaces;
using SharedLibraryCore.Helpers;
using SharedLibraryCore.Objects;
namespace IW4MAdmin.Plugins
{
public class Tests : IPlugin
{
public string Name => "Dev Tests";
public float Version => 0.1f;
public string Author => "RaidMax";
public async Task OnEventAsync(GameEvent E, Server S)
{
return;
if (E.Type == GameEvent.EventType.Start)
{
#region UNIT_TEST_LOG_CONNECT
for (int i = 1; i <= 8; i++)
{
var e = new GameEvent()
{
Type = GameEvent.EventType.Join,
Origin = new Player()
{
Name = $"Player{i}",
NetworkId = i,
ClientNumber = i - 1
},
Owner = S
};
S.Manager.GetEventHandler().AddEvent(e);
e.OnProcessed.Wait();
}
S.Logger.WriteAssert(S.ClientNum == 8, "UNIT_TEST_LOG_CONNECT failed client num check");
#endregion
#region UNIT_TEST_RCON_AUTHENTICATE
for (int i = 1; i <= 8; i++)
{
var e = new GameEvent()
{
Type = GameEvent.EventType.Connect,
Origin = new Player()
{
Name = $"Player{i}",
NetworkId = i,
ClientNumber = i - 1,
IPAddress = i,
Ping = 50,
CurrentServer = S
},
Owner = S,
};
S.Manager.GetEventHandler().AddEvent(e);
e.OnProcessed.Wait();
}
S.Logger.WriteAssert(S.GetPlayersAsList().Count(p => p.State == Player.ClientState.Connected) == 8,
"UNIT_TEST_RCON_AUTHENTICATE failed client num connected state check");
#endregion
}
//if (E.Type == GameEvent.EventType.Start)
//{
// #region PLAYER_HISTORY
// var rand = new Random(GetHashCode());
// var time = DateTime.UtcNow;
// await Task.Run(() =>
// {
// if (S.PlayerHistory.Count > 0)
// return;
// while (S.PlayerHistory.Count < 144)
// {
// S.PlayerHistory.Enqueue(new PlayerHistory(time, rand.Next(7, 18)));
// time = time.AddMinutes(PlayerHistory.UpdateInterval);
// }
// });
// #endregion
// #region PLUGIN_INFO
// Console.WriteLine("|Name |Alias|Description |Requires Target|Syntax |Required Level|");
// Console.WriteLine("|--------------| -----| --------------------------------------------------------| -----------------| -------------| ----------------|");
// foreach (var command in S.Manager.GetCommands().OrderByDescending(c => c.Permission).ThenBy(c => c.Name))
// {
// Console.WriteLine($"|{command.Name}|{command.Alias}|{command.Description}|{command.RequiresTarget}|{command.Syntax.Substring(8).EscapeMarkdown()}|{command.Permission}|");
// }
// #endregion
//}
}
public Task OnLoadAsync(IManager manager) => Task.CompletedTask;
public Task OnTickAsync(Server S)
{
return Task.CompletedTask;
/*
if ((DateTime.Now - Interval).TotalSeconds > 1)
{
var rand = new Random();
int index = rand.Next(0, 17);
var p = new Player()
{
Name = $"Test_{index}",
NetworkId = (long)$"_test_{index}".GetHashCode(),
ClientNumber = index,
Ping = 1,
IPAddress = $"127.0.0.{index}".ConvertToIP()
};
if (S.Players.ElementAt(index) != null)
await S.RemovePlayer(index);
// await S.AddPlayer(p);
Interval = DateTime.Now;
if (S.ClientNum > 0)
{
var victimPlayer = S.Players.Where(pl => pl != null).ToList()[rand.Next(0, S.ClientNum - 1)];
var attackerPlayer = S.Players.Where(pl => pl != null).ToList()[rand.Next(0, S.ClientNum - 1)];
await S.ExecuteEvent(new Event(Event.GType.Say, $"test_{attackerPlayer.ClientNumber}", victimPlayer, attackerPlayer, S));
string[] eventLine = null;
for (int i = 0; i < 1; i++)
{
if (S.GameName == Server.Game.IW4)
{
// attackerID ; victimID ; attackerOrigin ; victimOrigin ; Damage ; Weapon ; hitLocation ; meansOfDeath
var minimapInfo = StatsPlugin.MinimapConfig.IW4Minimaps().MapInfo.FirstOrDefault(m => m.MapName == S.CurrentMap.Name);
if (minimapInfo == null)
return;
eventLine = new string[]
{
"ScriptKill",
attackerPlayer.NetworkId.ToString(),
victimPlayer.NetworkId.ToString(),
new Vector3(rand.Next(minimapInfo.MaxRight, minimapInfo.MaxLeft), rand.Next(minimapInfo.MaxBottom, minimapInfo.MaxTop), rand.Next(0, 100)).ToString(),
new Vector3(rand.Next(minimapInfo.MaxRight, minimapInfo.MaxLeft), rand.Next(minimapInfo.MaxBottom, minimapInfo.MaxTop), rand.Next(0, 100)).ToString(),
rand.Next(50, 105).ToString(),
((StatsPlugin.IW4Info.WeaponName)rand.Next(0, Enum.GetValues(typeof(StatsPlugin.IW4Info.WeaponName)).Length - 1)).ToString(),
((StatsPlugin.IW4Info.HitLocation)rand.Next(0, Enum.GetValues(typeof(StatsPlugin.IW4Info.HitLocation)).Length - 1)).ToString(),
((StatsPlugin.IW4Info.MeansOfDeath)rand.Next(0, Enum.GetValues(typeof(StatsPlugin.IW4Info.MeansOfDeath)).Length - 1)).ToString()
};
}
else
{
eventLine = new string[]
{
"K",
victimPlayer.NetworkId.ToString(),
victimPlayer.ClientNumber.ToString(),
rand.Next(0, 1) == 0 ? "allies" : "axis",
victimPlayer.Name,
attackerPlayer.NetworkId.ToString(),
attackerPlayer.ClientNumber.ToString(),
rand.Next(0, 1) == 0 ? "allies" : "axis",
attackerPlayer.Name.ToString(),
((StatsPlugin.IW4Info.WeaponName)rand.Next(0, Enum.GetValues(typeof(StatsPlugin.IW4Info.WeaponName)).Length - 1)).ToString(), // Weapon
rand.Next(50, 105).ToString(), // Damage
((StatsPlugin.IW4Info.MeansOfDeath)rand.Next(0, Enum.GetValues(typeof(StatsPlugin.IW4Info.MeansOfDeath)).Length - 1)).ToString(), // Means of Death
((StatsPlugin.IW4Info.HitLocation)rand.Next(0, Enum.GetValues(typeof(StatsPlugin.IW4Info.HitLocation)).Length - 1)).ToString(), // Hit Location
};
}
var _event = Event.ParseEventString(eventLine, S);
await S.ExecuteEvent(_event);
}
}
}
*/
}
public Task OnUnloadAsync() => Task.CompletedTask;
}
}
#endif

View File

@ -0,0 +1,10 @@
using System;
using System.Collections.Generic;
using System.Text;
namespace Tests
{
class ServerTests
{
}
}

View File

@ -11,16 +11,18 @@
<DefineConstants>TRACE;DEBUG;NETCOREAPP2_0</DefineConstants> <DefineConstants>TRACE;DEBUG;NETCOREAPP2_0</DefineConstants>
</PropertyGroup> </PropertyGroup>
<Target Name="PostBuild" AfterTargets="PostBuildEvent"> <ItemGroup>
<Exec Command="copy &quot;$(TargetPath)&quot; &quot;$(SolutionDir)BUILD\Plugins&quot;" /> <PackageReference Include="xunit" Version="2.4.0" />
</Target> <PackageReference Include="xunit.runner.visualstudio" Version="2.4.0" />
</ItemGroup>
<ItemGroup> <ItemGroup>
<ProjectReference Include="..\..\Application\Application.csproj" />
<ProjectReference Include="..\..\SharedLibraryCore\SharedLibraryCore.csproj" /> <ProjectReference Include="..\..\SharedLibraryCore\SharedLibraryCore.csproj" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<PackageReference Update="Microsoft.NETCore.App"/> <PackageReference Update="Microsoft.NETCore.App" />
</ItemGroup> </ItemGroup>
</Project> </Project>

View File

@ -112,7 +112,7 @@ If you wish to further customize your experience of **IW4MAdmin**, the following
* `{{TOTALPLAYTIME}}` &mdash; displays the cumulative play time (in man-hours) on all monitored servers * `{{TOTALPLAYTIME}}` &mdash; displays the cumulative play time (in man-hours) on all monitored servers
* `{{VERSION}}` &mdash; displays the version of **IW4MAdmin** * `{{VERSION}}` &mdash; displays the version of **IW4MAdmin**
* `{{ADMINS}}` &mdash; displays the currently connected and *unmasked* privileged users online * `{{ADMINS}}` &mdash; displays the currently connected and *unmasked* privileged users online
* `{{NEXTMAP}} &dmash; displays the next map in rotation * `{{NEXTMAP}}` &mdash; displays the next map and gametype in rotation
`GlobalRules` `GlobalRules`
* Specifies the list of rules that apply to **all** servers` * Specifies the list of rules that apply to **all** servers`
@ -120,7 +120,7 @@ If you wish to further customize your experience of **IW4MAdmin**, the following
`Maps` `Maps`
* Specifies the list of maps for each supported game * Specifies the list of maps for each supported game
* `Name` * `Name`
* Specifies the name of the map as returned by the game * Specifies the name of the map as returned by the game (usually the file name sans the file extension)
* `Alias` * `Alias`
* Specifies the display name of the map (as seen while loading in) * Specifies the display name of the map (as seen while loading in)
___ ___
@ -181,7 +181,7 @@ All players are identified 5 separate ways
2. `IP` - The player's IP Address 2. `IP` - The player's IP Address
3. `Client ID` - The internal reference to a player, generated by **IW4MAdmin** 3. `Client ID` - The internal reference to a player, generated by **IW4MAdmin**
4. `Name` - The visible player name as it appears in game 4. `Name` - The visible player name as it appears in game
5. `Client Number` - The slot the client client occupies on the server. The number ranges between 0 and the max number of clients allowed on the server 5. `Client Number` - The slot the client occupies on a server. (The number ranges between 0 and the max number of clients allowed on the server)
For most commands players are identified by their `Name` For most commands players are identified by their `Name`
However, if they are currently offline, or their name contains un-typable characters, their `Client ID` must be used However, if they are currently offline, or their name contains un-typable characters, their `Client ID` must be used

View File

@ -43,6 +43,8 @@ namespace SharedLibraryCore.Database
currentPath = !RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ? currentPath = !RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ?
$"{Path.DirectorySeparatorChar}{currentPath}" : $"{Path.DirectorySeparatorChar}{currentPath}" :
currentPath; currentPath;
// todo: fix later
var connectionStringBuilder = new SqliteConnectionStringBuilder { DataSource = $"{currentPath}{Path.DirectorySeparatorChar}Database.db".Substring(6) }; var connectionStringBuilder = new SqliteConnectionStringBuilder { DataSource = $"{currentPath}{Path.DirectorySeparatorChar}Database.db".Substring(6) };
var connectionString = connectionStringBuilder.ToString(); var connectionString = connectionStringBuilder.ToString();
var connection = new SqliteConnection(connectionString); var connection = new SqliteConnection(connectionString);
@ -98,22 +100,28 @@ namespace SharedLibraryCore.Database
// adapted from // adapted from
// https://aleemkhan.wordpress.com/2013/02/28/dynamically-adding-dbset-properties-in-dbcontext-for-entity-framework-code-first/ // https://aleemkhan.wordpress.com/2013/02/28/dynamically-adding-dbset-properties-in-dbcontext-for-entity-framework-code-first/
#if !DEBUG //#if DEBUG == TRUE
foreach (string dllPath in Directory.GetFiles($"{Utilities.OperatingDirectory}Plugins")) // // foreach (string dllPath in Directory.GetFiles($"{Utilities.OperatingDirectory}Plugins"))
#else //#else
//todo: fix the debug thingie for entity scanning
IEnumerable<string> directoryFiles; IEnumerable<string> directoryFiles;
try
string pluginDir = $@"{Environment.CurrentDirectory}{Path.DirectorySeparatorChar}bin{Path.DirectorySeparatorChar}Debug{Path.DirectorySeparatorChar}netcoreapp2.0{Path.DirectorySeparatorChar}Plugins";
if (!Directory.Exists(pluginDir))
{ {
directoryFiles = Directory.GetFiles($@"{Environment.CurrentDirectory}{Path.DirectorySeparatorChar}bin{Path.DirectorySeparatorChar}Debug{Path.DirectorySeparatorChar}netcoreapp2.0{Path.DirectorySeparatorChar}Plugins").Where(f => f.Contains(".dll")); pluginDir = $@"{Environment.CurrentDirectory}{Path.DirectorySeparatorChar}Plugins";
if (!Directory.Exists(pluginDir))
{
pluginDir = Utilities.OperatingDirectory;
}
} }
catch (Exception) directoryFiles = Directory.GetFiles(pluginDir).Where(f => f.Contains(".dll"));
{
directoryFiles = Directory.GetFiles($@"{Environment.CurrentDirectory}{Path.DirectorySeparatorChar}Plugins").Where(f => f.Contains(".dll"));
}
foreach (string dllPath in directoryFiles) foreach (string dllPath in directoryFiles)
#endif //#endif
{ {
Assembly library; Assembly library;
try try

View File

@ -93,6 +93,7 @@ namespace SharedLibraryCore
queuedEvent.Type != EventType.Connect && queuedEvent.Type != EventType.Connect &&
queuedEvent.Type != EventType.Join && queuedEvent.Type != EventType.Join &&
queuedEvent.Type != EventType.Quit && queuedEvent.Type != EventType.Quit &&
queuedEvent.Type != EventType.Disconnect &&
// we don't care about unknown events // we don't care about unknown events
queuedEvent.Origin.NetworkId != 0; queuedEvent.Origin.NetworkId != 0;
} }

View File

@ -0,0 +1,29 @@
using System;
using System.Collections.Generic;
using System.Text;
namespace SharedLibraryCore.Interfaces
{
/// <summary>
/// represents the abtraction of game log reading
/// </summary>
public interface IGameLogReader
{
/// <summary>
/// get new events that have occured since the last poll
/// </summary>
/// <param name="server"></param>
/// <param name="fileSizeDiff"></param>
/// <param name="startPosition"></param>
/// <returns></returns>
ICollection<GameEvent> EventsFromLog(Server server, long fileSizeDiff, long startPosition);
/// <summary>
/// how long the log file is
/// </summary>
long Length { get; }
/// <summary>
/// how often to poll the log file
/// </summary>
int UpdateInterval { get; }
}
}

View File

@ -28,7 +28,10 @@ namespace SharedLibraryCore.Localization
get get
{ {
if (!Set.TryGetValue(key, out string value)) if (!Set.TryGetValue(key, out string value))
throw new Exception($"Invalid locale key {key}"); {
// throw new Exception($"Invalid locale key {key}");
return $"unknown locale key {key}";
}
return value; return value;
} }
} }

View File

@ -15,8 +15,21 @@ namespace SharedLibraryCore.Plugins
public static bool Load(IManager Manager) public static bool Load(IManager Manager)
{ {
string[] dllFileNames = Directory.GetFiles($"{Utilities.OperatingDirectory}Plugins{Path.DirectorySeparatorChar}", "*.dll"); string pluginDir = $"{Utilities.OperatingDirectory}Plugins{Path.DirectorySeparatorChar}";
string[] scriptFileNames = Directory.GetFiles($"{Utilities.OperatingDirectory}Plugins{Path.DirectorySeparatorChar}", "*.js"); string[] dllFileNames = null;
string[] scriptFileNames = null;
if (Directory.Exists(pluginDir))
{
dllFileNames = Directory.GetFiles($"{Utilities.OperatingDirectory}Plugins{Path.DirectorySeparatorChar}", "*.dll");
scriptFileNames = Directory.GetFiles($"{Utilities.OperatingDirectory}Plugins{Path.DirectorySeparatorChar}", "*.js");
}
else
{
dllFileNames = new string[0];
scriptFileNames = new string[0];
}
if (dllFileNames.Length == 0 && if (dllFileNames.Length == 0 &&
scriptFileNames.Length == 0) scriptFileNames.Length == 0)

View File

@ -40,7 +40,7 @@ namespace SharedLibraryCore.RCon
ILogger Log; ILogger Log;
int FailedSends; int FailedSends;
int FailedReceives; int FailedReceives;
DateTime LastQuery; static DateTime LastQuery;
string response; string response;
ManualResetEvent OnConnected; ManualResetEvent OnConnected;

View File

@ -18,7 +18,7 @@ namespace SharedLibraryCore
{ {
public static string OperatingDirectory = Path.GetDirectoryName(Assembly.GetEntryAssembly().Location) + Path.DirectorySeparatorChar; public static string OperatingDirectory = Path.GetDirectoryName(Assembly.GetEntryAssembly().Location) + Path.DirectorySeparatorChar;
public static Encoding EncodingType; public static Encoding EncodingType;
public static Localization.Layout CurrentLocalization; public static Localization.Layout CurrentLocalization = new Localization.Layout(new Dictionary<string, string>());
public static string HttpRequest(string location, string header, string headerValue) public static string HttpRequest(string location, string header, string headerValue)
{ {