001 /** 002 * 003 * Copyright 2003-2004 The Apache Software Foundation 004 * 005 * Licensed under the Apache License, Version 2.0 (the "License"); 006 * you may not use this file except in compliance with the License. 007 * 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.geronimo.mail.util; 019 020 import java.io.IOException; 021 import java.io.InputStream; 022 import java.io.OutputStream; 023 import java.io.PrintStream; 024 025 public class Base64Encoder 026 implements Encoder 027 { 028 protected final byte[] encodingTable = 029 { 030 (byte)'A', (byte)'B', (byte)'C', (byte)'D', (byte)'E', (byte)'F', (byte)'G', 031 (byte)'H', (byte)'I', (byte)'J', (byte)'K', (byte)'L', (byte)'M', (byte)'N', 032 (byte)'O', (byte)'P', (byte)'Q', (byte)'R', (byte)'S', (byte)'T', (byte)'U', 033 (byte)'V', (byte)'W', (byte)'X', (byte)'Y', (byte)'Z', 034 (byte)'a', (byte)'b', (byte)'c', (byte)'d', (byte)'e', (byte)'f', (byte)'g', 035 (byte)'h', (byte)'i', (byte)'j', (byte)'k', (byte)'l', (byte)'m', (byte)'n', 036 (byte)'o', (byte)'p', (byte)'q', (byte)'r', (byte)'s', (byte)'t', (byte)'u', 037 (byte)'v', 038 (byte)'w', (byte)'x', (byte)'y', (byte)'z', 039 (byte)'0', (byte)'1', (byte)'2', (byte)'3', (byte)'4', (byte)'5', (byte)'6', 040 (byte)'7', (byte)'8', (byte)'9', 041 (byte)'+', (byte)'/' 042 }; 043 044 protected byte padding = (byte)'='; 045 046 /* 047 * set up the decoding table. 048 */ 049 protected final byte[] decodingTable = new byte[256]; 050 051 protected void initialiseDecodingTable() 052 { 053 for (int i = 0; i < encodingTable.length; i++) 054 { 055 decodingTable[encodingTable[i]] = (byte)i; 056 } 057 } 058 059 public Base64Encoder() 060 { 061 initialiseDecodingTable(); 062 } 063 064 /** 065 * encode the input data producing a base 64 output stream. 066 * 067 * @return the number of bytes produced. 068 */ 069 public int encode( 070 byte[] data, 071 int off, 072 int length, 073 OutputStream out) 074 throws IOException 075 { 076 int modulus = length % 3; 077 int dataLength = (length - modulus); 078 int a1, a2, a3; 079 080 for (int i = off; i < off + dataLength; i += 3) 081 { 082 a1 = data[i] & 0xff; 083 a2 = data[i + 1] & 0xff; 084 a3 = data[i + 2] & 0xff; 085 086 out.write(encodingTable[(a1 >>> 2) & 0x3f]); 087 out.write(encodingTable[((a1 << 4) | (a2 >>> 4)) & 0x3f]); 088 out.write(encodingTable[((a2 << 2) | (a3 >>> 6)) & 0x3f]); 089 out.write(encodingTable[a3 & 0x3f]); 090 } 091 092 /* 093 * process the tail end. 094 */ 095 int b1, b2, b3; 096 int d1, d2; 097 098 switch (modulus) 099 { 100 case 0: /* nothing left to do */ 101 break; 102 case 1: 103 d1 = data[off + dataLength] & 0xff; 104 b1 = (d1 >>> 2) & 0x3f; 105 b2 = (d1 << 4) & 0x3f; 106 107 out.write(encodingTable[b1]); 108 out.write(encodingTable[b2]); 109 out.write(padding); 110 out.write(padding); 111 break; 112 case 2: 113 d1 = data[off + dataLength] & 0xff; 114 d2 = data[off + dataLength + 1] & 0xff; 115 116 b1 = (d1 >>> 2) & 0x3f; 117 b2 = ((d1 << 4) | (d2 >>> 4)) & 0x3f; 118 b3 = (d2 << 2) & 0x3f; 119 120 out.write(encodingTable[b1]); 121 out.write(encodingTable[b2]); 122 out.write(encodingTable[b3]); 123 out.write(padding); 124 break; 125 } 126 127 return (dataLength / 3) * 4 + ((modulus == 0) ? 0 : 4); 128 } 129 130 private boolean ignore( 131 char c) 132 { 133 return (c == '\n' || c =='\r' || c == '\t' || c == ' '); 134 } 135 136 /** 137 * decode the base 64 encoded byte data writing it to the given output stream, 138 * whitespace characters will be ignored. 139 * 140 * @return the number of bytes produced. 141 */ 142 public int decode( 143 byte[] data, 144 int off, 145 int length, 146 OutputStream out) 147 throws IOException 148 { 149 byte[] bytes; 150 byte b1, b2, b3, b4; 151 int outLen = 0; 152 153 int end = off + length; 154 155 while (end > 0) 156 { 157 if (!ignore((char)data[end - 1])) 158 { 159 break; 160 } 161 162 end--; 163 } 164 165 int i = off; 166 int finish = end - 4; 167 168 while (i < finish) 169 { 170 while ((i < finish) && ignore((char)data[i])) 171 { 172 i++; 173 } 174 175 b1 = decodingTable[data[i++]]; 176 177 while ((i < finish) && ignore((char)data[i])) 178 { 179 i++; 180 } 181 182 b2 = decodingTable[data[i++]]; 183 184 while ((i < finish) && ignore((char)data[i])) 185 { 186 i++; 187 } 188 189 b3 = decodingTable[data[i++]]; 190 191 while ((i < finish) && ignore((char)data[i])) 192 { 193 i++; 194 } 195 196 b4 = decodingTable[data[i++]]; 197 198 out.write((b1 << 2) | (b2 >> 4)); 199 out.write((b2 << 4) | (b3 >> 2)); 200 out.write((b3 << 6) | b4); 201 202 outLen += 3; 203 } 204 205 if (data[end - 2] == padding) 206 { 207 b1 = decodingTable[data[end - 4]]; 208 b2 = decodingTable[data[end - 3]]; 209 210 out.write((b1 << 2) | (b2 >> 4)); 211 212 outLen += 1; 213 } 214 else if (data[end - 1] == padding) 215 { 216 b1 = decodingTable[data[end - 4]]; 217 b2 = decodingTable[data[end - 3]]; 218 b3 = decodingTable[data[end - 2]]; 219 220 out.write((b1 << 2) | (b2 >> 4)); 221 out.write((b2 << 4) | (b3 >> 2)); 222 223 outLen += 2; 224 } 225 else 226 { 227 b1 = decodingTable[data[end - 4]]; 228 b2 = decodingTable[data[end - 3]]; 229 b3 = decodingTable[data[end - 2]]; 230 b4 = decodingTable[data[end - 1]]; 231 232 out.write((b1 << 2) | (b2 >> 4)); 233 out.write((b2 << 4) | (b3 >> 2)); 234 out.write((b3 << 6) | b4); 235 236 outLen += 3; 237 } 238 239 return outLen; 240 } 241 242 /** 243 * decode the base 64 encoded String data writing it to the given output stream, 244 * whitespace characters will be ignored. 245 * 246 * @return the number of bytes produced. 247 */ 248 public int decode( 249 String data, 250 OutputStream out) 251 throws IOException 252 { 253 byte[] bytes; 254 byte b1, b2, b3, b4; 255 int length = 0; 256 257 int end = data.length(); 258 259 while (end > 0) 260 { 261 if (!ignore(data.charAt(end - 1))) 262 { 263 break; 264 } 265 266 end--; 267 } 268 269 int i = 0; 270 int finish = end - 4; 271 272 while (i < finish) 273 { 274 while ((i < finish) && ignore(data.charAt(i))) 275 { 276 i++; 277 } 278 279 b1 = decodingTable[data.charAt(i++)]; 280 281 while ((i < finish) && ignore(data.charAt(i))) 282 { 283 i++; 284 } 285 b2 = decodingTable[data.charAt(i++)]; 286 287 while ((i < finish) && ignore(data.charAt(i))) 288 { 289 i++; 290 } 291 b3 = decodingTable[data.charAt(i++)]; 292 293 while ((i < finish) && ignore(data.charAt(i))) 294 { 295 i++; 296 } 297 b4 = decodingTable[data.charAt(i++)]; 298 299 out.write((b1 << 2) | (b2 >> 4)); 300 out.write((b2 << 4) | (b3 >> 2)); 301 out.write((b3 << 6) | b4); 302 303 length += 3; 304 } 305 306 if (data.charAt(end - 2) == padding) 307 { 308 b1 = decodingTable[data.charAt(end - 4)]; 309 b2 = decodingTable[data.charAt(end - 3)]; 310 311 out.write((b1 << 2) | (b2 >> 4)); 312 313 length += 1; 314 } 315 else if (data.charAt(end - 1) == padding) 316 { 317 b1 = decodingTable[data.charAt(end - 4)]; 318 b2 = decodingTable[data.charAt(end - 3)]; 319 b3 = decodingTable[data.charAt(end - 2)]; 320 321 out.write((b1 << 2) | (b2 >> 4)); 322 out.write((b2 << 4) | (b3 >> 2)); 323 324 length += 2; 325 } 326 else 327 { 328 b1 = decodingTable[data.charAt(end - 4)]; 329 b2 = decodingTable[data.charAt(end - 3)]; 330 b3 = decodingTable[data.charAt(end - 2)]; 331 b4 = decodingTable[data.charAt(end - 1)]; 332 333 out.write((b1 << 2) | (b2 >> 4)); 334 out.write((b2 << 4) | (b3 >> 2)); 335 out.write((b3 << 6) | b4); 336 337 length += 3; 338 } 339 340 return length; 341 } 342 343 /** 344 * decode the base 64 encoded byte data writing it to the provided byte array buffer. 345 * 346 * @return the number of bytes produced. 347 */ 348 public int decode(byte[] data, int off, int length, byte[] out) throws IOException 349 { 350 byte[] bytes; 351 byte b1, b2, b3, b4; 352 int outLen = 0; 353 354 int end = off + length; 355 356 while (end > 0) 357 { 358 if (!ignore((char)data[end - 1])) 359 { 360 break; 361 } 362 363 end--; 364 } 365 366 int i = off; 367 int finish = end - 4; 368 369 while (i < finish) 370 { 371 while ((i < finish) && ignore((char)data[i])) 372 { 373 i++; 374 } 375 376 b1 = decodingTable[data[i++]]; 377 378 while ((i < finish) && ignore((char)data[i])) 379 { 380 i++; 381 } 382 383 b2 = decodingTable[data[i++]]; 384 385 while ((i < finish) && ignore((char)data[i])) 386 { 387 i++; 388 } 389 390 b3 = decodingTable[data[i++]]; 391 392 while ((i < finish) && ignore((char)data[i])) 393 { 394 i++; 395 } 396 397 b4 = decodingTable[data[i++]]; 398 399 out[outLen++] = (byte)((b1 << 2) | (b2 >> 4)); 400 out[outLen++] = (byte)((b2 << 4) | (b3 >> 2)); 401 out[outLen++] = (byte)((b3 << 6) | b4); 402 } 403 404 if (data[end - 2] == padding) 405 { 406 b1 = decodingTable[data[end - 4]]; 407 b2 = decodingTable[data[end - 3]]; 408 409 out[outLen++] = (byte)((b1 << 2) | (b2 >> 4)); 410 } 411 else if (data[end - 1] == padding) 412 { 413 b1 = decodingTable[data[end - 4]]; 414 b2 = decodingTable[data[end - 3]]; 415 b3 = decodingTable[data[end - 2]]; 416 417 out[outLen++] = (byte)((b1 << 2) | (b2 >> 4)); 418 out[outLen++] = (byte)((b2 << 4) | (b3 >> 2)); 419 } 420 else 421 { 422 b1 = decodingTable[data[end - 4]]; 423 b2 = decodingTable[data[end - 3]]; 424 b3 = decodingTable[data[end - 2]]; 425 b4 = decodingTable[data[end - 1]]; 426 427 out[outLen++] = (byte)((b1 << 2) | (b2 >> 4)); 428 out[outLen++] = (byte)((b2 << 4) | (b3 >> 2)); 429 out[outLen++] = (byte)((b3 << 6) | b4); 430 } 431 432 return outLen; 433 } 434 435 /** 436 * Test if a character is a valid Base64 encoding character. This 437 * must be either a valid digit or the padding character ("="). 438 * 439 * @param ch The test character. 440 * 441 * @return true if this is valid in Base64 encoded data, false otherwise. 442 */ 443 public boolean isValidBase64(int ch) { 444 // 'A' has the value 0 in the decoding table, so we need a special one for that 445 return ch == padding || ch == 'A' || decodingTable[ch] != 0; 446 } 447 448 449 /** 450 * Perform RFC-2047 word encoding using Base64 data encoding. 451 * 452 * @param in The source for the encoded data. 453 * @param charset The charset tag to be added to each encoded data section. 454 * @param out The output stream where the encoded data is to be written. 455 * @param fold Controls whether separate sections of encoded data are separated by 456 * linebreaks or whitespace. 457 * 458 * @exception IOException 459 */ 460 public void encodeWord(InputStream in, String charset, OutputStream out, boolean fold) throws IOException 461 { 462 PrintStream writer = new PrintStream(out); 463 464 // encoded words are restricted to 76 bytes, including the control adornments. 465 int limit = 76 - 7 - charset.length(); 466 boolean firstLine = true; 467 StringBuffer encodedString = new StringBuffer(76); 468 469 while (true) { 470 // encode the next segment. 471 encode(in, encodedString, limit); 472 // if we're out of data, nothing will be encoded. 473 if (encodedString.length() == 0) { 474 break; 475 } 476 477 // if we have more than one segment, we need to insert separators. Depending on whether folding 478 // was requested, this is either a blank or a linebreak. 479 if (!firstLine) { 480 if (fold) { 481 writer.print("\r\n"); 482 } 483 else { 484 writer.print(" "); 485 } 486 } 487 488 // add the encoded word header 489 writer.print("=?"); 490 writer.print(charset); 491 writer.print("?B?"); 492 // the data 493 writer.print(encodedString.toString()); 494 // and the word terminator. 495 writer.print("?="); 496 writer.flush(); 497 498 // reset our string buffer for the next segment. 499 encodedString.setLength(0); 500 } 501 } 502 503 /** 504 * encode the input data producing a base 64 output stream. 505 * 506 * @return the number of bytes produced. 507 */ 508 public void encode(InputStream in, StringBuffer out, int limit) throws IOException 509 { 510 int count = limit / 4; 511 byte [] inBuffer = new byte[3]; 512 513 while (count-- > 0) { 514 515 int readCount = in.read(inBuffer); 516 // did we get a full triplet? that's an easy encoding. 517 if (readCount == 3) { 518 int a1 = inBuffer[0] & 0xff; 519 int a2 = inBuffer[1] & 0xff; 520 int a3 = inBuffer[2] & 0xff; 521 522 out.append((char)encodingTable[(a1 >>> 2) & 0x3f]); 523 out.append((char)encodingTable[((a1 << 4) | (a2 >>> 4)) & 0x3f]); 524 out.append((char)encodingTable[((a2 << 2) | (a3 >>> 6)) & 0x3f]); 525 out.append((char)encodingTable[a3 & 0x3f]); 526 527 } 528 else if (readCount <= 0) { 529 // eof condition, don'e entirely. 530 return; 531 } 532 else if (readCount == 1) { 533 int a1 = inBuffer[0] & 0xff; 534 out.append((char)encodingTable[(a1 >>> 2) & 0x3f]); 535 out.append((char)encodingTable[(a1 << 4) & 0x3f]); 536 out.append((char)padding); 537 out.append((char)padding); 538 return; 539 } 540 else if (readCount == 2) { 541 int a1 = inBuffer[0] & 0xff; 542 int a2 = inBuffer[1] & 0xff; 543 544 out.append((char)encodingTable[(a1 >>> 2) & 0x3f]); 545 out.append((char)encodingTable[((a1 << 4) | (a2 >>> 4)) & 0x3f]); 546 out.append((char)encodingTable[(a2 << 2) & 0x3f]); 547 out.append((char)padding); 548 return; 549 } 550 } 551 } 552 }