npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

multiredlock

v0.2.0

Published

A distributed lock algorithm for redis, see http://redis.io/topics/distlock

Downloads

9

Readme

multiredlock Build Status

A distributed lock algorithm for redis, see http://redis.io/topics/distlock

This is experimental software, use with your own risk.

Install

npm install multiredlock

Usage

Simple example, which can be found in example.js:

    var Redlock = require('multiredlock');
    
    var redlock = new Redlock([{host:'localhost', port:6379}]);
    
    redlock.on('connect', function() { 
      // Let's lock resource 'console' for 10 seconds!
      redlock.lock('console', 10000, function(err, lock) {
        if(err) {
          console.log(err);
          return;
        }
    
        // Do some stuff with the resource...
        console.log(lock);
    
        // Release lock when you're done with the resource
        redlock.unlock('console', lock.value);
      });
    });

Will output something like:

$ node example.js 
{ validity: 9899,
  resource: 'console',
  value: 'yourhost_1421844817874_6f554e61' }

Note: you must have redis-server running on localhost at port 6379.

API

Events

  • connect: emitted when multiredlock has connected to at least N/2 + 1 redis-servers, where N is the total amount of servers supplied to multiredlock. In other words, connect will be emitted when multiredlock is able to acquire locks.
  • disconnect: emitted when multiredlock is connected to less than N/2 + 1 redis-servers, meaning that it will not be able to acquire locks. However, unlocking currently locked resources is possible.

Variables

  • connected: true, when connected to at least N/2 + 1 redis-servers, false otherwise.

Constructor

The constructor takes two arguments, an array servers and an object options:

  • new Redlock() = new Redlock([{host:'localhost', port:6379}]);
  • new Redlock([{host:'192.168.0.100', port:6380}, {host:'example.com', port:6379}, {host:'localhost', port:6381}], options); will connect to three redis-servers residing at 192.168.0.100, example.com and localhost.

One may omit the port-property from the server object. Multiredlock will then will assume port number 6379.

An optional auth value may be passed in the server object to connect to a password protected Redis instance. e.g. auth: 'foobared' will connect with the password foobarded/

The object options may have the following properties:

  • id: an optional ID string to give for this instance of multiredlock. Is used in identifying locks. Defaults to the machine's hostname.
  • drift: set the maximum clock drift between servers in milliseconds. This will be subtracted from lock validity time. The default is 100ms.
  • debug: run multiredlock in debug mode. Boolean.

setRetry(retries, retryWait)

Sets the amount of retries and the maximum time to wait between retries when acquiring a lock. The actual time to retry will be chosen randomly between 0 and retryWait milliseconds before each retry. The default is 3 retries with 100ms maximum wait. This means that multiredlock will try to acquire a lock a total of 4 times before giving up. Setting retries to 0 will disable retrying.

lock(resource, ttl, callback)

Will try to acquire a lock for resource. The lock will be acquired if N/2 + 1 redis-servers approve of the locking attempt (in other words, they don't think the resource is already locked). Will retry acquiring lock according to retry policies if it is not granted at first time, see setRetry().

  • resource: the name of the resource to lock, arbitrary but must be a string.
  • ttl: Time To Live, the amount of milliseconds before the lock is released, should the resource not be unlocked with unlock before that.
  • callback: will be called with callback(err, lock), where:
    • err: error message if the lock was not acquired. null if lock was acquired.
    • lock: an object detailing the acquired lock. Has the following properties:
      • validity: how long the lock is valid, in milliseconds.
      • resource: the name of the resource that was locked.
      • value: unique id for the acquired lock. Needed when unlocking a resource or renewing the lock.

renew(resource, value, ttl, callback)

Sets a new TTL for an existing lock.

  • resource: name of the locked resource
  • value: value of the lock to renew. Can be found in the lock object provided by lock().
  • ttl: new Time To Live value for the lock in milliseconds.
  • callback: will be called with callback(err, lock), where:
    • err: error message if the lock was not renewed. null if lock was renewed.
    • lock: an object detailing the renewed lock. Has the following properties:
      • validity: how long the lock is valid, in milliseconds.
      • resource: the name of the resource that had its lock renewed.
      • value: same as the value of the lock before renewing.

Example:

    redlock.lock('test', 200, function(err, lock) {
        // Do some stuff with resource 'test'...
        // Realize that this stuff is going to take longer than 200ms
        redlock.renew('test', lock.value, 200, function(err, lock) {
            // If no errors occured, lock.validity is now a little less than 200ms
        }
    }

unlock(resource, value)

Unlocks a resource.

  • resource: the name of the resource to unlock.
  • value: the unique id of the lock to release. See lock().

Notice: unlock does not provide a callback, for unlocking attempts are best-effort, and there is no guarantee that the resource will be unlocked.

close()

Disconnect from all redis-servers.

This maybe useful for releasing the event loop for letting your program exit. Note that this effectively makes this multiredlock instance unusable.

Tests

This project has integration and failover tests implemented with Vagrant and Docker. The tests simulate a production environment where redis servers may crash.

To run these tests, your CPU needs to support VT-x, and you need at least 4GB memory. You also must have VirtualBox and Vagrant installed.

$ vagrant up
  (be patient...)
$ vagrant ssh

redlock:~$ cd redlock
redlock:~$ npm install
redlock:~$ npm test

If you wish to run only unit tests, you can do so by issuing mocha test/unit.js in the project root.


Special thanks to amv for sharing his thoughts on this project, and ordinary thanks to virta.