blob: 738b0bf44dac51f69507a2eddf62a285558614ae [file] [log] [blame]
/* Copyright (c) 2001-2010, The HSQL Development Group
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* Neither the name of the HSQL Development Group nor the names of its
* contributors may be used to endorse or promote products derived from this
* software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL HSQL DEVELOPMENT GROUP, HSQLDB.ORG,
* OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.hsqldb.lib;
import java.io.File;
import java.io.FileDescriptor;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.Random;
import org.hsqldb.lib.java.JavaSystem;
/**
* A collection of file management methods.<p>
* Also provides the default FileAccess implementation
*
* @author Campbell Boucher-Burnett (boucherb@users dot sourceforge.net)
* @author Fred Toussi (fredt@users dot sourceforge.net)
* @author Ocke Janssen oj@openoffice.org
* @version 1.9.0
* @since 1.7.2
*/
public class FileUtil implements FileAccess {
private static FileUtil fileUtil = new FileUtil();
private static FileAccessRes fileAccessRes = new FileAccessRes();
/** Creates a new instance of FileUtil */
FileUtil() {}
public static FileUtil getFileUtil() {
return fileUtil;
}
public static FileAccess getFileAccess(boolean isResource) {
return isResource ? (FileAccess) fileAccessRes
: (FileAccess) fileUtil;
}
public boolean isStreamElement(java.lang.String elementName) {
return (new File(elementName)).exists();
}
public InputStream openInputStreamElement(java.lang.String streamName)
throws java.io.IOException {
try {
return new FileInputStream(new File(streamName));
} catch (Throwable e) {
throw JavaSystem.toIOException(e);
}
}
public void createParentDirs(String filename) {
makeParentDirectories(new File(filename));
}
public void removeElement(String filename) {
if (isStreamElement(filename)) {
delete(filename);
}
}
public void renameElement(java.lang.String oldName,
java.lang.String newName) {
renameWithOverwrite(oldName, newName);
}
public java.io.OutputStream openOutputStreamElement(
java.lang.String streamName) throws java.io.IOException {
return new FileOutputStream(new File(streamName));
}
// end of FileAccess implementation
// a new File("...")'s path is not canonicalized, only resolved
// and normalized (e.g. redundant separator chars removed),
// so as of JDK 1.4.2, this is a valid test for case insensitivity,
// at least when it is assumed that we are dealing with a configuration
// that only needs to consider the host platform's native file system,
// even if, unlike for File.getCanonicalPath(), (new File("a")).exists() or
// (new File("A")).exits(), regardless of the hosting system's
// file path case sensitivity policy.
public final boolean fsIsIgnoreCase =
(new File("A")).equals(new File("a"));
// posix separator normalized to File.separator?
// CHECKME: is this true for every file system under Java?
public final boolean fsNormalizesPosixSeparator =
(new File("/")).getPath().endsWith(File.separator);
// for JDK 1.1 createTempFile
final Random random = new Random(System.currentTimeMillis());
/**
* Delete the named file
*/
public boolean delete(String filename) {
return (new File(filename)).delete();
}
/**
* Requests, in a JDK 1.1 compliant way, that the file or directory denoted
* by the given abstract pathname be deleted when the virtual machine
* terminates. <p>
*
* Deletion will be attempted only for JDK 1.2 and greater runtime
* environments and only upon normal termination of the virtual
* machine, as defined by the Java Language Specification. <p>
*
* Once deletion has been sucessfully requested, it is not possible to
* cancel the request. This method should therefore be used with care. <p>
*
* @param f the abstract pathname of the file be deleted when the virtual
* machine terminates
*/
public void deleteOnExit(File f) {
JavaSystem.deleteOnExit(f);
}
/**
* Return true or false based on whether the named file exists.
*/
public boolean exists(String filename) {
return (new File(filename)).exists();
}
public boolean exists(String fileName, boolean resource, Class cla) {
if (fileName == null || fileName.length() == 0) {
return false;
}
return resource ? null != cla.getResource(fileName)
: FileUtil.getFileUtil().exists(fileName);
}
/**
* Rename the file with oldname to newname. If a file with newname already
* exists, it is deleted before the renaming operation proceeds.
*
* If a file with oldname does not exist, no file will exist after the
* operation.
*/
private boolean renameWithOverwrite(String oldname, String newname) {
File file = new File(oldname);
boolean renamed = file.renameTo(new File(newname));
if (renamed) {
return true;
}
if (delete(newname)) {
return file.renameTo(new File(newname));
}
return false;
}
/**
* Retrieves the absolute path, given some path specification.
*
* @param path the path for which to retrieve the absolute path
* @return the absolute path
*/
public String absolutePath(String path) {
return (new File(path)).getAbsolutePath();
}
/**
* Retrieves the canonical file for the given file, in a
* JDK 1.1 complaint way.
*
* @param f the File for which to retrieve the absolute File
* @return the canonical File
*/
public File canonicalFile(File f) throws IOException {
return new File(f.getCanonicalPath());
}
/**
* Retrieves the canonical file for the given path, in a
* JDK 1.1 complaint way.
*
* @param path the path for which to retrieve the canonical File
* @return the canonical File
*/
public File canonicalFile(String path) throws IOException {
return new File(new File(path).getCanonicalPath());
}
/**
* Retrieves the canonical path for the given File, in a
* JDK 1.1 complaint way.
*
* @param f the File for which to retrieve the canonical path
* @return the canonical path
*/
public String canonicalPath(File f) throws IOException {
return f.getCanonicalPath();
}
/**
* Retrieves the canonical path for the given path, in a
* JDK 1.1 complaint way.
*
* @param path the path for which to retrieve the canonical path
* @return the canonical path
*/
public String canonicalPath(String path) throws IOException {
return new File(path).getCanonicalPath();
}
/**
* Retrieves the canonical path for the given path, or the absolute
* path if attemting to retrieve the canonical path fails.
*
* @param path the path for which to retrieve the canonical or
* absolute path
* @return the canonical or absolute path
*/
public String canonicalOrAbsolutePath(String path) {
try {
return canonicalPath(path);
} catch (Exception e) {
return absolutePath(path);
}
}
public void makeParentDirectories(File f) {
String parent = f.getParent();
if (parent != null) {
new File(parent).mkdirs();
} else {
// workaround for jdk 1.1 bug (returns null when there is a parent)
parent = f.getPath();
int index = parent.lastIndexOf('/');
if (index > 0) {
parent = parent.substring(0, index);
new File(parent).mkdirs();
}
}
}
public static String makeDirectories(String path) {
try {
File file = new File(path);
file.mkdirs();
return file.getCanonicalPath();
} catch (IOException e) {
return null;
}
}
public FileAccess.FileSync getFileSync(java.io.OutputStream os)
throws java.io.IOException {
return new FileSync((FileOutputStream) os);
}
public static class FileSync implements FileAccess.FileSync {
FileDescriptor outDescriptor;
FileSync(FileOutputStream os) throws IOException {
outDescriptor = os.getFD();
}
public void sync() throws IOException {
outDescriptor.sync();
}
}
public static class FileAccessRes implements FileAccess {
public boolean isStreamElement(String elementName) {
return getClass().getResource(elementName) != null;
}
public InputStream openInputStreamElement(String streamName)
throws IOException {
InputStream is;
try {
is = getClass().getResourceAsStream(streamName);
if (is == null) {
is = Thread.currentThread().getContextClassLoader()
.getResourceAsStream(streamName);
}
} catch (Throwable t) {
throw JavaSystem.toIOException(t);
}
return is;
}
public void createParentDirs(java.lang.String filename) {}
public void removeElement(java.lang.String filename) {}
public void renameElement(java.lang.String oldName,
java.lang.String newName) {}
public java.io.OutputStream openOutputStreamElement(String streamName)
throws IOException {
throw new IOException();
}
public FileAccess.FileSync getFileSync(OutputStream os)
throws IOException {
throw new IOException();
}
}
}