Migrated selectedServer reducer to RTK

This commit is contained in:
Alejandro Celaya 2022-11-11 19:31:05 +01:00
parent 6221f9ed05
commit d44fe945d8
5 changed files with 55 additions and 41 deletions

View file

@ -18,7 +18,8 @@ export const setUpStore = (container: IContainer) => configureStore({
devTools: !isProduction, devTools: !isProduction,
reducer: reducer(container), reducer: reducer(container),
preloadedState, preloadedState,
middleware: (defaultMiddlewaresIncludingReduxThunk) => defaultMiddlewaresIncludingReduxThunk( middleware: (defaultMiddlewaresIncludingReduxThunk) =>
{ immutableCheck: false, serializableCheck: false }, // State is too big for these defaultMiddlewaresIncludingReduxThunk({ immutableCheck: false, serializableCheck: false })// State is too big for these
).prepend(container.selectServerListener.middleware).concat(save(localStorageConfig)), .prepend(container.selectServerListener.middleware)
.concat(save(localStorageConfig)),
}); });

View file

@ -1,7 +1,6 @@
import { IContainer } from 'bottlejs'; import { IContainer } from 'bottlejs';
import { combineReducers } from '@reduxjs/toolkit'; import { combineReducers } from '@reduxjs/toolkit';
import { serversReducer } from '../servers/reducers/servers'; import { serversReducer } from '../servers/reducers/servers';
import selectedServerReducer from '../servers/reducers/selectedServer';
import shortUrlVisitsReducer from '../visits/reducers/shortUrlVisits'; import shortUrlVisitsReducer from '../visits/reducers/shortUrlVisits';
import tagVisitsReducer from '../visits/reducers/tagVisits'; import tagVisitsReducer from '../visits/reducers/tagVisits';
import domainVisitsReducer from '../visits/reducers/domainVisits'; import domainVisitsReducer from '../visits/reducers/domainVisits';
@ -15,7 +14,7 @@ import { ShlinkState } from '../container/types';
export default (container: IContainer) => combineReducers<ShlinkState>({ export default (container: IContainer) => combineReducers<ShlinkState>({
servers: serversReducer, servers: serversReducer,
selectedServer: selectedServerReducer, selectedServer: container.selectedServerReducer,
shortUrlsList: container.shortUrlsListReducer, shortUrlsList: container.shortUrlsListReducer,
shortUrlCreation: container.shortUrlCreationReducer, shortUrlCreation: container.shortUrlCreationReducer,
shortUrlDeletion: container.shortUrlDeletionReducer, shortUrlDeletion: container.shortUrlDeletionReducer,

View file

@ -1,20 +1,17 @@
import { createAction, createListenerMiddleware, PayloadAction } from '@reduxjs/toolkit'; import { createAction, createListenerMiddleware, createSlice, PayloadAction } from '@reduxjs/toolkit';
import { identity, memoizeWith, pipe } from 'ramda'; import { identity, memoizeWith, pipe } from 'ramda';
import { versionToPrintable, versionToSemVer as toSemVer } from '../../utils/helpers/version'; import { versionToPrintable, versionToSemVer as toSemVer } from '../../utils/helpers/version';
import { isReachableServer, SelectedServer } from '../data'; import { isReachableServer, SelectedServer } from '../data';
import { ShlinkHealth } from '../../api/types'; import { ShlinkHealth } from '../../api/types';
import { buildReducer, createAsyncThunk } from '../../utils/helpers/redux'; import { createAsyncThunk } from '../../utils/helpers/redux';
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder'; import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
export const SELECT_SERVER = 'shlink/selectedServer/selectServer'; const REDUCER_PREFIX = 'shlink/selectedServer';
export const RESET_SELECTED_SERVER = 'shlink/selectedServer/resetSelectedServer';
export const MIN_FALLBACK_VERSION = '1.0.0'; export const MIN_FALLBACK_VERSION = '1.0.0';
export const MAX_FALLBACK_VERSION = '999.999.999'; export const MAX_FALLBACK_VERSION = '999.999.999';
export const LATEST_VERSION_CONSTRAINT = 'latest'; export const LATEST_VERSION_CONSTRAINT = 'latest';
export type SelectServerAction = PayloadAction<SelectedServer>;
const versionToSemVer = pipe( const versionToSemVer = pipe(
(version: string) => (version === LATEST_VERSION_CONSTRAINT ? MAX_FALLBACK_VERSION : version), (version: string) => (version === LATEST_VERSION_CONSTRAINT ? MAX_FALLBACK_VERSION : version),
toSemVer(MIN_FALLBACK_VERSION), toSemVer(MIN_FALLBACK_VERSION),
@ -30,17 +27,11 @@ const getServerVersion = memoizeWith(
const initialState: SelectedServer = null; const initialState: SelectedServer = null;
export default buildReducer<SelectedServer, SelectServerAction>({ export const resetSelectedServer = createAction<void>(`${REDUCER_PREFIX}/resetSelectedServer`);
[RESET_SELECTED_SERVER]: () => initialState,
[SELECT_SERVER]: (_, { payload }) => payload,
[`${SELECT_SERVER}/fulfilled`]: (_, { payload }) => payload,
}, initialState);
export const resetSelectedServer = createAction<void>(RESET_SELECTED_SERVER); export const selectServer = (buildShlinkApiClient: ShlinkApiClientBuilder) => createAsyncThunk(
`${REDUCER_PREFIX}/selectServer`,
export const selectServer = ( async (serverId: string, { dispatch, getState }): Promise<SelectedServer> => {
buildShlinkApiClient: ShlinkApiClientBuilder,
) => createAsyncThunk(SELECT_SERVER, async (serverId: string, { dispatch, getState }): Promise<SelectedServer> => {
dispatch(resetSelectedServer()); dispatch(resetSelectedServer());
const { servers } = getState(); const { servers } = getState();
@ -62,10 +53,13 @@ export const selectServer = (
} catch (e) { } catch (e) {
return { ...selectedServer, serverNotReachable: true }; return { ...selectedServer, serverNotReachable: true };
} }
}); },
);
type SelectServerThunk = ReturnType<typeof selectServer>;
export const selectServerListener = ( export const selectServerListener = (
selectServerThunk: ReturnType<typeof selectServer>, selectServerThunk: SelectServerThunk,
loadMercureInfo: () => PayloadAction<any>, // TODO Consider setting actual type, if relevant loadMercureInfo: () => PayloadAction<any>, // TODO Consider setting actual type, if relevant
) => { ) => {
const listener = createListenerMiddleware(); const listener = createListenerMiddleware();
@ -79,3 +73,13 @@ export const selectServerListener = (
return listener; return listener;
}; };
export const selectedServerReducerCreator = (selectServerThunk: SelectServerThunk) => createSlice({
name: REDUCER_PREFIX,
initialState,
reducers: {},
extraReducers: (builder) => {
builder.addCase(resetSelectedServer, () => initialState);
builder.addCase(selectServerThunk.fulfilled, (_, { payload }) => payload as any);
},
});

View file

@ -1,3 +1,4 @@
import { prop } from 'ramda';
import Bottle from 'bottlejs'; import Bottle from 'bottlejs';
import { CreateServer } from '../CreateServer'; import { CreateServer } from '../CreateServer';
import { ServersDropdown } from '../ServersDropdown'; import { ServersDropdown } from '../ServersDropdown';
@ -5,7 +6,12 @@ import { DeleteServerModal } from '../DeleteServerModal';
import { DeleteServerButton } from '../DeleteServerButton'; import { DeleteServerButton } from '../DeleteServerButton';
import { EditServer } from '../EditServer'; import { EditServer } from '../EditServer';
import { ImportServersBtn } from '../helpers/ImportServersBtn'; import { ImportServersBtn } from '../helpers/ImportServersBtn';
import { resetSelectedServer, selectServer, selectServerListener } from '../reducers/selectedServer'; import {
resetSelectedServer,
selectedServerReducerCreator,
selectServer,
selectServerListener,
} from '../reducers/selectedServer';
import { createServers, deleteServer, editServer, setAutoConnect } from '../reducers/servers'; import { createServers, deleteServer, editServer, setAutoConnect } from '../reducers/servers';
import { fetchServers } from '../reducers/remoteServers'; import { fetchServers } from '../reducers/remoteServers';
import { ServerError } from '../helpers/ServerError'; import { ServerError } from '../helpers/ServerError';
@ -80,6 +86,8 @@ const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
// Reducers // Reducers
bottle.serviceFactory('selectServerListener', selectServerListener, 'selectServer', 'loadMercureInfo'); bottle.serviceFactory('selectServerListener', selectServerListener, 'selectServer', 'loadMercureInfo');
bottle.serviceFactory('selectedServerReducerCreator', selectedServerReducerCreator, 'selectServer');
bottle.serviceFactory('selectedServerReducer', prop('reducer'), 'selectedServerReducerCreator');
}; };
export default provideServices; export default provideServices;

View file

@ -1,8 +1,9 @@
import { v4 as uuid } from 'uuid'; import { v4 as uuid } from 'uuid';
import { Mock } from 'ts-mockery'; import { Mock } from 'ts-mockery';
import reducer, { import {
selectServer as selectServerCreator, selectServer as selectServerCreator,
resetSelectedServer, resetSelectedServer,
selectedServerReducerCreator,
MAX_FALLBACK_VERSION, MAX_FALLBACK_VERSION,
MIN_FALLBACK_VERSION, MIN_FALLBACK_VERSION,
} from '../../../src/servers/reducers/selectedServer'; } from '../../../src/servers/reducers/selectedServer';
@ -14,6 +15,7 @@ describe('selectedServerReducer', () => {
const health = jest.fn(); const health = jest.fn();
const buildApiClient = jest.fn().mockReturnValue(Mock.of<ShlinkApiClient>({ health })); const buildApiClient = jest.fn().mockReturnValue(Mock.of<ShlinkApiClient>({ health }));
const selectServer = selectServerCreator(buildApiClient); const selectServer = selectServerCreator(buildApiClient);
const { reducer } = selectedServerReducerCreator(selectServer);
afterEach(jest.clearAllMocks); afterEach(jest.clearAllMocks);