Skip to content

Commit

Permalink
AWS lambda - configurable flush timeout (#1960)
Browse files Browse the repository at this point in the history
  • Loading branch information
kuba-wu committed Jan 11, 2021
1 parent 4bf86be commit 179b225
Show file tree
Hide file tree
Showing 7 changed files with 77 additions and 8 deletions.
4 changes: 3 additions & 1 deletion instrumentation/aws-lambda-1.0/library/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,11 @@
This package contains libraries to help instrument AWS lambda functions in your code.

## Using wrappers
To use the instrumentation, configure `OTEL_LAMBDA_HANDLER` env property to your lambda handler method in following format `package.ClassName::methodName`
To use the instrumentation, configure `OTEL_INSTRUMENTATION_AWS_LAMBDA_HANDLER` env property to your lambda handler method in following format `package.ClassName::methodName`
and use one of wrappers as your lambda `Handler`.

In order to configure a span flush timeout (default is set to 1 second), please configure `OTEL_INSTRUMENTATION_AWS_LAMBDA_FLUSH_TIMEOUT` env property. The value is in seconds.

Available wrappers:
- `io.opentelemetry.instrumentation.awslambda.v1_0.TracingRequestWrapper` - for wrapping regular handlers (implementing `RequestHandler`)
- `io.opentelemetry.instrumentation.awslambda.v1_0.TracingRequestApiGatewayWrapper` - for wrapping regular handlers (implementing `RequestHandler`) proxied through API Gateway, enabling HTTP context propagation
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,18 +24,29 @@
*/
public abstract class TracingRequestHandler<I, O> implements RequestHandler<I, O> {

private static final long DEFAULT_FLUSH_TIMEOUT_SECONDS = 1;

private final AwsLambdaTracer tracer;
private final long flushTimeout;

/** Creates a new {@link TracingRequestHandler} which traces using the default {@link Tracer}. */
protected TracingRequestHandler(long flushTimeout) {
this.tracer = new AwsLambdaTracer();
this.flushTimeout = flushTimeout;
}

/** Creates a new {@link TracingRequestHandler} which traces using the default {@link Tracer}. */
protected TracingRequestHandler() {
this.tracer = new AwsLambdaTracer();
this.flushTimeout = DEFAULT_FLUSH_TIMEOUT_SECONDS;
}

/**
* Creates a new {@link TracingRequestHandler} which traces using the specified {@link Tracer}.
*/
protected TracingRequestHandler(Tracer tracer) {
this.tracer = new AwsLambdaTracer(tracer);
this.flushTimeout = DEFAULT_FLUSH_TIMEOUT_SECONDS;
}

/**
Expand All @@ -44,6 +55,7 @@ protected TracingRequestHandler(Tracer tracer) {
*/
protected TracingRequestHandler(AwsLambdaTracer tracer) {
this.tracer = tracer;
this.flushTimeout = DEFAULT_FLUSH_TIMEOUT_SECONDS;
}

private Map<String, String> getHeaders(I input) {
Expand Down Expand Up @@ -71,7 +83,9 @@ public final O handleRequest(I input, Context context) {
} else {
tracer.end(span);
}
OpenTelemetrySdk.getGlobalTracerManagement().forceFlush().join(1, TimeUnit.SECONDS);
OpenTelemetrySdk.getGlobalTracerManagement()
.forceFlush()
.join(flushTimeout, TimeUnit.SECONDS);
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,14 +23,27 @@
*/
public abstract class TracingRequestStreamHandler implements RequestStreamHandler {

private static final long DEFAULT_FLUSH_TIMEOUT_SECONDS = 1;

private final AwsLambdaTracer tracer;
private final long flushTimeout;

/**
* Creates a new {@link TracingRequestStreamHandler} which traces using the default {@link
* Tracer}.
*/
protected TracingRequestStreamHandler(long flushTimeout) {
this.tracer = new AwsLambdaTracer();
this.flushTimeout = flushTimeout;
}

/**
* Creates a new {@link TracingRequestStreamHandler} which traces using the default {@link
* Tracer}.
*/
protected TracingRequestStreamHandler() {
this.tracer = new AwsLambdaTracer();
this.flushTimeout = DEFAULT_FLUSH_TIMEOUT_SECONDS;
}

/**
Expand All @@ -39,6 +52,7 @@ protected TracingRequestStreamHandler() {
*/
protected TracingRequestStreamHandler(Tracer tracer) {
this.tracer = new AwsLambdaTracer(tracer);
this.flushTimeout = DEFAULT_FLUSH_TIMEOUT_SECONDS;
}

/**
Expand All @@ -47,6 +61,7 @@ protected TracingRequestStreamHandler(Tracer tracer) {
*/
protected TracingRequestStreamHandler(AwsLambdaTracer tracer) {
this.tracer = tracer;
this.flushTimeout = DEFAULT_FLUSH_TIMEOUT_SECONDS;
}

@Override
Expand All @@ -60,7 +75,9 @@ public void handleRequest(InputStream input, OutputStream output, Context contex
doHandleRequest(proxyRequest.freshStream(), new OutputStreamWrapper(output, span), context);
} catch (Throwable t) {
tracer.endExceptionally(span, t);
OpenTelemetrySdk.getGlobalTracerManagement().forceFlush().join(1, TimeUnit.SECONDS);
OpenTelemetrySdk.getGlobalTracerManagement()
.forceFlush()
.join(flushTimeout, TimeUnit.SECONDS);
throw t;
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,15 +13,19 @@

/**
* Wrapper for {@link TracingRequestStreamHandler}. Allows for wrapping a regular lambda, enabling
* single span tracing. Main lambda class should be configured as env property OTEL_LAMBDA_HANDLER
* in package.ClassName::methodName format. Lambda class must implement {@link
* RequestStreamHandler}.
* single span tracing. Main lambda class should be configured as env property
* OTEL_INSTRUMENTATION_AWS_LAMBDA_HANDLER in package.ClassName::methodName format. Lambda class
* must implement {@link RequestStreamHandler}.
*/
public class TracingRequestStreamWrapper extends TracingRequestStreamHandler {

// visible for testing
static WrappedLambda WRAPPED_LAMBDA = WrappedLambda.fromConfiguration();

public TracingRequestStreamWrapper() {
super(WrapperConfiguration.flushTimeout());
}

@Override
protected void doHandleRequest(InputStream inputStream, OutputStream output, Context context)
throws IOException {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,10 +11,14 @@

/**
* Base abstract wrapper for {@link TracingRequestHandler}. Provides: - delegation to a lambda via
* env property OTEL_LAMBDA_HANDLER in package.ClassName::methodName format
* env property OTEL_INSTRUMENTATION_AWS_LAMBDA_HANDLER in package.ClassName::methodName format
*/
abstract class TracingRequestWrapperBase<I, O> extends TracingRequestHandler<I, O> {

protected TracingRequestWrapperBase() {
super(WrapperConfiguration.flushTimeout());
}

// visible for testing
static WrappedLambda WRAPPED_LAMBDA = WrappedLambda.fromConfiguration();

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,8 @@
/** Model for wrapped lambda function (object, class, method). */
class WrappedLambda {

public static final String OTEL_LAMBDA_HANDLER_ENV_KEY = "OTEL_LAMBDA_HANDLER";
public static final String OTEL_LAMBDA_HANDLER_ENV_KEY =
"OTEL_INSTRUMENTATION_AWS_LAMBDA_HANDLER";

private final Object targetObject;
private final Class<?> targetClass;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
/*
* Copyright The OpenTelemetry Authors
* SPDX-License-Identifier: Apache-2.0
*/

package io.opentelemetry.instrumentation.awslambda.v1_0;

public final class WrapperConfiguration {

private WrapperConfiguration() {}

public static final String OTEL_LAMBDA_FLUSH_TIMEOUT_ENV_KEY =
"OTEL_INSTRUMENTATION_AWS_LAMBDA_FLUSH_TIMEOUT";
public static final long OTEL_LAMBDA_FLUSH_TIMEOUT_DEFAULT = 1;

public static final long flushTimeout() {
String lambdaFlushTimeout = System.getenv(OTEL_LAMBDA_FLUSH_TIMEOUT_ENV_KEY);
if (lambdaFlushTimeout != null && !lambdaFlushTimeout.isEmpty()) {
try {
return Long.parseLong(lambdaFlushTimeout);
} catch (NumberFormatException nfe) {
// ignored - default used
}
}
return OTEL_LAMBDA_FLUSH_TIMEOUT_DEFAULT;
}
}

0 comments on commit 179b225

Please sign in to comment.