
vite.config.js
import { ConfigEnv, UserConfig, loadEnv } from 'vite'; import { viteMockServe } from 'vite-plugin-mock'; import vue from '@vitejs/plugin-vue'; import vueJsx from '@vitejs/plugin-vue-jsx'; import svgLoader from 'vite-svg-loader'; import path from 'path'; const CWD = process.cwd(); // https://vitejs.dev/config/ export default ({ mode }: ConfigEnv): UserCOnfig=> { const { VITE_BASE_URL } = loadEnv(mode, CWD); return { base: VITE_BASE_URL, resolve: { alias: { '@': path.resolve(__dirname, './src'), }, }, css: { preprocessorOptions: { less: { modifyVars: { hack: `true; @import (reference) "${path.resolve('src/style/variables.less')}";`, }, math: 'strict', JavascriptEnabled: true, }, }, }, plugins: [ vue(), vueJsx(), viteMockServe({ mockPath: 'mock', localEnabled: true, }), svgLoader(), ], server: { port: 3002, proxy: { '/api': { target: "http://130.124.200.10:1000", changeOrigin: true, rewrite: (path) => path.replace(/^\/api/, '') } } }, }; }; 封装的 axios
// axios 配置 可自行根据项目进行更改,只需更改该文件即可,其他文件可以不动 import isString from 'lodash/isString'; import merge from 'lodash/merge'; import type { InternalAxiosRequestConfig } from 'axios'; import type { AxiosTransform, CreateAxiosOptions } from './AxiosTransform'; import { VAxios } from './Axios'; import proxy from '@/config/proxy'; import { joinTimestamp, formatRequestDate, setObjToUrlParams } from './utils'; import { TOKEN_NAME } from '@/config/global'; import { ContentTypeEnum } from '@/constants'; const env = import.meta.env.MODE || 'development'; // 如果是 mock 模式 或 没启用直连代理 就不配置 host 会走本地 Mock 拦截 或 Vite 代理 let host = env === 'mock' || !proxy.isRequestProxy ? '' : proxy[env].host; // 数据处理,方便区分多种处理方式 const transform: AxiosTransform = { // 处理请求数据。如果数据不是预期格式,可直接抛出错误 transformRequestHook: (res, options) => { const { isTransformResponse, isReturnNativeResponse } = options; // 如果 204 无内容直接返回 const method = res.config.method?.toLowerCase(); if (res.status === 204 || method === 'put' || method === 'patch') { return res; } // 是否返回原生响应头 比如:需要获取响应头时使用该属性 if (isReturnNativeResponse) { return res; } // 不进行任何处理,直接返回 // 用于页面代码可能需要直接获取 code ,data ,message 这些信息时开启 if (!isTransformResponse) { return res.data; } // 错误的时候返回 const { data } = res; if (!data) { throw new Error('请求接口错误'); } // 这里 code 为 后台统一的字段,需要在 types.ts 内修改为项目自己的接口返回格式 const { code } = data; // 这里逻辑可以根据项目进行修改 const hasSuccess = data && code === 0; if (hasSuccess) { return data.data; } throw new Error(`请求接口错误, 错误码: ${code}`); }, // 请求前处理配置 beforeRequestHook: (config, options) => { const { apiUrl, isJoinPrefix, urlPrefix, joinParamsToUrl, formatDate, joinTime = true } = options; // 添加接口前缀 if (isJoinPrefix && urlPrefix && isString(urlPrefix)) { config.url = `${urlPrefix}${config.url}`; } // 将 baseUrl 拼接 if (apiUrl && isString(apiUrl)) { config.url = `${apiUrl}${config.url}`; } const params = config.params || {}; const data = config.data || false; if (formatDate && data && !isString(data)) { formatRequestDate(data); } if (config.method?.toUpperCase() === 'GET') { if (!isString(params)) { // 给 get 请求加上时间戳参数,避免从缓存中拿数据。 config.params = Object.assign(params || {}, joinTimestamp(joinTime, false)); } else { // 兼容 restful 风格 config.url = `${config.url + params}${joinTimestamp(joinTime, true)}`; config.params = undefined; } } else if (!isString(params)) { if (formatDate) { formatRequestDate(params); } if ( Reflect.has(config, 'data') && config.data && (Object.keys(config.data).length > 0 || data instanceof FormData) ) { config.data = data; config.params = params; } else { // 非 GET 请求如果没有提供 data ,则将 params 视为 data config.data = params; config.params = undefined; } if (joinParamsToUrl) { config.url = setObjToUrlParams(config.url as string, { ...config.params, ...config.data }); } } else { // 兼容 restful 风格 config.url += params; config.params = undefined; } return config; }, // 请求拦截器处理 requestInterceptors: (config, options) => { // 请求之前处理 config const token = localStorage.getItem(TOKEN_NAME); if (token && (config as Recordable)?.requestOptions?.withToken !== false) { // jwt token (config as Recordable).headers.Authorization = options.authenticationScheme ? `${options.authenticationScheme} ${token}` : token; } return config as InternalAxiosRequestConfig; }, // 响应拦截器处理 responseInterceptors: (res) => { return res; }, // 响应错误处理 responseInterceptorsCatch: (error: any) => { const { config } = error; if (!config || !config.requestOptions.retry) return Promise.reject(error); config.retryCount = config.retryCount || 0; if (config.retryCount >= config.requestOptions.retry.count) return Promise.reject(error); config.retryCount += 1; const backoff = new Promise((resolve) => { setTimeout(() => { resolve(config); }, config.requestOptions.retry.delay || 1); }); config.headers = { ...config.headers, 'Content-Type': ContentTypeEnum.Json }; return backoff.then((config) => request.request(config)); }, }; function createAxios(opt?: Partial<CreateAxiosOptions>) { return new VAxios( merge( <CreateAxiosOptions>{ // https://developer.mozilla.org/en-US/docs/Web/HTTP/Authentication#authentication_schemes // 例如: authenticationScheme: 'Bearer' authenticationScheme: '', // 超时 timeout: 10 * 1000, // 携带 Cookie withCredentials: true, // 头信息 headers: { 'Content-Type': ContentTypeEnum.Json }, // 数据处理方式 transform, // 配置项,下面的选项都可以在独立的接口请求中覆盖 requestOptions: { // 接口地址 apiUrl: host, // 是否自动添加接口前缀 isJoinPrefix: true, // 接口前缀 // 例如: https://www.baidu.com/api // urlPrefix: '/api' urlPrefix: '/api', // 是否返回原生响应头 比如:需要获取响应头时使用该属性 isReturnNativeResponse: false, // 需要对返回数据进行处理 isTransformResponse: true, // post 请求的时候添加参数到 url joinParamsToUrl: false, // 格式化提交参数时间 formatDate: true, // 是否加入时间戳 joinTime: true, // 忽略重复请求 ignoreRepeatRequest: true, // 是否携带 token withToken: true, // 重试 retry: { count: 3, delay: 1000, }, }, }, opt || {}, ), ); } export const request = createAxios(); 配置地址
export default { isRequestProxy: true, development: { // 开发环境接口请求 // host: 'https://service-exndqyuk-1257786608.gz.apigw.tencentcs.com', host: 'http://130.124.200.10:1000', // 开发环境 cdn 路径 cdn: '', }, test: { // 测试环境接口地址 host: 'https://service-exndqyuk-1257786608.gz.apigw.tencentcs.com', // 测试环境 cdn 路径 cdn: '', }, release: { // 正式环境接口地址 host: 'https://service-bv448zsw-1257786608.gz.apigw.tencentcs.com', // 正式环境 cdn 路径 cdn: '', }, site: { // TDesign 部署特殊需要 与 release 功能一致 host: 'https://service-bv448zsw-1257786608.gz.apigw.tencentcs.com', // 正式环境 cdn 路径 cdn: '', }, }; 开发模式没有走代理,显示跨域
1 KotlinAmai 2023-02-21 15:50:22 +08:00 development: { // 开发环境接口请求 host: '/api', // 开发环境 cdn 路径 cdn: '', } |
2 estk 2023-02-21 15:50:31 +08:00 via iPhone Clash 开始增强模式,所有命令行都走它流量,其它软件不太懂 |
3 296727 2023-02-21 15:51:07 +08:00 你试试访问 IP:port/api ,是不是可以访问 那是不是你把请求地址写死了 都写死了为什么还会走代理 |
4 yuan321 OP @actar 可以了谢谢。不过很奇怪 我注释的腾讯的地址 https://service-exndqyuk-1257786608.gz.apigw.tencentcs.com 跨域也不报错,应该是他们后端做了允许跨域的,但是我们的后端用 koa 开发的也做了跨域处理的,但是死活都报跨域错误。。。。 |
5 KotlinAmai 2023-02-21 16:07:29 +08:00 @yuan321 你的配置里面 withCredentials 为 true ,那么就应该注意 跨域接口的响应头 Access-Control-Allow-Origin 的值就不能为 * ,你可以检查一下 |
6 yuan321 OP @actar 果然如此,当我把 withCredentials 改为 false 的时候,我直接请求服务器的地址,就算不用代理,也不报跨域的错误了。学到了这个 withCredentials 的知识了,原来这个需要后端返回 Access-Control-Allow-Origin 的值应该为域名才行。难怪之前怎么做都报跨域错误的,谢谢你, |
7 yuan321 OP @actar 我们在 koa 后端添加了如下跨域设置但是还是报跨域错误。``` app.use(async (ctx, next)=> { console.log(ctx.request.header.origin) ctx.set('Access-Control-Allow-Origin', ctx.request.header.origin); if (ctx.method == 'OPTIONS') { ctx.body = 200; } else { await next(); } });``` |
8 KotlinAmai 2023-02-21 20:33:54 +08:00 @yuan321 这个响应头也需要配置的 Access-Control-Allow-Methods ,可以检查一下。 Access-Control-Allow-Methods=GET,HEAD,PUT,POST,DELETE,PATCH 你用的 koa ,官方提供了一个 cors 的中间件 https://github.com/koajs/cors ,你们跨域用的中间件还是自己写的啊。 |