blob: 4bcbd8dc202142c0ead3a04a8fd8291fe59131c0 [file] [log] [blame]
package com.gemstone.gemfire.internal.redis.executor.string;
import java.util.List;
import com.gemstone.gemfire.cache.Region;
import com.gemstone.gemfire.internal.redis.ByteArrayWrapper;
import com.gemstone.gemfire.internal.redis.Command;
import com.gemstone.gemfire.internal.redis.Coder;
import com.gemstone.gemfire.internal.redis.ExecutionHandlerContext;
import com.gemstone.gemfire.internal.redis.RedisConstants.ArityDef;
public class DecrByExecutor extends StringExecutor {
private final String ERROR_VALUE_NOT_USABLE = "The value at this key cannot be decremented numerically";
private final String ERROR_DECREMENT_NOT_USABLE = "The decrementation on this key must be numeric";
private final String ERROR_OVERFLOW = "This decrementation cannot be performed due to overflow";
private final int DECREMENT_INDEX = 2;
@Override
public void executeCommand(Command command, ExecutionHandlerContext context) {
List<byte[]> commandElems = command.getProcessedCommand();
Region<ByteArrayWrapper, ByteArrayWrapper> r = context.getRegionCache().getStringsRegion();
if (commandElems.size() < 3) {
command.setResponse(Coder.getErrorResponse(context.getByteBufAllocator(), ArityDef.DECRBY));
return;
}
ByteArrayWrapper key = command.getKey();
checkAndSetDataType(key, context);
ByteArrayWrapper valueWrapper = r.get(key);
/*
* Try increment
*/
byte[] decrArray = commandElems.get(DECREMENT_INDEX);
String decrString = Coder.bytesToString(decrArray);
Long decrement;
try {
decrement = Long.parseLong(decrString);
} catch (NumberFormatException e) {
command.setResponse(Coder.getErrorResponse(context.getByteBufAllocator(), ERROR_DECREMENT_NOT_USABLE));
return;
}
/*
* Value does not exist
*/
if (valueWrapper == null) {
String negativeDecrString = decrString.charAt(0) == Coder.HYPHEN_ID ? decrString.substring(1) : "-" + decrString;
r.put(key, new ByteArrayWrapper(Coder.stringToBytes(negativeDecrString)));
command.setResponse(Coder.getIntegerResponse(context.getByteBufAllocator(), -decrement));
return;
}
/*
* Value exists
*/
String stringValue = Coder.bytesToString(valueWrapper.toBytes());
Long value;
try {
value = Long.parseLong(stringValue);
} catch (NumberFormatException e) {
command.setResponse(Coder.getErrorResponse(context.getByteBufAllocator(), ERROR_VALUE_NOT_USABLE));
return;
}
/*
* Check for overflow
* Negative decrement is used because the decrement is stored as a positive long
*/
if (value <= 0 && -decrement < (Long.MIN_VALUE - value)) {
command.setResponse(Coder.getErrorResponse(context.getByteBufAllocator(), ERROR_OVERFLOW));
return;
}
value -= decrement;
stringValue = "" + value;
r.put(key, new ByteArrayWrapper(Coder.stringToBytes(stringValue)));
command.setResponse(Coder.getIntegerResponse(context.getByteBufAllocator(), value));
}
}