mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2026-03-18 13:33:51 +00:00
Stop injecting dependencies in selectServer action
This commit is contained in:
@@ -5,13 +5,11 @@ import {
|
|||||||
ShlinkWebComponent,
|
ShlinkWebComponent,
|
||||||
} from '@shlinkio/shlink-web-component';
|
} from '@shlinkio/shlink-web-component';
|
||||||
import type { Settings } from '@shlinkio/shlink-web-component/settings';
|
import type { Settings } from '@shlinkio/shlink-web-component/settings';
|
||||||
import type { FC } from 'react';
|
|
||||||
import { memo } from 'react';
|
import { memo } from 'react';
|
||||||
import type { ShlinkApiClientBuilder } from '../api/services/ShlinkApiClientBuilder';
|
|
||||||
import type { FCWithDeps } from '../container/utils';
|
import type { FCWithDeps } from '../container/utils';
|
||||||
import { componentFactory, useDependencies } from '../container/utils';
|
import { componentFactory, useDependencies } from '../container/utils';
|
||||||
import { isReachableServer } from '../servers/data';
|
import { isReachableServer } from '../servers/data';
|
||||||
import type { WithSelectedServerProps } from '../servers/helpers/withSelectedServer';
|
import type { WithSelectedServerProps, WithSelectedServerPropsDeps } from '../servers/helpers/withSelectedServer';
|
||||||
import { withSelectedServer } from '../servers/helpers/withSelectedServer';
|
import { withSelectedServer } from '../servers/helpers/withSelectedServer';
|
||||||
import { NotFound } from './NotFound';
|
import { NotFound } from './NotFound';
|
||||||
|
|
||||||
@@ -19,10 +17,8 @@ type ShlinkWebComponentContainerProps = WithSelectedServerProps & {
|
|||||||
settings: Settings;
|
settings: Settings;
|
||||||
};
|
};
|
||||||
|
|
||||||
type ShlinkWebComponentContainerDeps = {
|
type ShlinkWebComponentContainerDeps = WithSelectedServerPropsDeps & {
|
||||||
buildShlinkApiClient: ShlinkApiClientBuilder,
|
|
||||||
TagColorsStorage: TagColorsStorage,
|
TagColorsStorage: TagColorsStorage,
|
||||||
ServerError: FC,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const ShlinkWebComponentContainer: FCWithDeps<
|
const ShlinkWebComponentContainer: FCWithDeps<
|
||||||
|
|||||||
@@ -1,26 +1,22 @@
|
|||||||
import { Button,useParsedQuery } from '@shlinkio/shlink-frontend-kit';
|
import { Button, useParsedQuery } from '@shlinkio/shlink-frontend-kit';
|
||||||
import type { FC } from 'react';
|
|
||||||
import { NoMenuLayout } from '../common/NoMenuLayout';
|
import { NoMenuLayout } from '../common/NoMenuLayout';
|
||||||
import type { FCWithDeps } from '../container/utils';
|
import type { FCWithDeps } from '../container/utils';
|
||||||
import { componentFactory } from '../container/utils';
|
import { componentFactory, useDependencies } from '../container/utils';
|
||||||
import { useGoBack } from '../utils/helpers/hooks';
|
import { useGoBack } from '../utils/helpers/hooks';
|
||||||
import type { ServerData } from './data';
|
import type { ServerData } from './data';
|
||||||
import { isServerWithId } from './data';
|
import { isServerWithId } from './data';
|
||||||
import { ServerForm } from './helpers/ServerForm';
|
import { ServerForm } from './helpers/ServerForm';
|
||||||
import type { WithSelectedServerProps } from './helpers/withSelectedServer';
|
import type { WithSelectedServerProps, WithSelectedServerPropsDeps } from './helpers/withSelectedServer';
|
||||||
import { withSelectedServer } from './helpers/withSelectedServer';
|
import { withSelectedServer } from './helpers/withSelectedServer';
|
||||||
|
|
||||||
type EditServerProps = WithSelectedServerProps & {
|
type EditServerProps = WithSelectedServerProps & {
|
||||||
editServer: (serverId: string, serverData: ServerData) => void;
|
editServer: (serverId: string, serverData: ServerData) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
type EditServerDeps = {
|
const EditServer: FCWithDeps<EditServerProps, WithSelectedServerPropsDeps> = withSelectedServer((
|
||||||
ServerError: FC;
|
|
||||||
};
|
|
||||||
|
|
||||||
const EditServer: FCWithDeps<EditServerProps, EditServerDeps> = withSelectedServer((
|
|
||||||
{ editServer, selectedServer, selectServer },
|
{ editServer, selectedServer, selectServer },
|
||||||
) => {
|
) => {
|
||||||
|
const { buildShlinkApiClient } = useDependencies(EditServer);
|
||||||
const goBack = useGoBack();
|
const goBack = useGoBack();
|
||||||
const { reconnect } = useParsedQuery<{ reconnect?: 'true' }>();
|
const { reconnect } = useParsedQuery<{ reconnect?: 'true' }>();
|
||||||
|
|
||||||
@@ -31,7 +27,7 @@ const EditServer: FCWithDeps<EditServerProps, EditServerDeps> = withSelectedServ
|
|||||||
const handleSubmit = (serverData: ServerData) => {
|
const handleSubmit = (serverData: ServerData) => {
|
||||||
editServer(selectedServer.id, serverData);
|
editServer(selectedServer.id, serverData);
|
||||||
if (reconnect === 'true') {
|
if (reconnect === 'true') {
|
||||||
selectServer(selectedServer.id);
|
selectServer({ serverId: selectedServer.id, buildShlinkApiClient });
|
||||||
}
|
}
|
||||||
goBack();
|
goBack();
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -2,34 +2,37 @@ import { Message } from '@shlinkio/shlink-frontend-kit';
|
|||||||
import type { FC } from 'react';
|
import type { FC } from 'react';
|
||||||
import { useEffect } from 'react';
|
import { useEffect } from 'react';
|
||||||
import { useParams } from 'react-router';
|
import { useParams } from 'react-router';
|
||||||
|
import type { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
||||||
import { NoMenuLayout } from '../../common/NoMenuLayout';
|
import { NoMenuLayout } from '../../common/NoMenuLayout';
|
||||||
import type { FCWithDeps } from '../../container/utils';
|
import type { FCWithDeps } from '../../container/utils';
|
||||||
import { useDependencies } from '../../container/utils';
|
import { useDependencies } from '../../container/utils';
|
||||||
import type { SelectedServer } from '../data';
|
import type { SelectedServer } from '../data';
|
||||||
import { isNotFoundServer } from '../data';
|
import { isNotFoundServer } from '../data';
|
||||||
|
import type { SelectServerOptions } from '../reducers/selectedServer';
|
||||||
|
|
||||||
export type WithSelectedServerProps = {
|
export type WithSelectedServerProps = {
|
||||||
selectServer: (serverId: string) => void;
|
selectServer: (options: SelectServerOptions) => void;
|
||||||
selectedServer: SelectedServer;
|
selectedServer: SelectedServer;
|
||||||
};
|
};
|
||||||
|
|
||||||
type WithSelectedServerPropsDeps = {
|
export type WithSelectedServerPropsDeps = {
|
||||||
ServerError: FC;
|
ServerError: FC;
|
||||||
|
buildShlinkApiClient: ShlinkApiClientBuilder;
|
||||||
};
|
};
|
||||||
|
|
||||||
export function withSelectedServer<T extends object>(
|
export function withSelectedServer<T extends object>(
|
||||||
WrappedComponent: FCWithDeps<WithSelectedServerProps & T, WithSelectedServerPropsDeps>,
|
WrappedComponent: FCWithDeps<WithSelectedServerProps & T, WithSelectedServerPropsDeps>,
|
||||||
) {
|
) {
|
||||||
const ComponentWrapper: FCWithDeps<WithSelectedServerProps & T, WithSelectedServerPropsDeps> = (props) => {
|
const ComponentWrapper: FCWithDeps<WithSelectedServerProps & T, WithSelectedServerPropsDeps> = (props) => {
|
||||||
const { ServerError } = useDependencies(ComponentWrapper);
|
const { ServerError, buildShlinkApiClient } = useDependencies(ComponentWrapper);
|
||||||
const params = useParams<{ serverId: string }>();
|
const params = useParams<{ serverId: string }>();
|
||||||
const { selectServer, selectedServer } = props;
|
const { selectServer, selectedServer } = props;
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (params.serverId) {
|
if (params.serverId) {
|
||||||
selectServer(params.serverId);
|
selectServer({ serverId: params.serverId, buildShlinkApiClient });
|
||||||
}
|
}
|
||||||
}, [params.serverId, selectServer]);
|
}, [buildShlinkApiClient, params.serverId, selectServer]);
|
||||||
|
|
||||||
if (!selectedServer) {
|
if (!selectedServer) {
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -29,9 +29,14 @@ const initialState: SelectedServer = null;
|
|||||||
|
|
||||||
export const resetSelectedServer = createAction<void>(`${REDUCER_PREFIX}/resetSelectedServer`);
|
export const resetSelectedServer = createAction<void>(`${REDUCER_PREFIX}/resetSelectedServer`);
|
||||||
|
|
||||||
export const selectServer = (buildShlinkApiClient: ShlinkApiClientBuilder) => createAsyncThunk(
|
export type SelectServerOptions = {
|
||||||
|
serverId: string;
|
||||||
|
buildShlinkApiClient: ShlinkApiClientBuilder;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const selectServer = createAsyncThunk(
|
||||||
`${REDUCER_PREFIX}/selectServer`,
|
`${REDUCER_PREFIX}/selectServer`,
|
||||||
async (serverId: string, { dispatch, getState }): Promise<SelectedServer> => {
|
async ({ serverId, buildShlinkApiClient }: SelectServerOptions, { dispatch, getState }): Promise<SelectedServer> => {
|
||||||
dispatch(resetSelectedServer());
|
dispatch(resetSelectedServer());
|
||||||
|
|
||||||
const { servers } = getState();
|
const { servers } = getState();
|
||||||
@@ -58,14 +63,11 @@ export const selectServer = (buildShlinkApiClient: ShlinkApiClientBuilder) => cr
|
|||||||
|
|
||||||
const { reducer } = createSlice({
|
const { reducer } = createSlice({
|
||||||
name: REDUCER_PREFIX,
|
name: REDUCER_PREFIX,
|
||||||
initialState,
|
initialState: initialState as SelectedServer,
|
||||||
reducers: {},
|
reducers: {},
|
||||||
extraReducers: (builder) => {
|
extraReducers: (builder) => {
|
||||||
builder.addCase(resetSelectedServer, () => initialState);
|
builder.addCase(resetSelectedServer, () => initialState);
|
||||||
builder.addCase(
|
builder.addCase(selectServer.fulfilled, (_, { payload }) => payload);
|
||||||
`${REDUCER_PREFIX}/selectServer/fulfilled`,
|
|
||||||
(_, { payload }: { payload: SelectedServer }) => payload,
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ export const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
|||||||
bottle.service('ServersExporter', ServersExporter, 'Storage', 'window', 'jsonToCsv');
|
bottle.service('ServersExporter', ServersExporter, 'Storage', 'window', 'jsonToCsv');
|
||||||
|
|
||||||
// Actions
|
// Actions
|
||||||
bottle.serviceFactory('selectServer', selectServer, 'buildShlinkApiClient', 'loadMercureInfo');
|
bottle.serviceFactory('selectServer', () => selectServer, 'buildShlinkApiClient', 'loadMercureInfo');
|
||||||
bottle.serviceFactory('createServers', () => createServers);
|
bottle.serviceFactory('createServers', () => createServers);
|
||||||
bottle.serviceFactory('deleteServer', () => deleteServer);
|
bottle.serviceFactory('deleteServer', () => deleteServer);
|
||||||
bottle.serviceFactory('editServer', () => editServer);
|
bottle.serviceFactory('editServer', () => editServer);
|
||||||
|
|||||||
@@ -7,14 +7,13 @@ import {
|
|||||||
MIN_FALLBACK_VERSION,
|
MIN_FALLBACK_VERSION,
|
||||||
resetSelectedServer,
|
resetSelectedServer,
|
||||||
selectedServerReducer as reducer,
|
selectedServerReducer as reducer,
|
||||||
selectServer as selectServerCreator,
|
selectServer,
|
||||||
} from '../../../src/servers/reducers/selectedServer';
|
} from '../../../src/servers/reducers/selectedServer';
|
||||||
|
|
||||||
describe('selectedServerReducer', () => {
|
describe('selectedServerReducer', () => {
|
||||||
const dispatch = vi.fn();
|
const dispatch = vi.fn();
|
||||||
const health = vi.fn();
|
const health = vi.fn();
|
||||||
const buildApiClient = vi.fn().mockReturnValue(fromPartial<ShlinkApiClient>({ health }));
|
const buildShlinkApiClient = vi.fn().mockReturnValue(fromPartial<ShlinkApiClient>({ health }));
|
||||||
const selectServer = selectServerCreator(buildApiClient);
|
|
||||||
|
|
||||||
describe('reducer', () => {
|
describe('reducer', () => {
|
||||||
it('returns default when action is RESET_SELECTED_SERVER', () =>
|
it('returns default when action is RESET_SELECTED_SERVER', () =>
|
||||||
@@ -22,7 +21,7 @@ describe('selectedServerReducer', () => {
|
|||||||
|
|
||||||
it('returns selected server when action is SELECT_SERVER', () => {
|
it('returns selected server when action is SELECT_SERVER', () => {
|
||||||
const payload = fromPartial<RegularServer>({ id: 'abc123' });
|
const payload = fromPartial<RegularServer>({ id: 'abc123' });
|
||||||
expect(reducer(null, selectServer.fulfilled(payload, '', ''))).toEqual(payload);
|
expect(reducer(null, selectServer.fulfilled(payload, '', { serverId: '', buildShlinkApiClient }))).toEqual(payload);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -49,10 +48,10 @@ describe('selectedServerReducer', () => {
|
|||||||
|
|
||||||
health.mockResolvedValue({ version: serverVersion });
|
health.mockResolvedValue({ version: serverVersion });
|
||||||
|
|
||||||
await selectServer(id)(dispatch, getState, {});
|
await selectServer({ serverId: id, buildShlinkApiClient })(dispatch, getState, {});
|
||||||
|
|
||||||
expect(getState).toHaveBeenCalledTimes(1);
|
expect(getState).toHaveBeenCalledTimes(1);
|
||||||
expect(buildApiClient).toHaveBeenCalledTimes(1);
|
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
||||||
expect(dispatch).toHaveBeenCalledTimes(3); // "Pending", "reset" and "fulfilled"
|
expect(dispatch).toHaveBeenCalledTimes(3); // "Pending", "reset" and "fulfilled"
|
||||||
expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ payload: expectedSelectedServer }));
|
expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ payload: expectedSelectedServer }));
|
||||||
});
|
});
|
||||||
@@ -64,7 +63,7 @@ describe('selectedServerReducer', () => {
|
|||||||
|
|
||||||
health.mockRejectedValue({});
|
health.mockRejectedValue({});
|
||||||
|
|
||||||
await selectServer(id)(dispatch, getState, {});
|
await selectServer({ serverId: id, buildShlinkApiClient })(dispatch, getState, {});
|
||||||
|
|
||||||
expect(health).toHaveBeenCalled();
|
expect(health).toHaveBeenCalled();
|
||||||
expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ payload: expectedSelectedServer }));
|
expect(dispatch).toHaveBeenLastCalledWith(expect.objectContaining({ payload: expectedSelectedServer }));
|
||||||
@@ -75,7 +74,7 @@ describe('selectedServerReducer', () => {
|
|||||||
const getState = vi.fn(() => fromPartial<ShlinkState>({ servers: {} }));
|
const getState = vi.fn(() => fromPartial<ShlinkState>({ servers: {} }));
|
||||||
const expectedSelectedServer: NotFoundServer = { serverNotFound: true };
|
const expectedSelectedServer: NotFoundServer = { serverNotFound: true };
|
||||||
|
|
||||||
await selectServer(id)(dispatch, getState, {});
|
await selectServer({ serverId: id, buildShlinkApiClient })(dispatch, getState, {});
|
||||||
|
|
||||||
expect(getState).toHaveBeenCalled();
|
expect(getState).toHaveBeenCalled();
|
||||||
expect(health).not.toHaveBeenCalled();
|
expect(health).not.toHaveBeenCalled();
|
||||||
|
|||||||
Reference in New Issue
Block a user