// Base64Decoder.java // $Id: Base64Decoder.java,v 1.1 2002/07/11 12:00:11 ohitz Exp $ // (c) COPYRIGHT MIT and INRIA, 1996. // Please first read the full copyright statement in file COPYRIGHT.html package iiuf.util; import java.io.* ; import java.math.* ; /** * Decode a BASE64 encoded input stream to some output stream. * This class implements BASE64 decoding, as specified in the * <a href="http://ds.internic.net/rfc/rfc1521.txt">MIME specification</a>. * @see iiuf.util.Base64Encoder */ public class Base64Decoder { private static final int BUFFER_SIZE = 1024 ; InputStream in = null ; OutputStream out = null ; boolean stringp = false ; private void printHex (int x) { int h = (x&0xf0) >> 4 ; int l = (x&0x0f) ; System.out.print ((new Character((char)((h>9) ? 'A'+h-10 : '0'+h))).toString() +(new Character((char)((l>9) ? 'A'+l-10 : '0'+l))).toString()); } private void printHex (byte buf[], int off, int len) { while (off < len) { printHex (buf[off++]) ; System.out.print (" ") ; } System.out.println ("") ; } private void printHex (String s) { byte bytes[] = s.getBytes(); printHex (bytes, 0, bytes.length) ; } private final int get1 (byte buf[], int off) { return ((buf[off] & 0x3f) << 2) | ((buf[off+1] & 0x30) >>> 4) ; } private final int get2 (byte buf[], int off) { return ((buf[off+1] & 0x0f) << 4) | ((buf[off+2] &0x3c) >>> 2) ; } private final int get3 (byte buf[], int off) { return ((buf[off+2] & 0x03) << 6) | (buf[off+3] & 0x3f) ; } private final int check (int ch) { if ((ch >= 'A') && (ch <= 'Z')) { return ch - 'A' ; } else if ((ch >= 'a') && (ch <= 'z')) { return ch - 'a' + 26 ; } else if ((ch >= '0') && (ch <= '9')) { return ch - '0' + 52 ; } else { switch (ch) { case '=': return 65 ; case '+': return 62 ; case '/': return 63 ; default: return -1 ; } } } /** * Do the actual decoding. * Process the input stream by decoding it and emiting the resulting bytes * into the output stream. * @exception IOException If the input or output stream accesses failed. * @exception Base64FormatException If the input stream is not compliant * with the BASE64 specification. */ public void process () throws IOException, Base64FormatException { byte buffer[] = new byte[BUFFER_SIZE] ; byte chunk[] = new byte[4] ; int got = -1 ; int ready = 0 ; fill: while ((got = in.read(buffer)) > 0) { int skiped = 0 ; while ( skiped < got ) { // Check for un-understood characters: while ( ready < 4 ) { if ( skiped >= got ) continue fill ; int ch = check (buffer[skiped++]) ; if ( ch >= 0 ) chunk[ready++] = (byte) ch ; } if ( chunk[2] == 65 ) { out.write(get1(chunk, 0)); return ; } else if ( chunk[3] == 65 ) { out.write(get1(chunk, 0)) ; out.write(get2(chunk, 0)) ; return ; } else { out.write(get1(chunk, 0)) ; out.write(get2(chunk, 0)) ; out.write(get3(chunk, 0)) ; } ready = 0 ; } } if ( ready != 0 ) throw new Base64FormatException ("Invalid length.") ; out.flush() ; } /** * Do the decoding, and return a String. * This methods should be called when the decoder is used in * <em>String</em> mode. It decodes the input string to an output string * that is returned. * @exception RuntimeException If the object wasn't constructed to * decode a String. * @exception Base64FormatException If the input string is not compliant * with the BASE64 specification. */ public String processString () throws Base64FormatException { if ( ! stringp ) throw new RuntimeException (this.getClass().getName() + "[processString]" + "invalid call (not a String)"); try { process() ; } catch (IOException e) { } return ((ByteArrayOutputStream) out).toString() ; } /** * Create a decoder to decode a String. * @param input The string to be decoded. */ public Base64Decoder (String input) { byte bytes[] = input.getBytes(); this.stringp = true ; this.in = new ByteArrayInputStream(bytes) ; this.out = new ByteArrayOutputStream () ; } /** * Create a decoder to decode a stream. * @param in The input stream (to be decoded). * @param out The output stream, to write decoded data to. */ public Base64Decoder (InputStream in, OutputStream out) { this.in = in ; this.out = out ; this.stringp = false ; } /** * Test the decoder. * Run it with one argument: the string to be decoded, it will print out * the decoded value. */ public static void main (String args[]) { if ( args.length == 1 ) { try { Base64Decoder b = new Base64Decoder (args[0]) ; System.out.println ("["+b.processString()+"]") ; } catch (Base64FormatException e) { System.out.println ("Invalid Base64 format !") ; System.exit (1) ; } } else if ((args.length == 2) && (args[0].equals("-f"))) { try { FileInputStream in = new FileInputStream(args[1]) ; Base64Decoder b = new Base64Decoder (in, System.out); b.process(); } catch (Exception ex) { System.out.println("error: " + ex.getMessage()); System.exit(1) ; } } else { System.out.println("Base64Decoder [strong] [-f file]"); } System.exit (0) ; } }