import { fromPartial } from '@total-typescript/shoehorn'; import type { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient'; import type { ShlinkVisitsOverview } from '../../../src/api/types'; import type { ShlinkState } from '../../../src/container/types'; import { createNewVisits } from '../../../src/shlink-web-component/visits/reducers/visitCreation'; import type { PartialVisitsSummary, VisitsOverview, } from '../../../src/shlink-web-component/visits/reducers/visitsOverview'; import { loadVisitsOverview as loadVisitsOverviewCreator, visitsOverviewReducerCreator, } from '../../../src/shlink-web-component/visits/reducers/visitsOverview'; import type { OrphanVisit } from '../../../src/shlink-web-component/visits/types'; describe('visitsOverviewReducer', () => { const getVisitsOverview = vi.fn(); const buildApiClientMock = () => fromPartial({ getVisitsOverview }); const loadVisitsOverview = loadVisitsOverviewCreator(buildApiClientMock); const { reducer } = visitsOverviewReducerCreator(loadVisitsOverview); describe('reducer', () => { const state = (payload: Partial = {}) => fromPartial(payload); it('returns loading on GET_OVERVIEW_START', () => { const { loading } = reducer( state({ loading: false, error: false }), loadVisitsOverview.pending(''), ); expect(loading).toEqual(true); }); it('stops loading and returns error on GET_OVERVIEW_ERROR', () => { const { loading, error } = reducer( state({ loading: true, error: false }), loadVisitsOverview.rejected(null, ''), ); expect(loading).toEqual(false); expect(error).toEqual(true); }); it('return visits overview on GET_OVERVIEW', () => { const action = loadVisitsOverview.fulfilled(fromPartial({ 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([ fromPartial({ visit: {} }), fromPartial({ visit: {} }), fromPartial({ visit: fromPartial({ visitedUrl: '' }) }), fromPartial({ visit: fromPartial({ visitedUrl: '' }) }), fromPartial({ visit: fromPartial({ 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([ fromPartial({ visit: {} }), fromPartial({ visit: { potentialBot: true } }), fromPartial({ visit: { potentialBot: true } }), fromPartial({ visit: fromPartial({ visitedUrl: '' }) }), fromPartial({ visit: fromPartial({ visitedUrl: '' }) }), fromPartial({ visit: fromPartial({ visitedUrl: '', potentialBot: true }) }), ]), ); expect(nonOrphanVisits).toEqual(expectedNonOrphanVisits); expect(orphanVisits).toEqual(expectedOrphanVisits); }); }); describe('loadVisitsOverview', () => { const dispatchMock = vi.fn(); const getState = () => fromPartial({}); 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 = fromPartial(serverResult); getVisitsOverview.mockResolvedValue(resolvedOverview); await loadVisitsOverview()(dispatchMock, getState, {}); expect(dispatchMock).toHaveBeenCalledTimes(2); expect(dispatchMock).toHaveBeenNthCalledWith(2, expect.objectContaining({ payload: dispatchedPayload })); expect(getVisitsOverview).toHaveBeenCalledTimes(1); }); }); });