using per-key basis queue (#5420)
* adding KeyPromiseQueue * nit * removing secondary object and using a tuple * using array * nits * some tests * Minor refinements * removing old adapter * dummy change, travis test not found * travis test missing, dummy change * revrting mistake * reverting mistake * indentation fix * additional tests for coverage * extending coverage * nits * fixing mistake * better code
This commit is contained in:
committed by
Arthur Cinader
parent
c3eb256139
commit
214aa2e450
@@ -1,83 +0,0 @@
|
||||
import redis from 'redis';
|
||||
import logger from '../../logger';
|
||||
|
||||
const DEFAULT_REDIS_TTL = 30 * 1000; // 30 seconds in milliseconds
|
||||
|
||||
function debug() {
|
||||
logger.debug.apply(logger, ['RedisCacheAdapter', ...arguments]);
|
||||
}
|
||||
|
||||
export class RedisCacheAdapter {
|
||||
constructor(redisCtx, ttl = DEFAULT_REDIS_TTL) {
|
||||
this.client = redis.createClient(redisCtx);
|
||||
this.p = Promise.resolve();
|
||||
this.ttl = ttl;
|
||||
}
|
||||
|
||||
get(key) {
|
||||
debug('get', key);
|
||||
this.p = this.p.then(() => {
|
||||
return new Promise(resolve => {
|
||||
this.client.get(key, function(err, res) {
|
||||
debug('-> get', key, res);
|
||||
if (!res) {
|
||||
return resolve(null);
|
||||
}
|
||||
resolve(JSON.parse(res));
|
||||
});
|
||||
});
|
||||
});
|
||||
return this.p;
|
||||
}
|
||||
|
||||
put(key, value, ttl = this.ttl) {
|
||||
value = JSON.stringify(value);
|
||||
debug('put', key, value, ttl);
|
||||
if (ttl === 0) {
|
||||
return this.p; // ttl of zero is a logical no-op, but redis cannot set expire time of zero
|
||||
}
|
||||
if (ttl < 0 || isNaN(ttl)) {
|
||||
ttl = DEFAULT_REDIS_TTL;
|
||||
}
|
||||
this.p = this.p.then(() => {
|
||||
return new Promise(resolve => {
|
||||
if (ttl === Infinity) {
|
||||
this.client.set(key, value, function() {
|
||||
resolve();
|
||||
});
|
||||
} else {
|
||||
this.client.psetex(key, ttl, value, function() {
|
||||
resolve();
|
||||
});
|
||||
}
|
||||
});
|
||||
});
|
||||
return this.p;
|
||||
}
|
||||
|
||||
del(key) {
|
||||
debug('del', key);
|
||||
this.p = this.p.then(() => {
|
||||
return new Promise(resolve => {
|
||||
this.client.del(key, function() {
|
||||
resolve();
|
||||
});
|
||||
});
|
||||
});
|
||||
return this.p;
|
||||
}
|
||||
|
||||
clear() {
|
||||
debug('clear');
|
||||
this.p = this.p.then(() => {
|
||||
return new Promise(resolve => {
|
||||
this.client.flushdb(function() {
|
||||
resolve();
|
||||
});
|
||||
});
|
||||
});
|
||||
return this.p;
|
||||
}
|
||||
}
|
||||
|
||||
export default RedisCacheAdapter;
|
||||
43
src/Adapters/Cache/RedisCacheAdapter/KeyPromiseQueue.js
Normal file
43
src/Adapters/Cache/RedisCacheAdapter/KeyPromiseQueue.js
Normal file
@@ -0,0 +1,43 @@
|
||||
// KeyPromiseQueue is a simple promise queue
|
||||
// used to queue operations per key basis.
|
||||
// Once the tail promise in the key-queue fulfills,
|
||||
// the chain on that key will be cleared.
|
||||
export class KeyPromiseQueue {
|
||||
constructor() {
|
||||
this.queue = {};
|
||||
}
|
||||
|
||||
enqueue(key, operation) {
|
||||
const tuple = this.beforeOp(key);
|
||||
const toAwait = tuple[1];
|
||||
const nextOperation = toAwait.then(operation);
|
||||
const wrappedOperation = nextOperation.then(result => {
|
||||
this.afterOp(key);
|
||||
return result;
|
||||
});
|
||||
tuple[1] = wrappedOperation;
|
||||
return wrappedOperation;
|
||||
}
|
||||
|
||||
beforeOp(key) {
|
||||
let tuple = this.queue[key];
|
||||
if (!tuple) {
|
||||
tuple = [0, Promise.resolve()];
|
||||
this.queue[key] = tuple;
|
||||
}
|
||||
tuple[0]++;
|
||||
return tuple;
|
||||
}
|
||||
|
||||
afterOp(key) {
|
||||
const tuple = this.queue[key];
|
||||
if (!tuple) {
|
||||
return;
|
||||
}
|
||||
tuple[0]--;
|
||||
if (tuple[0] <= 0) {
|
||||
delete this.queue[key];
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
101
src/Adapters/Cache/RedisCacheAdapter/index.js
Normal file
101
src/Adapters/Cache/RedisCacheAdapter/index.js
Normal file
@@ -0,0 +1,101 @@
|
||||
import redis from 'redis';
|
||||
import logger from '../../../logger';
|
||||
import { KeyPromiseQueue } from './KeyPromiseQueue';
|
||||
|
||||
const DEFAULT_REDIS_TTL = 30 * 1000; // 30 seconds in milliseconds
|
||||
const FLUSH_DB_KEY = '__flush_db__';
|
||||
|
||||
function debug() {
|
||||
logger.debug.apply(logger, ['RedisCacheAdapter', ...arguments]);
|
||||
}
|
||||
|
||||
const isValidTTL = ttl => typeof ttl === 'number' && ttl > 0;
|
||||
|
||||
export class RedisCacheAdapter {
|
||||
constructor(redisCtx, ttl = DEFAULT_REDIS_TTL) {
|
||||
this.ttl = isValidTTL(ttl) ? ttl : DEFAULT_REDIS_TTL;
|
||||
this.client = redis.createClient(redisCtx);
|
||||
this.queue = new KeyPromiseQueue();
|
||||
}
|
||||
|
||||
get(key) {
|
||||
debug('get', key);
|
||||
return this.queue.enqueue(
|
||||
key,
|
||||
() =>
|
||||
new Promise(resolve => {
|
||||
this.client.get(key, function(err, res) {
|
||||
debug('-> get', key, res);
|
||||
if (!res) {
|
||||
return resolve(null);
|
||||
}
|
||||
resolve(JSON.parse(res));
|
||||
});
|
||||
})
|
||||
);
|
||||
}
|
||||
|
||||
put(key, value, ttl = this.ttl) {
|
||||
value = JSON.stringify(value);
|
||||
debug('put', key, value, ttl);
|
||||
|
||||
if (ttl === 0) {
|
||||
// ttl of zero is a logical no-op, but redis cannot set expire time of zero
|
||||
return this.queue.enqueue(key, () => Promise.resolve());
|
||||
}
|
||||
|
||||
if (ttl === Infinity) {
|
||||
return this.queue.enqueue(
|
||||
key,
|
||||
() =>
|
||||
new Promise(resolve => {
|
||||
this.client.set(key, value, function() {
|
||||
resolve();
|
||||
});
|
||||
})
|
||||
);
|
||||
}
|
||||
|
||||
if (!isValidTTL(ttl)) {
|
||||
ttl = this.ttl;
|
||||
}
|
||||
|
||||
return this.queue.enqueue(
|
||||
key,
|
||||
() =>
|
||||
new Promise(resolve => {
|
||||
this.client.psetex(key, ttl, value, function() {
|
||||
resolve();
|
||||
});
|
||||
})
|
||||
);
|
||||
}
|
||||
|
||||
del(key) {
|
||||
debug('del', key);
|
||||
return this.queue.enqueue(
|
||||
key,
|
||||
() =>
|
||||
new Promise(resolve => {
|
||||
this.client.del(key, function() {
|
||||
resolve();
|
||||
});
|
||||
})
|
||||
);
|
||||
}
|
||||
|
||||
clear() {
|
||||
debug('clear');
|
||||
return this.queue.enqueue(
|
||||
FLUSH_DB_KEY,
|
||||
() =>
|
||||
new Promise(resolve => {
|
||||
this.client.flushdb(function() {
|
||||
resolve();
|
||||
});
|
||||
})
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
export default RedisCacheAdapter;
|
||||
Reference in New Issue
Block a user