Skip to content

Commit ee47d20

Browse files
authored
Feature/Proxy Implementation (#41)
* WIP Proxy Implementation * WIP Test Coverage * Fixing Tests
1 parent 78f1532 commit ee47d20

15 files changed

+473
-127
lines changed

.env.example

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
REDIS_HOST=localhost
2+
REDIS_PORT=6379

package.json

+7-5
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"private": false,
33
"author": "Matthew Oaxaca",
44
"license": "MIT",
5-
"version": "1.1.8",
5+
"version": "2.0.0",
66
"name": "async-redis",
77
"keywords": [
88
"redis",
@@ -30,20 +30,22 @@
3030
"scripts": {
3131
"coveralls": "nyc yarn test && nyc report --reporter=text-lcov | coveralls",
3232
"lint": "eslint --fix --ext .js, .",
33-
"test": "mocha",
34-
"test:unit": "mocha",
33+
"test": "mocha test/**/*.spec.js --config test/setup.js --exit",
34+
"test:unit": "mocha test/unit/*.spec.js --config test/setup.js --exit",
3535
"version:patch": "npm version patch",
3636
"version:minor": "npm version minor",
3737
"version:major": "npm version major"
3838
},
3939
"dependencies": {
40-
"redis": "^3.0.2",
41-
"redis-commands": "^1.6.0"
40+
"redis": "3.0.2"
4241
},
4342
"devDependencies": {
43+
"@types/node": "^14.0.27",
44+
"@types/redis": "^2.8.25",
4445
"chai": "^4.2.0",
4546
"chai-as-promised": "^7.1.1",
4647
"coveralls": "^3.1.0",
48+
"dotenv": "^8.2.0",
4749
"eslint": "^7.6.0",
4850
"mocha": "^8.1.0",
4951
"nyc": "^15.1.0"

src/index.d.ts

+162
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,162 @@
1+
import { Commands, RedisClient, ClientOpts, ServerInfo } from 'redis';
2+
import {EventEmitter} from "events";
3+
4+
type Callback<T> = (err: Error | null, reply: T) => void;
5+
6+
type Omit<T, K extends keyof T> = Pick<T, Exclude<keyof T, K>>;
7+
type Omitted = Omit<RedisClient, keyof Commands<boolean>>;
8+
type OkOrError = 'OK'|Error
9+
10+
interface OverloadedCommand<T, R> {
11+
(arg1: T, arg2: T, arg3: T, arg4: T, arg5: T, arg6: T): R;
12+
(arg1: T, arg2: T, arg3: T, arg4: T, arg5: T): R;
13+
(arg1: T, arg2: T, arg3: T, arg4: T): R;
14+
(arg1: T, arg2: T, arg3: T): R;
15+
(arg1: T, arg2: T | T[]): R;
16+
(arg1: T | T[]): R;
17+
(...args: Array<T>): R;
18+
}
19+
20+
interface Promisified<T = RedisClient> extends Omitted, Commands<Promise<boolean>> {}
21+
22+
interface AsyncRedisConstructor {
23+
new (port: number, host?: string, options?: ClientOpts): Promisified;
24+
new (unix_socket: string, options?: ClientOpts): Promisified;
25+
new (redis_url: string, options?: ClientOpts): Promisified;
26+
new (options?: ClientOpts): Promisified;
27+
28+
createClient(port: number, host?: string, options?: ClientOpts): Promisified;
29+
createClient(unix_socket: string, options?: ClientOpts): Promisified;
30+
createClient(redis_url: string, options?: ClientOpts): Promisified;
31+
createClient(options?: ClientOpts): Promisified;
32+
33+
decorate: (client: RedisClient) => Promisified;
34+
}
35+
36+
interface AsyncRedisEventHandlers extends EventEmitter {
37+
on(event: 'message' | 'message_buffer', listener: (channel: string, message: string) => void): this;
38+
on(event: 'pmessage' | 'pmessage_buffer', listener: (pattern: string, channel: string, message: string) => void): this;
39+
on(event: 'subscribe' | 'unsubscribe', listener: (channel: string, count: number) => void): this;
40+
on(event: 'psubscribe' | 'punsubscribe', listener: (pattern: string, count: number) => void): this;
41+
on(event: string, listener: (...args: any[]) => void): this;
42+
}
43+
44+
interface AsyncRedisCommands<R> {
45+
/**
46+
* Listen for all requests received by the server in real time.
47+
*/
48+
monitor(cb?: Callback<undefined>): any;
49+
MONITOR(cb?: Callback<undefined>): any;
50+
51+
/**
52+
* Get information and statistics about the server.
53+
*/
54+
info(): Promise<ServerInfo|boolean>;
55+
info(section?: string | string[]): Promise<ServerInfo|boolean>;
56+
INFO(): Promise<ServerInfo|boolean>;
57+
INFO(section?: string | string[]): Promise<ServerInfo|boolean>;
58+
59+
/**
60+
* Ping the server.
61+
*/
62+
ping(): Promise<string|boolean>;
63+
ping(message: string): Promise<string|boolean>;
64+
PING(): Promise<string|boolean>;
65+
PING(message: string): Promise<string|boolean>;
66+
67+
/**
68+
* Authenticate to the server.
69+
*/
70+
auth(password: string): Promise<string>;
71+
AUTH(password: string): Promise<string>;
72+
73+
/**
74+
* Get array of Redis command details.
75+
*
76+
* COUNT - Get total number of Redis commands.
77+
* GETKEYS - Extract keys given a full Redis command.
78+
* INFO - Get array of specific REdis command details.
79+
*/
80+
command(cb?: Callback<Array<[string, number, string[], number, number, number]>>): R;
81+
COMMAND(cb?: Callback<Array<[string, number, string[], number, number, number]>>): R;
82+
83+
/**
84+
* Get array of Redis command details.
85+
*
86+
* COUNT - Get array of Redis command details.
87+
* GETKEYS - Extract keys given a full Redis command.
88+
* INFO - Get array of specific Redis command details.
89+
* GET - Get the value of a configuration parameter.
90+
* REWRITE - Rewrite the configuration file with the in memory configuration.
91+
* SET - Set a configuration parameter to the given value.
92+
* RESETSTAT - Reset the stats returned by INFO.
93+
*/
94+
config: OverloadedCommand<string, boolean>;
95+
CONFIG: OverloadedCommand<string, boolean>;
96+
97+
/**
98+
* Return the number of keys in the selected database.
99+
*/
100+
dbsize(): Promise<number>;
101+
DBSIZE(): Promise<number>;
102+
103+
/**
104+
* OBJECT - Get debugging information about a key.
105+
* SEGFAULT - Make the server crash.
106+
*/
107+
debug: OverloadedCommand<string, boolean>;
108+
DEBUG: OverloadedCommand<string, boolean>;
109+
110+
/**
111+
* PubSub Commands
112+
*/
113+
/**
114+
* Post a message to a channel.
115+
*/
116+
publish(channel: string, value: string): Promise<number|boolean>;
117+
PUBLISH(channel: string, value: string): Promise<number|boolean>;
118+
119+
/**
120+
* CRUD Commands
121+
*/
122+
123+
/**
124+
* Append a value to a key.
125+
*/
126+
append(key: string, value: string): Promise<number>;
127+
APPEND(key: string, value: string): Promise<number>;
128+
129+
/**
130+
* Asynchronously rewrite the append-only file.
131+
*/
132+
bgrewriteaof(): Promise<OkOrError>;
133+
BGREWRITEAOF(): Promise<OkOrError>;
134+
135+
/**
136+
* Asynchronously save the dataset to disk.
137+
*/
138+
bgsave(): Promise<OkOrError>;
139+
BGSAVE(): Promise<OkOrError>;
140+
141+
/**
142+
* Determine if a key exists.
143+
*/
144+
exists: OverloadedCommand<string, R>;
145+
EXISTS: OverloadedCommand<string, R>;
146+
147+
/**
148+
* Set the string value of a key.
149+
*/
150+
set(key: string, value: string): Promise<string|boolean>;
151+
set(key: string, value: string, flag: string): Promise<string|boolean>;
152+
set(key: string, value: string, mode: string, duration: number): Promise<string|undefined>;
153+
set(key: string, value: string, mode: string, duration: number, flag: string): Promise<string|undefined>;
154+
SET(key: string, value: string): Promise<string|boolean>;
155+
SET(key: string, value: string, flag: string): Promise<string|boolean>;
156+
SET(key: string, value: string, mode: string, duration: number): Promise<string|undefined>;
157+
SET(key: string, value: string, mode: string, duration: number, flag: string): Promise<string|undefined>;
158+
}
159+
160+
interface AsyncRedisInterface extends AsyncRedisConstructor, AsyncRedisEventHandlers, AsyncRedisCommands<boolean> {}
161+
declare const AsyncRedis: AsyncRedisInterface;
162+
export = AsyncRedis;

src/index.js

+38-36
Original file line numberDiff line numberDiff line change
@@ -1,47 +1,49 @@
11
const redis = require('redis');
2-
const commands = require('redis-commands').list;
32
const objectDecorator = require('./object-decorator');
3+
const objectPromisify = require('./object-promisify');
4+
const redisCommands = require('./redis-commands');
45

5-
const AsyncRedis = function (args) {
6-
const client = Array.isArray(args) ? redis.createClient(...args) : redis.createClient(args);
7-
return AsyncRedis.decorate(client);
8-
};
9-
10-
AsyncRedis.createClient = (...args) => new AsyncRedis(args);
6+
const redisClients = new Map();
117

12-
const commandsToSkipSet = new Set(['multi']);
13-
const commandSet = new Set(commands.filter(c => !commandsToSkipSet.has(c)));
14-
const queueCommandSet = new Set(['batch', 'multi']);
15-
const multiCommandSet = new Set(['exec', 'exec_atomic']);
8+
/**
9+
* @return RedisClient
10+
*/
11+
const AsyncRedis = function (args=null) {
12+
if (args) {
13+
const serializedArgs = JSON.stringify(args);
14+
if (!redisClients.has(serializedArgs)) {
15+
redisClients.set(serializedArgs, Array.isArray(args) ? redis.createClient(...args) : redis.createClient(args));
16+
}
17+
this.setup(redisClients.get(serializedArgs));
18+
}
19+
};
1620

17-
const promisify = function (object, method) {
18-
return (...args) => new Promise((resolve, reject) => {
19-
args.push((error, ...results) => {
20-
if (error) {
21-
reject(error, ...results);
22-
} else {
23-
resolve(...results);
21+
AsyncRedis.prototype.setup = function(redisClient) {
22+
this.__redisClient = redisClient;
23+
const commandConfigs = redisCommands(redisClient);
24+
objectDecorator(redisClient, (name, method) => {
25+
if (commandConfigs.commands.has(name)) {
26+
objectPromisify(this, redisClient, name);
27+
} else if (commandConfigs.queueCommands.has(name)) {
28+
return (...args) => {
29+
const multi = method.apply(redisClient, args);
30+
return objectDecorator(multi, (multiName, multiMethod) => {
31+
if (commandConfigs.multiCommands.has(multiName)) {
32+
return objectPromisify(multi, multiMethod);
33+
}
34+
return multiMethod;
35+
});
2436
}
25-
});
26-
method.apply(object, args);
37+
}
2738
});
2839
};
2940

30-
AsyncRedis.decorate = redisClient => objectDecorator(redisClient, (name, method) => {
31-
if (commandSet.has(name)) {
32-
return promisify(redisClient, method);
33-
} else if (queueCommandSet.has(name)) {
34-
return (...args) => {
35-
const multi = method.apply(redisClient, args);
36-
return objectDecorator(multi, (multiName, multiMethod) => {
37-
if (multiCommandSet.has(multiName)) {
38-
return promisify(multi, multiMethod);
39-
}
40-
return multiMethod;
41-
});
42-
}
43-
}
44-
return method;
45-
});
41+
AsyncRedis.createClient = (...args) => new AsyncRedis(args);
42+
43+
AsyncRedis.decorate = (redisClient) => {
44+
const asyncClient = new AsyncRedis();
45+
asyncClient.setup(redisClient);
46+
return asyncClient;
47+
};
4648

4749
module.exports = AsyncRedis;

src/object-decorator.js

+4-3
Original file line numberDiff line numberDiff line change
@@ -4,12 +4,13 @@
44
* @returns {*}
55
*/
66
module.exports = (object, decorator) => {
7-
/* eslint-disable */
87
for (const prop in object) {
98
if (typeof object[prop] === 'function') {
10-
object[prop] = decorator(prop, object[prop]);
9+
const returned = decorator(prop, object[prop]);
10+
if (typeof returned === 'function') {
11+
object[prop] = returned;
12+
}
1113
}
1214
}
13-
/* eslint-enable */
1415
return object;
1516
};

src/object-promisify.js

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
/**
2+
* @param proxy
3+
* @param object
4+
* @param name
5+
* @returns void
6+
*/
7+
module.exports = (proxy, object, name) => {
8+
proxy[name] = (...args) => {
9+
return new Promise((resolve, reject) => {
10+
args.push((error, ...results) => {
11+
if (error) {
12+
reject(error, ...results);
13+
} else {
14+
resolve(...results);
15+
}
16+
});
17+
object[name](...args);
18+
});
19+
};
20+
};

src/redis-commands.js

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
const commandsToSkipSet = new Set(['multi']);
2+
const queueCommands = new Set(['batch', 'multi']);
3+
const multiCommands = new Set(['exec', 'exec_atomic']);
4+
5+
/**
6+
* @param redisClient
7+
* @returns {{queueCommands: Set<string>, multiCommands: Set<string>, commands: Set<any>}}
8+
*/
9+
module.exports = (redisClient) => {
10+
const commands = [];
11+
for (const prop in redisClient) {
12+
if (typeof redisClient[prop] === 'function') {
13+
commands.push(prop);
14+
}
15+
}
16+
return {
17+
commands: new Set(commands.filter(c => !commandsToSkipSet.has(c))),
18+
queueCommands,
19+
multiCommands,
20+
}
21+
};

0 commit comments

Comments
 (0)