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

com.opencsv.bean.FieldMapByPosition Maven / Gradle / Ivy

There is a newer version: 5.9
Show newest version
/*
 * Copyright 2018 Andrew Rucker Jones.
 *
 * 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.opencsv.bean;

import com.opencsv.ICSVParser;
import com.opencsv.exceptions.CsvRequiredFieldEmptyException;
import org.apache.commons.collections4.MultiValuedMap;
import org.apache.commons.collections4.Transformer;
import org.apache.commons.collections4.iterators.LazyIteratorChain;
import org.apache.commons.collections4.iterators.TransformIterator;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.text.TextStringBuilder;

import java.lang.reflect.Field;
import java.util.*;

/**
 * This class maintains a mapping from column position out of a CSV file to bean
 * fields.
 *
 * @author Andrew Rucker Jones
 * @since 4.2
 */
public class FieldMapByPosition extends AbstractFieldMap, T> implements Iterable> {
    
    private int maxIndex = Integer.MAX_VALUE;

    /** Holds a {@link java.util.Comparator} to sort columns on writing. */
    private Comparator writeOrder = null;
    
    /**
     * Initializes this {@link FieldMap}.
     * 
     * @param errorLocale The locale to be used for error messages
     */
    public FieldMapByPosition(final Locale errorLocale) {
        super(errorLocale);
    }
    
    /**
     * This method generates a header that can be used for writing beans of the
     * type provided back to a file.
     * The ordering of the headers can be determined with the
     * {@link java.util.Comparator} passed in to
     * {@link #setColumnOrderOnWrite(Comparator)}. Otherwise, it is ascending
     * according to position.
     */
    // The rest of the Javadoc is inherited.
    @Override
    public String[] generateHeader(final T bean) throws CsvRequiredFieldEmptyException {
        final List missingRequiredHeaders = new LinkedList<>();
        final SortedMap headerMap = new TreeMap<>(writeOrder);
        for(Map.Entry> entry : simpleMap.entrySet()) {
            headerMap.put(entry.getKey(), entry.getValue().getField().getName());
        }
        for(ComplexFieldMapEntry r : complexMapList) {
            @SuppressWarnings("unchecked")
            final MultiValuedMap m = (MultiValuedMap) r.getBeanField().getFieldValue(bean);
            boolean oneEntryMatched = false;
            if(m != null && !m.isEmpty()) {
                for(Map.Entry entry : m.entries()) {
                    Integer key = entry.getKey();
                    if(r.contains(key)) {
                        headerMap.put(entry.getKey(), r.getBeanField().getField().getName());
                        oneEntryMatched = true;
                    }
                }
            }
            if(m == null || m.isEmpty() || !oneEntryMatched) {
                if(r.getBeanField().isRequired()) {
                    missingRequiredHeaders.add(r.getBeanField().getField());
                }
            }
        }
        
        // Convert to an array of header "names".
        // Since the user can pass in an arbitrary collation, we have to
        // re-sort to get the highest value.
        SortedSet headerSet = new TreeSet<>(headerMap.keySet());
        int arraySize = headerSet.isEmpty() ? 0 : headerSet.last()+1;
        final String[] headers = new String[arraySize];
        int previousIndex = headerSet.isEmpty() ? 0 : headerSet.first();
        for(Integer i : headerSet) { // Fill in gaps
            for(int j = previousIndex+1; j < i ; j++) {
                headerMap.put(j, null);
            }
            previousIndex = i;
        }
        previousIndex = 0;
        for(String value : headerMap.values()) {
            headers[previousIndex++] = value;
        }
        
        // Report headers that should have been present
        if(!missingRequiredHeaders.isEmpty()) {
            TextStringBuilder sb = new TextStringBuilder();
            for(Field f : missingRequiredHeaders) {
                sb.appendSeparator(' '); sb.append(f.getName());
            }
            String errorMessage = String.format(
                    ResourceBundle
                            .getBundle(ICSVParser.DEFAULT_BUNDLE_NAME, errorLocale)
                            .getString("header.required.field.absent"),
                    sb.toString(),
                    StringUtils.join(headers, ' '));
            throw new CsvRequiredFieldEmptyException(bean.getClass(), missingRequiredHeaders, errorMessage);
        }
        
        return headers;
    }
    
    /**
     * @param rangeDefinition A string describing the column positions to be
     *   matched.
     * @see CsvBindAndJoinByPosition#position() 
     */
    // The rest of the Javadoc is inherited
    @Override
    public void putComplex(final String rangeDefinition, final BeanField field) {
        complexMapList.add(new PositionToBeanField<>(rangeDefinition, maxIndex, field, errorLocale));
    }
    
    /**
     * Sets the maximum index for all ranges specified in the entire field map.
     * No ranges or mappings are ever removed so as to preserve information
     * about required fields, but upper boundries are shortened as much as
     * possible. If ranges or individual column positions were specified that
     * lie wholly above {@code maxIndex}, these are preserved, though ranges
     * are shortened to a single value (the lower boundry).
     * 
     * @param maxIndex The maximum index in the data being imported
     */
    public void setMaxIndex(int maxIndex) {
        this.maxIndex = maxIndex;
        
        // Attenuate all ranges that end past the last index down to the last index
        for(PositionToBeanField p : complexMapList) {
            p.attenuateRanges(maxIndex);
        }
    }

    @Override
    public Iterator> iterator() {
        return new LazyIteratorChain>() {
            
            @Override
            protected Iterator> nextIterator(int count) {
                if(count <= complexMapList.size()) {
                    return complexMapList.get(count-1).iterator();
                }
                if(count == complexMapList.size()+1) {
                    return new TransformIterator>, FieldMapByPositionEntry>(
                            simpleMap.entrySet().iterator(),
                            new Transformer>, FieldMapByPositionEntry>() {
                                @Override
                                public FieldMapByPositionEntry transform(Map.Entry> input) {
                                    return new FieldMapByPositionEntry(input.getKey(), input.getValue());
                                }
                            });
                }
                return null;
            }
        };
    }

    /**
     * Sets the {@link java.util.Comparator} to be used to sort columns when
     * writing beans to a CSV file.
     *
     * @param writeOrder The {@link java.util.Comparator} to use. May be
     *   {@code null}, in which case the natural ordering is used.
     * @since 4.3
     */
    public void setColumnOrderOnWrite(Comparator writeOrder) {
        this.writeOrder = writeOrder;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy