{
"cells": [
{
"cell_type": "markdown",
"id": "c1915f05-dde7-48c3-8685-54106e8db9d3",
"metadata": {},
"source": [
"# Colley Ratings\n",
"\n",
"Colley Ratings are a mathematically sound, unbiased method for ranking football teams, relying exclusively on match outcomes rather than scores or margins of victory. \n",
"\n",
"Unlike some rating systems, the Colley method is specifically designed to be objective, using only wins, losses, and draws without subjective weighting. \n",
"\n",
"Its mathematical simplicity and stability make it highly suitable for quickly evaluating team performance, objectively comparing team strengths, and identifying discrepancies between actual team quality and public perception or bookmaker odds."
]
},
{
"cell_type": "code",
"execution_count": 1,
"id": "a23baca5-1387-4d56-9c8a-59bf2633bd84",
"metadata": {},
"outputs": [],
"source": [
"import penaltyblog as pb"
]
},
{
"cell_type": "code",
"execution_count": 2,
"id": "bea84cd8-7e75-4863-9ece-349870e19e53",
"metadata": {},
"outputs": [
{
"data": {
"text/html": [
"
\n",
"\n",
"
\n",
" \n",
" \n",
" | \n",
" date | \n",
" datetime | \n",
" season | \n",
" competition | \n",
" div | \n",
" time | \n",
" team_home | \n",
" team_away | \n",
" fthg | \n",
" ftag | \n",
" ... | \n",
" b365_cahh | \n",
" b365_caha | \n",
" pcahh | \n",
" pcaha | \n",
" max_cahh | \n",
" max_caha | \n",
" avg_cahh | \n",
" avg_caha | \n",
" goals_home | \n",
" goals_away | \n",
"
\n",
" \n",
" id | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
" | \n",
"
\n",
" \n",
" \n",
" \n",
" 1628812800---brentford---arsenal | \n",
" 2021-08-13 | \n",
" 2021-08-13 20:00:00 | \n",
" 2021-2022 | \n",
" ENG Premier League | \n",
" E0 | \n",
" 20:00 | \n",
" Brentford | \n",
" Arsenal | \n",
" 2 | \n",
" 0 | \n",
" ... | \n",
" 1.75 | \n",
" 2.05 | \n",
" 1.81 | \n",
" 2.13 | \n",
" 2.05 | \n",
" 2.17 | \n",
" 1.80 | \n",
" 2.09 | \n",
" 2 | \n",
" 0 | \n",
"
\n",
" \n",
" 1628899200---burnley---brighton | \n",
" 2021-08-14 | \n",
" 2021-08-14 15:00:00 | \n",
" 2021-2022 | \n",
" ENG Premier League | \n",
" E0 | \n",
" 15:00 | \n",
" Burnley | \n",
" Brighton | \n",
" 1 | \n",
" 2 | \n",
" ... | \n",
" 1.79 | \n",
" 2.15 | \n",
" 1.81 | \n",
" 2.14 | \n",
" 1.82 | \n",
" 2.19 | \n",
" 1.79 | \n",
" 2.12 | \n",
" 1 | \n",
" 2 | \n",
"
\n",
" \n",
" 1628899200---chelsea---crystal_palace | \n",
" 2021-08-14 | \n",
" 2021-08-14 15:00:00 | \n",
" 2021-2022 | \n",
" ENG Premier League | \n",
" E0 | \n",
" 15:00 | \n",
" Chelsea | \n",
" Crystal Palace | \n",
" 3 | \n",
" 0 | \n",
" ... | \n",
" 2.05 | \n",
" 1.75 | \n",
" 2.12 | \n",
" 1.81 | \n",
" 2.16 | \n",
" 1.93 | \n",
" 2.06 | \n",
" 1.82 | \n",
" 3 | \n",
" 0 | \n",
"
\n",
" \n",
" 1628899200---everton---southampton | \n",
" 2021-08-14 | \n",
" 2021-08-14 15:00:00 | \n",
" 2021-2022 | \n",
" ENG Premier League | \n",
" E0 | \n",
" 15:00 | \n",
" Everton | \n",
" Southampton | \n",
" 3 | \n",
" 1 | \n",
" ... | \n",
" 2.05 | \n",
" 1.88 | \n",
" 2.05 | \n",
" 1.88 | \n",
" 2.08 | \n",
" 1.90 | \n",
" 2.03 | \n",
" 1.86 | \n",
" 3 | \n",
" 1 | \n",
"
\n",
" \n",
" 1628899200---leicester---wolves | \n",
" 2021-08-14 | \n",
" 2021-08-14 15:00:00 | \n",
" 2021-2022 | \n",
" ENG Premier League | \n",
" E0 | \n",
" 15:00 | \n",
" Leicester | \n",
" Wolves | \n",
" 1 | \n",
" 0 | \n",
" ... | \n",
" 2.02 | \n",
" 1.91 | \n",
" 2.01 | \n",
" 1.92 | \n",
" 2.05 | \n",
" 1.95 | \n",
" 1.99 | \n",
" 1.89 | \n",
" 1 | \n",
" 0 | \n",
"
\n",
" \n",
"
\n",
"
5 rows × 111 columns
\n",
"
"
],
"text/plain": [
" date datetime \\\n",
"id \n",
"1628812800---brentford---arsenal 2021-08-13 2021-08-13 20:00:00 \n",
"1628899200---burnley---brighton 2021-08-14 2021-08-14 15:00:00 \n",
"1628899200---chelsea---crystal_palace 2021-08-14 2021-08-14 15:00:00 \n",
"1628899200---everton---southampton 2021-08-14 2021-08-14 15:00:00 \n",
"1628899200---leicester---wolves 2021-08-14 2021-08-14 15:00:00 \n",
"\n",
" season competition div \\\n",
"id \n",
"1628812800---brentford---arsenal 2021-2022 ENG Premier League E0 \n",
"1628899200---burnley---brighton 2021-2022 ENG Premier League E0 \n",
"1628899200---chelsea---crystal_palace 2021-2022 ENG Premier League E0 \n",
"1628899200---everton---southampton 2021-2022 ENG Premier League E0 \n",
"1628899200---leicester---wolves 2021-2022 ENG Premier League E0 \n",
"\n",
" time team_home team_away fthg \\\n",
"id \n",
"1628812800---brentford---arsenal 20:00 Brentford Arsenal 2 \n",
"1628899200---burnley---brighton 15:00 Burnley Brighton 1 \n",
"1628899200---chelsea---crystal_palace 15:00 Chelsea Crystal Palace 3 \n",
"1628899200---everton---southampton 15:00 Everton Southampton 3 \n",
"1628899200---leicester---wolves 15:00 Leicester Wolves 1 \n",
"\n",
" ftag ... b365_cahh b365_caha pcahh \\\n",
"id ... \n",
"1628812800---brentford---arsenal 0 ... 1.75 2.05 1.81 \n",
"1628899200---burnley---brighton 2 ... 1.79 2.15 1.81 \n",
"1628899200---chelsea---crystal_palace 0 ... 2.05 1.75 2.12 \n",
"1628899200---everton---southampton 1 ... 2.05 1.88 2.05 \n",
"1628899200---leicester---wolves 0 ... 2.02 1.91 2.01 \n",
"\n",
" pcaha max_cahh max_caha avg_cahh \\\n",
"id \n",
"1628812800---brentford---arsenal 2.13 2.05 2.17 1.80 \n",
"1628899200---burnley---brighton 2.14 1.82 2.19 1.79 \n",
"1628899200---chelsea---crystal_palace 1.81 2.16 1.93 2.06 \n",
"1628899200---everton---southampton 1.88 2.08 1.90 2.03 \n",
"1628899200---leicester---wolves 1.92 2.05 1.95 1.99 \n",
"\n",
" avg_caha goals_home goals_away \n",
"id \n",
"1628812800---brentford---arsenal 2.09 2 0 \n",
"1628899200---burnley---brighton 2.12 1 2 \n",
"1628899200---chelsea---crystal_palace 1.82 3 0 \n",
"1628899200---everton---southampton 1.86 3 1 \n",
"1628899200---leicester---wolves 1.89 1 0 \n",
"\n",
"[5 rows x 111 columns]"
]
},
"execution_count": 2,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"fbd = pb.scrapers.FootballData(\"ENG Premier League\", \"2021-2022\")\n",
"df = fbd.get_fixtures()\n",
"\n",
"df.head()"
]
},
{
"cell_type": "markdown",
"id": "6f3eb29e-a381-40bf-8d65-63f6744232f3",
"metadata": {},
"source": [
"### Colley ratings including tied scorelines"
]
},
{
"cell_type": "code",
"execution_count": 3,
"id": "9fc2e8b0-d55f-45c4-bb23-69cf0fdcf56c",
"metadata": {},
"outputs": [
{
"data": {
"text/html": [
"\n",
"\n",
"
\n",
" \n",
" \n",
" | \n",
" team | \n",
" rating | \n",
"
\n",
" \n",
" \n",
" \n",
" 0 | \n",
" Liverpool | \n",
" 1.904762 | \n",
"
\n",
" \n",
" 1 | \n",
" Man City | \n",
" 1.892857 | \n",
"
\n",
" \n",
" 2 | \n",
" Chelsea | \n",
" 1.791667 | \n",
"
\n",
" \n",
" 3 | \n",
" Tottenham | \n",
" 1.708333 | \n",
"
\n",
" \n",
" 4 | \n",
" Arsenal | \n",
" 1.672619 | \n",
"
\n",
" \n",
" 5 | \n",
" Man United | \n",
" 1.654762 | \n",
"
\n",
" \n",
" 6 | \n",
" Brighton | \n",
" 1.64881 | \n",
"
\n",
" \n",
" 7 | \n",
" Crystal Palace | \n",
" 1.625 | \n",
"
\n",
" \n",
" 8 | \n",
" West Ham | \n",
" 1.619048 | \n",
"
\n",
" \n",
" 9 | \n",
" Leicester | \n",
" 1.607143 | \n",
"
\n",
" \n",
" 10 | \n",
" Newcastle | \n",
" 1.583333 | \n",
"
\n",
" \n",
" 11 | \n",
" Wolves | \n",
" 1.559524 | \n",
"
\n",
" \n",
" 12 | \n",
" Southampton | \n",
" 1.541667 | \n",
"
\n",
" \n",
" 13 | \n",
" Brentford | \n",
" 1.529762 | \n",
"
\n",
" \n",
" 14 | \n",
" Aston Villa | \n",
" 1.511905 | \n",
"
\n",
" \n",
" 15 | \n",
" Burnley | \n",
" 1.511905 | \n",
"
\n",
" \n",
" 16 | \n",
" Leeds | \n",
" 1.505952 | \n",
"
\n",
" \n",
" 17 | \n",
" Everton | \n",
" 1.464286 | \n",
"
\n",
" \n",
" 18 | \n",
" Norwich | \n",
" 1.339286 | \n",
"
\n",
" \n",
" 19 | \n",
" Watford | \n",
" 1.327381 | \n",
"
\n",
" \n",
"
\n",
"
"
],
"text/plain": [
" team rating\n",
"0 Liverpool 1.904762\n",
"1 Man City 1.892857\n",
"2 Chelsea 1.791667\n",
"3 Tottenham 1.708333\n",
"4 Arsenal 1.672619\n",
"5 Man United 1.654762\n",
"6 Brighton 1.64881\n",
"7 Crystal Palace 1.625\n",
"8 West Ham 1.619048\n",
"9 Leicester 1.607143\n",
"10 Newcastle 1.583333\n",
"11 Wolves 1.559524\n",
"12 Southampton 1.541667\n",
"13 Brentford 1.529762\n",
"14 Aston Villa 1.511905\n",
"15 Burnley 1.511905\n",
"16 Leeds 1.505952\n",
"17 Everton 1.464286\n",
"18 Norwich 1.339286\n",
"19 Watford 1.327381"
]
},
"execution_count": 3,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"colley = pb.ratings.Colley(\n",
" df[\"goals_home\"], df[\"goals_away\"], df[\"team_home\"], df[\"team_away\"]\n",
")\n",
"colley.get_ratings()"
]
},
{
"cell_type": "markdown",
"id": "f2f7e397-82de-4086-87a4-a6e76fe1251c",
"metadata": {},
"source": [
"### Colley ratings excluding tied scorelines"
]
},
{
"cell_type": "code",
"execution_count": 4,
"id": "7515c4ce-e932-46f6-92da-4be85185a98f",
"metadata": {},
"outputs": [
{
"data": {
"text/html": [
"\n",
"\n",
"
\n",
" \n",
" \n",
" | \n",
" team | \n",
" rating | \n",
"
\n",
" \n",
" \n",
" \n",
" 0 | \n",
" Liverpool | \n",
" 0.809524 | \n",
"
\n",
" \n",
" 1 | \n",
" Man City | \n",
" 0.809524 | \n",
"
\n",
" \n",
" 2 | \n",
" Chelsea | \n",
" 0.678571 | \n",
"
\n",
" \n",
" 3 | \n",
" Tottenham | \n",
" 0.630952 | \n",
"
\n",
" \n",
" 4 | \n",
" Arsenal | \n",
" 0.607143 | \n",
"
\n",
" \n",
" 5 | \n",
" Man United | \n",
" 0.547619 | \n",
"
\n",
" \n",
" 6 | \n",
" West Ham | \n",
" 0.52381 | \n",
"
\n",
" \n",
" 7 | \n",
" Brighton | \n",
" 0.511905 | \n",
"
\n",
" \n",
" 8 | \n",
" Leicester | \n",
" 0.5 | \n",
"
\n",
" \n",
" 9 | \n",
" Crystal Palace | \n",
" 0.488095 | \n",
"
\n",
" \n",
" 10 | \n",
" Newcastle | \n",
" 0.47619 | \n",
"
\n",
" \n",
" 11 | \n",
" Wolves | \n",
" 0.47619 | \n",
"
\n",
" \n",
" 12 | \n",
" Brentford | \n",
" 0.440476 | \n",
"
\n",
" \n",
" 13 | \n",
" Aston Villa | \n",
" 0.428571 | \n",
"
\n",
" \n",
" 14 | \n",
" Southampton | \n",
" 0.416667 | \n",
"
\n",
" \n",
" 15 | \n",
" Leeds | \n",
" 0.392857 | \n",
"
\n",
" \n",
" 16 | \n",
" Burnley | \n",
" 0.380952 | \n",
"
\n",
" \n",
" 17 | \n",
" Everton | \n",
" 0.380952 | \n",
"
\n",
" \n",
" 18 | \n",
" Norwich | \n",
" 0.25 | \n",
"
\n",
" \n",
" 19 | \n",
" Watford | \n",
" 0.25 | \n",
"
\n",
" \n",
"
\n",
"
"
],
"text/plain": [
" team rating\n",
"0 Liverpool 0.809524\n",
"1 Man City 0.809524\n",
"2 Chelsea 0.678571\n",
"3 Tottenham 0.630952\n",
"4 Arsenal 0.607143\n",
"5 Man United 0.547619\n",
"6 West Ham 0.52381\n",
"7 Brighton 0.511905\n",
"8 Leicester 0.5\n",
"9 Crystal Palace 0.488095\n",
"10 Newcastle 0.47619\n",
"11 Wolves 0.47619\n",
"12 Brentford 0.440476\n",
"13 Aston Villa 0.428571\n",
"14 Southampton 0.416667\n",
"15 Leeds 0.392857\n",
"16 Burnley 0.380952\n",
"17 Everton 0.380952\n",
"18 Norwich 0.25\n",
"19 Watford 0.25"
]
},
"execution_count": 4,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"colley = pb.ratings.Colley(\n",
" df[\"goals_home\"],\n",
" df[\"goals_away\"],\n",
" df[\"team_home\"],\n",
" df[\"team_away\"],\n",
" include_draws=False,\n",
")\n",
"colley.get_ratings()"
]
}
],
"metadata": {
"kernelspec": {
"display_name": "venv",
"language": "python",
"name": "python3"
},
"language_info": {
"codemirror_mode": {
"name": "ipython",
"version": 3
},
"file_extension": ".py",
"mimetype": "text/x-python",
"name": "python",
"nbconvert_exporter": "python",
"pygments_lexer": "ipython3",
"version": "3.13.1"
}
},
"nbformat": 4,
"nbformat_minor": 5
}