blob: d46ba60e9c46475062e6a976df979bf0f8f7370e [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.cocoon.util;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.text.Collator;
import java.util.Arrays;
import java.util.Locale;
import org.apache.commons.lang.StringUtils;
import org.apache.log.Hierarchy;
/**
* A collection of <code>File</code>, <code>URL</code> and filename
* utility methods
*
* @author <a href="mailto:ricardo@apache.org">Ricardo Rocha</a>
* @author <a href="mailto:stefano@apache.org">Stefano Mazzocchi</a>
* @version CVS $Id$
*/
public class IOUtils {
// **********************
// Serialize Methods
// **********************
/**
* Dump a <code>String</code> to a text file.
*
* @param file The output file
* @param string The string to be dumped
* @exception IOException IO Error
* @deprecated To be removed in cocoon 2.3
*/
public static void serializeString(File file, String string)
throws IOException {
serializeString(file, string, null);
}
/**
* Dump a <code>String</code> to a text file.
*
* @param file The output file
* @param string The string to be dumped
* @param encoding The encoding for the output file or null for default platform encoding
* @exception IOException IO Error
* @deprecated To be removed in cocoon 2.3
*/
public static void serializeString(File file, String string, String encoding)
throws IOException {
final Writer fw =
(encoding == null)?
new FileWriter(file):
new OutputStreamWriter(new FileOutputStream(file), encoding);
try {
fw.write(string);
fw.flush();
} finally {
fw.close();
}
}
/**
* Load a text file contents as a <code>String<code>.
* This method does not perform enconding conversions
*
* @param file The input file
* @return The file contents as a <code>String</code>
* @exception IOException IO Error
*/
public static String deserializeString(File file)
throws IOException {
int len;
char[] chr = new char[4096];
final StringBuffer buffer = new StringBuffer();
final FileReader reader = new FileReader(file);
try {
while ((len = reader.read(chr)) > 0) {
buffer.append(chr, 0, len);
}
} finally {
reader.close();
}
return buffer.toString();
}
/**
* This method serializes an object to an output stream.
*
* @param file The output file
* @param object The object to be serialized
* @exception IOException IOError
* @deprecated To be removed in cocoon 2.3
*/
public static void serializeObject(File file, Object object)
throws IOException {
FileOutputStream fos = new FileOutputStream(file);
try {
ObjectOutputStream oos = new ObjectOutputStream(new BufferedOutputStream(fos));
oos.writeObject(object);
oos.flush();
} finally {
fos.close();
}
}
/**
* This method deserializes an object from an input stream.
*
* @param file The input file
* @return The deserialized object
* @exception IOException IOError
* @deprecated To be removed in cocoon 2.3
*/
public static Object deserializeObject(File file)
throws IOException, ClassNotFoundException {
FileInputStream fis = new FileInputStream(file);
Object object = null;
try {
ObjectInputStream ois = new ObjectInputStream(new BufferedInputStream(fis));
object = ois.readObject();
} finally {
fis.close();
}
return object;
}
/**
* These are java keywords as specified at the following URL (sorted alphabetically).
* http://java.sun.com/docs/books/jls/second_edition/html/lexical.doc.html#229308
*/
static final String keywords[] =
{
"abstract", "boolean", "break", "byte", "case",
"catch", "char", "class", "const", "continue",
"default", "do", "double", "else", "extends",
"final", "finally", "float", "for", "goto",
"if", "implements", "import", "instanceof", "int",
"interface", "long", "native", "new", "package",
"private", "protected", "public", "return", "short",
"static", "strictfp", "super", "switch", "synchronized",
"this", "throw", "throws", "transient", "try",
"void", "volatile", "while"
};
/** Collator for comparing the strings */
static final Collator englishCollator = Collator.getInstance(Locale.ENGLISH);
/** Use this character as suffix */
static final char keywordSuffix = '_';
/**
* checks if the input string is a valid java keyword.
* @return boolean true/false
*/
private static boolean isJavaKeyword(String keyword) {
return (Arrays.binarySearch(keywords, keyword, englishCollator) >= 0);
}
// **********************
// File Methods
// **********************
/**
* Return a modified filename suitable for replicating directory
* structures below the store's base directory. The following
* conversions are performed:
* <ul>
* <li>Path separators are converted to regular directory names</li>
* <li>File path components are transliterated to make them valid (?)
* programming language identifiers. This transformation may well
* generate collisions for unusual filenames.</li>
* </ul>
* @return The transformed filename
*/
public static String normalizedFilename(String filename) {
if ("".equals(filename)) {
return "";
}
filename = (File.separatorChar == '\\') ? filename.replace('/','\\') : filename.replace('\\','/');
String[] path = StringUtils.split(filename, File.separator);
int start = (path[0].length() == 0) ? 1 : 0;
StringBuffer buffer = new StringBuffer();
for (int i = start; i < path.length; i++) {
if (i > start) {
buffer.append(File.separator);
}
if(path[i].equals("..")) {
int lio;
for (lio = buffer.length() - 2; lio >= 0; lio--) {
if (buffer.substring(lio).startsWith(File.separator)) {
break;
}
}
if (lio >= 0) {
buffer.setLength(lio);
}
} else {
char[] chars = path[i].toCharArray();
if (chars.length < 1 || !Character.isLetter(chars[0])) {
buffer.append('_');
}
for (int j = 0; j < chars.length; j++) {
if (org.apache.cocoon.util.StringUtils.isAlphaNumeric(chars[j])) {
buffer.append(chars[j]);
} else {
buffer.append('_');
}
}
// Append the suffix if necessary.
if(isJavaKeyword(path[i]))
buffer.append(keywordSuffix);
}
}
return buffer.toString();
}
/**
* Remove file information from a filename returning only its path
* component
*
* @param filename The filename
* @return The path information
* @deprecated To be removed in cocoon 2.3
*/
public static String pathComponent(String filename) {
int i = filename.lastIndexOf(File.separator);
return (i > -1) ? filename.substring(0, i) : filename;
}
/**
* Remove path information from a filename returning only its file
* component
*
* @param filename The filename
* @return The filename sans path information
* @deprecated To be removed in cocoon 2.3
*/
public static String fileComponent(String filename) {
int i = filename.lastIndexOf(File.separator);
return (i > -1) ? filename.substring(i + 1) : filename;
}
/**
* Strip a filename of its <i>last</i> extension (the portion
* immediately following the last dot character, if any)
*
* @param filename The filename
* @return The filename sans extension
* @deprecated To be removed in cocoon 2.3
*/
public static String baseName(String filename) {
int i = filename.lastIndexOf('.');
return (i > -1) ? filename.substring(0, i) : filename;
}
/**
* Get the complete filename corresponding to a (typically relative)
* <code>File</code>.
* This method accounts for the possibility of an error in getting
* the filename's <i>canonical</i> path, returning the io/error-safe
* <i>absolute</i> form instead
*
* @param file The file
* @return The file's absolute filename
*/
public static String getFullFilename(File file) {
try {
return file.getCanonicalPath();
} catch (Exception e) {
Hierarchy.getDefaultHierarchy().getLoggerFor("cocoon").debug("IOUtils.getFullFilename", e);
return file.getAbsolutePath();
}
}
/**
* Return the path within a base directory
*/
public static String getContextFilePath(String directoryPath, String filePath) {
try
{
File directory = new File(directoryPath);
File file = new File(filePath);
directoryPath = directory.getCanonicalPath();
filePath = file.getCanonicalPath();
// If the context directory does not have a File.separator
// at the end then add one explicitly
if(!directoryPath.endsWith(File.separator)){
directoryPath += File.separator;
}
// If the context dir contains both kinds of separator
// then standardize on using the File.separator
if ((directoryPath.indexOf('/') !=-1) && (directoryPath.indexOf('\\') !=-1)) {
directoryPath = directoryPath.replace('\\', File.separator.charAt(0));
directoryPath = directoryPath.replace('/', File.separator.charAt(0));
}
// If the file path contains both kinds of separator
// then standardize on using the File.separator
if ((filePath.indexOf('/') !=-1) && (filePath.indexOf('\\') !=-1)) {
filePath = filePath.replace('\\', File.separator.charAt(0));
filePath = filePath.replace('/', File.separator.charAt(0));
}
if (filePath.startsWith(directoryPath)) {
filePath = filePath.substring(directoryPath.length());
}
} catch (Exception e){
Hierarchy.getDefaultHierarchy().getLoggerFor("cocoon").debug("IOUtils.getContextFilePath", e);
}
return filePath;
}
/**
* Return a file with the given filename creating the necessary
* directories if not present.
*
* @param filename The file
* @return The created File instance
*/
public static File createFile(File destDir, String filename) {
File file = new File(destDir, filename);
File parent = file.getParentFile();
if (parent != null) parent.mkdirs();
return file;
}
/**
* Returns a byte array from the given object.
*
* @param object to convert
* @return byte array from the object
* @deprecated To be removed in cocoon 2.3
*/
public static byte[] objectToBytes(Object object) throws IOException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream os = new ObjectOutputStream(baos);
os.writeObject(object);
return baos.toByteArray();
}
/**
* Returns a object from the given byte array.
*
* @param bytes array to convert
* @return object
* @deprecated To be removed in cocoon 2.3
*/
public static Object bytesToObject(byte[] bytes) throws IOException, ClassNotFoundException {
ByteArrayInputStream bais = new ByteArrayInputStream(bytes);
ObjectInputStream is = new ObjectInputStream(bais);
return is.readObject();
}
}