352684 Implemented logging of thread CPU utilisation in Thread Monitor

This commit is contained in:
Michael Gorovoy 2011-08-02 20:59:12 -04:00
parent 0a6c7ad846
commit a6d003a0bc
4 changed files with 222 additions and 17 deletions

View File

@ -9,3 +9,5 @@ To run ThreadMonitor on a Jetty installation that doesn't include jetty-monitor
java -jar start.jar etc/jetty-monitor.xml
If running Jetty on Java VM version 1.5, the -Dcom.sun.management.jmxremote option should be added to the command lines above in order to enable the JMX agent.
In order to log CPU utilization for threads that are above specified threshold, you need to follow instructions inside jetty-monitor.xml configuration file.

View File

@ -9,6 +9,13 @@
<Set name="scanInterval">2000</Set>
<Set name="busyThreshold">90</Set>
<Set name="stackDepth">3</Set>
<!-- To enable logging CPU utilization for threads above specified threshold, -->
<!-- uncomment the following lines, changing dump interval (in milliseconds) -->
<!-- and dump threshold (in percent) as desired. -->
<!--
<Set name="dumpInterval">10000</Arg>
<Set name="dumpThreshold">1</Arg>
-->
</New>
</Arg>
</Call>

View File

@ -19,6 +19,8 @@ import java.lang.management.ThreadInfo;
import java.lang.management.ThreadMXBean;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
@ -32,7 +34,9 @@ import org.eclipse.jetty.util.log.Logger;
public class ThreadMonitor extends AbstractLifeCycle implements Runnable
{
private int _scanInterval;
private int _dumpInterval;
private int _busyThreshold;
private int _dumpThreshold;
private int _stackDepth;
private ThreadMXBean _threadBean;
@ -70,7 +74,7 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
_busyThreshold = threshold;
_stackDepth = depth;
_logger = Log.getLogger(getClass().getName());
_logger = Log.getLogger(ThreadMonitor.class.getName());
_extInfo = new HashMap<Long, ExtThreadInfo>();
init();
@ -89,7 +93,19 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
}
/* ------------------------------------------------------------ */
public int getBusyThreshold()
public int getDumpInterval()
{
return _dumpInterval;
}
/* ------------------------------------------------------------ */
public void setDumpInterval(int ms)
{
_dumpInterval = ms;
}
/* ------------------------------------------------------------ */
public int getBusyThreshold()
{
return _busyThreshold;
}
@ -100,6 +116,18 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
_busyThreshold = percent;
}
/* ------------------------------------------------------------ */
public int getDumpThreshold()
{
return _dumpThreshold;
}
/* ------------------------------------------------------------ */
public void setDumpThreshold(int percent)
{
_dumpThreshold = percent;
}
/* ------------------------------------------------------------ */
public int getStackDepth()
{
@ -111,6 +139,13 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
{
_stackDepth = stackDepth;
}
/* ------------------------------------------------------------ */
public void enableDumpAll(int ms, int percent)
{
setDumpInterval(ms);
setDumpThreshold(percent);
}
/* ------------------------------------------------------------ */
/**
@ -249,6 +284,34 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
}
}
}
protected void dumpAll()
{
if (_extInfo.size() > 0)
{
List<ExtThreadInfo> sorted = new ArrayList<ExtThreadInfo>(_extInfo.values());
Collections.sort(sorted, new Comparator<ExtThreadInfo>() {
/* ------------------------------------------------------------ */
public int compare(ExtThreadInfo eti1, ExtThreadInfo eti2)
{
return (int)Math.signum(eti2.getCpuUtilization()-eti1.getCpuUtilization());
}
});
for (ExtThreadInfo info : sorted)
{
ThreadInfo threadInfo = info.getThreadInfo();
if (info.getCpuUtilization() > 1.0f)
{
_logger.info(String.format("Thread %s[%d] is using %.2f of CPU", threadInfo.getThreadName(), threadInfo.getThreadId(), info.getCpuUtilization()));
}
info.setDumpCpuTime(info.getLastCpuTime());
info.setDumpSampleTime(info.getLastSampleTime());
}
}
}
/* ------------------------------------------------------------ */
/**
@ -256,8 +319,9 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
*/
public void run()
{
long currTime;
long lastTime = 0;
long currTime = System.currentTimeMillis();
long lastTime = currTime;
long lastDumpTime = currTime;
while (!_done)
{
currTime = System.currentTimeMillis();
@ -285,6 +349,13 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
{
dump(threadInfo);
}
if (_dumpInterval > 0 && lastTime > lastDumpTime + _dumpInterval)
{
lastDumpTime = lastTime;
dumpAll();
}
}
}
@ -321,7 +392,9 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
long elapsedCpuTime = currCpuTime - currExtInfo.getLastCpuTime();
long elapsedNanoTime = currNanoTime - currExtInfo.getLastSampleTime();
if (((elapsedCpuTime * 100.0) / elapsedNanoTime) > _busyThreshold)
float cpuUtilization = Math.min((elapsedCpuTime * 100.0f) / elapsedNanoTime, 100.0f);
currExtInfo.setCpuUtilization(cpuUtilization);
if (cpuUtilization > _busyThreshold)
{
ThreadInfo currInfo = getThreadInfo(currId, Integer.MAX_VALUE);
if (currInfo != null)
@ -338,7 +411,12 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
}
else
{
currExtInfo = new ExtThreadInfo(currId);
currExtInfo = new ExtThreadInfo(getThreadInfo(currId, 0));
currExtInfo.setFirstCpuTime(currCpuTime);
currExtInfo.setFirstSampleTime(currNanoTime);
currExtInfo.setDumpCpuTime(currCpuTime);
currExtInfo.setDumpSampleTime(currNanoTime);
_extInfo.put(Long.valueOf(currId), currExtInfo);
}
@ -418,25 +496,72 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
/* ------------------------------------------------------------ */
private class ExtThreadInfo
{
private long _threadId;
private ThreadInfo _threadInfo;
private long _firstCpuTime;
private long _firstSampleTime;
private long _lastCpuTime;
private long _lastSampleTime;
private long _dumpCpuTime;
private long _dumpSampleTime;
private float _cpuUtilization;
private StackTraceElement[] _stackTrace;
/* ------------------------------------------------------------ */
public ExtThreadInfo(long threadId)
public ExtThreadInfo(ThreadInfo threadInfo)
{
_threadId = threadId;
_threadInfo = threadInfo;
}
/* ------------------------------------------------------------ */
/**
* @return thread id associated with the instance
*/
public long getThreadId()
public ThreadInfo getThreadInfo()
{
return _threadId;
return _threadInfo;
}
/* ------------------------------------------------------------ */
/**
* @return the first CPU time of the thread
*/
public long getFirstCpuTime()
{
return _firstCpuTime;
}
/* ------------------------------------------------------------ */
/**
* Set the first CPU time.
*
* @param ns new last CPU time
*/
public void setFirstCpuTime(long ns)
{
_firstCpuTime = ns;
}
/* ------------------------------------------------------------ */
/**
* @return the time of first sample
*/
public long getFirstSampleTime()
{
return _firstSampleTime;
}
/* ------------------------------------------------------------ */
/**
* Sets the first sample time.
*
* @param ns the time of first sample
*/
public void setFirstSampleTime(long ns)
{
_firstSampleTime = ns;
}
/* ------------------------------------------------------------ */
@ -452,11 +577,11 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
/**
* Set the last CPU time.
*
* @param lastCpuTime new last CPU time
* @param ns new last CPU time
*/
public void setLastCpuTime(long lastCpuTime)
public void setLastCpuTime(long ns)
{
this._lastCpuTime = lastCpuTime;
_lastCpuTime = ns;
}
/* ------------------------------------------------------------ */
@ -472,11 +597,73 @@ public class ThreadMonitor extends AbstractLifeCycle implements Runnable
/**
* Sets the last sample time.
*
* @param lastSampleTime the time of last sample
* @param ns the time of last sample
*/
public void setLastSampleTime(long lastSampleTime)
public void setLastSampleTime(long ns)
{
_lastSampleTime = lastSampleTime;
_lastSampleTime = ns;
}
/* ------------------------------------------------------------ */
/**
* @return the dump CPU time of the thread
*/
public long getDumpCpuTime()
{
return _dumpCpuTime;
}
/* ------------------------------------------------------------ */
/**
* Set the dump CPU time.
*
* @param ns new dump CPU time
*/
public void setDumpCpuTime(long ns)
{
_dumpCpuTime = ns;
}
/* ------------------------------------------------------------ */
/**
* @return the time of dump sample
*/
public long getDumpSampleTime()
{
return _dumpSampleTime;
}
/* ------------------------------------------------------------ */
/**
* Sets the dump sample time.
*
* @param ns the time of dump sample
*/
public void setDumpSampleTime(long ns)
{
_dumpSampleTime = ns;
}
/* ------------------------------------------------------------ */
/**
* Gets the CPU utilization.
*
* @return the CPU utilization percentage
*/
public float getCpuUtilization()
{
return _cpuUtilization;
}
/* ------------------------------------------------------------ */
/**
* Sets the CPU utilization.
*
* @param percentage the new CPU utilization percentage
*/
public void setCpuUtilization(float percentage)
{
_cpuUtilization = percentage;
}
/* ------------------------------------------------------------ */

View File

@ -30,6 +30,7 @@ public class ThreadMonitorTest
{
public final static int DURATION=9000;
private AtomicInteger count=new AtomicInteger(0);
private AtomicInteger countDump=new AtomicInteger(0);
@Test
public void monitorTest() throws Exception
@ -43,7 +44,14 @@ public class ThreadMonitorTest
count.incrementAndGet();
super.dump(threads);
}
@Override
protected void dumpAll()
{
countDump.incrementAndGet();
super.dumpAll();
}
};
monitor.enableDumpAll(2000,1);
monitor.start();
Spinner spinner = new Spinner();
@ -56,6 +64,7 @@ public class ThreadMonitorTest
monitor.stop();
assertTrue(count.get() >= 2);
assertTrue(countDump.get() >= 1);
}