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;
|
2010-08-28 22:05:41 -04:00
|
|
|
|
2022-07-05 15:33:34 +02:00
|
|
|
use DNW\Skills\Numerics\GaussianDistribution;
|
2010-08-28 22:05:41 -04:00
|
|
|
|
|
|
|
class TruncatedGaussianCorrectionFunctions
|
|
|
|
{
|
|
|
|
// These functions from the bottom of page 4 of the TrueSkill paper.
|
|
|
|
|
2010-10-08 21:44:36 -04:00
|
|
|
/**
|
|
|
|
* The "V" function where the team performance difference is greater than the draw margin.
|
2016-05-24 14:10:39 +02:00
|
|
|
*
|
2010-10-08 21:44:36 -04:00
|
|
|
* In the reference F# implementation, this is referred to as "the additive
|
|
|
|
* correction of a single-sided truncated Gaussian with unit variance."
|
2016-05-24 14:10:39 +02:00
|
|
|
*
|
|
|
|
* @param $teamPerformanceDifference
|
2023-08-01 13:53:19 +00:00
|
|
|
* @param $drawMargin In the paper, it's referred to as just
|
|
|
|
* "ε".
|
2016-05-24 14:10:39 +02:00
|
|
|
* @param $c
|
2010-10-08 21:44:36 -04:00
|
|
|
*/
|
2023-08-01 12:26:38 +00:00
|
|
|
public static function vExceedsMarginScaled(float $teamPerformanceDifference, float $drawMargin, float $c): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
2016-05-24 14:10:39 +02:00
|
|
|
return self::vExceedsMargin($teamPerformanceDifference / $c, $drawMargin / $c);
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
2023-08-01 12:26:38 +00:00
|
|
|
public static function vExceedsMargin(float $teamPerformanceDifference, float $drawMargin): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
|
|
|
$denominator = GaussianDistribution::cumulativeTo($teamPerformanceDifference - $drawMargin);
|
|
|
|
|
2016-05-24 14:10:39 +02:00
|
|
|
if ($denominator < 2.222758749e-162) {
|
2010-08-28 22:05:41 -04:00
|
|
|
return -$teamPerformanceDifference + $drawMargin;
|
|
|
|
}
|
|
|
|
|
2016-05-24 14:10:39 +02:00
|
|
|
return GaussianDistribution::at($teamPerformanceDifference - $drawMargin) / $denominator;
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
2010-10-08 21:44:36 -04:00
|
|
|
/**
|
|
|
|
* The "W" function where the team performance difference is greater than the draw margin.
|
2016-05-24 14:10:39 +02:00
|
|
|
*
|
2010-10-08 21:44:36 -04:00
|
|
|
* In the reference F# implementation, this is referred to as "the multiplicative
|
|
|
|
* correction of a single-sided truncated Gaussian with unit variance."
|
2016-05-24 16:31:21 +02:00
|
|
|
*
|
|
|
|
* @param $teamPerformanceDifference
|
|
|
|
* @param $drawMargin
|
|
|
|
* @param $c
|
2010-10-08 21:44:36 -04:00
|
|
|
*/
|
2023-08-01 12:26:38 +00:00
|
|
|
public static function wExceedsMarginScaled(float $teamPerformanceDifference, float $drawMargin, float $c): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
2016-05-24 14:10:39 +02:00
|
|
|
return self::wExceedsMargin($teamPerformanceDifference / $c, $drawMargin / $c);
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
2023-08-01 12:26:38 +00:00
|
|
|
public static function wExceedsMargin(float $teamPerformanceDifference, float $drawMargin): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
|
|
|
$denominator = GaussianDistribution::cumulativeTo($teamPerformanceDifference - $drawMargin);
|
|
|
|
|
2016-05-24 14:10:39 +02:00
|
|
|
if ($denominator < 2.222758749e-162) {
|
|
|
|
if ($teamPerformanceDifference < 0.0) {
|
2010-08-28 22:05:41 -04:00
|
|
|
return 1.0;
|
|
|
|
}
|
2022-07-05 15:55:47 +02:00
|
|
|
|
2010-08-28 22:05:41 -04:00
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
$vWin = self::vExceedsMargin($teamPerformanceDifference, $drawMargin);
|
2022-07-05 15:55:47 +02:00
|
|
|
|
2016-05-24 14:10:39 +02:00
|
|
|
return $vWin * ($vWin + $teamPerformanceDifference - $drawMargin);
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// the additive correction of a double-sided truncated Gaussian with unit variance
|
2023-08-01 12:26:38 +00:00
|
|
|
public static function vWithinMarginScaled(float $teamPerformanceDifference, float $drawMargin, float $c): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
2016-05-24 14:10:39 +02:00
|
|
|
return self::vWithinMargin($teamPerformanceDifference / $c, $drawMargin / $c);
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
2016-05-24 14:10:39 +02:00
|
|
|
|
2010-08-28 22:05:41 -04:00
|
|
|
// from F#:
|
2023-08-01 12:26:38 +00:00
|
|
|
public static function vWithinMargin(float $teamPerformanceDifference, float $drawMargin): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
|
|
|
$teamPerformanceDifferenceAbsoluteValue = abs($teamPerformanceDifference);
|
|
|
|
$denominator =
|
|
|
|
GaussianDistribution::cumulativeTo($drawMargin - $teamPerformanceDifferenceAbsoluteValue) -
|
|
|
|
GaussianDistribution::cumulativeTo(-$drawMargin - $teamPerformanceDifferenceAbsoluteValue);
|
|
|
|
|
2016-05-24 14:10:39 +02:00
|
|
|
if ($denominator < 2.222758749e-162) {
|
|
|
|
if ($teamPerformanceDifference < 0.0) {
|
2010-08-28 22:05:41 -04:00
|
|
|
return -$teamPerformanceDifference - $drawMargin;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -$teamPerformanceDifference + $drawMargin;
|
|
|
|
}
|
|
|
|
|
|
|
|
$numerator = GaussianDistribution::at(-$drawMargin - $teamPerformanceDifferenceAbsoluteValue) -
|
2016-05-24 14:10:39 +02:00
|
|
|
GaussianDistribution::at($drawMargin - $teamPerformanceDifferenceAbsoluteValue);
|
2010-08-28 22:05:41 -04:00
|
|
|
|
2016-05-24 14:10:39 +02:00
|
|
|
if ($teamPerformanceDifference < 0.0) {
|
|
|
|
return -$numerator / $denominator;
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
2016-05-24 14:10:39 +02:00
|
|
|
return $numerator / $denominator;
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// the multiplicative correction of a double-sided truncated Gaussian with unit variance
|
2023-08-01 12:26:38 +00:00
|
|
|
public static function wWithinMarginScaled(float $teamPerformanceDifference, float $drawMargin, float $c): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
2016-05-24 14:10:39 +02:00
|
|
|
return self::wWithinMargin($teamPerformanceDifference / $c, $drawMargin / $c);
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// From F#:
|
2023-08-01 12:26:38 +00:00
|
|
|
public static function wWithinMargin(float $teamPerformanceDifference, float $drawMargin): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
|
|
|
$teamPerformanceDifferenceAbsoluteValue = abs($teamPerformanceDifference);
|
|
|
|
$denominator = GaussianDistribution::cumulativeTo($drawMargin - $teamPerformanceDifferenceAbsoluteValue)
|
2016-05-24 14:10:39 +02:00
|
|
|
-
|
|
|
|
GaussianDistribution::cumulativeTo(-$drawMargin - $teamPerformanceDifferenceAbsoluteValue);
|
2010-08-28 22:05:41 -04:00
|
|
|
|
2016-05-24 14:10:39 +02:00
|
|
|
if ($denominator < 2.222758749e-162) {
|
2010-08-28 22:05:41 -04:00
|
|
|
return 1.0;
|
|
|
|
}
|
|
|
|
|
2010-08-29 16:40:03 -04:00
|
|
|
$vt = self::vWithinMargin($teamPerformanceDifferenceAbsoluteValue, $drawMargin);
|
2010-08-28 22:05:41 -04:00
|
|
|
|
2016-05-24 14:10:39 +02:00
|
|
|
return $vt * $vt +
|
|
|
|
(
|
|
|
|
($drawMargin - $teamPerformanceDifferenceAbsoluteValue)
|
|
|
|
*
|
|
|
|
GaussianDistribution::at(
|
2023-08-01 13:53:19 +00:00
|
|
|
$drawMargin - $teamPerformanceDifferenceAbsoluteValue
|
|
|
|
)
|
2016-05-24 14:10:39 +02:00
|
|
|
- (-$drawMargin - $teamPerformanceDifferenceAbsoluteValue)
|
|
|
|
*
|
|
|
|
GaussianDistribution::at(-$drawMargin - $teamPerformanceDifferenceAbsoluteValue)) / $denominator;
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
2022-07-05 15:55:47 +02:00
|
|
|
}
|