Files
trueskill/src/TrueSkill/TrueSkillFactorGraph.php

147 lines
4.6 KiB
PHP
Raw Normal View History

2022-07-05 15:55:47 +02:00
<?php
2024-02-02 14:53:38 +00:00
declare(strict_types=1);
2022-07-05 15:55:47 +02:00
namespace DNW\Skills\TrueSkill;
2022-07-05 15:33:34 +02:00
use DNW\Skills\FactorGraphs\FactorGraph;
use DNW\Skills\FactorGraphs\FactorList;
use DNW\Skills\FactorGraphs\ScheduleSequence;
use DNW\Skills\FactorGraphs\VariableFactory;
2022-07-05 15:55:47 +02:00
use DNW\Skills\GameInfo;
use DNW\Skills\Numerics\GaussianDistribution;
use DNW\Skills\Rating;
2023-08-03 09:13:19 +00:00
use DNW\Skills\Team;
2022-07-05 15:55:47 +02:00
use DNW\Skills\RatingContainer;
2023-08-02 10:10:57 +00:00
use DNW\Skills\FactorGraphs\FactorGraphLayer;
2022-07-05 15:33:34 +02:00
use DNW\Skills\TrueSkill\Layers\IteratedTeamDifferencesInnerLayer;
use DNW\Skills\TrueSkill\Layers\PlayerPerformancesToTeamPerformancesLayer;
use DNW\Skills\TrueSkill\Layers\PlayerPriorValuesToSkillsLayer;
use DNW\Skills\TrueSkill\Layers\PlayerSkillsToPerformancesLayer;
use DNW\Skills\TrueSkill\Layers\TeamDifferencesComparisonLayer;
use DNW\Skills\TrueSkill\Layers\TeamPerformancesToTeamPerformanceDifferencesLayer;
class TrueSkillFactorGraph extends FactorGraph
{
2023-08-02 09:36:44 +00:00
/**
* @var FactorGraphLayer[] $layers
*/
private array $layers;
2022-07-05 15:55:47 +02:00
2023-08-02 09:36:44 +00:00
private PlayerPriorValuesToSkillsLayer $priorLayer;
2023-08-02 14:24:19 +00:00
/**
2023-08-03 09:13:19 +00:00
* @param GameInfo $gameInfo Parameters for the game.
* @param Team[] $teams A mapping of team players and their ratings.
* @param int[] $teamRanks The ranks of the teams where 1 is first place. For a tie, repeat the number (e.g. 1, 2, 2).
2023-08-02 14:24:19 +00:00
*/
2023-08-01 13:53:19 +00:00
public function __construct(private readonly GameInfo $gameInfo, array $teams, array $teamRanks)
{
2023-08-01 13:53:19 +00:00
$this->priorLayer = new PlayerPriorValuesToSkillsLayer($this, $teams);
$newFactory = new VariableFactory(
2023-08-01 13:53:19 +00:00
fn () => GaussianDistribution::fromPrecisionMean(0, 0)
);
$this->setVariableFactory($newFactory);
2023-08-01 13:53:19 +00:00
$this->layers = [
$this->priorLayer,
new PlayerSkillsToPerformancesLayer($this),
new PlayerPerformancesToTeamPerformancesLayer($this),
new IteratedTeamDifferencesInnerLayer(
$this,
new TeamPerformancesToTeamPerformanceDifferencesLayer($this),
2023-08-01 13:53:19 +00:00
new TeamDifferencesComparisonLayer($this, $teamRanks)
),
2022-07-05 15:55:47 +02:00
];
}
2023-08-02 09:36:44 +00:00
public function getGameInfo(): GameInfo
{
2023-08-01 13:53:19 +00:00
return $this->gameInfo;
}
2023-08-01 12:43:58 +00:00
public function buildGraph(): void
{
2024-02-02 14:53:38 +00:00
$lastOutput = NULL;
2023-08-01 13:53:19 +00:00
$layers = $this->layers;
foreach ($layers as $currentLayer) {
2024-02-02 14:53:38 +00:00
if ($lastOutput != NULL) {
$currentLayer->setInputVariablesGroups($lastOutput);
}
$currentLayer->buildLayer();
$lastOutput = $currentLayer->getOutputVariablesGroups();
}
}
2023-08-01 12:43:58 +00:00
public function runSchedule(): void
{
$fullSchedule = $this->createFullSchedule();
2022-07-05 16:21:06 +02:00
$fullSchedule->visit();
}
2023-08-01 12:43:58 +00:00
public function getProbabilityOfRanking(): float
{
$factorList = new FactorList();
2023-08-01 13:53:19 +00:00
$layers = $this->layers;
foreach ($layers as $currentLayer) {
$localFactors = $currentLayer->getLocalFactors();
foreach ($localFactors as $currentFactor) {
$localCurrentFactor = $currentFactor;
$factorList->addFactor($localCurrentFactor);
}
}
$logZ = $factorList->getLogNormalization();
2022-07-05 15:55:47 +02:00
return exp($logZ);
}
2023-08-01 12:43:58 +00:00
private function createFullSchedule(): ScheduleSequence
{
2022-07-05 15:55:47 +02:00
$fullSchedule = [];
2023-08-01 13:53:19 +00:00
$layers = $this->layers;
foreach ($layers as $currentLayer) {
$currentPriorSchedule = $currentLayer->createPriorSchedule();
2024-02-02 14:53:38 +00:00
if ($currentPriorSchedule != NULL) {
$fullSchedule[] = $currentPriorSchedule;
}
}
2023-08-01 13:53:19 +00:00
$allLayersReverse = array_reverse($this->layers);
foreach ($allLayersReverse as $currentLayer) {
$currentPosteriorSchedule = $currentLayer->createPosteriorSchedule();
2024-02-02 14:53:38 +00:00
if ($currentPosteriorSchedule != NULL) {
$fullSchedule[] = $currentPosteriorSchedule;
}
}
2022-07-05 15:55:47 +02:00
return new ScheduleSequence('Full schedule', $fullSchedule);
}
2023-08-01 12:13:24 +00:00
public function getUpdatedRatings(): RatingContainer
{
$result = new RatingContainer();
2023-08-01 13:53:19 +00:00
$priorLayerOutputVariablesGroups = $this->priorLayer->getOutputVariablesGroups();
foreach ($priorLayerOutputVariablesGroups as $currentTeam) {
foreach ($currentTeam as $currentPlayer) {
$localCurrentPlayer = $currentPlayer->getKey();
2023-08-01 13:53:19 +00:00
$newRating = new Rating(
$currentPlayer->getValue()->getMean(),
$currentPlayer->getValue()->getStandardDeviation()
);
$result->setRating($localCurrentPlayer, $newRating);
}
}
return $result;
}
2022-07-05 15:55:47 +02:00
}