Slow Consumers can cause problems on non-durable topics since they can force the broker to keep old messages in RAM which once it fills up, forces the broker to slow down producers, causing the fast consumers to be slowed down. One option we could implement in the future is spooling to disk - but then spooling to disk could slow down the fast consumers too.
Currently we have a strategy that lets you configure the maximum number of matched messages the broker will keep around for a consumer in addition to its prefetch buffer. Once this maximum is reached, as new messages come in, older messages are discarded. This allows you to keep the RAM for current messages and keep sending messages to a slow consumer but to discard old messages.
Pending Message Limit Strategy
You can configure the
PendingMessageLimitStrategy implementation class on the destination map so that different regions of your topic namespace can have different strategies for dealing with slow consumers. For example you may want to use this strategy for prices which are very high volume but for orders and trades which are lower volume you might not wish to discard old messages.
The strategy calculates the maximum number of pending messages to be kept in RAM for a consumer (above its prefetch size). A value of zero means keep no messages around other than the prefetch amount. A value greater than zero will keep up to that amount of messages around, discarding the older messages as new messages come in. A value of
-1 disables the discarding of messages.
There are currently two different implementations of the strategy:
This strategy uses a constant limit for all consumers (above their prefetch size).
This strategy calculates the maximum number of pending messages using a multiplier of the consumers prefetch size. So you could for example keep around 2.5 times the prefetch count for each consumer.
Using the Prefetch Policy to Configure the Limit
The JMS Client has a prefetch policy you can use to configure the various prefetch limits for persistent and non persistent queues and topics. The prefetch policy also allows you to specify the
maximumPendingMessageLimit on a per connection/consumer basis. One minor difference with configuring this value; to simplify operation with non-JMS clients such as with OpenWire the value of zero is ignored; so the lowest value you can configure is
Configuring the Eviction Policy
We have a
MessageEvictionStrategy which is used to decide which message should be evicted on a slow consumer. The default implementation is:
However, you can write your own to use some application specific way of choosing messages for eviction. For example, if you are sending market data price updates you may wish to find an older price value, which might not be the oldest message.
propertyName is the JMS message property that specifies the price.
Another option could be to use the oldest message with the lowest priority message. Therefore if you have some high priority messages, evict the lower priority messages first even if they are newer.
The following example shows an ActiveMQ broker configuration file. Notice that for topics in the
PRICES.> wildcard range the
pendingMessageLimitStrategy property is set to only keep around
10 messages for each consumer above their prefetch buffer size.
It is advisable that if you know a particular consumer is going to be slow then set its prefetch size to something smaller than the fast consumers!
For example, if you know a particular server is quite slow and you have very high message rates and you have some very fast consumers then you might want to enable this feature and set the prefetch on the slow servers to be a little lower than on the fast servers.
Monitoring the Status of Slow Consumers
You can also use a JMX Console to view the statistics of the active subscriptions. This allows you to view the following statistics on a
The count of how many messages have been discarded during the lifetime of the subscription due to it being a slow consumer
The current number of messages matched and to be dispatched to the subscription as soon as some capacity is available in the prefetch buffer. So a non-zero value implies that the prefetch buffer is full for this subscription