request.js 5.8 KB
Newer Older
sin's avatar
sin committed
1 2 3 4 5
import fetch from 'dva/fetch';
import { notification } from 'antd';
import router from 'umi/router';
import hash from 'hash.js';
import { isAntdPro } from './utils';
6
import { getLoginToken } from './cache';
7
import { setAuthority } from './authority';
sin's avatar
sin committed
8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26

const codeMessage = {
  200: '服务器成功返回请求的数据。',
  201: '新建或修改数据成功。',
  202: '一个请求已经进入后台排队(异步任务)。',
  204: '删除数据成功。',
  400: '发出的请求有错误,服务器没有进行新建或修改数据的操作。',
  401: '用户没有权限(令牌、用户名、密码错误)。',
  403: '用户得到授权,但是访问是被禁止的。',
  404: '发出的请求针对的是不存在的记录,服务器没有进行操作。',
  406: '请求的格式不可得。',
  410: '请求的资源被永久删除,且不会再得到的。',
  422: '当创建一个对象时,发生一个验证错误。',
  500: '服务器发生错误,请检查服务器。',
  502: '网关错误。',
  503: '服务不可用,服务器暂时过载或维护。',
  504: '网关超时。',
};

27 28
// 需要调整 login 界面的 code
const redirectLoginCode = {
29 30 31 32
  1002001011: '访问令牌不存在',
  1002001012: '访问令牌已过期',
  1002001013: '访问令牌已失效',
  1002001015: '账号未登陆',
33 34
};

sin's avatar
sin committed
35
function checkStatus(response) {
sin's avatar
sin committed
36 37 38 39
  if (response.status >= 200 && response.status < 300) {
    return response;
  }
  const errortext = codeMessage[response.status] || response.statusText;
sin's avatar
sin committed
40
  notification.warning({
sin's avatar
sin committed
41 42 43
    message: `请求错误 ${response.status}: ${response.url}`,
    description: errortext,
  });
sin's avatar
sin committed
44

sin's avatar
sin committed
45 46 47 48
  const error = new Error(errortext);
  error.name = response.status;
  error.response = response;
  throw error;
sin's avatar
sin committed
49
}
sin's avatar
sin committed
50

51 52 53 54 55 56 57 58
function checkCode(result) {
  if (result.code === undefined || result.code === 0) {
    return result;
  }
  notification.warning({
    message: `请求错误 ${result.code}`,
    description: result.message,
  });
59 60 61 62 63 64 65

  // 重定向到登录界面
  if (redirectLoginCode[result.code]) {
    setAuthority('');
    window.location.reload();
  }

66 67 68 69 70
  const error = new Error(result.message);
  error.result = result;
  throw error;
}

sin's avatar
sin committed
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114
const cachedSave = (response, hashcode) => {
  /**
   * Clone a response data and store it in sessionStorage
   * Does not support data other than json, Cache only json
   */
  const contentType = response.headers.get('Content-Type');
  if (contentType && contentType.match(/application\/json/i)) {
    // All data is saved as text
    response
      .clone()
      .text()
      .then(content => {
        sessionStorage.setItem(hashcode, content);
        sessionStorage.setItem(`${hashcode}:timestamp`, Date.now());
      });
  }
  return response;
};

/**
 * Requests a URL, returning a promise.
 *
 * @param  {string} url       The URL we want to request
 * @param  {object} [option] The options we want to pass to "fetch"
 * @return {object}           An object containing either "data" or "err"
 */
export default function request(url, option) {
  const options = {
    expirys: isAntdPro(),
    ...option,
  };
  /**
   * Produce fingerprints based on url and parameters
   * Maybe url has the same parameters
   */
  const fingerprint = url + (options.body ? JSON.stringify(options.body) : '');
  const hashcode = hash
    .sha256()
    .update(fingerprint)
    .digest('hex');

  const defaultOptions = {
    credentials: 'include',
  };
115

sin's avatar
sin committed
116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
  const newOptions = { ...defaultOptions, ...options };
  if (
    newOptions.method === 'POST' ||
    newOptions.method === 'PUT' ||
    newOptions.method === 'DELETE'
  ) {
    if (!(newOptions.body instanceof FormData)) {
      newOptions.headers = {
        Accept: 'application/json',
        'Content-Type': 'application/json; charset=utf-8',
        ...newOptions.headers,
      };
      newOptions.body = JSON.stringify(newOptions.body);
    } else {
      // newOptions.body is FormData
      newOptions.headers = {
        Accept: 'application/json',
        ...newOptions.headers,
      };
    }
  }

138 139
  // 将登陆的 accessToken 放到 header
  const loginToken = getLoginToken();
sin's avatar
sin committed
140
  if (loginToken && loginToken.accessToken) {
141 142 143 144 145
    const headers = {
      ...newOptions.headers,
      Authorization: `Bearer ${loginToken.accessToken}`,
    };
    newOptions.headers = headers;
sin's avatar
sin committed
146
  }
147

sin's avatar
sin committed
148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173
  const expirys = options.expirys && 60;
  // options.expirys !== false, return the cache,
  if (options.expirys !== false) {
    const cached = sessionStorage.getItem(hashcode);
    const whenCached = sessionStorage.getItem(`${hashcode}:timestamp`);
    if (cached !== null && whenCached !== null) {
      const age = (Date.now() - whenCached) / 1000;
      if (age < expirys) {
        const response = new Response(new Blob([cached]));
        return response.json();
      }
      sessionStorage.removeItem(hashcode);
      sessionStorage.removeItem(`${hashcode}:timestamp`);
    }
  }
  return fetch(url, newOptions)
    .then(checkStatus)
    .then(response => cachedSave(response, hashcode))
    .then(response => {
      // DELETE and 204 do not return data by default
      // using .json will report an error.
      if (newOptions.method === 'DELETE' || response.status === 204) {
        return response.text();
      }
      return response.json();
    })
174
    .then(checkCode)
sin's avatar
sin committed
175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198
    .catch(e => {
      const status = e.name;
      if (status === 401) {
        // @HACK
        /* eslint-disable no-underscore-dangle */
        window.g_app._store.dispatch({
          type: 'login/logout',
        });
        return;
      }
      // environment should not be used
      if (status === 403) {
        router.push('/exception/403');
        return;
      }
      if (status <= 504 && status >= 500) {
        router.push('/exception/500');
        return;
      }
      if (status >= 404 && status < 422) {
        router.push('/exception/404');
      }
    });
}