001/*
002 * Licensed to the Apache Software Foundation (ASF) under one or more
003 * contributor license agreements.  See the NOTICE file distributed with
004 * this work for additional information regarding copyright ownership.
005 * The ASF licenses this file to You under the Apache License, Version 2.0
006 * (the "License"); you may not use this file except in compliance with
007 * the License.  You may obtain a copy of the License at
008 *
009 *      https://www.apache.org/licenses/LICENSE-2.0
010 *
011 * Unless required by applicable law or agreed to in writing, software
012 * distributed under the License is distributed on an "AS IS" BASIS,
013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
014 * See the License for the specific language governing permissions and
015 * limitations under the License.
016 */
017
018package org.apache.commons.codec.net;
019
020import java.io.UnsupportedEncodingException;
021import java.nio.charset.Charset;
022import java.nio.charset.StandardCharsets;
023import java.nio.charset.UnsupportedCharsetException;
024import java.util.BitSet;
025
026import org.apache.commons.codec.DecoderException;
027import org.apache.commons.codec.EncoderException;
028import org.apache.commons.codec.StringDecoder;
029import org.apache.commons.codec.StringEncoder;
030
031/**
032 * Similar to the Quoted-Printable content-transfer-encoding defined in
033 * <a href="http://www.ietf.org/rfc/rfc1521.txt">RFC 1521</a> and designed to allow text containing mostly ASCII
034 * characters to be decipherable on an ASCII terminal without decoding.
035 * <p>
036 * <a href="http://www.ietf.org/rfc/rfc1522.txt">RFC 1522</a> describes techniques to allow the encoding of non-ASCII
037 * text in various portions of a RFC 822 [2] message header, in a manner which is unlikely to confuse existing message
038 * handling software.
039 * </p>
040 * <p>
041 * This class is conditionally thread-safe.
042 * The instance field for encoding blanks is mutable {@link #setEncodeBlanks(boolean)}
043 * but is not volatile, and accesses are not synchronized.
044 * If an instance of the class is shared between threads, the caller needs to ensure that suitable synchronization
045 * is used to ensure safe publication of the value between threads, and must not invoke
046 * {@link #setEncodeBlanks(boolean)} after initial setup.
047 * </p>
048 *
049 * @see <a href="http://www.ietf.org/rfc/rfc1522.txt">MIME (Multipurpose Internet Mail Extensions) Part Two: Message
050 *          Header Extensions for Non-ASCII Text</a>
051 *
052 * @since 1.3
053 */
054public class QCodec extends RFC1522Codec implements StringEncoder, StringDecoder {
055    /**
056     * BitSet of printable characters as defined in RFC 1522.
057     */
058    private static final BitSet PRINTABLE_CHARS = new BitSet(256);
059
060    // Static initializer for printable chars collection
061    static {
062        // alpha characters
063        PRINTABLE_CHARS.set(' ');
064        PRINTABLE_CHARS.set('!');
065        PRINTABLE_CHARS.set('"');
066        PRINTABLE_CHARS.set('#');
067        PRINTABLE_CHARS.set('$');
068        PRINTABLE_CHARS.set('%');
069        PRINTABLE_CHARS.set('&');
070        PRINTABLE_CHARS.set('\'');
071        PRINTABLE_CHARS.set('(');
072        PRINTABLE_CHARS.set(')');
073        PRINTABLE_CHARS.set('*');
074        PRINTABLE_CHARS.set('+');
075        PRINTABLE_CHARS.set(',');
076        PRINTABLE_CHARS.set('-');
077        PRINTABLE_CHARS.set('.');
078        PRINTABLE_CHARS.set('/');
079        for (int i = '0'; i <= '9'; i++) {
080            PRINTABLE_CHARS.set(i);
081        }
082        PRINTABLE_CHARS.set(':');
083        PRINTABLE_CHARS.set(';');
084        PRINTABLE_CHARS.set('<');
085        PRINTABLE_CHARS.set('>');
086        PRINTABLE_CHARS.set('@');
087        for (int i = 'A'; i <= 'Z'; i++) {
088            PRINTABLE_CHARS.set(i);
089        }
090        PRINTABLE_CHARS.set('[');
091        PRINTABLE_CHARS.set('\\');
092        PRINTABLE_CHARS.set(']');
093        PRINTABLE_CHARS.set('^');
094        PRINTABLE_CHARS.set('`');
095        for (int i = 'a'; i <= 'z'; i++) {
096            PRINTABLE_CHARS.set(i);
097        }
098        PRINTABLE_CHARS.set('{');
099        PRINTABLE_CHARS.set('|');
100        PRINTABLE_CHARS.set('}');
101        PRINTABLE_CHARS.set('~');
102    }
103    private static final byte SPACE = 32;
104
105    private static final byte UNDERSCORE = 95;
106
107    private boolean encodeBlanks;
108
109    /**
110     * Default constructor.
111     */
112    public QCodec() {
113        this(StandardCharsets.UTF_8);
114    }
115
116    /**
117     * Constructor which allows for the selection of a default Charset.
118     *
119     * @param charset
120     *            the default string Charset to use.
121     *
122     * @see Charset
123     * @since 1.7
124     */
125    public QCodec(final Charset charset) {
126        super(charset);
127    }
128
129    /**
130     * Constructor which allows for the selection of a default Charset.
131     *
132     * @param charsetName
133     *            the Charset to use.
134     * @throws java.nio.charset.UnsupportedCharsetException
135     *             If the named Charset is unavailable
136     * @since 1.7 throws UnsupportedCharsetException if the named Charset is unavailable
137     * @see Charset
138     */
139    public QCodec(final String charsetName) {
140        this(Charset.forName(charsetName));
141    }
142
143    /**
144     * Decodes a quoted-printable object into its original form. Escaped characters are converted back to their original
145     * representation.
146     *
147     * @param obj
148     *            quoted-printable object to convert into its original form
149     * @return original object
150     * @throws DecoderException
151     *             Thrown if the argument is not a {@code String}. Thrown if a failure condition is encountered
152     *             during the decode process.
153     */
154    @Override
155    public Object decode(final Object obj) throws DecoderException {
156        if (obj == null) {
157            return null;
158        }
159        if (obj instanceof String) {
160            return decode((String) obj);
161        }
162        throw new DecoderException("Objects of type " + obj.getClass().getName() + " cannot be decoded using Q codec");
163    }
164
165    /**
166     * Decodes a quoted-printable string into its original form. Escaped characters are converted back to their original
167     * representation.
168     *
169     * @param str
170     *            quoted-printable string to convert into its original form
171     * @return original string
172     * @throws DecoderException
173     *             A decoder exception is thrown if a failure condition is encountered during the decode process.
174     */
175    @Override
176    public String decode(final String str) throws DecoderException {
177        try {
178            return decodeText(str);
179        } catch (final UnsupportedEncodingException e) {
180            throw new DecoderException(e.getMessage(), e);
181        }
182    }
183
184    @Override
185    protected byte[] doDecoding(final byte[] bytes) throws DecoderException {
186        if (bytes == null) {
187            return null;
188        }
189        boolean hasUnderscores = false;
190        for (final byte b : bytes) {
191            if (b == UNDERSCORE) {
192                hasUnderscores = true;
193                break;
194            }
195        }
196        if (hasUnderscores) {
197            final byte[] tmp = new byte[bytes.length];
198            for (int i = 0; i < bytes.length; i++) {
199                final byte b = bytes[i];
200                if (b != UNDERSCORE) {
201                    tmp[i] = b;
202                } else {
203                    tmp[i] = SPACE;
204                }
205            }
206            return QuotedPrintableCodec.decodeQuotedPrintable(tmp);
207        }
208        return QuotedPrintableCodec.decodeQuotedPrintable(bytes);
209    }
210
211    @Override
212    protected byte[] doEncoding(final byte[] bytes) {
213        if (bytes == null) {
214            return null;
215        }
216        final byte[] data = QuotedPrintableCodec.encodeQuotedPrintable(PRINTABLE_CHARS, bytes);
217        if (this.encodeBlanks) {
218            for (int i = 0; i < data.length; i++) {
219                if (data[i] == SPACE) {
220                    data[i] = UNDERSCORE;
221                }
222            }
223        }
224        return data;
225    }
226
227    /**
228     * Encodes an object into its quoted-printable form using the default Charset. Unsafe characters are escaped.
229     *
230     * @param obj
231     *            object to convert to quoted-printable form
232     * @return quoted-printable object
233     * @throws EncoderException
234     *             thrown if a failure condition is encountered during the encoding process.
235     */
236    @Override
237    public Object encode(final Object obj) throws EncoderException {
238        if (obj == null) {
239            return null;
240        }
241        if (obj instanceof String) {
242            return encode((String) obj);
243        }
244        throw new EncoderException("Objects of type " + obj.getClass().getName() + " cannot be encoded using Q codec");
245    }
246
247    /**
248     * Encodes a string into its quoted-printable form using the default Charset. Unsafe characters are escaped.
249     *
250     * @param sourceStr
251     *            string to convert to quoted-printable form
252     * @return quoted-printable string
253     * @throws EncoderException
254     *             thrown if a failure condition is encountered during the encoding process.
255     */
256    @Override
257    public String encode(final String sourceStr) throws EncoderException {
258        return encode(sourceStr, getCharset());
259    }
260
261    /**
262     * Encodes a string into its quoted-printable form using the specified Charset. Unsafe characters are escaped.
263     *
264     * @param sourceStr
265     *            string to convert to quoted-printable form
266     * @param sourceCharset
267     *            the Charset for sourceStr
268     * @return quoted-printable string
269     * @throws EncoderException
270     *             thrown if a failure condition is encountered during the encoding process.
271     * @since 1.7
272     */
273    public String encode(final String sourceStr, final Charset sourceCharset) throws EncoderException {
274        return encodeText(sourceStr, sourceCharset);
275    }
276
277    /**
278     * Encodes a string into its quoted-printable form using the specified Charset. Unsafe characters are escaped.
279     *
280     * @param sourceStr
281     *            string to convert to quoted-printable form
282     * @param sourceCharset
283     *            the Charset for sourceStr
284     * @return quoted-printable string
285     * @throws EncoderException
286     *             thrown if a failure condition is encountered during the encoding process.
287     */
288    public String encode(final String sourceStr, final String sourceCharset) throws EncoderException {
289        try {
290            return encodeText(sourceStr, sourceCharset);
291        } catch (final UnsupportedCharsetException e) {
292            throw new EncoderException(e.getMessage(), e);
293        }
294    }
295
296    @Override
297    protected String getEncoding() {
298        return "Q";
299    }
300
301    /**
302     * Tests if optional transformation of SPACE characters is to be used
303     *
304     * @return {@code true} if SPACE characters are to be transformed, {@code false} otherwise
305     */
306    public boolean isEncodeBlanks() {
307        return this.encodeBlanks;
308    }
309
310    /**
311     * Defines whether optional transformation of SPACE characters is to be used
312     *
313     * @param b
314     *            {@code true} if SPACE characters are to be transformed, {@code false} otherwise
315     */
316    public void setEncodeBlanks(final boolean b) {
317        this.encodeBlanks = b;
318    }
319}