com.helger.commons.codec.RFC1522QCodec Maven / Gradle / Ivy
/**
* 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);
}
}