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

com.moilioncircle.redis.replicator.rdb.BaseRdbParser Maven / Gradle / Ivy

Go to download

Redis Replicator is a redis RDB and Command parser written in java. It can parse,filter,broadcast the RDB and Command events in a real time manner. It also can synchronize redis data to your local cache or to database.

There is a newer version: 3.8.1
Show newest version
/*
 * Copyright 2016-2018 Leon Chen
 *
 * 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.moilioncircle.redis.replicator.rdb;

import com.moilioncircle.redis.replicator.io.RedisInputStream;
import com.moilioncircle.redis.replicator.util.ByteArray;
import com.moilioncircle.redis.replicator.util.Lzf;

import java.io.IOException;

import static com.moilioncircle.redis.replicator.Constants.RDB_14BITLEN;
import static com.moilioncircle.redis.replicator.Constants.RDB_32BITLEN;
import static com.moilioncircle.redis.replicator.Constants.RDB_64BITLEN;
import static com.moilioncircle.redis.replicator.Constants.RDB_6BITLEN;
import static com.moilioncircle.redis.replicator.Constants.RDB_ENCVAL;
import static com.moilioncircle.redis.replicator.Constants.RDB_ENC_INT16;
import static com.moilioncircle.redis.replicator.Constants.RDB_ENC_INT32;
import static com.moilioncircle.redis.replicator.Constants.RDB_ENC_INT8;
import static com.moilioncircle.redis.replicator.Constants.RDB_ENC_LZF;
import static com.moilioncircle.redis.replicator.Constants.RDB_LOAD_ENC;
import static com.moilioncircle.redis.replicator.Constants.RDB_LOAD_PLAIN;
import static com.moilioncircle.redis.replicator.Constants.ZIP_INT_16B;
import static com.moilioncircle.redis.replicator.Constants.ZIP_INT_24B;
import static com.moilioncircle.redis.replicator.Constants.ZIP_INT_32B;
import static com.moilioncircle.redis.replicator.Constants.ZIP_INT_64B;
import static com.moilioncircle.redis.replicator.Constants.ZIP_INT_8B;

/**
 * @author Leon Chen
 * @since 2.1.0
 */
public class BaseRdbParser {
    protected final RedisInputStream in;
    
    public BaseRdbParser(RedisInputStream in) {
        this.in = in;
    }
    
