File: README.rdoc

package info (click to toggle)
syncache 1.2-1
  • links: PTS, VCS
  • area: main
  • in suites: jessie, jessie-kfreebsd, wheezy
  • size: 204 kB
  • ctags: 222
  • sloc: ruby: 1,525; sh: 112; makefile: 9
file content (98 lines) | stat: -rw-r--r-- 3,701 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
= SynCache - thread-safe time-limited cache with flexible replacement policy

== Synopsys

  require 'syncache'

  @cache = SynCache::Cache.new

  @cache.fetch_or_add('key:1') do
    # expensive operation
  end

  @cache.flush(/^key:/)


== Description

SynCache::Cache stores cached objects in a Hash that is protected by an
advanced two-level locking mechanism. Two-level locking ensures that:

* Multiple threads can add and fetch objects in parallel without
  stepping on each other's toes.

* While one thread is working on a cache entry, other threads can access
  the rest of the cache with no waiting on the global lock, no race
  conditions nor deadlock or livelock situations.

* While one thread is performing a long and resource-intensive
  operation, other threads that request the same data with #fetch_or_add
  method will be put on hold, and as soon as the first thread completes
  the operation, the result will be returned to all threads. Without
  this feature, a steady stream of requests with less time between them
  than it takes to complete one request can easily bury a server under
  an avalanche of threads all wasting resources on the same expensive
  operation.

When number of cache entries exceeds the size limit, the least recently
accessed entries are replaced with new data. This replacement strategy
is controlled by the SynCache::CacheEntry class and can be changed by
overriding its #replacement_index method.

Cache entries are automatically invalidated when their +ttl+ (time to
live) is exceeded. Entries can be explicitly invalidated by #flush
method. The method can use <tt>===</tt> operator to compare cache keys
against flush base (so that base can be e.g. a Regexp). When invoked
without the +base+ parameter, it invalidates all entries.

The +flush_delay+ initialization option allows to limit cache's flush
rate. When this option is set, SynCache will make sure that at least
this many seconds (it can also be a fraction) pass between two flushes.
When extra flushes are requested, invalidation of flushed entries is
postponed until earliest time when next flush is allowed.


== SynCache DRb Server

SynCache::Cache object can be shared between multiple Ruby processes,
even across different computers. All you need is the
<tt>syncache-drb</tt> script shipped with this module. This script will
start a daemon that serves a SynCache::Cache object over dRuby protocol,
with $SAFE set to 1 for security.

To access a remote cache, you will need to use DRb library:

  require 'drb'

  # connect to the remote cache
  @cache = DRbObject.new_with_uri('druby://localhost:9000')

  # allow remote cache to access local objects from fetch_or_add blocks
  DRb.start_service('druby://localhost:0')

For the above to work properly, all your fetch_or_add blocks should be
thread-safe, because DRb will run them in their own threads. Also, if a
Ruby process crashes is the middle of such block, the key will remain
locked in the remote cache until its +ttl+ runs out.

To work around these limitations, you can wrap access to a remote cache
using a SynCache::RemoteCache object:

  require 'syncache'

  # connect to the remote cache
  @cache = SynCache::RemoteCache.new('druby://localhost:9000')

SynCache::RemoteCache implements its own version of fetch_or_add that
runs the supplied block locally in the current thread and would give up
and take over a locked key if the client that originally locked it takes
too long to free it up.


== Copying

  Copyright (c) 2002-2011  Dmitry Borodaenko <angdraug@debian.org>

  This program is free software.
  You can distribute/modify this program under the terms of the GNU
  General Public License version 3 or later.