一个项目从开发到上线,大致要经过本地开发、测试环境、预发布环境和正式环境。那么在代码中如何区分这几个环境,然后再与对应的后端接口进行对接呢?
我们经历多几个项目的磨练后,总结出几个区分不同环境的方法。
- 从 URL 的域名进行区分;
- 构建时传入不同的全局变量;
- 容器提前定义好全局变量(一般后端服务使用);
下面我们来一一进行讲解。
我们先来定义几个环境变量:
enum EnvType {
LOCAL = 'local', // 本地开发环境
TESTING = 'testing', // 测试环境
PRE = 'pre', // 预发布环境
PRODUCTION = 'production', // 正式环境
}
1. 从 URL 的域名进行区分 #
若不同环境使用不同的域名,最简单的就是从 URL 域名进行区分。我们从 URL 中获取到 hostname 或 origin,然后进行判断。
// 通过url检测环境
const checkUrlHost = (): EnvType => {
const { hostname } = window.location || { hostname: null };
if (/test/.test(hostname)) {
return EnvType.TESTING;
}
if (/pre/.test(hostname)) {
return EnvType.PRE;
}
if (/local/.test(hostname)) {
return EnvType.LOCAL;
}
return EnvType.PRODUCTION;
};
注意:不要用整个 URL 地址来进行匹配,因为中间的路径或者携带的参数,可能会干扰到判断。
最开始我们还不能区分域名时,都是在同一个域名里进行开发,然后用不同的路径来区分,不过代码里用的是location.href
来区分。然后我们参数里要带着用户的昵称,这个用户的昵称正好有个test
的字样。导致项目把这个链接识别成了测试环境,然后就去请求测试环境的接口了。
2. 构建时传入不同的全局变量 #
现在我们发布项目时,都会先进行构建,然后再把 html 发布出去。那么我们就可以在构建时,把当前要发布的环境对应的变量传进去。
如何传入全局变量,还得看您正在使用的框架,如 React 的脚手架 create-react-app,传入的全局变量必须以REACT_APP_
开头;Vue 的脚手架 vue-cli,传入的全局变量必须以VUE_APP_
开头。
我们以 create-react-app 为例,使用REACT_APP_SITE_ENV
来指定所在的环境。
构建时:
# 本地开发环境
$ REACT_APP_SITE_ENV=local npm run start
# 测试环境
$ REACT_APP_SITE_ENV=testing npm run build
# 预发布环境
$ REACT_APP_SITE_ENV=pre npm run build
# 正式环境
$ npm run build # 不传入
$ REACT_APP_SITE_ENV=production npm run build # 或传入production
代码中获取该全局变量:
// 通过全局变量来检测环境
const getEnv = (): EnvType => {
const { REACT_APP_SITE_ENV }: any = process.env; // 命令行传入
// REACT_APP_SITE_ENV只能是EnvType中的值
// 若没有传入,或者不是这里面的几个值,则返回production
if (REACT_APP_SITE_ENV) {
// eslint-disable-next-line no-restricted-syntax
for (const key in EnvType) {
const env = (EnvType as any)[key];
if (env === REACT_APP_SITE_ENV) {
return REACT_APP_SITE_ENV;
}
}
}
return EnvType.PRODUCTION;
};
这里我们在构建流水线里指定 REACT_APP_SITE_ENV 为不同的值即可。
3. 容器提前定义好全局变量 #
前端代码一般是只有构建过程,但后端代码(如 nodejs 等),通常是【启动】或者【构建 + 启动】的过程,若有构建过程,则像上面那样在流水线里指定全局变量即可。
而启动这个程序一般是在容器里启动的,而流水线里的全局变量在构建完成后就失效了,留不到启动这一步。因此,若是在容器中启动的程序,您可以在容器里配置好全局变量,这样程序在启动时,就自动能够获取到了。
至于怎么配置,还得看您公司自己的配置。
如我们的配置是:
4. 总结 #
这里我们主要讲解了下如何在项目中来区分环境,很多开发者还会再额外地定义一些变量或方法,来直接使用,如:
// 定义变量
export const IS_LOCALENV = ENV === EnvType.LOCAL; // 是否是本地环境
export const IS_TESTENV = ENV === EnvType.TESTING; // 是否测试环境
export const IS_PREENV = ENV === EnvType.PRE; // 是否是预发布环境
export const IS_PRODENV = ENV === EnvType.PRODUCTION; // 是否是正式环境
// 定义方法
export const isPro = () => ENV === EnvType.PRODUCTION; // 是否是正式环境
或许未来新的框架或者部署流程,还是出现新的定义环境的方式。