mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2024-12-24 01:48:18 +03:00
Use APi v3 by default, and fall back to v2 in case of not found errors
This commit is contained in:
parent
da6d45a72c
commit
d64abeecdc
4 changed files with 60 additions and 14 deletions
|
@ -1,5 +1,5 @@
|
||||||
import { isEmpty, isNil, reject } from 'ramda';
|
import { isEmpty, isNil, reject } from 'ramda';
|
||||||
import { AxiosInstance, AxiosResponse, Method } from 'axios';
|
import { AxiosError, AxiosInstance, AxiosResponse, Method } from 'axios';
|
||||||
import { ShortUrl, ShortUrlData } from '../../short-urls/data';
|
import { ShortUrl, ShortUrlData } from '../../short-urls/data';
|
||||||
import { OptionalString } from '../../utils/utils';
|
import { OptionalString } from '../../utils/utils';
|
||||||
import {
|
import {
|
||||||
|
@ -17,10 +17,12 @@ import {
|
||||||
ShlinkDomainRedirects,
|
ShlinkDomainRedirects,
|
||||||
ShlinkShortUrlsListParams,
|
ShlinkShortUrlsListParams,
|
||||||
ShlinkShortUrlsListNormalizedParams,
|
ShlinkShortUrlsListNormalizedParams,
|
||||||
|
ProblemDetailsError,
|
||||||
} from '../types';
|
} from '../types';
|
||||||
import { orderToString } from '../../utils/helpers/ordering';
|
import { orderToString } from '../../utils/helpers/ordering';
|
||||||
|
import { isRegularNotFound } from '../utils';
|
||||||
|
|
||||||
const buildShlinkBaseUrl = (url: string) => (url ? `${url}/rest/v2` : '');
|
const buildShlinkBaseUrl = (url: string, version: 2 | 3) => `${url}/rest/v${version}`;
|
||||||
const rejectNilProps = reject(isNil);
|
const rejectNilProps = reject(isNil);
|
||||||
const normalizeOrderByInParams = (params: ShlinkShortUrlsListParams): ShlinkShortUrlsListNormalizedParams => {
|
const normalizeOrderByInParams = (params: ShlinkShortUrlsListParams): ShlinkShortUrlsListNormalizedParams => {
|
||||||
const { orderBy = {}, ...rest } = params;
|
const { orderBy = {}, ...rest } = params;
|
||||||
|
@ -29,11 +31,14 @@ const normalizeOrderByInParams = (params: ShlinkShortUrlsListParams): ShlinkShor
|
||||||
};
|
};
|
||||||
|
|
||||||
export class ShlinkApiClient {
|
export class ShlinkApiClient {
|
||||||
|
private apiVersion: 2 | 3;
|
||||||
|
|
||||||
public constructor(
|
public constructor(
|
||||||
private readonly axios: AxiosInstance,
|
private readonly axios: AxiosInstance,
|
||||||
private readonly baseUrl: string,
|
private readonly baseUrl: string,
|
||||||
private readonly apiKey: string,
|
private readonly apiKey: string,
|
||||||
) {
|
) {
|
||||||
|
this.apiVersion = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
public readonly listShortUrls = async (params: ShlinkShortUrlsListParams = {}): Promise<ShlinkShortUrlsResponse> =>
|
public readonly listShortUrls = async (params: ShlinkShortUrlsListParams = {}): Promise<ShlinkShortUrlsResponse> =>
|
||||||
|
@ -118,10 +123,19 @@ export class ShlinkApiClient {
|
||||||
private readonly performRequest = async <T>(url: string, method: Method = 'GET', query = {}, body = {}): Promise<AxiosResponse<T>> =>
|
private readonly performRequest = async <T>(url: string, method: Method = 'GET', query = {}, body = {}): Promise<AxiosResponse<T>> =>
|
||||||
this.axios({
|
this.axios({
|
||||||
method,
|
method,
|
||||||
url: `${buildShlinkBaseUrl(this.baseUrl)}${url}`,
|
url: `${buildShlinkBaseUrl(this.baseUrl, this.apiVersion)}${url}`,
|
||||||
headers: { 'X-Api-Key': this.apiKey },
|
headers: { 'X-Api-Key': this.apiKey },
|
||||||
params: rejectNilProps(query),
|
params: rejectNilProps(query),
|
||||||
data: body,
|
data: body,
|
||||||
paramsSerializer: { indexes: false },
|
paramsSerializer: { indexes: false },
|
||||||
|
}).catch((e: AxiosError<ProblemDetailsError>) => {
|
||||||
|
if (!isRegularNotFound(e.response?.data)) {
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we capture a not found error, let's assume this Shlink version does not support API v3, so we decrease to
|
||||||
|
// v2 and retry
|
||||||
|
this.apiVersion = 2;
|
||||||
|
return this.performRequest(url, method, query, body);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -120,3 +120,8 @@ export interface InvalidShortUrlDeletion extends ProblemDetailsError {
|
||||||
type: 'INVALID_SHORTCODE_DELETION' | 'INVALID_SHORT_URL_DELETION';
|
type: 'INVALID_SHORTCODE_DELETION' | 'INVALID_SHORT_URL_DELETION';
|
||||||
threshold: number;
|
threshold: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface RegularNotFound extends ProblemDetailsError {
|
||||||
|
type: 'NOT_FOUND';
|
||||||
|
status: 404;
|
||||||
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { AxiosError } from 'axios';
|
import { AxiosError } from 'axios';
|
||||||
import { InvalidArgumentError, InvalidShortUrlDeletion, ProblemDetailsError } from '../types';
|
import { InvalidArgumentError, InvalidShortUrlDeletion, ProblemDetailsError, RegularNotFound } from '../types';
|
||||||
|
|
||||||
export const parseApiError = (e: AxiosError<ProblemDetailsError>) => e.response?.data;
|
export const parseApiError = (e: AxiosError<ProblemDetailsError>) => e.response?.data;
|
||||||
|
|
||||||
|
@ -8,3 +8,6 @@ export const isInvalidArgumentError = (error?: ProblemDetailsError): error is In
|
||||||
|
|
||||||
export const isInvalidDeletionError = (error?: ProblemDetailsError): error is InvalidShortUrlDeletion =>
|
export const isInvalidDeletionError = (error?: ProblemDetailsError): error is InvalidShortUrlDeletion =>
|
||||||
error?.type === 'INVALID_SHORTCODE_DELETION' || error?.type === 'INVALID_SHORT_URL_DELETION';
|
error?.type === 'INVALID_SHORTCODE_DELETION' || error?.type === 'INVALID_SHORT_URL_DELETION';
|
||||||
|
|
||||||
|
export const isRegularNotFound = (error?: ProblemDetailsError): error is RegularNotFound =>
|
||||||
|
error?.type === 'NOT_FOUND' && error?.status === 404;
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import { AxiosInstance, AxiosRequestConfig } from 'axios';
|
import { AxiosError, AxiosInstance, AxiosRequestConfig } from 'axios';
|
||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
|
import { ShlinkApiClient } from '../../../src/api/services/ShlinkApiClient';
|
||||||
import { OptionalString } from '../../../src/utils/utils';
|
import { OptionalString } from '../../../src/utils/utils';
|
||||||
|
@ -87,7 +87,7 @@ describe('ShlinkApiClient', () => {
|
||||||
|
|
||||||
expect({ data: expectedVisits }).toEqual(actualVisits);
|
expect({ data: expectedVisits }).toEqual(actualVisits);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
||||||
url: '/short-urls/abc123/visits',
|
url: expect.stringContaining('/short-urls/abc123/visits'),
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
@ -109,7 +109,7 @@ describe('ShlinkApiClient', () => {
|
||||||
|
|
||||||
expect({ data: expectedVisits }).toEqual(actualVisits);
|
expect({ data: expectedVisits }).toEqual(actualVisits);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
||||||
url: '/tags/foo/visits',
|
url: expect.stringContaining('/tags/foo/visits'),
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
@ -131,7 +131,7 @@ describe('ShlinkApiClient', () => {
|
||||||
|
|
||||||
expect({ data: expectedVisits }).toEqual(actualVisits);
|
expect({ data: expectedVisits }).toEqual(actualVisits);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
||||||
url: '/domains/foo.com/visits',
|
url: expect.stringContaining('/domains/foo.com/visits'),
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
@ -149,7 +149,7 @@ describe('ShlinkApiClient', () => {
|
||||||
|
|
||||||
expect(expectedShortUrl).toEqual(result);
|
expect(expectedShortUrl).toEqual(result);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
||||||
url: `/short-urls/${shortCode}`,
|
url: expect.stringContaining(`/short-urls/${shortCode}`),
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
params: domain ? { domain } : {},
|
params: domain ? { domain } : {},
|
||||||
}));
|
}));
|
||||||
|
@ -170,7 +170,7 @@ describe('ShlinkApiClient', () => {
|
||||||
|
|
||||||
expect(expectedResp).toEqual(result);
|
expect(expectedResp).toEqual(result);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
||||||
url: `/short-urls/${shortCode}`,
|
url: expect.stringContaining(`/short-urls/${shortCode}`),
|
||||||
method: 'PATCH',
|
method: 'PATCH',
|
||||||
params: domain ? { domain } : {},
|
params: domain ? { domain } : {},
|
||||||
}));
|
}));
|
||||||
|
@ -190,7 +190,10 @@ describe('ShlinkApiClient', () => {
|
||||||
const result = await listTags();
|
const result = await listTags();
|
||||||
|
|
||||||
expect({ tags: expectedTags }).toEqual(result);
|
expect({ tags: expectedTags }).toEqual(result);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({ url: '/tags', method: 'GET' }));
|
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
||||||
|
url: expect.stringContaining('/tags'),
|
||||||
|
method: 'GET',
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -203,7 +206,7 @@ describe('ShlinkApiClient', () => {
|
||||||
await deleteTags(tags);
|
await deleteTags(tags);
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
||||||
url: '/tags',
|
url: expect.stringContaining('/tags'),
|
||||||
method: 'DELETE',
|
method: 'DELETE',
|
||||||
params: { tags },
|
params: { tags },
|
||||||
}));
|
}));
|
||||||
|
@ -220,7 +223,7 @@ describe('ShlinkApiClient', () => {
|
||||||
await editTag(oldName, newName);
|
await editTag(oldName, newName);
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
||||||
url: '/tags',
|
url: expect.stringContaining('/tags'),
|
||||||
method: 'PUT',
|
method: 'PUT',
|
||||||
data: { oldName, newName },
|
data: { oldName, newName },
|
||||||
}));
|
}));
|
||||||
|
@ -235,7 +238,7 @@ describe('ShlinkApiClient', () => {
|
||||||
await deleteShortUrl(shortCode, domain);
|
await deleteShortUrl(shortCode, domain);
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
||||||
url: `/short-urls/${shortCode}`,
|
url: expect.stringContaining(`/short-urls/${shortCode}`),
|
||||||
method: 'DELETE',
|
method: 'DELETE',
|
||||||
params: domain ? { domain } : {},
|
params: domain ? { domain } : {},
|
||||||
}));
|
}));
|
||||||
|
@ -341,5 +344,26 @@ describe('ShlinkApiClient', () => {
|
||||||
expect(axiosSpy).toHaveBeenCalled();
|
expect(axiosSpy).toHaveBeenCalled();
|
||||||
expect(result).toEqual(resp);
|
expect(result).toEqual(resp);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('retries request if API version is not supported', async () => {
|
||||||
|
const axiosSpy = jest.fn()
|
||||||
|
.mockImplementationOnce(() => Promise.reject(Mock.of<AxiosError>({
|
||||||
|
response: {
|
||||||
|
data: { type: 'NOT_FOUND', status: 404 },
|
||||||
|
},
|
||||||
|
})))
|
||||||
|
.mockImplementation(createAxios({})) as unknown as AxiosInstance;
|
||||||
|
const { editDomainRedirects } = new ShlinkApiClient(axiosSpy, '', '');
|
||||||
|
|
||||||
|
await editDomainRedirects({ domain: 'foo' });
|
||||||
|
|
||||||
|
expect(axiosSpy).toHaveBeenCalledTimes(2);
|
||||||
|
expect(axiosSpy).toHaveBeenNthCalledWith(1, expect.objectContaining({
|
||||||
|
url: expect.stringContaining('/v3/'),
|
||||||
|
}));
|
||||||
|
expect(axiosSpy).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
||||||
|
url: expect.stringContaining('/v2/'),
|
||||||
|
}));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in a new issue