Issue #2775 LowResourceMonitor extendable (#2812)

* make LowResourceMonitor extendable #2775

Signed-off-by: olivier lamy <oliver.lamy@gmail.com>
This commit is contained in:
Olivier Lamy 2018-08-23 21:42:37 +10:00 committed by GitHub
parent 460f3fcb9b
commit 44e57f2170
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 678 additions and 239 deletions

View File

@ -8,7 +8,7 @@
<Configure id="Server" class="org.eclipse.jetty.server.Server"> <Configure id="Server" class="org.eclipse.jetty.server.Server">
<Call name="addBean"> <Call name="addBean">
<Arg> <Arg>
<New class="org.eclipse.jetty.server.LowResourceMonitor"> <New id="lowResourceMonitor" class="org.eclipse.jetty.server.LowResourceMonitor">
<Arg name="server"><Ref refid='Server'/></Arg> <Arg name="server"><Ref refid='Server'/></Arg>
<Set name="period"><Property name="jetty.lowresources.period" deprecated="lowresources.period" default="1000"/></Set> <Set name="period"><Property name="jetty.lowresources.period" deprecated="lowresources.period" default="1000"/></Set>
<Set name="lowResourcesIdleTimeout"><Property name="jetty.lowresources.idleTimeout" deprecated="lowresources.lowResourcesIdleTimeout" default="1000"/></Set> <Set name="lowResourcesIdleTimeout"><Property name="jetty.lowresources.idleTimeout" deprecated="lowresources.lowResourcesIdleTimeout" default="1000"/></Set>

View File

@ -18,6 +18,17 @@
package org.eclipse.jetty.server; package org.eclipse.jetty.server;
import org.eclipse.jetty.io.EndPoint;
import org.eclipse.jetty.util.annotation.ManagedAttribute;
import org.eclipse.jetty.util.annotation.ManagedObject;
import org.eclipse.jetty.util.annotation.Name;
import org.eclipse.jetty.util.component.ContainerLifeCycle;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.util.thread.ScheduledExecutorScheduler;
import org.eclipse.jetty.util.thread.Scheduler;
import org.eclipse.jetty.util.thread.ThreadPool;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
@ -27,66 +38,46 @@ import java.util.concurrent.Executor;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean; import java.util.concurrent.atomic.AtomicBoolean;
import org.eclipse.jetty.io.EndPoint;
import org.eclipse.jetty.util.annotation.ManagedAttribute;
import org.eclipse.jetty.util.annotation.ManagedObject;
import org.eclipse.jetty.util.annotation.Name;
import org.eclipse.jetty.util.component.AbstractLifeCycle;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.util.thread.ScheduledExecutorScheduler;
import org.eclipse.jetty.util.thread.Scheduler;
import org.eclipse.jetty.util.thread.ThreadPool;
/** /**
* A monitor for low resources *
* <p> * A monitor for low resources, low resources can be detected by:
* An instance of this class will monitor all the connectors of a server (or a set of connectors
* configured with {@link #setMonitoredConnectors(Collection)}) for a low resources state.
* <p>
* Low resources can be detected by:
* <ul> * <ul>
* <li>{@link ThreadPool#isLowOnThreads()} if {@link Connector#getExecutor()} is * <li>{@link ThreadPool#isLowOnThreads()} if {@link Connector#getExecutor()} is
* an instance of {@link ThreadPool} and {@link #setMonitorThreads(boolean)} is true.</li> * an instance of {@link ThreadPool} and {@link #setMonitorThreads(boolean)} is true.</li>
* <li>If {@link #setMaxMemory(long)} is non zero then low resources is detected if the JVMs * <li>If {@link #setMaxMemory(long)} is non zero then low resources is detected if the JVMs
* {@link Runtime} instance has {@link Runtime#totalMemory()} minus {@link Runtime#freeMemory()} * {@link Runtime} instance has {@link Runtime#totalMemory()} minus {@link Runtime#freeMemory()}
* greater than {@link #getMaxMemory()}</li> * greater than {@link #getMaxMemory()}</li>
* <li>If {@link #setMaxConnections(int)} is non zero then low resources is dected if the total number * <li>If {@link #setMaxConnections(int)} is non zero then low resources is detected if the total number
* of connections exceeds {@link #getMaxConnections()}. This feature is deprecated and replaced by * of connections exceeds {@link #getMaxConnections()}. This feature is deprecated and replaced by
* {@link ConnectionLimit}</li> * {@link ConnectionLimit}</li>
* </ul> * </ul>
* <p> *
* Once low resources state is detected, the cause is logged and all existing connections returned
* by {@link Connector#getConnectedEndPoints()} have {@link EndPoint#setIdleTimeout(long)} set
* to {@link #getLowResourcesIdleTimeout()}. New connections are not affected, however if the low
* resources state persists for more than {@link #getMaxLowResourcesTime()}, then the
* {@link #getLowResourcesIdleTimeout()} to all connections again. Once the low resources state is
* cleared, the idle timeout is reset to the connector default given by {@link Connector#getIdleTimeout()}.
* <p>
* If {@link #setAcceptingInLowResources(boolean)} is set to false (Default is true), then no new connections
* are accepted when in low resources state.
*/ */
@ManagedObject ("Monitor for low resource conditions and activate a low resource mode if detected") @ManagedObject ("Monitor for low resource conditions and activate a low resource mode if detected")
public class LowResourceMonitor extends AbstractLifeCycle public class LowResourceMonitor extends ContainerLifeCycle
{ {
private static final Logger LOG = Log.getLogger(LowResourceMonitor.class); private static final Logger LOG = Log.getLogger(LowResourceMonitor.class);
private final Server _server;
protected final Server _server;
private Scheduler _scheduler; private Scheduler _scheduler;
private Connector[] _monitoredConnectors; private Connector[] _monitoredConnectors;
private Set<AbstractConnector> _acceptingConnectors = new HashSet<>(); private Set<AbstractConnector> _acceptingConnectors = new HashSet<>();
private int _period=1000; private int _period=1000;
private int _maxConnections;
private long _maxMemory;
private int _lowResourcesIdleTimeout=1000; private int _lowResourcesIdleTimeout=1000;
private int _maxLowResourcesTime=0; private int _maxLowResourcesTime=0;
private boolean _monitorThreads=true;
private final AtomicBoolean _low = new AtomicBoolean(); private final AtomicBoolean _low = new AtomicBoolean();
private String _cause;
private String _reasons; private String _reasons;
private long _lowStarted; private long _lowStarted;
private boolean _acceptingInLowResources = true; private boolean _acceptingInLowResources = true;
private Set<LowResourceCheck> _lowResourceChecks = new HashSet<>();
private final Runnable _monitor = new Runnable() private final Runnable _monitor = new Runnable()
{ {
@Override @Override
@ -105,24 +96,111 @@ public class LowResourceMonitor extends AbstractLifeCycle
_server = server; _server = server;
} }
@ManagedAttribute("True if low available threads status is monitored")
public boolean getMonitorThreads()
{
return !getBeans(ConnectorsThreadPoolLowResourceCheck.class).isEmpty();
}
/**
* @param monitorThreads If true, check connectors executors to see if they are
* {@link ThreadPool} instances that are low on threads.
*/
public void setMonitorThreads(boolean monitorThreads)
{
if(monitorThreads)
// already configured?
if ( !getMonitorThreads() ) addLowResourceCheck( new ConnectorsThreadPoolLowResourceCheck() );
else
getBeans(ConnectorsThreadPoolLowResourceCheck.class).forEach(this::removeBean);
}
/**
* @return The maximum connections allowed for the monitored connectors before low resource handling is activated
* @deprecated Replaced by ConnectionLimit
*/
@ManagedAttribute("The maximum connections allowed for the monitored connectors before low resource handling is activated")
@Deprecated
public int getMaxConnections()
{
for(MaxConnectionsLowResourceCheck lowResourceCheck : getBeans(MaxConnectionsLowResourceCheck.class))
{
if (lowResourceCheck.getMaxConnections()>0)
{
return lowResourceCheck.getMaxConnections();
}
}
return -1;
}
/**
* @param maxConnections The maximum connections before low resources state is triggered
* @deprecated Replaced by ConnectionLimit
*/
@Deprecated
public void setMaxConnections(int maxConnections)
{
if (maxConnections>0)
{
if (getBeans(MaxConnectionsLowResourceCheck.class).isEmpty())
{
addLowResourceCheck(new MaxConnectionsLowResourceCheck(maxConnections));
} else
{
getBeans(MaxConnectionsLowResourceCheck.class).forEach( c -> c.setMaxConnections( maxConnections ) );
}
}
else
{
getBeans(ConnectorsThreadPoolLowResourceCheck.class).forEach(this::removeBean);
}
}
@ManagedAttribute("The reasons the monitored connectors are low on resources")
public String getReasons()
{
return _reasons;
}
protected void setReasons(String reasons)
{
_reasons = reasons;
}
@ManagedAttribute("Are the monitored connectors low on resources?") @ManagedAttribute("Are the monitored connectors low on resources?")
public boolean isLowOnResources() public boolean isLowOnResources()
{ {
return _low.get(); return _low.get();
} }
protected boolean enableLowOnResources(boolean expectedValue, boolean newValue)
{
return _low.compareAndSet(expectedValue, newValue);
}
@ManagedAttribute("The reason(s) the monitored connectors are low on resources") @ManagedAttribute("The reason(s) the monitored connectors are low on resources")
public String getLowResourcesReasons() public String getLowResourcesReasons()
{ {
return _reasons; return _reasons;
} }
protected void setLowResourcesReasons(String reasons)
{
_reasons = reasons;
}
@ManagedAttribute("Get the timestamp in ms since epoch that low resources state started") @ManagedAttribute("Get the timestamp in ms since epoch that low resources state started")
public long getLowResourcesStarted() public long getLowResourcesStarted()
{ {
return _lowStarted; return _lowStarted;
} }
public void setLowResourcesStarted(long lowStarted)
{
_lowStarted = lowStarted;
}
@ManagedAttribute("The monitored connectors. If null then all server connectors are monitored") @ManagedAttribute("The monitored connectors. If null then all server connectors are monitored")
public Collection<Connector> getMonitoredConnectors() public Collection<Connector> getMonitoredConnectors()
{ {
@ -142,6 +220,13 @@ public class LowResourceMonitor extends AbstractLifeCycle
_monitoredConnectors = monitoredConnectors.toArray(new Connector[monitoredConnectors.size()]); _monitoredConnectors = monitoredConnectors.toArray(new Connector[monitoredConnectors.size()]);
} }
protected Connector[] getMonitoredOrServerConnectors()
{
if (_monitoredConnectors!=null && _monitoredConnectors.length>0)
return _monitoredConnectors;
return _server.getConnectors();
}
@ManagedAttribute("If false, new connections are not accepted while in low resources") @ManagedAttribute("If false, new connections are not accepted while in low resources")
public boolean isAcceptingInLowResources() public boolean isAcceptingInLowResources()
{ {
@ -167,58 +252,6 @@ public class LowResourceMonitor extends AbstractLifeCycle
_period = periodMS; _period = periodMS;
} }
@ManagedAttribute("True if low available threads status is monitored")
public boolean getMonitorThreads()
{
return _monitorThreads;
}
/**
* @param monitorThreads If true, check connectors executors to see if they are
* {@link ThreadPool} instances that are low on threads.
*/
public void setMonitorThreads(boolean monitorThreads)
{
_monitorThreads = monitorThreads;
}
/**
* @return The maximum connections allowed for the monitored connectors before low resource handling is activated
* @deprecated Replaced by ConnectionLimit
*/
@ManagedAttribute("The maximum connections allowed for the monitored connectors before low resource handling is activated")
@Deprecated
public int getMaxConnections()
{
return _maxConnections;
}
/**
* @param maxConnections The maximum connections before low resources state is triggered
* @deprecated Replaced by ConnectionLimit
*/
@Deprecated
public void setMaxConnections(int maxConnections)
{
if (maxConnections>0)
LOG.warn("LowResourceMonitor.setMaxConnections is deprecated. Use ConnectionLimit.");
_maxConnections = maxConnections;
}
@ManagedAttribute("The maximum memory (in bytes) that can be used before low resources is triggered. Memory used is calculated as (totalMemory-freeMemory).")
public long getMaxMemory()
{
return _maxMemory;
}
/**
* @param maxMemoryBytes The maximum memory in bytes in use before low resources is triggered.
*/
public void setMaxMemory(long maxMemoryBytes)
{
_maxMemory = maxMemoryBytes;
}
@ManagedAttribute("The idletimeout in ms to apply to all existing connections when low resources is detected") @ManagedAttribute("The idletimeout in ms to apply to all existing connections when low resources is detected")
public int getLowResourcesIdleTimeout() public int getLowResourcesIdleTimeout()
{ {
@ -247,6 +280,99 @@ public class LowResourceMonitor extends AbstractLifeCycle
_maxLowResourcesTime = maxLowResourcesTimeMS; _maxLowResourcesTime = maxLowResourcesTimeMS;
} }
@ManagedAttribute("The maximum memory (in bytes) that can be used before low resources is triggered. Memory used is calculated as (totalMemory-freeMemory).")
public long getMaxMemory()
{
Collection<MemoryLowResourceCheck> beans = getBeans(MemoryLowResourceCheck.class);
if(beans.isEmpty())
{
return 0;
}
return beans.stream().findFirst().get().getMaxMemory();
}
/**
* @param maxMemoryBytes The maximum memory in bytes in use before low resources is triggered.
*/
public void setMaxMemory(long maxMemoryBytes)
{
if(maxMemoryBytes<=0)
{
return;
}
Collection<MemoryLowResourceCheck> beans = getBeans(MemoryLowResourceCheck.class);
if(beans.isEmpty())
addLowResourceCheck( new MemoryLowResourceCheck( maxMemoryBytes ) );
else
beans.forEach( lowResourceCheck -> lowResourceCheck.setMaxMemory( maxMemoryBytes ) );
}
public Set<LowResourceCheck> getLowResourceChecks()
{
return _lowResourceChecks;
}
public void setLowResourceChecks( Set<LowResourceCheck> lowResourceChecks )
{
updateBeans(_lowResourceChecks.toArray(),lowResourceChecks.toArray());
this._lowResourceChecks = lowResourceChecks;
}
public void addLowResourceCheck( LowResourceCheck lowResourceCheck )
{
addBean(lowResourceCheck);
this._lowResourceChecks.add(lowResourceCheck);
}
protected void monitor()
{
String reasons=null;
for(LowResourceCheck lowResourceCheck : _lowResourceChecks)
{
if(lowResourceCheck.isLowOnResources())
{
reasons = lowResourceCheck.toString();
break;
}
}
if (reasons!=null)
{
// Log the reasons if there is any change in the cause
if (!reasons.equals(getReasons()))
{
LOG.warn("Low Resources: {}",reasons);
setReasons(reasons);
}
// Enter low resources state?
if (enableLowOnResources(false,true))
{
setLowResourcesReasons(reasons);
setLowResourcesStarted(System.currentTimeMillis());
setLowResources();
}
// Too long in low resources state?
if ( getMaxLowResourcesTime()>0 && (System.currentTimeMillis()-getLowResourcesStarted())>getMaxLowResourcesTime())
setLowResources();
}
else
{
if (enableLowOnResources(true,false))
{
LOG.info("Low Resources cleared");
setLowResourcesReasons(null);
setLowResourcesStarted(0);
setReasons(null);
clearLowResources();
}
}
}
@Override @Override
protected void doStart() throws Exception protected void doStart() throws Exception
{ {
@ -257,6 +383,7 @@ public class LowResourceMonitor extends AbstractLifeCycle
_scheduler=new LRMScheduler(); _scheduler=new LRMScheduler();
_scheduler.start(); _scheduler.start();
} }
super.doStart(); super.doStart();
_scheduler.schedule(_monitor,_period,TimeUnit.MILLISECONDS); _scheduler.schedule(_monitor,_period,TimeUnit.MILLISECONDS);
@ -270,89 +397,6 @@ public class LowResourceMonitor extends AbstractLifeCycle
super.doStop(); super.doStop();
} }
protected Connector[] getMonitoredOrServerConnectors()
{
if (_monitoredConnectors!=null && _monitoredConnectors.length>0)
return _monitoredConnectors;
return _server.getConnectors();
}
protected void monitor()
{
String reasons=null;
String cause="";
int connections=0;
ThreadPool serverThreads = _server.getThreadPool();
if (_monitorThreads && serverThreads.isLowOnThreads())
{
reasons=low(reasons,"Server low on threads: "+serverThreads);
cause+="S";
}
for(Connector connector : getMonitoredOrServerConnectors())
{
connections+=connector.getConnectedEndPoints().size();
Executor executor = connector.getExecutor();
if (executor instanceof ThreadPool && executor!=serverThreads)
{
ThreadPool connectorThreads=(ThreadPool)executor;
if (_monitorThreads && connectorThreads.isLowOnThreads())
{
reasons=low(reasons,"Connector low on threads: "+connectorThreads);
cause+="T";
}
}
}
if (_maxConnections>0 && connections>_maxConnections)
{
reasons=low(reasons,"Max Connections exceeded: "+connections+">"+_maxConnections);
cause+="C";
}
long memory=Runtime.getRuntime().totalMemory()-Runtime.getRuntime().freeMemory();
if (_maxMemory>0 && memory>_maxMemory)
{
reasons=low(reasons,"Max memory exceeded: "+memory+">"+_maxMemory);
cause+="M";
}
if (reasons!=null)
{
// Log the reasons if there is any change in the cause
if (!cause.equals(_cause))
{
LOG.warn("Low Resources: {}",reasons);
_cause=cause;
}
// Enter low resources state?
if (_low.compareAndSet(false,true))
{
_reasons=reasons;
_lowStarted=System.currentTimeMillis();
setLowResources();
}
// Too long in low resources state?
if (_maxLowResourcesTime>0 && (System.currentTimeMillis()-_lowStarted)>_maxLowResourcesTime)
setLowResources();
}
else
{
if (_low.compareAndSet(true,false))
{
LOG.info("Low Resources cleared");
_reasons=null;
_lowStarted=0;
_cause=null;
clearLowResources();
}
}
}
protected void setLowResources() protected void setLowResources()
{ {
for(Connector connector : getMonitoredOrServerConnectors()) for(Connector connector : getMonitoredOrServerConnectors())
@ -387,15 +431,219 @@ public class LowResourceMonitor extends AbstractLifeCycle
_acceptingConnectors.clear(); _acceptingConnectors.clear();
} }
private String low(String reasons, String newReason) protected String low(String reasons, String newReason)
{ {
if (reasons==null) if (reasons==null)
return newReason; return newReason;
return reasons+", "+newReason; return reasons+", "+newReason;
} }
private static class LRMScheduler extends ScheduledExecutorScheduler private static class LRMScheduler extends ScheduledExecutorScheduler
{ {
} }
interface LowResourceCheck
{
boolean isLowOnResources();
String getReason();
}
//------------------------------------------------------
// default implementations for backward compat
//------------------------------------------------------
public class MainThreadPoolLowResourceCheck implements LowResourceCheck
{
private String reason;
public MainThreadPoolLowResourceCheck()
{
// no op
}
@Override
public boolean isLowOnResources()
{
ThreadPool serverThreads = _server.getThreadPool();
if (serverThreads.isLowOnThreads())
{
reason="Server low on threads: "+serverThreads;
return true;
}
return false;
}
@Override
public String getReason()
{
return reason;
}
@Override
public String toString()
{
return "Check if the server ThreadPool is lowOnThreads";
}
}
public class ConnectorsThreadPoolLowResourceCheck implements LowResourceCheck
{
private String reason;
public ConnectorsThreadPoolLowResourceCheck()
{
// no op
}
@Override
public boolean isLowOnResources()
{
ThreadPool serverThreads = _server.getThreadPool();
if(serverThreads.isLowOnThreads())
{
reason ="Server low on threads: "+serverThreads.getThreads()+", idleThreads:"+serverThreads.getIdleThreads();
return true;
}
for(Connector connector : getMonitoredConnectors())
{
Executor executor = connector.getExecutor();
if (executor instanceof ThreadPool && executor!=serverThreads)
{
ThreadPool connectorThreads=(ThreadPool)executor;
if (connectorThreads.isLowOnThreads())
{
reason ="Connector low on threads: "+connectorThreads;
return true;
}
}
}
return false;
}
@Override
public String getReason()
{
return reason;
}
@Override
public String toString()
{
return "Check if the ThreadPool from monitored connectors are lowOnThreads and if all connections number is higher than the allowed maxConnection";
}
}
@ManagedObject("Check max allowed connections on connectors")
public class MaxConnectionsLowResourceCheck implements LowResourceCheck
{
private String reason;
private int maxConnections;
public MaxConnectionsLowResourceCheck(int maxConnections)
{
this.maxConnections = maxConnections;
}
/**
* @return The maximum connections allowed for the monitored connectors before low resource handling is activated
* @deprecated Replaced by ConnectionLimit
*/
@ManagedAttribute("The maximum connections allowed for the monitored connectors before low resource handling is activated")
@Deprecated
public int getMaxConnections()
{
return maxConnections;
}
/**
* @param maxConnections The maximum connections before low resources state is triggered
* @deprecated Replaced by ConnectionLimit
*/
@Deprecated
public void setMaxConnections(int maxConnections)
{
if (maxConnections>0)
LOG.warn("LowResourceMonitor.setMaxConnections is deprecated. Use ConnectionLimit.");
this.maxConnections = maxConnections;
}
@Override
public boolean isLowOnResources()
{
int connections=0;
for(Connector connector : getMonitoredConnectors())
{
connections+=connector.getConnectedEndPoints().size();
}
if (maxConnections>0 && connections>maxConnections)
{
reason ="Max Connections exceeded: "+connections+">"+maxConnections;
return true;
}
return false;
}
@Override
public String getReason()
{
return reason;
}
@Override
public String toString()
{
return "All connections number is higher than the allowed maxConnection";
}
}
public class MemoryLowResourceCheck implements LowResourceCheck
{
private String reason;
private long maxMemory;
public MemoryLowResourceCheck(long maxMemory)
{
this.maxMemory = maxMemory;
}
@Override
public boolean isLowOnResources()
{
long memory=Runtime.getRuntime().totalMemory()-Runtime.getRuntime().freeMemory();
if (maxMemory>0 && memory>maxMemory)
{
reason = "Max memory exceeded: "+memory+">"+maxMemory;
return true;
}
return false;
}
public long getMaxMemory()
{
return maxMemory;
}
/**
* @param maxMemoryBytes The maximum memory in bytes in use before low resources is triggered.
*/
public void setMaxMemory( long maxMemoryBytes )
{
this.maxMemory = maxMemoryBytes;
}
@Override
public String getReason()
{
return reason;
}
@Override
public String toString()
{
return "Check if used memory is higher than the allowed max memory";
}
}
} }

View File

@ -0,0 +1,194 @@
//
// ========================================================================
// Copyright (c) 1995-2018 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.server;
import org.eclipse.jetty.toolchain.test.AdvancedRunner;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.util.thread.TimerScheduler;
import org.hamcrest.Matchers;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import java.io.IOException;
import java.io.InputStream;
import java.net.Socket;
import java.net.SocketTimeoutException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import static org.junit.Assert.*;
@RunWith(AdvancedRunner.class)
public class CustomResourcesMonitorTest
{
Server _server;
ServerConnector _connector;
FileOnDirectoryMonitor _fileOnDirectoryMonitor;
Path _monitoredPath;
LowResourceMonitor _lowResourceMonitor;
@Before
public void before() throws Exception
{
_server = new Server();
_server.addBean(new TimerScheduler());
_connector = new ServerConnector(_server);
_connector.setPort(0);
_connector.setIdleTimeout(35000);
_server.addConnector(_connector);
_server.setHandler(new DumpHandler());
_monitoredPath = Files.createTempDirectory( "jetty_test" );
_fileOnDirectoryMonitor=new FileOnDirectoryMonitor(_monitoredPath);
_lowResourceMonitor = new LowResourceMonitor(_server);
_server.addBean( _lowResourceMonitor );
_lowResourceMonitor.addLowResourceCheck( _fileOnDirectoryMonitor );
_server.start();
}
@After
public void after() throws Exception
{
_server.stop();
}
@Test
public void testFileOnDirectoryMonitor() throws Exception
{
int monitorPeriod = _lowResourceMonitor.getPeriod();
int lowResourcesIdleTimeout = _lowResourceMonitor.getLowResourcesIdleTimeout();
assertThat(lowResourcesIdleTimeout, Matchers.lessThanOrEqualTo(monitorPeriod));
int maxLowResourcesTime = 5 * monitorPeriod;
_lowResourceMonitor.setMaxLowResourcesTime(maxLowResourcesTime);
assertFalse(_fileOnDirectoryMonitor.isLowOnResources());
try(Socket socket0 = new Socket("localhost",_connector.getLocalPort()))
{
Path tmpFile = Files.createTempFile( _monitoredPath, "yup", ".tmp" );
// Write a file
Files.write(tmpFile, "foobar".getBytes());
// Wait a couple of monitor periods so that
// fileOnDirectoryMonitor detects it is in low mode.
Thread.sleep(2 * monitorPeriod);
assertTrue(_fileOnDirectoryMonitor.isLowOnResources());
// We already waited enough for fileOnDirectoryMonitor to close socket0.
assertEquals(-1, socket0.getInputStream().read());
// New connections are not affected by the
// low mode until maxLowResourcesTime elapses.
try(Socket socket1 = new Socket("localhost",_connector.getLocalPort()))
{
// Set a very short read timeout so we can test if the server closed.
socket1.setSoTimeout(1);
InputStream input1 = socket1.getInputStream();
assertTrue(_fileOnDirectoryMonitor.isLowOnResources());
try
{
input1.read();
fail();
}
catch (SocketTimeoutException expected)
{
}
// Wait a couple of lowResources idleTimeouts.
Thread.sleep(2 * lowResourcesIdleTimeout);
// Verify the new socket is still open.
assertTrue(_fileOnDirectoryMonitor.isLowOnResources());
try
{
input1.read();
fail();
}
catch (SocketTimeoutException expected)
{
}
Files.delete( tmpFile );
// Let the maxLowResourcesTime elapse.
Thread.sleep(maxLowResourcesTime);
assertFalse(_fileOnDirectoryMonitor.isLowOnResources());
}
}
}
static class FileOnDirectoryMonitor implements LowResourceMonitor.LowResourceCheck
{
private static final Logger LOG = Log.getLogger( FileOnDirectoryMonitor.class);
private final Path _pathToMonitor;
private String reason;
public FileOnDirectoryMonitor(Path pathToMonitor )
{
_pathToMonitor = pathToMonitor;
}
@Override
public boolean isLowOnResources()
{
try
{
Stream<Path> paths = Files.list( _pathToMonitor );
List<Path> content = paths.collect( Collectors.toList() );
if(!content.isEmpty())
{
reason= "directory not empty so enable low resources";
return true;
}
}
catch ( IOException e )
{
LOG.info( "ignore issue looking at directory content", e );
}
return false;
}
@Override
public String getReason()
{
return reason;
}
@Override
public String toString()
{
return getClass().getName();
}
}
}

