如何使用无服务器框架通过AWS API网关返回用Node.js编写的AWS Lambda函数的错误?

时间:2018-11-16 20:48:23

标签: node.js aws-lambda aws-api-gateway serverless-framework serverless

我正在编写供内部使用的API,并且这是我第一次使用serverless framework。我正在Node.js中编写Lambda函数,并使用AWS API Gateway进行连接。

在某些情况下,我想返回一个自定义错误消息,而我试图编写一个允许我执行此操作的函数。现在,每当Lambda进程失败时,我都会从API收到一条标准消息。在代码中,如果我尝试使用process.exit(1)终止进程,即使我已经使用callback()返回了错误,我也会得到一个通用错误:

{
    "message": "Internal server error"
}

如果我不使用process.exit(1),我会在日志中看到通过callback()返回的错误,但是该过程继续进行,最终超时:

{
    "message": "Endpoint request timed out"
}

我已经尝试了几种使用callback()方法返回错误的方法,但是到目前为止,我还没有成功。我尝试过这种方法:

async function return_error(callback, context, error, returnCode){
  console.error("FATAL ERROR: ", error);
  let ErrorObj = {
    errorType : "InternalServerError",
    httpStatus : 500,
    requestId : context.awsRequestId,
    errorMessage : error
}
  callback(JSON.stringify(ErrorObj));
  process.exit(1);
}

和这个:

async function return_error(callback, error, returnCode){
  console.error("FATAL ERROR: ", error);
  callback({
    isBase64Encoded: false,
    statusCode: returnCode,
    headers: { 'Content-Type': 'application/json' },
    body: JSON.stringify({"Error Message:": error})
  }, null);
  process.exit(1);
}

(很抱歉,两者之间的较小语法更改。)

到目前为止,我还无法通过API向用户返回任何错误。我的错误总是被记录下来,并且该功能继续。任何帮助,将不胜感激。谢谢!

作为参考,我的serverless.yml文件的相关部分:

service: #Name of service


provider:
  name: aws
  runtime: nodejs8.10
  role: #ARN of Iam role

functions:
  screenshot:
    handler: #Name of handler
    timeout: 30
    memorySize: 1280
    reservedConcurrency: 10
    events:
      - http: 
          method: get
          path: #path
          contentHandling: CONVERT_TO_BINARY
          authorizer:
            type: aws_iam

plugins:
  - serverless-plugin-chrome
  - serverless-apigw-binary
  - serverless-apigwy-binary
package:
  exclude:
    - node_modules/puppeteer/.local-chromium/** 

custom:
  apigwBinary:
    types:
      - '*/*'

2 个答案:

答案 0 :(得分:1)

Node.js的AWS错误回调无法如所宣传的那样工作。根据{{​​3}},所有要做的就是确保自定义错误扩展了Error原型。但是,经过十多个小时的测试,我发现这是完全不正确的。

返回将返回除{"message": "Internal server error"}之外的任何内容(即,如果您从API网关触发了Lambda函数)的错误回调的唯一方法是将错误视为成功完成。

TL; DR:callback(errorResponse, null)不起作用,但callback(null, errorResponse)起作用。

答案 1 :(得分:0)

您的lambda函数需要返回成功,以便APIgateway检测到您的响应。试试这个:

async function return_error(callback, error, returnCode){
  console.error("FATAL ERROR: ", error);
  callback(null, {
    isBase64Encoded: false,
    statusCode: returnCode,
    headers: { 'Content-Type': 'application/json' },
    body: JSON.stringify({"Error Message:": error})
  });
}