Marble testing is an expressive way to test observables by utilizing marble diagrams. This lesson will walk you through the syntax and features, preparing you to start writing marble tests today!

Grep two files from the rxjs

  • https://github.com/ReactiveX/rxjs/blob/master/spec/helpers/marble-testing.ts
  • https://github.com/ReactiveX/rxjs/blob/master/spec/helpers/test-helper.ts
/*
RxJS marble testing allows for a more natural style of testing observables.
To get started, you need to include a few helpers libraries, marble-testing.ts and test-helper.ts,
in your karma.conf or wallaby.js configuration file.
These files provide helpers for parsing marble diagrams and asserting against the subscription points and result
of your observables under test. For these examples I will be using Jasmine, but Mocha and Chai works just as well. Let's get started with the basics of marble testing! First, let's understand the pieces that make up a valid marble diagram. Dash: Indicates a passing of time, you can think of each dash as 10ms when it comes to your tests.
----- <----- 50ms
Characters: Each character inside the dash indicates an emission.
-----a-----b-----c <----- Emit 'a' at 60ms, 'b' at 120ms, 'c' at 180ms
Pipes |: Pipes indicate the completion point of an observable.
-----a| <----- Emit 'a' at 60ms then complete (70ms)
Parenthesis (): Parenthesis indicate multiple emissions in same time frame, think Observable.of(1,2,3)
-----(abc|) <----- Emit 'a''b''c' at 60ms then complete (60ms)
Caret ^: Indicates the starting point of a subscription, used with expectSubscription assertion.
^------- <----- Subscription point at caret.
Exclamation Point - !: Indicates the end point of a subscription, also used with expectSubscription assertion.
^------! <----- Subscription starts at caret, ends at exclamation point.
Pound Sign - #: Indicates an error
---a---# <----- Emit 'a' at 40ms, error at 80ms
There are also a few methods included to parse marble sequences and transpose values. cold(marbles: string, values?: object, error?: any) : Subscription starts when test begins
cold(--a--b--|, {a: 'Hello', b: 'World'}) <----- Emit 'Hello' at 30ms and 'World' at 60ms, complete at 90ms
hot(marbles: string, values?: object, error?: any) : Behaves like subscription starts at point of caret
hot(--^--a---b--|, {a: 'Goodbye', b: 'World'}) <----- Subscription begins at point of caret
*/

For example we want to test:

const source =       "---a---b---c--|";
const expected = "---a---b---c--|";

they should be equal.

Here each '-' means 1. frames.

'|' means completed.

The method we need to use is 'expectObservable' & 'cold':

    it('should parse marble diagrams', () => {
const source = cold('---a---b---c---|');
const expected = '---a---b---c---|'; expectObservable(source).toBe(expected)
});

Cold will treat the beginning of the diagram as a subscription point. Now the test passing.

But if we change a little bit:

    it('should parse marble diagrams', () => {
const source = cold('---a---b---c---|');
const expected = '---a--b---c---|'; expectObservable(source).toBe(expected)
});

It reports error:

    Expected
{"frame":30,"notification":{"kind":"N","value":"a","hasValue":true}}
{"frame":,"notification":{"kind":"N","value":"b","hasValue":true}}
{"frame":,"notification":{"kind":"N","value":"c","hasValue":true}}
{"frame":,"notification":{"kind":"C","hasValue":false}} to deep equal
{"frame":30,"notification":{"kind":"N","value":"a","hasValue":true}}
{"frame":,"notification":{"kind":"N","value":"b","hasValue":true}}
{"frame":,"notification":{"kind":"N","value":"c","hasValue":true}}
{"frame":,"notification":{"kind":"C","hasValue":false}}

Test 'concat' opreator:

    it('should work with cold observables', () => {
const obs1 = cold('-a---b-|');
const obs2 = cold('-c---d-|');
const expectedConcatRes = '-a---b--c---d-|'; expectObservable(obs1.concat(obs2)).toBe(expectedConcatRes)
});

'Hot' observable: Hot will actually let you identify the subscription point yourself:

When testing hot observables you can specify the subscription point using a caret '^', similar to how you specify subscriptions when utilizing the expectSubscriptions assertion.

    it('should work with hot observables', () => {
const obs1 = hot('---a--^--b---|');
const obs2 = hot('-----c---^-----------------d-|');
const expected = '---b--------------d-|'; expectObservable(obs1.concat(obs2)).toBe(expected);
});

Algin the ^, easy for read

Spread subscription and marble diagram:

    /*
For certain operators you may want to confirm the point at which
an observable is subscribed or unsubscribed. Marble testing makes this
possible by using the expectSubscriptions helper method. The cold and hot
methods return a subscriptions object, including the frame at which the observable
would be subscribed and unsubscribed. You can then assert against these
subscription points by supplying a diagram which indicates the expected behavior. ^ - Indicated the subscription point.
! - Indicates the point at which the observable was unsubscribed. Example subscriptions object: {"subscribedFrame":70,"unsubscribedFrame":140}
*/
it('should identify subscription points', () => {
const obs1 = cold('-a---b-|');
const obs2 = cold('-c---d-|')
const expected = '-a---b--c---d-|';
const sub1 = '^------!'
const sub2 = '-------^------!' expectObservable(obs1.concat(obs2)).toBe(expected);
expectSubscriptions(obs1.subscriptions).toBe(sub1);
expectSubscriptions(obs2.subscriptions).toBe(sub2);
})

Object to map the key and value:

    /*
Both the hot and cold methods, as well the the toBe method accept an object map as a
second parameter, indicating the values to output for the appropriate placeholder.
When the test is executed these values rather than the matching string in the marble diagram.
*/
it('should correctly sub in values', () => {
const values = {a: 3, b: 2};
const source = cold( '---a---b---|', values);
const expected = '---a---b---|'; expectObservable(source).toBe(expected, values);
});
    /*
Multiple emissions occuring in same time frame can be represented by grouping in parenthesis.
Complete and error symbols can also be included in the same grouping as simulated outputs.
*/
it('should handle emissions in same time frame', () => {
const obs1 = Observable.of(1,2,3,4);
const expected = '(abcd|)'; expectObservable(obs1).toBe(expected, {a: 1, b: 2, c: 3, d: 4});
});
    /*
For asynchronous tests RxJS supplies a TestScheduler.
How it works...
*/
it('should work with asynchronous operators', () => {
const obs1 = Observable
.interval(10, rxTestScheduler)
.take(5)
.filter(v => v % 2 === 0);
const expected = '-a-b-(c|)'; expectObservable(obs1).toBe(expected, {a: 0, b: 2, c: 4});
});

Error handling:

    /*
Observables that encounter errors are represented by the pound (#) sign.
In this case, our observable is retried twice before ultimately emitting an error.
A third value can be supplied to the toBe method specifying the error to be matched.
*/
it('should handle errors', () => {
const source = Observable.of(1,2,3,4)
.map(val => {
if(val > 3){
throw 'Number too high!';
};
return val;
})
.retry(2); const expected = '(abcabcabc#)'; expectObservable(source).toBe(expected, {a: 1, b: 2, c: 3, d: 4}, 'Number too high!');
});

[RxJS] Introduction to RxJS Marble Testing的更多相关文章

  1. [AngularJS + RxJS] Search with RxJS

    When doing search function, you always need to consider about the concurrent requests. AEvent ----(6 ...

  2. [RxJS] Split an RxJS Observable into groups with groupBy

    groupBy() is another RxJS operator to create higher order observables. In this lesson we will learn ...

  3. [RxJS] Split an RxJS observable conditionally with windowToggle

    There are variants of the window operator that allow you to split RxJS observables in different ways ...

  4. [RxJS] Split an RxJS observable with window

    Mapping the values of an observable to many inner observables is not the only way to create a higher ...

  5. RxJS v6 学习指南

    为什么要使用 RxJS RxJS 是一套处理异步编程的 API,那么我将从异步讲起. 前端编程中的异步有:事件(event).AJAX.动画(animation).定时器(timer). 异步常见的问 ...

  6. RxJS速成 (上)

    What is RxJS? RxJS是ReactiveX编程理念的JavaScript版本.ReactiveX是一种针对异步数据流的编程.简单来说,它将一切数据,包括HTTP请求,DOM事件或者普通数 ...

  7. RxJS速成 (下)

    上一部分: http://www.cnblogs.com/cgzl/p/8641738.html Subject Subject比较特殊, 它即是Observable又是Observer. 作为Obs ...

  8. RxJS入门

    一.RxJS是什么? 官方文档使用了一句话总结RxJS: Think of RxJS as Lodash for events.那么Lodash主要解决了什么问题?Lodash主要集成了一系列关于数组 ...

  9. RxJS 6有哪些新变化?

    我们的前端工程由Angular4升级到Angular6,rxjs也要升级到rxjs6.  rxjs6的语法做了很大的改动,幸亏引入了rxjs-compact包,否则升级工作会无法按时完成. 按照官方的 ...

随机推荐

  1. bjfu1284 判别正则表达式

    做解析器做得多的我,一上来就觉得要写解析器,麻烦,于是想偷懒用java的正则表达式类Pattern直接进行判断,结果wa了,原因是这题要求的正则表达式只是真正正则表达式的一个子集.比如|12是合法正则 ...

  2. 多校1005 HDU5785 Interesting (manacher)

    // 多校1005 HDU5785 Interesting // 题意:给你一个串,求相邻两个回文串左边端点*右边端点的和 // 思路:马拉车算出最长回文半径,求一个前缀和,既得到每个点对答案的贡献. ...

  3. PHP+Apache+MySQL+phpMyAdmin在win7系统下的环境配置

    配置方法在网上可以搜到很多,一步步来就好了,但是由于步骤比较多,需要耐心仔细一点点,这是我自己记录的成功步骤: 1.PHP+Apache+MySQL的安装:PHP网站开发 2.phpMyAdmin的配 ...

  4. [Hive - LanguageManual] Alter Table/Partition/Column

    Alter Table/Partition/Column Alter Table Rename Table Alter Table Properties Alter Table Comment Add ...

  5. data audit on hadoop fs

    最近项目中遇到了存储在HDFS上的数据格式不对,是由于数据中带有\r\n的字符,程序处理的时候没有考虑到这些情况.历史数据大概有一年的时间,需要把错误的数据或者重复的数据给删除了,保留正确的数据,项目 ...

  6. Windows Server 2012 四个版本对比

    Windows Server 2012 有4种版本: Foundation, Essentials, Standard and Datacenter. 版本 Foundation Essentials ...

  7. Apache Spark GraphX

    GraphX基于BSP模型,在Spark之上封装类似Pregel的接口,进行大规模同步全局的图计算,尤其是当用户进行多轮迭代时,基于Spark内存计算的优势尤为明显.

  8. homework-01 "最大子数组之和"的问题求解过程

    写在前面:我的算法能力很弱,并且也是第一次写博文,总之希望自己能在这次的课程中学到很多贴近实践的东西吧. 1.这次的程序是python写的,这也算是我第一次正正经经地拿python来写东西,结果上来说 ...

  9. hadoop streaming 编程

    概况 Hadoop Streaming 是一个工具, 代替编写Java的实现类,而利用可执行程序来完成map-reduce过程.一个最简单的程序 $HADOOP_HOME/bin/hadoop jar ...

  10. AutoCAD.NET二次开发:创建自定义菜单的两种方法比较

    目前我已经掌握的创建CAD菜单方法有两种: COM方式: http://www.cnblogs.com/bomb12138/p/3607929.html CUI方式: http://www.cnblo ...