我是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);
我感谢您提供的任何帮助。
答案 0 :(得分:0)
好,所以这绝对不是理想的解决方案,但我能够做到这一点。
实际上有很多库可以解析多部分表单数据。实际的问题是所有库都依赖于javax.servlet
包-最重要的是HttpServletRequest
类(还有更多)。
由于我们无法在AWS Lambda环境中访问javax.servlet
包类,因此我的解决方案是对其进行修复。
javax.servlet
package from GitHub并将其添加到您的lambda函数中。请参见下图-您可以看到我的类MultipartFormDataTest
在我的包com...
中,并且我也有javax.servlet
包在同一Java模块中。完成此操作后,我们可以使用一个或多个可以为我们完成所有工作的库。我发现可以解析文件内容的最佳库是Delight FileUpload-https://mvnrepository.com/artifact/org.javadelight/delight-fileupload。
添加该库后,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;
}