RequestModules.ts 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. /**
  2. * 这里写的是业务相关的:
  3. * * 请求数据处理函数。
  4. * * 自定义请求模块。
  5. * * 自定义错误报告处理函数。
  6. */
  7. import AppCofig, { isDev } from "@/common/config/AppCofig";
  8. import ApiCofig from "@/common/config/ApiCofig";
  9. import {
  10. RequestApiError, RequestApiResult, type RequestApiErrorType,
  11. RequestCoreInstance, RequestOptions,
  12. defaultResponseDataGetErrorInfo, defaultResponseDataHandlerCatch,
  13. RequestResponse,
  14. WebFetchImplementer,
  15. StringUtils,
  16. appendGetUrlParams,
  17. appendPostParams,
  18. } from "@imengyu/imengyu-utils";
  19. import type { DataModel, KeyValue, NewDataModel } from "@imengyu/js-request-transform";
  20. import { useAuthStore } from "@/stores/auth";
  21. import { Modal } from "ant-design-vue";
  22. /**
  23. * 不报告错误的 code
  24. */
  25. const notReportErrorCode = [401] as number[];
  26. const notReportMessages = [
  27. /请授权绑定手机号/g,
  28. ] as RegExp[];
  29. function matchNotReportMessage(str: string) {
  30. for (let i = 0; i < notReportMessages.length; i++) {
  31. if (notReportMessages[i].test(str))
  32. return true;
  33. }
  34. return false;
  35. }
  36. //请求拦截器
  37. function requestInceptor(url: string, req: RequestOptions) {
  38. //获取store中的token,追加到头;
  39. const autoStore = useAuthStore();
  40. if (StringUtils.isNullOrEmpty((req.header as KeyValue).token as string)) {
  41. req.header['token'] = autoStore.token;
  42. req.header['__token__'] = autoStore.token;
  43. }
  44. if (req.method == 'GET') {
  45. //追加GET参数
  46. url = appendGetUrlParams(url, 'main_body_id', ApiCofig.mainBodyId);
  47. } else {
  48. req.data = appendPostParams(req.data,'main_body_id', ApiCofig.mainBodyId);
  49. }
  50. return { newUrl: url, newReq: req };
  51. }
  52. //响应数据处理函数
  53. function responseDataHandler<T extends DataModel>(response: RequestResponse, req: RequestOptions, resultModelClass: NewDataModel|undefined, instance: RequestCoreInstance<T>, apiName: string | undefined): Promise<RequestApiResult<T>> {
  54. return new Promise<RequestApiResult<T>>((resolve, reject) => {
  55. const method = req.method || 'GET';
  56. response.json().then((json) => {
  57. if (response.ok) {
  58. if (!json) {
  59. reject(new RequestApiError(
  60. 'businessError',
  61. '后端未返回数据',
  62. '',
  63. response.status,
  64. null,
  65. null,
  66. req,
  67. apiName,
  68. response.url
  69. ));
  70. return;
  71. }
  72. //code == 0 错误
  73. if (json.code === 0) {
  74. handleError();
  75. return;
  76. }
  77. //处理后端的数据
  78. let message = '未知错误';
  79. let data = {} as any;
  80. //后端返回格式不统一,所以在这里处理格式
  81. if (typeof json.data === 'object') {
  82. data = json.data;
  83. message = json.data?.msg || response.statusText;
  84. }
  85. else {
  86. //否则返回上层对象
  87. data = json;
  88. message = json.msg || response.statusText;
  89. }
  90. resolve(new RequestApiResult(
  91. resultModelClass ?? instance.config.modelClassCreator,
  92. json?.code || response.status,
  93. message,
  94. data,
  95. json
  96. ));
  97. }
  98. else {
  99. handleError();
  100. }
  101. function handleError() {
  102. let errType : RequestApiErrorType = 'unknow';
  103. let errString = '';
  104. let errCodeStr = '';
  105. if (typeof json.message === 'string')
  106. errString = json.message;
  107. if (typeof json.msg === 'string')
  108. errString += json.msg;
  109. if (StringUtils.isStringAllEnglish(errString))
  110. errString = '服务器返回:' + errString;
  111. //错误处理
  112. if (errString) {
  113. //如果后端有返回错误信息,则收集错误信息并返回
  114. errType = 'businessError';
  115. if (typeof json.data === 'object' && json.data?.errmsg) {
  116. errString += '\n' + json.data.errmsg;
  117. }
  118. if (typeof json.errors === 'object') {
  119. for (const key in json.errors) {
  120. if (Object.prototype.hasOwnProperty.call(json.errors, key)) {
  121. errString += '\n' + json.errors[key];
  122. }
  123. }
  124. }
  125. } else {
  126. const res = defaultResponseDataGetErrorInfo(response, json);
  127. errType = res.errType;
  128. errString = res.errString;
  129. errCodeStr = res.errCodeStr;
  130. }
  131. reject(new RequestApiError(
  132. errType,
  133. errString,
  134. errCodeStr,
  135. response.status,
  136. null,
  137. null,
  138. req,
  139. apiName,
  140. response.url
  141. ));
  142. }
  143. }).catch((err) => {
  144. //错误统一处理
  145. defaultResponseDataHandlerCatch(method, req, response, null, err, apiName, response.url, reject, instance);
  146. });
  147. });
  148. }
  149. //错误报告处理
  150. function responseErrReoprtInceptor<T extends DataModel>(instance: RequestCoreInstance<T>, response: RequestApiError) {
  151. return (
  152. (response.errorType !== 'businessError' && response.errorType !== 'networkError') ||
  153. notReportErrorCode.indexOf(response.code) >= 0 ||
  154. matchNotReportMessage(response.errorMessage) === true
  155. );
  156. }
  157. //错误报告处理
  158. export function reportError<T extends DataModel>(instance: RequestCoreInstance<T>, response: RequestApiError | Error) {
  159. if (isDev) {
  160. if (response instanceof RequestApiError) {
  161. (globalThis as any).$error({
  162. message: `请求错误 ${response.apiName} : ${response.errorMessage}`,
  163. detail: response.toString() +
  164. '\r\n请求接口:' + response.apiName +
  165. '\r\n请求地址:' + response.apiUrl +
  166. '\r\n请求参数:' + JSON.stringify(response.rawRequest) +
  167. '\r\n返回参数:' + JSON.stringify(response.rawData) +
  168. '\r\n状态码:' + response.code +
  169. '\r\n信息:' + response.errorCodeMessage,
  170. type: 'error',
  171. });
  172. } else {
  173. (globalThis as any).$error({
  174. message: '错误报告 代码错误',
  175. detail: response?.stack || ('' + response),
  176. type: 'error',
  177. });
  178. }
  179. } else {
  180. let errMsg = '';
  181. if (response instanceof RequestApiError)
  182. errMsg = response.errorMessage + '。';
  183. errMsg += '服务出现了异常,请稍后重试或联系客服。';
  184. errMsg += '版本:' + AppCofig.version;
  185. Modal.error({
  186. title: '抱歉',
  187. content: errMsg,
  188. });
  189. }
  190. }
  191. /**
  192. * App服务请求模块
  193. */
  194. export class AppServerRequestModule<T extends DataModel> extends RequestCoreInstance<T> {
  195. constructor() {
  196. super(WebFetchImplementer);
  197. this.config.baseUrl = ApiCofig.serverProd;
  198. this.config.errCodes = []; //
  199. this.config.requestInceptor = requestInceptor;
  200. this.config.responseDataHandler = responseDataHandler;
  201. this.config.responseErrReoprtInceptor = responseErrReoprtInceptor;
  202. this.config.reportError = reportError;
  203. }
  204. }