PG::ConnectionBad: remaining connection slots are reserved for non-replication superuser connections
That’s an error from Postgres, and it’s telling you that you’ve used all of your available connections at your database (as opposed to your connection pool). Maybe not the most helpful error message in the world, but fortunately, understanding and solving it is pretty simple math.
This article is a companion to our Database Connections Calculator. They’re meant to be used together, so go ahead and pop it open in a new tab.
What does it do?
Our Database Connections Calculator is a tool for calculating how many potential database connections your app needs to have available in Postgres. It’s especially common to encounter connection errors on Heroku with their cheaper Heroku Postgres plans because the connection limits are so low.
Plan Name
Connection Limit
Essential-0
20
Essential-1
20
Essential-2
40
Standard-0
120
When you run multiple dynos with multiple processes, connections can add up quickly. Our tool helps you calculate and visualize those connections so you can choose the correct Postgres plan or adjust your app configuration to fit within your current Postgres plan.
Is the tool useful on platforms other than Heroku?
Yep, they all work pretty much the same! We’ll mostly be using Heroku jargon for consistency, but here’s how the terms map to other platforms:
Platform
Terms
Heroku
“app”
“process”
“dyno”
Render
“service group” or “environment”
“service”
“instance”
Fly
“app”
“process”
“machine”
Railway
“environment”
“service”
“instance”
Amazon ECS
“cluster”
“service”
“task”
Calculating potential database connections
Potential database connections are a factor of how many dynos (instances, etc.) you’re running, and what’s happening on those dynos. Dynos can be running multiple processes, each of which can share many database connections via a connection pool.
In the sections below, we’ll explain each of those components, and how to dial them into the calculator to determine your maximum potential database connections.
Dynos
You can see the number of running dynos by running heroku ps or in your Heroku dashboard.
If you’re using an autoscaling add-on such as Judoscale, make sure you use the maximum scale in your configured autoscale range.
Processes
It’s common to run multiple Puma or Unicorn processes (sometimes called “workers”) within web dynos. You can likewise run multiple Sidekiq, Resque, or Delayed Job processes in worker dynos. Running multiple processes lets you utilize multiple CPU cores for parallel processing.
For Puma, check the workers line in your puma.rb file. If this line is commented out, you’re running a single process.
If it refers to the WEB_CONCURRENCY environment variable, run heroku config:get WEB_CONCURRENCY or click “reveal config vars” in your Heroku settings page.
For Sidekiq, you can only run multiple processes with the sidekiqswarm command found in Sidekiq Enterprise. If you’re not using Sidekiq Enterprise, then you’re almost certainly only running a single Sidekiq process. Note that your Sidekiq concurrency setting configures threads, not processes.
Threads
This likely comes from your RAILS_MAX_THREADS environment variable, but you’ll need to check your code to be sure. For Puma, check the threads line in your puma.rb file.
For Sidekiq, your “concurrency” setting is the thread count. If you haven’t otherwise specified a concurrency setting in a sidekiq.yml or -c command line option, then this is also driven by RAILS_MAX_THREADS.
👀 Note
You generally want your thread count be equal or less than your database pool size. When you have more threads than pooled database connections, threads will need to wait for an available connection. This increases response time and sometimes raises ActiveRecord::ConnectionTimeoutError exceptions.
Your thread count will only impact your total database connections if it is smaller than your database pool (and there’s rarely a good reason to do so).
Rails Async Queries
Rails 7 introduced “async queries”—which allows for executing queries asynchronously using threads without blocking the main thread. This can be great for handling parallel queries without slowing down the application, but it increases the database connections required. If you’re using this feature, your application might need more connections than just your thread count.
Rails defaults the global_executor_concurrency to 4, which allows up to 4 additional threads needing database connections. When you enable this option in the calculator, it automatically adds four connections for each process. If you’ve changed this default concurrency setting, you’ll need to account for that when calculating the total number of connections.
Database pool
Every Rails process has it’s own database pool, which is shared by all threads in that process. This is configured via pool in your database.yml file.
By default it will use the RAILS_MAX_THREADS environment variable, so run heroku config:get RAILS_MAX_THREADS or view your settings page.
It’s also possible to override this environment variable (or any environment variable) in your Procfile or other executable. For example, I recommend customizing RAILS_MAX_THREADS for worker dynos like this:
This lets you configure database pools for web and worker dynos independently. If you’re doing something like this, you’ll need to get the WORKER_THREADS config var (or whatever you’ve named it) for your “workers” pool configuration.
Heroku’s Preboot feature allows zero-downtime deploys (rolling deploys) by running two simultaneous versions of your web dynos for several minutes. Traffic is switched from the old dynos to the new dynos once the new dynos are ready to handle requests.
During this short period, you’re running double the number of web dynos, so your web dynos can require up to twice as many database connections. Checking the “Preboot” box will double the number of potential connections from your web dynos.
Total database connections
This number represents the total possible connections with this configuration. Since database connections are created lazily, you might never actually use this many connections, but your database should support that scenario.
Also consider other sources of database connections:
One-off dynos. When you use the heroku run command, you’re create a new “one-off” dyno for as long as your command runs. This command might create new connections not factored into the calculator.
Migrations. If you run database migrations in a release phase, this happens on a one-off dyno and will create connections not factored into the calculator.
Scheduled tasks. If you use a scheduled task runner like Heroku Scheduler, this also uses one-off dynos.
Third-party tools. Do you connect to your database from third-party reporting or monitoring tools? Consider those connections as well.
PGBouncer
Large-scale applications often find themselves needing more than 500 database connections, which is the highest limit of any Heroku Postgres plan. In this scenario it makes sense to use PGBouncer, which is a server-side connection pool for Postgres and is supported natively by Heroku.
Rather than connecting directly to the database, your application will connect to PGBouncer, which runs on the Postgres server. PGBouncer connects to the database, and it will never use more than 75% of the database limit.
For example, if your Postgres limit is 400 connections, PGBouncer will establish up to 300 connections to Postgres. You can create up to 10,000 connections to PGBouncer, and PGBouncer manages access to the pool of 300 database connections.