gpt4 book ai didi

javascript - 如何使用 mockDOMSource 测试 Cycle.js 中的 Action 流?

转载 作者:搜寻专家 更新时间:2023-10-30 21:24:01 25 4
gpt4 key购买 nike

我意识到使用周期/时间可能有更好的方法,但我只是想了解基础知识。不知何故,我的 action$ 流似乎没有运行;我尝试使用 xs.periodic 构建多个模拟 dom。测试框架是mocha。

import 'mocha';
import {expect} from 'chai';
import xs from 'xstream';
import Stream from 'xstream';
import {mockDOMSource, DOMSource} from '@cycle/dom';
import {HTTPSource} from '@cycle/http';
import XStreamAdapter from '@cycle/xstream-adapter';

export interface Props {
displayAbs: boolean
}

export interface ISources {
DOM: DOMSource;
http: HTTPSource;
}

function testIntent(sources: ISources):Stream<Props> {
return xs.merge<Props>(
sources.DOM
.select('.absShow').events('click')
.mapTo( { displayAbs: true } ),
sources.DOM
.select('.absHide').events('click')
.mapTo( { displayAbs: false } )
).startWith( {displayAbs: false } );
}

describe( 'Test', ()=>{

describe( 'intent()', ()=>{

it('should change on click to shows and hides', () => {
let listenerGotEnd = false;

const mDOM$: Stream<DOMSource> = xs.periodic(1000).take(6).map(ii => {
if (ii % 2 == 0) {
return mockDOMSource(XStreamAdapter, {
'.absShow': {'click': xs.of({target: {}})}
})
}
else {
return mockDOMSource(XStreamAdapter, {
'.absHide': {'click': xs.of({target: {}})}
})
}
});

const action$ = mDOM$.map(mDOM => testIntent({
DOM: mDOM,
http: {} as HTTPSource,
})).flatten();


action$.addListener({
next: (x) => {
console.log("x is " + x.displayAbs);
},
error: (err) => {
console.log("error is:" + err);
throw err;
},
complete: () => { listenerGotEnd = true; }
});
expect(listenerGotEnd).to.equal(true);
});

});/* end of describe intent */

});

最佳答案

测试未运行的主要原因是因为它是异步的,因此在 mocha 中我们需要接收 done 回调,然后在测试完成时调用它。

不使用 @cycle/time ,这就是我编写此测试的方式:

import 'mocha';
import {expect} from 'chai';
import xs, {Stream} from 'xstream';
import {mockDOMSource, DOMSource} from '@cycle/dom';
import XStreamAdapter from '@cycle/xstream-adapter';

export interface Props {
displayAbs: boolean
}

export interface ISources {
DOM: DOMSource;
}

function testIntent(sources: ISources):Stream<Props> {
return xs.merge<Props>(
sources.DOM
.select('.absShow').events('click')
.mapTo( { displayAbs: true } ),
sources.DOM
.select('.absHide').events('click')
.mapTo( { displayAbs: false } )
).startWith( {displayAbs: false } );
}

describe('Test', () => {
describe('intent()', () => {
it('should change on click to shows and hides', (done) => {
const show$ = xs.create();
const hide$ = xs.create();

const DOM = mockDOMSource(XStreamAdapter, {
'.absShow': {
'click': show$
},

'.absHide': {
'click': hide$
}
});

const intent$ = testIntent({DOM});

const expectedValues = [
{displayAbs: false},
{displayAbs: true},
{displayAbs: false},
]

intent$.take(expectedValues.length).addListener({
next: (x) => {
expect(x).to.deep.equal(expectedValues.shift());
},
error: done,
complete: done
});

show$.shamefullySendNext({});
hide$.shamefullySendNext({});
});
});
});

此测试在 11 毫秒内运行,比使用 xs.periodic(1000).take(6)

快一点

为了比较,下面是我将如何使用 @cycle/time 编写它:

import {mockTimeSource} from '@cycle/time'

describe('Test', () => {
describe('intent()', () => {
it('should change on click to shows and hides', (done) => {
const Time = mockTimeSource();

const show$ = Time.diagram('---x-----');
const hide$ = Time.diagram('------x--');
const expected$ = Time.diagram('f--t--f--', {f: false, t: true});

const DOM = mockDOMSource({
'.absShow': {
'click': show$
},

'.absHide': {
'click': hide$
}
});

const intent$ = testIntent({DOM}).map(intent => intent.displayAbs);

Time.assertEqual(intent$, expected$);

Time.run(done);
});
});
});

第一个版本实际上是@cycle/time 在幕后为您做的,这只是一种稍微好一点的编写方式。有更好的错误消息也很好。

关于javascript - 如何使用 mockDOMSource 测试 Cycle.js 中的 Action 流?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42308185/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com