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
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
for managing the pgx
development environment
cargo pgx new
cargo pgx init
psql
via cargo pgx run
cargo pgx test
cargo pgx package
pgXX.rs
modulesextension_sql!
& extension_sql_file!
macrosCREATE EXTENSION ...;
panic!
s into Postgres ERROR
s that abort the transaction, not the processpanic!
and elog(ERROR)
#[pg_guard]
procedural macro to ensure the aboveDatum
is simply Option<T> where T: FromDatum
-- NULL
Datums are safely represented as Option::None
#[pg_test]
proc-macro for unit testing in-process within Postgres#[pg_extern]
to expose them to Postgresimpl std::iter::Iterator<Item = T> where T: IntoDatum
for automatic set-returning-functions (both RETURNS SETOF
and RETURNS TABLE (...)
variantsPostgres 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.
#[derive(PostgresType)]
to use a Rust struct as a Postgres type, represented as a CBOR-encoded object in-memory/on-disk, and JSON as human-readable
#[derive(PostgresEnum)]
to use a Rust enum as a Postgres enumMemoryContext
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
(minimum version 1.58) and cargo
cargo install rustfmt
git
libclang.so
libclang-dev
or clang
clang
-dynamic-list
(Linux) or -exported_symbols_list
(Mac).scl enable devtoolset-7
Note that a local Postgres installation is not required. pgx
will download and compile Postgres itself.
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
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.
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.
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();
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
:
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
There's probably more than are listed here, but a primary things of note are:
Threading is not really supported. Postgres is strictly single-threaded. As such, if you do venture into using threads, those threads MUST NOT call any internal Postgres function, or otherwise use any Postgres-provided pointer. There's also a potential problem with Postgres' use of 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
interactions are unknown right now.
pgx
uses lots of unsafe
Rust. That's generally the nature of the beast when doing FFI wrappers, so be aware.
Not all of Postgres' internals are included or even wrapped. This isn't due to it not being possible, it's simply due to it being an incredibly large task. If you identify internal Postgres APIs you need, open an issue and we'll get them exposed, at least through the pgx::pg_sys
module.
Windows is not supported. It could be, but will require a bit of work with cargo-pgx
and figuring out how to compile pgx
's "cshim" static library.
Sessions started before 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.
There's a few things on our immediate TODO list
pgx
does support creating trigger functions in Rust (need examples!)
but it doesn't automatically generate any of the DDL for them. This too likely needs a procmaro like #[pg_trigger]
cargo-pgx
subcommand and make use of https://github.com/zombodb/postgres-parser.#[derive(PostgresType/Enum)]
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.