mirror of
https://github.com/furyfire/trueskill.git
synced 2025-10-24 06:22:30 +02:00
119 lines
4.3 KiB
C#
119 lines
4.3 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Moserware.Numerics;
|
|
using Moserware.Skills.FactorGraphs;
|
|
using Moserware.Skills.TrueSkill.Layers;
|
|
|
|
namespace Moserware.Skills.TrueSkill
|
|
{
|
|
public class TrueSkillFactorGraph<TPlayer> :
|
|
FactorGraph<TrueSkillFactorGraph<TPlayer>, GaussianDistribution, Variable<GaussianDistribution>>
|
|
{
|
|
private readonly List<FactorGraphLayerBase<GaussianDistribution>> _Layers;
|
|
private readonly PlayerPriorValuesToSkillsLayer<TPlayer> _PriorLayer;
|
|
|
|
public TrueSkillFactorGraph(GameInfo gameInfo, IEnumerable<IDictionary<TPlayer, Rating>> teams, int[] teamRanks)
|
|
{
|
|
_PriorLayer = new PlayerPriorValuesToSkillsLayer<TPlayer>(this, teams);
|
|
GameInfo = gameInfo;
|
|
VariableFactory =
|
|
new VariableFactory<GaussianDistribution>(() => GaussianDistribution.FromPrecisionMean(0, 0));
|
|
|
|
_Layers = new List<FactorGraphLayerBase<GaussianDistribution>>
|
|
{
|
|
_PriorLayer,
|
|
new PlayerSkillsToPerformancesLayer<TPlayer>(this),
|
|
new PlayerPerformancesToTeamPerformancesLayer<TPlayer>(this),
|
|
new IteratedTeamDifferencesInnerLayer<TPlayer>(
|
|
this,
|
|
new TeamPerformancesToTeamPerformanceDifferencesLayer<TPlayer>(this),
|
|
new TeamDifferencesComparisonLayer<TPlayer>(this, teamRanks))
|
|
};
|
|
}
|
|
|
|
public GameInfo GameInfo { get; private set; }
|
|
|
|
public void BuildGraph()
|
|
{
|
|
object lastOutput = null;
|
|
|
|
foreach (var currentLayer in _Layers)
|
|
{
|
|
if (lastOutput != null)
|
|
{
|
|
currentLayer.SetRawInputVariablesGroups(lastOutput);
|
|
}
|
|
|
|
currentLayer.BuildLayer();
|
|
|
|
lastOutput = currentLayer.GetRawOutputVariablesGroups();
|
|
}
|
|
}
|
|
|
|
public void RunSchedule()
|
|
{
|
|
Schedule<GaussianDistribution> fullSchedule = CreateFullSchedule();
|
|
double fullScheduleDelta = fullSchedule.Visit();
|
|
}
|
|
|
|
public double GetProbabilityOfRanking()
|
|
{
|
|
var factorList = new FactorList<GaussianDistribution>();
|
|
|
|
foreach (var currentLayer in _Layers)
|
|
{
|
|
foreach (var currentFactor in currentLayer.UntypedFactors)
|
|
{
|
|
factorList.AddFactor(currentFactor);
|
|
}
|
|
}
|
|
|
|
double logZ = factorList.LogNormalization;
|
|
return Math.Exp(logZ);
|
|
}
|
|
|
|
private Schedule<GaussianDistribution> CreateFullSchedule()
|
|
{
|
|
var fullSchedule = new List<Schedule<GaussianDistribution>>();
|
|
|
|
foreach (var currentLayer in _Layers)
|
|
{
|
|
Schedule<GaussianDistribution> currentPriorSchedule = currentLayer.CreatePriorSchedule();
|
|
if (currentPriorSchedule != null)
|
|
{
|
|
fullSchedule.Add(currentPriorSchedule);
|
|
}
|
|
}
|
|
|
|
// Casting to IEnumerable to get the LINQ Reverse()
|
|
IEnumerable<FactorGraphLayerBase<GaussianDistribution>> allLayers = _Layers;
|
|
|
|
foreach (var currentLayer in allLayers.Reverse())
|
|
{
|
|
Schedule<GaussianDistribution> currentPosteriorSchedule = currentLayer.CreatePosteriorSchedule();
|
|
if (currentPosteriorSchedule != null)
|
|
{
|
|
fullSchedule.Add(currentPosteriorSchedule);
|
|
}
|
|
}
|
|
|
|
return new ScheduleSequence<GaussianDistribution>("Full schedule", fullSchedule);
|
|
}
|
|
|
|
public IDictionary<TPlayer, Rating> GetUpdatedRatings()
|
|
{
|
|
var result = new Dictionary<TPlayer, Rating>();
|
|
foreach (var currentTeam in _PriorLayer.OutputVariablesGroups)
|
|
{
|
|
foreach (var currentPlayer in currentTeam)
|
|
{
|
|
result[currentPlayer.Key] = new Rating(currentPlayer.Value.Mean,
|
|
currentPlayer.Value.StandardDeviation);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
}
|
|
} |