jdk/src/share/classes/javax/crypto/CipherOutputStream.java
changeset 2 90ce3da70b43
child 5506 202f599c92aa
equal deleted inserted replaced
0:fd16c54261b3 2:90ce3da70b43
       
     1 /*
       
     2  * Copyright 1997-2007 Sun Microsystems, Inc.  All Rights Reserved.
       
     3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
       
     4  *
       
     5  * This code is free software; you can redistribute it and/or modify it
       
     6  * under the terms of the GNU General Public License version 2 only, as
       
     7  * published by the Free Software Foundation.  Sun designates this
       
     8  * particular file as subject to the "Classpath" exception as provided
       
     9  * by Sun in the LICENSE file that accompanied this code.
       
    10  *
       
    11  * This code is distributed in the hope that it will be useful, but WITHOUT
       
    12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
       
    13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
       
    14  * version 2 for more details (a copy is included in the LICENSE file that
       
    15  * accompanied this code).
       
    16  *
       
    17  * You should have received a copy of the GNU General Public License version
       
    18  * 2 along with this work; if not, write to the Free Software Foundation,
       
    19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
       
    20  *
       
    21  * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
       
    22  * CA 95054 USA or visit www.sun.com if you need additional information or
       
    23  * have any questions.
       
    24  */
       
    25 
       
    26 package javax.crypto;
       
    27 
       
    28 import java.io.*;
       
    29 
       
    30 /**
       
    31  * A CipherOutputStream is composed of an OutputStream and a Cipher so
       
    32  * that write() methods first process the data before writing them out
       
    33  * to the underlying OutputStream.  The cipher must be fully
       
    34  * initialized before being used by a CipherOutputStream.
       
    35  *
       
    36  * <p> For example, if the cipher is initialized for encryption, the
       
    37  * CipherOutputStream will attempt to encrypt data before writing out the
       
    38  * encrypted data.
       
    39  *
       
    40  * <p> This class adheres strictly to the semantics, especially the
       
    41  * failure semantics, of its ancestor classes
       
    42  * java.io.OutputStream and java.io.FilterOutputStream.  This class
       
    43  * has exactly those methods specified in its ancestor classes, and
       
    44  * overrides them all.  Moreover, this class catches all exceptions
       
    45  * that are not thrown by its ancestor classes.
       
    46  *
       
    47  * <p> It is crucial for a programmer using this class not to use
       
    48  * methods that are not defined or overriden in this class (such as a
       
    49  * new method or constructor that is later added to one of the super
       
    50  * classes), because the design and implementation of those methods
       
    51  * are unlikely to have considered security impact with regard to
       
    52  * CipherOutputStream.
       
    53  *
       
    54  * @author  Li Gong
       
    55  * @see     java.io.OutputStream
       
    56  * @see     java.io.FilterOutputStream
       
    57  * @see     javax.crypto.Cipher
       
    58  * @see     javax.crypto.CipherInputStream
       
    59  *
       
    60  * @since 1.4
       
    61  */
       
    62 
       
    63 public class CipherOutputStream extends FilterOutputStream {
       
    64 
       
    65     // the cipher engine to use to process stream data
       
    66     private Cipher cipher;
       
    67 
       
    68     // the underlying output stream
       
    69     private OutputStream output;
       
    70 
       
    71     /* the buffer holding one byte of incoming data */
       
    72     private byte[] ibuffer = new byte[1];
       
    73 
       
    74     // the buffer holding data ready to be written out
       
    75     private byte[] obuffer;
       
    76 
       
    77     /**
       
    78      *
       
    79      * Constructs a CipherOutputStream from an OutputStream and a
       
    80      * Cipher.
       
    81      * <br>Note: if the specified output stream or cipher is
       
    82      * null, a NullPointerException may be thrown later when
       
    83      * they are used.
       
    84      *
       
    85      * @param os  the OutputStream object
       
    86      * @param c   an initialized Cipher object
       
    87      */
       
    88     public CipherOutputStream(OutputStream os, Cipher c) {
       
    89         super(os);
       
    90         output = os;
       
    91         cipher = c;
       
    92     };
       
    93 
       
    94     /**
       
    95      * Constructs a CipherOutputStream from an OutputStream without
       
    96      * specifying a Cipher. This has the effect of constructing a
       
    97      * CipherOutputStream using a NullCipher.
       
    98      * <br>Note: if the specified output stream is null, a
       
    99      * NullPointerException may be thrown later when it is used.
       
   100      *
       
   101      * @param os  the OutputStream object
       
   102      */
       
   103     protected CipherOutputStream(OutputStream os) {
       
   104         super(os);
       
   105         output = os;
       
   106         cipher = new NullCipher();
       
   107     }
       
   108 
       
   109     /**
       
   110      * Writes the specified byte to this output stream.
       
   111      *
       
   112      * @param      b   the <code>byte</code>.
       
   113      * @exception  IOException  if an I/O error occurs.
       
   114      * @since      JCE1.2
       
   115      */
       
   116     public void write(int b) throws IOException {
       
   117         ibuffer[0] = (byte) b;
       
   118         obuffer = cipher.update(ibuffer, 0, 1);
       
   119         if (obuffer != null) {
       
   120             output.write(obuffer);
       
   121             obuffer = null;
       
   122         }
       
   123     };
       
   124 
       
   125     /**
       
   126      * Writes <code>b.length</code> bytes from the specified byte array
       
   127      * to this output stream.
       
   128      * <p>
       
   129      * The <code>write</code> method of
       
   130      * <code>CipherOutputStream</code> calls the <code>write</code>
       
   131      * method of three arguments with the three arguments
       
   132      * <code>b</code>, <code>0</code>, and <code>b.length</code>.
       
   133      *
       
   134      * @param      b   the data.
       
   135      * @exception  NullPointerException if <code>b</code> is null.
       
   136      * @exception  IOException  if an I/O error occurs.
       
   137      * @see        javax.crypto.CipherOutputStream#write(byte[], int, int)
       
   138      * @since JCE1.2
       
   139      */
       
   140     public void write(byte b[]) throws IOException {
       
   141         write(b, 0, b.length);
       
   142     }
       
   143 
       
   144     /**
       
   145      * Writes <code>len</code> bytes from the specified byte array
       
   146      * starting at offset <code>off</code> to this output stream.
       
   147      *
       
   148      * @param      b     the data.
       
   149      * @param      off   the start offset in the data.
       
   150      * @param      len   the number of bytes to write.
       
   151      * @exception  IOException  if an I/O error occurs.
       
   152      * @since      JCE1.2
       
   153      */
       
   154     public void write(byte b[], int off, int len) throws IOException {
       
   155         obuffer = cipher.update(b, off, len);
       
   156         if (obuffer != null) {
       
   157             output.write(obuffer);
       
   158             obuffer = null;
       
   159         }
       
   160     }
       
   161 
       
   162     /**
       
   163      * Flushes this output stream by forcing any buffered output bytes
       
   164      * that have already been processed by the encapsulated cipher object
       
   165      * to be written out.
       
   166      *
       
   167      * <p>Any bytes buffered by the encapsulated cipher
       
   168      * and waiting to be processed by it will not be written out. For example,
       
   169      * if the encapsulated cipher is a block cipher, and the total number of
       
   170      * bytes written using one of the <code>write</code> methods is less than
       
   171      * the cipher's block size, no bytes will be written out.
       
   172      *
       
   173      * @exception  IOException  if an I/O error occurs.
       
   174      * @since      JCE1.2
       
   175      */
       
   176     public void flush() throws IOException {
       
   177         if (obuffer != null) {
       
   178             output.write(obuffer);
       
   179             obuffer = null;
       
   180         }
       
   181         output.flush();
       
   182     }
       
   183 
       
   184     /**
       
   185      * Closes this output stream and releases any system resources
       
   186      * associated with this stream.
       
   187      * <p>
       
   188      * This method invokes the <code>doFinal</code> method of the encapsulated
       
   189      * cipher object, which causes any bytes buffered by the encapsulated
       
   190      * cipher to be processed. The result is written out by calling the
       
   191      * <code>flush</code> method of this output stream.
       
   192      * <p>
       
   193      * This method resets the encapsulated cipher object to its initial state
       
   194      * and calls the <code>close</code> method of the underlying output
       
   195      * stream.
       
   196      *
       
   197      * @exception  IOException  if an I/O error occurs.
       
   198      * @since      JCE1.2
       
   199      */
       
   200     public void close() throws IOException {
       
   201         try {
       
   202             obuffer = cipher.doFinal();
       
   203         } catch (IllegalBlockSizeException e) {
       
   204             obuffer = null;
       
   205         } catch (BadPaddingException e) {
       
   206             obuffer = null;
       
   207         }
       
   208         try {
       
   209             flush();
       
   210         } catch (IOException ignored) {}
       
   211         out.close();
       
   212     }
       
   213 }