From 9cefdb7977909f8f647014a50c01682c5e78c14d Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sat, 18 Mar 2023 12:09:38 +0100 Subject: [PATCH] First refactor of redux tests to avoid covering RTK implementation details --- test/app/reducers/appUpdates.test.ts | 16 +---- test/common/reducer/sidebar.test.ts | 20 ++----- test/domains/reducers/domainRedirects.test.ts | 13 ---- test/domains/reducers/domainsList.test.ts | 59 +++++-------------- 4 files changed, 22 insertions(+), 86 deletions(-) diff --git a/test/app/reducers/appUpdates.test.ts b/test/app/reducers/appUpdates.test.ts index 1289132f..71c9992f 100644 --- a/test/app/reducers/appUpdates.test.ts +++ b/test/app/reducers/appUpdates.test.ts @@ -3,23 +3,11 @@ import { appUpdateAvailable, appUpdatesReducer, resetAppUpdate } from '../../../ describe('appUpdatesReducer', () => { describe('reducer', () => { it('returns true on APP_UPDATE_AVAILABLE', () => { - expect(appUpdatesReducer(undefined, { type: appUpdateAvailable.toString() })).toEqual(true); + expect(appUpdatesReducer(undefined, appUpdateAvailable())).toEqual(true); }); it('returns false on RESET_APP_UPDATE', () => { - expect(appUpdatesReducer(undefined, { type: resetAppUpdate.toString() })).toEqual(false); - }); - }); - - describe('appUpdateAvailable', () => { - it('creates expected action', () => { - expect(appUpdateAvailable()).toEqual({ type: appUpdateAvailable.toString() }); - }); - }); - - describe('resetAppUpdate', () => { - it('creates expected action', () => { - expect(resetAppUpdate()).toEqual({ type: resetAppUpdate.toString() }); + expect(appUpdatesReducer(undefined, resetAppUpdate())).toEqual(false); }); }); }); diff --git a/test/common/reducer/sidebar.test.ts b/test/common/reducer/sidebar.test.ts index 146529c5..96194952 100644 --- a/test/common/reducer/sidebar.test.ts +++ b/test/common/reducer/sidebar.test.ts @@ -3,22 +3,10 @@ import { sidebarNotPresent, sidebarPresent, sidebarReducer } from '../../../src/ describe('sidebarReducer', () => { describe('reducer', () => { it.each([ - [sidebarPresent.toString(), { sidebarPresent: true }], - [sidebarNotPresent.toString(), { sidebarPresent: false }], - ])('returns expected on %s', (type, expected) => { - expect(sidebarReducer(undefined, { type })).toEqual(expected); - }); - }); - - describe('sidebarPresent', () => { - it('returns expected action', () => { - expect(sidebarPresent()).toEqual({ type: sidebarPresent.toString() }); - }); - }); - - describe('sidebarNotPresent', () => { - it('returns expected action', () => { - expect(sidebarNotPresent()).toEqual({ type: sidebarNotPresent.toString() }); + [sidebarPresent, { sidebarPresent: true }], + [sidebarNotPresent, { sidebarPresent: false }], + ])('returns expected on %s', (actionCreator, expected) => { + expect(sidebarReducer(undefined, actionCreator())).toEqual(expected); }); }); }); diff --git a/test/domains/reducers/domainRedirects.test.ts b/test/domains/reducers/domainRedirects.test.ts index c33af060..99c41554 100644 --- a/test/domains/reducers/domainRedirects.test.ts +++ b/test/domains/reducers/domainRedirects.test.ts @@ -16,18 +16,6 @@ describe('domainRedirectsReducer', () => { const buildShlinkApiClient = () => Mock.of({ editDomainRedirects: editDomainRedirectsCall }); const editDomainRedirectsAction = editDomainRedirects(buildShlinkApiClient); - it('dispatches error when loading domains fails', async () => { - editDomainRedirectsCall.mockRejectedValue(new Error('error')); - - await editDomainRedirectsAction(Mock.of({ domain }))(dispatch, getState, {}); - - expect(dispatch).toHaveBeenCalledTimes(2); - expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ - type: editDomainRedirectsAction.rejected.toString(), - })); - expect(editDomainRedirectsCall).toHaveBeenCalledTimes(1); - }); - it('dispatches domain and redirects once loaded', async () => { editDomainRedirectsCall.mockResolvedValue(redirects); @@ -35,7 +23,6 @@ describe('domainRedirectsReducer', () => { expect(dispatch).toHaveBeenCalledTimes(2); expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ - type: editDomainRedirectsAction.fulfilled.toString(), payload: { domain, redirects }, })); expect(editDomainRedirectsCall).toHaveBeenCalledTimes(1); diff --git a/test/domains/reducers/domainsList.test.ts b/test/domains/reducers/domainsList.test.ts index ba880183..842f8ab5 100644 --- a/test/domains/reducers/domainsList.test.ts +++ b/test/domains/reducers/domainsList.test.ts @@ -4,6 +4,7 @@ import type { ShlinkDomainRedirects } from '../../../src/api/types'; import { parseApiError } from '../../../src/api/utils'; import type { ShlinkState } from '../../../src/container/types'; import type { Domain } from '../../../src/domains/data'; +import type { EditDomainRedirects } from '../../../src/domains/reducers/domainRedirects'; import { editDomainRedirects } from '../../../src/domains/reducers/domainRedirects'; import type { DomainsList } from '../../../src/domains/reducers/domainsList'; @@ -25,7 +26,7 @@ describe('domainsListReducer', () => { Mock.of({ domain: 'Boo', status: 'validating' }), ]; const domains = [...filteredDomains, Mock.of({ domain: 'bar', status: 'validating' })]; - const error = { type: 'NOT_FOUND', status: 404 }; + const error = { type: 'NOT_FOUND', status: 404 } as unknown as Error; const editDomainRedirectsThunk = editDomainRedirects(buildShlinkApiClient); const { reducer, listDomains: listDomainsAction, checkDomainHealth, filterDomains } = domainsListReducerCreator( buildShlinkApiClient, @@ -36,27 +37,25 @@ describe('domainsListReducer', () => { describe('reducer', () => { it('returns loading on LIST_DOMAINS_START', () => { - expect(reducer(undefined, { type: listDomainsAction.pending.toString() })).toEqual( + expect(reducer(undefined, listDomainsAction.pending(''))).toEqual( { domains: [], filteredDomains: [], loading: true, error: false }, ); }); it('returns error on LIST_DOMAINS_ERROR', () => { - expect(reducer(undefined, { type: listDomainsAction.rejected.toString(), error })).toEqual( + expect(reducer(undefined, listDomainsAction.rejected(error, ''))).toEqual( { domains: [], filteredDomains: [], loading: false, error: true, errorData: parseApiError(error) }, ); }); it('returns domains on LIST_DOMAINS', () => { expect( - reducer(undefined, { type: listDomainsAction.fulfilled.toString(), payload: { domains } }), + reducer(undefined, listDomainsAction.fulfilled({ domains }, '')), ).toEqual({ domains, filteredDomains: domains, loading: false, error: false }); }); it('filters domains on FILTER_DOMAINS', () => { - expect( - reducer(Mock.of({ domains }), { type: filterDomains.toString(), payload: 'oO' }), - ).toEqual({ domains, filteredDomains }); + expect(reducer(Mock.of({ domains }), filterDomains('oO'))).toEqual({ domains, filteredDomains }); }); it.each([ @@ -69,13 +68,14 @@ describe('domainsListReducer', () => { regular404Redirect: 'foo', invalidShortUrlRedirect: null, }; + const editDomainRedirects: EditDomainRedirects = { domain, redirects }; - expect(reducer(Mock.of({ domains, filteredDomains }), { - type: editDomainRedirectsThunk.fulfilled.toString(), - payload: { domain, redirects }, - })).toEqual({ - domains: domains.map(replaceRedirectsOnDomain({ domain, redirects })), - filteredDomains: filteredDomains.map(replaceRedirectsOnDomain({ domain, redirects })), + expect(reducer( + Mock.of({ domains, filteredDomains }), + editDomainRedirectsThunk.fulfilled(editDomainRedirects, '', editDomainRedirects), + )).toEqual({ + domains: domains.map(replaceRedirectsOnDomain(editDomainRedirects)), + filteredDomains: filteredDomains.map(replaceRedirectsOnDomain(editDomainRedirects)), }); }); @@ -86,10 +86,7 @@ describe('domainsListReducer', () => { ])('replaces status on proper domain on VALIDATE_DOMAIN', (domain) => { expect(reducer( Mock.of({ domains, filteredDomains }), - { - type: checkDomainHealth.fulfilled.toString(), - payload: { domain, status: 'valid' }, - }, + checkDomainHealth.fulfilled({ domain, status: 'valid' }, '', ''), )).toEqual({ domains: domains.map(replaceStatusOnDomain(domain, 'valid')), filteredDomains: filteredDomains.map(replaceStatusOnDomain(domain, 'valid')), @@ -98,32 +95,13 @@ describe('domainsListReducer', () => { }); describe('listDomains', () => { - it('dispatches error when loading domains fails', async () => { - listDomains.mockRejectedValue(new Error('error')); - - await listDomainsAction()(dispatch, getState, {}); - - expect(dispatch).toHaveBeenCalledTimes(2); - expect(dispatch).toHaveBeenNthCalledWith(1, expect.objectContaining({ - type: listDomainsAction.pending.toString(), - })); - expect(dispatch).toHaveBeenNthCalledWith(2, expect.objectContaining({ - type: listDomainsAction.rejected.toString(), - })); - expect(listDomains).toHaveBeenCalledTimes(1); - }); - it('dispatches domains once loaded', async () => { listDomains.mockResolvedValue({ data: domains }); await listDomainsAction()(dispatch, getState, {}); expect(dispatch).toHaveBeenCalledTimes(2); - expect(dispatch).toHaveBeenNthCalledWith(1, expect.objectContaining({ - type: listDomainsAction.pending.toString(), - })); - expect(dispatch).toHaveBeenNthCalledWith(2, expect.objectContaining({ - type: listDomainsAction.fulfilled.toString(), + expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ payload: { domains }, })); expect(listDomains).toHaveBeenCalledTimes(1); @@ -136,9 +114,7 @@ describe('domainsListReducer', () => { ['bar'], ['something'], ])('creates action as expected', (searchTerm) => { - expect(filterDomains(searchTerm)).toEqual( - expect.objectContaining({ type: filterDomains.toString(), payload: searchTerm }), - ); + expect(filterDomains(searchTerm).payload).toEqual(searchTerm); }); }); @@ -155,7 +131,6 @@ describe('domainsListReducer', () => { expect(getState).toHaveBeenCalledTimes(1); expect(health).not.toHaveBeenCalled(); expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ - type: checkDomainHealth.fulfilled.toString(), payload: { domain, status: 'invalid' }, })); }); @@ -174,7 +149,6 @@ describe('domainsListReducer', () => { expect(getState).toHaveBeenCalledTimes(1); expect(health).toHaveBeenCalledTimes(1); expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ - type: checkDomainHealth.fulfilled.toString(), payload: { domain, status: 'invalid' }, })); }); @@ -199,7 +173,6 @@ describe('domainsListReducer', () => { expect(getState).toHaveBeenCalledTimes(1); expect(health).toHaveBeenCalledTimes(1); expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ - type: checkDomainHealth.fulfilled.toString(), payload: { domain, status: expectedStatus }, })); });