MultiDbsLoadBalancer
Allow to setup load balancers sit on top of rails multi-databases.
Installation
gem "multi_dbs_load_balancer"
$ bundle install
$ rails g multi_dbs_load_balancer:installUsage
Declaring load balancers
# config/initializers/multi_dbs_load_balancer.rb
    load_balancer.db_down_time = 120
    load_balancer.redis_down_time = 120
    load_balancer.init :rr_load_balancer,
        [
            {role: :reading1}, 
            {role: :reading2},
            {role: :reading3},
        ],
        algorithm: :round_robin,
        redis: Redis.new(...)
    load_balancer.init :us_lrt_load_balancer,
        [
            {shard: :us, role: :reading1}, 
            {shard: :us, role: :reading2},
        ],
        algorithm: :least_response_time,
        redis: Redis.new(...)Now you could use them on controllers/services ...
# products_controller.rb
def index
    @products = ActiveRecord::Base.connected_through(:rr_load_balancer) { Product.all }
    # alias methods: connected_by, connected_through_load_balancer
endYou could also create and use a Middleware to wrap load balancer base on the request, for example:
class LoadBalancerMiddleware
    def initialize(app)
        @app = app
    end
    def call(env)
        request = ActionDispatch::Request.new(env)
        if is_something?(request)
            ActiveRecord::Base.connected_through(:rr_load_balancer) do
                @app.call(env)    
            end
        else
            @app.call(env)
        end
    end
    private def is_something?(request)
        # for example: check if reading request
        request.get? || request.head?
    end
end
Rails.application.config.app_middleware.use LoadBalancerMiddlewareNotes
- 
Support algorithms: round_robin,weight_round_robin,least_connection,least_response_time,hash,randomized- 
hashalgorithm require you pass the:sourceparameter@developers = Developer.connected_by(:hash_load_balancer, source: request.ip) { Developer.all } 
- 
weight_round_robinrequire setupweightfor each replica databaseload_balancing :wrr, [ {role: :reading1, weight: 5}, {role: :reading2, weight: 3}, {role: :reading3, weight: 2}, ], algorithm: :weight_round_robin 
 
- 
- 
Distribute If you launch multiple servers then you wish your load balancers will share states between servers, there're 3 algorithms that will do that if you provide a redis server: - 
round_robinwill share the current database
- 
least_connectionandleast_response_timewill share the sorted list of databases
 Other algorithms are independent on each server, so you don't need to provide a redis server for them. 
- 
- 
Fail-over All load balancers here are passive, they don't track database connections or redis connections. Whenever it could not connect to a database, it mark that database have down for db_down_timeseconds and ignore it on the next round, and try to connect to the next available database.After db_down_timeseconds, this database will be assumed available again and the load balancer will not ignore it and try to connect again.Whenever the redis-server has down (or you dont setup redis), distribute load balancers will process offline on each server until redis come back. 
Development
run test
rake setup_db
rake specContributing
Bug reports and pull requests are welcome on GitHub at https://github.com/[USERNAME]/rails_dbs_load_balancer.