Skip to content

Use Redis as a multi-process atomic cache to avoid thundering herds and long calculations!

License

Notifications You must be signed in to change notification settings

anujdas/atomic_redis_cache

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

10 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

AtomicRedisCache

Build Status

Gem Version

AtomicRedisCache is an overlay on top of Redis that allows you to use it much like the Rails cache (ActiveSupport::Cache). Usage centers around the .fetch() method, which reads from a key if present and valid, or else evaluates the passed block and saves it as the new value at that key. The foremost design goals apply mainly to webapps and are twofold:

  • when the cached value expires, recalculate in the first process to access it, serving the old value to all other processes until completion, thus avoiding the dogpile/thundering herd effect (http://en.wikipedia.org/wiki/Thundering_herd_problem)
  • when calculation takes too long (i.e., due to db, network calls) and a recently expired cached value is available, fail fast with it and try calculating again later (for a reasonable # of retries)

Installation

Add this line to your application's Gemfile:

gem 'atomic_redis_cache'

And then execute:

$ bundle

Or install it yourself as:

$ gem install atomic_redis_cache

Usage

Ensure that you set the value of AtomicRedisCache.redis. This can be either an instance of redis-rb (or compatible), or a lambda/Proc evaluating to one. Ex.

AtomicRedisCache.redis = Redis.new(:host => '127.0.0.1', :port => 6379)

Then use it as you would Rails.cache:

>> AtomicRedisCache.fetch('key') { Faraday.get('http://example.com').status }
=> 200  # network call made
>> AtomicRedisCache.fetch('key') { raise }
=> 200  # value read from cache; block not called
>> AtomicRedisCache.write('key', {:new => 'value'})
=> true
>> AtomicRedisCache.read('key')
=> {:new => "value"}
>> AtomicRedisCache.delete('key')
=> true
>> AtomicRedisCache.read('key')
=> nil

There are a couple of configurable options that can be set per fetch/write:

  • :expires_in - expiry in seconds; defaults to a day
  • :race_condition_ttl - time to lock down key for recalculation; default 30s
  • :max_retries - # of times to retry cache refresh before expiring; default 3

AtomicRedisCache is most useful when wrapped around expensive but rarely changing calls, such as network APIs, that are hit frequently by many processes. For instance,

AtomicRedisCache.fetch('search.results') do
  JSON.parse(Faraday.get('https://internal-api/search?q=value').body)
end

will prevent your internal-api from getting pounded when the cached expires.

Contributing

  1. Fork it ( https://github.com/anujdas/atomic_redis_cache/fork )
  2. Create your feature branch (git checkout -b my-new-feature)
  3. Commit your changes (git commit -am 'Add some feature')
  4. Push to the branch (git push origin my-new-feature)
  5. Create a new Pull Request

About

Use Redis as a multi-process atomic cache to avoid thundering herds and long calculations!

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages