pg-index-health – a static analysis tool for you PostgreSQL database

Hello there!

Since 2019, I have been developing an open-source tool called pg-index-health, which analyzes database structures and identifies potential issues. In one of my previous articles, I shared the story of how this tool came to life.

Over the years, pg-index-health has evolved and improved. In 2024, with the support of several contributors, I managed to address most of its remaining “growing pains” and bring the project to a state where it is ready for large-scale expansion.

The Growth of Databases with the Rise of Microservices

I’ve been working with PostgreSQL since 2015, and this fascinating journey began at the Yaroslavl-based company Tensor.

Back in 2015, it was still the era of monoliths with massive databases and a large number of tables. Typically, any changes to the structure of such databases required mandatory approval from an architect or a development lead, who served as the key knowledge holder. While this safeguarded against most errors, it slowed down the process of making changes and was entirely unscalable.

Gradually, people began transitioning to microservices.
The number of databases grew significantly, but the number of tables within each database, conversely, decreased. Now, each team started managing the structure of its own database independently. The centralized source of expertise disappeared, and database design errors began to multiply and propagate from one service to another.

The Testing Pyramid and Its Shapes

Most of you have probably heard of the testing pyramid. For monoliths, it has a fairly characteristic shape with a broad base of unit tests. For more details, I recommend Martin Fowler’s article.

Microservices have changed not only the approach to development but also the appearance of the testing pyramid. This shift was largely driven by the rise of containerization technologies (Docker, Testcontainers). Today the testing pyramid is no longer a pyramid at all. It can have a very bizarre shape. The most well-known examples are the Honeycomb and the Testing Trophy.

The modern trend is to write as few unit tests as possible, focusing on implementation details, and to prioritize component and integration tests that validate the actual functionality provided by the service.

My personal favorite is the Testing Trophy. At its base lies static code analysis, designed to guard against common errors.

The Importance of Static Code Analysis

Static analysis for Java and Kotlin code is now common practice. For Kotlin services, the tool of choice is typically detekt. For Java applications, the range of available tools (often referred to as linters) is broader. The main tools include Checkstyle, PMD, SpotBugs, and Error Prone. You can read more about them in my previous article.

Notably, both detekt and Checkstyle also handle code formatting, effectively functioning as formatters.

Static Analysis for Database Migrations

Modern microservices often include database migrations for creating and updating the database structure alongside application code.

In the Java ecosystem, the main tools for managing migrations are Liquibase and Flyway. Any changes to the database structure must always be documented in migrations. Even if changes are made manually during an incident in production, a migration must later be created to apply those changes across all environments.

Writing migrations in plain SQL is the best practice as it provides maximum flexibility and saves time compared to learning the XML dialects of tools like Liquibase. I touched on this in my article, “Six Tips for Using PostgreSQL in Functional Tests”.

Verifying SQL Migration Code

To verify the SQL code in migrations, I recommend using SQLFluff, which is essentially a Checkstyle equivalent for SQL. This linter supports multiple databases and dialects (including PostgreSQL) and can be integrated into your CI pipeline. It offers over 60 customizable rules, enabling you to manage table and column aliases, SQL command casing, indentation, column ordering in queries, and much more.

Compare the query with and without formatting:

-- well-formatted SQL
select
    pc.oid::regclass::text as table_name,
    pg_table_size(pc.oid) as table_size
from
    pg_catalog.pg_class pc
    inner join pg_catalog.pg_namespace nsp on nsp.oid = pc.relnamespace
where
    pc.relkind = 'r' and
    pc.oid not in (
        select c.conrelid as table_oid
        from pg_catalog.pg_constraint c
        where c.contype = 'p'
    ) and
    nsp.nspname = :schema_name_param::text
order by table_name;

Enter fullscreen mode Exit fullscreen mode

-- poorly formatted SQL
SELECT pc.oid::regclass::text AS table_name, pg_table_size(pc.oid) AS table_size
FROM pg_catalog.pg_class  pc
JOIN pg_catalog.pg_namespace AS nsp
ON nsp.oid =  pc.relnamespace
WHERE pc.relkind = 'r’ and pc.oid NOT in ( select c.conrelid as table_oid from pg_catalog.pg_constraint c where c.contype = 'p
)
and nsp.nspname  = :schema_name_param::text
ORDER BY  table_name;

Enter fullscreen mode Exit fullscreen mode

Well-formatted SQL code is much easier to read and understand. Most importantly, code reviews will no longer be bogged down by discussions about formatting preferences. SQLFluff enforces a consistent style, saving time.

SQLFluff in Action

This is what it looks like in a real pull request:

Here SQLFluff found a problem with formatting the return value in the select statement: when only one column is returned, we do not put it on a separate row. The second point is the incorrect column order in the selection results: first we return simple columns and only then the calculation results. And the third is the incorrect case for and in the join statement: I prefer to write all queries in lowercase.

For more examples of SQLFluff in use, check out my open-source projects: one, two.

Using Metadata to Analyze Database Structure

The structure of the database itself can also be checked. However, working with migrations is extremely inconvenient: there can be many of them; a new migration may fix errors in a previous migration, and so on. As a rule, we are more interested in the final structure of the database than its intermediate state.

Leveraging Information Schema

PostgreSQL (like many other relational databases) stores metadata about all objects and relationships between them and provides it externally in the form of information_schema. We can use queries to information_schema to identify any deviations, problems, or common errors (this is exactly what SchemaCrawler does).

Since we work only with PostgreSQL, instead of information_schema we can use system catalogs (the pg_catalog schema), which provide much more information about the internal structure of a specific database.

Cumulative Statistics System

In addition to metadata, PostgreSQL collects information about the operation of each database: what queries are executed, how they are executed, what access methods are used, etc. The Cumulative Statistics System is responsible for collecting this data.

By querying these statistics through system views and combining them with data from the system catalogs, we can:

Statistics can be reset manually. The date and time of the last reset are recorded in the system. It is important to consider this to understand whether the statistics can be trusted or not. For example, if you have some business logic that is executed once a month/quarter/half-year, the statistics need to be collected for a period of at least the interval mentioned above.

If a database cluster is used, then statistics are collected independently on each host and are not replicated within the cluster.

pg-index-health and Its Structure

The idea of analyzing database structure based on metadata within the database itself, as described above, has been implemented by me in the form of a tool called pg-index-health.

My solution includes the following components:

  • A set of checks in the form of SQL queries, placed in a separate repository (currently consisting of 25 checks). The queries are decoupled from the Java codebase and can be reused in projects written in other programming languages.
  • A domain model — a minimal set of classes that represent the results of the checks as objects.
  • The HighAvailabilityPgConnection abstraction for connecting to a database cluster consisting of multiple hosts.
  • Utilities for executing SQL queries and serializing results into domain model objects.
  • A Spring Boot starter for convenient and quick integration of checks into unit/component/integration tests.
  • A migration generator that can create corrective SQL migrations for identified issues.

Types of Checks

All checks (also referred to as diagnostics) are divided into two groups:

  • Runtime checks (require statistics).
  • Static checks (do not require statistics).

Runtime checks

Runtime checks are meaningful only when executed on a live database instance in production. These checks require accumulated statistics and aggregate this data from all hosts in the cluster.

Let’s consider a database cluster consisting of three hosts: primary, secondary, and async replica. Some services use clusters with similar topologies and execute heavy read queries only on the asynchronous replica to balance the load. Such queries are usually not executed on the primary host as they create additional load and negatively impact the latency of other queries.

As previously mentioned, in PostgreSQL, statistics are collected separately on each host and are not replicated within the cluster. Thus, you can easily encounter a situation where certain indexes are used and necessary only on the asynchronous replica. To make a reliable determination about whether an index is needed or not, it is necessary to run the check on each host in the cluster and aggregate the results.

Static checks

Static checks do not require accumulated statistics and can be executed on the primary host immediately after applying migrations. Of course, they can also be used on a production database to obtain data in real time. However, most checks are static, and they are particularly useful in tests, as they help catch and prevent common errors during the development phase.

How to Use pg-index-health

The primary use case for pg-index-health is adding tests to verify the database structure in your testing pipeline.

For Spring Boot applications, you need to add the starter to your test dependencies:

dependencies {
    testImplementation("io.github.mfvanek:pg-index-health-test-starter:0.14.4")
}

Enter fullscreen mode Exit fullscreen mode

Then add a standard test:

import io.github.mfvanek.pg.core.checks.common.DatabaseCheckOnHost;
import io.github.mfvanek.pg.core.checks.common.Diagnostic;
import io.github.mfvanek.pg.model.dbobject.DbObject;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.ActiveProfiles;

import java.util.List;

import static org.assertj.core.api.Assertions.assertThat;

@SpringBootTest
@ActiveProfiles("test")
class DatabaseStructureStaticAnalysisTest {

    @Autowired
    private List<DatabaseCheckOnHost<? extends DbObject>> checks;

    @Test
    void checksShouldWork() {
        assertThat(checks)
            .hasSameSizeAs(Diagnostic.values());

        checks.stream()
            .filter(DatabaseCheckOnHost::isStatic)
            .forEach(c -> assertThat(c.check())
                .as(c.getDiagnostic().name())
                .isEmpty());
    }
}

Enter fullscreen mode Exit fullscreen mode

In this test, all available checks are injected as a list. Then, only static checks are filtered and executed on the real database deployed in a container with applied migrations.

Ideally, each check should return an empty list. If there are any deviations when adding the next migration, the test will fail. The developer will be forced to pay attention to this and resolve the problem in any way: either fix it in his migration, or explicitly ignore it.

False Positives and Adding Exclusions

It is important to understand that pg-index-health, like any other static analyzer, may generate false positives. Additionally, some checks may not be relevant to your project. For example, it is considered good practice to document the database structure. PostgreSQL allows adding comments to almost all database objects. In a migration, this might look like the following:

create table if not exists demo.warehouse
(
    id bigint primary key generated always as identity,
    name varchar(255) not null
);

comment on table demo.warehouse is 'Information about the warehouses';
comment on column demo.warehouse.id is 'Unique identifier of the warehouse';
comment on column demo.warehouse.name is 'Human readable name of the warehouse';

Enter fullscreen mode Exit fullscreen mode

Within your team, you might agree not to do this. In that case, the results of corresponding checks (TABLES_WITHOUT_DESCRIPTION, COLUMNS_WITHOUT_DESCRIPTION, FUNCTIONS_WITHOUT_DESCRIPTION) become irrelevant for you.

You can either completely exclude these checks:

@Test
void checksShouldWork() {
    assertThat(checks)
        .hasSameSizeAs(Diagnostic.values());

    checks.stream()
        .filter(DatabaseCheckOnHost::isStatic)
        .filter(c -> c.getDiagnostic() != Diagnostic.TABLES_WITHOUT_DESCRIPTION &&
            c.getDiagnostic() != Diagnostic.COLUMNS_WITHOUT_DESCRIPTION)
        .forEach(c -> assertThat(c.check())
            .as(c.getDiagnostic().name())
            .isEmpty());
}

Enter fullscreen mode Exit fullscreen mode

Or simply ignore their results:

@Test
void checksShouldWork() {
    assertThat(checks)
        .hasSameSizeAs(Diagnostic.values());

    checks.stream()
        .filter(DatabaseCheckOnHost::isStatic)
        .forEach(c -> {
            final ListAssert<? extends DbObject> listAssert = assertThat(c.check())
                .as(c.getDiagnostic().name());
            switch (c.getDiagnostic()) {
                case TABLES_WITHOUT_DESCRIPTION, COLUMNS_WITHOUT_DESCRIPTION -> listAssert.hasSizeGreaterThanOrEqualTo(0); // ignored

                default -> listAssert.isEmpty();
            }
        });
}

Enter fullscreen mode Exit fullscreen mode

When introducing pg-index-health, you may often encounter situations where the database structure already has some deviations, but you do not want to address them immediately. At the same time, the check is relevant, and disabling it is not an option. In such cases, it is best to fix all deviations in the code:

@Test
void checksShouldWorkForAdditionalSchema() {
    final PgContext ctx = PgContext.of("additional_schema");
    checks.stream()
        .filter(DatabaseCheckOnHost::isStatic)
        .forEach(c -> {
            final ListAssert<? extends DbObject> listAssert = assertThat(c.check(ctx))
                .as(c.getDiagnostic().name());

            switch (c.getDiagnostic()) {
                case TABLES_WITHOUT_DESCRIPTION, TABLES_NOT_LINKED_TO_OTHERS ->
                    listAssert.hasSize(1)
                        .asInstanceOf(list(Table.class))
                        .containsExactly(
                            Table.of(ctx, "additional_table")
                        );

                default -> listAssert.isEmpty();
            }
        });
}

Enter fullscreen mode Exit fullscreen mode

