org.neo4j.kernel.impl.util.IoPrimitiveUtils Maven / Gradle / Ivy
/*
* Copyright (c) 2018-2020 "Graph Foundation,"
* Graph Foundation, Inc. [https://graphfoundation.org]
*
* This file is part of ONgDB.
*
* ONgDB is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
/*
* Copyright (c) 2002-2020 "Neo4j,"
* Neo4j Sweden AB [http://neo4j.com]
*
* This file is part of Neo4j.
*
* Neo4j is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package org.neo4j.kernel.impl.util;
import java.io.IOException;
import java.lang.reflect.Array;
import java.nio.ByteBuffer;
import java.nio.channels.ReadableByteChannel;
import java.util.HashMap;
import java.util.Map;
import org.neo4j.io.fs.StoreChannel;
import org.neo4j.storageengine.api.ReadableChannel;
import org.neo4j.storageengine.api.WritableChannel;
import org.neo4j.string.UTF8;
public abstract class IoPrimitiveUtils
{
public static String readString( ReadableChannel channel, int length ) throws IOException
{
assert length >= 0 : "invalid array length " + length;
byte[] chars = new byte[length];
channel.get( chars, length );
return UTF8.decode( chars );
}
public static void write3bLengthAndString( WritableChannel channel, String string ) throws IOException
{
byte[] chars = UTF8.encode( string );
// 3 bytes to represent the length (4 is a bit overkill)... maybe
// this space optimization is a bit overkill also :)
channel.putShort( (short)chars.length );
channel.put( (byte)(chars.length >> 16) );
channel.put(chars, chars.length);
}
public static String read3bLengthAndString( ReadableChannel channel ) throws IOException
{
short lengthShort = channel.getShort();
byte lengthByte = channel.get();
int length = (lengthByte << 16) | (lengthShort & 0xFFFF);
byte[] chars = new byte[length];
channel.get( chars, length );
return UTF8.decode( chars );
}
public static void write2bLengthAndString( WritableChannel channel, String string ) throws IOException
{
byte[] chars = UTF8.encode( string );
channel.putShort( (short)chars.length );
channel.put(chars, chars.length);
}
public static String read2bLengthAndString( ReadableChannel channel ) throws IOException
{
short length = channel.getShort();
return readString( channel, length );
}
private static char[] readCharArray( ReadableByteChannel channel,
ByteBuffer buffer, char[] charArray ) throws IOException
{
buffer.clear();
int charsLeft = charArray.length;
int maxSize = buffer.capacity() / 2;
int offset = 0; // offset in chars
while ( charsLeft > 0 )
{
if ( charsLeft > maxSize )
{
buffer.limit( maxSize * 2 );
charsLeft -= maxSize;
}
else
{
buffer.limit( charsLeft * 2 );
charsLeft = 0;
}
if ( channel.read( buffer ) != buffer.limit() )
{
return null;
}
buffer.flip();
int length = buffer.limit() / 2;
buffer.asCharBuffer().get( charArray, offset, length );
offset += length;
buffer.clear();
}
return charArray;
}
public static boolean readAndFlip( ReadableByteChannel channel, ByteBuffer buffer, int bytes )
throws IOException
{
buffer.clear();
buffer.limit( bytes );
while ( buffer.hasRemaining() )
{
int read = channel.read( buffer );
if ( read == -1 )
{
return false;
}
}
buffer.flip();
return true;
}
public static Integer readInt( ReadableByteChannel channel, ByteBuffer buffer ) throws IOException
{
return readAndFlip( channel, buffer, 4 ) ? buffer.getInt() : null;
}
public static byte[] readBytes( ReadableByteChannel channel, byte[] array ) throws IOException
{
return readBytes( channel, array, array.length );
}
public static byte[] readBytes( ReadableByteChannel channel, byte[] array, int length ) throws IOException
{
return readAndFlip( channel, ByteBuffer.wrap( array ), length ) ? array : null;
}
public static Map read2bMap( ReadableChannel channel ) throws IOException
{
short size = channel.getShort();
Map map = new HashMap<>( size );
for ( int i = 0; i < size; i++ )
{
String key = read2bLengthAndString( channel );
String value = read2bLengthAndString( channel );
map.put( key, value );
}
return map;
}
public static String readLengthAndString( ReadableByteChannel channel,
ByteBuffer buffer ) throws IOException
{
Integer length = readInt( channel, buffer );
if ( length != null )
{
char[] chars = new char[length];
chars = readCharArray( channel, buffer, chars );
return chars == null ? null : new String( chars );
}
return null;
}
public static void writeLengthAndString( StoreChannel channel, ByteBuffer buffer, String value )
throws IOException
{
char[] chars = value.toCharArray();
int length = chars.length;
writeInt( channel, buffer, length );
writeChars( channel, buffer, chars );
}
private static void writeChars( StoreChannel channel, ByteBuffer buffer, char[] chars )
throws IOException
{
int position = 0;
do
{
buffer.clear();
int leftToWrite = chars.length - position;
if ( leftToWrite * 2 < buffer.capacity() )
{
buffer.asCharBuffer().put( chars, position, leftToWrite );
buffer.limit( leftToWrite * 2);
channel.write( buffer );
position += leftToWrite;
}
else
{
int length = buffer.capacity() / 2;
buffer.asCharBuffer().put( chars, position, length );
buffer.limit( length * 2 );
channel.write( buffer );
position += length;
}
} while ( position < chars.length );
}
public static void writeInt( StoreChannel channel, ByteBuffer buffer, int value )
throws IOException
{
buffer.clear();
buffer.putInt( value );
buffer.flip();
channel.write( buffer );
}
public static Object[] asArray( Object propertyValue )
{
if ( propertyValue.getClass().isArray() )
{
int length = Array.getLength( propertyValue );
Object[] result = new Object[ length ];
for ( int i = 0; i < length; i++ )
{
result[ i ] = Array.get( propertyValue, i );
}
return result;
}
return new Object[] { propertyValue };
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy