org.apache.openejb.loader.Files Maven / Gradle / Ivy
/*
* 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.openejb.loader;
import java.io.File;
import java.io.FileFilter;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.security.DigestInputStream;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import java.util.regex.Pattern;
import static org.apache.openejb.loader.JarLocation.decode;
/**
* @version $Rev$ $Date$
*/
public class Files {
private static final Map DIGESTS = new HashMap<>();
private static final boolean IS_WINDOWS = System.getProperty("os.name", "unknown").toLowerCase(Locale.ENGLISH).startsWith("win");
public static File path(final String... parts) {
File dir = null;
for (final String part : parts) {
if (dir == null) {
dir = new File(part);
} else {
dir = new File(dir, part);
}
}
return null != dir ? dir.getAbsoluteFile() : dir;
}
public static File path(final File dir, final String... parts) {
File base = dir;
int idx = 0;
if (parts.length >= 1) {
final File partFile = new File(parts[0]);
if (partFile.exists() && partFile.isAbsolute()) {
base = partFile;
idx = 1;
}
}
for (int i = idx; i < parts.length; i++) {
base = new File(base, parts[i]);
}
return base.getAbsoluteFile();
}
public static List collect(final File dir, final String regex) {
return collect(dir, Pattern.compile(regex));
}
public static List collect(final File dir, final Pattern pattern) {
return collect(dir, new PatternFileFilter(pattern));
}
public static List collect(final File dir, final FileFilter filter) {
final List accepted = new ArrayList<>();
if (filter.accept(dir)) {
accepted.add(dir);
}
final File[] files = dir.listFiles();
if (files != null) {
for (final File file : files) {
accepted.addAll(collect(file, filter));
}
}
return accepted;
}
public static File exists(final File file, final String s) {
if (!file.exists()) {
throw new FileDoesNotExistException(s + " does not exist: " + file.getAbsolutePath());
}
return file;
}
public static File exists(final File file) {
if (!file.exists()) {
throw new FileDoesNotExistException("Does not exist: " + file.getAbsolutePath());
}
return file;
}
public static File dir(final File file) {
if (!file.isDirectory()) {
throw new FileRuntimeException("Not a directory: " + file.getAbsolutePath());
}
return file;
}
public static File touch(final File file) throws IOException {
if (!file.createNewFile()) {
throw new FileRuntimeException("Cannot create file: " + file.getAbsolutePath());
}
return file;
}
public static File file(final File file) {
exists(file);
if (!file.isFile()) {
throw new FileRuntimeException("Not a file: " + file.getAbsolutePath());
}
return file;
}
public static File notHidden(final File file) {
exists(file);
if (file.isHidden()) {
throw new FileRuntimeException("File is hidden: " + file.getAbsolutePath());
}
return file;
}
public static File writable(final File file) {
if (!file.canWrite()) {
throw new FileRuntimeException("Not writable: " + file.getAbsolutePath());
}
return file;
}
public static File readable(final File file) {
if (!file.canRead()) {
throw new FileRuntimeException("Not readable: " + file.getAbsolutePath());
}
return file;
}
public static File readableFile(final File file) {
return readable(file(file));
}
public static File mkdir(final File file) {
if (file.exists()) {
return file;
}
hackJDK4715154();
if (!file.mkdirs()) {
throw new FileRuntimeException("Cannot mkdir: " + file.getAbsolutePath());
}
return file;
}
private static void hackJDK4715154() {
if (IS_WINDOWS) {
//Known Windows bug JDK-4715154 and as of JDK8 still not fixable due to OS
System.gc();//NOPMD
}
}
public static File mkdir(final File file, final String name) {
return mkdir(new File(file, name));
}
public static File tmpdir() {
try {
File file;
try {
file = File.createTempFile("temp", "dir");
} catch (final Throwable e) {
//Use a local tmp directory
final File tmp = new File("tmp");
hackJDK4715154();
if (!tmp.exists() && !tmp.mkdirs()) {
throw new IOException("Failed to create local tmp directory: " + tmp.getAbsolutePath());
}
file = File.createTempFile("temp", "dir", tmp);
}
hackJDK4715154();
if (!file.delete()) {
throw new IOException("Failed to create temp dir. Delete failed");
}
mkdir(file);
deleteOnExit(file);
return file;
} catch (final IOException e) {
throw new FileRuntimeException(e);
}
}
public static File mkparent(final File file) {
mkdirs(file.getParentFile());
return file;
}
public static File mkdirs(final File file) {
if (!file.exists()) {
hackJDK4715154();
if (!file.mkdirs()) {
throw new FileRuntimeException("Cannot mkdirs: " + file.getAbsolutePath());
}
return file;
}
return dir(file);
}
// Shutdown hook for recursive DELETE on tmp directories
private static final List DELETE = new ArrayList<>();
static {
final ClassLoader loader = Thread.currentThread().getContextClassLoader();
Thread.currentThread().setContextClassLoader(Files.class.getClassLoader());
try {
final Thread deleteShutdownHook = new DeleteThread();
try {
Runtime.getRuntime().addShutdownHook(deleteShutdownHook);
} catch (final Throwable e) {
//Ignore
}
} finally {
Thread.currentThread().setContextClassLoader(loader);
}
}
public static void deleteOnExit(final File file) {
DELETE.add(file.getAbsolutePath());
flagForDeleteOnExit(file);
}
public static void flagForDeleteOnExit(final File file) {
if (file.exists()) {
if (file.isDirectory()) {
final File[] files = file.listFiles();
if (null != files) {
for (final File f : files) {
flagForDeleteOnExit(f);
}
}
}
try {
file.deleteOnExit();
} catch (final SecurityException e) {
//Ignore
}
}
}
private static void delete() {
for (final String path : DELETE) {
delete(new File(path));
}
}
/**
* Delete a file and all contents if specified file is a directory.
* If the DELETE fails then the file/s are flagged for DELETE on exit.
*
* @param file File
*/
public static void delete(final File file) {
if (null != file && file.exists()) {
if (file.isDirectory()) {
final File[] files = file.listFiles();
if (null != files) {
for (final File f : files) {
delete(f);
}
}
}
try {
hackJDK4715154();
if (!file.delete()) {
file.deleteOnExit();
}
} catch (final Throwable e) {
//Ignore
}
}
}
/**
* Delete a file and all contents if specified file is a directory
*
* @param file File
* @throws IllegalStateException on failure at any point
*/
public static void remove(final File file) {
if (null != file && file.exists()) {
if (file.isDirectory()) {
final File[] files = file.listFiles();
if (files != null) {
for (final File child : files) {
remove(child);
}
}
}
hackJDK4715154();
if (!file.delete()) {
throw new IllegalStateException("Could not delete file: " + file.getAbsolutePath());
}
}
}
public static File select(final File dir, final String pattern) {
final List matches = collect(dir, pattern);
if (matches.isEmpty()) {
throw new IllegalStateException(String.format("Missing '%s'", pattern));
}
if (matches.size() > 1) {
throw new IllegalStateException(String.format("Too many found '%s': %s", pattern, join(", ", matches)));
}
return matches.get(0);
}
private static String join(final String delimiter, final Collection collection) {
if (collection.isEmpty()) {
return "";
}
final StringBuilder sb = new StringBuilder();
for (final File obj : collection) {
sb.append(obj.getName()).append(delimiter);
}
return sb.substring(0, sb.length() - delimiter.length());
}
// return the token as url if simply a path otheriwse if ending by *.jar returning the list of
// files in the folder
/**
*
* @param path String
* @return List of files in the folder
*/
public static Set listJars(final String path) {
final Set set = new HashSet<>();
String token = path;
if (token.endsWith("*.jar")) {
token = token.substring(0, token.length() - "*.jar".length());
final File directory = new File(token);
if (!directory.isDirectory()) {
return set;
}
final String[] filenames = directory.list();
Arrays.sort(filenames);
for (final String rawFilename : filenames) {
final String filename = rawFilename.toLowerCase(Locale.ENGLISH);
if (!filename.endsWith(".jar")) {
continue;
}
final File file = new File(directory, rawFilename);
if (!file.isFile()) {
continue;
}
try {
set.add(file.toURI().toURL());
} catch (final MalformedURLException e) {
// no-op
}
}
} else {
// single file or directory
final File file = new File(token);
if (!file.exists()) {
return set;
}
try {
set.add(file.toURI().toURL());
} catch (final MalformedURLException e) {
// no-op
}
}
return set;
}
public static File toFile(final URL url) {
if ("jar".equals(url.getProtocol())) {
try {
final String spec = url.getFile();
int separator = spec.indexOf('!');
/*
* REMIND: we don't handle nested JAR URLs
*/
if (separator == -1) {
throw new MalformedURLException("no ! found in jar url spec:" + spec);
}
return toFile(new URL(spec.substring(0, separator++)));
} catch (final MalformedURLException e) {
throw new IllegalStateException(e);
}
} else if ("file".equals(url.getProtocol())) {
String path = decode(url.getFile());
if (path.endsWith("!")) {
path = path.substring(0, path.length() - 1);
}
return new File(path);
} else {
throw new IllegalArgumentException("Unsupported URL scheme: " + url.toExternalForm());
}
}
public static String hash(final Set urls, final String algo) {
final Collection files = new ArrayList<>();
for (final URL u : urls) {
final File file = toFile(u);
if (!file.isDirectory()) {
files.add(file);
} else {
files.addAll(Files.collect(file, TrueFilter.INSTANCE));
}
}
MessageDigest digest = DIGESTS.get(algo);
if (digest == null) {
try {
digest = MessageDigest.getInstance(algo);
} catch (final NoSuchAlgorithmException e) {
throw new LoaderRuntimeException(e);
}
DIGESTS.put(algo, digest);
}
for (final File file : files) {
if (!file.exists()) {
continue;
}
DigestInputStream is = null;
try {
is = new DigestInputStream(new FileInputStream(file), digest);
IO.copy(is, new NoopOutputStream()); // read the stream
} catch (final IOException e) {
// no-op: shouldn't occur here
} finally {
IO.close(is);
}
}
final byte[] hash = digest.digest();
digest.reset();
final StringBuilder sb = new StringBuilder("");
for (final byte b : hash) { // hex convertion
sb.append(Integer.toString((b & 0xff) + 0x100, 16).substring(1));
}
return sb.toString();
}
public static class NoopOutputStream extends OutputStream {
@Override
public void write(final int b) throws IOException {
// no-op
}
}
public static class FileRuntimeException extends RuntimeException {
public FileRuntimeException(final String str) {
super(str);
}
public FileRuntimeException(final Exception e) {
super(e);
}
}
public static class FileDoesNotExistException extends FileRuntimeException {
public FileDoesNotExistException(final String str) {
super(str);
}
public FileDoesNotExistException(final Exception e) {
super(e);
}
}
public static class PatternFileFilter implements FileFilter {
private final Pattern pattern;
public PatternFileFilter(final Pattern pattern) {
this.pattern = pattern;
}
@Override
public boolean accept(final File file) {
return pattern.matcher(file.getName()).matches();
}
}
public static class DeleteThread extends Thread {
@Override
public void run() {
delete();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy