[Unit testing RxJS] Test complex logic with time progression syntax
For example we have a search input:
const input$ = fromEvent(document.getElementById("#input"), "input");
input$
.pipe(
debounceTime(200),
pluck("target", "value"),
distinctUntilChanged(),
switchMap((searchTerm) => {
ajax
.getJSON(`${BASE_URL}?by_name=${searchTerm}`)
.pipe(catchError(() => EMPTY));
})
)
.subscribe();
We want to test the logic about the input box.
First thing we need to do is making it easy to mock ajax
request.
const typeahead =
(ajaxHelper = ajax) =>
(sourceObservable) => {
return sourceObservable.pipe(
debounceTime(200),
pluck("target", "value"),
distinctUntilChanged(),
switchMap((searchTerm) => {
ajaxHelper
.getJSON(`${BASE_URL}?by_name=${searchTerm}`)
.pipe(catchError(() => EMPTY));
})
);
};
input$.pipe(typeahead()).subscribe();
const { of, throwError } = require("rxjs");
const { delay } = require("rxjs/operators");
const { TestScheduler } = require("rxjs/testing");
const { typeahead } = require("..");
describe("THe typehead", () => {
beforeEach(() => {
testScheduler = new TestScheduler((actual, expected) => {
expect(actual).toEqual(expected);
});
});
it("should debounce input by 200ms", () => {
testScheduler.run((helpers) => {
const { cold, expectObservable } = helpers;
const searchTerm = "testing";
const source$ = cold("a", { a: { target: { value: searchTerm } } });
const final$ = source$.pipe(
typeahead({
getJSON: () => of(searchTerm).pipe(delay(300)),
})
);
const expected = "500ms a";
expectObservable(final$).toBe(expected, { a: searchTerm });
});
});
it("should cancel active requests if another value is emitted", () => {
testScheduler.run((helpers) => {
const { cold, expectObservable } = helpers;
const searchTerm = "testing";
const source$ = cold("a 250ms b", {
a: { target: { value: "first" } },
b: { target: { value: "second" } },
});
const final$ = source$.pipe(
typeahead({
getJSON: () => of(searchTerm).pipe(delay(300)),
})
);
const expected = "751ms b"; // 200 (debounce) + 250 (wait time) + 300 (network) + 1 (b)
expectObservable(final$).toBe(expected, { b: searchTerm });
});
});
it("should not emit duplciate values in a row", () => {
testScheduler.run((helpers) => {
const { cold, expectObservable } = helpers;
const searchTerm = "testing";
const source$ = cold("a 250ms b", {
a: { target: { value: "same" } },
b: { target: { value: "same" } },
});
const final$ = source$.pipe(
typeahead({
getJSON: () => of(searchTerm).pipe(delay(300)),
})
);
const expected = "500ms b"; // 200 (debounce) + 300 (network)
expectObservable(final$).toBe(expected, { b: searchTerm });
});
});
it("should ingore ajax errors", () => {
testScheduler.run((helpers) => {
const { cold, expectObservable } = helpers;
const searchTerm = "testing";
const source$ = cold("a 250ms b", {
a: { target: { value: "same" } },
b: { target: { value: "same" } },
});
const final$ = source$.pipe(
typeahead({
getJSON: () => throwError("error"),
})
);
const expected = ""; // in case of error, emit empty
expectObservable(final$).toBe(expected);
});
});
});
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具
2021-10-14 [Cloud Architect] 11. Protecting Data Stored in the Cloud
2020-10-14 [Kotlin] Using High Order Function
2020-10-14 [Kotlin] Exercise for Map
2020-10-14 [Kotlin] Working with Map
2020-10-14 [Kotlin] File I/O
2020-10-14 [Kotlin] Remove duplicate item from Array with toHashSet
2020-10-14 [RxJS] onErrorResumeNext