mirror of
https://github.com/furyfire/trueskill.git
synced 2025-01-16 17:59:29 +00:00
182 lines
7.2 KiB
PHP
182 lines
7.2 KiB
PHP
<?php namespace Moserware\Skills\TrueSkill;
|
|
|
|
use Moserware\Skills\GameInfo;
|
|
use Moserware\Skills\Guard;
|
|
use Moserware\Skills\ISupportPartialPlay;
|
|
use Moserware\Skills\ISupportPartialUpdate;
|
|
use Moserware\Skills\Numerics\BasicMath;
|
|
use Moserware\Skills\Numerics\DiagonalMatrix;
|
|
use Moserware\Skills\Numerics\Matrix;
|
|
use Moserware\Skills\Numerics\Vector;
|
|
use Moserware\Skills\PartialPlay;
|
|
use Moserware\Skills\PlayersRange;
|
|
use Moserware\Skills\RankSorter;
|
|
use Moserware\Skills\SkillCalculator;
|
|
use Moserware\Skills\SkillCalculatorSupportedOptions;
|
|
use Moserware\Skills\TeamsRange;
|
|
|
|
/**
|
|
* Calculates TrueSkill using a full factor graph.
|
|
*/
|
|
class FactorGraphTrueSkillCalculator extends SkillCalculator
|
|
{
|
|
public function __construct()
|
|
{
|
|
parent::__construct(SkillCalculatorSupportedOptions::PARTIAL_PLAY | SkillCalculatorSupportedOptions::PARTIAL_UPDATE, TeamsRange::atLeast(2), PlayersRange::atLeast(1));
|
|
}
|
|
|
|
public function calculateNewRatings(GameInfo $gameInfo,
|
|
array $teams,
|
|
array $teamRanks)
|
|
{
|
|
Guard::argumentNotNull($gameInfo, "gameInfo");
|
|
$this->validateTeamCountAndPlayersCountPerTeam($teams);
|
|
|
|
RankSorter::sort($teams, $teamRanks);
|
|
|
|
$factorGraph = new TrueSkillFactorGraph($gameInfo, $teams, $teamRanks);
|
|
$factorGraph->buildGraph();
|
|
$factorGraph->runSchedule();
|
|
|
|
$probabilityOfOutcome = $factorGraph->getProbabilityOfRanking();
|
|
|
|
return $factorGraph->getUpdatedRatings();
|
|
}
|
|
|
|
public function calculateMatchQuality(GameInfo $gameInfo, array $teams)
|
|
{
|
|
// We need to create the A matrix which is the player team assigments.
|
|
$teamAssignmentsList = $teams;
|
|
$skillsMatrix = $this->getPlayerCovarianceMatrix($teamAssignmentsList);
|
|
$meanVector = $this->getPlayerMeansVector($teamAssignmentsList);
|
|
$meanVectorTranspose = $meanVector->getTranspose();
|
|
|
|
$playerTeamAssignmentsMatrix = $this->createPlayerTeamAssignmentMatrix($teamAssignmentsList, $meanVector->getRowCount());
|
|
$playerTeamAssignmentsMatrixTranspose = $playerTeamAssignmentsMatrix->getTranspose();
|
|
|
|
$betaSquared = BasicMath::square($gameInfo->getBeta());
|
|
|
|
$start = Matrix::multiply($meanVectorTranspose, $playerTeamAssignmentsMatrix);
|
|
|
|
$aTa = Matrix::multiply(
|
|
Matrix::scalarMultiply($betaSquared, $playerTeamAssignmentsMatrixTranspose),
|
|
$playerTeamAssignmentsMatrix
|
|
);
|
|
|
|
$aTSA = Matrix::multiply(
|
|
Matrix::multiply($playerTeamAssignmentsMatrixTranspose, $skillsMatrix),
|
|
$playerTeamAssignmentsMatrix
|
|
);
|
|
|
|
$middle = Matrix::add($aTa, $aTSA);
|
|
|
|
$middleInverse = $middle->getInverse();
|
|
|
|
$end = Matrix::multiply($playerTeamAssignmentsMatrixTranspose, $meanVector);
|
|
|
|
$expPartMatrix = Matrix::scalarMultiply(-0.5, (Matrix::multiply(Matrix::multiply($start, $middleInverse), $end)));
|
|
$expPart = $expPartMatrix->getDeterminant();
|
|
|
|
$sqrtPartNumerator = $aTa->getDeterminant();
|
|
$sqrtPartDenominator = $middle->getDeterminant();
|
|
$sqrtPart = $sqrtPartNumerator / $sqrtPartDenominator;
|
|
|
|
$result = exp($expPart) * sqrt($sqrtPart);
|
|
|
|
return $result;
|
|
}
|
|
|
|
private static function getPlayerMeansVector(array $teamAssignmentsList)
|
|
{
|
|
// A simple vector of all the player means.
|
|
return new Vector(self::getPlayerRatingValues($teamAssignmentsList,
|
|
function ($rating) {
|
|
return $rating->getMean();
|
|
}));
|
|
}
|
|
|
|
private static function getPlayerCovarianceMatrix(array $teamAssignmentsList)
|
|
{
|
|
// This is a square matrix whose diagonal values represent the variance (square of standard deviation) of all
|
|
// players.
|
|
return new DiagonalMatrix(
|
|
self::getPlayerRatingValues($teamAssignmentsList,
|
|
function ($rating) {
|
|
return BasicMath::square($rating->getStandardDeviation());
|
|
}));
|
|
}
|
|
|
|
// Helper function that gets a list of values for all player ratings
|
|
private static function getPlayerRatingValues(array $teamAssignmentsList, $playerRatingFunction)
|
|
{
|
|
$playerRatingValues = array();
|
|
|
|
foreach ($teamAssignmentsList as $currentTeam) {
|
|
foreach ($currentTeam->getAllRatings() as $currentRating) {
|
|
$playerRatingValues[] = $playerRatingFunction($currentRating);
|
|
}
|
|
}
|
|
|
|
return $playerRatingValues;
|
|
}
|
|
|
|
private static function createPlayerTeamAssignmentMatrix($teamAssignmentsList, $totalPlayers)
|
|
{
|
|
// The team assignment matrix is often referred to as the "A" matrix. It's a matrix whose rows represent the players
|
|
// and the columns represent teams. At Matrix[row, column] represents that player[row] is on team[col]
|
|
// Positive values represent an assignment and a negative value means that we subtract the value of the next
|
|
// team since we're dealing with pairs. This means that this matrix always has teams - 1 columns.
|
|
// The only other tricky thing is that values represent the play percentage.
|
|
|
|
// For example, consider a 3 team game where team1 is just player1, team 2 is player 2 and player 3, and
|
|
// team3 is just player 4. Furthermore, player 2 and player 3 on team 2 played 25% and 75% of the time
|
|
// (e.g. partial play), the A matrix would be:
|
|
|
|
// A = this 4x2 matrix:
|
|
// | 1.00 0.00 |
|
|
// | -0.25 0.25 |
|
|
// | -0.75 0.75 |
|
|
// | 0.00 -1.00 |
|
|
|
|
$playerAssignments = array();
|
|
$totalPreviousPlayers = 0;
|
|
|
|
$teamAssignmentsListCount = count($teamAssignmentsList);
|
|
|
|
$currentColumn = 0;
|
|
|
|
for ($i = 0; $i < $teamAssignmentsListCount - 1; $i++) {
|
|
$currentTeam = $teamAssignmentsList[$i];
|
|
|
|
// Need to add in 0's for all the previous players, since they're not
|
|
// on this team
|
|
$playerAssignments[$currentColumn] = ($totalPreviousPlayers > 0) ? \array_fill(0, $totalPreviousPlayers, 0) : array();
|
|
|
|
foreach ($currentTeam->getAllPlayers() as $currentPlayer) {
|
|
$playerAssignments[$currentColumn][] = PartialPlay::getPartialPlayPercentage($currentPlayer);
|
|
// indicates the player is on the team
|
|
$totalPreviousPlayers++;
|
|
}
|
|
|
|
$rowsRemaining = $totalPlayers - $totalPreviousPlayers;
|
|
|
|
$nextTeam = $teamAssignmentsList[$i + 1];
|
|
foreach ($nextTeam->getAllPlayers() as $nextTeamPlayer) {
|
|
// Add a -1 * playing time to represent the difference
|
|
$playerAssignments[$currentColumn][] = -1 * PartialPlay::getPartialPlayPercentage($nextTeamPlayer);
|
|
$rowsRemaining--;
|
|
}
|
|
|
|
for ($ixAdditionalRow = 0; $ixAdditionalRow < $rowsRemaining; $ixAdditionalRow++) {
|
|
// Pad with zeros
|
|
$playerAssignments[$currentColumn][] = 0;
|
|
}
|
|
|
|
$currentColumn++;
|
|
}
|
|
|
|
$playerTeamAssignmentsMatrix = Matrix::fromColumnValues($totalPlayers, $teamAssignmentsListCount - 1, $playerAssignments);
|
|
|
|
return $playerTeamAssignmentsMatrix;
|
|
}
|
|
} |