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

org.neo4j.driver.internal.packstream.PackStream Maven / Gradle / Ivy

There is a newer version: 5.27.0
Show newest version
/**
 * Copyright (c) 2002-2016 "Neo Technology,"
 * Network Engine for Objects in Lund AB [http://neotechnology.com]
 *
 * This file is part of Neo4j.
 *
 * 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 org.neo4j.driver.internal.packstream;

import java.io.IOException;
import java.nio.channels.ReadableByteChannel;
import java.nio.channels.WritableByteChannel;
import java.nio.charset.Charset;
import java.util.List;
import java.util.Map;

import static java.lang.Integer.toHexString;
import static java.lang.String.format;
import static java.util.Collections.singletonList;

/**
 * PackStream is a messaging serialisation format heavily inspired by MessagePack.
 * The key differences are in the type system itself which (among other things) replaces extensions with structures.
 * The Packer and Unpacker implementations are also faster than their MessagePack counterparts.
 *
 * Note that several marker byte values are RESERVED for future use.
 * Extra markers should not be added casually and such additions must be follow a strict process involving both client and server software.
 *
 * The table below shows all allocated marker byte values.
 *
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
 * 
MarkerBinaryTypeDescription
00..7F0xxxxxxx+TINY_INTInteger 0 to 127
80..8F1000xxxxTINY_STRING
90..9F1001xxxxTINY_LIST
A0..AF1010xxxxTINY_MAP
B0..BF1011xxxxTINY_STRUCT
C011000000NULL
C111000001FLOAT_6464-bit floating point number (double)
C211000010FALSEBoolean false
C311000011TRUEBoolean true
C4..C7110001xxRESERVED
C811001000INT_88-bit signed integer
C911001001INT_816-bit signed integer
CA11001010INT_832-bit signed integer
CB11001011INT_864-bit signed integer
CC11001100BYTES_8Byte string (fewer than 28 bytes)
CD11001101BYTES_16Byte string (fewer than 216 bytes)
CE11001110BYTES_32Byte string (fewer than 232 bytes)
CF11001111RESERVED
D011010000STRING_8UTF-8 encoded string (fewer than 28 bytes)
D111010001STRING_16UTF-8 encoded string (fewer than 216 bytes)
D211010010STRING_32UTF-8 encoded string (fewer than 232 bytes)
D311010011RESERVED
D411010100LIST_8List (fewer than 28 items)
D511010101LIST_16List (fewer than 216 items)
D611010110LIST_32List (fewer than 232 items)
D711010111RESERVED
D811011000MAP_8Map (fewer than 28 key:value pairs)
D911011001MAP_16Map (fewer than 216 key:value pairs)
DA11011010MAP_32Map (fewer than 232 key:value pairs)
DB11011011RESERVED
DC11011100STRUCT_8Structure (fewer than 28 fields)
DD11011101STRUCT_16Structure (fewer than 216 fields)
DE11011110STRUCT_32Structure (fewer than 232 fields)
DF11011111RESERVED
E0..EF1110xxxxRESERVED
F0..FF1111xxxx-TINY_INTInteger -1 to -16
* */ public class PackStream { public static final byte TINY_STRING = (byte) 0x80; public static final byte TINY_LIST = (byte) 0x90; public static final byte TINY_MAP = (byte) 0xA0; public static final byte TINY_STRUCT = (byte) 0xB0; public static final byte NULL = (byte) 0xC0; public static final byte FLOAT_64 = (byte) 0xC1; public static final byte FALSE = (byte) 0xC2; public static final byte TRUE = (byte) 0xC3; public static final byte RESERVED_C4 = (byte) 0xC4; public static final byte RESERVED_C5 = (byte) 0xC5; public static final byte RESERVED_C6 = (byte) 0xC6; public static final byte RESERVED_C7 = (byte) 0xC7; public static final byte INT_8 = (byte) 0xC8; public static final byte INT_16 = (byte) 0xC9; public static final byte INT_32 = (byte) 0xCA; public static final byte INT_64 = (byte) 0xCB; public static final byte BYTES_8 = (byte) 0xCC; public static final byte BYTES_16 = (byte) 0xCD; public static final byte BYTES_32 = (byte) 0xCE; public static final byte RESERVED_CF = (byte) 0xCF; public static final byte STRING_8 = (byte) 0xD0; public static final byte STRING_16 = (byte) 0xD1; public static final byte STRING_32 = (byte) 0xD2; public static final byte RESERVED_D3 = (byte) 0xD3; public static final byte LIST_8 = (byte) 0xD4; public static final byte LIST_16 = (byte) 0xD5; public static final byte LIST_32 = (byte) 0xD6; public static final byte RESERVED_D7 = (byte) 0xD7; public static final byte MAP_8 = (byte) 0xD8; public static final byte MAP_16 = (byte) 0xD9; public static final byte MAP_32 = (byte) 0xDA; public static final byte RESERVED_DB = (byte) 0xDB; public static final byte STRUCT_8 = (byte) 0xDC; public static final byte STRUCT_16 = (byte) 0xDD; public static final byte RESERVED_DE = (byte) 0xDE; // TODO STRUCT_32? or the class javadoc is wrong? public static final byte RESERVED_DF = (byte) 0xDF; public static final byte RESERVED_E0 = (byte) 0xE0; public static final byte RESERVED_E1 = (byte) 0xE1; public static final byte RESERVED_E2 = (byte) 0xE2; public static final byte RESERVED_E3 = (byte) 0xE3; public static final byte RESERVED_E4 = (byte) 0xE4; public static final byte RESERVED_E5 = (byte) 0xE5; public static final byte RESERVED_E6 = (byte) 0xE6; public static final byte RESERVED_E7 = (byte) 0xE7; public static final byte RESERVED_E8 = (byte) 0xE8; public static final byte RESERVED_E9 = (byte) 0xE9; public static final byte RESERVED_EA = (byte) 0xEA; public static final byte RESERVED_EB = (byte) 0xEB; public static final byte RESERVED_EC = (byte) 0xEC; public static final byte RESERVED_ED = (byte) 0xED; public static final byte RESERVED_EE = (byte) 0xEE; public static final byte RESERVED_EF = (byte) 0xEF; private static final long PLUS_2_TO_THE_31 = 2147483648L; private static final long PLUS_2_TO_THE_15 = 32768L; private static final long PLUS_2_TO_THE_7 = 128L; private static final long MINUS_2_TO_THE_4 = -16L; private static final long MINUS_2_TO_THE_7 = -128L; private static final long MINUS_2_TO_THE_15 = -32768L; private static final long MINUS_2_TO_THE_31 = -2147483648L; private static final String EMPTY_STRING = ""; private static final Charset UTF_8 = Charset.forName( "UTF-8" ); private static final int DEFAULT_BUFFER_CAPACITY = 8192; private PackStream() {} public static class Packer { private PackOutput out; public Packer( PackOutput out ) { this.out = out; } public void reset( PackOutput out ) { this.out = out; } public void reset( WritableByteChannel channel ) { ((BufferedChannelOutput) out).reset( channel ); } public void flush() throws IOException { out.flush(); } public void packRaw( byte[] data ) throws IOException { out.writeBytes( data, 0, data.length ); } public void packNull() throws IOException { out.writeByte( NULL ); } public void pack( boolean value ) throws IOException { out.writeByte( value ? TRUE : FALSE ); } public void pack( long value ) throws IOException { if ( value >= MINUS_2_TO_THE_4 && value < PLUS_2_TO_THE_7) { out.writeByte( (byte) value ); } else if ( value >= MINUS_2_TO_THE_7 && value < MINUS_2_TO_THE_4 ) { out.writeByte( INT_8 ) .writeByte( (byte) value ); } else if ( value >= MINUS_2_TO_THE_15 && value < PLUS_2_TO_THE_15 ) { out.writeByte( INT_16 ) .writeShort( (short) value ); } else if ( value >= MINUS_2_TO_THE_31 && value < PLUS_2_TO_THE_31 ) { out.writeByte( INT_32 ) .writeInt( (int) value ); } else { out.writeByte( INT_64 ) .writeLong( value ); } } public void pack( double value ) throws IOException { out.writeByte( FLOAT_64 ) .writeDouble( value ); } public void pack( byte[] values ) throws IOException { if ( values == null ) { packNull(); } else { packBytesHeader( values.length ); packRaw( values ); } } public void pack( String value ) throws IOException { if ( value == null ) { packNull(); } else { byte[] utf8 = value.getBytes( UTF_8 ); packStringHeader( utf8.length ); packRaw( utf8 ); } } public void packString( byte[] utf8 ) throws IOException { if ( utf8 == null ) { packNull(); } else { packStringHeader( utf8.length ); packRaw( utf8 ); } } public void pack( List values ) throws IOException { if ( values == null ) { packNull(); } else { packListHeader( values.size() ); for ( Object value : values ) { pack( value ); } } } public void pack( Map values ) throws IOException { if ( values == null ) { packNull(); } else { packMapHeader( values.size() ); for ( Object key : values.keySet() ) { pack( key ); pack( values.get( key ) ); } } } public void pack( Object value ) throws IOException { if ( value == null ) { packNull(); } else if ( value instanceof Boolean ) { pack( (boolean) value ); } else if ( value instanceof boolean[] ) { pack( singletonList( value ) ); } else if ( value instanceof Byte ) { pack( (byte) value ); } else if ( value instanceof byte[] ) { pack( (byte[]) value ); } else if ( value instanceof Short ) { pack( (short) value ); } else if ( value instanceof short[] ) { pack( singletonList( value ) ); } else if ( value instanceof Integer ) { pack( (int) value ); } else if ( value instanceof int[] ) { pack( singletonList( value ) ); } else if ( value instanceof Long ) { pack( (long) value ); } else if ( value instanceof long[] ) { pack( singletonList( value ) ); } else if ( value instanceof Float ) { pack( (float) value ); } else if ( value instanceof float[] ) { pack( singletonList( value ) ); } else if ( value instanceof Double ) { pack( (double) value ); } else if ( value instanceof double[] ) { pack( singletonList( value ) ); } else if ( value instanceof Character ) { pack( Character.toString( (char) value ) ); } else if ( value instanceof char[] ) { pack( new String( (char[]) value ) ); } else if ( value instanceof String ) { pack( (String) value ); } else if ( value instanceof String[] ) { pack( singletonList( value ) ); } else if ( value instanceof List ) { pack( (List) value ); } else if ( value instanceof Map ) { pack( (Map) value ); } else { throw new UnPackable( format( "Cannot pack object %s", value ) );} } public void packBytesHeader( int size ) throws IOException { if ( size <= Byte.MAX_VALUE ) { out.writeByte( BYTES_8 ) .writeByte( (byte) size ); } else if ( size <= Short.MAX_VALUE ) { out.writeByte( BYTES_16 ) .writeShort( (short) size ); } else { out.writeByte( BYTES_32 ) .writeInt( size ); } } public void packStringHeader( int size ) throws IOException { if ( size < 0x10 ) { out.writeByte( (byte) (TINY_STRING | size) ); } else if ( size <= Byte.MAX_VALUE ) { out.writeByte( STRING_8 ) .writeByte( (byte) size ); } else if ( size <= Short.MAX_VALUE ) { out.writeByte( STRING_16 ) .writeShort( (short) size ); } else { out.writeByte( STRING_32 ) .writeInt( size ); } } public void packListHeader( int size ) throws IOException { if ( size < 0x10 ) { out.writeByte( (byte) (TINY_LIST | size) ); } else if ( size <= Byte.MAX_VALUE ) { out.writeByte( LIST_8 ) .writeByte( (byte) size ); } else if ( size <= Short.MAX_VALUE ) { out.writeByte( LIST_16 ) .writeShort( (short) size ); } else { out.writeByte( LIST_32 ) .writeInt( size ); } } public void packMapHeader( int size ) throws IOException { if ( size < 0x10 ) { out.writeByte( (byte) (TINY_MAP | size) ); } else if ( size <= Byte.MAX_VALUE ) { out.writeByte( MAP_8 ) .writeByte( (byte) size ); } else if ( size <= Short.MAX_VALUE ) { out.writeByte( MAP_16 ) .writeShort( (short) size ); } else { out.writeByte( MAP_32 ) .writeInt( size ); } } public void packStructHeader( int size, byte signature ) throws IOException { if ( size < 0x10 ) { out.writeByte( (byte) (TINY_STRUCT | size) ) .writeByte( signature ); } else if ( size <= Byte.MAX_VALUE ) { out.writeByte( STRUCT_8 ) .writeByte( (byte) size ) .writeByte( signature ); } else if ( size <= Short.MAX_VALUE ) { out.writeByte( STRUCT_16 ) .writeShort( (short) size ) .writeByte( signature ); } else { throw new Overflow( "Structures cannot have more than " + Short.MAX_VALUE + " fields" ); } } } public static class Unpacker { private PackInput in; public Unpacker( ReadableByteChannel channel ) { this( DEFAULT_BUFFER_CAPACITY ); reset( channel ); } public Unpacker( int bufferCapacity ) { assert bufferCapacity >= 8 : "Buffer must be at least 8 bytes."; this.in = new BufferedChannelInput( bufferCapacity ); } public Unpacker( PackInput in ) { this.in = in; } public Unpacker reset( ReadableByteChannel ch ) { ((BufferedChannelInput)in).reset( ch ); return this; } public boolean hasNext() throws IOException { return in.hasMoreData(); } public long unpackStructHeader() throws IOException { final byte markerByte = in.readByte(); final byte markerHighNibble = (byte) (markerByte & 0xF0); final byte markerLowNibble = (byte) (markerByte & 0x0F); if ( markerHighNibble == TINY_STRUCT ) { return markerLowNibble; } switch(markerByte) { case STRUCT_8: return unpackUINT8(); case STRUCT_16: return unpackUINT16(); default: throw new Unexpected( "Expected a struct, but got: " + toHexString( markerByte )); } } public byte unpackStructSignature() throws IOException { return in.readByte(); } public long unpackListHeader() throws IOException { final byte markerByte = in.readByte(); final byte markerHighNibble = (byte) (markerByte & 0xF0); final byte markerLowNibble = (byte) (markerByte & 0x0F); if ( markerHighNibble == TINY_LIST ) { return markerLowNibble; } switch(markerByte) { case LIST_8: return unpackUINT8(); case LIST_16: return unpackUINT16(); case LIST_32: return unpackUINT32(); default: throw new Unexpected( "Expected a list, but got: " + toHexString( markerByte & 0xFF )); } } public long unpackMapHeader() throws IOException { final byte markerByte = in.readByte(); final byte markerHighNibble = (byte) (markerByte & 0xF0); final byte markerLowNibble = (byte) (markerByte & 0x0F); if ( markerHighNibble == TINY_MAP ) { return markerLowNibble; } switch(markerByte) { case MAP_8: return unpackUINT8(); case MAP_16: return unpackUINT16(); case MAP_32: return unpackUINT32(); default: throw new Unexpected( "Expected a map, but got: " + toHexString( markerByte )); } } public long unpackLong() throws IOException { final byte markerByte = in.readByte(); if ( markerByte >= MINUS_2_TO_THE_4) { return markerByte; } switch(markerByte) { case INT_8: return in.readByte(); case INT_16: return in.readShort(); case INT_32: return in.readInt(); case INT_64: return in.readLong(); default: throw new Unexpected( "Expected an integer, but got: " + toHexString( markerByte )); } } public double unpackDouble() throws IOException { final byte markerByte = in.readByte(); if(markerByte == FLOAT_64) { return in.readDouble(); } throw new Unexpected( "Expected a double, but got: " + toHexString( markerByte )); } public String unpackString() throws IOException { final byte markerByte = in.readByte(); if( markerByte == TINY_STRING ) // Note no mask, so we compare to 0x80. { return EMPTY_STRING; } return new String(unpackUtf8(markerByte), UTF_8); } public byte[] unpackBytes() throws IOException { final byte markerByte = in.readByte(); switch(markerByte) { case BYTES_8: return unpackBytes( unpackUINT8() ); case BYTES_16: return unpackBytes( unpackUINT16() ); case BYTES_32: { long size = unpackUINT32(); if ( size <= Integer.MAX_VALUE ) { return unpackBytes( (int) size ); } else { throw new Overflow( "BYTES_32 too long for Java" ); } } default: throw new Unexpected( "Expected binary data, but got: 0x" + toHexString( markerByte & 0xFF )); } } /** * This may seem confusing. This method exists to move forward the internal pointer when encountering * a null value. The idiomatic usage would be someone using {@link #peekNextType()} to detect a null type, * and then this method to "skip past it". * @return null * @throws IOException if the unpacked value was not null */ public Object unpackNull() throws IOException { final byte markerByte = in.readByte(); if ( markerByte != NULL ) { throw new Unexpected( "Expected a null, but got: 0x" + toHexString( markerByte & 0xFF ) ); } return null; } private byte[] unpackUtf8(byte markerByte) throws IOException { final byte markerHighNibble = (byte) (markerByte & 0xF0); final byte markerLowNibble = (byte) (markerByte & 0x0F); if ( markerHighNibble == TINY_STRING ) { return unpackBytes( markerLowNibble ); } switch(markerByte) { case STRING_8: return unpackBytes( unpackUINT8() ); case STRING_16: return unpackBytes( unpackUINT16() ); case STRING_32: { long size = unpackUINT32(); if ( size <= Integer.MAX_VALUE ) { return unpackBytes( (int) size ); } else { throw new Overflow( "STRING_32 too long for Java" ); } } default: throw new Unexpected( "Expected a string, but got: 0x" + toHexString( markerByte & 0xFF )); } } public boolean unpackBoolean() throws IOException { final byte markerByte = in.readByte(); switch ( markerByte ) { case TRUE: return true; case FALSE: return false; default: throw new Unexpected( "Expected a boolean, but got: 0x" + toHexString( markerByte & 0xFF ) ); } } private int unpackUINT8() throws IOException { return in.readByte() & 0xFF; } private int unpackUINT16() throws IOException { return in.readShort() & 0xFFFF; } private long unpackUINT32() throws IOException { return in.readInt() & 0xFFFFFFFFL; } private byte[] unpackBytes( int size ) throws IOException { byte[] heapBuffer = new byte[size]; in.readBytes( heapBuffer, 0, heapBuffer.length ); return heapBuffer; } public PackType peekNextType() throws IOException { final byte markerByte = in.peekByte(); final byte markerHighNibble = (byte) (markerByte & 0xF0); switch(markerHighNibble) { case TINY_STRING: return PackType.STRING; case TINY_LIST: return PackType.LIST; case TINY_MAP: return PackType.MAP; case TINY_STRUCT: return PackType.STRUCT; } switch(markerByte) { case NULL: return PackType.NULL; case TRUE: case FALSE: return PackType.BOOLEAN; case FLOAT_64: return PackType.FLOAT; case BYTES_8: case BYTES_16: case BYTES_32: return PackType.BYTES; case STRING_8: case STRING_16: case STRING_32: return PackType.STRING; case LIST_8: case LIST_16: case LIST_32: return PackType.LIST; case MAP_8: case MAP_16: case MAP_32: return PackType.MAP; case STRUCT_8: case STRUCT_16: return PackType.STRUCT; default: return PackType.INTEGER; } } } public static class PackstreamException extends IOException { private static final long serialVersionUID = -1491422133282345421L; protected PackstreamException( String message ) { super( message ); } } public static class EndOfStream extends PackstreamException { private static final long serialVersionUID = 5102836237108105603L; public EndOfStream( String message ) { super( message ); } } public static class Overflow extends PackstreamException { private static final long serialVersionUID = -923071934446993659L; public Overflow( String message ) { super( message ); } } public static class Unexpected extends PackstreamException { private static final long serialVersionUID = 5004685868740125469L; public Unexpected( String message ) { super( message ); } } public static class UnPackable extends PackstreamException { private static final long serialVersionUID = 2408740707769711365L; public UnPackable( String message ) { super( message ); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy