Files
RSSHub/middleware/lru-cache.js
さくら 49de384d2e feat: add prettier and commit hooks (#144)
* feat: add prettier and commit hooks

* style(*): prettier format

* fix(eslint): parsing error: Unexpected token ...

* style(eslint): eslint --fix

* style(bilibili): fix `no-undef` and `no-unused-vars`

* style(jiandan): fix eqeqeq

* style(pixiv): disable `no-constant-condition` rule for eslint

* fix(pixiv): use template string

* style(toutiao): remove unused variables

* style(weibo): use `forEach` instead of `map`

The array has no return value, you should use `forEach` instead of `map`

* chore(hooks): add eslint hooks

* chore(hooks): switch from husky to yorkie

* style(zhihu): prettier format

* style(docs): fix vuepress custom container

https://vuepress.vuejs.org/guide/markdown.html#custom-containers

* chore(prettier): set a more reasonable `printWidth` value

* style(*): better prettier format
2018-05-17 10:52:29 +08:00

161 lines
3.9 KiB
JavaScript

// baed on https://github.com/coderhaoxin/koa-redis-cache
const pathToRegExp = require('path-to-regexp');
const readall = require('readall');
const crypto = require('crypto');
const lru = require('lru-cache');
module.exports = function(options = {}) {
const {
prefix = 'koa-cache:',
expire = 30 * 60, // 30 min
routes = ['(.*)'],
exclude = [],
passParam = '',
maxLength = Infinity,
ignoreQuery = false,
} = options;
const memoryCache = lru({
maxAge: expire * 1000,
max: maxLength,
});
return async function cache(ctx, next) {
const { url, path } = ctx.request;
const resolvedPrefix = typeof prefix === 'function' ? prefix.call(ctx, ctx) : prefix;
const key = resolvedPrefix + md5(ignoreQuery ? path : url);
const tkey = key + ':type';
let match = false;
let routeExpire = false;
for (let i = 0; i < routes.length; i++) {
let route = routes[i];
if (typeof routes[i] === 'object') {
route = routes[i].path;
routeExpire = routes[i].expire;
}
if (paired(route, path)) {
match = true;
break;
}
}
for (let j = 0; j < exclude.length; j++) {
if (paired(exclude[j], path)) {
match = false;
break;
}
}
if (!match || (passParam && ctx.request.query[passParam])) {
return await next();
}
let ok = false;
try {
ok = await getCache(ctx, key, tkey);
} catch (e) {
ok = false;
}
if (ok) {
return;
}
await next();
try {
const trueExpire = routeExpire || expire;
await setCache(ctx, key, tkey, trueExpire);
} catch (e) {} // eslint-disable-line no-empty
routeExpire = false;
};
/**
* getCache
*/
async function getCache(ctx, key, tkey) {
const value = memoryCache.get(key);
let type;
let ok = false;
if (value) {
ctx.response.status = 200;
type = memoryCache.get(tkey) || 'text/html';
// can happen if user specified return_buffers: true in redis options
if (Buffer.isBuffer(type)) {
type = type.toString();
}
ctx.response.set('X-Koa-Memory-Cache', 'true');
ctx.response.type = type;
try {
ctx.state.data = JSON.parse(value);
} catch (e) {
ctx.state.data = {};
}
ok = true;
}
return ok;
}
/**
* setCache
*/
async function setCache(ctx, key, tkey) {
ctx.state.data.lastBuildDate = new Date().toUTCString();
const body = JSON.stringify(ctx.state.data);
if (ctx.request.method !== 'GET' || !body) {
return;
}
if (Buffer.byteLength(body) > maxLength) {
return;
}
memoryCache.set(key, body);
await cacheType(ctx, tkey);
}
/**
* cacheType
*/
async function cacheType(ctx, tkey) {
const type = ctx.response.type;
if (type) {
memoryCache.set(tkey, type);
}
}
};
function paired(route, path) {
const options = {
sensitive: true,
strict: true,
};
return pathToRegExp(route, [], options).exec(path);
}
// eslint-disable-next-line no-unused-vars
function read(stream) {
return new Promise((resolve, reject) => {
readall(stream, (err, data) => {
if (err) {
reject(err);
} else {
resolve(data);
}
});
});
}
function md5(str) {
return crypto
.createHash('md5')
.update(str)
.digest('hex');
}