我是Node.js
的新手,并且一直在与第三方提供商一起处理示例项目,并且尝试使用Azure Key Vault存储配置值。
在执行其余操作之前,我无法进入wait
。我将尽我所能来详细说明。
sample project有一个名为agent.js
的文件,它是起始页/文件。在line 16(agent_config = require('./config/config.js')[process.env.LP_ACCOUNT][process.env.LP_USER]
)上,它调用带有值的配置文件。我正在尝试使用Key Vault设置这些值。我已经尝试了许多调用函数的组合,甚至实现了async / await
,但是agent_config
的值始终包含一个[Promise]
对象,而不包含Key Vault返回的数据。
如果我是对的,这是因为Key Vault本身也使用async / await
,并且在返回Key Vault值之前会返回配置文件。
在这种情况下如何添加/实施Key Vault?
这是我尝试过的:
首先将 agent.js 更新为
let agent_config = {};
try {
agent_config = require('./config/config.js')['123']['accountName'];
} catch (ex) {
log.warn(`[agent.js] Error loading config: ${ex}`)
}
console.log(agent_config);
./ config / config.js
const KeyVault = require('azure-keyvault');
const msRestAzure = require('ms-rest-azure');
const KEY_VAULT_URI = 'https://' + '{my vault}' + '.vault.azure.net/' || process.env['KEY_VAULT_URI'];
function getValue(secretName, secretVersion) {
msRestAzure.loginWithAppServiceMSI({ resource: 'https://vault.azure.net' }).then((credentials) => {
const client = new KeyVault.KeyVaultClient(credentials);
client.getSecret(KEY_VAULT_URI, secretName, secretVersion).then(
function (response) {
return response.Value;
});
});
}
module.exports = {
'123': {
'accountName': {
accountId: getValue('mySecretName', '')
}
}
};
结果
{accountId:undefined}
制作 getValue 一个async
函数并将其包装在另一个函数周围(尝试不进行包装并且也不起作用)
./ config / config.js
const KeyVault = require('azure-keyvault');
const msRestAzure = require('ms-rest-azure');
const KEY_VAULT_URI = 'https://' + '{my vault}' + '.vault.azure.net/' || process.env['KEY_VAULT_URI'];
async function getValue(secretName, secretVersion) {
msRestAzure.loginWithAppServiceMSI({ resource: 'https://vault.azure.net' }).then((credentials) => {
const client = new KeyVault.KeyVaultClient(credentials);
client.getSecret(KEY_VAULT_URI, secretName, secretVersion).then(
function (response) {
return response.Value;
});
});
}
async function config() {
module.exports = {
'123': {
'accountName': {
accountId: await getValue('mySecretName', '')
}
}
};
}
config();
结果
{}
制作 getValue 一个async
函数并将其包装在另一个函数周围(尝试不进行包装并且也不起作用)
./ config / config.js
const KeyVault = require('azure-keyvault');
const msRestAzure = require('ms-rest-azure');
const KEY_VAULT_URI = 'https://' + '{my vault}' + '.vault.azure.net/' || process.env['KEY_VAULT_URI'];
async function getValue(secretName, secretVersion) {
return msRestAzure.loginWithAppServiceMSI({ resource: 'https://vault.azure.net' })
.then((credentials) => {
const client = new KeyVault.KeyVaultClient(credentials);
return client.getSecret(KEY_VAULT_URI, secretName, secretVersion).then(
function (response) {
return response.Value;
});
});
}
module.exports = {
'123': {
'accountName': {
accountId: getValue('mySecretName', '')
}
}
};
config();
结果
{accountId:{
}}
我尝试了其他类似module.exports = async (value) =< {...}
的方法(通过其他问题/解决方案找到的方法都没有成功。
我开始认为我需要在agent.js
上做一些“等待”,但是我没有找到很好的信息。
任何帮助都会很棒!
答案 0 :(得分:2)
一个问题是您的getValue
函数未返回任何内容,因为您的返回值必须明确。
(而且没有兑现承诺,没有什么可等待的)
async function getValue(secretName, secretVersion) {
return msRestAzure.loginWithAppServiceMSI({ resource: 'https://vault.azure.net' })
.then((credentials) => {
const client = new KeyVault.KeyVaultClient(credentials);
return client.getSecret(KEY_VAULT_URI, secretName, secretVersion).then(
function (response) {
return response.Value;
});
});
}
您还可以使用箭头函数获得较少的明确回报。.
const getValue = async (secretName, secretVersion) =>
msRestAzure.loginWithAppServiceMSI({ resource: 'https://vault.azure.net' })
.then(credentials => {
const client = new KeyVault.KeyVaultClient(credentials);
return client.getSecret(KEY_VAULT_URI, secretName, secretVersion)
.then(response => response.Value);
});
引入了异步的Azure Key Vault读取,这意味着整个配置读取都是异步的。您无法做任何事情来解决这个问题。这意味着使用该配置的代码将需要适当地处理它。首先,导出一个异步函数,该函数将返回配置。
async function getConfig() {
return {
'123': {
'accountName': {
accountId: await getValue('mySecretName', '')
}
}
};
}
module.exports = getConfig;
在代理代码中,您可以调用该函数。这意味着您的代理代码也需要包装在一个函数中,所以也许是这样的。
const Bot = require('./bot/bot.js');
const getConfig = require('./config/config.js');
getConfig().then(agentConfig => {
const agent = new Bot(agentConfig);
agent.on(Bot.const.CONNECTED, data => {
log.info(`[agent.js] CONNECTED ${JSON.stringify(data)}`);
});
});
答案 1 :(得分:0)
包 azure-keyvault
已被弃用,取而代之的是新的包来分别处理 Keyvault 密钥、机密和证书。对于您的方案,您可以使用新的 @azure/keyvault-secrets 包与 Key Vault 通信,并使用新的 @azure/identity 包创建凭据。
const { SecretClient } = require("@azure/keyvault-secrets");
const { DefaultAzureCredential } = require("@azure/identity");
async function getValue(secretName, secretVersion) {
const credential = new DefaultAzureCredential();
const client = new SecretClient(KEY_VAULT_URI, credential);
const secret = await client.getSecret(secretName);
return secret.value;
}
DefaultAzureCredential
假定您已设置以下环境变量
要尝试其他凭据,请参阅 readme for @azure/identity 如果您要从较旧的 azure-keyvault 包迁移,请查看 migration guide 以了解主要更改