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

org.apache.phoenix.util.FirstLastNthValueDataContainer Maven / Gradle / Ivy

The 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.phoenix.util;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.util.LinkedList;
import java.util.ListIterator;
import java.util.Map;
import java.util.Map.Entry;
import java.util.TreeMap;

import org.apache.hadoop.hbase.util.Bytes;

/**
 * Container for data transfer between server and client aggregation (FIRST|LAST|NTH)_VALUE functions
 *
 */
public class FirstLastNthValueDataContainer {

    protected boolean isAscending = false;
    protected int offset;
    protected TreeMap> data;
    protected boolean isOrderValuesFixedLength = false;
    protected boolean isDataValuesFixedLength = false;

    public void setIsAscending(boolean ascending) {
        isAscending = ascending;
    }

    public void setData(TreeMap> topValues) {
        data = topValues;
    }

    public void setFixedWidthOrderValues(boolean fixedSize) {
        isOrderValuesFixedLength = fixedSize;
    }

    public void setFixedWidthDataValues(boolean fixedSize) {
        isDataValuesFixedLength = fixedSize;
    }

    public void setOffset(int offset) {
        this.offset = offset;
    }

    public void setPayload(byte[] payload) {
        if (payload[0] == (byte) 1) {
            isAscending = true;
        }

        int lengthOfOrderValues = Bytes.toInt(payload, 1);
        int lengthOfDataValues = Bytes.toInt(payload, 5);
        int sizeOfMap = Bytes.toInt(payload, 9);

        data = new TreeMap>(new Bytes.ByteArrayComparator());

        int payloadOffset = 13;

        for (; sizeOfMap != 0; sizeOfMap--) {
            byte[] key;
            byte[] value;

            if (lengthOfOrderValues != 0) {
                key = Bytes.copy(payload, payloadOffset, lengthOfOrderValues);
                payloadOffset += lengthOfOrderValues;
            } else {
                int l = Bytes.toInt(payload, payloadOffset);
                payloadOffset += 4;
                key = Bytes.copy(payload, payloadOffset, l);
                payloadOffset += l;
            }

            if (lengthOfDataValues != 0) {
                value = Bytes.copy(payload, payloadOffset, lengthOfDataValues);
                payloadOffset += lengthOfDataValues;
            } else {
                int l = Bytes.toInt(payload, payloadOffset);
                payloadOffset += 4;
                value = Bytes.copy(payload, payloadOffset, l);
                payloadOffset += l;
            }

            if(!data.containsKey(key)) {
                data.put(key, new LinkedList());
            }
            data.get(key).add(value);
        }

    }

    public byte[] getPayload() throws IOException {
        /*
        PAYLOAD STUCTURE

        what                    | size (bytes) | info
        is ascending            | 1            | 1 = asc, 0 = desc
        length of order by vals | 4            | 0 if dynamic length, size otherwise
        length of values        | 4            | 0 if dynamic length, size otherwise
      [ length of first order   | 4            | set if order is var length (optional) ]
        first order value       | n            | order by val
      [ length of first value   | 4            | set if value is var length (optional) ]
        first value             | n            | data val
        ... and so on, repeat order by values and data values


        example with fixed length for data and order by values
        0           | 0000 0004         | 0000 0004        | 0000 0001       | 0000 000FF  | ...
        is ascendig | length order vals | length data vals | first order val | first value | ... more values

        example with dynamic length for data (length will be zeros)
        0           | 0000 0000         | 0000 0000        | 0000 0004          | 0000 000FF        | ...
        is ascendig | length order vals | length data vals | first order length | first order value | ... more values

        */

        ByteArrayOutputStream bos = new ByteArrayOutputStream();

        bos.write(isAscending ? (byte) 1 : (byte) 0);

        Entry> firstEntry = data.firstEntry();
        if (isOrderValuesFixedLength) {
            bos.write(Bytes.toBytes(firstEntry.getKey().length));
        } else {
            bos.write(Bytes.toBytes(0));
        }

        if (isDataValuesFixedLength) {
            bos.write(Bytes.toBytes(firstEntry.getValue().getFirst().length));
        } else {
            bos.write(Bytes.toBytes(0));
        }

        int offsetForDataLength = bos.size();
        bos.write(new byte[4]); //space for number of elements
        int valuesCount = 0;

        for (Map.Entry> entry : data.entrySet()) {
            ListIterator it = entry.getValue().listIterator();
            while(it.hasNext()) {
                valuesCount++;
                byte[] itemValue = it.next();

                if (!isOrderValuesFixedLength) {
                    bos.write(Bytes.toBytes(entry.getKey().length));
                }
                bos.write(entry.getKey());

                if (!isDataValuesFixedLength) {
                    bos.write(Bytes.toBytes(itemValue.length));
                }
                bos.write(itemValue);
            }
        }

        byte[] outputArray = bos.toByteArray();
        //write number of elements
        System.arraycopy(Bytes.toBytes(valuesCount), 0, outputArray, offsetForDataLength, 4);
        return outputArray;
    }

    public boolean getIsAscending() {
        return isAscending;
    }

    public TreeMap> getData() {
        return data;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy