Skip to content

Commit 2317cf2

Browse files
nanodeathClaude
and
Claude
committed
feat(function-aws-api-proxy): add multipart/form-data support to ApiGatewayServletRequest
This adds support for parsing multipart/form-data in AWS API Gateway proxy requests. The implementation handles boundary extraction, part parsing, and field name extraction while maintaining compatibility with existing form handling. Co-Authored-By: Claude <[email protected]>
1 parent 895080e commit 2317cf2

6 files changed

+664
-5
lines changed

function-aws-api-proxy/src/main/java/io/micronaut/function/aws/proxy/ApiGatewayBinderRegistry.java

+6
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,9 @@
1818
import io.micronaut.context.annotation.Replaces;
1919
import io.micronaut.core.annotation.Internal;
2020
import io.micronaut.core.convert.ConversionService;
21+
import io.micronaut.function.aws.proxy.multipart.CompletedFileUploadBinder;
22+
import io.micronaut.function.aws.proxy.multipart.PartAnnotationRequestArgumentBinder;
23+
import io.micronaut.http.annotation.Part;
2124
import io.micronaut.http.bind.DefaultRequestBinderRegistry;
2225
import io.micronaut.http.bind.binders.DefaultBodyAnnotationBinder;
2326
import io.micronaut.http.bind.binders.RequestArgumentBinder;
@@ -39,5 +42,8 @@ class ApiGatewayBinderRegistry<T> extends ServletBinderRegistry<T> {
3942
DefaultBodyAnnotationBinder<T> defaultBodyAnnotationBinder
4043
) {
4144
super(mediaTypeCodecRegistry, conversionService, binders, defaultBodyAnnotationBinder);
45+
46+
CompletedFileUploadBinder completedFileUploadBinder = new CompletedFileUploadBinder();
47+
byAnnotation.put(Part.class, new PartAnnotationRequestArgumentBinder<>(conversionService, completedFileUploadBinder));
4248
}
4349
}

function-aws-api-proxy/src/main/java/io/micronaut/function/aws/proxy/ApiGatewayServletRequest.java

+45-5
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@
2929
import io.micronaut.core.util.CollectionUtils;
3030
import io.micronaut.core.util.StringUtils;
3131
import io.micronaut.core.util.SupplierUtil;
32+
import io.micronaut.function.aws.proxy.multipart.MultipartDataDecoder;
3233
import io.micronaut.http.CaseInsensitiveMutableHttpHeaders;
3334
import io.micronaut.http.FullHttpRequest;
3435
import io.micronaut.http.HttpMethod;
@@ -41,6 +42,7 @@
4142
import io.micronaut.http.body.stream.AvailableByteArrayBody;
4243
import io.micronaut.http.cookie.Cookie;
4344
import io.micronaut.http.cookie.Cookies;
45+
import io.micronaut.http.multipart.CompletedFileUpload;
4446
import io.micronaut.http.uri.UriBuilder;
4547
import io.micronaut.servlet.http.ByteArrayByteBuffer;
4648
import io.micronaut.servlet.http.MutableServletHttpRequest;
@@ -91,6 +93,8 @@ public abstract class ApiGatewayServletRequest<T, REQ, RES> implements MutableSe
9193
private Supplier<Optional<T>> body;
9294
private T parsedBody;
9395
private T overriddenBody;
96+
private final Supplier<Optional<MultipartDataDecoder>> multipartDataDecoder;
97+
private Map<String, CompletedFileUpload> fileUploads;
9498

9599
private ByteArrayByteBuffer<T> servletByteBuffer;
96100

@@ -111,6 +115,19 @@ protected ApiGatewayServletRequest(
111115
T built = parsedBody != null ? parsedBody : (T) bodyBuilder.buildBody(this::getInputStream, this);
112116
return Optional.ofNullable(built);
113117
});
118+
this.multipartDataDecoder = SupplierUtil.memoized(() -> {
119+
try {
120+
MediaType contentType = getContentType().orElse(null);
121+
if (MediaType.MULTIPART_FORM_DATA_TYPE.equals(contentType)) {
122+
return Optional.of(new MultipartDataDecoder(getBodyBytes(), getHeaders(), getCharacterEncoding()));
123+
}
124+
} catch (IOException e) {
125+
if (log.isDebugEnabled()) {
126+
log.debug("Error decoding multipart form data: {}", e.getMessage(), e);
127+
}
128+
}
129+
return Optional.empty();
130+
});
114131
}
115132

116133
@Override
@@ -266,16 +283,38 @@ public <B> MutableHttpRequest<B> body(B body) {
266283
*/
267284
protected MapListOfStringAndMapStringMutableHttpParameters getParametersFromBody(Map<String, String> queryStringParameters) {
268285
Map<String, List<String>> parameters = null;
269-
try {
270-
parameters = new QueryStringDecoder(new String(getBodyBytes(), getCharacterEncoding()), false).parameters();
271-
} catch (IOException ex) {
272-
if (log.isDebugEnabled()) {
273-
log.debug("Error decoding form data: " + ex.getMessage(), ex);
286+
MediaType contentType = getContentType().orElse(MediaType.APPLICATION_JSON_TYPE);
287+
288+
if (MediaType.APPLICATION_FORM_URLENCODED_TYPE.equals(contentType)) {
289+
try {
290+
parameters = new QueryStringDecoder(new String(getBodyBytes(), getCharacterEncoding()), false).parameters();
291+
} catch (IOException ex) {
292+
if (log.isDebugEnabled()) {
293+
log.debug("Error decoding form data: {}", ex.getMessage(), ex);
294+
}
274295
}
296+
} else if (MediaType.MULTIPART_FORM_DATA_TYPE.equals(contentType)) {
297+
parameters = multipartDataDecoder.get()
298+
.map(MultipartDataDecoder::parameters)
299+
.orElse(Collections.emptyMap());
275300
}
301+
276302
return new MapListOfStringAndMapStringMutableHttpParameters(conversionService, parameters, queryStringParameters);
277303
}
278304

305+
/**
306+
* Gets a map of uploaded files from the multipart request.
307+
* @return A map of field names to file uploads
308+
*/
309+
public Map<String, CompletedFileUpload> getFileUploads() {
310+
if (fileUploads != null) {
311+
log.trace("Skipping decoding file uploads as they have already been processed");
312+
} else {
313+
fileUploads = multipartDataDecoder.get().map(MultipartDataDecoder::fileUploads).orElse(Collections.emptyMap());
314+
}
315+
return Collections.unmodifiableMap(fileUploads);
316+
}
317+
279318
@Override
280319
public void setConversionService(ConversionService conversionService) {
281320
this.conversionService = conversionService;
@@ -355,6 +394,7 @@ protected MutableHttpParameters getParameters(@NonNull Supplier<Map<String, Stri
355394
Map<String, List<String>> multi = multiQueryStringParametersSupplier.get();
356395
Map<String, String> single = queryStringParametersSupplier.get();
357396
MediaType mediaType = getContentType().orElse(MediaType.APPLICATION_JSON_TYPE);
397+
358398
if (isFormSubmission(mediaType)) {
359399
return getParametersFromBody(MapCollapseUtils.collapse(MapCollapseUtils.collapse(multi, single)));
360400
} else {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
/*
2+
* Copyright 2017-2025 original authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package io.micronaut.function.aws.proxy.multipart;
17+
18+
import io.micronaut.core.annotation.Internal;
19+
import io.micronaut.core.bind.annotation.Bindable;
20+
import io.micronaut.core.convert.ArgumentConversionContext;
21+
import io.micronaut.core.type.Argument;
22+
import io.micronaut.function.aws.proxy.ApiGatewayServletRequest;
23+
import io.micronaut.http.HttpRequest;
24+
import io.micronaut.http.bind.binders.TypedRequestArgumentBinder;
25+
import io.micronaut.http.multipart.CompletedFileUpload;
26+
27+
import java.util.Optional;
28+
29+
/**
30+
* Binds {@link CompletedFileUpload} arguments from multipart requests in the AWS API Gateway context.
31+
*/
32+
@Internal
33+
public class CompletedFileUploadBinder implements TypedRequestArgumentBinder<CompletedFileUpload> {
34+
35+
static final Argument<CompletedFileUpload> TYPE = Argument.of(CompletedFileUpload.class);
36+
37+
@Override
38+
public Argument<CompletedFileUpload> argumentType() {
39+
return TYPE;
40+
}
41+
42+
@Override
43+
public BindingResult<CompletedFileUpload> bind(
44+
ArgumentConversionContext<CompletedFileUpload> context,
45+
HttpRequest<?> source) {
46+
47+
if (!(source instanceof ApiGatewayServletRequest)) {
48+
return BindingResult.UNSATISFIED;
49+
}
50+
51+
ApiGatewayServletRequest<?, ?, ?> request = (ApiGatewayServletRequest<?, ?, ?>) source;
52+
53+
// Get the parameter name to bind
54+
String paramName = context.getArgument()
55+
.getAnnotationMetadata()
56+
.stringValue(Bindable.class)
57+
.orElse(context.getArgument().getName());
58+
59+
// Try to find a file upload with the matching name
60+
final CompletedFileUpload fileUpload = request.getFileUploads().get(paramName);
61+
if (fileUpload != null) {
62+
return () -> Optional.of(fileUpload);
63+
}
64+
65+
return BindingResult.UNSATISFIED;
66+
}
67+
}

0 commit comments

Comments
 (0)