Skip to content
Open
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
import com.amazonaws.serverless.proxy.internal.testutils.Timer;
import com.amazonaws.serverless.proxy.model.AwsProxyRequest;
import com.amazonaws.serverless.proxy.model.AwsProxyResponse;
import com.amazonaws.serverless.proxy.model.Headers;
import com.amazonaws.services.lambda.runtime.Context;
import io.micronaut.core.annotation.Internal;
import io.micronaut.core.io.Writable;
Expand All @@ -38,6 +39,7 @@
import java.io.IOException;
import java.nio.charset.Charset;
import java.util.Base64;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

Expand Down Expand Up @@ -68,6 +70,7 @@ public AwsProxyResponse writeResponse(
Context lambdaContext) throws InvalidResponseObjectException {
Timer.start(TIMER_NAME);
AwsProxyResponse awsProxyResponse = containerResponse.getAwsResponse();

final Map<String, Cookie> cookies = containerResponse.getAllCookies();
if (CollectionUtils.isNotEmpty(cookies)) {
final io.netty.handler.codec.http.cookie.Cookie[] nettyCookies = cookies.values().stream().filter(c -> c instanceof NettyCookie).map(c -> ((NettyCookie) c).getNettyCookie()).toArray(io.netty.handler.codec.http.cookie.Cookie[]::new);
Expand Down Expand Up @@ -112,8 +115,20 @@ public AwsProxyResponse writeResponse(
status + " " +
Response.Status.fromStatusCode(status.getCode()).getReasonPhrase());
}

awsProxyResponse.setHeaders(getHeaders(awsProxyResponse));
Timer.stop(TIMER_NAME);
return awsProxyResponse;
}

private static Map<String, String> getHeaders(AwsProxyResponse awsProxyResponse) {
Headers multiValueHeaders = awsProxyResponse.getMultiValueHeaders();
Map<String, String> headers = new HashMap<>();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

use a linked hash map to retain order

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

resolved via 0f624cb

for (Map.Entry<String, List<String>> entry : multiValueHeaders.entrySet()) {
List<String> headerValues = entry.getValue();
if (headerValues != null && headerValues.size() == 1) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Use iterator.hasNext() and iterator.next() instead of calling size() and get(0) which is less efficient

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

check: No headers will be added if the size > 1, is this intended?

suggestion: Try this instead headerValues.stream().first().ifPresent(value -> headers.put(entry.getKey(), value));

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

resolved via 52b852e

headers.put(entry.getKey(), headerValues.get(0));
}
}
return headers;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,80 @@
package io.micronaut.function.aws.proxy

import com.amazonaws.serverless.proxy.internal.testutils.AwsProxyRequestBuilder
import com.amazonaws.serverless.proxy.internal.testutils.MockLambdaContext
import com.amazonaws.serverless.proxy.model.AwsProxyResponse
import com.amazonaws.services.lambda.runtime.Context
import io.micronaut.context.ApplicationContext
import io.micronaut.context.annotation.Requires
import io.micronaut.http.HttpHeaders
import io.micronaut.http.HttpMethod
import io.micronaut.http.HttpStatus
import io.micronaut.http.MediaType
import io.micronaut.http.annotation.Body
import io.micronaut.http.annotation.Controller
import io.micronaut.http.annotation.Get
import io.micronaut.http.annotation.Post
import io.micronaut.http.annotation.Status
import spock.lang.AutoCleanup
import spock.lang.Issue
import spock.lang.PendingFeature
import spock.lang.Shared
import spock.lang.Specification

class ContentTypeSpec extends Specification {
@Shared
@AutoCleanup
MicronautLambdaContainerHandler handler = new MicronautLambdaContainerHandler(
ApplicationContext.builder().properties([
'spec.name' : 'ContentTypeSpec',
'micronaut.security.enabled': false,
])
)
@Shared
Context lambdaContext = new MockLambdaContext()

void "verify controllers return json by default and mulitvalue headers are populated"() {
given:
AwsProxyRequestBuilder builder = new AwsProxyRequestBuilder('/json/bydefault', HttpMethod.GET.toString())
builder.header(HttpHeaders.ACCEPT, MediaType.APPLICATION_JSON)

when:
AwsProxyResponse response = handler.proxy(builder.build(), lambdaContext)

then:
response.statusCode == 200
response.body == '{"msg":"Hello world"}'

and: 'works with multivalue headers'
response.multiValueHeaders
["application/json"] == response.getMultiValueHeaders().get("Content-Type")
}

@Issue("https://github.com/micronaut-projects/micronaut-aws/issues/1330")
void "verify controllers return json by default and headers are populated"() {
given:
AwsProxyRequestBuilder builder = new AwsProxyRequestBuilder('/json/bydefault', HttpMethod.GET.toString())
builder.header(HttpHeaders.ACCEPT, MediaType.APPLICATION_JSON)

when:
AwsProxyResponse response = handler.proxy(builder.build(), lambdaContext)

then:
response.statusCode == 200
response.body == '{"msg":"Hello world"}'

response.headers
"application/json" == response.getHeaders().get("Content-Type")
}

@Controller('/json')
@Requires(property = 'spec.name', value = 'ContentTypeSpec')
static class BodyController {

@Get("/bydefault")
@Status(HttpStatus.OK)
Map<String, Object> index() {
[msg: "Hello world"]
}
}
}