Connect to your database
Supabase provides multiple methods to connect to your Postgres database, whether you’re working on the frontend, backend, or utilizing serverless functions.
How to connect to your Postgres databases
How you connect to your database depends on where you're connecting from:
- For frontend applications, use the Data API
- For Postgres clients, use a connection string
- For single sessions (for example, database GUIs) or Postgres native commands (for example, using client applications like pg_dump or specifying connections for replication) use the direct connection string if your environment supports IPv6
- For persistent clients, and support for both IPv4 and IPv6, use Supavisor session mode
- For temporary clients (for example, serverless or edge functions) use Supavisor transaction mode
Quickstarts
Data APIs and client libraries
The Data APIs allow you to interact with your database using REST or GraphQL requests. You can use these APIs to fetch and insert data from the frontend, as long as you have RLS enabled.
For convenience, you can also use the Supabase client libraries, which wrap the Data APIs with a developer-friendly interface and automatically handle authentication:
Direct connection
The direct connection string connects directly to your Postgres instance. It is ideal for persistent servers, such as virtual machines (VMs) and long-lasting containers. Examples include AWS EC2 machines, Fly.io VMs, and DigitalOcean Droplets.
Direct connections use IPv6 by default. If your environment doesn't support IPv6, use Supavisor session mode or get the IPv4 add-on.
The connection string looks like this:
1postgresql://postgres:[YOUR-PASSWORD]@db.apbkobhfnmcqqzqeeqss.supabase.co:5432/postgres
Get your project's direct connection string from your project dashboard by clicking Connect.
Shared pooler
Every Supabase project includes a free, shared connection pooler. This is ideal for persistent servers when IPv6 is not supported.
Supavisor session mode
The session mode connection string connects to your Postgres instance via a proxy.
The connection string looks like this:
1postgres://postgres.apbkobhfnmcqqzqeeqss:[YOUR-PASSWORD]@aws-0-[REGION].pooler.supabase.com:5432/postgres
Get your project's Session pooler connection string from your project dashboard by clicking Connect.
Supavisor transaction mode
The transaction mode connection string connects to your Postgres instance via a proxy which serves as a connection pooler. This is ideal for serverless or edge functions, which require many transient connections.
Transaction mode does not support prepared statements. To avoid errors, turn off prepared statements for your connection library.
The connection string looks like this:
1postgres://postgres.apbkobhfnmcqqzqeeqss:[YOUR-PASSWORD]@aws-0-[REGION].pooler.supabase.com:6543/postgres
Get your project's Transaction pooler connection string from your project dashboard by clicking Connect.
Dedicated pooler
For paying customers, we provision a Dedicated Pooler (PgBouncer) that's co-located with your Postgres database. This will require you to connect with IPv6 or, if that's not an option, you can use the IPv4 add-on.
The Dedicated Pooler ensures best performance and latency, while using up more of your project's compute resources. If your network supports IPv6 or you have the IPv4 add-on, we encourage you to use the Dedicated Pooler over the Shared Pooler.
Get your project's Dedicated pooler connection string from your project dashboard by clicking Connect.
PgBouncer always runs in Transaction mode and the current version does not support prepared statement (will be added in a few weeks).
More about connection pooling
Connection pooling improves database performance by reusing existing connections between queries. This reduces the overhead of establishing connections and improves scalability.
You can use an application-side pooler or a server-side pooler (Supabase automatically provides one called Supavisor), depending on whether your backend is persistent or serverless.
Application-side poolers
Application-side poolers are built into connection libraries and API servers, such as Prisma, SQLAlchemy, and PostgREST. They maintain several active connections with Postgres or a server-side pooler, reducing the overhead of establishing connections between queries. When deploying to static architecture, such as long-standing containers or VMs, application-side poolers are satisfactory on their own.
Serverside poolers
Postgres connections are like a WebSocket. Once established, they are preserved until the client (application server) disconnects. A server might only make a single 10 ms query, but needlessly reserve its database connection for seconds or longer.
Serverside-poolers, such as Supabase's Supavisor in transaction mode, sit between clients and the database and can be thought of as load balancers for Postgres connections.
They maintain hot connections with the database and intelligently share them with clients only when needed, maximizing the amount of queries a single connection can service. They're best used to manage queries from auto-scaling systems, such as edge and serverless functions.
Connecting with SSL
You should connect to your database using SSL wherever possible, to prevent snooping and man-in-the-middle attacks.
You can obtain your connection info and Server root certificate from your application's dashboard:
Resources
Troubleshooting and Postgres connection string FAQs
Below are answers to common challenges and queries.
What is a “connection refused” error?
A “Connection refused” error typically means your database isn’t reachable. Ensure your Supabase project is running, confirm your database’s connection string, check firewall settings, and validate network permissions.
What is the “FATAL: Password authentication failed” error?
This error occurs when your credentials are incorrect. Double-check your username and password from the Supabase dashboard. If the problem persists, reset your database password from the project settings.
How do you connect using IPv4?
Supabase’s default direct connection supports IPv6 only. To connect over IPv4, consider using the Supavisor session or transaction modes, or a connection pooler (shared or dedicated), which support both IPv4 and IPv6.
Where is the Postgres connection string in Supabase?
Your connection string is located in the Supabase Dashboard. Click the Connect button at the top of the page.
Can you use Supavisor and PgBouncer together?
You can technically use both, but it’s not recommended unless you’re specifically trying to increase the total number of concurrent client connections. In most cases, it is better to choose either PgBouncer or Supavisor for pooled or transaction-based traffic. Direct connections remain the best choice for long-lived sessions, and, if IPv4 is required for those sessions, Supavisor session mode can be used as an alternative. Running both poolers simultaneously increases the risk of hitting your database’s maximum connection limit on smaller compute tiers.
How does the default pool size work?
Supavisor and PgBouncer work independently, but both reference the same pool size setting. For example, if you set the pool size to 30, Supavisor can open up to 30 server-side connections to Postgres each for its session mode port (5432) and transaction mode port (6543), and PgBouncer can also open up to 30. If both poolers are active and reach their roles/modes limits at the same time, you could have as many as 60 backend connections hitting your database, in addition to any direct connections. You can adjust the pool size in Database settings in the dashboard.
What is the difference between client connections and backend connections?
There are two different limits to understand when working with poolers. The first is client connections, which refers to how many clients can connect to a pooler at the same time. This number is capped by your compute tier’s “max pooler clients” limit, and it applies independently to Supavisor and PgBouncer. The second is backend connections, which is the number of active connections a pooler opens to Postgres. This number is set by the pool size for that pooler.
12345Total backend load on Postgres = Direct connections + Supavisor backend connections (≤ supavisor_pool_size) + PgBouncer backend connections (≤ pgbouncer_pool_size)≤ Postgres max connections for your compute instance
What is the max pooler clients limit?
The “max pooler clients” limit for your compute tier applies separately to Supavisor and PgBouncer. One pooler reaching its client limit does not affect the other. When a pooler reaches this limit, it stops accepting new client connections until existing ones are closed, but the other pooler remains unaffected. You can check your tier’s connection limits in the compute and disk limits documentation.
Where can you see current connection usage?
You can track connection usage from the Reports section in your project dashboard. There are three key reports:
- Database Connections: shows total active connections by role (this includes direct and pooled connections).
- Dedicated Pooler Client Connections: shows the number of active client connections to PgBouncer.
- Shared Pooler (Supavisor) Client Connections: shows the number of active client connections to Supavisor.
Keep in mind that the Roles page is not real-time, it shows the connection count from the last refresh. If you need up-to-the-second data, set up Grafana or run the query against pg_stat_activity
directly in SQL Editor. We have a few helpful queries for checking connections.
12345678910-- Count connections by application and user nameselect count(usename), count(application_name), application_name, usenamefrom pg_stat_ssl join pg_stat_activity on pg_stat_ssl.pid = pg_stat_activity.pidgroup by usename, application_name;
123456789101112131415-- View all connections SELECT pg_stat_activity.pid, ssl AS ssl_connection, datname AS database, usename AS connected_role, application_name, client_addr, query, query_start, state, backend_startFROM pg_stat_sslJOIN pg_stat_activity ON pg_stat_ssl.pid = pg_stat_activity.pid;
Why are there active connections when the app is idle?
Even if your application isn’t making queries, some Supabase services keep persistent connections to your database. For example, Storage, PostgREST, and our health checker all maintain long-lived connections. You usually see a small baseline of active connections from these services.
Why do connection strings have different ports?
Different modes use different ports:
- Direct connection:
5432
(database server) - PgBouncer:
6543
(database server) - Supavisor transaction mode:
6543
(separate server) - Supavisor session mode:
5432
(separate server)
The port helps route the connection to the right pooler/mode.
Does connection pooling affect latency?
Because the dedicated pooler is hosted on the same machine as your database, it connects with lower latency than the shared pooler, which is hosted on a separate server. Direct connections have no pooler overhead but require IPv6 unless you have the IPv4 add-on.
How to choose the right connection method?
Direct connection:
- Best for: persistent backend services
- Limitation: IPv6 only
Shared pooler:
- Best for: general-purpose connections (supports IPv4 and IPv6)
- Supavisor session mode → persistent backend that require IPv4
- Supavisor transaction mode → serverless functions or short-lived tasks
Dedicated pooler (paid tier):
- Best for: high-performance apps that need dedicated resources
- Uses PgBouncer
You can follow the decision flow in the connection method diagram to quickly choose the right option for your environment.