blob: 971af76ec8b5a36a262e6c23a2e084be0ab5a2cc [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.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Random;
import java.util.Set;
import org.junit.Assert;
import org.junit.Test;
import org.apache.cassandra.cql3.CQL3Type;
import org.apache.cassandra.transport.ProtocolVersion;
import static org.apache.cassandra.db.marshal.ValueAccessors.ACCESSORS;
public class CollectionTypesTest
{
interface TypeFactory<T extends CollectionType> { T createType(AbstractType<?> keyType, AbstractType<?> valType); }
interface ValueFactory<T> { T createValue(ValueGenerator keyGen, ValueGenerator valGen, int size, Random random); }
static <CT extends CollectionType, T> void testSerializationDeserialization(TypeFactory<CT> typeFactory, ValueFactory<T> valueFactory, ValueGenerator keyType)
{
for (ValueGenerator valueType : ValueGenerator.GENERATORS)
{
CT type = typeFactory.createType(keyType != null ? keyType.getType() : null, valueType.getType());
CQL3Type.Collection cql3Type = new CQL3Type.Collection(type);
for (int i=0; i<500; i++)
{
Random random = new Random(i);
int size = random.nextInt(1000);
T expected = valueFactory.createValue(keyType, valueType, size, random);
for (ValueAccessor<Object> srcAccessor : ACCESSORS)
{
ByteBuffer srcBuffer = type.decompose(expected);
Object srcBytes = srcAccessor.convert(srcBuffer, ByteBufferAccessor.instance);
String srcString = type.getString(srcBytes, srcAccessor);
for (ValueAccessor<Object> dstAccessor : ACCESSORS)
{
Object dstBytes = dstAccessor.convert(srcBytes, srcAccessor);
String dstString = type.getString(dstBytes, dstAccessor);
T composed = (T) type.compose(dstBytes, dstAccessor);
Assert.assertEquals(expected, composed);
ValueAccessors.assertDataEquals(srcBytes, srcAccessor, dstBytes, dstAccessor);
Assert.assertEquals(srcString, dstString);
}
}
}
}
}
static <CT extends CollectionType, T> void testSerializationDeserialization(TypeFactory<CT> typeFactory, ValueFactory<T> valueFactory)
{
for (ValueGenerator keyType : ValueGenerator.GENERATORS)
testSerializationDeserialization(typeFactory, valueFactory, keyType);
}
private static List<Object> randomList(ValueGenerator keyGen, ValueGenerator valGen, int size, Random random)
{
List<Object> list = new ArrayList<>();
for (int k=0; k<size; k++)
list.add(valGen.nextValue(random));
return list;
}
@Test
public void list()
{
testSerializationDeserialization((k, v) -> ListType.getInstance(v, false), CollectionTypesTest::randomList, null);
}
private static Map<Object, Object> randomMap(ValueGenerator keyGen, ValueGenerator valGen, int size, Random random)
{
Map<Object, Object> map = new HashMap<>();
for (int k=0; k<size; k++)
map.put(keyGen.nextValue(random), valGen.nextValue(random));
return map;
}
@Test
public void map()
{
testSerializationDeserialization((k, v) -> MapType.getInstance(k, v, false), CollectionTypesTest::randomMap);
}
private static Set<Object> randomSet(ValueGenerator keyGen, ValueGenerator valGen, int size, Random random)
{
Set<Object> set = new HashSet<>();
for (int k=0; k<size; k++)
set.add(valGen.nextValue(random));
return set;
}
@Test
public void set()
{
testSerializationDeserialization((k, v) -> SetType.getInstance(v, false), CollectionTypesTest::randomSet, null);
}
}