diff --git a/agent-sdk/AsyncPropagationExample.md b/agent-sdk/AsyncPropagationExample.md new file mode 100644 index 000000000000..ae590fa50072 --- /dev/null +++ b/agent-sdk/AsyncPropagationExample.md @@ -0,0 +1,67 @@ +# AsyncContextProgation + +## Runnable instrumentation +Wrap the method you want to trace with `TraceRunnable.asyncEntry()`. + +```java +public class AsyncEntryExample { + private final ExecutorService executor = Executors.newSingleThreadExecutor(); + + @GetMapping(value = "/sdk-async-plugin/asyncEntry-propagation") + public String asyncEntryAndExecute() throws Exception { + CompletableFuture future = new CompletableFuture<>(); + + Runnable command = TraceRunnable.asyncEntry(() -> future.complete("asyncEntry-execute")); + executor.execute(command); + + Thread.sleep(1000); + + return future.get(); + } +} +``` + +## Executor instrumentation +Wrap the executor you want to trace to `TraceExecutors.wrapExecutorService(executr, true)`. + +```java +public class AutoExample { + + private final ExecutorService contextPropagationExecutor + = TraceExecutors.wrapExecutorService(Executors.newSingleThreadExecutor(), true); + + @GetMapping(value = "/sdk-async-plugin/auto-context-propagation") + public String autoWrapAndExecute() throws Exception { + CompletableFuture future = new CompletableFuture<>(); + + contextPropagationExecutor.execute(() -> future.complete("auto-execute")); + + Thread.sleep(1000); + + return future.get(); + } +} +``` + +## Executor and Runnable instrumentation +This method has high tracing precision. +```java +public class ManualExample { + private final ExecutorService traceExecutor + = TraceExecutors.wrapExecutorService(Executors.newSingleThreadExecutor()); + + @GetMapping(value = "/sdk-async-plugin/manual-context-propagation") + public String manualWrapAndExecute() throws Exception { + + CompletableFuture future = new CompletableFuture<>(); + + traceExecutor.execute(TraceRunnable.wrap(() -> future.complete("manual-execute"))); + + traceExecutor.execute(() -> "Not captured"); + + Thread.sleep(1000); + + return future.get(); + } +} +``` diff --git a/agent-sdk/pom.xml b/agent-sdk/pom.xml new file mode 100644 index 000000000000..93c04cca9acf --- /dev/null +++ b/agent-sdk/pom.xml @@ -0,0 +1,25 @@ + + + + + pinpoint + com.navercorp.pinpoint + 2.3.0-SNAPSHOT + + 4.0.0 + + pinpoint-agent-sdk + pinpoint-agent-sdk + + + + + + + + + + + + diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceCallable.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceCallable.java new file mode 100644 index 000000000000..be599db7e605 --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceCallable.java @@ -0,0 +1,49 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.sdk.v1.concurrent; + +import java.util.Objects; +import java.util.concurrent.Callable; + +/** + * {@link Callable} for TraceContext propagation + * @param return type + */ +public class TraceCallable implements Callable { + + public static Callable wrap(Callable delegate) { + return new TraceCallable<>(delegate); + } + + public static Callable asyncEntry(Callable delegate) { + return new TraceCallable<>(delegate); + } + + protected final Callable delegate; + + public TraceCallable(Callable delegate) { + this.delegate = Objects.requireNonNull(delegate, "delegate"); + } + + /** + * the starting point of the async execution + */ + @Override + public V call() throws Exception { + return delegate.call(); + } +} diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceExecutor.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceExecutor.java new file mode 100644 index 000000000000..fad76fa202de --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceExecutor.java @@ -0,0 +1,47 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.sdk.v1.concurrent; + +import com.navercorp.pinpoint.sdk.v1.concurrent.wrapper.CommandWrapper; + +import java.util.Objects; +import java.util.concurrent.Executor; + +/** + * {@link Executor} for TraceContext propagation. + *

{@link TraceScheduledExecutorService} marks the entry point of the async action. + */ +public class TraceExecutor implements Executor { + + protected final Executor delegate; + protected final CommandWrapper wrapper; + + public TraceExecutor(Executor delegate, CommandWrapper wrapper) { + this.delegate = Objects.requireNonNull(delegate, "delegate"); + this.wrapper = Objects.requireNonNull(wrapper, "wrapper"); + } + + + @Override + public void execute(Runnable command) { + Objects.requireNonNull(command, "command"); + + command = wrapper.wrap(command); + delegate.execute(command); + } + +} diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceExecutorService.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceExecutorService.java new file mode 100644 index 000000000000..55dc104cc052 --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceExecutorService.java @@ -0,0 +1,135 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.sdk.v1.concurrent; + +import com.navercorp.pinpoint.sdk.v1.concurrent.wrapper.CommandWrapper; + +import java.util.Collection; +import java.util.List; +import java.util.Objects; +import java.util.concurrent.Callable; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Future; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.TimeoutException; + +/* + * {@link ExecutorService} for TraceContext propagation. + *

{@link TraceExecutorService} marks the entry point of the async action. + */ + +public class TraceExecutorService implements ExecutorService { + + protected final ExecutorService delegate; + protected final CommandWrapper wrapper; + + public TraceExecutorService(ExecutorService delegate, CommandWrapper wrapper) { + this.delegate = Objects.requireNonNull(delegate, "delegate"); + this.wrapper = Objects.requireNonNull(wrapper, "wrapper"); + } + + @Override + public void shutdown() { + delegate.shutdown(); + } + + @Override + public List shutdownNow() { + return delegate.shutdownNow(); + } + + @Override + public boolean isShutdown() { + return delegate.isShutdown(); + } + + @Override + public boolean isTerminated() { + return delegate.isTerminated(); + } + + @Override + public boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException { + return delegate.awaitTermination(timeout, unit); + } + + @Override + public Future submit(Callable task) { + Objects.requireNonNull(task, "task"); + + task = wrapper.wrap(task); + return delegate.submit(task); + } + + @Override + public Future submit(Runnable task, T result) { + Objects.requireNonNull(task, "task"); + + task = wrapper.wrap(task); + return delegate.submit(task, result); + } + + @Override + public Future submit(Runnable task) { + Objects.requireNonNull(task, "task"); + + task = wrapper.wrap(task); + return delegate.submit(task); + } + + @Override + public List> invokeAll(Collection> tasks) throws InterruptedException { + Objects.requireNonNull(tasks, "tasks"); + + tasks = wrapper.wrap(tasks); + return delegate.invokeAll(tasks); + } + + @Override + public List> invokeAll(Collection> tasks, long timeout, TimeUnit unit) throws InterruptedException { + Objects.requireNonNull(tasks, "tasks"); + + tasks = wrapper.wrap(tasks); + return delegate.invokeAll(tasks, timeout, unit); + } + + @Override + public T invokeAny(Collection> tasks) throws InterruptedException, ExecutionException { + Objects.requireNonNull(tasks, "tasks"); + + tasks = wrapper.wrap(tasks); + return delegate.invokeAny(tasks); + } + + @Override + public T invokeAny(Collection> tasks, long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { + Objects.requireNonNull(tasks, "tasks"); + + tasks = wrapper.wrap(tasks); + return delegate.invokeAny(tasks, timeout, unit); + } + + @Override + public void execute(Runnable command) { + Objects.requireNonNull(command, "command"); + + command = wrapper.wrap(command); + delegate.execute(command); + } + +} diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceExecutors.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceExecutors.java new file mode 100644 index 000000000000..bf824fa173f2 --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceExecutors.java @@ -0,0 +1,73 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.sdk.v1.concurrent; + +import com.navercorp.pinpoint.sdk.v1.concurrent.wrapper.DefaultCommandWrapper; +import com.navercorp.pinpoint.sdk.v1.concurrent.wrapper.DisableCommandWrapper; +import com.navercorp.pinpoint.sdk.v1.concurrent.wrapper.CommandWrapper; + +import java.util.Objects; +import java.util.concurrent.Executor; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.ScheduledExecutorService; + +/** + * Utility class for Executor + */ +public class TraceExecutors { + + public static Executor wrapExecutor(Executor executor) { + return wrapExecutor(executor, false); + } + + public static Executor wrapExecutor(Executor executor, boolean autoThreadContextPropagation) { + Objects.requireNonNull(executor, "executor"); + + CommandWrapper wrapper = newCommandWrapper(autoThreadContextPropagation); + return new TraceExecutor(executor, wrapper); + } + + public static ExecutorService wrapExecutorService(ExecutorService executorService) { + return wrapExecutorService(executorService, false); + } + + public static ExecutorService wrapExecutorService(ExecutorService executorService, boolean autoThreadContextPropagation) { + Objects.requireNonNull(executorService, "executorService"); + + CommandWrapper wrapper = newCommandWrapper(autoThreadContextPropagation); + return new TraceExecutorService(executorService, wrapper); + } + + public static ScheduledExecutorService wrapScheduledExecutorService(ScheduledExecutorService executorService) { + return wrapScheduledExecutorService(executorService, false); + } + + public static ScheduledExecutorService wrapScheduledExecutorService(ScheduledExecutorService executorService, boolean autoThreadContextPropagation) { + Objects.requireNonNull(executorService, "executorService"); + + CommandWrapper wrapper = newCommandWrapper(autoThreadContextPropagation); + return new TraceScheduledExecutorService(executorService, wrapper); + } + + private static CommandWrapper newCommandWrapper(boolean autoThreadContextPropagation) { + if (autoThreadContextPropagation) { + return new DefaultCommandWrapper(); + } + return new DisableCommandWrapper(); + } + +} diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceForkJoinPool.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceForkJoinPool.java new file mode 100644 index 000000000000..fc98b6e06401 --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceForkJoinPool.java @@ -0,0 +1,115 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.sdk.v1.concurrent; + +import com.navercorp.pinpoint.sdk.v1.concurrent.wrapper.DefaultCommandWrapper; +import com.navercorp.pinpoint.sdk.v1.concurrent.wrapper.CommandWrapper; + +import java.util.Collection; +import java.util.List; +import java.util.concurrent.Callable; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.ForkJoinPool; +import java.util.concurrent.ForkJoinTask; +import java.util.concurrent.Future; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.TimeoutException; + +/** + * Not open yet. + */ +class TraceForkJoinPool extends ForkJoinPool { + + protected CommandWrapper commandWrapper = new DefaultCommandWrapper(); + + private TraceForkJoinPool() { + } + + private TraceForkJoinPool(int parallelism) { + super(parallelism); + } + + + private TraceForkJoinPool(int parallelism, + ForkJoinWorkerThreadFactory factory, + Thread.UncaughtExceptionHandler handler, + boolean asyncMode) { + super(parallelism, factory, handler, asyncMode); + } + + + + protected ForkJoinTask wrap(ForkJoinTask task) { + // TODO How to delegate ForkJoinTask? + return task; + } + + + + public ForkJoinTask submit(ForkJoinTask task) { + task = wrap(task); + return super.submit(task); + } + + + public ForkJoinTask submit(Callable task) { + task = commandWrapper.wrap(task); + return super.submit(task); + } + + + public ForkJoinTask submit(Runnable task, T result) { + task = commandWrapper.wrap(task); + return super.submit(task, result); + } + + + public ForkJoinTask submit(Runnable task) { + task = commandWrapper.wrap(task); + return super.submit(task); + } + + @Override + public T invokeAny(Collection> tasks) throws InterruptedException, ExecutionException { + tasks = commandWrapper.wrap(tasks); + return super.invokeAny(tasks); + } + + @Override + public T invokeAny(Collection> tasks, long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { + tasks = commandWrapper.wrap(tasks); + return super.invokeAny(tasks, timeout, unit); + } + + @Override + public List> invokeAll(Collection> tasks, long timeout, TimeUnit unit) throws InterruptedException { + tasks = commandWrapper.wrap(tasks); + return super.invokeAll(tasks, timeout, unit); + } + + public void execute(ForkJoinTask task) { + task = wrap(task); + super.execute(task); + } + + + public void execute(Runnable task) { + task = commandWrapper.wrap(task); + super.execute(task); + } + +} diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceRunnable.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceRunnable.java new file mode 100644 index 000000000000..9589223a5ad9 --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceRunnable.java @@ -0,0 +1,46 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.sdk.v1.concurrent; + +import java.util.Objects; + +/** + * {@link Runnable} for TraceContext propagation + */ +public class TraceRunnable implements Runnable { + + public static Runnable wrap(Runnable delegate) { + return new TraceRunnable(delegate); + } + + public static Runnable asyncEntry(Runnable delegate) { + return new TraceRunnable(delegate); + } + + protected final Runnable delegate; + + public TraceRunnable(Runnable runnable) { + this.delegate = Objects.requireNonNull(runnable, "delegate"); + } + + /** + * the starting point of the async execution + */ + public void run() { + this.delegate.run(); + } +} diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceScheduledExecutorService.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceScheduledExecutorService.java new file mode 100644 index 000000000000..e0e9d52a557f --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/TraceScheduledExecutorService.java @@ -0,0 +1,159 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.sdk.v1.concurrent; + +import com.navercorp.pinpoint.sdk.v1.concurrent.wrapper.CommandWrapper; + +import java.util.Collection; +import java.util.List; +import java.util.Objects; +import java.util.concurrent.Callable; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.Future; +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.ScheduledFuture; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.TimeoutException; + +/** + * {@link ScheduledExecutorService} for TraceContext propagation. + *

{@link TraceScheduledExecutorService} marks the entry point of the async action. + */ +public class TraceScheduledExecutorService implements ScheduledExecutorService { + + /** + * Inheritance is not recommended. + * No inheritance makes the trace more intuitive. + * e.g) class TraceExecutorService extends TraceExecutor {} + */ + + protected final ScheduledExecutorService delegate; + protected final CommandWrapper wrapper; + + public TraceScheduledExecutorService(ScheduledExecutorService delegate, CommandWrapper wrapper) { + this.delegate = Objects.requireNonNull(delegate, "delegate"); + this.wrapper = Objects.requireNonNull(wrapper, "wrapper"); + } + + @Override + public ScheduledFuture schedule(Runnable command, long delay, TimeUnit unit) { + command = wrapper.wrap(command); + return delegate.schedule(command, delay, unit); + } + + + @Override + public ScheduledFuture schedule(Callable callable, long delay, TimeUnit unit) { + callable = wrapper.wrap(callable); + return delegate.schedule(callable, delay, unit); + } + + @Override + public ScheduledFuture scheduleAtFixedRate(Runnable command, long initialDelay, long period, TimeUnit unit) { + command = wrapper.wrap(command); + return delegate.scheduleAtFixedRate(command, initialDelay, period, unit); + } + + @Override + public ScheduledFuture scheduleWithFixedDelay(Runnable command, long initialDelay, long delay, TimeUnit unit) { + command = wrapper.wrap(command); + return delegate.scheduleWithFixedDelay(command, initialDelay, delay, unit); + } + + @Override + public void shutdown() { + delegate.shutdown(); + } + + @Override + public List shutdownNow() { + return delegate.shutdownNow(); + } + + @Override + public boolean isShutdown() { + return delegate.isShutdown(); + } + + @Override + public boolean isTerminated() { + return delegate.isTerminated(); + } + + @Override + public boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException { + return delegate.awaitTermination(timeout, unit); + } + + @Override + public Future submit(Callable task) { + task = wrapper.wrap(task); + return delegate.submit(task); + } + + @Override + public Future submit(Runnable task, T result) { + task = wrapper.wrap(task); + return delegate.submit(task, result); + } + + @Override + public Future submit(Runnable task) { + task = wrapper.wrap(task); + return delegate.submit(task); + } + + @Override + public List> invokeAll(Collection> tasks) throws InterruptedException { +// tasks = wrapper.wrap(tasks); +// return delegate.invokeAll(wrapTasks); + + return delegate.invokeAll(tasks); + } + + + + @Override + public List> invokeAll(Collection> tasks, long timeout, TimeUnit unit) throws InterruptedException { +// tasks = wrapper.wrap(tasks); +// return delegate.invokeAll(tasks, timeout, unit); + + return delegate.invokeAll(tasks, timeout, unit); + } + + @Override + public T invokeAny(Collection> tasks) throws InterruptedException, ExecutionException { +// tasks = wrapper.wrap(tasks); +// return delegate.invokeAny(wrapTasks); + + return delegate.invokeAny(tasks); + } + + @Override + public T invokeAny(Collection> tasks, long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { +// tasks = wrapper.wrap(tasks); +// return delegate.invokeAny(wrapTasks, timeout, unit); + + return delegate.invokeAny(tasks, timeout, unit); + } + + @Override + public void execute(Runnable command) { + command = wrapper.wrap(command); + delegate.execute(command); + } +} diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/wrapper/CommandWrapper.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/wrapper/CommandWrapper.java new file mode 100644 index 000000000000..29616190aeb4 --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/wrapper/CommandWrapper.java @@ -0,0 +1,12 @@ +package com.navercorp.pinpoint.sdk.v1.concurrent.wrapper; + +import java.util.Collection; +import java.util.concurrent.Callable; + +public interface CommandWrapper { + Runnable wrap(Runnable command); + + Callable wrap(Callable callable); + + Collection> wrap(Collection> tasks); +} diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/wrapper/DefaultCommandWrapper.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/wrapper/DefaultCommandWrapper.java new file mode 100644 index 000000000000..95bf8393ef45 --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/wrapper/DefaultCommandWrapper.java @@ -0,0 +1,35 @@ +package com.navercorp.pinpoint.sdk.v1.concurrent.wrapper; + +import com.navercorp.pinpoint.sdk.v1.concurrent.TraceCallable; +import com.navercorp.pinpoint.sdk.v1.concurrent.TraceRunnable; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; +import java.util.concurrent.Callable; + +public class DefaultCommandWrapper implements CommandWrapper { + + public Runnable wrap(Runnable command) { + if (command instanceof TraceRunnable) { + return command; + } + return TraceRunnable.asyncEntry(command); + } + + public Callable wrap(Callable callable) { + if (callable instanceof TraceCallable) { + return callable; + } + return TraceCallable.asyncEntry(callable); + } + + public Collection> wrap(Collection> tasks) { + final List> wrapList = new ArrayList<>(tasks.size()); + for (Callable task : tasks) { + Callable wrapTask = wrap(task); + wrapList.add(wrapTask); + } + return wrapList; + } +} diff --git a/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/wrapper/DisableCommandWrapper.java b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/wrapper/DisableCommandWrapper.java new file mode 100644 index 000000000000..b2719d236d18 --- /dev/null +++ b/agent-sdk/src/main/java/com/navercorp/pinpoint/sdk/v1/concurrent/wrapper/DisableCommandWrapper.java @@ -0,0 +1,21 @@ +package com.navercorp.pinpoint.sdk.v1.concurrent.wrapper; + +import java.util.Collection; +import java.util.concurrent.Callable; + +public class DisableCommandWrapper implements CommandWrapper { + @Override + public Runnable wrap(Runnable command) { + return command; + } + + @Override + public Callable wrap(Callable callable) { + return callable; + } + + @Override + public Collection> wrap(Collection> tasks) { + return tasks; + } +} diff --git a/agent-testweb/agentsdk-async-testweb/README.md b/agent-testweb/agentsdk-async-testweb/README.md new file mode 100644 index 000000000000..cb7393c1fbda --- /dev/null +++ b/agent-testweb/agentsdk-async-testweb/README.md @@ -0,0 +1,16 @@ + +## Install +``` +$ mvnw -P pinpoint-agentsdk-async-testweb install -Dmaven.test.skip=true +``` + +## Run +``` +$ mvnw -P pinpoint-agentsdk-async-testweb spring-boot:start +``` +You can then access here: http://localhost:18080/ + +## Stop +``` +$ mvnw -P pinpoint-agentsdk-async-testweb spring-boot:stop +``` diff --git a/agent-testweb/agentsdk-async-testweb/pom.xml b/agent-testweb/agentsdk-async-testweb/pom.xml new file mode 100644 index 000000000000..c5275c3ac5b1 --- /dev/null +++ b/agent-testweb/agentsdk-async-testweb/pom.xml @@ -0,0 +1,29 @@ + + + 4.0.0 + + com.navercorp.pinpoint + pinpoint-agent-testweb + 2.3.0-SNAPSHOT + + + pinpoint-agentsdk-async-testweb + + jar + + + + ${pinpoint.agent.default.jvmargument} + + + + + + com.navercorp.pinpoint + pinpoint-agent-sdk + ${project.version} + + + + \ No newline at end of file diff --git a/agent-testweb/agentsdk-async-testweb/src/main/java/com/pinpointest/plugin/SdkAsyncPluginTestStarter.java b/agent-testweb/agentsdk-async-testweb/src/main/java/com/pinpointest/plugin/SdkAsyncPluginTestStarter.java new file mode 100644 index 000000000000..747a726aa501 --- /dev/null +++ b/agent-testweb/agentsdk-async-testweb/src/main/java/com/pinpointest/plugin/SdkAsyncPluginTestStarter.java @@ -0,0 +1,12 @@ +package com.pinpointest.plugin; + +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.SpringBootApplication; + +@SpringBootApplication +public class SdkAsyncPluginTestStarter { + + public static void main(String[] args) { + SpringApplication.run(SdkAsyncPluginTestStarter.class, args); + } +} diff --git a/agent-testweb/agentsdk-async-testweb/src/main/java/com/pinpointest/plugin/controller/ThreadContextPropagationController.java b/agent-testweb/agentsdk-async-testweb/src/main/java/com/pinpointest/plugin/controller/ThreadContextPropagationController.java new file mode 100644 index 000000000000..d9b283eb9524 --- /dev/null +++ b/agent-testweb/agentsdk-async-testweb/src/main/java/com/pinpointest/plugin/controller/ThreadContextPropagationController.java @@ -0,0 +1,104 @@ +package com.pinpointest.plugin.controller; + +import com.navercorp.pinpoint.sdk.v1.concurrent.TraceCallable; +import com.navercorp.pinpoint.sdk.v1.concurrent.TraceExecutors; +import com.navercorp.pinpoint.sdk.v1.concurrent.TraceRunnable; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.RestController; + +import javax.annotation.PreDestroy; +import java.util.Arrays; +import java.util.List; +import java.util.concurrent.Callable; +import java.util.concurrent.CompletableFuture; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.Future; +import java.util.concurrent.TimeUnit; + +@RestController +public class ThreadContextPropagationController { + private final Logger logger = LoggerFactory.getLogger(this.getClass()); + + private final ExecutorService traceExecutor = TraceExecutors.wrapExecutorService(Executors.newSingleThreadExecutor()); + + @GetMapping(value = "/sdk-async-plugin/manual-context-propagation") + public String manualWrapAndExecute() throws Exception { + + CompletableFuture future = new CompletableFuture<>(); + + traceExecutor.execute(TraceRunnable.wrap(() -> future.complete("manual-execute"))); + + Thread.sleep(1000); + + return future.get(); + } + + private final ExecutorService contextPropagationExecutor = TraceExecutors.wrapExecutorService(Executors.newSingleThreadExecutor(), true); + + @GetMapping(value = "/sdk-async-plugin/auto-context-propagation") + public String autoWrapAndExecute() throws Exception { + CompletableFuture future = new CompletableFuture<>(); + + contextPropagationExecutor.execute(() -> future.complete("auto-execute")); + + Thread.sleep(1000); + + return future.get(); + } + + private final ExecutorService executor = Executors.newSingleThreadExecutor(); + + @GetMapping(value = "/sdk-async-plugin/asyncEntry-propagation") + public String asyncEntryAndExecute() throws Exception { + CompletableFuture future = new CompletableFuture<>(); + + Runnable command = TraceRunnable.asyncEntry(() -> future.complete("asyncEntry-execute")); + executor.execute(command); + + Thread.sleep(1000); + + return future.get(); + } + + @GetMapping(value = "/sdk-async-plugin/call") + public String manualWrapAndCall() throws Exception { + Callable command = TraceCallable.wrap(() -> "asyncEntry-execute1"); + Future future = traceExecutor.submit(command); + + Thread.sleep(1000); + + return future.get(3000, TimeUnit.MILLISECONDS); + } + + @GetMapping(value = "/sdk-async-plugin/asyncEntryCall") + public String manualWrapAndAsyncEntryCall() throws Exception { + Callable command = TraceCallable.asyncEntry(() -> "asyncEntry-execute1"); + Future future = executor.submit(command); + + Thread.sleep(1000); + + return future.get(3000, TimeUnit.MILLISECONDS); + } + + + @GetMapping(value = "/sdk-async-plugin/invokeAll") + public String invokeAll() throws Exception { + Callable command1 = TraceCallable.asyncEntry(() -> "asyncEntry-execute1"); + Callable command2 = TraceCallable.asyncEntry(() -> "asyncEntry-execute1"); + List> futures = executor.invokeAll(Arrays.asList(command1, command2)); + + Thread.sleep(1000); + + return futures.get(0).get(3000, TimeUnit.MILLISECONDS); + } + + @PreDestroy + private void shutdown() { + this.traceExecutor.shutdown(); + this.contextPropagationExecutor.shutdown(); + this.executor.shutdown(); + } +} diff --git a/agent-testweb/agentsdk-async-testweb/src/main/resources/application.yml b/agent-testweb/agentsdk-async-testweb/src/main/resources/application.yml new file mode 100644 index 000000000000..2416cb576dc9 --- /dev/null +++ b/agent-testweb/agentsdk-async-testweb/src/main/resources/application.yml @@ -0,0 +1,12 @@ +# Defined in commandlineArgument of agent-test pom.xml + +#server: +# port: 18080 + +#logging: +# level: +# root: info + +#springdoc: +# swagger-ui: +# path: / \ No newline at end of file diff --git a/agent-testweb/pom.xml b/agent-testweb/pom.xml index 2200186e5e0e..b630b67ef563 100644 --- a/agent-testweb/pom.xml +++ b/agent-testweb/pom.xml @@ -51,6 +51,7 @@ + agentsdk-async-testweb thread-plugin-testweb paho-mqtt-plugin-testweb reactor-netty-plugin-testweb diff --git a/bootstraps/bootstrap-core/src/main/java/com/navercorp/pinpoint/bootstrap/instrument/transformer/TransformTemplate.java b/bootstraps/bootstrap-core/src/main/java/com/navercorp/pinpoint/bootstrap/instrument/transformer/TransformTemplate.java index fa9b9221dca2..6194cb7f0562 100644 --- a/bootstraps/bootstrap-core/src/main/java/com/navercorp/pinpoint/bootstrap/instrument/transformer/TransformTemplate.java +++ b/bootstraps/bootstrap-core/src/main/java/com/navercorp/pinpoint/bootstrap/instrument/transformer/TransformTemplate.java @@ -85,8 +85,6 @@ public void transform(String className, Class trans Objects.requireNonNull(className, "className"); Objects.requireNonNull(transformCallbackClass, "transformCallbackClass"); - - TransformCallbackChecker.validate(transformCallbackClass, parameterTypes); if (ParameterUtils.hasNull(parameterTypes)) { throw new IllegalArgumentException("null parameterType not supported"); diff --git a/commons/src/main/java/com/navercorp/pinpoint/common/trace/ServiceType.java b/commons/src/main/java/com/navercorp/pinpoint/common/trace/ServiceType.java index 7d05e7a830be..07e0eb7c63dc 100644 --- a/commons/src/main/java/com/navercorp/pinpoint/common/trace/ServiceType.java +++ b/commons/src/main/java/com/navercorp/pinpoint/common/trace/ServiceType.java @@ -32,6 +32,8 @@ * 5TEST * 7COLLECTOR * 100ASYNC + * 500SDK + * 510SDK_ASYNC * * * diff --git a/plugins/agentsdk-async/pom.xml b/plugins/agentsdk-async/pom.xml new file mode 100644 index 000000000000..d292f607ba34 --- /dev/null +++ b/plugins/agentsdk-async/pom.xml @@ -0,0 +1,25 @@ + + + + 4.0.0 + + com.navercorp.pinpoint + pinpoint-plugins + 2.3.0-SNAPSHOT + + + pinpoint-agentsdk-async-plugin + pinpoint-agentsdk-async-plugin + jar + + + + com.navercorp.pinpoint + pinpoint-bootstrap-core + provided + + + + + diff --git a/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncConfig.java b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncConfig.java new file mode 100644 index 000000000000..9b6844017ceb --- /dev/null +++ b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncConfig.java @@ -0,0 +1,41 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.plugin.sdk; + +import com.navercorp.pinpoint.bootstrap.config.ProfilerConfig; + +public class AgentSdkAsyncConfig { + + private final boolean enable; + + + public AgentSdkAsyncConfig(ProfilerConfig config) { + this.enable = config.readBoolean("profiler.agentsdk.enable", true); + } + + public boolean isEnable() { + return enable; + } + + @Override + public String toString() { + final StringBuilder sb = new StringBuilder("AgentSdkConfig{"); + sb.append("enable=").append(enable); + sb.append('}'); + return sb.toString(); + } +} diff --git a/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncConstants.java b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncConstants.java new file mode 100644 index 000000000000..1cf9ea9e8d74 --- /dev/null +++ b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncConstants.java @@ -0,0 +1,26 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.plugin.sdk; + +import com.navercorp.pinpoint.common.trace.ServiceType; +import com.navercorp.pinpoint.common.trace.ServiceTypeFactory; + +public class AgentSdkAsyncConstants { + public static final ServiceType AGENT_SDK_ASYNC = ServiceTypeFactory.of(510, "SDK_ASYNC", "SDK_ASYNC"); + public static final String AGENT_SDK_ASYNC_SCOPE = "SDK_ASYNC"; + +} diff --git a/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncPlugin.java b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncPlugin.java new file mode 100644 index 000000000000..27fb41d4969c --- /dev/null +++ b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncPlugin.java @@ -0,0 +1,161 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.plugin.sdk; + +import com.navercorp.pinpoint.bootstrap.async.AsyncContextAccessor; +import com.navercorp.pinpoint.bootstrap.instrument.InstrumentClass; +import com.navercorp.pinpoint.bootstrap.instrument.InstrumentException; +import com.navercorp.pinpoint.bootstrap.instrument.InstrumentMethod; +import com.navercorp.pinpoint.bootstrap.instrument.Instrumentor; +import com.navercorp.pinpoint.bootstrap.instrument.MethodFilters; +import com.navercorp.pinpoint.bootstrap.instrument.transformer.TransformCallback; +import com.navercorp.pinpoint.bootstrap.instrument.transformer.TransformTemplate; +import com.navercorp.pinpoint.bootstrap.instrument.transformer.TransformTemplateAware; +import com.navercorp.pinpoint.bootstrap.logging.PLogger; +import com.navercorp.pinpoint.bootstrap.logging.PLoggerFactory; +import com.navercorp.pinpoint.bootstrap.plugin.ProfilerPlugin; +import com.navercorp.pinpoint.bootstrap.plugin.ProfilerPluginSetupContext; +import com.navercorp.pinpoint.plugin.sdk.interceptor.AsyncEntryInterceptor; +import com.navercorp.pinpoint.plugin.sdk.interceptor.CommandInterceptor; +import com.navercorp.pinpoint.plugin.sdk.interceptor.ExecutorExecuteInterceptor; + + +import java.security.ProtectionDomain; + +public class AgentSdkAsyncPlugin implements ProfilerPlugin, TransformTemplateAware { + private final PLogger logger = PLoggerFactory.getLogger(getClass()); + + private TransformTemplate transformTemplate; + + + @Override + public void setup(ProfilerPluginSetupContext context) { + AgentSdkAsyncConfig agentSdkAsyncConfig = new AgentSdkAsyncConfig(context.getConfig()); + if (!agentSdkAsyncConfig.isEnable()) { + logger.info("AgentSdkAsyncConfig is disable"); + return; + } + + logger.info("AgentSdkAsyncConfig config={}", agentSdkAsyncConfig); + + String sdkPackage = "com.navercorp.pinpoint.sdk.v1.concurrent"; + + addTraceRunnableInterceptorTask(sdkPackage + ".TraceRunnable"); + addTraceCallableInterceptorTask(sdkPackage + ".TraceCallable"); + + + addTraceExecutor(sdkPackage + ".TraceExecutorService"); + addTraceExecutor(sdkPackage + ".TraceScheduledExecutorService"); + addTraceExecutor(sdkPackage + ".TraceExecutor"); + +// addAsyncTaskExecutor(sdkPackage + ".TraceForkJoinPool"); + + } + + + private void addTraceRunnableInterceptorTask(final String className) { + transformTemplate.transform(className, TraceRunnableCallback.class); + } + + public static class TraceRunnableCallback implements TransformCallback { + + public TraceRunnableCallback() { + } + + @Override + public byte[] doInTransform(Instrumentor instrumentor, ClassLoader loader, String className, Class classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws InstrumentException { + final InstrumentClass target = instrumentor.getInstrumentClass(loader, className, classfileBuffer); + target.addField(AsyncContextAccessor.class); + final InstrumentMethod callMethod = target.getDeclaredMethod("run"); + if (callMethod != null) { + callMethod.addInterceptor(CommandInterceptor.class); + } + InstrumentMethod asyncEntry = target.getDeclaredMethod("asyncEntry", "java.lang.Runnable"); + if (asyncEntry != null) { + asyncEntry.addInterceptor(AsyncEntryInterceptor.class); + } + return target.toBytecode(); + } + } + + private void addTraceCallableInterceptorTask(final String className) { + transformTemplate.transform(className, TraceCallableCallback.class); + } + + public static class TraceCallableCallback implements TransformCallback { + + public TraceCallableCallback() { + } + + @Override + public byte[] doInTransform(Instrumentor instrumentor, ClassLoader loader, String className, Class classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws InstrumentException { + final InstrumentClass target = instrumentor.getInstrumentClass(loader, className, classfileBuffer); + target.addField(AsyncContextAccessor.class); + final InstrumentMethod callMethod = target.getDeclaredMethod("call"); + if (callMethod != null) { + callMethod.addInterceptor(CommandInterceptor.class); + } + InstrumentMethod asyncEntry = target.getDeclaredMethod("asyncEntry", "java.util.concurrent.Callable"); + if (asyncEntry != null) { + asyncEntry.addInterceptor(AsyncEntryInterceptor.class); + } + return target.toBytecode(); + } + } + + + private void addTraceExecutor(final String className) { + transformTemplate.transform(className, ExecutorExecuteTransformCallback.class); + } + + public static class ExecutorExecuteTransformCallback implements TransformCallback { +// private final PLogger logger = PLoggerFactory.getLogger(getClass()); + + @Override + public byte[] doInTransform(Instrumentor instrumentor, ClassLoader classLoader, String className, Class classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws InstrumentException { + final InstrumentClass target = instrumentor.getInstrumentClass(classLoader, className, classfileBuffer); + for (InstrumentMethod m : target.getDeclaredMethods(MethodFilters.name("execute"))) { + m.addScopedInterceptor(ExecutorExecuteInterceptor.class, AgentSdkAsyncConstants.AGENT_SDK_ASYNC_SCOPE); + } + for (InstrumentMethod m : target.getDeclaredMethods(MethodFilters.name("submit"))) { + m.addScopedInterceptor(ExecutorExecuteInterceptor.class, AgentSdkAsyncConstants.AGENT_SDK_ASYNC_SCOPE); + } + for (InstrumentMethod m : target.getDeclaredMethods(MethodFilters.name("schedule"))) { + m.addScopedInterceptor(ExecutorExecuteInterceptor.class, AgentSdkAsyncConstants.AGENT_SDK_ASYNC_SCOPE); + } + for (InstrumentMethod m : target.getDeclaredMethods(MethodFilters.name("scheduleAtFixedRate"))) { + m.addScopedInterceptor(ExecutorExecuteInterceptor.class, AgentSdkAsyncConstants.AGENT_SDK_ASYNC_SCOPE); + } + for (InstrumentMethod m : target.getDeclaredMethods(MethodFilters.name("scheduleWithFixedDelay"))) { + m.addScopedInterceptor(ExecutorExecuteInterceptor.class, AgentSdkAsyncConstants.AGENT_SDK_ASYNC_SCOPE); + } +// for (InstrumentMethod m : target.getDeclaredMethods(MethodFilters.name("invokeAll"))) { +// m.addScopedInterceptor(ThreadPoolExecutorSubmitInterceptor.class, AgentSdkAsyncConstants.THREAD_POOL_EXECUTOR_SCOPE); +// } +// for (InstrumentMethod m : target.getDeclaredMethods(MethodFilters.name("invokeAny"))) { +// m.addScopedInterceptor(ThreadPoolExecutorSubmitInterceptor.class, AgentSdkAsyncConstants.THREAD_POOL_EXECUTOR_SCOPE); +// } + return target.toBytecode(); + } + } + + + @Override + public void setTransformTemplate(TransformTemplate transformTemplate) { + this.transformTemplate = transformTemplate; + } +} diff --git a/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncTraceMetadataProvider.java b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncTraceMetadataProvider.java new file mode 100644 index 000000000000..947f7010b568 --- /dev/null +++ b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/AgentSdkAsyncTraceMetadataProvider.java @@ -0,0 +1,27 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.plugin.sdk; + +import com.navercorp.pinpoint.common.trace.TraceMetadataProvider; +import com.navercorp.pinpoint.common.trace.TraceMetadataSetupContext; + +public class AgentSdkAsyncTraceMetadataProvider implements TraceMetadataProvider { + @Override + public void setup(TraceMetadataSetupContext context) { + context.addServiceType(AgentSdkAsyncConstants.AGENT_SDK_ASYNC); + } +} diff --git a/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/interceptor/AsyncEntryInterceptor.java b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/interceptor/AsyncEntryInterceptor.java new file mode 100644 index 000000000000..0bdd6e026b9c --- /dev/null +++ b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/interceptor/AsyncEntryInterceptor.java @@ -0,0 +1,105 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.plugin.sdk.interceptor; + +import com.navercorp.pinpoint.bootstrap.async.AsyncContextAccessor; +import com.navercorp.pinpoint.bootstrap.context.AsyncContext; +import com.navercorp.pinpoint.bootstrap.context.MethodDescriptor; +import com.navercorp.pinpoint.bootstrap.context.SpanEventRecorder; +import com.navercorp.pinpoint.bootstrap.context.Trace; +import com.navercorp.pinpoint.bootstrap.context.TraceContext; +import com.navercorp.pinpoint.bootstrap.interceptor.AroundInterceptor; +import com.navercorp.pinpoint.bootstrap.logging.PLogger; +import com.navercorp.pinpoint.bootstrap.logging.PLoggerFactory; +import com.navercorp.pinpoint.plugin.sdk.AgentSdkAsyncConstants; + +import java.util.Objects; + +/** + * TraceRunnable.asyncEntry(Runnable); + *

TraceCallable.asyncEntry(Callable); + */ +public class AsyncEntryInterceptor implements AroundInterceptor { + + private final PLogger logger = PLoggerFactory.getLogger(this.getClass()); + private final boolean isDebug = logger.isDebugEnabled(); + + private final TraceContext traceContext; + private final MethodDescriptor descriptor; + + public AsyncEntryInterceptor(TraceContext traceContext, MethodDescriptor descriptor) { + this.traceContext = Objects.requireNonNull(traceContext, "traceContext"); + this.descriptor = Objects.requireNonNull(descriptor, "descriptor"); + } + + @Override + public void before(Object target, Object[] args) { + if (isDebug) { + logger.beforeInterceptor(target, args); + } + + final Trace trace = traceContext.currentTraceObject(); + if (trace == null) { + return; + } + + trace.traceBlockBegin(); + + } + + private boolean validate(final Object result) { + if (!(result instanceof AsyncContextAccessor)) { + if (isDebug) { + logger.debug("Invalid result object. Need metadata accessor({}).", AsyncContextAccessor.class.getName()); + } + return false; + } + + return true; + } + + @Override + public void after(Object target, Object[] args, Object result, Throwable throwable) { + if (isDebug) { + logger.afterInterceptor(target, args, result, throwable); + } + + final Trace trace = traceContext.currentTraceObject(); + if (trace == null) { + return; + } + + try { + final SpanEventRecorder recorder = trace.currentSpanEventRecorder(); + recorder.recordApi(this.descriptor); + recorder.recordServiceType(AgentSdkAsyncConstants.AGENT_SDK_ASYNC); + recorder.recordException(throwable); + + boolean r = validate(result); + if (r) { + // make asynchronous trace-id + final AsyncContext asyncContext = recorder.recordNextAsyncContext(); + ((AsyncContextAccessor) result)._$PINPOINT$_setAsyncContext(asyncContext); + if (isDebug) { + logger.debug("Set asyncContext {}", asyncContext); + } + } + } finally { + trace.traceBlockEnd(); + } + } +} diff --git a/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/interceptor/CommandInterceptor.java b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/interceptor/CommandInterceptor.java new file mode 100644 index 000000000000..a584e6b0b699 --- /dev/null +++ b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/interceptor/CommandInterceptor.java @@ -0,0 +1,44 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.plugin.sdk.interceptor; + +import com.navercorp.pinpoint.bootstrap.context.AsyncContext; +import com.navercorp.pinpoint.bootstrap.context.MethodDescriptor; +import com.navercorp.pinpoint.bootstrap.context.SpanEventRecorder; +import com.navercorp.pinpoint.bootstrap.context.TraceContext; +import com.navercorp.pinpoint.bootstrap.interceptor.AsyncContextSpanEventSimpleAroundInterceptor; +import com.navercorp.pinpoint.plugin.sdk.AgentSdkAsyncConstants; + + +public class CommandInterceptor extends AsyncContextSpanEventSimpleAroundInterceptor { +// private final PLogger logger = PLoggerFactory.getLogger(this.getClass()); + + public CommandInterceptor(TraceContext traceContext, MethodDescriptor methodDescriptor) { + super(traceContext, methodDescriptor); + } + + @Override + protected void doInBeforeTrace(SpanEventRecorder recorder, AsyncContext asyncContext, Object target, Object[] args) { + } + + @Override + protected void doInAfterTrace(SpanEventRecorder recorder, Object target, Object[] args, Object result, Throwable throwable) { + recorder.recordApi(methodDescriptor); + recorder.recordServiceType(AgentSdkAsyncConstants.AGENT_SDK_ASYNC); + recorder.recordException(throwable); + } +} diff --git a/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/interceptor/ExecutorExecuteInterceptor.java b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/interceptor/ExecutorExecuteInterceptor.java new file mode 100644 index 000000000000..c677f9bce62c --- /dev/null +++ b/plugins/agentsdk-async/src/main/java/com/navercorp/pinpoint/plugin/sdk/interceptor/ExecutorExecuteInterceptor.java @@ -0,0 +1,104 @@ +/* + * Copyright 2021 NAVER Corp. + * + * 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. + */ + +package com.navercorp.pinpoint.plugin.sdk.interceptor; + +import com.navercorp.pinpoint.bootstrap.async.AsyncContextAccessor; +import com.navercorp.pinpoint.bootstrap.context.*; +import com.navercorp.pinpoint.bootstrap.interceptor.AroundInterceptor; +import com.navercorp.pinpoint.bootstrap.logging.PLogger; +import com.navercorp.pinpoint.bootstrap.logging.PLoggerFactory; +import com.navercorp.pinpoint.plugin.sdk.AgentSdkAsyncConstants; + +import java.util.Objects; + +public class ExecutorExecuteInterceptor implements AroundInterceptor { + + private final PLogger logger = PLoggerFactory.getLogger(this.getClass()); + private final boolean isDebug = logger.isDebugEnabled(); + + private final TraceContext traceContext; + private final MethodDescriptor descriptor; + + public ExecutorExecuteInterceptor(TraceContext traceContext, MethodDescriptor descriptor) { + this.traceContext = Objects.requireNonNull(traceContext, "traceContext"); + this.descriptor = Objects.requireNonNull(descriptor, "descriptor"); + } + + @Override + public void before(Object target, Object[] args) { + if (isDebug) { + logger.beforeInterceptor(target, args); + } + + final Trace trace = traceContext.currentTraceObject(); + if (trace == null) { + return; + } + + final SpanEventRecorder recorder = trace.traceBlockBegin(); + + + boolean r = validate(args); + if (r) { + // make asynchronous trace-id + final AsyncContext asyncContext = recorder.recordNextAsyncContext(); + ((AsyncContextAccessor) args[0])._$PINPOINT$_setAsyncContext(asyncContext); + if (isDebug) { + logger.debug("Set asyncContext {}", asyncContext); + } + } + } + + private boolean validate(final Object[] args) { + if (args == null || args.length < 1) { + if (isDebug) { + logger.debug("Invalid args object. args={}.", args); + } + return false; + } + + if (!(args[0] instanceof AsyncContextAccessor)) { + if (isDebug) { + logger.debug("Invalid args[0] object. Need metadata accessor({}).", AsyncContextAccessor.class.getName()); + } + return false; + } + + return true; + } + + @Override + public void after(Object target, Object[] args, Object result, Throwable throwable) { + if (isDebug) { + logger.afterInterceptor(target, args, result, throwable); + } + + final Trace trace = traceContext.currentTraceObject(); + if (trace == null) { + return; + } + + try { + final SpanEventRecorder recorder = trace.currentSpanEventRecorder(); + recorder.recordApi(this.descriptor); + recorder.recordServiceType(AgentSdkAsyncConstants.AGENT_SDK_ASYNC); + recorder.recordException(throwable); + } finally { + trace.traceBlockEnd(); + } + } +} diff --git a/plugins/agentsdk-async/src/main/resources/META-INF/services/com.navercorp.pinpoint.bootstrap.plugin.ProfilerPlugin b/plugins/agentsdk-async/src/main/resources/META-INF/services/com.navercorp.pinpoint.bootstrap.plugin.ProfilerPlugin new file mode 100644 index 000000000000..65b88a04d7be --- /dev/null +++ b/plugins/agentsdk-async/src/main/resources/META-INF/services/com.navercorp.pinpoint.bootstrap.plugin.ProfilerPlugin @@ -0,0 +1 @@ +com.navercorp.pinpoint.plugin.sdk.AgentSdkAsyncPlugin diff --git a/plugins/agentsdk-async/src/main/resources/META-INF/services/com.navercorp.pinpoint.common.trace.TraceMetadataProvider b/plugins/agentsdk-async/src/main/resources/META-INF/services/com.navercorp.pinpoint.common.trace.TraceMetadataProvider new file mode 100644 index 000000000000..400576013be9 --- /dev/null +++ b/plugins/agentsdk-async/src/main/resources/META-INF/services/com.navercorp.pinpoint.common.trace.TraceMetadataProvider @@ -0,0 +1 @@ +com.navercorp.pinpoint.plugin.sdk.AgentSdkAsyncTraceMetadataProvider \ No newline at end of file diff --git a/plugins/assembly/pom.xml b/plugins/assembly/pom.xml index aab90598439c..db4d722c1ff1 100644 --- a/plugins/assembly/pom.xml +++ b/plugins/assembly/pom.xml @@ -15,6 +15,12 @@ pom + + com.navercorp.pinpoint + pinpoint-agentsdk-async-plugin + ${project.version} + + com.navercorp.pinpoint pinpoint-common-servlet @@ -367,6 +373,12 @@ ${project.version} + + com.navercorp.pinpoint + pinpoint-agentsdk-async-plugin + ${project.version} + + com.navercorp.pinpoint pinpoint-process-plugin diff --git a/plugins/pom.xml b/plugins/pom.xml index 425778d3267f..62fa5c26ec49 100644 --- a/plugins/pom.xml +++ b/plugins/pom.xml @@ -106,6 +106,7 @@ process paho-mqtt rocketmq + agentsdk-async diff --git a/pom.xml b/pom.xml index 8cceb6076227..0353338d0eed 100644 --- a/pom.xml +++ b/pom.xml @@ -81,6 +81,8 @@ annotations agent + agent-sdk + agent-testweb agent-plugins bootstraps diff --git a/profiler/src/main/java/com/navercorp/pinpoint/profiler/instrument/ASMClass.java b/profiler/src/main/java/com/navercorp/pinpoint/profiler/instrument/ASMClass.java index 8d7343b1be3d..7cf9050365a4 100644 --- a/profiler/src/main/java/com/navercorp/pinpoint/profiler/instrument/ASMClass.java +++ b/profiler/src/main/java/com/navercorp/pinpoint/profiler/instrument/ASMClass.java @@ -77,6 +77,9 @@ public ASMClass(EngineComponent engineComponent, final InstrumentContext pluginC this.classNode = Objects.requireNonNull(classNode, "classNode"); } + public void test() { + + } public ClassLoader getClassLoader() { return this.classNode.getClassLoader(); } diff --git a/profiler/src/main/java/com/navercorp/pinpoint/profiler/transformer/PinpointClassFilter.java b/profiler/src/main/java/com/navercorp/pinpoint/profiler/transformer/PinpointClassFilter.java index 75462448ca56..21ce977d07a4 100644 --- a/profiler/src/main/java/com/navercorp/pinpoint/profiler/transformer/PinpointClassFilter.java +++ b/profiler/src/main/java/com/navercorp/pinpoint/profiler/transformer/PinpointClassFilter.java @@ -17,13 +17,26 @@ package com.navercorp.pinpoint.profiler.transformer; import java.security.ProtectionDomain; +import java.util.Objects; /** * @author emeroad */ public class PinpointClassFilter implements ClassFileFilter { + private static final String DEFAULT_PACKAGE = "com/navercorp/pinpoint/"; + private static final String[] DEFAULT_EXCLUDES = {"web/", "sdk/"}; + + private final String basePackage; + private final String[] excludes; + public PinpointClassFilter() { + this(DEFAULT_PACKAGE, DEFAULT_EXCLUDES); + } + + public PinpointClassFilter(String basePackage, String[] excludes) { + this.basePackage = Objects.requireNonNull(basePackage, "basePackage"); + this.excludes = Objects.requireNonNull(excludes, "excludes"); } @Override @@ -33,9 +46,11 @@ public boolean accept(ClassLoader classLoader, String className, Class classB } // Skip pinpoint packages too. - if (className.startsWith("com/navercorp/pinpoint/")) { - if (className.startsWith("com/navercorp/pinpoint/web/")) { - return CONTINUE; + if (className.startsWith(basePackage)) { + for (String exclude : excludes) { + if (className.startsWith(exclude, basePackage.length())) { + return CONTINUE; + } } return SKIP; } diff --git a/profiler/src/test/java/com/navercorp/pinpoint/profiler/transformer/PinpointClassFilterTest.java b/profiler/src/test/java/com/navercorp/pinpoint/profiler/transformer/PinpointClassFilterTest.java index 38fdc292e6b9..bf806465ded5 100644 --- a/profiler/src/test/java/com/navercorp/pinpoint/profiler/transformer/PinpointClassFilterTest.java +++ b/profiler/src/test/java/com/navercorp/pinpoint/profiler/transformer/PinpointClassFilterTest.java @@ -16,30 +16,38 @@ package com.navercorp.pinpoint.profiler.transformer; -import com.navercorp.pinpoint.common.util.ClassLoaderUtils; import org.junit.Assert; import org.junit.Test; -import java.net.URL; -import java.net.URLClassLoader; - /** * @author Woonduk Kang(emeroad) */ public class PinpointClassFilterTest { + @Test - public void testDoFilter_Package() throws Exception { + public void doFilter() throws Exception { + + ClassFileFilter filter = new PinpointClassFilter(); + + Assert.assertEquals(ClassFileFilter.CONTINUE, filter.accept(null, "java/test", null, null, null)); + Assert.assertEquals(ClassFileFilter.CONTINUE, filter.accept(null, "javax/test", null, null, null)); + Assert.assertEquals(ClassFileFilter.CONTINUE, filter.accept(null, "test", null, null, null)); + } + @Test + public void doFilter_Package() throws Exception { - final URLClassLoader agentClassLoader = new URLClassLoader(new URL[0]); ClassFileFilter filter = new PinpointClassFilter(); - final ClassLoader defaultClassLoader = ClassLoaderUtils.getDefaultClassLoader(); + Assert.assertEquals(ClassFileFilter.SKIP, filter.accept(null, "com/navercorp/pinpoint/", null, null, null)); + } + + @Test + public void doFilter_Package_exclude() throws Exception { + + ClassFileFilter filter = new PinpointClassFilter(); - Assert.assertSame("pinpoint", filter.accept(defaultClassLoader, "com/navercorp/pinpoint/", null, null, null), ClassFileFilter.SKIP); + Assert.assertEquals(ClassFileFilter.CONTINUE, filter.accept(null, "com/navercorp/pinpoint/web/", null, null, null)); - Assert.assertSame(filter.accept(defaultClassLoader, "java/test", null, null, null), ClassFileFilter.CONTINUE); - Assert.assertSame(filter.accept(defaultClassLoader, "javax/test", null, null, null), ClassFileFilter.CONTINUE); - Assert.assertSame(filter.accept(defaultClassLoader, "test", null, null, null), ClassFileFilter.CONTINUE); } -} +} \ No newline at end of file