Now, I would like to focus in more detail on the most frequently encountered issues.

Tables Without Primary Keys

Due to the specifics of the MVCC mechanism in PostgreSQL, situations like bloat can occur, where the size of your table (or index) rapidly grows due to a large number of dead tuples. This can happen, for example, as a result of long-running transactions or a one-time update of a large number of rows.

Garbage collection within the database is handled by the autovacuum process, but it does not free up the physical disk space occupied. The only way to effectively reduce the physical size of a table is by using the VACUUM FULL command, which requires an exclusive lock for the duration of the operation. For large tables, this can take several hours, making full vacuuming impractical for most modern services.

To address the issue of table bloat without downtime, third-party extensions like pg_repack are often used. One of the mandatory requirements of pg_repack is the presence of a primary key or some other uniqueness constraint on the target table. The TABLES_WITHOUT_PRIMARY_KEY diagnostic helps detect tables without primary keys and prevents maintenance problems in the future.

Below is an example of a table without a primary key. If bloat occurs in this table, pg_repack will be unable to process it and will return an error.

create table if not exists demo.payment
(
    id bigint not null, -- column is not marked as primary key
    order_id bigint references demo.orders (id),
    status int not null,
    created_at timestamp not null,
    payment_total decimal(22, 2) not null
);

Enter fullscreen mode Exit fullscreen mode

Duplicated Indexes

Our databases operate on hosts with limited resources, and disk space is one of them. When using Database-as-a-Service solutions, there is often a physical limitation on the maximum database size that cannot be changed.

Each index in a table is a separate entity on the disk. It occupies space and requires resources for maintenance, which slows down data insertion and updates. We create indexes to speed up searches or ensure the uniqueness of certain values. However, improper use of indexes can lead to a situation where their combined size exceeds the size of the useful data in the table itself. Thus, the number of indexes in a table should be minimal yet sufficient for its functionality.

I have encountered numerous cases where unnecessary indexes are created in migrations. For example, an index for a primary key is created automatically. While it is technically possible to manually index the id column, doing so is completely pointless.

create table if not exists demo.buyer
(
    id bigint primary key default nextval('demo.buyer_seq'), -- PK
    first_name varchar(255) not null,
    last_name varchar(255) not null,
    middle_name varchar(255),
    phone varchar(20) not null,
    email varchar(50) not null,
    ip_address varchar(100)
);

create unique index if not exists idx_buyer_pk on demo.buyer (id); -- possible but pointless

Enter fullscreen mode Exit fullscreen mode

A similar situation arises with unique constraints. When you mark a column (or a group of columns) with the unique keyword, PostgreSQL automatically creates an unique index for that column (or group of columns). Manually creating additional indexes is unnecessary. If done, this results in duplicated indexes. Such redundant indexes can and should be removed, and the DUPLICATED_INDEXES diagnostic can help identify them.

create table if not exists demo.order_item
(
    id bigint primary key default nextval('demo.order_item_seq'),
    order_id bigint not null references demo.orders (id),
    price decimal(22, 2) not null default 0,
    amount int not null default 0,
    sku varchar(255) not null,
    warehouse_id int not null,
    unique (sku, order_id) -- unique constraint

);

create unique index if not exists i_order_item_sku_order_id_unique on demo.order_item (sku, order_id); -- possible, but pointless

Enter fullscreen mode Exit fullscreen mode

Overlapping (intersecting) Indexes

Most indexes are created for a single column. When query optimization begins, more complex indexes may be added, involving multiple columns. This leads to scenarios where indexes are created for columns like A, A+B, and A+B+C. The first two indexes in this series can often be discarded since they are prefixes of the third (I recommend watching this video). Removing these redundant indexes can save significant disk space, and the INTERSECTED_INDEXES diagnostic is designed to detect such cases.

create table if not exists demo.buyer
(
    id bigint primary key default nextval('demo.buyer_seq'),
    first_name varchar(255) not null,
    last_name varchar(255) not null,
    middle_name varchar(255),
    phone varchar(20) not null
);

create index if not exists i_buyer_first_name
    on demo.buyer (first_name); -- can be removed

create index if not exists i_buyer_names
    on demo.buyer (first_name, last_name, middle_name);

create index if not exists i_buyer_id_phone
    on demo.buyer (id, phone); -- including the primary key in an index is almost always a bad idea

