com.couchbase.client.internal.AdaptiveThrottler Maven / Gradle / Ivy
/**
* Copyright (C) 2009-2013 Couchbase, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALING
* IN THE SOFTWARE.
*/
package com.couchbase.client.internal;
import com.couchbase.client.CouchbaseConnection;
import com.couchbase.client.CouchbaseProperties;
import java.net.InetSocketAddress;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import net.spy.memcached.BroadcastOpFactory;
import net.spy.memcached.MemcachedNode;
import net.spy.memcached.compat.SpyObject;
import net.spy.memcached.ops.Operation;
import net.spy.memcached.ops.OperationStatus;
import net.spy.memcached.ops.StatsOperation;
import net.spy.memcached.protocol.binary.BinaryOperationFactory;
/**
* The AdaptiveThrottler allows dynamic backoff of memcached operations to make
* sure the server is not overloaded to more then a certain level.
*/
public class AdaptiveThrottler extends SpyObject implements Throttler {
/**
* Under normal conditions after how many operations the stats should be
* re-checked.
*/
private final int normalStatsInterval;
/**
* Under high watermark conditions (10% of the remaining high_wat) after
* how many operations the stats should be re-checked.
*/
private final int highStatsInterval;
/**
* Under critical high watermark conditions (higher than 10% of the high_wat
* level) after how many operations the stats should be re-checked.
*/
private final int criticalStatsInterval;
/**
* The amount of time (in ms) to sleep when high watermark conditions are
* reached.
*/
private final int highSleep;
/**
* The amount of time (in ms) to sleep when critical high watermark conditions
* are reached.
*/
private final int criticalSleep;
/**
* The current amount of operations on the counter (needed to check when
* the stats need to be re-fetched).
*/
private int intervalCounter = 0;
/**
* Holds a reference to the CouchbaseConnection in order to schedule stats
* operations.
*/
private final CouchbaseConnection conn;
private final InetSocketAddress node;
/**
* Holds the current state of the throttler.
*/
private ThrottlerState currentState = ThrottlerState.NORMAL;
private final BinaryOperationFactory opFact;
/**
* Initialize the Throttler with sensible default settings.
*
* Also, when the appropriate property settings are loaded (see the
* CouchbaseProperties class for more information), those will be used
* instead of the default settings.
*
* Note that there is a second constructor available that lets you set all
* properties by hand. These are the default settings:
*
* - Normal Stats Interval Check: 10000 operations (normalStatsInterval)
* - High Stats Interval Check: 100 operations (highStatsInterval)
* - Critical Stats Interval Check: 10 operations (criticalStatsInterval)
* - Time of throttle when High: 1ms (high_sleep_time)
* - Time of throttle when critical: 3ms (critical_sleep_time)
*/
public AdaptiveThrottler(CouchbaseConnection conn,
BinaryOperationFactory opFact, InetSocketAddress node) {
this(conn, opFact, node,
Integer.parseInt(CouchbaseProperties.getProperty(
"normal_stats_interval", "10000")),
Integer.parseInt(CouchbaseProperties.getProperty(
"high_stats_interval", "100")),
Integer.parseInt(CouchbaseProperties.getProperty(
"critical_stats_interval", "10")),
Integer.parseInt(CouchbaseProperties.getProperty(
"high_sleep_time", "1")),
Integer.parseInt(CouchbaseProperties.getProperty(
"critical_sleep_time", "3"))
);
}
/**
* Construct the AdaptiveThrottler with all possible options.
*
* @param conn the CouchbaseConnection to work against.
* @param opFact the BinaryOperationFactory to work against.
* @param node the node for the throttler.
* @param normalStatsInterval After how many operations a check should be
* initialized when memory is below high_wat.
* @param highStatsInterval After how many operations a check should be
* initialized when memory is higher than high_wat (< 10%)
* @param criticalStatsInterval After how many operations a check should be
* initialized when memory is higher than high_wat (> 10%)
* @param highSleep The time (in ms) to throttle when high is reached.
* @param criticalSleep The time (in ms) to throttle when critical is reached.
*/
public AdaptiveThrottler(CouchbaseConnection conn,
BinaryOperationFactory opFact, InetSocketAddress node,
int normalStatsInterval, int highStatsInterval,
int criticalStatsInterval, int highSleep, int criticalSleep) {
this.conn = conn;
this.opFact = opFact;
this.node = node;
this.normalStatsInterval = normalStatsInterval;
this.highStatsInterval = highStatsInterval;
this.criticalStatsInterval = criticalStatsInterval;
this.highSleep = highSleep;
this.criticalSleep = criticalSleep;
logCreation();
}
/**
* Throttle if needed based on the given throttle constraints.
*/
@Override
public void throttle() {
++intervalCounter;
if(statsNeedFetch()) {
Map stats = gatherStats();
int throttleTime = throttleNeeded(stats);
if(throttleTime > 0) {
getLogger().debug("Throttling operation for " + throttleTime + "ms");
try {
Thread.sleep(throttleTime);
} catch (InterruptedException ex) {
getLogger().warn("Interrupted while Throttling!");
return;
}
}
intervalCounter = 0;
}
}
/**
* Checks if throttling is needed and returns the correct time to
* throttle (or 0 if no throttling is needed).
*
* @param stats stats to analyze for this node.
* @return the number of ms to sleep.
*/
private int throttleNeeded(Map stats) {
long highWater;
long memUsed;
try {
highWater = Long.parseLong(stats.get("ep_mem_high_wat"));
memUsed = Long.parseLong(stats.get("mem_used"));
} catch(NumberFormatException ex) {
getLogger().warn("Received throttle stats invalid, skipping interval.");
return 0;
}
if(memUsed >= (highWater + highWater/10)) {
currentState = ThrottlerState.CRITICAL;
return criticalSleep;
} else if(memUsed >= highWater) {
currentState = ThrottlerState.HIGH;
return highSleep;
} else {
currentState = ThrottlerState.NORMAL;
return 0;
}
}
/**
* Gather appropriate statistics from the cluster/node.
*/
private Map gatherStats() {
final Map> rv =
new HashMap>();
CountDownLatch blatch = conn.broadcastOperation(new BroadcastOpFactory() {
@Override
public Operation newOp(final MemcachedNode n,
final CountDownLatch latch) {
final InetSocketAddress sa = (InetSocketAddress)n.getSocketAddress();
rv.put(sa, new HashMap());
return opFact.stats(null, new StatsOperation.Callback() {
@Override
public void gotStat(String name, String val) {
rv.get(sa).put(name, val);
}
@SuppressWarnings("synthetic-access")
@Override
public void receivedStatus(OperationStatus status) {
if (!status.isSuccess()) {
getLogger().warn("Unsuccessful stats fetch: " + status);
}
}
@Override
public void complete() {
latch.countDown();
}
});
}
});
try {
blatch.await(1000, TimeUnit.MILLISECONDS);
} catch (InterruptedException e) {
throw new RuntimeException("Interrupted waiting for stats", e);
}
return rv.get(node);
}
/**
* Check if stats need to be fetched.
*
* @return true when stats need to be fetched.
*/
private boolean statsNeedFetch() {
if(currentState == ThrottlerState.NORMAL
&& intervalCounter >= normalStatsInterval) {
return true;
} else if(currentState == ThrottlerState.HIGH
&& intervalCounter >= highStatsInterval) {
return true;
} else if(currentState == ThrottlerState.CRITICAL
&& intervalCounter >= criticalStatsInterval) {
return true;
}
return false;
}
private void logCreation() {
getLogger().info("AdaptiveThrottler instantiated with options "
+ "normal_stats_interval: " + this.normalStatsInterval
+ " high_stats_interval: " + this.highStatsInterval
+ " critical_stats_interval: " + this.criticalStatsInterval
+ " high_sleep: " + this.highSleep
+ " critical_sleep: " + this.criticalSleep
+ " - for node " + this.node);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy