index.ts 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. import { isDev, ignoreTokens, config } from "@/config";
  2. import { locale, t } from "../locale";
  3. import { isNull, isObject, parse, storage } from "../utils";
  4. import { useStore } from "../store";
  5. import { useUi } from "@/uni_modules/cool-ui";
  6. // 请求参数类型定义
  7. export type RequestOptions = {
  8. url: string; // 请求地址
  9. method?: RequestMethod; // 请求方法
  10. data?: any; // 请求体数据
  11. params?: any; // URL参数
  12. header?: any; // 请求头
  13. timeout?: number; // 超时时间
  14. withCredentials?: boolean; // 是否携带凭证
  15. firstIpv4?: boolean; // 是否优先使用IPv4
  16. enableChunked?: boolean; // 是否启用分块传输
  17. };
  18. // 响应数据类型定义
  19. export type Response = {
  20. code?: number;
  21. message?: string;
  22. data?: any;
  23. };
  24. const ui = useUi();
  25. // 请求队列(用于等待token刷新后继续请求)
  26. let requests: ((token: string) => void)[] = [];
  27. // 标记token是否正在刷新
  28. let isRefreshing = false;
  29. // 判断当前url是否忽略token校验
  30. const isIgnoreToken = (url: string) => {
  31. return ignoreTokens.some((e) => {
  32. const pattern = e.replace(/\*/g, ".*");
  33. return new RegExp(pattern).test(url);
  34. });
  35. };
  36. /**
  37. * 通用请求方法
  38. * @param options 请求参数
  39. * @returns Promise<T>
  40. */
  41. export function request(options: RequestOptions): Promise<any | null> {
  42. let { url, method = "GET", data = {}, header = {}, timeout = 60000 } = options;
  43. const { user } = useStore();
  44. // 拼接基础url
  45. if (!url.startsWith("http")) {
  46. url = config.baseUrl + url;
  47. }
  48. // 开发环境下打印请求信息
  49. if (isDev) {
  50. console.log(`[${method}] ${url}`);
  51. }
  52. // 获取当前token
  53. let Authorization: string | null = user.token;
  54. // 如果是忽略token的接口,则不携带token
  55. if (isIgnoreToken(url)) {
  56. Authorization = null;
  57. }
  58. return new Promise((resolve, reject) => {
  59. // 发起请求的实际函数
  60. const next = () => {
  61. uni.request({
  62. url,
  63. method,
  64. data,
  65. header: {
  66. Authorization,
  67. language: locale.value,
  68. ...(header as UTSJSONObject)
  69. },
  70. timeout,
  71. success(res) {
  72. // 401 无权限
  73. if (res.statusCode == 401) {
  74. user.logout();
  75. ui.showToast({
  76. message: "无权限",
  77. type: "warn",
  78. })
  79. reject({ message: t("无权限") } as Response);
  80. }
  81. // 502 服务异常
  82. else if (res.statusCode == 502) {
  83. ui.showToast({
  84. message: "服务异常",
  85. type: "warn",
  86. })
  87. reject({
  88. message: t("服务异常")
  89. } as Response);
  90. }
  91. // 404 未找到
  92. else if (res.statusCode == 404) {
  93. ui.showToast({
  94. message: `[404] ${url}`,
  95. type: "warn",
  96. })
  97. return reject({
  98. message: `[404] ${url}`
  99. } as Response);
  100. }
  101. // 200 正常响应
  102. else if (res.statusCode == 200) {
  103. if (res.data == null) {
  104. resolve(null);
  105. } else if (!isObject(res.data as any)) {
  106. resolve(res.data);
  107. } else {
  108. // 解析响应数据
  109. const { code, message, data } = parse<Response>(
  110. res.data ?? { code: 0 }
  111. )!;
  112. if (code == 200 || code == 0) {
  113. resolve(data);
  114. } else {
  115. ui.showToast({
  116. message: message ?? t("服务异常"),
  117. type: "warn",
  118. })
  119. reject({ message: message } as Response);
  120. }
  121. }
  122. } else {
  123. ui.showToast({
  124. message: t("服务异常"),
  125. type: "warn",
  126. })
  127. reject({ message: t("服务异常") } as Response);
  128. }
  129. },
  130. // 网络请求失败
  131. fail(err) {
  132. reject({ message: err.errMsg } as Response);
  133. }
  134. });
  135. };
  136. next();
  137. });
  138. }
  139. export function useGet(url: string, params?: any, config?: RequestOptions): Promise<Response> {
  140. const options: RequestOptions = {
  141. url,
  142. params,
  143. method: 'GET',
  144. ...config,
  145. }
  146. return request(options)
  147. }
  148. export function usePost(url: string, data?: any, config?: RequestOptions): Promise<Response> {
  149. const options: RequestOptions = {
  150. url,
  151. data,
  152. method: 'POST',
  153. ...config,
  154. }
  155. return request(options)
  156. }
  157. export function usePut(url: string, data?: any, config?: RequestOptions): Promise<Response> {
  158. const options: RequestOptions = {
  159. url,
  160. data,
  161. method: 'PUT',
  162. ...config,
  163. }
  164. return request(options)
  165. }
  166. export function useDelete(url: string, data?: any, config?: RequestOptions): Promise<Response> {
  167. const options: RequestOptions = {
  168. url,
  169. data,
  170. method: 'DELETE',
  171. ...config,
  172. }
  173. return request(options)
  174. }