AWS 无服务器功能未响应图像

mbe*_*jda 5 amazon-web-services node.js serverless-framework aws-serverless

我正在尝试让 AWS API Gateway 使用图像进行响应。我的无服务器 Lambda 代码如下

const express = require('express');
const serverless = require('serverless-http');
const bodyParser = require('body-parser');
const request = require('request');
const fetch = require('node-fetch')
var Jimp = require('jimp');
const app = express()
app.use(bodyParser.json())
app.use(bodyParser.urlencoded({ extended: true }))

app.get('/image/:id', async(req, res) => {
    const id = req.params.id;

    var imgUrl = 'https://developer.salesforce.com/forums/profilephoto/729F00000005O41/T';
    let options = {};

    const image = await Jimp.read(imgUrl);
    image.getBuffer(Jimp.MIME_JPEG, (err, buffer) => {
        res.set('content-type', 'image/jpeg');
        res.send(buffer.toString('base64'));
    });
});
// wrap express app instance with serverless http function
module.exports.handler = serverless(app)
Run Code Online (Sandbox Code Playgroud)

无服务器.yml


provider:
  name: aws
  runtime: nodejs8.10
  stage: dev
  region: us-east-1
  memorySize: 512

custom:
  apigwBinary:
    types:           #list of mime-types
      - 'image/jpg'
      - 'image/jpeg'
      - 'image/png'
functions:
  avatarFunc:
    handler: index.handler
    events:
      - http:
          path: image/{id}
          method: get
          contentHandling: CONVERT_TO_BINARY

plugins:
  - serverless-offline
  - serverless-apigw-binary
Run Code Online (Sandbox Code Playgroud)

返回的图像是一个黑匣子。 在此输入图像描述

Sco*_*ols 5

好的。我刚刚经历过这个并想分享解决方案。该问题与 Serverless 和 AWS 之间的不匹配有关。所以我们要把它们放在同一页面上。

首先->无服务器配置

const binaryMimeTypes = {binary: [
  'image/*',
  'image/jpeg',
  'image/png',
  'image/svg+xml',
]};

module.exports.server = sls(app, binaryMimeTypes) 
Run Code Online (Sandbox Code Playgroud)

这将 Serverless 配置为将关联的 Mime 类型作为 Base64 提供服务。

第二-> AWS 配置

在 AWS API 中,选择网关,然后选择设置。向下滚动并添加以下二进制类型:

AWS配置

就是这样!现在应该可以了!

  • 我不知道在找到这个之前我对 `Buffer.from` `res.send` `res.end` 进行了多少次修订。AGW/服务器级别!啊。谢谢你! (2认同)

Tha*_*ssi 2

在 API Gateway 中处理二进制文件总是很麻烦。不过,我已经设法让它发挥作用。

您需要做的就是告诉 API Gateway 您的响应是用 base64 编码的。

这是一个可行的解决方案:

module.exports.hello = async (event, context) => {
  const imgUrl = 'https://developer.salesforce.com/forums/profilephoto/729F00000005O41/T';
  const image = await jimp.read(imgUrl);
  const buffer = await image.getBufferAsync(jimp.MIME_JPEG);
  return {
    statusCode: 200,
    headers: {
      'content-type': 'image/jpeg'
    },
    body: buffer.toString('base64'),
    isBase64Encoded: true
  };

};
Run Code Online (Sandbox Code Playgroud)

然而,我在这里看到的真正问题是Express正在为您管理路由,因此我认为您无法拦截 API GW 的响应来添加字段“isBase64Encoded”,所以恐怕您必须让这个API 由 API Gateway 而不是 Express 进行管理,以使其正常工作。

另外,Jimp 提供了一个getBufferAsync返回 Promise 的方法,因此您可以await使用它来使代码稍微简单一些。

希望能帮助到你!

编辑

我仍在尝试使其与 Express 一起使用,所以我发现了这个: https: //github.com/awslabs/aws-serverless-express/issues/99#issuecomment-332169739

我必须承认我没有测试,但如果您确实需要 Express 为您处理路线,它可能会起作用。

在此输入图像描述