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 * http://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 018 package org.apache.commons.codec.binary; 019 020 import static org.junit.Assert.assertEquals; 021 import static org.junit.Assert.assertFalse; 022 import static org.junit.Assert.assertTrue; 023 import static org.junit.Assert.fail; 024 025 import java.io.UnsupportedEncodingException; 026 import java.nio.charset.Charset; 027 import java.nio.charset.UnsupportedCharsetException; 028 import java.util.Arrays; 029 import java.util.Random; 030 031 import org.junit.Assert; 032 033 import org.apache.commons.codec.DecoderException; 034 import org.apache.commons.codec.EncoderException; 035 import org.junit.Test; 036 037 /** 038 * Tests {@link org.apache.commons.codec.binary.Hex}. 039 * 040 * @version $Id: HexTest.html 889935 2013-12-11 05:05:13Z ggregory $ 041 */ 042 public class HexTest { 043 044 private static final String BAD_ENCODING_NAME = "UNKNOWN"; 045 046 private final static boolean LOG = false; 047 048 private boolean charsetSanityCheck(final String name) { 049 final String source = "the quick brown dog jumped over the lazy fox"; 050 try { 051 final byte[] bytes = source.getBytes(name); 052 final String str = new String(bytes, name); 053 final boolean equals = source.equals(str); 054 if (equals == false) { 055 // Here with: 056 // 057 // Java Sun 1.4.2_19 x86 32-bits on Windows XP 058 // JIS_X0212-1990 059 // x-JIS0208 060 // 061 // Java Sun 1.5.0_17 x86 32-bits on Windows XP 062 // JIS_X0212-1990 063 // x-IBM834 064 // x-JIS0208 065 // x-MacDingbat 066 // x-MacSymbol 067 // 068 // Java Sun 1.6.0_14 x86 32-bits 069 // JIS_X0212-1990 070 // x-IBM834 071 // x-JIS0208 072 // x-MacDingbat 073 // x-MacSymbol 074 // 075 log("FAILED charsetSanityCheck=Interesting Java charset oddity: Roundtrip failed for " + name); 076 } 077 return equals; 078 } catch (final UnsupportedEncodingException e) { 079 // Should NEVER happen since we are getting the name from the Charset class. 080 if (LOG) { 081 log("FAILED charsetSanityCheck=" + name + ", e=" + e); 082 log(e); 083 } 084 return false; 085 } catch (final UnsupportedOperationException e) { 086 // Caught here with: 087 // x-JISAutoDetect on Windows XP and Java Sun 1.4.2_19 x86 32-bits 088 // x-JISAutoDetect on Windows XP and Java Sun 1.5.0_17 x86 32-bits 089 // x-JISAutoDetect on Windows XP and Java Sun 1.6.0_14 x86 32-bits 090 if (LOG) { 091 log("FAILED charsetSanityCheck=" + name + ", e=" + e); 092 log(e); 093 } 094 return false; 095 } 096 } 097 098 /** 099 * @param data 100 */ 101 private void checkDecodeHexOddCharacters(final char[] data) { 102 try { 103 Hex.decodeHex(data); 104 fail("An exception wasn't thrown when trying to decode an odd number of characters"); 105 } catch (final DecoderException e) { 106 // Expected exception 107 } 108 } 109 110 private void log(final String s) { 111 if (LOG) { 112 System.out.println(s); 113 System.out.flush(); 114 } 115 } 116 117 private void log(final Throwable t) { 118 if (LOG) { 119 t.printStackTrace(System.out); 120 System.out.flush(); 121 } 122 } 123 124 @Test 125 public void testCustomCharset() throws UnsupportedEncodingException, DecoderException { 126 for (final String name : Charset.availableCharsets().keySet()) { 127 testCustomCharset(name, "testCustomCharset"); 128 } 129 } 130 131 /** 132 * @param name 133 * @param parent 134 * TODO 135 * @throws UnsupportedEncodingException 136 * @throws DecoderException 137 */ 138 private void testCustomCharset(final String name, final String parent) throws UnsupportedEncodingException, DecoderException { 139 if (charsetSanityCheck(name) == false) { 140 return; 141 } 142 log(parent + "=" + name); 143 final Hex customCodec = new Hex(name); 144 // source data 145 final String sourceString = "Hello World"; 146 final byte[] sourceBytes = sourceString.getBytes(name); 147 // test 1 148 // encode source to hex string to bytes with charset 149 final byte[] actualEncodedBytes = customCodec.encode(sourceBytes); 150 // encode source to hex string... 151 String expectedHexString = Hex.encodeHexString(sourceBytes); 152 // ... and get the bytes in the expected charset 153 final byte[] expectedHexStringBytes = expectedHexString.getBytes(name); 154 Assert.assertTrue(Arrays.equals(expectedHexStringBytes, actualEncodedBytes)); 155 // test 2 156 String actualStringFromBytes = new String(actualEncodedBytes, name); 157 assertEquals(name + ", expectedHexString=" + expectedHexString + ", actualStringFromBytes=" + actualStringFromBytes, 158 expectedHexString, actualStringFromBytes); 159 // second test: 160 final Hex utf8Codec = new Hex(); 161 expectedHexString = "48656c6c6f20576f726c64"; 162 final byte[] decodedUtf8Bytes = (byte[]) utf8Codec.decode(expectedHexString); 163 actualStringFromBytes = new String(decodedUtf8Bytes, utf8Codec.getCharset()); 164 // sanity check: 165 assertEquals(name, sourceString, actualStringFromBytes); 166 // actual check: 167 final byte[] decodedCustomBytes = customCodec.decode(actualEncodedBytes); 168 actualStringFromBytes = new String(decodedCustomBytes, name); 169 assertEquals(name, sourceString, actualStringFromBytes); 170 } 171 172 @Test(expected=UnsupportedCharsetException.class) 173 public void testCustomCharsetBadName() { 174 new Hex(BAD_ENCODING_NAME); 175 } 176 177 @Test 178 public void testCustomCharsetToString() { 179 assertTrue(new Hex().toString().indexOf(Hex.DEFAULT_CHARSET_NAME) >= 0); 180 } 181 182 @Test 183 public void testDecodeArrayOddCharacters() { 184 try { 185 new Hex().decode(new byte[]{65}); 186 fail("An exception wasn't thrown when trying to decode an odd number of characters"); 187 } catch (final DecoderException e) { 188 // Expected exception 189 } 190 } 191 192 @Test 193 public void testDecodeBadCharacterPos0() { 194 try { 195 new Hex().decode("q0"); 196 fail("An exception wasn't thrown when trying to decode an illegal character"); 197 } catch (final DecoderException e) { 198 // Expected exception 199 } 200 } 201 202 @Test 203 public void testDecodeBadCharacterPos1() { 204 try { 205 new Hex().decode("0q"); 206 fail("An exception wasn't thrown when trying to decode an illegal character"); 207 } catch (final DecoderException e) { 208 // Expected exception 209 } 210 } 211 212 @Test 213 public void testDecodeClassCastException() { 214 try { 215 new Hex().decode(new int[]{65}); 216 fail("An exception wasn't thrown when trying to decode."); 217 } catch (final DecoderException e) { 218 // Expected exception 219 } 220 } 221 222 @Test 223 public void testDecodeHexOddCharacters1() { 224 checkDecodeHexOddCharacters(new char[]{'A'}); 225 } 226 227 @Test 228 public void testDecodeHexOddCharacters3() { 229 checkDecodeHexOddCharacters(new char[]{'A', 'B', 'C'}); 230 } 231 232 @Test 233 public void testDecodeHexOddCharacters5() { 234 checkDecodeHexOddCharacters(new char[]{'A', 'B', 'C', 'D', 'E'}); 235 } 236 237 @Test 238 public void testDecodeStringOddCharacters() { 239 try { 240 new Hex().decode("6"); 241 fail("An exception wasn't thrown when trying to decode an odd number of characters"); 242 } catch (final DecoderException e) { 243 // Expected exception 244 } 245 } 246 247 @Test 248 public void testDencodeEmpty() throws DecoderException { 249 assertTrue(Arrays.equals(new byte[0], Hex.decodeHex(new char[0]))); 250 assertTrue(Arrays.equals(new byte[0], new Hex().decode(new byte[0]))); 251 assertTrue(Arrays.equals(new byte[0], (byte[]) new Hex().decode(""))); 252 } 253 254 @Test 255 public void testEncodeClassCastException() { 256 try { 257 new Hex().encode(new int[]{65}); 258 fail("An exception wasn't thrown when trying to encode."); 259 } catch (final EncoderException e) { 260 // Expected exception 261 } 262 } 263 264 @Test 265 public void testEncodeDecodeRandom() throws DecoderException, EncoderException { 266 final Random random = new Random(); 267 268 final Hex hex = new Hex(); 269 for (int i = 5; i > 0; i--) { 270 final byte[] data = new byte[random.nextInt(10000) + 1]; 271 random.nextBytes(data); 272 273 // static API 274 final char[] encodedChars = Hex.encodeHex(data); 275 byte[] decodedBytes = Hex.decodeHex(encodedChars); 276 assertTrue(Arrays.equals(data, decodedBytes)); 277 278 // instance API with array parameter 279 final byte[] encodedStringBytes = hex.encode(data); 280 decodedBytes = hex.decode(encodedStringBytes); 281 assertTrue(Arrays.equals(data, decodedBytes)); 282 283 // instance API with char[] (Object) parameter 284 String dataString = new String(encodedChars); 285 char[] encodedStringChars = (char[]) hex.encode(dataString); 286 decodedBytes = (byte[]) hex.decode(encodedStringChars); 287 assertTrue(Arrays.equals(StringUtils.getBytesUtf8(dataString), decodedBytes)); 288 289 // instance API with String (Object) parameter 290 dataString = new String(encodedChars); 291 encodedStringChars = (char[]) hex.encode(dataString); 292 decodedBytes = (byte[]) hex.decode(new String(encodedStringChars)); 293 assertTrue(Arrays.equals(StringUtils.getBytesUtf8(dataString), decodedBytes)); 294 } 295 } 296 297 @Test 298 public void testEncodeEmpty() throws EncoderException { 299 assertTrue(Arrays.equals(new char[0], Hex.encodeHex(new byte[0]))); 300 assertTrue(Arrays.equals(new byte[0], new Hex().encode(new byte[0]))); 301 assertTrue(Arrays.equals(new char[0], (char[]) new Hex().encode(""))); 302 } 303 304 @Test 305 public void testEncodeZeroes() { 306 final char[] c = Hex.encodeHex(new byte[36]); 307 assertEquals("000000000000000000000000000000000000000000000000000000000000000000000000", new String(c)); 308 } 309 310 @Test 311 public void testHelloWorldLowerCaseHex() { 312 final byte[] b = StringUtils.getBytesUtf8("Hello World"); 313 final String expected = "48656c6c6f20576f726c64"; 314 char[] actual; 315 actual = Hex.encodeHex(b); 316 assertEquals(expected, new String(actual)); 317 actual = Hex.encodeHex(b, true); 318 assertEquals(expected, new String(actual)); 319 actual = Hex.encodeHex(b, false); 320 assertFalse(expected.equals(new String(actual))); 321 } 322 323 @Test 324 public void testHelloWorldUpperCaseHex() { 325 final byte[] b = StringUtils.getBytesUtf8("Hello World"); 326 final String expected = "48656C6C6F20576F726C64"; 327 char[] actual; 328 actual = Hex.encodeHex(b); 329 assertFalse(expected.equals(new String(actual))); 330 actual = Hex.encodeHex(b, true); 331 assertFalse(expected.equals(new String(actual))); 332 actual = Hex.encodeHex(b, false); 333 assertTrue(expected.equals(new String(actual))); 334 } 335 336 @Test 337 public void testRequiredCharset() throws UnsupportedEncodingException, DecoderException { 338 testCustomCharset("UTF-8", "testRequiredCharset"); 339 testCustomCharset("UTF-16", "testRequiredCharset"); 340 testCustomCharset("UTF-16BE", "testRequiredCharset"); 341 testCustomCharset("UTF-16LE", "testRequiredCharset"); 342 testCustomCharset("US-ASCII", "testRequiredCharset"); 343 testCustomCharset("ISO8859_1", "testRequiredCharset"); 344 } 345 }