20.1. Design Overview¶
20.1.1. Motivation¶
PostGIS is a widely used spatial database that is easily deployed, however it is limited in the total number of records it can handle in a single table. This module makes it possible to store much larger data sets using PostgreSQL’s native partitioning, by automatically sorting data into time-based partitions.
20.1.2. Overview¶
The PostGIS partitioning module is an extension of the GeoTools JDBC data store, implemented as a custom dialect. It is designed for sensor data, where each record has a timestamp and records are mostly ingested soon after they are generated.
As data is written, it gets sorted into partitions based on record timestamps. Recent data gets sorted
into smaller partitions, then into larger partitions once it has reached a certain age. The partitions are
sorted spatio-temporally, to reduce page reads during typical queries. Since the data is sorted, it can be
effectively indexed using a spatial BRIN
index, which is much smaller than a standard GIST
index. During
queries, PostgreSQL will automatically skip over partitions that don’t match the query, using standard partition
pruning. This allows massive data sets to be efficiently queried with temporal predicates, although non-temporal
queries may not perform well.
20.1.3. Table Design¶
The data is split into one primary view and three tables, names of which are prefixed by the feature type:
main view (named after the feature type) - a
UNION ALL
of the other tables, this is the view that should generally be used for all external reads and writes.
_wa
- the write-ahead table. All writes to the main view are delegated to this table using a trigger. The table is partitioned using table inheritance (as there may be overlap between the partitions), and is rolled over every 10 minutes. Only the most recent partition is ever written to.
_wa_partition
- recent data, partitioned into 10 minute increments using declarative partitioning. The most recent data is stored in this table in spatially-sorted order, copied out of the write ahead table when it is rolled over (after which the write ahead table partition is dropped). This table is designed to handle time-latent data, and to store enough data to fill an entire main partition, while still providing partition pruning for queries.
_partition
- the main data, partitioned using declarative time partitioning. Data is copied into this table from the_wa_partition
table in spatially-sorted order, once enough time has elapsed (after which the_wa_partition
partitions are dropped). It uses a BRIN index, which is small but performs well on sorted data. Keeping the data sorted also reduces the number of page hits required for most spatial queries.
20.1.4. Helper Tables¶
In addition to the ones above, the following additional tables are used:
_analyze_queue
- tracks partitions which have been modified. Modified partitions will haveANALYZE
invoked on them in a separate process.
_sort_queue
- tracks out-of-order inserts into the main partition table. This can be used to diagnose slow queries, as unsorted data can negatively impact the effectiveness of theBRIN
index.
20.1.5. Maintenance Scripts¶
Several PLPG/SQL procedures are used to move data around according to the design described above. The pg_cron
extension is used to schedule these tasks. The script names are prefixed by the feature type, and created based
on the feature type configuration (i.e. field names, partition size, etc). The procedures consist of:
_roll_wa
- creates a new write-ahead table every 10 minutes
_partition_maintenance
umbrella function for invoking all the partition scripts at once
_partition_wa
- moves data from old write ahead tables into the_wa_partition
table
_merge_wa_partitions
- moves data from the_wa_partition
table into the main_partition
table
_age_off
(if configured) - drops any partitions that have exceeded the maximum amount of data to keep
_analyze_partitions
- runsANALYZE
on any partitions that have been updated
_partition_sort
- manually called to re-sort a partition