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

com.apple.foundationdb.tuple.TupleUtil Maven / Gradle / Ivy

/*
 * TupleUtil.java
 *
 * This source file is part of the FoundationDB open source project
 *
 * Copyright 2013-2018 Apple Inc. and the FoundationDB project authors
 *
 * 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.apple.foundationdb.tuple;

import java.math.BigInteger;
import java.nio.BufferOverflowException;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.charset.CharacterCodingException;
import java.nio.charset.Charset;
import java.nio.charset.CharsetDecoder;
import java.nio.charset.CodingErrorAction;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.UUID;
import java.util.stream.Stream;

import com.apple.foundationdb.FDB;

class TupleUtil {
	private static final byte nil = 0x00;
	private static final Charset UTF8 = StandardCharsets.UTF_8;
	private static final BigInteger LONG_MIN_VALUE = BigInteger.valueOf(Long.MIN_VALUE);
	private static final BigInteger LONG_MAX_VALUE = BigInteger.valueOf(Long.MAX_VALUE);
	private static final int UUID_BYTES = 2 * Long.BYTES;
	private static final IterableComparator iterableComparator = new IterableComparator();

	private static final byte BYTES_CODE            = 0x01;
	private static final byte STRING_CODE           = 0x02;
	private static final byte NESTED_CODE           = 0x05;
	private static final byte INT_ZERO_CODE         = 0x14;
	private static final byte POS_INT_END           = 0x1d;
	private static final byte NEG_INT_START         = 0x0b;
	private static final byte FLOAT_CODE            = 0x20;
	private static final byte DOUBLE_CODE           = 0x21;
	private static final byte FALSE_CODE            = 0x26;
	private static final byte TRUE_CODE             = 0x27;
	private static final byte UUID_CODE             = 0x30;
	private static final byte VERSIONSTAMP_CODE     = 0x33;

	private static final byte[] NULL_ARR           = new byte[] {nil};
	private static final byte[] NULL_ESCAPED_ARR   = new byte[] {nil, (byte)0xFF};

	static class DecodeState {
		final List values;
		int end;
		int nullCount; // Basically a hack to allow findTerminator to return the terminator and null count

		DecodeState() {
			values = new ArrayList<>();
			end = 0;
		}

		void add(Object value, int end) {
			values.add(value);
			this.end = end;
		}

		int findNullTerminator(byte[] bytes, int from, int to) {
			nullCount = 0;
			int x = from;
			while(x < to) {
				if(bytes[x] == 0x00) {
					if(x + 1 >= to || bytes[x + 1] != (byte)0xFF) {
						return x;
					}
					else {
						nullCount++;
						x += 2;
					}
				}
				else {
					x += 1;
				}
			}
			throw new IllegalArgumentException("No terminator found for bytes starting at " + from);
		}
	}

	static class EncodeState {
		final ByteBuffer encodedBytes;
		int totalLength;
		int versionPos;

		EncodeState(ByteBuffer dest) {
			encodedBytes = dest;
			encodedBytes.order(ByteOrder.BIG_ENDIAN);
			totalLength = 0;
			versionPos = -1;
		}

		EncodeState add(byte[] encoded, int versionPos) {
			if(versionPos >= 0 && this.versionPos >= 0) {
				throw new IllegalArgumentException("Multiple incomplete Versionstamps included in Tuple");
			}
			encodedBytes.put(encoded);
			totalLength += encoded.length;
			this.versionPos = versionPos;
			return this;
		}

		EncodeState add(byte[] encoded) {
			encodedBytes.put(encoded);
			totalLength += encoded.length;
			return this;
		}

		EncodeState add(byte[] encoded, int offset, int length) {
			encodedBytes.put(encoded, offset, length);
			totalLength += length;
			return this;
		}

		EncodeState addNullEscaped(byte[] encoded) {
			int nullCount = ByteArrayUtil.nullCount(encoded);
			if(nullCount == 0) {
				encodedBytes.put(encoded);
			}
			else {
				ByteArrayUtil.replace(encoded, 0, encoded.length, NULL_ARR, NULL_ESCAPED_ARR, encodedBytes);
			}
			totalLength += encoded.length + nullCount;
			return this;
		}

		EncodeState add(byte b) {
			encodedBytes.put(b);
			totalLength++;
			return this;
		}

		EncodeState add(int i) {
			encodedBytes.putInt(i);
			totalLength += Integer.BYTES;
			return this;
		}

		EncodeState add(long l) {
			encodedBytes.putLong(l);
			totalLength += Long.BYTES;
			return this;
		}
	}

	private static boolean useOldVersionOffsetFormat() {
		return FDB.instance().getAPIVersion() < 520;
	}

	// These four functions are for adjusting the encoding of floating point numbers so
	// that when their byte representation is written out in big-endian order, unsigned
	// lexicographic byte comparison orders the values in the same way as the semantic
	// ordering of the values. This means flipping all bits for negative values and flipping
	// only the most-significant bit (i.e., the sign bit as all values in Java are signed)
	// in the case that the number is positive. For these purposes, 0.0 is positive and -0.0
	// is negative.

	private static int encodeFloatBits(float f) {
		int intBits = Float.floatToRawIntBits(f);
		return (intBits < 0) ? (~intBits) : (intBits ^ Integer.MIN_VALUE);
	}

	private static long encodeDoubleBits(double d) {
		long longBits = Double.doubleToRawLongBits(d);
		return (longBits < 0L) ? (~longBits) : (longBits ^ Long.MIN_VALUE);
	}

	private static float decodeFloatBits(int i) {
		int origBits = (i >= 0) ? (~i) : (i ^ Integer.MIN_VALUE);
		return Float.intBitsToFloat(origBits);
	}

	private static double decodeDoubleBits(long l) {
		long origBits = (l >= 0) ? (~l) : (l ^ Long.MIN_VALUE);
		return Double.longBitsToDouble(origBits);
	}

	// Get the minimal number of bytes in the representation of a long.
	private static int minimalByteCount(long i) {
		return (Long.SIZE + 7 - Long.numberOfLeadingZeros(i >= 0 ? i : -i)) / 8;
	}

	private static int minimalByteCount(BigInteger i) {
		int bitLength = (i.compareTo(BigInteger.ZERO) >= 0) ? i.bitLength() : i.negate().bitLength();
		return (bitLength + 7) / 8;
	}

	private static void adjustVersionPosition300(byte[] packed, int delta) {
		int offsetOffset = packed.length - Short.BYTES;
		ByteBuffer buffer = ByteBuffer.wrap(packed, offsetOffset, Short.BYTES).order(ByteOrder.LITTLE_ENDIAN);
		int versionPosition = buffer.getShort() + delta;
		if(versionPosition > 0xffff) {
			throw new IllegalArgumentException("Tuple has incomplete version at position " + versionPosition + " which is greater than the maximum " + 0xffff);
		}
		if(versionPosition < 0) {
			throw new IllegalArgumentException("Tuple has an incomplete version at a negative position");
		}
		buffer.position(offsetOffset);
		buffer.putShort((short)versionPosition);
	}

	private static void adjustVersionPosition520(byte[] packed, int delta) {
		int offsetOffset = packed.length - Integer.BYTES;
		ByteBuffer buffer = ByteBuffer.wrap(packed, offsetOffset, Integer.BYTES).order(ByteOrder.LITTLE_ENDIAN);
		int versionPosition = buffer.getInt() + delta;
		if(versionPosition < 0) {
			throw new IllegalArgumentException("Tuple has an incomplete version at a negative position");
		}
		buffer.position(offsetOffset);
		buffer.putInt(versionPosition);
	}

	static void adjustVersionPosition(byte[] packed, int delta) {
		if(useOldVersionOffsetFormat()) {
			adjustVersionPosition300(packed, delta);
		}
		else {
			adjustVersionPosition520(packed, delta);
		}
	}

	static int getCodeFor(Object o) {
		if(o == null)
			return nil;
		if(o instanceof byte[])
			return BYTES_CODE;
		if(o instanceof String)
			return STRING_CODE;
		if(o instanceof Float)
			return FLOAT_CODE;
		if(o instanceof Double)
			return DOUBLE_CODE;
		if(o instanceof Boolean)
			return FALSE_CODE;
		if(o instanceof UUID)
			return UUID_CODE;
		if(o instanceof Number)
			return INT_ZERO_CODE;
		if(o instanceof Versionstamp)
			return VERSIONSTAMP_CODE;
		if(o instanceof List)
			return NESTED_CODE;
		if(o instanceof Tuple)
			return NESTED_CODE;
		throw new IllegalArgumentException("Unsupported data type: " + o.getClass().getName());
	}

	static void encode(EncodeState state, Object t, boolean nested) {
		if(t == null) {
			if(nested) {
				state.add(NULL_ESCAPED_ARR);
			}
			else {
				state.add(nil);
			}
		}
		else if(t instanceof byte[])
			encode(state, (byte[]) t);
		else if(t instanceof String)
			encode(state, (String)t);
		else if(t instanceof Float)
			encode(state, (Float)t);
		else if(t instanceof Double)
			encode(state, (Double)t);
		else if(t instanceof Boolean)
			encode(state, (Boolean)t);
		else if(t instanceof UUID)
			encode(state, (UUID)t);
		else if(t instanceof BigInteger)
			encode(state, (BigInteger)t);
		else if(t instanceof Number)
			encode(state, ((Number)t).longValue());
		else if(t instanceof Versionstamp)
			encode(state, (Versionstamp)t);
		else if(t instanceof List)
			encode(state, (List)t);
		else if(t instanceof Tuple)
			encode(state, (Tuple)t);
		else
			throw new IllegalArgumentException("Unsupported data type: " + t.getClass().getName());
	}

	static void encode(EncodeState state, Object t) {
		encode(state, t, false);
	}

	static void encode(EncodeState state, byte[] bytes) {
		state.add(BYTES_CODE).addNullEscaped(bytes).add(nil);
	}

	static void encode(EncodeState state, String s) {
		StringUtil.validate(s);
		byte[] bytes = s.getBytes(UTF8);
		state.add(STRING_CODE).addNullEscaped(bytes).add(nil);
	}

	static void encode(EncodeState state, BigInteger i) {
		//System.out.println("Encoding integral " + i);
		if(i.equals(BigInteger.ZERO)) {
			state.add(INT_ZERO_CODE);
			return;
		}
		int n = minimalByteCount(i);
		if(n > 0xff) {
			throw new IllegalArgumentException("BigInteger magnitude is too large (more than 255 bytes)");
		}
		if(i.compareTo(BigInteger.ZERO) > 0) {
			byte[] bytes = i.toByteArray();
			if(n > Long.BYTES) {
				state.add(POS_INT_END);
				state.add((byte)n);
				state.add(bytes, bytes.length - n, n);
			}
			else {
				//System.out.println("  -- integral has 'n' of " + n + " and output bytes of " + bytes.length);
				state.add((byte)(INT_ZERO_CODE + n));
				state.add(bytes, bytes.length - n, n);
			}
		}
		else {
			byte[] bytes = i.subtract(BigInteger.ONE).toByteArray();
			if(n > Long.BYTES) {
				state.add(NEG_INT_START);
				state.add((byte)(n ^ 0xff));
				if(bytes.length >= n) {
					state.add(bytes, bytes.length - n, n);
				}
				else {
					for(int x = 0; x < n - bytes.length; x++) {
						state.add((byte)0x00);
					}
					state.add(bytes, 0, bytes.length);
				}
			}
			else {
				state.add((byte)(INT_ZERO_CODE - n));
				if(bytes.length >= n) {
					state.add(bytes, bytes.length - n, n);
				}
				else {
					for(int x = 0; x < n - bytes.length; x++) {
						state.add((byte)0x00);
					}
					state.add(bytes, 0, bytes.length);
				}
			}
		}
	}

	static void encode(EncodeState state, long i) {
		if(i == 0L) {
			state.add(INT_ZERO_CODE);
			return;
		}
		int n = minimalByteCount(i);
		// First byte encodes number of bytes (as difference from INT_ZERO_CODE)
		state.add((byte)(INT_ZERO_CODE + (i >= 0 ? n : -n)));
		// For positive integers, copy the bytes in big-endian order excluding leading 0x00 bytes.
		// For negative integers, copy the bytes of the one's complement representation excluding
		// the leading 0xff bytes. As Java stores negative values in two's complement, we subtract 1
		// from negative values.
		long val = Long.reverseBytes((i >= 0) ? i : (i - 1)) >> (Long.SIZE - 8 * n);
		for(int x = 0; x < n; x++) {
			state.add((byte)(val & 0xff));
			val >>= 8;
		}
	}

	static void encode(EncodeState state, Float f) {
		state.add(FLOAT_CODE).add(encodeFloatBits(f));
	}

	static void encode(EncodeState state, Double d) {
		state.add(DOUBLE_CODE).add(encodeDoubleBits(d));
	}

	static void encode(EncodeState state, Boolean b) {
		state.add(b ? TRUE_CODE : FALSE_CODE);
	}

	static void encode(EncodeState state, UUID uuid) {
		state.add(UUID_CODE).add(uuid.getMostSignificantBits()).add(uuid.getLeastSignificantBits());
	}

	static void encode(EncodeState state, Versionstamp v) {
		state.add(VERSIONSTAMP_CODE);
		if(v.isComplete()) {
			state.add(v.getBytes());
		}
		else {
			state.add(v.getBytes(), state.totalLength);
		}
	}

	static void encode(EncodeState state, List value) {
		state.add(NESTED_CODE);
		for(Object t : value) {
			encode(state, t, true);
		}
		state.add(nil);
	}

	static void encode(EncodeState state, Tuple value) {
		encode(state, value.elements);
	}

	static void decode(DecodeState state, byte[] rep, int pos, int last) {
		//System.out.println("Decoding '" + ArrayUtils.printable(rep) + "' at " + pos);

		// SOMEDAY: codes over 127 will be a problem with the signed Java byte mess
		int code = rep[pos];
		int start = pos + 1;
		if(code == nil) {
			state.add(null, start);
		}
		else if(code == BYTES_CODE) {
			int end = state.findNullTerminator(rep, start, last);
			//System.out.println("End of byte string: " + end);
			byte[] range;
			if(state.nullCount == 0) {
				range = Arrays.copyOfRange(rep, start, end);
			}
			else {
				ByteBuffer dest = ByteBuffer.allocate(end - start - state.nullCount);
				ByteArrayUtil.replace(rep, start, end - start, NULL_ESCAPED_ARR, NULL_ARR, dest);
				range = dest.array();
			}
			//System.out.println(" -> byte string contents: '" + ArrayUtils.printable(range) + "'");
			state.add(range, end + 1);
		}
		else if(code == STRING_CODE) {
			int end = state.findNullTerminator(rep, start, last);
			//System.out.println("End of UTF8 string: " + end);
			String str;
			ByteBuffer byteBuffer;
			if(state.nullCount == 0) {
				byteBuffer = ByteBuffer.wrap(rep, start, end - start);
			}
			else {
				byteBuffer = ByteBuffer.allocate(end - start - state.nullCount);
				ByteArrayUtil.replace(rep, start, end - start, NULL_ESCAPED_ARR, NULL_ARR, byteBuffer);
				byteBuffer.position(0);
			}
			try {
				CharsetDecoder decoder = UTF8.newDecoder().onMalformedInput(CodingErrorAction.REPORT);
				str = decoder.decode(byteBuffer).toString();
			}
			catch(CharacterCodingException e) {
				throw new IllegalArgumentException("malformed UTF-8 string", e);
			}
			//System.out.println(" -> UTF8 string contents: '" + str + "'");
			state.add(str, end + 1);
		}
		else if(code == FLOAT_CODE) {
			int rawFloatBits = ByteBuffer.wrap(rep, start, Float.BYTES).getInt();
			float res = decodeFloatBits(rawFloatBits);
			state.add(res, start + Float.BYTES);
		}
		else if(code == DOUBLE_CODE) {
			long rawDoubleBits = ByteBuffer.wrap(rep, start, Double.BYTES).getLong();
			double res = decodeDoubleBits(rawDoubleBits);
			state.add(res, start + Double.BYTES);
		}
		else if(code == FALSE_CODE) {
			state.add(false, start);
		}
		else if(code == TRUE_CODE) {
			state.add(true, start);
		}
		else if(code == UUID_CODE) {
			ByteBuffer bb = ByteBuffer.wrap(rep, start, UUID_BYTES).order(ByteOrder.BIG_ENDIAN);
			long msb = bb.getLong();
			long lsb = bb.getLong();
			state.add(new UUID(msb, lsb), start + UUID_BYTES);
		}
		else if(code == POS_INT_END) {
			int n = rep[start] & 0xff;
			byte[] intBytes = new byte[n + 1];
			System.arraycopy(rep, start + 1, intBytes, 1, n);
			BigInteger res = new BigInteger(intBytes);
			state.add(res, start + n + 1);
		}
		else if(code == NEG_INT_START) {
			int n = (rep[start] ^ 0xff) & 0xff;
			byte[] intBytes = new byte[n + 1];
			System.arraycopy(rep, start + 1, intBytes, 1, n);
			BigInteger origValue = new BigInteger(intBytes);
			BigInteger offset = BigInteger.ONE.shiftLeft(n*8).subtract(BigInteger.ONE);
			state.add(origValue.subtract(offset), start + n + 1);
		}
		else if(code > NEG_INT_START && code < POS_INT_END) {
			// decode a long
			boolean positive = code >= INT_ZERO_CODE;
			int n = positive ? code - INT_ZERO_CODE : INT_ZERO_CODE - code;
			int end = start + n;

			if(last < end) {
				throw new IllegalArgumentException("Invalid tuple (possible truncation)");
			}

			if(positive && (n < Long.BYTES || rep[start] > 0)) {
				long res = 0L;
				for(int i = start; i < end; i++) {
					res = (res << 8) | (rep[i] & 0xff);
				}
				state.add(res, end);
			}
			else if(!positive && (n < Long.BYTES || rep[start] < 0)) {
				long res = ~0L;
				for(int i = start; i < end; i++) {
					res = (res << 8) | (rep[i] & 0xff);
				}
				state.add(res + 1, end);
			}
			else {
				byte[] longBytes = new byte[9];
				System.arraycopy(rep, start, longBytes, longBytes.length-n, n);
				if (!positive)
					for(int i=longBytes.length-n; i= 0 && val.compareTo(LONG_MAX_VALUE) <= 0) {
					state.add(val.longValue(), end);
				} else {
					// This can occur if the thing can be represented with 8 bytes but requires using
					// the most-significant bit as a normal bit instead of the sign bit.
					state.add(val, end);
				}
			}
		}
		else if(code == VERSIONSTAMP_CODE) {
			if(start + Versionstamp.LENGTH > last) {
				throw new IllegalArgumentException("Invalid tuple (possible truncation)");
			}
			Versionstamp val = Versionstamp.fromBytes(Arrays.copyOfRange(rep, start, start + Versionstamp.LENGTH));
			state.add(val, start + Versionstamp.LENGTH);
		}
		else if(code == NESTED_CODE) {
			DecodeState subResult = new DecodeState();
			int endPos = start;
			boolean foundEnd = false;
			while(endPos < last) {
				if(rep[endPos] == nil) {
					if(endPos + 1 < last && rep[endPos+1] == (byte)0xff) {
						subResult.add(null, endPos + 2);
						endPos += 2;
					} else {
						endPos += 1;
						foundEnd = true;
						break;
					}
				} else {
					decode(subResult, rep, endPos, last);
					endPos = subResult.end;
				}
			}
			if(!foundEnd) {
				throw new IllegalArgumentException("No terminator found for nested tuple starting at " + start);
			}
			state.add(subResult.values, endPos);
		}
		else {
			throw new IllegalArgumentException("Unknown tuple data type " + code + " at index " + pos);
		}
	}

	static int compareItems(Object item1, Object item2) {
		if(item1 == item2) {
			// If we have pointer equality, just return 0 immediately.
			return 0;
		}
		int code1 = TupleUtil.getCodeFor(item1);
		int code2 = TupleUtil.getCodeFor(item2);

		if(code1 != code2) {
			return Integer.compare(code1, code2);
		}

		if(code1 == nil) {
			// All null's are equal. (Some may be more equal than others.)
			return 0;
		}
		if(code1 == BYTES_CODE) {
			return ByteArrayUtil.compareUnsigned((byte[])item1, (byte[])item2);
		}
		if(code1 == STRING_CODE) {
			return StringUtil.compareUtf8((String)item1, (String)item2);
		}
		if(code1 == INT_ZERO_CODE) {
			if(item1 instanceof Long && item2 instanceof Long) {
				// This should be the common case, so it's probably worth including as a way out.
				return Long.compare((Long)item1, (Long)item2);
			}
			else {
				BigInteger bi1;
				if (item1 instanceof BigInteger) {
					bi1 = (BigInteger) item1;
				} else {
					bi1 = BigInteger.valueOf(((Number) item1).longValue());
				}
				BigInteger bi2;
				if (item2 instanceof BigInteger) {
					bi2 = (BigInteger) item2;
				} else {
					bi2 = BigInteger.valueOf(((Number) item2).longValue());
				}
				return bi1.compareTo(bi2);
			}
		}
		if(code1 == FLOAT_CODE) {
			// This is done over vanilla float comparison basically to handle NaNs
			// sorting correctly.
			int fbits1 = encodeFloatBits((Float)item1);
			int fbits2 = encodeFloatBits((Float)item2);
			return Integer.compareUnsigned(fbits1, fbits2);
		}
		if(code1 == DOUBLE_CODE) {
			// This is done over vanilla double comparison basically to handle NaNs
			// sorting correctly.
			long dbits1 = encodeDoubleBits((Double)item1);
			long dbits2 = encodeDoubleBits((Double)item2);
			return Long.compareUnsigned(dbits1, dbits2);
		}
		if(code1 == FALSE_CODE) {
			return Boolean.compare((Boolean)item1, (Boolean)item2);
		}
		if(code1 == UUID_CODE) {
			// Java UUID.compareTo is signed, so we have to used the unsigned methods.
			UUID uuid1 = (UUID)item1;
			UUID uuid2 = (UUID)item2;
			int cmp1 = Long.compareUnsigned(uuid1.getMostSignificantBits(), uuid2.getMostSignificantBits());
			if(cmp1 != 0)
				return cmp1;
			return Long.compareUnsigned(uuid1.getLeastSignificantBits(), uuid2.getLeastSignificantBits());
		}
		if(code1 == VERSIONSTAMP_CODE) {
			return ((Versionstamp)item1).compareTo((Versionstamp)item2);
		}
		if(code1 == NESTED_CODE) {
			return iterableComparator.compare((Iterable)item1, (Iterable)item2);
		}
		throw new IllegalArgumentException("Unknown tuple data type: " + item1.getClass());
	}

	static List unpack(byte[] bytes) {
		try {
			DecodeState decodeState = new DecodeState();
			int pos = 0;
			int end = bytes.length;
			while (pos < end) {
				decode(decodeState, bytes, pos, end);
				pos = decodeState.end;
			}
			return decodeState.values;
		}
		catch(IndexOutOfBoundsException | BufferOverflowException e) {
			throw new IllegalArgumentException("Invalid tuple (possible truncation)", e);
		}
	}

	static void encodeAll(EncodeState state, List items) {
		for(Object t : items) {
			encode(state, t);
		}
	}

	static void pack(ByteBuffer dest, List items) {
		ByteOrder origOrder = dest.order();
		EncodeState state = new EncodeState(dest);
		encodeAll(state, items);
		dest.order(origOrder);
		if(state.versionPos >= 0) {
			throw new IllegalArgumentException("Incomplete Versionstamp included in vanilla tuple pack");
		}
	}

	static byte[] pack(List items, int expectedSize) {
		ByteBuffer dest = ByteBuffer.allocate(expectedSize);
		pack(dest, items);
		return dest.array();
	}

	static byte[] packWithVersionstamp(List items, int expectedSize) {
		ByteBuffer dest = ByteBuffer.allocate(expectedSize);
		EncodeState state = new EncodeState(dest);
		encodeAll(state, items);
		if(state.versionPos < 0) {
			throw new IllegalArgumentException("No incomplete Versionstamp included in tuple packInternal with versionstamp");
		}
		else {
			if(useOldVersionOffsetFormat() && state.versionPos > 0xffff) {
				throw new IllegalArgumentException("Tuple has incomplete version at position " + state.versionPos + " which is greater than the maximum " + 0xffff);
			}
			dest.order(ByteOrder.LITTLE_ENDIAN);
			if (useOldVersionOffsetFormat()) {
				dest.putShort((short)state.versionPos);
			} else {
				dest.putInt(state.versionPos);
			}
			return dest.array();
		}
	}

	static int getPackedSize(List items, boolean nested) {
		int packedSize = 0;
		for(Object item : items) {
			if(item == null)
				packedSize += nested ? 2 : 1;
			else if(item instanceof byte[]) {
				byte[] bytes = (byte[])item;
				packedSize += 2 + bytes.length + ByteArrayUtil.nullCount((byte[])item);
			}
			else if(item instanceof String) {
				int strPackedSize = StringUtil.packedSize((String)item);
				packedSize += 2 + strPackedSize;
			}
			else if(item instanceof Float)
				packedSize += 1 + Float.BYTES;
			else if(item instanceof Double)
				packedSize += 1 + Double.BYTES;
			else if(item instanceof Boolean)
				packedSize += 1;
			else if(item instanceof UUID)
				packedSize += 1 + UUID_BYTES;
			else if(item instanceof BigInteger) {
				BigInteger bigInt = (BigInteger)item;
				int byteCount = minimalByteCount(bigInt);
				// If byteCount <= 8, then the encoding uses 1 byte for both the size
				// and type code. If byteCount > 8, then there is 1 byte for the type code
				// and 1 byte for the length. In both cases, the value is followed by
				// the byte count.
				packedSize += byteCount + ((byteCount <= 8) ? 1 : 2);
			}
			else if(item instanceof Number)
				packedSize += 1 + minimalByteCount(((Number)item).longValue());
			else if(item instanceof Versionstamp) {
				packedSize += 1 + Versionstamp.LENGTH;
				Versionstamp versionstamp = (Versionstamp)item;
				if(!versionstamp.isComplete()) {
					int suffixSize = useOldVersionOffsetFormat() ? Short.BYTES : Integer.BYTES;
					packedSize += suffixSize;
				}
			}
			else if(item instanceof List)
				packedSize += 2 + getPackedSize((List)item, true);
			else if(item instanceof Tuple)
				packedSize += 2 + ((Tuple)item).getPackedSize(true);
			else
				throw new IllegalArgumentException("unknown type " + item.getClass() + " for tuple packing");
		}
		return packedSize;
	}

	static boolean hasIncompleteVersionstamp(Stream items) {
		return items.anyMatch(item -> {
			if(item == null) {
				return false;
			}
			else if(item instanceof Versionstamp) {
				return !((Versionstamp) item).isComplete();
			}
			else if(item instanceof Tuple) {
				return hasIncompleteVersionstamp(((Tuple) item).stream());
			}
			else if(item instanceof Collection) {
				return hasIncompleteVersionstamp(((Collection) item).stream());
			}
			else {
				return false;
			}
		});
	}

	public static void main(String[] args) {
		try {
			byte[] bytes = pack(Collections.singletonList(4), 2);
			DecodeState result = new DecodeState();
			decode(result, bytes, 0, bytes.length);
			int val = ((Number)result.values.get(0)).intValue();
			assert 4 == val;
		}
		catch(Exception e) {
			e.printStackTrace();
			System.out.println("Error " + e.getMessage());
		}

		try {
			byte[] bytes = pack(Collections.singletonList("\u021Aest \u0218tring"), 15);
			DecodeState result = new DecodeState();
			decode(result, bytes, 0, bytes.length);
			String string = (String)result.values.get(0);
			System.out.println("contents -> " + string);
			assert "\u021Aest \u0218tring".equals(string);
		}
		catch(Exception e) {
			e.printStackTrace();
			System.out.println("Error " + e.getMessage());
		}

		/*Object[] a = new Object[] { "\u0000a", -2, "b\u0001", 12345, ""};
		List o = Arrays.asList(a);
		byte[] packed = packInternal( o, null );
		System.out.println("packed length: " + packed.length);
		o = unpack( packed, 0, packed.length );
		System.out.println("unpacked elements: " + o);
		for(Object obj : o)
			System.out.println(" -> type: " + obj.getClass().getName());*/
	}
	private TupleUtil() {}
}