Warning
The AppCache
is deprecated and while I loved it and it worked really well for what it was, using it now isn't a great idea.
rails_appcache
This is a simple set of helpers for using the appcache from a Rails application
Why?
First, if you don't know what the appcache is, or you don't have (at least) a high-level understanding of how it works, read A high-level overview of the appcache.
How does rail_appcache help?
This gem serves appcache manifests from app/views/rails_appcache/manifests/<name>.appcache.erb
. You are (mostly) responsible for maintainig these files, but there is a generator for building them, and helpers for including things from the asset pipeline.
This gem provides the following:
- Automatically serve manifest files with the corret
text/appcache
mime type - Generators for building new manifests, with helpers for versioning (expiring) them
- A mountable engine for serving manifests
- Helper methods for exploding asset paths in development:
javascript_cache_path
stylesheet_cache_path
asset_cache_path
Example
For a small example of usage, checkout http://github.com/meagar/house-clock, a simple count-down clock. It's manifest lists the JavaScript and CSS, the background image (house.jpg) and some external CDN-hosted assets for Bootstrap.
It lists NETWORK: *
so that vairous tracking scripts from New Relic and Heroku can be injected into the page and continue to work while online.
Installation
Install the gem with gem install rails_appcache
or
- Add the gem to your gemfile:
gem 'rails_appcache'
- Bundle:
$ bundle
- Restart your server.
Usage
- Mount the engine in
config/routes.rb
:
mount RailsAppcache::Engine => '/'
- Generate a new manifest called "application":
$ rails g appcache_manifest application
create app/views/rails_appcache/manifests/application.appcache.erb
This will produce a file containing the following:
CACHE MANIFEST
# auto-expire appcache in dev
# <%= appcache_version_string %>
CACHE:
# by default, only the root path is cached
/
# Cache additional paths by using path helpers:
# <%= posts_path %>
# Cache our JS/CSS bundles
<%= stylesheet_cache_path 'application' %>
<%= javascript_cache_path 'application' %>
# Cache additional assets by using the asset_cache_path, or any of the Rails built-in asset pipeline helpers
# <%= asset_cache_path 'logo.png' %>
# or
# <%= asset_path 'logo.png' %>
# Make everything else accessible
# WITHOUT THIS LINE, your browser will 404 for anything not explicitly listed under CACHE:
NETWORK:
*
- Add any additional resources to cache to the manifest
Typically using path helpers. Your manifest might need to be quite long, but remember that you are limited to (roughly?) 5MB in most browsers.
- Include the manifest in your layout
Your <html>
tag should have a manifest attribute, containing the URL for your manifest. You can use the helper method appcache_manifest_path
to generate the correct URL:
# ERB:
<html manifest="<%= appcache_manifest_path('application') %>">
# Haml
%html{manifest: appcache_manifest_path('application')}
# Slim:
html manifest=appcache_manifest_path('application')
- Configure versions/expiration
There are sane defaults out-of-the-box, but you might want something specific to your usecase.
Similar to the existing configuration setting Rails provides, Rails.application.config.assets.version = '1.0'
, RailsAppcache provides a configuration value you should set/increment to expire your appcache manifests. This is actually pretty important; see appcache_version_string
below.
One typical solution is to use the Git commit ID which is currently deployed. If you're using Capistrano, this is availabe in a file called REVISION
in the root of your project, if you're using Git as a deploy tool, you can use Git directly:
# config/application.rb
# Capistrano
RailsAppcache.config.version = File.read(Rails.root.join('REVISION'))
# Or, pure Git
RailsAppcache.config.version = `git rev-parse HEAD`.strip
javascript_cache_path
and stylesheet_cache_path
These are helpers for making asset-path-served resources work inside appcache manifests.
During development, the asset pipeline "explodes" your manifests into many included JavaScript files, while during production the asset pipeline finger prints them.
Basically, given this JavaScript manifest:
// app/assets/application.js
//= require jquery
//= require underscore
//= require posts
in development (with config.assets.debug = true
in config/environments/development.rb), adding this file...
<%= javascript_include_tag 'application' %>
... will actually add four javascript tags:
<script src="/assets/application.js?body=1"></script>
<script src="/assets/jquery.js?body=1"></script>
<script src="/assets/underscore.js?body=1"></script>
<script src="/posts.js?body=1"></script>
while in production it will be a single tag:
<script src="/assets/application-1ef0203b2aab72467dc6261a2216e326.js""></script>
The obvious problem here is that, when working with the appcache, you need to list all of your resources, or they're going to 404 once your page is cached.
To help, you can use javascript_cache_path
and stylesheet_cache_path
in your appcache manifests, as direct analogs for javascript_include_tag
and stylesheet_link_tag
respectively:
CACHE MANIFEST
CACHE:
<%= javascript_cache_path 'application' %>
In development, this file will be served with the paths exploded:
CACHE MANIFEST
CACHE:
/assets/application.js?body=1
/assets/jquery.js?body=1
/assets/underscore.js?body=1
/posts.js?body=1
In production, you'll get the correctly fingerprinted URL.
appcache_version_string
Every request for a manifest contains a unique version string, such as /application-1.0.appcache
. This allows us to tell whether the browser is making a request for the current version of a manifest, and if not, serve a 404, causing the obsolete appcache to be thrown out. The closes the loop where a browser could have an old version of a cached file, request the new version, but fail to finish downloading the entire manifest before the user navigates. In this case (user navigation interrupting the manifest download), the browser would retain the old manifest, and continue to serve the legacy page if we weren't explicitly issuing a 404 and obsoleteing it.
This also gives you a chance in JavaScript to listen for the obsoletion event, and potentially redirect the user to a dedicated update page.
One common technique for expiring your appcache is to add a simple version string in a comment below the CACHE MANIFEST
line. This helper outputs a continually changing string in development (the unix timestamp) and the current revision in production, meaning that each time you deploy your app (assuming you're using Capistrano) you will get a new appcache manifest and your clients will re-download your app.
Doesn't this cause all of your assets to be expired on every single deploy?
Yes, but that's how the appcache works. You cannot expire individual things in the cache; if the manifest changes, every single file cached by that manifest is re-downloaded, so it really doesn't matter if we get much more clever in expiring the manifest. Typically, if your app is heavy on JavaScript, you'll probably need ot fully expire the appcache on each deploy.