Coverage Report - org.kuali.rice.ksb.security.SignatureSigningOutputStream
 
Classes in this File Line Coverage Branch Coverage Complexity
SignatureSigningOutputStream
0%
0/32
0%
0/8
3.667
 
 1  
 /*
 2  
  * Copyright 2005-2007 The Kuali Foundation
 3  
  * 
 4  
  * 
 5  
  * Licensed under the Educational Community License, Version 2.0 (the "License");
 6  
  * you may not use this file except in compliance with the License.
 7  
  * You may obtain a copy of the License at
 8  
  * 
 9  
  * http://www.opensource.org/licenses/ecl2.php
 10  
  * 
 11  
  * Unless required by applicable law or agreed to in writing, software
 12  
  * distributed under the License is distributed on an "AS IS" BASIS,
 13  
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 14  
  * See the License for the specific language governing permissions and
 15  
  * limitations under the License.
 16  
  */
 17  
 package org.kuali.rice.ksb.security;
 18  
 
 19  
 import java.io.BufferedOutputStream;
 20  
 import java.io.ByteArrayOutputStream;
 21  
 import java.io.IOException;
 22  
 import java.io.OutputStream;
 23  
 import java.security.GeneralSecurityException;
 24  
 
 25  
 import javax.servlet.ServletOutputStream;
 26  
 
 27  
 /**
 28  
  * An OutputStream which decorates another OutputStream with a wrapper that digitally
 29  
  * signs the data when the OutputStream is closed.  Since this class does not know where
 30  
  * the resulting digital signature will reside, a DigitalSigner will be invoked to
 31  
  * execute the actual signing of the message (i.e. put it in a header).
 32  
  * 
 33  
  * @author Kuali Rice Team (rice.collab@kuali.org)
 34  
  */
 35  
 public class SignatureSigningOutputStream extends ServletOutputStream {
 36  
 
 37  
         private boolean delayWrite;
 38  
         private DigitalSigner signer;
 39  
         private BufferedOutputStream bufferedDataHoldingStream;
 40  
         private ByteArrayOutputStream dataHoldingStream;
 41  
         private OutputStream wrappedOutputStream;
 42  
         
 43  
         /**
 44  
          * Constructs a SignatureSigningOutputStream with the given DigitalSigner and underlying OutputStream.
 45  
          * If true, the delayWrite boolean indicates that the stream should store all data internally until the
 46  
          * stream is closed, at which point it should forward all data to the wrapped OutputStream.  If delayWrite
 47  
          * is false, then the data will be forwarded immediately.
 48  
          */
 49  
         public SignatureSigningOutputStream(DigitalSigner signer, OutputStream wrappedOutputStream, boolean delayWrite) {
 50  0
                 super();
 51  0
                 this.delayWrite = delayWrite;
 52  0
                 if (delayWrite) {
 53  0
                     this.dataHoldingStream = new ByteArrayOutputStream();
 54  0
                     this.bufferedDataHoldingStream = new BufferedOutputStream(this.dataHoldingStream);
 55  
                 }
 56  0
                 this.wrappedOutputStream = wrappedOutputStream;
 57  0
                 this.signer = signer;
 58  0
         }
 59  
 
 60  
         public void write(int data) throws IOException {
 61  0
                 if (this.delayWrite) {
 62  0
                     this.bufferedDataHoldingStream.write(data);
 63  
                 } else {
 64  0
                     this.wrappedOutputStream.write(data);
 65  
                 }
 66  
                 try {
 67  0
                     this.signer.getSignature().update((byte)data);        
 68  0
                 } catch (GeneralSecurityException e) {
 69  0
                         IOException exception = new IOException("Error updating signature.");
 70  0
                         exception.initCause(e);
 71  0
                         throw exception;
 72  0
                 }
 73  0
         }
 74  
         
 75  
         @Override
 76  
         public void close() throws IOException {
 77  
                 // before we close, sign the message
 78  
                 try {
 79  0
                     this.signer.sign();
 80  0
                         if (this.delayWrite) {
 81  0
                             this.bufferedDataHoldingStream.close();
 82  0
                                 byte[] data = this.dataHoldingStream.toByteArray();
 83  0
                                 for (int index = 0; index < data.length; index++) {
 84  0
                                     this.wrappedOutputStream.write(data[index]);
 85  
                                 }
 86  
                         }
 87  0
                         this.wrappedOutputStream.close();
 88  0
                 } catch (Exception e) {
 89  0
                         IOException exception = new IOException("Error attaching digital signature to outbound response.");
 90  0
                         exception.initCause(e);
 91  0
                         throw exception;
 92  
                 } finally {
 93  0
                         super.close();
 94  0
                 }
 95  0
         }
 96  
 
 97  
 }