0

我在将文件从邮递员上传到 aws lambda + s3 时遇到问题。如果我理解正确,图像必须是 base64 字符串并通过 JSON 发送以与 lambda 和 API Gateway 一起使用,所以我将图像转换为 base64 并在邮递员中使用 base64 字符串

在此处输入图像描述

文件上传到 S3,但是当我下载 s3 对象并打开它时,我得到

在此处输入图像描述

所以我认为我没有正确上传它。我使用了base64到图像转换器并且图像出现了,所以base64字符串在通过邮递员发送之前是正确的,所以我的设置中的某些东西是关闭的。我究竟做错了什么?感谢您的帮助!

上传.js

const AWS = require('aws-sdk');
const s3 = new AWS.S3();
exports.handler = async (event, context, callback) => {
    let data = JSON.parse(event.body);
    let file = data.base64String;


    const s3Bucket = "upload-test3000";
    const objectName = "helloworld.jpg";
    const objectData = data.base64String;
    const objectType = "image/jpg";
    try {
        const params = {
            Bucket: s3Bucket,
            Key: objectName,
            Body: objectData,
            ContentType: objectType
        };
        const result = await s3.putObject(params).promise();
        return sendRes(200, `File uploaded successfully at https:/` + s3Bucket + `.s3.amazonaws.com/` + objectName);

    } catch (error) {
        return sendRes(404, error);
    }
};
const sendRes = (status, body) => {
    var response = {
        statusCode: status,
        headers: {
            "Content-Type": "application/json",
            "Access-Control-Allow-Headers": "Content-Type,X-Amz-Date,Authorization,X-Api-Key,X-Amz-Security-Token",
            "Access-Control-Allow-Methods": "OPTIONS,POST,PUT",
            "Access-Control-Allow-Credentials": true,
            "Access-Control-Allow-Origin": "*",
            "X-Requested-With": "*"
        },
        body: body
    };
    return response;
};

.png

4

3 回答 3

1

在构建参数时,您应该添加内容编码,否则您只是上传文本数据:

const params = {
  Bucket: s3Bucket,
  Key: objectName,
  Body: objectData,
  ContentType: objectType,
  ContentEncoding: 'base64'
};

编辑

好的,我已经检查了文件,我想您可能误解了将图像存储在 base64 中会发生什么。

Windows 或浏览器无法读取 base64 格式的 jpg 文件(据我所知),必须先转换它。当您在浏览器中有一个带有 base64 源的图像时,浏览器会即时处理此转换,但“helloworld.jpg”容器中的 base64 数据在 Windows 中如果不进行转换就毫无用处。

有两个选项,要么在到达您的服务器后进行转换,然后直接以 utf8 格式上传,要么在两者之间添加一层,根据请求转换图像。

于 2020-12-03T14:26:41.447 回答
0

我通过像这样添加 JSON 格式的 base64 字符串来让它工作

在此处输入图像描述

然后发送

let decodedImage = Buffer.from(encodedImage, 'base64');作为Body参数。

更新了 upload.js

const AWS = require('aws-sdk');
var s3 = new AWS.S3();


exports.handler = async (event) => {

    let encodedImage = JSON.parse(event.body).base64Data;
    let decodedImage = Buffer.from(encodedImage, 'base64');
    var filePath = "user-data/" + event.queryStringParameters.username + ".jpg"
    var params = {
        "Body": decodedImage,
        "Bucket": process.env.UploadBucket,
        "Key": filePath
    };

    try {
        let uploadOutput = await s3.upload(params).promise();
        let response = {
            "statusCode": 200,
            "body": JSON.stringify(uploadOutput),
            "isBase64Encoded": false
        };
        return response;
    }
    catch (err) {
        let response = {
            "statusCode": 500,
            "body": JSON.stringify(eerr),
            "isBase64Encoded": false
        };
        return response;
    }

};

我发现这篇文章非常有帮助

于 2020-12-04T20:14:39.717 回答
0

问题可能是 Body 中图像的传递格式,因为它没有按 s3.upload 参数的预期传递(它说 Body 键必须是 Buffer 才能传递)。

因此,简单的解决方案是将 Body 作为缓冲区传递,如果您的文件 存在于目录中的任何位置,则不要像传递

// Wrong Way 

const params = {
   Bucket: 'Your-Buket-Name',
   Key: 'abc.png', // destFileName i.e the name of file to be saved in s3 bucket
   Body: 'Path-To-File'
}

现在,问题是文件作为原始文本上传,格式已损坏,操作系统在下载时无法读取。

所以,让它工作通过它就像

// Correct Way According to aws-sdk library

const fs = require('fs'); 
const imageData = fs.readFileSync('Path-To-File'); // returns buffer
const params = {
   Bucket: 'Your-Buket-Name',
   Key: 'abc.png', // destFileName i.e the name of file to be saved in s3 bucket
   Body: imageData // image buffer
}
const uploadedFile = await s3.upload(params).promise();

注意:在回答期间我使用的是 -> "aws-sdk": "^2.1025.0"

希望这会对您或其他人有所帮助。谢谢!

于 2021-11-11T06:48:31.397 回答