Enter fullscreen mode Exit fullscreen mode

Foreign Keys Without Indexes

PostgreSQL allows the creation of foreign key constraints without specifying a supporting index, meaning that a table referencing another does not require and will not automatically create an index. In some cases, this might not be a problem and may not manifest at all. However, sometimes it can lead to incidents in production.

Let’s look at a small example (I’m using PostgreSQL 16.6):

create schema if not exists demo;

create table if not exists demo.orders
(
    id bigint primary key generated always as identity,
    user_id bigint not null,
    shop_id bigint not null,
    status int not null,
    created_at timestamptz not null default current_timestamp
);

create table if not exists demo.order_item
(
    id bigint primary key generated always as identity,
    order_id bigint not null references demo.orders (id),
    price decimal(22, 2) not null default 0,
    amount int not null default 0,
    sku varchar(255) not null,
    warehouse_id int
);

Enter fullscreen mode Exit fullscreen mode

We have an orders table and an order_item table. They are linked via a foreign key on the order_id column. The foreign key should always reference either the primary key or some unique constraint, which is satisfied in our case.

Let’s populate the tables with data and gather statistics. We’ll add 100,000 orders, with half having two items and the others having one.

insert into demo.orders (user_id, shop_id, status) 
select
    (ids.id % 10) + 1 as user_id,
    (ids.id % 4) + 1 as shop_id,
    1 as status -- new order
from generate_series(1, 100000) ids (id);

insert into demo.order_item (order_id, price, amount, sku)
select
    id as order_id,
    (random() + 1) * 1000.0 as price,
    (random() * 10) + 1 as amount,
    md5(random()::text) as sku
from demo.orders;

insert into demo.order_item (order_id, price, amount, sku)
select
    id as order_id,
    (random() + 1) * 2000.0 as price,
    (random() * 5) + 1 as amount,
    md5((random() + 1)::text) as sku
from demo.orders where id % 2 = 0;

-- gather statistics
vacuum analyze demo.orders, demo.order_item;

Enter fullscreen mode Exit fullscreen mode

If we attempt to retrieve items for an order with ID=100, we should successfully return 2 rows. Since there is an index on the id column in the orders table, it might seem that this query should be fast.

select oi.id, oi.order_id, oi.price, oi.amount, oi.sku, o.user_id, o.shop_id, o.status
from demo.orders o
join demo.order_item oi on oi.order_id = o.id
where o.id = 100 -- indexed on this column
order by oi.id;

Enter fullscreen mode Exit fullscreen mode

However, if we try to profile this query, we will see sequential scanning of the table in the execution plan. We should also be concerned about the large number of pages that need to be read (Buffers parameter).

explain (analyze, buffers)
select oi.id, oi.order_id, oi.price, oi.amount, oi.sku, o.user_id, o.shop_id, o.status
from demo.orders o
join demo.order_item oi on oi.order_id = o.id
where o.id = 100 -- indexed on this column
order by oi.id;

Enter fullscreen mode Exit fullscreen mode

Sort  (cost=3588.34..3588.35 rows=2 width=79) (actual time=25.244..25.248 rows=2 loops=1)
  Sort Key: oi.id
  Sort Method: quicksort  Memory: 25kB
  Buffers: shared hit=1711
  ->  Nested Loop  (cost=0.29..3588.33 rows=2 width=79) (actual time=0.083..25.212 rows=2 loops=1)
        Buffers: shared hit=1711
        ->  Index Scan using orders_pkey on orders o  (cost=0.29..8.31 rows=1 width=28) (actual time=0.040..0.052 rows=1 loops=1)
              Index Cond: (id = 100)
              Buffers: shared hit=6
        ->  Seq Scan on order_item oi  (cost=0.00..3580.00 rows=2 width=59) (actual time=0.036..25.145 rows=2 loops=1)
              Filter: (order_id = 100)
              Rows Removed by Filter: 149998
              Buffers: shared hit=1705
Planning:
  Buffers: shared hit=54
Planning Time: 0.793 ms
Execution Time: 25.314 ms

Enter fullscreen mode Exit fullscreen mode

If we create an index for a column with a foreign key, the situation will return to normal:

create index concurrently if not exists idx_order_item_order_id
    on demo.order_item (order_id);

Enter fullscreen mode Exit fullscreen mode

The sequential scan will disappear from the query plan, and the number of pages read will be significantly reduced:

Sort  (cost=19.54..19.55 rows=2 width=79) (actual time=0.110..0.113 rows=2 loops=1)
  Sort Key: oi.id
  Sort Method: quicksort  Memory: 25kB
  Buffers: shared hit=8
  ->  Nested Loop  (cost=0.71..19.53 rows=2 width=79) (actual time=0.076..0.086 rows=2 loops=1)
        Buffers: shared hit=8
        ->  Index Scan using orders_pkey on orders o  (cost=0.29..8.31 rows=1 width=28) (actual time=0.029..0.030 rows=1 loops=1)
              Index Cond: (id = 100)
              Buffers: shared hit=3
        ->  Index Scan using idx_order_item_order_id on order_item oi  (cost=0.42..11.20 rows=2 width=59) (actual time=0.033..0.040 rows=2 loops=1)
              Index Cond: (order_id = 100)
              Buffers: shared hit=5
Planning:
  Buffers: shared hit=15 read=1
Planning Time: 1.065 ms
Execution Time: 0.178 ms

Enter fullscreen mode Exit fullscreen mode

The FOREIGN_KEYS_WITHOUT_INDEX diagnostic will allow you to catch such cases early during development, preventing performance issues.

Should I create indexes or not?

It’s important to remember the issue of false positives: not all foreign key columns need to be indexed. Try to estimate the approximate table size in production; check your code for filtering, searching, or joining on the foreign key column. If you are 100% sure that you don’t need the index, you can simply add it to the exclusions. If you’re unsure, it’s better to create the index (it can always be removed later).

I’ve often encountered incidents where the database “slowed down” due to the absence of an index on a foreign key, but I haven’t seen any incidents where the database “slowed down” because of the presence of such indexes. Therefore, I disagree with the point made in the Percona blog article that foreign key indexes should not be created at all from the start. This is a DBA approach. Do you have a dedicated DBA on your team?

Null value in indexes

By default, PostgreSQL includes null values in btree indexes, but they are usually not needed there. All null values are unique, and you cannot simply retrieve a record where the column value is null. Most of the time, it’s better to exclude nulls from the index by creating partial indexes on nullable columns like where <A> is not null. The diagnostic INDEXES_WITH_NULL_VALUES helps to detect such cases.

Let’s consider an example with orders and order_items. The order_item table has a nullable column warehouse_id, which represents the warehouse ID.

create table if not exists demo.order_item
(
    id bigint primary key generated always as identity,
    order_id bigint not null references demo.orders (id),
    price decimal(22, 2) not null default 0,
    amount int not null default 0,
    sku varchar(255) not null,
    warehouse_id int
);

Enter fullscreen mode Exit fullscreen mode

Suppose we have several warehouses. After the order is paid, we begin to assemble it. We will update the status of some orders and mark them as paid.

-- index for efficient searching
create index concurrently if not exists idx_orders_creates_at_status
    on demo.orders (created_at, status);

-- update the status for several orders
update demo.orders
    set status = 2 -- paid order
where
    status = 1 -- new order
    and id in (
    select id from demo.orders where id % 4 = 0 order by id limit 100);

Enter fullscreen mode Exit fullscreen mode

Individual items in an order may be shipped from different warehouses according to an internal algorithm, considering logistics, stock, warehouse load, etc. After assigning the warehouse and updating the stock, we update the warehouse_id field for each item in the order (which was initially null).

update demo.order_item
    set warehouse_id = case when order_id % 8 = 0 then 1 else 2 end
where
    warehouse_id is null
    and order_id in (
    select id from demo.orders
    where
        status = 2
        and created_at >= current_timestamp - interval '1 day');

Enter fullscreen mode Exit fullscreen mode

We will need to search by a specific warehouse ID to know which items need to be completed and shipped. We take only paid orders for a certain time range.

select oi.id, oi.order_id, oi.price, oi.amount, oi.sku, o.user_id, o.shop_id, o.status
from demo.orders o
join demo.order_item oi on oi.order_id = o.id
where
    oi.warehouse_id = 2
    and o.status = 2
    and o.created_at >= current_timestamp - interval '1 day'
order by oi.id;

Enter fullscreen mode Exit fullscreen mode

The first solution would likely be a regular index on the warehouse_id column:

create index concurrently if not exists idx_order_item_warehouse_id
    on demo.order_item (warehouse_id);

Enter fullscreen mode Exit fullscreen mode

If we create such an index, it will be used without issues when searching for items for a specific warehouse. It might seem that this index should allow efficiently finding all items where the warehouse is not yet assigned, filtering records with the condition warehouse_id is null.

explain (analyze, buffers)
select * from demo.order_item where warehouse_id is null;

Enter fullscreen mode Exit fullscreen mode

However, if we look at the query execution plan, we will see sequential access there — the index is not used.

Seq Scan on order_item  (cost=0.00..3208.00 rows=149755 width=63) (actual time=0.214..24.909 rows=149800 loops=1)
  Filter: (warehouse_id IS NULL)
  Rows Removed by Filter: 200
  Buffers: shared hit=1708
Planning Time: 0.291 ms
Execution Time: 33.720 ms

Enter fullscreen mode Exit fullscreen mode

Of course, this is related to the specific distribution of data in the test database. The warehouse_id column has low cardinality, meaning the number of unique values in it is low. An index on this column has low selectivity. Index selectivity refers to the ratio of the number of distinct indexed values (i.e., cardinality) to the total number of rows in the table distinct / count(). For example, a unique index has a selectivity of one.

We can increase the selectivity of the index by removing null values and creating a partial index on the warehouse_id column.

create index concurrently if not exists idx_order_item_warehouse_id_without_nulls
    on demo.order_item (warehouse_id) where warehouse_id is not null;

Enter fullscreen mode Exit fullscreen mode

We will immediately see this index in the query plan:

Sort  (cost=860.61..860.61 rows=1 width=79) (actual time=0.628..0.640 rows=100 loops=1)
  Sort Key: oi.id
  Sort Method: quicksort  Memory: 35kB
  Buffers: shared hit=417
  ->  Nested Loop  (cost=0.44..860.60 rows=1 width=79) (actual time=0.052..0.546 rows=100 loops=1)
        Buffers: shared hit=417
        ->  Index Scan using idx_order_item_warehouse_id_without_nulls on order_item oi  (cost=0.14..246.83 rows=80 width=59) (actual time=0.023..0.069 rows=100 loops=1)
              Index Cond: (warehouse_id = 2)
              Buffers: shared hit=5
        ->  Index Scan using orders_pkey on orders o  (cost=0.29..7.67 rows=1 width=28) (actual time=0.003..0.003 rows=1 loops=100)
              Index Cond: (id = oi.order_id)
              Filter: ((status = 2) AND (created_at >= (CURRENT_TIMESTAMP - '1 day'::interval)))
              Buffers: shared hit=356
Planning:
  Buffers: shared hit=17
Planning Time: 0.835 ms
Execution Time: 0.702 ms

Enter fullscreen mode Exit fullscreen mode

If we compare the sizes of the indexes, we will see a significant difference. The partial index is much smaller and will be updated less frequently. Using this index, we save disk space and improve performance.

Query to get the size of the indexes

select
    x.indrelid::regclass::text as table_name,
    x.indexrelid::regclass::text as index_name,
    pg_relation_size(x.indexrelid) as index_size_bytes
from pg_index x
join pg_stat_all_indexes psai on x.indexrelid = psai.indexrelid
where psai.schemaname = 'demo'::text;

Enter fullscreen mode Exit fullscreen mode

table_name index_name index_size_bytes
demo.order_item demo.idx_order_item_warehouse_id 1056768
demo.order_item demo.idx_order_item_warehouse_id_without_nulls 16384

Plans for the Future

These are far from all the issues that pg-index-health can detect. The full list of diagnostics is available in the README of the project on GitHub, and it is regularly expanded.

Integrating pg-index-health into a Spring Boot application is quite simple. The overhead for running the checks is minimal. As a result, you will gain protection from common errors and issues. I encourage you to try implementing it!

In the near future, I plan to add full support for partitioned tables in all checks. Currently, this is implemented only for 11 out of 25 checks. I also want to expand the number of checks: there are already tickets for implementing at least 5 new checks. Additionally, in 2025, I plan to switch to Java 17 and Spring Boot 3.

Repository Links

Additional Material

原文链接:pg-index-health – a static analysis tool for you PostgreSQL database

© 版权声明
THE END
喜欢就支持一下吧
点赞10 分享
评论 抢沙发

请登录后发表评论

    暂无评论内容