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

com.robothy.s3.rest.utils.AwsChunkedDecodingInputStream Maven / Gradle / Ivy

package com.robothy.s3.rest.utils;

import java.io.IOException;
import java.io.InputStream;
import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;

/**
 * Skips V4 style signing metadata from input streams.
 * 

The original stream looks like this (newlines are CRLF):

* *
 * 5;chunk-signature=7ece820edcf094ce1ef6d643c8db60b67913e28831d9b0430efd2b56a9deec5e
 * 12345
 * 0;chunk-signature=ee2c094d7162170fcac17d2c76073cd834b0488bfe52e89e48599b8115c7ffa2
 * 
* *

The format of each chunk of data is:

* *
 * [hex-encoded-number-of-bytes-in-chunk];chunk-signature=[sha256-signature][crlf]
 * [payload-bytes-of-this-chunk][crlf]
 * 
* * @see * * AwsChunkedEncodingInputStream */ public class AwsChunkedDecodingInputStream extends InputStream { /** * That's the max chunk buffer size used in the AWS implementation. */ private static final int MAX_CHUNK_SIZE = 256 * 1024; private static final byte[] CRLF = "\r\n".getBytes(StandardCharsets.UTF_8); private static final byte[] DELIMITER = ";".getBytes(StandardCharsets.UTF_8); private final InputStream source; private int remainingInChunk = 0; private final ByteBuffer byteBuffer = ByteBuffer.allocate(MAX_CHUNK_SIZE); /** * Constructs a new {@link AwsChunkedDecodingInputStream}. * * @param source The {@link InputStream} to wrap. */ AwsChunkedDecodingInputStream(final InputStream source) { this.source = source; } @Override public int read() throws IOException { if (remainingInChunk == 0) { final byte[] hexLengthBytes = readUntil(DELIMITER); if (hexLengthBytes == null) { return -1; } remainingInChunk = Integer.parseInt(new String(hexLengthBytes, StandardCharsets.UTF_8).trim(), 16); if (remainingInChunk == 0) { return -1; } readUntil(CRLF); } remainingInChunk--; return source.read(); } @Override public void close() throws IOException { source.close(); } /** * Reads this stream until the byte sequence was found. * * @param endSequence The byte sequence to look for in the stream. The source stream is read * until the last bytes read are equal to this sequence. * * @return The bytes read before the end sequence started. */ private byte[] readUntil(final byte[] endSequence) throws IOException { byteBuffer.clear(); while (!endsWith(byteBuffer.asReadOnlyBuffer(), endSequence)) { final int c = source.read(); if (c < 0) { return null; } final byte unsigned = (byte) (c & 0xFF); byteBuffer.put(unsigned); } final byte[] result = new byte[byteBuffer.position() - endSequence.length]; byteBuffer.rewind(); byteBuffer.get(result); return result; } private boolean endsWith(final ByteBuffer buffer, final byte[] endSequence) { final int pos = buffer.position(); if (pos >= endSequence.length) { for (int i = 0; i < endSequence.length; i++) { if (buffer.get(pos - endSequence.length + i) != endSequence[i]) { return false; } } return true; } return false; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy