/* * Copyright (c) 2010, 2015, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.Reader; import java.io.Writer; import java.net.URI; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.Objects; import java.util.regex.Pattern; import javax.lang.model.element.Modifier; import javax.lang.model.element.NestingKind; import javax.tools.JavaFileManager.Location; import javax.tools.JavaFileObject; import javax.tools.StandardJavaFileManager; import com.sun.tools.javac.api.WrappingJavaFileManager; /** * A JavaFileManager that can throw IOException on attempting to read or write * selected files that match a regular expression. */ public class FileManager extends WrappingJavaFileManager implements StandardJavaFileManager { private static final String CANT_READ = "cantRead:"; private static final String CANT_WRITE = "cantWrite:"; private Pattern cantRead; private Pattern cantWrite; public FileManager(StandardJavaFileManager fm, List opts) { super(fm); for (String opt: opts) { if (opt.startsWith(CANT_READ)) cantRead = Pattern.compile(opt.substring(CANT_READ.length())); else if (opt.startsWith(CANT_WRITE)) cantWrite = Pattern.compile(opt.substring(CANT_WRITE.length())); else throw new IllegalArgumentException(opt); } } @Override protected JavaFileObject wrap(JavaFileObject fo) { return (fo == null) ? null : new WrappedFileObject(fo); } @Override protected JavaFileObject unwrap(JavaFileObject fo) { if (fo instanceof WrappedFileObject) return ((WrappedFileObject) fo).delegate; else return fo; } public Iterable getJavaFileObjectsFromFiles(Iterable files) { return wrap2(fileManager.getJavaFileObjectsFromFiles(files)); } public Iterable getJavaFileObjects(File... files) { return wrap2(fileManager.getJavaFileObjects(files)); } public Iterable getJavaFileObjectsFromStrings(Iterable names) { return wrap2(fileManager.getJavaFileObjectsFromStrings(names)); } public Iterable getJavaFileObjects(String... names) { return wrap2(fileManager.getJavaFileObjects(names)); } /* This method is regrettably necessary because WrappingJavaFileManager.wrap takes * Iterable fileObjects * instead of * Iterable fileObjects */ protected Iterable wrap2(Iterable fileObjects) { List mapped = new ArrayList(); for (JavaFileObject fileObject : fileObjects) mapped.add(wrap(fileObject)); return Collections.unmodifiableList(mapped); } public void setLocation(Location location, Iterable path) throws IOException { fileManager.setLocation(location, path); } public Iterable getLocation(Location location) { return fileManager.getLocation(location); } class WrappedFileObject implements JavaFileObject { WrappedFileObject(JavaFileObject fileObject) { delegate = Objects.requireNonNull(fileObject); } public Kind getKind() { return delegate.getKind(); } public boolean isNameCompatible(String simpleName, Kind kind) { return delegate.isNameCompatible(simpleName, kind); } public NestingKind getNestingKind() { return delegate.getNestingKind(); } public Modifier getAccessLevel() { return delegate.getAccessLevel(); } public URI toUri() { return delegate.toUri(); } public String getName() { return delegate.getName(); } public InputStream openInputStream() throws IOException { checkRead(); return delegate.openInputStream(); } public OutputStream openOutputStream() throws IOException { checkWrite(); return delegate.openOutputStream(); } public Reader openReader(boolean ignoreEncodingErrors) throws IOException { checkRead(); return delegate.openReader(ignoreEncodingErrors); } public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException { checkRead(); return delegate.getCharContent(ignoreEncodingErrors); } public Writer openWriter() throws IOException { checkWrite(); return delegate.openWriter(); } public long getLastModified() { return delegate.getLastModified(); } public boolean delete() { return delegate.delete(); } void checkRead() throws IOException { String canonName = getName().replace(File.separatorChar, '/'); if (cantRead != null && cantRead.matcher(canonName).matches()) throw new IOException("FileManager: Can't read"); } void checkWrite() throws IOException { String canonName = getName().replace(File.separatorChar, '/'); if (cantWrite != null && cantWrite.matcher(canonName).matches()) throw new IOException("FileManager: Can't write"); } JavaFileObject delegate; } }