mirror of
https://gitee.com/h_mo/uniapp-vue3-vite-ts-template
synced 2025-04-06 03:58:03 +08:00
types-将luch-request改成ts
This commit is contained in:
parent
55eb2d252a
commit
9395f17bf7
@ -18,6 +18,10 @@
|
|||||||
const title = ref('Hello');
|
const title = ref('Hello');
|
||||||
const url = import.meta.env.VITE_BASE_URL;
|
const url = import.meta.env.VITE_BASE_URL;
|
||||||
console.log('url', url);
|
console.log('url', url);
|
||||||
|
type Data = {
|
||||||
|
token: string;
|
||||||
|
};
|
||||||
|
|
||||||
const jumLogin = () => {
|
const jumLogin = () => {
|
||||||
console.log('/pages/login/index');
|
console.log('/pages/login/index');
|
||||||
|
|
||||||
@ -26,21 +30,18 @@
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
import { request } from '@/utils/http/index';
|
import { defHttp } from '@/utils/http/index';
|
||||||
const form = reactive({
|
const form = reactive({
|
||||||
email: 'catch@admin.com',
|
email: 'catch@admin.com',
|
||||||
password: 'catchadmin',
|
password: 'catchadmin',
|
||||||
});
|
});
|
||||||
const loginType = ref('');
|
const loginType = ref('');
|
||||||
const submit = () => {
|
const submit = () => {
|
||||||
request
|
defHttp
|
||||||
.post({
|
.post<Data>('/login', form)
|
||||||
url: '/login',
|
.then((res) => {
|
||||||
data: form,
|
|
||||||
})
|
|
||||||
.then((res: any) => {
|
|
||||||
loginType.value = '登录成功';
|
loginType.value = '登录成功';
|
||||||
console.log(res.message);
|
console.log(res.data.token);
|
||||||
})
|
})
|
||||||
.catch((err: any) => {
|
.catch((err: any) => {
|
||||||
loginType.value = '登录失败';
|
loginType.value = '登录失败';
|
||||||
|
48
src/types/http.d.ts
vendored
48
src/types/http.d.ts
vendored
@ -1,12 +1,28 @@
|
|||||||
type AnyObject = Record<string | number | symbol, any>;
|
type AnyObject = Record<string | number | symbol, any>;
|
||||||
|
export type HttpResponse<T = any> =
|
||||||
|
| HttpSuccess<T>
|
||||||
|
| HttpError<T>
|
||||||
|
| HttpDownloadResponse
|
||||||
|
| HttpUploadResponse<T>;
|
||||||
type HttpPromise<T> = Promise<HttpResponse<T>>;
|
type HttpPromise<T> = Promise<HttpResponse<T>>;
|
||||||
type Tasks = UniApp.RequestTask | UniApp.UploadTask | UniApp.DownloadTask;
|
type Tasks = UniApp.RequestTask | UniApp.UploadTask | UniApp.DownloadTask;
|
||||||
|
type Method =
|
||||||
|
| 'GET'
|
||||||
|
| 'POST'
|
||||||
|
| 'PUT'
|
||||||
|
| 'DELETE'
|
||||||
|
| 'CONNECT'
|
||||||
|
| 'HEAD'
|
||||||
|
| 'OPTIONS'
|
||||||
|
| 'TRACE'
|
||||||
|
| 'UPLOAD'
|
||||||
|
| 'DOWNLOAD';
|
||||||
export interface RequestTask {
|
export interface RequestTask {
|
||||||
abort: () => void;
|
abort: () => void;
|
||||||
offHeadersReceived: () => void;
|
offHeadersReceived: () => void;
|
||||||
onHeadersReceived: () => void;
|
onHeadersReceived: () => void;
|
||||||
}
|
}
|
||||||
export interface HttpRequestConfig<T = Tasks> {
|
export interface HttpRequestConfig<T = Tasks> extends Record<string, any> {
|
||||||
/** 请求基地址 */
|
/** 请求基地址 */
|
||||||
baseURL?: string;
|
baseURL?: string;
|
||||||
/** 请求服务器接口地址 */
|
/** 请求服务器接口地址 */
|
||||||
@ -32,20 +48,12 @@ export interface HttpRequestConfig<T = Tasks> {
|
|||||||
/** 要上传的文件对象,仅H5(2.6.15+)支持 */
|
/** 要上传的文件对象,仅H5(2.6.15+)支持 */
|
||||||
file?: File;
|
file?: File;
|
||||||
|
|
||||||
|
/** 文要上传的件类型, 支付宝小程序,且必填 **/
|
||||||
|
fileType?: string;
|
||||||
/** 请求头信息 */
|
/** 请求头信息 */
|
||||||
header?: AnyObject;
|
header?: AnyObject;
|
||||||
/** 请求方式 */
|
/** 请求方式 */
|
||||||
method?:
|
method?: Method;
|
||||||
| 'GET'
|
|
||||||
| 'POST'
|
|
||||||
| 'PUT'
|
|
||||||
| 'DELETE'
|
|
||||||
| 'CONNECT'
|
|
||||||
| 'HEAD'
|
|
||||||
| 'OPTIONS'
|
|
||||||
| 'TRACE'
|
|
||||||
| 'UPLOAD'
|
|
||||||
| 'DOWNLOAD';
|
|
||||||
/** 如果设为 json,会尝试对返回的数据做一次 JSON.parse */
|
/** 如果设为 json,会尝试对返回的数据做一次 JSON.parse */
|
||||||
dataType?: string;
|
dataType?: string;
|
||||||
/** 设置响应的数据类型,支付宝小程序不支持 */
|
/** 设置响应的数据类型,支付宝小程序不支持 */
|
||||||
@ -66,7 +74,7 @@ export interface HttpRequestConfig<T = Tasks> {
|
|||||||
/** 全局自定义验证器 */
|
/** 全局自定义验证器 */
|
||||||
validateStatus?: (statusCode: number) => boolean | void;
|
validateStatus?: (statusCode: number) => boolean | void;
|
||||||
}
|
}
|
||||||
export interface HttpResponse<T = any> {
|
export interface HttpSuccess<T = any> {
|
||||||
config: HttpRequestConfig;
|
config: HttpRequestConfig;
|
||||||
statusCode: number;
|
statusCode: number;
|
||||||
cookies: Array<string>;
|
cookies: Array<string>;
|
||||||
@ -80,14 +88,14 @@ export interface HttpUploadResponse<T = any> {
|
|||||||
data: T;
|
data: T;
|
||||||
errMsg: string;
|
errMsg: string;
|
||||||
}
|
}
|
||||||
export interface HttpDownloadResponse extends HttpResponse {
|
export interface HttpDownloadResponse extends HttpSuccess {
|
||||||
tempFilePath: string;
|
tempFilePath: string;
|
||||||
}
|
}
|
||||||
export interface HttpError {
|
export interface HttpError<T = any> {
|
||||||
config: HttpRequestConfig;
|
config: HttpRequestConfig;
|
||||||
statusCode?: number;
|
statusCode?: number;
|
||||||
cookies?: Array<string>;
|
cookies?: Array<string>;
|
||||||
data?: any;
|
data?: T;
|
||||||
errMsg: string;
|
errMsg: string;
|
||||||
header?: AnyObject;
|
header?: AnyObject;
|
||||||
}
|
}
|
||||||
@ -97,11 +105,15 @@ export interface HttpInterceptorManager<V, E = V> {
|
|||||||
onRejected?: (config: E) => Promise<E> | E,
|
onRejected?: (config: E) => Promise<E> | E,
|
||||||
): void;
|
): void;
|
||||||
eject(id: number): void;
|
eject(id: number): void;
|
||||||
|
forEach(h: any): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type InterceptorsRequest = HttpInterceptorManager<HttpRequestConfig, HttpRequestConfig>;
|
||||||
|
export type InterceptorsResponse = HttpInterceptorManager<HttpSuccess, HttpError>;
|
||||||
|
|
||||||
export interface Interceptors {
|
export interface Interceptors {
|
||||||
request: HttpInterceptorManager<HttpRequestConfig, HttpRequestConfig>;
|
request: InterceptorsRequest;
|
||||||
response: HttpInterceptorManager<HttpResponse, HttpError>;
|
response: InterceptorsResponse;
|
||||||
}
|
}
|
||||||
|
|
||||||
export abstract class HttpRequestAbstract {
|
export abstract class HttpRequestAbstract {
|
||||||
|
@ -2,7 +2,8 @@ import buildURL from '../helpers/buildURL';
|
|||||||
import buildFullPath from '../core/buildFullPath';
|
import buildFullPath from '../core/buildFullPath';
|
||||||
import settle from '../core/settle';
|
import settle from '../core/settle';
|
||||||
import { isUndefined } from '../utils';
|
import { isUndefined } from '../utils';
|
||||||
import { HttpRequestConfig } from '@/types/http';
|
import { HttpRequestConfig, HttpResponse } from '@/types/http';
|
||||||
|
import { cloneDeep } from 'lodash-es';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 返回可选值存在的配置
|
* 返回可选值存在的配置
|
||||||
@ -10,8 +11,8 @@ import { HttpRequestConfig } from '@/types/http';
|
|||||||
* @param {Object} config2 - 配置
|
* @param {Object} config2 - 配置
|
||||||
* @return {{}} - 存在的配置项
|
* @return {{}} - 存在的配置项
|
||||||
*/
|
*/
|
||||||
const mergeKeys = (keys, config2) => {
|
const mergeKeys = (keys: string[], config2: Record<string, any>) => {
|
||||||
let config = {};
|
const config: Record<string, any> = {};
|
||||||
keys.forEach((prop) => {
|
keys.forEach((prop) => {
|
||||||
if (!isUndefined(config2[prop])) {
|
if (!isUndefined(config2[prop])) {
|
||||||
config[prop] = config2[prop];
|
config[prop] = config2[prop];
|
||||||
@ -19,30 +20,55 @@ const mergeKeys = (keys, config2) => {
|
|||||||
});
|
});
|
||||||
return config;
|
return config;
|
||||||
};
|
};
|
||||||
export default (config: HttpRequestConfig) => {
|
|
||||||
return new Promise((resolve, reject) => {
|
export default <T>(config: HttpRequestConfig) => {
|
||||||
let fullPath = buildURL(buildFullPath(config.baseURL, config.url || ''), config.params);
|
return new Promise<T>((resolve, reject) => {
|
||||||
|
const fullPath = buildURL(buildFullPath(config.baseURL, config.url || ''), config.params);
|
||||||
const _config: UniApp.RequestOptions = {
|
const _config: UniApp.RequestOptions = {
|
||||||
url: fullPath,
|
url: fullPath,
|
||||||
header: config.header,
|
header: config.header,
|
||||||
complete: (response) => {
|
success: (res) => {
|
||||||
config.fullPath = fullPath;
|
|
||||||
response.config = config;
|
|
||||||
try {
|
try {
|
||||||
// 对可能字符串不是json 的情况容错
|
// 对可能字符串不是json 的情况容错
|
||||||
if (typeof response.data === 'string') {
|
if (typeof res.data === 'string') {
|
||||||
response.data = JSON.parse(response.data);
|
res.data = JSON.parse(res.data);
|
||||||
}
|
}
|
||||||
// eslint-disable-next-line no-empty
|
|
||||||
} catch (e) {}
|
} catch (e) {}
|
||||||
|
const cloneConfig = cloneDeep(config);
|
||||||
|
cloneConfig.fullPath = fullPath;
|
||||||
|
const response = {
|
||||||
|
...res,
|
||||||
|
config: cloneConfig,
|
||||||
|
} as HttpResponse;
|
||||||
settle(resolve, reject, response);
|
settle(resolve, reject, response);
|
||||||
},
|
},
|
||||||
|
fail: (err) => {
|
||||||
|
const cloneConfig = cloneDeep(config);
|
||||||
|
cloneConfig.fullPath = fullPath;
|
||||||
|
const response = {
|
||||||
|
...err,
|
||||||
|
config: cloneConfig,
|
||||||
|
} as HttpResponse;
|
||||||
|
settle(resolve, reject, response);
|
||||||
|
},
|
||||||
|
// complete: (response) => {
|
||||||
|
// config.fullPath = fullPath;
|
||||||
|
// response.config = config;
|
||||||
|
// try {
|
||||||
|
// // 对可能字符串不是json 的情况容错
|
||||||
|
// if (typeof response.data === 'string') {
|
||||||
|
// response.data = JSON.parse(response.data);
|
||||||
|
// }
|
||||||
|
// // eslint-disable-next-line no-empty
|
||||||
|
// } catch (e) {}
|
||||||
|
// settle(resolve, reject, response);
|
||||||
|
// },
|
||||||
};
|
};
|
||||||
let requestTask;
|
let requestTask;
|
||||||
if (config.method === 'UPLOAD') {
|
if (config.method === 'UPLOAD') {
|
||||||
delete _config.header['content-type'];
|
delete _config.header['content-type'];
|
||||||
delete _config.header['Content-Type'];
|
delete _config.header['Content-Type'];
|
||||||
let otherConfig = {
|
const otherConfig = {
|
||||||
// #ifdef MP-ALIPAY
|
// #ifdef MP-ALIPAY
|
||||||
fileType: config.fileType,
|
fileType: config.fileType,
|
||||||
// #endif
|
// #endif
|
||||||
@ -61,18 +87,20 @@ export default (config: HttpRequestConfig) => {
|
|||||||
// #endif
|
// #endif
|
||||||
'formData',
|
'formData',
|
||||||
];
|
];
|
||||||
requestTask = uni.uploadFile({
|
const uploadFileOption = {
|
||||||
..._config,
|
..._config,
|
||||||
...otherConfig,
|
...otherConfig,
|
||||||
...mergeKeys(optionalKeys, config),
|
...mergeKeys(optionalKeys, config),
|
||||||
});
|
} as UniApp.UploadFileOption;
|
||||||
|
requestTask = uni.uploadFile(uploadFileOption);
|
||||||
} else if (config.method === 'DOWNLOAD') {
|
} else if (config.method === 'DOWNLOAD') {
|
||||||
// #ifdef H5 || APP-PLUS
|
// #ifdef H5 || APP-PLUS
|
||||||
if (!isUndefined(config['timeout'])) {
|
if (!isUndefined(config['timeout'])) {
|
||||||
_config['timeout'] = config['timeout'];
|
_config['timeout'] = config['timeout'];
|
||||||
}
|
}
|
||||||
// #endif
|
// #endif
|
||||||
requestTask = uni.downloadFile(_config);
|
const downloadFileOption = _config as UniApp.DownloadFileOption;
|
||||||
|
requestTask = uni.downloadFile(downloadFileOption);
|
||||||
} else {
|
} else {
|
||||||
const optionalKeys = [
|
const optionalKeys = [
|
||||||
'data',
|
'data',
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
export class InterceptorManager1 {}
|
|
||||||
|
|
||||||
function InterceptorManager(): void {
|
function InterceptorManager(): void {
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
this.handlers = [];
|
this.handlers = [];
|
||||||
@ -28,7 +26,7 @@ InterceptorManager.prototype.use = function use(fulfilled: any, rejected: any) {
|
|||||||
*
|
*
|
||||||
* @param {Number} id The ID that was returned by `use`
|
* @param {Number} id The ID that was returned by `use`
|
||||||
*/
|
*/
|
||||||
InterceptorManager.prototype.eject = function eject(id) {
|
InterceptorManager.prototype.eject = function eject(id: number) {
|
||||||
if (this.handlers[id]) {
|
if (this.handlers[id]) {
|
||||||
this.handlers[id] = null;
|
this.handlers[id] = null;
|
||||||
}
|
}
|
||||||
@ -42,8 +40,8 @@ InterceptorManager.prototype.eject = function eject(id) {
|
|||||||
*
|
*
|
||||||
* @param {Function} fn The function to call for each interceptor
|
* @param {Function} fn The function to call for each interceptor
|
||||||
*/
|
*/
|
||||||
InterceptorManager.prototype.forEach = function forEach(fn) {
|
InterceptorManager.prototype.forEach = function forEach(fn: (h: any) => void) {
|
||||||
this.handlers.forEach((h) => {
|
this.handlers.forEach((h: any) => {
|
||||||
if (h !== null) {
|
if (h !== null) {
|
||||||
fn(h);
|
fn(h);
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@
|
|||||||
* github: https://github.com/lei-mu/luch-request
|
* github: https://github.com/lei-mu/luch-request
|
||||||
* DCloud: http://ext.dcloud.net.cn/plugin?id=392
|
* DCloud: http://ext.dcloud.net.cn/plugin?id=392
|
||||||
* HBuilderX: beat-3.0.4 alpha-3.0.4
|
* HBuilderX: beat-3.0.4 alpha-3.0.4
|
||||||
|
* 2022-06-09 改写成ts
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import dispatchRequest from './dispatchRequest';
|
import dispatchRequest from './dispatchRequest';
|
||||||
@ -17,11 +18,11 @@ import mergeConfig from './mergeConfig';
|
|||||||
import defaults from './defaults';
|
import defaults from './defaults';
|
||||||
import { cloneDeep } from 'lodash-es';
|
import { cloneDeep } from 'lodash-es';
|
||||||
import { isPlainObject } from '../utils';
|
import { isPlainObject } from '../utils';
|
||||||
import { HttpRequestConfig, Interceptors } from '@/types/http';
|
import { HttpRequestConfig, HttpResponse, Interceptors } from '@/types/http';
|
||||||
|
|
||||||
export default class Request {
|
export default class Request {
|
||||||
private readonly config: HttpRequestConfig | undefined;
|
private config: HttpRequestConfig;
|
||||||
private readonly interceptors: Interceptors;
|
interceptors: Interceptors;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {Object} arg - 全局配置
|
* @param {Object} arg - 全局配置
|
||||||
@ -44,7 +45,9 @@ export default class Request {
|
|||||||
}
|
}
|
||||||
this.config = cloneDeep({ ...defaults, ...arg });
|
this.config = cloneDeep({ ...defaults, ...arg });
|
||||||
this.interceptors = {
|
this.interceptors = {
|
||||||
|
// @ts-ignore
|
||||||
request: new InterceptorManager(),
|
request: new InterceptorManager(),
|
||||||
|
// @ts-ignore
|
||||||
response: new InterceptorManager(),
|
response: new InterceptorManager(),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@ -57,21 +60,24 @@ export default class Request {
|
|||||||
this.config = f(this.config);
|
this.config = f(this.config);
|
||||||
}
|
}
|
||||||
|
|
||||||
middleware(config) {
|
middleware<T>(config: HttpRequestConfig) {
|
||||||
config = mergeConfig(this.config, config);
|
config = mergeConfig(this.config, config);
|
||||||
let chain = [dispatchRequest, undefined];
|
const chain = [dispatchRequest, undefined];
|
||||||
let promise = Promise.resolve(config);
|
const response: HttpResponse<T> = {
|
||||||
|
config: config,
|
||||||
|
} as HttpResponse<T>;
|
||||||
|
let promise = Promise.resolve(response);
|
||||||
|
|
||||||
this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {
|
this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor: any) {
|
||||||
chain.unshift(interceptor.fulfilled, interceptor.rejected);
|
chain.unshift(interceptor.fulfilled, interceptor.rejected);
|
||||||
});
|
});
|
||||||
|
|
||||||
this.interceptors.response.forEach(function pushResponseInterceptors(interceptor) {
|
this.interceptors.response.forEach(function pushResponseInterceptors(interceptor: any) {
|
||||||
chain.push(interceptor.fulfilled, interceptor.rejected);
|
chain.push(interceptor.fulfilled, interceptor.rejected);
|
||||||
});
|
});
|
||||||
|
|
||||||
while (chain.length) {
|
while (chain.length) {
|
||||||
promise = promise.then(chain.shift(), chain.shift());
|
promise = promise.then<HttpResponse<T>>(chain.shift(), chain.shift());
|
||||||
}
|
}
|
||||||
|
|
||||||
return promise;
|
return promise;
|
||||||
@ -88,20 +94,20 @@ export default class Request {
|
|||||||
* @prop {Object} [options.method = config.method] - 请求方法
|
* @prop {Object} [options.method = config.method] - 请求方法
|
||||||
* @returns {Promise<unknown>}
|
* @returns {Promise<unknown>}
|
||||||
*/
|
*/
|
||||||
request(config = {}) {
|
request<T>(config = {}) {
|
||||||
return this.middleware(config);
|
return this.middleware<T>(config);
|
||||||
}
|
}
|
||||||
|
|
||||||
get(url, options = {}) {
|
get<T>(url: string, options = {}) {
|
||||||
return this.middleware({
|
return this.middleware<T>({
|
||||||
url,
|
url,
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
...options,
|
...options,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
post(url, data, options = {}) {
|
post<T>(url: string, data: AnyObject, options: Partial<HttpRequestConfig> = {}) {
|
||||||
return this.middleware({
|
return this.middleware<T>({
|
||||||
url,
|
url,
|
||||||
data,
|
data,
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
@ -110,87 +116,81 @@ export default class Request {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// #ifndef MP-ALIPAY
|
// #ifndef MP-ALIPAY
|
||||||
put(url, data, options = {}) {
|
put<T>(url: string, data: AnyObject, options: Partial<HttpRequestConfig> = {}) {
|
||||||
return this.middleware({
|
return this.middleware<T>({
|
||||||
url,
|
url,
|
||||||
data,
|
data,
|
||||||
method: 'PUT',
|
method: 'PUT',
|
||||||
...options,
|
...options,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// #endif
|
// #endif
|
||||||
|
|
||||||
// #ifdef APP-PLUS || H5 || MP-WEIXIN || MP-BAIDU
|
// #ifdef APP-PLUS || H5 || MP-WEIXIN || MP-BAIDU
|
||||||
delete(url, data, options = {}) {
|
delete<T>(url: string, data: AnyObject, options: Partial<HttpRequestConfig> = {}) {
|
||||||
return this.middleware({
|
return this.middleware<T>({
|
||||||
url,
|
url,
|
||||||
data,
|
data,
|
||||||
method: 'DELETE',
|
method: 'DELETE',
|
||||||
...options,
|
...options,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// #endif
|
// #endif
|
||||||
|
|
||||||
// #ifdef H5 || MP-WEIXIN
|
// #ifdef H5 || MP-WEIXIN
|
||||||
connect(url, data, options = {}) {
|
connect<T>(url: string, data: AnyObject, options: Partial<HttpRequestConfig> = {}) {
|
||||||
return this.middleware({
|
return this.middleware<T>({
|
||||||
url,
|
url,
|
||||||
data,
|
data,
|
||||||
method: 'CONNECT',
|
method: 'CONNECT',
|
||||||
...options,
|
...options,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// #endif
|
// #endif
|
||||||
|
|
||||||
// #ifdef H5 || MP-WEIXIN || MP-BAIDU
|
// #ifdef H5 || MP-WEIXIN || MP-BAIDU
|
||||||
head(url, data, options = {}) {
|
head<T>(url: string, data: AnyObject, options: Partial<HttpRequestConfig> = {}) {
|
||||||
return this.middleware({
|
return this.middleware<T>({
|
||||||
url,
|
url,
|
||||||
data,
|
data,
|
||||||
method: 'HEAD',
|
method: 'HEAD',
|
||||||
...options,
|
...options,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// #endif
|
// #endif
|
||||||
|
|
||||||
// #ifdef APP-PLUS || H5 || MP-WEIXIN || MP-BAIDU
|
// #ifdef APP-PLUS || H5 || MP-WEIXIN || MP-BAIDU
|
||||||
options(url, data, options = {}) {
|
options<T>(url: string, data: AnyObject, options: Partial<HttpRequestConfig> = {}) {
|
||||||
return this.middleware({
|
return this.middleware<T>({
|
||||||
url,
|
url,
|
||||||
data,
|
data,
|
||||||
method: 'OPTIONS',
|
method: 'OPTIONS',
|
||||||
...options,
|
...options,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// #endif
|
// #endif
|
||||||
|
|
||||||
// #ifdef H5 || MP-WEIXIN
|
// #ifdef H5 || MP-WEIXIN
|
||||||
trace(url, data, options = {}) {
|
trace<T>(url: string, data: AnyObject, options: Partial<HttpRequestConfig> = {}) {
|
||||||
return this.middleware({
|
return this.middleware<T>({
|
||||||
url,
|
url,
|
||||||
data,
|
data,
|
||||||
method: 'TRACE',
|
method: 'TRACE',
|
||||||
...options,
|
...options,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// #endif
|
// #endif
|
||||||
|
|
||||||
upload(url, config = {}) {
|
upload<T>(url: string, config: Partial<HttpRequestConfig> = {}) {
|
||||||
config.url = url;
|
config.url = url;
|
||||||
config.method = 'UPLOAD';
|
config.method = 'UPLOAD';
|
||||||
return this.middleware(config);
|
return this.middleware<T>(config);
|
||||||
}
|
}
|
||||||
|
|
||||||
download(url, config = {}) {
|
download<T>(url: string, config: Partial<HttpRequestConfig> = {}) {
|
||||||
config.url = url;
|
config.url = url;
|
||||||
config.method = 'DOWNLOAD';
|
config.method = 'DOWNLOAD';
|
||||||
return this.middleware(config);
|
return this.middleware<T>(config);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import adapter from '../adapters';
|
import adapter from '../adapters';
|
||||||
import { HttpRequestConfig } from '@/types/http';
|
import { HttpRequestConfig } from '@/types/http';
|
||||||
|
|
||||||
export default (config: HttpRequestConfig) => {
|
export default <T>(config: HttpRequestConfig) => {
|
||||||
return adapter(config);
|
return adapter<T>(config.config);
|
||||||
};
|
};
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import { deepMerge, isUndefined } from '../utils';
|
import { deepMerge, isUndefined } from '../utils';
|
||||||
|
import { HttpRequestConfig } from '@/types/http';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 合并局部配置优先的配置,如果局部有该配置项则用局部,如果全局有该配置项则用全局
|
* 合并局部配置优先的配置,如果局部有该配置项则用局部,如果全局有该配置项则用全局
|
||||||
@ -7,8 +8,12 @@ import { deepMerge, isUndefined } from '../utils';
|
|||||||
* @param {Object} config2 - 局部配置
|
* @param {Object} config2 - 局部配置
|
||||||
* @return {{}}
|
* @return {{}}
|
||||||
*/
|
*/
|
||||||
const mergeKeys = (keys, globalsConfig, config2) => {
|
const mergeKeys = (
|
||||||
let config = {};
|
keys: string[],
|
||||||
|
globalsConfig: HttpRequestConfig,
|
||||||
|
config2: Partial<HttpRequestConfig>,
|
||||||
|
) => {
|
||||||
|
const config: Partial<HttpRequestConfig> = {};
|
||||||
keys.forEach((prop) => {
|
keys.forEach((prop) => {
|
||||||
if (!isUndefined(config2[prop])) {
|
if (!isUndefined(config2[prop])) {
|
||||||
config[prop] = config2[prop];
|
config[prop] = config2[prop];
|
||||||
@ -24,14 +29,15 @@ const mergeKeys = (keys, globalsConfig, config2) => {
|
|||||||
* @param config2 - 当前的局部配置
|
* @param config2 - 当前的局部配置
|
||||||
* @return - 合并后的配置
|
* @return - 合并后的配置
|
||||||
*/
|
*/
|
||||||
export default (globalsConfig, config2 = {}) => {
|
export default (globalsConfig: HttpRequestConfig, config2: Partial<HttpRequestConfig> = {}) => {
|
||||||
const method = config2.method || globalsConfig.method || 'GET';
|
const method = config2.method || globalsConfig.method || 'GET';
|
||||||
let config = {
|
let config: Partial<HttpRequestConfig> = {
|
||||||
baseURL: globalsConfig.baseURL || '',
|
baseURL: globalsConfig.baseURL || '',
|
||||||
method: method,
|
method: method,
|
||||||
url: config2.url || '',
|
url: config2.url || '',
|
||||||
params: config2.params || {},
|
params: config2.params || {},
|
||||||
custom: { ...(globalsConfig.custom || {}), ...(config2.custom || {}) },
|
custom: { ...(globalsConfig.custom || {}), ...(config2.custom || {}) },
|
||||||
|
// @ts-ignore
|
||||||
header: deepMerge(globalsConfig.header || {}, config2.header || {}),
|
header: deepMerge(globalsConfig.header || {}, config2.header || {}),
|
||||||
};
|
};
|
||||||
const defaultToConfig2Keys = ['getTask', 'validateStatus'];
|
const defaultToConfig2Keys = ['getTask', 'validateStatus'];
|
||||||
@ -47,8 +53,10 @@ export default (globalsConfig, config2 = {}) => {
|
|||||||
}
|
}
|
||||||
// #endif
|
// #endif
|
||||||
} else if (method === 'UPLOAD') {
|
} else if (method === 'UPLOAD') {
|
||||||
|
if (config.header) {
|
||||||
delete config.header['content-type'];
|
delete config.header['content-type'];
|
||||||
delete config.header['Content-Type'];
|
delete config.header['Content-Type'];
|
||||||
|
}
|
||||||
const uploadKeys = [
|
const uploadKeys = [
|
||||||
// #ifdef APP-PLUS || H5
|
// #ifdef APP-PLUS || H5
|
||||||
'files',
|
'files',
|
||||||
|
@ -5,7 +5,9 @@
|
|||||||
* @param {Function} reject A function that rejects the promise.
|
* @param {Function} reject A function that rejects the promise.
|
||||||
* @param {object} response The response.
|
* @param {object} response The response.
|
||||||
*/
|
*/
|
||||||
export default function settle(resolve, reject, response) {
|
import { HttpResponse } from '@/types/http';
|
||||||
|
|
||||||
|
export default function settle(resolve: any, reject: any, response: HttpResponse) {
|
||||||
const validateStatus = response.config.validateStatus;
|
const validateStatus = response.config.validateStatus;
|
||||||
const status = response.statusCode;
|
const status = response.statusCode;
|
||||||
if (status && (!validateStatus || validateStatus(status))) {
|
if (status && (!validateStatus || validateStatus(status))) {
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
import * as utils from '../utils';
|
import * as utils from '../utils';
|
||||||
|
|
||||||
function encode(val) {
|
function encode(val: string | number | boolean) {
|
||||||
return encodeURIComponent(val)
|
return encodeURIComponent(val)
|
||||||
.replace(/%40/gi, '@')
|
.replace(/%40/gi, '@')
|
||||||
.replace(/%3A/gi, ':')
|
.replace(/%3A/gi, ':')
|
||||||
@ -31,7 +31,7 @@ export default function buildURL(url: string, params?: any) {
|
|||||||
} else {
|
} else {
|
||||||
const parts: string[] = [];
|
const parts: string[] = [];
|
||||||
|
|
||||||
utils.forEach(params, function serialize(val: string, key: string) {
|
utils.forEach(params, function serialize(val: any, key: string) {
|
||||||
if (val === null || typeof val === 'undefined') {
|
if (val === null || typeof val === 'undefined') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -42,7 +42,7 @@ export default function buildURL(url: string, params?: any) {
|
|||||||
val = [val];
|
val = [val];
|
||||||
}
|
}
|
||||||
|
|
||||||
utils.forEach(val, function parseValue(v) {
|
utils.forEach(val, function parseValue(v: any) {
|
||||||
if (utils.isDate(v)) {
|
if (utils.isDate(v)) {
|
||||||
v = v.toISOString();
|
v = v.toISOString();
|
||||||
} else if (utils.isObject(v)) {
|
} else if (utils.isObject(v)) {
|
||||||
|
@ -63,6 +63,8 @@ defHttp.interceptors.response.use(
|
|||||||
// if (response.data.code !== 200) { // 服务端返回的状态码不等于200,则reject()
|
// if (response.data.code !== 200) { // 服务端返回的状态码不等于200,则reject()
|
||||||
// return Promise.reject(response)
|
// return Promise.reject(response)
|
||||||
// }
|
// }
|
||||||
|
console.log('响应===', response);
|
||||||
|
|
||||||
return response;
|
return response;
|
||||||
},
|
},
|
||||||
(response) => {
|
(response) => {
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
import { cloneDeep, values } from 'lodash-es';
|
|
||||||
|
|
||||||
// utils is a library of generic helper functions non-specific to axios
|
// utils is a library of generic helper functions non-specific to axios
|
||||||
|
|
||||||
const toString = Object.prototype.toString;
|
const toString = Object.prototype.toString;
|
||||||
@ -80,7 +78,7 @@ export function forEach(obj: any, fn: any) {
|
|||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
// Iterate over object keys
|
// Iterate over object keys
|
||||||
for (let key in obj) {
|
for (const key in obj) {
|
||||||
if (Object.prototype.hasOwnProperty.call(obj, key)) {
|
if (Object.prototype.hasOwnProperty.call(obj, key)) {
|
||||||
fn.call(null, obj[key], key, obj);
|
fn.call(null, obj[key], key, obj);
|
||||||
}
|
}
|
||||||
@ -102,7 +100,7 @@ export function isBoolean(val: unknown): val is boolean {
|
|||||||
* @param {any} obj - 检测的对象
|
* @param {any} obj - 检测的对象
|
||||||
* @returns {boolean}
|
* @returns {boolean}
|
||||||
*/
|
*/
|
||||||
export function isPlainObject(obj: object): boolean {
|
export function isPlainObject(obj: unknown): boolean {
|
||||||
return Object.prototype.toString.call(obj) === '[object Object]';
|
return Object.prototype.toString.call(obj) === '[object Object]';
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -116,7 +114,7 @@ export function isPlainObject(obj: object): boolean {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
export function deepMerge(/* obj1, obj2, obj3, ... */) {
|
export function deepMerge(/* obj1, obj2, obj3, ... */) {
|
||||||
let result: any = {};
|
const result: any = {};
|
||||||
function assignValue(val: any, key: any) {
|
function assignValue(val: any, key: any) {
|
||||||
if (typeof result[key] === 'object' && typeof val === 'object') {
|
if (typeof result[key] === 'object' && typeof val === 'object') {
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
|
@ -1,262 +0,0 @@
|
|||||||
/* eslint-disable */
|
|
||||||
import { cloneDeep } from 'lodash-es';
|
|
||||||
|
|
||||||
var clone = (function () {
|
|
||||||
'use strict';
|
|
||||||
|
|
||||||
function _instanceof(obj, type) {
|
|
||||||
return type != null && obj instanceof type;
|
|
||||||
}
|
|
||||||
|
|
||||||
var nativeMap;
|
|
||||||
try {
|
|
||||||
nativeMap = Map;
|
|
||||||
} catch (_) {
|
|
||||||
// maybe a reference error because no `Map`. Give it a dummy value that no
|
|
||||||
// value will ever be an instanceof.
|
|
||||||
nativeMap = function () {};
|
|
||||||
}
|
|
||||||
|
|
||||||
var nativeSet;
|
|
||||||
try {
|
|
||||||
nativeSet = Set;
|
|
||||||
} catch (_) {
|
|
||||||
nativeSet = function () {};
|
|
||||||
}
|
|
||||||
|
|
||||||
var nativePromise;
|
|
||||||
try {
|
|
||||||
nativePromise = Promise;
|
|
||||||
} catch (_) {
|
|
||||||
nativePromise = function () {};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Clones (copies) an Object using deep copying.
|
|
||||||
*
|
|
||||||
* This function supports circular references by default, but if you are certain
|
|
||||||
* there are no circular references in your object, you can save some CPU time
|
|
||||||
* by calling clone(obj, false).
|
|
||||||
*
|
|
||||||
* Caution: if `circular` is false and `parent` contains circular references,
|
|
||||||
* your program may enter an infinite loop and crash.
|
|
||||||
*
|
|
||||||
* @param `parent` - the object to be cloned
|
|
||||||
* @param `circular` - set to true if the object to be cloned may contain
|
|
||||||
* circular references. (optional - true by default)
|
|
||||||
* @param `depth` - set to a number if the object is only to be cloned to
|
|
||||||
* a particular depth. (optional - defaults to Infinity)
|
|
||||||
* @param `prototype` - sets the prototype to be used when cloning an object.
|
|
||||||
* (optional - defaults to parent prototype).
|
|
||||||
* @param `includeNonEnumerable` - set to true if the non-enumerable properties
|
|
||||||
* should be cloned as well. Non-enumerable properties on the prototype
|
|
||||||
* chain will be ignored. (optional - false by default)
|
|
||||||
*/
|
|
||||||
function clone(parent, circular, depth, prototype, includeNonEnumerable) {
|
|
||||||
if (typeof circular === 'object') {
|
|
||||||
depth = circular.depth;
|
|
||||||
prototype = circular.prototype;
|
|
||||||
includeNonEnumerable = circular.includeNonEnumerable;
|
|
||||||
circular = circular.circular;
|
|
||||||
}
|
|
||||||
// maintain two arrays for circular references, where corresponding parents
|
|
||||||
// and children have the same index
|
|
||||||
var allParents = [];
|
|
||||||
var allChildren = [];
|
|
||||||
|
|
||||||
var useBuffer = typeof Buffer != 'undefined';
|
|
||||||
|
|
||||||
if (typeof circular == 'undefined') circular = true;
|
|
||||||
|
|
||||||
if (typeof depth == 'undefined') depth = Infinity;
|
|
||||||
|
|
||||||
// recurse this function so we don't reset allParents and allChildren
|
|
||||||
function _clone(parent, depth) {
|
|
||||||
// cloning null always returns null
|
|
||||||
if (parent === null) return null;
|
|
||||||
|
|
||||||
if (depth === 0) return parent;
|
|
||||||
|
|
||||||
var child;
|
|
||||||
var proto;
|
|
||||||
if (typeof parent != 'object') {
|
|
||||||
return parent;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (_instanceof(parent, nativeMap)) {
|
|
||||||
child = new nativeMap();
|
|
||||||
} else if (_instanceof(parent, nativeSet)) {
|
|
||||||
child = new nativeSet();
|
|
||||||
} else if (_instanceof(parent, nativePromise)) {
|
|
||||||
child = new nativePromise(function (resolve, reject) {
|
|
||||||
parent.then(
|
|
||||||
function (value) {
|
|
||||||
resolve(_clone(value, depth - 1));
|
|
||||||
},
|
|
||||||
function (err) {
|
|
||||||
reject(_clone(err, depth - 1));
|
|
||||||
},
|
|
||||||
);
|
|
||||||
});
|
|
||||||
} else if (clone.__isArray(parent)) {
|
|
||||||
child = [];
|
|
||||||
} else if (clone.__isRegExp(parent)) {
|
|
||||||
child = new RegExp(parent.source, __getRegExpFlags(parent));
|
|
||||||
if (parent.lastIndex) child.lastIndex = parent.lastIndex;
|
|
||||||
} else if (clone.__isDate(parent)) {
|
|
||||||
child = new Date(parent.getTime());
|
|
||||||
} else if (useBuffer && Buffer.isBuffer(parent)) {
|
|
||||||
if (Buffer.from) {
|
|
||||||
// Node.js >= 5.10.0
|
|
||||||
child = Buffer.from(parent);
|
|
||||||
} else {
|
|
||||||
// Older Node.js versions
|
|
||||||
child = new Buffer(parent.length);
|
|
||||||
parent.copy(child);
|
|
||||||
}
|
|
||||||
return child;
|
|
||||||
} else if (_instanceof(parent, Error)) {
|
|
||||||
child = Object.create(parent);
|
|
||||||
} else {
|
|
||||||
if (typeof prototype == 'undefined') {
|
|
||||||
proto = Object.getPrototypeOf(parent);
|
|
||||||
child = Object.create(proto);
|
|
||||||
} else {
|
|
||||||
child = Object.create(prototype);
|
|
||||||
proto = prototype;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (circular) {
|
|
||||||
var index = allParents.indexOf(parent);
|
|
||||||
|
|
||||||
if (index != -1) {
|
|
||||||
return allChildren[index];
|
|
||||||
}
|
|
||||||
allParents.push(parent);
|
|
||||||
allChildren.push(child);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (_instanceof(parent, nativeMap)) {
|
|
||||||
parent.forEach(function (value, key) {
|
|
||||||
var keyChild = _clone(key, depth - 1);
|
|
||||||
var valueChild = _clone(value, depth - 1);
|
|
||||||
child.set(keyChild, valueChild);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (_instanceof(parent, nativeSet)) {
|
|
||||||
parent.forEach(function (value) {
|
|
||||||
var entryChild = _clone(value, depth - 1);
|
|
||||||
child.add(entryChild);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
for (var i in parent) {
|
|
||||||
var attrs = Object.getOwnPropertyDescriptor(parent, i);
|
|
||||||
if (attrs) {
|
|
||||||
child[i] = _clone(parent[i], depth - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
var objProperty = Object.getOwnPropertyDescriptor(parent, i);
|
|
||||||
if (objProperty.set === 'undefined') {
|
|
||||||
// no setter defined. Skip cloning this property
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
child[i] = _clone(parent[i], depth - 1);
|
|
||||||
} catch (e) {
|
|
||||||
if (e instanceof TypeError) {
|
|
||||||
// when in strict mode, TypeError will be thrown if child[i] property only has a getter
|
|
||||||
// we can't do anything about this, other than inform the user that this property cannot be set.
|
|
||||||
continue;
|
|
||||||
} else if (e instanceof ReferenceError) {
|
|
||||||
//this may happen in non strict mode
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Object.getOwnPropertySymbols) {
|
|
||||||
var symbols = Object.getOwnPropertySymbols(parent);
|
|
||||||
for (var i = 0; i < symbols.length; i++) {
|
|
||||||
// Don't need to worry about cloning a symbol because it is a primitive,
|
|
||||||
// like a number or string.
|
|
||||||
var symbol = symbols[i];
|
|
||||||
var descriptor = Object.getOwnPropertyDescriptor(parent, symbol);
|
|
||||||
if (descriptor && !descriptor.enumerable && !includeNonEnumerable) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
child[symbol] = _clone(parent[symbol], depth - 1);
|
|
||||||
Object.defineProperty(child, symbol, descriptor);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (includeNonEnumerable) {
|
|
||||||
var allPropertyNames = Object.getOwnPropertyNames(parent);
|
|
||||||
for (var i = 0; i < allPropertyNames.length; i++) {
|
|
||||||
var propertyName = allPropertyNames[i];
|
|
||||||
var descriptor = Object.getOwnPropertyDescriptor(parent, propertyName);
|
|
||||||
if (descriptor && descriptor.enumerable) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
child[propertyName] = _clone(parent[propertyName], depth - 1);
|
|
||||||
Object.defineProperty(child, propertyName, descriptor);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return child;
|
|
||||||
}
|
|
||||||
|
|
||||||
return _clone(parent, depth);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Simple flat clone using prototype, accepts only objects, usefull for property
|
|
||||||
* override on FLAT configuration object (no nested props).
|
|
||||||
*
|
|
||||||
* USE WITH CAUTION! This may not behave as you wish if you do not know how this
|
|
||||||
* works.
|
|
||||||
*/
|
|
||||||
clone.clonePrototype = function clonePrototype(parent) {
|
|
||||||
if (parent === null) return null;
|
|
||||||
|
|
||||||
var c = function () {};
|
|
||||||
c.prototype = parent;
|
|
||||||
return new c();
|
|
||||||
};
|
|
||||||
|
|
||||||
// private utility functions
|
|
||||||
|
|
||||||
function __objToStr(o) {
|
|
||||||
return Object.prototype.toString.call(o);
|
|
||||||
}
|
|
||||||
clone.__objToStr = __objToStr;
|
|
||||||
|
|
||||||
function __isDate(o) {
|
|
||||||
return typeof o === 'object' && __objToStr(o) === '[object Date]';
|
|
||||||
}
|
|
||||||
clone.__isDate = __isDate;
|
|
||||||
|
|
||||||
function __isArray(o) {
|
|
||||||
return typeof o === 'object' && __objToStr(o) === '[object Array]';
|
|
||||||
}
|
|
||||||
clone.__isArray = __isArray;
|
|
||||||
|
|
||||||
function __isRegExp(o) {
|
|
||||||
return typeof o === 'object' && __objToStr(o) === '[object RegExp]';
|
|
||||||
}
|
|
||||||
clone.__isRegExp = __isRegExp;
|
|
||||||
|
|
||||||
function __getRegExpFlags(re) {
|
|
||||||
var flags = '';
|
|
||||||
if (re.global) flags += 'g';
|
|
||||||
if (re.ignoreCase) flags += 'i';
|
|
||||||
if (re.multiline) flags += 'm';
|
|
||||||
return flags;
|
|
||||||
}
|
|
||||||
clone.__getRegExpFlags = __getRegExpFlags;
|
|
||||||
|
|
||||||
return clone;
|
|
||||||
})();
|
|
||||||
|
|
||||||
export { cloneDeep };
|
|
@ -93,6 +93,7 @@ export const isServer = typeof window === 'undefined';
|
|||||||
export const isClient = !isServer;
|
export const isClient = !isServer;
|
||||||
|
|
||||||
export function isUrl(path: string): boolean {
|
export function isUrl(path: string): boolean {
|
||||||
|
// @ts-ignore
|
||||||
const reg =
|
const reg =
|
||||||
/(((^https?:(?:\/\/)?)(?:[-;:&=\+\$,\w]+@)?[A-Za-z0-9.-]+(?::\d+)?|(?:www.|[-;:&=\+\$,\w]+@)[A-Za-z0-9.-]+)((?:\/[\+~%\/.\w-_]*)?\??(?:[-\+=&;%@.\w_]*)#?(?:[\w]*))?)$/;
|
/(((^https?:(?:\/\/)?)(?:[-;:&=\+\$,\w]+@)?[A-Za-z0-9.-]+(?::\d+)?|(?:www.|[-;:&=\+\$,\w]+@)[A-Za-z0-9.-]+)((?:\/[\+~%\/.\w-_]*)?\??(?:[-\+=&;%@.\w_]*)#?(?:[\w]*))?)$/;
|
||||||
return reg.test(path);
|
return reg.test(path);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user