fix issue with duplicate key on top stats page
This commit is contained in:
parent
507688a175
commit
b27ae1517e
@ -117,7 +117,8 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Expression<Func<EFClientRankingHistory, bool>> GetNewRankingFunc(int? clientId = null, long? serverId = null)
|
public Expression<Func<EFClientRankingHistory, bool>> GetNewRankingFunc(int? clientId = null,
|
||||||
|
long? serverId = null)
|
||||||
{
|
{
|
||||||
return (ranking) => ranking.ServerId == serverId
|
return (ranking) => ranking.ServerId == serverId
|
||||||
&& ranking.Client.Level != Data.Models.Client.EFClient.Permission.Banned
|
&& ranking.Client.Level != Data.Models.Client.EFClient.Permission.Banned
|
||||||
@ -181,8 +182,17 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
})
|
})
|
||||||
.Take(60)
|
.Take(60)
|
||||||
.ToListAsync();
|
.ToListAsync();
|
||||||
|
|
||||||
|
if (rankingsDict.ContainsKey(clientId))
|
||||||
|
{
|
||||||
|
rankingsDict[clientId] = rankingsDict[clientId].Concat(eachRank).Distinct()
|
||||||
|
.OrderByDescending(ranking => ranking.CreatedDateTime).ToList();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
rankingsDict.Add(clientId, eachRank);
|
rankingsDict.Add(clientId, eachRank);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var statsInfo = await context.Set<EFClientStatistics>()
|
var statsInfo = await context.Set<EFClientStatistics>()
|
||||||
.Where(stat => clientIdsList.Contains(stat.ClientId))
|
.Where(stat => clientIdsList.Contains(stat.ClientId))
|
||||||
@ -195,7 +205,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
ClientId = s.Key,
|
ClientId = s.Key,
|
||||||
Kills = s.Sum(c => c.Kills),
|
Kills = s.Sum(c => c.Kills),
|
||||||
Deaths = s.Sum(c => c.Deaths),
|
Deaths = s.Sum(c => c.Deaths),
|
||||||
KDR = s.Sum(c => (c.Kills / (double) (c.Deaths == 0 ? 1 : c.Deaths)) * c.TimePlayed) /
|
KDR = s.Sum(c => (c.Kills / (double)(c.Deaths == 0 ? 1 : c.Deaths)) * c.TimePlayed) /
|
||||||
s.Sum(c => c.TimePlayed),
|
s.Sum(c => c.TimePlayed),
|
||||||
TotalTimePlayed = s.Sum(c => c.TimePlayed),
|
TotalTimePlayed = s.Sum(c => c.TimePlayed),
|
||||||
UpdatedAt = s.Max(c => c.UpdatedAt)
|
UpdatedAt = s.Max(c => c.UpdatedAt)
|
||||||
@ -282,7 +292,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
.Select(grp => new
|
.Select(grp => new
|
||||||
{
|
{
|
||||||
grp.Key,
|
grp.Key,
|
||||||
Ratings = grp.Select(r => new {r.Performance, r.Ranking, r.When})
|
Ratings = grp.Select(r => new { r.Performance, r.Ranking, r.When })
|
||||||
});
|
});
|
||||||
|
|
||||||
var iqStatsInfo = (from stat in context.Set<EFClientStatistics>()
|
var iqStatsInfo = (from stat in context.Set<EFClientStatistics>()
|
||||||
@ -296,7 +306,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
ClientId = s.Key,
|
ClientId = s.Key,
|
||||||
Kills = s.Sum(c => c.Kills),
|
Kills = s.Sum(c => c.Kills),
|
||||||
Deaths = s.Sum(c => c.Deaths),
|
Deaths = s.Sum(c => c.Deaths),
|
||||||
KDR = s.Sum(c => (c.Kills / (double) (c.Deaths == 0 ? 1 : c.Deaths)) * c.TimePlayed) /
|
KDR = s.Sum(c => (c.Kills / (double)(c.Deaths == 0 ? 1 : c.Deaths)) * c.TimePlayed) /
|
||||||
s.Sum(c => c.TimePlayed),
|
s.Sum(c => c.TimePlayed),
|
||||||
TotalTimePlayed = s.Sum(c => c.TimePlayed),
|
TotalTimePlayed = s.Sum(c => c.TimePlayed),
|
||||||
});
|
});
|
||||||
@ -394,7 +404,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
Port = sv.Port,
|
Port = sv.Port,
|
||||||
EndPoint = sv.ToString(),
|
EndPoint = sv.ToString(),
|
||||||
ServerId = serverId,
|
ServerId = serverId,
|
||||||
GameName = (Reference.Game?) sv.GameName,
|
GameName = (Reference.Game?)sv.GameName,
|
||||||
HostName = sv.Hostname
|
HostName = sv.Hostname
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -404,9 +414,9 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
}
|
}
|
||||||
|
|
||||||
// we want to set the gamename up if it's never been set, or it changed
|
// we want to set the gamename up if it's never been set, or it changed
|
||||||
else if (!server.GameName.HasValue || server.GameName.Value != (Reference.Game) sv.GameName)
|
else if (!server.GameName.HasValue || server.GameName.Value != (Reference.Game)sv.GameName)
|
||||||
{
|
{
|
||||||
server.GameName = (Reference.Game) sv.GameName;
|
server.GameName = (Reference.Game)sv.GameName;
|
||||||
ctx.Entry(server).Property(_prop => _prop.GameName).IsModified = true;
|
ctx.Entry(server).Property(_prop => _prop.GameName).IsModified = true;
|
||||||
ctx.SaveChanges();
|
ctx.SaveChanges();
|
||||||
}
|
}
|
||||||
@ -497,7 +507,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
{
|
{
|
||||||
Active = true,
|
Active = true,
|
||||||
HitCount = 0,
|
HitCount = 0,
|
||||||
Location = (int) hl
|
Location = (int)hl
|
||||||
}).ToList()
|
}).ToList()
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -517,7 +527,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
{
|
{
|
||||||
Active = true,
|
Active = true,
|
||||||
HitCount = 0,
|
HitCount = 0,
|
||||||
Location = (int) hl
|
Location = (int)hl
|
||||||
})
|
})
|
||||||
.ToList();
|
.ToList();
|
||||||
|
|
||||||
@ -549,9 +559,9 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
}
|
}
|
||||||
|
|
||||||
catch (DbUpdateException updateException) when (
|
catch (DbUpdateException updateException) when (
|
||||||
updateException.InnerException is PostgresException {SqlState: "23503"}
|
updateException.InnerException is PostgresException { SqlState: "23503" }
|
||||||
|| updateException.InnerException is SqliteException {SqliteErrorCode: 787}
|
|| updateException.InnerException is SqliteException { SqliteErrorCode: 787 }
|
||||||
|| updateException.InnerException is MySqlException {SqlState: "23503"})
|
|| updateException.InnerException is MySqlException { SqlState: "23503" })
|
||||||
{
|
{
|
||||||
_log.LogWarning("Trying to add {Client} to stats before they have been added to the database",
|
_log.LogWarning("Trying to add {Client} to stats before they have been added to the database",
|
||||||
pl.ToString());
|
pl.ToString());
|
||||||
@ -672,9 +682,9 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
ServerId = serverId,
|
ServerId = serverId,
|
||||||
DeathOrigin = vDeathOrigin,
|
DeathOrigin = vDeathOrigin,
|
||||||
KillOrigin = vKillOrigin,
|
KillOrigin = vKillOrigin,
|
||||||
DeathType = (int) ParseEnum<IW4Info.MeansOfDeath>.Get(type, typeof(IW4Info.MeansOfDeath)),
|
DeathType = (int)ParseEnum<IW4Info.MeansOfDeath>.Get(type, typeof(IW4Info.MeansOfDeath)),
|
||||||
Damage = int.Parse(damage),
|
Damage = int.Parse(damage),
|
||||||
HitLoc = (int) ParseEnum<IW4Info.HitLocation>.Get(hitLoc, typeof(IW4Info.HitLocation)),
|
HitLoc = (int)ParseEnum<IW4Info.HitLocation>.Get(hitLoc, typeof(IW4Info.HitLocation)),
|
||||||
WeaponReference = weapon,
|
WeaponReference = weapon,
|
||||||
ViewAngles = vViewAngles,
|
ViewAngles = vViewAngles,
|
||||||
TimeOffset = long.Parse(offset),
|
TimeOffset = long.Parse(offset),
|
||||||
@ -688,13 +698,13 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
AnglesList = snapshotAngles,
|
AnglesList = snapshotAngles,
|
||||||
IsAlive = isAlive == "1",
|
IsAlive = isAlive == "1",
|
||||||
TimeSinceLastAttack = long.Parse(lastAttackTime),
|
TimeSinceLastAttack = long.Parse(lastAttackTime),
|
||||||
GameName = (int) attacker.CurrentServer.GameName
|
GameName = (int)attacker.CurrentServer.GameName
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_log.LogError(ex, "Could not parse script hit data. Damage={Damage}, TimeOffset={Offset}, TimeSinceLastAttack={LastAttackTime}",
|
_log.LogError(ex,
|
||||||
|
"Could not parse script hit data. Damage={Damage}, TimeOffset={Offset}, TimeSinceLastAttack={LastAttackTime}",
|
||||||
damage, offset, lastAttackTime);
|
damage, offset, lastAttackTime);
|
||||||
|
|
||||||
return;
|
return;
|
||||||
@ -702,7 +712,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
|
|
||||||
hit.SetAdditionalProperty("HitLocationReference", hitLoc);
|
hit.SetAdditionalProperty("HitLocationReference", hitLoc);
|
||||||
|
|
||||||
if (hit.HitLoc == (int) IW4Info.HitLocation.shield)
|
if (hit.HitLoc == (int)IW4Info.HitLocation.shield)
|
||||||
{
|
{
|
||||||
// we don't care about shield hits
|
// we don't care about shield hits
|
||||||
return;
|
return;
|
||||||
@ -721,9 +731,9 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
await waiter.WaitAsync(Utilities.DefaultCommandTimeout, Plugin.ServerManager.CancellationToken);
|
await waiter.WaitAsync(Utilities.DefaultCommandTimeout, Plugin.ServerManager.CancellationToken);
|
||||||
|
|
||||||
// increment their hit count
|
// increment their hit count
|
||||||
if (hit.DeathType == (int) IW4Info.MeansOfDeath.MOD_PISTOL_BULLET ||
|
if (hit.DeathType == (int)IW4Info.MeansOfDeath.MOD_PISTOL_BULLET ||
|
||||||
hit.DeathType == (int) IW4Info.MeansOfDeath.MOD_RIFLE_BULLET ||
|
hit.DeathType == (int)IW4Info.MeansOfDeath.MOD_RIFLE_BULLET ||
|
||||||
hit.DeathType == (int) IW4Info.MeansOfDeath.MOD_HEAD_SHOT)
|
hit.DeathType == (int)IW4Info.MeansOfDeath.MOD_HEAD_SHOT)
|
||||||
{
|
{
|
||||||
clientStats.HitLocations.First(hl => hl.Location == hit.HitLoc).HitCount += 1;
|
clientStats.HitLocations.First(hl => hl.Location == hit.HitLoc).HitCount += 1;
|
||||||
}
|
}
|
||||||
@ -898,7 +908,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
new EFPenalty()
|
new EFPenalty()
|
||||||
{
|
{
|
||||||
AutomatedOffense = penalty.Type == Detection.DetectionType.Bone
|
AutomatedOffense = penalty.Type == Detection.DetectionType.Bone
|
||||||
? $"{penalty.Type}-{(int) penalty.Location}-{Math.Round(penalty.Value, 2)}@{penalty.HitCount}"
|
? $"{penalty.Type}-{(int)penalty.Location}-{Math.Round(penalty.Value, 2)}@{penalty.HitCount}"
|
||||||
: $"{penalty.Type}-{Math.Round(penalty.Value, 2)}@{penalty.HitCount}",
|
: $"{penalty.Type}-{Math.Round(penalty.Value, 2)}@{penalty.HitCount}",
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -915,7 +925,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
}
|
}
|
||||||
|
|
||||||
string flagReason = penalty.Type == Cheat.Detection.DetectionType.Bone
|
string flagReason = penalty.Type == Cheat.Detection.DetectionType.Bone
|
||||||
? $"{penalty.Type}-{(int) penalty.Location}-{Math.Round(penalty.Value, 2)}@{penalty.HitCount}"
|
? $"{penalty.Type}-{(int)penalty.Location}-{Math.Round(penalty.Value, 2)}@{penalty.HitCount}"
|
||||||
: $"{penalty.Type}-{Math.Round(penalty.Value, 2)}@{penalty.HitCount}";
|
: $"{penalty.Type}-{Math.Round(penalty.Value, 2)}@{penalty.HitCount}";
|
||||||
|
|
||||||
penaltyClient.AdministeredPenalties = new List<EFPenalty>()
|
penaltyClient.AdministeredPenalties = new List<EFPenalty>()
|
||||||
@ -1070,7 +1080,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public async Task UpdateStatHistory(EFClient client, EFClientStatistics clientStats)
|
public async Task UpdateStatHistory(EFClient client, EFClientStatistics clientStats)
|
||||||
{
|
{
|
||||||
int currentSessionTime = (int) (DateTime.UtcNow - client.LastConnection).TotalSeconds;
|
int currentSessionTime = (int)(DateTime.UtcNow - client.LastConnection).TotalSeconds;
|
||||||
|
|
||||||
// don't update their stat history if they haven't played long
|
// don't update their stat history if they haven't played long
|
||||||
if (currentSessionTime < 60)
|
if (currentSessionTime < 60)
|
||||||
@ -1282,7 +1292,8 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
|
|
||||||
if (performances.Any(performance => performance.TimePlayed >= minPlayTime))
|
if (performances.Any(performance => performance.TimePlayed >= minPlayTime))
|
||||||
{
|
{
|
||||||
var aggregateZScore = performances.WeightValueByPlaytime(nameof(EFClientStatistics.ZScore), minPlayTime);
|
var aggregateZScore =
|
||||||
|
performances.WeightValueByPlaytime(nameof(EFClientStatistics.ZScore), minPlayTime);
|
||||||
|
|
||||||
int? aggregateRanking = await context.Set<EFClientStatistics>()
|
int? aggregateRanking = await context.Set<EFClientStatistics>()
|
||||||
.Where(stat => stat.ClientId != clientId)
|
.Where(stat => stat.ClientId != clientId)
|
||||||
@ -1395,8 +1406,8 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
victimStats.EloRating = Math.Max(0, Math.Round(victimStats.EloRating, 2));
|
victimStats.EloRating = Math.Max(0, Math.Round(victimStats.EloRating, 2));
|
||||||
|
|
||||||
// update after calculation
|
// update after calculation
|
||||||
attackerStats.TimePlayed += (int) (DateTime.UtcNow - attackerStats.LastActive).TotalSeconds;
|
attackerStats.TimePlayed += (int)(DateTime.UtcNow - attackerStats.LastActive).TotalSeconds;
|
||||||
victimStats.TimePlayed += (int) (DateTime.UtcNow - victimStats.LastActive).TotalSeconds;
|
victimStats.TimePlayed += (int)(DateTime.UtcNow - victimStats.LastActive).TotalSeconds;
|
||||||
attackerStats.LastActive = DateTime.UtcNow;
|
attackerStats.LastActive = DateTime.UtcNow;
|
||||||
victimStats.LastActive = DateTime.UtcNow;
|
victimStats.LastActive = DateTime.UtcNow;
|
||||||
}
|
}
|
||||||
@ -1436,7 +1447,7 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
var spmMultiplier = 2.934 *
|
var spmMultiplier = 2.934 *
|
||||||
Math.Pow(
|
Math.Pow(
|
||||||
_servers[clientStats.ServerId]
|
_servers[clientStats.ServerId]
|
||||||
.TeamCount((IW4Info.Team) clientStats.Team == IW4Info.Team.Allies
|
.TeamCount((IW4Info.Team)clientStats.Team == IW4Info.Team.Allies
|
||||||
? IW4Info.Team.Axis
|
? IW4Info.Team.Axis
|
||||||
: IW4Info.Team.Allies), -0.454);
|
: IW4Info.Team.Allies), -0.454);
|
||||||
killSpm *= Math.Max(1, spmMultiplier);
|
killSpm *= Math.Max(1, spmMultiplier);
|
||||||
@ -1455,8 +1466,8 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
|
|
||||||
// calculate the weight of the new play time against last 10 hours of gameplay
|
// calculate the weight of the new play time against last 10 hours of gameplay
|
||||||
int totalPlayTime = (clientStats.TimePlayed == 0)
|
int totalPlayTime = (clientStats.TimePlayed == 0)
|
||||||
? (int) (DateTime.UtcNow - clientStats.LastActive).TotalSeconds
|
? (int)(DateTime.UtcNow - clientStats.LastActive).TotalSeconds
|
||||||
: clientStats.TimePlayed + (int) (DateTime.UtcNow - clientStats.LastActive).TotalSeconds;
|
: clientStats.TimePlayed + (int)(DateTime.UtcNow - clientStats.LastActive).TotalSeconds;
|
||||||
|
|
||||||
double SPMAgainstPlayWeight = timeSinceLastCalc / Math.Min(600, (totalPlayTime / 60.0));
|
double SPMAgainstPlayWeight = timeSinceLastCalc / Math.Min(600, (totalPlayTime / 60.0));
|
||||||
|
|
||||||
@ -1476,7 +1487,10 @@ namespace IW4MAdmin.Plugins.Stats.Helpers
|
|||||||
if (double.IsNaN(clientStats.SPM) || double.IsNaN(clientStats.Skill))
|
if (double.IsNaN(clientStats.SPM) || double.IsNaN(clientStats.Skill))
|
||||||
{
|
{
|
||||||
_log.LogWarning("clientStats SPM/Skill NaN {@killInfo}",
|
_log.LogWarning("clientStats SPM/Skill NaN {@killInfo}",
|
||||||
new {killSPM = killSpm, KDRWeight, totalPlayTime, SPMAgainstPlayWeight, clientStats, scoreDifference});
|
new
|
||||||
|
{
|
||||||
|
killSPM = killSpm, KDRWeight, totalPlayTime, SPMAgainstPlayWeight, clientStats, scoreDifference
|
||||||
|
});
|
||||||
clientStats.SPM = 0;
|
clientStats.SPM = 0;
|
||||||
clientStats.Skill = 0;
|
clientStats.Skill = 0;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user