FilterFileChannel.java

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      https://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.apache.commons.io.channels;

import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.MappedByteBuffer;
import java.nio.channels.Channel;
import java.nio.channels.FileChannel;
import java.nio.channels.FileLock;
import java.nio.channels.ReadableByteChannel;
import java.nio.channels.WritableByteChannel;
import java.util.Objects;

import org.apache.commons.io.build.AbstractStreamBuilder;

/**
 * Filters a {@link FileChannel}.
 * <p>
 * A {@code FilterFileChannel} wraps some other channel, which it uses as its basic source of data, possibly transforming the data along the way or providing
 * additional functionality. The class {@code FilterFileChannel} itself simply overrides methods of {@code FileChannel} with versions that pass all requests to
 * the wrapped channel. Subclasses of {@code FilterFileChannel} may of course override any methods declared or inherited by {@code FilterFileChannel}, and may
 * also provide additional fields and methods.
 * </p>
 * <p>
 * You construct s simple instance with the {@link FilterFileChannel#FilterFileChannel(FileChannel) channel constructor} and more advanced instances through the
 * {@link Builder}.
 * </p>
 *
 * @since 2.22.0
 */
public class FilterFileChannel extends FileChannel {

    /**
     * Builds instances of {@link FilterFileChannel} for subclasses.
     *
     * @param <F> The {@link FilterFileChannel} type.
     * @param <C> The {@link Channel} type wrapped by the FilterChannel.
     * @param <B> The builder type.
     */
    public abstract static class AbstractBuilder<F extends FilterFileChannel, C extends FileChannel, B extends AbstractBuilder<F, C, B>>
            extends AbstractStreamBuilder<F, AbstractBuilder<F, C, B>> {

        /**
         * Constructs instance for subclasses.
         */
        protected AbstractBuilder() {
            // empty
        }
    }

    /**
     * Builds instances of {@link FilterFileChannel}.
     */
    public static class Builder extends AbstractBuilder<FilterFileChannel, FileChannel, Builder> {

        /**
         * Builds instances of {@link FilterChannel}.
         */
        protected Builder() {
            // empty
        }

        @Override
        public FilterFileChannel get() throws IOException {
            return new FilterFileChannel(this);
        }
    }

    /**
     * Creates a new {@link Builder}.
     *
     * @return a new {@link Builder}.
     */
    public static Builder forFilterFileChannel() {
        return new Builder();
    }

    final FileChannel fileChannel;

    private FilterFileChannel(final Builder builder) throws IOException {
        this.fileChannel = builder.getChannel(FileChannel.class);
    }

    /**
     * Constructs a new instance.
     *
     * @param fileChannel the file channel to wrap.
     */
    public FilterFileChannel(final FileChannel fileChannel) {
        this.fileChannel = Objects.requireNonNull(fileChannel, "fileChannel");
    }

    @Override
    public boolean equals(final Object o) {
        return fileChannel.equals(o);
    }

    @Override
    public void force(final boolean metaData) throws IOException {
        fileChannel.force(metaData);
    }

    @Override
    public int hashCode() {
        return fileChannel.hashCode();
    }

    @Override
    protected void implCloseChannel() throws IOException {
        fileChannel.close();
    }

    @Override
    public FileLock lock(final long position, final long size, final boolean shared) throws IOException {
        return fileChannel.lock(position, size, shared);
    }

    @Override
    public MappedByteBuffer map(final MapMode mode, final long position, final long size) throws IOException {
        return fileChannel.map(mode, position, size);
    }

    @Override
    public long position() throws IOException {
        return fileChannel.position();
    }

    @Override
    public FileChannel position(final long newPosition) throws IOException {
        return fileChannel.position(newPosition);
    }

    @Override
    public int read(final ByteBuffer dst) throws IOException {
        return fileChannel.read(dst);
    }

    @Override
    public int read(final ByteBuffer dst, final long position) throws IOException {
        return fileChannel.read(dst, position);
    }

    @Override
    public long read(final ByteBuffer[] dsts, final int offset, final int length) throws IOException {
        return fileChannel.read(dsts, offset, length);
    }

    @Override
    public long size() throws IOException {
        return fileChannel.size();
    }

    @Override
    public String toString() {
        return fileChannel.toString();
    }

    @Override
    public long transferFrom(final ReadableByteChannel src, final long position, final long count) throws IOException {
        return fileChannel.transferFrom(src, position, count);
    }

    @Override
    public long transferTo(final long position, final long count, final WritableByteChannel target) throws IOException {
        return fileChannel.transferTo(position, count, target);
    }

    @Override
    public FileChannel truncate(final long size) throws IOException {
        return fileChannel.truncate(size);
    }

    @Override
    public FileLock tryLock(final long position, final long size, final boolean shared) throws IOException {
        return fileChannel.tryLock(position, size, shared);
    }

    /**
     * Unwraps this instance by returning the underlying {@link FileChannel}.
     * <p>
     * Use with caution.
     * </p>
     *
     * @return the underlying {@link FileChannel}.
     */
    public FileChannel unwrap() {
        return fileChannel;
    }

    @Override
    public int write(final ByteBuffer src) throws IOException {
        return fileChannel.write(src);
    }

    @Override
    public int write(final ByteBuffer src, final long position) throws IOException {
        return fileChannel.write(src, position);
    }

    @Override
    public long write(final ByteBuffer[] srcs, final int offset, final int length) throws IOException {
        return fileChannel.write(srcs, offset, length);
    }
}