com.azure.core.implementation.AsynchronousFileChannelAdapter 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.logging.ClientLogger;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.AsynchronousByteChannel;
import java.nio.channels.AsynchronousFileChannel;
import java.nio.channels.CompletionHandler;
import java.nio.channels.ReadPendingException;
import java.nio.channels.WritePendingException;
import java.util.Objects;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
/**
* Adapts {@link AsynchronousFileChannel} to {@link AsynchronousByteChannel}
*/
public class AsynchronousFileChannelAdapter implements AsynchronousByteChannel {
private static final ClientLogger LOGGER = new ClientLogger(AsynchronousFileChannelAdapter.class);
private final AsynchronousFileChannel fileChannel;
private static final AtomicLongFieldUpdater POSITION_ATOMIC_UPDATER
= AtomicLongFieldUpdater.newUpdater(AsynchronousFileChannelAdapter.class, "position");
private volatile long position;
// AsynchronousByteChannel implementation may disallow concurrent reads and writes.
private static final AtomicReferenceFieldUpdater PENDING_OPERATION_ATOMIC_UPDATER
= AtomicReferenceFieldUpdater.newUpdater(AsynchronousFileChannelAdapter.class, Operation.class,
"pendingOperation");
private volatile Operation pendingOperation = null;
/**
* Creates an instance of {@link AsynchronousFileChannelAdapter} that adapts {@link AsynchronousFileChannel} to
* {@link AsynchronousByteChannel}.
*
* @param fileChannel The {@link AsynchronousFileChannel} to adapt.
* @param position The position to start reading from or writing to.
*/
public AsynchronousFileChannelAdapter(AsynchronousFileChannel fileChannel, long position) {
this.fileChannel = Objects.requireNonNull(fileChannel);
this.position = position;
}
@Override
public void read(ByteBuffer dst, A attachment, CompletionHandler handler) {
beginOperation(Operation.READ);
fileChannel.read(dst, POSITION_ATOMIC_UPDATER.get(this), attachment,
new DelegatingCompletionHandler<>(handler, Operation.READ));
}
@Override
public Future read(ByteBuffer dst) {
beginOperation(Operation.READ);
CompletableFuture future = new CompletableFuture<>();
fileChannel.read(dst, POSITION_ATOMIC_UPDATER.get(this), dst,
new DelegatingCompletionHandler<>(future, Operation.READ));
return future;
}
@Override
public void write(ByteBuffer src, A attachment, CompletionHandler handler) {
beginOperation(Operation.WRITE);
// We're implementing channel interface here, i.e. we don't have to consume whole buffer in one shot.
// Caller is responsible for that.
fileChannel.write(src, POSITION_ATOMIC_UPDATER.get(this), attachment,
new DelegatingCompletionHandler<>(handler, Operation.WRITE));
}
@Override
public Future write(ByteBuffer src) {
beginOperation(Operation.WRITE);
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.
fileChannel.write(src, POSITION_ATOMIC_UPDATER.get(this), src,
new DelegatingCompletionHandler<>(future, Operation.WRITE));
return future;
}
@Override
public boolean isOpen() {
return fileChannel.isOpen();
}
@Override
public void close() throws IOException {
fileChannel.close();
}
private void beginOperation(Operation operation) {
if (!PENDING_OPERATION_ATOMIC_UPDATER.compareAndSet(this, null, operation)) {
switch (PENDING_OPERATION_ATOMIC_UPDATER.get(this)) {
case READ:
throw LOGGER.logExceptionAsError(new ReadPendingException());
case WRITE:
throw LOGGER.logExceptionAsError(new WritePendingException());
default:
throw LOGGER.logExceptionAsError(new IllegalStateException("Unknown channel operation"));
}
}
}
private void endOperation(Operation operation) {
if (!PENDING_OPERATION_ATOMIC_UPDATER.compareAndSet(this, operation, null)) {
throw LOGGER.logExceptionAsError(new IllegalStateException("There's no pending " + operation));
}
}
private enum Operation {
READ, WRITE
}
private final class DelegatingCompletionHandler implements CompletionHandler {
private final CompletionHandler handler;
private final CompletableFuture future;
private final Operation operation;
private DelegatingCompletionHandler(CompletionHandler handler, Operation operation) {
this.handler = handler;
this.future = null;
this.operation = operation;
}
private DelegatingCompletionHandler(CompletableFuture future, Operation operation) {
this.handler = null;
this.future = future;
this.operation = operation;
}
@Override
public void completed(Integer result, T attachment) {
if (result > 0) {
POSITION_ATOMIC_UPDATER.addAndGet(AsynchronousFileChannelAdapter.this, result);
}
endOperation(this.operation);
if (handler != null) {
handler.completed(result, attachment);
} else if (future != null) {
future.complete(result);
}
}
@Override
public void failed(Throwable exc, T attachment) {
endOperation(this.operation);
if (handler != null) {
handler.failed(exc, attachment);
} else if (future != null) {
future.completeExceptionally(exc);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy