1
0
mirror of https://github.com/avitex/elixir-glicko synced 2025-01-24 23:09:58 +00:00

Compare commits

...

5 Commits

Author SHA1 Message Date
Mikael Muszynski
6e22fcf0b0 Add .credo.exs to .formatter.exs (#4)
Additionally, run `mix format`.
2020-01-19 14:57:32 +00:00
Mikael Muszynski
db54b10e93 Change multi-line do: function definitions to do (#3)
Change the cases where a `do:` is followed by a `mix format` mandated
line break, or if the expression following the `do:` would span multiple
lines anyway.

The only exception to the rule above is the first change in the
changeset, as this was done for the function style of `initial_rating`
to be consistent with `initial_rating_deviation`.
2020-01-19 14:54:18 +00:00
Mikael Muszynski
1175f6b2c2 Update Elixir version (#2)
Update the project wide Elixir requirement to the latest stable version.
Additionally, update the OTP version in Travis to `v22.1`.
2020-01-19 14:43:22 +00:00
James Dyson
452eafc981
Update README and LICENSE 2020-01-19 14:24:52 +00:00
Mikael Muszynski
e22c34f2ff Run mix format (#1)
Add formatter configuration and run `mix format`.

Additionally, manually replace leading tabs in documentation examples
with four spaces.
2020-01-19 13:58:40 +00:00
12 changed files with 727 additions and 612 deletions

View File

@ -1,5 +1,6 @@
%{ %{
configs: [%{ configs: [
%{
name: "default", name: "default",
files: %{ files: %{
included: ["lib/", "test/"], included: ["lib/", "test/"],
@ -12,12 +13,10 @@
{Credo.Check.Consistency.SpaceAroundOperators}, {Credo.Check.Consistency.SpaceAroundOperators},
{Credo.Check.Consistency.SpaceInParentheses}, {Credo.Check.Consistency.SpaceInParentheses},
{Credo.Check.Consistency.TabsOrSpaces}, {Credo.Check.Consistency.TabsOrSpaces},
{Credo.Check.Design.AliasUsage, priority: :low}, {Credo.Check.Design.AliasUsage, priority: :low},
{Credo.Check.Design.DuplicatedCode, excluded_macros: []}, {Credo.Check.Design.DuplicatedCode, excluded_macros: []},
{Credo.Check.Design.TagTODO, exit_status: 2}, {Credo.Check.Design.TagTODO, exit_status: 2},
{Credo.Check.Design.TagFIXME}, {Credo.Check.Design.TagFIXME},
{Credo.Check.Readability.FunctionNames}, {Credo.Check.Readability.FunctionNames},
{Credo.Check.Readability.LargeNumbers}, {Credo.Check.Readability.LargeNumbers},
{Credo.Check.Readability.MaxLineLength, priority: :low, max_length: 80}, {Credo.Check.Readability.MaxLineLength, priority: :low, max_length: 80},
@ -35,7 +34,6 @@
{Credo.Check.Readability.VariableNames}, {Credo.Check.Readability.VariableNames},
{Credo.Check.Readability.Semicolons}, {Credo.Check.Readability.Semicolons},
{Credo.Check.Readability.SpaceAfterCommas}, {Credo.Check.Readability.SpaceAfterCommas},
{Credo.Check.Refactor.DoubleBooleanNegation}, {Credo.Check.Refactor.DoubleBooleanNegation},
{Credo.Check.Refactor.CondStatements}, {Credo.Check.Refactor.CondStatements},
{Credo.Check.Refactor.CyclomaticComplexity}, {Credo.Check.Refactor.CyclomaticComplexity},
@ -47,7 +45,6 @@
{Credo.Check.Refactor.Nesting}, {Credo.Check.Refactor.Nesting},
{Credo.Check.Refactor.PipeChainStart}, {Credo.Check.Refactor.PipeChainStart},
{Credo.Check.Refactor.UnlessWithElse}, {Credo.Check.Refactor.UnlessWithElse},
{Credo.Check.Warning.BoolOperationOnSameValues}, {Credo.Check.Warning.BoolOperationOnSameValues},
{Credo.Check.Warning.IExPry}, {Credo.Check.Warning.IExPry},
{Credo.Check.Warning.IoInspect}, {Credo.Check.Warning.IoInspect},
@ -62,7 +59,8 @@
{Credo.Check.Warning.UnusedRegexOperation}, {Credo.Check.Warning.UnusedRegexOperation},
{Credo.Check.Warning.UnusedStringOperation}, {Credo.Check.Warning.UnusedStringOperation},
{Credo.Check.Warning.UnusedTupleOperation}, {Credo.Check.Warning.UnusedTupleOperation},
{Credo.Check.Warning.RaiseInsideRescue}, {Credo.Check.Warning.RaiseInsideRescue}
]
}
] ]
}]
} }

3
.formatter.exs Normal file
View File

@ -0,0 +1,3 @@
[
inputs: ["{mix,.formatter,.credo}.exs", "{config,lib,test}/**/*.{ex,exs}"]
]

View File

@ -1,11 +1,11 @@
language: elixir language: elixir
elixir: elixir:
- 1.5.0 - 1.9.4
notifications: notifications:
recipients: recipients:
- theavitex@gmail.com - theavitex@gmail.com
otp_release: otp_release:
- 18.2 - 22.1
env: env:
- MIX_ENV=test - MIX_ENV=test
script: script:

View File

@ -1,6 +1,6 @@
MIT License MIT License
Copyright (c) 2017 James Dyson Copyright (c) 2017-2020 James Dyson <theavitex@gmail.com>
Permission is hereby granted, free of charge, to any person obtaining a copy Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal of this software and associated documentation files (the "Software"), to deal

View File

@ -10,10 +10,10 @@ Documentation hosted on [hexdocs](https://hexdocs.pm/glicko).
## Installation ## Installation
Add `glicko` to your list of dependencies in `mix.exs`: Add `glicko` to your list of dependencies in `mix.exs`:
```elixir ```elixir
def deps do def deps do
[{:glicko, "~> 0.6.0"}] [{:glicko, "~> 0.6.0"}]
end end
``` ```

View File

@ -39,7 +39,7 @@ defmodule Glicko do
alias __MODULE__.{ alias __MODULE__.{
Player, Player,
Result, Result
} }
@default_system_constant 0.8 @default_system_constant 0.8
@ -52,9 +52,13 @@ defmodule Glicko do
Returns a value between `0.0` and `1.0`. Returns a value between `0.0` and `1.0`.
""" """
@spec win_probability(player :: Player.t, opponent :: Player.t) :: float @spec win_probability(player :: Player.t(), opponent :: Player.t()) :: float
def win_probability(player, opponent) do def win_probability(player, opponent) do
win_probability(player |> Player.rating(:v2), opponent |> Player.rating(:v2), opponent |> Player.rating_deviation(:v2)) win_probability(
player |> Player.rating(:v2),
opponent |> Player.rating(:v2),
opponent |> Player.rating_deviation(:v2)
)
end end
@doc """ @doc """
@ -64,7 +68,11 @@ defmodule Glicko do
Returns a value between `0.0` and `1.0`. Returns a value between `0.0` and `1.0`.
""" """
@spec win_probability(player_rating :: Player.rating, opponent_rating :: Player.rating, opponent_rating_deviation :: Player.rating_deviation) :: float @spec win_probability(
player_rating :: Player.rating(),
opponent_rating :: Player.rating(),
opponent_rating_deviation :: Player.rating_deviation()
) :: float
def win_probability(player_rating, opponent_rating, opponent_rating_deviation) do def win_probability(player_rating, opponent_rating, opponent_rating_deviation) do
calc_e(player_rating, opponent_rating, calc_g(opponent_rating_deviation)) calc_e(player_rating, opponent_rating, calc_g(opponent_rating_deviation))
end end
@ -74,9 +82,13 @@ defmodule Glicko do
Returns a value between `0.0` and `1.0`. Returns a value between `0.0` and `1.0`.
""" """
@spec draw_probability(player :: Player.t, opponent :: Player.t) :: float @spec draw_probability(player :: Player.t(), opponent :: Player.t()) :: float
def draw_probability(player, opponent) do def draw_probability(player, opponent) do
draw_probability(player |> Player.rating(:v2), opponent |> Player.rating(:v2), opponent |> Player.rating_deviation(:v2)) draw_probability(
player |> Player.rating(:v2),
opponent |> Player.rating(:v2),
opponent |> Player.rating_deviation(:v2)
)
end end
@doc """ @doc """
@ -86,9 +98,14 @@ defmodule Glicko do
Returns a value between `0.0` and `1.0`. Returns a value between `0.0` and `1.0`.
""" """
@spec draw_probability(player_rating :: Player.rating, opponent_rating :: Player.rating, opponent_rating_deviation :: Player.rating_deviation) :: float @spec draw_probability(
player_rating :: Player.rating(),
opponent_rating :: Player.rating(),
opponent_rating_deviation :: Player.rating_deviation()
) :: float
def draw_probability(player_rating, opponent_rating, opponent_rating_deviation) do def draw_probability(player_rating, opponent_rating, opponent_rating_deviation) do
1 - abs(win_probability(player_rating, opponent_rating, opponent_rating_deviation) - 0.5) / 0.5 1 -
abs(win_probability(player_rating, opponent_rating, opponent_rating_deviation) - 0.5) / 0.5
end end
@doc """ @doc """
@ -96,16 +113,19 @@ defmodule Glicko do
Returns the updated player with the same version given to the function. Returns the updated player with the same version given to the function.
""" """
@spec new_rating(player :: Player.t, results :: list(Result.t), opts :: new_rating_opts) :: Player.t @spec new_rating(player :: Player.t(), results :: list(Result.t()), opts :: new_rating_opts) ::
Player.t()
def new_rating(player, results, opts \\ []) def new_rating(player, results, opts \\ [])
def new_rating(player, results, opts) when tuple_size(player) == 3 do def new_rating(player, results, opts) when tuple_size(player) == 3 do
do_new_rating(player, results, opts) do_new_rating(player, results, opts)
end end
def new_rating(player, results, opts) when tuple_size(player) == 2 do def new_rating(player, results, opts) when tuple_size(player) == 2 do
player player
|> Player.to_v2 |> Player.to_v2()
|> do_new_rating(results, opts) |> do_new_rating(results, opts)
|> Player.to_v1 |> Player.to_v1()
end end
defp do_new_rating({player_r, player_pre_rd, player_v}, [], _) do defp do_new_rating({player_r, player_pre_rd, player_v}, [], _) do
@ -113,6 +133,7 @@ defmodule Glicko do
{player_r, player_post_rd, player_v} {player_r, player_post_rd, player_v}
end end
defp do_new_rating({player_pre_r, player_pre_rd, player_pre_v}, results, opts) do defp do_new_rating({player_pre_r, player_pre_rd, player_pre_v}, results, opts) do
sys_const = Keyword.get(opts, :system_constant, @default_system_constant) sys_const = Keyword.get(opts, :system_constant, @default_system_constant)
conv_tol = Keyword.get(opts, :convergence_tolerance, @default_convergence_tolerance) conv_tol = Keyword.get(opts, :convergence_tolerance, @default_convergence_tolerance)
@ -126,17 +147,35 @@ defmodule Glicko do
alpha = calc_alpha(player_pre_v) alpha = calc_alpha(player_pre_v)
# Step 5.2 # Step 5.2
k = calc_k(alpha, delta, player_pre_rd_sq, variance_est, sys_const, 1) k = calc_k(alpha, delta, player_pre_rd_sq, variance_est, sys_const, 1)
{initial_a, initial_b} = iterative_algorithm_initial(
alpha, delta, player_pre_rd_sq, variance_est, sys_const, k {initial_a, initial_b} =
iterative_algorithm_initial(
alpha,
delta,
player_pre_rd_sq,
variance_est,
sys_const,
k
) )
# Step 5.3 # Step 5.3
initial_fa = calc_f(alpha, delta, player_pre_rd_sq, variance_est, sys_const, initial_a) initial_fa = calc_f(alpha, delta, player_pre_rd_sq, variance_est, sys_const, initial_a)
initial_fb = calc_f(alpha, delta, player_pre_rd_sq, variance_est, sys_const, initial_b) initial_fb = calc_f(alpha, delta, player_pre_rd_sq, variance_est, sys_const, initial_b)
# Step 5.4 # Step 5.4
a = iterative_algorithm_body( a =
alpha, delta, player_pre_rd_sq, variance_est, sys_const, conv_tol, iterative_algorithm_body(
initial_a, initial_b, initial_fa, initial_fb alpha,
delta,
player_pre_rd_sq,
variance_est,
sys_const,
conv_tol,
initial_a,
initial_b,
initial_fa,
initial_fb
) )
# Step 5.5 # Step 5.5
player_post_v = calc_new_player_volatility(a) player_post_v = calc_new_player_volatility(a)
# Step 6 # Step 6
@ -153,13 +192,14 @@ defmodule Glicko do
Enum.reduce(results, {0.0, 0.0}, fn result, {variance_estimate_acc, result_effect_acc} -> Enum.reduce(results, {0.0, 0.0}, fn result, {variance_estimate_acc, result_effect_acc} ->
opponent_rd_g = opponent_rd_g =
result result
|> Result.opponent_rating_deviation |> Result.opponent_rating_deviation()
|> calc_g |> calc_g
win_probability = calc_e(player_pre_r, Result.opponent_rating(result), opponent_rd_g) win_probability = calc_e(player_pre_r, Result.opponent_rating(result), opponent_rd_g)
{ {
variance_estimate_acc + :math.pow(opponent_rd_g, 2) * win_probability * (1 - win_probability), variance_estimate_acc +
:math.pow(opponent_rd_g, 2) * win_probability * (1 - win_probability),
result_effect_acc + opponent_rd_g * (Result.score(result) - win_probability) result_effect_acc + opponent_rd_g * (Result.score(result) - win_probability)
} }
end) end)
@ -195,11 +235,12 @@ defmodule Glicko do
end end
defp calc_player_post_base_rd(player_pre_rd_sq, player_pre_v) do defp calc_player_post_base_rd(player_pre_rd_sq, player_pre_v) do
:math.sqrt((:math.pow(player_pre_v, 2) + player_pre_rd_sq)) :math.sqrt(:math.pow(player_pre_v, 2) + player_pre_rd_sq)
end end
defp iterative_algorithm_initial(alpha, delta, player_pre_rd_sq, variance_est, sys_const, k) do defp iterative_algorithm_initial(alpha, delta, player_pre_rd_sq, variance_est, sys_const, k) do
initial_a = alpha initial_a = alpha
initial_b = initial_b =
if :math.pow(delta, 2) > player_pre_rd_sq + variance_est do if :math.pow(delta, 2) > player_pre_rd_sq + variance_est do
:math.log(:math.pow(delta, 2) - player_pre_rd_sq - variance_est) :math.log(:math.pow(delta, 2) - player_pre_rd_sq - variance_est)
@ -210,17 +251,41 @@ defmodule Glicko do
{initial_a, initial_b} {initial_a, initial_b}
end end
defp iterative_algorithm_body(alpha, delta, player_pre_rd_sq, variance_est, sys_const, conv_tol, a, b, fa, fb) do defp iterative_algorithm_body(
alpha,
delta,
player_pre_rd_sq,
variance_est,
sys_const,
conv_tol,
a,
b,
fa,
fb
) do
if abs(b - a) > conv_tol do if abs(b - a) > conv_tol do
c = a + (a - b) * fa / (fb - fa) c = a + (a - b) * fa / (fb - fa)
fc = calc_f(alpha, delta, player_pre_rd_sq, variance_est, sys_const, c) fc = calc_f(alpha, delta, player_pre_rd_sq, variance_est, sys_const, c)
{a, fa} = {a, fa} =
if fc * fb < 0 do if fc * fb < 0 do
{b, fb} {b, fb}
else else
{a, fa / 2} {a, fa / 2}
end end
iterative_algorithm_body(alpha, delta, player_pre_rd_sq, variance_est, sys_const, conv_tol, a, c, fa, fc)
iterative_algorithm_body(
alpha,
delta,
player_pre_rd_sq,
variance_est,
sys_const,
conv_tol,
a,
c,
fa,
fc
)
else else
a a
end end
@ -236,7 +301,7 @@ defmodule Glicko do
# g function # g function
defp calc_g(rd) do defp calc_g(rd) do
1 / :math.sqrt(1 + 3 * :math.pow(rd, 2) / :math.pow(:math.pi, 2)) 1 / :math.sqrt(1 + 3 * :math.pow(rd, 2) / :math.pow(:math.pi(), 2))
end end
# E function # E function

View File

@ -56,14 +56,20 @@ defmodule Glicko.Player do
""" """
@spec initial_rating(version) :: rating @spec initial_rating(version) :: rating
def initial_rating(_version = :v1), do: 1500.0 def initial_rating(_version = :v1), do: 1500.0
def initial_rating(_version = :v2), do: :v1 |> initial_rating |> scale_rating_to(:v2)
def initial_rating(_version = :v2) do
:v1 |> initial_rating |> scale_rating_to(:v2)
end
@doc """ @doc """
The recommended initial rating deviation value for a new player. The recommended initial rating deviation value for a new player.
""" """
@spec initial_rating_deviation(version) :: rating_deviation @spec initial_rating_deviation(version) :: rating_deviation
def initial_rating_deviation(_version = :v1), do: 350.0 def initial_rating_deviation(_version = :v1), do: 350.0
def initial_rating_deviation(_version = :v2), do: :v1 |> initial_rating_deviation |> scale_rating_deviation_to(:v2)
def initial_rating_deviation(_version = :v2) do
:v1 |> initial_rating_deviation |> scale_rating_deviation_to(:v2)
end
@doc """ @doc """
The recommended initial volatility value for a new player. The recommended initial volatility value for a new player.
@ -76,23 +82,27 @@ defmodule Glicko.Player do
If not overriden, will use the default values for an unrated player. If not overriden, will use the default values for an unrated player.
""" """
@spec new_v1([rating: rating, rating_deviation: rating_deviation]) :: v1 @spec new_v1(rating: rating, rating_deviation: rating_deviation) :: v1
def new_v1(opts \\ []) when is_list(opts), do: { def new_v1(opts \\ []) when is_list(opts) do
{
Keyword.get(opts, :rating, initial_rating(:v1)), Keyword.get(opts, :rating, initial_rating(:v1)),
Keyword.get(opts, :rating_deviation, initial_rating_deviation(:v1)), Keyword.get(opts, :rating_deviation, initial_rating_deviation(:v1))
} }
end
@doc """ @doc """
Creates a new v2 player. Creates a new v2 player.
If not overriden, will use default values for an unrated player. If not overriden, will use default values for an unrated player.
""" """
@spec new_v2([rating: rating, rating_deviation: rating_deviation, volatility: volatility]) :: v2 @spec new_v2(rating: rating, rating_deviation: rating_deviation, volatility: volatility) :: v2
def new_v2(opts \\ []) when is_list(opts), do: { def new_v2(opts \\ []) when is_list(opts) do
{
Keyword.get(opts, :rating, initial_rating(:v2)), Keyword.get(opts, :rating, initial_rating(:v2)),
Keyword.get(opts, :rating_deviation, initial_rating_deviation(:v2)), Keyword.get(opts, :rating_deviation, initial_rating_deviation(:v2)),
Keyword.get(opts, :volatility, initial_volatility()), Keyword.get(opts, :volatility, initial_volatility())
} }
end
@doc """ @doc """
Converts a v2 player to a v1. Converts a v2 player to a v1.
@ -103,10 +113,13 @@ defmodule Glicko.Player do
""" """
@spec to_v1(player :: t) :: v1 @spec to_v1(player :: t) :: v1
def to_v1({rating, rating_deviation}), do: {rating, rating_deviation} def to_v1({rating, rating_deviation}), do: {rating, rating_deviation}
def to_v1({rating, rating_deviation, _}), do: {
def to_v1({rating, rating_deviation, _}) do
{
rating |> scale_rating_to(:v1), rating |> scale_rating_to(:v1),
rating_deviation |> scale_rating_deviation_to(:v1), rating_deviation |> scale_rating_deviation_to(:v1)
} }
end
@doc """ @doc """
Converts a v1 player to a v2. Converts a v1 player to a v2.
@ -115,12 +128,17 @@ defmodule Glicko.Player do
""" """
@spec to_v2(player :: t, volatility :: volatility) :: v2 @spec to_v2(player :: t, volatility :: volatility) :: v2
def to_v2(player, volatility \\ initial_volatility()) def to_v2(player, volatility \\ initial_volatility())
def to_v2({rating, rating_deviation, volatility}, _volatility), do: {rating, rating_deviation, volatility}
def to_v2({rating, rating_deviation}, volatility), do: { def to_v2({rating, rating_deviation, volatility}, _volatility),
do: {rating, rating_deviation, volatility}
def to_v2({rating, rating_deviation}, volatility) do
{
rating |> scale_rating_to(:v2), rating |> scale_rating_to(:v2),
rating_deviation |> scale_rating_deviation_to(:v2), rating_deviation |> scale_rating_deviation_to(:v2),
volatility, volatility
} }
end
@doc """ @doc """
A version agnostic method for getting a player's rating. A version agnostic method for getting a player's rating.
@ -142,8 +160,13 @@ defmodule Glicko.Player do
def rating_deviation({_, rating_deviation}, nil), do: rating_deviation def rating_deviation({_, rating_deviation}, nil), do: rating_deviation
def rating_deviation({_, rating_deviation, _}, nil), do: rating_deviation def rating_deviation({_, rating_deviation, _}, nil), do: rating_deviation
def rating_deviation({_, rating_deviation}, :v1), do: rating_deviation def rating_deviation({_, rating_deviation}, :v1), do: rating_deviation
def rating_deviation({_, rating_deviation}, :v2), do: rating_deviation |> scale_rating_deviation_to(:v2)
def rating_deviation({_, rating_deviation, _}, :v1), do: rating_deviation |> scale_rating_deviation_to(:v1) def rating_deviation({_, rating_deviation}, :v2),
do: rating_deviation |> scale_rating_deviation_to(:v2)
def rating_deviation({_, rating_deviation, _}, :v1),
do: rating_deviation |> scale_rating_deviation_to(:v1)
def rating_deviation({_, rating_deviation, _}, :v2), do: rating_deviation def rating_deviation({_, rating_deviation, _}, :v2), do: rating_deviation
@doc """ @doc """
@ -170,23 +193,31 @@ defmodule Glicko.Player do
be 95% confident about a players strength being in a small interval of values. be 95% confident about a players strength being in a small interval of values.
""" """
@spec rating_interval(player :: t) :: {rating_low :: float, rating_high :: float} @spec rating_interval(player :: t) :: {rating_low :: float, rating_high :: float}
def rating_interval(player, as_version \\ nil), do: { def rating_interval(player, as_version \\ nil) do
{
rating(player, as_version) - rating_deviation(player, as_version) * 2, rating(player, as_version) - rating_deviation(player, as_version) * 2,
rating(player, as_version) + rating_deviation(player, as_version) * 2, rating(player, as_version) + rating_deviation(player, as_version) * 2
} }
end
@doc """ @doc """
Scales a player's rating. Scales a player's rating.
""" """
@spec scale_rating_to(rating :: rating, to_version :: version) :: rating @spec scale_rating_to(rating :: rating, to_version :: version) :: rating
def scale_rating_to(rating, _version = :v1), do: (rating * @magic_version_scale) + @magic_version_scale_rating def scale_rating_to(rating, _version = :v1),
def scale_rating_to(rating, _version = :v2), do: (rating - @magic_version_scale_rating) / @magic_version_scale do: rating * @magic_version_scale + @magic_version_scale_rating
def scale_rating_to(rating, _version = :v2),
do: (rating - @magic_version_scale_rating) / @magic_version_scale
@doc """ @doc """
Scales a player's rating deviation. Scales a player's rating deviation.
""" """
@spec scale_rating_deviation_to(rating_deviation :: rating_deviation, to_version :: version) :: rating_deviation @spec scale_rating_deviation_to(rating_deviation :: rating_deviation, to_version :: version) ::
def scale_rating_deviation_to(rating_deviation, _version = :v1), do: rating_deviation * @magic_version_scale rating_deviation
def scale_rating_deviation_to(rating_deviation, _version = :v2), do: rating_deviation / @magic_version_scale def scale_rating_deviation_to(rating_deviation, _version = :v1),
do: rating_deviation * @magic_version_scale
def scale_rating_deviation_to(rating_deviation, _version = :v2),
do: rating_deviation / @magic_version_scale
end end

View File

@ -14,7 +14,7 @@ defmodule Glicko.Result do
alias Glicko.Player alias Glicko.Player
@type t :: {Player.rating, Player.rating_deviation, score} @type t :: {Player.rating(), Player.rating_deviation(), score}
@type score :: float @type score :: float
@type score_shortcut :: :loss | :draw | :win @type score_shortcut :: :loss | :draw | :win
@ -29,11 +29,13 @@ defmodule Glicko.Result do
Supports passing either `:loss`, `:draw`, or `:win` as shortcuts. Supports passing either `:loss`, `:draw`, or `:win` as shortcuts.
""" """
@spec new(Player.rating, Player.rating_deviation, score | score_shortcut) :: t @spec new(Player.rating(), Player.rating_deviation(), score | score_shortcut) :: t
def new(opponent_rating, opponent_rating_deviation, score) when is_number(score) do def new(opponent_rating, opponent_rating_deviation, score) when is_number(score) do
{opponent_rating, opponent_rating_deviation, score} {opponent_rating, opponent_rating_deviation, score}
end end
def new(opponent_rating, opponent_rating_deviation, score_type) when is_atom(score_type) and score_type in @score_shortcuts do
def new(opponent_rating, opponent_rating_deviation, score_type)
when is_atom(score_type) and score_type in @score_shortcuts do
{opponent_rating, opponent_rating_deviation, Map.fetch!(@score_shortcut_map, score_type)} {opponent_rating, opponent_rating_deviation, Map.fetch!(@score_shortcut_map, score_type)}
end end
@ -42,7 +44,7 @@ defmodule Glicko.Result do
Supports passing either `:loss`, `:draw`, or `:win` as shortcuts. Supports passing either `:loss`, `:draw`, or `:win` as shortcuts.
""" """
@spec new(opponent :: Player.t, score :: score | score_shortcut) :: t @spec new(opponent :: Player.t(), score :: score | score_shortcut) :: t
def new(opponent, score) do def new(opponent, score) do
new(Player.rating(opponent, :v2), Player.rating_deviation(opponent, :v2), score) new(Player.rating(opponent, :v2), Player.rating_deviation(opponent, :v2), score)
end end
@ -50,19 +52,18 @@ defmodule Glicko.Result do
@doc """ @doc """
Convenience function for accessing an opponent's rating. Convenience function for accessing an opponent's rating.
""" """
@spec opponent_rating(result :: Result.t) :: Player.rating @spec opponent_rating(result :: Result.t()) :: Player.rating()
def opponent_rating(_result = {rating, _, _}), do: rating def opponent_rating(_result = {rating, _, _}), do: rating
@doc """ @doc """
Convenience function for accessing an opponent's rating deviation. Convenience function for accessing an opponent's rating deviation.
""" """
@spec opponent_rating_deviation(result :: Result.t) :: Player.rating_deviation @spec opponent_rating_deviation(result :: Result.t()) :: Player.rating_deviation()
def opponent_rating_deviation(_result = {_, rating_deviation, _}), do: rating_deviation def opponent_rating_deviation(_result = {_, rating_deviation, _}), do: rating_deviation
@doc """ @doc """
Convenience function for accessing the score. Convenience function for accessing the score.
""" """
@spec score(result :: Result.t) :: score @spec score(result :: Result.t()) :: score
def score(_result = {_, _, score}), do: score def score(_result = {_, _, score}), do: score
end end

22
mix.exs
View File

@ -5,26 +5,32 @@ defmodule Glicko.Mixfile do
Implementation of the Glicko rating system Implementation of the Glicko rating system
""" """
def project, do: [ def project do
[
app: :glicko, app: :glicko,
version: "0.6.0", version: "0.6.0",
elixir: "~> 1.5", elixir: "~> 1.9",
start_permanent: Mix.env == :prod, start_permanent: Mix.env() == :prod,
deps: deps(), deps: deps(),
package: package(), package: package(),
description: @description, description: @description
] ]
end
defp deps, do: [ defp deps do
[
{:inch_ex, "~> 0.5", only: :docs}, {:inch_ex, "~> 0.5", only: :docs},
{:ex_doc, "~> 0.16", only: :dev, runtime: false}, {:ex_doc, "~> 0.16", only: :dev, runtime: false},
{:credo, "~> 0.8", only: [:dev, :test], runtime: false}, {:credo, "~> 0.8", only: [:dev, :test], runtime: false}
] ]
end
defp package, do: [ defp package do
[
name: :glicko, name: :glicko,
maintainers: ["James Dyson"], maintainers: ["James Dyson"],
licenses: ["MIT"], licenses: ["MIT"],
links: %{"GitHub" => "https://github.com/avitex/elixir-glicko"}, links: %{"GitHub" => "https://github.com/avitex/elixir-glicko"}
] ]
end
end end

View File

@ -3,66 +3,75 @@ defmodule GlickoTest do
alias Glicko.{ alias Glicko.{
Player, Player,
Result, Result
} }
doctest Glicko doctest Glicko
@player [rating: 1500, rating_deviation: 200] |> Player.new_v1 |> Player.to_v2 @player [rating: 1500, rating_deviation: 200] |> Player.new_v1() |> Player.to_v2()
@results [ @results [
Result.new(Player.new_v1([rating: 1400, rating_deviation: 30]), :win), Result.new(Player.new_v1(rating: 1400, rating_deviation: 30), :win),
Result.new(Player.new_v1([rating: 1550, rating_deviation: 100]), :loss), Result.new(Player.new_v1(rating: 1550, rating_deviation: 100), :loss),
Result.new(Player.new_v1([rating: 1700, rating_deviation: 300]), :loss), Result.new(Player.new_v1(rating: 1700, rating_deviation: 300), :loss)
] ]
@valid_player_rating_after_results 1464.06 |> Player.scale_rating_to(:v2) @valid_player_rating_after_results 1464.06 |> Player.scale_rating_to(:v2)
@valid_player_rating_deviation_after_results 151.52 |> Player.scale_rating_deviation_to(:v2) @valid_player_rating_deviation_after_results 151.52 |> Player.scale_rating_deviation_to(:v2)
@valid_player_volatility_after_results 0.05999 @valid_player_volatility_after_results 0.05999
@valid_player_rating_deviation_after_no_results 200.2714 |> Player.scale_rating_deviation_to(:v2) @valid_player_rating_deviation_after_no_results 200.2714
|> Player.scale_rating_deviation_to(:v2)
describe "new rating" do describe "new rating" do
test "with results" do test "with results" do
player = Glicko.new_rating(@player, @results, [system_constant: 0.5]) player = Glicko.new_rating(@player, @results, system_constant: 0.5)
assert_in_delta Player.rating(player), @valid_player_rating_after_results, 1.0e-4 assert_in_delta Player.rating(player), @valid_player_rating_after_results, 1.0e-4
assert_in_delta Player.rating_deviation(player), @valid_player_rating_deviation_after_results, 1.0e-4
assert_in_delta Player.rating_deviation(player),
@valid_player_rating_deviation_after_results,
1.0e-4
assert_in_delta Player.volatility(player), @valid_player_volatility_after_results, 1.0e-5 assert_in_delta Player.volatility(player), @valid_player_volatility_after_results, 1.0e-5
end end
test "no results" do test "no results" do
player = Glicko.new_rating(@player, []) player = Glicko.new_rating(@player, [])
assert_in_delta Player.rating_deviation(player), @valid_player_rating_deviation_after_no_results, 1.0e-4 assert_in_delta Player.rating_deviation(player),
@valid_player_rating_deviation_after_no_results,
1.0e-4
end end
end end
describe "win probability" do describe "win probability" do
test "with same ratings" do test "with same ratings" do
assert Glicko.win_probability(Player.new_v1, Player.new_v1) == 0.5 assert Glicko.win_probability(Player.new_v1(), Player.new_v1()) == 0.5
end end
test "with better opponent" do test "with better opponent" do
assert Glicko.win_probability(Player.new_v1([rating: 1500]), Player.new_v1([rating: 1600])) < 0.5 assert Glicko.win_probability(Player.new_v1(rating: 1500), Player.new_v1(rating: 1600)) <
0.5
end end
test "with better player" do test "with better player" do
assert Glicko.win_probability(Player.new_v1([rating: 1600]), Player.new_v1([rating: 1500])) > 0.5 assert Glicko.win_probability(Player.new_v1(rating: 1600), Player.new_v1(rating: 1500)) >
0.5
end end
end end
describe "draw probability" do describe "draw probability" do
test "with same ratings" do test "with same ratings" do
assert Glicko.draw_probability(Player.new_v1, Player.new_v1) == 1 assert Glicko.draw_probability(Player.new_v1(), Player.new_v1()) == 1
end end
test "with better opponent" do test "with better opponent" do
assert Glicko.draw_probability(Player.new_v1([rating: 1500]), Player.new_v1([rating: 1600])) < 1 assert Glicko.draw_probability(Player.new_v1(rating: 1500), Player.new_v1(rating: 1600)) < 1
end end
test "with better player" do test "with better player" do
assert Glicko.draw_probability(Player.new_v1([rating: 1600]), Player.new_v1([rating: 1500])) < 1 assert Glicko.draw_probability(Player.new_v1(rating: 1600), Player.new_v1(rating: 1500)) < 1
end end
end end
end end

View File

@ -9,19 +9,21 @@ defmodule Glicko.PlayerTest do
@valid_v2_base {1.0, 2.0, 3.0} @valid_v2_base {1.0, 2.0, 3.0}
test "create v1" do test "create v1" do
assert @valid_v1_base == Player.new_v1([rating: 1.0, rating_deviation: 2.0]) assert @valid_v1_base == Player.new_v1(rating: 1.0, rating_deviation: 2.0)
end end
test "create v2" do test "create v2" do
assert @valid_v2_base == Player.new_v2([rating: 1.0, rating_deviation: 2.0, volatility: 3.0]) assert @valid_v2_base == Player.new_v2(rating: 1.0, rating_deviation: 2.0, volatility: 3.0)
end end
test "convert player v1 -> v2" do test "convert player v1 -> v2" do
assert {Player.scale_rating_to(1.0, :v2), Player.scale_rating_deviation_to(2.0, :v2), 3.0} == Player.to_v2(@valid_v1_base, 3.0) assert {Player.scale_rating_to(1.0, :v2), Player.scale_rating_deviation_to(2.0, :v2), 3.0} ==
Player.to_v2(@valid_v1_base, 3.0)
end end
test "convert player v2 -> v1" do test "convert player v2 -> v1" do
assert {Player.scale_rating_to(1.0, :v1), Player.scale_rating_deviation_to(2.0, :v1)} == Player.to_v1(@valid_v2_base) assert {Player.scale_rating_to(1.0, :v1), Player.scale_rating_deviation_to(2.0, :v1)} ==
Player.to_v1(@valid_v2_base)
end end
test "convert player v1 -> v1" do test "convert player v1 -> v1" do
@ -51,8 +53,8 @@ defmodule Glicko.PlayerTest do
test "rating interval" do test "rating interval" do
assert {rating_low, rating_high} = assert {rating_low, rating_high} =
[rating: 1850, rating_deviation: 50] [rating: 1850, rating_deviation: 50]
|> Player.new_v2 |> Player.new_v2()
|> Player.rating_interval |> Player.rating_interval()
assert_in_delta rating_low, 1750, 0.1 assert_in_delta rating_low, 1750, 0.1
assert_in_delta rating_high, 1950, 0.1 assert_in_delta rating_high, 1950, 0.1

View File

@ -3,12 +3,12 @@ defmodule Glicko.ResultTest do
alias Glicko.{ alias Glicko.{
Player, Player,
Result, Result
} }
doctest Result doctest Result
@opponent Player.new_v2 @opponent Player.new_v2()
@valid_game_result Result.new(@opponent, 0.0) @valid_game_result Result.new(@opponent, 0.0)