redis-semaphore
Version:
Distributed mutex and semaphore based on Redis
49 lines (44 loc) • 1.36 kB
text/typescript
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
}