TestBed
data:image/s3,"s3://crabby-images/03030/030307a061db2e32311b097ebca30ae70b1a68d6" alt=""
Some stats...
Test method | Duration |
---|---|
test (as-is) | 3s (stable) |
test (as-is) | 45s (flaky) |
(without optimization) | 7.5s (stable) |
data:image/s3,"s3://crabby-images/e6cc4/e6cc48fa6ff136f52f499e717b3ac8c0e7b3a995" alt=""
data:image/s3,"s3://crabby-images/88c1a/88c1ac6a85c3fe82d6872ed9d6be019c90f81ade" alt=""
data:image/s3,"s3://crabby-images/03030/030307a061db2e32311b097ebca30ae70b1a68d6" alt=""
E2E
(UI)
Integration
Unit
Isolation
Integration
Fast
Slow
Test pyramid
Components
E2E
(UI)
(Semi) Integration
Unit
Critical user flow testing
Render checks + Edge case testing
Isolated testing
data:image/s3,"s3://crabby-images/88c1a/88c1ac6a85c3fe82d6872ed9d6be019c90f81ade" alt=""
data:image/s3,"s3://crabby-images/03030/030307a061db2e32311b097ebca30ae70b1a68d6" alt=""
data:image/s3,"s3://crabby-images/03030/030307a061db2e32311b097ebca30ae70b1a68d6" alt=""
data:image/s3,"s3://crabby-images/88c1a/88c1ac6a85c3fe82d6872ed9d6be019c90f81ade" alt=""
?
data:image/s3,"s3://crabby-images/e6cc4/e6cc48fa6ff136f52f499e717b3ac8c0e7b3a995" alt=""
Production-like
Test env
+
E2E
App
data:image/s3,"s3://crabby-images/a839f/a839f7968785ca995842c786846b187084dbb5f3" alt=""
data:image/s3,"s3://crabby-images/88c1a/88c1ac6a85c3fe82d6872ed9d6be019c90f81ade" alt=""
App
data:image/s3,"s3://crabby-images/00bb8/00bb8cb5238527729d06e09e35c7ec7a4ddbfa07" alt=""/cdn.vox-cdn.com/uploads/chorus_image/image/59396599/ff12.0.jpg)
App
...
Critical flows
- Limited tests that prove our site main flows work
- Test simulates a user on our site
- Flows could be determined by business
data:image/s3,"s3://crabby-images/2609c/2609cb30689926a77785931380ab2d6efb4137b3" alt=""
E2E
(UI)
data:image/s3,"s3://crabby-images/88c1a/88c1ac6a85c3fe82d6872ed9d6be019c90f81ade" alt=""
data:image/s3,"s3://crabby-images/8c96a/8c96a29dbc0f04e29e73a7975235866ec4805a62" alt=""
==
All user flows work
==
Deploy with confidence
==
Automatic deployment?
So, what do we need to achieve this?
Production-like test env
data:image/s3,"s3://crabby-images/95786/957863d98a947f27ffd55f7a1c6dfd8fed5f280d" alt=""
Proper mock solution
data:image/s3,"s3://crabby-images/74374/7437418c40374ad54e48a7f268d936fbf77a8bda" alt=""
data:image/s3,"s3://crabby-images/e6cc4/e6cc48fa6ff136f52f499e717b3ac8c0e7b3a995" alt=""
- Strong coupling with GSM content schema
- Easier to configure E2E flows
- Easier to maintain
Component testing
Integration
Unit
TagSearch
Filters
...
...
FiltersPanel
data:image/s3,"s3://crabby-images/b7775/b77754bda94c1ebbb33b7be0e68d5636eb290259" alt=""
...
...
Smart/Container
Dumb/Presenter
Dumb/Presenter
[data]
(event)
Search
Unit testing components
FiltersPanel
data:image/s3,"s3://crabby-images/1065b/1065bb8f9a0c6d20648623986c00d480cc02e5ee" alt=""
<feature-filters-option
*ngFor="
let childFilter of filter.children
| slice
: 0
: (showAllOptions[filter.id]
? filter.children.length
: maxShown);
"
>
...
<div
...
[hidden]="
filter.children.length <= maxShown ||
showAllOptions[filter.id]
"
>
<rfs2-link
...
(rfs2Click)="showAll(filter.id)">
showAll(id: string) {
this.showAllOptions[id] = true;
}
data:image/s3,"s3://crabby-images/052be/052be00fe7b04f6511e908b0218af2a68914f789" alt=""
it('should show all filter children', () => {
component.showAll('123');
expect(component.showAllOptions['123']).toBeTruthy();
});
data:image/s3,"s3://crabby-images/6e489/6e489810610f3a21cb4bd82dd571fb65e4ff9840" alt=""
TestBed setup
describe('FiltersPanelComponent', () => {
let fixture: ComponentFixture<FiltersPanelComponent>;
let component: FiltersPanelComponent;
let pageObject: FiltersFeaturePageObject;
beforeEach(() => {
TestBed.configureTestingModule({
imports: [GlobalSitesTestingModule],
declarations: [FiltersPanelComponent, FiltersOptionComponent],
schemas: [CUSTOM_ELEMENTS_SCHEMA],
}).compileComponents();
fixture = TestBed.createComponent(FiltersPanelComponent);
component = fixture.componentInstance;
pageObject = new FiltersPanelPageObject(fixture.nativeElement);
});
it('should render', () => {
expect(component).not.toBeNull();
});
});
FiltersPanel
data:image/s3,"s3://crabby-images/1065b/1065bb8f9a0c6d20648623986c00d480cc02e5ee" alt=""
beforeEach(() => {
component.filters = FILTER_WITH_6_CHILDREN;
fixture.detectChanges();
});
it('should show a limited amount of children', () => {
expect(pageObject.childOptions.length).toBe(3);
});
it('should show all children on show more', () => {
pageObject.clickShowMore(0);
fixture.detectChanges();
expect(pageObject.childOptions.length).toBe(6);
});
FiltersPanel
it('should activate a filter option', () => {
const spy = jest.fn();
component.activate.subscribe(spy);
component.toggleFilter(DEACTIVE_FILTER);
expect(spy).toHaveBeenCalledTimes(1);
expect(spy).toHaveBeenCalledWith(DEACTIVE_FILTER);
});
data:image/s3,"s3://crabby-images/e91ef/e91ef5bddd4476a560b8b5beb70adbfccee87463" alt=""
it('should activate a filter option', () => {
const spy = jest.fn();
component.activate.subscribe(spy);
pageObject.filtersOptions[1].activate();
fixture.detectChanges();
expect(spy).toHaveBeenCalledTimes(1);
expect(spy).toHaveBeenCalledWith(DEACTIVE_FILTER);
});
activate(): void {
this.element.dispatchEvent(
new CustomEvent('rfs2Change', {
detail: true,
})
);
}
Filters
data:image/s3,"s3://crabby-images/b7775/b77754bda94c1ebbb33b7be0e68d5636eb290259" alt=""
data:image/s3,"s3://crabby-images/d33c3/d33c3becfb8cbf9726ca6fb358ed2520bac1d51a" alt=""
<div
*sensesIfBreakpoint="'lg'; to 'xl'"
<!-- Filters -->
...
<ng-container *sensesIfBreakpoint="'xs'; to 'md'">
<feature-filters-modal>
test.each([Breakpoint.xs, Breakpoint.sm, Breakpoint.md])(
'should render on breakpoint %p',
(breakpoint: Breakpoint) => {
windowObserver.breakpoint$ = of(
breakpoint
);
fixture.detectChanges();
expect(pageObject.getModalButton()).not.toBeNull();
}
);
test.each([Breakpoint.lg, Breakpoint.xl])(
'should not render on breakpoint %p',
(breakpoint: Breakpoint) => {
windowObserver.breakpoint$ = of(
breakpoint
);
fixture.detectChanges();
expect(pageObject.getModalButton()).toBeNull();
}
);
Desktop
Mobile
Integration testing components
Article render checks
it('should render', () => {
fixture.detectChanges();
expect(pageObject.relatedMedia.isRendered()).toBeTruthy();
expect(pageObject.relatedMedia.mediaCards.isRendered()).toBeTruthy();
expect(pageObject.relatedMedia.loadMore.isRendered()).toBeTruthy();
// ...
});
data:image/s3,"s3://crabby-images/200c1/200c1a7840c8a55307138e2247e54b6fdb775642" alt=""
Load more related media
Setup integration test
let legacySearchService: Mock<LegacySearchService, jest.SpyInstance>;
beforeEach(() => {
legacySearchService = TestBed.get(LegacySearchService);
jest.spyOn(legacySearchService, 'search');
jest
.spyOn(TestBed.get(HttpClient) as HttpClient, 'get')
.mockReturnValueOnce(of(SEARCH_RESPONSE_3))
.mockReturnValueOnce(of(SEARCH_RESPONSE_6))
.mockReturnValueOnce(of(SEARCH_RESPONSE_7));
});
Test load more results
it('should load more results', () => {
fixture.detectChanges();
expect(legacySearchService.search).toHaveBeenCalledTimes(1);
let searchParams: LegacySearchParams =
legacySearchService.search.mock.calls[0][0];
expect(searchParams.numberOfResults).toBe(3);
expect(pageObject.countMediaCards()).toBe(3);
pageObject.clickLoadMoreButton();
fixture.detectChanges();
expect(legacySearchService.search).toHaveBeenCalledTimes(2);
searchParams = legacySearchService.search.mock.calls[1][0];
expect(searchParams.numberOfResults).toBe(6);
expect(pageObject.countMediaCards()).toBe(6);
});
Test load more button
it('should show load more button only if more results exist', () => {
fixture.detectChanges();
expect(pageObject.relatedMedia.loadMore.isRendered()).toBeTruthy();
pageObject.clickLoadMoreButton(); // 3 -> 6 results
fixture.detectChanges();
pageObject.clickLoadMoreButton(); // 6 -> 7 results
fixture.detectChanges();
expect(pageObject.relatedMedia.loadMore.isRendered()).toBeFalsy();
});
hasLoadMore(): boolean {
return this.data.cards.length < this.totalCount;
}
Complex examples
Translations code
private sortingOptions = [SortDirection.DESC, SortDirection.RELEVANCY];
ngOnInit() {
this.translateService.setTranslations(translations);
this.sortingLinks = this.sortingOptions.map(sort => ({
name: this.translateService.instant(sort),
}));
}
Translations test
it('should translate sorting values', () => {
const DESC_TRANSLATED = 'DESC TRANSLATED';
const RELEVANCY_TRANSLATED = 'RELEVANCY TRANSLATED';
jest
.spyOn(sensesTranslateService, 'instant')
.mockReturnValueOnce(DESC_TRANSLATED)
.mockReturnValueOnce(RELEVANCY_TRANSLATED);
fixture.detectChanges();
expect(sensesTranslateService.instant).toHaveBeenCalledTimes(2);
expect(sensesTranslateService.instant).toHaveBeenNthCalledWith(
1,
SortDirection.DESC
);
expect(sensesTranslateService.instant).toHaveBeenNthCalledWith(
2,
SortDirection.RELEVANCY
);
expect(component.sortingLinks).toEqual([
{ name: DESC_TRANSLATED },
{ name: RELEVANCY_TRANSLATED },
]);
pageObject.toggleSortDropdownMenu();
fixture.detectChanges();
expect(pageObject.sortDropdownMenu.getLinkValues()).toEqual([
DESC_TRANSLATED,
RELEVANCY_TRANSLATED,
]);
});
Async code
// Async animation
animations: [
trigger('openModalAnimation', [
transition(':enter', [
style({ transform: 'translateX(100%)' }),
animate('300ms ease-out', style({ transform: 'translateX(0)' })),
]),
transition(':leave', [
animate('150ms ease-in', style({ transform: 'translateX(100%)' })),
]),
]),
],
<div (sensesClickOutside)="close.emit()">
...
</div>
setTimeout(() =>
this.document.addEventListener(
'click',
this.clickHandler,
this.useCapturing
)
);
Async test
beforeEach(fakeAsync(() => {
TestBed.get(WindowObserver).breakpoint$ = of(
Breakpoint.md
);
fixture.detectChanges();
pageObject.openModal();
fixture.detectChanges();
tick(CLICK_OUTSIDE_DIRECTIVE_TIMEOUT_DURATION); // 0
}));
it('should close filters modal', fakeAsync(() => {
document.dispatchEvent(new MouseEvent('click'));
fixture.detectChanges();
tick(FILTERS_MODAL_COMPONENT_ANIMATION_OUT_DURATION);
expect(pageObject.modal.isRendered()).toBeFalsy();
}));
Future nice-to-haves
- NgBullet (TestBed optimization)
- Spectator (TestBed wrapper)
- Jest snapshot testing
- Storybook visual regression testing
Questions
TestBed
By rachnerd
TestBed
- 176