从实战代码中学习,上述实战代码来源:芋道源码/yudao-mall-uniapp
该代码中,通过自定义 request 函数对 HTTP 请求进行了统一管理,并且结合了 Token 认证机制
根据实战中的Demo,给出一版通用的Demo:
封装request的时候,需要与token结合:
// utils/request.js
import { getAccessToken, setToken, removeToken } from '@/utils/auth';
import config from '@/config';
import errorCode from '@/utils/errorCode';
import { toast, showConfirm } from '@/utils/common';
let timeout = 10000;
let baseUrl = process.env.NODE_ENV === 'development' ? config.devbaseUrl : config.prodbaseUrl;
const request = config => {
const isToken = (config.headers || {}).isToken === false;
config.header = config.header || {};
if (getAccessToken() && !isToken) {
config.header['Authorization'] = 'Bearer ' + getAccessToken();
}
config.header['tenant-id'] = '1'; // 强制设置租户 ID
if (config.params) {
let url = config.url + '?' + tansParams(config.params);
url = url.slice(0, -1);
config.url = url;
}
return new Promise((resolve, reject) => {
uni.request({
method: config.method || 'get',
timeout: config.timeout || timeout,
url: config.baseUrl || baseUrl + config.url,
data: config.data,
header: config.header,
dataType: 'json'
}).then(response => {
let [error, res] = response;
if (error) {
toast('后端接口连接异常');
reject('后端接口连接异常');
return;
}
const code = res.data.code || 200;
const msg = errorCode[code] || res.data.msg || errorCode['default'];
if (code === 401) {
showConfirm('登录状态已过期,您可以继续留在该页面,或者重新登录?').then(res => {
if (res.confirm) {
removeToken();
uni.reLaunch({ url: '/pages/login' });
}
});
reject('无效的会话,或者会话已过期,请重新登录。');
} else if (code === 500) {
toast(msg);
reject('500');
} else if (code !== 200) {
toast(msg);
reject(code);
}
resolve(res.data);
}).catch(error => {
let { message } = error;
if (message === 'Network Error') message = '后端接口连接异常';
else if (message.includes('timeout')) message = '系统接口请求超时';
else if (message.includes('Request failed with status code')) message = '系统接口' + message.substr(message.length - 3) + '异常';
toast(message);
reject(error);
});
});
};
export default request;
对应的token文件:
// utils/auth.js
const AccessTokenKey = 'ACCESS_TOKEN';
const RefreshTokenKey = 'REFRESH_TOKEN';
export function getAccessToken() {
return uni.getStorageSync(AccessTokenKey);
}
export function getRefreshToken() {
return uni.getStorageSync(RefreshTokenKey);
}
export function setToken(token) {
uni.setStorageSync(AccessTokenKey, token.accessToken);
uni.setStorageSync(RefreshTokenKey, token.refreshToken);
}
export function removeToken() {
uni.removeStorageSync(AccessTokenKey);
uni.removeStorageSync(RefreshTokenKey);
}
相关接口请求:
// 在页面中调用封装的请求方法
import request from '@/utils/request';
export default {
methods: {
fetchData() {
request({
url: '/api/getData',
method: 'GET',
params: { id: 123 }
}).then(response => {
console.log('数据:', response);
}).catch(error => {
console.log('请求失败:', error);
});
}
}
}
process.env.NODE_ENV
是 Node.js 环境中用于获取当前应用运行环境的一个变量
在大多数前端框架(如 Vue、React)以及后端框架(如 Express)中,process.env.NODE_ENV 被广泛用于区分不同的开发环境
前端vue中可能已经标明了
在开发模式下:NODE_ENV=development npm run dev
在生产模式下:NODE_ENV=production npm run build
在 npm 脚本中,可以通过 cross-env 等工具来跨平台设置环境变量:
"scripts": {
"dev": "cross-env NODE_ENV=development vue-cli-service serve",
"build": "cross-env NODE_ENV=production vue-cli-service build"
}
另外一个接口超时时间,全局默认是20秒,如果时长不对,可以在单独某个接口设置:
// 上传图片
uploadImage(data) {
return upload({
url: '/infra/file/upload',
method: 'upload',
filePath: data.filePath,
timeout: 30000 // 设置超时时间为30秒
});
}