StringConverters.java

// Generated by delombok at Mon Jan 06 07:19:11 UTC 2025
package de.larssh.utils.text;

import static java.util.stream.Collectors.joining;
import java.io.IOException;
import java.io.Reader;
import java.io.StringReader;
import java.io.UncheckedIOException;
import java.io.UnsupportedEncodingException;
import java.net.URLDecoder;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.Base64;
import java.util.Collection;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;

/**
 * This class contains helper methods for decoding and encoding strings.
 */
public final class StringConverters {
	/**
	 * Decodes {@code value} as Base64 string.
	 *
	 * <p>
	 * This method is equivalent to using {@link Base64#getDecoder()} with the char
	 * set {@link Strings#DEFAULT_CHARSET}.
	 *
	 * @param value the value to be decoded
	 * @return the decoded value
	 */
	public static String decodeBase64(final String value) {
		return new String(Base64.getDecoder().decode(value), Strings.DEFAULT_CHARSET);
	}

	/**
	 * Decodes {@code value} using the Base64 MIME type decoding scheme.
	 *
	 * <p>
	 * This method is equivalent to using {@link Base64#getMimeDecoder()} with the
	 * char set {@link Strings#DEFAULT_CHARSET}.
	 *
	 * @param value the value to be decoded
	 * @return the decoded value
	 */
	public static String decodeBase64Mime(final String value) {
		return new String(Base64.getMimeDecoder().decode(value), Strings.DEFAULT_CHARSET);
	}

	/**
	 * Decodes {@code value} as Base64 URL and filename-safe string.
	 *
	 * <p>
	 * This method is equivalent to using {@link Base64#getUrlDecoder()} with the
	 * char set {@link Strings#DEFAULT_CHARSET}.
	 *
	 * @param value the value to be decoded
	 * @return the decoded value
	 */
	public static String decodeBase64Url(final String value) {
		return new String(Base64.getUrlDecoder().decode(value), Strings.DEFAULT_CHARSET);
	}

	/**
	 * Parses the CSV data given by {@code data}.
	 *
	 * <p>
	 * To process {@link Reader} input, refer to
	 * {@link Csv#parse(Reader, char, char)}.
	 *
	 * @param data      the CSV input data
	 * @param separator the CSV separator character
	 * @param escaper   the CSV escaping character
	 * @return an object representing the parsed CSV data
	 * @throws IllegalArgumentException on illegal {@code separator} or
	 *                                  {@code escaper} value
	 */
	@SuppressFBWarnings(value = "EXS_EXCEPTION_SOFTENING_NO_CONSTRAINTS", justification = "converting checked to unchecked exception, that should never be thrown at all")
	public static Csv decodeCsv(final String data, final char separator, final char escaper) {
		try (Reader reader = new StringReader(data)) {
			return Csv.parse(reader, separator, escaper);
		} catch (final IOException e) {
			throw new UncheckedIOException(e);
		}
	}

	/**
	 * Decodes an {@code application/x-www-form-urlencoded} string.
	 *
	 * <p>
	 * This method is equivalent to using {@link URLDecoder#decode(String)} with the
	 * char set {@link StandardCharsets#UTF_8} as the
	 * <a href= "https://www.w3.org/TR/html40/appendix/notes.html#non-ascii-chars">
	 * World Wide Web Consortium Recommendation</a> states that UTF-8 should be
	 * used.
	 *
	 * @param value the value to be decoded
	 * @return the decoded value
	 * @throws StringParseException on illegal or incomplete hex characters
	 */
	@SuppressFBWarnings(value = "EXS_EXCEPTION_SOFTENING_NO_CONSTRAINTS", justification = "converting checked to unchecked exception, that should never be thrown at all")
	public static String decodeUrl(final String value) throws StringParseException {
		try {
			return URLDecoder.decode(value, StandardCharsets.UTF_8.name());
		} catch (final IllegalArgumentException e) {
			throw new StringParseException(e, "Failed decoding URL.");
		} catch (final UnsupportedEncodingException e) {
			throw new UncheckedIOException(e);
		}
	}

	/**
	 * Translates {@code value} into Base64 format.
	 *
	 * <p>
	 * This method is equivalent to using {@link Base64#getEncoder()} with the char
	 * set {@link Strings#DEFAULT_CHARSET}.
	 *
	 * @param value the value to be encoded
	 * @return the encoded value
	 */
	public static String encodeBase64(final String value) {
		return new String(Base64.getEncoder().encode(value.getBytes(Strings.DEFAULT_CHARSET)), Strings.DEFAULT_CHARSET);
	}

	/**
	 * Translates {@code value} into Base64 MIME type encoding scheme.
	 *
	 * <p>
	 * This method is equivalent to using {@link Base64#getMimeEncoder()} with the
	 * char set {@link Strings#DEFAULT_CHARSET}.
	 *
	 * @param value the value to be encoded
	 * @return the encoded value
	 */
	public static String encodeBase64Mime(final String value) {
		return new String(Base64.getMimeEncoder().encode(value.getBytes(Strings.DEFAULT_CHARSET)), Strings.DEFAULT_CHARSET);
	}

	/**
	 * Translates {@code value} into Base64 URL and filename-safe format.
	 *
	 * <p>
	 * This method is equivalent to using {@link Base64#getUrlEncoder()} with the
	 * charset {@link Strings#DEFAULT_CHARSET}.
	 *
	 * @param value the value to be encoded
	 * @return the encoded value
	 */
	public static String encodeBase64Url(final String value) {
		return new String(Base64.getUrlEncoder().encode(value.getBytes(Strings.DEFAULT_CHARSET)), Strings.DEFAULT_CHARSET);
	}

	/**
	 * Encodes {@code data} as CSV document. Occurrences of special characters are
	 * encoded using {@code escaper} and values are separated using
	 * {@code separator}.
	 *
	 * <p>
	 * For more information on single value escaping see
	 * {@link #encodeCsvValue(String, char, char)}.
	 *
	 * @param data      the data to be encoded
	 * @param separator the CSV separator character
	 * @param escaper   the CSV escaping character
	 * @return the encoded CSV document
	 */
	public static String encodeCsv(final Collection<? extends Collection<String>> data, final char separator, final char escaper) {
		return  //
		data.stream().map(row -> encodeCsvRow(row, separator, escaper)).collect(joining(Strings.NEW_LINE));
	}

	/**
	 * Encodes {@code values} as one CSV row. Occurrences of special characters are
	 * encoded using {@code escaper} and values are separated using
	 * {@code separator}.
	 *
	 * <p>
	 * For more information on single value escaping see
	 * {@link #encodeCsvValue(String, char, char)}.
	 *
	 * @param values    the values to be encoded
	 * @param separator the CSV separator character
	 * @param escaper   the CSV escaping character
	 * @return the encoded CSV row
	 */
	public static String encodeCsvRow(final Collection<? extends String> values, final char separator, final char escaper) {
		return values.stream().map(value -> encodeCsvValue(value, separator, escaper)).collect(joining(Character.toString(separator)));
	}

	/**
	 * Encodes {@code value} into a single CSV value. All occurrences of
	 * {@code escaper} are escaped (doubled). If {@code value} contains
	 * inappropriate characters, it is surrounded by {@code escaper}.
	 *
	 * <p>
	 * Inappropriate characters are {@code '\r'}, {@code '\n'}, {@code escaper} and
	 * {@code separator}. The latter is used for the list of inappropriate
	 * characters only.
	 *
	 * <p>
	 * <b>Examples:</b>
	 * <table>
	 * <caption>Examples</caption>
	 * <tr>
	 * <th>Parameter</th>
	 * <th>Return Value</th>
	 * </tr>
	 * <tr>
	 * <td>EMPTY</td>
	 * <td>EMPTY</td>
	 * </tr>
	 * <tr>
	 * <td>abc</td>
	 * <td>abc</td>
	 * </tr>
	 * <tr>
	 * <td>abc"xyz</td>
	 * <td>"abc""xyz"</td>
	 * </tr>
	 * <tr>
	 * <td>"</td>
	 * <td>""""</td>
	 * </tr>
	 * </table>
	 *
	 * @param value     the value to be encoded
	 * @param separator the CSV separator character
	 * @param escaper   the CSV escaping character
	 * @return the encoded value
	 */
	@SuppressWarnings("PMD.CyclomaticComplexity")
	public static String encodeCsvValue(final String value, final char separator, final char escaper) {
		CsvParser.assertCsvInput(separator, escaper);
		final int length = value.length();
		int index = 0;
		boolean needsEscaping = false;
		for (; index < length && !needsEscaping; index += 1) {
			final char character = value.charAt(index);
			if (character == escaper) {
				needsEscaping = true;
				index -= 1;
			} else if (character == separator || character == '\r' || character == '\n') {
				needsEscaping = true;
			}
		}
		if (!needsEscaping) {
			return value;
		}
		final StringBuilder builder = new StringBuilder(length + 2);
		builder.append(escaper);
		if (index > 0) {
			builder.append(value, 0, index);
		}
		for (; index < length; index += 1) {
			final char character = value.charAt(index);
			if (character == escaper) {
				builder.append(escaper);
			}
			builder.append(character);
		}
		return builder.append(escaper).toString();
	}

	/**
	 * Translates a string into {@code application/x-www-form-urlencoded} format
	 * with the char set {@link StandardCharsets#UTF_8} as the
	 * <a href= "https://www.w3.org/TR/html40/appendix/notes.html#non-ascii-chars">
	 * World Wide Web Consortium Recommendation</a> states that UTF-8 should be
	 * used.
	 *
	 * @param value the value to be encoded
	 * @return the encoded value
	 */
	@SuppressFBWarnings(value = "EXS_EXCEPTION_SOFTENING_NO_CONSTRAINTS", justification = "converting checked to unchecked exception, that should never be thrown at all")
	public static String encodeUrl(final String value) {
		try {
			return URLEncoder.encode(value, StandardCharsets.UTF_8.name());
		} catch (final UnsupportedEncodingException e) {
			throw new UncheckedIOException(e);
		}
	}

	@java.lang.SuppressWarnings("all")
	@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
	@lombok.Generated
	private StringConverters() {
		throw new java.lang.UnsupportedOperationException("This is a utility class and cannot be instantiated");
	}
}