blob: 79f921920ddf0639041fbb7fcb0e347db52fc935 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.cassandra.db.marshal;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.regex.Pattern;
import com.google.common.base.Objects;
import com.google.common.collect.Lists;
import org.apache.cassandra.cql3.*;
import org.apache.cassandra.db.TypeSizes;
import org.apache.cassandra.exceptions.ConfigurationException;
import org.apache.cassandra.exceptions.SyntaxException;
import org.apache.cassandra.serializers.*;
import org.apache.cassandra.transport.ProtocolVersion;
import org.apache.cassandra.utils.JsonUtils;
import org.apache.cassandra.utils.bytecomparable.ByteComparable;
import org.apache.cassandra.utils.bytecomparable.ByteSource;
import org.apache.cassandra.utils.bytecomparable.ByteSourceInverse;
import static com.google.common.collect.Iterables.any;
import static com.google.common.collect.Iterables.transform;
/**
* This is essentially like a CompositeType, but it's not primarily meant for comparison, just
* to pack multiple values together so has a more friendly encoding.
*/
public class TupleType extends AbstractType<ByteBuffer>
{
private static final String COLON = ":";
private static final Pattern COLON_PAT = Pattern.compile(COLON);
private static final String ESCAPED_COLON = "\\\\:";
private static final Pattern ESCAPED_COLON_PAT = Pattern.compile(ESCAPED_COLON);
private static final String AT = "@";
private static final Pattern AT_PAT = Pattern.compile(AT);
private static final String ESCAPED_AT = "\\\\@";
private static final Pattern ESCAPED_AT_PAT = Pattern.compile(ESCAPED_AT);
protected final List<AbstractType<?>> types;
private final TupleSerializer serializer;
public TupleType(List<AbstractType<?>> types)
{
this(types, true);
}
protected TupleType(List<AbstractType<?>> types, boolean freezeInner)
{
super(ComparisonType.CUSTOM);
if (freezeInner)
this.types = Lists.newArrayList(transform(types, AbstractType::freeze));
else
this.types = types;
this.serializer = new TupleSerializer(fieldSerializers(types));
}
private static List<TypeSerializer<?>> fieldSerializers(List<AbstractType<?>> types)
{
int size = types.size();
List<TypeSerializer<?>> serializers = new ArrayList<>(size);
for (int i = 0; i < size; i++)
serializers.add(types.get(i).getSerializer());
return serializers;
}
public static TupleType getInstance(TypeParser parser) throws ConfigurationException, SyntaxException
{
List<AbstractType<?>> types = parser.getTypeParameters();
for (int i = 0; i < types.size(); i++)
types.set(i, types.get(i).freeze());
return new TupleType(types);
}
@Override
public <V> boolean referencesUserType(V name, ValueAccessor<V> accessor)
{
return any(types, t -> t.referencesUserType(name, accessor));
}
@Override
public TupleType withUpdatedUserType(UserType udt)
{
return referencesUserType(udt.name)
? new TupleType(Lists.newArrayList(transform(types, t -> t.withUpdatedUserType(udt))))
: this;
}
@Override
public AbstractType<?> expandUserTypes()
{
return new TupleType(Lists.newArrayList(transform(types, AbstractType::expandUserTypes)));
}
@Override
public boolean referencesDuration()
{
return allTypes().stream().anyMatch(f -> f.referencesDuration());
}
public AbstractType<?> type(int i)
{
return types.get(i);
}
public int size()
{
return types.size();
}
@Override
public List<AbstractType<?>> subTypes()
{
return types;
}
public List<AbstractType<?>> allTypes()
{
return types;
}
public boolean isTuple()
{
return true;
}
public <VL, VR> int compareCustom(VL left, ValueAccessor<VL> accessorL, VR right, ValueAccessor<VR> accessorR)
{
if (accessorL.isEmpty(left) || accessorR.isEmpty(right))
return Boolean.compare(accessorR.isEmpty(right), accessorL.isEmpty(left));
int offsetL = 0;
int offsetR = 0;
for (int i = 0; !accessorL.isEmptyFromOffset(left, offsetL) && !accessorR.isEmptyFromOffset(right, offsetR) && i < types.size(); i++)
{
AbstractType<?> comparator = types.get(i);
int sizeL = accessorL.getInt(left, offsetL);
offsetL += TypeSizes.INT_SIZE;
int sizeR = accessorR.getInt(right, offsetR);
offsetR += TypeSizes.INT_SIZE;
// Handle nulls
if (sizeL < 0)
{
if (sizeR < 0)
continue;
return -1;
}
if (sizeR < 0)
return 1;
VL valueL = accessorL.slice(left, offsetL, sizeL);
offsetL += sizeL;
VR valueR = accessorR.slice(right, offsetR, sizeR);
offsetR += sizeR;
int cmp = comparator.compare(valueL, accessorL, valueR, accessorR);
if (cmp != 0)
return cmp;
}
if (allRemainingComponentsAreNull(left, accessorL, offsetL) && allRemainingComponentsAreNull(right, accessorR, offsetR))
return 0;
if (accessorL.isEmptyFromOffset(left, offsetL))
return allRemainingComponentsAreNull(right, accessorR, offsetR) ? 0 : -1;
return allRemainingComponentsAreNull(left, accessorL, offsetL) ? 0 : 1;
}
private <T> boolean allRemainingComponentsAreNull(T v, ValueAccessor<T> accessor, int offset)
{
while (!accessor.isEmptyFromOffset(v, offset))
{
int size = accessor.getInt(v, offset);
offset += TypeSizes.INT_SIZE;
if (size >= 0)
return false;
}
return true;
}
@Override
public <V> ByteSource asComparableBytes(ValueAccessor<V> accessor, V data, ByteComparable.Version version)
{
switch (version)
{
case LEGACY:
return asComparableBytesLegacy(accessor, data);
case OSS50:
return asComparableBytesNew(accessor, data, version);
default:
throw new AssertionError();
}
}
private <V> ByteSource asComparableBytesLegacy(ValueAccessor<V> accessor, V data)
{
if (accessor.isEmpty(data))
return null;
V[] bufs = split(accessor, data); // this may be shorter than types.size -- other srcs remain null in that case
ByteSource[] srcs = new ByteSource[types.size()];
for (int i = 0; i < bufs.length; ++i)
srcs[i] = bufs[i] != null ? types.get(i).asComparableBytes(accessor, bufs[i], ByteComparable.Version.LEGACY) : null;
// We always have a fixed number of sources, with the trailing ones possibly being nulls.
// This can only result in a prefix if the last type in the tuple allows prefixes. Since that type is required
// to be weakly prefix-free, so is the tuple.
return ByteSource.withTerminatorLegacy(ByteSource.END_OF_STREAM, srcs);
}
private <V> ByteSource asComparableBytesNew(ValueAccessor<V> accessor, V data, ByteComparable.Version version)
{
if (accessor.isEmpty(data))
return null;
V[] bufs = split(accessor, data);
int lengthWithoutTrailingNulls = 0;
for (int i = 0; i < bufs.length; ++i)
if (bufs[i] != null)
lengthWithoutTrailingNulls = i + 1;
ByteSource[] srcs = new ByteSource[lengthWithoutTrailingNulls];
for (int i = 0; i < lengthWithoutTrailingNulls; ++i)
srcs[i] = bufs[i] != null ? types.get(i).asComparableBytes(accessor, bufs[i], version) : null;
// Because we stop early when there are trailing nulls, there needs to be an explicit terminator to make the
// type prefix-free.
return ByteSource.withTerminator(ByteSource.TERMINATOR, srcs);
}
@Override
public <V> V fromComparableBytes(ValueAccessor<V> accessor, ByteSource.Peekable comparableBytes, ByteComparable.Version version)
{
assert version == ByteComparable.Version.OSS50; // Reverse translation is not supported for the legacy version.
if (comparableBytes == null)
return accessor.empty();
V[] componentBuffers = accessor.createArray(types.size());
for (int i = 0; i < types.size(); ++i)
{
if (comparableBytes.peek() == ByteSource.TERMINATOR)
break; // the rest of the fields remain null
AbstractType<?> componentType = types.get(i);
ByteSource.Peekable component = ByteSourceInverse.nextComponentSource(comparableBytes);
if (component != null)
componentBuffers[i] = componentType.fromComparableBytes(accessor, component, version);
else
componentBuffers[i] = null;
}
// consume terminator
int terminator = comparableBytes.next();
assert terminator == ByteSource.TERMINATOR : String.format("Expected TERMINATOR (0x%2x) after %d components",
ByteSource.TERMINATOR,
types.size());
return buildValue(accessor, componentBuffers);
}
/**
* Split a tuple value into its component values.
*/
public <V> V[] split(ValueAccessor<V> accessor, V value)
{
return split(accessor, value, size(), this);
}
/**
* Split a tuple value into its component values.
*/
public static <V> V[] split(ValueAccessor<V> accessor, V value, int numberOfElements, TupleType type)
{
V[] components = accessor.createArray(numberOfElements);
int length = accessor.size(value);
int position = 0;
for (int i = 0; i < numberOfElements; i++)
{
if (position == length)
return Arrays.copyOfRange(components, 0, i);
if (position + 4 > length)
throw new MarshalException(String.format("Not enough bytes to read %dth component", i));
int size = accessor.getInt(value, position);
position += 4;
// size < 0 means null value
if (size >= 0)
{
if (position + size > length)
throw new MarshalException(String.format("Not enough bytes to read %dth component", i));
components[i] = accessor.slice(value, position, size);
position += size;
}
else
components[i] = null;
}
// error out if we got more values in the tuple/UDT than we expected
if (position < length)
{
throw new MarshalException(String.format("Expected %s %s for %s column, but got more",
numberOfElements, numberOfElements == 1 ? "value" : "values",
type.asCQL3Type()));
}
return components;
}
@SafeVarargs
public static <V> V buildValue(ValueAccessor<V> accessor, V... components)
{
int totalLength = 0;
for (V component : components)
totalLength += 4 + (component == null ? 0 : accessor.size(component));
int offset = 0;
V result = accessor.allocate(totalLength);
for (V component : components)
{
if (component == null)
{
offset += accessor.putInt(result, offset, -1);
}
else
{
offset += accessor.putInt(result, offset, accessor.size(component));
offset += accessor.copyTo(component, 0, result, accessor, offset, accessor.size(component));
}
}
return result;
}
public static ByteBuffer buildValue(ByteBuffer... components)
{
return buildValue(ByteBufferAccessor.instance, components);
}
@Override
public <V> String getString(V input, ValueAccessor<V> accessor)
{
if (input == null)
return "null";
StringBuilder sb = new StringBuilder();
int offset = 0;
for (int i = 0; i < size(); i++)
{
if (accessor.isEmptyFromOffset(input, offset))
return sb.toString();
if (i > 0)
sb.append(":");
AbstractType<?> type = type(i);
int size = accessor.getInt(input, offset);
offset += TypeSizes.INT_SIZE;
if (size < 0)
{
sb.append("@");
continue;
}
V field = accessor.slice(input, offset, size);
offset += size;
// We use ':' as delimiter, and @ to represent null, so escape them in the generated string
String fld = COLON_PAT.matcher(type.getString(field, accessor)).replaceAll(ESCAPED_COLON);
fld = AT_PAT.matcher(fld).replaceAll(ESCAPED_AT);
sb.append(fld);
}
return sb.toString();
}
public ByteBuffer fromString(String source)
{
// Split the input on non-escaped ':' characters
List<String> fieldStrings = AbstractCompositeType.split(source);
if (fieldStrings.size() > size())
throw new MarshalException(String.format("Invalid tuple literal: too many elements. Type %s expects %d but got %d",
asCQL3Type(), size(), fieldStrings.size()));
ByteBuffer[] fields = new ByteBuffer[fieldStrings.size()];
for (int i = 0; i < fieldStrings.size(); i++)
{
String fieldString = fieldStrings.get(i);
// We use @ to represent nulls
if (fieldString.equals("@"))
continue;
AbstractType<?> type = type(i);
fieldString = ESCAPED_COLON_PAT.matcher(fieldString).replaceAll(COLON);
fieldString = ESCAPED_AT_PAT.matcher(fieldString).replaceAll(AT);
fields[i] = type.fromString(fieldString);
}
return buildValue(fields);
}
@Override
public Term fromJSONObject(Object parsed) throws MarshalException
{
if (parsed instanceof String)
parsed = JsonUtils.decodeJson((String) parsed);
if (!(parsed instanceof List))
throw new MarshalException(String.format(
"Expected a list representation of a tuple, but got a %s: %s", parsed.getClass().getSimpleName(), parsed));
List<?> list = (List<?>) parsed;
if (list.size() > types.size())
throw new MarshalException(String.format("Tuple contains extra items (expected %s): %s", types.size(), parsed));
else if (types.size() > list.size())
throw new MarshalException(String.format("Tuple is missing items (expected %s): %s", types.size(), parsed));
List<Term> terms = new ArrayList<>(list.size());
Iterator<AbstractType<?>> typeIterator = types.iterator();
for (Object element : list)
{
if (element == null)
{
typeIterator.next();
terms.add(Constants.NULL_VALUE);
}
else
{
terms.add(typeIterator.next().fromJSONObject(element));
}
}
return new Tuples.DelayedValue(this, terms);
}
@Override
public String toJSONString(ByteBuffer buffer, ProtocolVersion protocolVersion)
{
ByteBuffer duplicated = buffer.duplicate();
int offset = 0;
StringBuilder sb = new StringBuilder("[");
for (int i = 0; i < types.size(); i++)
{
if (i > 0)
sb.append(", ");
ByteBuffer value = CollectionSerializer.readValue(duplicated, ByteBufferAccessor.instance, offset);
offset += CollectionSerializer.sizeOfValue(value, ByteBufferAccessor.instance);
if (value == null)
sb.append("null");
else
sb.append(types.get(i).toJSONString(value, protocolVersion));
}
return sb.append("]").toString();
}
public TypeSerializer<ByteBuffer> getSerializer()
{
return serializer;
}
@Override
public boolean isCompatibleWith(AbstractType<?> previous)
{
if (!(previous instanceof TupleType))
return false;
// Extending with new components is fine, removing is not
TupleType tt = (TupleType)previous;
if (size() < tt.size())
return false;
for (int i = 0; i < tt.size(); i++)
{
AbstractType<?> tprev = tt.type(i);
AbstractType<?> tnew = type(i);
if (!tnew.isCompatibleWith(tprev))
return false;
}
return true;
}
@Override
public boolean isValueCompatibleWithInternal(AbstractType<?> otherType)
{
if (!(otherType instanceof TupleType))
return false;
// Extending with new components is fine, removing is not
TupleType tt = (TupleType) otherType;
if (size() < tt.size())
return false;
for (int i = 0; i < tt.size(); i++)
{
AbstractType<?> tprev = tt.type(i);
AbstractType<?> tnew = type(i);
if (!tnew.isValueCompatibleWith(tprev))
return false;
}
return true;
}
@Override
public int hashCode()
{
return Objects.hashCode(types);
}
@Override
public boolean equals(Object o)
{
if(!(o instanceof TupleType))
return false;
TupleType that = (TupleType)o;
return types.equals(that.types);
}
@Override
public CQL3Type asCQL3Type()
{
return CQL3Type.Tuple.create(this);
}
@Override
public String toString()
{
return getClass().getName() + TypeParser.stringifyTypeParameters(types, true);
}
@Override
public ByteBuffer getMaskedValue()
{
ByteBuffer[] buffers = new ByteBuffer[types.size()];
for (int i = 0; i < types.size(); i++)
{
AbstractType<?> type = types.get(i);
buffers[i] = type.getMaskedValue();
}
return serializer.serialize(buildValue(buffers));
}
}