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

时间:2017-05-09 16:44:09

标签: java aws-lambda multipartform-data

我是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 +对象,但似乎我无法访问" name"部件的属性因此我不能区分相同类型的元件,例如"参数1"和" param2"。 我相信这可能与此类用于解析电子邮件的事实有关... 还有另一种方法来解析lambda函数中的这个多部分主体吗?这是我必须解析的代码(base64是包含正文的字符串):

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

我感谢您提供的任何帮助。

1 个答案:

答案 0 :(得分:0)

好,所以这绝对不是理想的解决方案,但我能够做到这一点。

问题

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

由于我们无法在AWS Lambda环境中访问javax.servlet包类,因此我的解决方案是对其进行修复。


解决方案

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

Javax Servlet Package added to Lambda Module

  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;
}