/***** BEGIN LICENSE BLOCK *****
 * Version: EPL 2.0/GPL 2.0/LGPL 2.1
 *
 * The contents of this file are subject to the Eclipse Public
 * 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.eclipse.org/legal/epl-v20.html
 *
 * Software distributed under the License is distributed on an "AS
 * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
 * implied. See the License for the specific language governing
 * rights and limitations under the License.
 *
 * Copyright (C) 2006 Ola Bini <ola@ologix.com>
 *
 * Alternatively, the contents of this file may be used under the terms of
 * either of the GNU General Public License Version 2 or later (the "GPL"),
 * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
 * in which case the provisions of the GPL or the LGPL are applicable instead
 * of those above. If you wish to allow use of your version of this file only
 * under the terms of either the GPL or the LGPL, and not to allow others to
 * use your version of this file under the terms of the EPL, indicate your
 * decision by deleting the provisions above and replace them with the notice
 * and other provisions required by the GPL or the LGPL. If you do not delete
 * the provisions above, a recipient may use your version of this file under
 * the terms of any one of the EPL, the GPL or the LGPL.
 ***** END LICENSE BLOCK *****/
$Id$
/** * $Id$ */
package org.jruby.util; import java.io.File; import java.io.FileFilter; import java.io.FileNotFoundException; import java.io.FilenameFilter; import java.io.IOException; import java.util.regex.Pattern; import jnr.posix.JavaSecuredFile; import org.jruby.Ruby; import org.jruby.platform.Platform; import org.jruby.runtime.ThreadContext;

This file acts as an alternative to NormalizedFile, due to the problems with current working directory.

/** * <p> * This file acts as an alternative to NormalizedFile, due to the problems with * current working directory. * </p> */
public class JRubyFile extends JavaSecuredFile { private static final long serialVersionUID = 435364547567567L; public static JRubyFile create(String cwd, String pathname) { if (pathname == null || pathname.length() == 0 || Ruby.isSecurityRestricted()) { return JRubyFile.DUMMY; } if (pathname.startsWith("file:")) { pathname = pathname.substring(5); } return createNoUnicodeConversion(cwd, pathname, new File(pathname)); } public static FileResource createResource(ThreadContext context, String pathname) { return createResource(context.runtime, pathname); } public static FileResource createResourceAsFile(Ruby runtime, String pathname) { return createResource(runtime, runtime.getCurrentDirectory(), pathname, true); } public static FileResource createRestrictedResource(String cwd, String pathname) { return createResource(null, cwd, pathname); } public static FileResource createResource(Ruby runtime, String pathname) { return createResource(runtime, runtime.getCurrentDirectory(), pathname, false); } public static FileResource createResource(Ruby runtime, String cwd, String pathname) { return createResource(runtime, cwd, pathname, false); } private static FileResource createResource(Ruby runtime, String cwd, String pathname, boolean isFile) { FileResource resource = EmptyFileResource.create(pathname); if (resource != null) return resource; // This will work against anything potentially containing a '!' in it and does not require a scheme. // (see test/test_java_on_load_path.rb: $LOAD_PATH << "test/test_jruby_1332.jar!"; require 'test_jruby_1332.rb' resource = JarResource.create(pathname); if (resource != null) return resource; if (Platform.IS_WINDOWS && (pathname.equalsIgnoreCase("nul") || pathname.equalsIgnoreCase("nul:"))) { return new NullDeviceResource(); } if (pathname.indexOf(':') > 0) { // scheme-oriented resources if (pathname.startsWith("classpath:")) { pathname = "uri:classloader:/" + pathname.substring(10); } // replace is needed for maven/jruby-complete/src/it/app_using_classpath_uri to work if (pathname.startsWith("uri:")) return URLResource.create(runtime, pathname, isFile); if (pathname.startsWith("file:")) { pathname = pathname.substring(5); if (pathname.length() == 0) return EmptyFileResource.INSTANCE; } } if (cwd != null && (cwd.startsWith("uri:") || cwd.startsWith("file:")) && !new File(pathname).isAbsolute()) { return createResource(runtime, null, cwd + '/' + pathname); } // If any other special resource types fail, count it as a filesystem backed resource. return new RegularFileResource(runtime != null ? runtime.getPosix() : null, create(cwd, pathname), pathname); } public static String normalizeSeps(String path) { return Platform.IS_WINDOWS ? path.replace(File.separatorChar, '/') : path; } private static JRubyFile createNoUnicodeConversion(String cwd, String pathname, File path) { if (path.isAbsolute()) { return new JRubyFile(path); } else if (Platform.IS_WINDOWS) { // File and company do not seem to recognize COM ports on Windows as absolute. Cheat! if (JRubyFile.isComPort(pathname)) { return new JRubyFile(pathname); // use raw path, not absolute path // Nor do they seem to recognize bare \ and / on Windows as absolute. Cheat! } else if (pathname.startsWith("/") || pathname.startsWith("\\")) { return new JRubyFile(path); } } if (cwd != null && cwd.startsWith("uri:") && !pathname.startsWith("uri:") && !pathname.contains("!/")) { return new JRubyFile(cwd + '/' + pathname); } path = new File(cwd, pathname); if (!path.isAbsolute()) { throw new IllegalArgumentException("Neither current working directory ("+cwd+") nor pathname ("+pathname+") led to an absolute path"); } return new JRubyFile(path); } public static String getFileProperty(String property) { return normalizeSeps(SafePropertyAccessor.getProperty(property, "/")); } private static final Pattern windowsComPattern = Pattern.compile("(?:\\/\\/\\.\\/|\\\\\\\\\\.\\\\)?COM\\d\\d?", Pattern.CASE_INSENSITIVE); // Checks to see if it's a windows com port path static boolean isComPort(String path) { if (!Platform.IS_WINDOWS) return false; int len = path.length(); // Look for both \\.\ and //./, but avoid COMxx (len != 5) as that isn't valid return len < 10 && len > 3 && len != 5 && windowsComPattern.matcher(path).matches(); } private JRubyFile(File file) { this(file.getAbsolutePath()); } public JRubyFile(String filename) { super(filename); } public JRubyFile(String parent, String child) { super(parent, child); } @Override public String getAbsolutePath() { final String path = super.getPath(); if (path.startsWith("uri:")) { // TODO better do not collapse // to / for uri: files return normalizeSeps(path.replaceFirst(":/([^/])", "://$1" )); } return normalizeSeps(new File(path).getAbsolutePath()); } @Override public String getCanonicalPath() throws IOException { try { return normalizeSeps(super.getCanonicalPath()); } catch (IOException e) { // usually IOExceptions don't tell us anything about the path, // so add an extra wrapper to give more debugging help. throw new IOException("Unable to canonicalize path: " + getAbsolutePath(), e); } } @Override public String getPath() { return normalizeSeps(super.getPath()); } final String getPathDefault() { return super.getPath(); } @Override public String toString() { return normalizeSeps(super.toString()); } @Override public File getAbsoluteFile() { return new JRubyFile(getAbsolutePath()); } @Override public File getCanonicalFile() throws IOException { return new JRubyFile(getCanonicalPath()); } @Override public String getParent() { String parent = super.getParent(); if (parent != null) { parent = normalizeSeps(parent); } return parent; } @Override public File getParentFile() { String parent = getParent(); if (parent == null) return this; return new JRubyFile(parent); } public static File[] listRoots() { File[] roots = File.listRoots(); JRubyFile[] smartRoots = new JRubyFile[roots.length]; for(int i = 0, j = roots.length; i < j; i++) { smartRoots[i] = new JRubyFile(roots[i].getPath()); } return smartRoots; } public static File createTempFile(String prefix, String suffix, File directory) throws IOException { return new JRubyFile(File.createTempFile(prefix, suffix,directory)); } public static File createTempFile(String prefix, String suffix) throws IOException { return new JRubyFile(File.createTempFile(prefix, suffix)); } @Override public String[] list(FilenameFilter filter) { String[] files = super.list(filter); if (files == null) { return null; } String[] smartFiles = new String[files.length]; for (int i = 0; i < files.length; i++) { smartFiles[i] = normalizeSeps(files[i]); } return smartFiles; } @Override public File[] listFiles() { return convertFiles(super.listFiles()); } @Override public File[] listFiles(final FileFilter filter) { return convertFiles(super.listFiles(filter)); } @Override public File[] listFiles(final FilenameFilter filter) { return convertFiles(super.listFiles(filter)); } private JRubyFile[] convertFiles(final File[] files) { if (files == null) return null; // non-existent directory final String absolutePath = super.getAbsolutePath(); JRubyFile[] smartFiles = new JRubyFile[files.length]; for (int i = 0; i < files.length; i++) { final File file = files[i]; smartFiles[i] = createNoUnicodeConversion(absolutePath, file.getPath(), file); } return smartFiles; } public static JRubyFile DUMMY = new JRubyFile("") { @Override public String getAbsolutePath() { return ""; } @Override public boolean isDirectory() { return false; } @Override public boolean exists() { return false; } @Override public String getCanonicalPath() throws IOException { throw new FileNotFoundException("File does not exist"); } @Override public String getPath() { return ""; } @Override public String toString() { return ""; } @Override public File getAbsoluteFile() { return this; } @Override public File getCanonicalFile() throws IOException { throw new FileNotFoundException("File does not exist"); } @Override public String getParent() { return ""; } @Override public File getParentFile() { return this; } @Override public String[] list(FilenameFilter filter) { return new String[0]; } @Override public File[] listFiles() { return new File[0]; } @Override public File[] listFiles(final FileFilter filter) { return new File[0]; } @Override public File[] listFiles(final FilenameFilter filter) { return new File[0]; } }; }