繁体   English   中英

使用 Java 在 AWS Lambda 上解析 multipart/form-data Body

[英]Parse multipart/form-data Body on AWS Lambda in Java

我是 AWS Lambda 的新手,我正在尝试实现一个 Lambda 函数,该函数接收包含编码为 multipart/form-data 的数据的 POST 请求。 消息使用 Lambda 代理集成通过 API 网关接收,当消息到达 Lambda 函数时,正文以 Base64 编码。 手动解码后,我看到它包含一个多部分的主体,如下所示:

-----WebKitFormBoundary3EZ0C3tbP2JpAmz4
Content-Disposition: form-data; name="param1"

value1
-----WebKitFormBoundary3EZ0C3tbP2JpAmz4
Content-Disposition: form-data; name="param2"

value2
------WebKitFormBoundary3EZ0C3tbP2JpAmz4
Content-Disposition: form-data; name="myfile"; filename="ivr.png"
Content-Type: image/png

PNG
... [binary stuff]
------WebKitFormBoundary3EZ0C3tbP2JpAmz4--

我需要的是在 java 8 中解析此消息,以便我可以访问各个部分。 我设法使用 +javax.mail.Multipart+ 对象来做到这一点,但似乎我无法访问部件的“名称”属性,因此我无法区分相同类型的元素,例如“param1”和“param2”。 我相信这可能与此类用于解析电子邮件的事实有关......是否有另一种方法可以在 lambda 函数中解析这个多部分主体? 这是我必须解析的代码(base64 是包含正文的字符串):

DataSource source = new ByteArrayDataSource(new ByteArrayInputStream(Base64.decodeBase64(base64)), "multipart/mixed");
MimeMultipart mp = new MimeMultipart(source);

我很感激你能提供的任何帮助。

好的,所以这绝对不是理想的解决方案,但我能够完成这项工作。

问题

实际上有很多库可以解析多部分表单数据。 实际问题是所有库都依赖javax.servlet包 - 最重要的是HttpServletRequest类(以及更多)。

由于我们无法在 AWS Lambda 环境中访问javax.servlet包类,因此我的解决方案是解决该问题。


解决方案

  1. 从 GitHub下载javax.servlet并将其添加到您的 lambda 函数中。 请参见下图 - 您可以看到我的类MultipartFormDataTest位于我的包com...并且我在同一个 Java 模块中也有javax.servlet包。

将 Javax Servlet 包添加到 Lambda 模块

  1. 一旦我们这样做了,我们就可以使用一个或多个可以为我们完成所有工作的库。 我发现可以解析文件内容的最好的库是 Delight FileUpload - https://mvnrepository.com/artifact/org.javadelight/delight-fileupload

  2. 添加该库后, getFilesFromMultipartFormData()下面的方法将返回ArrayList<File> ,其中列表中的每一项都代表请求中发送的File

/**
 * @param context context
 * @param body this value taken from the `request.getBody()`
 * @param contentType this value is taken from `request.headers().get("Content-Type")`
 * @return List of File objects
 */
private List<File> getFilesFromMultipartFormData(Context context, String body, String contentType) {
    ArrayList<File> files = new ArrayList<>();
    List<FileItem> fileItems = FileUpload.parse(body.getBytes(StandardCharsets.UTF_8), contentType);

    for(FileItem fileItem : fileItems) {
        if(fileItem == null) {
            continue;
        }

        logger.log("fileItem name: " + fileItem.getName());
        logger.log("fileItem content-type: " + fileItem.getContentType());
        logger.log("fileItem size: " + fileItem.getSize());

        // Note: instead of storing it locally, you can also directly store it to S3 for example
        try {
            // I'm setting the extension to .png but you can look at the fileItem.getContentType()
            // to make sure it is an image vs. pdf vs. some other format
            File temp = File.createTempFile(fileItem.getName(), ".png");
            Files.copy(fileItem.getInputStream(), temp.toPath(), StandardCopyOption.REPLACE_EXISTING);
            files.add(temp);
        } catch (Exception e) {
            continue;
        }
    }

    return files;
}

暂无
暂无

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

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