Browse Source

Add filtered response headers

master
Kornel Lesiński 9 years ago
committed by Kornel Lesiński
parent
commit
a20a6d47d6
  1. 12
      README.md
  2. 19
      index.js
  3. 2
      package.json
  4. 31
      test/responsetest.js

12
README.md

@ -52,17 +52,19 @@ If `options.shared` is true (default), then response is evaluated from perspecti
`options.cacheHeuristic` is a fraction of response's age that is used as a fallback cache duration. The default is 0.1 (10%), e.g. if a file hasn't been modified for 100 days, it'll be cached for 100*0.1 = 10 days.
### `satisfiesWithoutRevalidation(request)`
### `storable()`
If it returns `true`, then the given `request` matches the original response this cache policy has been created with, and the response can be reused without contacting the server.
Returns `true` if the response can be stored in a cache. If it's `false` then you MUST NOT store either the request or the response.
Note that you can't return the old response as-is. The old response must be cleaned up (remove hop-by-hop headers such as `TE` and `Connection`) and updated (add `Age` or rewrite `max-age`, etc.).
### `satisfiesWithoutRevalidation(request)`
If it returns `true`, then the given `request` matches the original response this cache policy has been created with, and the response can be reused without contacting the server. Note that the old response can't be returned without being updated, see `responseHeaders()`.
If it returns `false`, then the response may not be matching at all (e.g. it's for a different URL or method), or may require to be refreshed first.
### `storable()`
### `responseHeaders()`
Returns `true` if the response can be stored in a cache. If it's `false` then you MUST NOT store either the request or the response.
Returns updated, filtered set of response headers. Proxies MUST always remove hop-by-hop headers (such as `TE` and `Connection`) and update response age to avoid doubling cache time.
### `stale()`

19
index.js

@ -5,6 +5,8 @@ const statusCodeCacheableByDefault = [200, 203, 204, 206, 300, 301, 404, 405, 41
// This implementation does not understand partial responses (206)
const understoodStatuses = [200, 204, 301, 302, 303, 404, 410, 501];
const hopByHopHeaders = {'connection':true, 'keep-alive':true, 'proxy-authenticate':true, 'proxy-authorization':true, 'te':true, 'trailers':true, 'transfer-encoding':true, 'upgrade':true};
function parseCacheControl(header) {
const cc = {};
if (!header) return cc;
@ -133,6 +135,23 @@ CachePolicy.prototype = {
return true;
},
responseHeaders() {
const headers = {};
for(const name in this._resHeaders) {
if (hopByHopHeaders[name]) continue;
headers[name] = this._resHeaders[name];
}
// 9.1. Connection
if (this._resHeaders.connection) {
const tokens = this._resHeaders.connection.trim().split(/\s*,\s*/);
for(const name of tokens) {
delete headers[name];
}
}
headers.age = `${Math.round(this.age())}`;
return headers;
},
/**
* Value of the Date response header or current time if Date was demed invalid
* @return timestamp

2
package.json

@ -1,6 +1,6 @@
{
"name": "http-cache-semantics",
"version": "3.0.0",
"version": "3.1.0",
"description": "Parses Cache-Control headers and friends",
"main": "index.js",
"repository": "https://github.com/pornel/http-cache-semantics.git",

31
test/responsetest.js

@ -247,4 +247,35 @@ describe('Response headers', function() {
assert(!cache.stale());
assert.equal(333, cache.maxAge());
});
it('remove hop headers', function() {
let now = 10000;
class TimeTravellingPolicy extends CachePolicy {
now() {
return now;
}
}
const res = {headers:{
'te': 'deflate',
'date': 'now',
'custom': 'header',
'oompa': 'lumpa',
'connection': 'close, oompa, header',
'age': '10',
'cache-control': 'public, max-age=333',
}};
const cache = new TimeTravellingPolicy(req, res);
now += 1005;
const h = cache.responseHeaders();
assert(!h.connection);
assert(!h.te);
assert(!h.oompa);
assert.equal(h['cache-control'], 'public, max-age=333');
assert.equal(h.date, 'now', "date must stay the same for expires, age, etc");
assert.equal(h.custom, 'header');
assert.equal(h.age, '11');
assert.equal(res.headers.age, '10');
});
});

Loading…
Cancel
Save