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.
cargo-pgx
cargo pgx new
: Create new extensions quicklycargo pgx init
: Install new (or register existing) PostgreSQL installscargo pgx run
: Run your extension and interactively test it in psql
(or pgcli
)cargo pgx test
: Unit-test your extension across multiple PostgreSQL versionscargo pgx package
: Create installation packages for your extensionREADME.md
!cargo pgx schema
)extension_sql!
& extension_sql_file!
panic!
s into Postgres ERROR
s that abort the transaction, not the processpanic!
and elog(ERROR)
#[pg_guard]
procedural macro to ensure the aboveDatum
s are Option<T> where T: FromDatum
NULL
Datums are safely represented as Option::<T>::None
#[pg_extern]
to expose them to Postgrespgx::iter::SetOfIterator<'a, T>
for RETURNS SETOF
pgx::iter::TableIterator<'a, T>
for RETURNS TABLE (...)
#[pg_trigger]
#[derive(PostgresType)]
to use a Rust struct as a Postgres type
#[derive(PostgresEnum)]
to use a Rust enum as a Postgres enumpgx::composite_type!("Sample")
macroMemoryContext
system via pgx::PgMemoryContexts
pgx::PgBox<T>
(akin to alloc::boxed::Box<T>
)#[pg_guard]
proc-macro for guarding extern "C"
Rust functions that need to be passed into Postgreseprintln!
-like macrosunsafe
access to large parts of Postgres internals via the pgx::pg_sys
modulerustc
, cargo
, and rustfmt
. The recommended way to get these is from https://rustup.rs †git
libclang
apt install libclang-dev
or apt install clang
yum install clang
tar
bzip2
PostgreSQL's build dependencies ‡
† PGX has no MSRV policy, thus may require the latest stable version of Rust, available via Rustup
‡ A local PostgreSQL server installation is not required. cargo pgx
can download and compile PostgreSQL versions on its own.
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
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 v14 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();
Hello, my_extension (1 row) ```
For more details on how to manage pgx extensions see Managing pgx extensions.
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
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.
There's probably more than are listed here, but a primary things of note are:
sigprocmask
. This was being discussed on the -hackers list, even with a patch provided, but the conversation seems to have stalled (https://www.postgresql.org/message-id/flat/5EF20168.2040508%40anastigmatix.net#4533edb74194d30adfa04a6a2ce635ba).async
context remains unexplored.pgx
wraps a lot of unsafe
code, some of which has poorly-defined safety conditions. It may be easy to induce illogical and undesirable behaviors even from safe code with pgx
, and some of these wrappers may be fundamentally unsound. Please report any issues that may arise.pgx::pg_sys
module.cargo-pgx
and figuring out how to compile pgx
's "cshim" static library.ALTER EXTENSION my_extension UPDATE;
will continue to see the old version of my_extension
. New sessions will see the updated version of the extension.pgx
is used by many "in production", but it is not "1.0.0" or above, despite that being recommended by SemVer for production-quality software. This is because there are many unresolved soundness and ergonomics questions that will likely require breaking changes to resolve, in some cases requiring cutting-edge Rust features to be able to expose sound interfaces. While a 1.0.0 release is intended at some point, it seems prudent to wait until it seems like a 2.0.0 release would not be needed the next week and the remaining questions can be deferred.There's a few things on our immediate TODO list
cargo-pgx
subcommand and make use of https://github.com/zombodb/postgres-parser.#[derive(PostgresType/Enum)]
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
cratepgx
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.
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..."
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.
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.