/**
*Copyright [2009-2010] [dennis zhuang(killme2008@gmail.com)]
*Licensed under the Apache License, Version 2.0 (the "License");
*you may not use this file except in compliance with the License.
*You may obtain a copy of the License at
* http://www.apache.org/licenses/LICENSE-2.0
*Unless required by applicable law or agreed to in writing,
*software distributed under the License is distributed on an "AS IS" BASIS,
*WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
*either express or implied. See the License for the specific language governing permissions and limitations under the License
*/
/**
*Copyright [2009-2010] [dennis zhuang(killme2008@gmail.com)]
*Licensed under the Apache License, Version 2.0 (the "License");
*you may not use this file except in compliance with the License.
*You may obtain a copy of the License at
* http://www.apache.org/licenses/LICENSE-2.0
*Unless required by applicable law or agreed to in writing,
*software distributed under the License is distributed on an "AS IS" BASIS,
*WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
*either express or implied. See the License for the specific language governing permissions and limitations under the License
*/
package net.rubyeye.xmemcached.command.text;
import java.nio.ByteBuffer;
import java.util.concurrent.CountDownLatch;
import net.rubyeye.xmemcached.command.CommandType;
import net.rubyeye.xmemcached.impl.MemcachedTCPSession;
import net.rubyeye.xmemcached.transcoders.Transcoder;
import net.rubyeye.xmemcached.utils.ByteUtils;
/**
* CAS command for text protocol
*
* @author dennis
*
*/
public class TextCASCommand extends TextStoreCommand {
@SuppressWarnings("unchecked")
public TextCASCommand(String key, byte[] keyBytes, CommandType cmdType,
CountDownLatch latch, int exp, long cas, Object value,
boolean noreply, Transcoder transcoder) {
super(key, keyBytes, cmdType, latch, exp, cas, value, noreply,
transcoder);
}
private FailStatus failStatus;
static enum FailStatus {
NOT_FOUND, EXISTS
}
@Override
public final boolean decode(MemcachedTCPSession session, ByteBuffer buffer) {
if (buffer == null || !buffer.hasRemaining())
return false;
if (result == null) {
if (buffer.remaining() < 2)
return false;
byte first = buffer.get(buffer.position());
byte second = buffer.get(buffer.position() + 1);
if (first == 'S' && second == 'T') {
setResult(Boolean.TRUE);
countDownLatch();
// STORED\r\n
return ByteUtils.stepBuffer(buffer, 8);
} else if (first == 'N') {
setResult(Boolean.FALSE);
countDownLatch();
failStatus = FailStatus.NOT_FOUND;
// NOT_FOUND\r\n
return ByteUtils.stepBuffer(buffer, 11);
} else if (first == 'E' && second == 'X') {
setResult(Boolean.FALSE);
countDownLatch();
failStatus = FailStatus.EXISTS;
// EXISTS\r\n
return ByteUtils.stepBuffer(buffer, 8);
} else
return decodeError(session, buffer);
} else {
Boolean result = (Boolean) this.result;
if (result) {
return ByteUtils.stepBuffer(buffer, 8);
} else {
switch (this.failStatus) {
case NOT_FOUND:
return ByteUtils.stepBuffer(buffer, 11);
case EXISTS:
return ByteUtils.stepBuffer(buffer, 8);
default:
return decodeError(session, buffer);
}
}
}
}
}