UNPKG

86.8 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3const ava_1 = require("ava");
4const uuid_1 = require("uuid");
5const cache_1 = require("../src/cache");
6const throttle_1 = require("../src/throttle");
7const functions_1 = require("./fixtures/functions");
8const 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});
53ava_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}));
63ava_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}));
73ava_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}));
83ava_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}));
93ava_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});
200ava_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}));
212ava_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}));
223ava_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}));
234ava_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}));
250ava_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}));
260ava_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}));
270async 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}));
330function 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}
336ava_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}));
346ava_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}));
358ava_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}));
371ava_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}));
383ava_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}));
453ava_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});
543async function toArray(iterable) {
544 const result = [];
545 for await (const value of iterable) {
546 result.push(value);
547 }
548 return result;
549}
550async 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