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
pull/17356/head
Andy Wilkinson 5 years ago
parent a66c4d3096
commit c5cae28261

@ -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 @@
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;
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.
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
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(
MockMvcRestDocumentation
.document("threaddump/json", preprocessResponse(limit("threads")),
responseFields(fieldWithPath("threads").description("JVM's threads."), responseFields(fieldWithPath("threads").description("JVM's threads."),
fieldWithPath("threads.[].blockedCount") fieldWithPath("threads.[].blockedCount").description(
.description("Total number of times that the thread has been " + "blocked."), "Total number of times that the thread has been " + "blocked."),
fieldWithPath("threads.[].blockedTime") fieldWithPath("threads.[].blockedTime").description(
.description("Time in milliseconds that the thread has spent " "Time in milliseconds that the thread has spent "
+ "blocked. -1 if thread contention " + "monitoring is disabled."), + "blocked. -1 if thread contention "
+ "monitoring is disabled."),
fieldWithPath("threads.[].daemon") fieldWithPath("threads.[].daemon")
.description("Whether the thread is a daemon " .description("Whether the thread is a daemon "
+ "thread. Only available on Java 9 or " + "later.") + "thread. Only available on Java 9 or " + "later.")
.optional().type(JsonFieldType.BOOLEAN), .optional().type(JsonFieldType.BOOLEAN),
fieldWithPath("threads.[].inNative") fieldWithPath("threads.[].inNative")
.description("Whether the thread is executing native code."), .description("Whether the thread is executing native code."),
fieldWithPath("threads.[].lockName").description( fieldWithPath("threads.[].lockName")
"Description of the object on which the " + "thread is blocked, if any.") .description("Description of the object on which the "
+ "thread is blocked, if any.")
.optional().type(JsonFieldType.STRING), .optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].lockInfo") fieldWithPath("threads.[].lockInfo")
.description("Object for which the thread is blocked " + "waiting.").optional() .description(
.type(JsonFieldType.OBJECT), "Object for which the thread is blocked " + "waiting.")
.optional().type(JsonFieldType.OBJECT),
fieldWithPath("threads.[].lockInfo.className") fieldWithPath("threads.[].lockInfo.className")
.description("Fully qualified class name of the lock" + " object.").optional() .description(
.type(JsonFieldType.STRING), "Fully qualified class name of the lock" + " object.")
.optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].lockInfo.identityHashCode") fieldWithPath("threads.[].lockInfo.identityHashCode")
.description("Identity hash code of the lock object.").optional() .description("Identity hash code of the lock object.")
.type(JsonFieldType.NUMBER), .optional().type(JsonFieldType.NUMBER),
fieldWithPath("threads.[].lockedMonitors") fieldWithPath("threads.[].lockedMonitors")
.description("Monitors locked by this thread, if any"), .description("Monitors locked by this thread, if any"),
fieldWithPath("threads.[].lockedMonitors.[].className") fieldWithPath("threads.[].lockedMonitors.[].className")
.description("Class name of the lock object.").optional() .description("Class name of the lock object.").optional()
.type(JsonFieldType.STRING), .type(JsonFieldType.STRING),
fieldWithPath("threads.[].lockedMonitors.[].identityHashCode") fieldWithPath("threads.[].lockedMonitors.[].identityHashCode")
.description("Identity hash code of the lock " + "object.").optional() .description("Identity hash code of the lock " + "object.")
.type(JsonFieldType.NUMBER), .optional().type(JsonFieldType.NUMBER),
fieldWithPath("threads.[].lockedMonitors.[].lockedStackDepth") fieldWithPath("threads.[].lockedMonitors.[].lockedStackDepth")
.description("Stack depth where the monitor " + "was locked.").optional() .description("Stack depth where the monitor " + "was locked.")
.type(JsonFieldType.NUMBER), .optional().type(JsonFieldType.NUMBER),
subsectionWithPath("threads.[].lockedMonitors.[].lockedStackFrame") subsectionWithPath("threads.[].lockedMonitors.[].lockedStackFrame")
.description("Stack frame that locked the " + "monitor.").optional() .description("Stack frame that locked the " + "monitor.")
.type(JsonFieldType.OBJECT), .optional().type(JsonFieldType.OBJECT),
fieldWithPath("threads.[].lockedSynchronizers") fieldWithPath("threads.[].lockedSynchronizers")
.description("Synchronizers locked by this thread."), .description("Synchronizers locked by this thread."),
fieldWithPath("threads.[].lockedSynchronizers.[].className").description( fieldWithPath("threads.[].lockedSynchronizers.[].className")
"Class name of the locked " + "synchronizer.").optional() .description("Class name of the locked " + "synchronizer.")
.type(JsonFieldType.STRING), .optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].lockedSynchronizers.[].identityHashCode").description( fieldWithPath("threads.[].lockedSynchronizers.[].identityHashCode")
"Identity hash code of the locked " + "synchronizer.").optional() .description(
.type(JsonFieldType.NUMBER), "Identity hash code of the locked " + "synchronizer.")
fieldWithPath("threads.[].lockOwnerId").description( .optional().type(JsonFieldType.NUMBER),
"ID of the thread that owns the object on which " fieldWithPath("threads.[].lockOwnerId")
+ "the thread is blocked. `-1` if the " + "thread is not blocked."), .description("ID of the thread that owns the object on which "
+ "the thread is blocked. `-1` if the "
+ "thread is not blocked."),
fieldWithPath("threads.[].lockOwnerName") fieldWithPath("threads.[].lockOwnerName")
.description("Name of the thread that owns the " .description("Name of the thread that owns the "
+ "object on which the thread is " + "blocked, if any.") + "object on which the thread is " + "blocked, if any.")
.optional().type(JsonFieldType.STRING), .optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].priority") fieldWithPath("threads.[].priority")
.description("Priority of the thread. Only " + "available on Java 9 or later.") .description("Priority of the thread. Only "
+ "available on Java 9 or later.")
.optional().type(JsonFieldType.NUMBER), .optional().type(JsonFieldType.NUMBER),
fieldWithPath("threads.[].stackTrace").description("Stack trace of the thread."), fieldWithPath("threads.[].stackTrace")
fieldWithPath("threads.[].stackTrace.[].classLoaderName").description( .description("Stack trace of the thread."),
"Name of the class loader of the " + "class that contains the execution " fieldWithPath("threads.[].stackTrace.[].classLoaderName")
.description("Name of the class loader of the "
+ "class that contains the execution "
+ "point identified by this entry, if " + "point identified by this entry, if "
+ "any. Only available on Java 9 or " + "later.") + "any. Only available on Java 9 or " + "later.")
.optional().type(JsonFieldType.STRING), .optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].stackTrace.[].className").description( fieldWithPath("threads.[].stackTrace.[].className")
"Name of the class that contains the " + "execution point identified " .description("Name of the class that contains the "
+ "by this entry."), + "execution point identified " + "by this entry."),
fieldWithPath("threads.[].stackTrace.[].fileName") fieldWithPath("threads.[].stackTrace.[].fileName")
.description("Name of the source file that " + "contains the execution point " .description("Name of the source file that "
+ "contains the execution point "
+ "identified by this entry, if any.") + "identified by this entry, if any.")
.optional().type(JsonFieldType.STRING), .optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].stackTrace.[].lineNumber") fieldWithPath("threads.[].stackTrace.[].lineNumber")
.description("Line number of the execution " .description("Line number of the execution "
+ "point identified by this entry. " + "Negative if unknown."), + "point identified by this entry. "
fieldWithPath("threads.[].stackTrace.[].methodName").description("Name of the method."), + "Negative if unknown."),
fieldWithPath("threads.[].stackTrace.[].methodName")
.description("Name of the method."),
fieldWithPath("threads.[].stackTrace.[].moduleName") fieldWithPath("threads.[].stackTrace.[].moduleName")
.description("Name of the module that contains " .description("Name of the module that contains "
+ "the execution point identified by " + "the execution point identified by "
+ "this entry, if any. Only available " + "on Java 9 or later.") + "this entry, if any. Only available "
+ "on Java 9 or later.")
.optional().type(JsonFieldType.STRING), .optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].stackTrace.[].moduleVersion") fieldWithPath("threads.[].stackTrace.[].moduleVersion")
.description("Version of the module that " + "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. "
+ "Only available on Java 9 or later.") + "Only available on Java 9 or later.")
.optional().type(JsonFieldType.STRING), .optional().type(JsonFieldType.STRING),
fieldWithPath("threads.[].stackTrace.[].nativeMethod") fieldWithPath("threads.[].stackTrace.[].nativeMethod").description(
.description("Whether the execution point is a native " + "method."), "Whether the execution point is a native " + "method."),
fieldWithPath("threads.[].suspended").description("Whether the thread is suspended."), fieldWithPath("threads.[].suspended")
.description("Whether the thread is suspended."),
fieldWithPath("threads.[].threadId").description("ID of the thread."), fieldWithPath("threads.[].threadId").description("ID of the thread."),
fieldWithPath("threads.[].threadName").description("Name of the thread."), fieldWithPath("threads.[].threadName")
fieldWithPath("threads.[].threadState").description( .description("Name of the thread."),
"State of the thread (" + describeEnumValues(Thread.State.class) + ")."), fieldWithPath("threads.[].threadState")
fieldWithPath("threads.[].waitedCount").description( .description("State of the thread ("
"Total number of times that the thread has waited" + " for notification."), + describeEnumValues(Thread.State.class) + ")."),
fieldWithPath("threads.[].waitedCount")
.description("Total number of times that the thread has waited"
+ " for notification."),
fieldWithPath("threads.[].waitedTime") fieldWithPath("threads.[].waitedTime")
.description("Time in milliseconds that the thread has spent " .description("Time in milliseconds that the thread has spent "
+ "waiting. -1 if thread contention " + "monitoring is disabled")))); + "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 {

@ -0,0 +1,127 @@
/*
* 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;
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;
@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 {
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 @@
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 {
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);
}
}
} }

@ -0,0 +1,59 @@
/*
* 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();
}
}
}
Loading…
Cancel
Save