All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.kitfox.svg.xml.cpx.CPXInputStream Maven / Gradle / Ivy

The newest version!
/*
 * SVG Salamander
 * Copyright (c) 2004, Mark McKay
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or 
 * without modification, are permitted provided that the following
 * conditions are met:
 *
 *   - Redistributions of source code must retain the above 
 *     copyright notice, this list of conditions and the following
 *     disclaimer.
 *   - Redistributions in binary form must reproduce the above
 *     copyright notice, this list of conditions and the following
 *     disclaimer in the documentation and/or other materials 
 *     provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
 * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
 * OF THE POSSIBILITY OF SUCH DAMAGE. 
 * 
 * Mark McKay can be contacted at [email protected].  Salamander and other
 * projects can be found at http://www.kitfox.com
 *
 * Created on February 12, 2004, 10:34 AM
 */

package com.kitfox.svg.xml.cpx;

import com.kitfox.svg.SVGConst;
import java.io.*;
import java.util.zip.*;
import java.security.*;
import java.util.logging.Level;
import java.util.logging.Logger;

/**
 * This class reads/decodes the CPX file format.  This format is a simple
 * compression/encryption transformer for XML data.  This stream takes in
 * encrypted XML and outputs decrypted.  It does this by checking for a magic
 * number at the start of the stream.  If absent, it treats the stream as
 * raw XML data and passes it through unaltered.  This is to aid development
 * in debugging versions, where the XML files will not be in CPX format.
 *
 * See http://java.sun.com/developer/technicalArticles/Security/Crypto/
 *
 * @author Mark McKay
 * @author Mark McKay
 */
public class CPXInputStream extends FilterInputStream implements CPXConsts {


    SecureRandom sec = new SecureRandom();

    Inflater inflater = new Inflater();

    int xlateMode;

    //Keep header bytes in case this stream turns out to be plain text
    byte[] head = new byte[4];
    int headSize = 0;
    int headPtr = 0;

    boolean reachedEOF = false;
    byte[] inBuffer = new byte[2048];
    byte[] decryptBuffer = new byte[2048];

    /** 
     * Creates a new instance of CPXInputStream
     * @param in
     * @throws java.io.IOException
     */
    public CPXInputStream(InputStream in) throws IOException {
        super(in);

        //Determine processing type
        for (int i = 0; i < 4; i++)
        {
            int val = in.read();
            head[i] = (byte)val;
            if (val == -1 || head[i] != MAGIC_NUMBER[i])
            {
                headSize = i + 1;
                xlateMode = XL_PLAIN;
                return;
            }
        }

        xlateMode = XL_ZIP_CRYPT;
    }

    /**
     * We do not allow marking
     */
    @Override
    public boolean markSupported() { return false; }

    /**
     * Closes this input stream and releases any system resources
     * associated with the stream.
     * This
     * method simply performs in.close().
     *
     * @exception  IOException  if an I/O error occurs.
     * @see        java.io.FilterInputStream#in
     */
    @Override
    public void close() throws IOException {
        reachedEOF = true;
        in.close();
    }

    /**
     * Reads the next byte of data from this input stream. The value
     * byte is returned as an int in the range
     * 0 to 255. If no byte is available
     * because the end of the stream has been reached, the value
     * -1 is returned. This method blocks until input data
     * is available, the end of the stream is detected, or an exception
     * is thrown.
     * 

* This method * simply performs in.read() and returns the result. * * @return the next byte of data, or -1 if the end of the * stream is reached. * @exception IOException if an I/O error occurs. * @see java.io.FilterInputStream#in */ @Override public int read() throws IOException { final byte[] b = new byte[1]; int retVal = read(b, 0, 1); if (retVal == -1) return -1; return b[0]; } /** * Reads up to byte.length bytes of data from this * input stream into an array of bytes. This method blocks until some * input is available. *

* This method simply performs the call * read(b, 0, b.length) and returns * the result. It is important that it does * not do in.read(b) instead; * certain subclasses of FilterInputStream * depend on the implementation strategy actually * used. * * @param b the buffer into which the data is read. * @return the total number of bytes read into the buffer, or * -1 if there is no more data because the end of * the stream has been reached. * @exception IOException if an I/O error occurs. * @see java.io.FilterInputStream#read(byte[], int, int) */ @Override public int read(byte[] b) throws IOException { return read(b, 0, b.length); } /** * Reads up to len bytes of data from this input stream * into an array of bytes. This method blocks until some input is * available. *

* This method simply performs in.read(b, off, len) * and returns the result. * * @param b the buffer into which the data is read. * @param off the start offset of the data. * @param len the maximum number of bytes read. * @return the total number of bytes read into the buffer, or * -1 if there is no more data because the end of * the stream has been reached. * @exception IOException if an I/O error occurs. * @see java.io.FilterInputStream#in */ @Override public int read(byte[] b, int off, int len) throws IOException { if (reachedEOF) return -1; if (xlateMode == XL_PLAIN) { int count = 0; //Write header if appropriate while (headPtr < headSize && len > 0) { b[off++] = head[headPtr++]; count++; len--; } return (len == 0) ? count : count + in.read(b, off, len); } //Decrypt and inflate if (inflater.needsInput() && !decryptChunk()) { reachedEOF = true; //Read remaining bytes int numRead; try { numRead = inflater.inflate(b, off, len); } catch (Exception e) { Logger.getLogger(SVGConst.SVG_LOGGER).log(Level.WARNING, null, e); return -1; } if (!inflater.finished()) { Logger.getLogger(SVGConst.SVG_LOGGER).log(Level.WARNING, "Inflation imncomplete"); } return numRead == 0 ? -1 : numRead; } try { return inflater.inflate(b, off, len); } catch (DataFormatException e) { Logger.getLogger(SVGConst.SVG_LOGGER).log(Level.WARNING, null, e); return -1; } } /** * Call when inflater indicates that it needs more bytes. * @return - true if we decrypted more bytes to deflate, false if we * encountered the end of stream * @throws java.io.IOException */ protected boolean decryptChunk() throws IOException { while (inflater.needsInput()) { int numInBytes = in.read(inBuffer); if (numInBytes == -1) return false; // int numDecryptBytes = cipher.update(inBuffer, 0, numInBytes, decryptBuffer); // inflater.setInput(decryptBuffer, 0, numDecryptBytes); inflater.setInput(inBuffer, 0, numInBytes); } return true; } /** * This method returns 1 if we've not reached EOF, 0 if we have. Programs * should not rely on this to determine the number of bytes that can be * read without blocking. */ @Override public int available() { return reachedEOF ? 0 : 1; } /** * Skips bytes by reading them into a cached buffer */ @Override public long skip(long n) throws IOException { int skipSize = (int)n; if (skipSize > inBuffer.length) skipSize = inBuffer.length; return read(inBuffer, 0, skipSize); } } /* import java.security.KeyPairGenerator; import java.security.KeyPair; import java.security.KeyPairGenerator; import java.security.PrivateKey; import java.security.PublicKey; import java.security.SecureRandom; import java.security.Cipher; .... java.security.Security.addProvider(new cryptix.provider.Cryptix()); SecureRandom random = new SecureRandom(SecureRandom.getSeed(30)); KeyPairGenerator keygen = KeyPairGenerator.getInstance("RSA"); keygen.initialize(1024, random); keypair = keygen.generateKeyPair(); PublicKey pubkey = keypair.getPublic(); PrivateKey privkey = keypair.getPrivate(); */ /* * *Generate key pairs KeyPairGenerator keyGen = KeyPairGenerator.getInstance("DSA"); KeyGen.initialize(1024, new SecureRandom(userSeed)); KeyPair pair = KeyGen.generateKeyPair(); */





© 2015 - 2025 Weber Informatics LLC | Privacy Policy