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

com.azure.identity.implementation.intellij.HashedBlockInputStream Maven / Gradle / Ivy

/*
 * Copyright 2015 Jo Rabin
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
/*
 * Portions Copyright (c) Microsoft Corporation
 */

package com.azure.identity.implementation.intellij;

import java.io.ByteArrayInputStream;
import java.io.EOFException;
import java.io.IOException;
import java.io.InputStream;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;

/**
 * Takes an underlying input stream formatted as Hashed Blocks
 * and provides the content of the blocks as an input stream
 *
 * Taken From KeePassJava
 *
 * 

A Hashed block consists of: * *

    *
  1. A 4 byte block sequence number, increments from 0 *
  2. A 32 byte MD5 hash of the content *
  3. A 4 byte length field *
  4. Content *
* *

The stream of blocks is terminated with a 0 length 0 hash block. * *

Originally developed for KeePass. A KeePass hash block * stream is little endian, i.e. the sequence * number and length fields are low order byte first. * * @author Jo */ public class HashedBlockInputStream extends InputStream { private static MessageDigest md5; static { try { md5 = MessageDigest.getInstance("SHA-256"); } catch (NoSuchAlgorithmException e) { throw new IllegalStateException(e); } } private static final int HASH_SIZE = 32; private static final byte[] ZERO_HASH = new byte[HASH_SIZE]; private long expectedSequenceNumber = 0; private boolean littleEndian = false; private boolean done = false; private InputStream inputStream; private ByteArrayInputStream blockInputStream = new ByteArrayInputStream(new byte[0]); /** * Create a Big Endian Hash Block Input Stream * @param inputStream the input stream containing the hash blocks */ public HashedBlockInputStream(InputStream inputStream) { this(inputStream, false); } /** * Create a Hash Block Input Stream with choice of endian encoding * @param inputStream the input stream containing the hash blocks * @param littleEndian true if the stream is little endian encoded */ public HashedBlockInputStream(InputStream inputStream, boolean littleEndian) { this.inputStream = inputStream; this.littleEndian = littleEndian; } @Override public int read(byte[] b, int offset, int length) throws IOException { return get(b, offset, length); } @Override public int read() throws IOException { byte[] buffer = new byte[1]; if (get(buffer, 0, 1) != 1) { throw new IOException("Could not read int"); } return buffer[0] & 0xFF; } @Override public void close() throws IOException { inputStream.close(); } /** * Gets bytes from the internal buffer and replenishes the buffer as necessary * @param b a byte array to fill * @param offset the offset to strat from * @param length the number of bytes to return * @return the number of bytes actually returned, , -1 if end of file * @throws IOException */ protected int get(byte[] b, int offset, int length) throws IOException { if (done) { return -1; } int totalBytesRead = 0; int bytesRead; while ((bytesRead = blockInputStream.read(b, offset, length)) < length && !done) { if (bytesRead == -1) { load(); } else { offset += bytesRead; length -= bytesRead; totalBytesRead += bytesRead; } } return bytesRead > 0 ? totalBytesRead + bytesRead : totalBytesRead; } /** * Reload the internal buffer from the underlying input stream * @throws IOException */ protected void load() throws IOException { // read the sequence number of the block long sequenceNumber = readUInt(); if (sequenceNumber != expectedSequenceNumber) { throw new IllegalStateException("Expected sequence number " + expectedSequenceNumber + " got " + sequenceNumber); } expectedSequenceNumber++; // get the block hash byte[] hash = new byte[HASH_SIZE]; readFully(hash); // get the length long readLength = readUInt(); if (readLength < 0) { throw new IllegalStateException("Got negative length for block"); } // length 0 means end of file if (readLength == 0) { if (!Arrays.equals(hash, ZERO_HASH)) { throw new IllegalStateException("Block hash was not zero on final block"); } done = true; return; } // get the new buffer byte[] readBuffer = new byte[(int) readLength]; readFully(readBuffer); // check the hash md5.update(readBuffer); if (!Arrays.equals(md5.digest(), hash)) { throw new IllegalStateException("MD5 check failed while reading HashBlock"); } blockInputStream = new ByteArrayInputStream(readBuffer); } /** * Read an unsigned 4 byte int decoding from the endian format * @return a long holding the value read * @throws IOException */ private long readUInt() throws IOException { byte[] buf = new byte[4]; readFully(buf); if (littleEndian) { return buf[3] << 24 | (buf[2] & 0xFF) << 16 | (buf[1] & 0xFF) << 8 | (buf[0] & 0xFF); } return buf[0] << 24 | (buf[1] & 0xFF) << 16 | (buf[2] & 0xFF) << 8 | (buf[3] & 0xFF); } /** * Fill the buffer passed * @param buffer the buffer to fill * @throws IOException if the buffer could not be filled */ private void readFully(byte[] buffer) throws IOException { int bytesToRead = buffer.length; int bytesSoFar = 0; while (bytesSoFar < buffer.length) { int bytesRead = inputStream.read(buffer, bytesSoFar, bytesToRead); if (bytesRead <= 0) { throw new EOFException(); } bytesSoFar += bytesRead; bytesToRead -= bytesRead; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy