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

org.eobjects.analyzer.beans.datastructures.SelectFromMapTransformer Maven / Gradle / Ivy

Go to download

Components which manipulate rich data structures (lists, maps and more) for AnalyzerBeans

There is a newer version: 1.0.6
Show newest version
/**
 * eobjects.org AnalyzerBeans
 * Copyright (C) 2010 eobjects.org
 *
 * This copyrighted material is made available to anyone wishing to use, modify,
 * copy, or redistribute it subject to the terms and conditions of the GNU
 * Lesser General Public License, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
 * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License
 * for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this distribution; if not, write to:
 * Free Software Foundation, Inc.
 * 51 Franklin Street, Fifth Floor
 * Boston, MA  02110-1301  USA
 */
package org.eobjects.analyzer.beans.datastructures;

import java.lang.reflect.Array;
import java.util.Arrays;
import java.util.List;
import java.util.Map;

import javax.inject.Inject;

import org.eobjects.analyzer.beans.api.Categorized;
import org.eobjects.analyzer.beans.api.Configured;
import org.eobjects.analyzer.beans.api.Description;
import org.eobjects.analyzer.beans.api.OutputColumns;
import org.eobjects.analyzer.beans.api.Transformer;
import org.eobjects.analyzer.beans.api.TransformerBean;
import org.eobjects.analyzer.beans.categories.DataStructuresCategory;
import org.eobjects.analyzer.data.InputColumn;
import org.eobjects.analyzer.data.InputRow;

/**
 * Transformer for selecting values from maps.
 * 
 * @author Kasper Sørensen
 * @author Shekhar Gulati
 * @author Saurabh Gupta
 */
@TransformerBean("Select values from key/value map")
@Description("Given a specified list of keys, this transformer will select the values from a key/value map and place them as columns within the record")
@Categorized(DataStructuresCategory.class)
public class SelectFromMapTransformer implements Transformer {

    @Inject
    @Configured
    InputColumn> mapColumn;

    @Inject
    @Configured
    String[] keys;

    @Inject
    @Configured
    Class[] types;

    @Inject
    @Configured
    @Description("Verify that expected type and actual type are the same")
    boolean verifyTypes = false;

    public void setKeys(String[] keys) {
        this.keys = keys;
    }

    public void setTypes(Class[] types) {
        this.types = types;
    }

    public void setMapColumn(InputColumn> mapColumn) {
        this.mapColumn = mapColumn;
    }

    public void setVerifyTypes(boolean verifyTypes) {
        this.verifyTypes = verifyTypes;
    }

    @Override
    public OutputColumns getOutputColumns() {
        String[] keys = this.keys;
        Class[] types = this.types;
        if (keys.length != types.length) {
            // odd case sometimes encountered with invalid configurations or
            // while building a job
            final int length = Math.min(keys.length, types.length);
            keys = Arrays.copyOf(keys, length);
            types = Arrays.copyOf(types, length);
        }
        return new OutputColumns(keys, types);
    }

    @Override
    public Object[] transform(InputRow row) {
        final Map map = row.getValue(mapColumn);
        final Object[] result = new Object[keys.length];

        if (map == null) {
            return result;
        }

        for (int i = 0; i < keys.length; i++) {
            Object value = find(map, keys[i]);
            if (verifyTypes) {
                value = types[i].cast(value);
            }
            result[i] = value;
        }

        return result;
    }

    /**
     * Searches a map for a given key. The key can be a regular map key, or a
     * simple expression of the form:
     * 
     * 
    *
  • foo.bar (will lookup 'foo', and then 'bar' in a potential nested map) *
  • *
  • foo.bar[0].baz (will lookup 'foo', then 'bar' in a potential nested * map, then pick the first element in case it is a list/array and then pick * 'baz' from the potential map at that position). *
* * @param map * the map to search in * @param key * the key to resolve * @return the object in the map with the given key/expression. Or null if * it does not exist. */ public static Object find(Map map, String key) { if (map == null || key == null) { return null; } final Object result = map.get(key); if (result == null) { return find(map, key, 0); } return result; } private static Object find(Map map, String key, int fromIndex) { final int indexOfDot = key.indexOf('.', fromIndex); final int indexOfBracket = key.indexOf('[', fromIndex); int indexOfEndBracket = -1; int arrayIndex = -1; boolean hasDot = indexOfDot != -1; boolean hasBracket = indexOfBracket != -1; if (hasBracket) { // also check that there is an end-bracket indexOfEndBracket = key.indexOf("].", indexOfBracket); hasBracket = indexOfEndBracket != -1; if (hasBracket) { final String indexString = key.substring(indexOfBracket + 1, indexOfEndBracket); try { arrayIndex = Integer.parseInt(indexString); } catch (NumberFormatException e) { // not a valid array/list index hasBracket = false; } } } if (hasDot && hasBracket) { if (indexOfDot > indexOfBracket) { hasDot = false; } else { hasBracket = false; } } if (hasDot) { final String prefix = key.substring(0, indexOfDot); final Object nestedObject = map.get(prefix); if (nestedObject == null) { return find(map, key, indexOfDot + 1); } if (nestedObject instanceof Map) { final String remainingPart = key.substring(indexOfDot + 1); @SuppressWarnings("unchecked") final Map nestedMap = (Map) nestedObject; return find(nestedMap, remainingPart); } } if (hasBracket) { final String prefix = key.substring(0, indexOfBracket); final Object nestedObject = map.get(prefix); if (nestedObject == null) { return find(map, key, indexOfBracket + 1); } final String remainingPart = key.substring(indexOfEndBracket + 2); try { if (nestedObject instanceof List) { @SuppressWarnings("unchecked") final Map nestedMap = ((List>) nestedObject).get(arrayIndex); return find(nestedMap, remainingPart); } else if (nestedObject.getClass().isArray()) { @SuppressWarnings("unchecked") final Map nestedMap = (Map) Array.get(nestedObject, arrayIndex); return find(nestedMap, remainingPart); } } catch (IndexOutOfBoundsException e) { return null; } } return null; } }