Octopus vs Makara : Read-write adapters for ActiveRecord

Most of the web applications these days use database replication for their database setup. This involves a master database and multiple replicas known as slaves. The master database and the slaves share the data and the schema, and are always kept in a consistent state. In this architecture, all writes and updates must take place on the master. Reads, however, may take place on one or more slaves. This design makes writes to the database faster and improves the performance of the application. It also provides a highly reliable and highly available setup; if the master database were to go down, reads can still continue from the slaves.

In a Rails application, ActiveRecord is an ORM framework which is used to connect objects of the application to database tables. ActiveRecord provides adapters for almost all relational database systems. However, ActiveRecord does not support replication. This means that, if say, we are using MySQL as the database and the ActiveRecord MySQL adapter over it, although, MySQL supports replication, since ActiveRecord does not, your application will not be able to read from the slave databases.

There are a couple of gems which help you achieve this, Octopus and Makara being the most widely used. The default behaviour after using any of this gem becomes ‘all reads from slaves, writes to master’. Both perform the basic job of allowing reads from slaves pretty well, however, this post compares these two gems on a few intricate parameters and briefly explains which one might be suitable to use when.

For the purpose of comparison, lets assume our application has two databases, A and B. Each of these databases follow master-slave architecture, hence A has 2 slaves; A1 and A2, while B has 2 slaves; B1 and B2.

1. Slave failure

Suppose while making a read query on database A1, if it goes down, the expected behaviour would be that the query would be re-directed to either A2 or master A.

Octopus does not support re-direction to another slave or master. If a slave is being queried and it goes down, it gives a database error to the application. Makara handles slave failure by maintaining a blacklist. When a slave goes down it is blacklisted for a specific duration. In the meanwhile, queries will go to another slave. The blacklisted slave will be retried after the specified duration has passed.

2. Replication lag

Quoting from a MongoLab blogreplication lag is the delay between the time an operation occurs on the primary and the time that same operation gets applied on the secondary.  If some data is written to database A, it might take some time before that data is available on slaves A1 and A2. This might be a problem.

Octopus does not provide any solution to this, but advises that queries which might suffer from replication lag should be made on master rather than slaves ( Octopus provides an easy way to do this, more on this in the next point ).

Makara tries to solve the problem of replication lag by using a cookie. It has a “stick_to_master!” method which adds a cookie which will persist the master context for a duration which you can specify ( normally longer than the replication lag ). Suppose I create a new record in one request and immediately want to read it in the next request, the stick_to_master! method ensures that the read request comes to master rather than the slaves.

3. Flexibility to distribute reads across master and slave

The best way to solve replication lag is to query master rather than slave for queries which might suffer from replication lag. In this regard, Octopus easily allows configuring which reads should go to master and which should go to slave. By default all the reads are from slave, but using the “using(:master)” method, you can query from the master. Also, you can change the default, so that all reads can be from master, and then use “using(:slave)” method to query from the slaves when required.

Makara does not have such flexible configuration in-built. Its default is all reads from slave which cannot be changed. However, a good samaritan has written an extension which changes the default, so all reads can be from master and only those specified will be from slave.

4. Reading from slaves of multiple databases

Octopus, although the documentation says otherwise, does not support reading from slaves of multiple databases. Hence, when it needs data from database B, it should ideally query slave B1/B2 but it does not, the query goes to the master B. Makara supports querying slaves of multiple databases.

5. Error handling

Octopus doesn’t handle errors that well. Since Makara positions itself as an ActiveRecord adapter, it can catch errors from the database adaptors and handle them gracefully.

6. Community support

Octopus is more active than Makara and seems to be more in use. It also has a good community presence and questions / problems get replied to quickly. However, Makara is steadily gaining an audience and it attempts to overcome the drawbacks of Octopus.

Based on your requirements and priorities you will need to decide which one of these tools is more suitable for you. Hope this post gives you an insight into making that choice. Do post your suggestions and feedback. Happy coding!

More Reading :

Octopus : http://dandemeyere.com/blog/rails-replication-with-octopus

Makara : http://tech.taskrabbit.com/blog/2013/01/02/makara/

7 Comments

Add yours →

  1. Hey thank you for the great post. It looks like I will end up going with Makara. Can you elaborate on how to use stick_to_master! to avoid replication lag. I do not find any real examples online.

    Thank you,
    Ben

    Like

    • Thanks Ben! Regarding your question, replication lag in Makara can be avoided by reading from master in the below 3 ways :
      1) One common scenario where you may need to read from master is immediately after a record’s creation. This can be handled by using the option “sticky:true” in the configuration file.

      2) If you want to change the default behavior of Makara to make all reads from master and specific reads from slave, use this extension (https://github.com/ankane/shorts/blob/master/Scaling-Reads.md). This is recommended and we had also used this.

      3) If you want to continue with the default behavior of Makara, which is all reads from slaves, but want to redirect a specific query to master, use stick_to_master! Unfortunately, I have not used this method and hence cannot provide details about this.

      Hope this helps!

      Regards,
      Yahya.

      Like

  2. Hi, thank you for the article !
    However as of now octopus is no longer maintained and I would advise people to use makara instead (source: https://github.com/thiagopradi/octopus/issues/330#issuecomment-251219225)

    Regards,
    Dimitri

    Like

  3. Dude, really nice post. It really explains things that are really important when choosing either of these gems. I’m bookmarking this blog post.

    Thanks for sharing!

    Like

  4. Amazing! Thanks!

    Liked by 1 person

Leave a comment