mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2026-03-16 04:23:47 +00:00
Merge pull request #725 from acelaya-forks/feature/more-rtk
Feature/more rtk
This commit is contained in:
@@ -1,6 +1,7 @@
|
|||||||
import { Action } from 'redux';
|
import { Action } from 'redux';
|
||||||
import { ProblemDetailsError } from './errors';
|
import { ProblemDetailsError } from './errors';
|
||||||
|
|
||||||
|
/** @deprecated */
|
||||||
export interface ApiErrorAction extends Action<string> {
|
export interface ApiErrorAction extends Action<string> {
|
||||||
errorData?: ProblemDetailsError;
|
errorData?: ProblemDetailsError;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,19 +3,15 @@ import { combineReducers } from 'redux';
|
|||||||
import { serversReducer } from '../servers/reducers/servers';
|
import { serversReducer } from '../servers/reducers/servers';
|
||||||
import selectedServerReducer from '../servers/reducers/selectedServer';
|
import selectedServerReducer from '../servers/reducers/selectedServer';
|
||||||
import shortUrlsListReducer from '../short-urls/reducers/shortUrlsList';
|
import shortUrlsListReducer from '../short-urls/reducers/shortUrlsList';
|
||||||
import shortUrlCreationReducer from '../short-urls/reducers/shortUrlCreation';
|
|
||||||
import shortUrlDeletionReducer from '../short-urls/reducers/shortUrlDeletion';
|
|
||||||
import shortUrlEditionReducer from '../short-urls/reducers/shortUrlEdition';
|
|
||||||
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';
|
||||||
import orphanVisitsReducer from '../visits/reducers/orphanVisits';
|
import orphanVisitsReducer from '../visits/reducers/orphanVisits';
|
||||||
import nonOrphanVisitsReducer from '../visits/reducers/nonOrphanVisits';
|
import nonOrphanVisitsReducer from '../visits/reducers/nonOrphanVisits';
|
||||||
import shortUrlDetailReducer from '../short-urls/reducers/shortUrlDetail';
|
|
||||||
import tagsListReducer from '../tags/reducers/tagsList';
|
import tagsListReducer from '../tags/reducers/tagsList';
|
||||||
import tagDeleteReducer from '../tags/reducers/tagDelete';
|
import tagDeleteReducer from '../tags/reducers/tagDelete';
|
||||||
import tagEditReducer from '../tags/reducers/tagEdit';
|
import tagEditReducer from '../tags/reducers/tagEdit';
|
||||||
import settingsReducer from '../settings/reducers/settings';
|
import { settingsReducer } from '../settings/reducers/settings';
|
||||||
import visitsOverviewReducer from '../visits/reducers/visitsOverview';
|
import visitsOverviewReducer from '../visits/reducers/visitsOverview';
|
||||||
import { appUpdatesReducer } from '../app/reducers/appUpdates';
|
import { appUpdatesReducer } from '../app/reducers/appUpdates';
|
||||||
import { sidebarReducer } from '../common/reducers/sidebar';
|
import { sidebarReducer } from '../common/reducers/sidebar';
|
||||||
@@ -25,15 +21,15 @@ export default (container: IContainer) => combineReducers<ShlinkState>({
|
|||||||
servers: serversReducer,
|
servers: serversReducer,
|
||||||
selectedServer: selectedServerReducer,
|
selectedServer: selectedServerReducer,
|
||||||
shortUrlsList: shortUrlsListReducer,
|
shortUrlsList: shortUrlsListReducer,
|
||||||
shortUrlCreationResult: shortUrlCreationReducer,
|
shortUrlCreationResult: container.shortUrlCreationReducer,
|
||||||
shortUrlDeletion: shortUrlDeletionReducer,
|
shortUrlDeletion: container.shortUrlDeletionReducer,
|
||||||
shortUrlEdition: shortUrlEditionReducer,
|
shortUrlEdition: container.shortUrlEditionReducer,
|
||||||
|
shortUrlDetail: container.shortUrlDetailReducer,
|
||||||
shortUrlVisits: shortUrlVisitsReducer,
|
shortUrlVisits: shortUrlVisitsReducer,
|
||||||
tagVisits: tagVisitsReducer,
|
tagVisits: tagVisitsReducer,
|
||||||
domainVisits: domainVisitsReducer,
|
domainVisits: domainVisitsReducer,
|
||||||
orphanVisits: orphanVisitsReducer,
|
orphanVisits: orphanVisitsReducer,
|
||||||
nonOrphanVisits: nonOrphanVisitsReducer,
|
nonOrphanVisits: nonOrphanVisitsReducer,
|
||||||
shortUrlDetail: shortUrlDetailReducer,
|
|
||||||
tagsList: tagsListReducer,
|
tagsList: tagsListReducer,
|
||||||
tagDelete: tagDeleteReducer,
|
tagDelete: tagDeleteReducer,
|
||||||
tagEdit: tagEditReducer,
|
tagEdit: tagEditReducer,
|
||||||
|
|||||||
@@ -1,14 +1,10 @@
|
|||||||
import { Action } from 'redux';
|
import { createSlice, PayloadAction, PrepareAction } from '@reduxjs/toolkit';
|
||||||
import { dissoc, mergeDeepRight } from 'ramda';
|
import { mergeDeepRight } from 'ramda';
|
||||||
import { buildReducer } from '../../utils/helpers/redux';
|
|
||||||
import { RecursivePartial } from '../../utils/utils';
|
|
||||||
import { Theme } from '../../utils/theme';
|
import { Theme } from '../../utils/theme';
|
||||||
import { DateInterval } from '../../utils/dates/types';
|
import { DateInterval } from '../../utils/dates/types';
|
||||||
import { TagsOrder } from '../../tags/data/TagsListChildrenProps';
|
import { TagsOrder } from '../../tags/data/TagsListChildrenProps';
|
||||||
import { ShortUrlsOrder } from '../../short-urls/data';
|
import { ShortUrlsOrder } from '../../short-urls/data';
|
||||||
|
|
||||||
export const SET_SETTINGS = 'shlink/realTimeUpdates/SET_SETTINGS';
|
|
||||||
|
|
||||||
export const DEFAULT_SHORT_URLS_ORDERING: ShortUrlsOrder = {
|
export const DEFAULT_SHORT_URLS_ORDERING: ShortUrlsOrder = {
|
||||||
field: 'dateCreated',
|
field: 'dateCreated',
|
||||||
dir: 'DESC',
|
dir: 'DESC',
|
||||||
@@ -78,45 +74,37 @@ const initialState: Settings = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
type SettingsAction = Action & Settings;
|
type SettingsAction = PayloadAction<Settings>;
|
||||||
|
type SettingsPrepareAction = PrepareAction<Settings>;
|
||||||
|
|
||||||
type PartialSettingsAction = Action & RecursivePartial<Settings>;
|
const commonReducer = (state: Settings, { payload }: SettingsAction) => mergeDeepRight(state, payload);
|
||||||
|
const toReducer = (prepare: SettingsPrepareAction) => ({ reducer: commonReducer, prepare });
|
||||||
|
const toPreparedAction: SettingsPrepareAction = (payload: Settings) => ({ payload });
|
||||||
|
|
||||||
export default buildReducer<Settings, SettingsAction>({
|
const { reducer, actions } = createSlice({
|
||||||
[SET_SETTINGS]: (state, action) => mergeDeepRight(state, dissoc('type', action)),
|
name: 'settingsReducer',
|
||||||
}, initialState);
|
initialState,
|
||||||
|
reducers: {
|
||||||
export const toggleRealTimeUpdates = (enabled: boolean): PartialSettingsAction => ({
|
toggleRealTimeUpdates: toReducer((enabled: boolean) => toPreparedAction({ realTimeUpdates: { enabled } })),
|
||||||
type: SET_SETTINGS,
|
setRealTimeUpdatesInterval: toReducer((interval: number) => toPreparedAction({ realTimeUpdates: { interval } })),
|
||||||
realTimeUpdates: { enabled },
|
setShortUrlCreationSettings: toReducer(
|
||||||
|
(shortUrlCreation: ShortUrlCreationSettings) => toPreparedAction({ shortUrlCreation }),
|
||||||
|
),
|
||||||
|
setShortUrlsListSettings: toReducer((shortUrlsList: ShortUrlsListSettings) => toPreparedAction({ shortUrlsList })),
|
||||||
|
setUiSettings: toReducer((ui: UiSettings) => toPreparedAction({ ui })),
|
||||||
|
setVisitsSettings: toReducer((visits: VisitsSettings) => toPreparedAction({ visits })),
|
||||||
|
setTagsSettings: toReducer((tags: TagsSettings) => toPreparedAction({ tags })),
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
export const setRealTimeUpdatesInterval = (interval: number): PartialSettingsAction => ({
|
export const {
|
||||||
type: SET_SETTINGS,
|
toggleRealTimeUpdates,
|
||||||
realTimeUpdates: { interval },
|
setRealTimeUpdatesInterval,
|
||||||
});
|
setShortUrlCreationSettings,
|
||||||
|
setShortUrlsListSettings,
|
||||||
|
setUiSettings,
|
||||||
|
setVisitsSettings,
|
||||||
|
setTagsSettings,
|
||||||
|
} = actions;
|
||||||
|
|
||||||
export const setShortUrlCreationSettings = (settings: ShortUrlCreationSettings): PartialSettingsAction => ({
|
export const settingsReducer = reducer;
|
||||||
type: SET_SETTINGS,
|
|
||||||
shortUrlCreation: settings,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const setShortUrlsListSettings = (settings: ShortUrlsListSettings): PartialSettingsAction => ({
|
|
||||||
type: SET_SETTINGS,
|
|
||||||
shortUrlsList: settings,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const setUiSettings = (settings: UiSettings): PartialSettingsAction => ({
|
|
||||||
type: SET_SETTINGS,
|
|
||||||
ui: settings,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const setVisitsSettings = (settings: VisitsSettings): PartialSettingsAction => ({
|
|
||||||
type: SET_SETTINGS,
|
|
||||||
visits: settings,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const setTagsSettings = (settings: TagsSettings): PartialSettingsAction => ({
|
|
||||||
type: SET_SETTINGS,
|
|
||||||
tags: settings,
|
|
||||||
});
|
|
||||||
|
|||||||
@@ -6,16 +6,15 @@ import { ExternalLink } from 'react-external-link';
|
|||||||
import { useLocation, useParams } from 'react-router-dom';
|
import { useLocation, useParams } from 'react-router-dom';
|
||||||
import { SelectedServer } from '../servers/data';
|
import { SelectedServer } from '../servers/data';
|
||||||
import { Settings } from '../settings/reducers/settings';
|
import { Settings } from '../settings/reducers/settings';
|
||||||
import { OptionalString } from '../utils/utils';
|
import { ShortUrlIdentifier } from './data';
|
||||||
import { parseQuery } from '../utils/helpers/query';
|
import { parseQuery } from '../utils/helpers/query';
|
||||||
import { Message } from '../utils/Message';
|
import { Message } from '../utils/Message';
|
||||||
import { Result } from '../utils/Result';
|
import { Result } from '../utils/Result';
|
||||||
import { ShlinkApiError } from '../api/ShlinkApiError';
|
import { ShlinkApiError } from '../api/ShlinkApiError';
|
||||||
import { useGoBack, useToggle } from '../utils/helpers/hooks';
|
import { useGoBack } from '../utils/helpers/hooks';
|
||||||
import { ShortUrlFormProps } from './ShortUrlForm';
|
import { ShortUrlFormProps } from './ShortUrlForm';
|
||||||
import { ShortUrlDetail } from './reducers/shortUrlDetail';
|
import { ShortUrlDetail } from './reducers/shortUrlDetail';
|
||||||
import { EditShortUrlData } from './data';
|
import { EditShortUrl as EditShortUrlInfo, ShortUrlEdition } from './reducers/shortUrlEdition';
|
||||||
import { ShortUrlEdition } from './reducers/shortUrlEdition';
|
|
||||||
import { shortUrlDataFromShortUrl, urlDecodeShortCode } from './helpers';
|
import { shortUrlDataFromShortUrl, urlDecodeShortCode } from './helpers';
|
||||||
|
|
||||||
interface EditShortUrlConnectProps {
|
interface EditShortUrlConnectProps {
|
||||||
@@ -23,8 +22,8 @@ interface EditShortUrlConnectProps {
|
|||||||
selectedServer: SelectedServer;
|
selectedServer: SelectedServer;
|
||||||
shortUrlDetail: ShortUrlDetail;
|
shortUrlDetail: ShortUrlDetail;
|
||||||
shortUrlEdition: ShortUrlEdition;
|
shortUrlEdition: ShortUrlEdition;
|
||||||
getShortUrlDetail: (shortCode: string, domain: OptionalString) => void;
|
getShortUrlDetail: (shortUrl: ShortUrlIdentifier) => void;
|
||||||
editShortUrl: (shortUrl: string, domain: OptionalString, data: EditShortUrlData) => Promise<void>;
|
editShortUrl: (editShortUrl: EditShortUrlInfo) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const EditShortUrl = (ShortUrlForm: FC<ShortUrlFormProps>) => ({
|
export const EditShortUrl = (ShortUrlForm: FC<ShortUrlFormProps>) => ({
|
||||||
@@ -39,16 +38,15 @@ export const EditShortUrl = (ShortUrlForm: FC<ShortUrlFormProps>) => ({
|
|||||||
const params = useParams<{ shortCode: string }>();
|
const params = useParams<{ shortCode: string }>();
|
||||||
const goBack = useGoBack();
|
const goBack = useGoBack();
|
||||||
const { loading, error, errorData, shortUrl } = shortUrlDetail;
|
const { loading, error, errorData, shortUrl } = shortUrlDetail;
|
||||||
const { saving, error: savingError, errorData: savingErrorData } = shortUrlEdition;
|
const { saving, saved, error: savingError, errorData: savingErrorData } = shortUrlEdition;
|
||||||
const { domain } = parseQuery<{ domain?: string }>(search);
|
const { domain } = parseQuery<{ domain?: string }>(search);
|
||||||
const initialState = useMemo(
|
const initialState = useMemo(
|
||||||
() => shortUrlDataFromShortUrl(shortUrl, shortUrlCreationSettings),
|
() => shortUrlDataFromShortUrl(shortUrl, shortUrlCreationSettings),
|
||||||
[shortUrl, shortUrlCreationSettings],
|
[shortUrl, shortUrlCreationSettings],
|
||||||
);
|
);
|
||||||
const [savingSucceeded,, isSuccessful, isNotSuccessful] = useToggle();
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
params.shortCode && getShortUrlDetail(urlDecodeShortCode(params.shortCode), domain);
|
params.shortCode && getShortUrlDetail({ shortCode: urlDecodeShortCode(params.shortCode), domain });
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
if (loading) {
|
if (loading) {
|
||||||
@@ -88,18 +86,15 @@ export const EditShortUrl = (ShortUrlForm: FC<ShortUrlFormProps>) => ({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
isNotSuccessful();
|
editShortUrl({ ...shortUrl, data: shortUrlData });
|
||||||
editShortUrl(shortUrl.shortCode, shortUrl.domain, shortUrlData)
|
|
||||||
.then(isSuccessful)
|
|
||||||
.catch(isNotSuccessful);
|
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{savingError && (
|
{saved && savingError && (
|
||||||
<Result type="error" className="mt-3">
|
<Result type="error" className="mt-3">
|
||||||
<ShlinkApiError errorData={savingErrorData} fallbackMessage="An error occurred while updating short URL :(" />
|
<ShlinkApiError errorData={savingErrorData} fallbackMessage="An error occurred while updating short URL :(" />
|
||||||
</Result>
|
</Result>
|
||||||
)}
|
)}
|
||||||
{savingSucceeded && <Result type="success" className="mt-3">Short URL properly edited.</Result>}
|
{saved && !savingError && <Result type="success" className="mt-3">Short URL properly edited.</Result>}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -21,6 +21,11 @@ export interface ShortUrlData extends EditShortUrlData {
|
|||||||
findIfExists?: boolean;
|
findIfExists?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface ShortUrlIdentifier {
|
||||||
|
shortCode: string;
|
||||||
|
domain?: OptionalString;
|
||||||
|
}
|
||||||
|
|
||||||
export interface ShortUrl {
|
export interface ShortUrl {
|
||||||
shortCode: string;
|
shortCode: string;
|
||||||
shortUrl: string;
|
shortUrl: string;
|
||||||
@@ -47,11 +52,6 @@ export interface ShortUrlModalProps {
|
|||||||
toggle: () => void;
|
toggle: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ShortUrlIdentifier {
|
|
||||||
shortCode: string;
|
|
||||||
domain: OptionalString;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const SHORT_URLS_ORDERABLE_FIELDS = {
|
export const SHORT_URLS_ORDERABLE_FIELDS = {
|
||||||
dateCreated: 'Created at',
|
dateCreated: 'Created at',
|
||||||
shortCode: 'Short URL',
|
shortCode: 'Short URL',
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
import { Modal, ModalBody, ModalFooter, ModalHeader } from 'reactstrap';
|
import { Modal, ModalBody, ModalFooter, ModalHeader } from 'reactstrap';
|
||||||
import { identity, pipe } from 'ramda';
|
import { pipe } from 'ramda';
|
||||||
import { ShortUrlDeletion } from '../reducers/shortUrlDeletion';
|
import { ShortUrlDeletion } from '../reducers/shortUrlDeletion';
|
||||||
import { ShortUrlModalProps } from '../data';
|
import { ShortUrlIdentifier, ShortUrlModalProps } from '../data';
|
||||||
import { handleEventPreventingDefault, OptionalString } from '../../utils/utils';
|
import { handleEventPreventingDefault } from '../../utils/utils';
|
||||||
import { Result } from '../../utils/Result';
|
import { Result } from '../../utils/Result';
|
||||||
import { isInvalidDeletionError } from '../../api/utils';
|
import { isInvalidDeletionError } from '../../api/utils';
|
||||||
import { ShlinkApiError } from '../../api/ShlinkApiError';
|
import { ShlinkApiError } from '../../api/ShlinkApiError';
|
||||||
|
|
||||||
interface DeleteShortUrlModalConnectProps extends ShortUrlModalProps {
|
interface DeleteShortUrlModalConnectProps extends ShortUrlModalProps {
|
||||||
shortUrlDeletion: ShortUrlDeletion;
|
shortUrlDeletion: ShortUrlDeletion;
|
||||||
deleteShortUrl: (shortCode: string, domain: OptionalString) => Promise<void>;
|
deleteShortUrl: (shortUrl: ShortUrlIdentifier) => void;
|
||||||
resetDeleteShortUrl: () => void;
|
resetDeleteShortUrl: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -21,15 +21,9 @@ export const DeleteShortUrlModal = (
|
|||||||
|
|
||||||
useEffect(() => resetDeleteShortUrl, []);
|
useEffect(() => resetDeleteShortUrl, []);
|
||||||
|
|
||||||
const { error, errorData } = shortUrlDeletion;
|
const { loading, error, errorData } = shortUrlDeletion;
|
||||||
const close = pipe(resetDeleteShortUrl, toggle);
|
const close = pipe(resetDeleteShortUrl, toggle);
|
||||||
const handleDeleteUrl = handleEventPreventingDefault(() => {
|
const handleDeleteUrl = handleEventPreventingDefault(() => deleteShortUrl(shortUrl));
|
||||||
const { shortCode, domain } = shortUrl;
|
|
||||||
|
|
||||||
deleteShortUrl(shortCode, domain)
|
|
||||||
.then(toggle)
|
|
||||||
.catch(identity);
|
|
||||||
});
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal isOpen={isOpen} toggle={close} centered>
|
<Modal isOpen={isOpen} toggle={close} centered>
|
||||||
@@ -61,9 +55,9 @@ export const DeleteShortUrlModal = (
|
|||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
className="btn btn-danger"
|
className="btn btn-danger"
|
||||||
disabled={inputValue !== shortUrl.shortCode || shortUrlDeletion.loading}
|
disabled={inputValue !== shortUrl.shortCode || loading}
|
||||||
>
|
>
|
||||||
{shortUrlDeletion.loading ? 'Deleting...' : 'Delete'}
|
{loading ? 'Deleting...' : 'Delete'}
|
||||||
</button>
|
</button>
|
||||||
</ModalFooter>
|
</ModalFooter>
|
||||||
</form>
|
</form>
|
||||||
|
|||||||
@@ -24,8 +24,8 @@ export const ShortUrlsRowMenu = (
|
|||||||
QrCodeModal: ShortUrlModal,
|
QrCodeModal: ShortUrlModal,
|
||||||
) => ({ shortUrl, selectedServer }: ShortUrlsRowMenuProps) => {
|
) => ({ shortUrl, selectedServer }: ShortUrlsRowMenuProps) => {
|
||||||
const [isOpen, toggle] = useToggle();
|
const [isOpen, toggle] = useToggle();
|
||||||
const [isQrModalOpen, toggleQrCode] = useToggle();
|
const [isQrModalOpen,, openQrCodeModal, closeQrCodeModal] = useToggle();
|
||||||
const [isDeleteModalOpen, toggleDelete] = useToggle();
|
const [isDeleteModalOpen,, openDeleteModal, closeDeleteModal] = useToggle();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<DropdownBtnMenu toggle={toggle} isOpen={isOpen}>
|
<DropdownBtnMenu toggle={toggle} isOpen={isOpen}>
|
||||||
@@ -37,17 +37,17 @@ export const ShortUrlsRowMenu = (
|
|||||||
<FontAwesomeIcon icon={editIcon} fixedWidth /> Edit short URL
|
<FontAwesomeIcon icon={editIcon} fixedWidth /> Edit short URL
|
||||||
</DropdownItem>
|
</DropdownItem>
|
||||||
|
|
||||||
<DropdownItem onClick={toggleQrCode}>
|
<DropdownItem onClick={openQrCodeModal}>
|
||||||
<FontAwesomeIcon icon={qrIcon} fixedWidth /> QR code
|
<FontAwesomeIcon icon={qrIcon} fixedWidth /> QR code
|
||||||
</DropdownItem>
|
</DropdownItem>
|
||||||
<QrCodeModal shortUrl={shortUrl} isOpen={isQrModalOpen} toggle={toggleQrCode} />
|
<QrCodeModal shortUrl={shortUrl} isOpen={isQrModalOpen} toggle={closeQrCodeModal} />
|
||||||
|
|
||||||
<DropdownItem divider />
|
<DropdownItem divider />
|
||||||
|
|
||||||
<DropdownItem className="dropdown-item--danger" onClick={toggleDelete}>
|
<DropdownItem className="dropdown-item--danger" onClick={openDeleteModal}>
|
||||||
<FontAwesomeIcon icon={deleteIcon} fixedWidth /> Delete short URL
|
<FontAwesomeIcon icon={deleteIcon} fixedWidth /> Delete short URL
|
||||||
</DropdownItem>
|
</DropdownItem>
|
||||||
<DeleteShortUrlModal shortUrl={shortUrl} isOpen={isDeleteModalOpen} toggle={toggleDelete} />
|
<DeleteShortUrlModal shortUrl={shortUrl} isOpen={isDeleteModalOpen} toggle={closeDeleteModal} />
|
||||||
</DropdownBtnMenu>
|
</DropdownBtnMenu>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,16 +1,11 @@
|
|||||||
import { Action, Dispatch } from 'redux';
|
import { createSlice, PayloadAction } from '@reduxjs/toolkit';
|
||||||
import { GetState } from '../../container/types';
|
|
||||||
import { ShortUrl, ShortUrlData } from '../data';
|
import { ShortUrl, ShortUrlData } from '../data';
|
||||||
import { buildReducer, buildActionCreator } from '../../utils/helpers/redux';
|
import { createAsyncThunk } from '../../utils/helpers/redux';
|
||||||
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
||||||
import { parseApiError } from '../../api/utils';
|
import { parseApiError } from '../../api/utils';
|
||||||
import { ApiErrorAction } from '../../api/types/actions';
|
|
||||||
import { ProblemDetailsError } from '../../api/types/errors';
|
import { ProblemDetailsError } from '../../api/types/errors';
|
||||||
|
|
||||||
export const CREATE_SHORT_URL_START = 'shlink/createShortUrl/CREATE_SHORT_URL_START';
|
|
||||||
export const CREATE_SHORT_URL_ERROR = 'shlink/createShortUrl/CREATE_SHORT_URL_ERROR';
|
|
||||||
export const CREATE_SHORT_URL = 'shlink/createShortUrl/CREATE_SHORT_URL';
|
export const CREATE_SHORT_URL = 'shlink/createShortUrl/CREATE_SHORT_URL';
|
||||||
export const RESET_CREATE_SHORT_URL = 'shlink/createShortUrl/RESET_CREATE_SHORT_URL';
|
|
||||||
|
|
||||||
export interface ShortUrlCreation {
|
export interface ShortUrlCreation {
|
||||||
result: ShortUrl | null;
|
result: ShortUrl | null;
|
||||||
@@ -19,9 +14,7 @@ export interface ShortUrlCreation {
|
|||||||
errorData?: ProblemDetailsError;
|
errorData?: ProblemDetailsError;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface CreateShortUrlAction extends Action<string> {
|
export type CreateShortUrlAction = PayloadAction<ShortUrl>;
|
||||||
result: ShortUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
const initialState: ShortUrlCreation = {
|
const initialState: ShortUrlCreation = {
|
||||||
result: null,
|
result: null,
|
||||||
@@ -29,29 +22,33 @@ const initialState: ShortUrlCreation = {
|
|||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default buildReducer<ShortUrlCreation, CreateShortUrlAction & ApiErrorAction>({
|
export const shortUrlCreationReducerCreator = (buildShlinkApiClient: ShlinkApiClientBuilder) => {
|
||||||
[CREATE_SHORT_URL_START]: (state) => ({ ...state, saving: true, error: false }),
|
const createShortUrl = createAsyncThunk(CREATE_SHORT_URL, (data: ShortUrlData, { getState }): Promise<ShortUrl> => {
|
||||||
[CREATE_SHORT_URL_ERROR]: (state, { errorData }) => ({ ...state, saving: false, error: true, errorData }),
|
const { createShortUrl: shlinkCreateShortUrl } = buildShlinkApiClient(getState);
|
||||||
[CREATE_SHORT_URL]: (_, { result }) => ({ result, saving: false, error: false }),
|
return shlinkCreateShortUrl(data);
|
||||||
[RESET_CREATE_SHORT_URL]: () => initialState,
|
});
|
||||||
}, initialState);
|
|
||||||
|
|
||||||
export const createShortUrl = (buildShlinkApiClient: ShlinkApiClientBuilder) => (data: ShortUrlData) => async (
|
const { reducer, actions } = createSlice({
|
||||||
dispatch: Dispatch,
|
name: 'shortUrlCreationReducer',
|
||||||
getState: GetState,
|
initialState,
|
||||||
) => {
|
reducers: {
|
||||||
dispatch({ type: CREATE_SHORT_URL_START });
|
resetCreateShortUrl: () => initialState,
|
||||||
const { createShortUrl: shlinkCreateShortUrl } = buildShlinkApiClient(getState);
|
},
|
||||||
|
extraReducers: (builder) => {
|
||||||
|
builder.addCase(createShortUrl.pending, (state) => ({ ...state, saving: true, error: false }));
|
||||||
|
builder.addCase(
|
||||||
|
createShortUrl.rejected,
|
||||||
|
(state, { error }) => ({ ...state, saving: false, error: true, errorData: parseApiError(error) }),
|
||||||
|
);
|
||||||
|
builder.addCase(createShortUrl.fulfilled, (_, { payload: result }) => ({ result, saving: false, error: false }));
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
try {
|
const { resetCreateShortUrl } = actions;
|
||||||
const result = await shlinkCreateShortUrl(data);
|
|
||||||
|
|
||||||
dispatch<CreateShortUrlAction>({ type: CREATE_SHORT_URL, result });
|
return {
|
||||||
} catch (e: any) {
|
reducer,
|
||||||
dispatch<ApiErrorAction>({ type: CREATE_SHORT_URL_ERROR, errorData: parseApiError(e) });
|
createShortUrl,
|
||||||
|
resetCreateShortUrl,
|
||||||
throw e;
|
};
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const resetCreateShortUrl = buildActionCreator(RESET_CREATE_SHORT_URL);
|
|
||||||
|
|||||||
@@ -1,56 +1,57 @@
|
|||||||
import { Action, Dispatch } from 'redux';
|
import { createSlice, PayloadAction } from '@reduxjs/toolkit';
|
||||||
import { buildActionCreator, buildReducer } from '../../utils/helpers/redux';
|
import { createAsyncThunk } from '../../utils/helpers/redux';
|
||||||
import { GetState } from '../../container/types';
|
|
||||||
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
||||||
import { parseApiError } from '../../api/utils';
|
import { parseApiError } from '../../api/utils';
|
||||||
import { ApiErrorAction } from '../../api/types/actions';
|
|
||||||
import { ProblemDetailsError } from '../../api/types/errors';
|
import { ProblemDetailsError } from '../../api/types/errors';
|
||||||
|
import { ShortUrlIdentifier } from '../data';
|
||||||
|
|
||||||
export const DELETE_SHORT_URL_START = 'shlink/deleteShortUrl/DELETE_SHORT_URL_START';
|
|
||||||
export const DELETE_SHORT_URL_ERROR = 'shlink/deleteShortUrl/DELETE_SHORT_URL_ERROR';
|
|
||||||
export const SHORT_URL_DELETED = 'shlink/deleteShortUrl/SHORT_URL_DELETED';
|
export const SHORT_URL_DELETED = 'shlink/deleteShortUrl/SHORT_URL_DELETED';
|
||||||
export const RESET_DELETE_SHORT_URL = 'shlink/deleteShortUrl/RESET_DELETE_SHORT_URL';
|
|
||||||
|
|
||||||
export interface ShortUrlDeletion {
|
export interface ShortUrlDeletion {
|
||||||
shortCode: string;
|
shortCode: string;
|
||||||
loading: boolean;
|
loading: boolean;
|
||||||
|
deleted: boolean;
|
||||||
error: boolean;
|
error: boolean;
|
||||||
errorData?: ProblemDetailsError;
|
errorData?: ProblemDetailsError;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DeleteShortUrlAction extends Action<string> {
|
export type DeleteShortUrlAction = PayloadAction<ShortUrlIdentifier>;
|
||||||
shortCode: string;
|
|
||||||
domain?: string | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const initialState: ShortUrlDeletion = {
|
const initialState: ShortUrlDeletion = {
|
||||||
shortCode: '',
|
shortCode: '',
|
||||||
loading: false,
|
loading: false,
|
||||||
|
deleted: false,
|
||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default buildReducer<ShortUrlDeletion, DeleteShortUrlAction & ApiErrorAction>({
|
export const shortUrlDeletionReducerCreator = (buildShlinkApiClient: ShlinkApiClientBuilder) => {
|
||||||
[DELETE_SHORT_URL_START]: (state) => ({ ...state, loading: true, error: false }),
|
const deleteShortUrl = createAsyncThunk(
|
||||||
[DELETE_SHORT_URL_ERROR]: (state, { errorData }) => ({ ...state, errorData, loading: false, error: true }),
|
SHORT_URL_DELETED,
|
||||||
[SHORT_URL_DELETED]: (state, { shortCode }) => ({ ...state, shortCode, loading: false, error: false }),
|
async ({ shortCode, domain }: ShortUrlIdentifier, { getState }): Promise<ShortUrlIdentifier> => {
|
||||||
[RESET_DELETE_SHORT_URL]: () => initialState,
|
const { deleteShortUrl: shlinkDeleteShortUrl } = buildShlinkApiClient(getState);
|
||||||
}, initialState);
|
await shlinkDeleteShortUrl(shortCode, domain);
|
||||||
|
return { shortCode, domain };
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
export const deleteShortUrl = (buildShlinkApiClient: ShlinkApiClientBuilder) => (
|
const { actions, reducer } = createSlice({
|
||||||
shortCode: string,
|
name: 'shortUrlDeletion',
|
||||||
domain?: string | null,
|
initialState,
|
||||||
) => async (dispatch: Dispatch, getState: GetState) => {
|
reducers: {
|
||||||
dispatch({ type: DELETE_SHORT_URL_START });
|
resetDeleteShortUrl: () => initialState,
|
||||||
const { deleteShortUrl: shlinkDeleteShortUrl } = buildShlinkApiClient(getState);
|
},
|
||||||
|
extraReducers: (builder) => {
|
||||||
|
builder.addCase(deleteShortUrl.pending, (state) => ({ ...state, loading: true, error: false, deleted: false }));
|
||||||
|
builder.addCase(deleteShortUrl.rejected, (state, { error }) => (
|
||||||
|
{ ...state, errorData: parseApiError(error), loading: false, error: true, deleted: false }
|
||||||
|
));
|
||||||
|
builder.addCase(deleteShortUrl.fulfilled, (state, { payload }) => (
|
||||||
|
{ ...state, shortCode: payload.shortCode, loading: false, error: false, deleted: true }
|
||||||
|
));
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
try {
|
const { resetDeleteShortUrl } = actions;
|
||||||
await shlinkDeleteShortUrl(shortCode, domain);
|
|
||||||
dispatch<DeleteShortUrlAction>({ type: SHORT_URL_DELETED, shortCode, domain });
|
|
||||||
} catch (e: any) {
|
|
||||||
dispatch<ApiErrorAction>({ type: DELETE_SHORT_URL_ERROR, errorData: parseApiError(e) });
|
|
||||||
|
|
||||||
throw e;
|
return { reducer, deleteShortUrl, resetDeleteShortUrl };
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const resetDeleteShortUrl = buildActionCreator(RESET_DELETE_SHORT_URL);
|
|
||||||
|
|||||||
@@ -1,17 +1,12 @@
|
|||||||
import { Action, Dispatch } from 'redux';
|
import { createSlice, PayloadAction } from '@reduxjs/toolkit';
|
||||||
import { ShortUrl } from '../data';
|
import { ShortUrl, ShortUrlIdentifier } from '../data';
|
||||||
import { buildReducer } from '../../utils/helpers/redux';
|
import { createAsyncThunk } from '../../utils/helpers/redux';
|
||||||
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
||||||
import { OptionalString } from '../../utils/utils';
|
|
||||||
import { GetState } from '../../container/types';
|
|
||||||
import { shortUrlMatches } from '../helpers';
|
import { shortUrlMatches } from '../helpers';
|
||||||
import { parseApiError } from '../../api/utils';
|
import { parseApiError } from '../../api/utils';
|
||||||
import { ApiErrorAction } from '../../api/types/actions';
|
|
||||||
import { ProblemDetailsError } from '../../api/types/errors';
|
import { ProblemDetailsError } from '../../api/types/errors';
|
||||||
|
|
||||||
export const GET_SHORT_URL_DETAIL_START = 'shlink/shortUrlDetail/GET_SHORT_URL_DETAIL_START';
|
const GET_SHORT_URL_DETAIL = 'shlink/shortUrlDetail/GET_SHORT_URL_DETAIL';
|
||||||
export const GET_SHORT_URL_DETAIL_ERROR = 'shlink/shortUrlDetail/GET_SHORT_URL_DETAIL_ERROR';
|
|
||||||
export const GET_SHORT_URL_DETAIL = 'shlink/shortUrlDetail/GET_SHORT_URL_DETAIL';
|
|
||||||
|
|
||||||
export interface ShortUrlDetail {
|
export interface ShortUrlDetail {
|
||||||
shortUrl?: ShortUrl;
|
shortUrl?: ShortUrl;
|
||||||
@@ -20,35 +15,36 @@ export interface ShortUrlDetail {
|
|||||||
errorData?: ProblemDetailsError;
|
errorData?: ProblemDetailsError;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ShortUrlDetailAction extends Action<string> {
|
export type ShortUrlDetailAction = PayloadAction<ShortUrl>;
|
||||||
shortUrl: ShortUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
const initialState: ShortUrlDetail = {
|
const initialState: ShortUrlDetail = {
|
||||||
loading: false,
|
loading: false,
|
||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default buildReducer<ShortUrlDetail, ShortUrlDetailAction & ApiErrorAction>({
|
export const shortUrlDetailReducerCreator = (buildShlinkApiClient: ShlinkApiClientBuilder) => {
|
||||||
[GET_SHORT_URL_DETAIL_START]: () => ({ loading: true, error: false }),
|
const getShortUrlDetail = createAsyncThunk(
|
||||||
[GET_SHORT_URL_DETAIL_ERROR]: (_, { errorData }) => ({ loading: false, error: true, errorData }),
|
GET_SHORT_URL_DETAIL,
|
||||||
[GET_SHORT_URL_DETAIL]: (_, { shortUrl }) => ({ shortUrl, ...initialState }),
|
async ({ shortCode, domain }: ShortUrlIdentifier, { getState }): Promise<ShortUrl> => {
|
||||||
}, initialState);
|
const { shortUrlsList } = getState();
|
||||||
|
const alreadyLoaded = shortUrlsList?.shortUrls?.data.find((url) => shortUrlMatches(url, shortCode, domain));
|
||||||
|
|
||||||
export const getShortUrlDetail = (buildShlinkApiClient: ShlinkApiClientBuilder) => (
|
return alreadyLoaded ?? await buildShlinkApiClient(getState).getShortUrl(shortCode, domain);
|
||||||
shortCode: string,
|
},
|
||||||
domain: OptionalString,
|
);
|
||||||
) => async (dispatch: Dispatch, getState: GetState) => {
|
|
||||||
dispatch({ type: GET_SHORT_URL_DETAIL_START });
|
|
||||||
|
|
||||||
try {
|
const { reducer } = createSlice({
|
||||||
const { shortUrlsList } = getState();
|
name: 'shortUrlDetailReducer',
|
||||||
const shortUrl = shortUrlsList?.shortUrls?.data.find(
|
initialState,
|
||||||
(url) => shortUrlMatches(url, shortCode, domain),
|
reducers: {},
|
||||||
) ?? await buildShlinkApiClient(getState).getShortUrl(shortCode, domain);
|
extraReducers: (builder) => {
|
||||||
|
builder.addCase(getShortUrlDetail.pending, () => ({ loading: true, error: false }));
|
||||||
|
builder.addCase(getShortUrlDetail.rejected, (_, { error }) => (
|
||||||
|
{ loading: false, error: true, errorData: parseApiError(error) }
|
||||||
|
));
|
||||||
|
builder.addCase(getShortUrlDetail.fulfilled, (_, { payload: shortUrl }) => ({ ...initialState, shortUrl }));
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
dispatch<ShortUrlDetailAction>({ shortUrl, type: GET_SHORT_URL_DETAIL });
|
return { reducer, getShortUrlDetail };
|
||||||
} catch (e: any) {
|
|
||||||
dispatch<ApiErrorAction>({ type: GET_SHORT_URL_DETAIL_ERROR, errorData: parseApiError(e) });
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,55 +1,57 @@
|
|||||||
import { Action, Dispatch } from 'redux';
|
import { createSlice, PayloadAction } from '@reduxjs/toolkit';
|
||||||
import { buildReducer } from '../../utils/helpers/redux';
|
import { createAsyncThunk } from '../../utils/helpers/redux';
|
||||||
import { GetState } from '../../container/types';
|
import { EditShortUrlData, ShortUrl, ShortUrlIdentifier } from '../data';
|
||||||
import { OptionalString } from '../../utils/utils';
|
|
||||||
import { EditShortUrlData, ShortUrl } from '../data';
|
|
||||||
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
||||||
import { parseApiError } from '../../api/utils';
|
import { parseApiError } from '../../api/utils';
|
||||||
import { ApiErrorAction } from '../../api/types/actions';
|
|
||||||
import { ProblemDetailsError } from '../../api/types/errors';
|
import { ProblemDetailsError } from '../../api/types/errors';
|
||||||
|
|
||||||
export const EDIT_SHORT_URL_START = 'shlink/shortUrlEdition/EDIT_SHORT_URL_START';
|
|
||||||
export const EDIT_SHORT_URL_ERROR = 'shlink/shortUrlEdition/EDIT_SHORT_URL_ERROR';
|
|
||||||
export const SHORT_URL_EDITED = 'shlink/shortUrlEdition/SHORT_URL_EDITED';
|
export const SHORT_URL_EDITED = 'shlink/shortUrlEdition/SHORT_URL_EDITED';
|
||||||
|
|
||||||
export interface ShortUrlEdition {
|
export interface ShortUrlEdition {
|
||||||
shortUrl?: ShortUrl;
|
shortUrl?: ShortUrl;
|
||||||
saving: boolean;
|
saving: boolean;
|
||||||
error: boolean;
|
error: boolean;
|
||||||
|
saved: boolean;
|
||||||
errorData?: ProblemDetailsError;
|
errorData?: ProblemDetailsError;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ShortUrlEditedAction extends Action<string> {
|
export interface EditShortUrl extends ShortUrlIdentifier {
|
||||||
shortUrl: ShortUrl;
|
data: EditShortUrlData;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type ShortUrlEditedAction = PayloadAction<ShortUrl>;
|
||||||
|
|
||||||
const initialState: ShortUrlEdition = {
|
const initialState: ShortUrlEdition = {
|
||||||
saving: false,
|
saving: false,
|
||||||
|
saved: false,
|
||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default buildReducer<ShortUrlEdition, ShortUrlEditedAction & ApiErrorAction>({
|
export const shortUrlEditionReducerCreator = (buildShlinkApiClient: ShlinkApiClientBuilder) => {
|
||||||
[EDIT_SHORT_URL_START]: (state) => ({ ...state, saving: true, error: false }),
|
const editShortUrl = createAsyncThunk(
|
||||||
[EDIT_SHORT_URL_ERROR]: (state, { errorData }) => ({ ...state, saving: false, error: true, errorData }),
|
SHORT_URL_EDITED,
|
||||||
[SHORT_URL_EDITED]: (_, { shortUrl }) => ({ shortUrl, saving: false, error: false }),
|
({ shortCode, domain, data }: EditShortUrl, { getState }): Promise<ShortUrl> => {
|
||||||
}, initialState);
|
const { updateShortUrl } = buildShlinkApiClient(getState);
|
||||||
|
return updateShortUrl(shortCode, domain, data as any); // FIXME parse dates
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
export const editShortUrl = (buildShlinkApiClient: ShlinkApiClientBuilder) => (
|
const { reducer } = createSlice({
|
||||||
shortCode: string,
|
name: 'shortUrlEditionReducer',
|
||||||
domain: OptionalString,
|
initialState,
|
||||||
data: EditShortUrlData,
|
reducers: {},
|
||||||
) => async (dispatch: Dispatch, getState: GetState) => {
|
extraReducers: (builder) => {
|
||||||
dispatch({ type: EDIT_SHORT_URL_START });
|
builder.addCase(editShortUrl.pending, (state) => ({ ...state, saving: true, error: false, saved: false }));
|
||||||
|
builder.addCase(
|
||||||
|
editShortUrl.rejected,
|
||||||
|
(state, { error }) => ({ ...state, saving: false, error: true, saved: false, errorData: parseApiError(error) }),
|
||||||
|
);
|
||||||
|
builder.addCase(
|
||||||
|
editShortUrl.fulfilled,
|
||||||
|
(_, { payload: shortUrl }) => ({ shortUrl, saving: false, error: false, saved: true }),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const { updateShortUrl } = buildShlinkApiClient(getState);
|
return { reducer, editShortUrl };
|
||||||
|
|
||||||
try {
|
|
||||||
const shortUrl = await updateShortUrl(shortCode, domain, data as any); // FIXME parse dates;
|
|
||||||
|
|
||||||
dispatch<ShortUrlEditedAction>({ shortUrl, type: SHORT_URL_EDITED });
|
|
||||||
} catch (e: any) {
|
|
||||||
dispatch<ApiErrorAction>({ type: EDIT_SHORT_URL_ERROR, errorData: parseApiError(e) });
|
|
||||||
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -44,10 +44,11 @@ export default buildReducer<ShortUrlsList, ListShortUrlsCombinedAction>({
|
|||||||
[LIST_SHORT_URLS_START]: (state) => ({ ...state, loading: true, error: false }),
|
[LIST_SHORT_URLS_START]: (state) => ({ ...state, loading: true, error: false }),
|
||||||
[LIST_SHORT_URLS_ERROR]: () => ({ loading: false, error: true }),
|
[LIST_SHORT_URLS_ERROR]: () => ({ loading: false, error: true }),
|
||||||
[LIST_SHORT_URLS]: (_, { shortUrls }) => ({ loading: false, error: false, shortUrls }),
|
[LIST_SHORT_URLS]: (_, { shortUrls }) => ({ loading: false, error: false, shortUrls }),
|
||||||
[SHORT_URL_DELETED]: pipe(
|
[`${SHORT_URL_DELETED}/fulfilled`]: pipe( // TODO Do not hardcode action type here
|
||||||
(state: ShortUrlsList, { shortCode, domain }: DeleteShortUrlAction) => (!state.shortUrls ? state : assocPath(
|
(state: ShortUrlsList, { payload }: DeleteShortUrlAction) => (!state.shortUrls ? state : assocPath(
|
||||||
['shortUrls', 'data'],
|
['shortUrls', 'data'],
|
||||||
reject<ShortUrl, ShortUrl[]>((shortUrl) => shortUrlMatches(shortUrl, shortCode, domain), state.shortUrls.data),
|
reject<ShortUrl, ShortUrl[]>((shortUrl) =>
|
||||||
|
shortUrlMatches(shortUrl, payload.shortCode, payload.domain), state.shortUrls.data),
|
||||||
state,
|
state,
|
||||||
)),
|
)),
|
||||||
(state) => (!state.shortUrls ? state : assocPath(
|
(state) => (!state.shortUrls ? state : assocPath(
|
||||||
@@ -56,13 +57,13 @@ export default buildReducer<ShortUrlsList, ListShortUrlsCombinedAction>({
|
|||||||
state,
|
state,
|
||||||
)),
|
)),
|
||||||
),
|
),
|
||||||
[CREATE_VISITS]: (state, { createdVisits }) => assocPath(
|
[CREATE_VISITS]: (state, { payload }) => assocPath(
|
||||||
['shortUrls', 'data'],
|
['shortUrls', 'data'],
|
||||||
state.shortUrls?.data?.map(
|
state.shortUrls?.data?.map(
|
||||||
(currentShortUrl) => {
|
(currentShortUrl) => {
|
||||||
// Find the last of the new visit for this short URL, and pick the amount of visits from it
|
// Find the last of the new visit for this short URL, and pick the amount of visits from it
|
||||||
const lastVisit = last(
|
const lastVisit = last(
|
||||||
createdVisits.filter(
|
payload.createdVisits.filter(
|
||||||
({ shortUrl }) => shortUrl && shortUrlMatches(currentShortUrl, shortUrl.shortCode, shortUrl.domain),
|
({ shortUrl }) => shortUrl && shortUrlMatches(currentShortUrl, shortUrl.shortCode, shortUrl.domain),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
@@ -74,13 +75,13 @@ export default buildReducer<ShortUrlsList, ListShortUrlsCombinedAction>({
|
|||||||
),
|
),
|
||||||
state,
|
state,
|
||||||
),
|
),
|
||||||
[CREATE_SHORT_URL]: pipe(
|
[`${CREATE_SHORT_URL}/fulfilled`]: pipe( // TODO Do not hardcode action type here
|
||||||
// The only place where the list and the creation form coexist is the overview page.
|
// The only place where the list and the creation form coexist is the overview page.
|
||||||
// There we can assume we are displaying page 1, and therefore, we can safely prepend the new short URL.
|
// There we can assume we are displaying page 1, and therefore, we can safely prepend the new short URL.
|
||||||
// We can also remove the items above the amount that is displayed there.
|
// We can also remove the items above the amount that is displayed there.
|
||||||
(state: ShortUrlsList, { result }: CreateShortUrlAction) => (!state.shortUrls ? state : assocPath(
|
(state: ShortUrlsList, { payload }: CreateShortUrlAction) => (!state.shortUrls ? state : assocPath(
|
||||||
['shortUrls', 'data'],
|
['shortUrls', 'data'],
|
||||||
[result, ...state.shortUrls.data.slice(0, ITEMS_IN_OVERVIEW_PAGE - 1)],
|
[payload, ...state.shortUrls.data.slice(0, ITEMS_IN_OVERVIEW_PAGE - 1)],
|
||||||
state,
|
state,
|
||||||
)),
|
)),
|
||||||
(state: ShortUrlsList) => (!state.shortUrls ? state : assocPath(
|
(state: ShortUrlsList) => (!state.shortUrls ? state : assocPath(
|
||||||
@@ -89,7 +90,8 @@ export default buildReducer<ShortUrlsList, ListShortUrlsCombinedAction>({
|
|||||||
state,
|
state,
|
||||||
)),
|
)),
|
||||||
),
|
),
|
||||||
[SHORT_URL_EDITED]: (state, { shortUrl: editedShortUrl }) => (!state.shortUrls ? state : assocPath(
|
// TODO Do not hardcode action type here
|
||||||
|
[`${SHORT_URL_EDITED}/fulfilled`]: (state, { payload: editedShortUrl }) => (!state.shortUrls ? state : assocPath(
|
||||||
['shortUrls', 'data'],
|
['shortUrls', 'data'],
|
||||||
state.shortUrls.data.map((shortUrl) => {
|
state.shortUrls.data.map((shortUrl) => {
|
||||||
const { shortCode, domain } = editedShortUrl;
|
const { shortCode, domain } = editedShortUrl;
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import Bottle from 'bottlejs';
|
import Bottle from 'bottlejs';
|
||||||
|
import { prop } from 'ramda';
|
||||||
import { ShortUrlsFilteringBar } from '../ShortUrlsFilteringBar';
|
import { ShortUrlsFilteringBar } from '../ShortUrlsFilteringBar';
|
||||||
import { ShortUrlsList } from '../ShortUrlsList';
|
import { ShortUrlsList } from '../ShortUrlsList';
|
||||||
import { ShortUrlsRow } from '../helpers/ShortUrlsRow';
|
import { ShortUrlsRow } from '../helpers/ShortUrlsRow';
|
||||||
@@ -7,15 +8,15 @@ import { CreateShortUrl } from '../CreateShortUrl';
|
|||||||
import { DeleteShortUrlModal } from '../helpers/DeleteShortUrlModal';
|
import { DeleteShortUrlModal } from '../helpers/DeleteShortUrlModal';
|
||||||
import { CreateShortUrlResult } from '../helpers/CreateShortUrlResult';
|
import { CreateShortUrlResult } from '../helpers/CreateShortUrlResult';
|
||||||
import { listShortUrls } from '../reducers/shortUrlsList';
|
import { listShortUrls } from '../reducers/shortUrlsList';
|
||||||
import { createShortUrl, resetCreateShortUrl } from '../reducers/shortUrlCreation';
|
import { shortUrlCreationReducerCreator } from '../reducers/shortUrlCreation';
|
||||||
import { deleteShortUrl, resetDeleteShortUrl } from '../reducers/shortUrlDeletion';
|
import { shortUrlDeletionReducerCreator } from '../reducers/shortUrlDeletion';
|
||||||
import { editShortUrl } from '../reducers/shortUrlEdition';
|
import { shortUrlEditionReducerCreator } from '../reducers/shortUrlEdition';
|
||||||
|
import { shortUrlDetailReducerCreator } from '../reducers/shortUrlDetail';
|
||||||
import { ConnectDecorator } from '../../container/types';
|
import { ConnectDecorator } from '../../container/types';
|
||||||
import { ShortUrlsTable } from '../ShortUrlsTable';
|
import { ShortUrlsTable } from '../ShortUrlsTable';
|
||||||
import { QrCodeModal } from '../helpers/QrCodeModal';
|
|
||||||
import { ShortUrlForm } from '../ShortUrlForm';
|
import { ShortUrlForm } from '../ShortUrlForm';
|
||||||
import { EditShortUrl } from '../EditShortUrl';
|
import { EditShortUrl } from '../EditShortUrl';
|
||||||
import { getShortUrlDetail } from '../reducers/shortUrlDetail';
|
import { QrCodeModal } from '../helpers/QrCodeModal';
|
||||||
import { ExportShortUrlsBtn } from '../helpers/ExportShortUrlsBtn';
|
import { ExportShortUrlsBtn } from '../helpers/ExportShortUrlsBtn';
|
||||||
|
|
||||||
const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
||||||
@@ -55,18 +56,31 @@ const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
|||||||
bottle.serviceFactory('ExportShortUrlsBtn', ExportShortUrlsBtn, 'buildShlinkApiClient', 'ReportExporter');
|
bottle.serviceFactory('ExportShortUrlsBtn', ExportShortUrlsBtn, 'buildShlinkApiClient', 'ReportExporter');
|
||||||
bottle.decorator('ExportShortUrlsBtn', connect(['selectedServer']));
|
bottle.decorator('ExportShortUrlsBtn', connect(['selectedServer']));
|
||||||
|
|
||||||
|
// Reducers
|
||||||
|
bottle.serviceFactory('shortUrlCreationReducerCreator', shortUrlCreationReducerCreator, 'buildShlinkApiClient');
|
||||||
|
bottle.serviceFactory('shortUrlCreationReducer', prop('reducer'), 'shortUrlCreationReducerCreator');
|
||||||
|
|
||||||
|
bottle.serviceFactory('shortUrlEditionReducerCreator', shortUrlEditionReducerCreator, 'buildShlinkApiClient');
|
||||||
|
bottle.serviceFactory('shortUrlEditionReducer', prop('reducer'), 'shortUrlEditionReducerCreator');
|
||||||
|
|
||||||
|
bottle.serviceFactory('shortUrlDeletionReducerCreator', shortUrlDeletionReducerCreator, 'buildShlinkApiClient');
|
||||||
|
bottle.serviceFactory('shortUrlDeletionReducer', prop('reducer'), 'shortUrlDeletionReducerCreator');
|
||||||
|
|
||||||
|
bottle.serviceFactory('shortUrlDetailReducerCreator', shortUrlDetailReducerCreator, 'buildShlinkApiClient');
|
||||||
|
bottle.serviceFactory('shortUrlDetailReducer', prop('reducer'), 'shortUrlDetailReducerCreator');
|
||||||
|
|
||||||
// Actions
|
// Actions
|
||||||
bottle.serviceFactory('listShortUrls', listShortUrls, 'buildShlinkApiClient');
|
bottle.serviceFactory('listShortUrls', listShortUrls, 'buildShlinkApiClient');
|
||||||
|
|
||||||
bottle.serviceFactory('createShortUrl', createShortUrl, 'buildShlinkApiClient');
|
bottle.serviceFactory('createShortUrl', prop('createShortUrl'), 'shortUrlCreationReducerCreator');
|
||||||
bottle.serviceFactory('resetCreateShortUrl', () => resetCreateShortUrl);
|
bottle.serviceFactory('resetCreateShortUrl', prop('resetCreateShortUrl'), 'shortUrlCreationReducerCreator');
|
||||||
|
|
||||||
bottle.serviceFactory('deleteShortUrl', deleteShortUrl, 'buildShlinkApiClient');
|
bottle.serviceFactory('deleteShortUrl', prop('deleteShortUrl'), 'shortUrlDeletionReducerCreator');
|
||||||
bottle.serviceFactory('resetDeleteShortUrl', () => resetDeleteShortUrl);
|
bottle.serviceFactory('resetDeleteShortUrl', prop('resetDeleteShortUrl'), 'shortUrlDeletionReducerCreator');
|
||||||
|
|
||||||
bottle.serviceFactory('getShortUrlDetail', getShortUrlDetail, 'buildShlinkApiClient');
|
bottle.serviceFactory('getShortUrlDetail', prop('getShortUrlDetail'), 'shortUrlDetailReducerCreator');
|
||||||
|
|
||||||
bottle.serviceFactory('editShortUrl', editShortUrl, 'buildShlinkApiClient');
|
bottle.serviceFactory('editShortUrl', prop('editShortUrl'), 'shortUrlEditionReducerCreator');
|
||||||
};
|
};
|
||||||
|
|
||||||
export default provideServices;
|
export default provideServices;
|
||||||
|
|||||||
@@ -99,13 +99,13 @@ export default buildReducer<TagsList, TagsCombinedAction>({
|
|||||||
...state,
|
...state,
|
||||||
filteredTags: state.tags.filter((tag) => tag.toLowerCase().match(searchTerm.toLowerCase())),
|
filteredTags: state.tags.filter((tag) => tag.toLowerCase().match(searchTerm.toLowerCase())),
|
||||||
}),
|
}),
|
||||||
[CREATE_VISITS]: (state, { createdVisits }) => ({
|
[CREATE_VISITS]: (state, { payload }) => ({
|
||||||
...state,
|
...state,
|
||||||
stats: increaseVisitsForTags(calculateVisitsPerTag(createdVisits), state.stats),
|
stats: increaseVisitsForTags(calculateVisitsPerTag(payload.createdVisits), state.stats),
|
||||||
}),
|
}),
|
||||||
[CREATE_SHORT_URL]: ({ tags: stateTags, ...rest }, { result }) => ({
|
[`${CREATE_SHORT_URL}/fulfilled`]: ({ tags: stateTags, ...rest }, { payload }) => ({ // TODO Do not hardcode action type here
|
||||||
...rest,
|
...rest,
|
||||||
tags: stateTags.concat(result.tags.filter((tag) => !stateTags.includes(tag))), // More performant than [ ...new Set(...) ]
|
tags: stateTags.concat(payload.tags.filter((tag) => !stateTags.includes(tag))), // More performant than [ ...new Set(...) ]
|
||||||
}),
|
}),
|
||||||
}, initialState);
|
}, initialState);
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { ShlinkState } from '../../container/types';
|
|||||||
type ActionHandler<State, AT> = (currentState: State, action: AT) => State;
|
type ActionHandler<State, AT> = (currentState: State, action: AT) => State;
|
||||||
type ActionHandlerMap<State, AT> = Record<string, ActionHandler<State, AT>>;
|
type ActionHandlerMap<State, AT> = Record<string, ActionHandler<State, AT>>;
|
||||||
|
|
||||||
|
/** @deprecated */
|
||||||
export const buildReducer = <State, AT extends Action>(map: ActionHandlerMap<State, AT>, initialState: State) => (
|
export const buildReducer = <State, AT extends Action>(map: ActionHandlerMap<State, AT>, initialState: State) => (
|
||||||
state: State | undefined,
|
state: State | undefined,
|
||||||
action: AT,
|
action: AT,
|
||||||
@@ -16,6 +17,7 @@ export const buildReducer = <State, AT extends Action>(map: ActionHandlerMap<Sta
|
|||||||
return actionHandler ? actionHandler(currentState, action) : currentState;
|
return actionHandler ? actionHandler(currentState, action) : currentState;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/** @deprecated */
|
||||||
export const buildActionCreator = <T extends string>(type: T) => (): Action<T> => ({ type });
|
export const buildActionCreator = <T extends string>(type: T) => (): Action<T> => ({ type });
|
||||||
|
|
||||||
export const createAsyncThunk = <Returned, ThunkArg>(
|
export const createAsyncThunk = <Returned, ThunkArg>(
|
||||||
|
|||||||
@@ -14,11 +14,12 @@ import { NormalizedVisit, VisitsParams } from './types';
|
|||||||
import { CommonVisitsProps } from './types/CommonVisitsProps';
|
import { CommonVisitsProps } from './types/CommonVisitsProps';
|
||||||
import { toApiParams } from './types/helpers';
|
import { toApiParams } from './types/helpers';
|
||||||
import { urlDecodeShortCode } from '../short-urls/helpers';
|
import { urlDecodeShortCode } from '../short-urls/helpers';
|
||||||
|
import { ShortUrlIdentifier } from '../short-urls/data';
|
||||||
|
|
||||||
export interface ShortUrlVisitsProps extends CommonVisitsProps {
|
export interface ShortUrlVisitsProps extends CommonVisitsProps {
|
||||||
getShortUrlVisits: (shortCode: string, query?: ShlinkVisitsParams, doIntervalFallback?: boolean) => void;
|
getShortUrlVisits: (shortCode: string, query?: ShlinkVisitsParams, doIntervalFallback?: boolean) => void;
|
||||||
shortUrlVisits: ShortUrlVisitsState;
|
shortUrlVisits: ShortUrlVisitsState;
|
||||||
getShortUrlDetail: Function;
|
getShortUrlDetail: (shortUrl: ShortUrlIdentifier) => void;
|
||||||
shortUrlDetail: ShortUrlDetail;
|
shortUrlDetail: ShortUrlDetail;
|
||||||
cancelGetShortUrlVisits: () => void;
|
cancelGetShortUrlVisits: () => void;
|
||||||
}
|
}
|
||||||
@@ -44,7 +45,7 @@ export const ShortUrlVisits = ({ exportVisits }: ReportExporter) => boundToMercu
|
|||||||
);
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
getShortUrlDetail(urlDecodeShortCode(shortCode), domain);
|
getShortUrlDetail({ shortCode: urlDecodeShortCode(shortCode), domain });
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -56,10 +56,10 @@ export default buildReducer<DomainVisits, DomainVisitsCombinedAction>({
|
|||||||
[GET_DOMAIN_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
[GET_DOMAIN_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
||||||
[GET_DOMAIN_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
[GET_DOMAIN_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
||||||
[GET_DOMAIN_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
[GET_DOMAIN_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
||||||
[CREATE_VISITS]: (state, { createdVisits }) => {
|
[CREATE_VISITS]: (state, { payload }) => {
|
||||||
const { domain, visits, query = {} } = state;
|
const { domain, visits, query = {} } = state;
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
const newVisits = createdVisits
|
const newVisits = payload.createdVisits
|
||||||
.filter(({ shortUrl, visit }) =>
|
.filter(({ shortUrl, visit }) =>
|
||||||
shortUrl && domainMatches(shortUrl, domain) && isBetween(visit.date, startDate, endDate))
|
shortUrl && domainMatches(shortUrl, domain) && isBetween(visit.date, startDate, endDate))
|
||||||
.map(({ visit }) => visit);
|
.map(({ visit }) => visit);
|
||||||
|
|||||||
@@ -52,10 +52,10 @@ export default buildReducer<VisitsInfo, NonOrphanVisitsCombinedAction>({
|
|||||||
[GET_NON_ORPHAN_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
[GET_NON_ORPHAN_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
||||||
[GET_NON_ORPHAN_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
[GET_NON_ORPHAN_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
||||||
[GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
[GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
||||||
[CREATE_VISITS]: (state, { createdVisits }) => {
|
[CREATE_VISITS]: (state, { payload }) => {
|
||||||
const { visits, query = {} } = state;
|
const { visits, query = {} } = state;
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
const newVisits = createdVisits
|
const newVisits = payload.createdVisits
|
||||||
.filter(({ visit }) => isBetween(visit.date, startDate, endDate))
|
.filter(({ visit }) => isBetween(visit.date, startDate, endDate))
|
||||||
.map(({ visit }) => visit);
|
.map(({ visit }) => visit);
|
||||||
|
|
||||||
|
|||||||
@@ -55,10 +55,10 @@ export default buildReducer<VisitsInfo, OrphanVisitsCombinedAction>({
|
|||||||
[GET_ORPHAN_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
[GET_ORPHAN_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
||||||
[GET_ORPHAN_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
[GET_ORPHAN_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
||||||
[GET_ORPHAN_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
[GET_ORPHAN_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
||||||
[CREATE_VISITS]: (state, { createdVisits }) => {
|
[CREATE_VISITS]: (state, { payload }) => {
|
||||||
const { visits, query = {} } = state;
|
const { visits, query = {} } = state;
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
const newVisits = createdVisits
|
const newVisits = payload.createdVisits
|
||||||
.filter(({ visit, shortUrl }) => !shortUrl && isBetween(visit.date, startDate, endDate))
|
.filter(({ visit, shortUrl }) => !shortUrl && isBetween(visit.date, startDate, endDate))
|
||||||
.map(({ visit }) => visit);
|
.map(({ visit }) => visit);
|
||||||
|
|
||||||
|
|||||||
@@ -60,10 +60,10 @@ export default buildReducer<ShortUrlVisits, ShortUrlVisitsCombinedAction>({
|
|||||||
[GET_SHORT_URL_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
[GET_SHORT_URL_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
||||||
[GET_SHORT_URL_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
[GET_SHORT_URL_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
||||||
[GET_SHORT_URL_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
[GET_SHORT_URL_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
||||||
[CREATE_VISITS]: (state, { createdVisits }) => {
|
[CREATE_VISITS]: (state, { payload }) => {
|
||||||
const { shortCode, domain, visits, query = {} } = state;
|
const { shortCode, domain, visits, query = {} } = state;
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
const newVisits = createdVisits
|
const newVisits = payload.createdVisits
|
||||||
.filter(
|
.filter(
|
||||||
({ shortUrl, visit }) =>
|
({ shortUrl, visit }) =>
|
||||||
shortUrl && shortUrlMatches(shortUrl, shortCode, domain) && isBetween(visit.date, startDate, endDate),
|
shortUrl && shortUrlMatches(shortUrl, shortCode, domain) && isBetween(visit.date, startDate, endDate),
|
||||||
|
|||||||
@@ -53,10 +53,10 @@ export default buildReducer<TagVisits, TagsVisitsCombinedAction>({
|
|||||||
[GET_TAG_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
[GET_TAG_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
||||||
[GET_TAG_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
[GET_TAG_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
||||||
[GET_TAG_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
[GET_TAG_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
||||||
[CREATE_VISITS]: (state, { createdVisits }) => {
|
[CREATE_VISITS]: (state, { payload }) => {
|
||||||
const { tag, visits, query = {} } = state;
|
const { tag, visits, query = {} } = state;
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
const newVisits = createdVisits
|
const newVisits = payload.createdVisits
|
||||||
.filter(({ shortUrl, visit }) => shortUrl?.tags.includes(tag) && isBetween(visit.date, startDate, endDate))
|
.filter(({ shortUrl, visit }) => shortUrl?.tags.includes(tag) && isBetween(visit.date, startDate, endDate))
|
||||||
.map(({ visit }) => visit);
|
.map(({ visit }) => visit);
|
||||||
|
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
import { Action } from 'redux';
|
import { createAction, PayloadAction } from '@reduxjs/toolkit';
|
||||||
import { CreateVisit } from '../types';
|
import { CreateVisit } from '../types';
|
||||||
|
|
||||||
export const CREATE_VISITS = 'shlink/visitCreation/CREATE_VISITS';
|
export const CREATE_VISITS = 'shlink/visitCreation/CREATE_VISITS';
|
||||||
|
|
||||||
export interface CreateVisitsAction extends Action<typeof CREATE_VISITS> {
|
export type CreateVisitsAction = PayloadAction<{
|
||||||
createdVisits: CreateVisit[];
|
createdVisits: CreateVisit[];
|
||||||
}
|
}>;
|
||||||
|
|
||||||
export const createNewVisits = (createdVisits: CreateVisit[]): CreateVisitsAction => ({
|
export const createNewVisits = createAction(
|
||||||
type: CREATE_VISITS,
|
CREATE_VISITS,
|
||||||
createdVisits,
|
(createdVisits: CreateVisit[]) => ({ payload: { createdVisits } }),
|
||||||
});
|
);
|
||||||
|
|||||||
@@ -30,8 +30,8 @@ export default buildReducer<VisitsOverview, GetVisitsOverviewAction & CreateVisi
|
|||||||
[GET_OVERVIEW_START]: () => ({ ...initialState, loading: true }),
|
[GET_OVERVIEW_START]: () => ({ ...initialState, loading: true }),
|
||||||
[GET_OVERVIEW_ERROR]: () => ({ ...initialState, error: true }),
|
[GET_OVERVIEW_ERROR]: () => ({ ...initialState, error: true }),
|
||||||
[GET_OVERVIEW]: (_, { visitsCount, orphanVisitsCount }) => ({ ...initialState, visitsCount, orphanVisitsCount }),
|
[GET_OVERVIEW]: (_, { visitsCount, orphanVisitsCount }) => ({ ...initialState, visitsCount, orphanVisitsCount }),
|
||||||
[CREATE_VISITS]: ({ visitsCount, orphanVisitsCount = 0, ...rest }, { createdVisits }) => {
|
[CREATE_VISITS]: ({ visitsCount, orphanVisitsCount = 0, ...rest }, { payload }) => {
|
||||||
const { regularVisits, orphanVisits } = groupNewVisitsByType(createdVisits);
|
const { regularVisits, orphanVisits } = groupNewVisitsByType(payload.createdVisits);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...rest,
|
...rest,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import reducer, {
|
import {
|
||||||
SET_SETTINGS,
|
|
||||||
DEFAULT_SHORT_URLS_ORDERING,
|
DEFAULT_SHORT_URLS_ORDERING,
|
||||||
|
settingsReducer,
|
||||||
toggleRealTimeUpdates,
|
toggleRealTimeUpdates,
|
||||||
setRealTimeUpdatesInterval,
|
setRealTimeUpdatesInterval,
|
||||||
setShortUrlCreationSettings,
|
setShortUrlCreationSettings,
|
||||||
@@ -20,7 +20,9 @@ describe('settingsReducer', () => {
|
|||||||
|
|
||||||
describe('reducer', () => {
|
describe('reducer', () => {
|
||||||
it('returns realTimeUpdates when action is SET_SETTINGS', () => {
|
it('returns realTimeUpdates when action is SET_SETTINGS', () => {
|
||||||
expect(reducer(undefined, { type: SET_SETTINGS, realTimeUpdates })).toEqual(settings);
|
expect(
|
||||||
|
settingsReducer(undefined, { type: toggleRealTimeUpdates.toString(), payload: { realTimeUpdates } }),
|
||||||
|
).toEqual(settings);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -28,7 +30,10 @@ describe('settingsReducer', () => {
|
|||||||
it.each([[true], [false]])('updates settings with provided value and then loads updates again', (enabled) => {
|
it.each([[true], [false]])('updates settings with provided value and then loads updates again', (enabled) => {
|
||||||
const result = toggleRealTimeUpdates(enabled);
|
const result = toggleRealTimeUpdates(enabled);
|
||||||
|
|
||||||
expect(result).toEqual({ type: SET_SETTINGS, realTimeUpdates: { enabled } });
|
expect(result).toEqual({
|
||||||
|
type: toggleRealTimeUpdates.toString(),
|
||||||
|
payload: { realTimeUpdates: { enabled } },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -36,7 +41,10 @@ describe('settingsReducer', () => {
|
|||||||
it.each([[0], [1], [2], [10]])('updates settings with provided value and then loads updates again', (interval) => {
|
it.each([[0], [1], [2], [10]])('updates settings with provided value and then loads updates again', (interval) => {
|
||||||
const result = setRealTimeUpdatesInterval(interval);
|
const result = setRealTimeUpdatesInterval(interval);
|
||||||
|
|
||||||
expect(result).toEqual({ type: SET_SETTINGS, realTimeUpdates: { interval } });
|
expect(result).toEqual({
|
||||||
|
type: setRealTimeUpdatesInterval.toString(),
|
||||||
|
payload: { realTimeUpdates: { interval } },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -44,7 +52,10 @@ describe('settingsReducer', () => {
|
|||||||
it('creates action to set shortUrlCreation settings', () => {
|
it('creates action to set shortUrlCreation settings', () => {
|
||||||
const result = setShortUrlCreationSettings({ validateUrls: true });
|
const result = setShortUrlCreationSettings({ validateUrls: true });
|
||||||
|
|
||||||
expect(result).toEqual({ type: SET_SETTINGS, shortUrlCreation: { validateUrls: true } });
|
expect(result).toEqual({
|
||||||
|
type: setShortUrlCreationSettings.toString(),
|
||||||
|
payload: { shortUrlCreation: { validateUrls: true } },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -52,7 +63,10 @@ describe('settingsReducer', () => {
|
|||||||
it('creates action to set ui settings', () => {
|
it('creates action to set ui settings', () => {
|
||||||
const result = setUiSettings({ theme: 'dark' });
|
const result = setUiSettings({ theme: 'dark' });
|
||||||
|
|
||||||
expect(result).toEqual({ type: SET_SETTINGS, ui: { theme: 'dark' } });
|
expect(result).toEqual({
|
||||||
|
type: setUiSettings.toString(),
|
||||||
|
payload: { ui: { theme: 'dark' } },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -60,7 +74,10 @@ describe('settingsReducer', () => {
|
|||||||
it('creates action to set visits settings', () => {
|
it('creates action to set visits settings', () => {
|
||||||
const result = setVisitsSettings({ defaultInterval: 'last180Days' });
|
const result = setVisitsSettings({ defaultInterval: 'last180Days' });
|
||||||
|
|
||||||
expect(result).toEqual({ type: SET_SETTINGS, visits: { defaultInterval: 'last180Days' } });
|
expect(result).toEqual({
|
||||||
|
type: setVisitsSettings.toString(),
|
||||||
|
payload: { visits: { defaultInterval: 'last180Days' } },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -68,7 +85,10 @@ describe('settingsReducer', () => {
|
|||||||
it('creates action to set tags settings', () => {
|
it('creates action to set tags settings', () => {
|
||||||
const result = setTagsSettings({ defaultMode: 'list' });
|
const result = setTagsSettings({ defaultMode: 'list' });
|
||||||
|
|
||||||
expect(result).toEqual({ type: SET_SETTINGS, tags: { defaultMode: 'list' } });
|
expect(result).toEqual({
|
||||||
|
type: setTagsSettings.toString(),
|
||||||
|
payload: { tags: { defaultMode: 'list' } },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -76,7 +96,10 @@ describe('settingsReducer', () => {
|
|||||||
it('creates action to set short URLs list settings', () => {
|
it('creates action to set short URLs list settings', () => {
|
||||||
const result = setShortUrlsListSettings({ defaultOrdering: DEFAULT_SHORT_URLS_ORDERING });
|
const result = setShortUrlsListSettings({ defaultOrdering: DEFAULT_SHORT_URLS_ORDERING });
|
||||||
|
|
||||||
expect(result).toEqual({ type: SET_SETTINGS, shortUrlsList: { defaultOrdering: DEFAULT_SHORT_URLS_ORDERING } });
|
expect(result).toEqual({
|
||||||
|
type: setShortUrlsListSettings.toString(),
|
||||||
|
payload: { shortUrlsList: { defaultOrdering: DEFAULT_SHORT_URLS_ORDERING } },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -46,9 +46,16 @@ describe('<EditShortUrl />', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('shows error when saving data has failed', () => {
|
it('shows error when saving data has failed', () => {
|
||||||
setUp({}, { error: true });
|
setUp({}, { error: true, saved: true });
|
||||||
|
|
||||||
expect(screen.getByText('An error occurred while updating short URL :(')).toBeInTheDocument();
|
expect(screen.getByText('An error occurred while updating short URL :(')).toBeInTheDocument();
|
||||||
expect(screen.getByText('ShortUrlForm')).toBeInTheDocument();
|
expect(screen.getByText('ShortUrlForm')).toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('shows message when saving data succeeds', () => {
|
||||||
|
setUp({}, { error: false, saved: true });
|
||||||
|
|
||||||
|
expect(screen.getByText('Short URL properly edited.')).toBeInTheDocument();
|
||||||
|
expect(screen.getByText('ShortUrlForm')).toBeInTheDocument();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ describe('<DeleteShortUrlModal />', () => {
|
|||||||
shortCode: 'abc123',
|
shortCode: 'abc123',
|
||||||
longUrl: 'https://long-domain.com/foo/bar',
|
longUrl: 'https://long-domain.com/foo/bar',
|
||||||
});
|
});
|
||||||
const deleteShortUrl = jest.fn(async () => Promise.resolve());
|
const deleteShortUrl = jest.fn();
|
||||||
const setUp = (shortUrlDeletion: Partial<ShortUrlDeletion>) => renderWithEvents(
|
const setUp = (shortUrlDeletion: Partial<ShortUrlDeletion>) => renderWithEvents(
|
||||||
<DeleteShortUrlModal
|
<DeleteShortUrlModal
|
||||||
isOpen
|
isOpen
|
||||||
|
|||||||
@@ -1,19 +1,19 @@
|
|||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import reducer, {
|
import {
|
||||||
CREATE_SHORT_URL_START,
|
|
||||||
CREATE_SHORT_URL_ERROR,
|
|
||||||
CREATE_SHORT_URL,
|
|
||||||
RESET_CREATE_SHORT_URL,
|
|
||||||
createShortUrl,
|
|
||||||
resetCreateShortUrl,
|
|
||||||
CreateShortUrlAction,
|
CreateShortUrlAction,
|
||||||
|
shortUrlCreationReducerCreator,
|
||||||
} from '../../../src/short-urls/reducers/shortUrlCreation';
|
} from '../../../src/short-urls/reducers/shortUrlCreation';
|
||||||
import { ShortUrl } from '../../../src/short-urls/data';
|
import { ShortUrl } from '../../../src/short-urls/data';
|
||||||
import { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
|
import { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
|
||||||
import { ShlinkState } from '../../../src/container/types';
|
import { ShlinkState } from '../../../src/container/types';
|
||||||
|
|
||||||
describe('shortUrlCreationReducer', () => {
|
describe('shortUrlCreationReducer', () => {
|
||||||
const shortUrl = Mock.all<ShortUrl>();
|
const shortUrl = Mock.of<ShortUrl>();
|
||||||
|
const createShortUrlCall = jest.fn();
|
||||||
|
const buildShlinkApiClient = () => Mock.of<ShlinkApiClient>({ createShortUrl: createShortUrlCall });
|
||||||
|
const { reducer, createShortUrl, resetCreateShortUrl } = shortUrlCreationReducerCreator(buildShlinkApiClient);
|
||||||
|
|
||||||
|
afterEach(jest.resetAllMocks);
|
||||||
|
|
||||||
describe('reducer', () => {
|
describe('reducer', () => {
|
||||||
const action = (type: string, args: Partial<CreateShortUrlAction> = {}) => Mock.of<CreateShortUrlAction>(
|
const action = (type: string, args: Partial<CreateShortUrlAction> = {}) => Mock.of<CreateShortUrlAction>(
|
||||||
@@ -21,7 +21,7 @@ describe('shortUrlCreationReducer', () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
it('returns loading on CREATE_SHORT_URL_START', () => {
|
it('returns loading on CREATE_SHORT_URL_START', () => {
|
||||||
expect(reducer(undefined, action(CREATE_SHORT_URL_START))).toEqual({
|
expect(reducer(undefined, action(createShortUrl.pending.toString()))).toEqual({
|
||||||
result: null,
|
result: null,
|
||||||
saving: true,
|
saving: true,
|
||||||
error: false,
|
error: false,
|
||||||
@@ -29,7 +29,7 @@ describe('shortUrlCreationReducer', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('returns error on CREATE_SHORT_URL_ERROR', () => {
|
it('returns error on CREATE_SHORT_URL_ERROR', () => {
|
||||||
expect(reducer(undefined, action(CREATE_SHORT_URL_ERROR))).toEqual({
|
expect(reducer(undefined, action(createShortUrl.rejected.toString()))).toEqual({
|
||||||
result: null,
|
result: null,
|
||||||
saving: false,
|
saving: false,
|
||||||
error: true,
|
error: true,
|
||||||
@@ -37,7 +37,7 @@ describe('shortUrlCreationReducer', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('returns result on CREATE_SHORT_URL', () => {
|
it('returns result on CREATE_SHORT_URL', () => {
|
||||||
expect(reducer(undefined, action(CREATE_SHORT_URL, { result: shortUrl }))).toEqual({
|
expect(reducer(undefined, action(createShortUrl.fulfilled.toString(), { payload: shortUrl }))).toEqual({
|
||||||
result: shortUrl,
|
result: shortUrl,
|
||||||
saving: false,
|
saving: false,
|
||||||
error: false,
|
error: false,
|
||||||
@@ -45,7 +45,7 @@ describe('shortUrlCreationReducer', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('returns default state on RESET_CREATE_SHORT_URL', () => {
|
it('returns default state on RESET_CREATE_SHORT_URL', () => {
|
||||||
expect(reducer(undefined, action(RESET_CREATE_SHORT_URL))).toEqual({
|
expect(reducer(undefined, action(resetCreateShortUrl.toString()))).toEqual({
|
||||||
result: null,
|
result: null,
|
||||||
saving: false,
|
saving: false,
|
||||||
error: false,
|
error: false,
|
||||||
@@ -54,47 +54,43 @@ describe('shortUrlCreationReducer', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
describe('resetCreateShortUrl', () => {
|
describe('resetCreateShortUrl', () => {
|
||||||
it('returns proper action', () => expect(resetCreateShortUrl()).toEqual({ type: RESET_CREATE_SHORT_URL }));
|
it('returns proper action', () => expect(resetCreateShortUrl()).toEqual({ type: resetCreateShortUrl.toString() }));
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('createShortUrl', () => {
|
describe('createShortUrl', () => {
|
||||||
const createApiClientMock = (result: Promise<ShortUrl>) => Mock.of<ShlinkApiClient>({
|
|
||||||
createShortUrl: jest.fn().mockReturnValue(result),
|
|
||||||
});
|
|
||||||
const dispatch = jest.fn();
|
const dispatch = jest.fn();
|
||||||
const getState = () => Mock.all<ShlinkState>();
|
const getState = () => Mock.all<ShlinkState>();
|
||||||
|
|
||||||
afterEach(jest.resetAllMocks);
|
|
||||||
|
|
||||||
it('calls API on success', async () => {
|
it('calls API on success', async () => {
|
||||||
const apiClientMock = createApiClientMock(Promise.resolve(shortUrl));
|
createShortUrlCall.mockResolvedValue(shortUrl);
|
||||||
const dispatchable = createShortUrl(() => apiClientMock)({ longUrl: 'foo' });
|
await createShortUrl({ longUrl: 'foo' })(dispatch, getState, {});
|
||||||
|
|
||||||
await dispatchable(dispatch, getState);
|
expect(createShortUrlCall).toHaveBeenCalledTimes(1);
|
||||||
|
|
||||||
expect(apiClientMock.createShortUrl).toHaveBeenCalledTimes(1);
|
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
expect(dispatch).toHaveBeenCalledTimes(2);
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: CREATE_SHORT_URL_START });
|
expect(dispatch).toHaveBeenNthCalledWith(1, expect.objectContaining({
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: CREATE_SHORT_URL, result: shortUrl });
|
type: createShortUrl.pending.toString(),
|
||||||
|
}));
|
||||||
|
expect(dispatch).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
||||||
|
type: createShortUrl.fulfilled.toString(),
|
||||||
|
payload: shortUrl,
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('throws on error', async () => {
|
it('throws on error', async () => {
|
||||||
const error = 'Error';
|
const error = new Error('Error message');
|
||||||
const apiClientMock = createApiClientMock(Promise.reject(error));
|
createShortUrlCall.mockRejectedValue(error);
|
||||||
const dispatchable = createShortUrl(() => apiClientMock)({ longUrl: 'foo' });
|
|
||||||
|
|
||||||
expect.assertions(5);
|
await createShortUrl({ longUrl: 'foo' })(dispatch, getState, {});
|
||||||
|
|
||||||
try {
|
expect(createShortUrlCall).toHaveBeenCalledTimes(1);
|
||||||
await dispatchable(dispatch, getState);
|
|
||||||
} catch (e) {
|
|
||||||
expect(e).toEqual(error);
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(apiClientMock.createShortUrl).toHaveBeenCalledTimes(1);
|
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
expect(dispatch).toHaveBeenCalledTimes(2);
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: CREATE_SHORT_URL_START });
|
expect(dispatch).toHaveBeenNthCalledWith(1, expect.objectContaining({
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: CREATE_SHORT_URL_ERROR });
|
type: createShortUrl.pending.toString(),
|
||||||
|
}));
|
||||||
|
expect(dispatch).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
||||||
|
type: createShortUrl.rejected.toString(),
|
||||||
|
error: expect.objectContaining({ message: 'Error message' }),
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,45 +1,52 @@
|
|||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import reducer, {
|
import { shortUrlDeletionReducerCreator } from '../../../src/short-urls/reducers/shortUrlDeletion';
|
||||||
DELETE_SHORT_URL_ERROR,
|
|
||||||
DELETE_SHORT_URL_START,
|
|
||||||
RESET_DELETE_SHORT_URL,
|
|
||||||
SHORT_URL_DELETED,
|
|
||||||
resetDeleteShortUrl,
|
|
||||||
deleteShortUrl,
|
|
||||||
} from '../../../src/short-urls/reducers/shortUrlDeletion';
|
|
||||||
import { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
|
import { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
|
||||||
import { ProblemDetailsError } from '../../../src/api/types/errors';
|
import { ProblemDetailsError } from '../../../src/api/types/errors';
|
||||||
|
|
||||||
describe('shortUrlDeletionReducer', () => {
|
describe('shortUrlDeletionReducer', () => {
|
||||||
|
const deleteShortUrlCall = jest.fn();
|
||||||
|
const buildShlinkApiClient = () => Mock.of<ShlinkApiClient>({ deleteShortUrl: deleteShortUrlCall });
|
||||||
|
const { reducer, resetDeleteShortUrl, deleteShortUrl } = shortUrlDeletionReducerCreator(buildShlinkApiClient);
|
||||||
|
|
||||||
|
beforeEach(jest.clearAllMocks);
|
||||||
|
|
||||||
describe('reducer', () => {
|
describe('reducer', () => {
|
||||||
it('returns loading on DELETE_SHORT_URL_START', () =>
|
it('returns loading on DELETE_SHORT_URL_START', () =>
|
||||||
expect(reducer(undefined, { type: DELETE_SHORT_URL_START } as any)).toEqual({
|
expect(reducer(undefined, { type: deleteShortUrl.pending.toString() })).toEqual({
|
||||||
shortCode: '',
|
shortCode: '',
|
||||||
loading: true,
|
loading: true,
|
||||||
error: false,
|
error: false,
|
||||||
|
deleted: false,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('returns default on RESET_DELETE_SHORT_URL', () =>
|
it('returns default on RESET_DELETE_SHORT_URL', () =>
|
||||||
expect(reducer(undefined, { type: RESET_DELETE_SHORT_URL } as any)).toEqual({
|
expect(reducer(undefined, { type: resetDeleteShortUrl.toString() })).toEqual({
|
||||||
shortCode: '',
|
shortCode: '',
|
||||||
loading: false,
|
loading: false,
|
||||||
error: false,
|
error: false,
|
||||||
|
deleted: false,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('returns shortCode on SHORT_URL_DELETED', () =>
|
it('returns shortCode on SHORT_URL_DELETED', () =>
|
||||||
expect(reducer(undefined, { type: SHORT_URL_DELETED, shortCode: 'foo' } as any)).toEqual({
|
expect(reducer(undefined, {
|
||||||
|
type: deleteShortUrl.fulfilled.toString(),
|
||||||
|
payload: { shortCode: 'foo' },
|
||||||
|
})).toEqual({
|
||||||
shortCode: 'foo',
|
shortCode: 'foo',
|
||||||
loading: false,
|
loading: false,
|
||||||
error: false,
|
error: false,
|
||||||
|
deleted: true,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('returns errorData on DELETE_SHORT_URL_ERROR', () => {
|
it('returns errorData on DELETE_SHORT_URL_ERROR', () => {
|
||||||
const errorData = Mock.of<ProblemDetailsError>({ type: 'bar' });
|
const errorData = Mock.of<ProblemDetailsError>({ type: 'bar' });
|
||||||
|
const error = { response: { data: errorData } };
|
||||||
|
|
||||||
expect(reducer(undefined, { type: DELETE_SHORT_URL_ERROR, errorData } as any)).toEqual({
|
expect(reducer(undefined, { type: deleteShortUrl.rejected.toString(), error })).toEqual({
|
||||||
shortCode: '',
|
shortCode: '',
|
||||||
loading: false,
|
loading: false,
|
||||||
error: true,
|
error: true,
|
||||||
|
deleted: false,
|
||||||
errorData,
|
errorData,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -47,56 +54,47 @@ describe('shortUrlDeletionReducer', () => {
|
|||||||
|
|
||||||
describe('resetDeleteShortUrl', () => {
|
describe('resetDeleteShortUrl', () => {
|
||||||
it('returns expected action', () =>
|
it('returns expected action', () =>
|
||||||
expect(resetDeleteShortUrl()).toEqual({ type: RESET_DELETE_SHORT_URL }));
|
expect(resetDeleteShortUrl()).toEqual({ type: resetDeleteShortUrl.toString() }));
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('deleteShortUrl', () => {
|
describe('deleteShortUrl', () => {
|
||||||
const dispatch = jest.fn();
|
const dispatch = jest.fn();
|
||||||
const getState = jest.fn().mockReturnValue({ selectedServer: {} });
|
const getState = jest.fn().mockReturnValue({ selectedServer: {} });
|
||||||
|
|
||||||
afterEach(() => {
|
|
||||||
dispatch.mockReset();
|
|
||||||
getState.mockClear();
|
|
||||||
});
|
|
||||||
|
|
||||||
it.each(
|
it.each(
|
||||||
[[undefined], [null], ['example.com']],
|
[[undefined], [null], ['example.com']],
|
||||||
)('dispatches proper actions if API client request succeeds', async (domain) => {
|
)('dispatches proper actions if API client request succeeds', async (domain) => {
|
||||||
const apiClientMock = Mock.of<ShlinkApiClient>({
|
|
||||||
deleteShortUrl: jest.fn(() => ''),
|
|
||||||
});
|
|
||||||
const shortCode = 'abc123';
|
const shortCode = 'abc123';
|
||||||
|
|
||||||
await deleteShortUrl(() => apiClientMock)(shortCode, domain)(dispatch, getState);
|
await deleteShortUrl({ shortCode, domain })(dispatch, getState, {});
|
||||||
|
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
expect(dispatch).toHaveBeenCalledTimes(2);
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: DELETE_SHORT_URL_START });
|
expect(dispatch).toHaveBeenNthCalledWith(1, expect.objectContaining({ type: deleteShortUrl.pending.toString() }));
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: SHORT_URL_DELETED, shortCode, domain });
|
expect(dispatch).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
||||||
|
type: deleteShortUrl.fulfilled.toString(),
|
||||||
|
payload: { shortCode, domain },
|
||||||
|
}));
|
||||||
|
|
||||||
expect(apiClientMock.deleteShortUrl).toHaveBeenCalledTimes(1);
|
expect(deleteShortUrlCall).toHaveBeenCalledTimes(1);
|
||||||
expect(apiClientMock.deleteShortUrl).toHaveBeenCalledWith(shortCode, domain);
|
expect(deleteShortUrlCall).toHaveBeenCalledWith(shortCode, domain);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('dispatches proper actions if API client request fails', async () => {
|
it('dispatches proper actions if API client request fails', async () => {
|
||||||
const data = { foo: 'bar' };
|
const data = { foo: 'bar' };
|
||||||
const error = { response: { data } };
|
|
||||||
const apiClientMock = Mock.of<ShlinkApiClient>({
|
|
||||||
deleteShortUrl: jest.fn(async () => Promise.reject(error)),
|
|
||||||
});
|
|
||||||
const shortCode = 'abc123';
|
const shortCode = 'abc123';
|
||||||
|
|
||||||
try {
|
deleteShortUrlCall.mockRejectedValue({ response: { data } });
|
||||||
await deleteShortUrl(() => apiClientMock)(shortCode)(dispatch, getState);
|
|
||||||
} catch (e) {
|
await deleteShortUrl({ shortCode })(dispatch, getState, {});
|
||||||
expect(e).toEqual(error);
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
expect(dispatch).toHaveBeenCalledTimes(2);
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: DELETE_SHORT_URL_START });
|
expect(dispatch).toHaveBeenNthCalledWith(1, expect.objectContaining({ type: deleteShortUrl.pending.toString() }));
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: DELETE_SHORT_URL_ERROR, errorData: data });
|
expect(dispatch).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
||||||
|
type: deleteShortUrl.rejected.toString(),
|
||||||
|
}));
|
||||||
|
|
||||||
expect(apiClientMock.deleteShortUrl).toHaveBeenCalledTimes(1);
|
expect(deleteShortUrlCall).toHaveBeenCalledTimes(1);
|
||||||
expect(apiClientMock.deleteShortUrl).toHaveBeenCalledWith(shortCode, undefined);
|
expect(deleteShortUrlCall).toHaveBeenCalledWith(shortCode, undefined);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,31 +1,29 @@
|
|||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import reducer, {
|
import { ShortUrlDetailAction, shortUrlDetailReducerCreator } from '../../../src/short-urls/reducers/shortUrlDetail';
|
||||||
getShortUrlDetail,
|
|
||||||
GET_SHORT_URL_DETAIL_START,
|
|
||||||
GET_SHORT_URL_DETAIL_ERROR,
|
|
||||||
GET_SHORT_URL_DETAIL,
|
|
||||||
ShortUrlDetailAction,
|
|
||||||
} from '../../../src/short-urls/reducers/shortUrlDetail';
|
|
||||||
import { ShortUrl } from '../../../src/short-urls/data';
|
import { ShortUrl } from '../../../src/short-urls/data';
|
||||||
import { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
|
import { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
|
||||||
import { ShlinkState } from '../../../src/container/types';
|
import { ShlinkState } from '../../../src/container/types';
|
||||||
import { ShortUrlsList } from '../../../src/short-urls/reducers/shortUrlsList';
|
import { ShortUrlsList } from '../../../src/short-urls/reducers/shortUrlsList';
|
||||||
|
|
||||||
describe('shortUrlDetailReducer', () => {
|
describe('shortUrlDetailReducer', () => {
|
||||||
|
const getShortUrlCall = jest.fn();
|
||||||
|
const buildShlinkApiClient = () => Mock.of<ShlinkApiClient>({ getShortUrl: getShortUrlCall });
|
||||||
|
const { reducer, getShortUrlDetail } = shortUrlDetailReducerCreator(buildShlinkApiClient);
|
||||||
|
|
||||||
beforeEach(jest.clearAllMocks);
|
beforeEach(jest.clearAllMocks);
|
||||||
|
|
||||||
describe('reducer', () => {
|
describe('reducer', () => {
|
||||||
const action = (type: string) => Mock.of<ShortUrlDetailAction>({ type });
|
const action = (type: string) => Mock.of<ShortUrlDetailAction>({ type });
|
||||||
|
|
||||||
it('returns loading on GET_SHORT_URL_DETAIL_START', () => {
|
it('returns loading on GET_SHORT_URL_DETAIL_START', () => {
|
||||||
const state = reducer({ loading: false, error: false }, action(GET_SHORT_URL_DETAIL_START));
|
const state = reducer({ loading: false, error: false }, action(getShortUrlDetail.pending.toString()));
|
||||||
const { loading } = state;
|
const { loading } = state;
|
||||||
|
|
||||||
expect(loading).toEqual(true);
|
expect(loading).toEqual(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('stops loading and returns error on GET_SHORT_URL_DETAIL_ERROR', () => {
|
it('stops loading and returns error on GET_SHORT_URL_DETAIL_ERROR', () => {
|
||||||
const state = reducer({ loading: true, error: false }, action(GET_SHORT_URL_DETAIL_ERROR));
|
const state = reducer({ loading: true, error: false }, action(getShortUrlDetail.rejected.toString()));
|
||||||
const { loading, error } = state;
|
const { loading, error } = state;
|
||||||
|
|
||||||
expect(loading).toEqual(false);
|
expect(loading).toEqual(false);
|
||||||
@@ -34,7 +32,10 @@ describe('shortUrlDetailReducer', () => {
|
|||||||
|
|
||||||
it('return short URL on GET_SHORT_URL_DETAIL', () => {
|
it('return short URL on GET_SHORT_URL_DETAIL', () => {
|
||||||
const actionShortUrl = Mock.of<ShortUrl>({ longUrl: 'foo', shortCode: 'bar' });
|
const actionShortUrl = Mock.of<ShortUrl>({ longUrl: 'foo', shortCode: 'bar' });
|
||||||
const state = reducer({ loading: true, error: false }, { type: GET_SHORT_URL_DETAIL, shortUrl: actionShortUrl });
|
const state = reducer(
|
||||||
|
{ loading: true, error: false },
|
||||||
|
{ type: getShortUrlDetail.fulfilled.toString(), payload: actionShortUrl },
|
||||||
|
);
|
||||||
const { loading, error, shortUrl } = state;
|
const { loading, error, shortUrl } = state;
|
||||||
|
|
||||||
expect(loading).toEqual(false);
|
expect(loading).toEqual(false);
|
||||||
@@ -44,21 +45,22 @@ describe('shortUrlDetailReducer', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
describe('getShortUrlDetail', () => {
|
describe('getShortUrlDetail', () => {
|
||||||
const buildApiClientMock = (returned: Promise<ShortUrl>) => Mock.of<ShlinkApiClient>({
|
|
||||||
getShortUrl: jest.fn(async () => returned),
|
|
||||||
});
|
|
||||||
const dispatchMock = jest.fn();
|
const dispatchMock = jest.fn();
|
||||||
const buildGetState = (shortUrlsList?: ShortUrlsList) => () => Mock.of<ShlinkState>({ shortUrlsList });
|
const buildGetState = (shortUrlsList?: ShortUrlsList) => () => Mock.of<ShlinkState>({ shortUrlsList });
|
||||||
|
|
||||||
it('dispatches start and error when promise is rejected', async () => {
|
it('dispatches start and error when promise is rejected', async () => {
|
||||||
const ShlinkApiClient = buildApiClientMock(Promise.reject({}));
|
getShortUrlCall.mockRejectedValue({});
|
||||||
|
|
||||||
await getShortUrlDetail(() => ShlinkApiClient)('abc123', '')(dispatchMock, buildGetState());
|
await getShortUrlDetail({ shortCode: 'abc123', domain: '' })(dispatchMock, buildGetState(), {});
|
||||||
|
|
||||||
expect(dispatchMock).toHaveBeenCalledTimes(2);
|
expect(dispatchMock).toHaveBeenCalledTimes(2);
|
||||||
expect(dispatchMock).toHaveBeenNthCalledWith(1, { type: GET_SHORT_URL_DETAIL_START });
|
expect(dispatchMock).toHaveBeenNthCalledWith(1, expect.objectContaining({
|
||||||
expect(dispatchMock).toHaveBeenNthCalledWith(2, { type: GET_SHORT_URL_DETAIL_ERROR });
|
type: getShortUrlDetail.pending.toString(),
|
||||||
expect(ShlinkApiClient.getShortUrl).toHaveBeenCalledTimes(1);
|
}));
|
||||||
|
expect(dispatchMock).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
||||||
|
type: getShortUrlDetail.rejected.toString(),
|
||||||
|
}));
|
||||||
|
expect(getShortUrlCall).toHaveBeenCalledTimes(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
it.each([
|
it.each([
|
||||||
@@ -78,33 +80,44 @@ describe('shortUrlDetailReducer', () => {
|
|||||||
],
|
],
|
||||||
])('performs API call when short URL is not found in local state', async (shortUrlsList?: ShortUrlsList) => {
|
])('performs API call when short URL is not found in local state', async (shortUrlsList?: ShortUrlsList) => {
|
||||||
const resolvedShortUrl = Mock.of<ShortUrl>({ longUrl: 'foo', shortCode: 'abc123' });
|
const resolvedShortUrl = Mock.of<ShortUrl>({ longUrl: 'foo', shortCode: 'abc123' });
|
||||||
const ShlinkApiClient = buildApiClientMock(Promise.resolve(resolvedShortUrl));
|
getShortUrlCall.mockResolvedValue(resolvedShortUrl);
|
||||||
|
|
||||||
await getShortUrlDetail(() => ShlinkApiClient)('abc123', '')(dispatchMock, buildGetState(shortUrlsList));
|
await getShortUrlDetail({ shortCode: 'abc123', domain: '' })(dispatchMock, buildGetState(shortUrlsList), {});
|
||||||
|
|
||||||
expect(dispatchMock).toHaveBeenCalledTimes(2);
|
expect(dispatchMock).toHaveBeenCalledTimes(2);
|
||||||
expect(dispatchMock).toHaveBeenNthCalledWith(1, { type: GET_SHORT_URL_DETAIL_START });
|
expect(dispatchMock).toHaveBeenNthCalledWith(1, expect.objectContaining({
|
||||||
expect(dispatchMock).toHaveBeenNthCalledWith(2, { type: GET_SHORT_URL_DETAIL, shortUrl: resolvedShortUrl });
|
type: getShortUrlDetail.pending.toString(),
|
||||||
expect(ShlinkApiClient.getShortUrl).toHaveBeenCalledTimes(1);
|
}));
|
||||||
|
expect(dispatchMock).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
||||||
|
type: getShortUrlDetail.fulfilled.toString(),
|
||||||
|
payload: resolvedShortUrl,
|
||||||
|
}));
|
||||||
|
expect(getShortUrlCall).toHaveBeenCalledTimes(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('avoids API calls when short URL is found in local state', async () => {
|
it('avoids API calls when short URL is found in local state', async () => {
|
||||||
const foundShortUrl = Mock.of<ShortUrl>({ longUrl: 'foo', shortCode: 'abc123' });
|
const foundShortUrl = Mock.of<ShortUrl>({ longUrl: 'foo', shortCode: 'abc123' });
|
||||||
const ShlinkApiClient = buildApiClientMock(Promise.resolve(Mock.all<ShortUrl>()));
|
getShortUrlCall.mockResolvedValue(Mock.all<ShortUrl>());
|
||||||
|
|
||||||
await getShortUrlDetail(() => ShlinkApiClient)(foundShortUrl.shortCode, foundShortUrl.domain)(
|
await getShortUrlDetail(foundShortUrl)(
|
||||||
dispatchMock,
|
dispatchMock,
|
||||||
buildGetState(Mock.of<ShortUrlsList>({
|
buildGetState(Mock.of<ShortUrlsList>({
|
||||||
shortUrls: {
|
shortUrls: {
|
||||||
data: [foundShortUrl],
|
data: [foundShortUrl],
|
||||||
},
|
},
|
||||||
})),
|
})),
|
||||||
|
{},
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(dispatchMock).toHaveBeenCalledTimes(2);
|
expect(dispatchMock).toHaveBeenCalledTimes(2);
|
||||||
expect(dispatchMock).toHaveBeenNthCalledWith(1, { type: GET_SHORT_URL_DETAIL_START });
|
expect(dispatchMock).toHaveBeenNthCalledWith(1, expect.objectContaining({
|
||||||
expect(dispatchMock).toHaveBeenNthCalledWith(2, { type: GET_SHORT_URL_DETAIL, shortUrl: foundShortUrl });
|
type: getShortUrlDetail.pending.toString(),
|
||||||
expect(ShlinkApiClient.getShortUrl).not.toHaveBeenCalled();
|
}));
|
||||||
|
expect(dispatchMock).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
||||||
|
type: getShortUrlDetail.fulfilled.toString(),
|
||||||
|
payload: foundShortUrl,
|
||||||
|
}));
|
||||||
|
expect(getShortUrlCall).not.toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,11 +1,5 @@
|
|||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import reducer, {
|
import { ShortUrlEditedAction, shortUrlEditionReducerCreator } from '../../../src/short-urls/reducers/shortUrlEdition';
|
||||||
EDIT_SHORT_URL_START,
|
|
||||||
EDIT_SHORT_URL_ERROR,
|
|
||||||
SHORT_URL_EDITED,
|
|
||||||
editShortUrl,
|
|
||||||
ShortUrlEditedAction,
|
|
||||||
} from '../../../src/short-urls/reducers/shortUrlEdition';
|
|
||||||
import { ShlinkState } from '../../../src/container/types';
|
import { ShlinkState } from '../../../src/container/types';
|
||||||
import { ShortUrl } from '../../../src/short-urls/data';
|
import { ShortUrl } from '../../../src/short-urls/data';
|
||||||
import { SelectedServer } from '../../../src/servers/data';
|
import { SelectedServer } from '../../../src/servers/data';
|
||||||
@@ -14,48 +8,59 @@ describe('shortUrlEditionReducer', () => {
|
|||||||
const longUrl = 'https://shlink.io';
|
const longUrl = 'https://shlink.io';
|
||||||
const shortCode = 'abc123';
|
const shortCode = 'abc123';
|
||||||
const shortUrl = Mock.of<ShortUrl>({ longUrl, shortCode });
|
const shortUrl = Mock.of<ShortUrl>({ longUrl, shortCode });
|
||||||
|
const updateShortUrl = jest.fn().mockResolvedValue(shortUrl);
|
||||||
|
const buildShlinkApiClient = jest.fn().mockReturnValue({ updateShortUrl });
|
||||||
|
const { reducer, editShortUrl } = shortUrlEditionReducerCreator(buildShlinkApiClient);
|
||||||
|
|
||||||
|
afterEach(jest.clearAllMocks);
|
||||||
|
|
||||||
describe('reducer', () => {
|
describe('reducer', () => {
|
||||||
it('returns loading on EDIT_SHORT_URL_START', () => {
|
it('returns loading on EDIT_SHORT_URL_START', () => {
|
||||||
expect(reducer(undefined, Mock.of<ShortUrlEditedAction>({ type: EDIT_SHORT_URL_START }))).toEqual({
|
expect(reducer(undefined, Mock.of<ShortUrlEditedAction>({ type: editShortUrl.pending.toString() }))).toEqual({
|
||||||
saving: true,
|
saving: true,
|
||||||
|
saved: false,
|
||||||
error: false,
|
error: false,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('returns error on EDIT_SHORT_URL_ERROR', () => {
|
it('returns error on EDIT_SHORT_URL_ERROR', () => {
|
||||||
expect(reducer(undefined, Mock.of<ShortUrlEditedAction>({ type: EDIT_SHORT_URL_ERROR }))).toEqual({
|
expect(reducer(undefined, Mock.of<ShortUrlEditedAction>({ type: editShortUrl.rejected.toString() }))).toEqual({
|
||||||
saving: false,
|
saving: false,
|
||||||
|
saved: false,
|
||||||
error: true,
|
error: true,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('returns provided tags and shortCode on SHORT_URL_EDITED', () => {
|
it('returns provided tags and shortCode on SHORT_URL_EDITED', () => {
|
||||||
expect(reducer(undefined, { type: SHORT_URL_EDITED, shortUrl })).toEqual({
|
expect(reducer(undefined, { type: editShortUrl.fulfilled.toString(), payload: shortUrl })).toEqual({
|
||||||
shortUrl,
|
shortUrl,
|
||||||
saving: false,
|
saving: false,
|
||||||
|
saved: true,
|
||||||
error: false,
|
error: false,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('editShortUrl', () => {
|
describe('editShortUrl', () => {
|
||||||
const updateShortUrl = jest.fn().mockResolvedValue(shortUrl);
|
|
||||||
const buildShlinkApiClient = jest.fn().mockReturnValue({ updateShortUrl });
|
|
||||||
const dispatch = jest.fn();
|
const dispatch = jest.fn();
|
||||||
const createGetState = (selectedServer: SelectedServer = null) => () => Mock.of<ShlinkState>({ selectedServer });
|
const createGetState = (selectedServer: SelectedServer = null) => () => Mock.of<ShlinkState>({ selectedServer });
|
||||||
|
|
||||||
afterEach(jest.clearAllMocks);
|
afterEach(jest.clearAllMocks);
|
||||||
|
|
||||||
it.each([[undefined], [null], ['example.com']])('dispatches short URL on success', async (domain) => {
|
it.each([[undefined], [null], ['example.com']])('dispatches short URL on success', async (domain) => {
|
||||||
await editShortUrl(buildShlinkApiClient)(shortCode, domain, { longUrl })(dispatch, createGetState());
|
await editShortUrl({ shortCode, domain, data: { longUrl } })(dispatch, createGetState(), {});
|
||||||
|
|
||||||
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
||||||
expect(updateShortUrl).toHaveBeenCalledTimes(1);
|
expect(updateShortUrl).toHaveBeenCalledTimes(1);
|
||||||
expect(updateShortUrl).toHaveBeenCalledWith(shortCode, domain, { longUrl });
|
expect(updateShortUrl).toHaveBeenCalledWith(shortCode, domain, { longUrl });
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
expect(dispatch).toHaveBeenCalledTimes(2);
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: EDIT_SHORT_URL_START });
|
expect(dispatch).toHaveBeenNthCalledWith(1, expect.objectContaining({
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: SHORT_URL_EDITED, shortUrl });
|
type: editShortUrl.pending.toString(),
|
||||||
|
}));
|
||||||
|
expect(dispatch).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
||||||
|
type: editShortUrl.fulfilled.toString(),
|
||||||
|
payload: shortUrl,
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('dispatches error on failure', async () => {
|
it('dispatches error on failure', async () => {
|
||||||
@@ -63,18 +68,14 @@ describe('shortUrlEditionReducer', () => {
|
|||||||
|
|
||||||
updateShortUrl.mockRejectedValue(error);
|
updateShortUrl.mockRejectedValue(error);
|
||||||
|
|
||||||
try {
|
await editShortUrl({ shortCode, data: { longUrl } })(dispatch, createGetState(), {});
|
||||||
await editShortUrl(buildShlinkApiClient)(shortCode, undefined, { longUrl })(dispatch, createGetState());
|
|
||||||
} catch (e) {
|
|
||||||
expect(e).toBe(error);
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
||||||
expect(updateShortUrl).toHaveBeenCalledTimes(1);
|
expect(updateShortUrl).toHaveBeenCalledTimes(1);
|
||||||
expect(updateShortUrl).toHaveBeenCalledWith(shortCode, undefined, { longUrl });
|
expect(updateShortUrl).toHaveBeenCalledWith(shortCode, undefined, { longUrl });
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
expect(dispatch).toHaveBeenCalledTimes(2);
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: EDIT_SHORT_URL_START });
|
expect(dispatch).toHaveBeenNthCalledWith(1, expect.objectContaining({ type: editShortUrl.pending.toString() }));
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: EDIT_SHORT_URL_ERROR });
|
expect(dispatch).toHaveBeenNthCalledWith(2, expect.objectContaining({ type: editShortUrl.rejected.toString() }));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ describe('shortUrlsListReducer', () => {
|
|||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
expect(reducer(state, { type: SHORT_URL_DELETED, shortCode } as any)).toEqual({
|
expect(reducer(state, { type: `${SHORT_URL_DELETED}/fulfilled`, payload: { shortCode } } as any)).toEqual({
|
||||||
shortUrls: {
|
shortUrls: {
|
||||||
data: [{ shortCode, domain: 'example.com' }, { shortCode: 'foo' }],
|
data: [{ shortCode, domain: 'example.com' }, { shortCode: 'foo' }],
|
||||||
pagination: { totalItems: 9 },
|
pagination: { totalItems: 9 },
|
||||||
@@ -85,7 +85,7 @@ describe('shortUrlsListReducer', () => {
|
|||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
expect(reducer(state, { type: CREATE_VISITS, createdVisits } as any)).toEqual({
|
expect(reducer(state, { type: CREATE_VISITS, payload: { createdVisits } } as any)).toEqual({
|
||||||
shortUrls: {
|
shortUrls: {
|
||||||
data: [
|
data: [
|
||||||
{ shortCode, domain: 'example.com', visitsCount: 5 },
|
{ shortCode, domain: 'example.com', visitsCount: 5 },
|
||||||
@@ -142,7 +142,7 @@ describe('shortUrlsListReducer', () => {
|
|||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
expect(reducer(state, { type: CREATE_SHORT_URL, result: newShortUrl } as any)).toEqual({
|
expect(reducer(state, { type: `${CREATE_SHORT_URL}/fulfilled`, payload: newShortUrl } as any)).toEqual({
|
||||||
shortUrls: {
|
shortUrls: {
|
||||||
data: expectedData,
|
data: expectedData,
|
||||||
pagination: { totalItems: 16 },
|
pagination: { totalItems: 16 },
|
||||||
@@ -181,7 +181,7 @@ describe('shortUrlsListReducer', () => {
|
|||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
const result = reducer(state, { type: SHORT_URL_EDITED, shortUrl: editedShortUrl } as any);
|
const result = reducer(state, { type: `${SHORT_URL_EDITED}/fulfilled`, payload: editedShortUrl } as any);
|
||||||
|
|
||||||
expect(result.shortUrls?.data).toEqual(expectedList);
|
expect(result.shortUrls?.data).toEqual(expectedList);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -83,9 +83,9 @@ describe('tagsListReducer', () => {
|
|||||||
[['new', 'tag'], ['foo', 'bar', 'baz', 'foo2', 'fo', 'new', 'tag']],
|
[['new', 'tag'], ['foo', 'bar', 'baz', 'foo2', 'fo', 'new', 'tag']],
|
||||||
])('appends new short URL\'s tags to the list of tags on CREATE_SHORT_URL', (shortUrlTags, expectedTags) => {
|
])('appends new short URL\'s tags to the list of tags on CREATE_SHORT_URL', (shortUrlTags, expectedTags) => {
|
||||||
const tags = ['foo', 'bar', 'baz', 'foo2', 'fo'];
|
const tags = ['foo', 'bar', 'baz', 'foo2', 'fo'];
|
||||||
const result = Mock.of<ShortUrl>({ tags: shortUrlTags });
|
const payload = Mock.of<ShortUrl>({ tags: shortUrlTags });
|
||||||
|
|
||||||
expect(reducer(state({ tags }), { type: CREATE_SHORT_URL, result } as any)).toEqual({
|
expect(reducer(state({ tags }), { type: `${CREATE_SHORT_URL}/fulfilled`, payload } as any)).toEqual({
|
||||||
tags: expectedTags,
|
tags: expectedTags,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -135,7 +135,7 @@ describe('domainVisitsReducer', () => {
|
|||||||
|
|
||||||
const { visits } = reducer(prevState, {
|
const { visits } = reducer(prevState, {
|
||||||
type: CREATE_VISITS,
|
type: CREATE_VISITS,
|
||||||
createdVisits: [{ shortUrl, visit: { date: formatIsoDate(now) ?? undefined } }],
|
payload: { createdVisits: [{ shortUrl, visit: { date: formatIsoDate(now) ?? undefined } }] },
|
||||||
} as any);
|
} as any);
|
||||||
|
|
||||||
expect(visits).toHaveLength(expectedVisits);
|
expect(visits).toHaveLength(expectedVisits);
|
||||||
|
|||||||
@@ -105,10 +105,10 @@ describe('nonOrphanVisitsReducer', () => {
|
|||||||
const prevState = buildState({ ...state, visits: visitsMocks });
|
const prevState = buildState({ ...state, visits: visitsMocks });
|
||||||
const visit = Mock.of<Visit>({ date: formatIsoDate(now) ?? undefined });
|
const visit = Mock.of<Visit>({ date: formatIsoDate(now) ?? undefined });
|
||||||
|
|
||||||
const { visits } = reducer(
|
const { visits } = reducer(prevState, {
|
||||||
prevState,
|
type: CREATE_VISITS,
|
||||||
{ type: CREATE_VISITS, createdVisits: [{ visit }, { visit }] } as any,
|
payload: { createdVisits: [{ visit }, { visit }] },
|
||||||
);
|
} as any);
|
||||||
|
|
||||||
expect(visits).toHaveLength(expectedVisits);
|
expect(visits).toHaveLength(expectedVisits);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -105,10 +105,10 @@ describe('orphanVisitsReducer', () => {
|
|||||||
const prevState = buildState({ ...state, visits: visitsMocks });
|
const prevState = buildState({ ...state, visits: visitsMocks });
|
||||||
const visit = Mock.of<Visit>({ date: formatIsoDate(now) ?? undefined });
|
const visit = Mock.of<Visit>({ date: formatIsoDate(now) ?? undefined });
|
||||||
|
|
||||||
const { visits } = reducer(
|
const { visits } = reducer(prevState, {
|
||||||
prevState,
|
type: CREATE_VISITS,
|
||||||
{ type: CREATE_VISITS, createdVisits: [{ visit }, { visit }] } as any,
|
payload: { createdVisits: [{ visit }, { visit }] },
|
||||||
);
|
} as any);
|
||||||
|
|
||||||
expect(visits).toHaveLength(expectedVisits);
|
expect(visits).toHaveLength(expectedVisits);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -126,10 +126,10 @@ describe('shortUrlVisitsReducer', () => {
|
|||||||
visits: visitsMocks,
|
visits: visitsMocks,
|
||||||
});
|
});
|
||||||
|
|
||||||
const { visits } = reducer(
|
const { visits } = reducer(prevState, {
|
||||||
prevState,
|
type: CREATE_VISITS,
|
||||||
{ type: CREATE_VISITS, createdVisits: [{ shortUrl, visit: { date: formatIsoDate(now) ?? undefined } }] } as any,
|
payload: { createdVisits: [{ shortUrl, visit: { date: formatIsoDate(now) ?? undefined } }] },
|
||||||
);
|
} as any);
|
||||||
|
|
||||||
expect(visits).toHaveLength(expectedVisits);
|
expect(visits).toHaveLength(expectedVisits);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -128,7 +128,7 @@ describe('tagVisitsReducer', () => {
|
|||||||
|
|
||||||
const { visits } = reducer(prevState, {
|
const { visits } = reducer(prevState, {
|
||||||
type: CREATE_VISITS,
|
type: CREATE_VISITS,
|
||||||
createdVisits: [{ shortUrl, visit: { date: formatIsoDate(now) ?? undefined } }],
|
payload: { createdVisits: [{ shortUrl, visit: { date: formatIsoDate(now) ?? undefined } }] },
|
||||||
} as any);
|
} as any);
|
||||||
|
|
||||||
expect(visits).toHaveLength(expectedVisits);
|
expect(visits).toHaveLength(expectedVisits);
|
||||||
|
|||||||
@@ -9,9 +9,10 @@ describe('visitCreationReducer', () => {
|
|||||||
const visit = Mock.all<Visit>();
|
const visit = Mock.all<Visit>();
|
||||||
|
|
||||||
it('just returns the action with proper type', () => {
|
it('just returns the action with proper type', () => {
|
||||||
expect(createNewVisits([{ shortUrl, visit }])).toEqual(
|
expect(createNewVisits([{ shortUrl, visit }])).toEqual({
|
||||||
{ type: CREATE_VISITS, createdVisits: [{ shortUrl, visit }] },
|
type: CREATE_VISITS,
|
||||||
);
|
payload: { createdVisits: [{ shortUrl, visit }] },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -52,19 +52,21 @@ describe('visitsOverviewReducer', () => {
|
|||||||
state({ visitsCount: 100, orphanVisitsCount: providedOrphanVisitsCount }),
|
state({ visitsCount: 100, orphanVisitsCount: providedOrphanVisitsCount }),
|
||||||
{
|
{
|
||||||
type: CREATE_VISITS,
|
type: CREATE_VISITS,
|
||||||
createdVisits: [
|
payload: {
|
||||||
Mock.of<CreateVisit>({ visit: Mock.all<Visit>() }),
|
createdVisits: [
|
||||||
Mock.of<CreateVisit>({ visit: Mock.all<Visit>() }),
|
Mock.of<CreateVisit>({ visit: Mock.all<Visit>() }),
|
||||||
Mock.of<CreateVisit>({
|
Mock.of<CreateVisit>({ visit: Mock.all<Visit>() }),
|
||||||
visit: Mock.of<OrphanVisit>({ visitedUrl: '' }),
|
Mock.of<CreateVisit>({
|
||||||
}),
|
visit: Mock.of<OrphanVisit>({ visitedUrl: '' }),
|
||||||
Mock.of<CreateVisit>({
|
}),
|
||||||
visit: Mock.of<OrphanVisit>({ visitedUrl: '' }),
|
Mock.of<CreateVisit>({
|
||||||
}),
|
visit: Mock.of<OrphanVisit>({ visitedUrl: '' }),
|
||||||
Mock.of<CreateVisit>({
|
}),
|
||||||
visit: Mock.of<OrphanVisit>({ visitedUrl: '' }),
|
Mock.of<CreateVisit>({
|
||||||
}),
|
visit: Mock.of<OrphanVisit>({ visitedUrl: '' }),
|
||||||
],
|
}),
|
||||||
|
],
|
||||||
|
},
|
||||||
} as unknown as GetVisitsOverviewAction & CreateVisitsAction,
|
} as unknown as GetVisitsOverviewAction & CreateVisitsAction,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user