mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2025-01-11 02:37:22 +03:00
Deleted reducers for short URL tags and short URL meta
This commit is contained in:
parent
3ad0c4d009
commit
d703e5e182
11 changed files with 46 additions and 524 deletions
|
@ -1,12 +1,10 @@
|
||||||
import { MercureInfo } from '../mercure/reducers/mercureInfo';
|
import { MercureInfo } from '../mercure/reducers/mercureInfo';
|
||||||
import { SelectedServer, ServersMap } from '../servers/data';
|
import { SelectedServer, ServersMap } from '../servers/data';
|
||||||
import { Settings } from '../settings/reducers/settings';
|
import { Settings } from '../settings/reducers/settings';
|
||||||
import { ShortUrlMetaEdition } from '../short-urls/reducers/shortUrlMeta';
|
|
||||||
import { ShortUrlCreation } from '../short-urls/reducers/shortUrlCreation';
|
import { ShortUrlCreation } from '../short-urls/reducers/shortUrlCreation';
|
||||||
import { ShortUrlDeletion } from '../short-urls/reducers/shortUrlDeletion';
|
import { ShortUrlDeletion } from '../short-urls/reducers/shortUrlDeletion';
|
||||||
import { ShortUrlEdition } from '../short-urls/reducers/shortUrlEdition';
|
import { ShortUrlEdition } from '../short-urls/reducers/shortUrlEdition';
|
||||||
import { ShortUrlsListParams } from '../short-urls/reducers/shortUrlsListParams';
|
import { ShortUrlsListParams } from '../short-urls/reducers/shortUrlsListParams';
|
||||||
import { ShortUrlTags } from '../short-urls/reducers/shortUrlTags';
|
|
||||||
import { ShortUrlsList } from '../short-urls/reducers/shortUrlsList';
|
import { ShortUrlsList } from '../short-urls/reducers/shortUrlsList';
|
||||||
import { TagDeletion } from '../tags/reducers/tagDelete';
|
import { TagDeletion } from '../tags/reducers/tagDelete';
|
||||||
import { TagEdition } from '../tags/reducers/tagEdit';
|
import { TagEdition } from '../tags/reducers/tagEdit';
|
||||||
|
@ -25,8 +23,6 @@ export interface ShlinkState {
|
||||||
shortUrlsListParams: ShortUrlsListParams;
|
shortUrlsListParams: ShortUrlsListParams;
|
||||||
shortUrlCreationResult: ShortUrlCreation;
|
shortUrlCreationResult: ShortUrlCreation;
|
||||||
shortUrlDeletion: ShortUrlDeletion;
|
shortUrlDeletion: ShortUrlDeletion;
|
||||||
shortUrlTags: ShortUrlTags;
|
|
||||||
shortUrlMeta: ShortUrlMetaEdition;
|
|
||||||
shortUrlEdition: ShortUrlEdition;
|
shortUrlEdition: ShortUrlEdition;
|
||||||
shortUrlVisits: ShortUrlVisits;
|
shortUrlVisits: ShortUrlVisits;
|
||||||
tagVisits: TagVisits;
|
tagVisits: TagVisits;
|
||||||
|
|
|
@ -5,8 +5,6 @@ import shortUrlsListReducer from '../short-urls/reducers/shortUrlsList';
|
||||||
import shortUrlsListParamsReducer from '../short-urls/reducers/shortUrlsListParams';
|
import shortUrlsListParamsReducer from '../short-urls/reducers/shortUrlsListParams';
|
||||||
import shortUrlCreationReducer from '../short-urls/reducers/shortUrlCreation';
|
import shortUrlCreationReducer from '../short-urls/reducers/shortUrlCreation';
|
||||||
import shortUrlDeletionReducer from '../short-urls/reducers/shortUrlDeletion';
|
import shortUrlDeletionReducer from '../short-urls/reducers/shortUrlDeletion';
|
||||||
import shortUrlTagsReducer from '../short-urls/reducers/shortUrlTags';
|
|
||||||
import shortUrlMetaReducer from '../short-urls/reducers/shortUrlMeta';
|
|
||||||
import shortUrlEditionReducer from '../short-urls/reducers/shortUrlEdition';
|
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';
|
||||||
|
@ -28,8 +26,6 @@ export default combineReducers<ShlinkState>({
|
||||||
shortUrlsListParams: shortUrlsListParamsReducer,
|
shortUrlsListParams: shortUrlsListParamsReducer,
|
||||||
shortUrlCreationResult: shortUrlCreationReducer,
|
shortUrlCreationResult: shortUrlCreationReducer,
|
||||||
shortUrlDeletion: shortUrlDeletionReducer,
|
shortUrlDeletion: shortUrlDeletionReducer,
|
||||||
shortUrlTags: shortUrlTagsReducer,
|
|
||||||
shortUrlMeta: shortUrlMetaReducer,
|
|
||||||
shortUrlEdition: shortUrlEditionReducer,
|
shortUrlEdition: shortUrlEditionReducer,
|
||||||
shortUrlVisits: shortUrlVisitsReducer,
|
shortUrlVisits: shortUrlVisitsReducer,
|
||||||
tagVisits: tagVisitsReducer,
|
tagVisits: tagVisitsReducer,
|
||||||
|
|
|
@ -2,10 +2,11 @@ import { Action, Dispatch } from 'redux';
|
||||||
import { buildReducer } from '../../utils/helpers/redux';
|
import { buildReducer } from '../../utils/helpers/redux';
|
||||||
import { GetState } from '../../container/types';
|
import { GetState } from '../../container/types';
|
||||||
import { OptionalString } from '../../utils/utils';
|
import { OptionalString } from '../../utils/utils';
|
||||||
import { EditShortUrlData, ShortUrlIdentifier } from '../data';
|
import { EditShortUrlData, ShortUrl } from '../data';
|
||||||
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
||||||
import { ProblemDetailsError } from '../../api/types';
|
import { ProblemDetailsError } from '../../api/types';
|
||||||
import { parseApiError } from '../../api/utils';
|
import { parseApiError } from '../../api/utils';
|
||||||
|
import { supportsTagsInPatch } from '../../utils/helpers/features';
|
||||||
|
|
||||||
/* eslint-disable padding-line-between-statements */
|
/* eslint-disable padding-line-between-statements */
|
||||||
export const EDIT_SHORT_URL_START = 'shlink/shortUrlEdition/EDIT_SHORT_URL_START';
|
export const EDIT_SHORT_URL_START = 'shlink/shortUrlEdition/EDIT_SHORT_URL_START';
|
||||||
|
@ -14,15 +15,14 @@ export const SHORT_URL_EDITED = 'shlink/shortUrlEdition/SHORT_URL_EDITED';
|
||||||
/* eslint-enable padding-line-between-statements */
|
/* eslint-enable padding-line-between-statements */
|
||||||
|
|
||||||
export interface ShortUrlEdition {
|
export interface ShortUrlEdition {
|
||||||
shortCode: string | null;
|
shortUrl?: ShortUrl;
|
||||||
longUrl: string | null;
|
|
||||||
saving: boolean;
|
saving: boolean;
|
||||||
error: boolean;
|
error: boolean;
|
||||||
errorData?: ProblemDetailsError;
|
errorData?: ProblemDetailsError;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ShortUrlEditedAction extends Action<string>, ShortUrlIdentifier {
|
export interface ShortUrlEditedAction extends Action<string> {
|
||||||
longUrl: string;
|
shortUrl: ShortUrl;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ShortUrlEditionFailedAction extends Action<string> {
|
export interface ShortUrlEditionFailedAction extends Action<string> {
|
||||||
|
@ -30,8 +30,6 @@ export interface ShortUrlEditionFailedAction extends Action<string> {
|
||||||
}
|
}
|
||||||
|
|
||||||
const initialState: ShortUrlEdition = {
|
const initialState: ShortUrlEdition = {
|
||||||
shortCode: null,
|
|
||||||
longUrl: null,
|
|
||||||
saving: false,
|
saving: false,
|
||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
@ -39,7 +37,7 @@ const initialState: ShortUrlEdition = {
|
||||||
export default buildReducer<ShortUrlEdition, ShortUrlEditedAction & ShortUrlEditionFailedAction>({
|
export default buildReducer<ShortUrlEdition, ShortUrlEditedAction & ShortUrlEditionFailedAction>({
|
||||||
[EDIT_SHORT_URL_START]: (state) => ({ ...state, saving: true, error: false }),
|
[EDIT_SHORT_URL_START]: (state) => ({ ...state, saving: true, error: false }),
|
||||||
[EDIT_SHORT_URL_ERROR]: (state, { errorData }) => ({ ...state, saving: false, error: true, errorData }),
|
[EDIT_SHORT_URL_ERROR]: (state, { errorData }) => ({ ...state, saving: false, error: true, errorData }),
|
||||||
[SHORT_URL_EDITED]: (_, { shortCode, longUrl }) => ({ shortCode, longUrl, saving: false, error: false }),
|
[SHORT_URL_EDITED]: (_, { shortUrl }) => ({ shortUrl, saving: false, error: false }),
|
||||||
}, initialState);
|
}, initialState);
|
||||||
|
|
||||||
export const editShortUrl = (buildShlinkApiClient: ShlinkApiClientBuilder) => (
|
export const editShortUrl = (buildShlinkApiClient: ShlinkApiClientBuilder) => (
|
||||||
|
@ -49,13 +47,17 @@ export const editShortUrl = (buildShlinkApiClient: ShlinkApiClientBuilder) => (
|
||||||
) => async (dispatch: Dispatch, getState: GetState) => {
|
) => async (dispatch: Dispatch, getState: GetState) => {
|
||||||
dispatch({ type: EDIT_SHORT_URL_START });
|
dispatch({ type: EDIT_SHORT_URL_START });
|
||||||
|
|
||||||
// TODO Pass tags to the updateTags function if server version is lower than 2.6
|
const { selectedServer } = getState();
|
||||||
const { updateShortUrl } = buildShlinkApiClient(getState);
|
const sendTagsSeparately = !supportsTagsInPatch(selectedServer);
|
||||||
|
const { updateShortUrl, updateShortUrlTags } = buildShlinkApiClient(getState);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const { longUrl } = await updateShortUrl(shortCode, domain, data as any); // FIXME Parse dates
|
const [ shortUrl ] = await Promise.all([
|
||||||
|
updateShortUrl(shortCode, domain, data as any), // FIXME Parse dates
|
||||||
|
sendTagsSeparately && data.tags ? updateShortUrlTags(shortCode, domain, data.tags) : undefined,
|
||||||
|
]);
|
||||||
|
|
||||||
dispatch<ShortUrlEditedAction>({ shortCode, longUrl, domain, type: SHORT_URL_EDITED });
|
dispatch<ShortUrlEditedAction>({ shortUrl, type: SHORT_URL_EDITED });
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
dispatch<ShortUrlEditionFailedAction>({ type: EDIT_SHORT_URL_ERROR, errorData: parseApiError(e) });
|
dispatch<ShortUrlEditionFailedAction>({ type: EDIT_SHORT_URL_ERROR, errorData: parseApiError(e) });
|
||||||
|
|
||||||
|
|
|
@ -1,65 +0,0 @@
|
||||||
import { Dispatch, Action } from 'redux';
|
|
||||||
import { ShortUrlIdentifier, ShortUrlMeta } from '../data';
|
|
||||||
import { GetState } from '../../container/types';
|
|
||||||
import { buildActionCreator, buildReducer } from '../../utils/helpers/redux';
|
|
||||||
import { OptionalString } from '../../utils/utils';
|
|
||||||
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
|
||||||
import { ProblemDetailsError } from '../../api/types';
|
|
||||||
import { parseApiError } from '../../api/utils';
|
|
||||||
|
|
||||||
/* eslint-disable padding-line-between-statements */
|
|
||||||
export const EDIT_SHORT_URL_META_START = 'shlink/shortUrlMeta/EDIT_SHORT_URL_META_START';
|
|
||||||
export const EDIT_SHORT_URL_META_ERROR = 'shlink/shortUrlMeta/EDIT_SHORT_URL_META_ERROR';
|
|
||||||
export const SHORT_URL_META_EDITED = 'shlink/shortUrlMeta/SHORT_URL_META_EDITED';
|
|
||||||
export const RESET_EDIT_SHORT_URL_META = 'shlink/shortUrlMeta/RESET_EDIT_SHORT_URL_META';
|
|
||||||
/* eslint-enable padding-line-between-statements */
|
|
||||||
|
|
||||||
export interface ShortUrlMetaEdition {
|
|
||||||
shortCode: string | null;
|
|
||||||
meta: ShortUrlMeta;
|
|
||||||
saving: boolean;
|
|
||||||
error: boolean;
|
|
||||||
errorData?: ProblemDetailsError;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface ShortUrlMetaEditedAction extends Action<string>, ShortUrlIdentifier {
|
|
||||||
meta: ShortUrlMeta;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface ShortUrlMetaEditionFailedAction extends Action<string> {
|
|
||||||
errorData?: ProblemDetailsError;
|
|
||||||
}
|
|
||||||
|
|
||||||
const initialState: ShortUrlMetaEdition = {
|
|
||||||
shortCode: null,
|
|
||||||
meta: {},
|
|
||||||
saving: false,
|
|
||||||
error: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default buildReducer<ShortUrlMetaEdition, ShortUrlMetaEditedAction & ShortUrlMetaEditionFailedAction>({
|
|
||||||
[EDIT_SHORT_URL_META_START]: (state) => ({ ...state, saving: true, error: false }),
|
|
||||||
[EDIT_SHORT_URL_META_ERROR]: (state, { errorData }) => ({ ...state, saving: false, error: true, errorData }),
|
|
||||||
[SHORT_URL_META_EDITED]: (_, { shortCode, meta }) => ({ shortCode, meta, saving: false, error: false }),
|
|
||||||
[RESET_EDIT_SHORT_URL_META]: () => initialState,
|
|
||||||
}, initialState);
|
|
||||||
|
|
||||||
export const editShortUrlMeta = (buildShlinkApiClient: ShlinkApiClientBuilder) => (
|
|
||||||
shortCode: string,
|
|
||||||
domain: OptionalString,
|
|
||||||
meta: ShortUrlMeta,
|
|
||||||
) => async (dispatch: Dispatch, getState: GetState) => {
|
|
||||||
dispatch({ type: EDIT_SHORT_URL_META_START });
|
|
||||||
const { updateShortUrl } = buildShlinkApiClient(getState);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await updateShortUrl(shortCode, domain, meta);
|
|
||||||
dispatch<ShortUrlMetaEditedAction>({ shortCode, meta, domain, type: SHORT_URL_META_EDITED });
|
|
||||||
} catch (e) {
|
|
||||||
dispatch<ShortUrlMetaEditionFailedAction>({ type: EDIT_SHORT_URL_META_ERROR, errorData: parseApiError(e) });
|
|
||||||
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const resetShortUrlMeta = buildActionCreator(RESET_EDIT_SHORT_URL_META);
|
|
|
@ -1,74 +0,0 @@
|
||||||
import { Action, Dispatch } from 'redux';
|
|
||||||
import { prop } from 'ramda';
|
|
||||||
import { buildActionCreator, buildReducer } from '../../utils/helpers/redux';
|
|
||||||
import { GetState } from '../../container/types';
|
|
||||||
import { OptionalString } from '../../utils/utils';
|
|
||||||
import { ShortUrlIdentifier } from '../data';
|
|
||||||
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
|
||||||
import { ProblemDetailsError } from '../../api/types';
|
|
||||||
import { parseApiError } from '../../api/utils';
|
|
||||||
import { supportsTagsInPatch } from '../../utils/helpers/features';
|
|
||||||
|
|
||||||
/* eslint-disable padding-line-between-statements */
|
|
||||||
export const EDIT_SHORT_URL_TAGS_START = 'shlink/shortUrlTags/EDIT_SHORT_URL_TAGS_START';
|
|
||||||
export const EDIT_SHORT_URL_TAGS_ERROR = 'shlink/shortUrlTags/EDIT_SHORT_URL_TAGS_ERROR';
|
|
||||||
export const SHORT_URL_TAGS_EDITED = 'shlink/shortUrlTags/SHORT_URL_TAGS_EDITED';
|
|
||||||
export const RESET_EDIT_SHORT_URL_TAGS = 'shlink/shortUrlTags/RESET_EDIT_SHORT_URL_TAGS';
|
|
||||||
/* eslint-enable padding-line-between-statements */
|
|
||||||
|
|
||||||
export interface ShortUrlTags {
|
|
||||||
shortCode: string | null;
|
|
||||||
tags: string[];
|
|
||||||
saving: boolean;
|
|
||||||
error: boolean;
|
|
||||||
errorData?: ProblemDetailsError;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface EditShortUrlTagsAction extends Action<string>, ShortUrlIdentifier {
|
|
||||||
tags: string[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface EditShortUrlTagsFailedAction extends Action<string> {
|
|
||||||
errorData?: ProblemDetailsError;
|
|
||||||
}
|
|
||||||
|
|
||||||
const initialState: ShortUrlTags = {
|
|
||||||
shortCode: null,
|
|
||||||
tags: [],
|
|
||||||
saving: false,
|
|
||||||
error: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default buildReducer<ShortUrlTags, EditShortUrlTagsAction & EditShortUrlTagsFailedAction>({
|
|
||||||
[EDIT_SHORT_URL_TAGS_START]: (state) => ({ ...state, saving: true, error: false }),
|
|
||||||
[EDIT_SHORT_URL_TAGS_ERROR]: (state, { errorData }) => ({ ...state, saving: false, error: true, errorData }),
|
|
||||||
[SHORT_URL_TAGS_EDITED]: (_, { shortCode, tags }) => ({ shortCode, tags, saving: false, error: false }),
|
|
||||||
[RESET_EDIT_SHORT_URL_TAGS]: () => initialState,
|
|
||||||
}, initialState);
|
|
||||||
|
|
||||||
export const editShortUrlTags = (buildShlinkApiClient: ShlinkApiClientBuilder) => (
|
|
||||||
shortCode: string,
|
|
||||||
domain: OptionalString,
|
|
||||||
tags: string[],
|
|
||||||
) => async (dispatch: Dispatch, getState: GetState) => {
|
|
||||||
dispatch({ type: EDIT_SHORT_URL_TAGS_START });
|
|
||||||
const { selectedServer } = getState();
|
|
||||||
const tagsInPatch = supportsTagsInPatch(selectedServer);
|
|
||||||
const { updateShortUrlTags, updateShortUrl } = buildShlinkApiClient(getState);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const normalizedTags = await (
|
|
||||||
tagsInPatch
|
|
||||||
? updateShortUrl(shortCode, domain, { tags }).then(prop('tags'))
|
|
||||||
: updateShortUrlTags(shortCode, domain, tags)
|
|
||||||
);
|
|
||||||
|
|
||||||
dispatch<EditShortUrlTagsAction>({ tags: normalizedTags, shortCode, domain, type: SHORT_URL_TAGS_EDITED });
|
|
||||||
} catch (e) {
|
|
||||||
dispatch<EditShortUrlTagsFailedAction>({ type: EDIT_SHORT_URL_TAGS_ERROR, errorData: parseApiError(e) });
|
|
||||||
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const resetShortUrlsTags = buildActionCreator(RESET_EDIT_SHORT_URL_TAGS);
|
|
|
@ -2,15 +2,11 @@ import { assoc, assocPath, init, last, pipe, reject } from 'ramda';
|
||||||
import { Action, Dispatch } from 'redux';
|
import { Action, Dispatch } from 'redux';
|
||||||
import { shortUrlMatches } from '../helpers';
|
import { shortUrlMatches } from '../helpers';
|
||||||
import { CREATE_VISITS, CreateVisitsAction } from '../../visits/reducers/visitCreation';
|
import { CREATE_VISITS, CreateVisitsAction } from '../../visits/reducers/visitCreation';
|
||||||
import { ShortUrl, ShortUrlIdentifier } from '../data';
|
|
||||||
import { buildReducer } from '../../utils/helpers/redux';
|
import { buildReducer } from '../../utils/helpers/redux';
|
||||||
import { GetState } from '../../container/types';
|
import { GetState } from '../../container/types';
|
||||||
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
||||||
import { ShlinkShortUrlsResponse } from '../../api/types';
|
import { ShlinkShortUrlsResponse } from '../../api/types';
|
||||||
import { EditShortUrlTagsAction, SHORT_URL_TAGS_EDITED } from './shortUrlTags';
|
|
||||||
import { DeleteShortUrlAction, SHORT_URL_DELETED } from './shortUrlDeletion';
|
import { DeleteShortUrlAction, SHORT_URL_DELETED } from './shortUrlDeletion';
|
||||||
import { SHORT_URL_META_EDITED, ShortUrlMetaEditedAction } from './shortUrlMeta';
|
|
||||||
import { SHORT_URL_EDITED, ShortUrlEditedAction } from './shortUrlEdition';
|
|
||||||
import { ShortUrlsListParams } from './shortUrlsListParams';
|
import { ShortUrlsListParams } from './shortUrlsListParams';
|
||||||
import { CREATE_SHORT_URL, CreateShortUrlAction } from './shortUrlCreation';
|
import { CREATE_SHORT_URL, CreateShortUrlAction } from './shortUrlCreation';
|
||||||
|
|
||||||
|
@ -33,9 +29,6 @@ export interface ListShortUrlsAction extends Action<string> {
|
||||||
|
|
||||||
export type ListShortUrlsCombinedAction = (
|
export type ListShortUrlsCombinedAction = (
|
||||||
ListShortUrlsAction
|
ListShortUrlsAction
|
||||||
& EditShortUrlTagsAction
|
|
||||||
& ShortUrlEditedAction
|
|
||||||
& ShortUrlMetaEditedAction
|
|
||||||
& CreateVisitsAction
|
& CreateVisitsAction
|
||||||
& CreateShortUrlAction
|
& CreateShortUrlAction
|
||||||
& DeleteShortUrlAction
|
& DeleteShortUrlAction
|
||||||
|
@ -46,18 +39,6 @@ const initialState: ShortUrlsList = {
|
||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
const setPropFromActionOnMatchingShortUrl = <T extends ShortUrlIdentifier>(prop: keyof T) => (
|
|
||||||
state: ShortUrlsList,
|
|
||||||
{ shortCode, domain, [prop]: propValue }: T,
|
|
||||||
): ShortUrlsList => !state.shortUrls ? state : assocPath(
|
|
||||||
[ 'shortUrls', 'data' ],
|
|
||||||
state.shortUrls.data.map(
|
|
||||||
(shortUrl: ShortUrl) =>
|
|
||||||
shortUrlMatches(shortUrl, shortCode, domain) ? { ...shortUrl, [prop]: propValue } : shortUrl,
|
|
||||||
),
|
|
||||||
state,
|
|
||||||
);
|
|
||||||
|
|
||||||
export default buildReducer<ShortUrlsList, ListShortUrlsCombinedAction>({
|
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 }),
|
||||||
|
@ -74,9 +55,6 @@ export default buildReducer<ShortUrlsList, ListShortUrlsCombinedAction>({
|
||||||
state,
|
state,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
[SHORT_URL_TAGS_EDITED]: setPropFromActionOnMatchingShortUrl<EditShortUrlTagsAction>('tags'),
|
|
||||||
[SHORT_URL_META_EDITED]: setPropFromActionOnMatchingShortUrl<ShortUrlMetaEditedAction>('meta'),
|
|
||||||
[SHORT_URL_EDITED]: setPropFromActionOnMatchingShortUrl<ShortUrlEditedAction>('longUrl'),
|
|
||||||
[CREATE_VISITS]: (state, { createdVisits }) => assocPath(
|
[CREATE_VISITS]: (state, { createdVisits }) => assocPath(
|
||||||
[ 'shortUrls', 'data' ],
|
[ 'shortUrls', 'data' ],
|
||||||
state.shortUrls?.data?.map(
|
state.shortUrls?.data?.map(
|
||||||
|
|
|
@ -10,8 +10,6 @@ import CreateShortUrlResult from '../helpers/CreateShortUrlResult';
|
||||||
import { listShortUrls } from '../reducers/shortUrlsList';
|
import { listShortUrls } from '../reducers/shortUrlsList';
|
||||||
import { createShortUrl, resetCreateShortUrl } from '../reducers/shortUrlCreation';
|
import { createShortUrl, resetCreateShortUrl } from '../reducers/shortUrlCreation';
|
||||||
import { deleteShortUrl, resetDeleteShortUrl } from '../reducers/shortUrlDeletion';
|
import { deleteShortUrl, resetDeleteShortUrl } from '../reducers/shortUrlDeletion';
|
||||||
import { editShortUrlTags, resetShortUrlsTags } from '../reducers/shortUrlTags';
|
|
||||||
import { editShortUrlMeta, resetShortUrlMeta } from '../reducers/shortUrlMeta';
|
|
||||||
import { resetShortUrlParams } from '../reducers/shortUrlsListParams';
|
import { resetShortUrlParams } from '../reducers/shortUrlsListParams';
|
||||||
import { editShortUrl } from '../reducers/shortUrlEdition';
|
import { editShortUrl } from '../reducers/shortUrlEdition';
|
||||||
import { ConnectDecorator } from '../../container/types';
|
import { ConnectDecorator } from '../../container/types';
|
||||||
|
@ -61,9 +59,6 @@ const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
||||||
bottle.decorator('SearchBar', connect([ 'shortUrlsListParams' ], [ 'listShortUrls' ]));
|
bottle.decorator('SearchBar', connect([ 'shortUrlsListParams' ], [ 'listShortUrls' ]));
|
||||||
|
|
||||||
// Actions
|
// Actions
|
||||||
bottle.serviceFactory('editShortUrlTags', editShortUrlTags, 'buildShlinkApiClient');
|
|
||||||
bottle.serviceFactory('resetShortUrlsTags', () => resetShortUrlsTags);
|
|
||||||
|
|
||||||
bottle.serviceFactory('listShortUrls', listShortUrls, 'buildShlinkApiClient');
|
bottle.serviceFactory('listShortUrls', listShortUrls, 'buildShlinkApiClient');
|
||||||
bottle.serviceFactory('resetShortUrlParams', () => resetShortUrlParams);
|
bottle.serviceFactory('resetShortUrlParams', () => resetShortUrlParams);
|
||||||
|
|
||||||
|
@ -73,9 +68,6 @@ const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
||||||
bottle.serviceFactory('deleteShortUrl', deleteShortUrl, 'buildShlinkApiClient');
|
bottle.serviceFactory('deleteShortUrl', deleteShortUrl, 'buildShlinkApiClient');
|
||||||
bottle.serviceFactory('resetDeleteShortUrl', () => resetDeleteShortUrl);
|
bottle.serviceFactory('resetDeleteShortUrl', () => resetDeleteShortUrl);
|
||||||
|
|
||||||
bottle.serviceFactory('editShortUrlMeta', editShortUrlMeta, 'buildShlinkApiClient');
|
|
||||||
bottle.serviceFactory('resetShortUrlMeta', () => resetShortUrlMeta);
|
|
||||||
|
|
||||||
bottle.serviceFactory('getShortUrlDetail', getShortUrlDetail, 'buildShlinkApiClient');
|
bottle.serviceFactory('getShortUrlDetail', getShortUrlDetail, 'buildShlinkApiClient');
|
||||||
|
|
||||||
bottle.serviceFactory('editShortUrl', editShortUrl, 'buildShlinkApiClient');
|
bottle.serviceFactory('editShortUrl', editShortUrl, 'buildShlinkApiClient');
|
||||||
|
|
|
@ -7,16 +7,17 @@ import reducer, {
|
||||||
ShortUrlEditedAction,
|
ShortUrlEditedAction,
|
||||||
} from '../../../src/short-urls/reducers/shortUrlEdition';
|
} 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 { ReachableServer, SelectedServer } from '../../../src/servers/data';
|
||||||
|
|
||||||
describe('shortUrlEditionReducer', () => {
|
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 });
|
||||||
|
|
||||||
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: EDIT_SHORT_URL_START }))).toEqual({
|
||||||
longUrl: null,
|
|
||||||
shortCode: null,
|
|
||||||
saving: true,
|
saving: true,
|
||||||
error: false,
|
error: false,
|
||||||
});
|
});
|
||||||
|
@ -24,17 +25,14 @@ describe('shortUrlEditionReducer', () => {
|
||||||
|
|
||||||
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: EDIT_SHORT_URL_ERROR }))).toEqual({
|
||||||
longUrl: null,
|
|
||||||
shortCode: null,
|
|
||||||
saving: false,
|
saving: 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, longUrl, shortCode, domain: null })).toEqual({
|
expect(reducer(undefined, { type: SHORT_URL_EDITED, shortUrl })).toEqual({
|
||||||
longUrl,
|
shortUrl,
|
||||||
shortCode,
|
|
||||||
saving: false,
|
saving: false,
|
||||||
error: false,
|
error: false,
|
||||||
});
|
});
|
||||||
|
@ -42,31 +40,51 @@ describe('shortUrlEditionReducer', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('editShortUrl', () => {
|
describe('editShortUrl', () => {
|
||||||
const updateShortUrl = jest.fn().mockResolvedValue({ longUrl });
|
const updateShortUrl = jest.fn().mockResolvedValue(shortUrl);
|
||||||
const buildShlinkApiClient = jest.fn().mockReturnValue({ updateShortUrl });
|
const updateShortUrlTags = jest.fn().mockResolvedValue([]);
|
||||||
|
const buildShlinkApiClient = jest.fn().mockReturnValue({ updateShortUrl, updateShortUrlTags });
|
||||||
const dispatch = jest.fn();
|
const dispatch = jest.fn();
|
||||||
const getState = () => Mock.of<ShlinkState>();
|
const createGetState = (selectedServer: SelectedServer = null) => () => Mock.of<ShlinkState>({ selectedServer });
|
||||||
|
|
||||||
afterEach(jest.clearAllMocks);
|
afterEach(jest.clearAllMocks);
|
||||||
|
|
||||||
it.each([[ undefined ], [ null ], [ 'example.com' ]])('dispatches long 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, getState);
|
await editShortUrl(buildShlinkApiClient)(shortCode, domain, { 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, { type: EDIT_SHORT_URL_START });
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: SHORT_URL_EDITED, longUrl, shortCode, domain });
|
expect(dispatch).toHaveBeenNthCalledWith(2, { type: SHORT_URL_EDITED, shortUrl });
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it.each([
|
||||||
|
[ null, { tags: [ 'foo', 'bar' ] }, 1 ],
|
||||||
|
[ null, {}, 0 ],
|
||||||
|
[ Mock.of<ReachableServer>({ version: '2.6.0' }), {}, 0 ],
|
||||||
|
[ Mock.of<ReachableServer>({ version: '2.6.0' }), { tags: [ 'foo', 'bar' ] }, 0 ],
|
||||||
|
[ Mock.of<ReachableServer>({ version: '2.5.0' }), {}, 0 ],
|
||||||
|
[ Mock.of<ReachableServer>({ version: '2.5.0' }), { tags: [ 'foo', 'bar' ] }, 1 ],
|
||||||
|
])(
|
||||||
|
'sends tags separately when appropriate, based on selected server and the payload',
|
||||||
|
async (server, payload, expectedTagsCalls) => {
|
||||||
|
const getState = createGetState(server);
|
||||||
|
|
||||||
|
await editShortUrl(buildShlinkApiClient)(shortCode, null, payload)(dispatch, getState);
|
||||||
|
|
||||||
|
expect(updateShortUrl).toHaveBeenCalled();
|
||||||
|
expect(updateShortUrlTags).toHaveBeenCalledTimes(expectedTagsCalls);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
it('dispatches error on failure', async () => {
|
it('dispatches error on failure', async () => {
|
||||||
const error = new Error();
|
const error = new Error();
|
||||||
|
|
||||||
updateShortUrl.mockRejectedValue(error);
|
updateShortUrl.mockRejectedValue(error);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await editShortUrl(buildShlinkApiClient)(shortCode, undefined, { longUrl })(dispatch, getState);
|
await editShortUrl(buildShlinkApiClient)(shortCode, undefined, { longUrl })(dispatch, createGetState());
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
expect(e).toBe(error);
|
expect(e).toBe(error);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,100 +0,0 @@
|
||||||
import moment from 'moment';
|
|
||||||
import { Mock } from 'ts-mockery';
|
|
||||||
import reducer, {
|
|
||||||
EDIT_SHORT_URL_META_START,
|
|
||||||
EDIT_SHORT_URL_META_ERROR,
|
|
||||||
SHORT_URL_META_EDITED,
|
|
||||||
RESET_EDIT_SHORT_URL_META,
|
|
||||||
editShortUrlMeta,
|
|
||||||
resetShortUrlMeta,
|
|
||||||
} from '../../../src/short-urls/reducers/shortUrlMeta';
|
|
||||||
import { ShlinkState } from '../../../src/container/types';
|
|
||||||
|
|
||||||
describe('shortUrlMetaReducer', () => {
|
|
||||||
const meta = {
|
|
||||||
maxVisits: 50,
|
|
||||||
startDate: moment('2020-01-01').format(),
|
|
||||||
};
|
|
||||||
const shortCode = 'abc123';
|
|
||||||
|
|
||||||
describe('reducer', () => {
|
|
||||||
it('returns loading on EDIT_SHORT_URL_META_START', () => {
|
|
||||||
expect(reducer(undefined, { type: EDIT_SHORT_URL_META_START } as any)).toEqual({
|
|
||||||
meta: {},
|
|
||||||
shortCode: null,
|
|
||||||
saving: true,
|
|
||||||
error: false,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('returns error on EDIT_SHORT_URL_META_ERROR', () => {
|
|
||||||
expect(reducer(undefined, { type: EDIT_SHORT_URL_META_ERROR } as any)).toEqual({
|
|
||||||
meta: {},
|
|
||||||
shortCode: null,
|
|
||||||
saving: false,
|
|
||||||
error: true,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('returns provided tags and shortCode on SHORT_URL_META_EDITED', () => {
|
|
||||||
expect(reducer(undefined, { type: SHORT_URL_META_EDITED, meta, shortCode } as any)).toEqual({
|
|
||||||
meta,
|
|
||||||
shortCode,
|
|
||||||
saving: false,
|
|
||||||
error: false,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('goes back to initial state on RESET_EDIT_SHORT_URL_META', () => {
|
|
||||||
expect(reducer(undefined, { type: RESET_EDIT_SHORT_URL_META } as any)).toEqual({
|
|
||||||
meta: {},
|
|
||||||
shortCode: null,
|
|
||||||
saving: false,
|
|
||||||
error: false,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('editShortUrlMeta', () => {
|
|
||||||
const updateShortUrl = jest.fn().mockResolvedValue({});
|
|
||||||
const buildShlinkApiClient = jest.fn().mockReturnValue({ updateShortUrl });
|
|
||||||
const dispatch = jest.fn();
|
|
||||||
const getState = () => Mock.all<ShlinkState>();
|
|
||||||
|
|
||||||
afterEach(jest.clearAllMocks);
|
|
||||||
|
|
||||||
it.each([[ undefined ], [ null ], [ 'example.com' ]])('dispatches metadata on success', async (domain) => {
|
|
||||||
await editShortUrlMeta(buildShlinkApiClient)(shortCode, domain, meta)(dispatch, getState);
|
|
||||||
|
|
||||||
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrl).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrl).toHaveBeenCalledWith(shortCode, domain, meta);
|
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: EDIT_SHORT_URL_META_START });
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: SHORT_URL_META_EDITED, meta, shortCode, domain });
|
|
||||||
});
|
|
||||||
|
|
||||||
it('dispatches error on failure', async () => {
|
|
||||||
const error = new Error();
|
|
||||||
|
|
||||||
updateShortUrl.mockRejectedValue(error);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await editShortUrlMeta(buildShlinkApiClient)(shortCode, undefined, meta)(dispatch, getState);
|
|
||||||
} catch (e) {
|
|
||||||
expect(e).toBe(error);
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrl).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrl).toHaveBeenCalledWith(shortCode, undefined, meta);
|
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: EDIT_SHORT_URL_META_START });
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: EDIT_SHORT_URL_META_ERROR });
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('resetShortUrlMeta', () => {
|
|
||||||
it('creates expected action', () => expect(resetShortUrlMeta()).toEqual({ type: RESET_EDIT_SHORT_URL_META }));
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -1,131 +0,0 @@
|
||||||
import { Mock } from 'ts-mockery';
|
|
||||||
import reducer, {
|
|
||||||
EDIT_SHORT_URL_TAGS_ERROR,
|
|
||||||
EDIT_SHORT_URL_TAGS_START,
|
|
||||||
RESET_EDIT_SHORT_URL_TAGS,
|
|
||||||
resetShortUrlsTags,
|
|
||||||
SHORT_URL_TAGS_EDITED,
|
|
||||||
editShortUrlTags,
|
|
||||||
EditShortUrlTagsAction,
|
|
||||||
} from '../../../src/short-urls/reducers/shortUrlTags';
|
|
||||||
import { ShlinkState } from '../../../src/container/types';
|
|
||||||
import { ReachableServer, SelectedServer } from '../../../src/servers/data';
|
|
||||||
|
|
||||||
describe('shortUrlTagsReducer', () => {
|
|
||||||
const tags = [ 'foo', 'bar', 'baz' ];
|
|
||||||
const shortCode = 'abc123';
|
|
||||||
|
|
||||||
describe('reducer', () => {
|
|
||||||
const action = (type: string) => Mock.of<EditShortUrlTagsAction>({ type });
|
|
||||||
|
|
||||||
it('returns loading on EDIT_SHORT_URL_TAGS_START', () => {
|
|
||||||
expect(reducer(undefined, action(EDIT_SHORT_URL_TAGS_START))).toEqual({
|
|
||||||
tags: [],
|
|
||||||
shortCode: null,
|
|
||||||
saving: true,
|
|
||||||
error: false,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('returns error on EDIT_SHORT_URL_TAGS_ERROR', () => {
|
|
||||||
expect(reducer(undefined, action(EDIT_SHORT_URL_TAGS_ERROR))).toEqual({
|
|
||||||
tags: [],
|
|
||||||
shortCode: null,
|
|
||||||
saving: false,
|
|
||||||
error: true,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('returns provided tags and shortCode on SHORT_URL_TAGS_EDITED', () => {
|
|
||||||
expect(reducer(undefined, { type: SHORT_URL_TAGS_EDITED, tags, shortCode, domain: null })).toEqual({
|
|
||||||
tags,
|
|
||||||
shortCode,
|
|
||||||
saving: false,
|
|
||||||
error: false,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('goes back to initial state on RESET_EDIT_SHORT_URL_TAGS', () => {
|
|
||||||
expect(reducer(undefined, action(RESET_EDIT_SHORT_URL_TAGS))).toEqual({
|
|
||||||
tags: [],
|
|
||||||
shortCode: null,
|
|
||||||
saving: false,
|
|
||||||
error: false,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('resetShortUrlsTags', () => {
|
|
||||||
it('creates expected action', () => expect(resetShortUrlsTags()).toEqual({ type: RESET_EDIT_SHORT_URL_TAGS }));
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('editShortUrlTags', () => {
|
|
||||||
const updateShortUrlTags = jest.fn();
|
|
||||||
const updateShortUrl = jest.fn();
|
|
||||||
const buildShlinkApiClient = jest.fn().mockReturnValue({ updateShortUrlTags, updateShortUrl });
|
|
||||||
const dispatch = jest.fn();
|
|
||||||
const buildGetState = (selectedServer?: SelectedServer) => () => Mock.of<ShlinkState>({ selectedServer });
|
|
||||||
|
|
||||||
afterEach(jest.clearAllMocks);
|
|
||||||
|
|
||||||
it.each([[ undefined ], [ null ], [ 'example.com' ]])('dispatches normalized tags on success', async (domain) => {
|
|
||||||
const normalizedTags = [ 'bar', 'foo' ];
|
|
||||||
|
|
||||||
updateShortUrlTags.mockResolvedValue(normalizedTags);
|
|
||||||
|
|
||||||
await editShortUrlTags(buildShlinkApiClient)(shortCode, domain, tags)(dispatch, buildGetState());
|
|
||||||
|
|
||||||
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrlTags).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrlTags).toHaveBeenCalledWith(shortCode, domain, tags);
|
|
||||||
expect(updateShortUrl).not.toHaveBeenCalled();
|
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: EDIT_SHORT_URL_TAGS_START });
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(
|
|
||||||
2,
|
|
||||||
{ type: SHORT_URL_TAGS_EDITED, tags: normalizedTags, shortCode, domain },
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('calls updateShortUrl when server is version 2.6.0 or above', async () => {
|
|
||||||
const normalizedTags = [ 'bar', 'foo' ];
|
|
||||||
|
|
||||||
updateShortUrl.mockResolvedValue({ tags: normalizedTags });
|
|
||||||
|
|
||||||
await editShortUrlTags(buildShlinkApiClient)(shortCode, undefined, tags)(
|
|
||||||
dispatch,
|
|
||||||
buildGetState(Mock.of<ReachableServer>({ printableVersion: '', version: '2.6.0' })),
|
|
||||||
);
|
|
||||||
|
|
||||||
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrl).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrl).toHaveBeenCalledWith(shortCode, undefined, { tags });
|
|
||||||
expect(updateShortUrlTags).not.toHaveBeenCalled();
|
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: EDIT_SHORT_URL_TAGS_START });
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(
|
|
||||||
2,
|
|
||||||
{ type: SHORT_URL_TAGS_EDITED, tags: normalizedTags, shortCode },
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('dispatches error on failure', async () => {
|
|
||||||
const error = new Error();
|
|
||||||
|
|
||||||
updateShortUrlTags.mockRejectedValue(error);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await editShortUrlTags(buildShlinkApiClient)(shortCode, undefined, tags)(dispatch, buildGetState());
|
|
||||||
} catch (e) {
|
|
||||||
expect(e).toBe(error);
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(buildShlinkApiClient).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrlTags).toHaveBeenCalledTimes(1);
|
|
||||||
expect(updateShortUrlTags).toHaveBeenCalledWith(shortCode, undefined, tags);
|
|
||||||
expect(dispatch).toHaveBeenCalledTimes(2);
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(1, { type: EDIT_SHORT_URL_TAGS_START });
|
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(2, { type: EDIT_SHORT_URL_TAGS_ERROR });
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -5,15 +5,12 @@ import reducer, {
|
||||||
LIST_SHORT_URLS_START,
|
LIST_SHORT_URLS_START,
|
||||||
listShortUrls,
|
listShortUrls,
|
||||||
} from '../../../src/short-urls/reducers/shortUrlsList';
|
} from '../../../src/short-urls/reducers/shortUrlsList';
|
||||||
import { SHORT_URL_TAGS_EDITED } from '../../../src/short-urls/reducers/shortUrlTags';
|
|
||||||
import { SHORT_URL_DELETED } from '../../../src/short-urls/reducers/shortUrlDeletion';
|
import { SHORT_URL_DELETED } from '../../../src/short-urls/reducers/shortUrlDeletion';
|
||||||
import { SHORT_URL_META_EDITED } from '../../../src/short-urls/reducers/shortUrlMeta';
|
|
||||||
import { CREATE_VISITS } from '../../../src/visits/reducers/visitCreation';
|
import { CREATE_VISITS } from '../../../src/visits/reducers/visitCreation';
|
||||||
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 { ShlinkPaginator, ShlinkShortUrlsResponse } from '../../../src/api/types';
|
import { ShlinkPaginator, ShlinkShortUrlsResponse } from '../../../src/api/types';
|
||||||
import { CREATE_SHORT_URL } from '../../../src/short-urls/reducers/shortUrlCreation';
|
import { CREATE_SHORT_URL } from '../../../src/short-urls/reducers/shortUrlCreation';
|
||||||
import { SHORT_URL_EDITED } from '../../../src/short-urls/reducers/shortUrlEdition';
|
|
||||||
|
|
||||||
describe('shortUrlsListReducer', () => {
|
describe('shortUrlsListReducer', () => {
|
||||||
describe('reducer', () => {
|
describe('reducer', () => {
|
||||||
|
@ -36,66 +33,6 @@ describe('shortUrlsListReducer', () => {
|
||||||
error: true,
|
error: true,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('updates tags on matching URL on SHORT_URL_TAGS_EDITED', () => {
|
|
||||||
const shortCode = 'abc123';
|
|
||||||
const tags = [ 'foo', 'bar', 'baz' ];
|
|
||||||
const state = {
|
|
||||||
shortUrls: Mock.of<ShlinkShortUrlsResponse>({
|
|
||||||
data: [
|
|
||||||
Mock.of<ShortUrl>({ shortCode, tags: [] }),
|
|
||||||
Mock.of<ShortUrl>({ shortCode, tags: [], domain: 'example.com' }),
|
|
||||||
Mock.of<ShortUrl>({ shortCode: 'foo', tags: [] }),
|
|
||||||
],
|
|
||||||
}),
|
|
||||||
loading: false,
|
|
||||||
error: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
expect(reducer(state, { type: SHORT_URL_TAGS_EDITED, shortCode, tags, domain: null } as any)).toEqual({
|
|
||||||
shortUrls: {
|
|
||||||
data: [
|
|
||||||
{ shortCode, tags },
|
|
||||||
{ shortCode, tags: [], domain: 'example.com' },
|
|
||||||
{ shortCode: 'foo', tags: [] },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
loading: false,
|
|
||||||
error: false,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('updates meta on matching URL on SHORT_URL_META_EDITED', () => {
|
|
||||||
const shortCode = 'abc123';
|
|
||||||
const domain = 'example.com';
|
|
||||||
const meta = {
|
|
||||||
maxVisits: 5,
|
|
||||||
validSince: '2020-05-05',
|
|
||||||
};
|
|
||||||
const state = {
|
|
||||||
shortUrls: Mock.of<ShlinkShortUrlsResponse>({
|
|
||||||
data: [
|
|
||||||
Mock.of<ShortUrl>({ shortCode, meta: { maxVisits: 10 }, domain }),
|
|
||||||
Mock.of<ShortUrl>({ shortCode, meta: { maxVisits: 50 } }),
|
|
||||||
Mock.of<ShortUrl>({ shortCode: 'foo', meta: {} }),
|
|
||||||
],
|
|
||||||
}),
|
|
||||||
loading: false,
|
|
||||||
error: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
expect(reducer(state, { type: SHORT_URL_META_EDITED, shortCode, meta, domain } as any)).toEqual({
|
|
||||||
shortUrls: {
|
|
||||||
data: [
|
|
||||||
{ shortCode, meta, domain: 'example.com' },
|
|
||||||
{ shortCode, meta: { maxVisits: 50 } },
|
|
||||||
{ shortCode: 'foo', meta: {} },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
loading: false,
|
|
||||||
error: false,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('removes matching URL and reduces total on SHORT_URL_DELETED', () => {
|
it('removes matching URL and reduces total on SHORT_URL_DELETED', () => {
|
||||||
const shortCode = 'abc123';
|
const shortCode = 'abc123';
|
||||||
const state = {
|
const state = {
|
||||||
|
@ -123,33 +60,6 @@ describe('shortUrlsListReducer', () => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('updates edited short URL on SHORT_URL_EDITED', () => {
|
|
||||||
const shortCode = 'abc123';
|
|
||||||
const state = {
|
|
||||||
shortUrls: Mock.of<ShlinkShortUrlsResponse>({
|
|
||||||
data: [
|
|
||||||
Mock.of<ShortUrl>({ shortCode, longUrl: 'old' }),
|
|
||||||
Mock.of<ShortUrl>({ shortCode, domain: 'example.com', longUrl: 'foo' }),
|
|
||||||
Mock.of<ShortUrl>({ shortCode: 'foo', longUrl: 'bar' }),
|
|
||||||
],
|
|
||||||
}),
|
|
||||||
loading: false,
|
|
||||||
error: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
expect(reducer(state, { type: SHORT_URL_EDITED, shortCode, longUrl: 'newValue' } as any)).toEqual({
|
|
||||||
shortUrls: {
|
|
||||||
data: [
|
|
||||||
{ shortCode, longUrl: 'newValue' },
|
|
||||||
{ shortCode, longUrl: 'foo', domain: 'example.com' },
|
|
||||||
{ shortCode: 'foo', longUrl: 'bar' },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
loading: false,
|
|
||||||
error: false,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
const createNewShortUrlVisit = (visitsCount: number) => ({
|
const createNewShortUrlVisit = (visitsCount: number) => ({
|
||||||
shortUrl: { shortCode: 'abc123', visitsCount },
|
shortUrl: { shortCode: 'abc123', visitsCount },
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in a new issue