Files
kami-parse-server/src/Adapters/Cache/RedisCacheAdapter/index.js
Diamond Lewis 1666c3e382 [WIP] Enable test suite to be randomized (#7265)
* initial run

* Update ParseGraphQLServer.spec.js

* temporarily enable reporter

* Bump retry limit

* fix undefined database

* try to catch error

* Handle LiveQueryServers

* Update Config.js

* fast-fail false

* Remove usage of AppCache

* oops

* Update contributing guide

* enable debugger, try network retry attempt 1

* Fix ldap unbinding

* move non specs to support

* add missing mock adapter

* fix Parse.Push

* RestController should match batch.spec.js

* Remove request attempt limit

* handle index.spec.js

* Update CHANGELOG.md

* Handle error: tuple concurrently updated

* test transactions

* Clear RedisCache after every test

* LoggerController.spec.js

* Update schemas.spec.js

* finally fix transactions

* fix geopoint deadlock

* transaction with clean database

* batch.spec.js
2021-03-15 02:04:09 -05:00

130 lines
2.8 KiB
JavaScript

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(...args: any) {
const message = ['RedisCacheAdapter: ' + arguments[0]].concat(args.slice(1, args.length));
logger.debug.apply(logger, message);
}
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();
}
handleShutdown() {
if (!this.client) {
return Promise.resolve();
}
return new Promise(resolve => {
this.client.quit(err => {
if (err) {
logger.error('RedisCacheAdapter error on shutdown', { error: err });
}
resolve();
});
});
}
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();
});
})
);
}
// Used for testing
async getAllKeys() {
return new Promise((resolve, reject) => {
this.client.keys('*', (err, keys) => {
if (err) {
reject(err);
} else {
resolve(keys);
}
});
});
}
}
export default RedisCacheAdapter;