Skip to content
On this page

loadEnv 源码分析

env 的使用

ts
import { loadEnv } from 'vite'

loadEnv('development', process.cwd())
  • 检查 process.cwd()路径下.env.development.local、.env.development、.env.local、.env 这四个环境文件。
  • 输出 NODEENV 和 VITE开头的键值对。
  • VITE_开头的键值对后面的不会覆盖前面的。
  • NODE_ENV 的值后面的会覆盖前面的。

[node 中的 process.cwd()](/guide/node/process)

相关 API

源码

lookupFile

ts
// ↓加载node的path模块
var path$1 = require('path')
var fs$2 = require('fs')

// ↓传入的如果是一个包含'default'的key的对象,则输出对象的'default'的key对应的value。
function _interopDefaultLegacy(e) {
  return e && typeof e === 'object' && 'default' in e ? e['default'] : e
}

// ↓结合上面二者
var path__default = /*#__PURE__*/ _interopDefaultLegacy(path$1)
var fs__default = /*#__PURE__*/ _interopDefaultLegacy(fs$2)

// ↓检查dir下是否有formats中的路径,返回有就返回路径或者文件
function lookupFile(dir, formats, pathOnly = false) {
  for (const format of formats) {
    // ↓输出dir和formats连接后的路径
    const fullPath = path__default.join(dir, format)
    // ↓同步的检查该路径是否存在,并且该路径对应的是一个文件
    if (fs__default.existsSync(fullPath) && fs__default.statSync(fullPath).isFile()) {
      // ↓是否只要输出路径,否则输出文件
      return pathOnly ? fullPath : fs__default.readFileSync(fullPath, 'utf-8')
    }
  }
  // ↓上面的路径都不满足输出条件,那么再检查一遍检查传入的dir的目录名
  const parentDir = path__default.dirname(dir)
  if (parentDir !== dir) {
    return lookupFile(parentDir, formats, pathOnly)
  }
}

loadEnv 源码

ts
// 从传参的root目录下获取
// 按顺序 .env.${mode}.local、.env.${mode}、.env.local、.env这四个环境文件
// 输出文件内配置的对象
function loadEnv(mode, root, prefix = 'VITE_') {
  if (mode === 'local') {
    // ↓如果第一个参数传入'local',就报错:
    // ↓"local "不能用作模式名称,因为它与``.env文件的.local后缀冲突。
    throw new Error(
      `"local" cannot be used as a mode name because it conflicts with ` +
        `the .local postfix for .env files.`
    )
  }
  // ↓待输出的环境变量对象
  const env = {}
  // ↓要读取的四个文件名称的字符串数组
  const envFiles = [
    /** mode local file */ `.env.${mode}.local`,
    /** mode file */ `.env.${mode}`,
    /** local file */ `.env.local`,
    /** default file */ `.env`,
  ]
  // 检查是否有实际的以VITE_*开头的环境变量。

  // 这些通常是Node内联提供的env对象,并应优先考虑。
  for (const key in process.env) {
    if (key.startsWith(prefix) && env[key] === undefined) {
      env[key] = process.env[key]
    }
  }

  for (const file of envFiles) {
    // ↓检查根目录下是否有指定配置文件
    const path = lookupFile(root, [file], true)
    if (path) {
      // ↓以换行为单位输出文件中KEY=VAL格式的到结果对象中
      const parsed = main$2.parse(fs__default.readFileSync(path), {
        debug: !!process.env.DEBUG || undefined,
      })
      // ↓让环境变量互相使用,这个方法我没仔细研究。不是很懂
      main$1({
        parsed,
        // ↓防止process.env修改
        ignoreProcessEnv: true,
      })
      // 只输出以prefix开头的key
      for (const [key, value] of Object.entries(parsed)) {
        // ↓只有这个key在前面没有加载过才赋值
        if (key.startsWith(prefix) && env[key] === undefined) {
          env[key] = value
        } else if (key === 'NODE_ENV') {
          // 在.env文件中覆盖NODE_ENV。
          process.env.VITE_USER_NODE_ENV = value
        }
      }
    }
  }
  return env
}