Skip to content

Commit e298024

Browse files
authored
feat(redis): update redis commands (#120)
this updates both the https://github.com/antirez/redis-doc and https://github.com/NodeRedis/redis-commands dependencies. fixes #46 Mostly the changes are new commands, but some function can now be called like client.hset('myhash', ['field1', 'Hello']) rather than client.hset('myhash', 'field1', 'Hello'). This allows setting multiple fields, e.g. client.hset('myhash', ['field1', 'Hello'], ['field2', 'Goodbye']). The last arg of setbit can also now be number, rather than a string. Likely the string form will be dropped in the next major version. See the diff from the PR merging this change for examples in the snapshot tests.
1 parent 4713656 commit e298024

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

61 files changed

+7335
-3088
lines changed

README.md

+3-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ This package is a wrapper around node_redis and exclusively uses Promises. It pu
1414
## Usage
1515

1616
```cli
17-
npm install --save handy-redis
17+
npm install --save redis handy-redis
1818
```
1919

2020
ES6/TypeScript:
@@ -47,6 +47,8 @@ client
4747
The package is published with TypeScript types, with the redis documentation and response type attached to each command:
4848
![](./docs/intellisense.png)
4949

50+
Note: the [redis](https://npmjs.com/package/redis) package is listed as a peer dependency, so should be installed separately. If you need to use recent redis commands (e.g. `xadd` (recent at time of writing, at least)), you can run `npm install redis-commands` to tell the `redis` package to use more up-to-date commands than [redis](https://npmjs.com/package/redis) pulls in by default.
51+
5052
### Examples
5153

5254
See the [snapshot tests](https://github.com/mmkal/handy-redis/blob/master/test/generated/commands/__snapshots__) for tons of usage examples.

package-lock.json

+3-3
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,7 @@
7979
"lodash": "^4.17.14",
8080
"npm-run-all": "^4.1.5",
8181
"prettier": "^1.14.2",
82+
"redis-commands": "^1.5.0",
8283
"semantic-release": "^15.13.16",
8384
"shelljs": "^0.8.3",
8485
"shx": "^0.3.2",

scripts/cli-examples.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ import { log, warn, error } from "./log";
77
export const getExampleRuns = async () => {
88
const examples = getCliExamples();
99
const redisCli = spawn("docker", ["exec", "-i", "handy_redis", "redis-cli", "--no-raw"], { env: process.env });
10-
redisCli.stdin.setDefaultEncoding("utf-8");
10+
redisCli.stdin!.setDefaultEncoding("utf-8");
1111
const redisInteractor = {
1212
onstdout: (data: string) => log(data),
1313
onstderr: (data: string) => warn(data),
@@ -21,12 +21,12 @@ export const getExampleRuns = async () => {
2121
resolve(parseCommandOutput(command, null, data));
2222
};
2323
log(">", command);
24-
redisCli.stdin.write(`${command}\n`);
24+
redisCli.stdin!.write(`${command}\n`);
2525
}),
2626
};
2727

28-
redisCli.stdout.on("data", data => redisInteractor.onstdout(data.toString()));
29-
redisCli.stderr.on("data", data => redisInteractor.onstderr(data.toString()));
28+
redisCli.stdout!.on("data", data => redisInteractor.onstdout(data.toString()));
29+
redisCli.stderr!.on("data", data => redisInteractor.onstderr(data.toString()));
3030

3131
const runs = new Array<CliExampleRun>();
3232
for (const example of examples) {

scripts/command/generate.ts

+35-10
Original file line numberDiff line numberDiff line change
@@ -42,14 +42,13 @@ const typeFor = (arg: Argument): string => {
4242
const buildTypeScriptCommandInfo = (name: string, command: Command): BasicCommandInfo[] => {
4343
const baseArgs = (command.arguments || []).map((a, index, all) => {
4444
const nameParts = new Array<string | number>();
45-
if (a.name) {
46-
nameParts.push(a.name);
45+
if (a.command) {
46+
nameParts.push(a.command);
4747
}
48-
const previousArgsWithSameName = all
49-
.slice(0, index)
50-
.filter(other => other !== a && other.name === a.name);
51-
if (a.command && (nameParts.length === 0 || previousArgsWithSameName.length > 0)) {
52-
nameParts.unshift(a.command);
48+
if (Array.isArray(a.name)) {
49+
nameParts.push(...a.name);
50+
} else if (a.name && a.name.toUpperCase() !== a.command) {
51+
nameParts.push(a.name);
5352
}
5453
const argJson = JSON.stringify(a);
5554
if (all.map(other => JSON.stringify(other)).indexOf(argJson) !== index) {
@@ -59,7 +58,10 @@ const buildTypeScriptCommandInfo = (name: string, command: Command): BasicComman
5958
nameParts.push("arg");
6059
nameParts.push(index);
6160
}
62-
const argName = nameParts.join("_").replace(/\W/g, "_");
61+
const argName = nameParts
62+
.map(p => _.camelCase(p.toString()))
63+
.join("_")
64+
.replace(/idOr$/, "idOr$");
6365
return {
6466
...a,
6567
name: argName,
@@ -99,11 +101,34 @@ const buildTypeScriptCommandInfo = (name: string, command: Command): BasicComman
99101
.map(x => x!);
100102
};
101103

102-
export const getBasicCommands = _.once(() => {
103-
const referenceClient: { [methodName: string]: any } = createClient();
104+
const getCommandCollection = () => {
104105
const commandsJson = readFileSync(`${redisDoc}/commands.json`, "utf8");
105106
const commandCollection: CommandCollection = JSON.parse(commandsJson);
106107

108+
// hack: workaround while waiting for https://github.com/antirez/redis-doc/pulls/1231
109+
const expectedSetArgs = '[{"name":"key","type":"key"},{"name":"value","type":"string"},{"name":"expiration","type":"enum","enum":["EX seconds","PX milliseconds"],"optional":true},{"name":"condition","type":"enum","enum":["NX","XX"],"optional":true}]';
110+
if (JSON.stringify(commandCollection.SET.arguments) !== expectedSetArgs) {
111+
throw Error([
112+
"unexpected arguments value for command SET",
113+
"hack working around the issue mentioned in https://github.com/antirez/redis-doc/pulls/1231 might not be needed anymore",
114+
"expected: " + expectedSetArgs,
115+
"actual: " + JSON.stringify(commandCollection.SET.arguments),
116+
].join("\n"));
117+
}
118+
commandCollection.SET.arguments = [
119+
{ name: "key", type: "key" },
120+
{ name: "value", type: "string" },
121+
{ command: "EX", name: "seconds", type: "integer", optional: true },
122+
{ command: "PX", name: "milliseconds", type: "integer", optional: true },
123+
{ name: "condition", type: "enum", enum: ["NX", "XX"], optional: true }
124+
];
125+
return commandCollection;
126+
};
127+
128+
export const getBasicCommands = _.once(() => {
129+
const referenceClient: { [methodName: string]: any } = createClient();
130+
const commandCollection = getCommandCollection();
131+
107132
const basicCommands = _.flatten(Object.keys(commandCollection).map(name => {
108133
const methodName = simplifyName(name);
109134
if (typeof referenceClient[methodName] !== "function") {

scripts/generate-types.ts

+13
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,15 @@ import { useUnderlyingImpl } from "../src/overrides";
66
const generateClientInterface = async (getCommands: typeof getFullCommands) => {
77
const typescriptCommands = await getCommands();
88

9+
// hack: hset now can set multiple hash keys at once, so the generated type is (string, ...[string, string])
10+
// it used to be (string, string, string) - explicitly allow that here.
11+
// todo: remove this hack in next major version. Or earlier, if there's a better way to handle this class of command
12+
const hsetIndex = typescriptCommands.findIndex(c => c.name === "hset");
13+
typescriptCommands.splice(hsetIndex, 0, {
14+
...typescriptCommands[hsetIndex],
15+
args: ["key", "field", "value"].map(name => ({ name, type: "string" }))
16+
});
17+
918
const interfaceDeclarations = typescriptCommands.map(commandInfo => {
1019
const docs = [
1120
`/**`,
@@ -20,6 +29,10 @@ const generateClientInterface = async (getCommands: typeof getFullCommands) => {
2029
}
2130

2231
const argList = commandInfo.args
32+
// hack: setbit allowed a string as the last argument at one point.
33+
// for backwards compatibility, continue allowing it even though redis-doc has updated
34+
// todo: remove this hack in v2
35+
.map(a => commandInfo.name === "setbit" && a.name === "value" ? { ...a, type: "number | string" } : a)
2336
.map(a => `${a.name}: ${a.type}`)
2437
.join(`,${EOL}${twotabs}`);
2538

scripts/overloads.ts

+7-17
Original file line numberDiff line numberDiff line change
@@ -4,24 +4,14 @@ import { get } from "lodash";
44

55
export const getOverloads = (args: Argument[]): Argument[][] => {
66
if (args.length <= 1) {
7-
const overloads = [args];
8-
if (get(args, "[0].optional")) {
9-
overloads.push([]);
10-
}
11-
return overloads;
7+
return get(args, "[0].optional") ? [args, []] : [args];
128
}
13-
const first = args[0];
14-
const theRest = args.slice(1);
15-
let variations = new Array<Argument[]>();
16-
const overloadsOfTheRest = getOverloads(theRest);
17-
const argListsWithFirst = overloadsOfTheRest.map(argsSubList => [
18-
first,
19-
...argsSubList
20-
]);
21-
variations = variations.concat(argListsWithFirst);
22-
if (first.optional) {
23-
variations = variations.concat(overloadsOfTheRest);
9+
const head = args[0];
10+
const tailOverloads = getOverloads(args.slice(1));
11+
const overloads = tailOverloads.map(argsSubList => [head, ...argsSubList]);
12+
if (head.optional) {
13+
overloads.push(...tailOverloads);
2414
}
2515

26-
return variations;
16+
return overloads;
2717
};

scripts/redis-doc

0 commit comments

Comments
 (0)