为什么这个 Node 片段没有捕捉到被拒绝的 Promise?

Why is this Node snippet not catching rejected Promise?

我想了解为什么我遇到的 Promise 拒绝异常没有被 由我在以下代码中的初始 try/catch 处理(完整代码在 this branch 中)

在此特定错误中,nodeCf.deploy 在 index.js 中被调用。

发生错误是因为我应该使用 this.deployName,而不是 stack.deployName 在我的 console.log 声明中。

我不明白的是为什么这被认为是未处理的 Promise 拒绝。 我在 nodeCf.deploy 的初始调用中的 try/catch 不应该捕捉到这个吗?

// index.js:

  switch (args.action) {
    case 'deploy':
      try {
        await nodeCf.deploy(stacks, envVars);
      } catch (e) {
        console.log(`deployment failed: `, e);
        process.exit(1);
      }
      break;
  < ... >

// nodeCf module:
async deploy(stacks, envVars) {
  var stackOutputs = {};
  await Promise.each(stacks, async(stack) => {
    stackOutputs[stack.name] = await stack.deploy(envVars, 
      stackOutputs).outputs;
  });
}

// stack.deploy:
async deploy(envVars, stackOutputs) {
this.load(envVars, stackOutputs);
await ensureBucket(this.infraBucket);
const s3Resp = await this.uploadTemplate()
const stackResp = await ensureAwsCfStack({
  StackName: this.deployName,
  Parameters: this.parameters,
  Tags: this.tags,
  TemplateURL: s3Resp.Location,
  Capabilities: [ 'CAPABILITY_IAM', 
    'CAPABILITY_NAMED_IAM' ]
});
this.outputs = _.map(stackResp.Outputs, (it) => 
  _(it).pick(['OutputKey', 'OutputValue'])
    .toPairs()
    .unzip()
    .tail()
    .fromPairs()
    .value());
console.log(`deployed ${stack.deployName}`); // 'stack' is causing exception
return this;
}

这是你的问题:

stackOutputs[stack.name] = await stack.deploy(envVars, stackOutputs).outputs;

注意到 stack.deployasync function 了吗?这意味着它 returns 一个承诺 - 而不是带有 outputs 属性 的实例。您正在访问 .outputs 的承诺,即 undefined,它将被 await 编辑得很好,但承诺及其拒绝被忽略。

你需要写

stackOutputs[stack.name] = (await stack.deploy(envVars, stackOutputs)).outputs;
//                         ^                                         ^

或者为实例使用辅助变量。