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.ByteArrayOutputStream;
021import java.io.UnsupportedEncodingException;
022import java.util.BitSet;
023
024import org.apache.commons.codec.BinaryDecoder;
025import org.apache.commons.codec.BinaryEncoder;
026import org.apache.commons.codec.CharEncoding;
027import org.apache.commons.codec.DecoderException;
028import org.apache.commons.codec.EncoderException;
029import org.apache.commons.codec.StringDecoder;
030import org.apache.commons.codec.StringEncoder;
031import org.apache.commons.codec.binary.StringUtils;
032
033/**
034 * Implements the 'www-form-urlencoded' encoding scheme, also misleadingly known as URL encoding.
035 * <p>
036 * This codec is meant to be a replacement for standard Java classes {@link java.net.URLEncoder} and
037 * {@link java.net.URLDecoder} on older Java platforms, as these classes in Java versions below
038 * 1.4 rely on the platform's default charset encoding.
039 * </p>
040 * <p>
041 * This class is thread-safe as of 1.11
042 * </p>
043 *
044 * @see <a href="http://www.w3.org/TR/html4/interact/forms.html#h-17.13.4.1">Chapter 17.13.4 Form content types</a>
045 *           of the <a href="http://www.w3.org/TR/html4/">HTML 4.01 Specification</a>
046 *
047 * @since 1.2
048 */
049public class URLCodec implements BinaryEncoder, BinaryDecoder, StringEncoder, StringDecoder {
050
051    /**
052     * Release 1.5 made this field final.
053     */
054    protected static final byte ESCAPE_CHAR = '%';
055
056    /**
057     * BitSet of www-form-url safe characters.
058     * This is a copy of the internal BitSet which is now used for the conversion.
059     * Changes to this field are ignored.
060     * @deprecated 1.11 Will be removed in 2.0 (CODEC-230)
061     */
062    @Deprecated
063    protected static final BitSet WWW_FORM_URL;
064
065    private static final BitSet WWW_FORM_URL_SAFE = new BitSet(256);
066
067    // Static initializer for www_form_url
068    static {
069        // alpha characters
070        for (int i = 'a'; i <= 'z'; i++) {
071            WWW_FORM_URL_SAFE.set(i);
072        }
073        for (int i = 'A'; i <= 'Z'; i++) {
074            WWW_FORM_URL_SAFE.set(i);
075        }
076        // numeric characters
077        for (int i = '0'; i <= '9'; i++) {
078            WWW_FORM_URL_SAFE.set(i);
079        }
080        // special chars
081        WWW_FORM_URL_SAFE.set('-');
082        WWW_FORM_URL_SAFE.set('_');
083        WWW_FORM_URL_SAFE.set('.');
084        WWW_FORM_URL_SAFE.set('*');
085        // blank to be replaced with +
086        WWW_FORM_URL_SAFE.set(' ');
087
088        // Create a copy in case anyone (ab)uses it
089        WWW_FORM_URL = (BitSet) WWW_FORM_URL_SAFE.clone();
090    }
091
092    /**
093     * Decodes an array of URL safe 7-bit characters into an array of original bytes. Escaped characters are converted
094     * back to their original representation.
095     *
096     * @param bytes
097     *            array of URL safe characters
098     * @return array of original bytes
099     * @throws DecoderException
100     *             Thrown if URL decoding is unsuccessful
101     */
102    public static final byte[] decodeUrl(final byte[] bytes) throws DecoderException {
103        if (bytes == null) {
104            return null;
105        }
106        final ByteArrayOutputStream buffer = new ByteArrayOutputStream();
107        for (int i = 0; i < bytes.length; i++) {
108            final int b = bytes[i];
109            if (b == '+') {
110                buffer.write(' ');
111            } else if (b == ESCAPE_CHAR) {
112                try {
113                    final int u = Utils.digit16(bytes[++i]);
114                    final int l = Utils.digit16(bytes[++i]);
115                    buffer.write((char) ((u << 4) + l));
116                } catch (final ArrayIndexOutOfBoundsException e) {
117                    throw new DecoderException("Invalid URL encoding: ", e);
118                }
119            } else {
120                buffer.write(b);
121            }
122        }
123        return buffer.toByteArray();
124    }
125
126    /**
127     * Encodes an array of bytes into an array of URL safe 7-bit characters. Unsafe characters are escaped.
128     *
129     * @param urlsafe
130     *            bitset of characters deemed URL safe
131     * @param bytes
132     *            array of bytes to convert to URL safe characters
133     * @return array of bytes containing URL safe characters
134     */
135    public static final byte[] encodeUrl(BitSet urlsafe, final byte[] bytes) {
136        if (bytes == null) {
137            return null;
138        }
139        if (urlsafe == null) {
140            urlsafe = WWW_FORM_URL_SAFE;
141        }
142
143        final ByteArrayOutputStream buffer = new ByteArrayOutputStream();
144        for (final byte c : bytes) {
145            int b = c;
146            if (b < 0) {
147                b = 256 + b;
148            }
149            if (urlsafe.get(b)) {
150                if (b == ' ') {
151                    b = '+';
152                }
153                buffer.write(b);
154            } else {
155                buffer.write(ESCAPE_CHAR);
156                final char hex1 = Utils.hexDigit(b >> 4);
157                final char hex2 = Utils.hexDigit(b);
158                buffer.write(hex1);
159                buffer.write(hex2);
160            }
161        }
162        return buffer.toByteArray();
163    }
164
165    /**
166     * The default charset used for string decoding and encoding.
167     *
168     * @deprecated TODO: This field will be changed to a private final Charset in 2.0. (CODEC-126)
169     */
170    @Deprecated
171    protected volatile String charset; // added volatile: see CODEC-232
172
173    /**
174     * Default constructor.
175     */
176    public URLCodec() {
177        this(CharEncoding.UTF_8);
178    }
179
180    /**
181     * Constructor which allows for the selection of a default charset.
182     *
183     * @param charset the default string charset to use.
184     */
185    public URLCodec(final String charset) {
186        this.charset = charset;
187    }
188
189    /**
190     * Decodes an array of URL safe 7-bit characters into an array of original bytes. Escaped characters are converted
191     * back to their original representation.
192     *
193     * @param bytes
194     *            array of URL safe characters
195     * @return array of original bytes
196     * @throws DecoderException
197     *             Thrown if URL decoding is unsuccessful
198     */
199    @Override
200    public byte[] decode(final byte[] bytes) throws DecoderException {
201        return decodeUrl(bytes);
202    }
203
204    /**
205     * Decodes a URL safe object into its original form. Escaped characters are converted back to their original
206     * representation.
207     *
208     * @param obj
209     *            URL safe object to convert into its original form
210     * @return original object
211     * @throws DecoderException
212     *             Thrown if the argument is not a {@code String} or {@code byte[]}. Thrown if a failure
213     *             condition is encountered during the decode process.
214     */
215    @Override
216    public Object decode(final Object obj) throws DecoderException {
217        if (obj == null) {
218            return null;
219        }
220        if (obj instanceof byte[]) {
221            return decode((byte[]) obj);
222        }
223        if (obj instanceof String) {
224            return decode((String) obj);
225        }
226        throw new DecoderException("Objects of type " + obj.getClass().getName() + " cannot be URL decoded");
227    }
228
229    /**
230     * Decodes a URL safe string into its original form using the default string charset. Escaped characters are
231     * converted back to their original representation.
232     *
233     * @param str
234     *            URL safe string to convert into its original form
235     * @return original string
236     * @throws DecoderException
237     *             Thrown if URL decoding is unsuccessful
238     * @see #getDefaultCharset()
239     */
240    @Override
241    public String decode(final String str) throws DecoderException {
242        if (str == null) {
243            return null;
244        }
245        try {
246            return decode(str, getDefaultCharset());
247        } catch (final UnsupportedEncodingException e) {
248            throw new DecoderException(e.getMessage(), e);
249        }
250    }
251
252    /**
253     * Decodes a URL safe string into its original form using the specified encoding. Escaped characters are converted
254     * back to their original representation.
255     *
256     * @param str
257     *            URL safe string to convert into its original form
258     * @param charsetName
259     *            the original string charset
260     * @return original string
261     * @throws DecoderException
262     *             Thrown if URL decoding is unsuccessful
263     * @throws UnsupportedEncodingException
264     *             Thrown if charset is not supported
265     */
266    public String decode(final String str, final String charsetName)
267            throws DecoderException, UnsupportedEncodingException {
268        if (str == null) {
269            return null;
270        }
271        return new String(decode(StringUtils.getBytesUsAscii(str)), charsetName);
272    }
273
274    /**
275     * Encodes an array of bytes into an array of URL safe 7-bit characters. Unsafe characters are escaped.
276     *
277     * @param bytes
278     *            array of bytes to convert to URL safe characters
279     * @return array of bytes containing URL safe characters
280     */
281    @Override
282    public byte[] encode(final byte[] bytes) {
283        return encodeUrl(WWW_FORM_URL_SAFE, bytes);
284    }
285
286    /**
287     * Encodes an object into its URL safe form. Unsafe characters are escaped.
288     *
289     * @param obj
290     *            string to convert to a URL safe form
291     * @return URL safe object
292     * @throws EncoderException
293     *             Thrown if URL encoding is not applicable to objects of this type or if encoding is unsuccessful
294     */
295    @Override
296    public Object encode(final Object obj) throws EncoderException {
297        if (obj == null) {
298            return null;
299        }
300        if (obj instanceof byte[]) {
301            return encode((byte[]) obj);
302        }
303        if (obj instanceof String) {
304            return encode((String) obj);
305        }
306        throw new EncoderException("Objects of type " + obj.getClass().getName() + " cannot be URL encoded");
307    }
308
309    /**
310     * Encodes a string into its URL safe form using the default string charset. Unsafe characters are escaped.
311     *
312     * @param str
313     *            string to convert to a URL safe form
314     * @return URL safe string
315     * @throws EncoderException
316     *             Thrown if URL encoding is unsuccessful
317     *
318     * @see #getDefaultCharset()
319     */
320    @Override
321    public String encode(final String str) throws EncoderException {
322        if (str == null) {
323            return null;
324        }
325        try {
326            return encode(str, getDefaultCharset());
327        } catch (final UnsupportedEncodingException e) {
328            throw new EncoderException(e.getMessage(), e);
329        }
330    }
331
332    /**
333     * Encodes a string into its URL safe form using the specified string charset. Unsafe characters are escaped.
334     *
335     * @param str
336     *            string to convert to a URL safe form
337     * @param charsetName
338     *            the charset for str
339     * @return URL safe string
340     * @throws UnsupportedEncodingException
341     *             Thrown if charset is not supported
342     */
343    public String encode(final String str, final String charsetName) throws UnsupportedEncodingException {
344        if (str == null) {
345            return null;
346        }
347        return StringUtils.newStringUsAscii(encode(str.getBytes(charsetName)));
348    }
349
350    /**
351     * The default charset used for string decoding and encoding.
352     *
353     * @return the default string charset.
354     */
355    public String getDefaultCharset() {
356        return this.charset;
357    }
358
359    /**
360     * The {@code String} encoding used for decoding and encoding.
361     *
362     * @return Returns the encoding.
363     * @deprecated Use {@link #getDefaultCharset()}, will be removed in 2.0.
364     */
365    @Deprecated
366    public String getEncoding() {
367        return this.charset;
368    }
369
370}