import { Mock } from 'ts-mockery'; import type { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient'; import type { ShlinkVisitsOverview } from '../../../src/api/types'; import type { ShlinkState } from '../../../src/container/types'; import type { CreateVisitsAction } from '../../../src/visits/reducers/visitCreation'; import { createNewVisits } from '../../../src/visits/reducers/visitCreation'; import type { GetVisitsOverviewAction, ParsedVisitsOverview, PartialVisitsSummary, VisitsOverview } from '../../../src/visits/reducers/visitsOverview'; import { loadVisitsOverview as loadVisitsOverviewCreator, visitsOverviewReducerCreator, } from '../../../src/visits/reducers/visitsOverview'; import type { CreateVisit, OrphanVisit, Visit } from '../../../src/visits/types'; describe('visitsOverviewReducer', () => { const getVisitsOverview = jest.fn(); const buildApiClientMock = () => Mock.of({ getVisitsOverview }); const loadVisitsOverview = loadVisitsOverviewCreator(buildApiClientMock); const { reducer } = visitsOverviewReducerCreator(loadVisitsOverview); beforeEach(jest.clearAllMocks); describe('reducer', () => { const action = (type: string) => Mock.of({ type }) as GetVisitsOverviewAction & CreateVisitsAction; const state = (payload: Partial = {}) => Mock.of(payload); it('returns loading on GET_OVERVIEW_START', () => { const { loading } = reducer( state({ loading: false, error: false }), action(loadVisitsOverview.pending.toString()), ); expect(loading).toEqual(true); }); it('stops loading and returns error on GET_OVERVIEW_ERROR', () => { const { loading, error } = reducer( state({ loading: true, error: false }), action(loadVisitsOverview.rejected.toString()), ); expect(loading).toEqual(false); expect(error).toEqual(true); }); it('return visits overview on GET_OVERVIEW', () => { const action = loadVisitsOverview.fulfilled(Mock.of({ nonOrphanVisits: { total: 100 }, }), 'requestId'); const { loading, error, nonOrphanVisits } = reducer(state({ loading: true, error: false }), action); expect(loading).toEqual(false); expect(error).toEqual(false); expect(nonOrphanVisits.total).toEqual(100); }); it.each([ [50, 53], [0, 3], ])('returns updated amounts on CREATE_VISITS', (providedOrphanVisitsCount, expectedOrphanVisitsCount) => { const { nonOrphanVisits, orphanVisits } = reducer( state({ nonOrphanVisits: { total: 100 }, orphanVisits: { total: providedOrphanVisitsCount }, }), createNewVisits([ Mock.of({ visit: Mock.all() }), Mock.of({ visit: Mock.all() }), Mock.of({ visit: Mock.of({ visitedUrl: '' }), }), Mock.of({ visit: Mock.of({ visitedUrl: '' }), }), Mock.of({ visit: Mock.of({ visitedUrl: '' }), }), ]), ); expect(nonOrphanVisits.total).toEqual(102); expect(orphanVisits.total).toEqual(expectedOrphanVisitsCount); }); it.each([ [ {} satisfies Omit, {} satisfies Omit, { total: 103 } satisfies PartialVisitsSummary, { total: 203 } satisfies PartialVisitsSummary, ], [ { bots: 35 } satisfies Omit, { bots: 35 } satisfies Omit, { total: 103, bots: 37 } satisfies PartialVisitsSummary, { total: 203, bots: 36 } satisfies PartialVisitsSummary, ], [ { nonBots: 41, bots: 85 } satisfies Omit, { nonBots: 63, bots: 27 } satisfies Omit, { total: 103, nonBots: 42, bots: 87 } satisfies PartialVisitsSummary, { total: 203, nonBots: 65, bots: 28 } satisfies PartialVisitsSummary, ], [ { nonBots: 56 } satisfies Omit, { nonBots: 99 } satisfies Omit, { total: 103, nonBots: 57 } satisfies PartialVisitsSummary, { total: 203, nonBots: 101 } satisfies PartialVisitsSummary, ], ])('takes bots and non-bots into consideration when creating visits', ( initialNonOrphanVisits, initialOrphanVisits, expectedNonOrphanVisits, expectedOrphanVisits, ) => { const { nonOrphanVisits, orphanVisits } = reducer( state({ nonOrphanVisits: { total: 100, ...initialNonOrphanVisits }, orphanVisits: { total: 200, ...initialOrphanVisits }, }), createNewVisits([ Mock.of({ visit: Mock.all() }), Mock.of({ visit: Mock.of({ potentialBot: true }) }), Mock.of({ visit: Mock.of({ potentialBot: true }) }), Mock.of({ visit: Mock.of({ visitedUrl: '' }), }), Mock.of({ visit: Mock.of({ visitedUrl: '' }), }), Mock.of({ visit: Mock.of({ visitedUrl: '', potentialBot: true }), }), ]), ); expect(nonOrphanVisits).toEqual(expectedNonOrphanVisits); expect(orphanVisits).toEqual(expectedOrphanVisits); }); }); describe('loadVisitsOverview', () => { const dispatchMock = jest.fn(); const getState = () => Mock.of(); beforeEach(() => dispatchMock.mockReset()); it('dispatches start and error when promise is rejected', async () => { getVisitsOverview.mockRejectedValue(undefined); await loadVisitsOverview()(dispatchMock, getState, {}); expect(dispatchMock).toHaveBeenCalledTimes(2); expect(dispatchMock).toHaveBeenNthCalledWith(1, expect.objectContaining({ type: loadVisitsOverview.pending.toString(), })); expect(dispatchMock).toHaveBeenNthCalledWith(2, expect.objectContaining({ type: loadVisitsOverview.rejected.toString(), })); expect(getVisitsOverview).toHaveBeenCalledTimes(1); }); it.each([ [ // Shlink <3.5.0 { visitsCount: 50, orphanVisitsCount: 20 } satisfies ShlinkVisitsOverview, { nonOrphanVisits: { total: 50, nonBots: undefined, bots: undefined }, orphanVisits: { total: 20, nonBots: undefined, bots: undefined }, }, ], [ // Shlink >=3.5.0 { nonOrphanVisits: { total: 50, nonBots: 20, bots: 30 }, orphanVisits: { total: 50, nonBots: 20, bots: 30 }, visitsCount: 3, orphanVisitsCount: 3, } satisfies ShlinkVisitsOverview, { nonOrphanVisits: { total: 50, nonBots: 20, bots: 30 }, orphanVisits: { total: 50, nonBots: 20, bots: 30 }, }, ], ])('dispatches start and success when promise is resolved', async (serverResult, dispatchedPayload) => { const resolvedOverview = Mock.of(serverResult); getVisitsOverview.mockResolvedValue(resolvedOverview); await loadVisitsOverview()(dispatchMock, getState, {}); expect(dispatchMock).toHaveBeenCalledTimes(2); expect(dispatchMock).toHaveBeenNthCalledWith(1, expect.objectContaining({ type: loadVisitsOverview.pending.toString(), })); expect(dispatchMock).toHaveBeenNthCalledWith(2, expect.objectContaining({ type: loadVisitsOverview.fulfilled.toString(), payload: dispatchedPayload, })); expect(getVisitsOverview).toHaveBeenCalledTimes(1); }); }); });