/* * JBoss, Home of Professional Open Source * Copyright 2012, Red Hat Middleware LLC, and individual contributors * by the @authors tag. See the copyright.txt in the distribution for a * full listing of individual contributors. * * Licensed 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.jboss.shrinkwrap.impl.nio.file; import java.io.IOException; import java.nio.file.DirectoryStream; import java.nio.file.Path; import java.util.ArrayList; import java.util.Collection; import java.util.Iterator; import java.util.Map; import org.jboss.shrinkwrap.api.ArchivePath; import org.jboss.shrinkwrap.api.Node; /** * ShrinkWrap implementation of a {@link DirectoryStream} * * @author <a href="mailto:alr@jboss.org">Andrew Lee Rubinger</a> */ class ShrinkWrapDirectoryStream implements DirectoryStream<Path> { private final ShrinkWrapFileSystem fs; private final DirectoryStream.Filter<? super Path> filter; private final Path startingPath; private boolean closed = false; private boolean iteratorReturned = false; /** * Creates a new instance starting from startingPath with is required backing the specified * {@link ShrinkWrapFileSystem}, which is required. An optional {@link DirectoryStream.Filter} may be * specified as well. * * @param startingPath * @param fs * @param filter * @throws IllegalArgumentException * If the fs is not specified */ ShrinkWrapDirectoryStream(final Path startingPath, final ShrinkWrapFileSystem fs, final DirectoryStream.Filter<? super Path> filter) throws IllegalArgumentException { if (fs == null) { throw new IllegalArgumentException("File system must be specified"); } if (startingPath == null) { throw new IllegalArgumentException("Starting path must be specified"); } this.startingPath = startingPath.toAbsolutePath(); this.fs = fs; this.filter = filter; } /** * {@inheritDoc} * * @see java.io.Closeable#close() */ @Override public void close() throws IOException { this.closed = true; } /** * {@inheritDoc} * * @see java.nio.file.DirectoryStream#iterator() */ @Override public Iterator<Path> iterator() { if (closed) { throw new IllegalStateException("Directory Stream is closed"); } else if (iteratorReturned) { throw new IllegalStateException("Iterator was already returned"); } boolean finishedSuccessfully = true; try { // Translate ShrinkWrap API to NIO.2 API Path final Map<ArchivePath, Node> content = this.fs.getArchive().getContent(); final Collection<Path> newPaths = new ArrayList<>(content.size()); final Collection<ArchivePath> archivePaths = content.keySet(); for (final ArchivePath path : archivePaths) { final Path newPath = new ShrinkWrapPath(path, fs); if (!newPath.getParent().equals(startingPath)) { continue; } // If we have a filter, and it rejects this path try { if (filter != null && !(filter.accept(newPath))) { // Move along continue; } } catch (IOException ioe) { throw new RuntimeException("Error encountered during filtering", ioe); } // Add the new Path; the filter either wasn't specified or didn't reject this Path newPaths.add(newPath); } // Return return newPaths.iterator(); } catch (Throwable t) { finishedSuccessfully = false; throw t; } finally { if (finishedSuccessfully) { iteratorReturned = true; } } } }