mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2025-01-24 16:53:45 +03:00
Use /tags/stats endpoint when the server supports it
This commit is contained in:
parent
ddaec7c6ac
commit
b87b108e53
5 changed files with 40 additions and 8 deletions
|
@ -16,6 +16,7 @@ import type {
|
||||||
ShlinkShortUrlsResponse,
|
ShlinkShortUrlsResponse,
|
||||||
ShlinkTags,
|
ShlinkTags,
|
||||||
ShlinkTagsResponse,
|
ShlinkTagsResponse,
|
||||||
|
ShlinkTagsStatsResponse,
|
||||||
ShlinkVisits,
|
ShlinkVisits,
|
||||||
ShlinkVisitsOverview,
|
ShlinkVisitsOverview,
|
||||||
ShlinkVisitsParams,
|
ShlinkVisitsParams,
|
||||||
|
@ -85,10 +86,14 @@ export class ShlinkApiClient {
|
||||||
): Promise<ShortUrl> =>
|
): Promise<ShortUrl> =>
|
||||||
this.performRequest<ShortUrl>(`/short-urls/${shortCode}`, 'PATCH', { domain }, edit);
|
this.performRequest<ShortUrl>(`/short-urls/${shortCode}`, 'PATCH', { domain }, edit);
|
||||||
|
|
||||||
public readonly listTags = async (): Promise<ShlinkTags> =>
|
public readonly listTags = async (useTagsStatsEndpoint: boolean): Promise<ShlinkTags> =>
|
||||||
this.performRequest<{ tags: ShlinkTagsResponse }>('/tags', 'GET', { withStats: 'true' })
|
(useTagsStatsEndpoint
|
||||||
.then(({ tags }) => tags)
|
? this.performRequest<{ tags: ShlinkTagsStatsResponse }>('/tags/stats', 'GET')
|
||||||
.then(({ data, stats }) => ({ tags: data, stats }));
|
.then(({ tags }) => tags)
|
||||||
|
.then(({ data }) => ({ tags: data.map(({ tag }) => tag), stats: data }))
|
||||||
|
: this.performRequest<{ tags: ShlinkTagsResponse }>('/tags', 'GET', { withStats: 'true' })
|
||||||
|
.then(({ tags }) => tags)
|
||||||
|
.then(({ data, stats }) => ({ tags: data, stats })));
|
||||||
|
|
||||||
public readonly deleteTags = async (tags: string[]): Promise<{ tags: string[] }> =>
|
public readonly deleteTags = async (tags: string[]): Promise<{ tags: string[] }> =>
|
||||||
this.performEmptyRequest('/tags', 'DELETE', { tags }).then(() => ({ tags }));
|
this.performEmptyRequest('/tags', 'DELETE', { tags }).then(() => ({ tags }));
|
||||||
|
|
|
@ -31,9 +31,14 @@ export interface ShlinkTags {
|
||||||
|
|
||||||
export interface ShlinkTagsResponse {
|
export interface ShlinkTagsResponse {
|
||||||
data: string[];
|
data: string[];
|
||||||
|
/** @deprecated Present only when withStats=true is provided, which is deprecated */
|
||||||
stats: ShlinkTagsStats[];
|
stats: ShlinkTagsStats[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface ShlinkTagsStatsResponse {
|
||||||
|
data: ShlinkTagsStats[];
|
||||||
|
}
|
||||||
|
|
||||||
export interface ShlinkPaginator {
|
export interface ShlinkPaginator {
|
||||||
currentPage: number;
|
currentPage: number;
|
||||||
pagesCount: number;
|
pagesCount: number;
|
||||||
|
|
|
@ -5,6 +5,7 @@ import type { ShlinkTags } from '../../api/types';
|
||||||
import type { ProblemDetailsError } from '../../api/types/errors';
|
import type { ProblemDetailsError } from '../../api/types/errors';
|
||||||
import { parseApiError } from '../../api/utils';
|
import { parseApiError } from '../../api/utils';
|
||||||
import type { createShortUrl } from '../../short-urls/reducers/shortUrlCreation';
|
import type { createShortUrl } from '../../short-urls/reducers/shortUrlCreation';
|
||||||
|
import { supportedFeatures } from '../../utils/helpers/features';
|
||||||
import { createAsyncThunk } from '../../utils/helpers/redux';
|
import { createAsyncThunk } from '../../utils/helpers/redux';
|
||||||
import { createNewVisits } from '../../visits/reducers/visitCreation';
|
import { createNewVisits } from '../../visits/reducers/visitCreation';
|
||||||
import type { CreateVisit, Stats } from '../../visits/types';
|
import type { CreateVisit, Stats } from '../../visits/types';
|
||||||
|
@ -70,14 +71,14 @@ const calculateVisitsPerTag = (createdVisits: CreateVisit[]): TagIncrease[] => O
|
||||||
export const listTags = (buildShlinkApiClient: ShlinkApiClientBuilder, force = true) => createAsyncThunk(
|
export const listTags = (buildShlinkApiClient: ShlinkApiClientBuilder, force = true) => createAsyncThunk(
|
||||||
`${REDUCER_PREFIX}/listTags`,
|
`${REDUCER_PREFIX}/listTags`,
|
||||||
async (_: void, { getState }): Promise<ListTags> => {
|
async (_: void, { getState }): Promise<ListTags> => {
|
||||||
const { tagsList } = getState();
|
const { tagsList, selectedServer } = getState();
|
||||||
|
|
||||||
if (!force && !isEmpty(tagsList.tags)) {
|
if (!force && !isEmpty(tagsList.tags)) {
|
||||||
return tagsList;
|
return tagsList;
|
||||||
}
|
}
|
||||||
|
|
||||||
const { listTags: shlinkListTags } = buildShlinkApiClient(getState);
|
const { listTags: shlinkListTags } = buildShlinkApiClient(getState);
|
||||||
const { tags, stats = [] }: ShlinkTags = await shlinkListTags();
|
const { tags, stats = [] }: ShlinkTags = await shlinkListTags(supportedFeatures.tagsStats(selectedServer));
|
||||||
const processedStats = stats.reduce<TagsStatsMap>((acc, { tag, shortUrlsCount, visitsCount }) => {
|
const processedStats = stats.reduce<TagsStatsMap>((acc, { tag, shortUrlsCount, visitsCount }) => {
|
||||||
acc[tag] = { shortUrlsCount, visitsCount };
|
acc[tag] = { shortUrlsCount, visitsCount };
|
||||||
|
|
||||||
|
|
|
@ -14,11 +14,12 @@ export const supportedFeatures = {
|
||||||
defaultDomainRedirectsEdition: matchesMinVersion('2.10.0'),
|
defaultDomainRedirectsEdition: matchesMinVersion('2.10.0'),
|
||||||
nonOrphanVisits: matchesMinVersion('3.0.0'),
|
nonOrphanVisits: matchesMinVersion('3.0.0'),
|
||||||
allTagsFiltering: matchesMinVersion('3.0.0'),
|
allTagsFiltering: matchesMinVersion('3.0.0'),
|
||||||
|
tagsStats: matchesMinVersion('3.0.0'),
|
||||||
domainVisits: matchesMinVersion('3.1.0'),
|
domainVisits: matchesMinVersion('3.1.0'),
|
||||||
excludeBotsOnShortUrls: matchesMinVersion('3.4.0'),
|
excludeBotsOnShortUrls: matchesMinVersion('3.4.0'),
|
||||||
filterDisabledUrls: matchesMinVersion('3.4.0'),
|
filterDisabledUrls: matchesMinVersion('3.4.0'),
|
||||||
deviceLongUrls: matchesMinVersion('3.5.0'),
|
deviceLongUrls: matchesMinVersion('3.5.0'),
|
||||||
} as const;
|
} as const satisfies Record<string, ReturnType<typeof matchesMinVersion>>;
|
||||||
|
|
||||||
Object.freeze(supportedFeatures);
|
Object.freeze(supportedFeatures);
|
||||||
|
|
||||||
|
|
|
@ -202,7 +202,7 @@ describe('ShlinkApiClient', () => {
|
||||||
});
|
});
|
||||||
const { listTags } = buildApiClient();
|
const { listTags } = buildApiClient();
|
||||||
|
|
||||||
const result = await listTags();
|
const result = await listTags(false);
|
||||||
|
|
||||||
expect({ tags: expectedTags }).toEqual(result);
|
expect({ tags: expectedTags }).toEqual(result);
|
||||||
expect(fetchJson).toHaveBeenCalledWith(
|
expect(fetchJson).toHaveBeenCalledWith(
|
||||||
|
@ -210,6 +210,26 @@ describe('ShlinkApiClient', () => {
|
||||||
expect.objectContaining({ method: 'GET' }),
|
expect.objectContaining({ method: 'GET' }),
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('can use /tags/stats endpoint', async () => {
|
||||||
|
const expectedTags = ['foo', 'bar'];
|
||||||
|
const expectedStats = expectedTags.map((tag) => ({ tag, shortUrlsCount: 10, visitsCount: 10 }));
|
||||||
|
|
||||||
|
fetchJson.mockResolvedValue({
|
||||||
|
tags: {
|
||||||
|
data: expectedStats,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
const { listTags } = buildApiClient();
|
||||||
|
|
||||||
|
const result = await listTags(true);
|
||||||
|
|
||||||
|
expect({ tags: expectedTags, stats: expectedStats }).toEqual(result);
|
||||||
|
expect(fetchJson).toHaveBeenCalledWith(
|
||||||
|
expect.stringContaining('/tags/stats'),
|
||||||
|
expect.objectContaining({ method: 'GET' }),
|
||||||
|
);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('deleteTags', () => {
|
describe('deleteTags', () => {
|
||||||
|
|
Loading…
Add table
Reference in a new issue