Skip to content

Commit

Permalink
PAYARA-4123 Fixed error scenarios causing NPE
Browse files Browse the repository at this point in the history
- tracedAnnotation instance cannot be resolved if the event is for example
  ON_EXCEPTION invoked sooner than after REQUEST_MATCHED event.
- how to simulate: invoke unmapped context causing HTTP 404.
  • Loading branch information
David Matejcek authored and dmatej committed Sep 24, 2019
1 parent ddbda75 commit 603150a
Showing 1 changed file with 27 additions and 19 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -116,25 +116,34 @@ public OpenTracingRequestEventListener(final String applicationName, final Resou

@Override
public void onEvent(final RequestEvent event) {
LOG.info(() -> "onEvent(event.type=" + event.getType() + ")");
LOG.fine(() -> "onEvent(event.type=" + event.getType() + ")");

// onException is special, it can come in any phase of request processing.
// early phases are simply ignored.
if (Arrays.asList(Type.START, Type.MATCHING_START).contains(event.getType())) {
return;
}

final Traced tracedAnnotation = getTracedAnnotation();
final ContainerRequest requestContext = event.getContainerRequest();
if (!canTrace(requestContext, tracedAnnotation)) {
LOG.finest(() -> "canTrace(...) returned false, nothing to do.");
return;
}

try {
if (event.getType() == Type.REQUEST_MATCHED) {
onIncomingRequest(event, tracedAnnotation);
} else if (event.getType() == Type.ON_EXCEPTION) {
final Traced tracedAnnotation = getTracedAnnotation();
final ContainerRequest requestContext = event.getContainerRequest();
if (!canTrace(requestContext, tracedAnnotation)) {
LOG.finest(() -> "canTrace(...) returned false, nothing to do.");
return;
}
final String operationName = determineOperationName(requestContext, tracedAnnotation);
onIncomingRequest(event, operationName);
return;
}

final Scope activeScope = openTracing.getTracer(this.applicationName).scopeManager().active();
if (activeScope == null) {
LOG.finest(() -> "Could not find any active scope, nothing to do.");
return;
}

if (event.getType() == Type.ON_EXCEPTION) {
onException(event);
} else if (event.getType() == Type.RESP_FILTERS_FINISHED) {
onOutgoingResponse(event);
Expand All @@ -148,14 +157,14 @@ public void onEvent(final RequestEvent event) {
}


private void onIncomingRequest(final RequestEvent event, final Traced tracedAnnotation) {
LOG.fine(() -> "onIncomingRequest(event=" + event.getType() + ")");
private void onIncomingRequest(final RequestEvent event, final String operationName) {
LOG.fine(() -> "onIncomingRequest(event=" + event.getType() + ", operationName=" + operationName + ")");

final ContainerRequest requestContext = event.getContainerRequest();
final Tracer tracer = openTracing.getTracer(this.applicationName);

// Create a Span and instrument it with details about the request
final SpanBuilder spanBuilder = tracer.buildSpan(determineOperationName(requestContext, tracedAnnotation)) //
final SpanBuilder spanBuilder = tracer.buildSpan(operationName) //
.withTag(Tags.SPAN_KIND.getKey(), Tags.SPAN_KIND_SERVER) //
.withTag(Tags.HTTP_METHOD.getKey(), requestContext.getMethod()) //
.withTag(Tags.HTTP_URL.getKey(), toString(requestContext.getUriInfo())) //
Expand Down Expand Up @@ -320,12 +329,11 @@ private Scope getAlreadyActiveScope() {

/**
* Helper method that determines what the operation name of the span.
* Depends on completely initialized resourceInfo.
*
* @param requestContext The context of the request, obtained from the filter methods
* @param tracedAnnotation The Traced annotation obtained from the target method
* @return The name to use as the Span's operation name
*/
private String determineOperationName(final ContainerRequestContext requestContext, final Traced tracedAnnotation) {
private String determineOperationName(final ContainerRequestContext request, final Traced tracedAnnotation) {
if (tracedAnnotation != null) {
String operationName = OpenTracingCdiUtils.getConfigOverrideValue(
Traced.class, "operationName", resourceInfo, String.class)
Expand All @@ -334,7 +342,7 @@ private String determineOperationName(final ContainerRequestContext requestConte
// If the annotation or config override providing an empty name, just set it equal to the HTTP Method,
// followed by the method signature
if (operationName.equals("")) {
operationName = requestContext.getMethod() + ":"
operationName = request.getMethod() + ":"
+ resourceInfo.getResourceClass().getCanonicalName() + "."
+ resourceInfo.getResourceMethod().getName();
}
Expand All @@ -357,7 +365,7 @@ private String determineOperationName(final ContainerRequestContext requestConte

// If the provider is set to "http-path" and the class-level @Path annotation is actually present
if (operationNameProvider.equals("http-path") && classLevelAnnotation != null) {
String operationName = requestContext.getMethod() + ":";
String operationName = request.getMethod() + ":";

if (classLevelAnnotation.value().startsWith("/")) {
operationName += classLevelAnnotation.value();
Expand All @@ -379,7 +387,7 @@ private String determineOperationName(final ContainerRequestContext requestConte
}

// If we haven't returned by now, just go with the default ("class-method")
return requestContext.getMethod() + ":"
return request.getMethod() + ":"
+ resourceInfo.getResourceClass().getCanonicalName() + "."
+ resourceInfo.getResourceMethod().getName();
}
Expand Down

0 comments on commit 603150a

Please sign in to comment.