UNPKG

redis-semaphore

Version:

Distributed mutex and semaphore based on Redis

49 lines (44 loc) 1.36 kB
import createDebug from 'debug' import Redis from 'ioredis' import { acquireLua } from '../semaphore/acquire/lua' import { delay } from '../utils/index' import { getQuorum, smartSum } from '../utils/redlock' const debug = createDebug('redis-semaphore:redlock-semaphore:acquire') export interface Options { identifier: string lockTimeout: number acquireTimeout: number retryInterval: number } export async function acquireRedlockSemaphore( clients: Redis[], key: string, limit: number, options: Options ) { const { identifier, lockTimeout, acquireTimeout, retryInterval } = options const end = Date.now() + acquireTimeout const quorum = getQuorum(clients.length) let now: number while ((now = Date.now()) < end) { debug(key, identifier, 'attempt') const promises = clients.map(client => acquireLua(client, [key, limit, identifier, lockTimeout, now]) .then(result => +result) .catch(() => 0) ) const results = await Promise.all(promises) if (results.reduce(smartSum, 0) >= quorum) { debug(key, identifier, 'acquired') return true } else { const promises = clients.map(client => client.zrem(key, identifier).catch(() => 0) ) await Promise.all(promises) await delay(retryInterval) } } debug(key, identifier, 'timeout') return false }