My submission to the posit table contest

Creating a stunning table with TSA airport checking data

technical
polars
Author

Joram Mutenge

Published

June 13, 2024

This year, I decided to participate in the Posit Table Contest. Because I’m a fan of the great tables Python library I thought it would be great to enhance my mastery of this beautiful library by designing a table. I’ll use the Polars library for data munging and transformation.

snapshot of table

Data collection

The data was collected from the Transport Security Administration website, which has archived data on the number of airport passenger checkings done daily from 2019 to 2023. The data was then saved as a parquet file containing columns Date and Numbers.

Data transformation

Reading the original data and displaying five random records.

import polars as pl
from pathlib import Path

data = pl.read_parquet(f"{Path('../../../')}/datasets/tsa.parquet")
data.sample(5)
shape: (5, 2)
Date Numbers
date i64
2023-03-22 2203488
2023-11-14 2035761
2023-05-21 2673457
2020-01-27 2073819
2020-07-04 495309


Creating new columns to enable more data transformations.

df = (data
 .with_columns(Year=pl.col('Date').dt.year(),
             Month=pl.col('Date').dt.month(),
             Day=pl.col('Date').dt.day())
 .with_columns(pl.when(pl.col('Month').eq(11) & pl.col('Day').eq(28))
             .then(pl.lit('Thanksgiving'))
             .when(pl.col('Month').eq(12) & pl.col('Day').eq(25))
             .then(pl.lit('Christmas'))
             .when(pl.col('Month').eq(7) & pl.col('Day').eq(4))
             .then(pl.lit('July 4th'))
             .when(pl.col('Month').eq(5) & pl.col('Day').eq(27))
             .then(pl.lit('Memorial Day'))
             .otherwise(None)
             .alias('Holiday')
             )
)
df.head()
shape: (5, 6)
Date Numbers Year Month Day Holiday
date i64 i32 i8 i8 str
2019-01-01 2201765 2019 1 1 null
2019-01-02 2424225 2019 1 2 null
2019-01-03 2279384 2019 1 3 null
2019-01-04 2230078 2019 1 4 null
2019-01-05 2049460 2019 1 5 null


In the code below, I’m creating a single-row dataframe, showing the highest number of checkings for each year. I’m leveraging looping to avoid repeating myself five times. The five dataframes are combined into a single dataframe called highest_df.

# Create high dataframe
high_dfs = []
for year in df['Year'].unique().to_list():
    high_df = pl.DataFrame({'Year':year,
            'Date':(df.filter(pl.col('Year') == year).filter(pl.col('Numbers') == pl.col('Numbers').max())['Date']),
            'Numbers':df.filter(pl.col('Year') == year)['Numbers'].max(),
            'Holiday':'Highest Record',
            })
    high_dfs.append(high_df)
highest_df = pl.concat(high_dfs).with_columns(pl.col('Numbers').cast(pl.Int64))
highest_df.sample(3)
shape: (3, 4)
Year Date Numbers Holiday
i32 date i64 str
2019 2019-12-01 2882915 "Highest Record"
2021 2021-11-28 2458325 "Highest Record"
2022 2022-11-27 2639616 "Highest Record"


To create a dataframe with the lowest number of checkings, I repeat the above process, replacing max() with min().

# Create low dataframe
low_dfs = []
for year in df['Year'].unique().to_list():
    low_df = pl.DataFrame({'Year':year,
            'Date':(df.filter(pl.col('Year') == year).filter(pl.col('Numbers') == pl.col('Numbers').min())['Date']),
            'Numbers':df.filter(pl.col('Year') == year)['Numbers'].min(),
            'Holiday':'Lowest Record',
            })
    low_dfs.append(low_df)
lowest_df = pl.concat(low_dfs).with_columns(pl.col('Numbers').cast(pl.Int64))
lowest_df.sample(3)
shape: (3, 4)
Year Date Numbers Holiday
i32 date i64 str
2023 2023-01-31 1534786 "Lowest Record"
2022 2022-01-25 1063856 "Lowest Record"
2021 2021-01-26 508467 "Lowest Record"


In this code, I create another dataframe with the total number of checkings for each year by using sum().

# Create total dataframe
tot_dfs = []
for year in df['Year'].unique().to_list():
    tot_df = pl.DataFrame({'Year':year,
            'Date':None,
            'Numbers':df.filter(pl.col('Year') == year)['Numbers'].sum(),
            'Holiday':'Annual Checkings',
            'Distribution':None
            })
    tot_dfs.append(tot_df)
total_df = pl.concat(tot_dfs).with_columns(pl.col('Year').cast(pl.Int32))
total_df.sample(3)
shape: (3, 5)
Year Date Numbers Holiday Distribution
i32 null i64 str null
2019 null 848102043 "Annual Checkings" null
2022 null 760071362 "Annual Checkings" null
2020 null 339774756 "Annual Checkings" null


This code creates a dataframe where the value in Holiday is not null.

# Create holiday dataframe
hol_dfs = []
for year in df['Year'].unique().to_list():
    hol_df = (df
    .filter(pl.col('Year') == year)
    .filter(pl.col('Holiday').is_not_null())
    .select('Year','Date','Numbers','Holiday')
    )
    hol_dfs.append(hol_df)
holiday_dfs = pl.concat(hol_dfs)
holiday_dfs.sample(3)
shape: (3, 4)
Year Date Numbers Holiday
i32 date i64 str
2019 2019-12-25 1996541 "Christmas"
2019 2019-05-27 2512237 "Memorial Day"
2022 2022-11-28 2389672 "Thanksgiving"


The dataframe below creates a list of the average number of monthly checkings for each year. The created lists are row values for a column called Distribution.

# Create distribution dataframe
dist_dfs = []
for year in df['Year'].unique().to_list():
    dist_df = (df
    .filter(pl.col('Year') == year)
    .group_by('Month')
    .agg(pl.mean('Numbers'), pl.first('Year'))
    .sort('Month')
    .with_columns(Distribution=pl.col('Numbers').implode())
    .select('Year','Distribution').head(1)
    )
    dist_dfs.append(dist_df)
distribution_df = pl.concat(dist_dfs)
distribution_df.sample(3)
shape: (3, 2)
Year Distribution
i32 list[f64]
2022 [1.4902e6, 1.7396e6, … 2.1333e6]
2023 [1.9647e6, 2.0843e6, … 2.3407e6]
2021 [803976.419355, 879766.035714, … 1.9142e6]


Finally, I’m combining all the created dataframes into a single dataframe called TSA and adding another column Icon containing the names of the icons like the turkey and Christmas tree used in the stunning table. TSA is the dataframe used to make the stunning table.

# Combine all dataframes and add Icon column.
TSA = (pl.concat([holiday_dfs, highest_df, lowest_df])
.join(distribution_df, on='Year', how='inner')
.vstack(total_df)
.with_columns(pl.when(pl.col('Holiday') == "Thanksgiving")
            .then(pl.col('Distribution'))
            .otherwise(None)
            .alias('Distribution')
            )
.with_columns(pl.when(pl.col('Holiday') == "Memorial Day")
           .then(pl.lit('memorial.svg'))
           .when(pl.col('Holiday') == "July 4th")
           .then(pl.lit('flag.svg'))
           .when(pl.col('Holiday') == "Thanksgiving")
           .then(pl.lit('turkey.svg'))
           .when(pl.col('Holiday') == "Christmas")
           .then(pl.lit('christmas.svg'))
           .when(pl.col('Holiday') == "Highest Record")
           .then(pl.lit('high.svg'))
           .when(pl.col('Holiday') == "Lowest Record")
           .then(pl.lit('low.svg'))
           .otherwise(pl.lit('calendar.svg'))
           .alias('Icon')
           )
.select('Icon', 'Holiday', 'Year', 'Date', 'Numbers', 'Distribution')
.sort('Year')
)
TSA.sample(3)
shape: (3, 6)
Icon Holiday Year Date Numbers Distribution
str str i32 date i64 list[f64]
"memorial.svg" "Memorial Day" 2021 2021-05-27 1867067 null
"flag.svg" "July 4th" 2021 2021-07-04 1697422 null
"turkey.svg" "Thanksgiving" 2023 2023-11-28 2171943 [1.9647e6, 2.0843e6, … 2.3407e6]


After transforming the data into the desired format, I am now creating the table. The code below generates the table, which will be my final submission.

from great_tables import GT, html, loc, style, md, nanoplot_options

