com.azure.core.implementation.ByteCountingAsynchronousByteChannel Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of azure-core Show documentation
Show all versions of azure-core Show documentation
This package contains core types for Azure Java clients.
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
package com.azure.core.implementation;
import com.azure.core.util.ProgressReporter;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.AsynchronousByteChannel;
import java.nio.channels.CompletionHandler;
import java.util.Objects;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
/**
* Count bytes written and read to the target channel.
*/
public class ByteCountingAsynchronousByteChannel implements AsynchronousByteChannel {
private final AsynchronousByteChannel channel;
private final ProgressReporter readProgressReporter;
private final ProgressReporter writeProgressReporter;
private static final AtomicLongFieldUpdater BYTES_WRITTEN_ATOMIC_UPDATER
= AtomicLongFieldUpdater.newUpdater(ByteCountingAsynchronousByteChannel.class, "bytesWritten");
private volatile long bytesWritten;
private static final AtomicLongFieldUpdater BYTES_READ_ATOMIC_UPDATER
= AtomicLongFieldUpdater.newUpdater(ByteCountingAsynchronousByteChannel.class, "bytesRead");
private volatile long bytesRead;
/**
* Creates an instance of {@link ByteCountingAsynchronousByteChannel} that counts bytes written and read to the
* target channel.
*
* @param channel The {@link AsynchronousByteChannel} to adapt.
* @param readProgressReporter The {@link ProgressReporter} to report progress on read operations.
* @param writeProgressReporter The {@link ProgressReporter} to report progress on write operations.
*/
public ByteCountingAsynchronousByteChannel(AsynchronousByteChannel channel, ProgressReporter readProgressReporter,
ProgressReporter writeProgressReporter) {
this.channel = Objects.requireNonNull(channel, "'channel' must not be null");
this.readProgressReporter = readProgressReporter;
this.writeProgressReporter = writeProgressReporter;
}
@Override
public void read(ByteBuffer dst, A attachment, CompletionHandler handler) {
this.channel.read(dst, attachment,
new DelegatingCompletionHandler(handler, BYTES_READ_ATOMIC_UPDATER, readProgressReporter));
}
@Override
public Future read(ByteBuffer dst) {
CompletableFuture future = new CompletableFuture<>();
channel.read(dst, dst,
new DelegatingCompletionHandler<>(future, BYTES_READ_ATOMIC_UPDATER, readProgressReporter));
return future;
}
@Override
public void write(ByteBuffer src, A attachment, CompletionHandler handler) {
// We're implementing channel interface here, i.e. we don't have to consume whole buffer in one shot.
// Caller is responsible for that.
this.channel.write(src, attachment,
new DelegatingCompletionHandler(handler, BYTES_WRITTEN_ATOMIC_UPDATER, writeProgressReporter));
}
@Override
public Future write(ByteBuffer src) {
CompletableFuture future = new CompletableFuture<>();
// We're implementing channel interface here, i.e. we don't have to consume whole buffer in one shot.
// Caller is responsible for that.
channel.write(src, src,
new DelegatingCompletionHandler<>(future, BYTES_WRITTEN_ATOMIC_UPDATER, writeProgressReporter));
return future;
}
@Override
public boolean isOpen() {
return channel.isOpen();
}
@Override
public void close() throws IOException {
channel.close();
}
/**
* Gets the number of bytes written to the target channel.
*
* @return The number of bytes written to the target channel.
*/
public long getBytesWritten() {
return BYTES_WRITTEN_ATOMIC_UPDATER.get(this);
}
/**
* Gets the number of bytes read from the target channel.
*
* @return The number of bytes read from the target channel.
*/
public long getBytesRead() {
return BYTES_READ_ATOMIC_UPDATER.get(this);
}
private final class DelegatingCompletionHandler implements CompletionHandler {
private final CompletionHandler handler;
private final CompletableFuture future;
private final AtomicLongFieldUpdater atomicLongFieldUpdater;
private final ProgressReporter progressReporter;
private DelegatingCompletionHandler(CompletionHandler handler,
AtomicLongFieldUpdater atomicLongFieldUpdater,
ProgressReporter progressReporter) {
this.handler = handler;
this.future = null;
this.atomicLongFieldUpdater = atomicLongFieldUpdater;
this.progressReporter = progressReporter;
}
private DelegatingCompletionHandler(CompletableFuture future,
AtomicLongFieldUpdater atomicLongFieldUpdater,
ProgressReporter progressReporter) {
this.handler = null;
this.future = future;
this.atomicLongFieldUpdater = atomicLongFieldUpdater;
this.progressReporter = progressReporter;
}
@Override
public void completed(Integer result, T attachment) {
if (result > 0) {
atomicLongFieldUpdater.addAndGet(ByteCountingAsynchronousByteChannel.this, result);
if (progressReporter != null) {
progressReporter.reportProgress(result);
}
}
if (handler != null) {
handler.completed(result, attachment);
} else if (future != null) {
future.complete(result);
}
}
@Override
public void failed(Throwable exc, T attachment) {
if (handler != null) {
handler.failed(exc, attachment);
} else if (future != null) {
future.completeExceptionally(exc);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy