Skip to main content
Skip to main content

Join Order Benchmark (JOB)

The Join Order Benchmark (JOB) stresses the query optimizer with 113 analytical queries over a real-world, highly-correlated dataset (a snapshot of IMDb). Since its introduction, the JOB benchmark has become the de facto standard to assess the performance of relational database query optimizers, including cardinality estimation and join order optimization. Unlike synthetic benchmarks that assume uniform, independent data, JOB uses real data with skew and correlations, which makes it a hard test for join ordering and cardinality estimation.

The dataset holds about 74 million rows across 21 tables and takes around 1.15 GiB compressed in ClickHouse.

The 113 queries are organized into 33 families (133). Queries within a family (a, b, c, ...) share the same join graph but differ in their selection predicates.

References

Creating the tables

The JOB dataset is a snapshot of IMDb with 21 tables. The table definitions are available in init.sql in the ClickHouse repository.

Each table uses the MergeTree engine sorted by its primary key column id, mirroring the original PostgreSQL schema where every table declares id integer NOT NULL PRIMARY KEY. Nullable PostgreSQL columns map to Nullable(...) types.

Create the tables:

clickhouse client --query "CREATE DATABASE IF NOT EXISTS job"
clickhouse client --database job --queries-file init.sql

Loading the data

The dataset is available as Parquet files in a public S3 bucket.

Load all 21 tables directly from S3 using the s3 table function:

for table in aka_name aka_title cast_info char_name comp_cast_type company_name \
             company_type complete_cast info_type keyword kind_type link_type \
             movie_companies movie_info movie_info_idx movie_keyword movie_link \
             name person_info role_type title; do
    clickhouse client --database job --query \
        "INSERT INTO ${table} SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/${table}.parquet', 'Parquet')"
done

Alternatively, load each table with an explicit INSERT statement. Make sure to create the tables first using init.sql, then run the inserts against the job database:

INSERT INTO aka_name SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/aka_name.parquet', 'Parquet');
INSERT INTO aka_title SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/aka_title.parquet', 'Parquet');
INSERT INTO cast_info SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/cast_info.parquet', 'Parquet');
INSERT INTO char_name SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/char_name.parquet', 'Parquet');
INSERT INTO comp_cast_type SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/comp_cast_type.parquet', 'Parquet');
INSERT INTO company_name SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/company_name.parquet', 'Parquet');
INSERT INTO company_type SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/company_type.parquet', 'Parquet');
INSERT INTO complete_cast SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/complete_cast.parquet', 'Parquet');
INSERT INTO info_type SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/info_type.parquet', 'Parquet');
INSERT INTO keyword SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/keyword.parquet', 'Parquet');
INSERT INTO kind_type SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/kind_type.parquet', 'Parquet');
INSERT INTO link_type SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/link_type.parquet', 'Parquet');
INSERT INTO movie_companies SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/movie_companies.parquet', 'Parquet');
INSERT INTO movie_info SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/movie_info.parquet', 'Parquet');
INSERT INTO movie_info_idx SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/movie_info_idx.parquet', 'Parquet');
INSERT INTO movie_keyword SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/movie_keyword.parquet', 'Parquet');
INSERT INTO movie_link SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/movie_link.parquet', 'Parquet');
INSERT INTO name SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/name.parquet', 'Parquet');
INSERT INTO person_info SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/person_info.parquet', 'Parquet');
INSERT INTO role_type SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/role_type.parquet', 'Parquet');
INSERT INTO title SELECT * FROM s3('https://s3.eu-west-3.amazonaws.com/public-pme/join_bench/job/title.parquet', 'Parquet');

Detailed table sizes:

Tablesize (in rows)size (compressed in ClickHouse)
aka_name901,34331.86 MiB
aka_title361,47214.32 MiB
cast_info36,244,344296.25 MiB
char_name3,140,339107.95 MiB
comp_cast_type4132.00 B
company_name234,9978.38 MiB
company_type4162.00 B
complete_cast135,086748.80 KiB
info_type1131.25 KiB
keyword134,1701.88 MiB
kind_type7177.00 B
link_type18284.00 B
movie_companies2,609,12921.20 MiB
movie_info14,835,720300.46 MiB
movie_info_idx1,380,0358.01 MiB
movie_keyword4,523,93021.06 MiB
movie_link29,997178.21 KiB
name4,167,491131.16 MiB
person_info2,963,664154.12 MiB
role_type12246.00 B
title2,528,31278.04 MiB
Total74,190,1871.15 GiB

(Compressed sizes in ClickHouse are taken from system.tables.total_bytes and based on the above table definitions.)

Queries

The 113 JOB queries can be found here in the ClickHouse repository. The settings used to run them are in settings.json. See the README for known issues and notes on specific queries.

The queries reference the tables by name, so run them against the job database (for example, with clickhouse client --database job).

An example query (1a):

SELECT
    MIN(mc.note) AS production_note,
    MIN(t.title) AS movie_title,
    MIN(t.production_year) AS movie_year
FROM company_type AS ct, info_type AS it, movie_companies AS mc, movie_info_idx AS mi_idx, title AS t
WHERE (ct.kind = 'production companies') AND (it.info = 'top 250 rank') AND (mc.note NOT LIKE '%(as Metro-Goldwyn-Mayer Pictures)%') AND ((mc.note LIKE '%(co-production)%') OR (mc.note LIKE '%(presents)%')) AND (ct.id = mc.company_type_id) AND (t.id = mc.movie_id) AND (t.id = mi_idx.movie_id) AND (mc.movie_id = mi_idx.movie_id) AND (it.id = mi_idx.info_type_id);

Preparing the data from the original CSV files

The Parquet files above are derived from the original IMDb snapshot used by JOB, which is distributed as one CSV file per table (aka_name.csv, title.csv, ...). These CSVs use PostgreSQL COPY semantics with ESCAPE '\': a backslash escapes the quote character only inside a quoted field, while outside quotes a backslash is a literal character. ClickHouse expects RFC 4180 CSV (doubled quotes, no backslash escaping), so the files must be re-encoded first.

convert_csv.py performs that re-encoding. It reads the original CSV on stdin and writes standard CSV on stdout, doubling embedded quotes and preserving empty unquoted fields (which ClickHouse maps to NULL for Nullable columns).

To build the tables from the original CSVs:

clickhouse client --query "CREATE DATABASE IF NOT EXISTS job"
clickhouse client --database job --queries-file init.sql

for table in aka_name aka_title cast_info char_name comp_cast_type company_name \
             company_type complete_cast info_type keyword kind_type link_type \
             movie_companies movie_info movie_info_idx movie_keyword movie_link \
             name person_info role_type title; do
    python3 convert_csv.py < "${table}.csv" \
        | clickhouse client --database job --query "INSERT INTO ${table} FORMAT CSV"
done

Once the tables are populated, they can be exported to Parquet for faster re-import later, e.g. clickhouse client --database job --query "SELECT * FROM title ORDER BY id FORMAT Parquet" > title.parquet.