mirror of
https://github.com/furyfire/trueskill.git
synced 2025-01-16 01:47:39 +00:00
161 lines
5.8 KiB
PHP
161 lines
5.8 KiB
PHP
<?php
|
|
namespace Moserware\Skills\TrueSkill;
|
|
|
|
require_once(dirname(__FILE__) . '/../GameInfo.php');
|
|
require_once(dirname(__FILE__) . '/../Rating.php');
|
|
require_once(dirname(__FILE__) . '/../RatingContainer.php');
|
|
require_once(dirname(__FILE__) . '/../FactorGraphs/FactorGraph.php');
|
|
require_once(dirname(__FILE__) . '/../FactorGraphs/FactorList.php');
|
|
require_once(dirname(__FILE__) . '/../FactorGraphs/Schedule.php');
|
|
require_once(dirname(__FILE__) . '/../FactorGraphs/VariableFactory.php');
|
|
require_once(dirname(__FILE__) . '/../Numerics/GaussianDistribution.php');
|
|
require_once(dirname(__FILE__) . '/Layers/IteratedTeamDifferencesInnerLayer.php');
|
|
require_once(dirname(__FILE__) . '/Layers/PlayerPerformancesToTeamPerformancesLayer.php');
|
|
require_once(dirname(__FILE__) . '/Layers/PlayerPriorValuesToSkillsLayer.php');
|
|
require_once(dirname(__FILE__) . '/Layers/PlayerSkillsToPerformancesLayer.php');
|
|
require_once(dirname(__FILE__) . '/Layers/TeamDifferencesComparisonLayer.php');
|
|
require_once(dirname(__FILE__) . '/Layers/TeamPerformancesToTeamPerformanceDifferencesLayer.php');
|
|
|
|
use Moserware\Numerics\GaussianDistribution;
|
|
use Moserware\Skills\GameInfo;
|
|
use Moserware\Skills\Rating;
|
|
use Moserware\Skills\RatingContainer;
|
|
use Moserware\Skills\FactorGraphs\FactorGraph;
|
|
use Moserware\Skills\FactorGraphs\FactorList;
|
|
use Moserware\Skills\FactorGraphs\ScheduleSequence;
|
|
use Moserware\Skills\FactorGraphs\VariableFactory;
|
|
use Moserware\Skills\TrueSkill\Layers\IteratedTeamDifferencesInnerLayer;
|
|
use Moserware\Skills\TrueSkill\Layers\PlayerPerformancesToTeamPerformancesLayer;
|
|
use Moserware\Skills\TrueSkill\Layers\PlayerPriorValuesToSkillsLayer;
|
|
use Moserware\Skills\TrueSkill\Layers\PlayerSkillsToPerformancesLayer;
|
|
use Moserware\Skills\TrueSkill\Layers\TeamDifferencesComparisonLayer;
|
|
use Moserware\Skills\TrueSkill\Layers\TeamPerformancesToTeamPerformanceDifferencesLayer;
|
|
|
|
class TrueSkillFactorGraph extends FactorGraph
|
|
{
|
|
private $_gameInfo;
|
|
private $_layers;
|
|
private $_priorLayer;
|
|
|
|
public function __construct(GameInfo &$gameInfo, array &$teams, array $teamRanks)
|
|
{
|
|
$this->_priorLayer = new PlayerPriorValuesToSkillsLayer($this, $teams);
|
|
$this->_gameInfo = $gameInfo;
|
|
$newFactory = new VariableFactory(
|
|
function()
|
|
{
|
|
return GaussianDistribution::fromPrecisionMean(0, 0);
|
|
});
|
|
|
|
$this->setVariableFactory($newFactory);
|
|
$this->_layers = array(
|
|
$this->_priorLayer,
|
|
new PlayerSkillsToPerformancesLayer($this),
|
|
new PlayerPerformancesToTeamPerformancesLayer($this),
|
|
new IteratedTeamDifferencesInnerLayer(
|
|
$this,
|
|
new TeamPerformancesToTeamPerformanceDifferencesLayer($this),
|
|
new TeamDifferencesComparisonLayer($this, $teamRanks))
|
|
);
|
|
}
|
|
|
|
public function getGameInfo()
|
|
{
|
|
return $this->_gameInfo;
|
|
}
|
|
|
|
public function buildGraph()
|
|
{
|
|
$lastOutput = null;
|
|
|
|
$layers = &$this->_layers;
|
|
foreach ($layers as &$currentLayer)
|
|
{
|
|
if ($lastOutput != null)
|
|
{
|
|
$currentLayer->setInputVariablesGroups($lastOutput);
|
|
}
|
|
|
|
$currentLayer->buildLayer();
|
|
|
|
$lastOutput = &$currentLayer->getOutputVariablesGroups();
|
|
}
|
|
}
|
|
|
|
public function runSchedule()
|
|
{
|
|
$fullSchedule = $this->createFullSchedule();
|
|
$fullScheduleDelta = $fullSchedule->visit();
|
|
}
|
|
|
|
public function getProbabilityOfRanking()
|
|
{
|
|
$factorList = new FactorList();
|
|
|
|
$layers = &$this->_layers;
|
|
foreach ($layers as &$currentLayer)
|
|
{
|
|
$localFactors = &$currentLayer->getLocalFactors();
|
|
foreach ($localFactors as &$currentFactor)
|
|
{
|
|
$localCurrentFactor = &$currentFactor;
|
|
$factorList->addFactor($localCurrentFactor);
|
|
}
|
|
}
|
|
|
|
$logZ = $factorList->getLogNormalization();
|
|
return exp($logZ);
|
|
}
|
|
|
|
private function createFullSchedule()
|
|
{
|
|
$fullSchedule = array();
|
|
|
|
$layers = &$this->_layers;
|
|
foreach ($layers as &$currentLayer)
|
|
{
|
|
$currentPriorSchedule = $currentLayer->createPriorSchedule();
|
|
if ($currentPriorSchedule != null)
|
|
{
|
|
$fullSchedule[] = $currentPriorSchedule;
|
|
}
|
|
}
|
|
|
|
$allLayersReverse = \array_reverse($this->_layers);
|
|
|
|
foreach ($allLayersReverse as &$currentLayer)
|
|
{
|
|
$currentPosteriorSchedule = $currentLayer->createPosteriorSchedule();
|
|
if ($currentPosteriorSchedule != null)
|
|
{
|
|
$fullSchedule[] = $currentPosteriorSchedule;
|
|
}
|
|
}
|
|
|
|
return new ScheduleSequence("Full schedule", $fullSchedule);
|
|
}
|
|
|
|
public function getUpdatedRatings()
|
|
{
|
|
$result = new RatingContainer();
|
|
|
|
$priorLayerOutputVariablesGroups = &$this->_priorLayer->getOutputVariablesGroups();
|
|
foreach ($priorLayerOutputVariablesGroups as &$currentTeam)
|
|
{
|
|
foreach ($currentTeam as &$currentPlayer)
|
|
{
|
|
$localCurrentPlayer = &$currentPlayer->getKey();
|
|
$test = \spl_object_hash($localCurrentPlayer);
|
|
$newRating = new Rating($currentPlayer->getValue()->getMean(),
|
|
$currentPlayer->getValue()->getStandardDeviation());
|
|
|
|
$result->setRating($localCurrentPlayer, $newRating);
|
|
}
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
}
|
|
|
|
?>
|