You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
224 lines
6.4 KiB
TypeScript
224 lines
6.4 KiB
TypeScript
// tslint:disable: no-implicit-dependencies max-func-body-length no-unused-expression
|
|
|
|
import chai from 'chai';
|
|
import Sinon, * as sinon from 'sinon';
|
|
|
|
import { PromiseUtils } from '../../../../session/utils';
|
|
|
|
// tslint:disable-next-line: no-require-imports no-var-requires
|
|
import chaiAsPromised from 'chai-as-promised';
|
|
import {
|
|
allowOnlyOneAtATime,
|
|
hasAlreadyOneAtaTimeMatching,
|
|
sleepFor,
|
|
} from '../../../../session/utils/Promise';
|
|
import { TestUtils } from '../../../test-utils';
|
|
import { enableLogRedirect } from '../../../test-utils/utils';
|
|
|
|
chai.use(chaiAsPromised as any);
|
|
chai.should();
|
|
|
|
const { expect } = chai;
|
|
|
|
describe('Promise Utils', () => {
|
|
let pollSpy: sinon.SinonSpy<
|
|
[
|
|
(done: (arg: any) => void) => Promise<void> | void,
|
|
(Partial<PromiseUtils.PollOptions> | undefined)?
|
|
],
|
|
Promise<void>
|
|
>;
|
|
let waitForTaskSpy: sinon.SinonSpy<
|
|
[(done: (arg: any) => void) => Promise<void> | void, (number | undefined)?],
|
|
Promise<unknown>
|
|
>;
|
|
let waitUntilSpy: sinon.SinonSpy<
|
|
[() => Promise<boolean> | boolean, (number | undefined)?],
|
|
Promise<void>
|
|
>;
|
|
|
|
beforeEach(() => {
|
|
pollSpy = Sinon.spy(PromiseUtils, 'poll');
|
|
waitForTaskSpy = Sinon.spy(PromiseUtils, 'waitForTask');
|
|
waitUntilSpy = Sinon.spy(PromiseUtils, 'waitUntil');
|
|
TestUtils.stubWindowLog();
|
|
});
|
|
|
|
afterEach(() => {
|
|
Sinon.restore();
|
|
});
|
|
|
|
describe('poll', () => {
|
|
it('will call done on finished', async () => {
|
|
// completionSpy will be called on done
|
|
const completionSpy = Sinon.spy();
|
|
|
|
// tslint:disable-next-line: mocha-unneeded-done
|
|
const task = (done: any) => {
|
|
completionSpy();
|
|
done();
|
|
};
|
|
|
|
const promise = PromiseUtils.poll(task, { interval: 10 });
|
|
|
|
expect(pollSpy.callCount).to.equal(1);
|
|
expect(completionSpy.callCount).to.equal(1);
|
|
return promise;
|
|
});
|
|
|
|
it('can timeout a task', () => {
|
|
// completionSpy will be called on done
|
|
const completionSpy = Sinon.spy();
|
|
const task = (_done: any) => undefined;
|
|
|
|
const promise = PromiseUtils.poll(task, { timeoutMs: 1, interval: 10 });
|
|
|
|
expect(pollSpy.callCount).to.equal(1);
|
|
expect(completionSpy.callCount).to.equal(0);
|
|
return promise.should.eventually.be.rejectedWith('Periodic check timeout');
|
|
});
|
|
|
|
it('will recur according to interval option', async () => {
|
|
const expectedRecurrences = 4;
|
|
const timeout = 3000;
|
|
const interval = 3;
|
|
|
|
const recurrenceSpy = Sinon.spy();
|
|
const task = (done: any) => {
|
|
recurrenceSpy();
|
|
|
|
// Done after we've been called `expectedRecurrences` times
|
|
if (recurrenceSpy.callCount === expectedRecurrences) {
|
|
done();
|
|
}
|
|
};
|
|
|
|
const promise = PromiseUtils.poll(task, { timeoutMs: timeout, interval });
|
|
await promise;
|
|
|
|
expect(pollSpy.callCount).to.equal(1);
|
|
expect(recurrenceSpy.callCount).to.equal(expectedRecurrences);
|
|
});
|
|
});
|
|
|
|
describe('waitForTask', () => {
|
|
it('can wait for a task', async () => {
|
|
// completionSpy will be called on done
|
|
const completionSpy = Sinon.spy();
|
|
|
|
// tslint:disable-next-line: mocha-unneeded-done
|
|
const task = (done: any) => {
|
|
completionSpy();
|
|
done();
|
|
};
|
|
|
|
const promise = PromiseUtils.waitForTask(task);
|
|
|
|
expect(waitForTaskSpy.callCount).to.equal(1);
|
|
expect(completionSpy.callCount).to.equal(1);
|
|
return promise;
|
|
});
|
|
|
|
it('can timeout a task', () => {
|
|
// completionSpy will be called on done
|
|
const completionSpy = Sinon.spy();
|
|
const task = async (_done: any) => undefined;
|
|
|
|
const promise = PromiseUtils.waitForTask(task, 1);
|
|
|
|
expect(waitForTaskSpy.callCount).to.equal(1);
|
|
expect(completionSpy.callCount).to.equal(0);
|
|
return promise.should.eventually.be.rejectedWith('Task timed out');
|
|
});
|
|
});
|
|
|
|
describe('waitUntil', () => {
|
|
it('can wait for check', async () => {
|
|
const check = () => true;
|
|
const promise = PromiseUtils.waitUntil(check, 5);
|
|
|
|
expect(waitUntilSpy.callCount).to.equal(1);
|
|
return promise;
|
|
});
|
|
|
|
it('can timeout a check', () => {
|
|
const check = () => false;
|
|
const promise = PromiseUtils.waitUntil(check, 1);
|
|
|
|
expect(waitUntilSpy.callCount).to.equal(1);
|
|
return promise.should.eventually.be.rejectedWith('Periodic check timeout');
|
|
});
|
|
});
|
|
|
|
describe('allowOnlyOneAtATime', () => {
|
|
it('start if not running', async () => {
|
|
const spy = sinon.spy(async () => {
|
|
return sleepFor(10);
|
|
});
|
|
await allowOnlyOneAtATime('testing', spy);
|
|
expect(spy.callCount).to.be.eq(1);
|
|
});
|
|
|
|
it('starts only once if already running', async () => {
|
|
const spy = sinon.spy(async () => {
|
|
return sleepFor(10);
|
|
});
|
|
void allowOnlyOneAtATime('testing', spy);
|
|
|
|
await allowOnlyOneAtATime('testing', spy);
|
|
expect(spy.callCount).to.be.eq(1);
|
|
});
|
|
|
|
it('throw if took longer than expected timeout', async () => {
|
|
const spy = sinon.spy(async () => {
|
|
return sleepFor(10);
|
|
});
|
|
try {
|
|
await allowOnlyOneAtATime('testing', spy, 5);
|
|
throw new Error('should not get here');
|
|
} catch (e) {
|
|
// tslint:disable-next-line: no-console
|
|
console.error(e);
|
|
expect(e).to.be.be.eql(undefined, 'should be undefined');
|
|
}
|
|
|
|
expect(spy.callCount).to.be.eq(1);
|
|
});
|
|
|
|
it('does not throw if took less than expected timeout', async () => {
|
|
const spy = sinon.spy(async () => {
|
|
return sleepFor(10);
|
|
});
|
|
try {
|
|
await allowOnlyOneAtATime('testing', spy, 15);
|
|
throw new Error('should get here');
|
|
} catch (e) {
|
|
expect(e.message).to.be.be.eql('should get here');
|
|
}
|
|
|
|
expect(spy.callCount).to.be.eq(1);
|
|
});
|
|
});
|
|
|
|
describe('hasAlreadyOneAtaTimeMatching', () => {
|
|
it('returns true if already started', () => {
|
|
const spy = sinon.spy(async () => {
|
|
return sleepFor(10);
|
|
});
|
|
void allowOnlyOneAtATime('testing', spy);
|
|
expect(hasAlreadyOneAtaTimeMatching('testing')).to.be.eq(true, 'should be true');
|
|
});
|
|
|
|
it('returns false if not already started', () => {
|
|
expect(hasAlreadyOneAtaTimeMatching('testing2')).to.be.eq(false, 'should be false');
|
|
});
|
|
});
|
|
|
|
it('stubWindowLog is set to false before pushing', () => {
|
|
expect(
|
|
enableLogRedirect,
|
|
'If you see this message, just set `enableLogRedirect` to false in `ts/test/test-utils/utils/stubbing.ts`'
|
|
).to.be.false;
|
|
});
|
|
});
|