繁体   English   中英

部署时 API 网关 + Lambda 出现 502“内部服务器错误”

[英]502 "Internal Server Error" with API Gateway + Lambda when deploying

我正在使用 API-Gateway + Lambda 组合来处理 POST 请求。 我正在使用节点 + 无服务器框架。

当我离线运行无服务器时,我能够发送 POST 请求并将我的数据存储在 S3 上。 但是当我部署它并运行相同的 POST 请求时,我收到一条“502 内部服务器错误”消息。 因为它在本地工作但不在生产中,我很确定我有一些权限/配置问题。

    saveToS3(newData)
      .then(result => {
        callback(null, {
          statusCode: 200,
          headers: { 'Content-Type': 'application/json' },
          body: "Successfully added data!"
        });
      })
      .catch(e => callback(null, { statusCode: 500, body: JSON.stringify(e) }));

我检查过的:

  • 响应体是一个字符串
  • 在响应中包含状态码、正文、响应头
  • 使用回调模式返回我的响应(见上文)

没有检查的内容:

  • 一种可能的原因是我的 API 网关 api 无权调用 lambda 函数。 如何在serverless.yml文件中进行该设置?

我的yml:

service: myService

provider:
  name: aws
  runtime: nodejs12.x

  iamRoleStatements:
    - Effect: "Allow"
      Action:
        - "s3:GetObject"
        - "s3:PutObject"
      Resource: "arn:aws:s3:::myS3Bucket/*"

functions:
  saveToS3:
    handler: handler.saveToS3
    events:
      - http:
          path: users
          method: post
          cors: true
plugins:
  - serverless-offline

resources:
 Resources:
   NewResource:
     Type: AWS::S3::Bucket
     Properties:
       BucketName: myS3Bucket

发现问题,捂脸,因为我花了几个小时才找到它。

我有两个问题:

  • 我的主要 lambda 函数前面有一个“异步”,但我是用回调来实现它的。 删除“异步”修复了它。

  • 我的响应格式缺少“标题”和“isBase64Encoded”字段。 包括删除 502 错误(见下文)。

有用的链接: - https://docs.aws.amazon.com/apigateway/latest/developerguide/set-up-lambda-proxy-integrations.html#api-gateway-simple-proxy-for-lambda-output-format - (如果您使用的是无服务器框架) https://github.com/dherault/serverless-offline/issues/405

如果使用 API Gateway,请确保您的 lambda 函数的响应类似于下面的代码片段。 否则它会抛出“502 - Malformed Lambda Function”错误。

{
    "isBase64Encoded": true|false,
    "statusCode": httpStatusCode,
    "headers": { "headerName": "headerValue", ... },
    "multiValueHeaders": { "headerName": ["headerValue", "headerValue2", ...], ... },
    "body": "..."
}

如果您使用代理集成,您需要小心,因为对于每个可能的语法错误,它都会引发内部服务器错误。

catch(e => callback(null, { statusCode: 500, body: JSON.stringify(e) }));

错误可以不在这里设置标题吗? 如果不是,则可能是语法错误。

在我们的例子中,我们使用无服务器 + API 网关 + Lambda。 我们的主要app.ts导出一个由无服务器包装的异步处理程序。 例如( EnvironmentService只是环境配置的服务层):

const allowedBinaryMimeTypes = EnvironmentService.getFileMimeTypesAllowed();
...
const handler = serverless(app, { binary: allowedBinaryMimeTypes });

在修改任何代码之前,502 错误网关错误日志显示了以下内容 (AWS CloudWatch):

在此处输入图片说明

在此处输入图片说明

我们的解决方案是覆盖 serverless.yml 中的默认 aws 提供程序超时:

在哪里添加覆盖变量

S3Service 文件(调用 getObject):

    static async getObject(type: string, pathParams: APIGatewayProxyEventPathParameters): Promise<any> {
        const params = await this.setRequestParams(type, pathParams.id, pathParams.fileName);

        try {
            // get S3 object/file
            const data = await S3.getObject(params).promise();

            // some bug with AWS converting JPG objects to JPEGg
            const contentType = data.ContentType === 'image/jpeg' ? 'image/jpg' : data.ContentType;

            return {
                statusCode: 200,
                headers: {
                    'Access-Control-Allow-Headers': 'Content-Type',
                    'Access-Control-Allow-Origin': '*', // Required for CORS support to work
                    'Access-Control-Allow-Methods': 'OPTIONS,GET',
                    'Content-Type': contentType
                },
                body: data.Body.toString('base64'),
                isBase64Encoded: true
            };
        } catch (err) {
            console.error(err);
            throw err;
        }
    }

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM