2018-09-08 09:49:49 +03:00
|
|
|
import * as sinon from 'sinon';
|
|
|
|
import reducer, {
|
2018-12-18 16:32:02 +03:00
|
|
|
getShortUrlVisits,
|
2019-03-08 21:45:35 +03:00
|
|
|
cancelGetShortUrlVisits,
|
2018-09-08 09:49:49 +03:00
|
|
|
GET_SHORT_URL_VISITS_START,
|
|
|
|
GET_SHORT_URL_VISITS_ERROR,
|
|
|
|
GET_SHORT_URL_VISITS,
|
2019-03-04 21:21:46 +03:00
|
|
|
GET_SHORT_URL_VISITS_LARGE,
|
2019-03-08 21:40:43 +03:00
|
|
|
GET_SHORT_URL_VISITS_CANCEL,
|
2018-09-08 09:49:49 +03:00
|
|
|
} from '../../../src/visits/reducers/shortUrlVisits';
|
|
|
|
|
|
|
|
describe('shortUrlVisitsReducer', () => {
|
|
|
|
describe('reducer', () => {
|
|
|
|
it('returns loading on GET_SHORT_URL_VISITS_START', () => {
|
|
|
|
const state = reducer({ loading: false }, { type: GET_SHORT_URL_VISITS_START });
|
|
|
|
const { loading } = state;
|
|
|
|
|
|
|
|
expect(loading).toEqual(true);
|
|
|
|
});
|
|
|
|
|
2019-03-04 21:21:46 +03:00
|
|
|
it('returns loadingLarge on GET_SHORT_URL_VISITS_LARGE', () => {
|
|
|
|
const state = reducer({ loadingLarge: false }, { type: GET_SHORT_URL_VISITS_LARGE });
|
|
|
|
const { loadingLarge } = state;
|
|
|
|
|
|
|
|
expect(loadingLarge).toEqual(true);
|
|
|
|
});
|
|
|
|
|
2019-03-08 21:40:43 +03:00
|
|
|
it('returns cancelLoad on GET_SHORT_URL_VISITS_CANCEL', () => {
|
|
|
|
const state = reducer({ cancelLoad: false }, { type: GET_SHORT_URL_VISITS_CANCEL });
|
|
|
|
const { cancelLoad } = state;
|
|
|
|
|
|
|
|
expect(cancelLoad).toEqual(true);
|
|
|
|
});
|
|
|
|
|
2018-09-08 09:49:49 +03:00
|
|
|
it('stops loading and returns error on GET_SHORT_URL_VISITS_ERROR', () => {
|
|
|
|
const state = reducer({ loading: true, error: false }, { type: GET_SHORT_URL_VISITS_ERROR });
|
|
|
|
const { loading, error } = state;
|
|
|
|
|
|
|
|
expect(loading).toEqual(false);
|
|
|
|
expect(error).toEqual(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('return visits on GET_SHORT_URL_VISITS', () => {
|
|
|
|
const actionVisits = [{}, {}];
|
|
|
|
const state = reducer({ loading: true, error: false }, { type: GET_SHORT_URL_VISITS, visits: actionVisits });
|
|
|
|
const { loading, error, visits } = state;
|
|
|
|
|
|
|
|
expect(loading).toEqual(false);
|
|
|
|
expect(error).toEqual(false);
|
|
|
|
expect(visits).toEqual(actionVisits);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getShortUrlVisits', () => {
|
|
|
|
const buildApiClientMock = (returned) => ({
|
2019-01-10 22:05:02 +03:00
|
|
|
getShortUrlVisits: typeof returned === 'function' ? sinon.fake(returned) : sinon.fake.returns(returned),
|
2018-09-08 09:49:49 +03:00
|
|
|
});
|
|
|
|
const dispatchMock = sinon.spy();
|
2019-03-08 21:40:43 +03:00
|
|
|
const getState = () => ({
|
|
|
|
shortUrlVisits: { cancelVisits: false },
|
|
|
|
});
|
2018-09-08 09:49:49 +03:00
|
|
|
|
|
|
|
beforeEach(() => dispatchMock.resetHistory());
|
|
|
|
|
|
|
|
it('dispatches start and error when promise is rejected', async () => {
|
|
|
|
const ShlinkApiClient = buildApiClientMock(Promise.reject());
|
|
|
|
const expectedDispatchCalls = 2;
|
|
|
|
|
2018-12-18 16:32:02 +03:00
|
|
|
await getShortUrlVisits(() => ShlinkApiClient)('abc123')(dispatchMock, getState);
|
2018-09-08 09:49:49 +03:00
|
|
|
|
|
|
|
const [ firstCallArg ] = dispatchMock.getCall(0).args;
|
|
|
|
const { type: firstCallType } = firstCallArg;
|
|
|
|
|
|
|
|
const [ secondCallArg ] = dispatchMock.getCall(1).args;
|
|
|
|
const { type: secondCallType } = secondCallArg;
|
|
|
|
|
|
|
|
expect(dispatchMock.callCount).toEqual(expectedDispatchCalls);
|
|
|
|
expect(ShlinkApiClient.getShortUrlVisits.callCount).toEqual(1);
|
|
|
|
expect(firstCallType).toEqual(GET_SHORT_URL_VISITS_START);
|
|
|
|
expect(secondCallType).toEqual(GET_SHORT_URL_VISITS_ERROR);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('dispatches start and success when promise is resolved', async () => {
|
|
|
|
const resolvedVisits = [{}, {}];
|
2019-01-10 22:05:02 +03:00
|
|
|
const ShlinkApiClient = buildApiClientMock(Promise.resolve({
|
|
|
|
data: resolvedVisits,
|
|
|
|
pagination: {
|
|
|
|
currentPage: 1,
|
|
|
|
pagesCount: 1,
|
|
|
|
},
|
|
|
|
}));
|
2018-09-08 09:49:49 +03:00
|
|
|
const expectedDispatchCalls = 2;
|
|
|
|
|
2018-12-18 16:32:02 +03:00
|
|
|
await getShortUrlVisits(() => ShlinkApiClient)('abc123')(dispatchMock, getState);
|
2018-09-08 09:49:49 +03:00
|
|
|
|
|
|
|
const [ firstCallArg ] = dispatchMock.getCall(0).args;
|
|
|
|
const { type: firstCallType } = firstCallArg;
|
|
|
|
|
|
|
|
const [ secondCallArg ] = dispatchMock.getCall(1).args;
|
|
|
|
const { type: secondCallType, visits } = secondCallArg;
|
|
|
|
|
|
|
|
expect(dispatchMock.callCount).toEqual(expectedDispatchCalls);
|
|
|
|
expect(ShlinkApiClient.getShortUrlVisits.callCount).toEqual(1);
|
|
|
|
expect(firstCallType).toEqual(GET_SHORT_URL_VISITS_START);
|
|
|
|
expect(secondCallType).toEqual(GET_SHORT_URL_VISITS);
|
|
|
|
expect(visits).toEqual(resolvedVisits);
|
|
|
|
});
|
2019-01-10 22:05:02 +03:00
|
|
|
|
2019-01-10 22:11:32 +03:00
|
|
|
it('performs multiple API requests when response contains more pages', async () => {
|
2019-01-10 22:05:02 +03:00
|
|
|
const expectedRequests = 3;
|
|
|
|
const ShlinkApiClient = buildApiClientMock((shortCode, { page }) =>
|
|
|
|
Promise.resolve({
|
|
|
|
data: [{}, {}],
|
|
|
|
pagination: {
|
|
|
|
currentPage: page,
|
|
|
|
pagesCount: expectedRequests,
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
|
|
|
|
await getShortUrlVisits(() => ShlinkApiClient)('abc123')(dispatchMock, getState);
|
|
|
|
|
|
|
|
const [ secondCallArg ] = dispatchMock.getCall(1).args;
|
|
|
|
const { visits } = secondCallArg;
|
|
|
|
|
|
|
|
expect(ShlinkApiClient.getShortUrlVisits.callCount).toEqual(expectedRequests);
|
|
|
|
expect(visits).toEqual([{}, {}, {}, {}, {}, {}]);
|
|
|
|
});
|
2018-09-08 09:49:49 +03:00
|
|
|
});
|
2019-03-08 21:45:35 +03:00
|
|
|
|
|
|
|
describe('cancelGetShortUrlVisits', () => {
|
|
|
|
it('just returns the action with proper type', () =>
|
|
|
|
expect(cancelGetShortUrlVisits()).toEqual({ type: GET_SHORT_URL_VISITS_CANCEL }));
|
|
|
|
});
|
2018-09-08 09:49:49 +03:00
|
|
|
});
|