A PostgreSQL metric exporter for Prometheus written in Rust
pg_exporter is designed with a selective metrics approach:
- Modular collectors – Expose only the metrics you actually need instead of collecting everything by default.
- Avoid unnecessary metrics – Prevent exposing large numbers of unused metrics to Prometheus, reducing load and keeping monitoring efficient.
- Customizable collectors – Tailor the metrics to your specific requirements while maintaining compatibility with the official postgres_exporter.
- Low memory footprint – Designed to minimize memory usage and maximize efficiency while scraping metrics.
Install via Cargo:
cargo install pg_exporter
Or download the latest release from the releases page.
Container images are available at ghcr.io/nbari/pg_exporter:
# Using Docker
docker run -d \
-e PG_EXPORTER_DSN="postgresql://postgres_exporter@postgres-host:5432/postgres" \
-p 9432:9432 \
ghcr.io/nbari/pg_exporter:latest
# Using Podman
podman run -d \
-e PG_EXPORTER_DSN="postgresql://postgres_exporter@postgres-host:5432/postgres" \
-p 9432:9432 \
ghcr.io/nbari/pg_exporter:latestConnecting to host PostgreSQL from container:
- Docker Desktop (Mac/Windows): use
host.docker.internalinstead oflocalhost - Podman: use
host.containers.internalinstead oflocalhost - Linux with
--network=host: uselocalhostdirectly
Example with host connection:
podman run -d \
-e PG_EXPORTER_DSN="postgresql://[email protected]:5432/postgres" \
-p 9432:9432 \
ghcr.io/nbari/pg_exporter:latestRun the exporter and use the socket directory:
pg_exporter --dsn postgresql:///postgres?user=postgres_exporter
in pg_hba.conf you need to allow the user
postgres_exporterto connect, for example:
local all postgres_exporter trust
You can also specify a custom port, for example 9187:
pg_exporter --dsn postgresql://postgres_exporter@localhost:5432/postgres --port 9187
The following collectors are available:
--collector.defaultdefault--collector.activityactivity--collector.databasedatabase--collector.vacuumvacuum--collector.lockslocks--collector.statstat--collector.replicationreplication--collector.indexindex--collector.statementsstatements - Query performance metrics frompg_stat_statements(see detailed guide)--collector.tlstls - SSL/TLS certificate monitoring, connection encryption stats (PostgreSQL 9.5+)--collector.exporterexporter - Exporter self-monitoring (process metrics, scrape performance, cardinality tracking)
You can enable --collector.<name> or disable --no-collector.<name> For example,
to disable the vacuum collector:
pg_exporter --dsn postgresql:///postgres?user=postgres_exporter --no-collector.vacuum
This collectors are enabled by default:
defaultactivityvacuum
The project is structured as follows:
├── bin
├── cli
├── collectors
├── exporter
└── lib.rs
All the collectors are located in the collectors directory. Each collector is
in its own subdirectory, making it easy to manage and extend.
collectors
├── activity
│ ├── connections.rs
│ ├── mod.rs
│ └── wait.rs
├── config.rs
├── database
│ ├── catalog.rs
│ ├── mod.rs
│ ├── README.md
│ └── stats.rs
├── default
│ ├── mod.rs
│ ├── postmaster.rs
│ ├── settings.rs
│ └── version.rs
├── locks
│ ├── mod.rs
│ └── relations.rs
├── mod.rs <-- main file to register collectors
├── register_macro.rs
├── registry.rs
├── stat
│ ├── mod.rs
│ └── user_tables.rs
├── util.rs
└── vacuum
├── mod.rs
├── progress.rs
└── stats.rs
In mod.rs file inside the collectors directory, you can see how each
collector is registered. This modular approach allows for easy addition or
removal of collectors as needed.
Each collector can then be extended with more specific metrics. For example,
the vacuum collector has two files: progress.rs and stats.rs, this allows
for better organization and separation of concerns within the collector and
better testability. (or that is the plan).
The project includes unit tests for each collector and integration tests for the exporter as a whole. You can run the tests using:
just
need just installed, see just
To run with opentelemetry set the environment variable OTEL_EXPORTER_OTLP_ENDPOINT, for example:
OTEL_EXPORTER_OTLP_ENDPOINT="http://localhost:4317"
Then you can run the exporter and it will send traces to the specified endpoint.
To run postgres and jaeger locally
just postgres
just jaeger
just watch
For tracees add more verbosity with -v, for example:
cargo watch -x 'run -- --collector.vacuum -vv'
open jaeger at http://localhost:16686 and select the pg_exporter service to see the traces.
This project is a work in progress. Your feedback, suggestions, and contributions are always welcome!