Deflategate
On January 18, 2015, the Indianapolis Colts and the New England Patriots played the American Football Conference (AFC) championship game to determine which of those teams would play in the Super Bowl. After the game, there were allegations that the Patriots’ footballs had not been inflated as much as the regulations required; they were softer. This could be an advantage, as softer balls might be easier to catch.
For several weeks, the world of American football was consumed by accusations, denials, theories, and suspicions: the press labeled the topic Deflategate, after the Watergate political scandal of the 1970’s. The National Football League (NFL) commissioned an independent analysis. In this example, we will perform our own analysis of the data.
Pressure is often measured in pounds per square inch (psi). NFL rules stipulate that game balls must be inflated to have pressures in the range 12.5 psi and 13.5 psi. Each team plays with 12 balls. Teams have the responsibility of maintaining the pressure in their own footballs, but game officials inspect the balls. Before the start of the AFC game, all the Patriots’ balls were at about 12.5 psi. Most of the Colts’ balls were at about 13.0 psi. However, these pre-game data were not recorded.
During the second quarter, the Colts intercepted a Patriots ball. On the sidelines, they measured the pressure of the ball and determined that it was below the 12.5 psi threshold. Promptly, they informed officials.
At half-time, all the game balls were collected for inspection. Two officials, Clete Blakeman and Dyrol Prioleau, measured the pressure in each of the balls.
Here are the data. Each row corresponds to one football. Pressure is measured in psi. The Patriots ball that had been intercepted by the Colts was not inspected at half-time. Nor were most of the Colts’ balls – the officials simply ran out of time and had to relinquish the balls for the start of second half play.
football = Table.read_table(path_data + 'deflategate.csv')
football.show()
Team | Blakeman | Prioleau |
---|---|---|
Patriots | 11.5 | 11.8 |
Patriots | 10.85 | 11.2 |
Patriots | 11.15 | 11.5 |
Patriots | 10.7 | 11 |
Patriots | 11.1 | 11.45 |
Patriots | 11.6 | 11.95 |
Patriots | 11.85 | 12.3 |
Patriots | 11.1 | 11.55 |
Patriots | 10.95 | 11.35 |
Patriots | 10.5 | 10.9 |
Patriots | 10.9 | 11.35 |
Colts | 12.7 | 12.35 |
Colts | 12.75 | 12.3 |
Colts | 12.5 | 12.95 |
Colts | 12.55 | 12.15 |
For each of the 15 balls that were inspected, the two officials got different results. It is not uncommon that repeated measurements on the same object yield different results, especially when the measurements are performed by different people. So we will assign to each the ball the average of the two measurements made on that ball.
football = football.with_column(
'Combined', (football.column(1)+football.column(2))/2
).drop(1, 2)
football.show()
Team | Combined |
---|---|
Patriots | 11.65 |
Patriots | 11.025 |
Patriots | 11.325 |
Patriots | 10.85 |
Patriots | 11.275 |
Patriots | 11.775 |
Patriots | 12.075 |
Patriots | 11.325 |
Patriots | 11.15 |
Patriots | 10.7 |
Patriots | 11.125 |
Colts | 12.525 |
Colts | 12.525 |
Colts | 12.725 |
Colts | 12.35 |
At a glance, it seems apparent that the Patriots’ footballs were at a lower pressure than the Colts’ balls. Because some deflation is normal during the course of a game, the independent analysts decided to calculate the drop in pressure from the start of the game. Recall that the Patriots’ balls had all started out at about 12.5 psi, and the Colts’ balls at about 13.0 psi. Therefore the drop in pressure for the Patriots’ balls was computed as 12.5 minus the pressure at half-time, and the drop in pressure for the Colts’ balls was 13.0 minus the pressure at half-time.
We can calculate the drop in pressure for each football, by first setting up an array of the starting values. For this we will need an array consisting of 11 values each of which is 12.5, and another consisting of four values each of which is all 13. We will use the NumPy function np.ones
, which takes a count as its argument and returns an array of that many elements, each of which is 1.
np.ones(11)
array([ 1., 1., 1., 1., 1., 1., 1., 1., 1., 1., 1.])
patriots_start = 12.5 * np.ones(11)
colts_start = 13 * np.ones(4)
start = np.append(patriots_start, colts_start)
start
array([ 12.5, 12.5, 12.5, 12.5, 12.5, 12.5, 12.5, 12.5, 12.5,
12.5, 12.5, 13. , 13. , 13. , 13. ])
The drop in pressure for each football is the difference between the starting pressure and the combined pressure measurement.
drop = start - football.column('Combined')
football = football.with_column('Pressure Drop', drop)
football.show()
Team | Combined | Pressure Drop |
---|---|---|
Patriots | 11.65 | 0.85 |
Patriots | 11.025 | 1.475 |
Patriots | 11.325 | 1.175 |
Patriots | 10.85 | 1.65 |
Patriots | 11.275 | 1.225 |
Patriots | 11.775 | 0.725 |
Patriots | 12.075 | 0.425 |
Patriots | 11.325 | 1.175 |
Patriots | 11.15 | 1.35 |
Patriots | 10.7 | 1.8 |
Patriots | 11.125 | 1.375 |
Colts | 12.525 | 0.475 |
Colts | 12.525 | 0.475 |
Colts | 12.725 | 0.275 |
Colts | 12.35 | 0.65 |
It looks as though the Patriots’ drops were larger than the Colts’. Let’s look at the average drop in each of the two groups. We no longer need the combined scores.
football = football.drop('Combined')
football.group('Team', np.average)
Team | Pressure Drop average |
---|---|
Colts | 0.46875 |
Patriots | 1.20227 |
The average drop for the Patriots was about 1.2 psi compared to about 0.47 psi for the Colts.
The question now is why the Patriots’ footballs had a larger drop in pressure, on average, than the Colts footballs. Could it be due to chance?
The Hypotheses
How does chance come in here? Nothing was being selected at random. But we can make a chance model by hypothesizing that the 11 Patriots’ drops look like a random sample of 11 out of all the 15 drops, with the Colts’ drops being the remaining four. That’s a completely specified chance model under which we can simulate data. So it’s the null hypothesis.
For the alternative, we can take the position that the Patriots’ drops are too large, on average, to resemble a random sample drawn from all the drops.
Test Statistic
A natural statistic is the difference between the two average drops, which we will compute as “average drop for Patriots - average drop for Colts”. Large values of this statistic will favor the alternative hypothesis.
observed_means = football.group('Team', np.average).column(1)
observed_difference = observed_means.item(1) - observed_means.item(0)
observed_difference
0.733522727272728
This positive difference reflects the fact that the average drop in pressure of the Patriots’ balls was greater than that of the Colts.
Predicting the Statistic Under the Null Hypothesis
If the null hypothesis were true, then the Patriots’ drops would be comparable to 11 drops drawn at random without replacement from all 15 drops, and the Colts’ drops would be the remaining four. We can simulate this by randomly permuting all 15 drops and assigning each team the appropriate number of permuted values.
shuffled_drops = football.sample(with_replacement=False).column(1)
original_and_shuffled = football.with_column('Shuffled Drop', shuffled_drops)
original_and_shuffled.show()
Team | Pressure Drop | Shuffled Drop |
---|---|---|
Patriots | 0.85 | 1.175 |
Patriots | 1.475 | 1.175 |
Patriots | 1.175 | 1.65 |
Patriots | 1.65 | 1.475 |
Patriots | 1.225 | 1.225 |
Patriots | 0.725 | 0.475 |
Patriots | 0.425 | 0.725 |
Patriots | 1.175 | 0.65 |
Patriots | 1.35 | 0.85 |
Patriots | 1.8 | 0.425 |
Patriots | 1.375 | 0.275 |
Colts | 0.475 | 1.375 |
Colts | 0.475 | 0.475 |
Colts | 0.275 | 1.8 |
Colts | 0.65 | 1.35 |
How do all the group averages compare?
original_and_shuffled.group('Team', np.average)
Team | Pressure Drop average | Shuffled Drop average |
---|---|---|
Colts | 0.46875 | 1.25 |
Patriots | 1.20227 | 0.918182 |
The two teams’ average drop values are closer when the balls are randomly assigned to the two teams than they were for the balls actually used in the game.
Permutation Test
It’s time for a step that is now familiar. We will do repeated simulations of the test statistic under the null hypothesis, by repeatedly permuting the footballs and assigning random sets to the two teams.
In the last section we defined a function called permuted_sample_average_difference
to do this. Here is the definition again. The code is based on the steps we took to compare the averages of the shuffled data.
def permuted_sample_average_difference(table, label, group_label, repetitions):
tbl = table.select(group_label, label)
differences = make_array()
for i in np.arange(repetitions):
shuffled = tbl.sample(with_replacement = False).column(1)
original_and_shuffled = tbl.with_column('Shuffled Data', shuffled)
shuffled_means = original_and_shuffled.group(group_label, np.average).column(2)
simulated_difference = shuffled_means.item(1) - shuffled_means.item(0)
differences = np.append(differences, simulated_difference)
return differences
differences = permuted_sample_average_difference(football, 'Pressure Drop', 'Team', 10000)
The array differences
contains 10,000 values of the test statistic simulated under the null hypothesis.
Conclusion of the Test
To calculate the empirical P-value, it’s important to recall the alternative hypothesis, which is that the Patriots’ drops are too large to be the result of chance variation alone.
The “direction of the alternative” is towards large drops for the Patriots, with correspondingly large values for out test statistic “Patriots’ average - Colts’ average”. So the P-value is the chance (computed under the null hypothesis) of getting a test statistic equal to our observed value of 0.73352272727272805 or larger.
empirical_P = np.count_nonzero(differences >= observed_difference) / 10000
empirical_P
0.0041
That’s a pretty small P-value. To visualize this, here is the empirical distribution of the test statistic under the null hypothesis, with the observed statistic marked on the horizontal axis.
Table().with_column('Difference Between Group Averages', differences).hist()
plots.scatter(observed_difference, 0, color='red', s=30)
plots.title('Prediction Under the Null Hypothesis')
print('Observed Difference:', observed_difference)
print('Empirical P-value:', empirical_P)
Observed Difference: 0.733522727272728
Empirical P-value: 0.0041
As in previous examples of this test, the bulk of the distribution is centered around 0. Under the null hypothesis, the Patriots’ drops are a random sample of all 15 drops, and therefore so are the Colts’. Therefore the two sets of drops should be about equal on average, and therefore their difference should be around 0.
But the observed value of the test statistic is quite far away from the heart of the distribution. By any reasonable cutoff for what is “small”, the empirical P-value is small. So we end up rejecting the null hypothesis of randomness, and conclude that the Patriots drops were too large to reflect chance variation alone.
The independent investigative team analyzed the data in several different ways, taking into account the laws of physics. The final report said,
“[T]he average pressure drop of the Patriots game balls exceeded the average pressure drop of the Colts balls by 0.45 to 1.02 psi, depending on various possible assumptions regarding the gauges used, and assuming an initial pressure of 12.5 psi for the Patriots balls and 13.0 for the Colts balls.”
– Investigative report commissioned by the NFL regarding the AFC Championship game on January 18, 2015
Our analysis shows an average pressure drop of about 0.73 psi, which is close to the center of the interval “0.45 to 1.02 psi” and therefore consistent with the official analysis.
Remember that our test of hypotheses does not establish the reason why the difference is not due to chance. Establishing causality is usually more complex than running a test of hypotheses.
But the all-important question in the football world was about causation: the question was whether the excess drop of pressure in the Patriots’ footballs was deliberate. If you are curious about the answer given by the investigators, here is the full report.