All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.spy.memcached.tapmessage.ResponseMessage Maven / Gradle / Ivy

/**
 * Copyright (C) 2009-2012 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 net.spy.memcached.tapmessage;

import java.nio.ByteBuffer;
import java.util.LinkedList;
import java.util.List;
import net.spy.memcached.CachedData;
import net.spy.memcached.transcoders.SerializingTranscoder;

/**
 * A representation of a tap stream message sent from a tap stream server.
 */
public class ResponseMessage extends BaseMessage {
  // Offsets are given from the end of the header
  private static final int ENGINE_PRIVATE_OFFSET = 24;
  private static final int FLAGS_OFFSET = 26;
  private static final int TTL_OFFSET = 28;
  private static final int RESERVED1_OFFSET = 29;
  private static final int RESERVED2_OFFSET = 30;
  private static final int RESERVED3_OFFSET = 31;
  private static final int ITEM_FLAGS_OFFSET = 32;
  private static final int ITEM_EXPIRY_OFFSET = 36;
  private static final int KEY_OFFSET = 40;

  private final short engineprivate;
  private final List flags;
  private final byte ttl;
  private final byte reserved1;
  private final byte reserved2;
  private final byte reserved3;
  private final int itemflags;
  private int itemexpiry;
  private final int vbucketstate;
  private final long checkpoint;
  private final byte[] key;
  private final byte[] value;
  private final byte[] revid;

  /**
   * Creates a ResponseMessage from binary data.
   *
   * @param b The binary data sent from the tap stream server.
   */
  public ResponseMessage(byte[] b) {
    // TODO: This isn't the best way of doing this. In the future
    // this should be split into mutiple classes.
    super(b);
    if (!opcode.equals(TapOpcode.NOOP)) {
      engineprivate = decodeShort(b, ENGINE_PRIVATE_OFFSET);
      flags = TapResponseFlag.getFlags(decodeShort(b, FLAGS_OFFSET));
      ttl = b[TTL_OFFSET];
      reserved1 = b[RESERVED1_OFFSET];
      reserved2 = b[RESERVED2_OFFSET];
      reserved3 = b[RESERVED3_OFFSET];
    } else {
      engineprivate = 0;
      flags = new LinkedList();
      ttl = 0;
      reserved1 = 0;
      reserved2 = 0;
      reserved3 = 0;
    }

    if (opcode.equals(TapOpcode.MUTATION)) {
      if (flags.contains(TapResponseFlag.TAP_FLAG_NETWORK_BYTE_ORDER)) {
        itemflags = decodeInt(b, ITEM_FLAGS_OFFSET);
      } else {
        // handles Couchbase bug MB-4834
        itemflags = decodeIntHostOrder(b, ITEM_FLAGS_OFFSET);
      }
      itemexpiry = decodeInt(b, ITEM_EXPIRY_OFFSET);
      vbucketstate = 0;
      checkpoint = 0;
      revid = new byte[engineprivate];
      System.arraycopy(b, KEY_OFFSET, revid, 0, engineprivate);
      key = new byte[keylength];
      System.arraycopy(b, KEY_OFFSET + engineprivate, key, 0, keylength);
      value = new byte[b.length - keylength - engineprivate - KEY_OFFSET];
      System.arraycopy(b, (b.length - value.length), value, 0, value.length);
    } else if (opcode.equals(TapOpcode.DELETE)) {
      itemflags = 0;
      itemexpiry = 0;
      vbucketstate = 0;
      revid = new byte[engineprivate];
      System.arraycopy(b, 32, revid, 0, engineprivate);
      checkpoint = 0;
      key = new byte[keylength];
      System.arraycopy(b, 32 + engineprivate, key, 0, keylength);
      value = new byte[0];
    } else if (opcode.equals(TapOpcode.VBUCKETSET)) {
      itemflags = 0;
      itemexpiry = 0;
      vbucketstate = decodeInt(b, ITEM_FLAGS_OFFSET);
      checkpoint = 0;
      key = new byte[0];
      value = new byte[0];
      revid = new byte[0];
    } else if (opcode.equals(TapOpcode.START_CHECKPOINT)
      || opcode.equals(TapOpcode.END_CHECKPOINT)) {
      itemflags = 0;
      itemexpiry = 0;
      vbucketstate = 0;
      checkpoint = decodeLong(b, KEY_OFFSET);
      key = new byte[0];
      value = new byte[0];
      revid = new byte[0];
    } else if (opcode.equals(TapOpcode.OPAQUE)) {
      itemflags = 0;
      itemexpiry = 0;
      vbucketstate = decodeInt(b, ITEM_FLAGS_OFFSET);
      checkpoint = 0;
      key = new byte[0];
      value = new byte[0];
      revid = new byte[0];
    } else {
      itemflags = 0;
      itemexpiry = 0;
      vbucketstate = 0;
      checkpoint = 0;
      key = new byte[0];
      value = new byte[0];
      revid = new byte[0];
    }
  }

  /**
   * Gets the value of the engine private field. Not returned in a no-op
   * message.
   *
   * @return The engine private data.
   */
  public long getEnginePrivate() {
    return engineprivate;
  }

  /**
   * Gets the value of the flags field. Not returned in a no-op message.
   *
   * @return The flags data.
   */
  public List getFlags() {
    return flags;
  }

  /**
   * Gets the value of the time to live field. Not returned in a no-op message.
   *
   * @return The time to live value;
   */
  public int getTTL() {
    return ttl;
  }

  /**
   * Gets the value of the reserved1 field. Not returned in a no-op message.
   *
   * @return The reserved1 data.
   */
  protected int getReserved1() {
    return reserved1;
  }

  /**
   * Gets the value of the reserved2 field. Not returned in a no-op message.
   *
   * @return The reserved2 data.
   */
  protected int getReserved2() {
    return reserved2;
  }

  /**
   * Gets the value of the reserved3 field. Not returned in a no-op message.
   *
   * @return The reserved3 data.
   */
  protected int getReserved3() {
    return reserved3;
  }

  /**
   * Gets the state of the vbucket. Only returned with a tap vbucket state
   * message.
   *
   * @return the vbucket state
   */
  public int getVBucketState() {
    return vbucketstate;
  }

  /**
   * Gets the checkpoint of the vbucket.  Only returned with a start/end
   * checkpoint message.
   *
   * @return the checkpoint
   */
  public long getCheckpoint() {
    return checkpoint;
  }

  /**
   * Gets the value of the items flag field. Only returned with a tap mutation
   * message.
   *
   * @return The items flag data.
   */
  public int getItemFlags() {
    return itemflags;
  }

  /**
   * Gets the value of the item expiry field. Only returned with a tap mutation
   * message.
   *
   * @return The item expiry data.
   */
  public long getItemExpiry() {
    return itemexpiry;
  }

  /**
   * Gets the value of the key field. Only returned with a tap mutation
   * or tap delete message.
   *
   * @return The key data.
   */
  public String getKey() {
    return new String(key);
  }

  /**
   * Gets the value of the value field. Only returned with a tap mutation
   * message.
   *
   * @return The value data.
   */
  public byte[] getValue() {
    return value;
  }

  /**
   * Gets the value of the revid field. Only returned with a tap mutation
   * message.
   *
   * @return The revid of the document.
   */
  public byte[] getRevID() {
    return revid;
  }

  public ByteBuffer getBytes() {
    int bufSize = 0;
    bufSize += HEADER_LENGTH;
    if (opcode.equals(TapOpcode.MUTATION)) {
      bufSize += 16;
    }
    bufSize += getTotalbody();

    ByteBuffer bb = ByteBuffer.allocate(bufSize);
    bb.put(magic.getMagic());
    bb.put(opcode.getOpcode());
    bb.putShort(keylength);
    bb.put(extralength);
    bb.put(datatype);
    bb.putShort(vbucket);
    bb.putInt(totalbody);
    bb.putInt(opaque);
    bb.putLong(cas);

    if (opcode.equals(TapOpcode.NOOP)) {
      return bb;
    }

    bb.putShort(engineprivate);

    short flag = 0;
    for (int i = 0; i < flags.size(); i++) {
      flag |= flags.get(i).getFlags();
    }

    bb.putShort(flag);
    bb.put(ttl);
    bb.put(reserved1);
    bb.put(reserved2);
    bb.put(reserved3);

    if (opcode.equals(TapOpcode.MUTATION)) {
      bb.putInt(itemflags);
      bb.putInt(itemexpiry);
      bb.put(revid);
      bb.put(key);
      bb.put(value);
    } else if (opcode.equals(TapOpcode.DELETE)) {
      bb.put(revid);
      bb.put(key);
    } else if (opcode.equals(TapOpcode.VBUCKETSET)) {
      bb.putInt(vbucketstate);
    }
    return bb;
  }

  @Override
  public String toString() {
    return String.format("Key: %s, Flags: %d, TTL: %d, Size: %d\nValue: %s",
      getKey(), getItemFlags(), getTTL(), getValue().length, deserialize());
  }

  /**
   * Attempt to get the object represented by the given serialized bytes.
   */
  private Object deserialize() {
    SerializingTranscoder tc = new SerializingTranscoder();
    CachedData d = new CachedData(this.getItemFlags(), this.getValue(),
      CachedData.MAX_SIZE);
    Object rv = null;
    rv = tc.decode(d);
    return rv;
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy