2020-08-27 19:31:56 +03:00
|
|
|
import { Mock } from 'ts-mockery';
|
2018-12-21 12:58:51 +03:00
|
|
|
import reducer, {
|
|
|
|
LIST_SHORT_URLS,
|
|
|
|
LIST_SHORT_URLS_ERROR,
|
|
|
|
LIST_SHORT_URLS_START,
|
|
|
|
listShortUrls,
|
|
|
|
} from '../../../src/short-urls/reducers/shortUrlsList';
|
|
|
|
import { SHORT_URL_DELETED } from '../../../src/short-urls/reducers/shortUrlDeletion';
|
2020-08-27 19:31:56 +03:00
|
|
|
import { ShortUrl } from '../../../src/short-urls/data';
|
2022-05-28 11:47:39 +03:00
|
|
|
import { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
|
2020-12-22 11:49:13 +03:00
|
|
|
import { ShlinkPaginator, ShlinkShortUrlsResponse } from '../../../src/api/types';
|
2020-12-08 12:57:27 +03:00
|
|
|
import { CREATE_SHORT_URL } from '../../../src/short-urls/reducers/shortUrlCreation';
|
2021-04-24 18:58:37 +03:00
|
|
|
import { SHORT_URL_EDITED } from '../../../src/short-urls/reducers/shortUrlEdition';
|
2022-11-08 00:29:15 +03:00
|
|
|
import { createNewVisits } from '../../../src/visits/reducers/visitCreation';
|
2018-12-21 12:58:51 +03:00
|
|
|
|
|
|
|
describe('shortUrlsListReducer', () => {
|
2022-01-08 12:51:34 +03:00
|
|
|
const shortCode = 'abc123';
|
|
|
|
|
2018-12-21 12:58:51 +03:00
|
|
|
describe('reducer', () => {
|
|
|
|
it('returns loading on LIST_SHORT_URLS_START', () =>
|
2020-08-27 19:31:56 +03:00
|
|
|
expect(reducer(undefined, { type: LIST_SHORT_URLS_START } as any)).toEqual({
|
2018-12-21 12:58:51 +03:00
|
|
|
loading: true,
|
|
|
|
error: false,
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('returns short URLs on LIST_SHORT_URLS', () =>
|
2020-08-27 19:31:56 +03:00
|
|
|
expect(reducer(undefined, { type: LIST_SHORT_URLS, shortUrls: { data: [] } } as any)).toEqual({
|
|
|
|
shortUrls: { data: [] },
|
2018-12-21 12:58:51 +03:00
|
|
|
loading: false,
|
|
|
|
error: false,
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('returns error on LIST_SHORT_URLS_ERROR', () =>
|
2020-08-27 19:31:56 +03:00
|
|
|
expect(reducer(undefined, { type: LIST_SHORT_URLS_ERROR } as any)).toEqual({
|
2018-12-21 12:58:51 +03:00
|
|
|
loading: false,
|
|
|
|
error: true,
|
|
|
|
}));
|
|
|
|
|
2020-12-08 12:57:27 +03:00
|
|
|
it('removes matching URL and reduces total on SHORT_URL_DELETED', () => {
|
2018-12-21 12:58:51 +03:00
|
|
|
const state = {
|
2020-08-30 20:45:17 +03:00
|
|
|
shortUrls: Mock.of<ShlinkShortUrlsResponse>({
|
2018-12-21 12:58:51 +03:00
|
|
|
data: [
|
2020-08-27 19:31:56 +03:00
|
|
|
Mock.of<ShortUrl>({ shortCode }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode, domain: 'example.com' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'foo' }),
|
2018-12-21 12:58:51 +03:00
|
|
|
],
|
2020-12-08 12:57:27 +03:00
|
|
|
pagination: Mock.of<ShlinkPaginator>({
|
|
|
|
totalItems: 10,
|
|
|
|
}),
|
2020-08-30 20:45:17 +03:00
|
|
|
}),
|
2020-08-27 19:31:56 +03:00
|
|
|
loading: false,
|
|
|
|
error: false,
|
2018-12-21 12:58:51 +03:00
|
|
|
};
|
|
|
|
|
2022-11-06 15:06:55 +03:00
|
|
|
expect(reducer(state, { type: `${SHORT_URL_DELETED}/fulfilled`, payload: { shortCode } } as any)).toEqual({
|
2018-12-21 12:58:51 +03:00
|
|
|
shortUrls: {
|
2020-02-08 12:46:11 +03:00
|
|
|
data: [{ shortCode, domain: 'example.com' }, { shortCode: 'foo' }],
|
2020-12-08 12:57:27 +03:00
|
|
|
pagination: { totalItems: 9 },
|
|
|
|
},
|
|
|
|
loading: false,
|
|
|
|
error: false,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-02-27 22:03:51 +03:00
|
|
|
const createNewShortUrlVisit = (visitsCount: number) => ({
|
|
|
|
shortUrl: { shortCode: 'abc123', visitsCount },
|
|
|
|
});
|
|
|
|
|
|
|
|
it.each([
|
2022-03-26 14:17:42 +03:00
|
|
|
[[createNewShortUrlVisit(11)], 11],
|
|
|
|
[[createNewShortUrlVisit(30)], 30],
|
|
|
|
[[createNewShortUrlVisit(20), createNewShortUrlVisit(40)], 40],
|
|
|
|
[[{}], 10],
|
|
|
|
[[], 10],
|
2021-02-27 22:03:51 +03:00
|
|
|
])('updates visits count on CREATE_VISITS', (createdVisits, expectedCount) => {
|
2020-04-18 13:09:51 +03:00
|
|
|
const state = {
|
2020-08-30 20:45:17 +03:00
|
|
|
shortUrls: Mock.of<ShlinkShortUrlsResponse>({
|
2020-04-18 13:09:51 +03:00
|
|
|
data: [
|
2020-08-27 19:31:56 +03:00
|
|
|
Mock.of<ShortUrl>({ shortCode, domain: 'example.com', visitsCount: 5 }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode, visitsCount: 10 }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'foo', visitsCount: 8 }),
|
2020-04-18 13:09:51 +03:00
|
|
|
],
|
2020-08-30 20:45:17 +03:00
|
|
|
}),
|
2020-08-27 19:31:56 +03:00
|
|
|
loading: false,
|
|
|
|
error: false,
|
2020-04-18 13:09:51 +03:00
|
|
|
};
|
|
|
|
|
2022-11-08 00:29:15 +03:00
|
|
|
expect(reducer(state, { type: createNewVisits.toString(), payload: { createdVisits } } as any)).toEqual({
|
2020-04-18 13:09:51 +03:00
|
|
|
shortUrls: {
|
|
|
|
data: [
|
|
|
|
{ shortCode, domain: 'example.com', visitsCount: 5 },
|
2021-02-27 22:03:51 +03:00
|
|
|
{ shortCode, visitsCount: expectedCount },
|
2020-04-18 13:09:51 +03:00
|
|
|
{ shortCode: 'foo', visitsCount: 8 },
|
|
|
|
],
|
|
|
|
},
|
2020-08-27 19:31:56 +03:00
|
|
|
loading: false,
|
|
|
|
error: false,
|
2020-04-18 13:09:51 +03:00
|
|
|
});
|
|
|
|
});
|
2020-12-08 12:57:27 +03:00
|
|
|
|
2022-01-08 12:51:34 +03:00
|
|
|
it.each([
|
|
|
|
[
|
|
|
|
[
|
|
|
|
Mock.of<ShortUrl>({ shortCode }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode, domain: 'example.com' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'foo' }),
|
|
|
|
],
|
|
|
|
[{ shortCode: 'newOne' }, { shortCode }, { shortCode, domain: 'example.com' }, { shortCode: 'foo' }],
|
|
|
|
],
|
|
|
|
[
|
|
|
|
[
|
|
|
|
Mock.of<ShortUrl>({ shortCode }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'code' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'foo' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'bar' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'baz' }),
|
|
|
|
],
|
|
|
|
[{ shortCode: 'newOne' }, { shortCode }, { shortCode: 'code' }, { shortCode: 'foo' }, { shortCode: 'bar' }],
|
|
|
|
],
|
|
|
|
[
|
|
|
|
[
|
|
|
|
Mock.of<ShortUrl>({ shortCode }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'code' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'foo' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'bar' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'baz1' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'baz2' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'baz3' }),
|
|
|
|
],
|
|
|
|
[{ shortCode: 'newOne' }, { shortCode }, { shortCode: 'code' }, { shortCode: 'foo' }, { shortCode: 'bar' }],
|
|
|
|
],
|
|
|
|
])('prepends new short URL and increases total on CREATE_SHORT_URL', (data, expectedData) => {
|
2020-12-08 12:57:27 +03:00
|
|
|
const newShortUrl = Mock.of<ShortUrl>({ shortCode: 'newOne' });
|
|
|
|
const state = {
|
|
|
|
shortUrls: Mock.of<ShlinkShortUrlsResponse>({
|
2022-01-08 12:51:34 +03:00
|
|
|
data,
|
2020-12-08 12:57:27 +03:00
|
|
|
pagination: Mock.of<ShlinkPaginator>({
|
|
|
|
totalItems: 15,
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
loading: false,
|
|
|
|
error: false,
|
|
|
|
};
|
|
|
|
|
2022-11-06 12:11:44 +03:00
|
|
|
expect(reducer(state, { type: `${CREATE_SHORT_URL}/fulfilled`, payload: newShortUrl } as any)).toEqual({
|
2020-12-08 12:57:27 +03:00
|
|
|
shortUrls: {
|
2022-01-08 12:51:34 +03:00
|
|
|
data: expectedData,
|
2020-12-08 12:57:27 +03:00
|
|
|
pagination: { totalItems: 16 },
|
|
|
|
},
|
|
|
|
loading: false,
|
|
|
|
error: false,
|
|
|
|
});
|
|
|
|
});
|
2021-04-24 18:58:37 +03:00
|
|
|
|
|
|
|
it.each([
|
|
|
|
((): [ShortUrl, ShortUrl[], ShortUrl[]] => {
|
|
|
|
const editedShortUrl = Mock.of<ShortUrl>({ shortCode: 'notMatching' });
|
2022-03-26 14:17:42 +03:00
|
|
|
const list = [Mock.of<ShortUrl>({ shortCode: 'foo' }), Mock.of<ShortUrl>({ shortCode: 'bar' })];
|
2021-04-24 18:58:37 +03:00
|
|
|
|
2022-03-26 14:17:42 +03:00
|
|
|
return [editedShortUrl, list, list];
|
2021-04-24 18:58:37 +03:00
|
|
|
})(),
|
|
|
|
((): [ShortUrl, ShortUrl[], ShortUrl[]] => {
|
|
|
|
const editedShortUrl = Mock.of<ShortUrl>({ shortCode: 'matching', longUrl: 'new_one' });
|
|
|
|
const list = [
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'matching', longUrl: 'old_one' }),
|
|
|
|
Mock.of<ShortUrl>({ shortCode: 'bar' }),
|
|
|
|
];
|
2022-03-26 14:17:42 +03:00
|
|
|
const expectedList = [editedShortUrl, list[1]];
|
2021-04-24 18:58:37 +03:00
|
|
|
|
2022-03-26 14:17:42 +03:00
|
|
|
return [editedShortUrl, list, expectedList];
|
2021-04-24 18:58:37 +03:00
|
|
|
})(),
|
|
|
|
])('updates matching short URL on SHORT_URL_EDITED', (editedShortUrl, initialList, expectedList) => {
|
|
|
|
const state = {
|
|
|
|
shortUrls: Mock.of<ShlinkShortUrlsResponse>({
|
|
|
|
data: initialList,
|
|
|
|
pagination: Mock.of<ShlinkPaginator>({
|
|
|
|
totalItems: 15,
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
loading: false,
|
|
|
|
error: false,
|
|
|
|
};
|
|
|
|
|
2022-11-06 14:32:55 +03:00
|
|
|
const result = reducer(state, { type: `${SHORT_URL_EDITED}/fulfilled`, payload: editedShortUrl } as any);
|
2021-04-24 18:58:37 +03:00
|
|
|
|
|
|
|
expect(result.shortUrls?.data).toEqual(expectedList);
|
|
|
|
});
|
2018-12-21 12:58:51 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('listShortUrls', () => {
|
2019-04-19 13:41:59 +03:00
|
|
|
const dispatch = jest.fn();
|
|
|
|
const getState = jest.fn().mockReturnValue({ selectedServer: {} });
|
2018-12-21 12:58:51 +03:00
|
|
|
|
2020-08-27 19:31:56 +03:00
|
|
|
afterEach(jest.clearAllMocks);
|
2018-12-21 12:58:51 +03:00
|
|
|
|
|
|
|
it('dispatches proper actions if API client request succeeds', async () => {
|
2020-08-27 19:31:56 +03:00
|
|
|
const listShortUrlsMock = jest.fn().mockResolvedValue([]);
|
|
|
|
const apiClientMock = Mock.of<ShlinkApiClient>({ listShortUrls: listShortUrlsMock });
|
2018-12-21 12:58:51 +03:00
|
|
|
|
|
|
|
await listShortUrls(() => apiClientMock)()(dispatch, getState);
|
|
|
|
|
2019-04-19 13:52:55 +03:00
|
|
|
expect(dispatch).toHaveBeenCalledTimes(2);
|
|
|
|
expect(dispatch).toHaveBeenNthCalledWith(1, { type: LIST_SHORT_URLS_START });
|
2021-12-24 17:05:15 +03:00
|
|
|
expect(dispatch).toHaveBeenNthCalledWith(2, { type: LIST_SHORT_URLS, shortUrls: [] });
|
2018-12-21 12:58:51 +03:00
|
|
|
|
2020-08-27 19:31:56 +03:00
|
|
|
expect(listShortUrlsMock).toHaveBeenCalledTimes(1);
|
2018-12-21 12:58:51 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
it('dispatches proper actions if API client request fails', async () => {
|
2020-08-27 19:31:56 +03:00
|
|
|
const listShortUrlsMock = jest.fn().mockRejectedValue(undefined);
|
|
|
|
const apiClientMock = Mock.of<ShlinkApiClient>({ listShortUrls: listShortUrlsMock });
|
2018-12-21 12:58:51 +03:00
|
|
|
|
|
|
|
await listShortUrls(() => apiClientMock)()(dispatch, getState);
|
|
|
|
|
2019-04-19 13:52:55 +03:00
|
|
|
expect(dispatch).toHaveBeenCalledTimes(2);
|
|
|
|
expect(dispatch).toHaveBeenNthCalledWith(1, { type: LIST_SHORT_URLS_START });
|
2021-12-24 17:05:15 +03:00
|
|
|
expect(dispatch).toHaveBeenNthCalledWith(2, { type: LIST_SHORT_URLS_ERROR });
|
2018-12-21 12:58:51 +03:00
|
|
|
|
2020-08-27 19:31:56 +03:00
|
|
|
expect(listShortUrlsMock).toHaveBeenCalledTimes(1);
|
2018-12-21 12:58:51 +03:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|