This library lets you access the PostgreSQL 🐘 database management system from Emacs, using its network-level frontend/backend “wire” protocol. The module is capable of automatic type coercions from a range of SQL types to the equivalent Emacs Lisp type.
📖 You may be interested in the user manual.
This is a developer-oriented library, which won’t be useful to end users. If you’re looking for a browsing/editing interface to PostgreSQL in Emacs, you may be interested in PGmacs.
This library has support for:
-
SCRAM-SHA-256 authentication (the default authentication method since PostgreSQL version 14), as well as MD5 and password authentication. There is currently no support for authentication using client-side certificates.
-
Encrypted (TLS) connections with the PostgreSQL database, if your Emacs has been built with GnuTLS support. This includes support for authentication using client certificates.
-
Prepared statements using PostgreSQL’s extended query protocol, to avoid SQL injection attacks.
-
The PostgreSQL COPY protocol to copy preformatted data from an Emacs buffer to PostgreSQL, or to dump a PostgreSQL table or query result to an Emacs buffer in CSV or TSV format.
-
Asynchronous handling of LISTEN/NOTIFY notification messages from PostgreSQL, allowing the implementation of publish-subscribe architectures (PostgreSQL as an “event broker” or “message bus” and Emacs as event publisher and consumer).
-
Parsing various PostgreSQL types including integers, floats, array types, numerical ranges, JSON and JSONB objects into their native Emacs Lisp equivalents. The parsing support is user-extensible. Support for the HSTORE, pgvector and PostGIS extensions.
-
Connections over TCP or (on Unix machines) a local Unix socket.
Tested PostgreSQL versions: The code has been tested with versions 17.2, 16.4, 15.4, 13.8, 11.17, and 10.22 on Linux. It is also tested via GitHub actions on MacOS and Windows. This library also works, more or less, against other “PostgreSQL-compatible” databases. There are four main points where this compatibility may be problematic:
-
Compatibility with the PostgreSQL wire protocol. This is the most basic form of compatibility.
-
Compatibility with the PostgreSQL flavour of SQL, such as row expressions, non-standard functions such as
CHR
, data types such asBIT
,VARBIT
,JSON
andJSONB
, user-defined ENUMS and so on, functionality such asLISTEN
. Some databases that claim to be “Postgres compatible” don’t even support foreign keys, views, triggers, sequences, tablespaces and temporary tables (looking at you, Amazon Aurora DSQL). -
Implementation of the system tables that are used by certain pg-el functions, to retrieve the list of tables in a database, the list of types, and so on.
-
Establishing encrypted TLS connections to hosted services. Most PostgreSQL client libraries (in particular the official client library libpq) use OpenSSL for TLS support, whereas Emacs uses GnuTLS, and you may encounter incompatibilities.
The following PostgreSQL-compatible databases have been tested:
-
Neon “serverless PostgreSQL” works perfectly.
-
ParadeDB version 0.13.1 works perfectly (it's really a PostgreSQL extension rather than a distinct database implementation).
-
IvorySQL version 3.4 works perfectly (this fork of PostgreSQL adds some features for compatibility with Oracle).
-
The Timescale DB extension for time series data works perfectly (last tested with version 2.16.1).
-
The CitusDB extension for sharding PostgreSQL over multiple hosts works perfectly (last tested with Citus version 12.1.5, which is based on PostgreSQL 16.6).
-
The PgBouncer connection pooler for PostgreSQL works fine (tested with version 1.23 in the default session pooling mode).
-
Xata “serverless PostgreSQL” has many limitations including lack of support for
CREATE DATABASE
,CREATE COLLATION
, for XML processing, for temporary tables, for cursors, forEXPLAIN
, forCREATE EXTENSION
, for functions such aspg_notify
. -
YugabyteDB: last tested against version 2.23. Mostly working though the
pg_sequences
table is not implemented so certain tests fail. YugabyteDB does not have full compatibility with PostgreSQL SQL, and for exampleGENERATED ALWAYS AS
columns are not supported, andLISTEN
andNOTIFY
are not supported. It does support certain extensions such as pgvector, however. -
CrateDB: last tested with version 5.9.5. CrateDB does not support rows (e.g.
SELECT (1,2)
), does not support thetime
,varbit
,bytea
,jsonb
andhstore
types, does not handle a query which only contains an SQL comment, does not handle various PostgreSQL functions such asfactorial
, and does not return a correct type OID for text columns in rows returned from a prepared statement, doesn't support Unicode identifiers. -
CockroachDB: last tested with CockroachDB CCL v24.3. Note that this database does not implement the large object functionality, and its interpretation of SQL occasionally differs from that of PostgreSQL. Currently fails with an internal error on the SQL generated by our query for
pg-table-owner
, and fails on the boolean vector syntax b'1001000'. -
QuestDB: last tested against version 8.2.1. This has very limited PostgreSQL support, and does not support the
integer
type for example. -
Google Spanner: tested with the Spanner emulator (that reports itself as
PostgreSQL 14.1
) and the PGAdapter library that enables support for the PostgreSQL wire protocol. Spanner has very limited PostgreSQL compatibility, for example refusing to create tables that do not have a primary key. It does not recognize basic PostgreSQL types such as INT2. It also does not for example support theCHR
andMD5
functions, row expressions, and WHERE clauses without a FROM clause. -
YDB by Yandex last tested with version 23-4. Has very limited PostgreSQL compatibility. For example, an empty query string leads to a hung connection, and the
bit
type is returned as a string with the wrong oid. -
GreptimeDB: this time series database implements quite a lot of the PostgreSQL wire protocol, but the names it uses for types in the
pg_catalog.pg_types
table are not the same as those used by PostgreSQL (e.g.Int64
instead ofint8
), so our parsing machinery does not work. Last tested v0.12.0 in 2024-12. -
Hosted PostgreSQL services that have been tested: as of 2024-12 Railway.app is running a Debian build of PostgreSQL 16.4, and works fine; Aiven.io is running a Red Hat build of PostgreSQL 16.4 on Linux/Aarch64 and works fine.
-
Untested but likely to work: Amazon RDS, Google Cloud SQL, Azure Database for PostgreSQL, Amazon Auroa, Google AlloyDB, Materialize. You may however encounter difficulties with TLS connections, as noted above.
It does not work with the ClickHouse database, whose PostgreSQL support is too limited (no
implementation of the pg_types
system table, no support for basic PostgreSQL-flavoured SQL
commands such as SET
, no support for the extended query mechanism).
Tested Emacs versions: Tested with versions 30-prerelease, 29.4, 28.2, 27.2 and 26.3. Emacs versions older than 26.1 will not work against a recent PostgreSQL version (whose default configuration requires SCRAM-SHA-256 authentication), because they don’t include the GnuTLS support which we use to calculate HMACs. They may however work against a database set up to allow unauthenticated local connections. Emacs versions older than 28.1 (from April 2022) will not be able to use the extended query protocol (prepared statements), because they don’t have the necessary bindat functionality. It should however be easy to update the installed version of bindat.el for these older versions.
You may be interested in an alternative library emacs-libpq that enables access to PostgreSQL from Emacs by binding to the libpq library.
Install via the MELPA package archive by
including the following in your Emacs initialization file (.emacs.el
or init.el
):
(require 'package)
(add-to-list 'package-archives '("melpa" . "https://melpa.org/packages/") t)
then saying
M-x package-install RET pg
Alternatively, you can install the library from the latest GitHub revision using:
(unless (package-installed-p 'pg)
(package-vc-install "https://github.com/emarsden/pg-el" nil nil 'pg))
You can later update these to the latest version with M-x package-vc-upgrade RET pg RET
.
Thanks to Eric Ludlam for discovering a bug in the date parsing routines, to Hartmut Pilch and Yoshio Katayama for adding multibyte support, and to Doug McNaught and Pavel Janik for bug fixes.