2023-08-30 00:12:25 +03:00
|
|
|
import type { ShlinkApiClient } from '@shlinkio/shlink-js-sdk';
|
2023-04-13 22:48:29 +03:00
|
|
|
import { fromPartial } from '@total-typescript/shoehorn';
|
2023-02-18 13:11:01 +03:00
|
|
|
import type { ShlinkState } from '../../../src/container/types';
|
2023-07-24 21:14:59 +03:00
|
|
|
import type { NonReachableServer, NotFoundServer, RegularServer } from '../../../src/servers/data';
|
2022-11-11 21:31:05 +03:00
|
|
|
import {
|
2023-02-18 13:11:01 +03:00
|
|
|
MAX_FALLBACK_VERSION,
|
|
|
|
MIN_FALLBACK_VERSION,
|
2018-08-12 10:22:18 +03:00
|
|
|
resetSelectedServer,
|
2022-11-11 21:31:05 +03:00
|
|
|
selectedServerReducerCreator,
|
2023-02-18 13:11:01 +03:00
|
|
|
selectServer as selectServerCreator,
|
2018-08-12 10:22:18 +03:00
|
|
|
} from '../../../src/servers/reducers/selectedServer';
|
|
|
|
|
|
|
|
describe('selectedServerReducer', () => {
|
2023-05-27 12:57:26 +03:00
|
|
|
const dispatch = vi.fn();
|
|
|
|
const health = vi.fn();
|
|
|
|
const buildApiClient = vi.fn().mockReturnValue(fromPartial<ShlinkApiClient>({ health }));
|
2022-11-11 00:44:25 +03:00
|
|
|
const selectServer = selectServerCreator(buildApiClient);
|
2022-11-11 21:31:05 +03:00
|
|
|
const { reducer } = selectedServerReducerCreator(selectServer);
|
2022-11-11 00:44:25 +03:00
|
|
|
|
2018-08-26 11:52:45 +03:00
|
|
|
describe('reducer', () => {
|
2018-08-12 10:34:14 +03:00
|
|
|
it('returns default when action is RESET_SELECTED_SERVER', () =>
|
2023-03-18 14:35:33 +03:00
|
|
|
expect(reducer(null, resetSelectedServer())).toBeNull());
|
2018-08-12 10:34:14 +03:00
|
|
|
|
|
|
|
it('returns selected server when action is SELECT_SERVER', () => {
|
2023-04-13 22:48:29 +03:00
|
|
|
const payload = fromPartial<RegularServer>({ id: 'abc123' });
|
2023-03-18 14:35:33 +03:00
|
|
|
expect(reducer(null, selectServer.fulfilled(payload, '', ''))).toEqual(payload);
|
2018-08-12 10:22:18 +03:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2018-08-12 10:34:14 +03:00
|
|
|
describe('selectServer', () => {
|
2019-10-05 11:20:33 +03:00
|
|
|
const version = '1.19.0';
|
2023-05-27 12:57:26 +03:00
|
|
|
const createGetStateMock = (id: string) => vi.fn().mockReturnValue({
|
2022-12-31 18:56:22 +03:00
|
|
|
servers: {
|
|
|
|
[id]: { id },
|
|
|
|
},
|
|
|
|
});
|
2018-08-12 10:22:18 +03:00
|
|
|
|
2020-02-17 20:21:52 +03:00
|
|
|
it.each([
|
2022-03-26 14:17:42 +03:00
|
|
|
[version, version, `v${version}`],
|
|
|
|
['latest', MAX_FALLBACK_VERSION, 'latest'],
|
|
|
|
['%invalid_semver%', MIN_FALLBACK_VERSION, '%invalid_semver%'],
|
2020-03-05 13:58:35 +03:00
|
|
|
])('dispatches proper actions', async (serverVersion, expectedVersion, expectedPrintableVersion) => {
|
2024-10-02 13:06:22 +03:00
|
|
|
const id = crypto.randomUUID();
|
2020-04-27 14:21:07 +03:00
|
|
|
const getState = createGetStateMock(id);
|
2019-10-05 11:20:33 +03:00
|
|
|
const expectedSelectedServer = {
|
2022-12-31 18:56:22 +03:00
|
|
|
id,
|
2019-10-18 18:39:38 +03:00
|
|
|
version: expectedVersion,
|
2020-03-05 13:58:35 +03:00
|
|
|
printableVersion: expectedPrintableVersion,
|
2019-10-05 11:20:33 +03:00
|
|
|
};
|
2018-08-12 10:22:18 +03:00
|
|
|
|
2022-11-11 00:44:25 +03:00
|
|
|
health.mockResolvedValue({ version: serverVersion });
|
2019-10-18 18:39:38 +03:00
|
|
|
|
2022-11-11 00:44:25 +03:00
|
|
|
await selectServer(id)(dispatch, getState, {});
|
2018-08-12 10:22:18 +03:00
|
|
|
|
2020-04-27 14:21:07 +03:00
|
|
|
expect(getState).toHaveBeenCalledTimes(1);
|
2019-10-05 20:31:47 +03:00
|
|
|
expect(buildApiClient).toHaveBeenCalledTimes(1);
|
2023-03-18 14:35:33 +03:00
|
|
|
expect(dispatch).toHaveBeenCalledTimes(3); // "Pending", "reset" and "fulfilled"
|
|
|
|
expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ payload: expectedSelectedServer }));
|
2018-08-12 10:22:18 +03:00
|
|
|
});
|
2019-10-18 18:39:38 +03:00
|
|
|
|
2020-03-08 14:57:01 +03:00
|
|
|
it('dispatches error when health endpoint fails', async () => {
|
2024-10-02 13:06:22 +03:00
|
|
|
const id = crypto.randomUUID();
|
2020-04-27 14:21:07 +03:00
|
|
|
const getState = createGetStateMock(id);
|
2023-04-13 22:48:29 +03:00
|
|
|
const expectedSelectedServer = fromPartial<NonReachableServer>({ id, serverNotReachable: true });
|
2019-10-18 18:39:38 +03:00
|
|
|
|
2022-11-11 00:44:25 +03:00
|
|
|
health.mockRejectedValue({});
|
2019-10-18 18:39:38 +03:00
|
|
|
|
2022-11-11 00:44:25 +03:00
|
|
|
await selectServer(id)(dispatch, getState, {});
|
2019-10-18 18:39:38 +03:00
|
|
|
|
2022-11-11 00:44:25 +03:00
|
|
|
expect(health).toHaveBeenCalled();
|
2023-03-18 14:35:33 +03:00
|
|
|
expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ payload: expectedSelectedServer }));
|
2020-03-08 14:57:01 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
it('dispatches error when server is not found', async () => {
|
2024-10-02 13:06:22 +03:00
|
|
|
const id = crypto.randomUUID();
|
2023-05-27 12:57:26 +03:00
|
|
|
const getState = vi.fn(() => fromPartial<ShlinkState>({ servers: {} }));
|
2020-08-23 11:58:43 +03:00
|
|
|
const expectedSelectedServer: NotFoundServer = { serverNotFound: true };
|
2020-03-08 14:57:01 +03:00
|
|
|
|
2022-11-11 00:44:25 +03:00
|
|
|
await selectServer(id)(dispatch, getState, {});
|
2020-03-08 14:57:01 +03:00
|
|
|
|
2020-04-27 14:21:07 +03:00
|
|
|
expect(getState).toHaveBeenCalled();
|
2022-11-11 00:44:25 +03:00
|
|
|
expect(health).not.toHaveBeenCalled();
|
2023-03-18 14:35:33 +03:00
|
|
|
expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ payload: expectedSelectedServer }));
|
2019-10-18 18:39:38 +03:00
|
|
|
});
|
2018-08-12 10:22:18 +03:00
|
|
|
});
|
|
|
|
});
|