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

org.apache.cassandra.serializers.CollectionSerializer Maven / Gradle / Ivy

Go to download

The Apache Cassandra Project develops a highly scalable second-generation distributed database, bringing together Dynamo's fully distributed design and Bigtable's ColumnFamily-based data model.

There is a newer version: 5.0.0
Show newest version
/*
 * 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.serializers;

import java.nio.BufferUnderflowException;
import java.nio.ByteBuffer;
import java.util.Collection;
import java.util.List;
import java.util.function.Consumer;

import com.google.common.collect.Range;

import org.apache.cassandra.db.TypeSizes;
import org.apache.cassandra.db.marshal.ByteBufferAccessor;
import org.apache.cassandra.db.marshal.ValueAccessor;
import org.apache.cassandra.db.marshal.AbstractType;
import org.apache.cassandra.utils.ByteBufferUtil;

public abstract class CollectionSerializer extends TypeSerializer
{
    protected abstract List serializeValues(T value);
    protected abstract int getElementCount(T value);

    @Override
    public ByteBuffer serialize(T input)
    {
        List values = serializeValues(input);
        return pack(values, ByteBufferAccessor.instance, getElementCount(input));
    }

    public static ByteBuffer pack(Collection values, int elements)
    {
        return pack(values, ByteBufferAccessor.instance, elements);
    }

    public static  V pack(Collection values, ValueAccessor accessor, int elements)
    {
        int size = 0;
        for (V value : values)
            size += sizeOfValue(value, accessor);

        ByteBuffer result = ByteBuffer.allocate(sizeOfCollectionSize() + size);
        writeCollectionSize(result, elements);
        for (V value : values)
        {
            writeValue(result, value, accessor);
        }
        return accessor.valueOf((ByteBuffer) result.flip());
    }

    protected static void writeCollectionSize(ByteBuffer output, int elements)
    {
        output.putInt(elements);
    }

    public static  int readCollectionSize(V value, ValueAccessor accessor)
    {
        return accessor.toInt(value);
    }

    public static int sizeOfCollectionSize()
    {
        return TypeSizes.INT_SIZE;
    }

    public static  void writeValue(ByteBuffer output, V value, ValueAccessor accessor)
    {
        if (value == null)
        {
            output.putInt(-1);
            return;
        }

        output.putInt(accessor.size(value));
        accessor.write(value, output);
    }

    public static  V readValue(V input, ValueAccessor accessor, int offset)
    {
        int size = accessor.getInt(input, offset);
        if (size < 0)
            return null;

        return accessor.slice(input, offset + TypeSizes.INT_SIZE, size);
    }

    public static  V readNonNullValue(V input, ValueAccessor accessor, int offset)
    {
        V value = readValue(input, accessor, offset);
        if (value == null)
            throw new MarshalException("Null value read when not allowed");
        return value;
    }

    protected static void skipValue(ByteBuffer input)
    {
        int size = input.getInt();
        input.position(input.position() + size);
    }

    public static  int skipValue(V input, ValueAccessor accessor, int offset)
    {
        int size = accessor.getInt(input, offset);
        return TypeSizes.sizeof(size) + size;
    }

    public static  int sizeOfValue(V value, ValueAccessor accessor)
    {
        return value == null ? 4 : 4 + accessor.size(value);
    }

    /**
     * Extract an element from a serialized collection.
     * 

* Note that this is only supported to sets and maps. For sets, this mostly ends up being * a check for the presence of the provide key: it will return the key if it's present and * {@code null} otherwise. * * @param collection the serialized collection. This cannot be {@code null}. * @param key the key to extract (This cannot be {@code null} nor {@code ByteBufferUtil.UNSET_BYTE_BUFFER}). * @param comparator the type to use to compare the {@code key} value to those * in the collection. * @return the value associated with {@code key} if one exists, {@code null} otherwise */ public abstract ByteBuffer getSerializedValue(ByteBuffer collection, ByteBuffer key, AbstractType comparator); /** * Returns the slice of a collection directly from its serialized value. *

If the slice contains no elements an empty collection will be returned for frozen collections, and a * {@code null} one for non-frozen collections.

* * @param collection the serialized collection. This cannot be {@code null}. * @param from the left bound of the slice to extract. This cannot be {@code null} but if this is * {@code ByteBufferUtil.UNSET_BYTE_BUFFER}, then the returned slice starts at the beginning * of {@code collection}. * @param comparator the type to use to compare the {@code from} and {@code to} values to those * in the collection. * @param frozen {@code true} if the collection is a frozen one, {@code false} otherwise * @return a serialized collection corresponding to slice {@code [from, to]} of {@code collection}. */ public abstract ByteBuffer getSliceFromSerialized(ByteBuffer collection, ByteBuffer from, ByteBuffer to, AbstractType comparator, boolean frozen); /** * Returns the index of an element in a serialized collection. *

* Note that this is only supported by sets and maps, but not by lists. * * @param collection The serialized collection. This cannot be {@code null}. * @param key The key for which the index must be found. This cannot be {@code null} nor * {@link ByteBufferUtil#UNSET_BYTE_BUFFER}). * @param comparator The type to use to compare the {@code key} value to those in the collection. * @return The index of the element associated with {@code key} if one exists, {@code -1} otherwise. */ public abstract int getIndexFromSerialized(ByteBuffer collection, ByteBuffer key, AbstractType comparator); /** * Returns the range of indexes corresponding to the specified range of elements in the serialized collection. *

* Note that this is only supported by sets and maps, but not by lists. * * @param collection The serialized collection. This cannot be {@code null}. * @param from The left bound of the slice to extract. This cannot be {@code null} but if this is * {@link ByteBufferUtil#UNSET_BYTE_BUFFER}, then the returned slice starts at the beginning of the collection. * @param to The left bound of the slice to extract. This cannot be {@code null} but if this is * {@link ByteBufferUtil#UNSET_BYTE_BUFFER}, then the returned slice ends at the end of the collection. * @param comparator The type to use to compare the {@code from} and {@code to} values to those in the collection. * @return The range of indexes corresponding to specified range of elements. */ public abstract Range getIndexesRangeFromSerialized(ByteBuffer collection, ByteBuffer from, ByteBuffer to, AbstractType comparator); /** * Creates a new serialized map composed from the data from {@code input} between {@code startPos} * (inclusive) and {@code endPos} (exclusive), assuming that data holds {@code count} elements. */ protected ByteBuffer copyAsNewCollection(ByteBuffer input, int count, int startPos, int endPos) { int sizeLen = sizeOfCollectionSize(); if (count == 0) return ByteBuffer.allocate(sizeLen); int bodyLen = endPos - startPos; ByteBuffer output = ByteBuffer.allocate(sizeLen + bodyLen); writeCollectionSize(output, count); output.position(0); ByteBufferUtil.copyBytes(input, startPos, output, sizeLen, bodyLen); return output; } public void forEach(ByteBuffer input, Consumer action) { try { int collectionSize = readCollectionSize(input, ByteBufferAccessor.instance); int offset = sizeOfCollectionSize(); for (int i = 0; i < collectionSize; i++) { ByteBuffer value = readValue(input, ByteBufferAccessor.instance, offset); offset += sizeOfValue(value, ByteBufferAccessor.instance); action.accept(value); } } catch (BufferUnderflowException | IndexOutOfBoundsException e) { throw new MarshalException("Not enough bytes to read a set"); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy