MapLexicoder.java

/*
 * Licensed 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 com.facebook.presto.accumulo.serializers;

import org.apache.accumulo.core.client.lexicoder.Lexicoder;

import java.util.HashMap;
import java.util.Map;
import java.util.Map.Entry;

import static org.apache.accumulo.core.client.lexicoder.impl.ByteUtils.concat;
import static org.apache.accumulo.core.client.lexicoder.impl.ByteUtils.escape;
import static org.apache.accumulo.core.client.lexicoder.impl.ByteUtils.split;
import static org.apache.accumulo.core.client.lexicoder.impl.ByteUtils.unescape;

/**
 * Accumulo lexicoder for encoding a Java Map
 *
 * @param <K> Key data type
 * @param <V> Value data type
 */
public class MapLexicoder<K, V>
        implements Lexicoder<Map<K, V>>
{
    private final Lexicoder<K> keyLexicoder;
    private final Lexicoder<V> valueLexicoder;

    public MapLexicoder(
            Lexicoder<K> keyLexicoder,
            Lexicoder<V> valueLexicoder)
    {
        this.keyLexicoder = keyLexicoder;
        this.valueLexicoder = valueLexicoder;
    }

    @Override
    public byte[] encode(Map<K, V> v)
    {
        byte[][] elements = new byte[v.size() * 2][];
        int index = 0;
        for (Entry<K, V> entry : v.entrySet()) {
            elements[index++] = escape(keyLexicoder.encode(entry.getKey()));
            elements[index++] = escape(valueLexicoder.encode(entry.getValue()));
        }

        return concat(elements);
    }

    @Override
    public Map<K, V> decode(byte[] b)
    {
        byte[][] escapedElements = split(b);
        Map<K, V> decodedMap = new HashMap<>();
        for (int i = 0; i < escapedElements.length; i += 2) {
            K key = keyLexicoder.decode(unescape(escapedElements[i]));
            V value = valueLexicoder.decode(unescape(escapedElements[i + 1]));
            decodedMap.put(key, value);
        }

        return decodedMap;
    }
}