Processing rack_throttle exceptions in rails appli

2019-04-14 03:26发布

问题:

How can i process errors generated by rack-throttle gem when rate limit is excedeed? Now i just get a response containing the following:

Internal Server Error

undefined method `each' for "403 Forbidden (Rate Limit Exceeded)\n":String

Here is the stack trace

ERROR NoMethodError: undefined method `each' for "403 Forbidden (Rate Limit Exceeded)\n":String
/home/rkapitonov/.rvm/gems/ruby-1.9.2-p290/gems/activerecord-3.1.1/lib/active_record/query_cache.rb:45:in `each'
/home/rkapitonov/.rvm/gems/ruby-1.9.2-p290/gems/activerecord-3.1.1/lib/active_record/connection_adapters/abstract/connection_pool.rb:458:in `each'
/home/rkapitonov/.rvm/gems/ruby-1.9.2-p290/gems/rack-1.3.6/lib/rack/body_proxy.rb:23:in `method_missing'
/home/rkapitonov/.rvm/gems/ruby-1.9.2-p290/gems/railties-3.1.1/lib/rails/rack/content_length.rb:26:in `call'
/home/rkapitonov/.rvm/gems/ruby-1.9.2-p290/gems/railties-3.1.1/lib/rails/rack/log_tailer.rb:14:in `call'
/home/rkapitonov/.rvm/gems/ruby-1.9.2-p290/gems/rack-1.3.6/lib/rack/handler/webrick.rb:59:in `service'
/home/rkapitonov/.rvm/rubies/ruby-1.9.2-p290/lib/ruby/1.9.1/webrick/httpserver.rb:111:in `service'
/home/rkapitonov/.rvm/rubies/ruby-1.9.2-p290/lib/ruby/1.9.1/webrick/httpserver.rb:70:in `run'
/home/rkapitonov/.rvm/rubies/ruby-1.9.2-p290/lib/ruby/1.9.1/webrick/server.rb:183:in `block in start_thread'

How can i process this error and render some custom file or make a redirect?

回答1:

Finally i found a solution

The problem was in rack-throttle gem. Response body should be an array of strings, so to fix the issue i've just overridden http_error method and that did the trick for me. You should make http_status(code) + (message.nil? ? "\n" : " (#{message})\n") an array so just take it in square brackets.

class Rack::Throttle::Limiter
  def http_error(code, message = nil, headers = {})
    [code, {'Content-Type' => 'text/plain; charset=utf-8'}.merge(headers),
    [http_status(code) + (message.nil? ? "\n" : " (#{message})\n")]]
  end
end

Put this inside config/initializers/rack_throttle.rb. The name of the file may be changed to whatever you like.



回答2:

You need create a Rack::Middleware to check if there are an exceed limit from rack-throttle and do what you want.

Or you can override the method Rack::Throttle::Limiter#rate_limit_exceeded to do what you really want : https://github.com/datagraph/rack-throttle/blob/master/lib/rack/throttle/limiter.rb#L177

You can pass some option In your Rack::Trottle middelware too :

https://github.com/datagraph/rack-throttle/blob/master/lib/rack/throttle/limiter.rb#L19