test(docs-infra): add unit tests for rxjs examples (#34190)

This commit adds missing unit tests for all rxjs examples from the docs.

Closes #28017

PR Close #34190
This commit is contained in:
Sonu Kapoor 2019-12-02 13:15:58 -05:00 committed by Andrew Kushnir
parent a85109fd72
commit ba54671993
11 changed files with 341 additions and 199 deletions

View File

@ -1,3 +1,4 @@
// TODO: Add unit tests for this file.
// tslint:disable: no-output-native // tslint:disable: no-output-native
// #docregion // #docregion
import { Component, Output, OnInit, EventEmitter, NgModule } from '@angular/core'; import { Component, Output, OnInit, EventEmitter, NgModule } from '@angular/core';

View File

@ -2,7 +2,11 @@
"tests": [ "tests": [
{ {
"cmd": "yarn", "cmd": "yarn",
"args": [ "tsc", "--project", "./tsconfig.app.json" ] "args": ["tsc", "--project", "tsconfig.spec.json", "--module", "commonjs"]
},
{
"cmd": "yarn",
"args": ["jasmine", "out-tsc/**/*.spec.js"]
} }
] ]
} }

View File

@ -0,0 +1,55 @@
import { docRegionFromEvent, docRegionSubscriber } from './creating';
describe('observables', () => {
it('should create an observable using the constructor', () => {
const console = {log: jasmine.createSpy('log')};
docRegionSubscriber(console);
expect(console.log).toHaveBeenCalledTimes(4);
expect(console.log.calls.allArgs()).toEqual([
[1],
[2],
[3],
['Finished sequence'],
]);
});
it('should listen to input changes', () => {
let triggerInputChange;
const input = {
value: 'Test',
addEventListener: jasmine
.createSpy('addEvent')
.and.callFake((eventName: string, cb: (e) => void) => {
if (eventName === 'keydown') {
triggerInputChange = cb;
}
}),
removeEventListener: jasmine.createSpy('removeEventListener'),
};
const document = { getElementById: () => input };
docRegionFromEvent(document);
triggerInputChange({keyCode: 65});
expect(input.value).toBe('Test');
triggerInputChange({keyCode: 27});
expect(input.value).toBe('');
});
it('should call removeEventListener when unsubscribing', (doneFn: DoneFn) => {
const input = {
addEventListener: jasmine.createSpy('addEvent'),
removeEventListener: jasmine
.createSpy('removeEvent')
.and.callFake((eventName: string, cb: (e) => void) => {
if (eventName === 'keydown') {
doneFn();
}
})
};
const document = { getElementById: () => input };
const subscription = docRegionFromEvent(document);
subscription.unsubscribe();
});
});

View File

