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

com.couchbase.client.java.codec.Sdk2CompatibleLegacyTranscoder Maven / Gradle / Ivy

/*
 * Copyright (c) 2021 Couchbase, Inc.
 *
 * 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 com.couchbase.client.java.codec;

import com.couchbase.client.core.error.CouchbaseException;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.nio.charset.StandardCharsets;
import java.util.Date;
import java.util.regex.Pattern;
import java.util.zip.GZIPInputStream;
import java.util.zip.GZIPOutputStream;

/**
 * This transcoder is compatible with the Java SDK 2 "LegacyTranscoder", which makes it usable
 * back to Java SDK 1 as a result.
 *
 * You would really only want to move to this transcoder if you already had to use the LegacyTranscoder in SDK 2
 * because you were using couchbase since the SDK 1 days (you must really love couchbase, thanks! :-)).
 */
public class Sdk2CompatibleLegacyTranscoder implements Transcoder {

  public static final Sdk2CompatibleLegacyTranscoder INSTANCE = new Sdk2CompatibleLegacyTranscoder();

  public static final int DEFAULT_COMPRESSION_THRESHOLD = 16384;

  private static final Pattern DECIMAL_PATTERN = Pattern.compile("^-?\\d+$");

  // General flags
  static final int SERIALIZED = 1;
  static final int COMPRESSED = 2;

  // Special flags for specially handled types.
  private static final int SPECIAL_MASK = 0xff00;
  static final int SPECIAL_BOOLEAN = (1 << 8);
  static final int SPECIAL_INT = (2 << 8);
  static final int SPECIAL_LONG = (3 << 8);
  static final int SPECIAL_DATE = (4 << 8);
  static final int SPECIAL_BYTE = (5 << 8);
  static final int SPECIAL_FLOAT = (6 << 8);
  static final int SPECIAL_DOUBLE = (7 << 8);
  static final int SPECIAL_BYTEARRAY = (8 << 8);

  private final int compressionThreshold;

  public Sdk2CompatibleLegacyTranscoder() {
    this(DEFAULT_COMPRESSION_THRESHOLD);
  }

  public Sdk2CompatibleLegacyTranscoder(int compressionThreshold) {
    this.compressionThreshold = compressionThreshold;
  }

  @Override
  public EncodedValue encode(final Object input) {
    int flags = 0;

    boolean isJson = false;
    byte[] encoded;
    if (input instanceof String) {
      String c = (String) input;
      isJson = isJsonObject(c);
      encoded = c.getBytes(StandardCharsets.UTF_8);
    } else {
      if (input instanceof Long) {
        flags |= SPECIAL_LONG;
        encoded = encodeNum((Long) input, 8);
      } else if (input instanceof Integer) {
        flags |= SPECIAL_INT;
        encoded = encodeNum((Integer) input, 4);
      } else if (input instanceof Boolean) {
        flags |= SPECIAL_BOOLEAN;
        boolean b = (Boolean) input;
        encoded = new byte[1];
        encoded[0] = (byte) (b ? '1' : '0');
      } else if (input instanceof Date) {
        flags |= SPECIAL_DATE;
        encoded = encodeNum(((Date) input).getTime(), 8);
      } else if (input instanceof Byte) {
        flags |= SPECIAL_BYTE;
        encoded = new byte[] { (Byte) input };
      } else if (input instanceof Float) {
        flags |= SPECIAL_FLOAT;
        encoded = encodeNum(Float.floatToRawIntBits((Float) input), 4);
      } else if (input instanceof Double) {
        flags |= SPECIAL_DOUBLE;
        encoded = encodeNum(Double.doubleToRawLongBits((Double) input), 8);
      } else if (input instanceof byte[]) {
        flags |= SPECIAL_BYTEARRAY;
        encoded = (byte[]) input;
      } else {
        flags |= SERIALIZED;
        encoded = serialize(input);
      }
    }

    if (!isJson && encoded.length >= compressionThreshold) {
      byte[] compressed = compress(encoded);
      if (compressed.length < encoded.length) {
        encoded = compressed;
        flags |= COMPRESSED;
      }
    }

    return new EncodedValue(encoded, flags);
  }

  @Override
  @SuppressWarnings("unchecked")
  public  T decode(final Class target, byte[] input, final int flags) {
    Object decoded;
    if ((flags & COMPRESSED) != 0) {
      input = decompress(input);
    }
    int maskedFlags = flags & SPECIAL_MASK;
    if ((flags & SERIALIZED) != 0 && input != null) {
      decoded = deserialize(input);
    } else if (maskedFlags != 0 && input != null) {
      switch(maskedFlags) {
        case SPECIAL_BOOLEAN:
          decoded = input[0] == '1';
          break;
        case SPECIAL_INT:
          decoded = (int) decodeLong(input);
          break;
        case SPECIAL_LONG:
          decoded = decodeLong(input);
          break;
        case SPECIAL_DATE:
          decoded = new Date(decodeLong(input));
          break;
        case SPECIAL_BYTE:
          decoded = input[0];
          break;
        case SPECIAL_FLOAT:
          decoded = Float.intBitsToFloat((int) decodeLong(input));
          break;
        case SPECIAL_DOUBLE:
          decoded = Double.longBitsToDouble(decodeLong(input));
          break;
        case SPECIAL_BYTEARRAY:
          decoded = input;
          break;
        default:
          throw new CouchbaseException("Undecodeable with flags " + flags);
      }
    } else {
      decoded = new String(input, StandardCharsets.UTF_8);
    }

    return (T) decoded;
  }

  /**
   * Check if its a json object or not.
   *
   * Note that this code is not bullet proof, but it is copied over from as-is
   * in the spymemcached project, since its intended to be compatible with it.
   */
  private static boolean isJsonObject(final String s) {
    if (s == null || s.isEmpty()) {
      return false;
    }

    return s.startsWith("{") || s.startsWith("[")
      || "true".equals(s) || "false".equals(s)
      || "null".equals(s) || DECIMAL_PATTERN.matcher(s).matches();
  }

  protected byte[] compress(byte[] in) {
    if (in == null) {
      throw new NullPointerException("Can't compress null");
    }
    ByteArrayOutputStream bos = new ByteArrayOutputStream();
    GZIPOutputStream gz = null;
    try {
      gz = new GZIPOutputStream(bos);
      gz.write(in);
    } catch (IOException e) {
      throw new RuntimeException("IO exception compressing data", e);
    } finally {
      try {
        if (gz != null) {
          gz.close();
        }
      } catch(Exception ex) {
        // ignored.
      }
      try {
        bos.close();
      } catch(Exception ex) {
        // ignored.
      }
    }
    return bos.toByteArray();
  }

  protected byte[] decompress(byte[] in) {
    ByteArrayOutputStream bos = null;
    if(in != null) {
      ByteArrayInputStream bis = new ByteArrayInputStream(in);
      bos = new ByteArrayOutputStream();
      GZIPInputStream gis = null;
      try {
        gis = new GZIPInputStream(bis);

        byte[] buf = new byte[8192];
        int r;
        while ((r = gis.read(buf)) > 0) {
          bos.write(buf, 0, r);
        }
      } catch (IOException e) {
        bos = null;
      } finally {
        try {
          if (bos != null) {
            bos.close();
          }
        } catch(Exception ex) {
          // ignored
        }
        try {
          if (gis != null) {
            gis.close();
          }
        } catch(Exception ex) {
          // ignored
        }

        try {
          bis.close();
        } catch(Exception ex) {
          // ignored
        }
      }
    }
    return bos == null ? null : bos.toByteArray();
  }

  public static byte[] encodeNum(long l, int maxBytes) {
    byte[] rv = new byte[maxBytes];
    for (int i = 0; i < rv.length; i++) {
      int pos = rv.length - i - 1;
      rv[pos] = (byte) ((l >> (8 * i)) & 0xff);
    }
    int firstNon0 = 0;
    // Just looking for what we can reduce
    while (firstNon0 < rv.length && rv[firstNon0] == 0) {
      firstNon0++;
    }
    if (firstNon0 > 0) {
      byte[] tmp = new byte[rv.length - firstNon0];
      System.arraycopy(rv, firstNon0, tmp, 0, rv.length - firstNon0);
      rv = tmp;
    }
    return rv;
  }

  private static byte[] serialize(final Object content) {
    if (content == null) {
      throw new NullPointerException("Can't serialize null");
    }
    byte[] rv;
    try (ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream os = new ObjectOutputStream(bos)) {
      os.writeObject(content);
      os.close();
      bos.close();
      rv = bos.toByteArray();
    } catch (IOException e) {
      throw new IllegalArgumentException("Non-serializable object", e);
    }
    return rv;
  }

  protected Object deserialize(byte[] in) {
    Object rv=null;
    ByteArrayInputStream bis = null;
    ObjectInputStream is = null;
    try {
      if(in != null) {
        bis=new ByteArrayInputStream(in);
        is=new ObjectInputStream(bis);
        rv=is.readObject();
        is.close();
        bis.close();
      }
    } catch (IOException e) {
      throw new CouchbaseException("Caught IOException decoding bytes of data", e);
    } catch (ClassNotFoundException e) {
      throw new CouchbaseException("Caught ClassNotFoundException decoding bytes of data", e);
    } finally {
      try {
        if (is != null) {
          is.close();
        }
      } catch(Exception ex) {
        // ignored
      }
      try {
        if (bis != null) {
          bis.close();
        }
      } catch(Exception ex) {
        // ignored
      }
    }
    return rv;
  }

  static long decodeLong(byte[] b) {
    long rv = 0;
    for (byte i : b) {
      rv = (rv << 8) | (i < 0 ? 256 + i : i);
    }
    return rv;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy