MCPcopy
hub / github.com/redis/node-redis

github.com/redis/node-redis @bloom@6.1.0 sqlite

repository ↗ · DeepWiki ↗ · release bloom@6.1.0 ↗ · compare 2 versions
2,527 symbols 9,402 edges 1,337 files 209 documented · 8%
README

Node-Redis

Tests Coverage License

Discord Twitch YouTube Twitter

node-redis is a modern, high performance Redis client for Node.js.

How do I Redis?

Learn for free at Redis University

Build faster with the Redis Launchpad

Try the Redis Cloud

Dive in developer tutorials

Join the Redis community

Work at Redis

Installation

Start a redis via docker:

docker run -p 6379:6379 -d redis:8.0-rc1

To install node-redis, simply:

npm install redis

"redis" is the "whole in one" package that includes all the other packages. If you only need a subset of the commands, you can install the individual packages. See the list below.

Packages

Name Description
redis The client with all the "redis-stack" modules
@redis/client The base clients (i.e RedisClient, RedisCluster, etc.)
@redis/bloom Redis Bloom commands
@redis/json Redis JSON commands
@redis/search RediSearch commands
@redis/time-series Redis Time-Series commands
@redis/entraid Secure token-based authentication for Redis clients using Microsoft Entra ID

Looking for a high-level library to handle object mapping? See redis-om-node!

Usage

Basic Example

import { createClient } from "redis";

const client = await createClient()
  .on("error", (err) => console.log("Redis Client Error", err))
  .connect();

await client.set("key", "value");
const value = await client.get("key");
client.destroy();

The above code connects to localhost on port 6379. To connect to a different host or port, use a connection string in the format redis[s]://[[username][:password]@][host][:port][/db-number]:

createClient({
  url: "redis://alice:foobared@awesome.redis.server:6380",
});

You can also use discrete parameters, UNIX sockets, and even TLS to connect. Details can be found in the client configuration guide.

To check if the the client is connected and ready to send commands, use client.isReady which returns a boolean. client.isOpen is also available. This returns true when the client's underlying socket is open, and false when it isn't (for example when the client is still connecting or reconnecting after a network error).

Redis Commands

There is built-in support for all of the out-of-the-box Redis commands. They are exposed using the raw Redis command names (HSET, HGETALL, etc.) and a friendlier camel-cased version (hSet, hGetAll, etc.):

// raw Redis commands
await client.HSET("key", "field", "value");
await client.HGETALL("key");

// friendly JavaScript commands
await client.hSet("key", "field", "value");
await client.hGetAll("key");

Modifiers to commands are specified using a JavaScript object:

await client.set("key", "value", {
  EX: 10,
  NX: true,
});

Replies will be transformed into useful data structures:

await client.hGetAll("key"); // { field1: 'value1', field2: 'value2' }
await client.hVals("key"); // ['value1', 'value2']

Buffers are supported as well:

const client = createClient().withTypeMapping({
  [RESP_TYPES.BLOB_STRING]: Buffer
});

await client.hSet("key", "field", Buffer.from("value")); // 'OK'
await client.hGet("key", "field"); // { field: <Buffer 76 61 6c 75 65> }

For commands that return serialized binary payloads, such as DUMP, map blob strings to Buffer before using the result with commands like RESTORE:

const binaryClient = createClient().withTypeMapping({
  [RESP_TYPES.BLOB_STRING]: Buffer
});

const dump = await binaryClient.dump("source");
await binaryClient.restore("destination", 0, dump);

Unsupported Redis Commands

If you want to run commands and/or use arguments that Node Redis doesn't know about (yet!) use .sendCommand():

await client.sendCommand(["SET", "key", "value", "NX"]); // 'OK'

await client.sendCommand(["HGETALL", "key"]); // ['key1', 'field1', 'key2', 'field2']

Note: the API is different when using a cluster.

Transactions (Multi/Exec)

Start a transaction by calling .multi(), then chaining your commands. When you're done, call .exec() and you'll get an array back with your results:

await client.set("another-key", "another-value");

const [setKeyReply, otherKeyValue] = await client
  .multi()
  .set("key", "value")
  .get("another-key")
  .exec(); // ['OK', 'another-value']

You can also watch keys by calling .watch(). Your transaction will abort if any of the watched keys change.

Blocking Commands

In v4, RedisClient had the ability to create a pool of connections using an "Isolation Pool" on top of the "main" connection. However, there was no way to use the pool without a "main" connection:

const client = await createClient()
  .on("error", (err) => console.error(err))
  .connect();

await client.ping(client.commandOptions({ isolated: true }));

In v5 we've extracted this pool logic into its own class—RedisClientPool:

const pool = await createClientPool()
  .on("error", (err) => console.error(err))
  .connect();

await pool.ping();

Pub/Sub

See the Pub/Sub overview.

Scan Iterator

SCAN results can be looped over using async iterators:

for await (const keys of client.scanIterator()) {
  console.log(keys, await client.mGet(keys));
}

This works with HSCAN, SSCAN, and ZSCAN too:

for await (const { field, value } of client.hScanIterator("hash")) {
}
for await (const member of client.sScanIterator("set")) {
}
for await (const { score, value } of client.zScanIterator("sorted-set")) {
}

You can override the default options by providing a configuration object:

client.scanIterator({
  TYPE: "string", // `SCAN` only
  MATCH: "pattern*",
  COUNT: 100,
});

Compare-and-Set/Delete (CAS/CAD)

Note: CAS/CAD operations were introduced in Redis 8.4

// Conditionally update only if the current value matches
await client.set("key", "new-value", { condition: "IFEQ", matchValue: "old-value" });

// Conditionally delete only if the current value matches
await client.delEx("key", { condition: "IFEQ", matchValue: "expected-value" });

Local Digest

Note: This feature requires the optional @node-rs/xxhash peer dependency.

The digest helper computes an XXH3 64-bit hash locally, matching what Redis computes via the DIGEST command. This is useful for CAS/CAD operations with large values where comparing the full value would be inefficient.

npm install @node-rs/xxhash
import { digest } from "redis";

const hash = await digest("my-value");

await client.set("key", "new-value", { condition: "IFDEQ", matchValue: hash });

Disconnecting

The QUIT command has been deprecated in Redis 7.2 and should now also be considered deprecated in Node-Redis. Instead of sending a QUIT command to the server, the client can simply close the network connection.

client.QUIT/quit() is replaced by client.close(). and, to avoid confusion, client.disconnect() has been renamed to client.destroy().

client.destroy();

Client Side Caching

Node Redis v5 adds support for Client Side Caching, which enables clients to cache query results locally. The Redis server will notify the client when cached results are no longer valid.

// Enable client side caching with RESP3
const client = createClient({
  RESP: 3,
  clientSideCache: {
    ttl: 0,             // Time-to-live (0 = no expiration)
    maxEntries: 0,      // Maximum entries (0 = unlimited)
    evictPolicy: "LRU"  // Eviction policy: "LRU" or "FIFO"
  }
});

See the V5 documentation for more details and advanced usage.

Auto-Pipelining

Node Redis will automatically pipeline requests that are made during the same "tick".

client.set("Tm9kZSBSZWRpcw==", "users:1");
client.sAdd("users:1:tokens", "Tm9kZSBSZWRpcw==");

Of course, if you don't do something with your Promises you're certain to get unhandled Promise exceptions. To take advantage of auto-pipelining and handle your Promises, use Promise.all().

await Promise.all([
  client.set("Tm9kZSBSZWRpcw==", "users:1"),
  client.sAdd("users:1:tokens", "Tm9kZSBSZWRpcw=="),
]);

Sentinel

Look at the sentinel section to figure out how to use this library with sentinels.

Programmability

See the Programmability overview.

Clustering

Check out the Clustering Guide when using Node Redis to connect to a Redis Cluster.

OpenTelemetry

OpenTelemetry Metrics Instrumentation

import { createClient, OpenTelemetry } from "redis";

OpenTelemetry.init({
  metrics: {
    enabled: true
  }
});

const client = createClient()

await client.connect();
// ... use the client as usual

Important: Initializing OpenTelemetry only enables node-redis metrics instrumentation and requires both @opentelemetry/api and an OpenTelemetry SDK configured in your application.

Important: Initialize OpenTelemetry before creating Redis clients. For SDK/provider/exporter setup, verification, and advanced configuration, see:

Diagnostics Channel

Node Redis publishes telemetry through Node.js diagnostics_channel, enabling APM tools and custom instrumentation to observe commands, connections, and internal events. See the Diagnostics Channel guide for the full channel reference and usage examples.

Events

The Node Redis client class is an Nodejs EventEmitter and it emits an event each time the network status changes:

Name When Listener arguments
connect Initiating a connection to the server No arguments
ready Client is ready to use No arguments
end Connection has been closed (via .disconnect()) No arguments
error An error has occurred—usually

Extension points exported contracts — how you extend this code

IdentityProvider (Interface)
(no doc) [6 implementers]
packages/client/lib/authx/identity-provider.ts
IFaultInjectorClient (Interface)
(no doc) [4 implementers]
packages/test-utils/lib/fault-injector/types.ts
TsRangeMultiAggrOptions (Interface)
(no doc)
packages/time-series/lib/commands/RANGE_MULTIAGGR.ts
FtSearchOptions (Interface)
(no doc)
packages/search/lib/commands/SEARCH.ts
RedisDefaultModules (Interface)
(no doc)
packages/redis/index.ts
TestConfig (Interface)
(no doc)
packages/entraid/integration-tests/entraid-integration.spec.ts
JsonSetOptions (Interface)
(no doc)
packages/json/lib/commands/SET.ts
CfReserveOptions (Interface)
(no doc)
packages/bloom/lib/commands/cuckoo/RESERVE.ts

Core symbols most depended-on inside this repo

parseArgs
called by 1188
packages/client/lib/commands/generic-transformers.ts
push
called by 1011
packages/client/lib/client/parser.ts
log
called by 705
packages/test-utils/lib/proxy/redis-proxy.ts
push
called by 649
packages/client/lib/sentinel/wait-queue.ts
testWithClient
called by 433
packages/test-utils/lib/index.ts
testAll
called by 345
packages/test-utils/lib/index.ts
set
called by 320
packages/client/lib/client/cache.ts
pushKey
called by 310
packages/client/lib/client/parser.ts

Shape

Function 1,031
Method 950
Interface 355
Class 187
Enum 4

Languages

TypeScript100%

Modules by API surface

packages/client/lib/sentinel/index.ts103 symbols
packages/client/lib/client/cache.ts103 symbols
packages/client/lib/client/index.ts97 symbols
packages/client/lib/RESP/decoder.ts73 symbols
packages/client/lib/sentinel/test-util.ts54 symbols
packages/client/lib/errors.ts53 symbols
packages/client/lib/client/commands-queue.ts48 symbols
packages/client/lib/cluster/cluster-slots.ts47 symbols
packages/client/lib/cluster/index.ts46 symbols
packages/client/lib/commands/generic-transformers.ts45 symbols
packages/client/lib/opentelemetry/metrics.ts44 symbols
packages/test-utils/lib/index.ts37 symbols

Dependencies from manifests, versioned

@azure/identity4.13.1 · 1×
@azure/msal-common16.5.2 · 1×
@azure/msal-node5.1.5 · 1×
@eslint/js9.39.4 · 1×
@istanbuljs/nyc-config-typescript1.0.2 · 1×
@node-rs/xxhash1.7.6 · 1×
@opentelemetry/api1.9.0 · 1×
@opentelemetry/sdk-metrics2.2.0 · 1×
@redis/bloom6.0.1 · 1×
@redis/client6.0.1 · 1×
@redis/json6.0.1 · 1×
@redis/search6.0.1 · 1×

For agents

$ claude mcp add node-redis \
  -- python -m otcore.mcp_server <graph>

⬇ download graph artifact