Running big monolithic Ruby on Rails applications on Heroku could be challenging,
especially when there are large amount of data. For example, if Rails needs to
load too many data from database to process a request, and render complex views
to generate a response, it may take too much time, and use too much memory. When
it takes more than 30 seconds, Heroku will close the HTTP connection and log an
H12 - Request timeout
error. When Rails uses more memory than the quota, Heroku
will log a R14 - Memory quota exceeded
error.
Here are some relating articles, and some personal opinions regarding those problems.
Some fixes do not not need much effort to put in place, but you can see the effect immediately.
I recommend that all Ruby webapps run at least 3 processes per server or container. This maximizes routing performance. ... ... Routing at higher layers (that is, at the load balancer or Heroku’s HTTP mesh) is far more difficult to do so efficiently, because the load balancer usually has no idea whether or not the servers its routing to are busy or not.
-- Configuring Puma, Unicorn and Passenger for Maximum Efficiency
More about Heroku's router:
A set of routers automatically routes HTTP requests from your app’s hostname(s) to your web dynos. The router uses a random selection algorithm to distribute traffic across your web dynos.
To make these memory leaks manageable, GitLab comes with the unicorn-worker-killer gem. This gem monkey-patches the Unicorn workers to do a memory self-check after every 16 requests. If the memory of the Unicorn worker exceeds a pre-set limit then the worker process exits. The Unicorn master then automatically replaces the worker process.
Available killers:
Because Ruby is relatively a slower language, it makes sense to move heavy computations to other components of the system. For example, use database for grouping, sorting and other calculations.
Once Ruby VM requests some memory form OS, it seldom returns the memory to OS.
However, one also need to consider that
free()
does not always mean the decrease of the amount of memory in use. If it does not return memory to OS, the amount of memory in use of the process never decrease. And, depending on the implementation ofmalloc()
, although doingfree()
it often does not cause returning memory to OS.
Therefore, a Ruby on Rails application should load least data.
- Filter data in database
- Only
select
necessary columns - Caching data in memory to avoid repeated allocating
- Use in-place modification when possible
Compared to Ruby profiling and GC tuning, those are just some low-hanging fruits, but they are effective enough for web applications under moderate load.
For example, the time complexity of Array.include?
is O(n), while the time complexity of Set.include?
is O(1).
The overall performance drop for multiple threads that work and multiple that invalidate is around 23,8%. Which means that if you have many threads (lets say you use Puma), and you work heavily with OpenStruct (or invalidate method cache in a different way) in all of them, you could get up to almost 24% more switching to something else. Of course probably you will gain less, because every library has some impact on the performance, but in general, the less you invalidate method cache, the faster you go.
-- Ruby global method cache invalidation impact on a single and multithreaded applications
And here is another list of Things that clear Ruby's method cache, with code examples:
According to Heroku's document Request Timeout, "the router will terminate the request if it takes longer than 30 seconds to complete.” 30s should be enough for most requests, however, some requests may need more time. For example, a request to export a large amount of data. This can be done in an asynchronous way. But if it has to be synchronous, Rails's Streaming could be helpful.
If the application takes more than a few seconds to boot, deployment will be possible to create timeout errors. It happens like this. During the application is booting, requests are held by Heroku router. When the application is ready to accept requests, Heroku router sends all those requests to the application immediately. If the server does not have the capacity to process those requests, some of them will result in timeout errors.
Set log_min_duration_statement
to a positive value representing how many milliseconds the query has to run before it's logged.
Add indexes for slow queries, then verify their effectiveness with explain
.
The most powerful tool at our disposal for understanding and optimizing SQL queries is EXPLAIN ANALYZE.
If a query only needs some special types of data, partial indexes might be helpful.
Because of MVCC, dead tuples will accumulate over time, resulting in bloated tables and indexes. Vacuuming can keep the database clean. You may need to tune relating configurations to reduce its impact on database performance.
Once vacuum has fallen behind it consumes more resources when it does run and it interferes with normal query operation. This can lead to a vicious cycle where database administrators mistakenly reconfigure the “resource hog” autovacuum to make it run less frequently or not at all. Autovacuum is not the enemy, and turning it off is disastrous.
Partitioning breaks large tables in a series of smaller tables, so that the database server can load less data into memory to execute some queries. When PostgreSQL can keep all data in memory, all queries would be faster. If there is not enough memory, PostgreSQL needs to load data from disk to memory, slowing down the responses dramatically.
pgslice is a great tool to automate the process. However it can only work with tables using integer as primary key.
Performance problems in some popular open source Rails applications and the ways to fix them:
-
ORM API Misuse
- Inefficient computation (IC) (
exists?
vsany?
) - Unnecessary Computation (UC) (a query inside a loop body, that could be computed once outside the loop body)
- Inefficient data accessing (ID) (N+1 selects problem)
- Unnecessary data retrieval (UD) (an application retrieves persistent data that it doesn’t then use)
- Inefficient Rendering (IR) (calls link_to within a loop to generate an anchor)
- Inefficient computation (IC) (
-
Database design issues
- Missing Fields (MF) (repeatedly computes a value that it could just store)
- Missing Database Indexes (MI)
-
Application design issues
- Content Display Trade-offs (DT) (returning all records instead of using pagination)
- Application Functionality Trade-offs (FT) (a side information on a page that is actually quite expensive to compute)
Here is the original paper.
The default Linux malloc will not release memory back to the OS unless it is at the end of the Ruby heap.
Some tips for performance. My favorite ones are Measure twice, optimize once and 80⁄20 rule.
Using jemalloc on Heroku, which may "reduce your Ruby application’s memory usage and response time".
Set MALLOC_ARENA_MAX=2 everywhere you start Sidekiq and enjoy your extra memory.
It looks like jemalloc gives conservatively an 11% speedup for a big concurrent Rails app.
Multithreaded Ruby programs may be consuming 2 to 4 times the amount of memory that they really need, due to fragmentation caused by per-thread memory arenas in malloc. To fix this, you can reduce the maximum number of arenas by setting the MALLOC_ARENA_MAX environment variable or by switching to an allocator with better performance, such as jemalloc.
A great summary about scaling a Rails app on Heroku.
When there are large amount of data in the database, simple migration task like adding a NOT NULL
column can bring servers down.
Learn how to make database schema changes without downtime from this article.
There is also a wonderful gem, strong_migrations. Detailed solutions are provided in its README.
Nearly all of my performance investigations start with identifying slow queries, or views that are running far more queries than are necessary.
Look for uses of present?, none?, any?, blank? and empty? on objects which may be ActiveRecord::Relations. Are you just going to load the entire array later if the relation is present? If so, add load to the call (e.g. @my_relation.load.any?) Be careful with your use of exists? - it ALWAYS executes a SQL query. Only use it in cases where that is appropriate - otherwise use present? or any other the other methods which use empty? Be extremely careful using where in instance methods on ActiveRecord objects - they break preloading and often cause N+1s when used in rendering collections. count always executes a SQL query - audit its use in your codebase, and determine if a size check would be more appropriate.