You've already forked node-redis
mirror of
https://github.com/redis/node-redis.git
synced 2025-08-09 00:22:08 +03:00
* update workflows & README
* add .deepsource.toml
* fix client.quit, add error events on cluster, fix some "deepsource.io" warnings
* Release 4.0.0-rc.1
* add cluster.duplicate, add some tests
* fix #1650 - add support for Buffer in some commands, add GET_BUFFER command
* fix GET and GET_BUFFER return type
* update FAQ
* Update invalid code example in README.md (#1654)
* Update invalid code example in README.md
* Update README.md
Co-authored-by: Leibale Eidelman <leibale1998@gmail.com>
* fix #1652
* ref #1653 - better types
* better types
* fix 54124793ad
* Update GEOSEARCHSTORE.spec.ts
* fix #1660 - add support for client.HSET('key', 'field', 'value')
* upgrade dependencies, update README
* fix #1659 - add support for db-number in client options url
* fix README, remove unused import, downgrade typedoc & typedoc-plugin-markdown
* update client-configurations.md
* fix README
* add CLUSTER_SLOTS, add some tests
* fix "createClient with url" test with redis 5
* remove unused imports
* Release 4.0.0-rc.2
* add missing semicolon
* replace empty "transformReply" functions with typescript "declare"
* fix EVAL & EVALSHA, add some tests, npm update
* fix #1665 - add ZRANGEBYLEX, ZRANGEBYSCORE, ZRANGEBYSCORE_WITHSCORES
* new issue templates
* add all COMMAND commands
* run COMMAND & COMMAND INFO tests only on redis >6
* Create SECURITY.md
* fix #1671 - add support for all client configurations in cluster
* ref #1671 - add support for defaults
* remove some commands from cluster, npm update, clean code,
* lock benny version
* fix #1674 - remove `isolationPoolOptions` when creating isolated connection
* increase test coverage
* update .npmignore
* Release 4.0.0-rc.3
Co-authored-by: Richard Samuelsson <noobtoothfairy@gmail.com>
133 lines
4.7 KiB
TypeScript
133 lines
4.7 KiB
TypeScript
import COMMANDS from './commands';
|
|
import { RedisCommand, RedisCommandArguments, RedisCommandRawReply, RedisModules, RedisPlugins, RedisScript, RedisScripts } from '../commands';
|
|
import RedisMultiCommand, { RedisMultiQueuedCommand } from '../multi-command';
|
|
import { extendWithCommands, extendWithModulesAndScripts } from '../commander';
|
|
|
|
type RedisClientMultiCommandSignature<C extends RedisCommand, M extends RedisModules, S extends RedisScripts> =
|
|
(...args: Parameters<C['transformArguments']>) => RedisClientMultiCommandType<M, S>;
|
|
|
|
type WithCommands<M extends RedisModules, S extends RedisScripts> = {
|
|
[P in keyof typeof COMMANDS]: RedisClientMultiCommandSignature<(typeof COMMANDS)[P], M, S>
|
|
};
|
|
|
|
type WithModules<M extends RedisModules, S extends RedisScripts> = {
|
|
[P in keyof M]: {
|
|
[C in keyof M[P]]: RedisClientMultiCommandSignature<M[P][C], M, S>;
|
|
};
|
|
};
|
|
|
|
type WithScripts<M extends RedisModules, S extends RedisScripts> = {
|
|
[P in keyof S]: RedisClientMultiCommandSignature<S[P], M, S>
|
|
};
|
|
|
|
export type RedisClientMultiCommandType<M extends RedisModules = {}, S extends RedisScripts = {}> =
|
|
RedisClientMultiCommand & WithCommands<M, S> & WithModules<M, S> & WithScripts<M, S>;
|
|
|
|
export type RedisClientMultiExecutor = (queue: Array<RedisMultiQueuedCommand>, chainId?: symbol) => Promise<Array<RedisCommandRawReply>>;
|
|
|
|
export default class RedisClientMultiCommand {
|
|
readonly #multi = new RedisMultiCommand();
|
|
readonly #executor: RedisClientMultiExecutor;
|
|
|
|
static extend<M extends RedisModules, S extends RedisScripts>(
|
|
plugins?: RedisPlugins<M, S>
|
|
): new (...args: ConstructorParameters<typeof RedisMultiCommand>) => RedisClientMultiCommandType<M, S> {
|
|
return <any>extendWithModulesAndScripts({
|
|
BaseClass: RedisClientMultiCommand,
|
|
modules: plugins?.modules,
|
|
modulesCommandsExecutor: RedisClientMultiCommand.prototype.commandsExecutor,
|
|
scripts: plugins?.scripts,
|
|
scriptsExecutor: RedisClientMultiCommand.prototype.scriptsExecutor
|
|
});
|
|
}
|
|
|
|
readonly v4: Record<string, any> = {};
|
|
|
|
constructor(executor: RedisClientMultiExecutor, legacyMode = false) {
|
|
this.#executor = executor;
|
|
if (legacyMode) {
|
|
this.#legacyMode();
|
|
}
|
|
}
|
|
|
|
#legacyMode(): void {
|
|
this.v4.addCommand = this.addCommand.bind(this);
|
|
(this as any).addCommand = (...args: Array<string | Buffer | Array<string | Buffer>>): this => {
|
|
this.#multi.addCommand(args.flat());
|
|
return this;
|
|
};
|
|
this.v4.exec = this.exec.bind(this);
|
|
(this as any).exec = (callback?: (err: Error | null, replies?: Array<unknown>) => unknown): void => {
|
|
this.v4.exec()
|
|
.then((reply: Array<unknown>) => {
|
|
if (!callback) return;
|
|
|
|
callback(null, reply);
|
|
})
|
|
.catch((err: Error) => {
|
|
if (!callback) {
|
|
// this.emit('error', err);
|
|
return;
|
|
}
|
|
|
|
callback(err);
|
|
});
|
|
};
|
|
|
|
for (const name of Object.keys(COMMANDS)) {
|
|
this.#defineLegacyCommand(name);
|
|
}
|
|
}
|
|
|
|
#defineLegacyCommand(name: string): void {
|
|
(this as any).v4[name] = (this as any)[name].bind(this.v4);
|
|
(this as any)[name] = (...args: Array<unknown>): void => (this as any).addCommand(name, args);
|
|
}
|
|
|
|
commandsExecutor(command: RedisCommand, args: Array<unknown>): this {
|
|
return this.addCommand(
|
|
command.transformArguments(...args),
|
|
command.transformReply
|
|
);
|
|
}
|
|
|
|
addCommand(args: RedisCommandArguments, transformReply?: RedisCommand['transformReply']): this {
|
|
this.#multi.addCommand(args, transformReply);
|
|
return this;
|
|
}
|
|
|
|
scriptsExecutor(script: RedisScript, args: Array<unknown>): this {
|
|
this.#multi.addScript(script, args);
|
|
return this;
|
|
}
|
|
|
|
async exec(execAsPipeline = false): Promise<Array<RedisCommandRawReply>> {
|
|
if (execAsPipeline) {
|
|
return this.execAsPipeline();
|
|
}
|
|
|
|
const commands = this.#multi.exec();
|
|
if (!commands) return [];
|
|
|
|
return this.#multi.handleExecReplies(
|
|
await this.#executor(commands, RedisMultiCommand.generateChainId())
|
|
);
|
|
}
|
|
|
|
EXEC = this.exec;
|
|
|
|
async execAsPipeline(): Promise<Array<RedisCommandRawReply>> {
|
|
if (!this.#multi.queue.length) return [];
|
|
|
|
return this.#multi.transformReplies(
|
|
await this.#executor(this.#multi.queue)
|
|
);
|
|
}
|
|
}
|
|
|
|
extendWithCommands({
|
|
BaseClass: RedisClientMultiCommand,
|
|
commands: COMMANDS,
|
|
executor: RedisClientMultiCommand.prototype.commandsExecutor
|
|
});
|