com.univocity.parsers.conversions.BooleanConversion Maven / Gradle / Ivy
/*******************************************************************************
* Copyright 2014 Univocity Software Pty Ltd
*
* 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.univocity.parsers.conversions;
import com.univocity.parsers.common.*;
import java.util.*;
/**
* Converts Strings to Booleans and vice versa
*
* This class supports multiple representations of boolean values. For example, you can define conversions from different Strings such as "Yes, Y, 1" to true, and
* "No, N, 0" to false.
*
*
The reverse conversion from a Boolean to String (in {@link BooleanConversion#revert(Boolean)} will return the first String provided in this class constructor
*
Using the previous example, a call to {@code revert(true)} will produce "Yes" and a call {@code revert(false)} will produce "No".
*
* @author Univocity Software Pty Ltd - [email protected]
*/
public class BooleanConversion extends ObjectConversion {
private String defaultForTrue;
private String defaultForFalse;
private final Set falseValues = new LinkedHashSet();
private final Set trueValues = new LinkedHashSet();
/**
* Creates conversions from String to Boolean.
* This default constructor assumes the output of a conversion should be null when input is null
* The list of Strings that identify "true" the list of Strings that identify "false" are mandatory.
*
* @param valuesForTrue Strings that identify the boolean value true. The first element will be returned when executing {@code revert(true)}
* @param valuesForFalse Strings that identify the boolean value false. The first element will be returned when executing {@code #revert(false)}
*/
public BooleanConversion(String[] valuesForTrue, String[] valuesForFalse) {
this(null, null, valuesForTrue, valuesForFalse);
}
/**
* Creates a Conversion from String to Boolean with default values to return when the input is null.
*
The list of Strings that identify "true" the list of Strings that identify "false" are mandatory.
*
* @param valueIfStringIsNull default Boolean value to be returned when the input String is null. Used when {@link ObjectConversion#execute(String)} is invoked.
* @param valueIfObjectIsNull default String value to be returned when a Boolean input is null. Used when {@link BooleanConversion#revert(Boolean)} is invoked.
* @param valuesForTrue Strings that identify the boolean value true. The first element will be returned when executing {@code revert(true)}
* @param valuesForFalse Strings that identify the boolean value false. The first element will be returned when executing {@code #revert(false)}
*/
public BooleanConversion(Boolean valueIfStringIsNull, String valueIfObjectIsNull, String[] valuesForTrue, String[] valuesForFalse) {
super(valueIfStringIsNull, valueIfObjectIsNull);
ArgumentUtils.notEmpty("Values for true", valuesForTrue);
ArgumentUtils.notEmpty("Values for false", valuesForFalse);
Collections.addAll(falseValues, valuesForFalse);
Collections.addAll(trueValues, valuesForTrue);
normalize(falseValues);
normalize(trueValues);
for (String falseValue : falseValues) {
if (trueValues.contains(falseValue)) {
throw new DataProcessingException("Ambiguous string representation for both false and true values: '" + falseValue + '\'');
}
}
defaultForTrue = valuesForTrue[0];
defaultForFalse = valuesForFalse[0];
}
/**
* Converts a Boolean back to a String
*
The return value depends on the list of values for true/false provided in the constructor of this class.
*
* @param input the Boolean to be converted to a String
*
* @return a String representation for this boolean value, or the value of {@link BooleanConversion#getValueIfObjectIsNull()} if the Boolean input is null.
*/
@Override
public String revert(Boolean input) {
if (input != null) {
if (Boolean.FALSE.equals(input)) {
return defaultForFalse;
}
if (Boolean.TRUE.equals(input)) {
return defaultForTrue;
}
}
return getValueIfObjectIsNull();
}
/**
* Converts a String to a Boolean
*
* @param input a String to be converted into a Boolean value.
*
* @return true if the input String is part of {@link BooleanConversion#trueValues}, false if the input String is part of {@link BooleanConversion#falseValues}, or {@link BooleanConversion#getValueIfStringIsNull()} if the input String is null.
*/
@Override
protected Boolean fromString(String input) {
if (input != null) {
return getBoolean(input, trueValues, falseValues);
}
return super.getValueIfStringIsNull();
}
/**
* Returns the {@code Boolean} value represented by a {@code String}, as defined by sets of Strings that represent {@code true} and {@code false} values.
* @param booleanString the value that represents either {@code true} or {@code false}
* @param trueValues a set of possible string values that represent {@code true}. If empty, then "true" will be assumed as the only acceptable representation.
* @param falseValues a set of possible string values that represent {@code false}. If empty, then "false" will be assumed as the only acceptable representation.
* @return the boolean value that the input string represents
* @throws DataProcessingException if the input string does not match any value provided in neither set of possible values.
*/
public static Boolean getBoolean(String booleanString, String[] trueValues, String[] falseValues) {
trueValues = trueValues == null || trueValues.length == 0 ? new String[]{"true"} : trueValues;
falseValues = falseValues == null || falseValues.length == 0 ? new String[]{"false"} : falseValues;
BooleanConversion tmp = new BooleanConversion(trueValues, falseValues);
return getBoolean(booleanString, tmp.trueValues, tmp.falseValues);
}
private static Boolean getBoolean(String defaultString, Set trueValues, Set falseValues) {
String normalized = normalize(defaultString);
if (falseValues.contains(normalized)) {
return Boolean.FALSE;
}
if (trueValues.contains(normalized)) {
return Boolean.TRUE;
}
DataProcessingException exception = new DataProcessingException("Unable to convert '{value}' to Boolean. Allowed Strings are: " + trueValues + " for true; and " + falseValues + " for false.");
exception.setValue(defaultString);
throw exception;
}
/**
* Normalizes a given String by trimming whitespaces and converting it to lower case.
*
* @param string a String to be normalized.
*
* @return the normalized version of the original String.
*/
private static String normalize(String string) {
if (string == null) {
return null;
}
return string.trim().toLowerCase();
}
/**
* Normalizes the Strings in a given collection by trimming all elements and converting them to lower case.
*
* @param strings a String collection with elements to be normalized. The original contents of the collection will be modified.
*/
private static void normalize(Collection strings) {
LinkedHashSet normalized = new LinkedHashSet(strings.size());
for (String string : strings) {
if (string == null) {
normalized.add(null);
} else {
normalized.add(string.trim().toLowerCase());
}
}
strings.clear();
strings.addAll(normalized);
}
}