Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Spring-webflux 5.3 instrumentation #121

Merged
merged 2 commits into from
Dec 15, 2020
Merged
Show file tree
Hide file tree
Changes from all 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
17 changes: 17 additions & 0 deletions instrumentation/spring-webflux-5.3.0/NOTICE.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
This product contains a modified part of OpenTelemetry:

* License:

Copyright 2019 The OpenTelemetry Authors

Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
in compliance with the License. You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software distributed under the License
is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
or implied. See the License for the specific language governing permissions and limitations under
the License.

* Homepage: https://github.com/open-telemetry/opentelemetry-java-instrumentation/blob/master/LICENSE
40 changes: 40 additions & 0 deletions instrumentation/spring-webflux-5.3.0/build.gradle
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
dependencies {
implementation(project(":agent-bridge"))
implementation("org.springframework:spring-webflux:5.3.0")

testImplementation("org.springframework:spring-context:5.3.1")
testImplementation("io.projectreactor.netty:reactor-netty:1.0.1")

testImplementation(project(":instrumentation:spring-webclient-5.0"))
}

sourceCompatibility = JavaVersion.VERSION_1_8
targetCompatibility = JavaVersion.VERSION_1_8

compileTestJava {
sourceCompatibility = JavaVersion.VERSION_1_8
targetCompatibility = JavaVersion.VERSION_1_8
}

compileJava.options.bootstrapClasspath = null

jar {
manifest { attributes 'Implementation-Title': 'com.newrelic.instrumentation.spring-webflux-5.3.0'}
}

verifyInstrumentation {
passesOnly('org.springframework:spring-webflux:[5.3.0,)')
excludeRegex '.*.M[0-9]'
excludeRegex '.*.RC[0-9]'
}

test {
onlyIf {
!project.hasProperty('test7')
}
}

site {
title 'Spring webclient'
type 'Messaging'
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,140 @@
/*
*
* * Copyright 2020 New Relic Corporation. All rights reserved.
* * SPDX-License-Identifier: Apache-2.0
*
*/

package com.nr.agent.instrumentation.spring.reactive;

import com.newrelic.agent.bridge.Token;
import com.newrelic.api.agent.NewRelic;
import com.newrelic.api.agent.Trace;
import org.reactivestreams.Publisher;
import org.reactivestreams.Subscription;
import org.springframework.web.reactive.function.server.RequestPredicate;
import org.springframework.web.reactive.function.server.ServerRequest;
import reactor.core.CoreSubscriber;
import reactor.core.publisher.Mono;
import reactor.core.publisher.Operators;
import reactor.util.context.Context;

import java.util.function.Function;

public class Util {
public static final String NR_TXN_NAME = "newrelic-transaction-name";
public static final String NR_TOKEN = "newrelic-token";

public static <T> Mono<T> setTransactionToken(Mono<T> mono, Token token) {
return mono.<T>transform(tokenLift(token));
}

public static <T> Function<? super Mono<T>, ? extends Publisher<T>> tokenLift(
Token token) {
return Operators.lift(
(scannable, subscriber) -> new TokenLinkingSubscriber<T>(subscriber, token));
}

public static class TokenLinkingSubscriber<T> implements CoreSubscriber<T> {

private final CoreSubscriber<? super T> subscriber;
private final Context context;

public TokenLinkingSubscriber(
CoreSubscriber<? super T> subscriber, Token token) {
this.subscriber = subscriber;
this.context = subscriber.currentContext().put(NR_TOKEN, token);
}

@Override
public void onSubscribe(Subscription s) {
subscriber.onSubscribe(s);
}

@Override
public void onNext(T t) {
subscriber.onNext(t);
}

@Override
public void onError(Throwable t) {
withNRError(() -> subscriber.onError(t), t);
}

@Override
public void onComplete() {
withNRToken(() -> subscriber.onComplete());
}

@Override
public Context currentContext() {
return context;
}

@Trace(async = true, excludeFromTransactionTrace = true)
private void withNRToken(Runnable runnable) {
Token token = currentContext().get(NR_TOKEN);
if (token != null) {
token.linkAndExpire();
}
runnable.run();
Copy link
Contributor

Choose a reason for hiding this comment

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

I noticed that this line is missing in the same file in spring-webflux-5.1.0 and spring-webflux-5.0.0 modules. Is that a bug there or here?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

It was the issue fixed in #114

Copy link
Contributor

Choose a reason for hiding this comment

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

Ah I see so this branch just hasn't been rebased to in include that code.

}

@Trace(async = true, excludeFromTransactionTrace = true)
private void withNRError(Runnable runnable, Throwable throwable) {
Token token = currentContext().get(NR_TOKEN);
if (token != null && token.isActive()) {
token.linkAndExpire();
NewRelic.noticeError(throwable);
}
runnable.run();
}
}

public static RequestPredicate createRequestPredicate(final String name,
final RequestPredicate originalRequestPredicate) {
return new RequestPredicate() {
@Override
public boolean test(ServerRequest request) {
final boolean matched = originalRequestPredicate.test(request);
if (matched) {
Util.addPath(request, "QueryParameter/" + name);
}
return matched;
}

@Override
public String toString() {
return "";
}
};
}

public static RequestPredicate createPathExtensionPredicate(String extension,
RequestPredicate originalRequestPredicate) {
return new RequestPredicate() {
@Override
public boolean test(ServerRequest request) {
final boolean matched = originalRequestPredicate.test(request);
if (matched) {
Util.addPath(request, "PathExtension/" + extension);
}
return matched;
}

@Override
public String toString() {
return "";
}
};
}

public static void addPath(ServerRequest request, String name) {
Token token = (Token) request.attributes().get(NR_TOKEN);
if (token != null && !name.isEmpty()) {
request.attributes().computeIfAbsent(NR_TXN_NAME, k -> "");
String existingName = (String) request.attributes().get(NR_TXN_NAME);
request.attributes().put(NR_TXN_NAME, existingName + name);
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
/*
*
* * Copyright 2020 New Relic Corporation. All rights reserved.
* * SPDX-License-Identifier: Apache-2.0
*
*/

package org.springframework.http;

import com.newrelic.agent.bridge.AgentBridge;
import com.newrelic.api.agent.Token;
import com.newrelic.api.agent.weaver.MatchType;
import com.newrelic.api.agent.weaver.NewField;
import com.newrelic.api.agent.weaver.Weave;
import com.newrelic.api.agent.weaver.Weaver;
import org.reactivestreams.Publisher;
import org.springframework.core.io.buffer.DataBuffer;
import reactor.core.publisher.Mono;

@Weave(type = MatchType.Interface, originalName = "org.springframework.http.ReactiveHttpOutputMessage")
public class ReactiveHttpOutputMessage_Instrumentation {

@NewField
public Token token;

public Mono<Void> writeWith(Publisher<? extends DataBuffer> body) {
try {
if (this.token != null) {
this.token.expire();
this.token = null;
}
} catch (Throwable t) {
AgentBridge.instrumentation.noticeInstrumentationError(t, Weaver.getImplementationTitle());
}

return Weaver.callOriginal();
}

public Mono<Void> writeAndFlushWith(
Publisher<? extends Publisher<? extends DataBuffer>> body) {
try {
if (this.token != null) {
this.token.expire();
this.token = null;
}
} catch (Throwable t) {
AgentBridge.instrumentation.noticeInstrumentationError(t, Weaver.getImplementationTitle());
}
return Weaver.callOriginal();
}

public Mono<Void> setComplete() {
try {
if (this.token != null) {
this.token.expire();
this.token = null;
}
} catch (Throwable t) {
AgentBridge.instrumentation.noticeInstrumentationError(t, Weaver.getImplementationTitle());
}
return Weaver.callOriginal();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
/*
*
* * Copyright 2020 New Relic Corporation. All rights reserved.
* * SPDX-License-Identifier: Apache-2.0
*
*/

package org.springframework.web.reactive;

import com.newrelic.api.agent.Token;
import com.newrelic.api.agent.weaver.Weave;
import com.nr.agent.instrumentation.spring.reactive.Util;
import org.springframework.http.codec.HttpMessageReader;
import org.springframework.web.server.ServerWebExchange;

import java.util.List;
import java.util.Map;

@Weave(originalName = "org.springframework.web.reactive.function.server.DefaultServerRequest")
abstract class DefaultServerRequest_Instrumentation {

DefaultServerRequest_Instrumentation(ServerWebExchange exchange, List<HttpMessageReader<?>> messageReaders) {
final Token token = exchange == null ? null : exchange.getAttribute(Util.NR_TOKEN);
if (token != null) {
attributes().put(Util.NR_TOKEN, token);
}
}

public abstract Map<String, Object> attributes();
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
package org.springframework.web.reactive;

import com.newrelic.agent.bridge.Token;
import com.newrelic.api.agent.Trace;
import com.newrelic.api.agent.weaver.MatchType;
import com.newrelic.api.agent.weaver.Weave;
import com.newrelic.api.agent.weaver.Weaver;
import com.nr.agent.instrumentation.spring.reactive.Util;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;

// Based on OpenTelemetry instrumentation
// https://github.com/open-telemetry/opentelemetry-java-instrumentation/blob/master/instrumentation/spring/spring-webflux-5.0/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/spring/webflux/server/DispatcherHandlerAdvice.java
@Weave(type =MatchType.ExactClass, originalName = "org.springframework.web.reactive.DispatcherHandler")
public class DispatchHandler_Instrumentation {
@Trace
public Mono<Void> handle(ServerWebExchange exchange) {
final Token token = exchange == null ? null : exchange.getAttribute(Util.NR_TOKEN);

Mono<Void> original = Weaver.callOriginal();

if (token != null) {
return Util.setTransactionToken(original, token);
}
return original;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
/*
*
* * Copyright 2020 New Relic Corporation. All rights reserved.
* * SPDX-License-Identifier: Apache-2.0
*
*/

package org.springframework.web.reactive;

import com.newrelic.api.agent.weaver.Weave;
import com.newrelic.api.agent.weaver.Weaver;
import com.nr.agent.instrumentation.spring.reactive.Util;
import org.springframework.core.io.Resource;
import org.springframework.web.reactive.function.server.ServerRequest;
import org.springframework.web.util.pattern.PathPattern;
import reactor.core.publisher.Mono;

@Weave(originalName = "org.springframework.web.reactive.function.server.PathResourceLookupFunction")
class PathResourceLookupFunction_Instrumentation {
private final PathPattern pattern = Weaver.callOriginal();

private final Resource location = Weaver.callOriginal();

public Mono<Resource> apply(ServerRequest request) {
Mono<Resource> result = Weaver.callOriginal();
if (!Mono.empty().equals(result)) {
Util.addPath(request, pattern.getPatternString());
}
return result;
}
}
Loading