Skip to content

Commit

Permalink
Adds ability to exclude deadlocked threads from exported metrics
Browse files Browse the repository at this point in the history
Finding deadlocked threads might be an expensive operation if there are
a large number of threads in the jvm since it requires a safepoint.

Signed-off-by: Andreas Skoog <[email protected]>
  • Loading branch information
askoog committed Jan 13, 2021
1 parent 683466f commit 0765982
Show file tree
Hide file tree
Showing 2 changed files with 73 additions and 12 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
*/
public class ThreadExports extends Collector {
private final ThreadMXBean threadBean;
private boolean includeDeadlockedThreads = true;

public ThreadExports() {
this(ManagementFactory.getThreadMXBean());
Expand All @@ -41,6 +42,14 @@ public ThreadExports(ThreadMXBean threadBean) {
this.threadBean = threadBean;
}

public final boolean isIncludeDeadlockedThreads() {
return includeDeadlockedThreads;
}

public final void setIncludeDeadlockedThreads(boolean includeDeadlockedThreads) {
this.includeDeadlockedThreads = includeDeadlockedThreads;
}

void addThreadMetrics(List<MetricFamilySamples> sampleFamilies) {
sampleFamilies.add(
new GaugeMetricFamily(
Expand All @@ -66,18 +75,20 @@ void addThreadMetrics(List<MetricFamilySamples> sampleFamilies) {
"Started thread count of a JVM",
threadBean.getTotalStartedThreadCount()));

sampleFamilies.add(
new GaugeMetricFamily(
"jvm_threads_deadlocked",
"Cycles of JVM-threads that are in deadlock waiting to acquire object monitors or ownable synchronizers",
nullSafeArrayLength(threadBean.findDeadlockedThreads())));

sampleFamilies.add(
new GaugeMetricFamily(
"jvm_threads_deadlocked_monitor",
"Cycles of JVM-threads that are in deadlock waiting to acquire object monitors",
nullSafeArrayLength(threadBean.findMonitorDeadlockedThreads())));

if (includeDeadlockedThreads) {
sampleFamilies.add(
new GaugeMetricFamily(
"jvm_threads_deadlocked",
"Cycles of JVM-threads that are in deadlock waiting to acquire object monitors or ownable synchronizers",
nullSafeArrayLength(threadBean.findDeadlockedThreads())));

sampleFamilies.add(
new GaugeMetricFamily(
"jvm_threads_deadlocked_monitor",
"Cycles of JVM-threads that are in deadlock waiting to acquire object monitors",
nullSafeArrayLength(threadBean.findMonitorDeadlockedThreads())));
}

GaugeMetricFamily threadStateFamily = new GaugeMetricFamily(
"jvm_threads_state",
"Current count of threads by state",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
import java.util.Arrays;

import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNull;
import static org.mockito.Mockito.when;

public class ThreadExportsTest {
Expand Down Expand Up @@ -97,4 +98,53 @@ public void testThreadPools() {
"jvm_threads_state", STATE_LABEL, STATE_TERMINATED_LABEL),
.0000001);
}

@Test
public void testThreadPoolsNoDeadlocked() {
collectorUnderTest.setIncludeDeadlockedThreads(false);

assertNull(registry.getSampleValue(
"jvm_threads_deadlocked", EMPTY_LABEL, EMPTY_LABEL));
assertNull(registry.getSampleValue(
"jvm_threads_deadlocked_monitor", EMPTY_LABEL, EMPTY_LABEL));

assertEquals(
300L,
registry.getSampleValue(
"jvm_threads_current", EMPTY_LABEL, EMPTY_LABEL),
.0000001);
assertEquals(
200L,
registry.getSampleValue(
"jvm_threads_daemon", EMPTY_LABEL, EMPTY_LABEL),
.0000001);
assertEquals(
301L,
registry.getSampleValue(
"jvm_threads_peak", EMPTY_LABEL, EMPTY_LABEL),
.0000001);
assertEquals(
503L,
registry.getSampleValue(
"jvm_threads_started_total", EMPTY_LABEL, EMPTY_LABEL),
.0000001);

assertEquals(
1L,
registry.getSampleValue(
"jvm_threads_state", STATE_LABEL, STATE_BLOCKED_LABEL),
.0000001);

assertEquals(
2L,
registry.getSampleValue(
"jvm_threads_state", STATE_LABEL, STATE_RUNNABLE_LABEL),
.0000001);

assertEquals(
0L,
registry.getSampleValue(
"jvm_threads_state", STATE_LABEL, STATE_TERMINATED_LABEL),
.0000001);
}
}

0 comments on commit 0765982

Please sign in to comment.