/*
* 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.io.sstable;
import java.io.*;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import org.apache.cassandra.db.TypeSizes;
import org.apache.cassandra.db.marshal.AbstractType;
import org.apache.cassandra.io.util.FileDataInput;
import org.apache.cassandra.io.util.FileMark;
import org.apache.cassandra.io.util.FileUtils;
import org.apache.cassandra.utils.*;
/**
* Provides helper to serialize, deserialize and use column indexes.
*/
public class IndexHelper
{
public static void skipBloomFilter(DataInput in) throws IOException
{
int size = in.readInt();
FileUtils.skipBytesFully(in, size);
}
/**
* Skip the index
* @param in the data input from which the index should be skipped
* @throws IOException if an I/O error occurs.
*/
public static void skipIndex(DataInput in) throws IOException
{
/* read only the column index list */
int columnIndexSize = in.readInt();
/* skip the column index data */
if (in instanceof FileDataInput)
{
FileUtils.skipBytesFully(in, columnIndexSize);
}
else
{
// skip bytes
byte[] skip = new byte[columnIndexSize];
in.readFully(skip);
}
}
/**
* Deserialize the index into a structure and return it
*
* @param in - input source
*
* @return ArrayList<IndexInfo> - list of de-serialized indexes
* @throws IOException if an I/O error occurs.
*/
public static List<IndexInfo> deserializeIndex(FileDataInput in) throws IOException
{
int columnIndexSize = in.readInt();
if (columnIndexSize == 0)
return Collections.<IndexInfo>emptyList();
ArrayList<IndexInfo> indexList = new ArrayList<IndexInfo>();
FileMark mark = in.mark();
while (in.bytesPastMark(mark) < columnIndexSize)
{
indexList.add(IndexInfo.deserialize(in));
}
assert in.bytesPastMark(mark) == columnIndexSize;
return indexList;
}
/**
* The index of the IndexInfo in which a scan starting with @name should begin.
*
* @param name
* name of the index
*
* @param indexList
* list of the indexInfo objects
*
* @param comparator
* comparator type
*
* @param reversed
* is name reversed
*
* @return int index
*/
public static int indexFor(ByteBuffer name, List<IndexInfo> indexList, AbstractType<?> comparator, boolean reversed, int lastIndex)
{
if (name.remaining() == 0 && reversed)
return indexList.size() - 1;
if (lastIndex >= indexList.size())
return -1;
IndexInfo target = new IndexInfo(name, name, 0, 0);
/*
Take the example from the unit test, and say your index looks like this:
[0..5][10..15][20..25]
and you look for the slice [13..17].
When doing forward slice, we we doing a binary search comparing 13 (the start of the query)
to the lastName part of the index slot. You'll end up with the "first" slot, going from left to right,
that may contain the start.
When doing a reverse slice, we do the same thing, only using as a start column the end of the query,
i.e. 17 in this example, compared to the firstName part of the index slots. bsearch will give us the
first slot where firstName > start ([20..25] here), so we subtract an extra one to get the slot just before.
*/
int startIdx = 0;
List<IndexInfo> toSearch = indexList;
if (lastIndex >= 0)
{
if (reversed)
{
toSearch = indexList.subList(0, lastIndex + 1);
}
else
{
startIdx = lastIndex;
toSearch = indexList.subList(lastIndex, indexList.size());
}
}
int index = Collections.binarySearch(toSearch, target, getComparator(comparator, reversed));
return startIdx + (index < 0 ? -index - (reversed ? 2 : 1) : index);
}
public static Comparator<IndexInfo> getComparator(final AbstractType<?> nameComparator, boolean reversed)
{
return reversed ? nameComparator.indexReverseComparator : nameComparator.indexComparator;
}
public static class IndexInfo
{
public final long width;
public final ByteBuffer lastName;
public final ByteBuffer firstName;
public final long offset;
public IndexInfo(ByteBuffer firstName, ByteBuffer lastName, long offset, long width)
{
this.firstName = firstName;
this.lastName = lastName;
this.offset = offset;
this.width = width;
}
public void serialize(DataOutput out) throws IOException
{
ByteBufferUtil.writeWithShortLength(firstName, out);
ByteBufferUtil.writeWithShortLength(lastName, out);
out.writeLong(offset);
out.writeLong(width);
}
public int serializedSize(TypeSizes typeSizes)
{
int firstNameSize = firstName.remaining();
int lastNameSize = lastName.remaining();
return typeSizes.sizeof((short) firstNameSize) + firstNameSize +
typeSizes.sizeof((short) lastNameSize) + lastNameSize +
typeSizes.sizeof(offset) + typeSizes.sizeof(width);
}
public static IndexInfo deserialize(DataInput in) throws IOException
{
return new IndexInfo(ByteBufferUtil.readWithShortLength(in), ByteBufferUtil.readWithShortLength(in), in.readLong(), in.readLong());
}
public long memorySize()
{
return ObjectSizes.getFieldSize(// firstName
ObjectSizes.getReferenceSize() +
// lastName
ObjectSizes.getReferenceSize() +
TypeSizes.NATIVE.sizeof(offset) +
TypeSizes.NATIVE.sizeof(width))
+ ObjectSizes.getSize(firstName) + ObjectSizes.getSize(lastName);
}
}
}