@ -1,8 +1,9 @@
// #docplaster
import { Observable } from 'rxjs'; import { Observable } from 'rxjs';
export function docRegionSubscriber(console) {
// #docregion subscriber // #docregion subscriber
// This function runs when subscribe() is called // This function runs when subscribe() is called
function sequenceSubscriber(observer) { function sequenceSubscriber(observer) {
// synchronously deliver 1, 2, and 3, then complete // synchronously deliver 1, 2, and 3, then complete
@ -30,8 +31,8 @@ sequence.subscribe({
// 2 // 2
// 3 // 3
// Finished sequence // Finished sequence
// #enddocregion subscriber // #enddocregion subscriber
}
// #docregion fromevent // #docregion fromevent
@ -51,16 +52,18 @@ function fromEvent(target, eventName) {
// #enddocregion fromevent // #enddocregion fromevent
export function docRegionFromEvent(document) {
// #docregion fromevent_use // #docregion fromevent_use
const ESC_KEY = 27; const ESC_KEY = 27;
const nameInput = document.getElementById('name') as HTMLInputElement; const nameInput = document.getElementById('name') as HTMLInputElement;
const subscription = fromEvent(nameInput, 'keydown') const subscription = fromEvent(nameInput, 'keydown').subscribe((e: KeyboardEvent) => {
.subscribe((e: KeyboardEvent) => {
if (e.keyCode === ESC_KEY) { if (e.keyCode === ESC_KEY) {
nameInput.value = ''; nameInput.value = '';
} }
}); });
// #enddocregion fromevent_use // #enddocregion fromevent_use
return subscription;
}

View File

@ -1,5 +1,5 @@
// TODO: Add unit tests for this file.
import { Observable } from 'rxjs'; import { Observable } from 'rxjs';
// #docregion // #docregion
// Create an Observable that will start listening to geolocation updates // Create an Observable that will start listening to geolocation updates

View File

@ -0,0 +1,48 @@
import { docRegionDelaySequence, docRegionMulticastSequence } from './multicasting';
describe('multicasting', () => {
let console;
beforeEach(() => {
jasmine.clock().install();
console = {log: jasmine.createSpy('log')};
});
afterEach(() => {
jasmine.clock().uninstall();
});
it('should create an observable and emit in sequence', () => {
docRegionDelaySequence(console);
jasmine.clock().tick(10000);
expect(console.log).toHaveBeenCalledTimes(12);
expect(console.log.calls.allArgs()).toEqual([
[1],
['1st subscribe: 1'],
['2nd subscribe: 1'],
[2],
['1st subscribe: 2'],
['2nd subscribe: 2'],
[3],
['Finished sequence'],
['1st subscribe: 3'],
['1st sequence finished.'],
['2nd subscribe: 3'],
['2nd sequence finished.']
]);
});
it('should create an observable and multicast the emissions', () => {
docRegionMulticastSequence(console);
jasmine.clock().tick(10000);
expect(console.log).toHaveBeenCalledTimes(7);
expect(console.log.calls.allArgs()).toEqual([
['1st subscribe: 1'],
['1st subscribe: 2'],
['2nd subscribe: 2'],
['1st subscribe: 3'],
['2nd subscribe: 3'],
['1st sequence finished.'],
['2nd sequence finished.']
]);
});
});

View File

@ -1,8 +1,9 @@
// #docplaster
import { Observable } from 'rxjs'; import { Observable } from 'rxjs';
export function docRegionDelaySequence(console) {
// #docregion delay_sequence // #docregion delay_sequence
function sequenceSubscriber(observer) { function sequenceSubscriber(observer) {
const seq = [1, 2, 3]; const seq = [1, 2, 3];
let timeoutId; let timeoutId;
@ -23,9 +24,11 @@ function sequenceSubscriber(observer) {
doInSequence(seq, 0); doInSequence(seq, 0);
// Unsubscribe should clear the timeout to stop execution // Unsubscribe should clear the timeout to stop execution
return {unsubscribe() { return {
unsubscribe() {
clearTimeout(timeoutId); clearTimeout(timeoutId);
}}; }
};
} }
// Create a new Observable that will deliver the above sequence // Create a new Observable that will deliver the above sequence
@ -71,9 +74,10 @@ setTimeout(() => {
// (at 3.5 seconds): 2nd sequence finished // (at 3.5 seconds): 2nd sequence finished
// #enddocregion subscribe_twice // #enddocregion subscribe_twice
}
export function docRegionMulticastSequence(console) {
// #docregion multicast_sequence // #docregion multicast_sequence
function multicastSequenceSubscriber() { function multicastSequenceSubscriber() {
const seq = [1, 2, 3]; const seq = [1, 2, 3];
// Keep track of each observer (one for every active subscription) // Keep track of each observer (one for every active subscription)
@ -84,7 +88,7 @@ function multicastSequenceSubscriber() {
// Return the subscriber function (runs when subscribe() // Return the subscriber function (runs when subscribe()
// function is invoked) // function is invoked)
return (observer) => { return observer => {
observers.push(observer); observers.push(observer);
// When this is the first subscription, start the sequence // When this is the first subscription, start the sequence
if (observers.length === 1) { if (observers.length === 1) {
@ -153,3 +157,4 @@ setTimeout(() => {
// (at 3 seconds): 2nd sequence finished // (at 3 seconds): 2nd sequence finished
// #enddocregion multicast_sequence // #enddocregion multicast_sequence
}

View File

@ -0,0 +1,19 @@
import { docRegionObserver } from './subscribing';
describe('subscribing', () => {
it('should subscribe and emit', () => {
const console = {log: jasmine.createSpy('log')};
docRegionObserver(console);
expect(console.log).toHaveBeenCalledTimes(8);
expect(console.log.calls.allArgs()).toEqual([
['Observer got a next value: 1'],
['Observer got a next value: 2'],
['Observer got a next value: 3'],
['Observer got a complete notification'],
['Observer got a next value: 1'],
['Observer got a next value: 2'],
['Observer got a next value: 3'],
['Observer got a complete notification'],
]);
});
});

View File

@ -1,6 +1,7 @@
// #docplaster
import { of } from 'rxjs';
import { Observable, of } from 'rxjs'; export function docRegionObserver(console) {
// #docregion observer // #docregion observer
// Create simple observable that emits three values // Create simple observable that emits three values
@ -15,6 +16,7 @@ const myObserver = {
// Execute with the observer object // Execute with the observer object
myObservable.subscribe(myObserver); myObservable.subscribe(myObserver);
// Logs: // Logs:
// Observer got a next value: 1 // Observer got a next value: 1
// Observer got a next value: 2 // Observer got a next value: 2
@ -30,3 +32,4 @@ myObservable.subscribe(
() => console.log('Observer got a complete notification') () => console.log('Observer got a complete notification')
); );
// #enddocregion sub_fn // #enddocregion sub_fn
}

View File

@ -2,7 +2,11 @@
"tests": [ "tests": [
{ {
"cmd": "yarn", "cmd": "yarn",
"args": [ "tsc", "--project", "./tsconfig.app.json" ] "args": ["tsc", "--project", "tsconfig.spec.json", "--module", "commonjs"]
},
{
"cmd": "yarn",
"args": ["jasmine", "out-tsc/**/*.spec.js"]
} }
] ]
} }

View File

@ -1,4 +1,4 @@
// TODO: Add unit tests for this file.
import { pipe, range, timer, zip } from 'rxjs'; import { pipe, range, timer, zip } from 'rxjs';
import { ajax } from 'rxjs/ajax'; import { ajax } from 'rxjs/ajax';
import { retryWhen, map, mergeMap } from 'rxjs/operators'; import { retryWhen, map, mergeMap } from 'rxjs/operators';