1 | ;
|
2 | Object.defineProperty(exports, "__esModule", { value: true });
|
3 | const ava_1 = require("ava");
|
4 | const uuid_1 = require("uuid");
|
5 | const cache_1 = require("../src/cache");
|
6 | const throttle_1 = require("../src/throttle");
|
7 | const functions_1 = require("./fixtures/functions");
|
8 | const util_1 = require("./fixtures/util");
|
9 | (0, ava_1.default)("deferred resolves its promise", async (t) => {
|
10 | const deferred = new throttle_1.Deferred();
|
11 | let resolved = false;
|
12 | deferred.promise.then(_ => (resolved = true));
|
13 | t.is(resolved, false);
|
14 | deferred.resolve();
|
15 | await deferred.promise;
|
16 | t.is(resolved, true);
|
17 | });
|
18 | (0, ava_1.default)("deferred rejects its promise", async (t) => {
|
19 | const deferred = new throttle_1.Deferred();
|
20 | let rejected = false;
|
21 | t.is(rejected, false);
|
22 | deferred.reject();
|
23 | try {
|
24 | await deferred.promise;
|
25 | }
|
26 | catch (_) {
|
27 | rejected = true;
|
28 | }
|
29 | t.is(rejected, true);
|
30 | });
|
31 | (0, ava_1.default)("deferred resolves only once", async (t) => {
|
32 | const deferred = new throttle_1.Deferred();
|
33 | let value = 0;
|
34 | deferred.promise.then(_ => value++);
|
35 | deferred.resolve();
|
36 | await deferred.promise;
|
37 | t.is(value, 1);
|
38 | deferred.resolve();
|
39 | await deferred.promise;
|
40 | t.is(value, 1);
|
41 | });
|
42 | (0, ava_1.default)("deferred cannot reject after resolving", async (t) => {
|
43 | const deferred = new throttle_1.Deferred();
|
44 | let value = 0;
|
45 | deferred.promise.then(_ => value++);
|
46 | deferred.resolve();
|
47 | await deferred.promise;
|
48 | t.is(value, 1);
|
49 | deferred.reject();
|
50 | await deferred.promise;
|
51 | t.is(value, 1);
|
52 | });
|
53 | ava_1.default.serial("funnel defaults to infinite concurrency (tested with 200)", t => (0, util_1.withClock)(async () => {
|
54 | const funnel = new throttle_1.Funnel(0);
|
55 | const promises = [];
|
56 | const N = 200;
|
57 | for (let i = 0; i < N; i++) {
|
58 | promises.push(funnel.push(() => (0, functions_1.timer)(300)));
|
59 | }
|
60 | const times = await Promise.all(promises);
|
61 | t.is((0, util_1.measureConcurrency)(times), N);
|
62 | }));
|
63 | ava_1.default.serial("funnel single concurrency is mutually exclusive", t => (0, util_1.withClock)(async () => {
|
64 | const funnel = new throttle_1.Funnel(1);
|
65 | const promises = [];
|
66 | const N = 10;
|
67 | for (let i = 0; i < N; i++) {
|
68 | promises.push(funnel.push(() => (0, functions_1.timer)(10)));
|
69 | }
|
70 | const times = await Promise.all(promises);
|
71 | t.is((0, util_1.measureConcurrency)(times), 1);
|
72 | }));
|
73 | ava_1.default.serial("funnel handles concurrency level 2", t => (0, util_1.withClock)(async () => {
|
74 | const funnel = new throttle_1.Funnel(2);
|
75 | const promises = [];
|
76 | const N = 10;
|
77 | for (let i = 0; i < N; i++) {
|
78 | promises.push(funnel.push(() => (0, functions_1.timer)(20)));
|
79 | }
|
80 | const times = await Promise.all(promises);
|
81 | t.is((0, util_1.measureConcurrency)(times), 2);
|
82 | }));
|
83 | ava_1.default.serial("funnel handles concurrency level 10", t => (0, util_1.withClock)(async () => {
|
84 | const funnel = new throttle_1.Funnel(10);
|
85 | const promises = [];
|
86 | const N = 100;
|
87 | for (let i = 0; i < N; i++) {
|
88 | promises.push(funnel.push(() => (0, functions_1.timer)(20)));
|
89 | }
|
90 | const times = await Promise.all(promises);
|
91 | t.is((0, util_1.measureConcurrency)(times), 10);
|
92 | }));
|
93 | ava_1.default.serial("funnel resumes after finishing a worker", t => (0, util_1.withClock)(async () => {
|
94 | const funnel = new throttle_1.Funnel(1);
|
95 | const time1 = await funnel.push(() => (0, functions_1.timer)(10));
|
96 | const time2 = await funnel.push(() => (0, functions_1.timer)(10));
|
97 | t.is((0, util_1.measureConcurrency)([time1, time2]), 1);
|
98 | }));
|
99 | (0, ava_1.default)("funnel clearing", async (t) => {
|
100 | const funnel = new throttle_1.Funnel(1);
|
101 | let count = 0;
|
102 | const promise0 = funnel.push(async () => count++);
|
103 | const promise1 = funnel.push(async () => count++);
|
104 | const promise2 = funnel.push(async () => count++);
|
105 | funnel.clear();
|
106 | t.is(await Promise.race([promise0, promise1, promise2, (0, util_1.sleep)(100).then(_ => "done")]), "done");
|
107 | t.is(count, 0);
|
108 | });
|
109 | (0, ava_1.default)("funnel gets executed asynchronously, not at the moment of push", async (t) => {
|
110 | const funnel = new throttle_1.Funnel(1);
|
111 | let n = 0;
|
112 | funnel.push(async () => {
|
113 | n++;
|
114 | });
|
115 | t.is(n, 0);
|
116 | await funnel.all();
|
117 | t.is(n, 1);
|
118 | });
|
119 | (0, ava_1.default)("funnel handles promise rejections without losing concurrency", async (t) => {
|
120 | const funnel = new throttle_1.Funnel(1);
|
121 | let executed = false;
|
122 | await t.throwsAsync(funnel.push(() => Promise.reject(new Error("message"))), { message: "message" });
|
123 | await funnel.push(async () => {
|
124 | executed = true;
|
125 | });
|
126 | t.is(executed, true);
|
127 | });
|
128 | (0, ava_1.default)("funnel.all() waits for all requests to finish", async (t) => {
|
129 | const funnel = new throttle_1.Funnel(1);
|
130 | let executed = false;
|
131 | funnel.push(async () => {
|
132 | await (0, util_1.sleep)(200);
|
133 | executed = true;
|
134 | return "first";
|
135 | });
|
136 | funnel.push(async () => "second");
|
137 | t.is(executed, false);
|
138 | const result = await funnel.all();
|
139 | t.is(result.length, 2);
|
140 | t.is(result[0], "first");
|
141 | t.is(result[1], "second");
|
142 | t.is(executed, true);
|
143 | });
|
144 | (0, ava_1.default)("funnel.all() ignores errors and waits for other requests to finish", async (t) => {
|
145 | const funnel = new throttle_1.Funnel(1);
|
146 | funnel.push(async () => {
|
147 | throw new Error();
|
148 | });
|
149 | funnel.push(async () => {
|
150 | await (0, util_1.sleep)(100);
|
151 | return "done";
|
152 | });
|
153 | const result = await funnel.all();
|
154 | t.is(result.length, 2);
|
155 | t.falsy(result[0]);
|
156 | t.is(result[1], "done");
|
157 | });
|
158 | (0, ava_1.default)("retryOp() retries failures", async (t) => {
|
159 | let attempts = 0;
|
160 | await (0, throttle_1.retryOp)(2, async () => {
|
161 | attempts++;
|
162 | throw new Error();
|
163 | }).catch(_ => { });
|
164 | t.is(attempts, 3);
|
165 | });
|
166 | (0, ava_1.default)("funnel shouldRetry parameter retries failures", async (t) => {
|
167 | const funnel = new throttle_1.Funnel(1, 2);
|
168 | let attempts = 0;
|
169 | let errors = 0;
|
170 | funnel
|
171 | .push(async () => {
|
172 | attempts++;
|
173 | throw Error();
|
174 | })
|
175 | .catch(_ => errors++);
|
176 | await funnel.all();
|
177 | t.is(attempts, 3);
|
178 | t.is(errors, 1);
|
179 | });
|
180 | (0, ava_1.default)("funnel cancellation", async (t) => {
|
181 | const funnel = new throttle_1.Funnel(1);
|
182 | let executed = 0;
|
183 | const promise = funnel.push(async () => {
|
184 | executed++;
|
185 | }, 0, () => "cancelled");
|
186 | await t.throwsAsync(promise);
|
187 | t.is(executed, 0);
|
188 | });
|
189 | (0, ava_1.default)("funnel processed and error counts", async (t) => {
|
190 | const funnel = new throttle_1.Funnel(2);
|
191 | funnel.push(async () => { });
|
192 | funnel.push(async () => Promise.reject());
|
193 | funnel.push(async () => { });
|
194 | funnel.push(async () => Promise.reject());
|
195 | funnel.push(async () => { });
|
196 | await funnel.all();
|
197 | t.is(funnel.processed, 3);
|
198 | t.is(funnel.errors, 2);
|
199 | });
|
200 | ava_1.default.serial("pump works for concurrency level 1", t => (0, util_1.withClock)(async () => {
|
201 | let executed = 0;
|
202 | const pump = new throttle_1.Pump({ concurrency: 1 }, () => {
|
203 | executed++;
|
204 | return (0, util_1.sleep)(100);
|
205 | });
|
206 | t.is(executed, 0);
|
207 | pump.start();
|
208 | await (0, util_1.sleep)(300);
|
209 | pump.stop();
|
210 | t.true(executed === 3);
|
211 | }));
|
212 | ava_1.default.serial("pump works for concurrency level 10", t => (0, util_1.withClock)(async () => {
|
213 | let executed = 0;
|
214 | const pump = new throttle_1.Pump({ concurrency: 10 }, () => {
|
215 | executed++;
|
216 | return (0, util_1.sleep)(100);
|
217 | });
|
218 | pump.start();
|
219 | await (0, util_1.sleep)(100);
|
220 | pump.stop();
|
221 | t.is(executed, 10);
|
222 | }));
|
223 | ava_1.default.serial("pump handles promise rejections without losing concurrency", t => (0, util_1.withClock)(async () => {
|
224 | let executed = 0;
|
225 | const pump = new throttle_1.Pump({ concurrency: 1, verbose: false }, () => {
|
226 | executed++;
|
227 | return (0, util_1.sleep)(100).then(_ => Promise.reject("hi"));
|
228 | });
|
229 | pump.start();
|
230 | await (0, util_1.sleep)(500);
|
231 | pump.stop();
|
232 | t.is(executed, 5);
|
233 | }));
|
234 | ava_1.default.serial("pump drain", t => (0, util_1.withClock)(async () => {
|
235 | let started = 0;
|
236 | let finished = 0;
|
237 | const N = 5;
|
238 | const pump = new throttle_1.Pump({ concurrency: N }, async () => {
|
239 | started++;
|
240 | await (0, util_1.sleep)(100);
|
241 | finished++;
|
242 | });
|
243 | t.is(started, 0);
|
244 | t.is(finished, 0);
|
245 | pump.start();
|
246 | await pump.drain();
|
247 | t.is(started, N);
|
248 | t.is(finished, N);
|
249 | }));
|
250 | ava_1.default.serial("memoize returns cached results for the same key", t => (0, util_1.withClock)(async () => {
|
251 | const promises = [];
|
252 | const N = 10;
|
253 | const timerFn = (0, throttle_1.throttle)({ memoize: true, concurrency: 1, rate: 10 }, _ => (0, functions_1.timer)(10));
|
254 | for (let i = 0; i < N; i++) {
|
255 | promises.push(timerFn("key"));
|
256 | }
|
257 | const times = await Promise.all(promises);
|
258 | t.is((0, util_1.measureConcurrency)(times), N);
|
259 | }));
|
260 | ava_1.default.serial("memoize runs the worker for different keys", t => (0, util_1.withClock)(async () => {
|
261 | const promises = [];
|
262 | const N = 10;
|
263 | const timerFn = (0, throttle_1.throttle)({ memoize: true, concurrency: 1, rate: 10 }, _ => (0, functions_1.timer)(10));
|
264 | for (let i = 0; i < N; i++) {
|
265 | promises.push(timerFn(i));
|
266 | }
|
267 | const times = await Promise.all(promises);
|
268 | t.is((0, util_1.measureConcurrency)(times), 1);
|
269 | }));
|
270 | async function withCache(fn) {
|
271 | const nonce = (0, uuid_1.v4)();
|
272 | const cache = new cache_1.PersistentCache(`.faast/test/${nonce}`);
|
273 | await fn(cache).catch(console.error);
|
274 | await cache.clear({ leaveEmptyDir: false });
|
275 | }
|
276 | (0, ava_1.default)("caching saves values and skips re-execution", t => withCache(async (cache) => {
|
277 | let counter = 0;
|
278 | function fn(_) {
|
279 | return Promise.resolve(counter++);
|
280 | }
|
281 | const mfn = (0, throttle_1.cacheFn)(cache, fn);
|
282 | await mfn(0);
|
283 | await mfn(7);
|
284 | await mfn(0);
|
285 | t.is(counter, 2);
|
286 | const mfn2 = (0, throttle_1.cacheFn)(cache, fn);
|
287 | await mfn2(0);
|
288 | await mfn2(7);
|
289 | await mfn2(0);
|
290 | await mfn2(10);
|
291 | t.is(counter, 3);
|
292 | }));
|
293 | (0, ava_1.default)("cache works with string arguments", async (t) => withCache(async (cache) => {
|
294 | let counter = 0;
|
295 | function fn(_) {
|
296 | return Promise.resolve(counter++);
|
297 | }
|
298 | const mfn = (0, throttle_1.cacheFn)(cache, fn);
|
299 | await mfn("a");
|
300 | await mfn("b");
|
301 | await mfn("a");
|
302 | t.is(counter, 2);
|
303 | }));
|
304 | (0, ava_1.default)("cache works with object arguments", async (t) => withCache(async (cache) => {
|
305 | let counter = 0;
|
306 | function fn(_) {
|
307 | return Promise.resolve(counter++);
|
308 | }
|
309 | const mfn = (0, throttle_1.cacheFn)(cache, fn);
|
310 | await mfn({ f: "field", i: 42 });
|
311 | await mfn({ f: "field", i: 1 });
|
312 | await mfn({ f: "other", i: 42 });
|
313 | await mfn({ f: "field", i: 42 });
|
314 | t.is(counter, 3);
|
315 | }));
|
316 | (0, ava_1.default)("cache does not save rejected promises from cached function", async (t) => withCache(async (cache) => {
|
317 | let counter = 0;
|
318 | function fn(_) {
|
319 | counter++;
|
320 | return Promise.reject(new Error("rejection"));
|
321 | }
|
322 | let caught = 0;
|
323 | const mfn = (0, throttle_1.cacheFn)(cache, fn);
|
324 | await mfn(1).catch(_ => caught++);
|
325 | await mfn(2).catch(_ => caught++);
|
326 | await mfn(1).catch(_ => caught++);
|
327 | t.is(counter, 3);
|
328 | t.is(caught, 3);
|
329 | }));
|
330 | function measureMaxRequestRatePerSecond(timings) {
|
331 | const requestsPerSecondStartingAt = timings
|
332 | .map(t => t.start)
|
333 | .map(t => timings.filter(({ start }) => start >= t && start < t + 1000).length);
|
334 | return Math.max(...requestsPerSecondStartingAt);
|
335 | }
|
336 | ava_1.default.serial("rate limiter restricts max request rate per second", t => (0, util_1.withClock)(async () => {
|
337 | const requestRate = 10;
|
338 | const rateLimiter = new throttle_1.RateLimiter(requestRate);
|
339 | const promises = [];
|
340 | for (let i = 0; i < 15; i++) {
|
341 | promises.push(rateLimiter.push(() => (0, functions_1.timer)(0)));
|
342 | }
|
343 | const timings = await Promise.all(promises);
|
344 | t.is(measureMaxRequestRatePerSecond(timings), requestRate);
|
345 | }));
|
346 | ava_1.default.serial("rate limiter works across second boundaries", t => (0, util_1.withClock)(async () => {
|
347 | const requestRate = 10;
|
348 | const rateLimiter = new throttle_1.RateLimiter(requestRate);
|
349 | const promises = [];
|
350 | promises.push(rateLimiter.push(() => (0, functions_1.timer)(0)));
|
351 | await (0, util_1.sleep)(900);
|
352 | for (let i = 0; i < 15; i++) {
|
353 | promises.push(rateLimiter.push(() => (0, functions_1.timer)(0)));
|
354 | }
|
355 | const timings = await Promise.all(promises);
|
356 | t.is(measureMaxRequestRatePerSecond(timings), requestRate);
|
357 | }));
|
358 | ava_1.default.serial("rate limiter bursting allows for request rate beyond target rate", t => (0, util_1.withClock)(async () => {
|
359 | const requestRate = 10;
|
360 | const maxBurst = 5;
|
361 | const rateLimiter = new throttle_1.RateLimiter(requestRate, maxBurst);
|
362 | const promises = [];
|
363 | for (let i = 0; i < 15; i++) {
|
364 | promises.push(rateLimiter.push(() => (0, functions_1.timer)(0)));
|
365 | }
|
366 | const timings = await Promise.all(promises);
|
367 | const maxRate = measureMaxRequestRatePerSecond(timings);
|
368 | t.true(maxRate <= maxBurst + requestRate);
|
369 | t.true(maxRate > maxBurst);
|
370 | }));
|
371 | ava_1.default.serial("throttle limits max concurrency and rate", t => (0, util_1.withClock)(async () => {
|
372 | const concurrency = 10;
|
373 | const rate = 10;
|
374 | const timerFn = (0, throttle_1.throttle)({ concurrency, rate }, functions_1.timer);
|
375 | const promises = [];
|
376 | for (let i = 0; i < 15; i++) {
|
377 | promises.push(timerFn(1000));
|
378 | }
|
379 | const times = await Promise.all(promises);
|
380 | t.is((0, util_1.measureConcurrency)(times), concurrency);
|
381 | t.is(measureMaxRequestRatePerSecond(times), rate);
|
382 | }));
|
383 | ava_1.default.serial("throttle limits rate with single concurrency", t => (0, util_1.withClock)(async () => {
|
384 | const concurrency = 1;
|
385 | const rate = 10;
|
386 | const processTimeMs = 200;
|
387 | const timerFn = (0, throttle_1.throttle)({ concurrency, rate }, functions_1.timer);
|
388 | const promises = [];
|
389 | for (let i = 0; i < 10; i++) {
|
390 | promises.push(timerFn(processTimeMs));
|
391 | }
|
392 | const times = await Promise.all(promises);
|
393 | t.is((0, util_1.measureConcurrency)(times), concurrency);
|
394 | t.true(measureMaxRequestRatePerSecond(times) <= 1000 / processTimeMs + 1);
|
395 | }));
|
396 | (0, ava_1.default)("throttle memoize option", async (t) => {
|
397 | const concurrency = 1;
|
398 | const rate = 100;
|
399 | let counter = 0;
|
400 | const N = 5;
|
401 | async function fn(_) {
|
402 | counter++;
|
403 | }
|
404 | const throttledFn = (0, throttle_1.throttle)({ concurrency, rate, memoize: true }, fn);
|
405 | const promises = [];
|
406 | for (let i = 0; i < N; i++) {
|
407 | promises.push(throttledFn(i));
|
408 | }
|
409 | for (let i = 0; i < N; i++) {
|
410 | promises.push(throttledFn(i));
|
411 | }
|
412 | await Promise.all(promises);
|
413 | t.is(counter, N);
|
414 | });
|
415 | (0, ava_1.default)("throttle cache option persists values", async (t) => withCache(async (cache) => {
|
416 | const concurrency = 1;
|
417 | const rate = 100;
|
418 | let counter = 0;
|
419 | async function fn(_) {
|
420 | return counter++;
|
421 | }
|
422 | const throttledFn = (0, throttle_1.throttle)({ concurrency, rate, cache }, fn);
|
423 | const v = await throttledFn(10);
|
424 | t.is(v, 0);
|
425 | const throttledFn2 = (0, throttle_1.throttle)({ concurrency, rate, cache }, fn);
|
426 | const u1 = await throttledFn2(10);
|
427 | const u2 = await throttledFn2(20);
|
428 | t.is(u1, 0);
|
429 | t.is(u2, 1);
|
430 | t.is(counter, 2);
|
431 | }));
|
432 | (0, ava_1.default)("throttle cache and memoize options work together", async (t) => withCache(async (cache) => {
|
433 | const concurrency = 1;
|
434 | const rate = 100;
|
435 | let counter = 0;
|
436 | async function fn(_) {
|
437 | return counter++;
|
438 | }
|
439 | const throttledFn = (0, throttle_1.throttle)({ concurrency, rate, memoize: true, cache }, fn);
|
440 | const v = await throttledFn(10);
|
441 | const v2 = await throttledFn(10);
|
442 | t.is(v, 0);
|
443 | t.is(v2, 0);
|
444 | const throttledFn2 = (0, throttle_1.throttle)({ concurrency, rate, memoize: true, cache }, fn);
|
445 | const u1 = await throttledFn2(10);
|
446 | const u2 = await throttledFn2(20);
|
447 | const u3 = await throttledFn2(10);
|
448 | t.is(u1, 0);
|
449 | t.is(u2, 1);
|
450 | t.is(u3, 0);
|
451 | t.is(counter, 2);
|
452 | }));
|
453 | ava_1.default.serial("throttle cancellation", async (t) => (0, util_1.withClock)(async () => {
|
454 | const concurrency = 10;
|
455 | const rate = 100;
|
456 | let counter = 0;
|
457 | async function fn(_) {
|
458 | return counter++;
|
459 | }
|
460 | const cancel = new throttle_1.Deferred();
|
461 | const throttledFn = (0, throttle_1.throttle)({ concurrency, rate, memoize: true, cancel: cancel.promise }, fn);
|
462 | throttledFn(1);
|
463 | throttledFn(2);
|
464 | throttledFn(3);
|
465 | await (0, util_1.sleep)(100);
|
466 | t.is(counter, 3);
|
467 | counter = 0;
|
468 | throttledFn(1);
|
469 | throttledFn(2);
|
470 | throttledFn(3);
|
471 | cancel.resolve();
|
472 | await (0, util_1.sleep)(100);
|
473 | t.is(counter, 0);
|
474 | }));
|
475 | (0, ava_1.default)("AsyncQueue works with enqueue before dequeue", async (t) => {
|
476 | const q = new throttle_1.AsyncQueue();
|
477 | q.enqueue(42);
|
478 | t.is(await q.next(), 42);
|
479 | });
|
480 | (0, ava_1.default)("AsyncQueue works with multiple enqueues before dequeue", async (t) => {
|
481 | const q = new throttle_1.AsyncQueue();
|
482 | q.enqueue(42);
|
483 | q.enqueue(43);
|
484 | t.is(await q.next(), 42);
|
485 | t.is(await q.next(), 43);
|
486 | });
|
487 | (0, ava_1.default)("AsyncQueue works with dequeue before enqueue", async (t) => {
|
488 | const q = new throttle_1.AsyncQueue();
|
489 | const promise = q.next();
|
490 | q.enqueue(42);
|
491 | t.is(await promise, 42);
|
492 | });
|
493 | (0, ava_1.default)("AsyncQueue works with multiple dequeues before enqueue", async (t) => {
|
494 | const q = new throttle_1.AsyncQueue();
|
495 | const promises = [q.next(), q.next()];
|
496 | q.enqueue(42);
|
497 | q.enqueue(43);
|
498 | t.deepEqual(await Promise.all(promises), [42, 43]);
|
499 | });
|
500 | (0, ava_1.default)("AsyncQueue transition from more enqueues to more dequeues", async (t) => {
|
501 | const q = new throttle_1.AsyncQueue();
|
502 | q.enqueue(42);
|
503 | t.is(await q.next(), 42);
|
504 | const promise = q.next();
|
505 | q.enqueue(100);
|
506 | t.is(await promise, 100);
|
507 | });
|
508 | (0, ava_1.default)("AsyncQueue transition from more dequeues to more enqueues", async (t) => {
|
509 | const q = new throttle_1.AsyncQueue();
|
510 | const promise = q.next();
|
511 | q.enqueue(42);
|
512 | q.enqueue(100);
|
513 | t.is(await promise, 42);
|
514 | t.is(await q.next(), 100);
|
515 | });
|
516 | (0, ava_1.default)("AsyncQueue handles async enqueueing", async (t) => {
|
517 | const q = new throttle_1.AsyncQueue();
|
518 | const promise = q.next();
|
519 | setTimeout(() => q.enqueue(99), 100);
|
520 | t.is(await promise, 99);
|
521 | });
|
522 | (0, ava_1.default)("AsyncQueue handles async dequeueing", async (t) => {
|
523 | t.plan(1);
|
524 | const q = new throttle_1.AsyncQueue();
|
525 | q.enqueue(88);
|
526 | await new Promise(resolve => setTimeout(async () => {
|
527 | t.is(await q.next(), 88);
|
528 | resolve();
|
529 | }, 100));
|
530 | });
|
531 | (0, ava_1.default)("AsyncQueue clear", async (t) => {
|
532 | const q = new throttle_1.AsyncQueue();
|
533 | q.enqueue(1);
|
534 | q.clear();
|
535 | q.enqueue(2);
|
536 | t.is(await q.next(), 2);
|
537 | const p1 = q.next();
|
538 | q.clear();
|
539 | const p2 = q.next();
|
540 | q.enqueue(3);
|
541 | t.is(await p2, 3);
|
542 | });
|
543 | async function toArray(iterable) {
|
544 | const result = [];
|
545 | for await (const value of iterable) {
|
546 | result.push(value);
|
547 | }
|
548 | return result;
|
549 | }
|
550 | async function take(q, n) {
|
551 | const result = [];
|
552 | for (let i = 0; i < n; i++) {
|
553 | result.push(await q.next());
|
554 | }
|
555 | return result;
|
556 | }
|
557 | (0, ava_1.default)("AsyncIterableQueue done function finishes iterator", async (t) => {
|
558 | const q = new throttle_1.AsyncIterableQueue();
|
559 | q.push(10);
|
560 | q.done();
|
561 | t.deepEqual(await toArray(q), [10]);
|
562 | // test times out if the done function doesn't work.
|
563 | });
|
564 | (0, ava_1.default)("AsyncIterableQueue done function finishes iterator with pending dequeus", async (t) => {
|
565 | const q = new throttle_1.AsyncIterableQueue();
|
566 | const value = q.next();
|
567 | q.done();
|
568 | t.is((await value).done, true);
|
569 | });
|
570 | (0, ava_1.default)("AsyncOrderedQueue reorders according to sequence value", async (t) => {
|
571 | const q = new throttle_1.AsyncOrderedQueue();
|
572 | q.push(42, 1);
|
573 | q.push(-42, 0);
|
574 | t.deepEqual(await take(q, 2), [-42, 42]);
|
575 | });
|
576 | (0, ava_1.default)("AsyncOrderedQueue takes the first value with a given sequence value", async (t) => {
|
577 | const q = new throttle_1.AsyncOrderedQueue();
|
578 | q.push(100, 1);
|
579 | q.push(101, 1);
|
580 | q.push(42, 0);
|
581 | t.deepEqual(await take(q, 2), [42, 100]);
|
582 | });
|
583 | (0, ava_1.default)("AsyncOrderedQueue pushImmediate pre-empts arrival order", async (t) => {
|
584 | const q = new throttle_1.AsyncOrderedQueue();
|
585 | q.push(42, 0);
|
586 | q.push(44, 2);
|
587 | q.pushImmediate(100);
|
588 | q.push(43, 1);
|
589 | t.is(await q.next(), 42);
|
590 | t.is(await q.next(), 100);
|
591 | t.is(await q.next(), 43);
|
592 | t.is(await q.next(), 44);
|
593 | });
|
594 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"unit-throttle.test.js","sourceRoot":"","sources":["../../test/unit-throttle.test.ts"],"names":[],"mappings":"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","sourcesContent":["import test from \"ava\";\nimport { v4 as uuidv4 } from \"uuid\";\nimport { PersistentCache } from \"../src/cache\";\nimport {\n    AsyncOrderedQueue,\n    AsyncQueue,\n    cacheFn,\n    Deferred,\n    Funnel,\n    Pump,\n    RateLimiter,\n    retryOp,\n    throttle,\n    AsyncIterableQueue\n} from \"../src/throttle\";\nimport { timer, Timing } from \"./fixtures/functions\";\nimport { measureConcurrency, sleep, withClock } from \"./fixtures/util\";\n\ntest(\"deferred resolves its promise\", async t => {\n    const deferred = new Deferred();\n    let resolved = false;\n    deferred.promise.then(_ => (resolved = true));\n    t.is(resolved, false);\n    deferred.resolve();\n    await deferred.promise;\n    t.is(resolved, true);\n});\n\ntest(\"deferred rejects its promise\", async t => {\n    const deferred = new Deferred();\n    let rejected = false;\n    t.is(rejected, false);\n    deferred.reject();\n    try {\n        await deferred.promise;\n    } catch (_) {\n        rejected = true;\n    }\n    t.is(rejected, true);\n});\n\ntest(\"deferred resolves only once\", async t => {\n    const deferred = new Deferred();\n    let value = 0;\n    deferred.promise.then(_ => value++);\n\n    deferred.resolve();\n    await deferred.promise;\n    t.is(value, 1);\n\n    deferred.resolve();\n    await deferred.promise;\n    t.is(value, 1);\n});\n\ntest(\"deferred cannot reject after resolving\", async t => {\n    const deferred = new Deferred();\n    let value = 0;\n    deferred.promise.then(_ => value++);\n\n    deferred.resolve();\n    await deferred.promise;\n    t.is(value, 1);\n\n    deferred.reject();\n    await deferred.promise;\n    t.is(value, 1);\n});\n\ntest.serial(\"funnel defaults to infinite concurrency (tested with 200)\", t =>\n    withClock(async () => {\n        const funnel = new Funnel<Timing>(0);\n        const promises = [];\n        const N = 200;\n        for (let i = 0; i < N; i++) {\n            promises.push(funnel.push(() => timer(300)));\n        }\n        const times = await Promise.all(promises);\n        t.is(measureConcurrency(times), N);\n    })\n);\n\ntest.serial(\"funnel single concurrency is mutually exclusive\", t =>\n    withClock(async () => {\n        const funnel = new Funnel<Timing>(1);\n        const promises = [];\n        const N = 10;\n        for (let i = 0; i < N; i++) {\n            promises.push(funnel.push(() => timer(10)));\n        }\n        const times = await Promise.all(promises);\n        t.is(measureConcurrency(times), 1);\n    })\n);\n\ntest.serial(\"funnel handles concurrency level 2\", t =>\n    withClock(async () => {\n        const funnel = new Funnel<Timing>(2);\n        const promises = [];\n        const N = 10;\n        for (let i = 0; i < N; i++) {\n            promises.push(funnel.push(() => timer(20)));\n        }\n        const times = await Promise.all(promises);\n        t.is(measureConcurrency(times), 2);\n    })\n);\n\ntest.serial(\"funnel handles concurrency level 10\", t =>\n    withClock(async () => {\n        const funnel = new Funnel<Timing>(10);\n        const promises = [];\n        const N = 100;\n        for (let i = 0; i < N; i++) {\n            promises.push(funnel.push(() => timer(20)));\n        }\n        const times = await Promise.all(promises);\n        t.is(measureConcurrency(times), 10);\n    })\n);\n\ntest.serial(\"funnel resumes after finishing a worker\", t =>\n    withClock(async () => {\n        const funnel = new Funnel<Timing>(1);\n        const time1 = await funnel.push(() => timer(10));\n        const time2 = await funnel.push(() => timer(10));\n        t.is(measureConcurrency([time1, time2]), 1);\n    })\n);\n\ntest(\"funnel clearing\", async t => {\n    const funnel = new Funnel<number>(1);\n    let count = 0;\n    const promise0 = funnel.push(async () => count++);\n    const promise1 = funnel.push(async () => count++);\n    const promise2 = funnel.push(async () => count++);\n    funnel.clear();\n    t.is(\n        await Promise.race([promise0, promise1, promise2, sleep(100).then(_ => \"done\")]),\n        \"done\"\n    );\n    t.is(count, 0);\n});\ntest(\"funnel gets executed asynchronously, not at the moment of push\", async t => {\n    const funnel = new Funnel(1);\n    let n = 0;\n    funnel.push(async () => {\n        n++;\n    });\n    t.is(n, 0);\n    await funnel.all();\n    t.is(n, 1);\n});\ntest(\"funnel handles promise rejections without losing concurrency\", async t => {\n    const funnel = new Funnel<void>(1);\n    let executed = false;\n    await t.throwsAsync(\n        funnel.push(() => Promise.reject(new Error(\"message\"))),\n        { message: \"message\" }\n    );\n    await funnel.push(async () => {\n        executed = true;\n    });\n    t.is(executed, true);\n});\ntest(\"funnel.all() waits for all requests to finish\", async t => {\n    const funnel = new Funnel<string>(1);\n    let executed = false;\n    funnel.push(async () => {\n        await sleep(200);\n        executed = true;\n        return \"first\";\n    });\n    funnel.push(async () => \"second\");\n    t.is(executed, false);\n    const result = await funnel.all();\n    t.is(result.length, 2);\n    t.is(result[0], \"first\");\n    t.is(result[1], \"second\");\n    t.is(executed, true);\n});\ntest(\"funnel.all() ignores errors and waits for other requests to finish\", async t => {\n    const funnel = new Funnel<string>(1);\n    funnel.push(async () => {\n        throw new Error();\n    });\n    funnel.push(async () => {\n        await sleep(100);\n        return \"done\";\n    });\n    const result = await funnel.all();\n    t.is(result.length, 2);\n    t.falsy(result[0]);\n    t.is(result[1], \"done\");\n});\n\ntest(\"retryOp() retries failures\", async t => {\n    let attempts = 0;\n    await retryOp(2, async () => {\n        attempts++;\n        throw new Error();\n    }).catch(_ => {});\n    t.is(attempts, 3);\n});\n\ntest(\"funnel shouldRetry parameter retries failures\", async t => {\n    const funnel = new Funnel<string>(1, 2);\n    let attempts = 0;\n    let errors = 0;\n    funnel\n        .push(async () => {\n            attempts++;\n            throw Error();\n        })\n        .catch(_ => errors++);\n    await funnel.all();\n    t.is(attempts, 3);\n    t.is(errors, 1);\n});\n\ntest(\"funnel cancellation\", async t => {\n    const funnel = new Funnel(1);\n    let executed = 0;\n\n    const promise = funnel.push(\n        async () => {\n            executed++;\n        },\n        0,\n        () => \"cancelled\"\n    );\n    await t.throwsAsync(promise);\n    t.is(executed, 0);\n});\n\ntest(\"funnel processed and error counts\", async t => {\n    const funnel = new Funnel(2);\n    funnel.push(async () => {});\n    funnel.push(async () => Promise.reject());\n    funnel.push(async () => {});\n    funnel.push(async () => Promise.reject());\n    funnel.push(async () => {});\n\n    await funnel.all();\n    t.is(funnel.processed, 3);\n    t.is(funnel.errors, 2);\n});\n\ntest.serial(\"pump works for concurrency level 1\", t =>\n    withClock(async () => {\n        let executed = 0;\n        const pump = new Pump({ concurrency: 1 }, () => {\n            executed++;\n            return sleep(100);\n        });\n        t.is(executed, 0);\n        pump.start();\n        await sleep(300);\n        pump.stop();\n        t.true(executed === 3);\n    })\n);\n\ntest.serial(\"pump works for concurrency level 10\", t =>\n    withClock(async () => {\n        let executed = 0;\n        const pump = new Pump({ concurrency: 10 }, () => {\n            executed++;\n            return sleep(100);\n        });\n        pump.start();\n        await sleep(100);\n        pump.stop();\n        t.is(executed, 10);\n    })\n);\n\ntest.serial(\"pump handles promise rejections without losing concurrency\", t =>\n    withClock(async () => {\n        let executed = 0;\n        const pump = new Pump({ concurrency: 1, verbose: false }, () => {\n            executed++;\n            return sleep(100).then(_ => Promise.reject(\"hi\"));\n        });\n        pump.start();\n        await sleep(500);\n        pump.stop();\n        t.is(executed, 5);\n    })\n);\n\ntest.serial(\"pump drain\", t =>\n    withClock(async () => {\n        let started = 0;\n        let finished = 0;\n        const N = 5;\n\n        const pump = new Pump({ concurrency: N }, async () => {\n            started++;\n            await sleep(100);\n            finished++;\n        });\n\n        t.is(started, 0);\n        t.is(finished, 0);\n\n        pump.start();\n        await pump.drain();\n        t.is(started, N);\n        t.is(finished, N);\n    })\n);\n\ntest.serial(\"memoize returns cached results for the same key\", t =>\n    withClock(async () => {\n        const promises = [];\n        const N = 10;\n        const timerFn = throttle({ memoize: true, concurrency: 1, rate: 10 }, _ =>\n            timer(10)\n        );\n        for (let i = 0; i < N; i++) {\n            promises.push(timerFn(\"key\"));\n        }\n        const times = await Promise.all(promises);\n        t.is(measureConcurrency(times), N);\n    })\n);\n\ntest.serial(\"memoize runs the worker for different keys\", t =>\n    withClock(async () => {\n        const promises = [];\n        const N = 10;\n        const timerFn = throttle({ memoize: true, concurrency: 1, rate: 10 }, _ =>\n            timer(10)\n        );\n        for (let i = 0; i < N; i++) {\n            promises.push(timerFn(i));\n        }\n        const times = await Promise.all(promises);\n        t.is(measureConcurrency(times), 1);\n    })\n);\n\nasync function withCache(fn: (cache: PersistentCache) => Promise<void>) {\n    const nonce = uuidv4();\n    const cache = new PersistentCache(`.faast/test/${nonce}`);\n    await fn(cache).catch(console.error);\n    await cache.clear({ leaveEmptyDir: false });\n}\n\ntest(\"caching saves values and skips re-execution\", t =>\n    withCache(async cache => {\n        let counter = 0;\n        function fn(_: number) {\n            return Promise.resolve(counter++);\n        }\n        const mfn = cacheFn(cache, fn);\n        await mfn(0);\n        await mfn(7);\n        await mfn(0);\n        t.is(counter, 2);\n\n        const mfn2 = cacheFn(cache, fn);\n        await mfn2(0);\n        await mfn2(7);\n        await mfn2(0);\n        await mfn2(10);\n        t.is(counter, 3);\n    }));\n\ntest(\"cache works with string arguments\", async t =>\n    withCache(async cache => {\n        let counter = 0;\n        function fn(_: string) {\n            return Promise.resolve(counter++);\n        }\n        const mfn = cacheFn(cache, fn);\n        await mfn(\"a\");\n        await mfn(\"b\");\n        await mfn(\"a\");\n        t.is(counter, 2);\n    }));\n\ntest(\"cache works with object arguments\", async t =>\n    withCache(async cache => {\n        let counter = 0;\n        function fn(_: { f: string; i: number }) {\n            return Promise.resolve(counter++);\n        }\n        const mfn = cacheFn(cache, fn);\n        await mfn({ f: \"field\", i: 42 });\n        await mfn({ f: \"field\", i: 1 });\n        await mfn({ f: \"other\", i: 42 });\n        await mfn({ f: \"field\", i: 42 });\n        t.is(counter, 3);\n    }));\n\ntest(\"cache does not save rejected promises from cached function\", async t =>\n    withCache(async cache => {\n        let counter = 0;\n        function fn(_: number) {\n            counter++;\n            return Promise.reject(new Error(\"rejection\"));\n        }\n        let caught = 0;\n        const mfn = cacheFn(cache, fn);\n        await mfn(1).catch(_ => caught++);\n        await mfn(2).catch(_ => caught++);\n        await mfn(1).catch(_ => caught++);\n        t.is(counter, 3);\n        t.is(caught, 3);\n    }));\n\nfunction measureMaxRequestRatePerSecond(timings: Timing[]) {\n    const requestsPerSecondStartingAt = timings\n        .map(t => t.start)\n        .map(t => timings.filter(({ start }) => start >= t && start < t + 1000).length);\n    return Math.max(...requestsPerSecondStartingAt);\n}\n\ntest.serial(\"rate limiter restricts max request rate per second\", t =>\n    withClock(async () => {\n        const requestRate = 10;\n        const rateLimiter = new RateLimiter<Timing>(requestRate);\n        const promises: Promise<Timing>[] = [];\n        for (let i = 0; i < 15; i++) {\n            promises.push(rateLimiter.push(() => timer(0)));\n        }\n        const timings = await Promise.all(promises);\n        t.is(measureMaxRequestRatePerSecond(timings), requestRate);\n    })\n);\n\ntest.serial(\"rate limiter works across second boundaries\", t =>\n    withClock(async () => {\n        const requestRate = 10;\n        const rateLimiter = new RateLimiter<Timing>(requestRate);\n        const promises: Promise<Timing>[] = [];\n        promises.push(rateLimiter.push(() => timer(0)));\n        await sleep(900);\n        for (let i = 0; i < 15; i++) {\n            promises.push(rateLimiter.push(() => timer(0)));\n        }\n        const timings = await Promise.all(promises);\n        t.is(measureMaxRequestRatePerSecond(timings), requestRate);\n    })\n);\n\ntest.serial(\"rate limiter bursting allows for request rate beyond target rate\", t =>\n    withClock(async () => {\n        const requestRate = 10;\n        const maxBurst = 5;\n        const rateLimiter = new RateLimiter<Timing>(requestRate, maxBurst);\n        const promises: Promise<Timing>[] = [];\n        for (let i = 0; i < 15; i++) {\n            promises.push(rateLimiter.push(() => timer(0)));\n        }\n        const timings = await Promise.all(promises);\n        const maxRate = measureMaxRequestRatePerSecond(timings);\n        t.true(maxRate <= maxBurst + requestRate);\n        t.true(maxRate > maxBurst);\n    })\n);\n\ntest.serial(\"throttle limits max concurrency and rate\", t =>\n    withClock(async () => {\n        const concurrency = 10;\n        const rate = 10;\n        const timerFn = throttle({ concurrency, rate }, timer);\n        const promises = [];\n        for (let i = 0; i < 15; i++) {\n            promises.push(timerFn(1000));\n        }\n\n        const times = await Promise.all(promises);\n        t.is(measureConcurrency(times), concurrency);\n        t.is(measureMaxRequestRatePerSecond(times), rate);\n    })\n);\n\ntest.serial(\"throttle limits rate with single concurrency\", t =>\n    withClock(async () => {\n        const concurrency = 1;\n        const rate = 10;\n        const processTimeMs = 200;\n        const timerFn = throttle({ concurrency, rate }, timer);\n\n        const promises = [];\n        for (let i = 0; i < 10; i++) {\n            promises.push(timerFn(processTimeMs));\n        }\n\n        const times = await Promise.all(promises);\n        t.is(measureConcurrency(times), concurrency);\n        t.true(measureMaxRequestRatePerSecond(times) <= 1000 / processTimeMs + 1);\n    })\n);\n\ntest(\"throttle memoize option\", async t => {\n    const concurrency = 1;\n    const rate = 100;\n    let counter = 0;\n    const N = 5;\n    async function fn(_: number) {\n        counter++;\n    }\n    const throttledFn = throttle({ concurrency, rate, memoize: true }, fn);\n\n    const promises = [];\n    for (let i = 0; i < N; i++) {\n        promises.push(throttledFn(i));\n    }\n    for (let i = 0; i < N; i++) {\n        promises.push(throttledFn(i));\n    }\n\n    await Promise.all(promises);\n    t.is(counter, N);\n});\n\ntest(\"throttle cache option persists values\", async t =>\n    withCache(async cache => {\n        const concurrency = 1;\n        const rate = 100;\n        let counter = 0;\n\n        async function fn(_: number) {\n            return counter++;\n        }\n\n        const throttledFn = throttle({ concurrency, rate, cache }, fn);\n\n        const v = await throttledFn(10);\n        t.is(v, 0);\n\n        const throttledFn2 = throttle({ concurrency, rate, cache }, fn);\n\n        const u1 = await throttledFn2(10);\n        const u2 = await throttledFn2(20);\n\n        t.is(u1, 0);\n        t.is(u2, 1);\n        t.is(counter, 2);\n    }));\n\ntest(\"throttle cache and memoize options work together\", async t =>\n    withCache(async cache => {\n        const concurrency = 1;\n        const rate = 100;\n        let counter = 0;\n\n        async function fn(_: number) {\n            return counter++;\n        }\n\n        const throttledFn = throttle({ concurrency, rate, memoize: true, cache }, fn);\n\n        const v = await throttledFn(10);\n        const v2 = await throttledFn(10);\n        t.is(v, 0);\n        t.is(v2, 0);\n\n        const throttledFn2 = throttle({ concurrency, rate, memoize: true, cache }, fn);\n\n        const u1 = await throttledFn2(10);\n        const u2 = await throttledFn2(20);\n        const u3 = await throttledFn2(10);\n\n        t.is(u1, 0);\n        t.is(u2, 1);\n        t.is(u3, 0);\n\n        t.is(counter, 2);\n    }));\n\ntest.serial(\"throttle cancellation\", async t =>\n    withClock(async () => {\n        const concurrency = 10;\n        const rate = 100;\n        let counter = 0;\n\n        async function fn(_: number) {\n            return counter++;\n        }\n        const cancel = new Deferred();\n        const throttledFn = throttle(\n            { concurrency, rate, memoize: true, cancel: cancel.promise },\n            fn\n        );\n        throttledFn(1);\n        throttledFn(2);\n        throttledFn(3);\n        await sleep(100);\n        t.is(counter, 3);\n\n        counter = 0;\n        throttledFn(1);\n        throttledFn(2);\n        throttledFn(3);\n        cancel.resolve();\n        await sleep(100);\n        t.is(counter, 0);\n    })\n);\n\ntest(\"AsyncQueue works with enqueue before dequeue\", async t => {\n    const q = new AsyncQueue<number>();\n    q.enqueue(42);\n    t.is(await q.next(), 42);\n});\n\ntest(\"AsyncQueue works with multiple enqueues before dequeue\", async t => {\n    const q = new AsyncQueue<number>();\n    q.enqueue(42);\n    q.enqueue(43);\n    t.is(await q.next(), 42);\n    t.is(await q.next(), 43);\n});\n\ntest(\"AsyncQueue works with dequeue before enqueue\", async t => {\n    const q = new AsyncQueue<number>();\n    const promise = q.next();\n    q.enqueue(42);\n    t.is(await promise, 42);\n});\n\ntest(\"AsyncQueue works with multiple dequeues before enqueue\", async t => {\n    const q = new AsyncQueue<number>();\n    const promises = [q.next(), q.next()];\n    q.enqueue(42);\n    q.enqueue(43);\n    t.deepEqual(await Promise.all(promises), [42, 43]);\n});\n\ntest(\"AsyncQueue transition from more enqueues to more dequeues\", async t => {\n    const q = new AsyncQueue<number>();\n    q.enqueue(42);\n    t.is(await q.next(), 42);\n    const promise = q.next();\n    q.enqueue(100);\n    t.is(await promise, 100);\n});\n\ntest(\"AsyncQueue transition from more dequeues to more enqueues\", async t => {\n    const q = new AsyncQueue<number>();\n    const promise = q.next();\n    q.enqueue(42);\n    q.enqueue(100);\n    t.is(await promise, 42);\n    t.is(await q.next(), 100);\n});\n\ntest(\"AsyncQueue handles async enqueueing\", async t => {\n    const q = new AsyncQueue<number>();\n    const promise = q.next();\n    setTimeout(() => q.enqueue(99), 100);\n    t.is(await promise, 99);\n});\n\ntest(\"AsyncQueue handles async dequeueing\", async t => {\n    t.plan(1);\n    const q = new AsyncQueue<number>();\n    q.enqueue(88);\n    await new Promise<void>(resolve =>\n        setTimeout(async () => {\n            t.is(await q.next(), 88);\n            resolve();\n        }, 100)\n    );\n});\n\ntest(\"AsyncQueue clear\", async t => {\n    const q = new AsyncQueue<number>();\n    q.enqueue(1);\n    q.clear();\n    q.enqueue(2);\n    t.is(await q.next(), 2);\n\n    const p1 = q.next();\n    q.clear();\n    const p2 = q.next();\n    q.enqueue(3);\n    t.is(await p2, 3);\n});\n\nasync function toArray<T>(iterable: AsyncIterable<T> | Iterable<T>) {\n    const result = [];\n    for await (const value of iterable) {\n        result.push(value);\n    }\n    return result;\n}\n\nasync function take<T>(q: AsyncOrderedQueue<T>, n: number) {\n    const result = [];\n    for (let i = 0; i < n; i++) {\n        result.push(await q.next());\n    }\n    return result;\n}\n\ntest(\"AsyncIterableQueue done function finishes iterator\", async t => {\n    const q = new AsyncIterableQueue<number>();\n    q.push(10);\n    q.done();\n    t.deepEqual(await toArray(q), [10]);\n    // test times out if the done function doesn't work.\n});\n\ntest(\"AsyncIterableQueue done function finishes iterator with pending dequeus\", async t => {\n    const q = new AsyncIterableQueue<number>();\n    const value = q.next();\n    q.done();\n    t.is((await value).done, true);\n});\n\ntest(\"AsyncOrderedQueue reorders according to sequence value\", async t => {\n    const q = new AsyncOrderedQueue<number>();\n    q.push(42, 1);\n    q.push(-42, 0);\n    t.deepEqual(await take(q, 2), [-42, 42]);\n});\n\ntest(\"AsyncOrderedQueue takes the first value with a given sequence value\", async t => {\n    const q = new AsyncOrderedQueue<number>();\n    q.push(100, 1);\n    q.push(101, 1);\n    q.push(42, 0);\n    t.deepEqual(await take(q, 2), [42, 100]);\n});\n\ntest(\"AsyncOrderedQueue pushImmediate pre-empts arrival order\", async t => {\n    const q = new AsyncOrderedQueue<number>();\n    q.push(42, 0);\n    q.push(44, 2);\n    q.pushImmediate(100);\n    q.push(43, 1);\n\n    t.is(await q.next(), 42);\n    t.is(await q.next(), 100);\n    t.is(await q.next(), 43);\n    t.is(await q.next(), 44);\n});\n"]} |
\ | No newline at end of file |