Skip to main content

How to change table for Rails models on the fly

2 min read

Use Case: Maybe you need your multi tenant application database to partition horizontally (shard). Probably you want to store different customers data in different tables like customer_01, customer_02. But you don't want multiple models for these, but want to access data with single Model interface.

With Rails, you can achieve this very easily. Rails ActiveRecord::Base interface provides us a class method table_name. We'll use this to achieve our goal. We'll change this in runtime, dynamically.

Here is an example:

cTable4 ❯ rails c -e production
Running via Spring preloader in process 60375
Loading production environment (Rails 4.2.4)
irb(main):001:0> Book.first
  Book Load (0.1ms)  SELECT  "books".* FROM "books"  ORDER BY "books"."id" ASC LIMIT 1
=> #<Book id: 1, name: "ELoquent Ruby", author: "Russ", year: "2011", created_at: "2016-01-07 08:40:47", updated_at: "2016-01-07 08:40:47">
irb(main):002:0> Book.table_name = 'titles'
=> "titles"
irb(main):003:0> Book.first
  Book Load (0.2ms)  SELECT  "titles".* FROM "titles"  ORDER BY "titles"."id" ASC LIMIT 1
=> #<Book id: 1, title: "Practical OOD Ruby", author: "Sandi Metz">
irb(main):004:0>

For convenience, you might want to set some helper class methods.

class Customer < ActiveRecord::Base
  def self.change_to(id)
    self.table_name = "customer_#{id}"
  end
end

Now you can simple call that method and change the table:

Customer.change_to(06)

Yes, it's that simple. Happy development :)

If you've any other tips in mind, don't forget to share with us in the comments.