Test for empty/blank headers the same way for input and output

This commit is contained in:
Gary Gregory 2022-10-23 09:09:26 -04:00
parent b6c63e47d9
commit de47291978
4 changed files with 84 additions and 64 deletions

View File

@ -49,7 +49,7 @@ import java.util.Objects;
import java.util.Set; import java.util.Set;
/** /**
* Specifies the format of a CSV file and parses input. * Specifies the format of a CSV file for parsing and writing.
* *
* <h2>Using predefined formats</h2> * <h2>Using predefined formats</h2>
* *
@ -174,6 +174,9 @@ import java.util.Set;
* <p> * <p>
* This class is immutable. * This class is immutable.
* </p> * </p>
* <p>
* Not all settings are used for both parsing and writing.
* </p>
*/ */
public final class CSVFormat implements Serializable { public final class CSVFormat implements Serializable {
@ -1210,6 +1213,10 @@ public final class CSVFormat implements Serializable {
return contains(source, CR) || contains(source, LF); return contains(source, CR) || contains(source, LF);
} }
static boolean isBlank(final String value) {
return value == null || value.trim().isEmpty();
}
/** /**
* Returns true if the given character is a line break character. * Returns true if the given character is a line break character.
* *
@ -1232,10 +1239,12 @@ public final class CSVFormat implements Serializable {
return c != null && isLineBreak(c.charValue()); return c != null && isLineBreak(c.charValue());
} }
/** Same test as in as {@link String#trim()}. */
private static boolean isTrimChar(final char ch) { private static boolean isTrimChar(final char ch) {
return ch <= SP; return ch <= SP;
} }
/** Same test as in as {@link String#trim()}. */
private static boolean isTrimChar(final CharSequence charSequence, final int pos) { private static boolean isTrimChar(final CharSequence charSequence, final int pos) {
return isTrimChar(charSequence.charAt(pos)); return isTrimChar(charSequence.charAt(pos));
} }
@ -2250,8 +2259,16 @@ public final class CSVFormat implements Serializable {
return sb.toString(); return sb.toString();
} }
String trim(final String value) {
return getTrim() ? value.trim() : value;
}
/** /**
* Verifies the validity and consistency of the attributes, and throws an IllegalArgumentException if necessary. * Verifies the validity and consistency of the attributes, and throws an {@link IllegalArgumentException} if necessary.
* <p>
* Because an instance can be used for both writing an parsing, not all conditions can be tested here. For example allowMissingColumnNames is only used for
* parsing, so it cannot be used here.
* </p>
* *
* @throws IllegalArgumentException Throw when any attribute is invalid or inconsistent with other attributes. * @throws IllegalArgumentException Throw when any attribute is invalid or inconsistent with other attributes.
*/ */
@ -2289,11 +2306,11 @@ public final class CSVFormat implements Serializable {
final Set<String> dupCheckSet = new HashSet<>(headers.length); final Set<String> dupCheckSet = new HashSet<>(headers.length);
final boolean rejectEmpty = duplicateHeaderMode != DuplicateHeaderMode.ALLOW_EMPTY; final boolean rejectEmpty = duplicateHeaderMode != DuplicateHeaderMode.ALLOW_EMPTY;
for (final String header : headers) { for (final String header : headers) {
final boolean empty = header == null || header.isEmpty(); final boolean blank = isBlank(header);
if (rejectEmpty && empty) { if (rejectEmpty && blank) {
throw new IllegalArgumentException("Header is empty"); throw new IllegalArgumentException("Header is empty");
} }
if (!empty && !dupCheckSet.add(header)) { if (!blank && !dupCheckSet.add(header)) {
throw new IllegalArgumentException(String.format("Header '%s' is a duplicate in %s", header, Arrays.toString(headers))); throw new IllegalArgumentException(String.format("Header '%s' is a duplicate in %s", header, Arrays.toString(headers)));
} }
} }
@ -2301,7 +2318,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} that allows duplicate header names. * Builds a new {@code CSVFormat} that allows duplicate header names.
* *
* @return a new {@code CSVFormat} that allows duplicate header names * @return a new {@code CSVFormat} that allows duplicate header names
* @since 1.7 * @since 1.7
@ -2313,7 +2330,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with duplicate header names behavior set to the given value. * Builds a new {@code CSVFormat} with duplicate header names behavior set to the given value.
* *
* @param allowDuplicateHeaderNames the duplicate header names behavior, true to allow, false to disallow. * @param allowDuplicateHeaderNames the duplicate header names behavior, true to allow, false to disallow.
* @return a new {@code CSVFormat} with duplicate header names behavior set to the given value. * @return a new {@code CSVFormat} with duplicate header names behavior set to the given value.
@ -2327,7 +2344,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the missing column names behavior of the format set to {@code true}. * Builds a new {@code CSVFormat} with the missing column names behavior of the format set to {@code true}.
* *
* @return A new CSVFormat that is equal to this but with the specified missing column names behavior. * @return A new CSVFormat that is equal to this but with the specified missing column names behavior.
* @see Builder#setAllowMissingColumnNames(boolean) * @see Builder#setAllowMissingColumnNames(boolean)
@ -2340,7 +2357,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the missing column names behavior of the format set to the given value. * Builds a new {@code CSVFormat} with the missing column names behavior of the format set to the given value.
* *
* @param allowMissingColumnNames the missing column names behavior, {@code true} to allow missing column names in the header line, {@code false} to cause * @param allowMissingColumnNames the missing column names behavior, {@code true} to allow missing column names in the header line, {@code false} to cause
* an {@link IllegalArgumentException} to be thrown. * an {@link IllegalArgumentException} to be thrown.
@ -2353,7 +2370,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with whether to flush on close. * Builds a new {@code CSVFormat} with whether to flush on close.
* *
* @param autoFlush whether to flush on close. * @param autoFlush whether to flush on close.
* *
@ -2367,7 +2384,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the comment start marker of the format set to the specified character. * Builds a new {@code CSVFormat} with the comment start marker of the format set to the specified character.
* *
* Note that the comment start character is only recognized at the start of a line. * Note that the comment start character is only recognized at the start of a line.
* *
@ -2382,7 +2399,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the comment start marker of the format set to the specified character. * Builds a new {@code CSVFormat} with the comment start marker of the format set to the specified character.
* *
* Note that the comment start character is only recognized at the start of a line. * Note that the comment start character is only recognized at the start of a line.
* *
@ -2397,7 +2414,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the delimiter of the format set to the specified character. * Builds a new {@code CSVFormat} with the delimiter of the format set to the specified character.
* *
* @param delimiter the delimiter character * @param delimiter the delimiter character
* @return A new CSVFormat that is equal to this with the specified character as delimiter * @return A new CSVFormat that is equal to this with the specified character as delimiter
@ -2410,7 +2427,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the escape character of the format set to the specified character. * Builds a new {@code CSVFormat} with the escape character of the format set to the specified character.
* *
* @param escape the escape character * @param escape the escape character
* @return A new CSVFormat that is equal to this but with the specified character as the escape character * @return A new CSVFormat that is equal to this but with the specified character as the escape character
@ -2423,7 +2440,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the escape character of the format set to the specified character. * Builds a new {@code CSVFormat} with the escape character of the format set to the specified character.
* *
* @param escape the escape character, use {@code null} to disable * @param escape the escape character, use {@code null} to disable
* @return A new CSVFormat that is equal to this but with the specified character as the escape character * @return A new CSVFormat that is equal to this but with the specified character as the escape character
@ -2436,7 +2453,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} using the first record as header. * Builds a new {@code CSVFormat} using the first record as header.
* *
* <p> * <p>
* Calling this method is equivalent to calling: * Calling this method is equivalent to calling:
@ -2463,7 +2480,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the header of the format defined by the enum class. * Builds a new {@code CSVFormat} with the header of the format defined by the enum class.
* *
* <p> * <p>
* Example: * Example:
@ -2493,7 +2510,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the header of the format set from the result set metadata. The header can either be parsed automatically from the * Builds a new {@code CSVFormat} with the header of the format set from the result set metadata. The header can either be parsed automatically from the
* input file with: * input file with:
* *
* <pre> * <pre>
@ -2521,7 +2538,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the header of the format set from the result set metadata. The header can either be parsed automatically from the * Builds a new {@code CSVFormat} with the header of the format set from the result set metadata. The header can either be parsed automatically from the
* input file with: * input file with:
* *
* <pre> * <pre>
@ -2549,7 +2566,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the header of the format set to the given values. The header can either be parsed automatically from the input file * Builds a new {@code CSVFormat} with the header of the format set to the given values. The header can either be parsed automatically from the input file
* with: * with:
* *
* <pre> * <pre>
@ -2576,7 +2593,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the header comments of the format set to the given values. The comments will be printed first, before the headers. * Builds a new {@code CSVFormat} with the header comments of the format set to the given values. The comments will be printed first, before the headers.
* This setting is ignored by the parser. * This setting is ignored by the parser.
* *
* <pre> * <pre>
@ -2595,7 +2612,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the empty line skipping behavior of the format set to {@code true}. * Builds a new {@code CSVFormat} with the empty line skipping behavior of the format set to {@code true}.
* *
* @return A new CSVFormat that is equal to this but with the specified empty line skipping behavior. * @return A new CSVFormat that is equal to this but with the specified empty line skipping behavior.
* @see Builder#setIgnoreEmptyLines(boolean) * @see Builder#setIgnoreEmptyLines(boolean)
@ -2608,7 +2625,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the empty line skipping behavior of the format set to the given value. * Builds a new {@code CSVFormat} with the empty line skipping behavior of the format set to the given value.
* *
* @param ignoreEmptyLines the empty line skipping behavior, {@code true} to ignore the empty lines between the records, {@code false} to translate empty * @param ignoreEmptyLines the empty line skipping behavior, {@code true} to ignore the empty lines between the records, {@code false} to translate empty
* lines to empty records. * lines to empty records.
@ -2621,7 +2638,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the header ignore case behavior set to {@code true}. * Builds a new {@code CSVFormat} with the header ignore case behavior set to {@code true}.
* *
* @return A new CSVFormat that will ignore case header name. * @return A new CSVFormat that will ignore case header name.
* @see Builder#setIgnoreHeaderCase(boolean) * @see Builder#setIgnoreHeaderCase(boolean)
@ -2634,7 +2651,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with whether header names should be accessed ignoring case. * Builds a new {@code CSVFormat} with whether header names should be accessed ignoring case.
* *
* @param ignoreHeaderCase the case mapping behavior, {@code true} to access name/values, {@code false} to leave the mapping as is. * @param ignoreHeaderCase the case mapping behavior, {@code true} to access name/values, {@code false} to leave the mapping as is.
* @return A new CSVFormat that will ignore case header name if specified as {@code true} * @return A new CSVFormat that will ignore case header name if specified as {@code true}
@ -2647,7 +2664,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the parser trimming behavior of the format set to {@code true}. * Builds a new {@code CSVFormat} with the parser trimming behavior of the format set to {@code true}.
* *
* @return A new CSVFormat that is equal to this but with the specified parser trimming behavior. * @return A new CSVFormat that is equal to this but with the specified parser trimming behavior.
* @see Builder#setIgnoreSurroundingSpaces(boolean) * @see Builder#setIgnoreSurroundingSpaces(boolean)
@ -2660,7 +2677,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the parser trimming behavior of the format set to the given value. * Builds a new {@code CSVFormat} with the parser trimming behavior of the format set to the given value.
* *
* @param ignoreSurroundingSpaces the parser trimming behavior, {@code true} to remove the surrounding spaces, {@code false} to leave the spaces as is. * @param ignoreSurroundingSpaces the parser trimming behavior, {@code true} to remove the surrounding spaces, {@code false} to leave the spaces as is.
* @return A new CSVFormat that is equal to this but with the specified trimming behavior. * @return A new CSVFormat that is equal to this but with the specified trimming behavior.
@ -2672,7 +2689,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with conversions to and from null for strings on input and output. * Builds a new {@code CSVFormat} with conversions to and from null for strings on input and output.
* <ul> * <ul>
* <li><strong>Reading:</strong> Converts strings equal to the given {@code nullString} to {@code null} when reading records.</li> * <li><strong>Reading:</strong> Converts strings equal to the given {@code nullString} to {@code null} when reading records.</li>
* <li><strong>Writing:</strong> Writes {@code null} as the given {@code nullString} when writing records.</li> * <li><strong>Writing:</strong> Writes {@code null} as the given {@code nullString} when writing records.</li>
@ -2688,7 +2705,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the quoteChar of the format set to the specified character. * Builds a new {@code CSVFormat} with the quoteChar of the format set to the specified character.
* *
* @param quoteChar the quote character * @param quoteChar the quote character
* @return A new CSVFormat that is equal to this but with the specified character as quoteChar * @return A new CSVFormat that is equal to this but with the specified character as quoteChar
@ -2701,7 +2718,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the quoteChar of the format set to the specified character. * Builds a new {@code CSVFormat} with the quoteChar of the format set to the specified character.
* *
* @param quoteChar the quote character, use {@code null} to disable. * @param quoteChar the quote character, use {@code null} to disable.
* @return A new CSVFormat that is equal to this but with the specified character as quoteChar * @return A new CSVFormat that is equal to this but with the specified character as quoteChar
@ -2714,7 +2731,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the output quote policy of the format set to the specified value. * Builds a new {@code CSVFormat} with the output quote policy of the format set to the specified value.
* *
* @param quoteMode the quote policy to use for output. * @param quoteMode the quote policy to use for output.
* *
@ -2727,7 +2744,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the record separator of the format set to the specified character. * Builds a new {@code CSVFormat} with the record separator of the format set to the specified character.
* *
* <p> * <p>
* <strong>Note:</strong> This setting is only used during printing and does not affect parsing. Parsing currently only works for inputs with '\n', '\r' and * <strong>Note:</strong> This setting is only used during printing and does not affect parsing. Parsing currently only works for inputs with '\n', '\r' and
@ -2744,7 +2761,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the record separator of the format set to the specified String. * Builds a new {@code CSVFormat} with the record separator of the format set to the specified String.
* *
* <p> * <p>
* <strong>Note:</strong> This setting is only used during printing and does not affect parsing. Parsing currently only works for inputs with '\n', '\r' and * <strong>Note:</strong> This setting is only used during printing and does not affect parsing. Parsing currently only works for inputs with '\n', '\r' and
@ -2762,7 +2779,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with skipping the header record set to {@code true}. * Builds a new {@code CSVFormat} with skipping the header record set to {@code true}.
* *
* @return A new CSVFormat that is equal to this but with the specified skipHeaderRecord setting. * @return A new CSVFormat that is equal to this but with the specified skipHeaderRecord setting.
* @see Builder#setSkipHeaderRecord(boolean) * @see Builder#setSkipHeaderRecord(boolean)
@ -2776,7 +2793,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with whether to skip the header record. * Builds a new {@code CSVFormat} with whether to skip the header record.
* *
* @param skipHeaderRecord whether to skip the header record. * @param skipHeaderRecord whether to skip the header record.
* @return A new CSVFormat that is equal to this but with the specified skipHeaderRecord setting. * @return A new CSVFormat that is equal to this but with the specified skipHeaderRecord setting.
@ -2789,7 +2806,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with the record separator of the format set to the operating system's line separator string, typically CR+LF on Windows * Builds a new {@code CSVFormat} with the record separator of the format set to the operating system's line separator string, typically CR+LF on Windows
* and LF on Linux. * and LF on Linux.
* *
* <p> * <p>
@ -2807,7 +2824,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} to add a trailing delimiter. * Builds a new {@code CSVFormat} to add a trailing delimiter.
* *
* @return A new CSVFormat that is equal to this but with the trailing delimiter setting. * @return A new CSVFormat that is equal to this but with the trailing delimiter setting.
* @since 1.3 * @since 1.3
@ -2819,7 +2836,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with whether to add a trailing delimiter. * Builds a new {@code CSVFormat} with whether to add a trailing delimiter.
* *
* @param trailingDelimiter whether to add a trailing delimiter. * @param trailingDelimiter whether to add a trailing delimiter.
* @return A new CSVFormat that is equal to this but with the specified trailing delimiter setting. * @return A new CSVFormat that is equal to this but with the specified trailing delimiter setting.
@ -2832,7 +2849,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} to trim leading and trailing blanks. See {@link #getTrim()} for details of where this is used. * Builds a new {@code CSVFormat} to trim leading and trailing blanks. See {@link #getTrim()} for details of where this is used.
* *
* @return A new CSVFormat that is equal to this but with the trim setting on. * @return A new CSVFormat that is equal to this but with the trim setting on.
* @since 1.3 * @since 1.3
@ -2844,7 +2861,7 @@ public final class CSVFormat implements Serializable {
} }
/** /**
* Returns a new {@code CSVFormat} with whether to trim leading and trailing blanks. See {@link #getTrim()} for details of where this is used. * Builds a new {@code CSVFormat} with whether to trim leading and trailing blanks. See {@link #getTrim()} for details of where this is used.
* *
* @param trim whether to trim leading and trailing blanks. * @param trim whether to trim leading and trailing blanks.
* @return A new CSVFormat that is equal to this but with the specified trim setting. * @return A new CSVFormat that is equal to this but with the specified trim setting.

View File

@ -441,12 +441,11 @@ public final class CSVParser implements Iterable<CSVRecord>, Closeable {
} }
private void addRecordValue(final boolean lastRecord) { private void addRecordValue(final boolean lastRecord) {
final String input = this.reusableToken.content.toString(); final String input = this.format.trim(this.reusableToken.content.toString());
final String inputClean = this.format.getTrim() ? input.trim() : input; if (lastRecord && input.isEmpty() && this.format.getTrailingDelimiter()) {
if (lastRecord && inputClean.isEmpty() && this.format.getTrailingDelimiter()) {
return; return;
} }
this.recordList.add(handleNull(inputClean)); this.recordList.add(handleNull(input));
} }
/** /**
@ -502,8 +501,8 @@ public final class CSVParser implements Iterable<CSVRecord>, Closeable {
if (headerRecord != null) { if (headerRecord != null) {
for (int i = 0; i < headerRecord.length; i++) { for (int i = 0; i < headerRecord.length; i++) {
final String header = headerRecord[i]; final String header = headerRecord[i];
final boolean emptyHeader = header == null || header.trim().isEmpty(); final boolean blankHeader = CSVFormat.isBlank(header);
if (emptyHeader && !this.format.getAllowMissingColumnNames()) { if (blankHeader && !this.format.getAllowMissingColumnNames()) {
throw new IllegalArgumentException( throw new IllegalArgumentException(
"A header name is missing in " + Arrays.toString(headerRecord)); "A header name is missing in " + Arrays.toString(headerRecord));
} }
@ -513,7 +512,7 @@ public final class CSVParser implements Iterable<CSVRecord>, Closeable {
final boolean duplicatesAllowed = headerMode == DuplicateHeaderMode.ALLOW_ALL; final boolean duplicatesAllowed = headerMode == DuplicateHeaderMode.ALLOW_ALL;
final boolean emptyDuplicatesAllowed = headerMode == DuplicateHeaderMode.ALLOW_EMPTY; final boolean emptyDuplicatesAllowed = headerMode == DuplicateHeaderMode.ALLOW_EMPTY;
if (containsHeader && !duplicatesAllowed && !(emptyHeader && emptyDuplicatesAllowed)) { if (containsHeader && !duplicatesAllowed && !(blankHeader && emptyDuplicatesAllowed)) {
throw new IllegalArgumentException( throw new IllegalArgumentException(
String.format( String.format(
"The header contains a duplicate name: \"%s\" in %s. If this is valid then use CSVFormat.Builder.setDuplicateHeaderMode().", "The header contains a duplicate name: \"%s\" in %s. If this is valid then use CSVFormat.Builder.setDuplicateHeaderMode().",

View File

@ -31,7 +31,7 @@ public enum DuplicateHeaderMode {
ALLOW_ALL, ALLOW_ALL,
/** /**
* Allows duplicate headers only if they're empty strings or null. * Allows duplicate headers only if they're empty, blank, or null strings.
*/ */
ALLOW_EMPTY, ALLOW_EMPTY,

View File

@ -68,24 +68,25 @@ public class CSVDuplicateHeaderTest {
// Duplicate empty names // Duplicate empty names
Arguments.of(DuplicateHeaderMode.DISALLOW, false, new String[] {"", ""}, false), Arguments.of(DuplicateHeaderMode.DISALLOW, false, new String[] {"", ""}, false),
// Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, false, new String[] {"", ""}, false),
// Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] {"", ""}, false),
Arguments.of(DuplicateHeaderMode.DISALLOW, true, new String[] {"", ""}, false), Arguments.of(DuplicateHeaderMode.DISALLOW, true, new String[] {"", ""}, false),
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] {"", ""}, true), Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] {"", ""}, true),
Arguments.of(DuplicateHeaderMode.ALLOW_ALL, true, new String[] {"", ""}, true), Arguments.of(DuplicateHeaderMode.ALLOW_ALL, true, new String[] {"", ""}, true),
// Duplicate blank names // Duplicate blank names (1 space)
Arguments.of(DuplicateHeaderMode.DISALLOW, false, new String[] {" ", " "}, false), Arguments.of(DuplicateHeaderMode.DISALLOW, false, new String[] {" ", " "}, false),
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, false, new String[] {" ", " "}, false),
// Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] {" ", " "}, false),
Arguments.of(DuplicateHeaderMode.DISALLOW, true, new String[] {" ", " "}, false), Arguments.of(DuplicateHeaderMode.DISALLOW, true, new String[] {" ", " "}, false),
// Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] {" ", " "}, true), Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] {" ", " "}, true),
Arguments.of(DuplicateHeaderMode.ALLOW_ALL, true, new String[] {" ", " "}, true), Arguments.of(DuplicateHeaderMode.ALLOW_ALL, true, new String[] {" ", " "}, true),
// Duplicate blank names (3 spaces)
Arguments.of(DuplicateHeaderMode.DISALLOW, false, new String[] {" ", " "}, false),
Arguments.of(DuplicateHeaderMode.DISALLOW, true, new String[] {" ", " "}, false),
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] {" ", " "}, true),
Arguments.of(DuplicateHeaderMode.ALLOW_ALL, true, new String[] {" ", " "}, true),
// Duplicate non-empty and empty names // Duplicate non-empty and empty names
Arguments.of(DuplicateHeaderMode.DISALLOW, false, new String[] {"A", "A", "", ""}, false), Arguments.of(DuplicateHeaderMode.DISALLOW, false, new String[] {"A", "A", "", ""}, false),
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, false, new String[] {"A", "A", "", ""}, false), Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, false, new String[] {"A", "A", "", ""}, false),
// Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] {"A", "A", "", ""}, false),
Arguments.of(DuplicateHeaderMode.DISALLOW, true, new String[] {"A", "A", "", ""}, false), Arguments.of(DuplicateHeaderMode.DISALLOW, true, new String[] {"A", "A", "", ""}, false),
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] {"A", "A", "", ""}, false), Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] {"A", "A", "", ""}, false),
Arguments.of(DuplicateHeaderMode.ALLOW_ALL, true, new String[] {"A", "A", "", ""}, true), Arguments.of(DuplicateHeaderMode.ALLOW_ALL, true, new String[] {"A", "A", "", ""}, true),
@ -93,7 +94,6 @@ public class CSVDuplicateHeaderTest {
// Duplicate non-empty and blank names // Duplicate non-empty and blank names
Arguments.of(DuplicateHeaderMode.DISALLOW, false, new String[] {"A", "A", " ", " "}, false), Arguments.of(DuplicateHeaderMode.DISALLOW, false, new String[] {"A", "A", " ", " "}, false),
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, false, new String[] {"A", "A", " ", " "}, false), Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, false, new String[] {"A", "A", " ", " "}, false),
// Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] {"A", "A", " ", " "}, false),
Arguments.of(DuplicateHeaderMode.DISALLOW, true, new String[] {"A", "A", " ", " "}, false), Arguments.of(DuplicateHeaderMode.DISALLOW, true, new String[] {"A", "A", " ", " "}, false),
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] {"A", "A", " ", " "}, false), Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] {"A", "A", " ", " "}, false),
Arguments.of(DuplicateHeaderMode.ALLOW_ALL, true, new String[] {"A", "A", " ", " "}, true) Arguments.of(DuplicateHeaderMode.ALLOW_ALL, true, new String[] {"A", "A", " ", " "}, true)
@ -106,10 +106,14 @@ public class CSVDuplicateHeaderTest {
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, false, new String[] { "", "" }, false), Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, false, new String[] { "", "" }, false),
Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] { "", "" }, false), Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] { "", "" }, false),
// Duplicate blank names // Duplicate blank names (1 space)
Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] { " ", " " }, false), Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] { " ", " " }, false),
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] { " ", " " }, true), Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] { " ", " " }, true),
// Duplicate blank names (3 spaces)
Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] { " ", " " }, false),
Arguments.of(DuplicateHeaderMode.ALLOW_EMPTY, true, new String[] { " ", " " }, true),
// Duplicate non-empty and empty names // Duplicate non-empty and empty names
Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] { "A", "A", "", "" }, false), Arguments.of(DuplicateHeaderMode.ALLOW_ALL, false, new String[] { "A", "A", "", "" }, false),
@ -118,12 +122,12 @@ public class CSVDuplicateHeaderTest {
} }
/** /**
* Test duplicate headers with the CSVFormat. * Tests duplicate headers with the CSVFormat.
* *
* @param duplicateHeaderMode the duplicate header mode * @param duplicateHeaderMode the duplicate header mode
* @param allowMissingColumnNames the allow missing column names flag * @param allowMissingColumnNames the allow missing column names flag (only used for parsing)
* @param headers the headers * @param headers the headers
* @param valid true if the settings are expected to be valid * @param valid true if the settings are expected to be valid, otherwise expect a IllegalArgumentException
*/ */
@ParameterizedTest @ParameterizedTest
@MethodSource(value = {"duplicateHeaderData"}) @MethodSource(value = {"duplicateHeaderData"})
@ -146,12 +150,12 @@ public class CSVDuplicateHeaderTest {
} }
/** /**
* Test duplicate headers with the CSVParser. * Tests duplicate headers with the CSVParser.
* *
* @param duplicateHeaderMode the duplicate header mode * @param duplicateHeaderMode the duplicate header mode
* @param allowMissingColumnNames the allow missing column names flag * @param allowMissingColumnNames the allow missing column names flag (only used for parsing)
* @param headers the headers (joined with the CSVFormat delimiter to create a string input) * @param headers the headers (joined with the CSVFormat delimiter to create a string input)
* @param valid true if the settings are expected to be valid * @param valid true if the settings are expected to be valid, otherwise expect a IllegalArgumentException
* @throws IOException Signals that an I/O exception has occurred. * @throws IOException Signals that an I/O exception has occurred.
*/ */
@ParameterizedTest @ParameterizedTest