blob: 8a7ff754e014e6bb2b3ac73f6d01fb7fca426c8e [file] [log] [blame]
/*
* 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
*
* http://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.lucene.mockfile;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URI;
import java.nio.channels.AsynchronousFileChannel;
import java.nio.channels.FileChannel;
import java.nio.channels.SeekableByteChannel;
import java.nio.file.AccessMode;
import java.nio.file.CopyOption;
import java.nio.file.DirectoryStream;
import java.nio.file.DirectoryStream.Filter;
import java.nio.file.FileStore;
import java.nio.file.FileSystem;
import java.nio.file.LinkOption;
import java.nio.file.OpenOption;
import java.nio.file.Path;
import java.nio.file.ProviderMismatchException;
import java.nio.file.attribute.BasicFileAttributes;
import java.nio.file.attribute.FileAttribute;
import java.nio.file.attribute.FileAttributeView;
import java.nio.file.spi.FileSystemProvider;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.ExecutorService;
/**
* A {@code FilterFileSystemProvider} contains another
* {@code FileSystemProvider}, which it uses as its basic
* source of data, possibly transforming the data along the
* way or providing additional functionality.
*/
public abstract class FilterFileSystemProvider extends FileSystemProvider {
/**
* The underlying {@code FileSystemProvider}.
*/
protected final FileSystemProvider delegate;
/**
* The underlying {@code FileSystem} instance.
*/
protected FileSystem fileSystem;
/**
* The URI scheme for this provider.
*/
protected final String scheme;
/**
* Construct a {@code FilterFileSystemProvider} indicated by
* the specified {@code scheme} and wrapping functionality of the
* provider of the specified base filesystem.
* @param scheme URI scheme
* @param delegateInstance specified base filesystem.
*/
public FilterFileSystemProvider(String scheme, FileSystem delegateInstance) {
this.scheme = Objects.requireNonNull(scheme);
Objects.requireNonNull(delegateInstance);
this.delegate = delegateInstance.provider();
this.fileSystem = new FilterFileSystem(this, delegateInstance);
}
/**
* Construct a {@code FilterFileSystemProvider} indicated by
* the specified {@code scheme} and wrapping functionality of the
* provider. You must set the singleton {@code filesystem} yourself.
* @param scheme URI scheme
* @param delegate specified base provider.
*/
public FilterFileSystemProvider(String scheme, FileSystemProvider delegate) {
this.scheme = Objects.requireNonNull(scheme);
this.delegate = Objects.requireNonNull(delegate);
}
@Override
public String getScheme() {
return scheme;
}
@Override
public FileSystem newFileSystem(URI uri, Map<String,?> env) throws IOException {
if (fileSystem == null) {
throw new IllegalStateException("subclass did not initialize singleton filesystem");
}
return fileSystem;
}
@Override
public FileSystem newFileSystem(Path path, Map<String,?> env) throws IOException {
if (fileSystem == null) {
throw new IllegalStateException("subclass did not initialize singleton filesystem");
}
return fileSystem;
}
@Override
public FileSystem getFileSystem(URI uri) {
if (fileSystem == null) {
throw new IllegalStateException("subclass did not initialize singleton filesystem");
}
return fileSystem;
}
@Override
public Path getPath(URI uri) {
if (fileSystem == null) {
throw new IllegalStateException("subclass did not initialize singleton filesystem");
}
Path path = delegate.getPath(uri);
return new FilterPath(path, fileSystem);
}
@Override
public void createDirectory(Path dir, FileAttribute<?>... attrs) throws IOException {
delegate.createDirectory(toDelegate(dir), attrs);
}
@Override
public void delete(Path path) throws IOException {
delegate.delete(toDelegate(path));
}
@Override
public void copy(Path source, Path target, CopyOption... options) throws IOException {
delegate.copy(toDelegate(source), toDelegate(target), options);
}
@Override
public void move(Path source, Path target, CopyOption... options) throws IOException {
delegate.move(toDelegate(source), toDelegate(target), options);
}
@Override
public boolean isSameFile(Path path, Path path2) throws IOException {
return delegate.isSameFile(toDelegate(path), toDelegate(path2));
}
@Override
public boolean isHidden(Path path) throws IOException {
return delegate.isHidden(toDelegate(path));
}
@Override
public FileStore getFileStore(Path path) throws IOException {
return delegate.getFileStore(toDelegate(path));
}
@Override
public void checkAccess(Path path, AccessMode... modes) throws IOException {
delegate.checkAccess(toDelegate(path), modes);
}
@Override
public <V extends FileAttributeView> V getFileAttributeView(Path path, Class<V> type, LinkOption... options) {
return delegate.getFileAttributeView(toDelegate(path), type, options);
}
@Override
public <A extends BasicFileAttributes> A readAttributes(Path path, Class<A> type, LinkOption... options) throws IOException {
return delegate.readAttributes(toDelegate(path), type, options);
}
@Override
public Map<String,Object> readAttributes(Path path, String attributes, LinkOption... options) throws IOException {
return delegate.readAttributes(toDelegate(path), attributes, options);
}
@Override
public void setAttribute(Path path, String attribute, Object value, LinkOption... options) throws IOException {
delegate.setAttribute(toDelegate(path), attribute, value, options);
}
@Override
public InputStream newInputStream(Path path, OpenOption... options) throws IOException {
return delegate.newInputStream(toDelegate(path), options);
}
@Override
public OutputStream newOutputStream(Path path, OpenOption... options) throws IOException {
return delegate.newOutputStream(toDelegate(path), options);
}
@Override
public FileChannel newFileChannel(Path path, Set<? extends OpenOption> options, FileAttribute<?>... attrs) throws IOException {
return delegate.newFileChannel(toDelegate(path), options, attrs);
}
@Override
public AsynchronousFileChannel newAsynchronousFileChannel(Path path, Set<? extends OpenOption> options, ExecutorService executor, FileAttribute<?>... attrs) throws IOException {
return delegate.newAsynchronousFileChannel(toDelegate(path), options, executor, attrs);
}
@Override
public SeekableByteChannel newByteChannel(Path path, Set<? extends OpenOption> options, FileAttribute<?>... attrs) throws IOException {
return delegate.newByteChannel(toDelegate(path), options, attrs);
}
@Override
public DirectoryStream<Path> newDirectoryStream(Path dir, final Filter<? super Path> filter) throws IOException {
Filter<Path> wrappedFilter = new Filter<Path>() {
@Override
public boolean accept(Path entry) throws IOException {
return filter.accept(new FilterPath(entry, fileSystem));
}
};
return new FilterDirectoryStream(delegate.newDirectoryStream(toDelegate(dir), wrappedFilter), fileSystem);
}
@Override
public void createSymbolicLink(Path link, Path target, FileAttribute<?>... attrs) throws IOException {
delegate.createSymbolicLink(toDelegate(link), toDelegate(target), attrs);
}
@Override
public void createLink(Path link, Path existing) throws IOException {
delegate.createLink(toDelegate(link), toDelegate(existing));
}
@Override
public boolean deleteIfExists(Path path) throws IOException {
return delegate.deleteIfExists(toDelegate(path));
}
@Override
public Path readSymbolicLink(Path link) throws IOException {
return delegate.readSymbolicLink(toDelegate(link));
}
protected Path toDelegate(Path path) {
if (path instanceof FilterPath) {
FilterPath fp = (FilterPath) path;
if (fp.fileSystem != fileSystem) {
throw new ProviderMismatchException("mismatch, expected: " + fileSystem.provider().getClass() + ", got: " + fp.fileSystem.provider().getClass());
}
return fp.delegate;
} else {
throw new ProviderMismatchException("mismatch, expected: FilterPath, got: " + path.getClass());
}
}
/**
* Override to trigger some behavior when the filesystem is closed.
* <p>
* This is always called for each FilterFileSystemProvider in the chain.
*/
protected void onClose() {
}
@Override
public String toString() {
return getClass().getSimpleName() + "(" + delegate + ")";
}
}