Logo

pgx

Build Postgres Extensions with Rust!

cargo test --all crates.io badge docs.rs badge Twitter Follow Discord Chat

pgx is a framework for developing PostgreSQL extensions in Rust and strives to be as idiomatic and safe as possible.

pgx supports Postgres v11-v15.

Feel free to join our Discord Server.

Key Features

System Requirements

How to: GCC 7 on CentOS 7

In order to use GCC 7, install scl and enter the GCC 7 development environment:

bash yum install centos-release-scl yum install devtoolset-7 scl enable devtoolset-7 bash

Getting Started

First install the cargo-pgx sub-command and initialize the development environment:

bash cargo install --locked cargo-pgx cargo pgx init

The init command downloads PostgreSQL versions v10 through v15 compiles them to ~/.pgx/, and runs initdb. It's also possible to use an existing (user-writable) PostgreSQL install, or install a subset of versions, see the README.md of cargo-pgx for details.

bash cargo pgx new my_extension cd my_extension

This will create a new directory for the extension crate.

``` $ tree . ├── Cargo.toml ├── my_extension.control ├── sql └── src └── lib.rs

2 directories, 3 files ```

The new extension includes an example, so you can go ahead and run it right away.

bash cargo pgx run

This compiles the extension to a shared library, copies it to the specified Postgres installation, starts that Postgres instance and connects you to a database named the same as the extension.

Once cargo-pgx drops us into psql we can load the extension and do a SELECT on the example function.

```sql myextension=# CREATE EXTENSION myextension; CREATE EXTENSION

myextension=# SELECT hellomy_extension();

hellomyextension

Hello, my_extension (1 row) ```

For more details on how to manage pgx extensions see Managing pgx extensions.

Upgrading

You can upgrade your current cargo-pgx installation by passing the --force flag to cargo install:

bash cargo install --force --locked cargo-pgx

As new Postgres versions are supported by pgx, you can re-run the pgx init process to download and compile them:

bash cargo pgx init

Mapping of Postgres types to Rust

Postgres Type | Rust Type (as Option<T>) --------------|----------- bytea | Vec<u8> or &[u8] (zero-copy) text | String or &str (zero-copy) varchar | String or &str (zero-copy) or char "char" | i8 smallint | i16 integer | i32 bigint | i64 oid | u32 real | f32 double precision | f64 bool | bool json | pgx::Json(serde_json::Value) jsonb | pgx::JsonB(serde_json::Value) date | pgx::Date time | pgx::Time timestamp | pgx::Timestamp time with time zone | pgx::TimeWithTimeZone timestamp with time zone | pgx::TimestampWithTimeZone anyarray | pgx::AnyArray anyelement | pgx::AnyElement box | pgx::pg_sys::BOX point | pgx::pgx_sys::Point tid | pgx::pg_sys::ItemPointerData cstring | &std::ffi::CStr inet | pgx::Inet(String) -- TODO: needs better support numeric | pgx::Numeric<P, S> or pgx::AnyNumeric void | () ARRAY[]::<type> | Vec<Option<T>> or pgx::Array<T> (zero-copy) NULL | Option::None internal | pgx::PgBox<T> where T is any Rust/Postgres struct uuid | pgx::Uuid([u8; 16])

There are also IntoDatum and FromDatum traits for implementing additional type conversions, along with #[derive(PostgresType)] and #[derive(PostgresEnum)] for automatic conversion of custom types.

Digging Deeper

Caveats & Known Issues

There's probably more than are listed here, but a primary things of note are:

TODO

There's a few things on our immediate TODO list

Feature Flags

PGX has optional feature flags for Rust code that do not involve configuring the version of Postgres used, but rather extend additional support for other kinds of Rust code. These are not included by default.

"time-crate": interop with the time crate

pgx once used direct interop with the excellent time crate. However, due to complications involving performance and accurate interop with Postgres, this feature is now considered deprecated in favor of a lower-overhead interop. You may still request implementations of TryFrom<time::Type> for pgx::MatchingType and From<time::Type> for pgx::MatchingType by enabling the "time-crate" feature.

Experimental Features

Adding pgx = { version = "0.5.0", features = ["postgrestd"] } to your Cargo.toml will enable a highly experimental variant of pgx designed for integration with postgrestd, a modified Rust standard library that executes the Rust runtime atop the Postgres runtime, instead of using the operating system's ordinary C runtime. This reduces the programmatic and performance impedance between Rust and Postgres. This feature is neither complete, nor is it completely enabled just by enabling the feature, as it requires additional code not in this crate in the form of the modified sysroot.

Because the postgrestd feature is designed around control over std, some of pgx's insulating guard code around the C FFI with Postgres is disabled. Combined with its "pre-alpha" stage, you should assume this feature can enable undefined behavior, even if you know what you are doing. Especially if you know exactly what you're doing, in fact, as that almost certainly means you are developing this feature, and further extending both runtimes in ways neither initially had imagined. If you absolutely must enable this feature, you may wish to discuss it first on [Discord].

Adding pgx = { version = "0.5.0", features = ["plrust"] } to your Cargo.toml will enable an even more experimental variant of the above with special carve-outs specifically for usage with PL/Rust. This feature may not last long, as it is likely that code may move into a separate crate.

As a reminder: "THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND..."

Contributing

We are most definitely open to contributions of any kind. Bug Reports, Feature Requests, Documentation, and even sponsorships.

If you'd like to contribute code via a Pull Request, please make it against our develop branch. The master branch is meant to represent what is currently available on crates.io.

Providing wrappers for Postgres' internals is not a straightforward task, and completely wrapping it is going to take quite a bit of time. pgx is generally ready for use now, and it will continue to be developed as time goes on. Your feedback about what you'd like to be able to do with pgx is greatly appreciated.

License

Portions Copyright 2019-2021 ZomboDB, LLC. Portions Copyright 2021-2022 Technology Concepts & Design, Inc. <support@tcdi.com>. All rights reserved. Use of this source code is governed by the MIT license that can be found in the LICENSE file.