Skip to content
Discord Get Started

Migrate from Neon

This guide walks through migrating a PostgreSQL database from Neon to DB9. The process uses standard PostgreSQL tooling (pg_dump for export) and the DB9 CLI for import.

  • SQL compatibility — DB9 supports the same DML, DDL, joins, CTEs, window functions, and subqueries you use in Neon. Most queries work without changes.
  • PostgreSQL drivers — Any driver that connects via pgwire (node-postgres, psycopg, pgx, JDBC) works with DB9.
  • ORM compatibility — Prisma, Drizzle, SQLAlchemy, TypeORM, Sequelize, Knex, and GORM are tested and supported.
  • Data types — Common types (TEXT, INTEGER, BIGINT, BOOLEAN, TIMESTAMPTZ, UUID, JSONB, arrays, vectors) work identically.
AreaNeonDB9
Connection stringpostgresql://user:pass@ep-*.neon.tech/dbnamepostgresql://tenant.role@pg.db9.io:5433/postgres
Connection poolingBuilt-in PgBouncer (transaction mode)No built-in pooler — use application-side pooling
BranchingCopy-on-write, instant for any sizeFull data copy, async (seconds to minutes)
ComputeAutoscaling, scale-to-zeroFixed per-database, always on
Serverless driver@neondatabase/serverless (HTTP/WebSocket)Standard pgwire only — no HTTP SQL endpoint
Extensions40+ community extensions9 built-in (http, vector, fs9, pg_cron, embedding, hstore, uuid-ossp, parquet, zhparser)
ReplicationLogical replication supportedNot supported
Row-level securitySupportedNot supported
Table partitioningSupportedNot supported
LISTEN/NOTIFYSupportedNot supported
Port54325433
Database nameCustom (e.g., neondb)Always postgres

Review the Compatibility Matrix for the full list of supported and unsupported features.

  • Access to your Neon database (direct/unpooled connection string)
  • pg_dump installed locally (comes with PostgreSQL client tools)
  • DB9 CLI installed: curl -fsSL https://get.db9.io | sh
  • A DB9 account: db9 create --name my-app to create your target database

Use pg_dump with Neon’s direct (unpooled) connection string. Do not use the pooled connection — pg_dump requires a direct connection.

Terminal
pg_dump --no-owner --no-privileges --no-comments \
"postgresql://user:pass@ep-cool-name-123456.us-east-2.aws.neon.tech/neondb?sslmode=require" \
> export.sql
Terminal
pg_dump --schema-only --no-owner --no-privileges \
"postgresql://user:pass@ep-cool-name-123456.us-east-2.aws.neon.tech/neondb?sslmode=require" \
> schema.sql

Flags explained:

  • --no-owner — omits ALTER ... OWNER TO statements that reference Neon-specific roles
  • --no-privileges — omits GRANT/REVOKE statements
  • --no-comments — omits COMMENT ON statements that may reference Neon internals

Use plain SQL format (default). DB9 does not support pg_restore with the custom (-Fc) or directory (-Fd) formats — import via SQL text only.

The pg_dump output may contain statements that DB9 does not support. Remove or comment out:

  • CREATE EXTENSION for extensions DB9 does not have — DB9 supports 9 built-in extensions. Remove any CREATE EXTENSION for extensions not in: http, uuid-ossp, hstore, fs9, pg_cron, parquet, zhparser, vector, embedding.
  • CREATE PUBLICATION / CREATE SUBSCRIPTION — DB9 does not support logical replication.
  • Row-level security policiesCREATE POLICY, ALTER TABLE ... ENABLE ROW LEVEL SECURITY.
  • Table partitioningPARTITION BY, CREATE TABLE ... PARTITION OF.
  • Advisory lock callspg_advisory_lock(), pg_try_advisory_lock().
  • Custom types with WHILE loops in PL/pgSQL — DB9 supports basic PL/pgSQL but not WHILE, EXECUTE, or exception handling.
  • Locale settings — DB9 accepts and ignores locale parameters from pg_dump, so these are safe to leave in.

A quick way to identify issues:

Terminal
# Check for unsupported extensions
grep "CREATE EXTENSION" export.sql
# Check for partitioning
grep -i "PARTITION" export.sql
# Check for RLS
grep -i "ROW LEVEL SECURITY\|CREATE POLICY" export.sql
# Check for replication
grep -i "PUBLICATION\|SUBSCRIPTION" export.sql
Terminal
# Create a new database
db9 create --name my-app --show-connection-string

This returns immediately with the connection string and credentials. Save them for your application config.

Section titled “Option A: CLI import (recommended for most databases)”
Terminal
db9 db sql my-app -f export.sql

This executes the SQL file against your DB9 database via the API. Suitable for databases up to the dump limits (50,000 rows or 16 MB per table).

Option B: Direct psql import (for larger databases)

Section titled “Option B: Direct psql import (for larger databases)”

For larger exports, use psql with DB9’s connection string directly:

Terminal
psql "$(db9 db status my-app --json | jq -r .connection_string)" -f export.sql

This streams the SQL through the pgwire protocol and handles larger files without the API dump limits.

If your export is large and you split schema from data, you can use COPY for bulk loading:

Terminal
# Import schema first
psql "$(db9 db status my-app --json | jq -r .connection_string)" -f schema.sql
# Then import data via COPY (pg_dump with --data-only --inserts=off uses COPY by default)
pg_dump --data-only --no-owner \
"postgresql://user:pass@ep-cool-name-123456.us-east-2.aws.neon.tech/neondb?sslmode=require" \
| psql "$(db9 db status my-app --json | jq -r .connection_string)"

DB9 supports COPY in CSV and TEXT formats over pgwire.

Replace the Neon connection string with DB9’s:

Diff
DATABASE_URL=postgresql://user:pass@ep-cool-name-123456.us-east-2.aws.neon.tech/neondb?sslmode=require
DATABASE_URL=postgresql://a1b2c3d4e5f6.admin@pg.db9.io:5433/postgres?sslmode=require

Key differences:

  • Username: DB9 uses {tenant_id}.{role} format (e.g., a1b2c3d4e5f6.admin)
  • Port: 5433, not 5432
  • Database: Always postgres
  • Host: pg.db9.io (not region-specific endpoints)

If you use @neondatabase/serverless, replace it with a standard PostgreSQL driver:

Diff
import { neon } from '@neondatabase/serverless';
const sql = neon(process.env.DATABASE_URL);
const result = await sql`SELECT * FROM users`;
import pg from 'pg';
const pool = new pg.Pool({ connectionString: process.env.DATABASE_URL });
const result = await pool.query('SELECT * FROM users');

DB9 uses standard pgwire (TCP), so pg (node-postgres), psycopg, pgx, and other standard drivers work without modification.

Neon provides built-in PgBouncer. DB9 does not include a connection pooler. If your application opens many connections, configure pooling at the application level:

TypeScript
// node-postgres pool
const pool = new pg.Pool({
connectionString: process.env.DATABASE_URL,
max: 10, // DB9 handles concurrent connections well
idleTimeoutMillis: 30000,
});

For ORMs, see the integration guides: Prisma, Drizzle, SQLAlchemy.

If you run code in edge/serverless environments (Cloudflare Workers, Vercel Edge Functions) that relied on Neon’s HTTP driver, you need to move database queries to a Node.js runtime. DB9 requires TCP connections via pgwire — it does not support HTTP or WebSocket SQL endpoints.

See the Next.js guide for patterns that work with both Server Components and API routes.

Terminal
db9 db dump my-app --ddl-only

Compare the output with your original schema to confirm all tables, indexes, and constraints were created.

Run a count on your key tables to verify data was imported:

Terminal
db9 db sql my-app -q "SELECT count(*) FROM users"
db9 db sql my-app -q "SELECT count(*) FROM orders"

Compare row counts against the source Neon database.

The most reliable validation is running your application’s existing test suite against the DB9 database. Update DATABASE_URL in your test environment and run:

Terminal
DATABASE_URL="$(db9 db status my-app --json | jq -r .connection_string)" npm test

If your tests fail, check these common differences:

  • SERIALIZABLE isolation — DB9 accepts SET TRANSACTION ISOLATION LEVEL SERIALIZABLE but runs as REPEATABLE READ
  • LISTEN/NOTIFY — not supported; use polling or an external message queue
  • Advisory locks — not supported; use SELECT ... FOR UPDATE for row-level locking
  • Row-level security — not supported; enforce access control in your application layer

If you need to revert:

  1. Your Neon database is unchanged — switch DATABASE_URL back to the Neon connection string.
  2. If you need to export data created in DB9 back to Neon:
Terminal
# Export from DB9
db9 db dump my-app -o db9-export.sql
# Import to Neon (use direct/unpooled connection)
psql "postgresql://user:pass@ep-cool-name-123456.us-east-2.aws.neon.tech/neondb?sslmode=require" \
-f db9-export.sql

The db9 db dump command outputs plain SQL (up to 50,000 rows or 16 MB per table). For larger databases, use psql to stream individual tables with COPY.

  • No zero-downtime migration — DB9 does not support logical replication, so you cannot stream changes from Neon in real time. Plan a maintenance window or accept a brief cutover period.
  • Extension gaps — If your Neon database uses extensions not in DB9’s built-in set (e.g., PostGIS, pg_trgm, pgcrypto), those features will not be available. Check your CREATE EXTENSION statements.
  • Dump size limits — The db9 db sql -f API import has limits (50,000 rows, 16 MB per table). For larger databases, use direct psql connection for import.
  • Branching model — Neon branches are copy-on-write and instant. DB9 branches are full copies and take longer for large databases. Adjust CI workflows that depend on instant branching.
  • Autoscaling — Neon can scale compute to zero when idle. DB9 databases are always on. This affects cost for rarely-used databases.