File Coverage

blib/lib/XAO/Cache.pm
Criterion Covered Total %
statement 56 57 98.2
branch 23 34 67.6
condition 16 29 55.1
subroutine 8 8 100.0
pod 4 4 100.0
total 107 132 81.0


line stmt bran cond sub pod time code
1             =head1 NAME
2              
3             XAO::Cache - generic interface for caching various data
4              
5             =head1 SYNOPSIS
6              
7             my $cache=XAO::Cache->new(
8             retrieve => &real_retrieve,
9             coords => [qw(outer inner)],
10             size => 100,
11             expire => 30*60,
12             backend => 'Cache::Memory',
13             );
14            
15             my $d1=$cache->get(outer => 123, inner => 'foo');
16              
17             my $d2=$cache->get($self, outer => 234, extra => 'bar');
18              
19             =head1 DESCRIPTION
20              
21             NOTE: It is almost always better to use Config::cache() method instead
22             of creating a cache directly with its new() method. That will also save
23             on the initialization step - cache object themselves are cached and
24             reused in that case.
25              
26             XAO::Cache is a generic cache implementation for caching various "slow"
27             data such as database content, results of remote requests and so on.
28              
29             There is no operation of storing data into the cache. Instead cache
30             is provided with a method to retrieve requested content whenever
31             required. On subsequent calls a cached value would be returned until
32             either expiration time is elapsed or cache has overgrown its maximum
33             size. In which case the real query will be made again to actually
34             retrieve data.
35              
36             That means that cache always returns valid data or throws an error if
37             that is not possible.
38              
39             To force the cache to use "retrieve" to get a new value that is stored
40             in the cache give an extra "force_update" parameter to the get() method.
41              
42             =head1 METHODS
43              
44             Here is the alphabetically arranged list of methods:
45              
46             =over
47              
48             =cut
49              
50             ###############################################################################
51             package XAO::Cache;
52 3     3   503 use strict;
  3         5  
  3         87  
53 3     3   14 use XAO::Utils;
  3         6  
  3         174  
54 3     3   17 use XAO::Errors qw(XAO::E::Cache);
  3         6  
  3         52  
55 3     3   19 use XAO::Objects;
  3         6  
  3         2629  
56              
57             our $VERSION=2.1;
58              
59             ###############################################################################
60              
61             =item drop ($%)
62              
63             Removes an element from cache. Useful to make cache aware of changes in
64             the cached element -- when cached data are no longer valid.
65              
66             Arguments must contain a list of coordinates the same as in get()
67             method.
68              
69             =cut
70              
71             sub drop ($%) {
72 8     8 1 148 my $self=shift;
73 8         48 my $backend=$self->{'backend'};
74            
75 8 50 33     32 my $object=ref($_[0]) && ref($_[0]) ne 'HASH' ? shift(@_) : undef;
76 8         21 my $args=get_args(\@_);
77              
78 8         15 my @c=map { $args->{$_} } @{$self->{'coords'}};
  16         35  
  8         13  
79 8 50       22 defined($c[0]) ||
80             throw XAO::E::Cache "get - no first coordinate ($args->{coords}->[0])";
81              
82 8         29 $backend->drop(\@c);
83             }
84              
85             ###############################################################################
86              
87             =item drop_all ($)
88              
89             Remove all elements from the cache.
90              
91             =cut
92              
93             sub drop_all ($) {
94 1     1 1 11 my $self=shift;
95              
96 1 50       11 if($self->{'backend'}->can('drop_all')) {
97 1         5 $self->{'backend'}->drop_all();
98             }
99             else {
100 0         0 eprint "Cache backend '$self->{'backend'}' does not support drop_all()";
101             }
102             }
103              
104             ###############################################################################
105              
106             =item get ($%)
107              
108             Retrieve a data element from the cache. The cache can decide to use real
109             'retrieve' method to get the data or return previously stored value
110             instead.
111              
112             All arguments given to the get() method will be passed to 'retrieve'
113             method. As a special case if retrieve is a method of some class then a
114             reference to object of that class must be the first argument followed by
115             a hash with arguments.
116              
117             Example of calling 'retrieve' as a function:
118              
119             $cache->get(foo => 123, bar => 234);
120              
121             Example of calling 'retrieve' as a method:
122              
123             $cache->get($object, foo => 123, bar => 123);
124              
125             Example of forcing an update of cache value:
126              
127             $cache->get(foo => 123, bar => 234, force_update => 1);
128              
129             =cut
130              
131             sub get ($@) {
132 1517     1517 1 4041373 my $self=shift;
133 1517         2122 my $backend=$self->{'backend'};
134            
135 1517 100 66     3456 my $object=ref($_[0]) && ref($_[0]) ne 'HASH' ? shift(@_) : undef;
136 1517         3742 my $args=get_args(\@_);
137              
138 1517         2137 my @c=map { $args->{$_} } @{$self->{'coords'}};
  2156         4659  
  1517         2685  
139 1517 50       3044 defined($c[0]) ||
140             throw XAO::E::Cache "get - no first coordinate ($args->{coords}->[0])";
141              
142             # Get method will return undef for non-existent. Or a reference to
143             # value (possibly an undef) when a value exists.
144             #
145 1517 100       4133 my $data_ref=$args->{'force_update'} ? undef : $backend->get(\@c);
146              
147 1517 100       3949 return $$data_ref if defined $data_ref;
148              
149 802 100       1177 my $data=&{$self->{'retrieve'}}($object ? ($object) : (),$args);
  802         1721  
150              
151 802         5999 $backend->put(\@c => \$data);
152              
153 802         2593 return $data;
154             }
155              
156             ###############################################################################
157              
158             =item new (%)
159              
160             Creates a new independent instance of a cache. When that instance is
161             destroyed all cache content is destroyed as well. Arguments are:
162              
163             =over
164              
165             =item backend
166              
167             Type of backend that will actuall keep values in cache.
168             Can be either a XAO object name or an object reference.
169              
170             Default is 'Cache::Memory' (XAO::DO::Cache::Memory).
171              
172             =item coords
173              
174             Coordinates of a data element in the cache -- reference to an array that
175             keeps names of arguments identifying a data element in the cache. The
176             order of elements in the list is significant -- first element is
177             mandatory, the rest is optional.
178              
179             A combination of all coordinates must uniquely identify a cached data
180             element among all others in the cache. For instance, if you create a
181             cache with customers, then 'customer_id' will most probably be your only
182             coordinate. But if to retrieve a data element you need element type and
183             id then your coordinates will be:
184              
185             coords => ['type', 'id']
186              
187             There is no default for coordinates.
188              
189             B: Coordinates are supposed to be text strings meeting isprint()
190             criteria.
191              
192             =item expire
193              
194             Expiration time for data elements in the cache. Default is no expiration
195             time.
196              
197             =item retrieve
198              
199             Reference to a method or a subroutine that will actually retrieve data
200             element when there is no element in the cache or cache element has
201             expired.
202              
203             The subroutine gets all parameters passed to cache's get() method.
204              
205             Cache does not perform any checks for correctness of result, so if for
206             some reason retrieval cannot be performed an error should be thrown
207             instead of returning undef or other indicator of failure.
208              
209             =item size
210              
211             Optional maximum size of the cache in Kbytes. If not specified then only
212             expiration time will be used as a criteria to throw a data element out
213             of cache.
214              
215             =item value_maxlength
216              
217             Maximum length of an individual value to be stored. Values longer than
218             this size may be ignored by the cache, but it is still safe to return
219             then from the retrieve() method. They MAY just not be cached.
220              
221             Primarily this is useful for memcached configuration to match what the
222             memcached server is going to reject anyway.
223              
224             =back
225              
226             If there is a current project and that project Config object holds a
227             /cache/config data then that data is used for default values, providing
228             a way to, for instance, change cache backend globally for all project
229             caches.
230              
231             The configuration is structured like this:
232              
233             cache => {
234             config => {
235             common => {
236             backend => 'Cache::Memcached',
237             },
238             foo_cache => {
239             backend => 'Cache::Memory',
240             size => 1_000_000,
241             },
242             },
243             },
244              
245             For a cache named foo_cache the backend would be 'Cache::Memory' and for
246             all other caches -- 'Cache::Memcached' in that case.
247              
248             =cut
249              
250             sub new ($%) {
251 9     9 1 301 my $proto=shift;
252 9         64 my $args=get_args(\@_);
253              
254             # Checking if there is a site configuration and some default
255             # parameters in it.
256             #
257             my $config=$args->{'sitename'}
258 9 50 66     69 ? XAO::Projects::get_project($args->{'sitename'})
259             : (XAO::Projects::get_current_project_name() && XAO::Projects::get_current_project());
260              
261 9 100 66     93 if($config && $config->can('get')) {
262             $args=merge_refs(
263             $config->get('/cache/config/common') || { },
264 8 100 50     406 ($args->{'name'} ? ($config->get('/cache/config/'.$args->{'name'})) : ()),
265             $args,
266             );
267             }
268              
269             # Backend -- can be an object reference or an object name
270             #
271 9   100     72 my $backend=$args->{'backend'} || 'Cache::Memory';
272             ### dprint "Created cache '",$args->{'name'},"', backend='$backend'";
273 9 50       87 $backend=XAO::Objects->new(objname => $backend) unless ref($backend);
274              
275             # Retrieve function must be a code reference
276             #
277 9   33     26 my $retrieve=$args->{'retrieve'} ||
278             throw XAO::E::Cache "new - no 'retrive' argument";
279 9 50       32 ref($retrieve) eq 'CODE' ||
280             throw XAO::E::Cache "new - 'retrive' must be a code reference";
281              
282             # Coords must be an array reference or a single scalar
283             #
284 9   0     23 my $coords=$args->{'coords'} || $args->{'coordinates'} ||
285             throw XAO::E::Cache "new - no 'coords' argument";
286              
287 9 50       26 $coords=[ $coords ] if !ref($coords);
288              
289 9 50       40 ref($coords) eq 'ARRAY' ||
290             throw XAO::E::Cache "new - 'coords' must be an array reference";
291              
292 9 50       29 (grep { $_ eq 'force_update' } @$coords) &&
  11         65  
293             throw XAO::E::Cache "new - cannot use 'force_update' as a coordinate";
294              
295             my $self={
296             name => $args->{'name'},
297             backend => $backend,
298             coords => $coords,
299             expire => $args->{'expire'} || 0,
300             retrieve => $retrieve,
301 9   100     100 size => ($args->{'size'} || 0)*1024,
      100        
302             };
303              
304             # Setting up back-end parameters
305             #
306 9         59 $backend->setup($args);
307              
308             # Old caches used to have 'exists' method, which is now obsolete.
309             # It requires at least a double key calculation, and in the case of
310             # memcached also a double network trip.
311             #
312 9 50       61 !$backend->can('exists') ||
313             throw XAO::E::Cache "new - backend '$backend' supports an obsolete 'exists' method, upgrade it";
314              
315             # Done, blessing
316             #
317 9   33     109 bless $self,ref($proto) || $proto;
318             }
319              
320             ###############################################################################
321             1;
322             __END__