diff --git a/cibuild.sh b/cibuild.sh
index fada17f74..9df39a503 100755
--- a/cibuild.sh
+++ b/cibuild.sh
@@ -388,12 +388,12 @@ do
mkdir -p /tmp/web/dist
mkdir -p /tmp/web/examples
- mkdir -p /tmp/web/benchmarks
+ mkdir -p /tmp/web/benchmark
PGLITE=$(pwd)/packages/pglite
cp -r ${PGLITE}/dist/* /tmp/web/dist/
cp -r ${PGLITE}/examples/* /tmp/web/examples/
- cp -r ${WORKSPACE}/packages/benchmark/dist/* /tmp/web/benchmarks/
+ cp -r ${WORKSPACE}/packages/benchmark/dist/* /tmp/web/benchmark/
;;
esac
shift
diff --git a/docs/benchmarks.md b/docs/benchmarks.md
index 491469e4c..78b226554 100644
--- a/docs/benchmarks.md
+++ b/docs/benchmarks.md
@@ -18,7 +18,23 @@
# Benchmarks
-There are two sets of benchmarks, one testing [round trip time](#round-trip-time-benchmarks) for both PGlite and wa-sqlite, and [another](#sqlite-benchmark-suite) based on the [wa-sqlite bechmarks](https://rhashimoto.github.io/wa-sqlite/demo/benchmarks.html).
+There are two sets of micro-benchmarks, one testing [round trip time](#round-trip-time-benchmarks) for both PGlite and wa-sqlite, and [another](#sqlite-benchmark-suite) based on the [SQLite speed test](https://sqlite.org/src/file?name=tool/speedtest.tcl&ci=trunk) which was ported for the [wa-sqlite benchmarks](https://rhashimoto.github.io/wa-sqlite/demo/benchmarks.html).
+
+We also have a set of [native baseline](#native-baseline) results where we have compared native SQLite (via the Node better-sqlite3 package) to full Postgres.
+
+Comparing Postgres to SQlite is a little difficult as they are quite different databases, particularly when you then throw in the complexities of WASM. Therefore these benchmarks provide a view of performance only as a starting point to investigate the difference between the two and the improvements we can make going forward.
+
+The other thing to consider when analysing the speed is the performance of various different VFS implementations providing persistance to both PGlite and wa-sqlite, the the performance of the underlying storage.
+
+The key finding are:
+
+1. wa-sqlite is a little faster than PGlite when run purely in memory. This is be expected as it is a simpler database with fewer features, its designed to go fast. Having said that, PGlite is not slow, its well withing the range you would expect when [comparing native SQLite to Postgres](#native-baseline).
+
+2. For single row CRUD inserts and updates, PGlite is faster then wa-sqlite. This is likely due to PGlite using the Posrgres WAL, whereas wa-sqlite is only using the SQLite rollback journal mode and not its WAL.
+
+3. An fsync or flush to the underlying storage can be quite slow, particularly in the browser with IndexedDB for PGlite, or OPFS for wa-sqlite. Both offer some level of "relaxed durability" that can be used to accelerate these queriers, and is likely suitable for many embedded use cases.
+
+We are going to continue to use these micro-benchmarks to feed back into the development of PGlite, and update them and the findings as we move forward.
These results below were run on a M2 Macbook Air.
@@ -26,6 +42,8 @@ These results below were run on a M2 Macbook Air.
These tests run a series of inserts/updates/deletes to find the average time to execute the type of CRUD operations that are regularly used in an app.
+Values are average ms - lower is better.
+
![](./public/img/benckmark/rtt.svg)
| Test | PGlite Memory | PGlite IDB | PGlite IDB
_relaxed durability_ | PGlite OPFS AHP | PGlite OPFS AHP
_relaxed durability_ | SQLite Memory | SQLite IDB | SQLite IDB
_relaxed durability_ | SQLite IDB BatchAtomic | SQLite IDB BatchAtomic
_relaxed durability_ | SQLite OPFS | SQLite OPFS AHP |
@@ -47,6 +65,8 @@ These tests run a series of inserts/updates/deletes to find the average time to
The SQLite benchmark suite, converted to web for wa-sqlite, performs a number of large queries to test the performance of the sql engin.
+Values are seconds to complete the test - lower is better.
+
![](./public/img/benckmark/sqlite-suite.svg)
| Test | PGlite
Memory | PGlite
IDB FS | PGlite
IDB FS
_relaxed durability_ | PGlite
OPFS Access Handle Pool | PGlite
OPFS Access Handle Pool
_relaxed durability_ | wa-sqlite
Memory (sync) | wa-sqlite
Memory (async) | wa-sqlite
DB Minimal | wa-sqlite
IDB Minimal
_relaxed durability_ | wa-sqlite
IDB Batch Atomic | wa-sqlite
IDB Batch Atomic
_relaxed durability_ | wa-sqlite
OPFS | wa-sqlite
OPFS Access Handle Pool |
diff --git a/docs/docs/about.md b/docs/docs/about.md
index 4e72e5487..cec37e0f3 100644
--- a/docs/docs/about.md
+++ b/docs/docs/about.md
@@ -1 +1,37 @@
-# What is PGlite
\ No newline at end of file
+# What is PGlite
+
+PGlite is a WASM Postgres build packaged into a TypeScript/JavaScript client library that enables you to run Postgres in the browser, Node.js and Bun, with no need to install any other dependencies. It's under 3mb gzipped, and has support for many [Postgres extensions](../extensions/), including [pgvector](../extensions/#pgvector).
+
+Unlike previous "Postgres in the browser" projects, PGlite does not use a Linux virtual machine - it is simply Postgres in WASM.
+
+It's being developed by [ElectricSQL](https://electric-sql.com/) for our use case of embedding into applications, either locally or at the edge, allowing users to sync a subset of their Postgres database.
+
+However, there are many more use cases for PGlite beyond it's use as an embedded application databases:
+
+- Unit and CI testing
+ PGlite is very fast to start and tare down, perfect for unit tests, you can a unique fresh Postgres for each test.
+
+- Local development
+ You can use PGlite as an alternative to a full local Postgres for local development, masivly simplifyinf your development environmant.
+
+- Remote development, or local web containers
+ As PGlite is so light weight it can be easily embedded into remote containerised development environments, or in-browser [web containers](https://webcontainers.io).
+
+- On-device or edge AI and RAG
+ PGlite has full support for [pgvector](../extensions/#pgvector), enabling a local or edge retrieval augmented generation (RAG) workflow.
+
+We are very keen to establish PGlite as an open source, and open contribution, project, working to build a community around it to develop its capabilities for all use cases.
+
+Getting started with PGlite is super easy, just install and import the NPM package, then create a your embded database:
+
+```js
+import { PGlite } from "@electric-sql/pglite";
+
+const db = new PGlite();
+await db.query("select 'Hello world' as message;");
+// -> { rows: [ { message: "Hello world" } ] }
+```
+
+It can be used as an ephemeral in-memory database, or with persistence either to the file system (Node/Bun) or indexedDB (Browser).
+
+Read more in our [getting started guide](./index.md).
diff --git a/docs/docs/filesystems.md b/docs/docs/filesystems.md
index 3ec59f386..8a5d85082 100644
--- a/docs/docs/filesystems.md
+++ b/docs/docs/filesystems.md
@@ -24,6 +24,12 @@ To use the in-memory FS you can use one of these methods:
})
```
+### Platform Support
+
+| Node | Bun | Chrome | Safari | Firefox |
+|------|-----|--------|--------|---------|
+| ✓ | ✓ | ✓ | ✓ | ✓ |
+
## Node FS
The Node FS uses the Node.js file system API to implement a VFS for PGLite. It is bailable in both Node and Bun.
@@ -42,6 +48,12 @@ To use the Node FS you can use one of these methods:
})
```
+#### Platform Support
+
+| Node | Bun | Chrome | Safari | Firefox |
+|------|-----|--------|--------|---------|
+| ✓ | ✓ | | | |
+
## IndexedDB FS
The IndexedDB FS persistes the database to IndexedDB in the browser. It's a layer over the in-memory filesystem, loading all files for the database into memory on start, and flushing them to IndexedDB after each query.
@@ -62,6 +74,12 @@ To use the IndexedDB FS you can use one of these methods:
The IndexedDB filesystem works at the file level, storing hole files as blobs in IndexedDB. Flushing whole files can take a few milliseconds after each query, to aid in building resposive apps we provide a `relaxedDurability` mode that can be [configured when starting](./api.md#options) PGlite. Under this mode the results of a query are returned imediatly, and the flush to IndexedDB is scheduled to happen asynchronous afterwards. Typically this is immediately after the query returns with no delay.
+### Platform Support
+
+| Node | Bun | Chrome | Safari | Firefox |
+|------|-----|--------|--------|---------|
+| | | ✓ | ✓ | ✓ |
+
## OPFS AHP FS
The OPFS AHP filesystem is built on top of the [Origin Private Filesystem](https://developer.mozilla.org/en-US/docs/Web/API/File_System_API/Origin_private_file_system) in the browser and uses an "access handle pool". It is only available when PGlite is run in a Web Worker, this could be any worker you configure, however we provide a [Multi Tab Worker](./multi-tab-worker.md) to aid in using PGlite from multiple tabs in the browser.
@@ -80,6 +98,14 @@ To use the OPFS AHP FS you can use one of these methods:
})
```
+### Platform Support
+
+| Node | Bun | Chrome | Safari | Firefox |
+|------|-----|--------|--------|---------|
+| | | ✓ | | ✓ |
+
+Unfortunately Safari appears to have a limit of 252 open sync access handles, this prevents this VFS from working as a standard Postgres install has between 300-800 files.
+
### What is an "access handle pool"?
The Origin Private Filesystem API provides both asynchronous ans synchronous methods, bit the synchronous are limited to read, write and flush. You are unable to travers the filesystem or open files synchronously. PGlite is a fully synchronous WASM build of Postgres and unable to call async APIs while handling a query. While it is possible to build an async WASM Postgres using [Asyncify](https://emscripten.org/docs/porting/asyncify.html), it adds significant overhead in both file size and performance.
diff --git a/docs/docs/index.md b/docs/docs/index.md
index 1a2319dad..251919dbd 100644
--- a/docs/docs/index.md
+++ b/docs/docs/index.md
@@ -1,46 +1,46 @@
# Getting started with PGlite
-PGlite is a WASM Postgres build packaged into a TypeScript client library that enables you to run Postgres in the browser, Node.js and Bun, with no need to install any other dependencies. It is only 2.6mb gzipped.
+PGlite can be used in both Node/Bun or the browser, and cen be used with any JavaScript framework.
-```js
-import { PGlite } from "@electric-sql/pglite";
+## Install and start in Node/Bun
-const db = new PGlite();
-await db.query("select 'Hello world' as message;");
-// -> { rows: [ { message: "Hello world" } ] }
-```
+Install into your project:
-It can be used as an ephemeral in-memory database, or with persistence either to the file system (Node/Bun) or indexedDB (Browser).
+::: code-group
-Unlike previous "Postgres in the browser" projects, PGlite does not use a Linux virtual machine - it is simply Postgres in WASM.
+```bash [npm]
+npm install @electric-sql/pglite
+```
-## Node/Bun
+```bash [pnpm]
+pnpm install @electric-sql/pglite
+```
-Install into your project:
+```bash [yarn]
+yarn add @electric-sql/pglite
+```
-```bash
-npm install @electric-sql/pglite
+```bash [bun]
+bun install @electric-sql/pglite
```
+:::
+
To use the in-memory Postgres:
```js
-
import { PGlite } from "@electric-sql/pglite";
const db = new PGlite();
-await db.query("select 'Hello world' as message;");
-// -> { rows: [ { message: "Hello world" } ] }
```
-or to persist to the filesystem:
+or to persist to the native filesystem:
```js
-
const db = new PGlite("./path/to/pgdata");
```
-## Browser
+## Install and start in the browser
It can be installed and imported using your usual package manager:
@@ -57,12 +57,87 @@ Then for an in-memory Postgres:
```js
const db = new PGlite()
-await db.query("select 'Hello world' as message;")
-// -> { rows: [ { message: "Hello world" } ] }
```
or to persist the database to indexedDB:
```js
const db = new PGlite("idb://my-pgdata");
-```
\ No newline at end of file
+```
+
+## Making a query
+
+There are two method for querying the database, `.query` and `.exec`, the former support parameters, and the latter multiple statements.
+
+First, lets crate a table and insert some test data using the `.exec` method:
+
+```js
+await db.exec(`
+ CREATE TABLE IF NOT EXISTS todo (
+ id SERIAL PRIMARY KEY,
+ task TEXT,
+ done BOOLEAN DEFAULT false
+ );
+ INSERT INTO todo (task, done) VALUES ('Install PGlite from NPM', true);
+ INSERT INTO todo (task, done) VALUES ('Load PGlite', true);
+ INSERT INTO todo (task, done) VALUES ('Create a table', true);
+ INSERT INTO todo (task, done) VALUES ('Insert some data', true);
+ INSERT INTO todo (task) VALUES ('Update a task');
+`)
+```
+
+The `.exec` method is perfect for migrations, or batch inserts with raw SQL.
+
+Now, lets retrieve an item using `.query` method:
+
+```js
+const ret = await db.query(`
+ SELECT * from todo WHERE id = 1;
+`)
+console.log(ret.rows)
+
+// Output:
+[
+ {
+ id: 1,
+ task: "Install PGlite from NPM"
+ }
+]
+```
+
+## Using parametrised queries
+
+When working with user supplied values its always best to use parametrised queries, these are supported on the `.query` method.
+
+We can use this to update a task:
+
+```js
+const ret = await db.query(
+ "UPDATE todo SET task = $2, done = $3 WHERE id = $1",
+ [
+ 5,
+ "Update a task using parametrised queries",
+ true
+ ]
+)
+```
+
+## What next?
+
+- To learn more about [querying](./api.md#query) and [transactions](./api.md#transaction) you can read the main [PGlite API documentation](./api.md).
+
+- There is also a [live-query extension](./live-queries.md) that enables reactive queries to update a UI when the underlying database changes.
+
+- PGlite has a number of built in [virtual file systems](./filesystems.md) to provided persistance to the database.
+
+- There are [framework hooks](./framework-hooks.md) to make working with PGlite within React and Vue much easer with less boilerplate.
+
+- As PGlite only has single exclusive connection to the database, we provide a [multi-tab worker](./multi-tab-worker.md) to enable sharing a PGlite instance between multiple browser tabs.
+
+- There is a [REPL component](./repl.md) that can be easily embedded into a web-app to aid in debugging and development, or as part of a database application itself.
+
+- We maintain a [list of ORMs and query builders](./orm-support.md) that support PGlite.
+
+- PGlite supports both Postgres extensions and PGlite Plugins via its [extensions API](./api.md#optionsextensions), and there is a list of [supported extensions](../extensions/).
+
+- We have a [page of examples](../examples.md) that you can open to test out PGlite in the browser.
\ No newline at end of file
diff --git a/docs/extensions/development.md b/docs/extensions/development.md
index ec027b93e..eb483ebd3 100644
--- a/docs/extensions/development.md
+++ b/docs/extensions/development.md
@@ -1 +1,62 @@
-# Extension Development
\ No newline at end of file
+# Extension Development
+
+PGlite has support for both Postgres extensions and it's own plugin api that allows a developer to augment a PGlite instance with an additional API.
+
+## Extension API
+
+::: warning
+The extension API is not yet stable and may change in a future release.
+:::
+
+PGlite extensions are an object with the following interface:
+
+```ts
+export interface Extension {
+ name: string;
+ setup: ExtensionSetup;
+}
+
+export type ExtensionSetup = (
+ pg: PGliteInterface,
+ emscriptenOpts: any,
+ clientOnly?: boolean,
+) => Promise
+ the [PGlite](../docs/api.md) instance that the enstension is being added to
+- `emscriptenOpts`
+ the options currently configured to pass to the [Emscrption Module factory](https://emscripten.org/docs/api_reference/module.html), including any [Emscript FS](https://emscripten.org/docs/api_reference/Filesystem-API.html).
+- `clientOnly`
+ A boolean indicating if this instace of the extension is "client only", meaning that it is on the main thread and doesnt have direct access to the underlying WASM. When true `emscriptenOpts` and `bundlePath` should not re returned as they will have no affect.
+
+The returned object has these properties - all are optional:
+
+- `emscriptenOpts`
+ Any augmented or altered configuration to pass to the [Emscrption Module factory](https://emscripten.org/docs/api_reference/module.html).
+- `namespaceObj`
+ An object to add as a namespace to the PGlite instance, this can provide access to additional methods or properties that your extension would like to expose.
+- `bundlePath`
+ The path to the Postgres extension tarball - see [Building Postgres Extensions](#building-postgres-extensions)
+- `init`
+ An initiation function that will be run after the PGlite instance and Postgres runtime has started, but before the instance is marked as ready for external usage. You can use this to perform any initiation your extension needs to perform on the database at startup.
+- `close`
+ A function that will be called when the user calls `close()` on their PGlite instance, this is called before the database has been shut down.
+
+An example of a PGlite extension that auments the PGlite instance is the [live query extension](../docs/live-queries.md).
+
+## Building Postgres Extensions
+
+We are still working on documentation, and examples, showing how to build Postgres extensions for use with PGlite. Please check back soon, or reach out on [Discord](https://discord.com/channels/933657521581858818/1212676471588520006) if you would like to try building a particular extension for PGlite.
diff --git a/packages/pglite/examples/dump-data-dir.html b/packages/pglite/examples/dump-data-dir.html
new file mode 100644
index 000000000..e69f600d1
--- /dev/null
+++ b/packages/pglite/examples/dump-data-dir.html
@@ -0,0 +1,22 @@
+
+
+
+ PGlite Dump Datadir Example
+PGlite OPFS Example
+Worker Thread -
+
+opfs-worker.js
Main Thread
+PGlite Worker Example
worker-process.js