display(
    GT(TSA, rowname_col="Icon", groupname_col="Year")
    .tab_stubhead(label=html('<b style="font-family: Inter, sans-serif; font-weight: 500;">Year</b>'))
    .tab_header(title=html('''
        <link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;900&display=swap" rel="stylesheet">
        <h2 style="text-align:center; display: flex; align-items: center; justify-content: center; font-family: Inter, sans-serif; font-weight: 500; color: #014eac;">
            <img src="assets/plane3.svg" alt="Plane Icon" style="margin-right: 34px; height: 54px;">
            TSA Airport Checkings<br>on Major Holidays
            <img src="assets/plane3.svg" alt="Plane Icon" style="margin-left: 34px; height: 54px;">
        </h2>
    '''))
    .tab_options(container_width="100%",
                 table_background_color='#F0FFF0',
                 heading_background_color="#C0C0C0",
                 column_labels_background_color="#696969",
                 row_group_font_weight='bold',
                 row_group_background_color='#C0C0C0',
                 source_notes_font_size='12px',
                 row_group_padding='8px',
                 table_font_names='Inter')
    .cols_label(Date=html('<b style="font-family: Inter, sans-serif; font-weight: 500;">Date</b>'),
                Numbers=html('<b style="font-family: Inter, sans-serif; font-weight: 500;">Checkings</b>'),
                Distribution=html('<b style="font-family: Inter, sans-serif; font-weight: 500;">Avg Monthly Checkings</b>'),
                Holiday='')
    .fmt_number(columns='Numbers', decimals=0)
    .cols_width(cases={'Date':'120px'})
    .fmt_date(columns="Date", date_style="day_m")
    .tab_style(style=style.text(color='#556B2F', weight='bold'),
               locations=loc.body(rows=pl.col("Holiday") == "Annual Checkings"))
    .tab_style(style=style.text(color='black', weight='normal'),
               locations=loc.body(columns="Holiday"))
    .sub_missing(missing_text='')
    .fmt_nanoplot(columns="Distribution", reference_line="mean",
                  options=nanoplot_options(data_point_radius=12,
                                           data_point_stroke_color="black",
                                           data_point_stroke_width=4,
                                           data_point_fill_color="white",
                                           data_line_type="straight",
                                           data_line_stroke_color="brown",
                                           data_line_stroke_width=2,
                                           data_area_fill_color="#FF8C00",
                                           vertical_guide_stroke_color="green"))
    .fmt_image("Icon", path="assets")
    .tab_source_note(source_note=md("**Source:** [TSA Passenger Volumes](https://www.tsa.gov/travel/passenger-volumes)<br/>**Designer:** Joram Mutenge<br/>*www.jorammutenge.com*"))
)

Plane Icon TSA Airport Checkings
on Major Holidays Plane Icon

Year Date Checkings Avg Monthly Checkings
2019
Memorial Day 27 May 2,512,237
July 4th 4 Jul 2,088,760
Thanksgiving 28 Nov 1,591,158
2.32M2.55M1.99M1.99M2.09M2.35M2.33M2.41M2.55M2.54M2.45M2.22M2.32M2.26M2.36M
Christmas 25 Dec 1,996,541
Highest Record 1 Dec 2,882,915
Lowest Record 28 Nov 1,591,158
Annual Checkings 848,102,043
2020
Memorial Day 27 May 283,387
July 4th 4 Jul 495,309
Thanksgiving 28 Nov 966,412
931K2.14M136K2.09M2.14M1.20M136K253K511K708K742K756K867K874K896K
Christmas 25 Dec 683,838
Highest Record 14 Feb 2,507,588
Lowest Record 14 Apr 113,147
Annual Checkings 339,774,756
2021
Memorial Day 27 May 1,867,067
July 4th 4 Jul 1,697,422
Thanksgiving 28 Nov 2,458,325
1.60M2.06M804K804K880K1.23M1.40M1.62M1.91M2.06M1.86M1.71M1.86M1.93M1.91M
Christmas 25 Dec 1,535,935
Highest Record 28 Nov 2,458,325
Lowest Record 26 Jan 508,467
Annual Checkings 585,250,987
2022
Memorial Day 27 May 2,397,928
July 4th 4 Jul 2,086,854
Thanksgiving 28 Nov 2,389,672
2.08M2.29M1.49M1.49M1.74M2.06M2.12M2.18M2.29M2.29M2.19M2.12M2.19M2.16M2.13M
Christmas 25 Dec 1,800,463
Highest Record 27 Nov 2,639,616
Lowest Record 25 Jan 1,063,856
Annual Checkings 760,071,362
2023
Memorial Day 27 May 2,206,475
July 4th 4 Jul 2,007,441
Thanksgiving 28 Nov 2,171,943
2.35M2.58M1.96M1.96M2.08M2.32M2.34M2.41M2.57M2.58M2.43M2.34M2.44M2.39M2.34M
Christmas 25 Dec 2,028,266
Highest Record 26 Nov 2,908,785
Lowest Record 31 Jan 1,534,786
Annual Checkings 858,548,196
Source: TSA Passenger Volumes
Designer: Joram Mutenge
www.jorammutenge.com


Get the full code here. If you want to learn how to transform data like I did in this post, check out my Polars course.