    /**
     * "expiry time in seconds". After that, expiry time is read as a 4 byte unsigned int
     * 

* * @return seconds * @throws IOException when read timeout */ public int rdbLoadTime() throws IOException { return in.readInt(4); } /** * "expiry time in ms". After that, expiry time is read as a 8 byte unsigned long *

* * @return millisecond * @throws IOException when read timeout */ public long rdbLoadMillisecondTime() throws IOException { return in.readLong(8); } /** * read bytes 1 or 2 or 5 *

* 1. |00xxxxxx| remaining 6 bits represent the length *

* 2. |01xxxxxx|xxxxxxxx| the combined 14 bits represent the length *

* 3. |10xxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx| the remaining 6 bits are discarded.Additional 4 bytes represent the length(big endian in version6) *

* 4. |11xxxxxx| the remaining 6 bits are read.and then the next object is encoded in a special format.so we set isencoded = true *

* * @return tuple(len, isencoded) * @throws IOException when read timeout * @see #rdbLoadIntegerObject * @see #rdbLoadLzfStringObject */ public Len rdbLoadLen() throws IOException { boolean isencoded = false; int rawByte = in.read(); int type = (rawByte & 0xC0) >> 6; long value; if (type == RDB_ENCVAL) { isencoded = true; value = rawByte & 0x3F; } else if (type == RDB_6BITLEN) { value = rawByte & 0x3F; } else if (type == RDB_14BITLEN) { value = ((rawByte & 0x3F) << 8) | in.read(); } else if (rawByte == RDB_32BITLEN) { value = in.readInt(4, false); } else if (rawByte == RDB_64BITLEN) { value = in.readLong(8, false); } else { throw new AssertionError("unexpected len-type:" + type); } return new Len(value, isencoded); } /** * @param enctype 0,1,2 * @param flags RDB_LOAD_ENC: encoded string.RDB_LOAD_PLAIN | RDB_LOAD_NONE:raw bytes * @return ByteArray rdb byte array object * @throws IOException when read timeout */ public ByteArray rdbLoadIntegerObject(int enctype, int flags) throws IOException { boolean plain = (flags & RDB_LOAD_PLAIN) != 0; boolean encode = (flags & RDB_LOAD_ENC) != 0; byte[] value; switch (enctype) { case RDB_ENC_INT8: value = in.readBytes(1).first(); break; case RDB_ENC_INT16: value = in.readBytes(2).first(); break; case RDB_ENC_INT32: value = in.readBytes(4).first(); break; default: value = new byte[]{0x00}; break; } if (plain) { return new ByteArray(value); } else if (encode) { // createStringObjectFromLongLong(val); return new ByteArray(String.valueOf(in.readInt(value)).getBytes()); } else { // createObject(OBJ_STRING,sdsfromlonglong(val)); return new ByteArray(value); } } /** * |11xxxxxx| remaining 6bit is 3,then lzf compressed string follows *

* lzf format *

* |lzf flag|clen:1 or 2 or 5 bytes|len:1 or 2 or 5 bytes | lzf compressed bytes | *

* |11xxxxxx|xxxxxxxx|....|xxxxxxxx|xxxxxxxx|....|xxxxxxxx|xxxxxxxx|xxxxxxxx|............xxxxxxxx| *

* * @param flags RDB_LOAD_ENC: encoded string.RDB_LOAD_PLAIN | RDB_LOAD_NONE:raw bytes * @return ByteArray rdb byte array object * @throws IOException when read timeout * @see #rdbLoadLen */ public ByteArray rdbLoadLzfStringObject(int flags) throws IOException { boolean plain = (flags & RDB_LOAD_PLAIN) != 0; boolean encode = (flags & RDB_LOAD_ENC) != 0; long clen = rdbLoadLen().len; long len = rdbLoadLen().len; // if (plain || sds) { // return val; // } else { // return createObject(OBJ_STRING,val); // } if (plain) { return Lzf.decode(in.readBytes(clen), len); } else if (encode) { return Lzf.decode(in.readBytes(clen), len); } else { return Lzf.decode(in.readBytes(clen), len); } } /** * 1.|11xxxxxx|xxxxxxxx| remaining 6bit is 0, then an 8 bit integer follows *

* 2.|11xxxxxx|xxxxxxxx|xxxxxxxx| remaining 6bit is 1, then an 16 bit integer follows *

* 3.|11xxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx| remaining 6bit is 2, then an 32 bit integer follows *

* 4.|11xxxxxx| remaining 6bit is 3,then lzf compressed string follows *

* * @param flags RDB_LOAD_ENC: encoded string.RDB_LOAD_PLAIN | RDB_LOAD_NONE:raw bytes * @return ByteArray rdb byte array object * @throws IOException when read timeout * @see #rdbLoadIntegerObject * @see #rdbLoadLzfStringObject */ public ByteArray rdbGenericLoadStringObject(int flags) throws IOException { boolean plain = (flags & RDB_LOAD_PLAIN) != 0; boolean encode = (flags & RDB_LOAD_ENC) != 0; Len lenObj = rdbLoadLen(); long len = (int) lenObj.len; boolean isencoded = lenObj.isencoded; if (isencoded) { switch ((int) len) { case RDB_ENC_INT8: case RDB_ENC_INT16: case RDB_ENC_INT32: return rdbLoadIntegerObject((int) len, flags); case RDB_ENC_LZF: return rdbLoadLzfStringObject(flags); default: throw new AssertionError("unknown RdbParser encoding type:" + len); } } if (plain) { return in.readBytes(len); } else if (encode) { // createStringObject(NULL,len) return in.readBytes(len); } else { // createRawStringObject(NULL,len); return in.readBytes(len); } } /** * @return ByteArray rdb object with byte[] * @throws IOException when read timeout */ public ByteArray rdbLoadPlainStringObject() throws IOException { return rdbGenericLoadStringObject(RDB_LOAD_PLAIN); } /** * @return ByteArray rdb object with byte[] * @throws IOException when read timeout */ public ByteArray rdbLoadEncodedStringObject() throws IOException { return rdbGenericLoadStringObject(RDB_LOAD_ENC); } public double rdbLoadDoubleValue() throws IOException { int len = in.read(); switch (len) { case 255: return Double.NEGATIVE_INFINITY; case 254: return Double.POSITIVE_INFINITY; case 253: return Double.NaN; default: byte[] bytes = in.readBytes(len).first(); return Double.valueOf(new String(bytes)); } } public double rdbLoadBinaryDoubleValue() throws IOException { return Double.longBitsToDouble(in.readLong(8)); } /** * @return single precision float * @throws IOException io exception * @since 2.2.0 */ public float rdbLoadBinaryFloatValue() throws IOException { return Float.intBitsToFloat(in.readInt(4)); } /** * @see #rdbLoadLen */ public static class Len { public final long len; public final boolean isencoded; private Len(long len, boolean isencoded) { this.len = len; this.isencoded = isencoded; } } public static class StringHelper { private StringHelper() { } public static String str(RedisInputStream in, int len) throws IOException { return in.readString(len); } public static byte[] bytes(RedisInputStream in, int len) throws IOException { return in.readBytes(len).first(); } public static long skip(RedisInputStream in, long len) throws IOException { return in.skip(len); } /* * * * : * |xxxxxxxx| if first byte value < 254. then 1 byte as prev len. * |xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx| if first byte >=254 then next 4 byte as prev len. * * : * |00xxxxxx| remaining 6 bit as string len. * |01xxxxxx|xxxxxxxx| combined 14 bit as string len. * |10xxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx| next 4 byte as string len. * |11111110|xxxxxxxx| next 1 byte as 8bit int * |11000000|xxxxxxxx|xxxxxxxx| next 2 bytes as 16bit int * |11110000|xxxxxxxx|xxxxxxxx|xxxxxxxx| next 3 bytes as 24bit int * |11010000|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx| next 4 bytes as 32bit int * |11100000|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx| next 8 bytes as 64bit long * |11xxxxxx| next 6 bit value as int value */ public static byte[] zipListEntry(RedisInputStream in) throws IOException { int prevlen = in.read(); if (prevlen >= 254) { prevlen = in.readInt(4); } int special = in.read(); switch (special >> 6) { case 0: int len = special & 0x3F; return bytes(in, len); case 1: len = ((special & 0x3F) << 8) | in.read(); return bytes(in, len); case 2: //bigEndian len = in.readInt(4, false); return bytes(in, len); default: break; } switch (special) { case ZIP_INT_8B: return String.valueOf(in.readInt(1)).getBytes(); case ZIP_INT_16B: return String.valueOf(in.readInt(2)).getBytes(); case ZIP_INT_24B: return String.valueOf(in.readInt(3)).getBytes(); case ZIP_INT_32B: return String.valueOf(in.readInt(4)).getBytes(); case ZIP_INT_64B: return String.valueOf(in.readLong(8)).getBytes(); default: //6BIT return String.valueOf(special - 0xF1).getBytes(); } } /* * * * : * |0xxxxxxx| 7 bit unsigned integer * |10xxxxxx| 6 bit unsigned integer as string length. then read the `length` bytes as string. * |110xxxxx|xxxxxxxx| 13 bit signed integer * |1110xxxx|xxxxxxxx| string with length up to 4095 * |11110001|xxxxxxxx|xxxxxxxx| next 2 bytes as 16bit int * |11110010|xxxxxxxx|xxxxxxxx|xxxxxxxx| next 3 bytes as 24bit int * |11110011|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx| next 4 bytes as 32bit int * |11110100|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx| next 8 bytes as 64bit long * |11110000|xxxxxxxx|xxxxxxxx|xxxxxxxx|xxxxxxxx| next 4 bytes as string length. then read the `length` bytes as string. * * : * TBD * * : * TBD */ public static byte[] listPackEntry(RedisInputStream in) throws IOException { int special = in.read(); byte[] value; long skip; if ((special & 0x80) == 0) { skip = 1; value = String.valueOf(special & 0x7F).getBytes(); } else if ((special & 0xC0) == 0x80) { int len = special & 0x3F; skip = 1 + len; value = bytes(in, len); } else if ((special & 0xE0) == 0xC0) { skip = 2; int next = in.read(); value = String.valueOf((((special & 0x1F) << 8) | next) << 19 >> 19).getBytes(); } else if ((special & 0xFF) == 0xF1) { skip = 3; value = String.valueOf(in.readInt(2)).getBytes(); } else if ((special & 0xFF) == 0xF2) { skip = 4; value = String.valueOf(in.readInt(3)).getBytes(); } else if ((special & 0xFF) == 0xF3) { skip = 5; value = String.valueOf(in.readInt(4)).getBytes(); } else if ((special & 0xFF) == 0xF4) { skip = 9; value = String.valueOf(in.readLong(8)).getBytes(); } else if ((special & 0xF0) == 0xE0) { int len = ((special & 0x0F) << 8) | in.read(); skip = 2 + len; value = bytes(in, len); } else if ((special & 0xFF) == 0xf0) { int len = in.readInt(4, false); skip = 5 + len; value = bytes(in, len); } else { throw new UnsupportedOperationException(String.valueOf(special)); } // if (skip <= 127) { in.skip(1); } else if (skip < 16383) { in.skip(2); } else if (skip < 2097151) { in.skip(3); } else if (skip < 268435455) { in.skip(4); } else { in.skip(5); } return value; } } public static class LenHelper { private LenHelper() { } //zip hash public static int zmlen(RedisInputStream in) throws IOException { return in.read(); } public static int free(RedisInputStream in) throws IOException { return in.read(); } public static int zmElementLen(RedisInputStream in) throws IOException { int len = in.read(); if (len >= 0 && len <= 253) { return len; } else if (len == 254) { return in.readInt(4, false); } else { return len; } } //zip list public static int zlbytes(RedisInputStream in) throws IOException { return in.readInt(4); } public static int zlend(RedisInputStream in) throws IOException { return in.read(); } public static int zltail(RedisInputStream in) throws IOException { return in.readInt(4); } public static int zllen(RedisInputStream in) throws IOException { return in.readInt(2); } //int set public static int encoding(RedisInputStream in) throws IOException { return in.readInt(4); } public static long lenOfContent(RedisInputStream in) throws IOException { return in.readUInt(4); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy