Nodejs Memcache Client
npm install memcache




- Getting Started
- Installation
- Basic Usage
- Custom Connection
- API
- Constructor
- Properties
- Connection Management
- Node Management
- Data Storage Operations
- String Modification Operations
- Deletion & Expiration
- Numeric Operations
- Server Management & Statistics
- Validation
- Helper Functions
- Hooks and Events
- Events
- Available Events
- Hooks
- Available Hooks
- get(key)
- set(key, value, exptime?, flags?)
- [gets(keys[])](#getskeys)
- add(key, value, exptime?, flags?)
- replace(key, value, exptime?, flags?)
- append(key, value)
- prepend(key, value)
- delete(key)
- incr(key, value?)
- decr(key, value?)
- touch(key, exptime)
- Hook Examples
- Distribution Algorithms
- KetamaHash (Default)
- ModulaHash
- Choosing an Algorithm
- Retry Configuration
- Basic Retry Setup
- Backoff Strategies
- Idempotent Safety
- Methods Without Retry Support
- SASL Authentication
- Enabling SASL Authentication
- SASL Options
- Per-Node SASL Configuration
- Authentication Events
- Server Configuration
- Contributing
- License and Copyright
``bash`
npm install memcache
or with pnpm:
`bash`
pnpm add memcache
`javascript
import { Memcache } from 'memcache';
// Create a new client
const client = new Memcache();
// Set a value
await client.set('mykey', 'Hello, Memcache!');
// Get a value
const value = await client.get('mykey');
console.log(value); // ['Hello, Memcache!']
// Delete a value
await client.delete('mykey');
// Close the connection
await client.quit();
`
You can also just pass in the uri into the constructor
`javascript
// Single node as string
const client = new Memcache('localhost:11211');
// Single node with protocol
const client = new Memcache('memcache://192.168.1.100:11211');
// Multiple nodes with options
const client = new Memcache({
nodes: ['localhost:11211', 'server2:11211'],
timeout: 10000
});
`
You can specify multiple Memcache nodes by passing an array of connection strings:
`javascript
import { Memcache } from 'memcache';
// Create a client with multiple nodes
const client = new Memcache({
nodes: ['localhost:11211', '192.168.1.100:11211', 'memcache://192.168.1.101:11211']
});
// Set and get values (automatically distributed across nodes)
await client.set('mykey', 'Hello, Memcache!');
const value = await client.get('mykey');
console.log(value); // ['Hello, Memcache!']
// Close the connection
await client.quit();
`
You can also pass an array of MemcacheNode instances for advanced configuration:
`javascript
import { Memcache, createNode } from 'memcache';
// Create nodes with custom settings
const node1 = createNode('localhost', 11211, { weight: 2 });
const node2 = createNode('192.168.1.100', 11211, { weight: 1 });
const node3 = createNode('192.168.1.101', 11211, { weight: 1 });
// Create a client with MemcacheNode instances
const client = new Memcache({
nodes: [node1, node2, node3],
timeout: 10000
});
// node1 will receive twice as much traffic due to higher weight
await client.set('mykey', 'Hello, Memcache!');
const value = await client.get('mykey');
console.log(value); // ['Hello, Memcache!']
// Close the connection
await client.quit();
`
`typescript`
new Memcache(options?: string | MemcacheOptions)
Creates a new Memcache client instance. You can pass either:
- A string representing a single node URI (uses default settings)
- A MemcacheOptions object for custom configuration
Examples:
`javascript
// Single node as string
const client = new Memcache('localhost:11211');
// Single node with protocol
const client = new Memcache('memcache://192.168.1.100:11211');
// Multiple nodes with options
const client = new Memcache({
nodes: ['localhost:11211', 'server2:11211'],
timeout: 10000
});
`
- nodes?: (string | MemcacheNode)[] - Array of node URIs or MemcacheNode instances["localhost:11211", "memcache://192.168.1.100:11212"]
- Examples: timeout?: number
- - Operation timeout in milliseconds (default: 5000)keepAlive?: boolean
- - Keep connection alive (default: true)keepAliveDelay?: number
- - Keep alive delay in milliseconds (default: 1000)hash?: HashProvider
- - Hash provider for consistent hashing (default: KetamaHash)retries?: number
- - Number of retry attempts for failed commands (default: 0)retryDelay?: number
- - Base delay in milliseconds between retries (default: 100)retryBackoff?: RetryBackoffFunction
- - Function to calculate backoff delay (default: fixed delay)retryOnlyIdempotent?: boolean
- - Only retry commands marked as idempotent (default: true)
).$3
Get or set the hash provider used for consistent hashing distribution.$3
Get or set the timeout for operations in milliseconds (default: 5000).$3
Get or set the keepAlive setting. Updates all existing nodes. Requires reconnect() to apply changes.$3
Get or set the keep alive delay in milliseconds. Updates all existing nodes. Requires reconnect() to apply changes.$3
Get or set the number of retry attempts for failed commands (default: 0).$3
Get or set the base delay in milliseconds between retry attempts (default: 100).$3
Get or set the backoff function for calculating retry delays.$3
Get or set whether retries are restricted to idempotent commands only (default: true).Connection Management
$3
Connect to all Memcache servers or a specific node.$3
Disconnect all connections.$3
Reconnect all nodes by disconnecting and connecting them again.$3
Quit all connections gracefully.$3
Check if any node is connected to a Memcache server.Node Management
$3
Get an array of all MemcacheNode instances.$3
Get a specific node by its ID (e.g., "localhost:11211").$3
Add a new node to the cluster. Throws error if node already exists.$3
Remove a node from the cluster.$3
Get the nodes for a given key using consistent hashing. Automatically connects to nodes if not already connected.$3
Parse a URI string into host and port. Supports formats:
- Simple: "localhost:11211" or "localhost"
- Protocol: "memcache://localhost:11211", "tcp://localhost:11211"
- IPv6: "[::1]:11211" or "memcache://[2001:db8::1]:11212"
- Unix socket: "/var/run/memcached.sock" or "unix:///var/run/memcached.sock"Data Storage Operations
$3
Get a value from the Memcache server. Returns the first successful result from replica nodes.$3
Get multiple values from the Memcache server. Returns a Map with keys to values.$3
Set a value in the Memcache server. Returns true only if all replica nodes succeed.
- exptime - Expiration time in seconds (default: 0 = never expire)
- flags - Flags/metadata (default: 0)$3
Add a value (only if key doesn't exist). Returns true only if all replica nodes succeed.$3
Replace a value (only if key exists). Returns true only if all replica nodes succeed.$3
Check-And-Set: Store a value only if it hasn't been modified since last fetch. Returns true only if all replica nodes succeed.String Modification Operations
$3
Append a value to an existing key. Returns true only if all replica nodes succeed.$3
Prepend a value to an existing key. Returns true only if all replica nodes succeed.Deletion & Expiration
$3
Delete a value from the Memcache server. Returns true only if all replica nodes succeed.$3
Update expiration time without retrieving value. Returns true only if all replica nodes succeed.Numeric Operations
$3
Increment a value. Returns the new value or undefined on failure.
- value - Amount to increment (default: 1)$3
Decrement a value. Returns the new value or undefined on failure.
- value - Amount to decrement (default: 1)Server Management & Statistics
$3
Flush all values from all Memcache servers. Returns true if all nodes successfully flushed.
- delay - Optional delay in seconds before flushing$3
Get statistics from all Memcache servers. Returns a Map of node IDs to their stats.$3
Get the Memcache server version from all nodes. Returns a Map of node IDs to version strings.Validation
$3
Validates a Memcache key according to protocol requirements. Throws error if:
- Key is empty
- Key exceeds 250 characters
- Key contains spaces, newlines, or null charactersHelper Functions
$3
Factory function to create a new MemcacheNode instance.`javascript
import { createNode } from 'memcache';const node = createNode('localhost', 11211, {
timeout: 5000,
keepAlive: true,
weight: 1
});
`Hooks and Events
The Memcache client extends Hookified to provide powerful hooks and events for monitoring and customizing behavior.
Events
The client emits various events during operations that you can listen to:
`javascript
const client = new Memcache();// Connection events
client.on('connect', () => {
console.log('Connected to Memcache server');
});
client.on('close', () => {
console.log('Connection closed');
});
client.on('error', (error) => {
console.error('Error:', error);
});
client.on('timeout', () => {
console.log('Connection timeout');
});
// Cache hit/miss events
client.on('hit', (key, value) => {
console.log(
Cache hit for key: ${key});
});client.on('miss', (key) => {
console.log(
Cache miss for key: ${key});
});
`Available Events
-
connect - Emitted when connection to Memcache server is established
- close - Emitted when connection is closed
- error - Emitted when an error occurs
- timeout - Emitted when a connection timeout occurs
- hit - Emitted when a key is found in cache (includes key and value)
- miss - Emitted when a key is not found in cache
- quit - Emitted when quit command is sent
- warn - Emitted for warning messages
- info - Emitted for informational messagesHooks
Hooks allow you to intercept and modify behavior before and after operations. Every operation supports
before and after hooks.`javascript
const client = new Memcache();// Add a before hook for get operations
client.onHook('before:get', async ({ key }) => {
console.log(
Getting key: ${key});
});// Add an after hook for set operations
client.onHook('after:set', async ({ key, value, success }) => {
if (success) {
console.log(
Successfully set ${key});
}
});// Hooks can be async and modify behavior
client.onHook('before:set', async ({ key, value }) => {
console.log(
About to set ${key} = ${value});
// Perform validation, logging, etc.
});
`Available Hooks
All operations support before and after hooks with specific parameters:
get(key)
- before:get - { key }
- after:get - { key, value } (value is array or undefined)set(key, value, exptime?, flags?)
- before:set - { key, value, exptime, flags }
- after:set - { key, value, exptime, flags, success }gets(keys[])
- before:gets - { keys }
- after:gets - { keys, values } (values is a Map)add(key, value, exptime?, flags?)
- before:add - { key, value, exptime, flags }
- after:add - { key, value, exptime, flags, success }replace(key, value, exptime?, flags?)
- before:replace - { key, value, exptime, flags }
- after:replace - { key, value, exptime, flags, success }append(key, value)
- before:append - { key, value }
- after:append - { key, value, success }prepend(key, value)
- before:prepend - { key, value }
- after:prepend - { key, value, success }delete(key)
- before:delete - { key }
- after:delete - { key, success }incr(key, value?)
- before:incr - { key, value }
- after:incr - { key, value, newValue }decr(key, value?)
- before:decr - { key, value }
- after:decr - { key, value, newValue }touch(key, exptime)
- before:touch - { key, exptime }
- after:touch - { key, exptime, success }Hook Examples
`javascript
const client = new Memcache();// Log all get operations
client.onHook('before:get', async ({ key }) => {
console.log(
[GET] Fetching key: ${key});
});client.onHook('after:get', async ({ key, value }) => {
console.log(
[GET] Key: ${key}, Found: ${value !== undefined});
});// Log all set operations with timing
client.onHook('before:set', async (context) => {
context.startTime = Date.now();
});
client.onHook('after:set', async (context) => {
const duration = Date.now() - context.startTime;
console.log(
[SET] Key: ${context.key}, Success: ${context.success}, Time: ${duration}ms);
});
`Distribution Algorithms
Memcache supports pluggable distribution algorithms to determine how keys are distributed across nodes. You can configure the algorithm using the
hash option.KetamaHash (Default)
KetamaHash uses the Ketama consistent hashing algorithm, which minimizes key redistribution when nodes are added or removed. This is the default and recommended algorithm for production environments with dynamic scaling.
`javascript
import { Memcache } from 'memcache';// KetamaHash is used by default
const client = new Memcache({
nodes: ['server1:11211', 'server2:11211', 'server3:11211']
});
`Characteristics:
- Minimal key redistribution (~1/n keys move when adding/removing nodes)
- Uses virtual nodes for better distribution
- Supports weighted nodes
- Best for production environments with dynamic scaling
ModulaHash
ModulaHash uses a simple modulo-based hashing algorithm (
hash(key) % nodeCount). This is a simpler algorithm that may redistribute all keys when nodes change.`javascript
import { Memcache, ModulaHash } from 'memcache';// Use ModulaHash for distribution
const client = new Memcache({
nodes: ['server1:11211', 'server2:11211', 'server3:11211'],
hash: new ModulaHash()
});
// With a custom hash algorithm (default is sha1)
const client2 = new Memcache({
nodes: ['server1:11211', 'server2:11211'],
hash: new ModulaHash('md5')
});
`Characteristics:
- Simple and fast algorithm
- All keys may be redistributed when nodes are added or removed
- Supports weighted nodes (nodes with higher weight appear more in the distribution)
- Best for fixed-size clusters or testing environments
$3
ModulaHash supports weighted nodes, where nodes with higher weights receive proportionally more keys:
`javascript
import { Memcache, ModulaHash, createNode } from 'memcache';// Create nodes with different weights
const node1 = createNode('server1', 11211, { weight: 3 }); // 3x traffic
const node2 = createNode('server2', 11211, { weight: 1 }); // 1x traffic
const client = new Memcache({
nodes: [node1, node2],
hash: new ModulaHash()
});
// server1 will receive approximately 75% of keys
// server2 will receive approximately 25% of keys
`Choosing an Algorithm
| Feature | KetamaHash | ModulaHash |
|---------|------------|------------|
| Key redistribution on node change | Minimal (~1/n keys) | All keys may move |
| Complexity | Higher (virtual nodes) | Lower (simple modulo) |
| Performance | Slightly slower | Faster |
| Best for | Dynamic scaling | Fixed clusters |
| Weighted nodes | Yes | Yes |
Use KetamaHash (default) when:
- Your cluster size may change dynamically
- You want to minimize cache invalidation during scaling
- You're running in production
Use ModulaHash when:
- Your cluster size is fixed
- You prefer simplicity over minimal redistribution
- You're in a testing or development environment
Retry Configuration
The Memcache client supports automatic retry of failed commands with configurable backoff strategies.
Basic Retry Setup
Enable retries by setting the
retries option:`javascript
import { Memcache } from 'memcache';const client = new Memcache({
nodes: ['localhost:11211'],
retries: 3, // Retry up to 3 times
retryDelay: 100 // 100ms between retries
});
`You can also modify retry settings at runtime:
`javascript
client.retries = 5;
client.retryDelay = 200;
`Backoff Strategies
The client includes two built-in backoff functions:
$3
`javascript
import { Memcache, defaultRetryBackoff } from 'memcache';const client = new Memcache({
retries: 3,
retryDelay: 100,
retryBackoff: defaultRetryBackoff // 100ms, 100ms, 100ms
});
`$3
`javascript
import { Memcache, exponentialRetryBackoff } from 'memcache';const client = new Memcache({
retries: 3,
retryDelay: 100,
retryBackoff: exponentialRetryBackoff // 100ms, 200ms, 400ms
});
`$3
You can provide your own backoff function:
`javascript
const client = new Memcache({
retries: 3,
retryDelay: 100,
retryBackoff: (attempt, baseDelay) => {
// Exponential backoff with jitter
const delay = baseDelay * Math.pow(2, attempt);
return delay + Math.random() delay 0.1;
}
});
`The backoff function receives:
-
attempt - The current attempt number (0-indexed)
- baseDelay - The configured retryDelay valueIdempotent Safety
Important: By default, retries are only performed for commands explicitly marked as idempotent. This prevents accidental double-execution of non-idempotent operations like
incr, decr, append, and prepend.$3
If a network timeout occurs after the server applies a mutation but before the client receives the response, retrying would apply the mutation twice:
- Counter incremented twice instead of once
- Data appended twice instead of once
$3
For read operations (always safe to retry):
`javascript
// Mark read operations as idempotent
await client.execute('get mykey', nodes, { idempotent: true });
`For idempotent writes (safe to retry):
`javascript
// SET with the same value is idempotent
await client.execute('set mykey 0 0 5\r\nhello', nodes, { idempotent: true });
`Disable safety for all commands (use with caution):
`javascript
const client = new Memcache({
retries: 3,
retryOnlyIdempotent: false // Allow retries for ALL commands
});
`$3
|
retryOnlyIdempotent | idempotent flag | Retries enabled? |
|-----------------------|-------------------|------------------|
| true (default) | false (default) | No |
| true (default) | true | Yes |
| false | (any) | Yes |$3
The following methods do not use the retry mechanism and have their own error handling:
-
get() - Returns undefined on failure
- gets() - Returns partial results on node failure
- flush() - Operates directly on nodes
- stats() - Operates directly on nodes
- version() - Operates directly on nodesTo use retries with read operations, use the
execute() method directly:`javascript
const nodes = await client.getNodesByKey('mykey');
const results = await client.execute('get mykey', nodes, { idempotent: true });
`SASL Authentication
The Memcache client supports SASL (Simple Authentication and Security Layer) authentication using the PLAIN mechanism. This allows you to connect to memcached servers that require authentication.
Enabling SASL Authentication
`javascript
import { Memcache } from 'memcache';const client = new Memcache({
nodes: ['localhost:11211'],
sasl: {
username: 'myuser',
password: 'mypassword',
},
});
await client.connect();
// Client is now authenticated and ready to use
`SASL Options
The
sasl option accepts an object with the following properties:-
username: string - The username for authentication (required)
- password: string - The password for authentication (required)
- mechanism?: 'PLAIN' - The SASL mechanism to use (default: 'PLAIN')Currently, only the PLAIN mechanism is supported.
Binary Protocol Methods
Important: Memcached servers with SASL enabled (
-S flag) require the binary protocol for all operations after authentication. The standard text-based methods (client.get(), client.set(), etc.) will not work on SASL-enabled servers.Use the
binary* methods on nodes for SASL-enabled servers:`javascript
import { Memcache } from 'memcache';const client = new Memcache({
nodes: ['localhost:11211'],
sasl: { username: 'user', password: 'pass' },
});
await client.connect();
// Access the node directly for binary operations
const node = client.nodes[0];
// Binary protocol operations
await node.binarySet('mykey', 'myvalue', 3600); // Set with 1 hour expiry
const value = await node.binaryGet('mykey'); // Get value
await node.binaryDelete('mykey'); // Delete key
// Other binary operations
await node.binaryAdd('newkey', 'value'); // Add (only if not exists)
await node.binaryReplace('existingkey', 'newvalue'); // Replace (only if exists)
await node.binaryIncr('counter', 1); // Increment
await node.binaryDecr('counter', 1); // Decrement
await node.binaryAppend('mykey', '-suffix'); // Append to value
await node.binaryPrepend('mykey', 'prefix-'); // Prepend to value
await node.binaryTouch('mykey', 7200); // Update expiration
await node.binaryFlush(); // Flush all
const version = await node.binaryVersion(); // Get server version
const stats = await node.binaryStats(); // Get server stats
`Per-Node SASL Configuration
You can also configure SASL credentials when creating individual nodes:
`javascript
import { createNode } from 'memcache';// Create a node with SASL credentials
const node = createNode('localhost', 11211, {
sasl: { username: 'user', password: 'pass' },
});
// Connect and use binary methods
await node.connect();
await node.binarySet('mykey', 'hello');
const value = await node.binaryGet('mykey');
`Authentication Events
You can listen for authentication events on both nodes and the client:
`javascript
import { Memcache, MemcacheNode } from 'memcache';// Node-level events
const node = new MemcacheNode('localhost', 11211, {
sasl: { username: 'user', password: 'pass' },
});
node.on('authenticated', () => {
console.log('Node authenticated successfully');
});
node.on('error', (error) => {
if (error.message.includes('SASL authentication failed')) {
console.error('Authentication failed:', error.message);
}
});
await node.connect();
// Client-level events (forwarded from nodes)
const client = new Memcache({
nodes: ['localhost:11211'],
sasl: { username: 'user', password: 'pass' },
});
client.on('authenticated', () => {
console.log('Client authenticated');
});
await client.connect();
`$3
-
node.hasSaslCredentials - Returns true if SASL credentials are configured
- node.isAuthenticated - Returns true if the node has successfully authenticatedServer Configuration
To use SASL authentication, your memcached server must be configured with SASL support:
1. Build memcached with SASL support - Ensure memcached was compiled with
--enable-sasl2. Create SASL users - Use
saslpasswd2 to create users:
`bash
saslpasswd2 -a memcached -c username
`3. Configure SASL mechanism - Create
/etc/sasl2/memcached.conf:
`
mech_list: plain
`4. Start memcached with SASL - Use the
-S flag:
`bash
memcached -S -m 64 -p 11211
``For more details, see the memcached SASL documentation.
Please read our Contributing Guidelines and also our Code of Conduct.