| 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)); |
| |
| } |
| |
| } |