UNPKG

hastily

Version:

express middleware to simulate fastly cdn

86 lines (58 loc) 7.35 kB
# hastily [![npm downloads](https://img.shields.io/npm/v/hastily)](https://npmjs.com/package/hastily) [![CircleCI](https://img.shields.io/circleci/build/github/zetlen/hastily?token=%5B%21%5BCircleCI%5D%28https%3A%2F%2Fcircleci.com%2Fgh%2Fzetlen%2Fhastily.svg%3Fstyle%3Dsvg%26circle-token%3D53542f77b0d4d5418e830f5a108ae6eb64cf5d87%29%5D%28https%3A%2F%2Fcircleci.com%2Fgh%2Fzetlen%2Fhastily%29&logo=circleci)](https://circleci.com/gh/zetlen/hastily) [![codecov](https://img.shields.io/codecov/c/github/zetlen/hastily?token=b42598abc62c411cb485401eeb1a03a3&logo=)](https://codecov.io/gh/zetlen/hastily) ![snyk](https://img.shields.io/snyk/vulnerabilities/npm/hastily) ![peer dependency](https://img.shields.io/npm/dependency-version/hastily/peer/sharp) [![npm downloads](https://img.shields.io/npm/dt/hastily?logo=npm)](https://npmjs.com/package/hastily) ### Drop-in compatible Express middleware to replicate the [Fastly Image Optimization API][fastly-api] ## Usage The middleware works a lot like [compression](https://npmjs.org/packages/compression), a standard Express middleware which transparently compresses any text-based response body that emits from something in the middleware chain. So `hastily` works as an add-on to an Express server or middleware that is already serving images. ```js import express, { static } from 'express'; import { imageopto } from 'hastily'; const app = express(); app.use('/images', imageopto(), static('/www/images')); app.listen(8000); ``` You now have an app which can serve any image from `/www/images`, and optimize it with URL parameters from the [Fastly Image Optimization API][fastly-api]. ## Behavior ### Request Filtering Hastily is meant to be registered as a middleware within an app that may be handling non-image requests as well as image requests. By default, it will only attempt to transform responses that appear to be uncompressed images. Hastily verifies this in several steps. 1. **By URL:** The default `filter` function for the imageopto middleware is `hastily.hasSupportedExtension(req: Request)` It. checks a request's URL path for an extension that indicates a Sharp-supported file type. It reads the supported extensions from Sharp itself. If a file does not have an image-file extension, it will do nothing. **Supply an alternative function as the `filter` property of imageopto options to override this behavior.** ```js import { imageopto, hasSupportedExtension } from 'hastily'; // Don't require an extension. Require a certain base directory and query param. imageopto({ filter: (req) => req.path.startsWith('/media/') && 'optimize' in req.query, }); // Require an extension, OR an 'imageServer' path. hastily.imageopto({ filter: (req) => hasSupportedExtension(req) || req.path === '/imageServer', }); ``` 1. **By method and status:** Hastily will not transform any response unless the request was a `GET` and the response code is in the `2xx` range. 1. **By Cache-control:** If the response has a `Cache-Control` header containing `no-transform`, Hastily will respect that. 1. **By dedupe:** Both Hastily and Fastly (which Hastily emulates) set telltale headers on responses after processing them. If a `fastly-io-info` header is present, OR an `X-Optimized: hastily` header is present, Hastily will assume a transform has already occurred and won't attempt another. **This behavior can be overridden by `options.force`**, but you shouldn't do that. 1. **By content type:** Hastily will try to get the `Content-Type` header of the response. If it cannot detect a content type, OR if Sharp does not support the content type, Hastily will not attempt to transform. 1. **By content encoding:** Hastily will not attempt to decompress gzipped responses. **Images shouldn't be gzipped** (or deflated, or brotli'd, or what have you) and it's a mistake to configure a server to do so by default. The available gzip algorithms are designed for text data, and they don't compress binary data hardly at all. If your server is gzipping images before Hastily handles them, **fix this configuration** for the highest speeds. ### Debugging The Hastily middleware logs debugs, warnings and errors to standard error, which is what web server stacks expect. When `NODE_ENV` is `production`, only warnings will be logged, as parseable JSON lines. When `NODE_ENV` is _not_ `production`, Hastily respects the Node convention of path syntax in the `DEBUG` environment variable to determine log level. `DEBUG=hastily:*` will show all debug data in a pretty format. You can limit logging to subject areas by using paths like `DEBUG=hastily:request,hastily:params,hastily:splice` instead. ### [Full API doc at zetlen.github.io/hastily](https://zetlen.github.io/hastily) ## TODO - [x] implement resize and crop mappers - [x] throw on unsupported - [x] implement enable and disable for upscaling in resize - [x] implement format, auto=webp, and quality params in post-manip phase - [x] add unit tests - [ ] add image-diff automated testing - [ ] implement sharpen, mapping [amt, radius, threshold] to libvips sharpen params - [ ] implement brightness, contrast, saturation by figuring out percentage to multiplier mapping - [ ] use image.metadata() to implement relative and context-based methods - [ ] add header-based methods - [ ] montage - [ ] overlay [fastly-api]: https://docs.fastly.com/api/imageopto/