Skip to content

Commit

Permalink
Fix serialisation exception on default lambda events (#4724)
Browse files Browse the repository at this point in the history
Adds `FAIL_ON_UNKNOWN_PROPERTIES` to the paramter parser on the lambda
as events coming from AWS have more fields than those represented in the
libraries provided by AWS. Adds a custom `JodaModule` to the same
parser to support `ScheduledEvent` which for now uses Joda Time. At the
same time avoid the real Joda Module to not have extra dependencies.

Resolves: #4645
  • Loading branch information
acm19 committed Jan 14, 2022
1 parent 872c6c7 commit a806937
Show file tree
Hide file tree
Showing 4 changed files with 335 additions and 3 deletions.
6 changes: 4 additions & 2 deletions instrumentation/aws-lambda-1.0/library/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,10 @@ dependencies {

implementation("io.opentelemetry:opentelemetry-extension-aws")

// 1.2.0 allows to get the function ARN
testLibrary("com.amazonaws:aws-lambda-java-core:1.2.0")
// allows to get the function ARN
testLibrary("com.amazonaws:aws-lambda-java-core:1.2.1")
// allows to get the default events
testLibrary("com.amazonaws:aws-lambda-java-events:3.10.0")

testImplementation("com.fasterxml.jackson.core:jackson-databind")
testImplementation("commons-io:commons-io:2.2")
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
/*
* Copyright The OpenTelemetry Authors
* SPDX-License-Identifier: Apache-2.0
*/

package io.opentelemetry.instrumentation.awslambda.v1_0;

import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonDeserializer;
import com.fasterxml.jackson.databind.module.SimpleModule;
import io.opentelemetry.testing.internal.jackson.core.JsonTokenId;
import java.io.IOException;
import javax.annotation.Nullable;
import org.joda.time.DateTime;
import org.joda.time.format.DateTimeFormat;
import org.joda.time.format.DateTimeFormatter;

/**
* Jackson module to be able to properly parse standard events as provided by <a
* href="https://github.com/aws/aws-lambda-java-libs/tree/master/aws-lambda-java-events">AWS</a>.
*
* <p>It enables parsing of {@link DateTime} which is used by some standard events. A custom module
* was used as opposed to Jackson standard module for parsing Joda to avoid adding more libraries to
* the Java Agent.
*
* <p>Supporting custom POJOs using Joda is out of the scope of this class.
*/
class CustomJodaModule extends SimpleModule {
public CustomJodaModule() {
super();
addDeserializer(DateTime.class, new DateTimeDeserialiser());
}

@Override
public String getModuleName() {
return getClass().getSimpleName();
}

@Override
public int hashCode() {
return getClass().hashCode();
}

@Override
public boolean equals(@Nullable Object o) {
return this == o;
}

private static class DateTimeDeserialiser extends JsonDeserializer<DateTime> {
private final DateTimeFormatter dateFormatter =
DateTimeFormat.forPattern("yyyy-MM-dd'T'HH:mm:ssZ");

@Override
public DateTime deserialize(JsonParser p, DeserializationContext ctxt) throws IOException {
if (JsonTokenId.ID_STRING != p.getCurrentTokenId()) {
throw new IllegalArgumentException("Only stream input is accepted");
}

String value = p.getText().trim();
return value.isEmpty() ? null : dateFormatter.parseDateTime(value);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
package io.opentelemetry.instrumentation.awslambda.v1_0;

import com.amazonaws.services.lambda.runtime.Context;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import io.opentelemetry.sdk.OpenTelemetrySdk;
import io.opentelemetry.sdk.autoconfigure.AutoConfiguredOpenTelemetrySdk;
Expand All @@ -19,7 +20,10 @@
*/
abstract class TracingRequestWrapperBase<I, O> extends TracingRequestHandler<I, O> {

protected static final ObjectMapper OBJECT_MAPPER = new ObjectMapper();
protected static final ObjectMapper OBJECT_MAPPER =
new ObjectMapper()
.registerModule(new CustomJodaModule())
.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
private final WrappedLambda wrappedLambda;
private final Method targetMethod;
private final BiFunction<I, Class, Object> parameterMapper;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,262 @@
/*
* Copyright The OpenTelemetry Authors
* SPDX-License-Identifier: Apache-2.0
*/

package io.opentelemetry.instrumentation.awslambda.v1_0;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.mockito.Mockito.mock;

import com.amazonaws.services.lambda.runtime.Context;
import com.amazonaws.services.lambda.runtime.RequestHandler;
import com.amazonaws.services.lambda.runtime.events.KinesisEvent;
import com.amazonaws.services.lambda.runtime.events.S3Event;
import com.amazonaws.services.lambda.runtime.events.SNSEvent;
import com.amazonaws.services.lambda.runtime.events.SQSEvent;
import com.amazonaws.services.lambda.runtime.events.ScheduledEvent;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import io.opentelemetry.sdk.OpenTelemetrySdk;
import java.util.HashMap;
import java.util.Map;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.ValueSource;

class TracingRequestWrapperStandardEventsTest {
private static final String SUCCESS = "success";
private static final ObjectMapper OBJECT_MAPPER = new ObjectMapper();
private static final Map<Class<?>, String> EVENTS_JSON = buildEventExamples();

private final OpenTelemetrySdk sdk = OpenTelemetrySdk.builder().build();
private final Context context = mock(Context.class);
private TracingRequestWrapper wrapper;

private static Map<Class<?>, String> buildEventExamples() {
Map<Class<?>, String> events = new HashMap<>();
events.put(
ScheduledEventRequestHandler.class,
"{\n"
+ " \"version\": \"0\",\n"
+ " \"id\": \"53dc4d37-cffa-4f76-80c9-8b7d4a4d2eaa\",\n"
+ " \"detail-type\": \"Scheduled Event\",\n"
+ " \"source\": \"aws.events\",\n"
+ " \"account\": \"123456789012\",\n"
+ " \"time\": \"2015-10-08T16:53:06Z\",\n"
+ " \"region\": \"us-east-1\",\n"
+ " \"resources\": [\n"
+ " \"arn:aws:events:us-east-1:123456789012:rule/my-scheduled-rule\"\n"
+ " ],\n"
+ " \"detail\": {}\n"
+ "}");
events.put(
KinesisEventRequestHandler.class,
"{\n"
+ " \"Records\": [\n"
+ " {\n"
+ " \"kinesis\": {\n"
+ " \"kinesisSchemaVersion\": \"1.0\",\n"
+ " \"partitionKey\": \"1\",\n"
+ " \"sequenceNumber\": \"49590338271490256608559692538361571095921575989136588898\",\n"
+ " \"data\": \"SGVsbG8sIHRoaXMgaXMgYSB0ZXN0Lg==\",\n"
+ " \"approximateArrivalTimestamp\": 1545084650.987\n"
+ " },\n"
+ " \"eventSource\": \"aws:kinesis\",\n"
+ " \"eventVersion\": \"1.0\",\n"
+ " \"eventID\": \"shardId-000000000006:49590338271490256608559692538361571095921575989136588898\",\n"
+ " \"eventName\": \"aws:kinesis:record\",\n"
+ " \"invokeIdentityArn\": \"arn:aws:iam::123456789012:role/lambda-role\",\n"
+ " \"awsRegion\": \"us-east-2\",\n"
+ " \"eventSourceARN\": \"arn:aws:kinesis:us-east-2:123456789012:stream/lambda-stream\"\n"
+ " },\n"
+ " {\n"
+ " \"kinesis\": {\n"
+ " \"kinesisSchemaVersion\": \"1.0\",\n"
+ " \"partitionKey\": \"1\",\n"
+ " \"sequenceNumber\": \"49590338271490256608559692540925702759324208523137515618\",\n"
+ " \"data\": \"VGhpcyBpcyBvbmx5IGEgdGVzdC4=\",\n"
+ " \"approximateArrivalTimestamp\": 1545084711.166\n"
+ " },\n"
+ " \"eventSource\": \"aws:kinesis\",\n"
+ " \"eventVersion\": \"1.0\",\n"
+ " \"eventID\": \"shardId-000000000006:49590338271490256608559692540925702759324208523137515618\",\n"
+ " \"eventName\": \"aws:kinesis:record\",\n"
+ " \"invokeIdentityArn\": \"arn:aws:iam::123456789012:role/lambda-role\",\n"
+ " \"awsRegion\": \"us-east-2\",\n"
+ " \"eventSourceARN\": \"arn:aws:kinesis:us-east-2:123456789012:stream/lambda-stream\"\n"
+ " }\n"
+ " ]\n"
+ "}");
events.put(
SqsEventRequestHandler.class,
"{\n"
+ " \"Records\": [\n"
+ " {\n"
+ " \"messageId\": \"059f36b4-87a3-44ab-83d2-661975830a7d\",\n"
+ " \"receiptHandle\": \"AQEBwJnKyrHigUMZj6rYigCgxlaS3SLy0a...\",\n"
+ " \"body\": \"Test message.\",\n"
+ " \"attributes\": {\n"
+ " \"ApproximateReceiveCount\": \"1\",\n"
+ " \"SentTimestamp\": \"1545082649183\",\n"
+ " \"SenderId\": \"AIDAIENQZJOLO23YVJ4VO\",\n"
+ " \"ApproximateFirstReceiveTimestamp\": \"1545082649185\"\n"
+ " },\n"
+ " \"messageAttributes\": {},\n"
+ " \"md5OfBody\": \"e4e68fb7bd0e697a0ae8f1bb342846b3\",\n"
+ " \"eventSource\": \"aws:sqs\",\n"
+ " \"eventSourceARN\": \"arn:aws:sqs:us-east-2:123456789012:my-queue\",\n"
+ " \"awsRegion\": \"us-east-2\"\n"
+ " },\n"
+ " {\n"
+ " \"messageId\": \"2e1424d4-f796-459a-8184-9c92662be6da\",\n"
+ " \"receiptHandle\": \"AQEBzWwaftRI0KuVm4tP+/7q1rGgNqicHq...\",\n"
+ " \"body\": \"Test message.\",\n"
+ " \"attributes\": {\n"
+ " \"ApproximateReceiveCount\": \"1\",\n"
+ " \"SentTimestamp\": \"1545082650636\",\n"
+ " \"SenderId\": \"AIDAIENQZJOLO23YVJ4VO\",\n"
+ " \"ApproximateFirstReceiveTimestamp\": \"1545082650649\"\n"
+ " },\n"
+ " \"messageAttributes\": {},\n"
+ " \"md5OfBody\": \"e4e68fb7bd0e697a0ae8f1bb342846b3\",\n"
+ " \"eventSource\": \"aws:sqs\",\n"
+ " \"eventSourceARN\": \"arn:aws:sqs:us-east-2:123456789012:my-queue\",\n"
+ " \"awsRegion\": \"us-east-2\"\n"
+ " }\n"
+ " ]\n"
+ "}");
events.put(
S3EventRequestHandler.class,
"{\n"
+ " \"Records\": [\n"
+ " {\n"
+ " \"eventVersion\": \"2.1\",\n"
+ " \"eventSource\": \"aws:s3\",\n"
+ " \"awsRegion\": \"us-east-2\",\n"
+ " \"eventTime\": \"2019-09-03T19:37:27.192Z\",\n"
+ " \"eventName\": \"ObjectCreated:Put\",\n"
+ " \"userIdentity\": {\n"
+ " \"principalId\": \"AWS:AIDAINPONIXQXHT3IKHL2\"\n"
+ " },\n"
+ " \"requestParameters\": {\n"
+ " \"sourceIPAddress\": \"205.255.255.255\"\n"
+ " },\n"
+ " \"responseElements\": {\n"
+ " \"x-amz-request-id\": \"D82B88E5F771F645\",\n"
+ " \"x-amz-id-2\": \"vlR7PnpV2Ce81l0PRw6jlUpck7Jo5ZsQjryTjKlc5aLWGVHPZLj5NeC6qMa0emYBDXOo6QBU0Wo=\"\n"
+ " },\n"
+ " \"s3\": {\n"
+ " \"s3SchemaVersion\": \"1.0\",\n"
+ " \"configurationId\": \"828aa6fc-f7b5-4305-8584-487c791949c1\",\n"
+ " \"bucket\": {\n"
+ " \"name\": \"DOC-EXAMPLE-BUCKET\",\n"
+ " \"ownerIdentity\": {\n"
+ " \"principalId\": \"A3I5XTEXAMAI3E\"\n"
+ " },\n"
+ " \"arn\": \"arn:aws:s3:::lambda-artifacts-deafc19498e3f2df\"\n"
+ " },\n"
+ " \"object\": {\n"
+ " \"key\": \"b21b84d653bb07b05b1e6b33684dc11b\",\n"
+ " \"size\": 1305107,\n"
+ " \"eTag\": \"b21b84d653bb07b05b1e6b33684dc11b\",\n"
+ " \"sequencer\": \"0C0F6F405D6ED209E1\"\n"
+ " }\n"
+ " }\n"
+ " }\n"
+ " ]\n"
+ "}");
events.put(
SnsEventRequestHandler.class,
"{\n"
+ " \"Records\": [\n"
+ " {\n"
+ " \"EventVersion\": \"1.0\",\n"
+ " \"EventSubscriptionArn\": \"arn:aws:sns:us-east-2:123456789012:sns-lambda:21be56ed-a058-49f5-8c98-aedd2564c486\",\n"
+ " \"EventSource\": \"aws:sns\",\n"
+ " \"Sns\": {\n"
+ " \"SignatureVersion\": \"1\",\n"
+ " \"Timestamp\": \"2019-01-02T12:45:07.000Z\",\n"
+ " \"Signature\": \"tcc6faL2yUC6dgZdmrwh1Y4cGa/ebXEkAi6RibDsvpi+tE/1+82j...65r==\",\n"
+ " \"SigningCertUrl\": \"https://sns.us-east-2.amazonaws.com/SimpleNotificationService-ac565b8b1a6c5d002d285f9598aa1d9b.pem\",\n"
+ " \"MessageId\": \"95df01b4-ee98-5cb9-9903-4c221d41eb5e\",\n"
+ " \"Message\": \"Hello from SNS!\",\n"
+ " \"MessageAttributes\": {\n"
+ " \"Test\": {\n"
+ " \"Type\": \"String\",\n"
+ " \"Value\": \"TestString\"\n"
+ " },\n"
+ " \"TestBinary\": {\n"
+ " \"Type\": \"Binary\",\n"
+ " \"Value\": \"TestBinary\"\n"
+ " }\n"
+ " },\n"
+ " \"Type\": \"Notification\",\n"
+ " \"UnsubscribeUrl\": \"https://sns.us-east-2.amazonaws.com/?Action=Unsubscribe&amp;SubscriptionArn=arn:aws:sns:us-east-2:123456789012:test-lambda:21be56ed-a058-49f5-8c98-aedd2564c486\",\n"
+ " \"TopicArn\":\"arn:aws:sns:us-east-2:123456789012:sns-lambda\",\n"
+ " \"Subject\": \"TestInvoke\"\n"
+ " }\n"
+ " }\n"
+ " ]\n"
+ "}");

return events;
}

private TracingRequestWrapper buildWrapper(Class<?> targetClass) {
WrappedLambda wrappedLambda = new WrappedLambda(targetClass, "handleRequest");

return new TracingRequestWrapper(sdk, wrappedLambda, TracingRequestWrapper::map);
}

@ParameterizedTest
@ValueSource(
classes = {
ScheduledEventRequestHandler.class,
KinesisEventRequestHandler.class,
SqsEventRequestHandler.class,
S3EventRequestHandler.class,
SnsEventRequestHandler.class
})
void handleScheduledEvent(Class<?> targetClass) throws JsonProcessingException {
wrapper = buildWrapper(targetClass);
Object parsedScheduledEvent =
OBJECT_MAPPER.readValue(EVENTS_JSON.get(targetClass), Object.class);
assertEquals(SUCCESS, wrapper.doHandleRequest(parsedScheduledEvent, context));
}

public static class ScheduledEventRequestHandler
implements RequestHandler<ScheduledEvent, String> {
@Override
public String handleRequest(ScheduledEvent i, Context cntxt) {
return SUCCESS;
}
}

public static class KinesisEventRequestHandler implements RequestHandler<KinesisEvent, String> {
@Override
public String handleRequest(KinesisEvent i, Context cntxt) {
return SUCCESS;
}
}

public static class SqsEventRequestHandler implements RequestHandler<SQSEvent, String> {
@Override
public String handleRequest(SQSEvent i, Context cntxt) {
return SUCCESS;
}
}

public static class S3EventRequestHandler implements RequestHandler<S3Event, String> {
@Override
public String handleRequest(S3Event i, Context cntxt) {
return SUCCESS;
}
}

public static class SnsEventRequestHandler implements RequestHandler<SNSEvent, String> {
@Override
public String handleRequest(SNSEvent i, Context cntxt) {
return SUCCESS;
}
}
}

0 comments on commit a806937

Please sign in to comment.