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

com.azure.core.util.FluxUtil Maven / Gradle / Ivy

// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.

package com.azure.core.util;

import com.azure.core.http.HttpHeaderName;
import com.azure.core.http.HttpHeaders;
import com.azure.core.http.policy.ExponentialBackoffOptions;
import com.azure.core.http.policy.RetryOptions;
import com.azure.core.http.rest.PagedFlux;
import com.azure.core.http.rest.Response;
import com.azure.core.implementation.AsynchronousByteChannelWriteSubscriber;
import com.azure.core.implementation.ByteBufferCollector;
import com.azure.core.implementation.OutputStreamWriteSubscriber;
import com.azure.core.implementation.RetriableDownloadFlux;
import com.azure.core.implementation.TypeUtil;
import com.azure.core.util.io.IOUtils;
import com.azure.core.util.logging.ClientLogger;
import com.azure.core.util.logging.LoggingEventBuilder;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
import reactor.core.CoreSubscriber;
import reactor.core.Exceptions;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.core.publisher.Operators;
import reactor.core.scheduler.Schedulers;
import reactor.util.context.ContextView;

import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.lang.reflect.Type;
import java.nio.ByteBuffer;
import java.nio.channels.AsynchronousByteChannel;
import java.nio.channels.AsynchronousFileChannel;
import java.nio.channels.CompletionHandler;
import java.nio.channels.FileChannel;
import java.nio.channels.WritableByteChannel;
import java.nio.file.StandardOpenOption;
import java.util.Collections;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
import java.util.function.BiFunction;
import java.util.function.Function;
import java.util.function.Supplier;

/**
 * Utility type exposing methods to deal with {@link Flux}.
 */
public final class FluxUtil {
    private static final byte[] EMPTY_BYTE_ARRAY = new byte[0];
    private static final ClientLogger LOGGER = new ClientLogger(FluxUtil.class);

    /**
     * Checks if a type is Flux<ByteBuffer>.
     *
     * @param entityType the type to check
     * @return whether the type represents a Flux that emits ByteBuffer
     */
    public static boolean isFluxByteBuffer(Type entityType) {
        if (TypeUtil.isTypeOrSubTypeOf(entityType, Flux.class)) {
            final Type innerType = TypeUtil.getTypeArguments(entityType)[0];
            return TypeUtil.isTypeOrSubTypeOf(innerType, ByteBuffer.class);
        }
        return false;
    }

    /**
     * Adds progress reporting to the provided {@link Flux} of {@link ByteBuffer}.
     *
     * 

* Each {@link ByteBuffer} that's emitted from the {@link Flux} will report {@link ByteBuffer#remaining()}. *

*

* When {@link Flux} is resubscribed the progress is reset. If the flux is not replayable, resubscribing * can result in empty or partial data then progress reporting might not be accurate. *

*

* If {@link ProgressReporter} is not provided, i.e. is {@code null}, * then this method returns unmodified {@link Flux}. *

* * @param flux A {@link Flux} to report progress on. * @param progressReporter Optional {@link ProgressReporter}. * @return A {@link Flux} that reports progress, or original {@link Flux} if {@link ProgressReporter} is not * provided. */ public static Flux addProgressReporting(Flux flux, ProgressReporter progressReporter) { if (progressReporter == null) { return flux; } return Mono.just(progressReporter).flatMapMany(reporter -> { /* * Each time there is a new subscription, we will rewind the progress. This is desirable specifically for * retries, which resubscribe on each try. The first time this flowable is subscribed to, the reset will be * a noop as there will have been no progress made. Subsequent rewinds will work as expected. */ reporter.reset(); /* * Every time we emit some data, report it to the Tracker, which will pass it on to the end user. */ return flux.doOnNext(buffer -> reporter.reportProgress(buffer.remaining())); }); } /** * Collects ByteBuffers emitted by a Flux into a byte array. * * @param stream A stream which emits ByteBuffer instances. * @return A Mono which emits the concatenation of all the ByteBuffer instances given by the source Flux. * @throws IllegalStateException If the combined size of the emitted ByteBuffers is greater than {@link * Integer#MAX_VALUE}. */ public static Mono collectBytesInByteBufferStream(Flux stream) { return stream.collect(ByteBufferCollector::new, ByteBufferCollector::write) .map(ByteBufferCollector::toByteArray); } /** * Collects ByteBuffers emitted by a Flux into a byte array. *

* Unlike {@link #collectBytesInByteBufferStream(Flux)}, this method accepts a second parameter {@code sizeHint}. * This size hint allows for optimizations when creating the initial buffer to reduce the number of times it needs * to be resized while concatenating emitted ByteBuffers. * * @param stream A stream which emits ByteBuffer instances. * @param sizeHint A hint about the expected stream size. * @return A Mono which emits the concatenation of all the ByteBuffer instances given by the source Flux. * @throws IllegalArgumentException If {@code sizeHint} is equal to or less than {@code 0}. * @throws IllegalStateException If the combined size of the emitted ByteBuffers is greater than {@link * Integer#MAX_VALUE}. */ public static Mono collectBytesInByteBufferStream(Flux stream, int sizeHint) { return stream.collect(() -> new ByteBufferCollector(sizeHint), ByteBufferCollector::write) .map(ByteBufferCollector::toByteArray); } /** * Collects ByteBuffers returned in a network response into a byte array. *

* The {@code headers} are inspected for containing an {@code Content-Length} which determines if a size hinted * collection, {@link #collectBytesInByteBufferStream(Flux, int)}, or default collection, {@link * #collectBytesInByteBufferStream(Flux)}, will be used. * * @param stream A network response ByteBuffer stream. * @param headers The HTTP headers of the response. * @return A Mono which emits the collected network response ByteBuffers. * @throws NullPointerException If {@code headers} is null. * @throws IllegalStateException If the size of the network response is greater than {@link Integer#MAX_VALUE}. */ public static Mono collectBytesFromNetworkResponse(Flux stream, HttpHeaders headers) { Objects.requireNonNull(headers, "'headers' cannot be null."); String contentLengthHeader = headers.getValue(HttpHeaderName.CONTENT_LENGTH); if (contentLengthHeader == null) { return FluxUtil.collectBytesInByteBufferStream(stream); } else { try { int contentLength = Integer.parseInt(contentLengthHeader); if (contentLength > 0) { return FluxUtil.collectBytesInByteBufferStream(stream, contentLength); } else { return Mono.just(EMPTY_BYTE_ARRAY); } } catch (NumberFormatException ex) { return FluxUtil.collectBytesInByteBufferStream(stream); } } } /** * Gets the content of the provided ByteBuffer as a byte array. This method will create a new byte array even if the * ByteBuffer can have optionally backing array. * * @param byteBuffer the byte buffer * @return the byte array */ public static byte[] byteBufferToArray(ByteBuffer byteBuffer) { int length = byteBuffer.remaining(); byte[] byteArray = new byte[length]; byteBuffer.get(byteArray); return byteArray; } /** * Creates a {@link Flux} that is capable of resuming a download by applying retry logic when an error occurs. * * @param downloadSupplier Supplier of the initial download. * @param onDownloadErrorResume {@link BiFunction} of {@link Throwable} and {@link Long} which is used to resume * downloading when an error occurs. * @param maxRetries The maximum number of times a download can be resumed when an error occurs. * @return A {@link Flux} that downloads reliably. */ public static Flux createRetriableDownloadFlux(Supplier> downloadSupplier, BiFunction> onDownloadErrorResume, int maxRetries) { return createRetriableDownloadFlux(downloadSupplier, onDownloadErrorResume, createDefaultRetryOptions(maxRetries), 0L); } /** * Creates a {@link Flux} that is capable of resuming a download by applying retry logic when an error occurs. * * @param downloadSupplier Supplier of the initial download. * @param onDownloadErrorResume {@link BiFunction} of {@link Throwable} and {@link Long} which is used to resume * downloading when an error occurs. * @param maxRetries The maximum number of times a download can be resumed when an error occurs. * @param position The initial offset for the download. * @return A {@link Flux} that downloads reliably. */ public static Flux createRetriableDownloadFlux(Supplier> downloadSupplier, BiFunction> onDownloadErrorResume, int maxRetries, long position) { return createRetriableDownloadFlux(downloadSupplier, onDownloadErrorResume, createDefaultRetryOptions(maxRetries), position); } private static RetryOptions createDefaultRetryOptions(int maxRetries) { return new RetryOptions(new ExponentialBackoffOptions().setMaxRetries(Math.max(0, maxRetries))); } /** * Creates a {@link Flux} that is capable of resuming a download by applying retry logic when an error occurs. * * @param downloadSupplier Supplier of the initial download. * @param onDownloadErrorResume {@link BiFunction} of {@link Throwable} and {@link Long} which is used to resume * downloading when an error occurs. * @param retryOptions The options for retrying. * @param position The initial offset for the download. * @return A {@link Flux} that downloads reliably. */ public static Flux createRetriableDownloadFlux(Supplier> downloadSupplier, BiFunction> onDownloadErrorResume, RetryOptions retryOptions, long position) { RetryOptions options = (retryOptions == null) ? new RetryOptions(new ExponentialBackoffOptions()) : retryOptions; return new RetriableDownloadFlux(downloadSupplier, onDownloadErrorResume, options, position); } /** * Converts an {@link InputStream} into a {@link Flux} of {@link ByteBuffer} using a chunk size of 4096. *

* Given that {@link InputStream} is not guaranteed to be replayable the returned {@link Flux} should be considered * non-replayable as well. *

* If the passed {@link InputStream} is {@code null} {@link Flux#empty()} will be returned. * * @param inputStream The {@link InputStream} to convert into a {@link Flux}. * @return A {@link Flux} of {@link ByteBuffer ByteBuffers} that contains the contents of the stream. */ public static Flux toFluxByteBuffer(InputStream inputStream) { return toFluxByteBuffer(inputStream, 4096); } /** * Converts an {@link InputStream} into a {@link Flux} of {@link ByteBuffer}. *

* Given that {@link InputStream} is not guaranteed to be replayable the returned {@link Flux} should be considered * non-replayable as well. *

* If the passed {@link InputStream} is {@code null} {@link Flux#empty()} will be returned. * * @param inputStream The {@link InputStream} to convert into a {@link Flux}. * @param chunkSize The requested size for each {@link ByteBuffer}. * @return A {@link Flux} of {@link ByteBuffer ByteBuffers} that contains the contents of the stream. * @throws IllegalArgumentException If {@code chunkSize} is less than or equal to {@code 0}. */ public static Flux toFluxByteBuffer(InputStream inputStream, int chunkSize) { if (chunkSize <= 0) { return Flux.error(new IllegalArgumentException("'chunkSize' must be greater than 0.")); } if (inputStream == null) { return Flux.empty(); } // If the InputStream is an instance of FileInputStream we should be able to leverage the FileChannel backing // the FileInputStream to generated MappedByteBuffers which aren't loaded into memory until the content is // consumed. This at least defers the memory usage until later and may also provide downstream calls ways to // optimize if they have special cases for MappedByteBuffer. if (inputStream instanceof FileInputStream) { FileChannel fileChannel = ((FileInputStream) inputStream).getChannel(); return Flux.generate(() -> fileChannel, (channel, sink) -> { try { long channelPosition = channel.position(); long channelSize = channel.size(); if (channelPosition == channelSize) { // End of File has been reached, signal completion. channel.close(); sink.complete(); } else { // Determine the size of the next MappedByteBuffer, either the remaining File contents or the // expected chunk size. int nextByteBufferSize = (int) Math.min(chunkSize, channelSize - channelPosition); sink.next(channel.map(FileChannel.MapMode.READ_ONLY, channelPosition, nextByteBufferSize)); // FileChannel.map doesn't update the FileChannel's position as reading would, so the position // needs to be updated based on the number of bytes mapped. channel.position(channelPosition + nextByteBufferSize); } } catch (IOException ex) { sink.error(ex); } return channel; }); } return Flux.generate(() -> inputStream, (stream, sink) -> { byte[] buffer = new byte[chunkSize]; try { int offset = 0; while (offset < chunkSize) { int readCount = inputStream.read(buffer, offset, chunkSize - offset); // We have finished reading the stream, trigger onComplete. if (readCount == -1) { // If there were bytes read before reaching the end emit the buffer before completing. if (offset > 0) { sink.next(ByteBuffer.wrap(buffer, 0, offset)); } sink.complete(); return stream; } offset += readCount; } sink.next(ByteBuffer.wrap(buffer)); } catch (IOException ex) { sink.error(ex); } return stream; }).filter(ByteBuffer::hasRemaining); } /** * This method converts the incoming {@code deferContextual} from {@link reactor.util.context.Context Reactor * Context} to {@link Context Azure Context} and calls the given lambda function with this context and returns a * single entity of type {@code T} *

* If the reactor context is empty, {@link Context#NONE} will be used to call the lambda function *

* *

Code samples

* *
     * String prefix = "Hello, ";
     * Mono<String> response = FluxUtil
     *     .withContext(context -> serviceCallReturnsSingle(prefix, context));
     * 
* * * @param serviceCall The lambda function that makes the service call into which azure context will be passed * @param The type of response returned from the service call * @return The response from service call */ public static Mono withContext(Function> serviceCall) { return withContext(serviceCall, Collections.emptyMap()); } /** * This method converts the incoming {@code deferContextual} from {@link reactor.util.context.Context Reactor * Context} to {@link Context Azure Context}, adds the specified context attributes and calls the given lambda * function with this context and returns a single entity of type {@code T} *

* If the reactor context is empty, {@link Context#NONE} will be used to call the lambda function *

* * @param serviceCall serviceCall The lambda function that makes the service call into which azure context will be * passed * @param contextAttributes The map of attributes sent by the calling method to be set on {@link Context}. * @param The type of response returned from the service call * @return The response from service call */ public static Mono withContext(Function> serviceCall, Map contextAttributes) { return Mono.deferContextual(context -> { final Context[] azureContext = new Context[]{Context.NONE}; if (!CoreUtils.isNullOrEmpty(contextAttributes)) { contextAttributes.forEach((key, value) -> azureContext[0] = azureContext[0].addData(key, value)); } if (!context.isEmpty()) { context.stream().forEach(entry -> azureContext[0] = azureContext[0].addData(entry.getKey(), entry.getValue())); } return serviceCall.apply(azureContext[0]); }); } /** * Converts the incoming content to Mono. * * @param The type of the Response, which will be returned in the Mono. * @param response whose {@link Response#getValue() value} is to be converted * @return The converted {@link Mono} */ public static Mono toMono(Response response) { return Mono.justOrEmpty(response.getValue()); } /** * Propagates a {@link RuntimeException} through the error channel of {@link Mono}. * * @param logger The {@link ClientLogger} to log the exception. * @param ex The {@link RuntimeException}. * @param The return type. * @return A {@link Mono} that terminates with error wrapping the {@link RuntimeException}. */ public static Mono monoError(ClientLogger logger, RuntimeException ex) { return Mono.error(logger.logExceptionAsError(Exceptions.propagate(ex))); } /** * Propagates a {@link RuntimeException} through the error channel of {@link Mono}. * * @param logBuilder The {@link LoggingEventBuilder} with context to log the exception. * @param ex The {@link RuntimeException}. * @param The return type. * @return A {@link Mono} that terminates with error wrapping the {@link RuntimeException}. */ public static Mono monoError(LoggingEventBuilder logBuilder, RuntimeException ex) { return Mono.error(logBuilder.log(Exceptions.propagate(ex))); } /** * Propagates a {@link RuntimeException} through the error channel of {@link Flux}. * * @param logger The {@link ClientLogger} to log the exception. * @param ex The {@link RuntimeException}. * @param The return type. * @return A {@link Flux} that terminates with error wrapping the {@link RuntimeException}. */ public static Flux fluxError(ClientLogger logger, RuntimeException ex) { return Flux.error(logger.logExceptionAsError(Exceptions.propagate(ex))); } /** * Propagates a {@link RuntimeException} through the error channel of {@link PagedFlux}. * * @param logger The {@link ClientLogger} to log the exception. * @param ex The {@link RuntimeException}. * @param The return type. * @return A {@link PagedFlux} that terminates with error wrapping the {@link RuntimeException}. */ public static PagedFlux pagedFluxError(ClientLogger logger, RuntimeException ex) { return new PagedFlux<>(() -> monoError(logger, ex)); } /** * This method converts the incoming {@code deferContextual} from {@link reactor.util.context.Context Reactor * Context} to {@link Context Azure Context} and calls the given lambda function with this context and returns a * collection of type {@code T} *

* If the reactor context is empty, {@link Context#NONE} will be used to call the lambda function *

* *

Code samples

* *
     * String prefix = "Hello, ";
     * Flux<String> response = FluxUtil
     *     .fluxContext(context -> serviceCallReturnsCollection(prefix, context));
     * 
* * * @param serviceCall The lambda function that makes the service call into which the context will be passed * @param The type of response returned from the service call * @return The response from service call */ public static Flux fluxContext(Function> serviceCall) { return Flux.deferContextual(context -> serviceCall.apply(toAzureContext(context))); } /** * Converts a reactor context to azure context. If the reactor context is {@code null} or empty, {@link * Context#NONE} will be returned. * * @param context The reactor context * @return The azure context */ private static Context toAzureContext(ContextView context) { final Context[] azureContext = new Context[]{Context.NONE}; if (!context.isEmpty()) { context.stream().forEach(entry -> azureContext[0] = azureContext[0].addData(entry.getKey(), entry.getValue())); } return azureContext[0]; } /** * Converts an Azure context to Reactor context. If the Azure context is {@code null} or empty, {@link * reactor.util.context.Context#empty()} will be returned. * * @param context The Azure context. * @return The Reactor context. */ public static reactor.util.context.Context toReactorContext(Context context) { if (context == null) { return reactor.util.context.Context.empty(); } reactor.util.context.Context returnContext = reactor.util.context.Context.empty(); Context[] contextChain = context.getContextChain(); for (Context toAdd : contextChain) { // Filter out null value entries as Reactor's context doesn't allow null values. if (toAdd == null || toAdd.getValue() == null) { continue; } returnContext = returnContext.put(toAdd.getKey(), toAdd.getValue()); } return returnContext; } /** * Writes the {@link ByteBuffer ByteBuffers} emitted by a {@link Flux} of {@link ByteBuffer} to an {@link * OutputStream}. *

* The {@code stream} is not closed by this call, closing of the {@code stream} is managed by the caller. *

* The response {@link Mono} will emit an error if {@code content} or {@code stream} are null. Additionally, an * error will be emitted if an exception occurs while writing the {@code content} to the {@code stream}. * * @param content The {@link Flux} of {@link ByteBuffer} content. * @param stream The {@link OutputStream} being written into. * @return A {@link Mono} which emits a completion status once the {@link Flux} has been written to the {@link * OutputStream}, or an error status if writing fails. */ public static Mono writeToOutputStream(Flux content, OutputStream stream) { if (content == null && stream == null) { return monoError(LOGGER, new NullPointerException("'content' and 'stream' cannot be null.")); } else if (content == null) { return monoError(LOGGER, new NullPointerException("'content' cannot be null.")); } else if (stream == null) { return monoError(LOGGER, new NullPointerException("'stream' cannot be null.")); } return Mono.create(emitter -> content.subscribe(new OutputStreamWriteSubscriber(emitter, stream, LOGGER))); } /** * Writes the {@link ByteBuffer ByteBuffers} emitted by a {@link Flux} of {@link ByteBuffer} to an {@link * AsynchronousFileChannel}. *

* The {@code outFile} is not closed by this call, closing of the {@code outFile} is managed by the caller. *

* The response {@link Mono} will emit an error if {@code content} or {@code outFile} are null. Additionally, an * error will be emitted if the {@code outFile} wasn't opened with the proper open options, such as {@link * StandardOpenOption#WRITE}. * * @param content The {@link Flux} of {@link ByteBuffer} content. * @param outFile The {@link AsynchronousFileChannel}. * @return A {@link Mono} which emits a completion status once the {@link Flux} has been written to the {@link * AsynchronousFileChannel}. * @throws NullPointerException When {@code content} is null. * @throws NullPointerException When {@code outFile} is null. */ public static Mono writeFile(Flux content, AsynchronousFileChannel outFile) { return writeFile(content, outFile, 0); } /** * Writes the {@link ByteBuffer ByteBuffers} emitted by a {@link Flux} of {@link ByteBuffer} to an {@link * AsynchronousFileChannel} starting at the given {@code position} in the file. *

* The {@code outFile} is not closed by this call, closing of the {@code outFile} is managed by the caller. *

* The response {@link Mono} will emit an error if {@code content} or {@code outFile} are null or {@code position} * is less than 0. Additionally, an error will be emitted if the {@code outFile} wasn't opened with the proper open * options, such as {@link StandardOpenOption#WRITE}. * * @param content The {@link Flux} of {@link ByteBuffer} content. * @param outFile The {@link AsynchronousFileChannel}. * @param position The position in the file to begin writing the {@code content}. * @return A {@link Mono} which emits a completion status once the {@link Flux} has been written to the {@link * AsynchronousFileChannel}. * @throws NullPointerException When {@code content} is null. * @throws NullPointerException When {@code outFile} is null. * @throws IllegalArgumentException When {@code position} is negative. */ public static Mono writeFile(Flux content, AsynchronousFileChannel outFile, long position) { if (content == null && outFile == null) { return monoError(LOGGER, new NullPointerException("'content' and 'outFile' cannot be null.")); } else if (content == null) { return monoError(LOGGER, new NullPointerException("'content' cannot be null.")); } else if (outFile == null) { return monoError(LOGGER, new NullPointerException("'outFile' cannot be null.")); } else if (position < 0) { return monoError(LOGGER, new IllegalArgumentException("'position' cannot be less than 0.")); } return writeToAsynchronousByteChannel(content, IOUtils.toAsynchronousByteChannel(outFile, position)); } /** * Writes the {@link ByteBuffer ByteBuffers} emitted by a {@link Flux} of {@link ByteBuffer} to an {@link * AsynchronousByteChannel}. *

* The {@code channel} is not closed by this call, closing of the {@code channel} is managed by the caller. *

* The response {@link Mono} will emit an error if {@code content} or {@code channel} are null. * * @param content The {@link Flux} of {@link ByteBuffer} content. * @param channel The {@link AsynchronousByteChannel}. * @return A {@link Mono} which emits a completion status once the {@link Flux} has been written to the {@link * AsynchronousByteChannel}. * @throws NullPointerException When {@code content} is null. * @throws NullPointerException When {@code channel} is null. */ public static Mono writeToAsynchronousByteChannel(Flux content, AsynchronousByteChannel channel) { if (content == null && channel == null) { return monoError(LOGGER, new NullPointerException("'content' and 'channel' cannot be null.")); } else if (content == null) { return monoError(LOGGER, new NullPointerException("'content' cannot be null.")); } else if (channel == null) { return monoError(LOGGER, new NullPointerException("'channel' cannot be null.")); } return Mono.create(emitter -> content.subscribe( new AsynchronousByteChannelWriteSubscriber(channel, emitter))); } /** * Writes the {@link ByteBuffer ByteBuffers} emitted by a {@link Flux} of {@link ByteBuffer} to an {@link * WritableByteChannel}. *

* The {@code channel} is not closed by this call, closing of the {@code channel} is managed by the caller. *

* The response {@link Mono} will emit an error if {@code content} or {@code channel} are null. * * @param content The {@link Flux} of {@link ByteBuffer} content. * @param channel The {@link WritableByteChannel}. * @return A {@link Mono} which emits a completion status once the {@link Flux} has been written to the {@link * WritableByteChannel}. * @throws NullPointerException When {@code content} is null. * @throws NullPointerException When {@code channel} is null. */ public static Mono writeToWritableByteChannel(Flux content, WritableByteChannel channel) { if (content == null && channel == null) { return monoError(LOGGER, new NullPointerException("'content' and 'channel' cannot be null.")); } else if (content == null) { return monoError(LOGGER, new NullPointerException("'content' cannot be null.")); } else if (channel == null) { return monoError(LOGGER, new NullPointerException("'channel' cannot be null.")); } return content.publishOn(Schedulers.boundedElastic()) .map(buffer -> { while (buffer.hasRemaining()) { try { channel.write(buffer); } catch (IOException e) { throw Exceptions.propagate(e); } } return buffer; }).then(); } /** * Creates a {@link Flux} from an {@link AsynchronousFileChannel} which reads part of a file into chunks of the * given size. * * @param fileChannel The file channel. * @param chunkSize the size of file chunks to read. * @param offset The offset in the file to begin reading. * @param length The number of bytes to read from the file. * @return the Flux. */ public static Flux readFile(AsynchronousFileChannel fileChannel, int chunkSize, long offset, long length) { return new FileReadFlux(fileChannel, chunkSize, offset, length); } /** * Creates a {@link Flux} from an {@link AsynchronousFileChannel} which reads part of a file. * * @param fileChannel The file channel. * @param offset The offset in the file to begin reading. * @param length The number of bytes to read from the file. * @return the Flux. */ public static Flux readFile(AsynchronousFileChannel fileChannel, long offset, long length) { return readFile(fileChannel, DEFAULT_CHUNK_SIZE, offset, length); } /** * Creates a {@link Flux} from an {@link AsynchronousFileChannel} which reads the entire file. * * @param fileChannel The file channel. * @return The AsyncInputStream. */ public static Flux readFile(AsynchronousFileChannel fileChannel) { try { long size = fileChannel.size(); return readFile(fileChannel, DEFAULT_CHUNK_SIZE, 0, size); } catch (IOException e) { return Flux.error(new RuntimeException("Failed to read the file.", e)); } } private static final int DEFAULT_CHUNK_SIZE = 1024 * 64; private static final class FileReadFlux extends Flux { private final AsynchronousFileChannel fileChannel; private final int chunkSize; private final long offset; private final long length; FileReadFlux(AsynchronousFileChannel fileChannel, int chunkSize, long offset, long length) { this.fileChannel = fileChannel; this.chunkSize = chunkSize; this.offset = offset; this.length = length; } @Override public void subscribe(CoreSubscriber actual) { FileReadSubscription subscription = new FileReadSubscription(actual, fileChannel, chunkSize, offset, length); actual.onSubscribe(subscription); } static final class FileReadSubscription implements Subscription, CompletionHandler { private static final int NOT_SET = -1; private static final long serialVersionUID = -6831808726875304256L; // private final Subscriber subscriber; private volatile long position; // private final AsynchronousFileChannel fileChannel; private final int chunkSize; private final long offset; private final long length; // private volatile boolean done; private Throwable error; private volatile ByteBuffer next; private volatile boolean cancelled; // volatile int wip; static final AtomicIntegerFieldUpdater WIP = AtomicIntegerFieldUpdater.newUpdater(FileReadSubscription.class, "wip"); volatile long requested; static final AtomicLongFieldUpdater REQUESTED = AtomicLongFieldUpdater.newUpdater(FileReadSubscription.class, "requested"); // FileReadSubscription(Subscriber subscriber, AsynchronousFileChannel fileChannel, int chunkSize, long offset, long length) { this.subscriber = subscriber; // this.fileChannel = fileChannel; this.chunkSize = chunkSize; this.offset = offset; this.length = length; // this.position = NOT_SET; } //region Subscription implementation @Override public void request(long n) { if (Operators.validate(n)) { Operators.addCap(REQUESTED, this, n); drain(); } } @Override public void cancel() { this.cancelled = true; } //endregion //region CompletionHandler implementation @Override public void completed(Integer bytesRead, ByteBuffer buffer) { if (!cancelled) { if (bytesRead == -1) { done = true; } else { // use local variable to perform fewer volatile reads long pos = position; int bytesWanted = Math.min(bytesRead, maxRequired(pos)); long position2 = pos + bytesWanted; //noinspection NonAtomicOperationOnVolatileField position = position2; buffer.position(bytesWanted); buffer.flip(); next = buffer; if (position2 >= offset + length) { done = true; } } drain(); } } @Override public void failed(Throwable exc, ByteBuffer attachment) { if (!cancelled) { // must set error before setting done to true // so that is visible in drain loop error = exc; done = true; drain(); } } //endregion private void drain() { if (WIP.getAndIncrement(this) != 0) { return; } // on first drain (first request) we initiate the first read if (position == NOT_SET) { position = offset; doRead(); } int missed = 1; while (true) { if (cancelled) { return; } if (REQUESTED.get(this) > 0) { boolean emitted = false; // read d before next to avoid race boolean d = done; ByteBuffer bb = next; if (bb != null) { next = null; subscriber.onNext(bb); emitted = true; } if (d) { if (error != null) { subscriber.onError(error); } else { subscriber.onComplete(); } // exit without reducing wip so that further drains will be NOOP return; } if (emitted) { // do this after checking d to avoid calling read // when done Operators.produced(REQUESTED, this, 1); // doRead(); } } missed = WIP.addAndGet(this, -missed); if (missed == 0) { return; } } } private void doRead() { // use local variable to limit volatile reads long pos = position; ByteBuffer innerBuf = ByteBuffer.allocate(Math.min(chunkSize, maxRequired(pos))); fileChannel.read(innerBuf, pos, innerBuf, this); } private int maxRequired(long pos) { long maxRequired = offset + length - pos; if (maxRequired <= 0) { return 0; } else { int m = (int) (maxRequired); // support really large files by checking for overflow if (m < 0) { return Integer.MAX_VALUE; } else { return m; } } } } } // Private Ctr private FluxUtil() { } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy