Scaling the Monolith

Many companies start out with a single web application. As the team and codebase grow, things feel less organized and common tasks like booting the app and running the test suite take longer and longer. It can be tempting to turn to microservices to alleviate some of this pain. However, distributed systems add a significant amount of complexity and mental overhead.

Before you decide to split apart your app, there are a number of tactics you can use to scale it majestically. Spend a significant amount of time trying to solve your existing problems before making big changes.

The topics we’ll cover are:

The examples are geared towards Rails apps, but the principles apply to any codebase.

Code

Rails models and controllers tend to get larger and larger. Rails introduced concerns as one way to address this. Concerns allow you to pull out related logic into a separate file.

Service objects are another nice pattern for this. Here’s an example of a service object. There’s not a standard way to create service objects, but it’s a good idea to decide on a convention for your app. You can use gems like Interactor to establish one.

Use namespaces to organize code.

class Admin::UsersController < Admin::BaseController
end

Some teams also prefer to use Rails engines, although I’m not a fan of this approach. Here’s a good comparison of the pros and cons of each.

Errors

As the team grows, it’s important that errors get routed to the right place. You can use the ownership gem to help with this. Add it to controllers, jobs, and rake tasks.

class WelcomeJob < ApplicationJob
  owner :growth
end

git blame can help with assigning initial owners.

Boot Times & Memory

As your app accumulates more gems and files, its boot time and memory usage grow. There have been a number of projects over the years to speed up boot time. Spring was introduced in Rails 4.1 and keeps your app running in the background so it doesn’t have to boot every time you run a new command.

Last year, Shopify released Bootsnap, which caches expensive loading computations. It’s now part of Rails 5.2 and can be used with earlier versions of Rails as well. With Bootsnap, “the core Shopify platform - a rather large monolithic application - boots about 75% faster, dropping from around 25s to 6.5s.”

Another tactic is lazy loading files. Instead of incurring a speed and memory penalty at startup to load files, you can incur it the first time a request or job requires it. If it’s never needed, it’s never loaded. You can specify which gems to load in your Gemfile.

gem 'groupdate', require: false

You can also use different Bundler groups to selectively load gems for different environments.

group :web do
  gem 'rack-attack'
end

group :admin_web do
  gem 'activeadmin'
end

group :worker do
  gem 'premailer-rails'
end

Read how to set it up here.

Use Bumbler to see how long each gem takes to load and Derailed Benchmarks to see memory usage. Focus on the top ones and leave the rest.

If a gem is slow, there’s a chance it may be doing a lot of work upfront. You can try to debug the gem and fix it. Here’s an example of speeding up initial load time by only reading a CSV file when it’s needed.

Testing

As the number of tests grow, the test suite can become slow. TestProf provides a number of tools to profile and optimize your tests. You can also use a library like Database Cleaner to quickly clean the database after tests.

In development, you can use Guard for Minitest or RSpec to automatically run tests when relevant files are modified. Also make sure it’s easy to manually run common subsets of tests. You can use tags in RSpec for this.

rspec --tags growth

The key to speeding up the entire test suite is parallelization. Stripe has a great post about how they were able to get three hours of tests to run in three minutes. With continuous integration, split tests across multiple machines. Both Travis and Circle support this. You can use ParallelTests in development to use all the cores on your machine. Rails 6 will run tests in parallel by default.

Another way to speed up tests is to change your schema dump format to SQL.

config.active_record.schema_format = :sql

This allows you to load the database schema for tests without booting the Rails app. With Postgres, you can use:

psql < db/structure.sql

To prevent slow tests from being added, automatically fail tests that take too long. With RSpec, you can do:

RSpec.configure do |config|
  config.around(:each) do |example|
    duration = Benchmark.realtime(&example)
    raise "Test took over 2 seconds to run" if duration > 2
  end
end

Start with a higher value and ratchet it down as you fix tests that are slow. You can see the slowest tests with:

rspec --profile

As the number of tests grows, there’s a higher chance of a random network issue causing an individual test to fail. Automatically retry failing tests to cut down on noise. With RSpec, you can use RSpec::Retry for this.

require "rspec/retry"

RSpec.configure do |config|
  config.around(:each) do |example|
    example.run_with_retry retry: 2 # must be 2 to retry once (shrug)
  end
end

For test failures, make sure they get routed to the committer. You can use webhooks from your CI platform to do this.

Databases

Modern relational databases can scale extremely well if you follow best practices.

One of the most important things you can do is set a statement timeout to prevent bad queries from taking too many resources.

production:
  variables:
    statement_timeout: 250 # ms

It’s also good to track which queries consume the most CPU time. With Postgres, you can use PgHero for this.

PgHero

Use Marginalia to make it easy to identity the origin of queries. This adds a comment to the end of queries like /*application:Datakick,controller:items,action:edit*/ so you can see where they’re coming from.

Add defensive measures as well. For instance, pause low priority job queues automatically when the database CPU gets too high.

Sidekiq::Queue.new("low").pause!

As the team grows, so does the chance of someone accidentally running a migration that takes down the site. Strong Migrations can help prevent downtime due to database migrations. It raises an error if you try to run an unsafe operation and gives instructions for a better way to do it.

Strong Migrations

Some tables can accumulate a lot of columns. You can split them into multiple tables based off concern that have a 1-to-1 relationship.

Scale reads by fixing N+1 queries and caching frequent queries. Bullet can help you identify N+1 queries. If you still have high load after spending a good amount of time on these, use Distribute Reads for replicas.

Scale writes and space with additional databases. Use Multiverse to manage them. This can also be good if you have business domains with different workloads. It adds complexity and removes the ability to join certain tables, but can increase stability.

Partitioning is another strategy for space for tables that only need recent data. You can use pgslice for Postgres.

While Rails has built-in connection pooling, connections can become an issue when you have a lot of servers. With Postgres, use a connection pooler like PgBouncer when you start to hit 500 connections.

Be hesitant to introduce new data stores. Most of the time you can just table data. It’s often not worth having another technology to manage if your current stack can do the job.

Stability

Your monolith is one codebase, but you can increase stability by isolating different parts of the app in production. Have separate load balancers and web servers for your customer site and admin site so customers aren’t impacted if the admin site goes down. Use separate workers for different groups of queues so a backed up queue or bad job won’t affect the whole system.

You can separate by business domain, which will be aligned with teams if you have vertical teams. This also allows you to scale different parts of your app independently as if they were different services.

Conclusion

As you’ve seen, there are a number of things you can do to scale your monolith. Focus on developer happiness and productivity as well as system stability. Keep track of metrics over time that impact developers, like boot time, test suite time, and deploy time. It’s also good to invest in projects that make it comfortable to ship code fast, like quick rollbacks. Overall, spend a decent amount of time trying to solve your exact pain points before breaking your app apart to solve them.

Published November 7, 2018

Thanks to Arnaud Ferreri, Brandon Leonardo, and Nick Elser for reading drafts of this.


You might also enjoy

navigator.sendBeacon and Rails

Argon2 with Devise

Bulk Upsert in Ruby/Rails


All code examples are public domain.
Use them however you’d like (licensed under CC0).