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


Logo

🚨 NOTICE 🚨

This repo has relocated from https://github.com/zombodb/pgx to this location (https://github.com/tcdi/pgx). You may need to update your remote in .git/config to reflect this change.

pgx

Build Postgres Extensions with Rust!

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

pgx supports Postgres v10-v14.

Feel free to join our Discord Server.

Key Features

A Managed Development Environment

Target Multiple Postgres Versions

Automatic Schema Generation

Safety First

First-class UDF support

Most Postgres Data Types Transparently Converted 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(String) -- TODO: needs better support 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.

Easy Custom Types

Server Programming Interface (SPI)

Advanced Features

System Requirements

Note that a local Postgres installation is not required. pgx will download and compile Postgres itself.

Getting Started

1. Install cargo-pgx

First you'll want to install the pgx cargo sub-command from crates.io. You'll use it almost exclusively during your development and testing workflow.

shell script $ cargo install cargo-pgx

2. Initialize it

Next, pgx needs to be initialized. You only need to do this once.

shell script $ cargo pgx init

The init command downloads Postgres versions v10, v11, v12, v13, v14 compiles them to ~/.pgx/, and runs initdb. These installations are needed by pgx not only for auto-generating Rust bindings from each version's header files, but also for pgx's test framework.

See the documentation for cargo-pgx for details on how to limit the required postgres versions.

3. Create a new extension

shell script $ cargo pgx new my_extension $ cd my_extension

This will create a new directory for the extension crate.

my_extension/ ├── Cargo.toml ├── my_extension.control ├── sql │   ├── lib.generated.sql │   └── load-order.txt └── src └── lib.rs

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

4. Run your extension

shell script $ cargo pgx run pg13 # or pg10 or pg11 or pg12 or pg14

This compiles the extension to a shared library, copies it to the specified Postgres installation (in ~/.pgx/), starts that Postgres instance and connects you, via psql, to a database named for the extension.

The first time, compilation takes a few minutes as pgx needs to generate almost 200k lines of Rust "bindings" from Postgres' header files.

Once compiled you'll be placed in a psql shell, for, in this case, Postgres 13. Now, we can load the extension and do a SELECT on the example function.

```console myextension=# CREATE EXTENSION myextension; CREATE EXTENSION

myextension=# SELECT hellomy_extension();

hellomyextension

Hello, my_extension (1 row) ```

5. Detailed cargo pgx usage

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:

shell script $ cargo install --force cargo-pgx

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

shell script $ cargo pgx init

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

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.