View File

@ -19,27 +19,27 @@
package org.eclipse.jetty.server; package org.eclipse.jetty.server;
import java.io.InputStream;
import java.net.Socket;
import java.net.SocketTimeoutException;
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
import java.util.concurrent.CountDownLatch;
import org.eclipse.jetty.toolchain.test.AdvancedRunner; import org.eclipse.jetty.toolchain.test.AdvancedRunner;
import org.eclipse.jetty.util.thread.QueuedThreadPool; import org.eclipse.jetty.util.thread.QueuedThreadPool;
import org.eclipse.jetty.util.thread.TimerScheduler; import org.eclipse.jetty.util.thread.TimerScheduler;
import org.hamcrest.Matchers; import org.hamcrest.Matchers;
import org.junit.After; import org.junit.After;
import org.junit.Assert;
import org.junit.Before; import org.junit.Before;
import org.junit.Ignore; import org.junit.Ignore;
import org.junit.Test; import org.junit.Test;
import org.junit.runner.RunWith; import org.junit.runner.RunWith;
import java.io.InputStream;
import java.net.Socket;
import java.net.SocketTimeoutException;
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
import java.util.Collections;
import java.util.concurrent.CountDownLatch;
import static org.hamcrest.Matchers.equalTo; import static org.hamcrest.Matchers.equalTo;
import static org.hamcrest.Matchers.not; import static org.hamcrest.Matchers.not;
import static org.junit.Assert.assertThat; import static org.junit.Assert.*;
@RunWith(AdvancedRunner.class) @RunWith(AdvancedRunner.class)
public class LowResourcesMonitorTest public class LowResourcesMonitorTest
@ -71,6 +71,7 @@ public class LowResourcesMonitorTest
_lowResourcesMonitor.setLowResourcesIdleTimeout(200); _lowResourcesMonitor.setLowResourcesIdleTimeout(200);
_lowResourcesMonitor.setMaxConnections(20); _lowResourcesMonitor.setMaxConnections(20);
_lowResourcesMonitor.setPeriod(900); _lowResourcesMonitor.setPeriod(900);
_lowResourcesMonitor.setMonitoredConnectors( Collections.singleton( _connector ) );
_server.addBean(_lowResourcesMonitor); _server.addBean(_lowResourcesMonitor);
_server.start(); _server.start();
@ -86,19 +87,17 @@ public class LowResourcesMonitorTest
@Test @Test
public void testLowOnThreads() throws Exception public void testLowOnThreads() throws Exception
{ {
_lowResourcesMonitor.setMonitorThreads(true);
Thread.sleep(1200); Thread.sleep(1200);
_threadPool.setMaxThreads(_threadPool.getThreads()-_threadPool.getIdleThreads()+10); _threadPool.setMaxThreads(_threadPool.getThreads()-_threadPool.getIdleThreads()+10);
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertFalse(_lowResourcesMonitor.isLowOnResources()); assertFalse(_lowResourcesMonitor.getReasons(), _lowResourcesMonitor.isLowOnResources());
final CountDownLatch latch = new CountDownLatch(1); final CountDownLatch latch = new CountDownLatch(1);
for (int i=0;i<100;i++) for (int i=0;i<100;i++)
{ {
_threadPool.execute(new Runnable() _threadPool.execute(() ->
{
@Override
public void run()
{ {
try try
{ {
@ -108,16 +107,15 @@ public class LowResourcesMonitorTest
{ {
e.printStackTrace(); e.printStackTrace();
} }
}
}); });
} }
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertTrue(_lowResourcesMonitor.isLowOnResources()); assertTrue(_lowResourcesMonitor.isLowOnResources());
latch.countDown(); latch.countDown();
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertFalse(_lowResourcesMonitor.isLowOnResources()); assertFalse(_lowResourcesMonitor.getReasons(),_lowResourcesMonitor.isLowOnResources());
} }
@ -125,10 +123,13 @@ public class LowResourcesMonitorTest
public void testNotAccepting() throws Exception public void testNotAccepting() throws Exception
{ {
_lowResourcesMonitor.setAcceptingInLowResources(false); _lowResourcesMonitor.setAcceptingInLowResources(false);
_lowResourcesMonitor.setMonitorThreads( true );
Thread.sleep(1200); Thread.sleep(1200);
_threadPool.setMaxThreads(_threadPool.getThreads()-_threadPool.getIdleThreads()+10); int maxThreads = _threadPool.getThreads()-_threadPool.getIdleThreads()+10;
System.out.println("maxThreads:"+maxThreads);
_threadPool.setMaxThreads(maxThreads);
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertFalse(_lowResourcesMonitor.isLowOnResources()); assertFalse(_lowResourcesMonitor.getReasons(),_lowResourcesMonitor.isLowOnResources());
for (AbstractConnector c : _server.getBeans(AbstractConnector.class)) for (AbstractConnector c : _server.getBeans(AbstractConnector.class))
assertThat(c.isAccepting(),Matchers.is(true)); assertThat(c.isAccepting(),Matchers.is(true));
@ -136,10 +137,7 @@ public class LowResourcesMonitorTest
final CountDownLatch latch = new CountDownLatch(1); final CountDownLatch latch = new CountDownLatch(1);
for (int i=0;i<100;i++) for (int i=0;i<100;i++)
{ {
_threadPool.execute(new Runnable() _threadPool.execute(() ->
{
@Override
public void run()
{ {
try try
{ {
@ -149,18 +147,17 @@ public class LowResourcesMonitorTest
{ {
e.printStackTrace(); e.printStackTrace();
} }
}
}); });
} }
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertTrue(_lowResourcesMonitor.isLowOnResources()); assertTrue(_lowResourcesMonitor.isLowOnResources());
for (AbstractConnector c : _server.getBeans(AbstractConnector.class)) for (AbstractConnector c : _server.getBeans(AbstractConnector.class))
assertThat(c.isAccepting(),Matchers.is(false)); assertThat(c.isAccepting(),Matchers.is(false));
latch.countDown(); latch.countDown();
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertFalse(_lowResourcesMonitor.isLowOnResources()); assertFalse(_lowResourcesMonitor.getReasons(),_lowResourcesMonitor.isLowOnResources());
for (AbstractConnector c : _server.getBeans(AbstractConnector.class)) for (AbstractConnector c : _server.getBeans(AbstractConnector.class))
assertThat(c.isAccepting(),Matchers.is(true)); assertThat(c.isAccepting(),Matchers.is(true));
} }
@ -172,7 +169,7 @@ public class LowResourcesMonitorTest
{ {
_lowResourcesMonitor.setMaxMemory(Runtime.getRuntime().totalMemory()-Runtime.getRuntime().freeMemory()+(100*1024*1024)); _lowResourcesMonitor.setMaxMemory(Runtime.getRuntime().totalMemory()-Runtime.getRuntime().freeMemory()+(100*1024*1024));
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertFalse(_lowResourcesMonitor.isLowOnResources()); assertFalse(_lowResourcesMonitor.getReasons(),_lowResourcesMonitor.isLowOnResources());
byte[] data = new byte[100*1024*1024]; byte[] data = new byte[100*1024*1024];
Arrays.fill(data,(byte)1); Arrays.fill(data,(byte)1);
@ -180,13 +177,13 @@ public class LowResourcesMonitorTest
assertThat(hash,not(equalTo(0))); assertThat(hash,not(equalTo(0)));
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertTrue(_lowResourcesMonitor.isLowOnResources()); assertTrue(_lowResourcesMonitor.isLowOnResources());
data=null; data=null;
System.gc(); System.gc();
System.gc(); System.gc();
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertFalse(_lowResourcesMonitor.isLowOnResources()); assertFalse(_lowResourcesMonitor.getReasons(),_lowResourcesMonitor.isLowOnResources());
} }
@ -194,26 +191,27 @@ public class LowResourcesMonitorTest
public void testMaxConnectionsAndMaxIdleTime() throws Exception public void testMaxConnectionsAndMaxIdleTime() throws Exception
{ {
_lowResourcesMonitor.setMaxMemory(0); _lowResourcesMonitor.setMaxMemory(0);
Assert.assertFalse(_lowResourcesMonitor.isLowOnResources()); assertFalse(_lowResourcesMonitor.getReasons(),_lowResourcesMonitor.isLowOnResources());
assertEquals( 20, _lowResourcesMonitor.getMaxConnections() );
Socket[] socket = new Socket[_lowResourcesMonitor.getMaxConnections()+1]; Socket[] socket = new Socket[_lowResourcesMonitor.getMaxConnections()+1];
for (int i=0;i<socket.length;i++) for (int i=0;i<socket.length;i++)
socket[i]=new Socket("localhost",_connector.getLocalPort()); socket[i]=new Socket("localhost",_connector.getLocalPort());
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertTrue(_lowResourcesMonitor.isLowOnResources()); assertTrue(_lowResourcesMonitor.isLowOnResources());
try(Socket newSocket = new Socket("localhost",_connector.getLocalPort())) try(Socket newSocket = new Socket("localhost",_connector.getLocalPort()))
{ {
// wait for low idle time to close sockets, but not new Socket // wait for low idle time to close sockets, but not new Socket
Thread.sleep(1200); Thread.sleep(1200);
Assert.assertFalse(_lowResourcesMonitor.isLowOnResources()); assertFalse(_lowResourcesMonitor.getReasons(),_lowResourcesMonitor.isLowOnResources());
for (int i=0;i<socket.length;i++) for (int i=0;i<socket.length;i++)
Assert.assertEquals(-1,socket[i].getInputStream().read()); assertEquals(-1,socket[i].getInputStream().read());
newSocket.getOutputStream().write("GET / HTTP/1.0\r\n\r\n".getBytes(StandardCharsets.UTF_8)); newSocket.getOutputStream().write("GET / HTTP/1.0\r\n\r\n".getBytes(StandardCharsets.UTF_8));
Assert.assertEquals('H',newSocket.getInputStream().read()); assertEquals('H',newSocket.getInputStream().read());
} }
} }
@ -222,11 +220,11 @@ public class LowResourcesMonitorTest
{ {
int monitorPeriod = _lowResourcesMonitor.getPeriod(); int monitorPeriod = _lowResourcesMonitor.getPeriod();
int lowResourcesIdleTimeout = _lowResourcesMonitor.getLowResourcesIdleTimeout(); int lowResourcesIdleTimeout = _lowResourcesMonitor.getLowResourcesIdleTimeout();
Assert.assertThat(lowResourcesIdleTimeout, Matchers.lessThan(monitorPeriod)); assertThat(lowResourcesIdleTimeout, Matchers.lessThan(monitorPeriod));
int maxLowResourcesTime = 5 * monitorPeriod; int maxLowResourcesTime = 5 * monitorPeriod;
_lowResourcesMonitor.setMaxLowResourcesTime(maxLowResourcesTime); _lowResourcesMonitor.setMaxLowResourcesTime(maxLowResourcesTime);
Assert.assertFalse(_lowResourcesMonitor.isLowOnResources()); assertFalse(_lowResourcesMonitor.getReasons(),_lowResourcesMonitor.isLowOnResources());
try(Socket socket0 = new Socket("localhost",_connector.getLocalPort())) try(Socket socket0 = new Socket("localhost",_connector.getLocalPort()))
{ {
@ -236,10 +234,10 @@ public class LowResourcesMonitorTest
// Wait a couple of monitor periods so that // Wait a couple of monitor periods so that
// lowResourceMonitor detects it is in low mode. // lowResourceMonitor detects it is in low mode.
Thread.sleep(2 * monitorPeriod); Thread.sleep(2 * monitorPeriod);
Assert.assertTrue(_lowResourcesMonitor.isLowOnResources()); assertTrue(_lowResourcesMonitor.isLowOnResources());
// We already waited enough for lowResourceMonitor to close socket0. // We already waited enough for lowResourceMonitor to close socket0.
Assert.assertEquals(-1, socket0.getInputStream().read()); assertEquals(-1, socket0.getInputStream().read());
// New connections are not affected by the // New connections are not affected by the
// low mode until maxLowResourcesTime elapses. // low mode until maxLowResourcesTime elapses.
@ -249,11 +247,11 @@ public class LowResourcesMonitorTest
socket1.setSoTimeout(1); socket1.setSoTimeout(1);
InputStream input1 = socket1.getInputStream(); InputStream input1 = socket1.getInputStream();
Assert.assertTrue(_lowResourcesMonitor.isLowOnResources()); assertTrue(_lowResourcesMonitor.isLowOnResources());
try try
{ {
input1.read(); input1.read();
Assert.fail(); fail();
} }
catch (SocketTimeoutException expected) catch (SocketTimeoutException expected)
{ {
@ -263,22 +261,21 @@ public class LowResourcesMonitorTest
Thread.sleep(2 * lowResourcesIdleTimeout); Thread.sleep(2 * lowResourcesIdleTimeout);
// Verify the new socket is still open. // Verify the new socket is still open.
Assert.assertTrue(_lowResourcesMonitor.isLowOnResources()); assertTrue(_lowResourcesMonitor.isLowOnResources());
try try
{ {
input1.read(); input1.read();
Assert.fail(); fail();
} }
catch (SocketTimeoutException expected) catch (SocketTimeoutException expected)
{ {
} }
// Let the maxLowResourcesTime elapse. // Let the maxLowResourcesTime elapse.
Thread.sleep(maxLowResourcesTime); Thread.sleep(maxLowResourcesTime);
Assert.assertTrue(_lowResourcesMonitor.isLowOnResources()); assertTrue(_lowResourcesMonitor.isLowOnResources());
// Now also the new socket should be closed. // Now also the new socket should be closed.
Assert.assertEquals(-1, input1.read()); assertEquals(-1, input1.read());
} }
} }
} }