axios.js 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. import Axios from 'axios'
  2. import defu from 'defu'
  3. // Axios.prototype cannot be modified
  4. const axiosExtra = {
  5. setBaseURL (baseURL) {
  6. this.defaults.baseURL = baseURL
  7. },
  8. setHeader (name, value, scopes = 'common') {
  9. for (const scope of Array.isArray(scopes) ? scopes : [ scopes ]) {
  10. if (!value) {
  11. delete this.defaults.headers[scope][name];
  12. continue
  13. }
  14. this.defaults.headers[scope][name] = value
  15. }
  16. },
  17. setToken (token, type, scopes = 'common') {
  18. const value = !token ? null : (type ? type + ' ' : '') + token
  19. this.setHeader('Authorization', value, scopes)
  20. },
  21. onRequest(fn) {
  22. this.interceptors.request.use(config => fn(config) || config)
  23. },
  24. onResponse(fn) {
  25. this.interceptors.response.use(response => fn(response) || response)
  26. },
  27. onRequestError(fn) {
  28. this.interceptors.request.use(undefined, error => fn(error) || Promise.reject(error))
  29. },
  30. onResponseError(fn) {
  31. this.interceptors.response.use(undefined, error => fn(error) || Promise.reject(error))
  32. },
  33. onError(fn) {
  34. this.onRequestError(fn)
  35. this.onResponseError(fn)
  36. },
  37. create(options) {
  38. return createAxiosInstance(defu(options, this.defaults))
  39. }
  40. }
  41. // Request helpers ($get, $post, ...)
  42. for (const method of ['request', 'delete', 'get', 'head', 'options', 'post', 'put', 'patch']) {
  43. axiosExtra['$' + method] = function () { return this[method].apply(this, arguments).then(res => res && res.data) }
  44. }
  45. const extendAxiosInstance = axios => {
  46. for (const key in axiosExtra) {
  47. axios[key] = axiosExtra[key].bind(axios)
  48. }
  49. }
  50. const createAxiosInstance = axiosOptions => {
  51. // Create new axios instance
  52. const axios = Axios.create(axiosOptions)
  53. axios.CancelToken = Axios.CancelToken
  54. axios.isCancel = Axios.isCancel
  55. // Extend axios proto
  56. extendAxiosInstance(axios)
  57. // Intercept to apply default headers
  58. axios.onRequest((config) => {
  59. config.headers = { ...axios.defaults.headers.common, ...config.headers }
  60. })
  61. // Setup interceptors
  62. setupProgress(axios)
  63. return axios
  64. }
  65. const setupProgress = (axios) => {
  66. if (process.server) {
  67. return
  68. }
  69. // A noop loading inteterface for when $nuxt is not yet ready
  70. const noopLoading = {
  71. finish: () => { },
  72. start: () => { },
  73. fail: () => { },
  74. set: () => { }
  75. }
  76. const $loading = () => {
  77. const $nuxt = typeof window !== 'undefined' && window['$nuxt']
  78. return ($nuxt && $nuxt.$loading && $nuxt.$loading.set) ? $nuxt.$loading : noopLoading
  79. }
  80. let currentRequests = 0
  81. axios.onRequest(config => {
  82. if (config && config.progress === false) {
  83. return
  84. }
  85. currentRequests++
  86. })
  87. axios.onResponse(response => {
  88. if (response && response.config && response.config.progress === false) {
  89. return
  90. }
  91. currentRequests--
  92. if (currentRequests <= 0) {
  93. currentRequests = 0
  94. $loading().finish()
  95. }
  96. })
  97. axios.onError(error => {
  98. if (error && error.config && error.config.progress === false) {
  99. return
  100. }
  101. currentRequests--
  102. if (Axios.isCancel(error)) {
  103. if (currentRequests <= 0) {
  104. currentRequests = 0
  105. $loading().finish()
  106. }
  107. return
  108. }
  109. $loading().fail()
  110. $loading().finish()
  111. })
  112. const onProgress = e => {
  113. if (!currentRequests || !e.total) {
  114. return
  115. }
  116. const progress = ((e.loaded * 100) / (e.total * currentRequests))
  117. $loading().set(Math.min(100, progress))
  118. }
  119. axios.defaults.onUploadProgress = onProgress
  120. axios.defaults.onDownloadProgress = onProgress
  121. }
  122. export default (ctx, inject) => {
  123. // runtimeConfig
  124. const runtimeConfig = ctx.$config && ctx.$config.axios || {}
  125. // baseURL
  126. const baseURL = process.browser
  127. ? (runtimeConfig.browserBaseURL || runtimeConfig.browserBaseUrl || runtimeConfig.baseURL || runtimeConfig.baseUrl || '/')
  128. : (runtimeConfig.baseURL || runtimeConfig.baseUrl || process.env._AXIOS_BASE_URL_ || 'http://localhost:3000/')
  129. // Create fresh objects for all default header scopes
  130. // Axios creates only one which is shared across SSR requests!
  131. // https://github.com/mzabriskie/axios/blob/master/lib/defaults.js
  132. const headers = {
  133. "common": {
  134. "Accept": "application/json, text/plain, */*"
  135. },
  136. "delete": {},
  137. "get": {},
  138. "head": {},
  139. "post": {},
  140. "put": {},
  141. "patch": {}
  142. }
  143. const axiosOptions = {
  144. baseURL,
  145. headers
  146. }
  147. // Proxy SSR request headers headers
  148. if (process.server && ctx.req && ctx.req.headers) {
  149. const reqHeaders = { ...ctx.req.headers }
  150. for (const h of ["accept","cf-connecting-ip","cf-ray","content-length","content-md5","content-type","host","x-forwarded-host","x-forwarded-port","x-forwarded-proto"]) {
  151. delete reqHeaders[h]
  152. }
  153. axiosOptions.headers.common = { ...reqHeaders, ...axiosOptions.headers.common }
  154. }
  155. if (process.server) {
  156. // Don't accept brotli encoding because Node can't parse it
  157. axiosOptions.headers.common['accept-encoding'] = 'gzip, deflate'
  158. }
  159. const axios = createAxiosInstance(axiosOptions)
  160. // Inject axios to the context as $axios
  161. ctx.$axios = axios
  162. inject('axios', axios)
  163. }