42 lines
1.1 KiB
JavaScript
42 lines
1.1 KiB
JavaScript
|
import semaphore from 'semaphore';
|
||
|
|
||
|
export const asyncLock = () => {
|
||
|
let lock = semaphore(1);
|
||
|
|
||
|
const acquire = (timeout = 15000) => {
|
||
|
const promise = new Promise(resolve => {
|
||
|
// this makes sure a caller doesn't gets stuck forever awaiting on the lock
|
||
|
const timeoutId = setTimeout(() => {
|
||
|
// we reset the lock in that case to allow future consumers to use it without being blocked
|
||
|
lock = semaphore(1);
|
||
|
resolve(false);
|
||
|
}, timeout);
|
||
|
|
||
|
lock.take(() => {
|
||
|
clearTimeout(timeoutId);
|
||
|
resolve(true);
|
||
|
});
|
||
|
});
|
||
|
|
||
|
return promise;
|
||
|
};
|
||
|
|
||
|
const release = () => {
|
||
|
try {
|
||
|
// suppress too many calls to leave error
|
||
|
lock.leave();
|
||
|
} catch (e) {
|
||
|
// calling 'leave' too many times might not be good behavior
|
||
|
// but there is no reason to completely fail on it
|
||
|
if (e.message !== 'leave called too many times.') {
|
||
|
throw e;
|
||
|
} else {
|
||
|
console.warn('leave called too many times.');
|
||
|
lock = semaphore(1);
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
return { acquire, release };
|
||
|
};
|