ferguson
v0.6.0
Published
An asset manager
Downloads
7
Readme
Ferguson is a highly-configurable asset manager for node.js with the following features
- Framework-agnostic compilation and compression
- Hot-reloading
- Cache-busting
- Asset bundles
- Inline assets
Overview
Install the library with npm
$ npm install ferguson
Setup a ferguson instance and bind it to an express application
var ferguson = require('ferguson');
var assetManager = ferguson('/path/to/assets', options);
assetManager.bind(app);
Your templates now have access to a view helper
{{ asset('favicon.ico') }}
<!--[if lt IE 9]>
{{ asset('js/ie8.js', { include: ['js/html5shiv.js', 'js/respond.js'] });
<![endif]-->
{{ asset('js/smallFile.js', { inline: true }) }}
{{ asset('css/styles.less') }}
which generates something like
<link rel="shortcut icon" href="/asset-a2029888991a8a83-favicon.ico" />
<!--[if lt IE 9]>
<script src="/js/asset-0ba08226c3bd0e46-ie8.js" type="text/javascript"></script>
<![endif]-->
<script type="text/javascript">var foo = "bar";</script>
<link rel="stylesheet" href="/css/asset-b5d5d67465f661c1-styles.css" />
Assets are compiled and compressed on demand. Compiled asset filenames only change when one of the included assets is modified, i.e. they're safe to cache forever.
Adding your own formats is easy
var stylus = require('stylus');
assetManager.registerCompiler('.styl', '.css', function (path, buffer, options, callback) {
stylus.render(buffer.toString(), callback);
});
Now you can call asset('css/foo.styl')
to render an asset with stylus.
Options
The following options are available when creating a new ferguson instance with ferguson(dir, options)
- hash (default:
md5
) - the hashing algorithm to use. - hashLength (default:
16
) - the maximum length of the hash in the filenames of compiled assets. - viewHelper (default:
asset
) - the name of the view helper. - servePrefix (default:
/
) - the path to serve assets from. - urlPrefix (default:
null
) - prefix asset URLs with this. - maxAge (default:
2419200000
- 4 weeks) - the Cache-Control max-age value (in milliseconds). - compress (default:
false
) - whether to compress assets. - hotReload (default:
false
) - whether to enable hot-reloading of assets. - wrapJavascript (default:
false
) - whether to wrap compiled JS in an IIFE. - separateBundles (default:
false
) - generate separate tags for each asset in a bundle. - html5 (default:
false
) - generate HTML5-compatible tags, e.g. omit the type attribute from a<script>
tag.
The following setup is recommended
var production = process.env.NODE_ENV === 'production';
var assetManager = ferguson('/path/to/assets', {
compress: production
, hotReload: !production
, separateBundles: !production
});
Asset Definitions
The following options are available when defining an asset with asset(file, options)
- include - one or more files that make up the asset bundle. Glob is supported.
- urlPrefix - prefix the asset URL with this. This overrides the library's
urlPrefix
option. - attributes - an object containing additional HTML attributes.
- dependencies - one or more files that ferguson should take into account when generating cache-busting hashes (see the Compilers section below for an explanation). Glob is supported.
- inline - whether to inline the asset instead of referencing an external resource.
Here's an example definition
{{ asset('foo.jpg', { attributes: { alt: 'Foo & bar' }, urlPrefix: 'http://example.com' }) }}
which generates
<img src="http://example.com/asset-d3b07384d113edec-foo.jpg" alt="Foo & bar" />
There are two variations of the view helper: one to output the asset's path and another to output the asset's full URL (if you've provided a urlPrefix
)
<script type="text/javascript" src="{{ asset.url('foo.js') }}"></script>
<script type="text/javascript" src="//example.com{{ asset.path('foo.js') }}"></script>
It's also possible to define assets when you create a ferguson instance
var assetManager = ferguson('/path/to/assets');
assetManager.asset('ie8.js', { include: ['html5shiv.js', 'respond.js'] });
Once defined, assets can be referenced by name
{{ asset('ie8.js') }}
Caching
Ferguson generates a cache-busting hash based on the contents of each included asset. This means that the compiled assets are safe to cache indefinitely. Any modifications to your assets will cause the hash (and filename) to change, forcing clients to re-download the asset. The library will automatically cleanup old compiled assets.
Ferguson will write the compiled assets to the static assets directory that you specify. This allows you to serve up raw and compiled assets from the same place. For example, it allows nginx users to use a try_files
to serve up all assets directly.
Ferguson keeps an index (filename, hash, mtime) of each file in your static assets directory. The index is persisted across restarts using a manifest file, .asset-manifest
. This reduces the amount of hashing required; assets are only hashed when a modification or new file is detected.
If your static assets live in a folder named static
then you might want to add the following two lines to your .gitignore
static/.asset-manifest
static/**/asset-*
For assets that don't change very often, e.g. favicons, you can omit the cache busting hash entirely by building your own asset URL
<link rel="shortcut icon" href="{{ asset.prefix }}favicon.ico" />
Compilers
Ferguson does not ship with any compilers since it's trivial to add your own
var less = require('less');
assetManager.registerCompiler('.less', '.css', function (path, buffer, options, callback) {
less.render(buffer.toString(), callback);
});
Since ferguson is framework-agnostic, there are some cases where you will need to manually specify an asset's dependencies. For example, less allows you to @import
a file and browserify allows you to require()
a file. In order to generate a correct cache-busting hash, ferguson needs to know about these imports, since the hash is based on the contents of the final compiled asset.
Let's say you have a file called style.less
which @import
's a variables.less
. You'll need to define your asset like so
{{ asset('style.less', { dependencies: ['variables.less'] }) }}
Ferguson will generate the cache-busting hash based on the contents of both style.less
and variables.less
. Without specifying the dependency, only updates to style.less
would cause a new hash to be generated.
A lazy way to specify dependencies is to use a glob pattern that matches everything
{{ asset('style.less', { dependencies: '**/*.less' }) }}
You can also define synchronous compressors by omitting the callback
var marked = require('marked');
assetManager.registerCompiler('.md', '.html', function (path, buffer, options) {
return marked(buffer.toString());
});
Compressors
Ferguson ships with a JS minifier (uglifyjs) and CSS compressor (clean-css) which are both enabled when the compress
option is true.
You can add additional compressors or override an existing one
var yui = require('yuicompressor');
assetManager.registerCompressor('.js', function (buffer, options, callback) {
yui.compress(buffer.toString(), options, callback);
});
Just like compiler definitions, you can define a synchronous compressor by omitting the callback.
Inline assets
You can generate inline assets using {{ asset(yourAsset, { inline: true }) }}
.
Ferguson will wrap Javascript in a <script>
tag, wrap CSS in a <style>
tag and base64-encode and inline images into an <img>
tag. It will output all other assets as-is.
You can add or override an inline formatter
assetManager.registerInlineFormat('.html', function (filename, buffer, options, attributes) {
return util.format('<script type="text/x-template">%s<script>', buffer);
})
Since the inline formatters have access to the binary contents of the asset, the following is possible
var mime = require('mime');
assetManager.registerInlineFormat(imageExt, function (filename, buffer, options, attributes) {
return util.format('<img src="data:%s;base64,%s" />',
mime.lookup(filename),
buffer.toString('base64'));
});
The following call
{{ asset('1x1.gif', { inline: true }) }}
would output the following
<img src="" />
Since asset inlining happens synchronously, you can only use the inline
option with synchronous compilers and compressors. Attempting to inline an asset that uses either an asynchronous compiler or compressor will fail with an error.
Tag Formats
Ferguson knows how to generate HTML tags for css
, js
, ico
, jpg
, gif
, png
, svg
and bmp
assets.
You can add and override tag formats if necessary
assetManager.registerTagFormat('.js', function (url, options, attributes) {
return util.format('<custom-tag src="%s" />', url);
})
Alternatively, you could use the asset.url()
view helper to output the asset URL directly
<custom-tag src="{{ asset.url('foo.js') }}" />
Events
The ferguson instance will emit the following events
assetManager.on('error', function (err) {
// an error occurred
})
assetManager.on('delete', function (file) {
// a stale version of an asset was removed
});
assetManager.on('change', function (file) {
// a change was detected in the specified file
});
Multi-process Environments
Ferguson doesn't know how to serve up compiled assets until they are defined. If you plan on using the cluster module, there's a chance that only one process could render a template which contains the asset definition. If another process in the cluster gets a request for the compiled asset and a cached copy doesn't exist, then it won't know what to do since only the process that defines the asset knows how to compile it.
The solution is to define your assets when you initialise the library. That way, all processes in the cluster will see the definition
var assetManager = ferguson('/path/to/assets');
assetManager.asset('ie8.js', { include: ['html5shiv.js', 'respond.js'] });
Developers
To debug all aspects of the library, run the process with the DEBUG
env set
$ node yourapp.js DEBUG=ferguson
The test suite can be run with
$ make test
To increase verbosity use V=1 make test
. To run a subset of tests use TEST=<pattern> make test
To run coverage analysis and view a HTML report
$ make coverage-html
To run the lint tool
$ make lint
Naming
The name (and logo) is derived from one of the worlds most successful football managers, Sir Alex Ferguson.
License (MIT)
Copyright (c) 2013 Sydney Stockholm [email protected]
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.