shlink-web-client/test/short-urls/reducers/shortUrlsList.test.ts

206 lines
7.4 KiB
TypeScript
Raw Normal View History

2023-04-13 23:47:13 +03:00
import { fromPartial } from '@total-typescript/shoehorn';
2023-02-18 13:11:01 +03:00
import type { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
2023-04-13 23:47:13 +03:00
import type { ShlinkShortUrlsResponse } from '../../../src/api/types';
import type { ShortUrl } from '../../../src/short-urls/data';
2023-02-18 13:11:01 +03:00
import { createShortUrl as createShortUrlCreator } from '../../../src/short-urls/reducers/shortUrlCreation';
import { shortUrlDeleted } from '../../../src/short-urls/reducers/shortUrlDeletion';
import { editShortUrl as editShortUrlCreator } from '../../../src/short-urls/reducers/shortUrlEdition';
2022-11-09 21:13:44 +03:00
import {
listShortUrls as listShortUrlsCreator,
shortUrlsListReducerCreator,
2018-12-21 12:58:51 +03:00
} from '../../../src/short-urls/reducers/shortUrlsList';
import { createNewVisits } from '../../../src/visits/reducers/visitCreation';
import type { CreateVisit } from '../../../src/visits/types';
2018-12-21 12:58:51 +03:00
describe('shortUrlsListReducer', () => {
2022-01-08 12:51:34 +03:00
const shortCode = 'abc123';
2023-05-27 12:57:26 +03:00
const listShortUrlsMock = vi.fn();
2023-04-13 23:47:13 +03:00
const buildShlinkApiClient = () => fromPartial<ShlinkApiClient>({ listShortUrls: listShortUrlsMock });
2022-11-09 21:13:44 +03:00
const listShortUrls = listShortUrlsCreator(buildShlinkApiClient);
const editShortUrl = editShortUrlCreator(buildShlinkApiClient);
const createShortUrl = createShortUrlCreator(buildShlinkApiClient);
const { reducer } = shortUrlsListReducerCreator(listShortUrls, editShortUrl, createShortUrl);
2022-11-09 21:13:44 +03:00
2023-05-27 12:57:26 +03:00
afterEach(vi.clearAllMocks);
2022-01-08 12:51:34 +03:00
2018-12-21 12:58:51 +03:00
describe('reducer', () => {
it('returns loading on LIST_SHORT_URLS_START', () =>
expect(reducer(undefined, listShortUrls.pending(''))).toEqual({
2018-12-21 12:58:51 +03:00
loading: true,
error: false,
}));
it('returns short URLs on LIST_SHORT_URLS', () =>
2023-04-13 23:47:13 +03:00
expect(reducer(undefined, listShortUrls.fulfilled(fromPartial({ data: [] }), ''))).toEqual({
shortUrls: { data: [] },
2018-12-21 12:58:51 +03:00
loading: false,
error: false,
}));
it('returns error on LIST_SHORT_URLS_ERROR', () =>
expect(reducer(undefined, listShortUrls.rejected(null, ''))).toEqual({
2018-12-21 12:58:51 +03:00
loading: false,
error: true,
}));
it('removes matching URL and reduces total on SHORT_URL_DELETED', () => {
2018-12-21 12:58:51 +03:00
const state = {
2023-04-13 23:47:13 +03:00
shortUrls: fromPartial<ShlinkShortUrlsResponse>({
2018-12-21 12:58:51 +03:00
data: [
2023-04-13 23:47:13 +03:00
{ shortCode },
{ shortCode, domain: 'example.com' },
{ shortCode: 'foo' },
2018-12-21 12:58:51 +03:00
],
2023-04-13 23:47:13 +03:00
pagination: { totalItems: 10 },
}),
loading: false,
error: false,
2018-12-21 12:58:51 +03:00
};
2023-04-13 23:47:13 +03:00
expect(reducer(state, shortUrlDeleted(fromPartial({ shortCode })))).toEqual({
2018-12-21 12:58:51 +03:00
shortUrls: {
data: [{ shortCode, domain: 'example.com' }, { shortCode: 'foo' }],
pagination: { totalItems: 9 },
},
loading: false,
error: false,
});
});
2023-04-13 23:47:13 +03:00
const createNewShortUrlVisit = (visitsCount: number) => fromPartial<CreateVisit>({
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],
])('updates visits count on CREATE_VISITS', (createdVisits, expectedCount) => {
const state = {
2023-04-13 23:47:13 +03:00
shortUrls: fromPartial<ShlinkShortUrlsResponse>({
data: [
2023-04-13 23:47:13 +03:00
{ shortCode, domain: 'example.com', visitsCount: 5 },
{ shortCode, visitsCount: 10 },
{ shortCode: 'foo', visitsCount: 8 },
],
}),
loading: false,
error: false,
};
expect(reducer(state, createNewVisits(createdVisits))).toEqual({
shortUrls: {
data: [
{ shortCode, domain: 'example.com', visitsCount: 5 },
{ shortCode, visitsCount: expectedCount },
{ shortCode: 'foo', visitsCount: 8 },
],
},
loading: false,
error: false,
});
});
2022-01-08 12:51:34 +03:00
it.each([
[
[
2023-04-13 23:47:13 +03:00
fromPartial<ShortUrl>({ shortCode }),
fromPartial<ShortUrl>({ shortCode, domain: 'example.com' }),
fromPartial<ShortUrl>({ shortCode: 'foo' }),
2022-01-08 12:51:34 +03:00
],
[{ shortCode: 'newOne' }, { shortCode }, { shortCode, domain: 'example.com' }, { shortCode: 'foo' }],
],
[
[
2023-04-13 23:47:13 +03:00
fromPartial<ShortUrl>({ shortCode }),
fromPartial<ShortUrl>({ shortCode: 'code' }),
fromPartial<ShortUrl>({ shortCode: 'foo' }),
fromPartial<ShortUrl>({ shortCode: 'bar' }),
fromPartial<ShortUrl>({ shortCode: 'baz' }),
2022-01-08 12:51:34 +03:00
],
[{ shortCode: 'newOne' }, { shortCode }, { shortCode: 'code' }, { shortCode: 'foo' }, { shortCode: 'bar' }],
],
[
[
2023-04-13 23:47:13 +03:00
fromPartial<ShortUrl>({ shortCode }),
fromPartial<ShortUrl>({ shortCode: 'code' }),
fromPartial<ShortUrl>({ shortCode: 'foo' }),
fromPartial<ShortUrl>({ shortCode: 'bar' }),
fromPartial<ShortUrl>({ shortCode: 'baz1' }),
fromPartial<ShortUrl>({ shortCode: 'baz2' }),
fromPartial<ShortUrl>({ shortCode: 'baz3' }),
2022-01-08 12:51:34 +03:00
],
[{ shortCode: 'newOne' }, { shortCode }, { shortCode: 'code' }, { shortCode: 'foo' }, { shortCode: 'bar' }],
],
])('prepends new short URL and increases total on CREATE_SHORT_URL', (data, expectedData) => {
2023-04-13 23:47:13 +03:00
const newShortUrl = fromPartial<ShortUrl>({ shortCode: 'newOne' });
const state = {
2023-04-13 23:47:13 +03:00
shortUrls: fromPartial<ShlinkShortUrlsResponse>({
2022-01-08 12:51:34 +03:00
data,
2023-04-13 23:47:13 +03:00
pagination: { totalItems: 15 },
}),
loading: false,
error: false,
};
2023-04-13 23:47:13 +03:00
expect(reducer(state, createShortUrl.fulfilled(newShortUrl, '', fromPartial({})))).toEqual({
shortUrls: {
2022-01-08 12:51:34 +03:00
data: expectedData,
pagination: { totalItems: 16 },
},
loading: false,
error: false,
});
});
it.each([
((): [ShortUrl, ShortUrl[], ShortUrl[]] => {
2023-04-13 23:47:13 +03:00
const editedShortUrl = fromPartial<ShortUrl>({ shortCode: 'notMatching' });
const list: ShortUrl[] = [fromPartial({ shortCode: 'foo' }), fromPartial({ shortCode: 'bar' })];
2022-03-26 14:17:42 +03:00
return [editedShortUrl, list, list];
})(),
((): [ShortUrl, ShortUrl[], ShortUrl[]] => {
2023-04-13 23:47:13 +03:00
const editedShortUrl = fromPartial<ShortUrl>({ shortCode: 'matching', longUrl: 'new_one' });
const list: ShortUrl[] = [
fromPartial({ shortCode: 'matching', longUrl: 'old_one' }),
fromPartial({ shortCode: 'bar' }),
];
2022-03-26 14:17:42 +03:00
const expectedList = [editedShortUrl, list[1]];
2022-03-26 14:17:42 +03:00
return [editedShortUrl, list, expectedList];
})(),
])('updates matching short URL on SHORT_URL_EDITED', (editedShortUrl, initialList, expectedList) => {
const state = {
2023-04-13 23:47:13 +03:00
shortUrls: fromPartial<ShlinkShortUrlsResponse>({
data: initialList,
2023-04-13 23:47:13 +03:00
pagination: { totalItems: 15 },
}),
loading: false,
error: false,
};
2023-04-13 23:47:13 +03:00
const result = reducer(state, editShortUrl.fulfilled(editedShortUrl, '', fromPartial({})));
expect(result.shortUrls?.data).toEqual(expectedList);
});
2018-12-21 12:58:51 +03:00
});
describe('listShortUrls', () => {
2023-05-27 12:57:26 +03:00
const dispatch = vi.fn();
const getState = vi.fn().mockReturnValue({ selectedServer: {} });
2018-12-21 12:58:51 +03:00
it('dispatches proper actions if API client request succeeds', async () => {
2022-11-09 21:13:44 +03:00
listShortUrlsMock.mockResolvedValue({});
2018-12-21 12:58:51 +03:00
2022-11-09 21:13:44 +03:00
await listShortUrls()(dispatch, getState, {});
2018-12-21 12:58:51 +03:00
expect(dispatch).toHaveBeenCalledTimes(2);
expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ payload: {} }));
2018-12-21 12:58:51 +03:00
expect(listShortUrlsMock).toHaveBeenCalledTimes(1);
2018-12-21 12:58:51 +03:00
});
});
});