mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2025-01-09 09:47:28 +03:00
Merge pull request #571 from acelaya-forks/feature/non-orphan-visits
Feature/non orphan visits
This commit is contained in:
commit
071eaddfd1
26 changed files with 640 additions and 74 deletions
|
@ -7,6 +7,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
### Added
|
### Added
|
||||||
* [#558](https://github.com/shlinkio/shlink-web-client/pull/558) Added dark text for tags where the generated background is too light, improving its legibility.
|
* [#558](https://github.com/shlinkio/shlink-web-client/pull/558) Added dark text for tags where the generated background is too light, improving its legibility.
|
||||||
|
* [#570](https://github.com/shlinkio/shlink-web-client/pull/570) Added new section to load non-orphan visits all together.
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
* [#567](https://github.com/shlinkio/shlink-web-client/pull/567) Improved Shlink 3.0.0 compatibility by checking the `INVALID_SHORT_URL_DELETION` error code when deleting short URLs.
|
* [#567](https://github.com/shlinkio/shlink-web-client/pull/567) Improved Shlink 3.0.0 compatibility by checking the `INVALID_SHORT_URL_DELETION` error code when deleting short URLs.
|
||||||
|
|
|
@ -60,6 +60,10 @@ export default class ShlinkApiClient {
|
||||||
this.performRequest<{ visits: ShlinkVisits }>('/visits/orphan', 'GET', query)
|
this.performRequest<{ visits: ShlinkVisits }>('/visits/orphan', 'GET', query)
|
||||||
.then(({ data }) => data.visits);
|
.then(({ data }) => data.visits);
|
||||||
|
|
||||||
|
public readonly getNonOrphanVisits = async (query?: Omit<ShlinkVisitsParams, 'domain'>): Promise<ShlinkVisits> =>
|
||||||
|
this.performRequest<{ visits: ShlinkVisits }>('/visits/non-orphan', 'GET', query)
|
||||||
|
.then(({ data }) => data.visits);
|
||||||
|
|
||||||
public readonly getVisitsOverview = async (): Promise<ShlinkVisitsOverview> =>
|
public readonly getVisitsOverview = async (): Promise<ShlinkVisitsOverview> =>
|
||||||
this.performRequest<{ visits: ShlinkVisitsOverview }>('/visits', 'GET')
|
this.performRequest<{ visits: ShlinkVisitsOverview }>('/visits', 'GET')
|
||||||
.then(({ data }) => data.visits);
|
.then(({ data }) => data.visits);
|
||||||
|
|
|
@ -5,7 +5,7 @@ import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import { withSelectedServer } from '../servers/helpers/withSelectedServer';
|
import { withSelectedServer } from '../servers/helpers/withSelectedServer';
|
||||||
import { useSwipeable, useToggle } from '../utils/helpers/hooks';
|
import { useSwipeable, useToggle } from '../utils/helpers/hooks';
|
||||||
import { supportsDomainRedirects, supportsOrphanVisits } from '../utils/helpers/features';
|
import { supportsDomainRedirects, supportsNonOrphanVisits, supportsOrphanVisits } from '../utils/helpers/features';
|
||||||
import { isReachableServer } from '../servers/data';
|
import { isReachableServer } from '../servers/data';
|
||||||
import NotFound from './NotFound';
|
import NotFound from './NotFound';
|
||||||
import { AsideMenuProps } from './AsideMenu';
|
import { AsideMenuProps } from './AsideMenu';
|
||||||
|
@ -19,6 +19,7 @@ const MenuLayout = (
|
||||||
ShortUrlVisits: FC,
|
ShortUrlVisits: FC,
|
||||||
TagVisits: FC,
|
TagVisits: FC,
|
||||||
OrphanVisits: FC,
|
OrphanVisits: FC,
|
||||||
|
NonOrphanVisits: FC,
|
||||||
ServerError: FC,
|
ServerError: FC,
|
||||||
Overview: FC,
|
Overview: FC,
|
||||||
EditShortUrl: FC,
|
EditShortUrl: FC,
|
||||||
|
@ -33,6 +34,7 @@ const MenuLayout = (
|
||||||
}
|
}
|
||||||
|
|
||||||
const addOrphanVisitsRoute = supportsOrphanVisits(selectedServer);
|
const addOrphanVisitsRoute = supportsOrphanVisits(selectedServer);
|
||||||
|
const addNonOrphanVisitsRoute = supportsNonOrphanVisits(selectedServer);
|
||||||
const addManageDomainsRoute = supportsDomainRedirects(selectedServer);
|
const addManageDomainsRoute = supportsDomainRedirects(selectedServer);
|
||||||
const burgerClasses = classNames('menu-layout__burger-icon', { 'menu-layout__burger-icon--active': sidebarVisible });
|
const burgerClasses = classNames('menu-layout__burger-icon', { 'menu-layout__burger-icon--active': sidebarVisible });
|
||||||
const swipeableProps = useSwipeable(showSidebar, hideSidebar);
|
const swipeableProps = useSwipeable(showSidebar, hideSidebar);
|
||||||
|
@ -55,6 +57,7 @@ const MenuLayout = (
|
||||||
<Route path="/server/:serverId/short-code/:shortCode/edit" component={EditShortUrl} />
|
<Route path="/server/:serverId/short-code/:shortCode/edit" component={EditShortUrl} />
|
||||||
<Route path="/server/:serverId/tag/:tag/visits" component={TagVisits} />
|
<Route path="/server/:serverId/tag/:tag/visits" component={TagVisits} />
|
||||||
{addOrphanVisitsRoute && <Route path="/server/:serverId/orphan-visits" component={OrphanVisits} />}
|
{addOrphanVisitsRoute && <Route path="/server/:serverId/orphan-visits" component={OrphanVisits} />}
|
||||||
|
{addNonOrphanVisitsRoute && <Route path="/server/:serverId/non-orphan-visits" component={NonOrphanVisits} />}
|
||||||
<Route exact path="/server/:serverId/manage-tags" component={TagsList} />
|
<Route exact path="/server/:serverId/manage-tags" component={TagsList} />
|
||||||
{addManageDomainsRoute && <Route exact path="/server/:serverId/manage-domains" component={ManageDomains} />}
|
{addManageDomainsRoute && <Route exact path="/server/:serverId/manage-domains" component={ManageDomains} />}
|
||||||
<Route
|
<Route
|
||||||
|
|
|
@ -41,6 +41,7 @@ const provideServices = (bottle: Bottle, connect: ConnectDecorator, withRouter:
|
||||||
'ShortUrlVisits',
|
'ShortUrlVisits',
|
||||||
'TagVisits',
|
'TagVisits',
|
||||||
'OrphanVisits',
|
'OrphanVisits',
|
||||||
|
'NonOrphanVisits',
|
||||||
'ServerError',
|
'ServerError',
|
||||||
'Overview',
|
'Overview',
|
||||||
'EditShortUrl',
|
'EditShortUrl',
|
||||||
|
|
|
@ -25,6 +25,7 @@ export interface ShlinkState {
|
||||||
shortUrlVisits: ShortUrlVisits;
|
shortUrlVisits: ShortUrlVisits;
|
||||||
tagVisits: TagVisits;
|
tagVisits: TagVisits;
|
||||||
orphanVisits: VisitsInfo;
|
orphanVisits: VisitsInfo;
|
||||||
|
nonOrphanVisits: VisitsInfo;
|
||||||
shortUrlDetail: ShortUrlDetail;
|
shortUrlDetail: ShortUrlDetail;
|
||||||
tagsList: TagsList;
|
tagsList: TagsList;
|
||||||
tagDelete: TagDeletion;
|
tagDelete: TagDeletion;
|
||||||
|
|
|
@ -8,6 +8,7 @@ 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 orphanVisitsReducer from '../visits/reducers/orphanVisits';
|
import orphanVisitsReducer from '../visits/reducers/orphanVisits';
|
||||||
|
import nonOrphanVisitsReducer from '../visits/reducers/nonOrphanVisits';
|
||||||
import shortUrlDetailReducer from '../short-urls/reducers/shortUrlDetail';
|
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';
|
||||||
|
@ -29,6 +30,7 @@ export default combineReducers<ShlinkState>({
|
||||||
shortUrlVisits: shortUrlVisitsReducer,
|
shortUrlVisits: shortUrlVisitsReducer,
|
||||||
tagVisits: tagVisitsReducer,
|
tagVisits: tagVisitsReducer,
|
||||||
orphanVisits: orphanVisitsReducer,
|
orphanVisits: orphanVisitsReducer,
|
||||||
|
nonOrphanVisits: nonOrphanVisitsReducer,
|
||||||
shortUrlDetail: shortUrlDetailReducer,
|
shortUrlDetail: shortUrlDetailReducer,
|
||||||
tagsList: tagsListReducer,
|
tagsList: tagsListReducer,
|
||||||
tagDelete: tagDeleteReducer,
|
tagDelete: tagDeleteReducer,
|
||||||
|
|
|
@ -1,13 +0,0 @@
|
||||||
@import '../utils/base';
|
|
||||||
|
|
||||||
.overview__card.overview__card {
|
|
||||||
text-align: center;
|
|
||||||
border-top: 3px solid var(--brand-color);
|
|
||||||
color: inherit;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.overview__card-title {
|
|
||||||
text-transform: uppercase;
|
|
||||||
color: $textPlaceholder;
|
|
||||||
}
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { FC, useEffect } from 'react';
|
import { FC, useEffect } from 'react';
|
||||||
import { Card, CardBody, CardHeader, CardText, CardTitle, Row } from 'reactstrap';
|
import { Card, CardBody, CardHeader, Row } from 'reactstrap';
|
||||||
import { Link, useHistory } from 'react-router-dom';
|
import { Link, useHistory } from 'react-router-dom';
|
||||||
import { ITEMS_IN_OVERVIEW_PAGE, ShortUrlsList as ShortUrlsListState } from '../short-urls/reducers/shortUrlsList';
|
import { ITEMS_IN_OVERVIEW_PAGE, ShortUrlsList as ShortUrlsListState } from '../short-urls/reducers/shortUrlsList';
|
||||||
import { prettify } from '../utils/helpers/numbers';
|
import { prettify } from '../utils/helpers/numbers';
|
||||||
|
@ -11,8 +11,9 @@ import { VisitsOverview } from '../visits/reducers/visitsOverview';
|
||||||
import { Versions } from '../utils/helpers/version';
|
import { Versions } from '../utils/helpers/version';
|
||||||
import { Topics } from '../mercure/helpers/Topics';
|
import { Topics } from '../mercure/helpers/Topics';
|
||||||
import { ShlinkShortUrlsListParams } from '../api/types';
|
import { ShlinkShortUrlsListParams } from '../api/types';
|
||||||
|
import { supportsNonOrphanVisits, supportsOrphanVisits } from '../utils/helpers/features';
|
||||||
import { getServerId, SelectedServer } from './data';
|
import { getServerId, SelectedServer } from './data';
|
||||||
import './Overview.scss';
|
import { HighlightCard } from './helpers/HighlightCard';
|
||||||
|
|
||||||
interface OverviewConnectProps {
|
interface OverviewConnectProps {
|
||||||
shortUrlsList: ShortUrlsListState;
|
shortUrlsList: ShortUrlsListState;
|
||||||
|
@ -41,6 +42,8 @@ export const Overview = (
|
||||||
const { loading: loadingTags } = tagsList;
|
const { loading: loadingTags } = tagsList;
|
||||||
const { loading: loadingVisits, visitsCount, orphanVisitsCount } = visitsOverview;
|
const { loading: loadingVisits, visitsCount, orphanVisitsCount } = visitsOverview;
|
||||||
const serverId = getServerId(selectedServer);
|
const serverId = getServerId(selectedServer);
|
||||||
|
const linkToOrphanVisits = supportsOrphanVisits(selectedServer);
|
||||||
|
const linkToNonOrphanVisits = supportsNonOrphanVisits(selectedServer);
|
||||||
const history = useHistory();
|
const history = useHistory();
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -52,40 +55,33 @@ export const Overview = (
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Row>
|
<Row>
|
||||||
<div className="col-md-6 col-xl-3">
|
<div className="col-lg-6 col-xl-3 mb-3">
|
||||||
<Card className="overview__card mb-3" body>
|
<HighlightCard title="Visits" link={linkToNonOrphanVisits && `/server/${serverId}/non-orphan-visits`}>
|
||||||
<CardTitle tag="h5" className="overview__card-title">Visits</CardTitle>
|
{loadingVisits ? 'Loading...' : prettify(visitsCount)}
|
||||||
<CardText tag="h2">{loadingVisits ? 'Loading...' : prettify(visitsCount)}</CardText>
|
</HighlightCard>
|
||||||
</Card>
|
|
||||||
</div>
|
</div>
|
||||||
<div className="col-md-6 col-xl-3">
|
<div className="col-lg-6 col-xl-3 mb-3">
|
||||||
<Card className="overview__card mb-3" body tag={Link} to={`/server/${serverId}/orphan-visits`}>
|
<HighlightCard title="Orphan visits" link={linkToOrphanVisits && `/server/${serverId}/orphan-visits`}>
|
||||||
<CardTitle tag="h5" className="overview__card-title">Orphan visits</CardTitle>
|
|
||||||
<CardText tag="h2">
|
|
||||||
<ForServerVersion minVersion="2.6.0">
|
<ForServerVersion minVersion="2.6.0">
|
||||||
{loadingVisits ? 'Loading...' : prettify(orphanVisitsCount ?? 0)}
|
{loadingVisits ? 'Loading...' : prettify(orphanVisitsCount ?? 0)}
|
||||||
</ForServerVersion>
|
</ForServerVersion>
|
||||||
<ForServerVersion maxVersion="2.5.*">
|
<ForServerVersion maxVersion="2.5.*">
|
||||||
<small className="text-muted"><i>Shlink 2.6 is needed</i></small>
|
<small className="text-muted"><i>Shlink 2.6 is needed</i></small>
|
||||||
</ForServerVersion>
|
</ForServerVersion>
|
||||||
</CardText>
|
</HighlightCard>
|
||||||
</Card>
|
|
||||||
</div>
|
</div>
|
||||||
<div className="col-md-6 col-xl-3">
|
<div className="col-lg-6 col-xl-3 mb-3">
|
||||||
<Card className="overview__card mb-3" body tag={Link} to={`/server/${serverId}/list-short-urls/1`}>
|
<HighlightCard title="Short URLs" link={`/server/${serverId}/list-short-urls/1`}>
|
||||||
<CardTitle tag="h5" className="overview__card-title">Short URLs</CardTitle>
|
|
||||||
<CardText tag="h2">
|
|
||||||
{loading ? 'Loading...' : prettify(shortUrls?.pagination.totalItems ?? 0)}
|
{loading ? 'Loading...' : prettify(shortUrls?.pagination.totalItems ?? 0)}
|
||||||
</CardText>
|
</HighlightCard>
|
||||||
</Card>
|
|
||||||
</div>
|
</div>
|
||||||
<div className="col-md-6 col-xl-3">
|
<div className="col-lg-6 col-xl-3 mb-3">
|
||||||
<Card className="overview__card mb-3" body tag={Link} to={`/server/${serverId}/manage-tags`}>
|
<HighlightCard title="Tags" link={`/server/${serverId}/manage-tags`}>
|
||||||
<CardTitle tag="h5" className="overview__card-title">Tags</CardTitle>
|
{loadingTags ? 'Loading...' : prettify(tagsList.tags.length)}
|
||||||
<CardText tag="h2">{loadingTags ? 'Loading...' : prettify(tagsList.tags.length)}</CardText>
|
</HighlightCard>
|
||||||
</Card>
|
|
||||||
</div>
|
</div>
|
||||||
</Row>
|
</Row>
|
||||||
|
|
||||||
<Card className="mb-3">
|
<Card className="mb-3">
|
||||||
<CardHeader>
|
<CardHeader>
|
||||||
<span className="d-sm-none">Create a short URL</span>
|
<span className="d-sm-none">Create a short URL</span>
|
||||||
|
|
21
src/servers/helpers/HighlightCard.scss
Normal file
21
src/servers/helpers/HighlightCard.scss
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
@import '../../utils/base';
|
||||||
|
|
||||||
|
.highlight-card.highlight-card {
|
||||||
|
text-align: center;
|
||||||
|
border-top: 3px solid var(--brand-color);
|
||||||
|
color: inherit;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.highlight-card__link-icon {
|
||||||
|
position: absolute;
|
||||||
|
right: 5px;
|
||||||
|
bottom: 5px;
|
||||||
|
opacity: 0.1;
|
||||||
|
transform: rotate(-45deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.highlight-card__title {
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: $textPlaceholder;
|
||||||
|
}
|
21
src/servers/helpers/HighlightCard.tsx
Normal file
21
src/servers/helpers/HighlightCard.tsx
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
import { FC } from 'react';
|
||||||
|
import { Card, CardText, CardTitle } from 'reactstrap';
|
||||||
|
import { Link } from 'react-router-dom';
|
||||||
|
import { faArrowAltCircleRight as linkIcon } from '@fortawesome/free-regular-svg-icons';
|
||||||
|
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
||||||
|
import './HighlightCard.scss';
|
||||||
|
|
||||||
|
export interface HighlightCardProps {
|
||||||
|
title: string;
|
||||||
|
link?: string | false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const buildExtraProps = (link?: string | false) => !link ? {} : { tag: Link, to: link };
|
||||||
|
|
||||||
|
export const HighlightCard: FC<HighlightCardProps> = ({ children, title, link }) => (
|
||||||
|
<Card className="highlight-card" body {...buildExtraProps(link)}>
|
||||||
|
{link && <FontAwesomeIcon size="3x" className="highlight-card__link-icon" icon={linkIcon} />}
|
||||||
|
<CardTitle tag="h5" className="highlight-card__title">{title}</CardTitle>
|
||||||
|
<CardText tag="h2">{children}</CardText>
|
||||||
|
</Card>
|
||||||
|
);
|
|
@ -25,3 +25,5 @@ export const supportsDomainRedirects = supportsQrErrorCorrection;
|
||||||
export const supportsForwardQuery = serverMatchesVersions({ minVersion: '2.9.0' });
|
export const supportsForwardQuery = serverMatchesVersions({ minVersion: '2.9.0' });
|
||||||
|
|
||||||
export const supportsDefaultDomainRedirectsEdition = serverMatchesVersions({ minVersion: '2.10.0' });
|
export const supportsDefaultDomainRedirectsEdition = serverMatchesVersions({ minVersion: '2.10.0' });
|
||||||
|
|
||||||
|
export const supportsNonOrphanVisits = serverMatchesVersions({ minVersion: '3.0.0' });
|
||||||
|
|
44
src/visits/NonOrphanVisits.tsx
Normal file
44
src/visits/NonOrphanVisits.tsx
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
import { RouteComponentProps } from 'react-router';
|
||||||
|
import { boundToMercureHub } from '../mercure/helpers/boundToMercureHub';
|
||||||
|
import { ShlinkVisitsParams } from '../api/types';
|
||||||
|
import { Topics } from '../mercure/helpers/Topics';
|
||||||
|
import VisitsStats from './VisitsStats';
|
||||||
|
import { NormalizedVisit, VisitsInfo, VisitsParams } from './types';
|
||||||
|
import { VisitsExporter } from './services/VisitsExporter';
|
||||||
|
import { CommonVisitsProps } from './types/CommonVisitsProps';
|
||||||
|
import { toApiParams } from './types/helpers';
|
||||||
|
import { NonOrphanVisitsHeader } from './NonOrphanVisitsHeader';
|
||||||
|
|
||||||
|
export interface NonOrphanVisitsProps extends CommonVisitsProps, RouteComponentProps {
|
||||||
|
getNonOrphanVisits: (params?: ShlinkVisitsParams, doIntervalFallback?: boolean) => void;
|
||||||
|
nonOrphanVisits: VisitsInfo;
|
||||||
|
cancelGetNonOrphanVisits: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const NonOrphanVisits = ({ exportVisits }: VisitsExporter) => boundToMercureHub(({
|
||||||
|
history: { goBack },
|
||||||
|
match: { url },
|
||||||
|
getNonOrphanVisits,
|
||||||
|
nonOrphanVisits,
|
||||||
|
cancelGetNonOrphanVisits,
|
||||||
|
settings,
|
||||||
|
selectedServer,
|
||||||
|
}: NonOrphanVisitsProps) => {
|
||||||
|
const exportCsv = (visits: NormalizedVisit[]) => exportVisits('non_orphan_visits.csv', visits);
|
||||||
|
const loadVisits = (params: VisitsParams, doIntervalFallback?: boolean) =>
|
||||||
|
getNonOrphanVisits(toApiParams(params), doIntervalFallback);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<VisitsStats
|
||||||
|
getVisits={loadVisits}
|
||||||
|
cancelGetVisits={cancelGetNonOrphanVisits}
|
||||||
|
visitsInfo={nonOrphanVisits}
|
||||||
|
baseUrl={url}
|
||||||
|
settings={settings}
|
||||||
|
exportCsv={exportCsv}
|
||||||
|
selectedServer={selectedServer}
|
||||||
|
>
|
||||||
|
<NonOrphanVisitsHeader nonOrphanVisits={nonOrphanVisits} goBack={goBack} />
|
||||||
|
</VisitsStats>
|
||||||
|
);
|
||||||
|
}, () => [ Topics.visits ]);
|
14
src/visits/NonOrphanVisitsHeader.tsx
Normal file
14
src/visits/NonOrphanVisitsHeader.tsx
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
import VisitsHeader from './VisitsHeader';
|
||||||
|
import { VisitsInfo } from './types';
|
||||||
|
import './ShortUrlVisitsHeader.scss';
|
||||||
|
|
||||||
|
interface NonOrphanVisitsHeaderProps {
|
||||||
|
nonOrphanVisits: VisitsInfo;
|
||||||
|
goBack: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const NonOrphanVisitsHeader = ({ nonOrphanVisits, goBack }: NonOrphanVisitsHeaderProps) => {
|
||||||
|
const { visits } = nonOrphanVisits;
|
||||||
|
|
||||||
|
return <VisitsHeader title="Non-orphan visits" goBack={goBack} visits={visits} />;
|
||||||
|
};
|
88
src/visits/reducers/nonOrphanVisits.ts
Normal file
88
src/visits/reducers/nonOrphanVisits.ts
Normal file
|
@ -0,0 +1,88 @@
|
||||||
|
import { Action, Dispatch } from 'redux';
|
||||||
|
import {
|
||||||
|
Visit,
|
||||||
|
VisitsFallbackIntervalAction,
|
||||||
|
VisitsInfo,
|
||||||
|
VisitsLoadProgressChangedAction,
|
||||||
|
} from '../types';
|
||||||
|
import { buildActionCreator, buildReducer } from '../../utils/helpers/redux';
|
||||||
|
import { ShlinkApiClientBuilder } from '../../api/services/ShlinkApiClientBuilder';
|
||||||
|
import { GetState } from '../../container/types';
|
||||||
|
import { ShlinkVisitsParams } from '../../api/types';
|
||||||
|
import { ApiErrorAction } from '../../api/types/actions';
|
||||||
|
import { isBetween } from '../../utils/helpers/date';
|
||||||
|
import { getVisitsWithLoader, lastVisitLoaderForLoader } from './common';
|
||||||
|
import { CREATE_VISITS, CreateVisitsAction } from './visitCreation';
|
||||||
|
|
||||||
|
/* eslint-disable padding-line-between-statements */
|
||||||
|
export const GET_NON_ORPHAN_VISITS_START = 'shlink/orphanVisits/GET_NON_ORPHAN_VISITS_START';
|
||||||
|
export const GET_NON_ORPHAN_VISITS_ERROR = 'shlink/orphanVisits/GET_NON_ORPHAN_VISITS_ERROR';
|
||||||
|
export const GET_NON_ORPHAN_VISITS = 'shlink/orphanVisits/GET_NON_ORPHAN_VISITS';
|
||||||
|
export const GET_NON_ORPHAN_VISITS_LARGE = 'shlink/orphanVisits/GET_NON_ORPHAN_VISITS_LARGE';
|
||||||
|
export const GET_NON_ORPHAN_VISITS_CANCEL = 'shlink/orphanVisits/GET_NON_ORPHAN_VISITS_CANCEL';
|
||||||
|
export const GET_NON_ORPHAN_VISITS_PROGRESS_CHANGED = 'shlink/orphanVisits/GET_NON_ORPHAN_VISITS_PROGRESS_CHANGED';
|
||||||
|
export const GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL = 'shlink/orphanVisits/GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL';
|
||||||
|
/* eslint-enable padding-line-between-statements */
|
||||||
|
|
||||||
|
export interface NonOrphanVisitsAction extends Action<string> {
|
||||||
|
visits: Visit[];
|
||||||
|
query?: ShlinkVisitsParams;
|
||||||
|
}
|
||||||
|
|
||||||
|
type NonOrphanVisitsCombinedAction = NonOrphanVisitsAction
|
||||||
|
& VisitsLoadProgressChangedAction
|
||||||
|
& VisitsFallbackIntervalAction
|
||||||
|
& CreateVisitsAction
|
||||||
|
& ApiErrorAction;
|
||||||
|
|
||||||
|
const initialState: VisitsInfo = {
|
||||||
|
visits: [],
|
||||||
|
loading: false,
|
||||||
|
loadingLarge: false,
|
||||||
|
error: false,
|
||||||
|
cancelLoad: false,
|
||||||
|
progress: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default buildReducer<VisitsInfo, NonOrphanVisitsCombinedAction>({
|
||||||
|
[GET_NON_ORPHAN_VISITS_START]: () => ({ ...initialState, loading: true }),
|
||||||
|
[GET_NON_ORPHAN_VISITS_ERROR]: (_, { errorData }) => ({ ...initialState, error: true, errorData }),
|
||||||
|
[GET_NON_ORPHAN_VISITS]: (state, { visits, query }) => ({ ...state, visits, query, loading: false, error: false }),
|
||||||
|
[GET_NON_ORPHAN_VISITS_LARGE]: (state) => ({ ...state, loadingLarge: true }),
|
||||||
|
[GET_NON_ORPHAN_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
||||||
|
[GET_NON_ORPHAN_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
||||||
|
[GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL]: (state, { fallbackInterval }) => ({ ...state, fallbackInterval }),
|
||||||
|
[CREATE_VISITS]: (state, { createdVisits }) => {
|
||||||
|
const { visits, query = {} } = state;
|
||||||
|
const { startDate, endDate } = query;
|
||||||
|
const newVisits = createdVisits
|
||||||
|
.filter(({ visit }) => isBetween(visit.date, startDate, endDate))
|
||||||
|
.map(({ visit }) => visit);
|
||||||
|
|
||||||
|
return { ...state, visits: [ ...newVisits, ...visits ] };
|
||||||
|
},
|
||||||
|
}, initialState);
|
||||||
|
|
||||||
|
export const getNonOrphanVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => (
|
||||||
|
query: ShlinkVisitsParams = {},
|
||||||
|
doIntervalFallback = false,
|
||||||
|
) => async (dispatch: Dispatch, getState: GetState) => {
|
||||||
|
const { getNonOrphanVisits } = buildShlinkApiClient(getState);
|
||||||
|
const visitsLoader = async (page: number, itemsPerPage: number) =>
|
||||||
|
getNonOrphanVisits({ ...query, page, itemsPerPage });
|
||||||
|
const lastVisitLoader = lastVisitLoaderForLoader(doIntervalFallback, getNonOrphanVisits);
|
||||||
|
const shouldCancel = () => getState().orphanVisits.cancelLoad;
|
||||||
|
const extraFinishActionData: Partial<NonOrphanVisitsAction> = { query };
|
||||||
|
const actionMap = {
|
||||||
|
start: GET_NON_ORPHAN_VISITS_START,
|
||||||
|
large: GET_NON_ORPHAN_VISITS_LARGE,
|
||||||
|
finish: GET_NON_ORPHAN_VISITS,
|
||||||
|
error: GET_NON_ORPHAN_VISITS_ERROR,
|
||||||
|
progress: GET_NON_ORPHAN_VISITS_PROGRESS_CHANGED,
|
||||||
|
fallbackToInterval: GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL,
|
||||||
|
};
|
||||||
|
|
||||||
|
return getVisitsWithLoader(visitsLoader, lastVisitLoader, extraFinishActionData, actionMap, dispatch, shouldCancel);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const cancelGetNonOrphanVisits = buildActionCreator(GET_NON_ORPHAN_VISITS_CANCEL);
|
|
@ -59,7 +59,7 @@ export default buildReducer<VisitsInfo, OrphanVisitsCombinedAction>({
|
||||||
const { visits, query = {} } = state;
|
const { visits, query = {} } = state;
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
const newVisits = createdVisits
|
const newVisits = createdVisits
|
||||||
.filter(({ visit }) => isBetween(visit.date, startDate, endDate))
|
.filter(({ visit, shortUrl }) => !shortUrl && isBetween(visit.date, startDate, endDate))
|
||||||
.map(({ visit }) => visit);
|
.map(({ visit }) => visit);
|
||||||
|
|
||||||
return { ...state, visits: [ ...newVisits, ...visits ] };
|
return { ...state, visits: [ ...newVisits, ...visits ] };
|
||||||
|
|
|
@ -1,12 +1,14 @@
|
||||||
import Bottle from 'bottlejs';
|
import Bottle from 'bottlejs';
|
||||||
import ShortUrlVisits from '../ShortUrlVisits';
|
|
||||||
import { cancelGetShortUrlVisits, getShortUrlVisits } from '../reducers/shortUrlVisits';
|
|
||||||
import MapModal from '../helpers/MapModal';
|
import MapModal from '../helpers/MapModal';
|
||||||
import { createNewVisits } from '../reducers/visitCreation';
|
import { createNewVisits } from '../reducers/visitCreation';
|
||||||
|
import ShortUrlVisits from '../ShortUrlVisits';
|
||||||
import TagVisits from '../TagVisits';
|
import TagVisits from '../TagVisits';
|
||||||
import { cancelGetTagVisits, getTagVisits } from '../reducers/tagVisits';
|
|
||||||
import { OrphanVisits } from '../OrphanVisits';
|
import { OrphanVisits } from '../OrphanVisits';
|
||||||
|
import { NonOrphanVisits } from '../NonOrphanVisits';
|
||||||
|
import { cancelGetShortUrlVisits, getShortUrlVisits } from '../reducers/shortUrlVisits';
|
||||||
|
import { cancelGetTagVisits, getTagVisits } from '../reducers/tagVisits';
|
||||||
import { cancelGetOrphanVisits, getOrphanVisits } from '../reducers/orphanVisits';
|
import { cancelGetOrphanVisits, getOrphanVisits } from '../reducers/orphanVisits';
|
||||||
|
import { cancelGetNonOrphanVisits, getNonOrphanVisits } from '../reducers/nonOrphanVisits';
|
||||||
import { ConnectDecorator } from '../../container/types';
|
import { ConnectDecorator } from '../../container/types';
|
||||||
import { loadVisitsOverview } from '../reducers/visitsOverview';
|
import { loadVisitsOverview } from '../reducers/visitsOverview';
|
||||||
import * as visitsParser from './VisitsParser';
|
import * as visitsParser from './VisitsParser';
|
||||||
|
@ -34,6 +36,12 @@ const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
||||||
[ 'getOrphanVisits', 'cancelGetOrphanVisits', 'createNewVisits', 'loadMercureInfo' ],
|
[ 'getOrphanVisits', 'cancelGetOrphanVisits', 'createNewVisits', 'loadMercureInfo' ],
|
||||||
));
|
));
|
||||||
|
|
||||||
|
bottle.serviceFactory('NonOrphanVisits', NonOrphanVisits, 'VisitsExporter');
|
||||||
|
bottle.decorator('NonOrphanVisits', connect(
|
||||||
|
[ 'nonOrphanVisits', 'mercureInfo', 'settings', 'selectedServer' ],
|
||||||
|
[ 'getNonOrphanVisits', 'cancelGetNonOrphanVisits', 'createNewVisits', 'loadMercureInfo' ],
|
||||||
|
));
|
||||||
|
|
||||||
// Services
|
// Services
|
||||||
bottle.serviceFactory('VisitsParser', () => visitsParser);
|
bottle.serviceFactory('VisitsParser', () => visitsParser);
|
||||||
bottle.service('VisitsExporter', VisitsExporter, 'window', 'csvjson');
|
bottle.service('VisitsExporter', VisitsExporter, 'window', 'csvjson');
|
||||||
|
@ -48,6 +56,9 @@ const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
||||||
bottle.serviceFactory('getOrphanVisits', getOrphanVisits, 'buildShlinkApiClient');
|
bottle.serviceFactory('getOrphanVisits', getOrphanVisits, 'buildShlinkApiClient');
|
||||||
bottle.serviceFactory('cancelGetOrphanVisits', () => cancelGetOrphanVisits);
|
bottle.serviceFactory('cancelGetOrphanVisits', () => cancelGetOrphanVisits);
|
||||||
|
|
||||||
|
bottle.serviceFactory('getNonOrphanVisits', getNonOrphanVisits, 'buildShlinkApiClient');
|
||||||
|
bottle.serviceFactory('cancelGetNonOrphanVisits', () => cancelGetNonOrphanVisits);
|
||||||
|
|
||||||
bottle.serviceFactory('createNewVisits', () => createNewVisits);
|
bottle.serviceFactory('createNewVisits', () => createNewVisits);
|
||||||
bottle.serviceFactory('loadVisitsOverview', loadVisitsOverview, 'buildShlinkApiClient');
|
bottle.serviceFactory('loadVisitsOverview', loadVisitsOverview, 'buildShlinkApiClient');
|
||||||
};
|
};
|
||||||
|
|
|
@ -313,6 +313,20 @@ describe('ShlinkApiClient', () => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('getNonOrphanVisits', () => {
|
||||||
|
it('returns non-orphan visits', async () => {
|
||||||
|
const expectedData: Visit[] = [];
|
||||||
|
const resp = { visits: expectedData };
|
||||||
|
const axiosSpy = createAxiosMock({ data: resp });
|
||||||
|
const { getNonOrphanVisits } = new ShlinkApiClient(axiosSpy, '', '');
|
||||||
|
|
||||||
|
const result = await getNonOrphanVisits();
|
||||||
|
|
||||||
|
expect(axiosSpy).toHaveBeenCalled();
|
||||||
|
expect(result).toEqual(expectedData);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe('editDomainRedirects', () => {
|
describe('editDomainRedirects', () => {
|
||||||
it('returns the redirects', async () => {
|
it('returns the redirects', async () => {
|
||||||
const resp = { baseUrlRedirect: null, regular404Redirect: 'foo', invalidShortUrlRedirect: 'bar' };
|
const resp = { baseUrlRedirect: null, regular404Redirect: 'foo', invalidShortUrlRedirect: 'bar' };
|
||||||
|
|
|
@ -11,7 +11,7 @@ import { SemVer } from '../../src/utils/helpers/version';
|
||||||
describe('<MenuLayout />', () => {
|
describe('<MenuLayout />', () => {
|
||||||
const ServerError = jest.fn();
|
const ServerError = jest.fn();
|
||||||
const C = jest.fn();
|
const C = jest.fn();
|
||||||
const MenuLayout = createMenuLayout(C, C, C, C, C, C, C, ServerError, C, C, C);
|
const MenuLayout = createMenuLayout(C, C, C, C, C, C, C, C, ServerError, C, C, C);
|
||||||
let wrapper: ShallowWrapper;
|
let wrapper: ShallowWrapper;
|
||||||
const createWrapper = (selectedServer: SelectedServer) => {
|
const createWrapper = (selectedServer: SelectedServer) => {
|
||||||
wrapper = shallow(
|
wrapper = shallow(
|
||||||
|
@ -52,6 +52,9 @@ describe('<MenuLayout />', () => {
|
||||||
[ '2.5.0' as SemVer, 8 ],
|
[ '2.5.0' as SemVer, 8 ],
|
||||||
[ '2.6.0' as SemVer, 9 ],
|
[ '2.6.0' as SemVer, 9 ],
|
||||||
[ '2.7.0' as SemVer, 9 ],
|
[ '2.7.0' as SemVer, 9 ],
|
||||||
|
[ '2.8.0' as SemVer, 10 ],
|
||||||
|
[ '2.10.0' as SemVer, 10 ],
|
||||||
|
[ '3.0.0' as SemVer, 11 ],
|
||||||
])('has expected amount of routes based on selected server\'s version', (version, expectedAmountOfRoutes) => {
|
])('has expected amount of routes based on selected server\'s version', (version, expectedAmountOfRoutes) => {
|
||||||
const selectedServer = Mock.of<ReachableServer>({ version });
|
const selectedServer = Mock.of<ReachableServer>({ version });
|
||||||
const wrapper = createWrapper(selectedServer).dive();
|
const wrapper = createWrapper(selectedServer).dive();
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
import { FC } from 'react';
|
import { FC } from 'react';
|
||||||
import { shallow, ShallowWrapper } from 'enzyme';
|
import { mount, ReactWrapper } from 'enzyme';
|
||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import { CardText } from 'reactstrap';
|
import { Link, MemoryRouter } from 'react-router-dom';
|
||||||
import { Link } from 'react-router-dom';
|
|
||||||
import { ShortUrlsList as ShortUrlsListState } from '../../src/short-urls/reducers/shortUrlsList';
|
import { ShortUrlsList as ShortUrlsListState } from '../../src/short-urls/reducers/shortUrlsList';
|
||||||
import { Overview as overviewCreator } from '../../src/servers/Overview';
|
import { Overview as overviewCreator } from '../../src/servers/Overview';
|
||||||
import { TagsList } from '../../src/tags/reducers/tagsList';
|
import { TagsList } from '../../src/tags/reducers/tagsList';
|
||||||
|
@ -10,9 +9,10 @@ import { VisitsOverview } from '../../src/visits/reducers/visitsOverview';
|
||||||
import { MercureInfo } from '../../src/mercure/reducers/mercureInfo';
|
import { MercureInfo } from '../../src/mercure/reducers/mercureInfo';
|
||||||
import { ReachableServer } from '../../src/servers/data';
|
import { ReachableServer } from '../../src/servers/data';
|
||||||
import { prettify } from '../../src/utils/helpers/numbers';
|
import { prettify } from '../../src/utils/helpers/numbers';
|
||||||
|
import { HighlightCard } from '../../src/servers/helpers/HighlightCard';
|
||||||
|
|
||||||
describe('<Overview />', () => {
|
describe('<Overview />', () => {
|
||||||
let wrapper: ShallowWrapper;
|
let wrapper: ReactWrapper;
|
||||||
const ShortUrlsTable = () => null;
|
const ShortUrlsTable = () => null;
|
||||||
const CreateShortUrl = () => null;
|
const CreateShortUrl = () => null;
|
||||||
const ForServerVersion: FC = ({ children }) => <>{children}</>;
|
const ForServerVersion: FC = ({ children }) => <>{children}</>;
|
||||||
|
@ -25,7 +25,8 @@ describe('<Overview />', () => {
|
||||||
};
|
};
|
||||||
const serverId = '123';
|
const serverId = '123';
|
||||||
const createWrapper = (loading = false) => {
|
const createWrapper = (loading = false) => {
|
||||||
wrapper = shallow(
|
wrapper = mount(
|
||||||
|
<MemoryRouter>
|
||||||
<Overview
|
<Overview
|
||||||
listShortUrls={listShortUrls}
|
listShortUrls={listShortUrls}
|
||||||
listTags={listTags}
|
listTags={listTags}
|
||||||
|
@ -37,8 +38,9 @@ describe('<Overview />', () => {
|
||||||
createNewVisits={jest.fn()}
|
createNewVisits={jest.fn()}
|
||||||
loadMercureInfo={jest.fn()}
|
loadMercureInfo={jest.fn()}
|
||||||
mercureInfo={Mock.all<MercureInfo>()}
|
mercureInfo={Mock.all<MercureInfo>()}
|
||||||
/>,
|
/>
|
||||||
).dive(); // Dive is needed as this component is wrapped in a HOC
|
</MemoryRouter>,
|
||||||
|
);
|
||||||
|
|
||||||
return wrapper;
|
return wrapper;
|
||||||
};
|
};
|
||||||
|
@ -47,7 +49,7 @@ describe('<Overview />', () => {
|
||||||
|
|
||||||
it('displays loading messages when still loading', () => {
|
it('displays loading messages when still loading', () => {
|
||||||
const wrapper = createWrapper(true);
|
const wrapper = createWrapper(true);
|
||||||
const cards = wrapper.find(CardText);
|
const cards = wrapper.find(HighlightCard);
|
||||||
|
|
||||||
expect(cards).toHaveLength(4);
|
expect(cards).toHaveLength(4);
|
||||||
cards.forEach((card) => expect(card.html()).toContain('Loading...'));
|
cards.forEach((card) => expect(card.html()).toContain('Loading...'));
|
||||||
|
@ -55,7 +57,7 @@ describe('<Overview />', () => {
|
||||||
|
|
||||||
it('displays amounts in cards after finishing loading', () => {
|
it('displays amounts in cards after finishing loading', () => {
|
||||||
const wrapper = createWrapper();
|
const wrapper = createWrapper();
|
||||||
const cards = wrapper.find(CardText);
|
const cards = wrapper.find(HighlightCard);
|
||||||
|
|
||||||
expect(cards).toHaveLength(4);
|
expect(cards).toHaveLength(4);
|
||||||
expect(cards.at(0).html()).toContain(prettify(3456));
|
expect(cards.at(0).html()).toContain(prettify(3456));
|
||||||
|
@ -75,8 +77,10 @@ describe('<Overview />', () => {
|
||||||
const wrapper = createWrapper();
|
const wrapper = createWrapper();
|
||||||
const links = wrapper.find(Link);
|
const links = wrapper.find(Link);
|
||||||
|
|
||||||
expect(links).toHaveLength(2);
|
expect(links).toHaveLength(4);
|
||||||
expect(links.at(0).prop('to')).toEqual(`/server/${serverId}/create-short-url`);
|
expect(links.at(0).prop('to')).toEqual(`/server/${serverId}/list-short-urls/1`);
|
||||||
expect(links.at(1).prop('to')).toEqual(`/server/${serverId}/list-short-urls/1`);
|
expect(links.at(1).prop('to')).toEqual(`/server/${serverId}/manage-tags`);
|
||||||
|
expect(links.at(2).prop('to')).toEqual(`/server/${serverId}/create-short-url`);
|
||||||
|
expect(links.at(3).prop('to')).toEqual(`/server/${serverId}/list-short-urls/1`);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
65
test/servers/helpers/HighlightCard.test.tsx
Normal file
65
test/servers/helpers/HighlightCard.test.tsx
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
import { shallow, ShallowWrapper } from 'enzyme';
|
||||||
|
import { ReactNode } from 'react';
|
||||||
|
import { Card, CardText, CardTitle } from 'reactstrap';
|
||||||
|
import { Link } from 'react-router-dom';
|
||||||
|
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
||||||
|
import { HighlightCard, HighlightCardProps } from '../../../src/servers/helpers/HighlightCard';
|
||||||
|
|
||||||
|
describe('<HighlightCard />', () => {
|
||||||
|
let wrapper: ShallowWrapper;
|
||||||
|
const createWrapper = (props: HighlightCardProps & { children?: ReactNode }) => {
|
||||||
|
wrapper = shallow(<HighlightCard {...props} />);
|
||||||
|
|
||||||
|
return wrapper;
|
||||||
|
};
|
||||||
|
|
||||||
|
afterEach(() => wrapper?.unmount());
|
||||||
|
|
||||||
|
it.each([
|
||||||
|
[ undefined ],
|
||||||
|
[ false ],
|
||||||
|
])('renders expected components', (link) => {
|
||||||
|
const wrapper = createWrapper({ title: 'foo', link: link as undefined | false });
|
||||||
|
|
||||||
|
expect(wrapper.find(Card)).toHaveLength(1);
|
||||||
|
expect(wrapper.find(CardTitle)).toHaveLength(1);
|
||||||
|
expect(wrapper.find(CardText)).toHaveLength(1);
|
||||||
|
expect(wrapper.find(FontAwesomeIcon)).toHaveLength(0);
|
||||||
|
expect(wrapper.prop('tag')).not.toEqual(Link);
|
||||||
|
expect(wrapper.prop('to')).not.toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
it.each([
|
||||||
|
[ 'foo' ],
|
||||||
|
[ 'bar' ],
|
||||||
|
[ 'baz' ],
|
||||||
|
])('renders provided title', (title) => {
|
||||||
|
const wrapper = createWrapper({ title });
|
||||||
|
const cardTitle = wrapper.find(CardTitle);
|
||||||
|
|
||||||
|
expect(cardTitle.html()).toContain(`>${title}<`);
|
||||||
|
});
|
||||||
|
|
||||||
|
it.each([
|
||||||
|
[ 'foo' ],
|
||||||
|
[ 'bar' ],
|
||||||
|
[ 'baz' ],
|
||||||
|
])('renders provided children', (children) => {
|
||||||
|
const wrapper = createWrapper({ title: 'foo', children });
|
||||||
|
const cardText = wrapper.find(CardText);
|
||||||
|
|
||||||
|
expect(cardText.html()).toContain(`>${children}<`);
|
||||||
|
});
|
||||||
|
|
||||||
|
it.each([
|
||||||
|
[ 'foo' ],
|
||||||
|
[ 'bar' ],
|
||||||
|
[ 'baz' ],
|
||||||
|
])('adds extra props when a link is provided', (link) => {
|
||||||
|
const wrapper = createWrapper({ title: 'foo', link });
|
||||||
|
|
||||||
|
expect(wrapper.find(FontAwesomeIcon)).toHaveLength(1);
|
||||||
|
expect(wrapper.prop('tag')).toEqual(Link);
|
||||||
|
expect(wrapper.prop('to')).toEqual(link);
|
||||||
|
});
|
||||||
|
});
|
47
test/visits/NonOrphanVisits.test.tsx
Normal file
47
test/visits/NonOrphanVisits.test.tsx
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
import { shallow } from 'enzyme';
|
||||||
|
import { Mock } from 'ts-mockery';
|
||||||
|
import { History, Location } from 'history';
|
||||||
|
import { match } from 'react-router';
|
||||||
|
import { NonOrphanVisits as createNonOrphanVisits } from '../../src/visits/NonOrphanVisits';
|
||||||
|
import { MercureBoundProps } from '../../src/mercure/helpers/boundToMercureHub';
|
||||||
|
import { VisitsInfo } from '../../src/visits/types';
|
||||||
|
import VisitsStats from '../../src/visits/VisitsStats';
|
||||||
|
import { NonOrphanVisitsHeader } from '../../src/visits/NonOrphanVisitsHeader';
|
||||||
|
import { Settings } from '../../src/settings/reducers/settings';
|
||||||
|
import { VisitsExporter } from '../../src/visits/services/VisitsExporter';
|
||||||
|
import { SelectedServer } from '../../src/servers/data';
|
||||||
|
|
||||||
|
describe('<NonOrphanVisits />', () => {
|
||||||
|
it('wraps visits stats and header', () => {
|
||||||
|
const goBack = jest.fn();
|
||||||
|
const getNonOrphanVisits = jest.fn();
|
||||||
|
const cancelGetNonOrphanVisits = jest.fn();
|
||||||
|
const nonOrphanVisits = Mock.all<VisitsInfo>();
|
||||||
|
const NonOrphanVisits = createNonOrphanVisits(Mock.all<VisitsExporter>());
|
||||||
|
|
||||||
|
const wrapper = shallow(
|
||||||
|
<NonOrphanVisits
|
||||||
|
{...Mock.of<MercureBoundProps>({ mercureInfo: {} })}
|
||||||
|
getNonOrphanVisits={getNonOrphanVisits}
|
||||||
|
nonOrphanVisits={nonOrphanVisits}
|
||||||
|
cancelGetNonOrphanVisits={cancelGetNonOrphanVisits}
|
||||||
|
history={Mock.of<History>({ goBack })}
|
||||||
|
location={Mock.all<Location>()}
|
||||||
|
match={Mock.of<match>({ url: 'the_base_url' })}
|
||||||
|
settings={Mock.all<Settings>()}
|
||||||
|
selectedServer={Mock.all<SelectedServer>()}
|
||||||
|
/>,
|
||||||
|
).dive();
|
||||||
|
const stats = wrapper.find(VisitsStats);
|
||||||
|
const header = wrapper.find(NonOrphanVisitsHeader);
|
||||||
|
|
||||||
|
expect(stats).toHaveLength(1);
|
||||||
|
expect(header).toHaveLength(1);
|
||||||
|
expect(stats.prop('cancelGetVisits')).toEqual(cancelGetNonOrphanVisits);
|
||||||
|
expect(stats.prop('visitsInfo')).toEqual(nonOrphanVisits);
|
||||||
|
expect(stats.prop('baseUrl')).toEqual('the_base_url');
|
||||||
|
expect(stats.prop('isOrphanVisits')).not.toBeDefined();
|
||||||
|
expect(header.prop('nonOrphanVisits')).toEqual(nonOrphanVisits);
|
||||||
|
expect(header.prop('goBack')).toEqual(goBack);
|
||||||
|
});
|
||||||
|
});
|
21
test/visits/NonOrphanVisitsHeader.test.tsx
Normal file
21
test/visits/NonOrphanVisitsHeader.test.tsx
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
import { shallow } from 'enzyme';
|
||||||
|
import { Mock } from 'ts-mockery';
|
||||||
|
import { NonOrphanVisitsHeader } from '../../src/visits/NonOrphanVisitsHeader';
|
||||||
|
import VisitsHeader from '../../src/visits/VisitsHeader';
|
||||||
|
import { Visit, VisitsInfo } from '../../src/visits/types';
|
||||||
|
|
||||||
|
describe('<NonOrphanVisitsHeader />', () => {
|
||||||
|
it('wraps a VisitsHeader with provided data', () => {
|
||||||
|
const visits: Visit[] = [];
|
||||||
|
const orphanVisits = Mock.of<VisitsInfo>({ visits });
|
||||||
|
const goBack = jest.fn();
|
||||||
|
|
||||||
|
const wrapper = shallow(<NonOrphanVisitsHeader nonOrphanVisits={orphanVisits} goBack={goBack} />);
|
||||||
|
const visitsHeader = wrapper.find(VisitsHeader);
|
||||||
|
|
||||||
|
expect(visitsHeader).toHaveLength(1);
|
||||||
|
expect(visitsHeader.prop('visits')).toEqual(visits);
|
||||||
|
expect(visitsHeader.prop('goBack')).toEqual(goBack);
|
||||||
|
expect(visitsHeader.prop('title')).toEqual('Non-orphan visits');
|
||||||
|
});
|
||||||
|
});
|
|
@ -1,7 +1,7 @@
|
||||||
import { shallow } from 'enzyme';
|
import { shallow } from 'enzyme';
|
||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import { History, Location } from 'history';
|
import { History, Location } from 'history';
|
||||||
import { match } from 'react-router'; // eslint-disable-line @typescript-eslint/no-unused-vars
|
import { match } from 'react-router';
|
||||||
import { OrphanVisits as createOrphanVisits } from '../../src/visits/OrphanVisits';
|
import { OrphanVisits as createOrphanVisits } from '../../src/visits/OrphanVisits';
|
||||||
import { MercureBoundProps } from '../../src/mercure/helpers/boundToMercureHub';
|
import { MercureBoundProps } from '../../src/mercure/helpers/boundToMercureHub';
|
||||||
import { VisitsInfo } from '../../src/visits/types';
|
import { VisitsInfo } from '../../src/visits/types';
|
||||||
|
@ -40,6 +40,7 @@ describe('<OrphanVisits />', () => {
|
||||||
expect(stats.prop('cancelGetVisits')).toEqual(cancelGetOrphanVisits);
|
expect(stats.prop('cancelGetVisits')).toEqual(cancelGetOrphanVisits);
|
||||||
expect(stats.prop('visitsInfo')).toEqual(orphanVisits);
|
expect(stats.prop('visitsInfo')).toEqual(orphanVisits);
|
||||||
expect(stats.prop('baseUrl')).toEqual('the_base_url');
|
expect(stats.prop('baseUrl')).toEqual('the_base_url');
|
||||||
|
expect(stats.prop('isOrphanVisits')).toEqual(true);
|
||||||
expect(header.prop('orphanVisits')).toEqual(orphanVisits);
|
expect(header.prop('orphanVisits')).toEqual(orphanVisits);
|
||||||
expect(header.prop('goBack')).toEqual(goBack);
|
expect(header.prop('goBack')).toEqual(goBack);
|
||||||
});
|
});
|
||||||
|
|
|
@ -48,6 +48,7 @@ describe('<ShortUrlVisits />', () => {
|
||||||
const visitHeader = wrapper.find(ShortUrlVisitsHeader);
|
const visitHeader = wrapper.find(ShortUrlVisitsHeader);
|
||||||
|
|
||||||
expect(visitStats).toHaveLength(1);
|
expect(visitStats).toHaveLength(1);
|
||||||
|
expect(visitStats.prop('isOrphanVisits')).not.toBeDefined();
|
||||||
expect(visitHeader).toHaveLength(1);
|
expect(visitHeader).toHaveLength(1);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -44,6 +44,7 @@ describe('<TagVisits />', () => {
|
||||||
const visitHeader = wrapper.find(TagVisitsHeader);
|
const visitHeader = wrapper.find(TagVisitsHeader);
|
||||||
|
|
||||||
expect(visitStats).toHaveLength(1);
|
expect(visitStats).toHaveLength(1);
|
||||||
|
expect(visitStats.prop('isOrphanVisits')).not.toBeDefined();
|
||||||
expect(visitHeader).toHaveLength(1);
|
expect(visitHeader).toHaveLength(1);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
213
test/visits/reducers/nonOrphanVisits.test.ts
Normal file
213
test/visits/reducers/nonOrphanVisits.test.ts
Normal file
|
@ -0,0 +1,213 @@
|
||||||
|
import { Mock } from 'ts-mockery';
|
||||||
|
import { addDays, formatISO, subDays } from 'date-fns';
|
||||||
|
import reducer, {
|
||||||
|
getNonOrphanVisits,
|
||||||
|
cancelGetNonOrphanVisits,
|
||||||
|
GET_NON_ORPHAN_VISITS_START,
|
||||||
|
GET_NON_ORPHAN_VISITS_ERROR,
|
||||||
|
GET_NON_ORPHAN_VISITS,
|
||||||
|
GET_NON_ORPHAN_VISITS_LARGE,
|
||||||
|
GET_NON_ORPHAN_VISITS_CANCEL,
|
||||||
|
GET_NON_ORPHAN_VISITS_PROGRESS_CHANGED,
|
||||||
|
GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL,
|
||||||
|
} from '../../../src/visits/reducers/nonOrphanVisits';
|
||||||
|
import { CREATE_VISITS } from '../../../src/visits/reducers/visitCreation';
|
||||||
|
import { rangeOf } from '../../../src/utils/utils';
|
||||||
|
import { Visit, VisitsInfo } from '../../../src/visits/types';
|
||||||
|
import { ShlinkVisits } from '../../../src/api/types';
|
||||||
|
import ShlinkApiClient from '../../../src/api/services/ShlinkApiClient';
|
||||||
|
import { ShlinkState } from '../../../src/container/types';
|
||||||
|
import { formatIsoDate } from '../../../src/utils/helpers/date';
|
||||||
|
import { DateInterval } from '../../../src/utils/dates/types';
|
||||||
|
|
||||||
|
describe('nonOrphanVisitsReducer', () => {
|
||||||
|
const now = new Date();
|
||||||
|
const visitsMocks = rangeOf(2, () => Mock.all<Visit>());
|
||||||
|
|
||||||
|
describe('reducer', () => {
|
||||||
|
const buildState = (data: Partial<VisitsInfo>) => Mock.of<VisitsInfo>(data);
|
||||||
|
|
||||||
|
it('returns loading on GET_NON_ORPHAN_VISITS_START', () => {
|
||||||
|
const state = reducer(buildState({ loading: false }), { type: GET_NON_ORPHAN_VISITS_START } as any);
|
||||||
|
const { loading } = state;
|
||||||
|
|
||||||
|
expect(loading).toEqual(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns loadingLarge on GET_NON_ORPHAN_VISITS_LARGE', () => {
|
||||||
|
const state = reducer(buildState({ loadingLarge: false }), { type: GET_NON_ORPHAN_VISITS_LARGE } as any);
|
||||||
|
const { loadingLarge } = state;
|
||||||
|
|
||||||
|
expect(loadingLarge).toEqual(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns cancelLoad on GET_NON_ORPHAN_VISITS_CANCEL', () => {
|
||||||
|
const state = reducer(buildState({ cancelLoad: false }), { type: GET_NON_ORPHAN_VISITS_CANCEL } as any);
|
||||||
|
const { cancelLoad } = state;
|
||||||
|
|
||||||
|
expect(cancelLoad).toEqual(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('stops loading and returns error on GET_NON_ORPHAN_VISITS_ERROR', () => {
|
||||||
|
const state = reducer(buildState({ loading: true, error: false }), { type: GET_NON_ORPHAN_VISITS_ERROR } as any);
|
||||||
|
const { loading, error } = state;
|
||||||
|
|
||||||
|
expect(loading).toEqual(false);
|
||||||
|
expect(error).toEqual(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('return visits on GET_NON_ORPHAN_VISITS', () => {
|
||||||
|
const actionVisits = [{}, {}];
|
||||||
|
const state = reducer(
|
||||||
|
buildState({ loading: true, error: false }),
|
||||||
|
{ type: GET_NON_ORPHAN_VISITS, visits: actionVisits } as any,
|
||||||
|
);
|
||||||
|
const { loading, error, visits } = state;
|
||||||
|
|
||||||
|
expect(loading).toEqual(false);
|
||||||
|
expect(error).toEqual(false);
|
||||||
|
expect(visits).toEqual(actionVisits);
|
||||||
|
});
|
||||||
|
|
||||||
|
it.each([
|
||||||
|
[{}, visitsMocks.length + 2 ],
|
||||||
|
[
|
||||||
|
Mock.of<VisitsInfo>({
|
||||||
|
query: { endDate: formatIsoDate(subDays(now, 1)) ?? undefined },
|
||||||
|
}),
|
||||||
|
visitsMocks.length,
|
||||||
|
],
|
||||||
|
[
|
||||||
|
Mock.of<VisitsInfo>({
|
||||||
|
query: { startDate: formatIsoDate(addDays(now, 1)) ?? undefined },
|
||||||
|
}),
|
||||||
|
visitsMocks.length,
|
||||||
|
],
|
||||||
|
[
|
||||||
|
Mock.of<VisitsInfo>({
|
||||||
|
query: {
|
||||||
|
startDate: formatIsoDate(subDays(now, 5)) ?? undefined,
|
||||||
|
endDate: formatIsoDate(subDays(now, 2)) ?? undefined,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
visitsMocks.length,
|
||||||
|
],
|
||||||
|
[
|
||||||
|
Mock.of<VisitsInfo>({
|
||||||
|
query: {
|
||||||
|
startDate: formatIsoDate(subDays(now, 5)) ?? undefined,
|
||||||
|
endDate: formatIsoDate(addDays(now, 3)) ?? undefined,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
visitsMocks.length + 2,
|
||||||
|
],
|
||||||
|
])('prepends new visits on CREATE_VISIT', (state, expectedVisits) => {
|
||||||
|
const prevState = buildState({ ...state, visits: visitsMocks });
|
||||||
|
const visit = Mock.of<Visit>({ date: formatIsoDate(now) ?? undefined });
|
||||||
|
|
||||||
|
const { visits } = reducer(
|
||||||
|
prevState,
|
||||||
|
{ type: CREATE_VISITS, createdVisits: [{ visit }, { visit }] } as any,
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(visits).toHaveLength(expectedVisits);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns new progress on GET_NON_ORPHAN_VISITS_PROGRESS_CHANGED', () => {
|
||||||
|
const state = reducer(undefined, { type: GET_NON_ORPHAN_VISITS_PROGRESS_CHANGED, progress: 85 } as any);
|
||||||
|
|
||||||
|
expect(state).toEqual(expect.objectContaining({ progress: 85 }));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns fallbackInterval on GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL', () => {
|
||||||
|
const fallbackInterval: DateInterval = 'last30Days';
|
||||||
|
const state = reducer(undefined, { type: GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL, fallbackInterval } as any);
|
||||||
|
|
||||||
|
expect(state).toEqual(expect.objectContaining({ fallbackInterval }));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('getNonOrphanVisits', () => {
|
||||||
|
type GetVisitsReturn = Promise<ShlinkVisits> | ((query: any) => Promise<ShlinkVisits>);
|
||||||
|
|
||||||
|
const buildApiClientMock = (returned: GetVisitsReturn) => Mock.of<ShlinkApiClient>({
|
||||||
|
getNonOrphanVisits: jest.fn(typeof returned === 'function' ? returned : async () => returned),
|
||||||
|
});
|
||||||
|
const dispatchMock = jest.fn();
|
||||||
|
const getState = () => Mock.of<ShlinkState>({
|
||||||
|
orphanVisits: { cancelLoad: false },
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(jest.resetAllMocks);
|
||||||
|
|
||||||
|
it('dispatches start and error when promise is rejected', async () => {
|
||||||
|
const ShlinkApiClient = buildApiClientMock(Promise.reject({}));
|
||||||
|
|
||||||
|
await getNonOrphanVisits(() => ShlinkApiClient)()(dispatchMock, getState);
|
||||||
|
|
||||||
|
expect(dispatchMock).toHaveBeenCalledTimes(2);
|
||||||
|
expect(dispatchMock).toHaveBeenNthCalledWith(1, { type: GET_NON_ORPHAN_VISITS_START });
|
||||||
|
expect(dispatchMock).toHaveBeenNthCalledWith(2, { type: GET_NON_ORPHAN_VISITS_ERROR });
|
||||||
|
expect(ShlinkApiClient.getNonOrphanVisits).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it.each([
|
||||||
|
[ undefined ],
|
||||||
|
[{}],
|
||||||
|
])('dispatches start and success when promise is resolved', async (query) => {
|
||||||
|
const visits = visitsMocks.map((visit) => ({ ...visit, visitedUrl: '' }));
|
||||||
|
const ShlinkApiClient = buildApiClientMock(Promise.resolve({
|
||||||
|
data: visits,
|
||||||
|
pagination: {
|
||||||
|
currentPage: 1,
|
||||||
|
pagesCount: 1,
|
||||||
|
totalItems: 1,
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
await getNonOrphanVisits(() => ShlinkApiClient)(query)(dispatchMock, getState);
|
||||||
|
|
||||||
|
expect(dispatchMock).toHaveBeenCalledTimes(2);
|
||||||
|
expect(dispatchMock).toHaveBeenNthCalledWith(1, { type: GET_NON_ORPHAN_VISITS_START });
|
||||||
|
expect(dispatchMock).toHaveBeenNthCalledWith(2, { type: GET_NON_ORPHAN_VISITS, visits, query: query ?? {} });
|
||||||
|
expect(ShlinkApiClient.getNonOrphanVisits).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it.each([
|
||||||
|
[
|
||||||
|
[ Mock.of<Visit>({ date: formatISO(subDays(new Date(), 5)) }) ],
|
||||||
|
{ type: GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL, fallbackInterval: 'last7Days' },
|
||||||
|
],
|
||||||
|
[
|
||||||
|
[ Mock.of<Visit>({ date: formatISO(subDays(new Date(), 200)) }) ],
|
||||||
|
{ type: GET_NON_ORPHAN_VISITS_FALLBACK_TO_INTERVAL, fallbackInterval: 'last365Days' },
|
||||||
|
],
|
||||||
|
[[], expect.objectContaining({ type: GET_NON_ORPHAN_VISITS }) ],
|
||||||
|
])('dispatches fallback interval when the list of visits is empty', async (lastVisits, expectedSecondDispatch) => {
|
||||||
|
const buildVisitsResult = (data: Visit[] = []): ShlinkVisits => ({
|
||||||
|
data,
|
||||||
|
pagination: {
|
||||||
|
currentPage: 1,
|
||||||
|
pagesCount: 1,
|
||||||
|
totalItems: 1,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
const getShlinkOrphanVisits = jest.fn()
|
||||||
|
.mockResolvedValueOnce(buildVisitsResult())
|
||||||
|
.mockResolvedValueOnce(buildVisitsResult(lastVisits));
|
||||||
|
const ShlinkApiClient = Mock.of<ShlinkApiClient>({ getNonOrphanVisits: getShlinkOrphanVisits });
|
||||||
|
|
||||||
|
await getNonOrphanVisits(() => ShlinkApiClient)({}, true)(dispatchMock, getState);
|
||||||
|
|
||||||
|
expect(dispatchMock).toHaveBeenCalledTimes(2);
|
||||||
|
expect(dispatchMock).toHaveBeenNthCalledWith(1, { type: GET_NON_ORPHAN_VISITS_START });
|
||||||
|
expect(dispatchMock).toHaveBeenNthCalledWith(2, expectedSecondDispatch);
|
||||||
|
expect(getShlinkOrphanVisits).toHaveBeenCalledTimes(2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('cancelGetNonOrphanVisits', () => {
|
||||||
|
it('just returns the action with proper type', () =>
|
||||||
|
expect(cancelGetNonOrphanVisits()).toEqual({ type: GET_NON_ORPHAN_VISITS_CANCEL }));
|
||||||
|
});
|
||||||
|
});
|
Loading…
Reference in a new issue