com.azure.core.implementation.RetriableDownloadFlux Maven / Gradle / Ivy
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
package com.azure.core.implementation;
import com.azure.core.util.logging.ClientLogger;
import com.azure.core.util.logging.LogLevel;
import reactor.core.CoreSubscriber;
import reactor.core.publisher.Flux;
import java.nio.ByteBuffer;
import java.util.function.BiFunction;
import java.util.function.Supplier;
/**
* A {@code Flux} implementation which is capable of performing a retriable download by applying a resume
* operation if an error occurs during the download.
*/
public final class RetriableDownloadFlux extends Flux {
private static final ClientLogger LOGGER = new ClientLogger(RetriableDownloadFlux.class);
private final Supplier> downloadSupplier;
private final BiFunction> onDownloadErrorResume;
private final int maxRetries;
private final long position;
private final int retryCount;
/**
* Creates a RetriableDownloadFlux.
*
* @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.
*/
public RetriableDownloadFlux(Supplier> downloadSupplier,
BiFunction> onDownloadErrorResume, int maxRetries, long position) {
this(downloadSupplier, onDownloadErrorResume, maxRetries, position, 0);
}
private RetriableDownloadFlux(Supplier> downloadSupplier,
BiFunction> onDownloadErrorResume, int maxRetries, long position,
int retryCount) {
this.downloadSupplier = downloadSupplier;
this.onDownloadErrorResume = onDownloadErrorResume;
this.maxRetries = maxRetries;
this.position = position;
this.retryCount = retryCount;
}
@Override
public void subscribe(CoreSubscriber super ByteBuffer> actual) {
final long[] currentPosition = new long[]{position};
downloadSupplier.get()
.map(buffer -> {
currentPosition[0] += buffer.remaining();
return buffer;
})
.onErrorResume(Exception.class, exception -> {
int updatedRetryCount = retryCount + 1;
if (updatedRetryCount > maxRetries) {
LOGGER.log(LogLevel.ERROR, () -> "Exhausted all retry attempts while downloading, "
+ maxRetries + " of " + maxRetries + ".", exception);
return Flux.error(exception);
}
LOGGER.log(LogLevel.INFORMATIONAL,
() -> "Using retry attempt " + updatedRetryCount + " of " + maxRetries + " while downloading.",
exception);
return new RetriableDownloadFlux(() -> onDownloadErrorResume.apply(exception, currentPosition[0]),
onDownloadErrorResume, maxRetries, currentPosition[0], updatedRetryCount);
})
.subscribe(actual);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy