Commit c5cae282 authored by Andy Wilkinson's avatar Andy Wilkinson

Add support for plain text thread dumps to the threaddump endpoint

When a request that accepts text/plain is received, the threaddump
endpoint will now return a thread dump in plain text. The format of
this text is modelled after the output produced by JVisualVM when
connecting to a remote process over JMX. Note that this output does
not include all of the information in, for example, JStack's output
as it is not available via Java 8's ThreadInfo API.

Rather than the custom formatting logic, using ThreadInfo's toString()
method was considered but its output is documented as being undefined
and implementation specific. The implementation used while developing
this feature produced output that did not match that of JStack or
JVisualVM and truncated stack traces quite considerably.

At the time of writing the format produced by the endpoint could be
consumed by both Thread Dump Analyzer [1] and https://fastthread.io.

Closes gh-2339

[1] https://github.com/irockel/tda
parent a66c4d30
...@@ -5,25 +5,39 @@ The `threaddump` endpoint provides a thread dump from the application's JVM. ...@@ -5,25 +5,39 @@ The `threaddump` endpoint provides a thread dump from the application's JVM.
[[threaddump-retrieving]] [[threaddump-retrieving-json]]
== Retrieving the Thread Dump == Retrieving the Thread Dump as JSON
To retrieve the thread dump, make a `GET` request to `/actuator/threaddump`, as shown To retrieve the thread dump as JSON, make a `GET` request to `/actuator/threaddump` with
in the following curl-based example: an appropriate `Accept` header, as shown in the following curl-based example:
include::{snippets}threaddump/curl-request.adoc[] include::{snippets}threaddump/json/curl-request.adoc[]
The resulting response is similar to the following: The resulting response is similar to the following:
include::{snippets}threaddump/http-response.adoc[] include::{snippets}threaddump/json/http-response.adoc[]
[[threaddump-retrieving-response-structure]] [[threaddump-retrieving-json-response-structure]]
=== Response Structure === Response Structure
The response contains details of the JVM's threads. The following table describes the The response contains details of the JVM's threads. The following table describes the
structure of the response: structure of the response:
[cols="3,1,2"] [cols="3,1,2"]
include::{snippets}threaddump/response-fields.adoc[] include::{snippets}threaddump/json/response-fields.adoc[]
[[threaddump-retrieving-text]]
== Retrieving the Thread Dump as Text
To retrieve the thread dump as text, make a `GET` request to `/actuator/threaddump` that
accepts `text/plain`, as shown in the following curl-based example:
include::{snippets}threaddump/text/curl-request.adoc[]
The resulting response is similar to the following:
include::{snippets}threaddump/text/http-response.adoc[]
...@@ -16,6 +16,7 @@ ...@@ -16,6 +16,7 @@
package org.springframework.boot.actuate.autoconfigure.endpoint.web.documentation; package org.springframework.boot.actuate.autoconfigure.endpoint.web.documentation;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.CountDownLatch; import java.util.concurrent.CountDownLatch;
import java.util.concurrent.locks.ReentrantLock; import java.util.concurrent.locks.ReentrantLock;
...@@ -25,7 +26,9 @@ import org.springframework.boot.actuate.management.ThreadDumpEndpoint; ...@@ -25,7 +26,9 @@ import org.springframework.boot.actuate.management.ThreadDumpEndpoint;
import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import; import org.springframework.context.annotation.Import;
import org.springframework.http.MediaType;
import org.springframework.restdocs.mockmvc.MockMvcRestDocumentation; import org.springframework.restdocs.mockmvc.MockMvcRestDocumentation;
import org.springframework.restdocs.operation.preprocess.ContentModifyingOperationPreprocessor;
import org.springframework.restdocs.payload.JsonFieldType; import org.springframework.restdocs.payload.JsonFieldType;
import static org.springframework.restdocs.operation.preprocess.Preprocessors.preprocessResponse; import static org.springframework.restdocs.operation.preprocess.Preprocessors.preprocessResponse;
...@@ -43,7 +46,7 @@ import static org.springframework.test.web.servlet.result.MockMvcResultMatchers. ...@@ -43,7 +46,7 @@ import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.
class ThreadDumpEndpointDocumentationTests extends MockMvcEndpointDocumentationTests { class ThreadDumpEndpointDocumentationTests extends MockMvcEndpointDocumentationTests {
@Test @Test
void threadDump() throws Exception { void jsonThreadDump() throws Exception {
ReentrantLock lock = new ReentrantLock(); ReentrantLock lock = new ReentrantLock();
CountDownLatch latch = new CountDownLatch(1); CountDownLatch latch = new CountDownLatch(1);
new Thread(() -> { new Thread(() -> {
...@@ -60,106 +63,137 @@ class ThreadDumpEndpointDocumentationTests extends MockMvcEndpointDocumentationT ...@@ -60,106 +63,137 @@ class ThreadDumpEndpointDocumentationTests extends MockMvcEndpointDocumentationT
Thread.currentThread().interrupt(); Thread.currentThread().interrupt();
} }
}).start(); }).start();
this.mockMvc.perform(get("/actuator/threaddump")).andExpect(status().isOk()) this.mockMvc.perform(get("/actuator/threaddump").accept(MediaType.APPLICATION_JSON)).andExpect(
.andDo(MockMvcRestDocumentation.document("threaddump", preprocessResponse(limit("threads")), status().isOk()).andDo(
responseFields(fieldWithPath("threads").description("JVM's threads."), MockMvcRestDocumentation
fieldWithPath("threads.[].blockedCount") .document("threaddump/json", preprocessResponse(limit("threads")),
.description("Total number of times that the thread has been " + "blocked."), responseFields(fieldWithPath("threads").description("JVM's threads."),
fieldWithPath("threads.[].blockedTime") fieldWithPath("threads.[].blockedCount").description(
.description("Time in milliseconds that the thread has spent " "Total number of times that the thread has been " + "blocked."),
+ "blocked. -1 if thread contention " + "monitoring is disabled."), fieldWithPath("threads.[].blockedTime").description(
fieldWithPath("threads.[].daemon") "Time in milliseconds that the thread has spent "
.description("Whether the thread is a daemon " + "blocked. -1 if thread contention "
+ "thread. Only available on Java 9 or " + "later.") + "monitoring is disabled."),
.optional().type(JsonFieldType.BOOLEAN), fieldWithPath("threads.[].daemon")
fieldWithPath("threads.[].inNative") .description("Whether the thread is a daemon "
.description("Whether the thread is executing native code."), + "thread. Only available on Java 9 or " + "later.")
fieldWithPath("threads.[].lockName").description( .optional().type(JsonFieldType.BOOLEAN),
"Description of the object on which the " + "thread is blocked, if any.") fieldWithPath("threads.[].inNative")
.optional().type(JsonFieldType.STRING), .description("Whether the thread is executing native code."),
fieldWithPath("threads.[].lockInfo") fieldWithPath("threads.[].lockName")
.description("Object for which the thread is blocked " + "waiting.").optional() .description("Description of the object on which the "
.type(JsonFieldType.OBJECT), + "thread is blocked, if any.")
fieldWithPath("threads.[].lockInfo.className") .optional().type(JsonFieldType.STRING),
.description("Fully qualified class name of the lock" + " object.").optional() fieldWithPath("threads.[].lockInfo")
.type(JsonFieldType.STRING), .description(
fieldWithPath("threads.[].lockInfo.identityHashCode") "Object for which the thread is blocked " + "waiting.")
.description("Identity hash code of the lock object.").optional() .optional().type(JsonFieldType.OBJECT),
.type(JsonFieldType.NUMBER), fieldWithPath("threads.[].lockInfo.className")
fieldWithPath("threads.[].lockedMonitors") .description(
.description("Monitors locked by this thread, if any"), "Fully qualified class name of the lock" + " object.")
fieldWithPath("threads.[].lockedMonitors.[].className") .optional().type(JsonFieldType.STRING),
.description("Class name of the lock object.").optional() fieldWithPath("threads.[].lockInfo.identityHashCode")
.type(JsonFieldType.STRING), .description("Identity hash code of the lock object.")
fieldWithPath("threads.[].lockedMonitors.[].identityHashCode") .optional().type(JsonFieldType.NUMBER),
.description("Identity hash code of the lock " + "object.").optional() fieldWithPath("threads.[].lockedMonitors")
.type(JsonFieldType.NUMBER), .description("Monitors locked by this thread, if any"),
fieldWithPath("threads.[].lockedMonitors.[].lockedStackDepth") fieldWithPath("threads.[].lockedMonitors.[].className")
.description("Stack depth where the monitor " + "was locked.").optional() .description("Class name of the lock object.").optional()
.type(JsonFieldType.NUMBER), .type(JsonFieldType.STRING),
subsectionWithPath("threads.[].lockedMonitors.[].lockedStackFrame") fieldWithPath("threads.[].lockedMonitors.[].identityHashCode")
.description("Stack frame that locked the " + "monitor.").optional() .description("Identity hash code of the lock " + "object.")
.type(JsonFieldType.OBJECT), .optional().type(JsonFieldType.NUMBER),
fieldWithPath("threads.[].lockedSynchronizers") fieldWithPath("threads.[].lockedMonitors.[].lockedStackDepth")
.description("Synchronizers locked by this thread."), .description("Stack depth where the monitor " + "was locked.")
fieldWithPath("threads.[].lockedSynchronizers.[].className").description( .optional().type(JsonFieldType.NUMBER),
"Class name of the locked " + "synchronizer.").optional() subsectionWithPath("threads.[].lockedMonitors.[].lockedStackFrame")
.type(JsonFieldType.STRING), .description("Stack frame that locked the " + "monitor.")
fieldWithPath("threads.[].lockedSynchronizers.[].identityHashCode").description( .optional().type(JsonFieldType.OBJECT),
"Identity hash code of the locked " + "synchronizer.").optional() fieldWithPath("threads.[].lockedSynchronizers")
.type(JsonFieldType.NUMBER), .description("Synchronizers locked by this thread."),
fieldWithPath("threads.[].lockOwnerId").description( fieldWithPath("threads.[].lockedSynchronizers.[].className")
"ID of the thread that owns the object on which " .description("Class name of the locked " + "synchronizer.")
+ "the thread is blocked. `-1` if the " + "thread is not blocked."), .optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].lockOwnerName") fieldWithPath("threads.[].lockedSynchronizers.[].identityHashCode")
.description("Name of the thread that owns the " .description(
+ "object on which the thread is " + "blocked, if any.") "Identity hash code of the locked " + "synchronizer.")
.optional().type(JsonFieldType.STRING), .optional().type(JsonFieldType.NUMBER),
fieldWithPath("threads.[].priority") fieldWithPath("threads.[].lockOwnerId")
.description("Priority of the thread. Only " + "available on Java 9 or later.") .description("ID of the thread that owns the object on which "
.optional().type(JsonFieldType.NUMBER), + "the thread is blocked. `-1` if the "
fieldWithPath("threads.[].stackTrace").description("Stack trace of the thread."), + "thread is not blocked."),
fieldWithPath("threads.[].stackTrace.[].classLoaderName").description( fieldWithPath("threads.[].lockOwnerName")
"Name of the class loader of the " + "class that contains the execution " .description("Name of the thread that owns the "
+ "point identified by this entry, if " + "object on which the thread is " + "blocked, if any.")
+ "any. Only available on Java 9 or " + "later.") .optional().type(JsonFieldType.STRING),
.optional().type(JsonFieldType.STRING), fieldWithPath("threads.[].priority")
fieldWithPath("threads.[].stackTrace.[].className").description( .description("Priority of the thread. Only "
"Name of the class that contains the " + "execution point identified " + "available on Java 9 or later.")
+ "by this entry."), .optional().type(JsonFieldType.NUMBER),
fieldWithPath("threads.[].stackTrace.[].fileName") fieldWithPath("threads.[].stackTrace")
.description("Name of the source file that " + "contains the execution point " .description("Stack trace of the thread."),
+ "identified by this entry, if any.") fieldWithPath("threads.[].stackTrace.[].classLoaderName")
.optional().type(JsonFieldType.STRING), .description("Name of the class loader of the "
fieldWithPath("threads.[].stackTrace.[].lineNumber") + "class that contains the execution "
.description("Line number of the execution " + "point identified by this entry, if "
+ "point identified by this entry. " + "Negative if unknown."), + "any. Only available on Java 9 or " + "later.")
fieldWithPath("threads.[].stackTrace.[].methodName").description("Name of the method."), .optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].stackTrace.[].moduleName") fieldWithPath("threads.[].stackTrace.[].className")
.description("Name of the module that contains " .description("Name of the class that contains the "
+ "the execution point identified by " + "execution point identified " + "by this entry."),
+ "this entry, if any. Only available " + "on Java 9 or later.") fieldWithPath("threads.[].stackTrace.[].fileName")
.optional().type(JsonFieldType.STRING), .description("Name of the source file that "
fieldWithPath("threads.[].stackTrace.[].moduleVersion") + "contains the execution point "
.description("Version of the module that " + "contains the execution point " + "identified by this entry, if any.")
+ "identified by this entry, if any. " .optional().type(JsonFieldType.STRING),
+ "Only available on Java 9 or later.") fieldWithPath("threads.[].stackTrace.[].lineNumber")
.optional().type(JsonFieldType.STRING), .description("Line number of the execution "
fieldWithPath("threads.[].stackTrace.[].nativeMethod") + "point identified by this entry. "
.description("Whether the execution point is a native " + "method."), + "Negative if unknown."),
fieldWithPath("threads.[].suspended").description("Whether the thread is suspended."), fieldWithPath("threads.[].stackTrace.[].methodName")
fieldWithPath("threads.[].threadId").description("ID of the thread."), .description("Name of the method."),
fieldWithPath("threads.[].threadName").description("Name of the thread."), fieldWithPath("threads.[].stackTrace.[].moduleName")
fieldWithPath("threads.[].threadState").description( .description("Name of the module that contains "
"State of the thread (" + describeEnumValues(Thread.State.class) + ")."), + "the execution point identified by "
fieldWithPath("threads.[].waitedCount").description( + "this entry, if any. Only available "
"Total number of times that the thread has waited" + " for notification."), + "on Java 9 or later.")
fieldWithPath("threads.[].waitedTime") .optional().type(JsonFieldType.STRING),
.description("Time in milliseconds that the thread has spent " fieldWithPath("threads.[].stackTrace.[].moduleVersion")
+ "waiting. -1 if thread contention " + "monitoring is disabled")))); .description("Version of the module that "
+ "contains the execution point "
+ "identified by this entry, if any. "
+ "Only available on Java 9 or later.")
.optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].stackTrace.[].nativeMethod").description(
"Whether the execution point is a native " + "method."),
fieldWithPath("threads.[].suspended")
.description("Whether the thread is suspended."),
fieldWithPath("threads.[].threadId").description("ID of the thread."),
fieldWithPath("threads.[].threadName")
.description("Name of the thread."),
fieldWithPath("threads.[].threadState")
.description("State of the thread ("
+ describeEnumValues(Thread.State.class) + ")."),
fieldWithPath("threads.[].waitedCount")
.description("Total number of times that the thread has waited"
+ " for notification."),
fieldWithPath("threads.[].waitedTime")
.description("Time in milliseconds that the thread has spent "
+ "waiting. -1 if thread contention "
+ "monitoring is disabled"))));
latch.countDown(); latch.countDown();
} }
@Test
void textThreadDump() throws Exception {
this.mockMvc.perform(get("/actuator/threaddump").accept(MediaType.TEXT_PLAIN)).andExpect(status().isOk())
.andDo(MockMvcRestDocumentation.document("threaddump/text",
preprocessResponse(new ContentModifyingOperationPreprocessor((bytes, mediaType) -> {
String content = new String(bytes, StandardCharsets.UTF_8);
return content.substring(0, content.indexOf("\"main\" - Thread")).getBytes();
}))));
}
@Configuration(proxyBeanMethods = false) @Configuration(proxyBeanMethods = false)
@Import(BaseDocumentationConfiguration.class) @Import(BaseDocumentationConfiguration.class)
static class TestConfiguration { static class TestConfiguration {
......
/*
* Copyright 2012-2019 the original author or 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
*
* https://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 org.springframework.boot.actuate.management;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.lang.management.LockInfo;
import java.lang.management.ManagementFactory;
import java.lang.management.MonitorInfo;
import java.lang.management.RuntimeMXBean;
import java.lang.management.ThreadInfo;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;
/**
* Formats a thread dump as plain text.
*
* @author Andy Wilkinson
*/
class PlainTextThreadDumpFormatter {
String format(ThreadInfo[] threads) {
StringWriter dump = new StringWriter();
PrintWriter writer = new PrintWriter(dump);
writePreamble(writer);
for (ThreadInfo info : threads) {
writeThread(writer, info);
}
return dump.toString();
}
private void writePreamble(PrintWriter writer) {
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
writer.println(dateFormat.format(new Date()));
RuntimeMXBean runtime = ManagementFactory.getRuntimeMXBean();
writer.printf("Full thread dump %s (%s %s):%n", runtime.getVmName(), runtime.getVmVersion(),
System.getProperty("java.vm.info"));
writer.println();
}
private void writeThread(PrintWriter writer, ThreadInfo info) {
writer.printf("\"%s\" - Thread t@%d%n", info.getThreadName(), info.getThreadId());
writer.printf(" %s: %s%n", Thread.State.class.getCanonicalName(), info.getThreadState());
writeStackTrace(writer, info, info.getLockedMonitors());
writer.println();
writeLockedOwnableSynchronizers(writer, info);
writer.println();
}
private void writeStackTrace(PrintWriter writer, ThreadInfo info, MonitorInfo[] lockedMonitors) {
int depth = 0;
for (StackTraceElement element : info.getStackTrace()) {
writeStackTraceElement(writer, element, info, lockedMonitorsForDepth(lockedMonitors, depth), depth == 0);
depth++;
}
}
private List<MonitorInfo> lockedMonitorsForDepth(MonitorInfo[] lockedMonitors, int depth) {
return Stream.of(lockedMonitors).filter((lockedMonitor) -> lockedMonitor.getLockedStackDepth() == depth)
.collect(Collectors.toList());
}
private void writeStackTraceElement(PrintWriter writer, StackTraceElement element, ThreadInfo info,
List<MonitorInfo> lockedMonitors, boolean firstElement) {
writer.printf("\tat %s%n", element.toString());
LockInfo lockInfo = info.getLockInfo();
if (firstElement && lockInfo != null) {
if (element.getClassName().equals(Object.class.getName()) && element.getMethodName().equals("wait")) {
if (lockInfo != null) {
writer.printf("\t- waiting on %s%n", format(lockInfo));
}
}
else {
String lockOwner = info.getLockOwnerName();
if (lockOwner != null) {
writer.printf("\t- waiting to lock %s owned by \"%s\" t@%d%n", format(lockInfo), lockOwner,
info.getLockOwnerId());
}
else {
writer.printf("\t- parking to wait for %s%n", format(lockInfo));
}
}
}
writeMonitors(writer, lockedMonitors);
}
private String format(LockInfo lockInfo) {
return String.format("<%x> (a %s)", lockInfo.getIdentityHashCode(), lockInfo.getClassName());
}
private void writeMonitors(PrintWriter writer, List<MonitorInfo> lockedMonitorsAtCurrentDepth) {
for (MonitorInfo lockedMonitor : lockedMonitorsAtCurrentDepth) {
writer.printf("\t- locked %s%n", format(lockedMonitor));
}
}
private void writeLockedOwnableSynchronizers(PrintWriter writer, ThreadInfo info) {
writer.println(" Locked ownable synchronizers:");
LockInfo[] lockedSynchronizers = info.getLockedSynchronizers();
if (lockedSynchronizers == null || lockedSynchronizers.length == 0) {
writer.println("\t- None");
}
else {
for (LockInfo lockedSynchronizer : lockedSynchronizers) {
writer.printf("\t- Locked %s%n", format(lockedSynchronizer));
}
}
}
}
...@@ -20,6 +20,7 @@ import java.lang.management.ManagementFactory; ...@@ -20,6 +20,7 @@ import java.lang.management.ManagementFactory;
import java.lang.management.ThreadInfo; import java.lang.management.ThreadInfo;
import java.util.Arrays; import java.util.Arrays;
import java.util.List; import java.util.List;
import java.util.function.Function;
import org.springframework.boot.actuate.endpoint.annotation.Endpoint; import org.springframework.boot.actuate.endpoint.annotation.Endpoint;
import org.springframework.boot.actuate.endpoint.annotation.ReadOperation; import org.springframework.boot.actuate.endpoint.annotation.ReadOperation;
...@@ -34,9 +35,20 @@ import org.springframework.boot.actuate.endpoint.annotation.ReadOperation; ...@@ -34,9 +35,20 @@ import org.springframework.boot.actuate.endpoint.annotation.ReadOperation;
@Endpoint(id = "threaddump") @Endpoint(id = "threaddump")
public class ThreadDumpEndpoint { public class ThreadDumpEndpoint {
private final PlainTextThreadDumpFormatter plainTextFormatter = new PlainTextThreadDumpFormatter();
@ReadOperation @ReadOperation
public ThreadDumpDescriptor threadDump() { public ThreadDumpDescriptor threadDump() {
return new ThreadDumpDescriptor(Arrays.asList(ManagementFactory.getThreadMXBean().dumpAllThreads(true, true))); return getFormattedThreadDump(ThreadDumpDescriptor::new);
}
@ReadOperation(produces = "text/plain;charset=UTF-8")
public String textThreadDump() {
return getFormattedThreadDump(this.plainTextFormatter::format);
}
private <T> T getFormattedThreadDump(Function<ThreadInfo[], T> formatter) {
return formatter.apply(ManagementFactory.getThreadMXBean().dumpAllThreads(true, true));
} }
/** /**
...@@ -46,8 +58,8 @@ public class ThreadDumpEndpoint { ...@@ -46,8 +58,8 @@ public class ThreadDumpEndpoint {
private final List<ThreadInfo> threads; private final List<ThreadInfo> threads;
private ThreadDumpDescriptor(List<ThreadInfo> threads) { private ThreadDumpDescriptor(ThreadInfo[] threads) {
this.threads = threads; this.threads = Arrays.asList(threads);
} }
public List<ThreadInfo> getThreads() { public List<ThreadInfo> getThreads() {
......
...@@ -16,6 +16,12 @@ ...@@ -16,6 +16,12 @@
package org.springframework.boot.actuate.management; package org.springframework.boot.actuate.management;
import java.lang.Thread.State;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThat;
...@@ -33,4 +39,87 @@ class ThreadDumpEndpointTests { ...@@ -33,4 +39,87 @@ class ThreadDumpEndpointTests {
assertThat(new ThreadDumpEndpoint().threadDump().getThreads().size()).isGreaterThan(0); assertThat(new ThreadDumpEndpoint().threadDump().getThreads().size()).isGreaterThan(0);
} }
@Test
void dumpThreadsAsText() throws InterruptedException {
Object contendedMonitor = new Object();
Object monitor = new Object();
CountDownLatch latch = new CountDownLatch(1);
Thread awaitCountDownLatchThread = new Thread(() -> {
try {
latch.await();
}
catch (InterruptedException ex) {
Thread.currentThread().interrupt();
}
}, "Awaiting CountDownLatch");
awaitCountDownLatchThread.start();
Thread contendedMonitorThread = new Thread(() -> {
synchronized (contendedMonitor) {
// Intentionally empty
}
}, "Waiting for monitor");
Thread waitOnMonitorThread = new Thread(() -> {
synchronized (monitor) {
try {
monitor.wait();
}
catch (InterruptedException ex) {
Thread.currentThread().interrupt();
}
}
}, "Waiting on monitor");
waitOnMonitorThread.start();
ReadWriteLock readWriteLock = new ReentrantReadWriteLock();
Lock writeLock = readWriteLock.writeLock();
new Thread(() -> {
writeLock.lock();
try {
latch.await();
}
catch (InterruptedException ex) {
Thread.currentThread().interrupt();
}
finally {
writeLock.unlock();
}
}, "Holding write lock").start();
while (writeLock.tryLock()) {
writeLock.unlock();
}
awaitState(waitOnMonitorThread, State.WAITING);
awaitState(awaitCountDownLatchThread, State.WAITING);
String threadDump;
synchronized (contendedMonitor) {
contendedMonitorThread.start();
awaitState(contendedMonitorThread, State.BLOCKED);
threadDump = new ThreadDumpEndpoint().textThreadDump();
}
latch.countDown();
synchronized (monitor) {
monitor.notifyAll();
}
System.out.println(threadDump);
assertThat(threadDump)
.containsPattern(String.format("\t- parking to wait for <[0-9a-z]+> \\(a %s\\$Sync\\)",
CountDownLatch.class.getName().replace(".", "\\.")))
.contains(String.format("\t- locked <%s> (a java.lang.Object)", hexIdentityHashCode(contendedMonitor)))
.contains(String.format("\t- waiting to lock <%s> (a java.lang.Object) owned by \"%s\" t@%d",
hexIdentityHashCode(contendedMonitor), Thread.currentThread().getName(),
Thread.currentThread().getId()))
.contains(String.format("\t- waiting on <%s> (a java.lang.Object)", hexIdentityHashCode(monitor)))
.containsPattern(
String.format("Locked ownable synchronizers:%n\t- Locked <[0-9a-z]+> \\(a %s\\$NonfairSync\\)",
ReentrantReadWriteLock.class.getName().replace(".", "\\.")));
}
private String hexIdentityHashCode(Object object) {
return Integer.toHexString(System.identityHashCode(object));
}
private void awaitState(Thread thread, State state) throws InterruptedException {
while (thread.getState() != state) {
Thread.sleep(50);
}
}
} }
/*
* Copyright 2012-2019 the original author or 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
*
* https://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 org.springframework.boot.actuate.management;
import org.springframework.boot.actuate.endpoint.web.test.WebEndpointTest;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.MediaType;
import org.springframework.test.web.reactive.server.WebTestClient;
import static org.assertj.core.api.Assertions.assertThat;
/**
* Integration tests for {@link ThreadDumpEndpoint} exposed by Jersey, Spring MVC, and
* WebFlux.
*
* @author Andy Wilkinson
*/
class ThreadDumpEndpointWebIntegrationTests {
@WebEndpointTest
void getRequestWithJsonAcceptHeaderShouldProduceJsonThreadDumpResponse(WebTestClient client) throws Exception {
client.get().uri("/actuator/threaddump").accept(MediaType.APPLICATION_JSON).exchange().expectStatus().isOk()
.expectHeader().contentType(MediaType.APPLICATION_JSON);
}
@WebEndpointTest
void getRequestWithTextPlainAcceptHeaderShouldProduceTextPlainResponse(WebTestClient client) throws Exception {
String response = client.get().uri("/actuator/threaddump").accept(MediaType.TEXT_PLAIN).exchange()
.expectStatus().isOk().expectHeader().contentType("text/plain;charset=UTF-8").expectBody(String.class)
.returnResult().getResponseBody();
assertThat(response).contains("Full thread dump");
}
@Configuration(proxyBeanMethods = false)
public static class TestConfiguration {
@Bean
public ThreadDumpEndpoint endpoint() {
return new ThreadDumpEndpoint();
}
}
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment