如何使用 ts-node 加载环境变量?



我尝试了一些实现,但没有一个成功。

第一次尝试

在 package.json 脚本中使用eval"fetch:data": "eval $(cat .env) ts-node -O '{"module":"commonjs"}' ./bin/build-api-data.ts".

这会导致 JSON 解析错误,因为 eval 出于某种原因删除了我的引号。

undefined:1
{module:commonjs}
^
SyntaxError: Unexpected token m in JSON at position 1

第二次尝试

使用 dotenv,我在这里遇到的问题是这是一个竞争条件,导致如下错误:

$ CANDID_ENV=local ts-node -O '{"module":"commonjs"}' ./bin/build-api-data.ts
/Users/lassiter.gregg/code/candidco-web/node_modules/contentful/dist/webpack:/contentful/contentful.js:49
throw new TypeError('Expected parameter accessToken')
^
TypeError: Expected parameter accessToken

代码示例

import fs from 'fs';
import path from 'path';
import fetchApiData from '../lib/apiData';
import dotEnv from 'dotenv-safe';
const { CANDID_ENV } = process.env;
const isLocalBuild = CANDID_ENV === 'local';
console.log(dotEnv);
const API_DATA_FILENAME = 'api_data.json';
const ensureDirectoryExistence = filePath => {
var dirname = path.dirname(filePath);
if (fs.existsSync(dirname)) {
return true;
}
ensureDirectoryExistence(dirname);
fs.mkdirSync(dirname);
};
const writeData = (filename, data) => {
const filePath = path.join(__dirname, '..', '.data', filename);
ensureDirectoryExistence(filePath);
fs.writeFileSync(filePath, JSON.stringify(data));
console.log('API data stored', filePath);
};
const fetchAndStoreApiData = async () => {
console.log('Fetching all API data');
await dotEnv.config({
path: isLocalBuild ? './.env' : `./.env.${CANDID_ENV}`,
});
const newData = await fetchApiData();
writeData(API_DATA_FILENAME, newData);
};
const init = async () => {
fetchAndStoreApiData();
};
if (require.main === module) {
init();
}

在上面的例子中,我尝试在文件顶部、初始化和函数中执行dotenv.config,如您所见。它总是抛出相同的错误,即内容未获取所需的 env 变量。也就是说,如果我登录process.env并注释掉与fetchApiData相关的代码,那么我会看到所有环境变量。这就是为什么我认为这是一个比赛时间条件,但无法找到与我自己的问题类似的东西。

此外,使这更加棘手的是,这是一个自定义脚本,必须在节点和 esnext 环境中工作。因此,我使用我不太喜欢但尚未发现的语法遇到了相当多的棘手的导入/导出问题(例如export = someFunction)。

我是否正确看到,您正在尝试使用使用 env 变量初始化的变量配置 dotenv?我认为这不会奏效。 Dotenv 的工作是将 env 变量加载到 process.env。您必须尽早在应用程序中配置它。

更多关于它的信息在这里: https://www.npmjs.com/package/dotenv

我遇到了类似的问题,最终加载了带有"require"标志dotenv模块:

> ts-node --help
...
-r, --require [path]            Require a node module before execution
...

因此,拥有一个简单的模块来调用dotenv就足够了:

// env-loader.ts
require("dotenv").config({ path: "../.env" });
ts-node -r ./env-loader.ts other-script.ts

以便可以在执行脚本之前加载 .env 或其他环境变量源。

事实上,我能够设置环境变量而无需操作任何 ts 代码。我不得不:

npm install dotenv --save

在项目的根目录中定义 .env 后,我能够像这样简单地调用它:

npx ts-node -r dotenv/config ./src/run.ts

感谢@CKK和@Skezo在上面提供帮助。

最新更新