org.apache.camel.dataformat.csv.CsvUnmarshaller Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of camel-csv Show documentation
Show all versions of camel-csv Show documentation
Camel CSV data format support
/*
* 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.camel.dataformat.csv;
import java.io.Closeable;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import org.apache.camel.Exchange;
import org.apache.camel.support.ExchangeHelper;
import org.apache.camel.util.IOHelper;
import org.apache.commons.csv.CSVFormat;
import org.apache.commons.csv.CSVParser;
import org.apache.commons.csv.CSVRecord;
/**
* This class unmarshal CSV into lists or maps depending on the configuration.
*/
abstract class CsvUnmarshaller {
protected final CSVFormat format;
protected final CsvDataFormat dataFormat;
protected final CsvRecordConverter> converter;
private CsvUnmarshaller(CSVFormat format, CsvDataFormat dataFormat) {
this.format = format;
this.dataFormat = dataFormat;
this.converter = extractConverter(dataFormat);
}
public static CsvUnmarshaller create(CSVFormat format, CsvDataFormat dataFormat) {
// If we want to capture the header record, thus the header must be either fixed or automatic
if (dataFormat.isCaptureHeaderRecord() && format.getHeader() == null) {
format = format.withHeader();
}
// If we want to use maps, thus the header must be either fixed or automatic
if ((dataFormat.isUseMaps() || dataFormat.isUseOrderedMaps()) && format.getHeader() == null) {
format = format.withHeader();
}
// If we want to skip the header record it must automatic otherwise it's not working
if (format.getSkipHeaderRecord() && format.getHeader() == null) {
format = format.withHeader();
}
if (dataFormat.isLazyLoad()) {
return new StreamCsvUnmarshaller(format, dataFormat);
}
return new BulkCsvUnmarshaller(format, dataFormat);
}
/**
* Unmarshal the CSV
*
* @param exchange Exchange (used for accessing type converter)
* @param body the input
* @return Unmarshalled CSV
* @throws Exception if error during unmarshalling
*/
public abstract Object unmarshal(Exchange exchange, Object body) throws Exception;
private static CsvRecordConverter> extractConverter(CsvDataFormat dataFormat) {
if (dataFormat.getRecordConverter() != null) {
return dataFormat.getRecordConverter();
} else if (dataFormat.isUseOrderedMaps()) {
return CsvRecordConverters.orderedMapConverter();
} else if (dataFormat.isUseMaps()) {
return CsvRecordConverters.mapConverter();
} else {
return CsvRecordConverters.listConverter();
}
}
//region Implementations
/**
* This class reads all the CSV into one big list.
*/
private static final class BulkCsvUnmarshaller extends CsvUnmarshaller {
private BulkCsvUnmarshaller(CSVFormat format, CsvDataFormat dataFormat) {
super(format, dataFormat);
}
@Override
public Object unmarshal(Exchange exchange, Object body) throws Exception {
Reader reader = exchange.getContext().getTypeConverter().tryConvertTo(Reader.class, exchange, body);
if (reader == null) {
// fallback to input stream
InputStream is = exchange.getContext().getTypeConverter().mandatoryConvertTo(InputStream.class, exchange, body);
reader = new InputStreamReader(is, ExchangeHelper.getCharsetName(exchange));
}
CSVParser parser
= new CSVParser(reader, format);
try {
if (dataFormat.isCaptureHeaderRecord()) {
exchange.getMessage().setHeader(CsvConstants.HEADER_RECORD, parser.getHeaderNames());
}
return asList(parser.iterator(), converter);
} finally {
IOHelper.close(parser);
}
}
private List asList(Iterator iterator, CsvRecordConverter converter) {
List answer = new ArrayList<>();
while (iterator.hasNext()) {
answer.add(converter.convertRecord(iterator.next()));
}
return answer;
}
}
/**
* This class streams the content of the CSV
*/
@SuppressWarnings("unchecked")
private static final class StreamCsvUnmarshaller extends CsvUnmarshaller {
private StreamCsvUnmarshaller(CSVFormat format, CsvDataFormat dataFormat) {
super(format, dataFormat);
}
@Override
public Object unmarshal(Exchange exchange, Object body) throws Exception {
Reader reader = exchange.getContext().getTypeConverter().tryConvertTo(Reader.class, exchange, body);
if (reader == null) {
// fallback to input stream
InputStream is = exchange.getContext().getTypeConverter().mandatoryConvertTo(InputStream.class, exchange, body);
reader = new InputStreamReader(is, ExchangeHelper.getCharsetName(exchange));
}
try {
CSVParser parser = new CSVParser(reader, format);
CsvIterator> answer = new CsvIterator<>(parser, converter);
// add to UoW, so we can close the iterator, so it can release any resources
exchange.getExchangeExtension().addOnCompletion(new CsvUnmarshalOnCompletion(answer));
return answer;
} catch (Exception e) {
IOHelper.close(reader);
throw e;
}
}
}
/**
* This class converts the CSV iterator into the proper result type.
*
* @param Converted type
*/
private static final class CsvIterator implements Iterator, Closeable {
private final CSVParser parser;
private final Iterator iterator;
private final CsvRecordConverter converter;
private CsvIterator(CSVParser parser, CsvRecordConverter converter) {
this.parser = parser;
this.iterator = parser.iterator();
this.converter = converter;
}
@Override
public boolean hasNext() {
return iterator.hasNext();
}
@Override
public T next() {
return converter.convertRecord(iterator.next());
}
@Override
public void remove() {
iterator.remove();
}
@Override
public void close() throws IOException {
if (!parser.isClosed()) {
parser.close();
}
}
}
//endregion
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy