Skip to content

Commit

Permalink
Merge pull request #11 from asyncth/add-full-trueskill
Browse files Browse the repository at this point in the history
Implement TrueSkill for more than two teams
  • Loading branch information
atomflunder authored Jun 8, 2024
2 parents 8bc74e9 + 0f02755 commit 4adbb98
Show file tree
Hide file tree
Showing 4 changed files with 1,477 additions and 6 deletions.
276 changes: 276 additions & 0 deletions src/trueskill/factor_graph.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,276 @@
use std::cell::RefCell;
use std::collections::HashMap;
use std::rc::Rc;

use super::gaussian::Gaussian;

#[derive(Clone, Debug, Default, PartialEq)]
pub struct Variable {
pub gaussian: Gaussian,
messages: HashMap<u32, Gaussian>,
}

impl Variable {
pub fn new() -> Self {
Self {
gaussian: Gaussian::default(),
messages: HashMap::new(),
}
}

fn set(&mut self, val: Gaussian) -> f64 {
let delta = self.delta(val);

self.gaussian.pi = val.pi;
self.gaussian.tau = val.tau;

delta
}

fn update_message(&mut self, factor_id: u32, message: Gaussian) -> f64 {
let old_message = self.messages[&factor_id];
let v = self.messages.entry(factor_id).or_default();
*v = message;

self.set(self.gaussian / old_message * message)
}

fn update_value(&mut self, factor_id: u32, val: Gaussian) -> f64 {
let old_message = self.messages[&factor_id];
let v = self.messages.entry(factor_id).or_default();
*v = val * old_message / self.gaussian;

self.set(val)
}

fn delta(&self, other: Gaussian) -> f64 {
let pi_delta = (self.gaussian.pi - other.pi).abs();
if pi_delta.is_infinite() {
return 0.0;
}

(self.gaussian.tau - other.tau).abs().max(pi_delta.sqrt())
}
}

pub struct PriorFactor {
id: u32,
pub variable: Rc<RefCell<Variable>>,
val: Gaussian,
dynamic: f64,
}

impl PriorFactor {
pub fn new(id: u32, variable: Rc<RefCell<Variable>>, val: Gaussian, dynamic: f64) -> Self {
variable.borrow_mut().messages.entry(id).or_default();

Self {
id,
variable,
val,
dynamic,
}
}

pub fn down(&mut self) -> f64 {
let sigma = self.val.sigma().hypot(self.dynamic);
let value = Gaussian::with_mu_sigma(self.val.mu(), sigma);
self.variable.borrow_mut().update_value(self.id, value)
}
}

pub struct LikelihoodFactor {
id: u32,
mean: Rc<RefCell<Variable>>,
value: Rc<RefCell<Variable>>,
variance: f64,
}

impl LikelihoodFactor {
pub fn new(
id: u32,
mean: Rc<RefCell<Variable>>,
value: Rc<RefCell<Variable>>,
variance: f64,
) -> Self {
mean.borrow_mut().messages.entry(id).or_default();
value.borrow_mut().messages.entry(id).or_default();

Self {
id,
mean,
value,
variance,
}
}

pub fn down(&mut self) -> f64 {
let msg = {
let mean = self.mean.borrow();
mean.gaussian / mean.messages[&self.id]
};
let a = self.calc_a(msg);
self.value
.borrow_mut()
.update_message(self.id, Gaussian::with_pi_tau(a * msg.pi, a * msg.tau))
}

pub fn up(&mut self) -> f64 {
let msg = {
let value = self.value.borrow();
value.gaussian / value.messages[&self.id]
};
let a = self.calc_a(msg);
self.mean
.borrow_mut()
.update_message(self.id, Gaussian::with_pi_tau(a * msg.pi, a * msg.tau))
}

fn calc_a(&self, gaussian: Gaussian) -> f64 {
self.variance.mul_add(gaussian.pi, 1.0).recip()
}
}

pub struct SumFactor {
id: u32,
sum: Rc<RefCell<Variable>>,
terms: Vec<Rc<RefCell<Variable>>>,
coeffs: Vec<f64>,
}

impl SumFactor {
pub fn new(
id: u32,
sum: Rc<RefCell<Variable>>,
terms: Vec<Rc<RefCell<Variable>>>,
coeffs: Vec<f64>,
) -> Self {
sum.borrow_mut().messages.entry(id).or_default();
for term in &terms {
term.borrow_mut().messages.entry(id).or_default();
}

Self {
id,
sum,
terms,
coeffs,
}
}

pub fn down(&mut self) -> f64 {
let msgs: Vec<Gaussian> = self
.terms
.iter()
.map(|term| term.borrow().messages[&self.id])
.collect();
self.update(&self.sum, &self.terms, &msgs, &self.coeffs)
}

pub fn up(&mut self, index: usize) -> f64 {
let coeff = self.coeffs[index];
let mut coeffs = Vec::new();
for (x, c) in self.coeffs.iter().enumerate() {
if coeff == 0.0 {
coeffs.push(0.0);
} else if x == index {
coeffs.push(coeff.recip());
} else {
coeffs.push(-(*c) / coeff);
}
}

let mut vals = self.terms.clone();
vals[index] = self.sum.clone();
let msgs: Vec<Gaussian> = vals
.iter()
.map(|val| val.borrow().messages[&self.id])
.collect();

self.update(&self.terms[index], &vals, &msgs, &coeffs)
}

#[inline]
pub fn terms_len(&self) -> usize {
self.terms.len()
}

fn update(
&self,
var: &Rc<RefCell<Variable>>,
vals: &[Rc<RefCell<Variable>>],
msgs: &[Gaussian],
coeffs: &[f64],
) -> f64 {
let mut pi_inv = 0.0_f64;
let mut mu = 0.0;

for ((val, msg), coeff) in vals.iter().zip(msgs).zip(coeffs) {
let div = val.borrow().gaussian / *msg;
mu += coeff * div.mu();
if pi_inv.is_infinite() {
continue;
}

if div.pi == 0.0 {
pi_inv = f64::INFINITY;
} else {
pi_inv += coeff.powi(2) / div.pi;
}
}

let pi = pi_inv.recip();
let tau = pi * mu;

var.borrow_mut()
.update_message(self.id, Gaussian::with_pi_tau(pi, tau))
}
}

pub struct TruncateFactor {
id: u32,
variable: Rc<RefCell<Variable>>,
v_func: Box<dyn Fn(f64, f64) -> f64>,
w_func: Box<dyn Fn(f64, f64) -> f64>,
draw_margin: f64,
}

impl TruncateFactor {
pub fn new(
id: u32,
variable: Rc<RefCell<Variable>>,
v_func: Box<dyn Fn(f64, f64) -> f64>,
w_func: Box<dyn Fn(f64, f64) -> f64>,
draw_margin: f64,
) -> Self {
variable.borrow_mut().messages.entry(id).or_default();

Self {
id,
variable,
v_func,
w_func,
draw_margin,
}
}

pub fn up(&mut self) -> f64 {
let div = {
let variable = self.variable.borrow();
variable.gaussian / variable.messages[&self.id]
};
let pi_sqrt = div.pi.sqrt();
let arg_1 = div.tau / pi_sqrt;
let arg_2 = self.draw_margin * pi_sqrt;
let v = (self.v_func)(arg_1, arg_2);
let w = (self.w_func)(arg_1, arg_2);
let denom = 1.0 - w;

let pi = div.pi / denom;
let tau = pi_sqrt.mul_add(v, div.tau) / denom;

self.variable
.borrow_mut()
.update_value(self.id, Gaussian::with_pi_tau(pi, tau))
}
}
65 changes: 65 additions & 0 deletions src/trueskill/gaussian.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
use std::cmp::Ordering;
use std::ops::{Div, Mul};

#[derive(Copy, Clone, Debug, Default, PartialEq)]
pub struct Gaussian {
pub pi: f64,
pub tau: f64,
}

impl Gaussian {
pub fn with_mu_sigma(mu: f64, sigma: f64) -> Self {
assert_ne!(sigma, 0.0, "sigma^2 needs to be greater than 0");

let pi = sigma.powi(-2);
Self { pi, tau: pi * mu }
}

pub const fn with_pi_tau(pi: f64, tau: f64) -> Self {
Self { pi, tau }
}

pub fn mu(&self) -> f64 {
if self.pi == 0.0 {
return 0.0;
}

self.tau / self.pi
}

pub fn sigma(&self) -> f64 {
if self.pi == 0.0 {
return f64::INFINITY;
}

self.pi.recip().sqrt()
}
}

impl Mul for Gaussian {
type Output = Self;

fn mul(self, rhs: Self) -> Self::Output {
Self {
pi: self.pi + rhs.pi,
tau: self.tau + rhs.tau,
}
}
}

impl Div for Gaussian {
type Output = Self;

fn div(self, rhs: Self) -> Self::Output {
Self {
pi: self.pi - rhs.pi,
tau: self.tau - rhs.tau,
}
}
}

impl PartialOrd for Gaussian {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
self.mu().partial_cmp(&other.mu())
}
}
Loading

0 comments on commit 4adbb98

Please sign in to comment.