mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2024-12-23 09:30:31 +03:00
Migrated selectedServer reducer to RTK
This commit is contained in:
parent
6221f9ed05
commit
d44fe945d8
5 changed files with 55 additions and 41 deletions
|
@ -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)),
|
||||||
});
|
});
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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,42 +27,39 @@ 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`,
|
||||||
|
async (serverId: string, { dispatch, getState }): Promise<SelectedServer> => {
|
||||||
|
dispatch(resetSelectedServer());
|
||||||
|
|
||||||
export const selectServer = (
|
const { servers } = getState();
|
||||||
buildShlinkApiClient: ShlinkApiClientBuilder,
|
const selectedServer = servers[serverId];
|
||||||
) => createAsyncThunk(SELECT_SERVER, async (serverId: string, { dispatch, getState }): Promise<SelectedServer> => {
|
|
||||||
dispatch(resetSelectedServer());
|
|
||||||
|
|
||||||
const { servers } = getState();
|
if (!selectedServer) {
|
||||||
const selectedServer = servers[serverId];
|
return { serverNotFound: true };
|
||||||
|
}
|
||||||
|
|
||||||
if (!selectedServer) {
|
try {
|
||||||
return { serverNotFound: true };
|
const { health } = buildShlinkApiClient(selectedServer);
|
||||||
}
|
const { version, printableVersion } = await getServerVersion(serverId, health);
|
||||||
|
|
||||||
try {
|
return {
|
||||||
const { health } = buildShlinkApiClient(selectedServer);
|
...selectedServer,
|
||||||
const { version, printableVersion } = await getServerVersion(serverId, health);
|
version,
|
||||||
|
printableVersion,
|
||||||
|
};
|
||||||
|
} catch (e) {
|
||||||
|
return { ...selectedServer, serverNotReachable: true };
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
return {
|
type SelectServerThunk = ReturnType<typeof selectServer>;
|
||||||
...selectedServer,
|
|
||||||
version,
|
|
||||||
printableVersion,
|
|
||||||
};
|
|
||||||
} catch (e) {
|
|
||||||
return { ...selectedServer, serverNotReachable: true };
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
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);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue