NAME

HTTP::Caching - The RFC 7234 compliant brains to do caching right

VERSION

Version 0.01 Alpha 01

SYNOPSIS

my $chi_cache = CHI->new(
    driver          => 'File',
    root_dir        => '/tmp/HTTP_Caching',
    file_extension  => '.cache',
    l1_cache        => {
        driver          => 'Memory',
        global          => 1,
        max_size        => 1024*1024
    }
);

my $ua = LWP::UserAgent->new();

my $http_caching = HTTP::Caching->new(
    cache         => $chi_cache,
    cache_type    => 'private',
    forwarder     => sub { return $ua->request(shift) }
);

my $rqst = HTTP::Request->new( GET => 'http://example.com' );

my $resp = $http_caching->make_request( $rqst );

DESCRIPTION

This module tries to provide caching for HTTP responses based on RFC 7234 Hypertext Transfer Protocol (HTTP/1.1): Caching.

Basicly it looks like the following steps below:

  • For a presented request, it will check with the cache if there is a suitable response available AND if it can be served or that it needs to be revalidated with an upstream server.

  • If there was no response available at all, or non were suitable, the (modified) request will simply be forwarded.

  • Depending on the response it gets back, it will do one of the following dependingon the response status code:

    200 OK

    it will update the cache and serve the response as is

    304 Not Modified

    the cached version is valid, update the cache with new header info and serve the cached response

    500 Server Error

    in general, this is an error, and pass that onto the caller, however, in some cases it could be fine to serve a (stale) cached response

The above is a over-simplified version of the RFC

METHODS

new

my $http_caching = HTTP::Caching->new(
    cache           => $chi_cache,
    cache_type      => 'private',
    cache_request   => 'max-age=86400, min-fresh=60',
    forwarder       => sub { return $ua->request(shift) }
);

Constructs a new HTTP::Caching object that knows how to find cached responses and will forward if needed.

The following attributes are used:

cache

A CHI compatible cache. To have most benifits from a cache, it could be stored on a files-system if you like to have it between processes. A L1 Cache is very useful because of the way HTTP Caching has to handle the responses. Since it is possible that there are multiple responses associated with a specific URI, it needs to inspect all variants and if needed, validate against the upstream server. After the response comes back, it then needs to reselct a stored response, based on the the recieved response. The L1-cache will only be used for the request and response headers, the actual response body or the payload will be stored in the primary cache, on filesystem.

See CHI for more information about creating a cache. The one shown in the synopsis is a good starter.

cache_type

This must either be 'private' or 'public'. For most LWP::UserAgents, it can be 'private' as it will probably not be shared with other processes on the same macine. If this module is being used at the serverside in a Plack::Middleware then the cache will be used by all other clients connecting to the server, and thus should be set to 'public'.

Responses to Authenticated request should not be held in public caches and also those responses that specifacally have their cache-control headerfield set to 'private'.

cache_control_request

A string that contains the Cache-control header-field settings that will be sent as default with the request. So you do not have to set those each time. See RFC 7234 Section 5.2.1 for the list of available cache-control directives.

cache_control_response

Like the above, but those will be set for each response. This is useful for server side caching. See RFC 7234 Section 5.2.2.

forwarder

This CodeRef must be a callback function that accepts a HTTP::Request and returns a HTTP::Response. Since this module does not know how to do a request it will use the forwarder. It will be used to sent of validation requests with If-None-Match and/or If-Modified-Since header-fields. Or if it does not have a stored response it will send the original full request (with the extra directives from cache_request).

Failing to return a HTTP::Response might cause the module to die or generate a response itself with status code 502 Bad Gateway.

make_request

This is the only public provided method and will take a HTTP::Request. Like described above, it might have to forward the (modified) request throug the CodeRef in the forwarder attribute.

It will return a HTTP::Response from cache or a new retrieved one. This might be a HTTP respons with a 500 Error message.

In other cases it might die and let the caller know what was wrong, or send another 5XX Error.