com.izforge.izpack.util.IoHelper Maven / Gradle / Ivy
/*
* IzPack - Copyright 2001-2008 Julien Ponge, All Rights Reserved.
*
* http://izpack.org/
* http://izpack.codehaus.org/
*
* Copyright 2004 Elmar Klaus Bartz
*
* 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 com.izforge.izpack.util;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.FilterOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.StringTokenizer;
import java.util.zip.ZipEntry;
import java.util.zip.ZipException;
import java.util.zip.ZipInputStream;
import org.apache.tools.zip.ZipOutputStream;
import com.izforge.izpack.api.data.Variables;
import com.izforge.izpack.api.substitutor.SubstitutionType;
import com.izforge.izpack.api.substitutor.VariableSubstitutor;
/**
*
* Class with some IO related helper.
*
*/
public class IoHelper
{
// This class uses the same values for family and flavor as
// TargetFactory. But this class should not depends on TargetFactory,
// because it is possible that TargetFactory is not bound. Therefore
// the definition here again.
// ------------------------------------------------------------------------
// Constant Definitions
// ------------------------------------------------------------------------
/**
* Placeholder during translatePath computing
*/
private static final String MASKED_SLASH_PLACEHOLDER = "~&_&~";
private static Properties envVars = null;
/**
* Default constructor
*/
private IoHelper()
{
}
/**
* Copies the contents of inFile into outFile.
*
* @param inFile path of file which should be copied
* @param outFile path of file to create and copy the contents of inFile into
*/
public static void copyFile(String inFile, String outFile) throws IOException
{
copyFile(new File(inFile), new File(outFile));
}
/**
* Creates an in- and output stream for the given File objects and copies all the data from the
* specified input to the specified output.
*
* @param inFile File object for input
* @param outFile File object for output
* @throws IOException if an I/O error occurs
*/
public static void copyFile(File inFile, File outFile) throws IOException
{
copyFile(inFile, outFile, null, null);
}
/**
* Creates an in- and output stream for the given File objects and copies all the data from the
* specified input to the specified output. If permissions is not null, a chmod will be done on
* the output file.
*
* @param inFile File object for input
* @param outFile File object for output
* @param permissions permissions for the output file
* @throws IOException if an I/O error occurs
*/
public static void copyFile(File inFile, File outFile, String permissions) throws IOException
{
copyFile(inFile, outFile, permissions, null);
}
/**
* Creates an in- and output stream for the given File objects and copies all the data from the
* specified input to the specified output. If the VariableSubstitutor is not null, a substition
* will be done during copy.
*
* @param inFile File object for input
* @param outFile File object for output
* @param vss substitutor which is used during copying
* @throws IOException if an I/O error occurs
*/
public static void copyFile(File inFile, File outFile, VariableSubstitutor vss)
throws IOException
{
copyFile(inFile, outFile, null, vss);
}
/**
* Creates an in- and output stream for the given File objects and copies all the data from the
* specified input to the specified output. If the VariableSubstitutor is not null, a substition
* will be done during copy. If permissions is not null, a chmod will be done on the output
* file.
*
* @param inFile File object for input
* @param outFile File object for output
* @param permissions permissions for the output file
* @param vs substitutor which is used during copying
* @throws IOException if an I/O error occurs
*/
public static void copyFile(File inFile, File outFile, String permissions,
VariableSubstitutor vs) throws IOException
{
copyFile(inFile, outFile, permissions, vs, null);
}
/**
* Creates an in- and output stream for the given File objects and copies all the data from the
* specified input to the specified output. If the VariableSubstitutor is not null, a substition
* will be done during copy. If permissions is not null, a chmod will be done on the output
* file. If type is not null, that type is used as file type at substitution.
*
* @param inFile File object for input
* @param outFile File object for output
* @param permissions permissions for the output file
* @param vs substitutor which is used during copying
* @param type file type for the substitutor
* @throws IOException if an I/O error occurs
*/
public static void copyFile(File inFile, File outFile, String permissions,
VariableSubstitutor vs, SubstitutionType type) throws IOException
{
FileOutputStream out = new FileOutputStream(outFile);
FileInputStream in = new FileInputStream(inFile);
copyStream(in, out, vs, type);
if (permissions != null && IoHelper.supported("chmod"))
{
chmod(outFile.getAbsolutePath(), permissions);
}
}
/**
* Copies an input stream to an output stream. Both streams must already
* be open. If the VariableSubstitutor is not null, a substition
* will be done during copy.
*
* @param in stream object for input
* @param out stream object for output
* @param vs substitutor which is used during copying
* @param type file type for the substitutor
* @throws IOException if an I/O error occurs
*/
public static void copyStream(InputStream in, OutputStream out, VariableSubstitutor vs, SubstitutionType type)
throws IOException
{
if (vs == null)
{
byte[] buffer = new byte[5120];
long bytesCopied = 0;
int bytesInBuffer;
while ((bytesInBuffer = in.read(buffer)) != -1)
{
out.write(buffer, 0, bytesInBuffer);
bytesCopied += bytesInBuffer;
}
in.close();
out.close();
}
else
{
BufferedInputStream bin = new BufferedInputStream(in, 5120);
BufferedOutputStream bout = new BufferedOutputStream(out, 5120);
try
{
vs.substitute(bin, bout, type, null);
}
catch (Exception e)
{
throw new IOException("Substitution failed during copying a stream(" + e.getMessage() + ")");
}
bin.close();
bout.close();
}
}
/**
* Creates a temp file with delete on exit rule. The extension is extracted from the template if
* possible, else the default extension is used. The contents of template will be copied into
* the temporary file.
*
* @param template file to copy from and define file extension
* @param defaultExtension file extension if no is contained in template
* @return newly created and filled temporary file
* @throws IOException
*/
public static File copyToTempFile(File template, String defaultExtension) throws IOException
{
return copyToTempFile(template, defaultExtension, null);
}
/**
* Creates a temp file with delete on exit rule. The contents of the input stream will be copied into
* the temporary file. If the variable substitutor is not null, variables will be replaced
* during copying.
*
* @param is input stream to copy from
* @param ext file extension
* @param vss substitutor which is used during copying
* @return newly created and filled temporary file
* @throws IOException
*/
public static File copyToTempFile(InputStream is, String ext,
VariableSubstitutor vss) throws IOException
{
File tmpFile = File.createTempFile("izpack_io", ext);
tmpFile.deleteOnExit();
IoHelper.copyStream(is, new FileOutputStream(tmpFile), vss, null);
return tmpFile;
}
/**
* Creates a temp file with delete on exit rule. The extension is extracted from the template if
* possible, else the default extension is used. The contents of template will be copied into
* the temporary file. If the variable substitutor is not null, variables will be replaced
* during copying.
*
* @param template file to copy from and define file extension
* @param defaultExtension file extension if no is contained in template
* @param vss substitutor which is used during copying
* @return newly created and filled temporary file
* @throws IOException
*/
public static File copyToTempFile(File template, String defaultExtension,
VariableSubstitutor vss) throws IOException
{
String path = template.getCanonicalPath();
int pos = path.lastIndexOf('.');
String ext = path.substring(pos);
if (ext == null)
{
ext = defaultExtension;
}
File tmpFile = File.createTempFile("izpack_io", ext);
tmpFile.deleteOnExit();
IoHelper.copyFile(template, tmpFile, vss);
return tmpFile;
}
/**
* Creates a temp file with delete on exit rule. The extension is extracted from the template if
* possible, else the default extension is used. The contents of template will be copied into
* the temporary file.
*
* @param template file to copy from and define file extension
* @param defaultExtension file extension if no is contained in template
* @return newly created and filled temporary file
* @throws IOException
*/
public static File copyToTempFile(String template, String defaultExtension) throws IOException
{
return copyToTempFile(new File(template), defaultExtension);
}
/**
* Changes the permissions of the given file to the given POSIX permissions.
*
* @param file the file for which the permissions should be changed
* @param permissions POSIX permissions to be set
* @throws IOException if an I/O error occurs
*/
public static void chmod(File file, String permissions) throws IOException
{
chmod(file.getAbsolutePath(), permissions);
}
/**
* Changes the permissions of the given file to the given POSIX permissions. This method will be
* raised an exception, if the OS is not UNIX.
*
* @param path the absolute path of the file for which the permissions should be changed
* @param permissions POSIX permissions to be set
* @throws IOException if an I/O error occurs
*/
public static void chmod(String path, String permissions) throws IOException
{
// Perform UNIX
if (OsVersion.IS_UNIX)
{
String[] params = {"chmod", permissions, path};
String[] output = new String[2];
FileExecutor fe = new FileExecutor();
fe.executeCommand(params, output);
}
else
{
throw new IOException("Sorry, chmod not supported yet on " + OsVersion.OS_NAME + ".");
}
}
/**
* Returns the free (disk) space for the given path. If it is not ascertainable -1 returns.
*
* @param path path for which the free space should be detected
* @return the free space for the given path
*/
public static long getFreeSpace(String path)
{
long retval = -1;
if (OsVersion.IS_WINDOWS)
{
String command = "cmd.exe";
if (System.getProperty("os.name").toLowerCase().contains("windows 9"))
{
return (-1);
}
String[] params = {command, "/C", "\"dir /D /-C \"" + path + "\"\""};
String[] output = new String[2];
FileExecutor fe = new FileExecutor();
fe.executeCommand(params, output);
retval = extractLong(output[0], -3, 3, "%");
}
else if (OsVersion.IS_SUNOS)
{
String[] params = {"df", "-k", path};
String[] output = new String[2];
FileExecutor fe = new FileExecutor();
fe.executeCommand(params, output);
retval = extractLong(output[0], -3, 3, "%") * 1024;
}
else if (OsVersion.IS_HPUX)
{
String[] params = {"bdf", path};
String[] output = new String[2];
FileExecutor fe = new FileExecutor();
fe.executeCommand(params, output);
retval = extractLong(output[0], -3, 3, "%") * 1024;
}
else if (OsVersion.IS_UNIX)
{
String[] params = {"df", "-Pk", path};
String[] output = new String[2];
FileExecutor fe = new FileExecutor();
fe.executeCommand(params, output);
retval = extractLong(output[0], -3, 3, "%") * 1024;
}
return retval;
}
/**
* Returns whether the given method will be supported with the given environment. Some methods
* of this class are not supported on all operation systems.
*
* @param method name of the method
* @return true if the method will be supported with the current enivronment else false
* @throws RuntimeException if the given method name does not exist
*/
public static boolean supported(String method)
{
if ("getFreeSpace".equals(method))
{
if (OsVersion.IS_UNIX)
{
return true;
}
if (OsVersion.IS_WINDOWS)
{ // getFreeSpace do not work on Windows 98.
if (System.getProperty("os.name").toLowerCase().contains("windows 9"))
{
return (false);
}
return (true);
}
}
else if ("chmod".equals(method))
{
if (OsVersion.IS_UNIX)
{
return true;
}
}
else if ("copyFile".equals(method))
{
return true;
}
else if ("getPrimaryGroup".equals(method))
{
if (OsVersion.IS_UNIX)
{
return true;
}
}
else if ("getenv".equals(method))
{
return true;
}
else
{
throw new RuntimeException("method name " + method + "not supported by this method");
}
return false;
}
/**
* Returns the first existing parent directory in a path
*
* @param path path which should be scanned
* @return the first existing parent directory in a path
*/
public static File existingParent(File path)
{
File result = path;
while (!result.exists())
{
if (result.getParent() == null)
{
return result;
}
result = result.getParentFile();
}
return result;
}
/**
* Extracts a long value from a string in a special manner. The string will be broken into
* tokens with a standard StringTokenizer. Arround the assumed place (with the given half range)
* the tokens are scaned reverse for a token which represents a long. if useNotIdentifier is not
* null, tokens which are contains this string will be ignored. The first founded long returns.
*
* @param in the string which should be parsed
* @param assumedPlace token number which should contain the value
* @param halfRange half range for detection range
* @param useNotIdentifier string which determines tokens which should be ignored
* @return founded long
*/
private static long extractLong(String in, int assumedPlace, int halfRange,
String useNotIdentifier)
{
long retval = -1;
StringTokenizer st = new StringTokenizer(in);
int length = st.countTokens();
int i;
int currentRange = 0;
String[] interestedEntries = new String[halfRange + halfRange];
for (i = 0; i < length - halfRange + assumedPlace; ++i)
{
st.nextToken(); // Forget this entries.
}
for (i = 0; i < halfRange + halfRange; ++i)
{ // Put the interesting Strings into an intermediaer array.
if (st.hasMoreTokens())
{
interestedEntries[i] = st.nextToken();
currentRange++;
}
}
for (i = currentRange - 1; i >= 0; --i)
{
if (useNotIdentifier != null && interestedEntries[i].contains(useNotIdentifier))
{
continue;
}
try
{
retval = Long.parseLong(interestedEntries[i]);
}
catch (NumberFormatException nfe)
{
continue;
}
break;
}
return retval;
}
/**
* Returns the primary group of the current user. This feature will be supported only on Unix.
* On other systems null returns.
*
* @return the primary group of the current user
*/
public static String getPrimaryGroup()
{
if (supported("getPrimaryGroup"))
{
if (OsVersion.IS_SUNOS)
{ // Standard id of SOLARIS do not support -gn.
String[] params = {"id"};
String[] output = new String[2];
FileExecutor fileExecutor = new FileExecutor();
fileExecutor.executeCommand(params, output);
// No we have "uid=%u(%s) gid=%u(%s)"
if (output[0] != null)
{
StringTokenizer tokenizer = new StringTokenizer(output[0], "()");
int length = tokenizer.countTokens();
if (length >= 4)
{
for (int i = 0; i < 3; ++i)
{
tokenizer.nextToken();
}
return (tokenizer.nextToken());
}
}
return (null);
}
else
{
String[] params = {"id", "-gn"};
String[] output = new String[2];
FileExecutor fe = new FileExecutor();
fe.executeCommand(params, output);
return output[0];
}
}
else
{
return null;
}
}
/**
* Returns a string resulting from replacing all occurrences of what in this string with with.
* In opposite to the String.replaceAll method this method do not use regular expression or
* other methods which are only available in JRE 1.4 and later. This method was special made to
* mask masked slashes to avert a conversion during path translation.
*
* @param destination string for which the replacing should be performed
* @param what what string should be replaced
* @param with with what string what should be replaced
* @return a new String object if what was found in the given string, else the given string self
*/
public static String replaceString(String destination, String what, String with)
{
if (destination.contains(what))
{ // what found, with (placeholder) not included in destination ->
// perform changing.
StringBuffer buf = new StringBuffer();
int last = 0;
int current = destination.indexOf(what);
int whatLength = what.length();
while (current >= 0)
{ // Do not use Methods from JRE 1.4 and higher ...
if (current > 0)
{
buf.append(destination.substring(last, current));
}
buf.append(with);
last = current + whatLength;
current = destination.indexOf(what, last);
}
if (destination.length() > last)
{
buf.append(destination.substring(last));
}
return buf.toString();
}
return destination;
}
/**
* Translates a relative path to a local system path.
*
* @param destination the path to translate
* @param variables used to replaces variables in the path
* @return the translated path
*/
public static String translatePath(String destination, Variables variables)
{
destination = variables.replace(destination);
return translatePath(destination);
}
/**
* Translates a relative path to a local system path.
*
* @param destination The path to translate.
* @return The translated path.
* @deprecated see {@link #translatePath(String, Variables)}
*/
@Deprecated
public static String translatePath(String destination, VariableSubstitutor vs)
{
// Parse for variables
try
{
destination = vs.substitute(destination);
}
catch (Exception e)
{
// ignore
}
return translatePath(destination);
}
/**
* Translates a path.
*
* @param destination the path to translate
* @return the translated path
*/
private static String translatePath(String destination)
{
// Convert the file separator characters
// destination = destination.replace('/', File.separatorChar);
// Undo the conversion if the slashes was masked with
// a backslash
// Not all occurencies of slashes are path separators. To differ
// between it we allow to mask a slash with a backslash infront.
// Unfortunately we cannot use String.replaceAll because it
// handles backslashes in the replacement string in a special way
// and the method exist only beginning with JRE 1.4.
// Therefore the little bit crude way following ...
if (destination.contains("\\/") && !destination.contains(MASKED_SLASH_PLACEHOLDER))
{ // Masked slash found, placeholder not included in destination ->
// perform masking.
destination = replaceString(destination, "\\/", MASKED_SLASH_PLACEHOLDER);
// Masked slashes changed to MASKED_SLASH_PLACEHOLDER.
// Replace unmasked slashes.
destination = destination.replace('/', File.separatorChar);
// Replace the MASKED_SLASH_PLACEHOLDER to slashes; masking
// backslashes will
// be removed.
destination = replaceString(destination, MASKED_SLASH_PLACEHOLDER, "/");
}
else
{
destination = destination.replace('/', File.separatorChar);
}
return destination;
}
/**
* Returns the value of the environment variable given by key. This method is a work around for
* VM versions which do not support getenv in an other way. At the first call all environment
* variables will be loaded via an exec. On Windows keys are not case sensitive.
*
* @param key variable name for which the value should be resolved
* @return the value of the environment variable given by key
*/
public static String getenv(String key)
{
if (envVars == null)
{
loadEnv();
}
if (envVars == null)
{
return (null);
}
if (OsVersion.IS_WINDOWS)
{
key = key.toUpperCase();
}
return (String) (envVars.get(key));
}
/**
* Loads all environment variables via an exec.
*/
private static void loadEnv()
{
String[] output = new String[2];
String[] params;
if (OsVersion.IS_WINDOWS)
{
String command = "cmd.exe";
if (System.getProperty("os.name").toLowerCase().contains("windows 9"))
{
command = "command.com";
}
String[] paramst = {command, "/C", "set"};
params = paramst;
}
else
{
String[] paramst = {"env"};
params = paramst;
}
FileExecutor fe = new FileExecutor();
fe.executeCommand(params, output);
if (output[0].length() <= 0)
{
return;
}
String lineSep = System.getProperty("line.separator");
StringTokenizer tokenizer = new StringTokenizer(output[0], lineSep);
envVars = new Properties();
String var = null;
while (tokenizer.hasMoreTokens())
{
String line = tokenizer.nextToken();
if (line.indexOf('=') == -1)
{ // May be a env var with a new line in it.
if (var == null)
{
var = lineSep + line;
}
else
{
var += lineSep + line;
}
}
else
{ // New var, perform the previous one.
setEnvVar(var);
var = line;
}
}
setEnvVar(var);
}
/**
* Extracts key and value from the given string var. The key should be separated from the value
* by a sign. On Windows all chars of the key are translated to upper case.
*
* @param var
*/
private static void setEnvVar(String var)
{
if (var == null)
{
return;
}
int index = var.indexOf('=');
if (index < 0)
{
return;
}
String key = var.substring(0, index);
// On windows change all key chars to upper.
if (OsVersion.IS_WINDOWS)
{
key = key.toUpperCase();
}
envVars.setProperty(key, var.substring(index + 1));
}
/**
* Copies specified contents of one jar to another.
*
*
* TODO: it would be useful to be able to keep signature information from signed jar files, can
* we combine manifests and still have their content signed?
*/
public static void copyZip(ZipInputStream zin, org.apache.tools.zip.ZipOutputStream out,
List files, Map> alreadyWrittenFiles)
throws IOException
{
ZipEntry zentry;
if (!alreadyWrittenFiles.containsKey(out))
{
alreadyWrittenFiles.put(out, new HashSet());
}
Set currentSet = alreadyWrittenFiles.get(out);
while ((zentry = zin.getNextEntry()) != null)
{
String currentName = zentry.getName();
String testName = currentName.replace('/', '.');
testName = testName.replace('\\', '.');
if (files != null)
{
boolean founded = false;
for (String doInclude : files)
{ // Make "includes" self to support regex.
if (testName.matches(doInclude))
{
founded = true;
break;
}
}
if (!founded)
{
continue;
}
}
if (currentSet.contains(currentName))
{
continue;
}
try
{
// Get input file date and time.
long fileTime = zentry.getTime();
copyStreamToJar(zin, out, currentName, fileTime);
zin.closeEntry();
currentSet.add(currentName);
}
catch (ZipException x)
{
// This avoids any problem that can occur with duplicate
// directories. for instance all META-INF data in jars
// unfortunately this do not work with the apache ZipOutputStream...
}
}
}
public static void copyStreamToJar(InputStream zin, ZipOutputStream out, String currentName, long fileTime)
throws IOException
{
// Create new entry for zip file.
org.apache.tools.zip.ZipEntry newEntry = new org.apache.tools.zip.ZipEntry(currentName);
// Make sure there is date and time set.
if (fileTime != -1)
{
newEntry.setTime(fileTime); // If found set it into output file.
}
out.putNextEntry(newEntry);
if (zin != null)
{
copyStream(zin, out);
}
out.closeEntry();
}
public static void copyStreamToJar(InputStream zin, java.util.zip.ZipOutputStream out, String currentName,
long fileTime) throws IOException
{
// Create new entry for zip file.
org.apache.tools.zip.ZipEntry newEntry = new org.apache.tools.zip.ZipEntry(currentName);
// Make sure there is date and time set.
if (fileTime != -1)
{
newEntry.setTime(fileTime); // If found set it into output file.
}
out.putNextEntry(newEntry);
if (zin != null)
{
copyStream(zin, out);
}
out.closeEntry();
}
/**
* Copies all the data from the specified input stream to the specified output stream.
*
* @param in the input stream to read
* @param out the output stream to write
* @return the total number of bytes copied
* @throws IOException if an I/O error occurs
*/
public static long copyStream(InputStream in, OutputStream out) throws IOException
{
byte[] buffer = new byte[5120];
long bytesCopied = 0;
int bytesInBuffer;
while ((bytesInBuffer = in.read(buffer)) != -1)
{
out.write(buffer, 0, bytesInBuffer);
bytesCopied += bytesInBuffer;
}
return bytesCopied;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy