-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
👩💻 dx: First draft for
defer
utility.
- Loading branch information
1 parent
3d14f2e
commit 1aa42bc
Showing
3 changed files
with
314 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,233 @@ | ||
import {assert} from 'chai'; | ||
|
||
import {isNode, isomorphic} from '../../_test/fixtures'; | ||
|
||
import {cancelAll, defer, flushAll} from './defer'; | ||
import sleep from './sleep'; | ||
|
||
isomorphic(__filename, () => { | ||
it('should queue to macrotask queue', async () => { | ||
const x: number[] = []; | ||
|
||
defer(() => x.push(1)); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
await Promise.resolve().then(() => { | ||
assert.deepEqual(x, []); | ||
}); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, [1]); | ||
}); | ||
|
||
it('should allow cancellation', async () => { | ||
const x: number[] = []; | ||
|
||
const deferred = defer(() => x.push(1)); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
await Promise.resolve().then(() => { | ||
assert.deepEqual(x, []); | ||
}); | ||
|
||
deferred.cancel(); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, []); | ||
}); | ||
|
||
it('should allow flushing before microtask queue', async () => { | ||
const x: number[] = []; | ||
|
||
const deferred = defer(() => x.push(1)); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
deferred.flush(); | ||
|
||
await Promise.resolve().then(() => { | ||
assert.deepEqual(x, [1]); | ||
}); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, [1]); | ||
}); | ||
|
||
it('should flush after main loop', async () => { | ||
const x: number[] = []; | ||
|
||
const deferred = defer(() => x.push(1)); | ||
|
||
deferred.flush(); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
await Promise.resolve().then(() => { | ||
assert.deepEqual(x, [1]); | ||
}); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, [1]); | ||
}); | ||
|
||
it('should catch errors', async () => { | ||
const x: number[] = []; | ||
|
||
defer(() => { | ||
x.push(1); | ||
throw new Error('test'); | ||
}); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, [1]); | ||
}); | ||
|
||
it('should catch errors when flushing', async () => { | ||
const x: number[] = []; | ||
|
||
const deferred = defer(() => { | ||
x.push(1); | ||
throw new Error('test'); | ||
}); | ||
|
||
deferred.flush(); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, [1]); | ||
}); | ||
|
||
it('should allow cancellation of all deferred computations', async () => { | ||
const x: number[] = []; | ||
|
||
defer(() => x.push(1)); | ||
defer(() => x.push(2)); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
await Promise.resolve().then(() => { | ||
assert.deepEqual(x, []); | ||
}); | ||
|
||
cancelAll(); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, []); | ||
}); | ||
|
||
it('should allow flushing all deferred computations before microtask queue', async () => { | ||
const x: number[] = []; | ||
|
||
defer(() => x.push(1)); | ||
defer(() => x.push(2)); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
flushAll(); | ||
|
||
await Promise.resolve().then(() => { | ||
assert.deepEqual(x, [1, 2]); | ||
}); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, [1, 2]); | ||
}); | ||
|
||
it('should flush all after main loop', async () => { | ||
const x: number[] = []; | ||
|
||
defer(() => x.push(1)); | ||
defer(() => x.push(2)); | ||
|
||
flushAll(); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
await Promise.resolve().then(() => { | ||
assert.deepEqual(x, [1, 2]); | ||
}); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, [1, 2]); | ||
}); | ||
|
||
it('should execute in order', async () => { | ||
const x: number[] = []; | ||
|
||
defer(() => { | ||
x.push(1); | ||
}); | ||
defer(() => { | ||
x.push(2); | ||
}); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
await Promise.resolve().then(() => { | ||
assert.deepEqual(x, []); | ||
}); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, [1, 2]); | ||
}); | ||
|
||
it('should respect timeout', async () => { | ||
const x: number[] = []; | ||
|
||
const delay = isNode() ? 5 : 1; | ||
|
||
defer(() => { | ||
x.push(1); | ||
}, delay); | ||
defer(() => { | ||
x.push(2); | ||
}); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
await Promise.all([ | ||
sleep(delay).then(() => { | ||
assert.deepEqual(x, [2, 1]); | ||
}), | ||
sleep(0).then(() => { | ||
assert.deepEqual(x, [2]); | ||
}), | ||
Promise.resolve().then(() => { | ||
assert.deepEqual(x, []); | ||
}), | ||
]); | ||
}); | ||
|
||
it('should allow passing arguments', async () => { | ||
const x: number[] = []; | ||
defer( | ||
(a, b) => { | ||
x.push(a, b); | ||
}, | ||
0, | ||
1, | ||
2, | ||
); | ||
|
||
assert.deepEqual(x, []); | ||
|
||
await Promise.resolve().then(() => { | ||
assert.deepEqual(x, []); | ||
}); | ||
|
||
await sleep(0); | ||
|
||
assert.deepEqual(x, [1, 2]); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,79 @@ | ||
import type Timeout from '../types/Timeout'; | ||
|
||
import createPromise from './createPromise'; | ||
|
||
type Resolve = (value?: any) => void; | ||
type Reject = (reason?: any) => void; | ||
|
||
type Callback<A extends any[]> = (...args: A) => void; | ||
|
||
const _pending = new Set<Deferred>(); | ||
|
||
export class Deferred { | ||
#timeout: Timeout; | ||
#resolve: Resolve; | ||
#reject: Reject; | ||
|
||
constructor(timeout: Timeout, resolve: Resolve, reject: Reject) { | ||
this.#timeout = timeout; | ||
this.#resolve = resolve; | ||
this.#reject = reject; | ||
} | ||
|
||
cancel() { | ||
if (!_pending.has(this)) return; | ||
_pending.delete(this); | ||
clearTimeout(this.#timeout); | ||
this.#reject(); | ||
} | ||
|
||
flush() { | ||
if (!_pending.has(this)) return; | ||
_pending.delete(this); | ||
clearTimeout(this.#timeout); | ||
this.#resolve(); | ||
} | ||
} | ||
|
||
export const defer = <A extends any[]>( | ||
callback: Callback<A>, | ||
timeout?: number, | ||
...args: A | ||
): Deferred => { | ||
const {promise, resolve, reject} = createPromise(); | ||
promise | ||
.then( | ||
() => { | ||
_pending.delete(deferred); | ||
callback(...args); | ||
}, | ||
|
||
() => { | ||
// NOTE This handles cancellation. | ||
}, | ||
) | ||
.catch((error: unknown) => { | ||
console.error({error}); | ||
}); | ||
const deferred = new Deferred(setTimeout(resolve, timeout), resolve, reject); | ||
_pending.add(deferred); | ||
return deferred; | ||
}; | ||
|
||
const _cancelAll = (pending: Iterable<Deferred>) => { | ||
for (const deferred of pending) deferred.cancel(); | ||
}; | ||
|
||
export const cancelAll = () => { | ||
_cancelAll(_pending); | ||
_pending.clear(); | ||
}; | ||
|
||
const _flushAll = (pending: Iterable<Deferred>) => { | ||
for (const deferred of pending) deferred.flush(); | ||
}; | ||
|
||
export const flushAll = () => { | ||
_flushAll(_pending); | ||
_pending.clear(); | ||
}; |