mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2024-12-23 09:30:31 +03:00
commit
a0767417b3
24 changed files with 286 additions and 343 deletions
55
package-lock.json
generated
55
package-lock.json
generated
|
@ -13,7 +13,6 @@
|
||||||
"@fortawesome/free-solid-svg-icons": "^6.2.0",
|
"@fortawesome/free-solid-svg-icons": "^6.2.0",
|
||||||
"@fortawesome/react-fontawesome": "^0.2.0",
|
"@fortawesome/react-fontawesome": "^0.2.0",
|
||||||
"@reduxjs/toolkit": "^1.9.0",
|
"@reduxjs/toolkit": "^1.9.0",
|
||||||
"axios": "^1.1.2",
|
|
||||||
"bootstrap": "^5.2.2",
|
"bootstrap": "^5.2.2",
|
||||||
"bottlejs": "^2.0.1",
|
"bottlejs": "^2.0.1",
|
||||||
"bowser": "^2.11.0",
|
"bowser": "^2.11.0",
|
||||||
|
@ -7141,7 +7140,8 @@
|
||||||
"node_modules/asynckit": {
|
"node_modules/asynckit": {
|
||||||
"version": "0.4.0",
|
"version": "0.4.0",
|
||||||
"resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
|
"resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
|
||||||
"integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k="
|
"integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"node_modules/at-least-node": {
|
"node_modules/at-least-node": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
|
@ -7194,16 +7194,6 @@
|
||||||
"node": ">=4"
|
"node": ">=4"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/axios": {
|
|
||||||
"version": "1.1.2",
|
|
||||||
"resolved": "https://registry.npmjs.org/axios/-/axios-1.1.2.tgz",
|
|
||||||
"integrity": "sha512-bznQyETwElsXl2RK7HLLwb5GPpOLlycxHCtrpDR/4RqqBzjARaOTo3jz4IgtntWUYee7Ne4S8UHd92VCuzPaWA==",
|
|
||||||
"dependencies": {
|
|
||||||
"follow-redirects": "^1.15.0",
|
|
||||||
"form-data": "^4.0.0",
|
|
||||||
"proxy-from-env": "^1.1.0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/axobject-query": {
|
"node_modules/axobject-query": {
|
||||||
"version": "2.2.0",
|
"version": "2.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-2.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-2.2.0.tgz",
|
||||||
|
@ -8468,6 +8458,7 @@
|
||||||
"version": "1.0.8",
|
"version": "1.0.8",
|
||||||
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
|
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
|
||||||
"integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
|
"integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
|
||||||
|
"dev": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"delayed-stream": "~1.0.0"
|
"delayed-stream": "~1.0.0"
|
||||||
},
|
},
|
||||||
|
@ -9085,6 +9076,7 @@
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
|
||||||
"integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=",
|
"integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=",
|
||||||
|
"dev": true,
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.4.0"
|
"node": ">=0.4.0"
|
||||||
}
|
}
|
||||||
|
@ -11251,6 +11243,7 @@
|
||||||
"version": "1.15.2",
|
"version": "1.15.2",
|
||||||
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz",
|
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz",
|
||||||
"integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==",
|
"integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==",
|
||||||
|
"dev": true,
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
"type": "individual",
|
"type": "individual",
|
||||||
|
@ -11422,6 +11415,7 @@
|
||||||
"version": "4.0.0",
|
"version": "4.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
|
||||||
"integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
|
"integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
|
||||||
|
"dev": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"asynckit": "^0.4.0",
|
"asynckit": "^0.4.0",
|
||||||
"combined-stream": "^1.0.8",
|
"combined-stream": "^1.0.8",
|
||||||
|
@ -17388,6 +17382,7 @@
|
||||||
"version": "1.52.0",
|
"version": "1.52.0",
|
||||||
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
|
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
|
||||||
"integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
|
"integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
|
||||||
|
"dev": true,
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 0.6"
|
"node": ">= 0.6"
|
||||||
}
|
}
|
||||||
|
@ -17396,6 +17391,7 @@
|
||||||
"version": "2.1.35",
|
"version": "2.1.35",
|
||||||
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
|
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
|
||||||
"integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
|
"integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
|
||||||
|
"dev": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"mime-db": "1.52.0"
|
"mime-db": "1.52.0"
|
||||||
},
|
},
|
||||||
|
@ -19182,11 +19178,6 @@
|
||||||
"node": ">= 0.10"
|
"node": ">= 0.10"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/proxy-from-env": {
|
|
||||||
"version": "1.1.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
|
|
||||||
"integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="
|
|
||||||
},
|
|
||||||
"node_modules/psl": {
|
"node_modules/psl": {
|
||||||
"version": "1.8.0",
|
"version": "1.8.0",
|
||||||
"resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz",
|
"resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz",
|
||||||
|
@ -32124,7 +32115,8 @@
|
||||||
"asynckit": {
|
"asynckit": {
|
||||||
"version": "0.4.0",
|
"version": "0.4.0",
|
||||||
"resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
|
"resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
|
||||||
"integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k="
|
"integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"at-least-node": {
|
"at-least-node": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
|
@ -32152,16 +32144,6 @@
|
||||||
"integrity": "sha512-gd1kmb21kwNuWr6BQz8fv6GNECPBnUasepcoLbekws23NVBLODdsClRZ+bQ8+9Uomf3Sm3+Vwn0oYG9NvwnJCw==",
|
"integrity": "sha512-gd1kmb21kwNuWr6BQz8fv6GNECPBnUasepcoLbekws23NVBLODdsClRZ+bQ8+9Uomf3Sm3+Vwn0oYG9NvwnJCw==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"axios": {
|
|
||||||
"version": "1.1.2",
|
|
||||||
"resolved": "https://registry.npmjs.org/axios/-/axios-1.1.2.tgz",
|
|
||||||
"integrity": "sha512-bznQyETwElsXl2RK7HLLwb5GPpOLlycxHCtrpDR/4RqqBzjARaOTo3jz4IgtntWUYee7Ne4S8UHd92VCuzPaWA==",
|
|
||||||
"requires": {
|
|
||||||
"follow-redirects": "^1.15.0",
|
|
||||||
"form-data": "^4.0.0",
|
|
||||||
"proxy-from-env": "^1.1.0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"axobject-query": {
|
"axobject-query": {
|
||||||
"version": "2.2.0",
|
"version": "2.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-2.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-2.2.0.tgz",
|
||||||
|
@ -33100,6 +33082,7 @@
|
||||||
"version": "1.0.8",
|
"version": "1.0.8",
|
||||||
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
|
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
|
||||||
"integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
|
"integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
|
||||||
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"delayed-stream": "~1.0.0"
|
"delayed-stream": "~1.0.0"
|
||||||
}
|
}
|
||||||
|
@ -33579,7 +33562,8 @@
|
||||||
"delayed-stream": {
|
"delayed-stream": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
|
||||||
"integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk="
|
"integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"depd": {
|
"depd": {
|
||||||
"version": "1.1.2",
|
"version": "1.1.2",
|
||||||
|
@ -35246,7 +35230,8 @@
|
||||||
"follow-redirects": {
|
"follow-redirects": {
|
||||||
"version": "1.15.2",
|
"version": "1.15.2",
|
||||||
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz",
|
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz",
|
||||||
"integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA=="
|
"integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"fork-ts-checker-webpack-plugin": {
|
"fork-ts-checker-webpack-plugin": {
|
||||||
"version": "6.5.2",
|
"version": "6.5.2",
|
||||||
|
@ -35351,6 +35336,7 @@
|
||||||
"version": "4.0.0",
|
"version": "4.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
|
||||||
"integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
|
"integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
|
||||||
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"asynckit": "^0.4.0",
|
"asynckit": "^0.4.0",
|
||||||
"combined-stream": "^1.0.8",
|
"combined-stream": "^1.0.8",
|
||||||
|
@ -39845,12 +39831,14 @@
|
||||||
"mime-db": {
|
"mime-db": {
|
||||||
"version": "1.52.0",
|
"version": "1.52.0",
|
||||||
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
|
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
|
||||||
"integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg=="
|
"integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"mime-types": {
|
"mime-types": {
|
||||||
"version": "2.1.35",
|
"version": "2.1.35",
|
||||||
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
|
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
|
||||||
"integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
|
"integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
|
||||||
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"mime-db": "1.52.0"
|
"mime-db": "1.52.0"
|
||||||
}
|
}
|
||||||
|
@ -41140,11 +41128,6 @@
|
||||||
"ipaddr.js": "1.9.1"
|
"ipaddr.js": "1.9.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"proxy-from-env": {
|
|
||||||
"version": "1.1.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
|
|
||||||
"integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="
|
|
||||||
},
|
|
||||||
"psl": {
|
"psl": {
|
||||||
"version": "1.8.0",
|
"version": "1.8.0",
|
||||||
"resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz",
|
"resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz",
|
||||||
|
|
|
@ -30,7 +30,6 @@
|
||||||
"@fortawesome/free-solid-svg-icons": "^6.2.0",
|
"@fortawesome/free-solid-svg-icons": "^6.2.0",
|
||||||
"@fortawesome/react-fontawesome": "^0.2.0",
|
"@fortawesome/react-fontawesome": "^0.2.0",
|
||||||
"@reduxjs/toolkit": "^1.9.0",
|
"@reduxjs/toolkit": "^1.9.0",
|
||||||
"axios": "^1.1.2",
|
|
||||||
"bootstrap": "^5.2.2",
|
"bootstrap": "^5.2.2",
|
||||||
"bottlejs": "^2.0.1",
|
"bottlejs": "^2.0.1",
|
||||||
"bowser": "^2.11.0",
|
"bowser": "^2.11.0",
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
import { isEmpty, isNil, reject } from 'ramda';
|
import { isEmpty, isNil, reject } from 'ramda';
|
||||||
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 {
|
||||||
|
@ -20,7 +19,8 @@ import {
|
||||||
} from '../types';
|
} from '../types';
|
||||||
import { orderToString } from '../../utils/helpers/ordering';
|
import { orderToString } from '../../utils/helpers/ordering';
|
||||||
import { isRegularNotFound, parseApiError } from '../utils';
|
import { isRegularNotFound, parseApiError } from '../utils';
|
||||||
import { ProblemDetailsError } from '../types/errors';
|
import { stringifyQuery } from '../../utils/helpers/query';
|
||||||
|
import { JsonFetch } from '../../utils/types';
|
||||||
|
|
||||||
const buildShlinkBaseUrl = (url: string, version: 2 | 3) => `${url}/rest/v${version}`;
|
const buildShlinkBaseUrl = (url: string, version: 2 | 3) => `${url}/rest/v${version}`;
|
||||||
const rejectNilProps = reject(isNil);
|
const rejectNilProps = reject(isNil);
|
||||||
|
@ -34,7 +34,7 @@ export class ShlinkApiClient {
|
||||||
private apiVersion: 2 | 3;
|
private apiVersion: 2 | 3;
|
||||||
|
|
||||||
public constructor(
|
public constructor(
|
||||||
private readonly axios: AxiosInstance,
|
private readonly fetch: JsonFetch,
|
||||||
private readonly baseUrl: string,
|
private readonly baseUrl: string,
|
||||||
private readonly apiKey: string,
|
private readonly apiKey: string,
|
||||||
) {
|
) {
|
||||||
|
@ -43,42 +43,40 @@ export class ShlinkApiClient {
|
||||||
|
|
||||||
public readonly listShortUrls = async (params: ShlinkShortUrlsListParams = {}): Promise<ShlinkShortUrlsResponse> =>
|
public readonly listShortUrls = async (params: ShlinkShortUrlsListParams = {}): Promise<ShlinkShortUrlsResponse> =>
|
||||||
this.performRequest<{ shortUrls: ShlinkShortUrlsResponse }>('/short-urls', 'GET', normalizeOrderByInParams(params))
|
this.performRequest<{ shortUrls: ShlinkShortUrlsResponse }>('/short-urls', 'GET', normalizeOrderByInParams(params))
|
||||||
.then(({ data }) => data.shortUrls);
|
.then(({ shortUrls }) => shortUrls);
|
||||||
|
|
||||||
public readonly createShortUrl = async (options: ShortUrlData): Promise<ShortUrl> => {
|
public readonly createShortUrl = async (options: ShortUrlData): Promise<ShortUrl> => {
|
||||||
const filteredOptions = reject((value) => isEmpty(value) || isNil(value), options as any);
|
const filteredOptions = reject((value) => isEmpty(value) || isNil(value), options as any);
|
||||||
|
|
||||||
return this.performRequest<ShortUrl>('/short-urls', 'POST', {}, filteredOptions)
|
return this.performRequest<ShortUrl>('/short-urls', 'POST', {}, filteredOptions);
|
||||||
.then((resp) => resp.data);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public readonly getShortUrlVisits = async (shortCode: string, query?: ShlinkVisitsParams): Promise<ShlinkVisits> =>
|
public readonly getShortUrlVisits = async (shortCode: string, query?: ShlinkVisitsParams): Promise<ShlinkVisits> =>
|
||||||
this.performRequest<{ visits: ShlinkVisits }>(`/short-urls/${shortCode}/visits`, 'GET', query)
|
this.performRequest<{ visits: ShlinkVisits }>(`/short-urls/${shortCode}/visits`, 'GET', query)
|
||||||
.then(({ data }) => data.visits);
|
.then(({ visits }) => visits);
|
||||||
|
|
||||||
public readonly getTagVisits = async (tag: string, query?: Omit<ShlinkVisitsParams, 'domain'>): Promise<ShlinkVisits> =>
|
public readonly getTagVisits = async (tag: string, query?: Omit<ShlinkVisitsParams, 'domain'>): Promise<ShlinkVisits> =>
|
||||||
this.performRequest<{ visits: ShlinkVisits }>(`/tags/${tag}/visits`, 'GET', query)
|
this.performRequest<{ visits: ShlinkVisits }>(`/tags/${tag}/visits`, 'GET', query)
|
||||||
.then(({ data }) => data.visits);
|
.then(({ visits }) => visits);
|
||||||
|
|
||||||
public readonly getDomainVisits = async (domain: string, query?: Omit<ShlinkVisitsParams, 'domain'>): Promise<ShlinkVisits> =>
|
public readonly getDomainVisits = async (domain: string, query?: Omit<ShlinkVisitsParams, 'domain'>): Promise<ShlinkVisits> =>
|
||||||
this.performRequest<{ visits: ShlinkVisits }>(`/domains/${domain}/visits`, 'GET', query)
|
this.performRequest<{ visits: ShlinkVisits }>(`/domains/${domain}/visits`, 'GET', query)
|
||||||
.then(({ data }) => data.visits);
|
.then(({ visits }) => visits);
|
||||||
|
|
||||||
public readonly getOrphanVisits = async (query?: Omit<ShlinkVisitsParams, 'domain'>): Promise<ShlinkVisits> =>
|
public readonly getOrphanVisits = async (query?: Omit<ShlinkVisitsParams, 'domain'>): Promise<ShlinkVisits> =>
|
||||||
this.performRequest<{ visits: ShlinkVisits }>('/visits/orphan', 'GET', query)
|
this.performRequest<{ visits: ShlinkVisits }>('/visits/orphan', 'GET', query)
|
||||||
.then(({ data }) => data.visits);
|
.then(({ visits }) => visits);
|
||||||
|
|
||||||
public readonly getNonOrphanVisits = async (query?: Omit<ShlinkVisitsParams, 'domain'>): Promise<ShlinkVisits> =>
|
public readonly getNonOrphanVisits = async (query?: Omit<ShlinkVisitsParams, 'domain'>): Promise<ShlinkVisits> =>
|
||||||
this.performRequest<{ visits: ShlinkVisits }>('/visits/non-orphan', 'GET', query)
|
this.performRequest<{ visits: ShlinkVisits }>('/visits/non-orphan', 'GET', query)
|
||||||
.then(({ data }) => data.visits);
|
.then(({ visits }) => visits);
|
||||||
|
|
||||||
public readonly getVisitsOverview = async (): Promise<ShlinkVisitsOverview> =>
|
public readonly getVisitsOverview = async (): Promise<ShlinkVisitsOverview> =>
|
||||||
this.performRequest<{ visits: ShlinkVisitsOverview }>('/visits', 'GET')
|
this.performRequest<{ visits: ShlinkVisitsOverview }>('/visits')
|
||||||
.then(({ data }) => data.visits);
|
.then(({ visits }) => visits);
|
||||||
|
|
||||||
public readonly getShortUrl = async (shortCode: string, domain?: OptionalString): Promise<ShortUrl> =>
|
public readonly getShortUrl = async (shortCode: string, domain?: OptionalString): Promise<ShortUrl> =>
|
||||||
this.performRequest<ShortUrl>(`/short-urls/${shortCode}`, 'GET', { domain })
|
this.performRequest<ShortUrl>(`/short-urls/${shortCode}`, 'GET', { domain });
|
||||||
.then(({ data }) => data);
|
|
||||||
|
|
||||||
public readonly deleteShortUrl = async (shortCode: string, domain?: OptionalString): Promise<void> =>
|
public readonly deleteShortUrl = async (shortCode: string, domain?: OptionalString): Promise<void> =>
|
||||||
this.performRequest(`/short-urls/${shortCode}`, 'DELETE', { domain })
|
this.performRequest(`/short-urls/${shortCode}`, 'DELETE', { domain })
|
||||||
|
@ -89,11 +87,11 @@ export class ShlinkApiClient {
|
||||||
domain: OptionalString,
|
domain: OptionalString,
|
||||||
edit: ShlinkShortUrlData,
|
edit: ShlinkShortUrlData,
|
||||||
): Promise<ShortUrl> =>
|
): Promise<ShortUrl> =>
|
||||||
this.performRequest<ShortUrl>(`/short-urls/${shortCode}`, 'PATCH', { domain }, edit).then(({ data }) => data);
|
this.performRequest<ShortUrl>(`/short-urls/${shortCode}`, 'PATCH', { domain }, edit);
|
||||||
|
|
||||||
public readonly listTags = async (): Promise<ShlinkTags> =>
|
public readonly listTags = async (): Promise<ShlinkTags> =>
|
||||||
this.performRequest<{ tags: ShlinkTagsResponse }>('/tags', 'GET', { withStats: 'true' })
|
this.performRequest<{ tags: ShlinkTagsResponse }>('/tags', 'GET', { withStats: 'true' })
|
||||||
.then((resp) => resp.data.tags)
|
.then(({ tags }) => tags)
|
||||||
.then(({ data, stats }) => ({ tags: data, stats }));
|
.then(({ data, stats }) => ({ tags: data, stats }));
|
||||||
|
|
||||||
public readonly deleteTags = async (tags: string[]): Promise<{ tags: string[] }> =>
|
public readonly deleteTags = async (tags: string[]): Promise<{ tags: string[] }> =>
|
||||||
|
@ -104,31 +102,28 @@ export class ShlinkApiClient {
|
||||||
this.performRequest('/tags', 'PUT', {}, { oldName, newName })
|
this.performRequest('/tags', 'PUT', {}, { oldName, newName })
|
||||||
.then(() => ({ oldName, newName }));
|
.then(() => ({ oldName, newName }));
|
||||||
|
|
||||||
public readonly health = async (): Promise<ShlinkHealth> =>
|
public readonly health = async (): Promise<ShlinkHealth> => this.performRequest<ShlinkHealth>('/health', 'GET');
|
||||||
this.performRequest<ShlinkHealth>('/health', 'GET')
|
|
||||||
.then((resp) => resp.data);
|
|
||||||
|
|
||||||
public readonly mercureInfo = async (): Promise<ShlinkMercureInfo> =>
|
public readonly mercureInfo = async (): Promise<ShlinkMercureInfo> =>
|
||||||
this.performRequest<ShlinkMercureInfo>('/mercure-info', 'GET')
|
this.performRequest<ShlinkMercureInfo>('/mercure-info', 'GET');
|
||||||
.then((resp) => resp.data);
|
|
||||||
|
|
||||||
public readonly listDomains = async (): Promise<ShlinkDomainsResponse> =>
|
public readonly listDomains = async (): Promise<ShlinkDomainsResponse> =>
|
||||||
this.performRequest<{ domains: ShlinkDomainsResponse }>('/domains', 'GET').then(({ data }) => data.domains);
|
this.performRequest<{ domains: ShlinkDomainsResponse }>('/domains').then(({ domains }) => domains);
|
||||||
|
|
||||||
public readonly editDomainRedirects = async (
|
public readonly editDomainRedirects = async (
|
||||||
domainRedirects: ShlinkEditDomainRedirects,
|
domainRedirects: ShlinkEditDomainRedirects,
|
||||||
): Promise<ShlinkDomainRedirects> =>
|
): Promise<ShlinkDomainRedirects> =>
|
||||||
this.performRequest<ShlinkDomainRedirects>('/domains/redirects', 'PATCH', {}, domainRedirects).then(({ data }) => data);
|
this.performRequest<ShlinkDomainRedirects>('/domains/redirects', 'PATCH', {}, domainRedirects);
|
||||||
|
|
||||||
private readonly performRequest = async <T>(url: string, method: Method = 'GET', query = {}, body = {}): Promise<AxiosResponse<T>> =>
|
private readonly performRequest = async <T>(url: string, method = 'GET', query = {}, body?: object): Promise<T> => {
|
||||||
this.axios({
|
const normalizedQuery = stringifyQuery(rejectNilProps(query));
|
||||||
|
const stringifiedQuery = isEmpty(normalizedQuery) ? '' : `?${normalizedQuery}`;
|
||||||
|
|
||||||
|
return this.fetch<T>(`${buildShlinkBaseUrl(this.baseUrl, this.apiVersion)}${url}${stringifiedQuery}`, {
|
||||||
method,
|
method,
|
||||||
url: `${buildShlinkBaseUrl(this.baseUrl, this.apiVersion)}${url}`,
|
body: body && JSON.stringify(body),
|
||||||
headers: { 'X-Api-Key': this.apiKey },
|
headers: { 'X-Api-Key': this.apiKey },
|
||||||
params: rejectNilProps(query),
|
}).catch((e: unknown) => {
|
||||||
data: body,
|
|
||||||
paramsSerializer: { indexes: false },
|
|
||||||
}).catch((e: AxiosError<ProblemDetailsError>) => {
|
|
||||||
if (!isRegularNotFound(parseApiError(e))) {
|
if (!isRegularNotFound(parseApiError(e))) {
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
|
@ -138,4 +133,5 @@ export class ShlinkApiClient {
|
||||||
this.apiVersion = 2;
|
this.apiVersion = 2;
|
||||||
return this.performRequest(url, method, query, body);
|
return this.performRequest(url, method, query, body);
|
||||||
});
|
});
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { AxiosInstance } from 'axios';
|
|
||||||
import { hasServerData, ServerWithId } from '../../servers/data';
|
import { hasServerData, ServerWithId } from '../../servers/data';
|
||||||
import { GetState } from '../../container/types';
|
import { GetState } from '../../container/types';
|
||||||
import { ShlinkApiClient } from './ShlinkApiClient';
|
import { ShlinkApiClient } from './ShlinkApiClient';
|
||||||
|
import { JsonFetch } from '../../utils/types';
|
||||||
|
|
||||||
const apiClients: Record<string, ShlinkApiClient> = {};
|
const apiClients: Record<string, ShlinkApiClient> = {};
|
||||||
|
|
||||||
|
@ -16,14 +16,14 @@ const getSelectedServerFromState = (getState: GetState): ServerWithId => {
|
||||||
return selectedServer;
|
return selectedServer;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const buildShlinkApiClient = (axios: AxiosInstance) => (getStateOrSelectedServer: GetState | ServerWithId) => {
|
export const buildShlinkApiClient = (fetch: JsonFetch) => (getStateOrSelectedServer: GetState | ServerWithId) => {
|
||||||
const { url, apiKey } = isGetState(getStateOrSelectedServer)
|
const { url, apiKey } = isGetState(getStateOrSelectedServer)
|
||||||
? getSelectedServerFromState(getStateOrSelectedServer)
|
? getSelectedServerFromState(getStateOrSelectedServer)
|
||||||
: getStateOrSelectedServer;
|
: getStateOrSelectedServer;
|
||||||
const clientKey = `${url}_${apiKey}`;
|
const clientKey = `${url}_${apiKey}`;
|
||||||
|
|
||||||
if (!apiClients[clientKey]) {
|
if (!apiClients[clientKey]) {
|
||||||
apiClients[clientKey] = new ShlinkApiClient(axios, url, apiKey);
|
apiClients[clientKey] = new ShlinkApiClient(fetch, url, apiKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
return apiClients[clientKey];
|
return apiClients[clientKey];
|
||||||
|
|
|
@ -2,7 +2,7 @@ import Bottle from 'bottlejs';
|
||||||
import { buildShlinkApiClient } from './ShlinkApiClientBuilder';
|
import { buildShlinkApiClient } from './ShlinkApiClientBuilder';
|
||||||
|
|
||||||
const provideServices = (bottle: Bottle) => {
|
const provideServices = (bottle: Bottle) => {
|
||||||
bottle.serviceFactory('buildShlinkApiClient', buildShlinkApiClient, 'axios');
|
bottle.serviceFactory('buildShlinkApiClient', buildShlinkApiClient, 'jsonFetch');
|
||||||
};
|
};
|
||||||
|
|
||||||
export default provideServices;
|
export default provideServices;
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
import { AxiosError } from 'axios';
|
|
||||||
import {
|
import {
|
||||||
ErrorTypeV2,
|
ErrorTypeV2,
|
||||||
ErrorTypeV3,
|
ErrorTypeV3,
|
||||||
|
@ -8,11 +7,10 @@ import {
|
||||||
RegularNotFound,
|
RegularNotFound,
|
||||||
} from '../types/errors';
|
} from '../types/errors';
|
||||||
|
|
||||||
const isAxiosError = (e: unknown): e is AxiosError<ProblemDetailsError> => !!e && typeof e === 'object' && 'response' in e;
|
const isProblemDetails = (e: unknown): e is ProblemDetailsError =>
|
||||||
|
!!e && typeof e === 'object' && ['type', 'detail', 'title', 'status'].every((prop) => prop in e);
|
||||||
|
|
||||||
export const parseApiError = (e: unknown): ProblemDetailsError | undefined => (
|
export const parseApiError = (e: unknown): ProblemDetailsError | undefined => (isProblemDetails(e) ? e : undefined);
|
||||||
isAxiosError(e) ? e.response?.data : undefined
|
|
||||||
);
|
|
||||||
|
|
||||||
export const isInvalidArgumentError = (error?: ProblemDetailsError): error is InvalidArgumentError =>
|
export const isInvalidArgumentError = (error?: ProblemDetailsError): error is InvalidArgumentError =>
|
||||||
error?.type === ErrorTypeV2.INVALID_ARGUMENT || error?.type === ErrorTypeV3.INVALID_ARGUMENT;
|
error?.type === ErrorTypeV2.INVALID_ARGUMENT || error?.type === ErrorTypeV3.INVALID_ARGUMENT;
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
import { AxiosInstance } from 'axios';
|
import { Fetch } from '../../utils/types';
|
||||||
import { saveUrl } from '../../utils/helpers/files';
|
import { saveUrl } from '../../utils/helpers/files';
|
||||||
|
|
||||||
export class ImageDownloader {
|
export class ImageDownloader {
|
||||||
public constructor(private readonly axios: AxiosInstance, private readonly window: Window) {}
|
public constructor(private readonly fetch: Fetch, private readonly window: Window) {}
|
||||||
|
|
||||||
public async saveImage(imgUrl: string, filename: string): Promise<void> {
|
public async saveImage(imgUrl: string, filename: string): Promise<void> {
|
||||||
const { data } = await this.axios.get(imgUrl, { responseType: 'blob' });
|
const data = await this.fetch(imgUrl).then((resp) => resp.blob());
|
||||||
const url = URL.createObjectURL(data);
|
const url = URL.createObjectURL(data);
|
||||||
|
|
||||||
saveUrl(this.window, url, filename);
|
saveUrl(this.window, url, filename);
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
import axios from 'axios';
|
|
||||||
import Bottle from 'bottlejs';
|
import Bottle from 'bottlejs';
|
||||||
import { ScrollToTop } from '../ScrollToTop';
|
import { ScrollToTop } from '../ScrollToTop';
|
||||||
import { MainHeader } from '../MainHeader';
|
import { MainHeader } from '../MainHeader';
|
||||||
|
@ -12,14 +11,16 @@ import { withoutSelectedServer } from '../../servers/helpers/withoutSelectedServ
|
||||||
import { sidebarNotPresent, sidebarPresent } from '../reducers/sidebar';
|
import { sidebarNotPresent, sidebarPresent } from '../reducers/sidebar';
|
||||||
import { ImageDownloader } from './ImageDownloader';
|
import { ImageDownloader } from './ImageDownloader';
|
||||||
import { ReportExporter } from './ReportExporter';
|
import { ReportExporter } from './ReportExporter';
|
||||||
|
import { jsonFetch } from '../../utils/helpers/fetch';
|
||||||
|
|
||||||
const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
||||||
// Services
|
// Services
|
||||||
bottle.constant('window', (global as any).window);
|
bottle.constant('window', (global as any).window);
|
||||||
bottle.constant('console', global.console);
|
bottle.constant('console', global.console);
|
||||||
bottle.constant('axios', axios);
|
bottle.constant('fetch', (global as any).fetch.bind(global));
|
||||||
|
bottle.serviceFactory('jsonFetch', jsonFetch, 'fetch');
|
||||||
|
|
||||||
bottle.service('ImageDownloader', ImageDownloader, 'axios', 'window');
|
bottle.service('ImageDownloader', ImageDownloader, 'fetch', 'window');
|
||||||
bottle.service('ReportExporter', ReportExporter, 'window', 'jsonToCsv');
|
bottle.service('ReportExporter', ReportExporter, 'window', 'jsonToCsv');
|
||||||
|
|
||||||
// Components
|
// Components
|
||||||
|
|
|
@ -1,19 +1,15 @@
|
||||||
import { pipe, prop } from 'ramda';
|
|
||||||
import { AxiosInstance } from 'axios';
|
|
||||||
import pack from '../../../package.json';
|
import pack from '../../../package.json';
|
||||||
import { hasServerData, ServerData } from '../data';
|
import { hasServerData, ServerData } from '../data';
|
||||||
import { createServers } from './servers';
|
import { createServers } from './servers';
|
||||||
import { createAsyncThunk } from '../../utils/helpers/redux';
|
import { createAsyncThunk } from '../../utils/helpers/redux';
|
||||||
|
import { JsonFetch } from '../../utils/types';
|
||||||
|
|
||||||
const responseToServersList = pipe(
|
const responseToServersList = (data: any): ServerData[] => (Array.isArray(data) ? data.filter(hasServerData) : []);
|
||||||
prop<any, any>('data'),
|
|
||||||
(data: any): ServerData[] => (Array.isArray(data) ? data.filter(hasServerData) : []),
|
|
||||||
);
|
|
||||||
|
|
||||||
export const fetchServers = ({ get }: AxiosInstance) => createAsyncThunk(
|
export const fetchServers = (fetch: JsonFetch) => createAsyncThunk(
|
||||||
'shlink/remoteServers/fetchServers',
|
'shlink/remoteServers/fetchServers',
|
||||||
async (_: void, { dispatch }): Promise<void> => {
|
async (_: void, { dispatch }): Promise<void> => {
|
||||||
const resp = await get(`${pack.homepage}/servers.json`);
|
const resp = await fetch<any>(`${pack.homepage}/servers.json`);
|
||||||
const result = responseToServersList(resp);
|
const result = responseToServersList(resp);
|
||||||
|
|
||||||
dispatch(createServers(result));
|
dispatch(createServers(result));
|
||||||
|
|
|
@ -80,7 +80,7 @@ const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
||||||
bottle.serviceFactory('deleteServer', () => deleteServer);
|
bottle.serviceFactory('deleteServer', () => deleteServer);
|
||||||
bottle.serviceFactory('editServer', () => editServer);
|
bottle.serviceFactory('editServer', () => editServer);
|
||||||
bottle.serviceFactory('setAutoConnect', () => setAutoConnect);
|
bottle.serviceFactory('setAutoConnect', () => setAutoConnect);
|
||||||
bottle.serviceFactory('fetchServers', fetchServers, 'axios');
|
bottle.serviceFactory('fetchServers', fetchServers, 'jsonFetch');
|
||||||
|
|
||||||
bottle.serviceFactory('resetSelectedServer', () => resetSelectedServer);
|
bottle.serviceFactory('resetSelectedServer', () => resetSelectedServer);
|
||||||
|
|
||||||
|
|
10
src/utils/helpers/fetch.ts
Normal file
10
src/utils/helpers/fetch.ts
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
export const jsonFetch = (fetch: typeof window.fetch) => <T>(url: string, options?: RequestInit) => fetch(url, options)
|
||||||
|
.then(async (resp) => {
|
||||||
|
const parsed = await resp.json();
|
||||||
|
|
||||||
|
if (!resp.ok) {
|
||||||
|
throw parsed; // eslint-disable-line @typescript-eslint/no-throw-literal
|
||||||
|
}
|
||||||
|
|
||||||
|
return parsed as T;
|
||||||
|
});
|
|
@ -1 +1,5 @@
|
||||||
export type MediaMatcher = (query: string) => MediaQueryList;
|
export type MediaMatcher = (query: string) => MediaQueryList;
|
||||||
|
|
||||||
|
export type Fetch = typeof window.fetch;
|
||||||
|
|
||||||
|
export type JsonFetch = <T>(url: string, options?: RequestInit) => Promise<T>;
|
||||||
|
|
|
@ -20,20 +20,20 @@ type VisitsLoader = (page: number, itemsPerPage: number) => Promise<ShlinkVisits
|
||||||
type LastVisitLoader = () => Promise<Visit | undefined>;
|
type LastVisitLoader = () => Promise<Visit | undefined>;
|
||||||
|
|
||||||
interface VisitsAsyncThunkOptions<T extends LoadVisits = LoadVisits, R extends VisitsLoaded = VisitsLoaded> {
|
interface VisitsAsyncThunkOptions<T extends LoadVisits = LoadVisits, R extends VisitsLoaded = VisitsLoaded> {
|
||||||
name: string;
|
typePrefix: string;
|
||||||
createLoaders: (params: T, getState: () => ShlinkState) => [VisitsLoader, LastVisitLoader];
|
createLoaders: (params: T, getState: () => ShlinkState) => [VisitsLoader, LastVisitLoader];
|
||||||
getExtraFulfilledPayload: (params: T) => Partial<R>;
|
getExtraFulfilledPayload: (params: T) => Partial<R>;
|
||||||
shouldCancel: (getState: () => ShlinkState) => boolean;
|
shouldCancel: (getState: () => ShlinkState) => boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const createVisitsAsyncThunk = <T extends LoadVisits = LoadVisits, R extends VisitsLoaded = VisitsLoaded>(
|
export const createVisitsAsyncThunk = <T extends LoadVisits = LoadVisits, R extends VisitsLoaded = VisitsLoaded>(
|
||||||
{ name, createLoaders, getExtraFulfilledPayload, shouldCancel }: VisitsAsyncThunkOptions<T, R>,
|
{ typePrefix, createLoaders, getExtraFulfilledPayload, shouldCancel }: VisitsAsyncThunkOptions<T, R>,
|
||||||
) => {
|
) => {
|
||||||
const progressChangedAction = createAction<number>(`${name}/progressChanged`);
|
const progressChangedAction = createAction<number>(`${typePrefix}/progressChanged`);
|
||||||
const largeAction = createAction<void>(`${name}/large`);
|
const largeAction = createAction<void>(`${typePrefix}/large`);
|
||||||
const fallbackToIntervalAction = createAction<DateInterval>(`${name}/fallbackToInterval`);
|
const fallbackToIntervalAction = createAction<DateInterval>(`${typePrefix}/fallbackToInterval`);
|
||||||
|
|
||||||
const asyncThunk = createAsyncThunk(name, async (params: T, { getState, dispatch }): Promise<R> => {
|
const asyncThunk = createAsyncThunk(typePrefix, async (params: T, { getState, dispatch }): Promise<R> => {
|
||||||
const [visitsLoader, lastVisitLoader] = createLoaders(params, getState);
|
const [visitsLoader, lastVisitLoader] = createLoaders(params, getState);
|
||||||
|
|
||||||
const loadVisitsInParallel = async (pages: number[]): Promise<Visit[]> =>
|
const loadVisitsInParallel = async (pages: number[]): Promise<Visit[]> =>
|
||||||
|
@ -97,11 +97,15 @@ export const lastVisitLoaderForLoader = (
|
||||||
return async () => loader({ page: 1, itemsPerPage: 1 }).then(({ data }) => data[0]);
|
return async () => loader({ page: 1, itemsPerPage: 1 }).then(({ data }) => data[0]);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
interface VisitsReducerOptions<State extends VisitsInfo, AT extends ReturnType<typeof createVisitsAsyncThunk>> {
|
||||||
|
name: string;
|
||||||
|
asyncThunkCreator: AT;
|
||||||
|
initialState: State;
|
||||||
|
filterCreatedVisits: (state: State, createdVisits: CreateVisit[]) => CreateVisit[];
|
||||||
|
}
|
||||||
|
|
||||||
export const createVisitsReducer = <State extends VisitsInfo, AT extends ReturnType<typeof createVisitsAsyncThunk>>(
|
export const createVisitsReducer = <State extends VisitsInfo, AT extends ReturnType<typeof createVisitsAsyncThunk>>(
|
||||||
name: string,
|
{ name, asyncThunkCreator, initialState, filterCreatedVisits }: VisitsReducerOptions<State, AT>,
|
||||||
asyncThunkCreator: AT,
|
|
||||||
initialState: State,
|
|
||||||
filterCreatedVisits: (state: State, createdVisits: CreateVisit[]) => CreateVisit[],
|
|
||||||
) => {
|
) => {
|
||||||
const { asyncThunk, largeAction, fallbackToIntervalAction, progressChangedAction } = asyncThunkCreator;
|
const { asyncThunk, largeAction, fallbackToIntervalAction, progressChangedAction } = asyncThunkCreator;
|
||||||
const { reducer, actions } = createSlice({
|
const { reducer, actions } = createSlice({
|
||||||
|
@ -127,10 +131,10 @@ export const createVisitsReducer = <State extends VisitsInfo, AT extends ReturnT
|
||||||
|
|
||||||
builder.addCase(createNewVisits, (state, { payload }) => {
|
builder.addCase(createNewVisits, (state, { payload }) => {
|
||||||
const { visits } = state;
|
const { visits } = state;
|
||||||
// @ts-expect-error TODO Fix the state inferred type
|
// @ts-expect-error TODO Fix type inference
|
||||||
const newVisits = filterCreatedVisits(state, payload.createdVisits).map(({ visit }) => visit);
|
const newVisits = filterCreatedVisits(state, payload.createdVisits).map(({ visit }) => visit);
|
||||||
|
|
||||||
return { ...state, visits: [...newVisits, ...visits] };
|
return !newVisits.length ? state : { ...state, visits: [...newVisits, ...visits] };
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -27,7 +27,7 @@ const initialState: DomainVisits = {
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getDomainVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
export const getDomainVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
||||||
name: `${REDUCER_PREFIX}/getDomainVisits`,
|
typePrefix: `${REDUCER_PREFIX}/getDomainVisits`,
|
||||||
createLoaders: ({ domain, query = {}, doIntervalFallback = false }: LoadDomainVisits, getState) => {
|
createLoaders: ({ domain, query = {}, doIntervalFallback = false }: LoadDomainVisits, getState) => {
|
||||||
const { getDomainVisits: getVisits } = buildShlinkApiClient(getState);
|
const { getDomainVisits: getVisits } = buildShlinkApiClient(getState);
|
||||||
const visitsLoader = async (page: number, itemsPerPage: number) => getVisits(
|
const visitsLoader = async (page: number, itemsPerPage: number) => getVisits(
|
||||||
|
@ -43,17 +43,17 @@ export const getDomainVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) =>
|
||||||
});
|
});
|
||||||
|
|
||||||
export const domainVisitsReducerCreator = (
|
export const domainVisitsReducerCreator = (
|
||||||
getVisitsCreator: ReturnType<typeof getDomainVisits>,
|
asyncThunkCreator: ReturnType<typeof getDomainVisits>,
|
||||||
) => createVisitsReducer(
|
) => createVisitsReducer({
|
||||||
REDUCER_PREFIX,
|
name: REDUCER_PREFIX,
|
||||||
// @ts-expect-error TODO Fix type inference
|
|
||||||
getVisitsCreator,
|
|
||||||
initialState,
|
initialState,
|
||||||
({ domain, query = {} }, createdVisits) => {
|
// @ts-expect-error TODO Fix type inference
|
||||||
|
asyncThunkCreator,
|
||||||
|
filterCreatedVisits: ({ domain, query = {} }, createdVisits) => {
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
return createdVisits.filter(
|
return createdVisits.filter(
|
||||||
({ shortUrl, visit }) =>
|
({ shortUrl, visit }) =>
|
||||||
shortUrl && domainMatches(shortUrl, domain) && isBetween(visit.date, startDate, endDate),
|
shortUrl && domainMatches(shortUrl, domain) && isBetween(visit.date, startDate, endDate),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
});
|
||||||
|
|
|
@ -15,7 +15,7 @@ const initialState: VisitsInfo = {
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getNonOrphanVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
export const getNonOrphanVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
||||||
name: `${REDUCER_PREFIX}/getNonOrphanVisits`,
|
typePrefix: `${REDUCER_PREFIX}/getNonOrphanVisits`,
|
||||||
createLoaders: ({ query = {}, doIntervalFallback = false }, getState) => {
|
createLoaders: ({ query = {}, doIntervalFallback = false }, getState) => {
|
||||||
const { getNonOrphanVisits: shlinkGetNonOrphanVisits } = buildShlinkApiClient(getState);
|
const { getNonOrphanVisits: shlinkGetNonOrphanVisits } = buildShlinkApiClient(getState);
|
||||||
const visitsLoader = async (page: number, itemsPerPage: number) =>
|
const visitsLoader = async (page: number, itemsPerPage: number) =>
|
||||||
|
@ -29,13 +29,13 @@ export const getNonOrphanVisits = (buildShlinkApiClient: ShlinkApiClientBuilder)
|
||||||
});
|
});
|
||||||
|
|
||||||
export const nonOrphanVisitsReducerCreator = (
|
export const nonOrphanVisitsReducerCreator = (
|
||||||
getVisitsCreator: ReturnType<typeof getNonOrphanVisits>,
|
asyncThunkCreator: ReturnType<typeof getNonOrphanVisits>,
|
||||||
) => createVisitsReducer(
|
) => createVisitsReducer({
|
||||||
REDUCER_PREFIX,
|
name: REDUCER_PREFIX,
|
||||||
getVisitsCreator,
|
|
||||||
initialState,
|
initialState,
|
||||||
({ query = {} }, createdVisits) => {
|
asyncThunkCreator,
|
||||||
|
filterCreatedVisits: ({ query = {} }, createdVisits) => {
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
return createdVisits.filter(({ visit }) => isBetween(visit.date, startDate, endDate));
|
return createdVisits.filter(({ visit }) => isBetween(visit.date, startDate, endDate));
|
||||||
},
|
},
|
||||||
);
|
});
|
||||||
|
|
|
@ -24,13 +24,12 @@ const matchesType = (visit: OrphanVisit, orphanVisitsType?: OrphanVisitType) =>
|
||||||
!orphanVisitsType || orphanVisitsType === visit.type;
|
!orphanVisitsType || orphanVisitsType === visit.type;
|
||||||
|
|
||||||
export const getOrphanVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
export const getOrphanVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
||||||
name: `${REDUCER_PREFIX}/getOrphanVisits`,
|
typePrefix: `${REDUCER_PREFIX}/getOrphanVisits`,
|
||||||
createLoaders: ({ orphanVisitsType, query = {}, doIntervalFallback = false }: LoadOrphanVisits, getState) => {
|
createLoaders: ({ orphanVisitsType, query = {}, doIntervalFallback = false }: LoadOrphanVisits, getState) => {
|
||||||
const { getOrphanVisits: getVisits } = buildShlinkApiClient(getState);
|
const { getOrphanVisits: getVisits } = buildShlinkApiClient(getState);
|
||||||
const visitsLoader = async (page: number, itemsPerPage: number) => getVisits({ ...query, page, itemsPerPage })
|
const visitsLoader = async (page: number, itemsPerPage: number) => getVisits({ ...query, page, itemsPerPage })
|
||||||
.then((result) => {
|
.then((result) => {
|
||||||
const visits = result.data.filter((visit) => isOrphanVisit(visit) && matchesType(visit, orphanVisitsType));
|
const visits = result.data.filter((visit) => isOrphanVisit(visit) && matchesType(visit, orphanVisitsType));
|
||||||
|
|
||||||
return { ...result, data: visits };
|
return { ...result, data: visits };
|
||||||
});
|
});
|
||||||
const lastVisitLoader = lastVisitLoaderForLoader(doIntervalFallback, getVisits);
|
const lastVisitLoader = lastVisitLoaderForLoader(doIntervalFallback, getVisits);
|
||||||
|
@ -42,13 +41,13 @@ export const getOrphanVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) =>
|
||||||
});
|
});
|
||||||
|
|
||||||
export const orphanVisitsReducerCreator = (
|
export const orphanVisitsReducerCreator = (
|
||||||
getVisitsCreator: ReturnType<typeof getOrphanVisits>,
|
asyncThunkCreator: ReturnType<typeof getOrphanVisits>,
|
||||||
) => createVisitsReducer(
|
) => createVisitsReducer({
|
||||||
REDUCER_PREFIX,
|
name: REDUCER_PREFIX,
|
||||||
getVisitsCreator,
|
|
||||||
initialState,
|
initialState,
|
||||||
({ query = {} }, createdVisits) => {
|
asyncThunkCreator,
|
||||||
|
filterCreatedVisits: ({ query = {} }, createdVisits) => {
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
return createdVisits.filter(({ visit, shortUrl }) => !shortUrl && isBetween(visit.date, startDate, endDate));
|
return createdVisits.filter(({ visit, shortUrl }) => !shortUrl && isBetween(visit.date, startDate, endDate));
|
||||||
},
|
},
|
||||||
);
|
});
|
||||||
|
|
|
@ -25,7 +25,7 @@ const initialState: ShortUrlVisits = {
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getShortUrlVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
export const getShortUrlVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
||||||
name: `${REDUCER_PREFIX}/getShortUrlVisits`,
|
typePrefix: `${REDUCER_PREFIX}/getShortUrlVisits`,
|
||||||
createLoaders: ({ shortCode, query = {}, doIntervalFallback = false }: LoadShortUrlVisits, getState) => {
|
createLoaders: ({ shortCode, query = {}, doIntervalFallback = false }: LoadShortUrlVisits, getState) => {
|
||||||
const { getShortUrlVisits: shlinkGetShortUrlVisits } = buildShlinkApiClient(getState);
|
const { getShortUrlVisits: shlinkGetShortUrlVisits } = buildShlinkApiClient(getState);
|
||||||
const visitsLoader = async (page: number, itemsPerPage: number) => shlinkGetShortUrlVisits(
|
const visitsLoader = async (page: number, itemsPerPage: number) => shlinkGetShortUrlVisits(
|
||||||
|
@ -46,17 +46,17 @@ export const getShortUrlVisits = (buildShlinkApiClient: ShlinkApiClientBuilder)
|
||||||
});
|
});
|
||||||
|
|
||||||
export const shortUrlVisitsReducerCreator = (
|
export const shortUrlVisitsReducerCreator = (
|
||||||
getVisitsCreator: ReturnType<typeof getShortUrlVisits>,
|
asyncThunkCreator: ReturnType<typeof getShortUrlVisits>,
|
||||||
) => createVisitsReducer(
|
) => createVisitsReducer({
|
||||||
REDUCER_PREFIX,
|
name: REDUCER_PREFIX,
|
||||||
// @ts-expect-error TODO Fix type inference
|
|
||||||
getVisitsCreator,
|
|
||||||
initialState,
|
initialState,
|
||||||
({ shortCode, domain, query = {} }, createdVisits) => {
|
// @ts-expect-error TODO Fix type inference
|
||||||
|
asyncThunkCreator,
|
||||||
|
filterCreatedVisits: ({ shortCode, domain, query = {} }: ShortUrlVisits, createdVisits) => {
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
return createdVisits.filter(
|
return createdVisits.filter(
|
||||||
({ shortUrl, visit }) =>
|
({ shortUrl, visit }) =>
|
||||||
shortUrl && shortUrlMatches(shortUrl, shortCode, domain) && isBetween(visit.date, startDate, endDate),
|
shortUrl && shortUrlMatches(shortUrl, shortCode, domain) && isBetween(visit.date, startDate, endDate),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
});
|
||||||
|
|
|
@ -24,7 +24,7 @@ const initialState: TagVisits = {
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getTagVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
export const getTagVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => createVisitsAsyncThunk({
|
||||||
name: `${REDUCER_PREFIX}/getTagVisits`,
|
typePrefix: `${REDUCER_PREFIX}/getTagVisits`,
|
||||||
createLoaders: ({ tag, query = {}, doIntervalFallback = false }: LoadTagVisits, getState) => {
|
createLoaders: ({ tag, query = {}, doIntervalFallback = false }: LoadTagVisits, getState) => {
|
||||||
const { getTagVisits: getVisits } = buildShlinkApiClient(getState);
|
const { getTagVisits: getVisits } = buildShlinkApiClient(getState);
|
||||||
const visitsLoader = async (page: number, itemsPerPage: number) => getVisits(
|
const visitsLoader = async (page: number, itemsPerPage: number) => getVisits(
|
||||||
|
@ -39,15 +39,15 @@ export const getTagVisits = (buildShlinkApiClient: ShlinkApiClientBuilder) => cr
|
||||||
shouldCancel: (getState) => getState().tagVisits.cancelLoad,
|
shouldCancel: (getState) => getState().tagVisits.cancelLoad,
|
||||||
});
|
});
|
||||||
|
|
||||||
export const tagVisitsReducerCreator = (getTagVisitsCreator: ReturnType<typeof getTagVisits>) => createVisitsReducer(
|
export const tagVisitsReducerCreator = (asyncThunkCreator: ReturnType<typeof getTagVisits>) => createVisitsReducer({
|
||||||
REDUCER_PREFIX,
|
name: REDUCER_PREFIX,
|
||||||
// @ts-expect-error TODO Fix type inference
|
|
||||||
getTagVisitsCreator,
|
|
||||||
initialState,
|
initialState,
|
||||||
({ tag, query = {} }, createdVisits) => {
|
// @ts-expect-error TODO Fix type inference
|
||||||
|
asyncThunkCreator,
|
||||||
|
filterCreatedVisits: ({ tag, query = {} }: TagVisits, createdVisits) => {
|
||||||
const { startDate, endDate } = query;
|
const { startDate, endDate } = query;
|
||||||
return createdVisits.filter(
|
return createdVisits.filter(
|
||||||
({ shortUrl, visit }) => shortUrl?.tags.includes(tag) && isBetween(visit.date, startDate, endDate),
|
({ shortUrl, visit }) => shortUrl?.tags.includes(tag) && isBetween(visit.date, startDate, endDate),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
});
|
||||||
|
|
|
@ -1,16 +1,16 @@
|
||||||
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';
|
||||||
import { ShlinkDomain, ShlinkVisitsOverview } from '../../../src/api/types';
|
import { ShlinkDomain, ShlinkVisits, ShlinkVisitsOverview } from '../../../src/api/types';
|
||||||
import { ShortUrl, ShortUrlsOrder } from '../../../src/short-urls/data';
|
import { ShortUrl, ShortUrlsOrder } from '../../../src/short-urls/data';
|
||||||
import { Visit } from '../../../src/visits/types';
|
import { JsonFetch } from '../../../src/utils/types';
|
||||||
|
import { ErrorTypeV2, ErrorTypeV3 } from '../../../src/api/types/errors';
|
||||||
|
|
||||||
describe('ShlinkApiClient', () => {
|
describe('ShlinkApiClient', () => {
|
||||||
const createAxios = (data: AxiosRequestConfig) => (async () => Promise.resolve(data)) as unknown as AxiosInstance;
|
const buildFetch = (data: any) => jest.fn().mockResolvedValue(data);
|
||||||
const createAxiosMock = (data: AxiosRequestConfig = {}) => jest.fn(createAxios(data)) as unknown as AxiosInstance;
|
const buildRejectedFetch = (error: any) => jest.fn().mockRejectedValueOnce(error);
|
||||||
const createApiClient = (data: AxiosRequestConfig) => new ShlinkApiClient(createAxios(data), '', '');
|
const buildApiClient = (fetch: JsonFetch) => new ShlinkApiClient(fetch, '', '');
|
||||||
const shortCodesWithDomainCombinations: [ string, OptionalString ][] = [
|
const shortCodesWithDomainCombinations: [string, OptionalString][] = [
|
||||||
['abc123', null],
|
['abc123', null],
|
||||||
['abc123', undefined],
|
['abc123', undefined],
|
||||||
['abc123', 'example.com'],
|
['abc123', 'example.com'],
|
||||||
|
@ -20,11 +20,9 @@ describe('ShlinkApiClient', () => {
|
||||||
const expectedList = ['foo', 'bar'];
|
const expectedList = ['foo', 'bar'];
|
||||||
|
|
||||||
it('properly returns short URLs list', async () => {
|
it('properly returns short URLs list', async () => {
|
||||||
const { listShortUrls } = createApiClient({
|
const { listShortUrls } = buildApiClient(buildFetch({
|
||||||
data: {
|
shortUrls: expectedList,
|
||||||
shortUrls: expectedList,
|
}));
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const actualList = await listShortUrls();
|
const actualList = await listShortUrls();
|
||||||
|
|
||||||
|
@ -32,20 +30,16 @@ describe('ShlinkApiClient', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it.each([
|
it.each([
|
||||||
[{ field: 'visits', dir: 'DESC' } as ShortUrlsOrder, 'visits-DESC'],
|
[{ field: 'visits', dir: 'DESC' } as ShortUrlsOrder, '?orderBy=visits-DESC'],
|
||||||
[{ field: 'longUrl', dir: 'ASC' } as ShortUrlsOrder, 'longUrl-ASC'],
|
[{ field: 'longUrl', dir: 'ASC' } as ShortUrlsOrder, '?orderBy=longUrl-ASC'],
|
||||||
[{ field: 'longUrl', dir: undefined } as ShortUrlsOrder, undefined],
|
[{ field: 'longUrl', dir: undefined } as ShortUrlsOrder, ''],
|
||||||
])('parses orderBy in params', async (orderBy, expectedOrderBy) => {
|
])('parses orderBy in params', async (orderBy, expectedOrderBy) => {
|
||||||
const axiosSpy = createAxiosMock({
|
const fetch = buildFetch({ data: expectedList });
|
||||||
data: expectedList,
|
const { listShortUrls } = buildApiClient(fetch);
|
||||||
});
|
|
||||||
const { listShortUrls } = new ShlinkApiClient(axiosSpy, '', '');
|
|
||||||
|
|
||||||
await listShortUrls({ orderBy });
|
await listShortUrls({ orderBy });
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(expect.stringContaining(`/short-urls${expectedOrderBy}`), expect.anything());
|
||||||
params: { orderBy: expectedOrderBy },
|
|
||||||
}));
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -55,39 +49,38 @@ describe('ShlinkApiClient', () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
it('returns create short URL', async () => {
|
it('returns create short URL', async () => {
|
||||||
const { createShortUrl } = createApiClient({ data: shortUrl });
|
const { createShortUrl } = buildApiClient(buildFetch(shortUrl));
|
||||||
const result = await createShortUrl({ longUrl: '' });
|
const result = await createShortUrl({ longUrl: '' });
|
||||||
|
|
||||||
expect(result).toEqual(shortUrl);
|
expect(result).toEqual(shortUrl);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('removes all empty options', async () => {
|
it('removes all empty options', async () => {
|
||||||
const axiosSpy = createAxiosMock({ data: shortUrl });
|
const fetch = buildFetch({ data: shortUrl });
|
||||||
const { createShortUrl } = new ShlinkApiClient(axiosSpy, '', '');
|
const { createShortUrl } = buildApiClient(fetch);
|
||||||
|
|
||||||
await createShortUrl({ longUrl: 'bar', customSlug: undefined, maxVisits: null });
|
await createShortUrl({ longUrl: 'bar', customSlug: undefined, maxVisits: null });
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({ data: { longUrl: 'bar' } }));
|
expect(fetch).toHaveBeenCalledWith(expect.anything(), expect.objectContaining({
|
||||||
|
body: JSON.stringify({ longUrl: 'bar' }),
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('getShortUrlVisits', () => {
|
describe('getShortUrlVisits', () => {
|
||||||
it('properly returns short URL visits', async () => {
|
it('properly returns short URL visits', async () => {
|
||||||
const expectedVisits = ['foo', 'bar'];
|
const expectedVisits = ['foo', 'bar'];
|
||||||
const axiosSpy = createAxiosMock({
|
const fetch = buildFetch({
|
||||||
data: {
|
visits: {
|
||||||
visits: {
|
data: expectedVisits,
|
||||||
data: expectedVisits,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
const { getShortUrlVisits } = new ShlinkApiClient(axiosSpy, '', '');
|
const { getShortUrlVisits } = buildApiClient(fetch);
|
||||||
|
|
||||||
const actualVisits = await getShortUrlVisits('abc123', {});
|
const actualVisits = await getShortUrlVisits('abc123', {});
|
||||||
|
|
||||||
expect({ data: expectedVisits }).toEqual(actualVisits);
|
expect({ data: expectedVisits }).toEqual(actualVisits);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(expect.stringContaining('/short-urls/abc123/visits'), expect.objectContaining({
|
||||||
url: expect.stringContaining('/short-urls/abc123/visits'),
|
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
@ -96,20 +89,17 @@ describe('ShlinkApiClient', () => {
|
||||||
describe('getTagVisits', () => {
|
describe('getTagVisits', () => {
|
||||||
it('properly returns tag visits', async () => {
|
it('properly returns tag visits', async () => {
|
||||||
const expectedVisits = ['foo', 'bar'];
|
const expectedVisits = ['foo', 'bar'];
|
||||||
const axiosSpy = createAxiosMock({
|
const fetch = buildFetch({
|
||||||
data: {
|
visits: {
|
||||||
visits: {
|
data: expectedVisits,
|
||||||
data: expectedVisits,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
const { getTagVisits } = new ShlinkApiClient(axiosSpy, '', '');
|
const { getTagVisits } = buildApiClient(fetch);
|
||||||
|
|
||||||
const actualVisits = await getTagVisits('foo', {});
|
const actualVisits = await getTagVisits('foo', {});
|
||||||
|
|
||||||
expect({ data: expectedVisits }).toEqual(actualVisits);
|
expect({ data: expectedVisits }).toEqual(actualVisits);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(expect.stringContaining('/tags/foo/visits'), expect.objectContaining({
|
||||||
url: expect.stringContaining('/tags/foo/visits'),
|
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
@ -118,20 +108,17 @@ describe('ShlinkApiClient', () => {
|
||||||
describe('getDomainVisits', () => {
|
describe('getDomainVisits', () => {
|
||||||
it('properly returns domain visits', async () => {
|
it('properly returns domain visits', async () => {
|
||||||
const expectedVisits = ['foo', 'bar'];
|
const expectedVisits = ['foo', 'bar'];
|
||||||
const axiosSpy = createAxiosMock({
|
const fetch = buildFetch({
|
||||||
data: {
|
visits: {
|
||||||
visits: {
|
data: expectedVisits,
|
||||||
data: expectedVisits,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
const { getDomainVisits } = new ShlinkApiClient(axiosSpy, '', '');
|
const { getDomainVisits } = buildApiClient(fetch);
|
||||||
|
|
||||||
const actualVisits = await getDomainVisits('foo.com', {});
|
const actualVisits = await getDomainVisits('foo.com', {});
|
||||||
|
|
||||||
expect({ data: expectedVisits }).toEqual(actualVisits);
|
expect({ data: expectedVisits }).toEqual(actualVisits);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(expect.stringContaining('/domains/foo.com/visits'), expect.objectContaining({
|
||||||
url: expect.stringContaining('/domains/foo.com/visits'),
|
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
@ -140,19 +127,17 @@ describe('ShlinkApiClient', () => {
|
||||||
describe('getShortUrl', () => {
|
describe('getShortUrl', () => {
|
||||||
it.each(shortCodesWithDomainCombinations)('properly returns short URL', async (shortCode, domain) => {
|
it.each(shortCodesWithDomainCombinations)('properly returns short URL', async (shortCode, domain) => {
|
||||||
const expectedShortUrl = { foo: 'bar' };
|
const expectedShortUrl = { foo: 'bar' };
|
||||||
const axiosSpy = createAxiosMock({
|
const fetch = buildFetch(expectedShortUrl);
|
||||||
data: expectedShortUrl,
|
const { getShortUrl } = buildApiClient(fetch);
|
||||||
});
|
const expectedQuery = domain ? `?domain=${domain}` : '';
|
||||||
const { getShortUrl } = new ShlinkApiClient(axiosSpy, '', '');
|
|
||||||
|
|
||||||
const result = await getShortUrl(shortCode, domain);
|
const result = await getShortUrl(shortCode, domain);
|
||||||
|
|
||||||
expect(expectedShortUrl).toEqual(result);
|
expect(expectedShortUrl).toEqual(result);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(
|
||||||
url: expect.stringContaining(`/short-urls/${shortCode}`),
|
expect.stringContaining(`/short-urls/${shortCode}${expectedQuery}`),
|
||||||
method: 'GET',
|
expect.objectContaining({ method: 'GET' }),
|
||||||
params: domain ? { domain } : {},
|
);
|
||||||
}));
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -163,53 +148,49 @@ describe('ShlinkApiClient', () => {
|
||||||
validSince: '2025-01-01T10:00:00+01:00',
|
validSince: '2025-01-01T10:00:00+01:00',
|
||||||
};
|
};
|
||||||
const expectedResp = Mock.of<ShortUrl>();
|
const expectedResp = Mock.of<ShortUrl>();
|
||||||
const axiosSpy = createAxiosMock({ data: expectedResp });
|
const fetch = buildFetch(expectedResp);
|
||||||
const { updateShortUrl } = new ShlinkApiClient(axiosSpy, '', '');
|
const { updateShortUrl } = buildApiClient(fetch);
|
||||||
|
const expectedQuery = domain ? `?domain=${domain}` : '';
|
||||||
|
|
||||||
const result = await updateShortUrl(shortCode, domain, meta);
|
const result = await updateShortUrl(shortCode, domain, meta);
|
||||||
|
|
||||||
expect(expectedResp).toEqual(result);
|
expect(expectedResp).toEqual(result);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(
|
||||||
url: expect.stringContaining(`/short-urls/${shortCode}`),
|
expect.stringContaining(`/short-urls/${shortCode}${expectedQuery}`),
|
||||||
method: 'PATCH',
|
expect.objectContaining({ method: 'PATCH' }),
|
||||||
params: domain ? { domain } : {},
|
);
|
||||||
}));
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('listTags', () => {
|
describe('listTags', () => {
|
||||||
it('properly returns list of tags', async () => {
|
it('properly returns list of tags', async () => {
|
||||||
const expectedTags = ['foo', 'bar'];
|
const expectedTags = ['foo', 'bar'];
|
||||||
const axiosSpy = createAxiosMock({
|
const fetch = buildFetch({
|
||||||
data: {
|
tags: {
|
||||||
tags: { data: expectedTags },
|
data: expectedTags,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
const { listTags } = new ShlinkApiClient(axiosSpy, '', '');
|
const { listTags } = buildApiClient(fetch);
|
||||||
|
|
||||||
const result = await listTags();
|
const result = await listTags();
|
||||||
|
|
||||||
expect({ tags: expectedTags }).toEqual(result);
|
expect({ tags: expectedTags }).toEqual(result);
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(expect.stringContaining('/tags'), expect.objectContaining({ method: 'GET' }));
|
||||||
url: expect.stringContaining('/tags'),
|
|
||||||
method: 'GET',
|
|
||||||
}));
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('deleteTags', () => {
|
describe('deleteTags', () => {
|
||||||
it('properly deletes provided tags', async () => {
|
it('properly deletes provided tags', async () => {
|
||||||
const tags = ['foo', 'bar'];
|
const tags = ['foo', 'bar'];
|
||||||
const axiosSpy = createAxiosMock();
|
const fetch = buildFetch({});
|
||||||
const { deleteTags } = new ShlinkApiClient(axiosSpy, '', '');
|
const { deleteTags } = buildApiClient(fetch);
|
||||||
|
|
||||||
await deleteTags(tags);
|
await deleteTags(tags);
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(
|
||||||
url: expect.stringContaining('/tags'),
|
expect.stringContaining(`/tags?${tags.map((tag) => `tags%5B%5D=${tag}`).join('&')}`),
|
||||||
method: 'DELETE',
|
expect.objectContaining({ method: 'DELETE' }),
|
||||||
params: { tags },
|
);
|
||||||
}));
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -217,31 +198,30 @@ describe('ShlinkApiClient', () => {
|
||||||
it('properly edits provided tag', async () => {
|
it('properly edits provided tag', async () => {
|
||||||
const oldName = 'foo';
|
const oldName = 'foo';
|
||||||
const newName = 'bar';
|
const newName = 'bar';
|
||||||
const axiosSpy = createAxiosMock();
|
const fetch = buildFetch({});
|
||||||
const { editTag } = new ShlinkApiClient(axiosSpy, '', '');
|
const { editTag } = buildApiClient(fetch);
|
||||||
|
|
||||||
await editTag(oldName, newName);
|
await editTag(oldName, newName);
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(expect.stringContaining('/tags'), expect.objectContaining({
|
||||||
url: expect.stringContaining('/tags'),
|
|
||||||
method: 'PUT',
|
method: 'PUT',
|
||||||
data: { oldName, newName },
|
body: JSON.stringify({ oldName, newName }),
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('deleteShortUrl', () => {
|
describe('deleteShortUrl', () => {
|
||||||
it.each(shortCodesWithDomainCombinations)('properly deletes provided short URL', async (shortCode, domain) => {
|
it.each(shortCodesWithDomainCombinations)('properly deletes provided short URL', async (shortCode, domain) => {
|
||||||
const axiosSpy = createAxiosMock({});
|
const fetch = buildFetch({});
|
||||||
const { deleteShortUrl } = new ShlinkApiClient(axiosSpy, '', '');
|
const { deleteShortUrl } = buildApiClient(fetch);
|
||||||
|
const expectedQuery = domain ? `?domain=${domain}` : '';
|
||||||
|
|
||||||
await deleteShortUrl(shortCode, domain);
|
await deleteShortUrl(shortCode, domain);
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalledWith(expect.objectContaining({
|
expect(fetch).toHaveBeenCalledWith(
|
||||||
url: expect.stringContaining(`/short-urls/${shortCode}`),
|
expect.stringContaining(`/short-urls/${shortCode}${expectedQuery}`),
|
||||||
method: 'DELETE',
|
expect.objectContaining({ method: 'DELETE' }),
|
||||||
params: domain ? { domain } : {},
|
);
|
||||||
}));
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -251,12 +231,12 @@ describe('ShlinkApiClient', () => {
|
||||||
status: 'pass',
|
status: 'pass',
|
||||||
version: '1.19.0',
|
version: '1.19.0',
|
||||||
};
|
};
|
||||||
const axiosSpy = createAxiosMock({ data: expectedData });
|
const fetch = buildFetch(expectedData);
|
||||||
const { health } = new ShlinkApiClient(axiosSpy, '', '');
|
const { health } = buildApiClient(fetch);
|
||||||
|
|
||||||
const result = await health();
|
const result = await health();
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalled();
|
expect(fetch).toHaveBeenCalled();
|
||||||
expect(result).toEqual(expectedData);
|
expect(result).toEqual(expectedData);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -267,12 +247,12 @@ describe('ShlinkApiClient', () => {
|
||||||
token: 'abc.123.def',
|
token: 'abc.123.def',
|
||||||
mercureHubUrl: 'http://example.com/.well-known/mercure',
|
mercureHubUrl: 'http://example.com/.well-known/mercure',
|
||||||
};
|
};
|
||||||
const axiosSpy = createAxiosMock({ data: expectedData });
|
const fetch = buildFetch(expectedData);
|
||||||
const { mercureInfo } = new ShlinkApiClient(axiosSpy, '', '');
|
const { mercureInfo } = buildApiClient(fetch);
|
||||||
|
|
||||||
const result = await mercureInfo();
|
const result = await mercureInfo();
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalled();
|
expect(fetch).toHaveBeenCalled();
|
||||||
expect(result).toEqual(expectedData);
|
expect(result).toEqual(expectedData);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -280,13 +260,12 @@ describe('ShlinkApiClient', () => {
|
||||||
describe('listDomains', () => {
|
describe('listDomains', () => {
|
||||||
it('returns domains', async () => {
|
it('returns domains', async () => {
|
||||||
const expectedData = { data: [Mock.all<ShlinkDomain>(), Mock.all<ShlinkDomain>()] };
|
const expectedData = { data: [Mock.all<ShlinkDomain>(), Mock.all<ShlinkDomain>()] };
|
||||||
const resp = { domains: expectedData };
|
const fetch = buildFetch({ domains: expectedData });
|
||||||
const axiosSpy = createAxiosMock({ data: resp });
|
const { listDomains } = buildApiClient(fetch);
|
||||||
const { listDomains } = new ShlinkApiClient(axiosSpy, '', '');
|
|
||||||
|
|
||||||
const result = await listDomains();
|
const result = await listDomains();
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalled();
|
expect(fetch).toHaveBeenCalled();
|
||||||
expect(result).toEqual(expectedData);
|
expect(result).toEqual(expectedData);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -294,76 +273,67 @@ describe('ShlinkApiClient', () => {
|
||||||
describe('getVisitsOverview', () => {
|
describe('getVisitsOverview', () => {
|
||||||
it('returns visits overview', async () => {
|
it('returns visits overview', async () => {
|
||||||
const expectedData = Mock.all<ShlinkVisitsOverview>();
|
const expectedData = Mock.all<ShlinkVisitsOverview>();
|
||||||
const resp = { visits: expectedData };
|
const fetch = buildFetch({ visits: expectedData });
|
||||||
const axiosSpy = createAxiosMock({ data: resp });
|
const { getVisitsOverview } = buildApiClient(fetch);
|
||||||
const { getVisitsOverview } = new ShlinkApiClient(axiosSpy, '', '');
|
|
||||||
|
|
||||||
const result = await getVisitsOverview();
|
const result = await getVisitsOverview();
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalled();
|
expect(fetch).toHaveBeenCalled();
|
||||||
expect(result).toEqual(expectedData);
|
expect(result).toEqual(expectedData);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('getOrphanVisits', () => {
|
describe('getOrphanVisits', () => {
|
||||||
it('returns orphan visits', async () => {
|
it('returns orphan visits', async () => {
|
||||||
const expectedData: Visit[] = [];
|
const fetch = buildFetch({ visits: Mock.of<ShlinkVisits>({ data: [] }) });
|
||||||
const resp = { visits: expectedData };
|
const { getOrphanVisits } = buildApiClient(fetch);
|
||||||
const axiosSpy = createAxiosMock({ data: resp });
|
|
||||||
const { getOrphanVisits } = new ShlinkApiClient(axiosSpy, '', '');
|
|
||||||
|
|
||||||
const result = await getOrphanVisits();
|
const result = await getOrphanVisits();
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalled();
|
expect(fetch).toHaveBeenCalled();
|
||||||
expect(result).toEqual(expectedData);
|
expect(result).toEqual({ data: [] });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('getNonOrphanVisits', () => {
|
describe('getNonOrphanVisits', () => {
|
||||||
it('returns non-orphan visits', async () => {
|
it('returns non-orphan visits', async () => {
|
||||||
const expectedData: Visit[] = [];
|
const fetch = buildFetch({ visits: Mock.of<ShlinkVisits>({ data: [] }) });
|
||||||
const resp = { visits: expectedData };
|
const { getNonOrphanVisits } = buildApiClient(fetch);
|
||||||
const axiosSpy = createAxiosMock({ data: resp });
|
|
||||||
const { getNonOrphanVisits } = new ShlinkApiClient(axiosSpy, '', '');
|
|
||||||
|
|
||||||
const result = await getNonOrphanVisits();
|
const result = await getNonOrphanVisits();
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalled();
|
expect(fetch).toHaveBeenCalled();
|
||||||
expect(result).toEqual(expectedData);
|
expect(result).toEqual({ data: [] });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
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' };
|
||||||
const axiosSpy = createAxiosMock({ data: resp });
|
const fetch = buildFetch(resp);
|
||||||
const { editDomainRedirects } = new ShlinkApiClient(axiosSpy, '', '');
|
const { editDomainRedirects } = buildApiClient(fetch);
|
||||||
|
|
||||||
const result = await editDomainRedirects({ domain: 'foo' });
|
const result = await editDomainRedirects({ domain: 'foo' });
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalled();
|
expect(fetch).toHaveBeenCalled();
|
||||||
expect(result).toEqual(resp);
|
expect(result).toEqual(resp);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('retries request if API version is not supported', async () => {
|
it.each([
|
||||||
const axiosSpy = jest.fn()
|
['NOT_FOUND'],
|
||||||
.mockImplementationOnce(() => Promise.reject(Mock.of<AxiosError>({
|
[ErrorTypeV2.NOT_FOUND],
|
||||||
response: {
|
[ErrorTypeV3.NOT_FOUND],
|
||||||
data: { type: 'NOT_FOUND', status: 404 },
|
])('retries request if API version is not supported', async (type) => {
|
||||||
},
|
const fetch = buildRejectedFetch({ type, detail: 'detail', title: 'title', status: 404 }).mockImplementation(
|
||||||
})))
|
buildFetch({}),
|
||||||
.mockImplementation(createAxios({})) as unknown as AxiosInstance;
|
);
|
||||||
const { editDomainRedirects } = new ShlinkApiClient(axiosSpy, '', '');
|
const { editDomainRedirects } = buildApiClient(fetch);
|
||||||
|
|
||||||
await editDomainRedirects({ domain: 'foo' });
|
await editDomainRedirects({ domain: 'foo' });
|
||||||
|
|
||||||
expect(axiosSpy).toHaveBeenCalledTimes(2);
|
expect(fetch).toHaveBeenCalledTimes(2);
|
||||||
expect(axiosSpy).toHaveBeenNthCalledWith(1, expect.objectContaining({
|
expect(fetch).toHaveBeenNthCalledWith(1, expect.stringContaining('/v3/'), expect.anything());
|
||||||
url: expect.stringContaining('/v3/'),
|
expect(fetch).toHaveBeenNthCalledWith(2, expect.stringContaining('/v2/'), expect.anything());
|
||||||
}));
|
|
||||||
expect(axiosSpy).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
|
||||||
url: expect.stringContaining('/v2/'),
|
|
||||||
}));
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,16 +1,14 @@
|
||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import { AxiosInstance } from 'axios';
|
|
||||||
import { buildShlinkApiClient } from '../../../src/api/services/ShlinkApiClientBuilder';
|
import { buildShlinkApiClient } from '../../../src/api/services/ShlinkApiClientBuilder';
|
||||||
import { ReachableServer, SelectedServer } from '../../../src/servers/data';
|
import { ReachableServer, SelectedServer } from '../../../src/servers/data';
|
||||||
import { ShlinkState } from '../../../src/container/types';
|
import { ShlinkState } from '../../../src/container/types';
|
||||||
|
|
||||||
describe('ShlinkApiClientBuilder', () => {
|
describe('ShlinkApiClientBuilder', () => {
|
||||||
const axiosMock = Mock.all<AxiosInstance>();
|
const fetch = jest.fn();
|
||||||
const server = (data: Partial<ReachableServer>) => Mock.of<ReachableServer>(data);
|
const server = (data: Partial<ReachableServer>) => Mock.of<ReachableServer>(data);
|
||||||
|
|
||||||
const createBuilder = () => {
|
const createBuilder = () => {
|
||||||
const builder = buildShlinkApiClient(axiosMock);
|
const builder = buildShlinkApiClient(fetch);
|
||||||
|
|
||||||
return (selectedServer: SelectedServer) => builder(() => Mock.of<ShlinkState>({ selectedServer }));
|
return (selectedServer: SelectedServer) => builder(() => Mock.of<ShlinkState>({ selectedServer }));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -44,7 +42,7 @@ describe('ShlinkApiClientBuilder', () => {
|
||||||
it('does not fetch from state when provided param is already selected server', () => {
|
it('does not fetch from state when provided param is already selected server', () => {
|
||||||
const url = 'url';
|
const url = 'url';
|
||||||
const apiKey = 'apiKey';
|
const apiKey = 'apiKey';
|
||||||
const apiClient = buildShlinkApiClient(axiosMock)(server({ url, apiKey }));
|
const apiClient = buildShlinkApiClient(fetch)(server({ url, apiKey }));
|
||||||
|
|
||||||
expect(apiClient['baseUrl']).toEqual(url); // eslint-disable-line @typescript-eslint/dot-notation
|
expect(apiClient['baseUrl']).toEqual(url); // eslint-disable-line @typescript-eslint/dot-notation
|
||||||
expect(apiClient['apiKey']).toEqual(apiKey); // eslint-disable-line @typescript-eslint/dot-notation
|
expect(apiClient['apiKey']).toEqual(apiKey); // eslint-disable-line @typescript-eslint/dot-notation
|
||||||
|
|
|
@ -1,25 +1,22 @@
|
||||||
import { Mock } from 'ts-mockery';
|
|
||||||
import { AxiosInstance } from 'axios';
|
|
||||||
import { ImageDownloader } from '../../../src/common/services/ImageDownloader';
|
import { ImageDownloader } from '../../../src/common/services/ImageDownloader';
|
||||||
import { windowMock } from '../../__mocks__/Window.mock';
|
import { windowMock } from '../../__mocks__/Window.mock';
|
||||||
|
|
||||||
describe('ImageDownloader', () => {
|
describe('ImageDownloader', () => {
|
||||||
const get = jest.fn();
|
const fetch = jest.fn();
|
||||||
const axios = Mock.of<AxiosInstance>({ get });
|
|
||||||
let imageDownloader: ImageDownloader;
|
let imageDownloader: ImageDownloader;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
jest.clearAllMocks();
|
jest.clearAllMocks();
|
||||||
(global as any).URL = { createObjectURL: () => '' };
|
(global as any).URL = { createObjectURL: () => '' };
|
||||||
|
|
||||||
imageDownloader = new ImageDownloader(axios, windowMock);
|
imageDownloader = new ImageDownloader(fetch, windowMock);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('calls URL with response type blob', async () => {
|
it('calls URL with response type blob', async () => {
|
||||||
get.mockResolvedValue({ data: {} });
|
fetch.mockResolvedValue({ blob: () => new Blob() });
|
||||||
|
|
||||||
await imageDownloader.saveImage('/foo/bar.png', 'my-image.png');
|
await imageDownloader.saveImage('/foo/bar.png', 'my-image.png');
|
||||||
|
|
||||||
expect(get).toHaveBeenCalledWith('/foo/bar.png', { responseType: 'blob' });
|
expect(fetch).toHaveBeenCalledWith('/foo/bar.png');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import { AxiosError } from 'axios';
|
|
||||||
import {
|
import {
|
||||||
DomainsList,
|
DomainsList,
|
||||||
replaceRedirectsOnDomain,
|
replaceRedirectsOnDomain,
|
||||||
|
@ -25,11 +24,7 @@ describe('domainsListReducer', () => {
|
||||||
Mock.of<Domain>({ domain: 'Boo', status: 'validating' }),
|
Mock.of<Domain>({ domain: 'Boo', status: 'validating' }),
|
||||||
];
|
];
|
||||||
const domains = [...filteredDomains, Mock.of<Domain>({ domain: 'bar', status: 'validating' })];
|
const domains = [...filteredDomains, Mock.of<Domain>({ domain: 'bar', status: 'validating' })];
|
||||||
const error = Mock.of<AxiosError>({
|
const error = { type: 'NOT_FOUND', status: 404 };
|
||||||
response: {
|
|
||||||
data: { type: 'NOT_FOUND', status: 404 },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
const editDomainRedirectsThunk = editDomainRedirects(buildShlinkApiClient);
|
const editDomainRedirectsThunk = editDomainRedirects(buildShlinkApiClient);
|
||||||
const { reducer, listDomains: listDomainsAction, checkDomainHealth, filterDomains } = domainsListReducerCreator(
|
const { reducer, listDomains: listDomainsAction, checkDomainHealth, filterDomains } = domainsListReducerCreator(
|
||||||
buildShlinkApiClient,
|
buildShlinkApiClient,
|
||||||
|
|
|
@ -1,5 +1,3 @@
|
||||||
import { Mock } from 'ts-mockery';
|
|
||||||
import { AxiosInstance } from 'axios';
|
|
||||||
import { fetchServers } from '../../../src/servers/reducers/remoteServers';
|
import { fetchServers } from '../../../src/servers/reducers/remoteServers';
|
||||||
import { createServers } from '../../../src/servers/reducers/servers';
|
import { createServers } from '../../../src/servers/reducers/servers';
|
||||||
|
|
||||||
|
@ -8,27 +6,24 @@ describe('remoteServersReducer', () => {
|
||||||
|
|
||||||
describe('fetchServers', () => {
|
describe('fetchServers', () => {
|
||||||
const dispatch = jest.fn();
|
const dispatch = jest.fn();
|
||||||
const get = jest.fn();
|
const fetch = jest.fn();
|
||||||
const axios = Mock.of<AxiosInstance>({ get });
|
|
||||||
|
|
||||||
it.each([
|
it.each([
|
||||||
[
|
[
|
||||||
{
|
[
|
||||||
data: [
|
{
|
||||||
{
|
id: '111',
|
||||||
id: '111',
|
name: 'acel.me from servers.json',
|
||||||
name: 'acel.me from servers.json',
|
url: 'https://acel.me',
|
||||||
url: 'https://acel.me',
|
apiKey: '07fb8a96-8059-4094-a24c-80a7d5e7e9b0',
|
||||||
apiKey: '07fb8a96-8059-4094-a24c-80a7d5e7e9b0',
|
},
|
||||||
},
|
{
|
||||||
{
|
id: '222',
|
||||||
id: '222',
|
name: 'Local from servers.json',
|
||||||
name: 'Local from servers.json',
|
url: 'http://localhost:8000',
|
||||||
url: 'http://localhost:8000',
|
apiKey: '7a531c75-134e-4d5c-86e0-a71b7167b57a',
|
||||||
apiKey: '7a531c75-134e-4d5c-86e0-a71b7167b57a',
|
},
|
||||||
},
|
],
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
111: {
|
111: {
|
||||||
id: '111',
|
id: '111',
|
||||||
|
@ -45,26 +40,24 @@ describe('remoteServersReducer', () => {
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
{
|
[
|
||||||
data: [
|
{
|
||||||
{
|
id: '111',
|
||||||
id: '111',
|
name: 'acel.me from servers.json',
|
||||||
name: 'acel.me from servers.json',
|
url: 'https://acel.me',
|
||||||
url: 'https://acel.me',
|
apiKey: '07fb8a96-8059-4094-a24c-80a7d5e7e9b0',
|
||||||
apiKey: '07fb8a96-8059-4094-a24c-80a7d5e7e9b0',
|
},
|
||||||
},
|
{
|
||||||
{
|
id: '222',
|
||||||
id: '222',
|
name: 'Invalid',
|
||||||
name: 'Invalid',
|
},
|
||||||
},
|
{
|
||||||
{
|
id: '333',
|
||||||
id: '333',
|
name: 'Local from servers.json',
|
||||||
name: 'Local from servers.json',
|
url: 'http://localhost:8000',
|
||||||
url: 'http://localhost:8000',
|
apiKey: '7a531c75-134e-4d5c-86e0-a71b7167b57a',
|
||||||
apiKey: '7a531c75-134e-4d5c-86e0-a71b7167b57a',
|
},
|
||||||
},
|
],
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
111: {
|
111: {
|
||||||
id: '111',
|
id: '111',
|
||||||
|
@ -83,8 +76,8 @@ describe('remoteServersReducer', () => {
|
||||||
['<html></html>', {}],
|
['<html></html>', {}],
|
||||||
[{}, {}],
|
[{}, {}],
|
||||||
])('tries to fetch servers from remote', async (mockedValue, expectedNewServers) => {
|
])('tries to fetch servers from remote', async (mockedValue, expectedNewServers) => {
|
||||||
get.mockResolvedValue(mockedValue);
|
fetch.mockResolvedValue(mockedValue);
|
||||||
const doFetchServers = fetchServers(axios);
|
const doFetchServers = fetchServers(fetch);
|
||||||
|
|
||||||
await doFetchServers()(dispatch, jest.fn(), {});
|
await doFetchServers()(dispatch, jest.fn(), {});
|
||||||
|
|
||||||
|
@ -95,7 +88,7 @@ describe('remoteServersReducer', () => {
|
||||||
expect(dispatch).toHaveBeenNthCalledWith(3, expect.objectContaining({
|
expect(dispatch).toHaveBeenNthCalledWith(3, expect.objectContaining({
|
||||||
type: doFetchServers.fulfilled.toString(),
|
type: doFetchServers.fulfilled.toString(),
|
||||||
}));
|
}));
|
||||||
expect(get).toHaveBeenCalledTimes(1);
|
expect(fetch).toHaveBeenCalledTimes(1);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -43,8 +43,8 @@ describe('shortUrlDeletionReducer', () => {
|
||||||
}));
|
}));
|
||||||
|
|
||||||
it('returns errorData on DELETE_SHORT_URL_ERROR', () => {
|
it('returns errorData on DELETE_SHORT_URL_ERROR', () => {
|
||||||
const errorData = Mock.of<ProblemDetailsError>({ type: 'bar' });
|
const errorData = Mock.of<ProblemDetailsError>({ type: 'bar', detail: 'detail', title: 'title', status: 400 });
|
||||||
const error = { response: { data: errorData } };
|
const error = errorData;
|
||||||
|
|
||||||
expect(reducer(undefined, { type: deleteShortUrl.rejected.toString(), error })).toEqual({
|
expect(reducer(undefined, { type: deleteShortUrl.rejected.toString(), error })).toEqual({
|
||||||
shortCode: '',
|
shortCode: '',
|
||||||
|
|
Loading…
Reference in a new issue