为什么我无法更新节点js中的env变量

问题描述 投票:0回答:3

我想在节点js中更新我的环境变量,但我无法更新它的环境变量,我尝试使用

console.log(process.env.DB_HOST)
但我仍然得到旧值,在这里我添加了我的整个代码,任何人都可以查看并帮助我解决这个问题,

function exec_api() { 
    return new Promise(async function (resolve) {
        const execSync = require('child_process').exec;
        //let child_process_obj = execSync('DB_HOST='+process.env.UNITTEST_DB_HOST+' DB_DATABASE='+process.env.UNITTEST_DB_DATABASE+'  DB_USERNAME='+process.env.UNITTEST_DB_USERNAME+' DB_PASSWORD='+process.env.UNITTEST_DB_PASSWORD+'  PORT='+process.env.UNITTEST_SERVICE_PORT+' ./node_modules/.bin/nodemon main.js'); 
        await execSync('export DB_HOST=' + process.env.UNITTEST_DB_HOST);
        await execSync('export DB_DATABASE=' + process.env.UNITTEST_DB_DATABASE);
        await execSync('export DB_USERNAME=' + process.env.UNITTEST_DB_USERNAME);
        await execSync('export DB_PASSWORD=' + process.env.UNITTEST_DB_PASSWORD);
        await execSync('export PORT=' + process.env.UNITTEST_API_BACKEND_PORT);
        let child_process_obj = await execSync('node main.js');
        unittest_api_backend_process_id = child_process_obj.pid;
        resolve(true);
    });
}
node.js node-modules
3个回答
5
投票

TLDR:只需更改 process.env

要更改、添加或删除环境变量,请使用

process.env
。以下是显示其工作原理的测试代码:

main.js:

console.log(process.env.DB_DATABASE);

exec.js:

const execSync = require ('child_process').execSync;

process.env.DB_DATABASE = 'foo'; // this is ALL you need to do

console.log(execSync('node main.js').toString('utf8'));

使用上面的两个文件,如果运行

node exec.js
,您将看到控制台中打印出
foo
。这是从
main.js
打印的,它继承了
exec.js
的环境。

所以你需要在代码中做的是:

function exec_api() { 
    return new Promise(function (resolve) {
        const exec = require('child_process').exec;

        // The following is node.js equivalent of bash "export":
        process.env.DB_HOST = process.env.UNITTEST_DB_HOST;
        process.env.DB_DATABASE = process.env.UNITTEST_DB_DATABASE;
        process.env.DB_USERNAME = process.env.UNITTEST_DB_USERNAME;
        process.env.DB_PASSWORD = process.env.UNITTEST_DB_PASSWORD;
        process.env.PORT = process.env.UNITTEST_SERVICE_PORT;

        let child_process_obj = exec('node main.js', {
            stdio: ['inherit', 'inherit', 'inherit']
        });
        
        unittest_api_backend_process_id = child_process_obj.pid;
        
        resolve(true);
    });
}

请注意,如果您希望在

main.js
结束时返回承诺,您需要执行以下操作:

function exec_api() { 
    return new Promise(function (resolve) {
        const exec = require('child_process').exec;

        // The following is node.js equivalent of bash "export":
        process.env.DB_HOST = process.env.UNITTEST_DB_HOST;
        process.env.DB_DATABASE = process.env.UNITTEST_DB_DATABASE;
        process.env.DB_USERNAME = process.env.UNITTEST_DB_USERNAME;
        process.env.DB_PASSWORD = process.env.UNITTEST_DB_PASSWORD;
        process.env.PORT = process.env.UNITTEST_SERVICE_PORT;

        let child_process_obj = exec('node main.js', {
            stdio: ['inherit', 'inherit', 'inherit']
        });
        
        unittest_api_backend_process_id = child_process_obj.pid;
        
        child_process_obj.on('exit', () => resolve(true));

        // ^^^ Cannot use `await` as the API is not promise based
        //     but event based instead.
    });
}

长话短说:完整解释为什么
export
不起作用

在unixen上,环境变量,甚至整个环境,包括当前工作目录、根目录(可以通过chroot更改)等都不是shell的功能。它们是流程的特征。

我们可能熟悉某些 shell 为子进程设置环境变量的

export
语法,但这就是 shell 的语法。它与环境变量本身无关。例如,C/C++ 不要使用
export
,而是使用
setenv()
函数来设置环境变量(事实上,内部这就是 bash/sh/ksh 等在实现
export
时所做的事情)。

在node.js中,读取和设置环境变量的机制是通过

process.env

为什么要求 shell 做这件事不起作用

这不仅仅是一个 Node.js 问题。它在 bash 中也不起作用:

exporter.sh:

#! /bin/bash

export DB_DATABASE=$1

exec.sh:

#! /bin/bash

./exporter.sh foo
echo $DB_DATABASE ;# does not print "foo"

这是unixen的核心安全功能:其他用户不应该被允许扰乱你的进程。在环境情况下执行此策略的方式是,只有父进程可以设置子进程的环境。子进程不允许设置父进程的环境。假设子进程“属于”父进程,因此您应该被允许对程序执行您想要的操作 - 但由于父进程(您)不属于子进程,因此不允许子进程执行此操作扰乱父母的环境。 这就是为什么您尝试使用

export

不起作用。它确实有效(变量确实是在子 shell 中创建的),但不允许更改其父进程(node.js 进程)的环境

    


0
投票
export

时,它会指示 shell 设置环境变量。


当您从代码中调用

exec

时,您并没有运行这样的 shell,原因是提取每个命令的输出将成为一个挑战。


这使得

export

成为一个被忽略的命令。


您可以通过将

option 对象传递给

execSync 来解决此问题:



execSync('node main.js', {
    env: {
        DB_HOST: 'localhost',
         // More envs...
    }
}



0
投票
.env

文件来存储端口;那么这对我有用。


复制
    .env
  1. 文件内的内容
    现在,删除 
  2. .env
  3. 文件。
    创建新的
  4. .env
  5. 并将内容粘贴进去。
    现在就跑(
  6. npm install dotenv
  7. )
    运行你的服务器
© www.soinside.com 2019 - 2024. All rights reserved.