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

org.infinispan.commons.marshall.MarshallUtil Maven / Gradle / Ivy

package org.infinispan.commons.marshall;

import net.jcip.annotations.Immutable;
import org.infinispan.commons.util.Util;

import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.util.Collection;
import java.util.Map;
import java.util.Objects;
import java.util.UUID;

/**
 * MarshallUtil.
 *
 * @author Galder Zamarreño
 * @since 4.0
 */
@Immutable
public class MarshallUtil {

   private static final byte NULL_VALUE = -1;

   /**
    * Marshall the {@code map} to the {@code ObjectOutput}.
    * 

* {@code null} maps are supported. * * @param map {@link Map} to marshall. * @param out {@link ObjectOutput} to write. It must be non-null. * @param Key type of the map. * @param Value type of the map. * @param Type of the {@link Map}. * @throws IOException If any of the usual Input/Output related exceptions occur. */ public static > void marshallMap(T map, ObjectOutput out) throws IOException { final int mapSize = map == null ? NULL_VALUE : map.size(); marshallInt(out, mapSize); if (mapSize <= 0) return; for (Map.Entry me : map.entrySet()) { out.writeObject(me.getKey()); out.writeObject(me.getValue()); } } /** * Unmarshall the {@link Map}. *

* If the marshalled map is {@link null}, then the {@link MapBuilder} is not invoked. * * @param in {@link ObjectInput} to read. * @param builder {@link MapBuilder} to create the concrete {@link Map} implementation. * @return The populated {@link Map} created by the {@link MapBuilder} or {@code null}. * @throws IOException If any of the usual Input/Output related exceptions occur. * @throws ClassNotFoundException If the class of a serialized object cannot be found. * @see {@link #marshallMap(Map, ObjectOutput)} */ public static > T unmarshallMap(ObjectInput in, MapBuilder builder) throws IOException, ClassNotFoundException { final int size = unmarshallInt(in); if (size == NULL_VALUE) { return null; } final T map = Objects.requireNonNull(builder, "MapBuilder must be non-null").build(size); for (int i = 0; i < size; i++) //noinspection unchecked map.put((K) in.readObject(), (V) in.readObject()); return map; } /** * Marshall the {@link UUID} by sending the most and lest significant bits. *

* This method supports {@code null} if {@code checkNull} is set to {@link true}. * * @param uuid {@link UUID} to marshall. * @param out {@link ObjectOutput} to write. * @param checkNull If {@code true}, it checks if {@code uuid} is {@code null}. * @throws IOException If any of the usual Input/Output related exceptions occur. */ public static void marshallUUID(UUID uuid, ObjectOutput out, boolean checkNull) throws IOException { if (checkNull) { if (uuid == null) { out.writeBoolean(true); return; } out.writeBoolean(false); } out.writeLong(uuid.getMostSignificantBits()); out.writeLong(uuid.getLeastSignificantBits()); } /** * Unmarshall {@link UUID}. * * @param in {@link ObjectInput} to read. * @param checkNull If {@code true}, it checks if the {@link UUID} marshalled was {@link null}. * @return {@link UUID} marshalled. * @throws IOException If any of the usual Input/Output related exceptions occur. * @see {@link #marshallUUID(UUID, ObjectOutput, boolean)}. */ public static UUID unmarshallUUID(ObjectInput in, boolean checkNull) throws IOException { if (checkNull && in.readBoolean()) { return null; } return new UUID(in.readLong(), in.readLong()); } /** * Marshall arrays. *

* This method supports {@code null} {@code array}. * * @param array Array to marshall. * @param out {@link ObjectOutput} to write. * @param Array type. * @throws IOException If any of the usual Input/Output related exceptions occur. */ public static void marshallArray(E[] array, ObjectOutput out) throws IOException { final int size = array == null ? NULL_VALUE : array.length; marshallInt(out, size); if (size <= 0) { return; } for (int i = 0; i < size; ++i) { out.writeObject(array[i]); } } /** * Unmarshall arrays. * * @param in {@link ObjectInput} to read. * @param builder {@link ArrayBuilder} to build the array. * @param Array type. * @return The populated array. * @throws IOException If any of the usual Input/Output related exceptions occur. * @throws ClassNotFoundException If the class of a serialized object cannot be found. * @see {@link #marshallArray(Object[], ObjectOutput)}. */ public static E[] unmarshallArray(ObjectInput in, ArrayBuilder builder) throws IOException, ClassNotFoundException { final int size = unmarshallInt(in); if (size == NULL_VALUE) { return null; } else if (size == 0) { //noinspection unchecked return (E[]) Util.EMPTY_OBJECT_ARRAY; } final E[] array = Objects.requireNonNull(builder, "ArrayBuilder must be non-null").build(size); for (int i = 0; i < size; ++i) { //noinspection unchecked array[i] = (E) in.readObject(); } return array; } /** * Marshall a {@link Collection}. *

* This method supports {@code null} {@code collection}. * * @param collection {@link Collection} to marshal. * @param out {@link ObjectOutput} to write. * @param Collection's element type. * @throws IOException If any of the usual Input/Output related exceptions occur. */ public static void marshallCollection(Collection collection, ObjectOutput out) throws IOException { final int size = collection == null ? NULL_VALUE : collection.size(); marshallInt(out, size); if (size <= 0) { return; } for (E e : collection) { out.writeObject(e); } } /** * Unmarshal a {@link Collection}. * * @param in {@link ObjectInput} to read. * @param builder {@link CollectionBuilder} builds the concrete {@link Collection} based on size. * @param Collection's element type. * @param {@link Collection} implementation. * @return The concrete {@link Collection} implementation. * @throws IOException If any of the usual Input/Output related exceptions occur. * @throws ClassNotFoundException If the class of a serialized object cannot be found. */ public static > T unmarshallCollection(ObjectInput in, CollectionBuilder builder) throws IOException, ClassNotFoundException { final int size = unmarshallInt(in); if (size == NULL_VALUE) { return null; } T collection = Objects.requireNonNull(builder, "CollectionBuilder must be non-null").build(size); for (int i = 0; i < size; ++i) { //noinspection unchecked collection.add((E) in.readObject()); } return collection; } /** * Same as {@link #unmarshallCollection(ObjectInput, CollectionBuilder)}. *

* Used when the size of the {@link Collection} is not needed for it construction. * * @see {@link #unmarshallCollection(ObjectInput, CollectionBuilder)}. */ public static > T unmarshallCollectionUnbounded(ObjectInput in, UnboundedCollectionBuilder builder) throws IOException, ClassNotFoundException { final int size = unmarshallInt(in); if (size == NULL_VALUE) { return null; } T collection = Objects.requireNonNull(builder, "UnboundedCollectionBuilder must be non-null").build(); for (int i = 0; i < size; ++i) { //noinspection unchecked collection.add((E) in.readObject()); } return collection; } /** * Marshall the {@link String}. *

* Same behavior as {@link ObjectOutput#writeUTF(String)} but it checks for {@code null}. If the {@code string} is * never {@code null}, it is better to use {@link ObjectOutput#writeUTF(String)}. * * @param string {@link String} to marshall. * @param out {@link ObjectOutput} to write. * @throws IOException If any of the usual Input/Output related exceptions occur. */ public static void marshallString(String string, ObjectOutput out) throws IOException { if (string == null) { out.writeBoolean(true); return; } out.writeBoolean(false); out.writeUTF(string); } /** * Unmarshall a {@link String}. * * @param in {@link ObjectInput} to read. * @return The {@link String} or {@code null}. * @throws IOException If any of the usual Input/Output related exceptions occur. * @see {@link #marshallString(String, ObjectOutput)}. */ public static String unmarshallString(ObjectInput in) throws IOException { if (in.readBoolean()) { return null; } return in.readUTF(); } /** * Same as {@link #marshallArray(Object[], ObjectOutput)} but specialized for byte arrays. * * @see {@link #marshallArray(Object[], ObjectOutput)}. */ public static void marshallByteArray(byte[] array, ObjectOutput out) throws IOException { final int size = array == null ? NULL_VALUE : array.length; marshallInt(out, size); if (size <= 0) { return; } out.write(array); } /** * Same as {@link #unmarshallArray(ObjectInput, ArrayBuilder)} but specialzed for byte array. *

* No {@link ArrayBuilder} is necessary. * * @see {@link #unmarshallArray(ObjectInput, ArrayBuilder)}. */ public static byte[] unmarshallByteArray(ObjectInput in) throws IOException { final int size = unmarshallInt(in); if (size == NULL_VALUE) { return null; } else if (size == 0) { return Util.EMPTY_BYTE_ARRAY; } byte[] array = new byte[size]; in.readFully(array); return array; } /** * A special marshall implementation for integer. *

* This method supports negative values but they are handles as {@link #NULL_VALUE}. It means that the real value is * lost and {@link #NULL_VALUE} is returned by {@link #unmarshallInt(ObjectInput)}. *

* The integer is marshalled in a variable length from 1 to 5 bytes. Negatives values are always marshalled in 1 * byte. * * @param out {@link ObjectOutput} to write. * @param value Integer value to marshall. * @throws IOException If any of the usual Input/Output related exceptions occur. */ public static void marshallInt(ObjectOutput out, int value) throws IOException { if (value < 0) { out.writeByte(0x80); //meaning it is a negative value! return; } if ((value & ~0x3F) == 0) { // fits in 1 byte out.writeByte(value & 0x3F); //first bit is 0 (== positive) and second bit is zero (== not more bytes) return; } out.writeByte((value & 0x3F) | 0x40); //set second bit to 1 (== more bytes) value >>>= 6; //6 bits written so far //normal code for unsigned int. only the first byte is special while ((value & ~0x7F) != 0) { out.writeByte((byte) ((value & 0x7f) | 0x80)); value >>>= 7; } out.writeByte((byte) value); } /** * Unmarshall an integer. * * @param in {@link ObjectInput} to read. * @return The integer value or {@link #NULL_VALUE} if the original value was negative. * @throws IOException If any of the usual Input/Output related exceptions occur. * @see {@link #marshallInt(ObjectOutput, int)}. */ public static int unmarshallInt(ObjectInput in) throws IOException { byte b = in.readByte(); if ((b & 0x80) != 0) { return NULL_VALUE; //negative value } int i = b & 0x3F; if ((b & 0x40) == 0) { return i; } int shift = 6; do { b = in.readByte(); i |= (b & 0x7F) << shift; shift += 7; } while ((b & 0x80) != 0); return i; } public static > void marshallEnum(E e, ObjectOutput output) throws IOException { if (e == null) { output.writeByte(NULL_VALUE); } else { output.writeByte(e.ordinal()); } } public static > E unmarshallEnum(ObjectInput input, EnumBuilder builder) throws IOException { final byte ordinal = input.readByte(); if (ordinal == NULL_VALUE) { return null; } try { return Objects.requireNonNull(builder).build(ordinal); } catch (ArrayIndexOutOfBoundsException e) { throw new IOException("Unknown enum.", e); } } public interface ArrayBuilder { E[] build(int size); } public interface CollectionBuilder> { T build(int size); } public interface UnboundedCollectionBuilder> { T build(); } public interface MapBuilder> { T build(int size); } public interface EnumBuilder> { E build(int ordinal); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy