Home | Wiki | OI 1.x Docs | OI 2.x Docs |
OpenInteract2::Manual::Caching - Storing generated data for later reuse
Caching can not only provide a dramatic speedup to certain types of content, it can mean the difference between scaling when you hit the big time and falling on your face. Even much highly dynamic content can be cached.
There are two different levels of caching in OpenInteract:
Content: Once a component has been generated with a particular set of parameters, we don't want to regenerate it again.
Templates: Once a template has been parsed we don't want to repeat those actions again. This is fairly minor and should be handled by whatever templating system you're using.
Content caching is still fairly young in OpenInteract, and it's not appropriate (or useful) for all purposes. It's best when used on pages that contain a lot of data or require a good deal of processing.
The following keys in your server configuration are used in caching:
cache.use
: If set to true, content caching is enabled; if
set to false, it is disabled.
cache.cleanup
: If true we delete and recreate the cache
directory every time the server starts up. This is recommended.
cache.class
: Cache implementation to use. Currently only
OpenInteract2::Cache::File is supported.
cache.directory
: The directory where we put the cached
content. This is normally $WEBSITE_DIR/cache/content
.
cache.default_expire
: number of seconds used for the
cached content expiration. You can override this on a case-by-case
basis.
cache.max_size
: Max size the cache should be allowed to
grow, in bytes.
cache.directory_depth
: If you find that retrieving cached
content is slow because of the number of items in a particular
directory, increase this number.
There is really only one requirement for enabling caching. Well, two. The first (or zeroth) is that you must subclass OpenInteract2::Action. You almost certainly already do this, so it's not much a requirement.
The real requirement: caching must be configured in your action using
the parameter cache_param
and, optionally, cache_expire
.
We need to ensure that we can uniquely identify each request for content. Generally this is done by associating a unique key with each different set of dynamic content. We create this unique key by combining the task with a number of action parameters and their values.
For instance, in the 'news' action you have a 'latest' task. This
brings up the latest n news items. You'd want the cached data to
reflect this number so you set it in the cache_param
section of the
action. Here's an example:
[news cache_param] latest = num_items
Here we've told the caching system to associate content from the 'latest' task with the variable 'num_items'. So when we get a request:
/news/latest/?num_items=10
We'll create a unique cache key like the following:
latest;num_items=10
and use this to get and set the cached content. (The actual key may not look like this, it's just an example.)
You can also associate multiple parameters with a task. For instance, say we also used a variable 'country' to further specify which latest news items are retrieved:
[news cache_param] latest = num_items latest = country
And a corresponding key might look like:
latest;country=USA;num_items=10
Now the cached content depends on 'num_items' and 'country'. All of these requests would cache their content to different places and remain totally separate from one another:
/news/latest/?num_items=10 /news/latest/?num_items=15 /news/latest/?num_items=10&country=USA /news/latest/?num_items=10&country=France /news/latest/?num_items=15&country=USA
You can also control when cached content expires using the
cache_expire
section of the action:
[news cache_expire] latest = 600 display = 3600 home = 600
The values are in seconds. Here we've said the content generated by
the 'latest' and 'home' tasks should be cached for 10 minutes, and the
'display' task for one hour. You can manually delete cache entries
using the clear_cache()
action method.
Notice that we included an extra task here, 'home'. It has no
dependencies on any parameters so we don't need to specify any in
cache_param
.
If you don't list your task in cache_expire
content generated by
it will not be cached.
The execute
method of
OpenInteract2::Action takes care of this for
you. The only aspect you need to be aware of is setting up your
parameters for the cache key. Before execute()
checks the cache it
makes a call to initialize_cache_params()
. This allows you the
chance to return parameter values that will determine the cache
key. It's useful to specify values that might be the combination of
several request parameters (like a date) or a parameter that doesn't
normally vary by request (like the day of the week).
Much of the time, however, you won't need to set any additional parameters. Normally you'll depend on GET/POST parameters passed from the user. We already have access to those through the OpenInteract2::Request object, so we go ahead and use them if necessary.
Additionally there are a couple of implicit parameters you can use to segment your cache entries:
user_id: ID of the current user
theme_id: ID of the current theme
If you specify any of these a reasonable default is supplied.
So to find a value we check, in order and taking the first defined value:
Return value from initialize_cache_params()
Value of parameter in action
Value of parameter in request
Default value if an implicit parameter.
For instance, in the above example we specified the parameters for the 'display' task as 'news_id'. If this were passed in via the request we'd don't have to change our 'display' task at all to use caching. Even if the task is called programmatically by another action we won't have to change it since the 'news_id' can be set via the action parameter.
Ah, but what happens if someone passes in a news object directly?
sub _calling_display_task { my ( $self ) = @_; my $fakenews = $self->_create_news_object( type => 'onion' ); my $display_action = CTX->lookup_action( 'news' ); return $display_action->execute({ task => 'display', news => $fakenews }); }
Now our automatic parameter discovery won't work. This is where the
initialize_cache_params()
comes in handy. In our 'news' action we
can have:
sub initialize_cache_params { my ( $self ) = @_; my %params = (); if ( my $news = $self->param( 'news' ) ) { $params{news_id} = $news->id; } return \%params; }
And everything will work!
You have the option of clearing the cache whenever you manipulate data. For instance, if you edit the title of a news story you do not want the old title to appear in the story listing. And if you delete a story and mark it as inactive because it's inappropriate, you do not want it in your headline listing.
So whenever you modify data, it's normally best to call
clear_cache()
. This method is inherited from
OpenInteract2::Action like the others. Here's
an example:
sub update { my ( $self ) = @_; my $request = CTX->request; my $thingy_id = $self->param( 'thingy_id' ) || $request->param( 'thingy_id' ); my $thingy = eval { CTX->lookup_object( 'thingy' )->fetch( $thingy_id ); }; if ( $@ ) { ... } $thingy->{foo} = $request->param( 'foo' ); eval { $thingy->save }; if ( $@ ) { $self->param_add( error_msg => "Cannot save thingy: $@" ); return $self->execute({ task => 'display_form', thingy => $thingy }); } else { $self->clear_cache(); $self->param_add({ status_msg => "Thingy updated ok" }); return $self->execute({ task => 'list' }); } }
So when the 'list' method is called after a successful update()
on
the object, the previously cached content will have been deleted and
the content will be regenerated anew.
If you're an admin user you frequently see functionality that normal users do not see: Edit or Remove links next to an object, etc. You do not want to cache this content, since users shouldn't see this information. (Normal users modifying the object shouldn't be an issue, since security should take care of it.)
As a result, any user defined as an administrator will not view or save cached content. "Defined as an administrator" means that a call to the following will return true:
my $is_admin = CTX->request->auth_is_admin;
This section discusses what the preferred templating engine (Template Toolkit) does and how it's handled in OpenInteract. Your engine may handle it differently.
Instead of parsing a template every time you request it, the Template Toolkit (TT) will translate the template to Perl code and, if allowed, cache it in memory. Keeping templates in memory will make your website much faster.
TT will also save your compiled template to the filesystem. This is useful for successive starts of your website -- if the template if found in the compile directory TT doesn't need to parse it again, even though you've stopped and restarted your server since it was first read.
The following keys from your server configuration control caching and compiling:
content_generator.TT.cache_size
: This is the main parameter, describing
how many cached templates TT will hold in memory. The only restriction
on a high value is your memory, so experiment with as high a number as
possible.
If you set this to 0 then caching will be disabled. This is useful when you're doing debugging on your site, but it can make things noticably slower if you have lots of requests, where 'lots' means 'more than a handful'.
dir.cache_tt
: The directory where we store the compiled templates
in the filesystem. This is normally $WEBSITE_DIR/cache/tt
content_generator.TT.cache_expire
: Sets the expiration (in seconds)
for how long the templates remain cached in memory before they're
reparsed.
content_generator.TT.compile_cleanup
: If set to a true value, we
clean out the template compile directory when the Apache server starts
up.
content_generator.TT.compile_ext
: Extension of the file created when the
template is compiled to the filesystem.
That's it! You can monitor the process of template caching by setting
the OI2.TEMPLATE
logging key to 'DEBUG':
Old value: log4perl.logger.OI2.TEMPLATE = WARN New value: log4perl.logger.OI2.TEMPLATE = DEBUG
Be warned: this produces a prodigious amount of messages.
Copyright (c) 2002-2003 Chris Winters. All rights reserved.
Chris Winters <chris@cwinters.com>
Generated from the OpenInteract 1.99_04 source.