mirror of
https://github.com/DIYgod/RSSHub.git
synced 2025-12-15 10:15:23 +08:00
app: fix redis cache
This commit is contained in:
@@ -3,6 +3,12 @@ module.exports = {
|
|||||||
cacheType: process.env.CACHE_TYPE || 'memory', // support memory and redis, set empty to disable cache
|
cacheType: process.env.CACHE_TYPE || 'memory', // support memory and redis, set empty to disable cache
|
||||||
cacheExpire: process.env.CACHE_EXPIRE || 5 * 60,
|
cacheExpire: process.env.CACHE_EXPIRE || 5 * 60,
|
||||||
ua: process.env.UA || 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/65.0.3325.181 Safari/537.36',
|
ua: process.env.UA || 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/65.0.3325.181 Safari/537.36',
|
||||||
|
redis: {
|
||||||
|
url: process.env.REDIS_URL || 'redis://localhost:6379/',
|
||||||
|
options: { // 支持这些参数 https://github.com/NodeRedis/node_redis#options-object-properties
|
||||||
|
password: process.env.REDIS_PASSWORD || null,
|
||||||
|
}
|
||||||
|
},
|
||||||
pixiv: {
|
pixiv: {
|
||||||
client_id: 'MOBrBDS8blbauoSck0ZfDbtuzpyT',
|
client_id: 'MOBrBDS8blbauoSck0ZfDbtuzpyT',
|
||||||
client_secret: 'lsACyCD94FhDUtGTXi3QzcFE2uU1hqtDaKeqrdwj',
|
client_secret: 'lsACyCD94FhDUtGTXi3QzcFE2uU1hqtDaKeqrdwj',
|
||||||
|
|||||||
10
index.js
10
index.js
@@ -6,8 +6,8 @@ const config = require('./config');
|
|||||||
const onerror = require('./middleware/onerror');
|
const onerror = require('./middleware/onerror');
|
||||||
const header = require('./middleware/header.js');
|
const header = require('./middleware/header.js');
|
||||||
const utf8 = require('./middleware/utf8');
|
const utf8 = require('./middleware/utf8');
|
||||||
const memoryCache = require('./middleware/cache.js');
|
const memoryCache = require('./middleware/lru-cache.js');
|
||||||
const redisCache = require('koa-redis-cache');
|
const redisCache = require('./middleware/redis-cache.js');
|
||||||
const filter = require('./middleware/filter.js');
|
const filter = require('./middleware/filter.js');
|
||||||
const template = require('./middleware/template.js');
|
const template = require('./middleware/template.js');
|
||||||
const favicon = require('koa-favicon');
|
const favicon = require('koa-favicon');
|
||||||
@@ -52,8 +52,12 @@ if (config.cacheType === 'memory') {
|
|||||||
app.use(
|
app.use(
|
||||||
redisCache({
|
redisCache({
|
||||||
expire: config.cacheExpire,
|
expire: config.cacheExpire,
|
||||||
|
ignoreQuery: true,
|
||||||
onerror: (e) => {
|
onerror: (e) => {
|
||||||
logger.error('cache error', e);
|
logger.error('Redis error: ', e);
|
||||||
|
},
|
||||||
|
onconnect: () => {
|
||||||
|
logger.info('Redis connect.');
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
// transform from https://github.com/coderhaoxin/koa-redis-cache
|
// baed on https://github.com/coderhaoxin/koa-redis-cache
|
||||||
|
|
||||||
const pathToRegExp = require('path-to-regexp');
|
const pathToRegExp = require('path-to-regexp');
|
||||||
const readall = require('readall');
|
const readall = require('readall');
|
||||||
177
middleware/redis-cache.js
Normal file
177
middleware/redis-cache.js
Normal file
@@ -0,0 +1,177 @@
|
|||||||
|
// baed on https://github.com/coderhaoxin/koa-redis-cache
|
||||||
|
|
||||||
|
const pathToRegExp = require('path-to-regexp');
|
||||||
|
const wrapper = require('co-redis');
|
||||||
|
const readall = require('readall');
|
||||||
|
const crypto = require('crypto');
|
||||||
|
const Redis = require('redis');
|
||||||
|
|
||||||
|
module.exports = function (options = {}) {
|
||||||
|
let redisAvailable = false;
|
||||||
|
|
||||||
|
const {
|
||||||
|
prefix = 'koa-redis-cache:',
|
||||||
|
expire = 30 * 60, // 30 min
|
||||||
|
routes = ['(.*)'],
|
||||||
|
exclude = [],
|
||||||
|
passParam = '',
|
||||||
|
maxLength = Infinity,
|
||||||
|
ignoreQuery = false,
|
||||||
|
onerror = function () { },
|
||||||
|
onconnect = function () {},
|
||||||
|
} = options;
|
||||||
|
|
||||||
|
const {
|
||||||
|
host:redisHost = 'localhost',
|
||||||
|
port:redisPort = 6379,
|
||||||
|
url:redisUrl = `redis://${redisHost}:${redisPort}/`,
|
||||||
|
options:redisOptions = {}
|
||||||
|
} = options.redis || {};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* redisClient
|
||||||
|
*/
|
||||||
|
const redisClient = wrapper(Redis.createClient(redisUrl, redisOptions));
|
||||||
|
redisClient.on('error', (error) => {
|
||||||
|
redisAvailable = false;
|
||||||
|
onerror(error);
|
||||||
|
});
|
||||||
|
redisClient.on('end', () => {
|
||||||
|
redisAvailable = false;
|
||||||
|
});
|
||||||
|
redisClient.on('connect', () => {
|
||||||
|
redisAvailable = true;
|
||||||
|
onconnect();
|
||||||
|
});
|
||||||
|
|
||||||
|
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 (!redisAvailable || !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) { }
|
||||||
|
routeExpire = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* getCache
|
||||||
|
*/
|
||||||
|
async function getCache (ctx, key, tkey) {
|
||||||
|
const value = await redisClient.get(key);
|
||||||
|
let type;
|
||||||
|
let ok = false;
|
||||||
|
|
||||||
|
if (value) {
|
||||||
|
ctx.response.status = 200;
|
||||||
|
type = await redisClient.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-Redis-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, expire) {
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
await redisClient.setex(key, expire, body);
|
||||||
|
|
||||||
|
await cacheType(ctx, tkey, expire);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* cacheType
|
||||||
|
*/
|
||||||
|
async function cacheType (ctx, tkey, expire) {
|
||||||
|
const type = ctx.response.type;
|
||||||
|
if (type) {
|
||||||
|
await redisClient.setex(tkey, expire, type);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function paired (route, path) {
|
||||||
|
const options = {
|
||||||
|
sensitive: true,
|
||||||
|
strict: true,
|
||||||
|
};
|
||||||
|
|
||||||
|
return pathToRegExp(route, [], options).exec(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
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');
|
||||||
|
}
|
||||||
@@ -28,6 +28,7 @@
|
|||||||
"art-template": "4.12.2",
|
"art-template": "4.12.2",
|
||||||
"axios": "0.18.0",
|
"axios": "0.18.0",
|
||||||
"cheerio": "1.0.0-rc.2",
|
"cheerio": "1.0.0-rc.2",
|
||||||
|
"co-redis": "2.1.1",
|
||||||
"crypto": "1.0.1",
|
"crypto": "1.0.1",
|
||||||
"eslint": "4.19.1",
|
"eslint": "4.19.1",
|
||||||
"form-data": "^2.3.2",
|
"form-data": "^2.3.2",
|
||||||
@@ -35,11 +36,11 @@
|
|||||||
"json-bigint": "0.2.3",
|
"json-bigint": "0.2.3",
|
||||||
"koa": "2.5.1",
|
"koa": "2.5.1",
|
||||||
"koa-favicon": "2.0.1",
|
"koa-favicon": "2.0.1",
|
||||||
"koa-redis-cache": "3.0.0",
|
|
||||||
"koa-router": "7.4.0",
|
"koa-router": "7.4.0",
|
||||||
"lru-cache": "4.1.3",
|
"lru-cache": "4.1.3",
|
||||||
"path-to-regexp": "2.2.1",
|
"path-to-regexp": "2.2.1",
|
||||||
"readall": "1.0.0",
|
"readall": "1.0.0",
|
||||||
|
"redis": "2.8.0",
|
||||||
"twit": "2.2.9",
|
"twit": "2.2.9",
|
||||||
"winston": "3.0.0-rc3"
|
"winston": "3.0.0-rc3"
|
||||||
},
|
},
|
||||||
|
|||||||
25
yarn.lock
25
yarn.lock
@@ -807,7 +807,7 @@ clone@^1.0.2:
|
|||||||
version "1.0.4"
|
version "1.0.4"
|
||||||
resolved "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz#da309cc263df15994c688ca902179ca3c7cd7c7e"
|
resolved "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz#da309cc263df15994c688ca902179ca3c7cd7c7e"
|
||||||
|
|
||||||
co-redis@2:
|
co-redis@2.1.1:
|
||||||
version "2.1.1"
|
version "2.1.1"
|
||||||
resolved "https://registry.npmjs.org/co-redis/-/co-redis-2.1.1.tgz#8bc3717977c9ee5bfb6b28f39ae01bcf724d2dde"
|
resolved "https://registry.npmjs.org/co-redis/-/co-redis-2.1.1.tgz#8bc3717977c9ee5bfb6b28f39ae01bcf724d2dde"
|
||||||
dependencies:
|
dependencies:
|
||||||
@@ -2980,15 +2980,6 @@ koa-mount@^3.0.0:
|
|||||||
debug "^2.6.1"
|
debug "^2.6.1"
|
||||||
koa-compose "^3.2.1"
|
koa-compose "^3.2.1"
|
||||||
|
|
||||||
koa-redis-cache@3.0.0:
|
|
||||||
version "3.0.0"
|
|
||||||
resolved "https://registry.npmjs.org/koa-redis-cache/-/koa-redis-cache-3.0.0.tgz#e4c33ebc9d9c08fd50a669bca4573fabb4b2c584"
|
|
||||||
dependencies:
|
|
||||||
co-redis "2"
|
|
||||||
path-to-regexp "1"
|
|
||||||
readall "1"
|
|
||||||
redis "2"
|
|
||||||
|
|
||||||
koa-router@7.4.0:
|
koa-router@7.4.0:
|
||||||
version "7.4.0"
|
version "7.4.0"
|
||||||
resolved "https://registry.npmjs.org/koa-router/-/koa-router-7.4.0.tgz#aee1f7adc02d5cb31d7d67465c9eacc825e8c5e0"
|
resolved "https://registry.npmjs.org/koa-router/-/koa-router-7.4.0.tgz#aee1f7adc02d5cb31d7d67465c9eacc825e8c5e0"
|
||||||
@@ -3960,16 +3951,16 @@ path-parse@^1.0.5:
|
|||||||
version "1.0.5"
|
version "1.0.5"
|
||||||
resolved "https://registry.npmjs.org/path-parse/-/path-parse-1.0.5.tgz#3c1adf871ea9cd6c9431b6ea2bd74a0ff055c4c1"
|
resolved "https://registry.npmjs.org/path-parse/-/path-parse-1.0.5.tgz#3c1adf871ea9cd6c9431b6ea2bd74a0ff055c4c1"
|
||||||
|
|
||||||
path-to-regexp@1, path-to-regexp@^1.1.1:
|
path-to-regexp@2.2.1:
|
||||||
|
version "2.2.1"
|
||||||
|
resolved "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-2.2.1.tgz#90b617025a16381a879bc82a38d4e8bdeb2bcf45"
|
||||||
|
|
||||||
|
path-to-regexp@^1.1.1:
|
||||||
version "1.7.0"
|
version "1.7.0"
|
||||||
resolved "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-1.7.0.tgz#59fde0f435badacba103a84e9d3bc64e96b9937d"
|
resolved "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-1.7.0.tgz#59fde0f435badacba103a84e9d3bc64e96b9937d"
|
||||||
dependencies:
|
dependencies:
|
||||||
isarray "0.0.1"
|
isarray "0.0.1"
|
||||||
|
|
||||||
path-to-regexp@2.2.1:
|
|
||||||
version "2.2.1"
|
|
||||||
resolved "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-2.2.1.tgz#90b617025a16381a879bc82a38d4e8bdeb2bcf45"
|
|
||||||
|
|
||||||
path-type@^3.0.0:
|
path-type@^3.0.0:
|
||||||
version "3.0.0"
|
version "3.0.0"
|
||||||
resolved "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz#cef31dc8e0a1a3bb0d105c0cd97cf3bf47f4e36f"
|
resolved "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz#cef31dc8e0a1a3bb0d105c0cd97cf3bf47f4e36f"
|
||||||
@@ -4506,7 +4497,7 @@ readable-stream@1.0:
|
|||||||
isarray "0.0.1"
|
isarray "0.0.1"
|
||||||
string_decoder "~0.10.x"
|
string_decoder "~0.10.x"
|
||||||
|
|
||||||
readall@1, readall@1.0.0:
|
readall@1.0.0:
|
||||||
version "1.0.0"
|
version "1.0.0"
|
||||||
resolved "https://registry.npmjs.org/readall/-/readall-1.0.0.tgz#8b4bfc412e687dbfbb00e0a6ede7e3a0f001b7c7"
|
resolved "https://registry.npmjs.org/readall/-/readall-1.0.0.tgz#8b4bfc412e687dbfbb00e0a6ede7e3a0f001b7c7"
|
||||||
dependencies:
|
dependencies:
|
||||||
@@ -4536,7 +4527,7 @@ redis-parser@^2.6.0:
|
|||||||
version "2.6.0"
|
version "2.6.0"
|
||||||
resolved "https://registry.npmjs.org/redis-parser/-/redis-parser-2.6.0.tgz#52ed09dacac108f1a631c07e9b69941e7a19504b"
|
resolved "https://registry.npmjs.org/redis-parser/-/redis-parser-2.6.0.tgz#52ed09dacac108f1a631c07e9b69941e7a19504b"
|
||||||
|
|
||||||
redis@2:
|
redis@2.8.0:
|
||||||
version "2.8.0"
|
version "2.8.0"
|
||||||
resolved "https://registry.npmjs.org/redis/-/redis-2.8.0.tgz#202288e3f58c49f6079d97af7a10e1303ae14b02"
|
resolved "https://registry.npmjs.org/redis/-/redis-2.8.0.tgz#202288e3f58c49f6079d97af7a10e1303ae14b02"
|
||||||
dependencies:
|
dependencies:
|
||||||
|
|||||||
Reference in New Issue
Block a user