Merge pull request #781 from WalkerWatch/docupdates/Chapter20
Chapter 20 cleanup.
This commit is contained in:
commit
b45fbbfd04
|
@ -17,11 +17,11 @@
|
||||||
[[optimizing]]
|
[[optimizing]]
|
||||||
== Optimizing Jetty
|
== Optimizing Jetty
|
||||||
|
|
||||||
There are many ways to optimize Jetty, which vary depending on the situation.
|
There are many ways to optimize Jetty which vary depending on the situation.
|
||||||
Are you trying to optimize for number of requests within a given amount of time?
|
Are you trying to optimize for number of requests within a given amount of time?
|
||||||
Are you trying to optimize the serving of static content?
|
Are you trying to optimize the serving of static content?
|
||||||
Do you have a large bit of hardware that you want to give entirely over to Jetty to use to its heart's delight?
|
Do you have a large bit of hardware that you want to give entirely over to Jetty to use to its heart's delight?
|
||||||
Here are a few of the many different ways to optimize Jetty.
|
This chapter examines a few of the many different ways to optimize Jetty.
|
||||||
|
|
||||||
include::garbage-collection.adoc[]
|
include::garbage-collection.adoc[]
|
||||||
include::high-load.adoc[]
|
include::high-load.adoc[]
|
||||||
|
|
|
@ -19,7 +19,7 @@
|
||||||
|
|
||||||
Tuning the JVM garbage collection (GC) can greatly improve Jetty performance.
|
Tuning the JVM garbage collection (GC) can greatly improve Jetty performance.
|
||||||
Specifically, you can avoid pauses while the system performs full garbage collections.
|
Specifically, you can avoid pauses while the system performs full garbage collections.
|
||||||
Optimal tuning of the GC depends on the behaviour of the application and requires detailed analysis, however there are general recommendations.
|
Optimal tuning of the GC depends on the behavior of the application and requires detailed analysis, however there are general recommendations.
|
||||||
|
|
||||||
See official https://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/[Java 8 Garbage Collection documentation] for further assistance.
|
See official https://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/[Java 8 Garbage Collection documentation] for further assistance.
|
||||||
|
|
||||||
|
@ -29,8 +29,9 @@ See official https://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/
|
||||||
These options are general to the Oracle JVM, and work in a Java 8 installation.
|
These options are general to the Oracle JVM, and work in a Java 8 installation.
|
||||||
They provide good information about how your JVM is running; based on that initial information, you can then tune more finely.
|
They provide good information about how your JVM is running; based on that initial information, you can then tune more finely.
|
||||||
|
|
||||||
The most important thing you can do for yourself when worrying about GC is to capture the GC activity so that you can analyze what is happening and how often it happens.
|
The most important thing you can do for yourself when considering GC is to capture the GC activity so that you can analyze what is happening and how often it happens.
|
||||||
|
|
||||||
|
[source,screen, subs="{sub-order}"]
|
||||||
....
|
....
|
||||||
-verbose:gc
|
-verbose:gc
|
||||||
-Xloggc:/path/to/myjettybase/logs/gc.log
|
-Xloggc:/path/to/myjettybase/logs/gc.log
|
||||||
|
@ -45,21 +46,23 @@ The most important thing you can do for yourself when worrying about GC is to ca
|
||||||
|
|
||||||
There are not many recommended options for GC that can apply to nearly all users.
|
There are not many recommended options for GC that can apply to nearly all users.
|
||||||
|
|
||||||
However, the most obvious one is to disable explicit GC (this is performed regularly by RMI and can introduce an abnormal amount of GC pauses)
|
However, the most obvious one is to disable explicit GC (this is performed regularly by RMI and can introduce an abnormal amount of GC pauses).
|
||||||
|
|
||||||
|
[source,screen, subs="{sub-order}"]
|
||||||
....
|
....
|
||||||
-XX:+DisableExplicitGC
|
-XX:+DisableExplicitGC
|
||||||
....
|
....
|
||||||
|
|
||||||
Before you apply any other GC tuning options, monitor your GC logs to see if https://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/cms.html[tuning the CMS] makes sense for your environment.
|
Before you apply any other GC tuning options, monitor your GC logs to see if https://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/cms.html[tuning the CMS] makes sense for your environment.
|
||||||
|
|
||||||
A common setup for those just starting out with GC tuning is included below for reference reasons.
|
A common setup for those just starting out with GC tuning is included below for reference.
|
||||||
|
|
||||||
____
|
____
|
||||||
[CAUTION]
|
[CAUTION]
|
||||||
This example configuration below could have a negative effect on your application performance, so continued monitoring of your GC log before and after is highly recommended to know if the configuration was beneficial or not.
|
This example configuration below could have a negative effect on your application performance, so continued monitoring of your GC log before and after is highly recommended to know if the configuration was beneficial or not.
|
||||||
____
|
____
|
||||||
|
|
||||||
|
[source,screen, subs="{sub-order}"]
|
||||||
....
|
....
|
||||||
-XX:MaxGCPauseMillis=250
|
-XX:MaxGCPauseMillis=250
|
||||||
-XX:+UseConcMarkSweepGC
|
-XX:+UseConcMarkSweepGC
|
||||||
|
|
|
@ -21,11 +21,12 @@ Configuring Jetty for high load, whether for load testing or for production, req
|
||||||
|
|
||||||
==== Load Generation for Load Testing
|
==== Load Generation for Load Testing
|
||||||
|
|
||||||
The load generation machines must have their OS, JVM, etc., tuned just as much as the server machines.
|
Machines handling load generation must have their OS, JVM, etc., tuned just as much as the server machines.
|
||||||
|
|
||||||
The load generation should not be over the local network on the server machine, as this has unrealistic performance and latency as well as different packet sizes and transport characteristics.
|
The load generation should not be over the local network on the server machine, as this has unrealistic performance and latency as well as different packet sizes and transport characteristics.
|
||||||
|
|
||||||
The load generator should generate a realistic load:
|
The load generator should generate a realistic load.
|
||||||
|
Avoid the following pitfalls:
|
||||||
|
|
||||||
* A common mistake is that load generators often open relatively few connections that are extremely busy sending as many requests as possible over each connection.
|
* A common mistake is that load generators often open relatively few connections that are extremely busy sending as many requests as possible over each connection.
|
||||||
This causes the measured throughput to be limited by request latency (see http://blogs.webtide.com/gregw/entry/lies_damned_lies_and_benchmarks[Lies, Damned Lies and Benchmarks] for an analysis of such an issue).
|
This causes the measured throughput to be limited by request latency (see http://blogs.webtide.com/gregw/entry/lies_damned_lies_and_benchmarks[Lies, Damned Lies and Benchmarks] for an analysis of such an issue).
|
||||||
|
@ -34,9 +35,9 @@ This often results in accept queues filling and limitations due to file descript
|
||||||
* A load generator should model the traffic profile from the normal clients of the server.
|
* A load generator should model the traffic profile from the normal clients of the server.
|
||||||
For browsers, this is often between two and six connections that are mostly idle and that are used in sporadic bursts with read times in between.
|
For browsers, this is often between two and six connections that are mostly idle and that are used in sporadic bursts with read times in between.
|
||||||
The connections are typically long held HTTP/1.1 connections.
|
The connections are typically long held HTTP/1.1 connections.
|
||||||
* Load generators should be written in asynchronous programming style, so that a limited number of threads does not restrict the maximum number of users that can be simulated.
|
* Load generators should be written in asynchronously so that a limited number of threads does not restrict the maximum number of users that can be simulated.
|
||||||
If the generator is not asynchronous, a thread pool of 2000 may only be able to simulate 500 or fewer users.
|
If the generator is not asynchronous, a thread pool of 2000 may only be able to simulate 500 or fewer users.
|
||||||
The Jetty HttpClient is an ideal choice for building a load generator, as it is asynchronous and can simulate many thousands of connections (see the CometD Load Tester for a good example of a realistic load generator).
|
The Jetty `HttpClient` is an ideal choice for building a load generator as it is asynchronous and can simulate many thousands of connections (see the CometD Load Tester for a good example of a realistic load generator).
|
||||||
|
|
||||||
==== Operating System Tuning
|
==== Operating System Tuning
|
||||||
|
|
||||||
|
@ -44,11 +45,12 @@ Both the server machine and any load generating machines need to be tuned to sup
|
||||||
|
|
||||||
===== Linux
|
===== Linux
|
||||||
|
|
||||||
Linux does a reasonable job of self-configuring TCP/IP, but there are a few limits and defaults that you should increase. You can configure most of them in `/etc/security/limits.conf` or via `sysctl`.
|
Linux does a reasonable job of self-configuring TCP/IP, but there are a few limits and defaults that you should increase.
|
||||||
|
You can configure most of these in `/etc/security/limits.conf` or via `sysctl`.
|
||||||
|
|
||||||
====== TCP Buffer Sizes
|
====== TCP Buffer Sizes
|
||||||
|
|
||||||
You should increase TCP buffer sizes to at least 16MB for 10G paths and tune the auto-tuning (although you now need to consider buffer bloat).
|
You should increase TCP buffer sizes to at least 16MB for 10G paths and tune the auto-tuning (keep in mind that you need to consider buffer bloat).
|
||||||
|
|
||||||
[source, screen, subs="{sub-order}"]
|
[source, screen, subs="{sub-order}"]
|
||||||
....
|
....
|
||||||
|
@ -56,44 +58,39 @@ $ sysctl -w net.core.rmem_max=16777216
|
||||||
$ sysctl -w net.core.wmem_max=16777216
|
$ sysctl -w net.core.wmem_max=16777216
|
||||||
$ sysctl -w net.ipv4.tcp_rmem="4096 87380 16777216"
|
$ sysctl -w net.ipv4.tcp_rmem="4096 87380 16777216"
|
||||||
$ sysctl -w net.ipv4.tcp_wmem="4096 16384 16777216"
|
$ sysctl -w net.ipv4.tcp_wmem="4096 16384 16777216"
|
||||||
|
|
||||||
....
|
....
|
||||||
|
|
||||||
====== Queue Sizes
|
====== Queue Sizes
|
||||||
|
|
||||||
`net.core.somaxconn` controls the size of the connection listening queue.
|
`net.core.somaxconn` controls the size of the connection listening queue.
|
||||||
The default value is 128; if you are running a high-volume server and connections are getting refused at a TCP level, you need to increase this.
|
The default value is 128.
|
||||||
This is a very tweakable setting in such a case: if you set it too high, resource problems occur as it tries to notify a server of a large number of connections, and many remain pending, but if you set it too low, refused connections occur.
|
If you are running a high-volume server and connections are getting refused at a TCP level, you need to increase this value.
|
||||||
|
This setting can take a bit of finesse to get correct: if you set it too high, resource problems occur as it tries to notify a server of a large number of connections, and many remain pending, but if you set it too low, refused connections occur.
|
||||||
|
|
||||||
[source, screen, subs="{sub-order}"]
|
[source, screen, subs="{sub-order}"]
|
||||||
....
|
....
|
||||||
$ sysctl -w net.core.somaxconn=4096
|
$ sysctl -w net.core.somaxconn=4096
|
||||||
|
|
||||||
....
|
....
|
||||||
|
|
||||||
The `net.core.netdev_max_backlog` controls the size of the incoming packet queue for upper-layer (java) processing.
|
The `net.core.netdev_max_backlog` controls the size of the incoming packet queue for upper-layer (Java) processing.
|
||||||
The default (2048) may be increased and other related parameters (TODO MORE EXPLANATION) adjusted with:
|
The default (2048) may be increased and other related parameters adjusted with:
|
||||||
|
|
||||||
[source, screen, subs="{sub-order}"]
|
[source, screen, subs="{sub-order}"]
|
||||||
....
|
....
|
||||||
|
|
||||||
$ sysctl -w net.core.netdev_max_backlog=16384
|
$ sysctl -w net.core.netdev_max_backlog=16384
|
||||||
$ sysctl -w net.ipv4.tcp_max_syn_backlog=8192
|
$ sysctl -w net.ipv4.tcp_max_syn_backlog=8192
|
||||||
$ sysctl -w net.ipv4.tcp_syncookies=1
|
$ sysctl -w net.ipv4.tcp_syncookies=1
|
||||||
|
|
||||||
|
|
||||||
....
|
....
|
||||||
|
|
||||||
====== Ports
|
====== Ports
|
||||||
|
|
||||||
If many outgoing connections are made (for example, on load generators), the operating system might run low on ports. Thus it is best to increase the port range, and allow reuse of sockets in TIME_WAIT:
|
If many outgoing connections are made (for example, on load generators), the operating system might run low on ports.
|
||||||
|
Thus it is best to increase the port range, and allow reuse of sockets in `TIME_WAIT`:
|
||||||
|
|
||||||
[source, screen, subs="{sub-order}"]
|
[source, screen, subs="{sub-order}"]
|
||||||
....
|
....
|
||||||
|
|
||||||
$ sysctl -w net.ipv4.ip_local_port_range="1024 65535"
|
$ sysctl -w net.ipv4.ip_local_port_range="1024 65535"
|
||||||
$ sysctl -w net.ipv4.tcp_tw_recycle=1
|
$ sysctl -w net.ipv4.tcp_tw_recycle=1
|
||||||
|
|
||||||
....
|
....
|
||||||
|
|
||||||
====== File Descriptors
|
====== File Descriptors
|
||||||
|
@ -104,7 +101,6 @@ These can be increased for a specific user in `/etc/security/limits.conf`:
|
||||||
....
|
....
|
||||||
theusername hard nofile 40000
|
theusername hard nofile 40000
|
||||||
theusername soft nofile 40000
|
theusername soft nofile 40000
|
||||||
|
|
||||||
....
|
....
|
||||||
|
|
||||||
====== Congestion Control
|
====== Congestion Control
|
||||||
|
@ -115,7 +111,6 @@ To get a list of congestion control algorithms that are available in your kernel
|
||||||
[source, screen, subs="{sub-order}"]
|
[source, screen, subs="{sub-order}"]
|
||||||
....
|
....
|
||||||
$ sysctl net.ipv4.tcp_available_congestion_control
|
$ sysctl net.ipv4.tcp_available_congestion_control
|
||||||
|
|
||||||
....
|
....
|
||||||
|
|
||||||
If cubic and/or htcp are not listed, you need to research the control algorithms for your kernel.
|
If cubic and/or htcp are not listed, you need to research the control algorithms for your kernel.
|
||||||
|
@ -124,7 +119,6 @@ You can try setting the control to cubic with:
|
||||||
[source, screen, subs="{sub-order}"]
|
[source, screen, subs="{sub-order}"]
|
||||||
....
|
....
|
||||||
$ sysctl -w net.ipv4.tcp_congestion_control=cubic
|
$ sysctl -w net.ipv4.tcp_congestion_control=cubic
|
||||||
|
|
||||||
....
|
....
|
||||||
|
|
||||||
====== Mac OS
|
====== Mac OS
|
||||||
|
@ -147,7 +141,7 @@ Make sure to use persistent HTTP/1.1 connections.
|
||||||
* Use the -server option
|
* Use the -server option
|
||||||
* Jetty Tuning
|
* Jetty Tuning
|
||||||
|
|
||||||
====== Connectors
|
//====== Connectors
|
||||||
|
|
||||||
====== Acceptors
|
====== Acceptors
|
||||||
|
|
||||||
|
@ -160,4 +154,4 @@ Must not be configured for less than the number of expected connections.
|
||||||
====== Thread Pool
|
====== Thread Pool
|
||||||
|
|
||||||
Configure with goal of limiting memory usage maximum available.
|
Configure with goal of limiting memory usage maximum available.
|
||||||
Typically >50 and <500
|
Typically this is >50 and <500
|
||||||
|
|
|
@ -17,12 +17,12 @@
|
||||||
[[limit-load]]
|
[[limit-load]]
|
||||||
=== Limiting Load
|
=== Limiting Load
|
||||||
|
|
||||||
To achieve optimal fair handling for all users of a server, it can be necessary to limit the resources that each user/connection can utilize so as to maximize throughput for the server or to ensure that the entire server runs within the limitations of it's runtime
|
To achieve optimal fair handling for all users of a server, it can be necessary to limit the resources that each user/connection can utilize so as to maximize throughput for the server or to ensure that the entire server runs within the limitations of it's runtime.
|
||||||
|
|
||||||
==== Low Resources Monitor
|
==== Low Resources Monitor
|
||||||
|
|
||||||
An instance of link:{JDURL}/org/eclipse/jetty/server/LowResourcesMonitor.html[LowResourcesMonitor] may be added to a Jetty Server to monitor for low resources situations and to take action to limit the number of idle connections on the server.
|
An instance of link:{JDURL}/org/eclipse/jetty/server/LowResourcesMonitor.html[LowResourcesMonitor] may be added to a Jetty server to monitor for low resources situations and to take action to limit the number of idle connections on the server.
|
||||||
To configure the low resources monitor, you can uncomment the jetty-lowresources.xml line from the start.ini configuration file, which has the effect of including the following XML configuration:
|
To configure the low resources monitor, you can enable the the `lowresources.mod` on the command line, which has the effect of including the following XML configuration:
|
||||||
|
|
||||||
[source, xml, subs="{sub-order}"]
|
[source, xml, subs="{sub-order}"]
|
||||||
----
|
----
|
||||||
|
@ -31,15 +31,11 @@ include::{SRCDIR}/jetty-server/src/main/config/etc/jetty-lowresources.xml[]
|
||||||
|
|
||||||
The monitor is configured with a period in milliseconds at which it will scan the server looking for a low resources condition, which may be one of:
|
The monitor is configured with a period in milliseconds at which it will scan the server looking for a low resources condition, which may be one of:
|
||||||
|
|
||||||
* If monitorThreads is configured as true and a connectors Executor is an instance of link:{JDURL}/org/eclipse/jetty/util/thread/ThreadPool.html[ThreadPool], then its isLowOnThreads() method is used to detect low resources.
|
* If `monitorThreads` is configured as true and a connectors Executor is an instance of link:{JDURL}/org/eclipse/jetty/util/thread/ThreadPool.html[ThreadPool], then its `isLowOnThreads()` method is used to detect low resources.
|
||||||
* If maxConnections is configured to a number >0 then if the total number of connections from all monitored connectors exceeds this value, then low resources state is entered.
|
* If `maxConnections` is configured to a number >0 then if the total number of connections from all monitored connectors exceeds this value, then low resources state is entered.
|
||||||
* If the maxMemory field is configured to a number of bytes >0 then if the JVMs total memory minus its idle memory exceeds this value, then low resources state is entered.
|
* If the `maxMemory` field is configured to a number of bytes >0 then if the JVMs total memory minus its idle memory exceeds this value, then low resources state is entered.
|
||||||
|
|
||||||
Once low resources state is detected, then the monitor will iterate over all existing connections and set their IdleTimeout to its configured lowResourcesIdleTimeout in milliseconds.
|
Once low resources state is detected, then the monitor will iterate over all existing connections and set their `IdleTimeout` to its configured `lowResourcesIdleTimeout` in milliseconds.
|
||||||
This allows the idle time of existing connections to be reduced so that the connection is quickly closed if no further request are received.
|
This allows the idle time of existing connections to be reduced so that the connection is quickly closed if no further request are received.
|
||||||
|
|
||||||
If the low resources state persists longer than the time in milliseconds configured for the maxLowResourcesTime field, the the lowResourcesIdleTimeout is repeatedly applied so that new connections as well as existing connections will be limited.
|
If the low resources state persists longer than the time in milliseconds configured for the `maxLowResourcesTime` field, the the `lowResourcesIdleTimeout` is repeatedly applied so that new connections as well as existing connections will be limited.
|
||||||
|
|
||||||
==== Denial Of Service Filter
|
|
||||||
|
|
||||||
TBD (see http://wiki.eclipse.org/Jetty/Reference/DoSFilter[DoSFilter]).
|
|
||||||
|
|
Loading…
Reference in New Issue