123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322 |
- import { Observable } from '../Observable';
- import { Notification } from '../Notification';
- import { ColdObservable } from './ColdObservable';
- import { HotObservable } from './HotObservable';
- import { SubscriptionLog } from './SubscriptionLog';
- import { VirtualTimeScheduler, VirtualAction } from '../scheduler/VirtualTimeScheduler';
- import { AsyncScheduler } from '../scheduler/AsyncScheduler';
- const defaultMaxFrame = 750;
- export class TestScheduler extends VirtualTimeScheduler {
- constructor(assertDeepEqual) {
- super(VirtualAction, defaultMaxFrame);
- this.assertDeepEqual = assertDeepEqual;
- this.hotObservables = [];
- this.coldObservables = [];
- this.flushTests = [];
- this.runMode = false;
- }
- createTime(marbles) {
- const indexOf = marbles.indexOf('|');
- if (indexOf === -1) {
- throw new Error('marble diagram for time should have a completion marker "|"');
- }
- return indexOf * TestScheduler.frameTimeFactor;
- }
- createColdObservable(marbles, values, error) {
- if (marbles.indexOf('^') !== -1) {
- throw new Error('cold observable cannot have subscription offset "^"');
- }
- if (marbles.indexOf('!') !== -1) {
- throw new Error('cold observable cannot have unsubscription marker "!"');
- }
- const messages = TestScheduler.parseMarbles(marbles, values, error, undefined, this.runMode);
- const cold = new ColdObservable(messages, this);
- this.coldObservables.push(cold);
- return cold;
- }
- createHotObservable(marbles, values, error) {
- if (marbles.indexOf('!') !== -1) {
- throw new Error('hot observable cannot have unsubscription marker "!"');
- }
- const messages = TestScheduler.parseMarbles(marbles, values, error, undefined, this.runMode);
- const subject = new HotObservable(messages, this);
- this.hotObservables.push(subject);
- return subject;
- }
- materializeInnerObservable(observable, outerFrame) {
- const messages = [];
- observable.subscribe((value) => {
- messages.push({ frame: this.frame - outerFrame, notification: Notification.createNext(value) });
- }, (err) => {
- messages.push({ frame: this.frame - outerFrame, notification: Notification.createError(err) });
- }, () => {
- messages.push({ frame: this.frame - outerFrame, notification: Notification.createComplete() });
- });
- return messages;
- }
- expectObservable(observable, subscriptionMarbles = null) {
- const actual = [];
- const flushTest = { actual, ready: false };
- const subscriptionParsed = TestScheduler.parseMarblesAsSubscriptions(subscriptionMarbles, this.runMode);
- const subscriptionFrame = subscriptionParsed.subscribedFrame === Number.POSITIVE_INFINITY ?
- 0 : subscriptionParsed.subscribedFrame;
- const unsubscriptionFrame = subscriptionParsed.unsubscribedFrame;
- let subscription;
- this.schedule(() => {
- subscription = observable.subscribe(x => {
- let value = x;
- if (x instanceof Observable) {
- value = this.materializeInnerObservable(value, this.frame);
- }
- actual.push({ frame: this.frame, notification: Notification.createNext(value) });
- }, (err) => {
- actual.push({ frame: this.frame, notification: Notification.createError(err) });
- }, () => {
- actual.push({ frame: this.frame, notification: Notification.createComplete() });
- });
- }, subscriptionFrame);
- if (unsubscriptionFrame !== Number.POSITIVE_INFINITY) {
- this.schedule(() => subscription.unsubscribe(), unsubscriptionFrame);
- }
- this.flushTests.push(flushTest);
- const { runMode } = this;
- return {
- toBe(marbles, values, errorValue) {
- flushTest.ready = true;
- flushTest.expected = TestScheduler.parseMarbles(marbles, values, errorValue, true, runMode);
- }
- };
- }
- expectSubscriptions(actualSubscriptionLogs) {
- const flushTest = { actual: actualSubscriptionLogs, ready: false };
- this.flushTests.push(flushTest);
- const { runMode } = this;
- return {
- toBe(marbles) {
- const marblesArray = (typeof marbles === 'string') ? [marbles] : marbles;
- flushTest.ready = true;
- flushTest.expected = marblesArray.map(marbles => TestScheduler.parseMarblesAsSubscriptions(marbles, runMode));
- }
- };
- }
- flush() {
- const hotObservables = this.hotObservables;
- while (hotObservables.length > 0) {
- hotObservables.shift().setup();
- }
- super.flush();
- this.flushTests = this.flushTests.filter(test => {
- if (test.ready) {
- this.assertDeepEqual(test.actual, test.expected);
- return false;
- }
- return true;
- });
- }
- static parseMarblesAsSubscriptions(marbles, runMode = false) {
- if (typeof marbles !== 'string') {
- return new SubscriptionLog(Number.POSITIVE_INFINITY);
- }
- const len = marbles.length;
- let groupStart = -1;
- let subscriptionFrame = Number.POSITIVE_INFINITY;
- let unsubscriptionFrame = Number.POSITIVE_INFINITY;
- let frame = 0;
- for (let i = 0; i < len; i++) {
- let nextFrame = frame;
- const advanceFrameBy = (count) => {
- nextFrame += count * this.frameTimeFactor;
- };
- const c = marbles[i];
- switch (c) {
- case ' ':
- if (!runMode) {
- advanceFrameBy(1);
- }
- break;
- case '-':
- advanceFrameBy(1);
- break;
- case '(':
- groupStart = frame;
- advanceFrameBy(1);
- break;
- case ')':
- groupStart = -1;
- advanceFrameBy(1);
- break;
- case '^':
- if (subscriptionFrame !== Number.POSITIVE_INFINITY) {
- throw new Error('found a second subscription point \'^\' in a ' +
- 'subscription marble diagram. There can only be one.');
- }
- subscriptionFrame = groupStart > -1 ? groupStart : frame;
- advanceFrameBy(1);
- break;
- case '!':
- if (unsubscriptionFrame !== Number.POSITIVE_INFINITY) {
- throw new Error('found a second subscription point \'^\' in a ' +
- 'subscription marble diagram. There can only be one.');
- }
- unsubscriptionFrame = groupStart > -1 ? groupStart : frame;
- break;
- default:
- if (runMode && c.match(/^[0-9]$/)) {
- if (i === 0 || marbles[i - 1] === ' ') {
- const buffer = marbles.slice(i);
- const match = buffer.match(/^([0-9]+(?:\.[0-9]+)?)(ms|s|m) /);
- if (match) {
- i += match[0].length - 1;
- const duration = parseFloat(match[1]);
- const unit = match[2];
- let durationInMs;
- switch (unit) {
- case 'ms':
- durationInMs = duration;
- break;
- case 's':
- durationInMs = duration * 1000;
- break;
- case 'm':
- durationInMs = duration * 1000 * 60;
- break;
- default:
- break;
- }
- advanceFrameBy(durationInMs / this.frameTimeFactor);
- break;
- }
- }
- }
- throw new Error('there can only be \'^\' and \'!\' markers in a ' +
- 'subscription marble diagram. Found instead \'' + c + '\'.');
- }
- frame = nextFrame;
- }
- if (unsubscriptionFrame < 0) {
- return new SubscriptionLog(subscriptionFrame);
- }
- else {
- return new SubscriptionLog(subscriptionFrame, unsubscriptionFrame);
- }
- }
- static parseMarbles(marbles, values, errorValue, materializeInnerObservables = false, runMode = false) {
- if (marbles.indexOf('!') !== -1) {
- throw new Error('conventional marble diagrams cannot have the ' +
- 'unsubscription marker "!"');
- }
- const len = marbles.length;
- const testMessages = [];
- const subIndex = runMode ? marbles.replace(/^[ ]+/, '').indexOf('^') : marbles.indexOf('^');
- let frame = subIndex === -1 ? 0 : (subIndex * -this.frameTimeFactor);
- const getValue = typeof values !== 'object' ?
- (x) => x :
- (x) => {
- if (materializeInnerObservables && values[x] instanceof ColdObservable) {
- return values[x].messages;
- }
- return values[x];
- };
- let groupStart = -1;
- for (let i = 0; i < len; i++) {
- let nextFrame = frame;
- const advanceFrameBy = (count) => {
- nextFrame += count * this.frameTimeFactor;
- };
- let notification;
- const c = marbles[i];
- switch (c) {
- case ' ':
- if (!runMode) {
- advanceFrameBy(1);
- }
- break;
- case '-':
- advanceFrameBy(1);
- break;
- case '(':
- groupStart = frame;
- advanceFrameBy(1);
- break;
- case ')':
- groupStart = -1;
- advanceFrameBy(1);
- break;
- case '|':
- notification = Notification.createComplete();
- advanceFrameBy(1);
- break;
- case '^':
- advanceFrameBy(1);
- break;
- case '#':
- notification = Notification.createError(errorValue || 'error');
- advanceFrameBy(1);
- break;
- default:
- if (runMode && c.match(/^[0-9]$/)) {
- if (i === 0 || marbles[i - 1] === ' ') {
- const buffer = marbles.slice(i);
- const match = buffer.match(/^([0-9]+(?:\.[0-9]+)?)(ms|s|m) /);
- if (match) {
- i += match[0].length - 1;
- const duration = parseFloat(match[1]);
- const unit = match[2];
- let durationInMs;
- switch (unit) {
- case 'ms':
- durationInMs = duration;
- break;
- case 's':
- durationInMs = duration * 1000;
- break;
- case 'm':
- durationInMs = duration * 1000 * 60;
- break;
- default:
- break;
- }
- advanceFrameBy(durationInMs / this.frameTimeFactor);
- break;
- }
- }
- }
- notification = Notification.createNext(getValue(c));
- advanceFrameBy(1);
- break;
- }
- if (notification) {
- testMessages.push({ frame: groupStart > -1 ? groupStart : frame, notification });
- }
- frame = nextFrame;
- }
- return testMessages;
- }
- run(callback) {
- const prevFrameTimeFactor = TestScheduler.frameTimeFactor;
- const prevMaxFrames = this.maxFrames;
- TestScheduler.frameTimeFactor = 1;
- this.maxFrames = Number.POSITIVE_INFINITY;
- this.runMode = true;
- AsyncScheduler.delegate = this;
- const helpers = {
- cold: this.createColdObservable.bind(this),
- hot: this.createHotObservable.bind(this),
- flush: this.flush.bind(this),
- expectObservable: this.expectObservable.bind(this),
- expectSubscriptions: this.expectSubscriptions.bind(this),
- };
- try {
- const ret = callback(helpers);
- this.flush();
- return ret;
- }
- finally {
- TestScheduler.frameTimeFactor = prevFrameTimeFactor;
- this.maxFrames = prevMaxFrames;
- this.runMode = false;
- AsyncScheduler.delegate = undefined;
- }
- }
- }
|