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

software.amazon.nio.spi.s3.S3SeekableByteChannel Maven / Gradle / Ivy

Go to download

A Java NIO.2 service provider for S3, allowing Java NIO operations to be performed on paths using the `s3` scheme. This package implements the service provider interface (SPI) defined for Java NIO.2 in JDK 1.7 providing "plug-in" non-blocking access to S3 objects for Java applications using Java NIO.2 for file access.

There is a newer version: 2.2.0
Show newest version
/*
 * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
 * SPDX-License-Identifier: Apache-2.0
 */

package software.amazon.nio.spi.s3;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import software.amazon.awssdk.services.s3.S3AsyncClient;
import software.amazon.awssdk.services.s3.model.HeadObjectResponse;
import software.amazon.nio.spi.s3.config.S3NioSpiConfiguration;
import software.amazon.nio.spi.s3.util.TimeOutUtils;

import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.*;
import java.nio.file.OpenOption;
import java.nio.file.StandardOpenOption;
import java.util.Collections;
import java.util.Set;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

public class S3SeekableByteChannel implements SeekableByteChannel {

    private static final Logger LOGGER = LoggerFactory.getLogger(S3SeekableByteChannel.class);

    private long position;
    private final S3AsyncClient s3Client;
    private final S3Path path;
    private final ReadableByteChannel readDelegate;
    private final S3WritableByteChannel writeDelegate;

    private boolean closed;
    private long size = -1L;

    protected S3SeekableByteChannel(S3Path s3Path, S3AsyncClient s3Client) throws IOException {
        this(s3Path, s3Client, Collections.singleton(StandardOpenOption.READ));
    }

    /**
     * @deprecated startAt is only a valid parameter for the read mode and is
     * therefore discouraged to be used during creation of the channel
     */
    @Deprecated
    protected S3SeekableByteChannel(S3Path s3Path, S3AsyncClient s3Client, long startAt) throws IOException {
        this(s3Path, s3Client, startAt, Collections.singleton(StandardOpenOption.READ), null, null);
    }

    /**
     * @deprecated startAt is only a valid parameter for the read mode and is
     * therefore discouraged to be used during creation of the channel
     */
    @Deprecated
    protected S3SeekableByteChannel(S3Path s3Path, long startAt) throws IOException {
        this(s3Path, S3ClientStore.getInstance().getAsyncClientForBucketName(s3Path.bucketName()), startAt);
    }

    protected S3SeekableByteChannel(S3Path s3Path) throws IOException {
        this(s3Path, S3ClientStore.getInstance().getAsyncClientForBucketName(s3Path.bucketName()), Collections.singleton(StandardOpenOption.READ));
    }

    protected S3SeekableByteChannel(S3Path s3Path, S3AsyncClient s3Client, Set options) throws IOException {
        this(s3Path, s3Client, 0L, options, null, null);
    }

    protected S3SeekableByteChannel(S3Path s3Path, S3AsyncClient s3Client, Set options, long timeout, TimeUnit timeUnit) throws IOException {
        this(s3Path, s3Client, 0L, options, timeout, timeUnit);
    }

    private S3SeekableByteChannel(S3Path s3Path, S3AsyncClient s3Client, long startAt, Set options, Long timeout, TimeUnit timeUnit) throws IOException {
        position = startAt;
        path = s3Path;
        closed = false;
        this.s3Client = s3Client;
        s3Path.getFileSystem().registerOpenChannel(this);

        if (options.contains(StandardOpenOption.WRITE) && options.contains(StandardOpenOption.READ)) {
            throw new IOException("This channel does not support read and write access simultaneously");
        }
        if (options.contains(StandardOpenOption.SYNC) || options.contains(StandardOpenOption.DSYNC)) {
            throw new IOException("The SYNC/DSYNC options is not supported");
        }

        // later we will add a constructor that allows providing delegates for composition

        S3NioSpiConfiguration config = new S3NioSpiConfiguration();
        if (options.contains(StandardOpenOption.WRITE)) {
            LOGGER.info("using S3WritableByteChannel as write delegate for path '{}'", s3Path.toUri());
            readDelegate = null;
            writeDelegate = new S3WritableByteChannel(s3Path, s3Client, options, timeout, timeUnit);
            position = 0L;
        } else if (options.contains(StandardOpenOption.READ) || options.size() == 0) {
            LOGGER.info("using S3ReadAheadByteChannel as read delegate for path '{}'", s3Path.toUri());
            readDelegate = new S3ReadAheadByteChannel(s3Path, config.getMaxFragmentSize(), config.getMaxFragmentNumber(), s3Client, this, timeout, timeUnit);
            writeDelegate = null;
        } else {
            throw new IOException("Invalid channel mode");
        }
    }

