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

com.helger.commons.codec.RFC1522QCodec Maven / Gradle / Ivy

There is a newer version: 5.0.12
Show newest version
/**
 * Copyright (C) 2014-2015 Philip Helger (www.helger.com)
 * philip[at]helger[dot]com
 *
 * 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.helger.commons.codec;

import java.nio.charset.Charset;
import java.util.BitSet;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;

import com.helger.commons.ValueEnforcer;
import com.helger.commons.annotation.ReturnsMutableCopy;
import com.helger.commons.charset.CCharset;

/**
 * Similar to the Quoted-Printable content-transfer-encoding defined in RFC 1521 and designed to allow
 * text containing mostly ASCII characters to be decipherable on an ASCII
 * terminal without decoding.
 * 

* RFC 1522 describes * techniques to allow the encoding of non-ASCII text in various portions of a * RFC 822 [2] message header, in a manner which is unlikely to confuse existing * message handling software. *

* This class is conditionally thread-safe. The instance field m_bEncodeBlanks * is mutable {@link #setEncodeBlanks(boolean)} but is not volatile, and * accesses are not synchronised. If an instance of the class is shared between * threads, the caller needs to ensure that suitable synchronisation is used to * ensure safe publication of the value between threads, and must not invoke * {@link #setEncodeBlanks(boolean)} after initial setup. * * @see MIME (Multipurpose * Internet Mail Extensions) Part Two: Message Header Extensions for * Non-ASCII Text */ public class RFC1522QCodec extends AbstractRFC1522Codec { public static final boolean DEFAULT_ENCODE_BLANKS = false; /** * BitSet of printable characters as defined in RFC 1522. */ private static final BitSet PRINTABLE_CHARS = new BitSet (256); // Static initializer for printable chars collection static { // alpha characters PRINTABLE_CHARS.set (' '); PRINTABLE_CHARS.set ('!'); PRINTABLE_CHARS.set ('"'); PRINTABLE_CHARS.set ('#'); PRINTABLE_CHARS.set ('$'); PRINTABLE_CHARS.set ('%'); PRINTABLE_CHARS.set ('&'); PRINTABLE_CHARS.set ('\''); PRINTABLE_CHARS.set ('('); PRINTABLE_CHARS.set (')'); PRINTABLE_CHARS.set ('*'); PRINTABLE_CHARS.set ('+'); PRINTABLE_CHARS.set (','); PRINTABLE_CHARS.set ('-'); PRINTABLE_CHARS.set ('.'); PRINTABLE_CHARS.set ('/'); for (int i = '0'; i <= '9'; i++) PRINTABLE_CHARS.set (i); PRINTABLE_CHARS.set (':'); PRINTABLE_CHARS.set (';'); PRINTABLE_CHARS.set ('<'); PRINTABLE_CHARS.set ('>'); PRINTABLE_CHARS.set ('@'); for (int i = 'A'; i <= 'Z'; i++) PRINTABLE_CHARS.set (i); PRINTABLE_CHARS.set ('['); PRINTABLE_CHARS.set ('\\'); PRINTABLE_CHARS.set (']'); PRINTABLE_CHARS.set ('^'); PRINTABLE_CHARS.set ('`'); for (int i = 'a'; i <= 'z'; i++) PRINTABLE_CHARS.set (i); PRINTABLE_CHARS.set ('{'); PRINTABLE_CHARS.set ('|'); PRINTABLE_CHARS.set ('}'); PRINTABLE_CHARS.set ('~'); } private static final byte BLANK = 32; private static final byte UNDERSCORE = 95; /** * The default charset used for string decoding and encoding. */ private final Charset m_aCharset; private boolean m_bEncodeBlanks = DEFAULT_ENCODE_BLANKS; /** * Default constructor with the UTF-8 charset. */ public RFC1522QCodec () { this (CCharset.CHARSET_UTF_8_OBJ); } /** * Constructor which allows for the selection of a default charset. * * @param aCharset * the default string charset to use. */ public RFC1522QCodec (@Nonnull final Charset aCharset) { m_aCharset = ValueEnforcer.notNull (aCharset, "Charset"); } @Override protected String getRFC1522Encoding () { return "Q"; } @Nonnull public Charset getCharset () { return m_aCharset; } /** * Tests if optional transformation of SPACE characters is to be used * * @return {@code true} if SPACE characters are to be transformed, * {@code false} otherwise */ public boolean isEncodeBlanks () { return m_bEncodeBlanks; } /** * Defines whether optional transformation of SPACE characters is to be used * * @param bEncodeBlanks * {@code true} if SPACE characters are to be transformed, * {@code false} otherwise */ public void setEncodeBlanks (final boolean bEncodeBlanks) { m_bEncodeBlanks = bEncodeBlanks; } @Nullable @ReturnsMutableCopy public byte [] getEncoded (@Nullable final byte [] aBuffer) { if (aBuffer == null) return null; final byte [] data = QuotedPrintableCodec.getEncodedQuotedPrintable (PRINTABLE_CHARS, aBuffer); if (m_bEncodeBlanks) for (int i = 0; i < data.length; i++) if (data[i] == BLANK) data[i] = UNDERSCORE; return data; } @Nullable @ReturnsMutableCopy public byte [] getDecoded (@Nullable final byte [] aBuffer) throws DecodeException { if (aBuffer == null) return null; boolean bHasUnderscores = false; for (final byte b : aBuffer) if (b == UNDERSCORE) { bHasUnderscores = true; break; } if (bHasUnderscores) { final byte [] tmp = new byte [aBuffer.length]; for (int i = 0; i < aBuffer.length; i++) { final byte b = aBuffer[i]; if (b != UNDERSCORE) tmp[i] = b; else tmp[i] = BLANK; } return QuotedPrintableCodec.getDecodedQuotedPrintable (tmp); } return QuotedPrintableCodec.getDecodedQuotedPrintable (aBuffer); } /** * Encodes a string into its quoted-printable form using the default charset. * Unsafe characters are escaped. * * @param sText * string to convert to quoted-printable form * @return quoted-printable string * @throws EncodeException * thrown if a failure condition is encountered during the encoding * process. */ @Nullable public String getEncodedText (@Nullable final String sText) throws EncodeException { return super.getEncodedText (sText, getCharset ()); } /** * Decodes a quoted-printable string into its original form. Escaped * characters are converted back to their original representation. * * @param sText * quoted-printable string to convert into its original form * @return original string * @throws DecodeException * A decoder exception is thrown if a failure condition is encountered * during the decode process. */ @Override @Nullable public String getDecodedText (@Nullable final String sText) throws DecodeException { return super.getDecodedText (sText); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy