request.js 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. import axios from "axios";
  2. import { ElNotification, ElMessageBox, ElMessage, ElLoading } from "element-plus";
  3. import { getToken } from "/src/utils/auth";
  4. import errorCode from "/src/utils/errorCode";
  5. import { tansParams, blobValidate } from "/src/utils/ruoyi";
  6. import cache from "/src/plugins/cache";
  7. import { saveAs } from "file-saver";
  8. import useUserStore from "/src/store/modules/user";
  9. let downloadLoadingInstance;
  10. // 是否显示重新登录
  11. export let isRelogin = {
  12. show: false,
  13. };
  14. axios.defaults.headers["Content-Type"] = "application/json;charset=utf-8";
  15. // 创建axios实例
  16. const service = axios.create({
  17. // axios中请求配置有baseURL选项,表示请求URL公共部分
  18. baseURL: import.meta.env.VITE_APP_BASE_API,
  19. // 超时
  20. timeout: 100000,
  21. });
  22. // request拦截器
  23. service.interceptors.request.use(
  24. (config) => {
  25. // 是否需要设置 token
  26. const isToken = (config.headers || {}).isToken === false;
  27. // 是否需要防止数据重复提交
  28. const isRepeatSubmit = (config.headers || {}).repeatSubmit === false;
  29. if (getToken() && !isToken) {
  30. config.headers["Authorization"] = "Bearer " + getToken(); // 让每个请求携带自定义token 请根据实际情况自行修改
  31. }
  32. // get请求映射params参数
  33. if (config.method === "get" && config.params) {
  34. let url = config.url + "?" + tansParams(config.params);
  35. url = url.slice(0, -1);
  36. config.params = {};
  37. config.url = url;
  38. }
  39. if (!isRepeatSubmit && (config.method === "post" || config.method === "put")) {
  40. const requestObj = {
  41. url: config.url,
  42. data: typeof config.data === "object" ? JSON.stringify(config.data) : config.data,
  43. time: new Date().getTime(),
  44. };
  45. const sessionObj = cache.session.getJSON("sessionObj");
  46. if (sessionObj === undefined || sessionObj === null || sessionObj === "") {
  47. cache.session.setJSON("sessionObj", requestObj);
  48. } else {
  49. const s_url = sessionObj.url; // 请求地址
  50. const s_data = sessionObj.data; // 请求数据
  51. const s_time = sessionObj.time; // 请求时间
  52. const interval = 1000; // 间隔时间(ms),小于此时间视为重复提交
  53. if (s_data === requestObj.data && requestObj.time - s_time < interval && s_url === requestObj.url) {
  54. console.log(requestObj);
  55. const message = "数据正在处理,请勿重复提交";
  56. return Promise.reject(new Error(message));
  57. } else {
  58. cache.session.setJSON("sessionObj", requestObj);
  59. }
  60. }
  61. }
  62. return config;
  63. },
  64. (error) => {
  65. console.log(error);
  66. Promise.reject(error);
  67. }
  68. );
  69. // 响应拦截器
  70. service.interceptors.response.use(
  71. (res) => {
  72. // 未设置状态码则默认成功状态
  73. const code = res.data.code || 200;
  74. // 获取错误信息
  75. const msg = errorCode[code] || res.data.msg || errorCode["default"];
  76. // 二进制数据则直接返回
  77. if (res.request.responseType === "blob" || res.request.responseType === "arraybuffer") {
  78. return res.data;
  79. }
  80. if (code === 401) {
  81. if (!isRelogin.show) {
  82. isRelogin.show = true;
  83. ElMessageBox.confirm("登录状态已过期,您可以继续留在该页面,或者重新登录", "系统提示", {
  84. confirmButtonText: "重新登录",
  85. cancelButtonText: "取消",
  86. type: "warning",
  87. })
  88. .then(() => {
  89. isRelogin.show = false;
  90. useUserStore()
  91. .logOut()
  92. .then(() => {
  93. location.href = "/index";
  94. });
  95. })
  96. .catch(() => {
  97. isRelogin.show = false;
  98. });
  99. }
  100. return Promise.reject("无效的会话,或者会话已过期,请重新登录。");
  101. } else if (code === 500) {
  102. ElMessage({
  103. message: msg,
  104. type: "error",
  105. });
  106. return Promise.reject(new Error(msg));
  107. } else if (code === 601) {
  108. ElMessage({
  109. message: msg,
  110. type: "warning",
  111. });
  112. return Promise.reject(new Error(msg));
  113. } else if (code !== 200) {
  114. ElNotification.error({
  115. title: msg,
  116. });
  117. return Promise.reject("error");
  118. } else {
  119. return Promise.resolve(res.data);
  120. }
  121. },
  122. (error) => {
  123. let { message } = error;
  124. if (message == "Network Error") {
  125. message = "后端接口连接异常";
  126. } else if (message.includes("timeout")) {
  127. message = "系统接口请求超时";
  128. } else if (message.includes("Request failed with status code")) {
  129. message = "系统接口" + message.substr(message.length - 3) + "异常";
  130. } else if (message.includes("java.lang.NoClassDefFoundError")) {
  131. message = "服务器部署中,请稍后再试";
  132. }
  133. ElMessage({
  134. message: message,
  135. type: "error",
  136. duration: 5 * 1000,
  137. });
  138. return Promise.reject(error);
  139. }
  140. );
  141. // 通用下载方法
  142. export function download(url, params, filename, config) {
  143. downloadLoadingInstance = ElLoading.service({
  144. text: "正在下载数据,请稍候",
  145. background: "rgba(0, 0, 0, 0.7)",
  146. });
  147. return service
  148. .post(url, params, {
  149. transformRequest: [
  150. (params) => {
  151. return tansParams(params);
  152. },
  153. ],
  154. headers: {
  155. "Content-Type": "application/x-www-form-urlencoded",
  156. },
  157. responseType: "blob",
  158. ...config,
  159. })
  160. .then(async (data) => {
  161. const isLogin = await blobValidate(data);
  162. if (isLogin) {
  163. const blob = new Blob([data]);
  164. saveAs(blob, filename);
  165. } else {
  166. const resText = await data.text();
  167. const rspObj = JSON.parse(resText);
  168. const errMsg = errorCode[rspObj.code] || rspObj.msg || errorCode["default"];
  169. ElMessage.error(errMsg);
  170. }
  171. downloadLoadingInstance.close();
  172. })
  173. .catch((r) => {
  174. console.error(r);
  175. ElMessage.error("下载文件出现错误,请联系管理员!");
  176. downloadLoadingInstance.close();
  177. });
  178. }
  179. export function get(url, params = {}) {
  180. return new Promise((resolve, reject) => {
  181. service
  182. .get(url, {
  183. params: params,
  184. })
  185. .then(
  186. (response) => {
  187. setTimeout(() => {}, 500);
  188. resolve(response);
  189. },
  190. (err) => {
  191. reject(err);
  192. }
  193. );
  194. });
  195. }
  196. /**
  197. * 封装post请求
  198. * @param url
  199. * @param data
  200. * @returns {Promise}
  201. */
  202. export function post(url, data = {}, method) {
  203. return new Promise((resolve, reject) => {
  204. service({
  205. method: method || "post",
  206. url: url,
  207. data: data,
  208. })
  209. .then((res) => {
  210. resolve(res.data);
  211. })
  212. .catch((err) => {
  213. reject(err);
  214. });
  215. });
  216. }
  217. export function postFile(url, data = {}, method) {
  218. return new Promise((resolve, reject) => {
  219. service({
  220. method: method || "post",
  221. url: url,
  222. data: data,
  223. responseType: "blob",
  224. })
  225. .then((res) => {
  226. resolve(res);
  227. })
  228. .catch((err) => {
  229. reject(err);
  230. });
  231. });
  232. }
  233. export function getFile(url, data = {}, method) {
  234. return new Promise((resolve, reject) => {
  235. service({
  236. method: method || "get",
  237. url: url,
  238. params: data,
  239. responseType: "blob",
  240. })
  241. .then((res) => {
  242. resolve(res);
  243. })
  244. .catch((err) => {
  245. reject(err);
  246. });
  247. });
  248. }
  249. export function postUploadFile(url, data = {}, method) {
  250. return new Promise((resolve, reject) => {
  251. service({
  252. method: method || "post",
  253. url: url,
  254. data: data,
  255. headers: {
  256. "Content-Type": "multipart/form-data",
  257. },
  258. })
  259. .then((res) => {
  260. resolve(res);
  261. })
  262. .catch((err) => {
  263. reject(err);
  264. });
  265. });
  266. }
  267. export function postUploadAndDownloadFile(url, data = {}, method) {
  268. return new Promise((resolve, reject) => {
  269. service({
  270. method: method || "post",
  271. url: url,
  272. data: data,
  273. headers: {
  274. "Content-Type": "multipart/form-data",
  275. },
  276. responseType: "blob",
  277. })
  278. .then((res) => {
  279. resolve(res);
  280. })
  281. .catch((err) => {
  282. reject(err);
  283. });
  284. });
  285. }
  286. export default service;