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

software.amazon.awssdk.utils.LookaheadInputStream Maven / Gradle / Ivy

There is a newer version: 2.29.16
Show newest version
/*
 * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License").
 * You may not use this file except in compliance with the License.
 * A copy of the License is located at
 *
 *  http://aws.amazon.com/apache2.0
 *
 * or in the "license" file accompanying this file. This file 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.
 */

package software.amazon.awssdk.utils;

import java.io.FilterInputStream;
import java.io.IOException;
import java.io.InputStream;
import software.amazon.awssdk.annotations.SdkProtectedApi;

/**
 * A wrapper for an {@link InputStream} that allows {@link #peek()}ing one byte ahead in the stream. This is useful for
 * detecting the end of a stream without actually consuming any data in the process (e.g. so the stream can be passed to
 * another library that doesn't handle end-of-stream as the first byte well).
 */
@SdkProtectedApi
public class LookaheadInputStream extends FilterInputStream {
    private Integer next;
    private Integer nextAtMark;

    public LookaheadInputStream(InputStream in) {
        super(in);
    }

    public int peek() throws IOException {
        if (next == null) {
            next = read();
        }

        return next;
    }

    @Override
    public int read() throws IOException {
        if (next == null) {
            return super.read();
        }

        Integer next = this.next;
        this.next = null;
        return next;
    }

    @Override
    public int read(byte[] b, int off, int len) throws IOException {
        if (next == null) {
            return super.read(b, off, len);
        }

        if (len == 0) {
            return 0;
        }

        if (next == -1) {
            return -1;
        }

        b[off] = (byte) (int) next;
        next = null;

        if (len == 1) {
            return 1;
        }

        return super.read(b, off + 1, b.length - 1) + 1;
    }

    @Override
    public long skip(long n) throws IOException {
        if (next == null) {
            return super.skip(n);
        }

        if (n == 0) {
            return 0;
        }

        if (next == -1) {
            return 0;
        }

        next = null;

        if (n == 1) {
            return 1;
        }

        return super.skip(n - 1);
    }

    @Override
    public int available() throws IOException {
        if (next == null) {
            return super.available();
        }

        return super.available() + 1;
    }

    @Override
    public synchronized void mark(int readlimit) {
        if (next == null) {
            super.mark(readlimit);
        } else {
            nextAtMark = next;
            super.mark(readlimit - 1);
        }
    }

    @Override
    public synchronized void reset() throws IOException {
        next = nextAtMark;
        super.reset();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy