Elote is a powerful Python library for implementing and comparing rating systems. Whether you're ranking chess players, sports teams, or prioritizing features in your product backlog, Elote provides a simple, elegant API for all your competitive ranking needs.
- Overview
- Features
- Installation
- Quick Start
- Usage Examples
- Development
- Contributing
- Blog Posts
- References
Rating systems allow you to rank competitors based on their performance in head-to-head matchups. The most famous example is the Elo rating system used in chess, but these systems have applications far beyond sports:
- Ranking products based on A/B comparisons
- Prioritizing features through pairwise voting
- Creating recommendation systems
- Matchmaking in games and competitions
- Collaborative filtering and ranking
Elote makes implementing these systems simple and intuitive, with a clean API that handles all the mathematical complexity for you.
Currently implemented rating systems:
- Elo - The classic chess rating system
- Glicko-1 - An improvement on Elo that accounts for rating reliability
- Glicko-2 - A further improvement on Glicko that adds volatility tracking
- TrueSkill - Microsoft's Bayesian skill rating system for multiplayer games
- ECF - The English Chess Federation rating system
- DWZ - The Deutsche Wertungszahl (German evaluation number) system
pip install elote
We use a modern Python packaging approach with pyproject.toml
. Most things you need are in the Makefile
:
# Using Make (recommended)
make install-dev
# Or using pip
pip install -e ".[dev]"
# Or using uv
uv pip install -e ".[dev]"
- Python 3.8 or higher
from elote import EloCompetitor
# Create two competitors with different initial ratings
player1 = EloCompetitor(initial_rating=1500)
player2 = EloCompetitor(initial_rating=1600)
# Get win probability
print(f"Player 2 win probability: {player2.expected_score(player1):.2%}")
# Record a match result
player1.beat(player2) # Player 1 won!
# Ratings are automatically updated
print(f"Player 1 new rating: {player1.rating}")
print(f"Player 2 new rating: {player2.rating}")
Elote is built around two main concepts: Competitors and Arenas.
Competitors represent the entities you're rating. Here's how to use them:
from elote import EloCompetitor
good = EloCompetitor(initial_rating=400)
better = EloCompetitor(initial_rating=500)
# Check win probabilities
print(f"Probability of better beating good: {better.expected_score(good):.2%}")
print(f"Probability of good beating better: {good.expected_score(better):.2%}")
Output:
Probability of better beating good: 64.01%
Probability of good beating better: 35.99%
If a match occurs, updating ratings is simple:
# If good wins (an upset!)
good.beat(better)
# OR
better.lost_to(good)
# Check updated probabilities
print(f"Probability of better beating good: {better.expected_score(good):.2%}")
print(f"Probability of good beating better: {good.expected_score(better):.2%}")
Output:
Probability of better beating good: 61.25%
Probability of good beating better: 38.75%
Arenas handle large numbers of matchups automatically. The LambdaArena
takes a comparison function and manages all competitors for you:
from elote import LambdaArena
import json
import random
# Define a comparison function (returns True if a beats b)
def comparison(a, b):
return a > b
# Generate 1000 random matchups between numbers 1-10
matchups = [(random.randint(1, 10), random.randint(1, 10)) for _ in range(1000)]
# Create arena and run tournament
arena = LambdaArena(comparison)
arena.tournament(matchups)
# Display final rankings
print("Arena results:")
print(json.dumps(arena.leaderboard(), indent=4))
This example effectively implements a sorting algorithm using a rating system - not efficient, but demonstrates how Elote works with any comparable objects!
The project includes a Makefile that simplifies common development tasks:
# Run tests
make test
# Run tests with coverage
make test-cov
# Lint code
make lint
# Auto-fix linting issues
make lint-fix
# Format code
make format
# Build package
make build
# Build documentation
make docs
Contributions are welcome! If you'd like to help improve Elote:
- Check the issues for open tasks
- Fork the repository
- Create a feature branch
- Add your changes
- Submit a pull request
For major changes, please open an issue first to discuss what you'd like to change.
Here are some blog posts about Elote:
- Elote: A Python Package for Rating Systems - Introduction to the library
- Using Cursor for Library Maintenance - How Cursor helps maintain Elote
- Year's End: Looking Back at 2017 - Reflections including Elote development
- Glicko Rating System
- Glicko-2 Rating System
- Massey Ratings
- Elo, Arpad (1978). The Rating of Chessplayers, Past and Present. Arco. ISBN 0-668-04721-6.
- ECF Grading System
- Deutsche Wertungszahl
- TrueSkill: A Bayesian Skill Rating System