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

org.elasticsearch.common.io.Streams Maven / Gradle / Ivy

There is a newer version: 8.13.2
Show newest version
/*
 * Licensed to Elasticsearch under one or more contributor
 * license agreements. See the NOTICE file distributed with
 * this work for additional information regarding copyright
 * ownership. Elasticsearch licenses this file to you 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.
 */

package org.elasticsearch.common.io;

import org.elasticsearch.common.bytes.BytesReference;
import org.elasticsearch.common.io.stream.BytesStream;
import org.elasticsearch.common.io.stream.BytesStreamOutput;
import org.elasticsearch.common.io.stream.StreamOutput;

import java.io.BufferedReader;
import java.io.FilterInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.Reader;
import java.io.StringWriter;
import java.io.Writer;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.function.Consumer;

/**
 * Simple utility methods for file and stream copying.
 * All copy methods use a block size of 4096 bytes,
 * and close all affected streams when done.
 * 

* Mainly for use within the framework, * but also useful for application code. */ public abstract class Streams { public static final int BUFFER_SIZE = 1024 * 8; //--------------------------------------------------------------------- // Copy methods for java.io.InputStream / java.io.OutputStream //--------------------------------------------------------------------- public static long copy(InputStream in, OutputStream out) throws IOException { return copy(in, out, new byte[BUFFER_SIZE]); } /** * Copy the contents of the given InputStream to the given OutputStream. * Closes both streams when done. * * @param in the stream to copy from * @param out the stream to copy to * @return the number of bytes copied * @throws IOException in case of I/O errors */ public static long copy(InputStream in, OutputStream out, byte[] buffer) throws IOException { Objects.requireNonNull(in, "No InputStream specified"); Objects.requireNonNull(out, "No OutputStream specified"); // Leverage try-with-resources to close in and out so that exceptions in close() are either propagated or added as suppressed // exceptions to the main exception try (InputStream in2 = in; OutputStream out2 = out) { return doCopy(in2, out2, buffer); } } private static long doCopy(InputStream in, OutputStream out, byte[] buffer) throws IOException { long byteCount = 0; int bytesRead; while ((bytesRead = in.read(buffer)) != -1) { out.write(buffer, 0, bytesRead); byteCount += bytesRead; } out.flush(); return byteCount; } /** * Copy the contents of the given byte array to the given OutputStream. * Closes the stream when done. * * @param in the byte array to copy from * @param out the OutputStream to copy to * @throws IOException in case of I/O errors */ public static void copy(byte[] in, OutputStream out) throws IOException { Objects.requireNonNull(in, "No input byte array specified"); Objects.requireNonNull(out, "No OutputStream specified"); try (OutputStream out2 = out) { out2.write(in); } } //--------------------------------------------------------------------- // Copy methods for java.io.Reader / java.io.Writer //--------------------------------------------------------------------- /** * Copy the contents of the given Reader to the given Writer. * Closes both when done. * * @param in the Reader to copy from * @param out the Writer to copy to * @return the number of characters copied * @throws IOException in case of I/O errors */ public static int copy(Reader in, Writer out) throws IOException { Objects.requireNonNull(in, "No Reader specified"); Objects.requireNonNull(out, "No Writer specified"); // Leverage try-with-resources to close in and out so that exceptions in close() are either propagated or added as suppressed // exceptions to the main exception try (Reader in2 = in; Writer out2 = out) { return doCopy(in2, out2); } } private static int doCopy(Reader in, Writer out) throws IOException { int byteCount = 0; char[] buffer = new char[BUFFER_SIZE]; int bytesRead; while ((bytesRead = in.read(buffer)) != -1) { out.write(buffer, 0, bytesRead); byteCount += bytesRead; } out.flush(); return byteCount; } /** * Copy the contents of the given String to the given output Writer. * Closes the write when done. * * @param in the String to copy from * @param out the Writer to copy to * @throws IOException in case of I/O errors */ public static void copy(String in, Writer out) throws IOException { Objects.requireNonNull(in, "No input String specified"); Objects.requireNonNull(out, "No Writer specified"); try (Writer out2 = out) { out2.write(in); } } /** * Copy the contents of the given Reader into a String. * Closes the reader when done. * * @param in the reader to copy from * @return the String that has been copied to * @throws IOException in case of I/O errors */ public static String copyToString(Reader in) throws IOException { StringWriter out = new StringWriter(); copy(in, out); return out.toString(); } public static int readFully(Reader reader, char[] dest) throws IOException { return readFully(reader, dest, 0, dest.length); } public static int readFully(Reader reader, char[] dest, int offset, int len) throws IOException { int read = 0; while (read < len) { final int r = reader.read(dest, offset + read, len - read); if (r == -1) { break; } read += r; } return read; } public static int readFully(InputStream reader, byte[] dest) throws IOException { return readFully(reader, dest, 0, dest.length); } public static int readFully(InputStream reader, byte[] dest, int offset, int len) throws IOException { int read = 0; while (read < len) { final int r = reader.read(dest, offset + read, len - read); if (r == -1) { break; } read += r; } return read; } public static List readAllLines(InputStream input) throws IOException { final List lines = new ArrayList<>(); readAllLines(input, lines::add); return lines; } public static void readAllLines(InputStream input, Consumer consumer) throws IOException { try (BufferedReader reader = new BufferedReader(new InputStreamReader(input, StandardCharsets.UTF_8))) { String line; while ((line = reader.readLine()) != null) { consumer.accept(line); } } } /** * Wraps an {@link InputStream} such that it's {@code close} method becomes a noop * * @param stream {@code InputStream} to wrap * @return wrapped {@code InputStream} */ public static InputStream noCloseStream(InputStream stream) { return new FilterInputStream(stream) { @Override public void close() { // noop } }; } /** * Wraps the given {@link BytesStream} in a {@link StreamOutput} that simply flushes when * close is called. */ public static BytesStream flushOnCloseStream(BytesStream os) { return new FlushOnCloseOutputStream(os); } /** * Reads all bytes from the given {@link InputStream} and closes it afterwards. */ public static BytesReference readFully(InputStream in) throws IOException { try (InputStream inputStream = in) { BytesStreamOutput out = new BytesStreamOutput(); copy(inputStream, out); return out.bytes(); } } /** * A wrapper around a {@link BytesStream} that makes the close operation a flush. This is * needed as sometimes a stream will be closed but the bytes that the stream holds still need * to be used and the stream cannot be closed until the bytes have been consumed. */ private static class FlushOnCloseOutputStream extends BytesStream { private final BytesStream delegate; private FlushOnCloseOutputStream(BytesStream bytesStreamOutput) { this.delegate = bytesStreamOutput; } @Override public void writeByte(byte b) throws IOException { delegate.writeByte(b); } @Override public void writeBytes(byte[] b, int offset, int length) throws IOException { delegate.writeBytes(b, offset, length); } @Override public void flush() throws IOException { delegate.flush(); } @Override public void close() throws IOException { flush(); } @Override public void reset() throws IOException { delegate.reset(); } @Override public BytesReference bytes() { return delegate.bytes(); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy