The Neon serverless driver is a low-latency Postgres driver for JavaScript and TypeScript that allows you to query data from serverless and edge environments over HTTP or WebSockets in place of TCP. The driver's low-latency capability is due to message pipelining and other optimizations.
The Neon serverless driver is now generally available (GA)
The GA version of the Neon serverless driver, v1.0.0 and higher, requires Node.js version 19 or higher. It also includes a breaking change but only if you're calling the HTTP query template function as a conventional function. For details, please see the 1.0.0 release notes or read the blog post.
When to query over HTTP vs WebSockets:
- HTTP: Querying over an HTTP fetch request is faster for single, non-interactive transactions, also referred to as "one-shot queries". Issuing multiple queries via a single, non-interactive transaction is also supported. See Use the driver over HTTP.
- WebSockets: If you require session or interactive transaction support or compatibility with node-postgres (the popular npm pgpackage), use WebSockets. See Use the driver over WebSockets.
Install the Neon serverless driver
You can install the driver with your preferred JavaScript package manager. For example:
npm install @neondatabase/serverlessThe driver includes TypeScript types (the equivalent of @types/pg). No additional installation is required.
note
The Neon serverless driver is also available as a JavaScript Registry (JSR) package: https://jsr.io/@neon/serverless. The JavaScript Registry (JSR) is a package registry for JavaScript and TypeScript. JSR works with many runtimes (Node.js, Deno, browsers, and more) and is backward compatible with npm.
Configure your Neon database connection
You can obtain a connection string for your database by clicking the Connect button on your Project Dashboard. Your Neon connection string will look something like this:
DATABASE_URL=postgresql://[user]:[password]@[neon_hostname]/[dbname]The examples that follow assume that your database connection string is assigned to a DATABASE_URL variable in your application's environment file.
Use the driver over HTTP
The Neon serverless driver uses the neon function for queries over HTTP. The function returns a query function that can only be used as a template function for improved safety against SQL injection vulnerabilities.
For example:
import { neon } from '@neondatabase/serverless';
const sql = neon(process.env.DATABASE_URL);
const id = 1;
// Safe and convenient template function usage
const result = await sql`SELECT * FROM table WHERE id = ${id}`;
// For manually parameterized queries, use the query() function
const result = await sql.query('SELECT * FROM table WHERE id = $1', [id]);
// For interpolating trusted strings (like column or table names), use the unsafe() function
const table = condition ? 'table1' : 'table2'; // known-safe string values
const result = await sql`SELECT * FROM ${sql.unsafe(table)} WHERE id = ${id}`;
// Alternatively, use template literals for known-safe values
const table = condition ? sql`table1` : sql`table2`;
const result = await sql`SELECT * FROM ${table} WHERE id = ${id}`;SQL template queries are fully composable, including those with parameters:
const name = 'Olivia';
const limit = 1;
const whereClause = sql`WHERE name = ${name}`;
const limitClause = sql`LIMIT ${limit}`;
// Parameters are numbered appropriately at query time
const result = await sql`SELECT * FROM table ${whereClause} ${limitClause}`;You can use raw SQL queries or tools such as Drizzle-ORM, kysely, Zapatos, and others for type safety.
import { neon } from '@neondatabase/serverless';
const sql = neon(process.env.DATABASE_URL);
const posts = await sql`SELECT * FROM posts WHERE id = ${postId}`;
// or using query() for parameterized queries
const posts = await sql.query('SELECT * FROM posts WHERE id = $1', [postId]);
// `posts` is now [{ id: 12, title: 'My post', ... }] (or undefined)note
The maximum request size and response size for queries over HTTP is 64 MB.
neon function configuration options
The neon(...) function returns a query function that can be used as a template function, with additional properties for special cases:
import { neon } from '@neondatabase/serverless';
const sql = neon(process.env.DATABASE_URL);
// Use as a template function (recommended)
const rows = await sql`SELECT * FROM posts WHERE id = ${postId}`;
// Use query() for manually parameterized queries
const rows = await sql.query('SELECT * FROM posts WHERE id = $1', [postId]);
// Use unsafe() for trusted string interpolation
const table = 'posts'; // trusted value
const rows = await sql`SELECT * FROM ${sql.unsafe(table)} WHERE id = ${postId}`;By default, the query function returns only the rows resulting from the provided SQL query, and it returns them as an array of objects where the keys are column names. For example:
const rows = await sql`SELECT * FROM posts WHERE id = ${postId}`;
// -> [{ id: 12, title: "My post", ... }]You can customize the return format using the configuration options fullResults and arrayMode. These options are available both on the neon(...) function and on the query function it returns.
- 
arrayMode: boolean,falseby defaultThe default arrayModevalue isfalse. When it is true, rows are returned as an array of arrays instead of an array of objects:const sql = neon(process.env.DATABASE_URL, { arrayMode: true }); const rows = await sql`SELECT * FROM posts WHERE id = ${postId}`; // -> [[12, "My post", ...]]Or, with the same effect when using query(): const sql = neon(process.env.DATABASE_URL); const rows = await sql.query('SELECT * FROM posts WHERE id = $1', [postId], { arrayMode: true }); // -> [[12, "My post", ...]]
- 
fullResults: booleanThe default fullResultsvalue isfalse. When it istrue, additional metadata is returned alongside the result rows, which are then found in therowsproperty of the return value. The metadata matches what would be returned bynode-postgres:const sql = neon(process.env.DATABASE_URL, { fullResults: true }); const results = await sql`SELECT * FROM posts WHERE id = ${postId}`; /* -> { rows: [{ id: 12, title: "My post", ... }], fields: [ { name: "id", dataTypeID: 23, ... }, { name: "title", dataTypeID: 25, ... }, ... ], rowCount: 1, rowAsArray: false, command: "SELECT" } */Or, with the same effect when using query(): const sql = neon(process.env.DATABASE_URL); const results = await sql.query('SELECT * FROM posts WHERE id = $1', [postId], { fullResults: true, }); // -> { ... same as above ... }
- 
fetchOptions: Record<string, any>The fetchOptionsoption can also be passed to eitherneon(...)or thequeryfunction. This option takes an object that is merged with the options to thefetchcall.For example, to increase the priority of every database fetchrequest:import { neon } from '@neondatabase/serverless'; const sql = neon(process.env.DATABASE_URL, { fetchOptions: { priority: 'high' } }); const rows = await sql`SELECT * FROM posts WHERE id = ${postId}`;Or to implement a fetchtimeout:import { neon } from '@neondatabase/serverless'; const sql = neon(process.env.DATABASE_URL); const abortController = new AbortController(); const timeout = setTimeout(() => abortController.abort('timed out'), 10000); const rows = await sql('SELECT * FROM posts WHERE id = $1', [postId], { fetchOptions: { signal: abortController.signal }, }); // throws an error if no result received within 10s clearTimeout(timeout);
For additional details, see Options and configuration.
Issue multiple queries with the transaction() function
The transaction(queriesOrFn, options) function is exposed as a property on the query function. It allows multiple queries to be executed within a single, non-interactive transaction.
The first argument to transaction(), queriesOrFn, is either an array of queries or a non-async function that receives a query function as its argument and returns an array of queries.
The array-of-queries case looks like this:
import { neon } from '@neondatabase/serverless';
const sql = neon(process.env.DATABASE_URL);
const showLatestN = 10;
const [posts, tags] = await sql.transaction(
  [sql`SELECT * FROM posts ORDER BY posted_at DESC LIMIT ${showLatestN}`, sql`SELECT * FROM tags`],
  {
    isolationLevel: 'RepeatableRead',
    readOnly: true,
  }
);Or as an example of the function case:
const [authors, tags] = await neon(process.env.DATABASE_URL).transaction((txn) => [
  txn`SELECT * FROM authors`,
  txn`SELECT * FROM tags`,
]);The optional second argument to transaction(), options, has the same keys as the options to the ordinary query function — arrayMode, fullResults and fetchOptions — plus three additional keys that concern the transaction configuration. These transaction-related keys are: isolationMode, readOnly and deferrable.
Note that options cannot be supplied for individual queries within a transaction. Query and transaction options must instead be passed as the second argument of the transaction() function. For example, this arrayMode setting is ineffective (and TypeScript won't compile it): await sql.transaction([sql('SELECT now()', [], { arrayMode: true })]). Instead, use await sql.transaction([sql('SELECT now()')], { arrayMode: true }).
- 
isolationModeThis option selects a Postgres transaction isolation mode. If present, it must be one of ReadUncommitted,ReadCommitted,RepeatableRead, orSerializable.
- 
readOnlyIf true, this option ensures that aREAD ONLYtransaction is used to execute the queries passed. This is a boolean option. The default value isfalse.
- 
deferrableIf true(and ifreadOnlyis alsotrue, andisolationModeisSerializable), this option ensures that aDEFERRABLEtransaction is used to execute the queries passed. This is a boolean option. The default value isfalse.
For additional details, see transaction(...) function.
Use the driver over WebSockets
The Neon serverless driver supports the Pool and Client constructors for querying over WebSockets.
The Pool and Client constructors, provide session and transaction support, as well as node-postgres compatibility. You can find the API guide for the Pool and Client constructors in the node-postgres documentation.
Consider using the driver with Pool or Client in the following scenarios:
- You already use node-postgresin your code base and would like to migrate to using@neondatabase/serverless.
- You are writing a new code base and want to use a package that expects a node-postgres-compatibledriver.
- Your backend service uses sessions / interactive transactions with multiple queries per connection.
You can use the Neon serverless driver in the same way you would use node-postgres with Pool and Client. Where you usually import pg, import @neondatabase/serverless instead.
import { Pool } from '@neondatabase/serverless';
const pool = new Pool({ connectionString: process.env.DATABASE_URL });
const posts = await pool.query('SELECT * FROM posts WHERE id =$1', [postId]);
pool.end();Pool and Client usage notes
- 
In Node.js and some other environments, there's no built-in WebSocket support. In these cases, supply a WebSocket constructor function. import { Pool, neonConfig } from '@neondatabase/serverless'; import ws from 'ws'; neonConfig.webSocketConstructor = ws;
- 
In serverless environments such as Vercel Edge Functions or Cloudflare Workers, WebSocket connections can't outlive a single request. That means PoolorClientobjects must be connected, used and closed within a single request handler. Don't create them outside a request handler; don't create them in one handler and try to reuse them in another; and to avoid exhausting available connections, don't forget to close them.
For examples that demonstrate these points, see Pool and Client.
Advanced configuration options
For advanced configuration options, see neonConfig configuration, in the Neon serverless driver GitHub readme.
Developing locally with the Neon serverless driver
The Neon serverless driver enables you to query data over HTTP or WebSockets instead of TCP, even though Postgres does not natively support these connection methods. To use the Neon serverless driver locally, you must run a local instance of Neon's proxy and configure it to connect to your local Postgres database.
For a step-by-step guide to setting up a local environment, refer to this community guide: Local Development with Neon. The guide demonstrates how to use a community-developed Docker Compose file to configure a local Postgres database and a Neon proxy service. This setup allows connections over both WebSockets and HTTP.
Example applications
Explore the example applications that use the Neon serverless driver.
UNESCO World Heritage sites app
Neon provides an example application to help you get started with the Neon serverless driver. The application generates a JSON listing of the 10 nearest UNESCO World Heritage sites using IP geolocation (data copyright © 1992 – 2022 UNESCO/World Heritage Centre).

There are different implementations of the application to choose from.
- Raw SQL + Vercel Edge Functions- Demonstrates using raw SQL with Neon's serverless driver on Vercel Edge Functions 
- Raw SQL via https + Vercel Edge Functions- Demonstrates Neon's serverless driver over HTTP on Vercel Edge Functions 
- Raw SQL + Cloudflare Workers- Demonstrates using the Neon serverless driver on Cloudflare Workers and employs caching for high performance. 
- Kysely + Vercel Edge Functions- Demonstrates using kysely and kysely-codegen with Neon's serverless driver on Vercel Edge Functions 
- Zapatos + Vercel Edge Functions- Demonstrates using Zapatos with Neon's serverless driver on Vercel Edge Functions 
- Neon + pgTyped on Vercel Edge Functions- Demonstrates using using pgTyped with Neon's serverless driver on Vercel Edge Functions 
- Neon + Knex on Vercel Edge Functions- Demonstrates using using Knex with Neon's serverless driver on Vercel Edge Functions 
Ping Thing
The Ping Thing application pings a Neon Serverless Postgres database using a Vercel Edge Function and shows the journey your request makes. You can read more about this application in the accompanying blog post: How to use Postgres at the Edge
Neon serverless driver GitHub repository and changelog
The GitHub repository and changelog for the Neon serverless driver are found here.
References
- Fetch API
- node-postgres
- Drizzle-ORM
- Schema migration with Neon Postgres and Drizzle ORM
- kysely
- Zapatos
- Vercel Edge Functions
- Cloudflare Workers
- Use Neon with Cloudflare Workers
Need help?
Join our Discord Server to ask questions or see what others are doing with Neon. Users on paid plans can open a support ticket from the console. For more details, see Getting Support.