trueskill/Skills/TrueSkill/TrueSkillFactorGraph.cs

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;
}
}
}