    /**
     * Reads a sequence of bytes from this channel into the given buffer.
     *
     * 

Bytes are read starting at this channel's current position, and * then the position is updated with the number of bytes actually read. * Otherwise, this method behaves exactly as specified in the {@link * ReadableByteChannel} interface. * * @param dst the destination buffer * @return the number of bytes read or -1 if no more bytes can be read. */ @Override public int read(ByteBuffer dst) throws IOException { validateOpen(); if (readDelegate == null) { throw new NonReadableChannelException(); } return readDelegate.read(dst); } /** * Writes a sequence of bytes to this channel from the given buffer. * *

Bytes are written starting at this channel's current position, unless * the channel is connected to an entity such as a file that is opened with * the {@link StandardOpenOption#APPEND APPEND} option, in * which case the position is first advanced to the end. The entity to which * the channel is connected is grown, if necessary, to accommodate the * written bytes, and then the position is updated with the number of bytes * actually written. Otherwise, this method behaves exactly as specified by * the {@link WritableByteChannel} interface. * * @param src the src of the bytes to write to this channel */ @Override public int write(ByteBuffer src) throws IOException { validateOpen(); if (writeDelegate == null) { throw new NonWritableChannelException(); } int length = src.remaining(); this.position += length; return writeDelegate.write(src); } /** * Returns this channel's position. * * @return This channel's position, * a non-negative integer counting the number of bytes * from the beginning of the entity to the current position */ @Override public long position() throws IOException { validateOpen(); synchronized (this) { return position; } } /** * Sets this channel's position. * *

Setting the position to a value that is greater than the current size * is legal but does not change the size of the entity. A later attempt to * read bytes at such a position will immediately return an end-of-file * indication. A later attempt to write bytes at such a position will cause * the entity to grow to accommodate the new bytes; the values of any bytes * between the previous end-of-file and the newly-written bytes are * unspecified. * *

Setting the channel's position is not recommended when connected to * an entity, typically a file, that is opened with the {@link * StandardOpenOption#APPEND APPEND} option. When opened for * append, the position is first advanced to the end before writing. * * @param newPosition The new position, a non-negative integer counting * the number of bytes from the beginning of the entity * @return This channel * @throws ClosedChannelException If this channel is closed * @throws IllegalArgumentException If the new position is negative * @throws IOException If some other I/O error occurs */ @Override public SeekableByteChannel position(long newPosition) throws IOException { if (newPosition < 0) throw new IllegalArgumentException("newPosition cannot be < 0"); if (!isOpen()) { throw new ClosedChannelException(); } // this is only valid to read channels if (readDelegate == null) { throw new NonReadableChannelException(); } synchronized (this) { position = newPosition; return this; } } /** * Returns the current size of entity to which this channel is connected. * * @return The current size, measured in bytes * @throws IOException If some other I/O error occurs */ @Override public long size() throws IOException { validateOpen(); if (size < 0) { long timeOut = TimeOutUtils.TIMEOUT_TIME_LENGTH_1; TimeUnit unit = TimeUnit.MINUTES; LOGGER.debug("requesting size of '{}'", path.toUri()); synchronized (this) { final HeadObjectResponse headObjectResponse; try { headObjectResponse = s3Client.headObject(builder -> builder .bucket(path.bucketName()) .key(path.getKey())).get(timeOut, unit); } catch (InterruptedException e) { Thread.currentThread().interrupt(); throw new RuntimeException(e); } catch (ExecutionException e) { throw new IOException(e); } catch (TimeoutException e) { throw TimeOutUtils.logAndGenerateExceptionOnTimeOut(LOGGER, "size", timeOut, unit); } LOGGER.debug("size of '{}' is '{}'", path.toUri(), headObjectResponse.contentLength()); this.size = headObjectResponse.contentLength(); } } return this.size; } /** * Truncates the entity, to which this channel is connected, to the given * size. * *

If the given size is less than the current size then the entity is * truncated, discarding any bytes beyond the new end. If the given size is * greater than or equal to the current size then the entity is not modified. * In either case, if the current position is greater than the given size * then it is set to that size. * *

An implementation of this interface may prohibit truncation when * connected to an entity, typically a file, opened with the {@link * StandardOpenOption#APPEND APPEND} option. * * @param size The new size, a non-negative byte count * @return This channel */ @Override public SeekableByteChannel truncate(long size) { throw new UnsupportedOperationException("Currently not supported"); } /** * Tells whether this channel is open. * * @return {@code true} if, and only if, this channels delegate is open */ @Override public boolean isOpen() { synchronized (this) { return !this.closed; } } /** * Closes this channel. * *

After a channel is closed, any further attempt to invoke I/O * operations upon it will cause a {@link ClosedChannelException} to be * thrown. * *

If this channel is already closed then invoking this method has no * effect. * *

This method may be invoked at any time. If some other thread has * already invoked it, however, then another invocation will block until * the first invocation is complete, after which it will return without * effect.

*/ @Override public void close() throws IOException { synchronized (this) { if (readDelegate != null) { readDelegate.close(); } if (writeDelegate != null) { writeDelegate.close(); } closed = true; path.getFileSystem().deregisterClosedChannel(this); } } private void validateOpen() throws ClosedChannelException { if (this.closed) { throw new ClosedChannelException(); } } /** * Access the underlying {@code ReadableByteChannel} used for reading * @return the channel. May be null if opened for writing only */ protected ReadableByteChannel getReadDelegate() { return this.readDelegate; } /** * Access the underlying {@code WritableByteChannel} used for writing * @return the channel. May be null if opened for reading only */ protected WritableByteChannel getWriteDelegate() { return this.writeDelegate; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy