All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.kuali.common.util.LocationUtils Maven / Gradle / Ivy

There is a newer version: 4.4.17
Show newest version
/**
 * Copyright 2010-2014 The Kuali Foundation
 *
 * Licensed under the Educational Community 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.opensource.org/licenses/ecl2.php
 *
 * 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.kuali.common.util;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.PrintStream;
import java.io.Reader;
import java.io.StringReader;
import java.io.Writer;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Properties;

import org.apache.commons.io.FileUtils;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.core.io.DefaultResourceLoader;
import org.springframework.core.io.Resource;
import org.springframework.core.io.ResourceLoader;
import org.springframework.util.Assert;

public class LocationUtils {

	private static final Logger logger = LoggerFactory.getLogger(LocationUtils.class);

	private static final String FILE_PREFIX = "file:";
	private static final String BACK_SLASH = "\\";
	private static final String FORWARD_SLASH = "/";
	private static final String SLASH_DOT_SLASH = "/./";
	private static final String DOT_DOT_SLASH = "../";
	private static final String SLASH_DOT_DOT = "/..";
	private static final String CLASSPATH = "classpath:";
	private static final String MD5 = "MD5";

	/**
	 * Get the MD5 checksum of the location
	 */
	public static String getMD5Checksum(String location) {
		return getChecksum(location, MD5);
	}

	/**
	 * Get the MD5 checksum of the file
	 */
	public static String getMD5Checksum(File file) {
		return getChecksum(getCanonicalPath(file), MD5);
	}

	/**
	 * Open a PrintStream to the indicated file. Parent directories are created if necessary.
	 */
	public static final PrintStream openPrintStream(File file) throws IOException {
		return new PrintStream(FileUtils.openOutputStream(file));
	}

	/**
	 * Open a Writer to the indicated file. Parent directories are created if necessary.
	 */
	public static final Writer openWriter(File file) throws IOException {
		touch(file);
		return new FileWriter(file);
	}

	/**
	 * Open a Writer to the location (It must be a writable file on the local file system). Parent directories are created if necessary.
	 */
	public static final Writer openWriter(String location) throws IOException {
		return openWriter(new File(location));
	}

	public static Properties getLocationProperties(LocationPropertiesContext context) {

		Assert.notNull(context, "context is null");

		Properties properties = context.getProperties();
		String keySuffix = context.getKeySuffix();
		String locationPropertiesSuffix = context.getLocationPropertiesSuffix();
		String encoding = context.getEncoding();

		Assert.notNull(properties, "properties is null");
		Assert.notNull(keySuffix, "keySuffix is null");
		Assert.notNull(locationPropertiesSuffix, "locationPropertiesSuffix is null");

		List keys = PropertyUtils.getEndsWithKeys(properties, keySuffix);

		Properties locationProperties = new Properties();
		for (String key : keys) {
			String location = properties.getProperty(key);
			if (!exists(location)) {
				continue;
			}
			String propertiesLocation = location + locationPropertiesSuffix;
			if (!exists(propertiesLocation)) {
				continue;
			}
			Properties p = PropertyUtils.load(propertiesLocation, encoding);
			locationProperties.putAll(p);
		}
		logger.info("Located {} properties for {} location listings", locationProperties.size(), keys.size());
		return locationProperties;
	}

	public static TextMetaData getTextMetaData(File file) {
		return getTextMetaData(getCanonicalPath(file));
	}

	public static TextMetaData getTextMetaData(String location) {
		long lines = 0;
		long size = 0;
		BufferedReader in = null;
		try {
			in = getBufferedReader(location);
			String s = in.readLine();
			while (s != null) {
				lines++;
				size += s.length();
				s = in.readLine();
			}
			return new TextMetaData(lines, size);
		} catch (IOException e) {
			throw new IllegalStateException(e);
		} finally {
			IOUtils.closeQuietly(in);
		}
	}

	/**
	 * Count the lines of text in the file.
	 * 
	 * @param file
	 *            The file to read lines of text from
	 * @return A count representing the number of lines of text
	 * @throws IllegalStateException
	 *             If there is an i/o exception reading the file
	 * @see BufferedReader
	 */
	public static long getLineCount(File file) {
		return getLineCount(file, null);
	}

	public static long getLineCount(File file, String encoding) {
		return getLineCount(getCanonicalPath(file));
	}

	/**
	 * Count the lines of text in the location.
	 * 
	 * @param location
	 *            The location to read lines of text from
	 * @return A count representing the number of lines of text
	 * @throws IllegalStateException
	 *             If there is an i/o exception reading the file
	 * @see BufferedReader
	 * @deprecated Use getLineCount(location,encoding) instead
	 */
	@Deprecated
	public static long getLineCount(String location) {
		return getLineCount(location, null);
	}

	/**
	 * Count the lines of text in the location.
	 * 
	 * @param location
	 *            The location to read lines of text from
	 * @return A count representing the number of lines of text
	 * @throws IllegalStateException
	 *             If there is an i/o exception reading the file
	 * @see BufferedReader
	 */
	public static long getLineCount(String location, String encoding) {
		long count = 0;
		BufferedReader in = null;
		try {
			in = getBufferedReader(location, encoding);
			while (in.readLine() != null) {
				count++;
			}
			return count;
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} finally {
			IOUtils.closeQuietly(in);
		}
	}

	public static final void copyLocationsToFiles(List locations, List files) {
		Assert.isTrue(locations.size() == files.size());
		for (int i = 0; i < locations.size(); i++) {
			String location = locations.get(i);
			File destination = files.get(i);
			copyLocationToFile(location, destination);
		}
	}

	/**
	 * Return the text that appears after classpath:. Throws IllegalArgumentException if location does not start with classpath:
	 */
	public static final String getClasspathFilename(String location) {
		return getClasspathFilenames(Arrays.asList(location)).get(0);
	}

	/**
	 * Return the text that appears after classpath:. Throws IllegalArgumentException if any locations do not start with classpath:
	 */
	public static final List getClasspathFilenames(List locations) {
		List classpathFilenames = new ArrayList();
		for (String location : locations) {
			if (!isClasspathLocation(location)) {
				throw new IllegalArgumentException(location + " must start with " + CLASSPATH);
			} else {
				classpathFilenames.add(StringUtils.substring(location, CLASSPATH.length()));
			}
		}
		return classpathFilenames;
	}

	/**
	 * Return true if location starts with classpath:
	 */
	public static final boolean isClasspathLocation(String location) {
		return StringUtils.startsWith(location, CLASSPATH);
	}

	public static final List getNormalizedPathFragments(String absolutePath, boolean directory) {
		String normalized = getNormalizedAbsolutePath(absolutePath);
		String[] tokens = StringUtils.split(normalized, FORWARD_SLASH);
		List fragments = new ArrayList();
		StringBuilder sb = new StringBuilder();
		sb.append(FORWARD_SLASH);
		int length = directory ? tokens.length : tokens.length - 1;
		for (int i = 0; i < length; i++) {
			if (i != 0) {
				sb.append(FORWARD_SLASH);
			}
			sb.append(tokens[i]);
			fragments.add(sb.toString());
		}
		return fragments;
	}

	public static final List getCanonicalPaths(List files) {
		List paths = new ArrayList();
		for (File file : files) {
			String path = getCanonicalPath(file);
			paths.add(path);
		}
		return paths;
	}

	public static final List getLocations(String location, LocationType type, String encoding) {
		switch (type) {
		case LOCATION:
			return Collections.singletonList(location);
		case LOCATIONLIST:
			return getLocations(location, encoding);
		default:
			throw new IllegalArgumentException("Location type '" + type + "' is unknown");
		}
	}

	public static final List getLocations(String location, LocationType type) {
		return getLocations(location, type, null);
	}

	public static final List getLocations(String locationListing) {
		return getLocations(Collections.singletonList(locationListing), null);
	}

	public static final List getLocations(String locationListing, String encoding) {
		return getLocations(Collections.singletonList(locationListing), encoding);
	}

	public static final List getLocations(List locationListings) {
		return getLocations(locationListings, null);
	}

	public static final void copyLocationToFile(String location, File destination) {
		Assert.notNull(location);
		Assert.notNull(destination);
		logger.debug("Copying [{}]->[{}]", location, destination);
		InputStream in = null;
		try {
			in = getInputStream(location);
			FileUtils.copyInputStreamToFile(in, destination);
		} catch (IOException e) {
			throw new IllegalStateException(e);
		} finally {
			IOUtils.closeQuietly(in);
		}
	}

	public static final List getFiles(File dir, List filenames) {
		List files = new ArrayList();
		for (String filename : filenames) {
			File file = new File(dir, filename);
			files.add(file);
		}
		return files;
	}

	public static final List getFilenames(List locations) {
		Assert.notNull(locations);
		List filenames = new ArrayList();
		for (String location : locations) {
			filenames.add(getFilename(location));
		}
		return filenames;
	}

	/**
	 * Throw IllegalArgumentException if the locationListing does not exist, or if any of the locations inside the locationListing do not exist
	 */
	public static final void validateLocationListing(String locationListing) {
		validateLocationListings(Collections.singletonList(locationListing));
	}

	/**
	 * Throw IllegalArgumentException if any of the locationListings do not exist, or if any of the locations inside any of the locationListings do not exist
	 */
	public static final void validateLocationListings(List locationListings) {
		for (String locationListing : locationListings) {
			validateLocation(locationListing);
			List locations = getLocations(locationListing);
			validateLocations(locations);
		}
	}

	public static final void validateLocations(List locations) {
		for (String location : locations) {
			validateLocation(location);
		}
	}

	/**
	 * Throw IllegalArgumentException if the location does not exist
	 */
	public static final void validateLocation(String location) {
		validateLocation(location, "[" + location + "] does not exist");
	}

	/**
	 * Throw IllegalArgumentException if the location does not exist
	 */
	public static final void validateLocation(String location, String message) {
		Assert.isTrue(exists(location), message);
	}

	public static final List getLocations(List locationListings, String encoding) {
		List locations = new ArrayList();
		for (String locationListing : locationListings) {
			List lines = readLines(locationListing, encoding);
			locations.addAll(lines);
		}
		return locations;
	}

	public static final String getCanonicalURLString(File file) {
		if (file == null) {
			return null;
		}
		String path = getCanonicalPath(file);
		File canonical = new File(path);
		return getURLString(canonical);
	}

	public static final void validateNormalizedPath(String originalPath, String normalizedPath) {
		if (CollectionUtils.containsAny(normalizedPath, Arrays.asList(SLASH_DOT_DOT, SLASH_DOT_SLASH, DOT_DOT_SLASH))) {
			throw new IllegalArgumentException("[" + originalPath + "] could not be normalized. Normalized path [" + normalizedPath + "]");
		}
	}

	/**
	 * Resolve and remove .. and . from absolutePath after converting any back slashes to forward slashes
	 */
	public static final String getNormalizedAbsolutePath(String absolutePath) {
		if (absolutePath == null) {
			return null;
		}
		String replaced = StringUtils.replace(absolutePath, BACK_SLASH, FORWARD_SLASH);
		boolean absolute = StringUtils.startsWith(replaced, FORWARD_SLASH);
		if (!absolute) {
			throw new IllegalArgumentException("[" + absolutePath + "] is not an absolute path.");
		}
		String prefixed = FILE_PREFIX + replaced;
		try {
			URI rawURI = new URI(prefixed);
			URI normalizedURI = rawURI.normalize();
			URL normalizedURL = normalizedURI.toURL();
			String externalForm = normalizedURL.toExternalForm();
			String trimmed = StringUtils.substring(externalForm, FILE_PREFIX.length());
			validateNormalizedPath(absolutePath, trimmed);
			return trimmed;
		} catch (MalformedURLException e) {
			throw new IllegalArgumentException(e);
		} catch (URISyntaxException e) {
			throw new IllegalArgumentException(e);
		}
	}

	public static final String getURLString(File file) {
		if (file == null) {
			return null;
		}
		try {
			URI uri = file.toURI();
			URL url = uri.toURL();
			return url.toExternalForm();
		} catch (MalformedURLException e) {
			throw new IllegalArgumentException(e);
		}
	}

	public static final void forceMkdir(File file) {
		try {
			FileUtils.forceMkdir(file);
		} catch (IOException e) {
			throw new IllegalArgumentException("Unexpected IO error", e);
		}
	}

	public static final void touch(File file) {
		try {
			FileUtils.touch(file);
		} catch (IOException e) {
			throw new IllegalArgumentException("Unexpected IO error", e);
		}
	}

	public static final String getCanonicalPath(File file) {
		try {
			return file.getCanonicalPath();
		} catch (IOException e) {
			throw new IllegalArgumentException("Unexpected IO error", e);
		}
	}

	/**
	 * Null safe method to unconditionally attempt to delete filename without throwing an exception. If filename is a directory, delete it and all
	 * sub-directories.
	 */
	public static final boolean deleteFileQuietly(String filename) {
		File file = getFileQuietly(filename);
		return FileUtils.deleteQuietly(file);
	}

	/**
	 * Null safe method for getting a File handle from filename. If filename is null, null is returned.
	 */
	public static final File getFileQuietly(String filename) {
		if (filename == null) {
			return null;
		} else {
			return new File(filename);
		}
	}

	/**
	 * Get the contents of file as a String using the platform's default character encoding.
	 */
	public static final String toString(File file) {
		return toString(file, null);
	}

	/**
	 * Get the contents of file as a String using the specified character encoding.
	 */
	public static final String toString(File file, String encoding) {
		return toString(getCanonicalPath(file), encoding);
	}

	/**
	 * Get the contents of location as a String using the platform's default character encoding.
	 */
	public static final String toString(String location) {
		return toString(location, null);
	}

	/**
	 * Get the contents of location as a String using the specified character encoding.
	 */
	public static final String toString(String location, String encoding) {
		InputStream in = null;
		try {
			in = getInputStream(location);
			if (encoding == null) {
				return IOUtils.toString(in);
			} else {
				return IOUtils.toString(in, encoding);
			}
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} finally {
			IOUtils.closeQuietly(in);
		}
	}

	/**
	 * Get the contents of s as a list of String's one entry per line
	 */
	public static final List readLinesFromString(String s) {
		Reader reader = getBufferedReaderFromString(s);
		return readLinesAndClose(reader);
	}

	public static final List readLinesAndClose(InputStream in) {
		return readLinesAndClose(in, null);
	}

	public static final List readLinesAndClose(InputStream in, String encoding) {
		Reader reader = null;
		try {
			reader = getBufferedReader(in, encoding);
			return readLinesAndClose(reader);
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} finally {
			IOUtils.closeQuietly(reader);
		}
	}

	public static final List readLinesAndClose(Reader reader) {
		try {
			return IOUtils.readLines(reader);
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} finally {
			IOUtils.closeQuietly(reader);
		}
	}

	/**
	 * Get the contents of file as a list of String's one entry per line using the platform default encoding
	 */
	public static final List readLines(File file) {
		return readLines(getCanonicalPath(file));
	}

	/**
	 * Get the contents of location as a list of String's one entry per line using the platform default encoding
	 */
	public static final List readLines(String location) {
		return readLines(location, null);
	}

	/**
	 * Get the contents of location as a list of String's one entry per line using the encoding indicated.
	 */
	public static final List readLines(String location, String encoding) {
		Reader reader = null;
		try {
			reader = getBufferedReader(location, encoding);
			return readLinesAndClose(reader);
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} finally {
			IOUtils.closeQuietly(reader);
		}
	}

	/**
	 * Return a BufferedReader for the location indicated using the platform default encoding.
	 */
	public static final BufferedReader getBufferedReader(String location) throws IOException {
		return getBufferedReader(location, null);
	}

	/**
	 * Return a BufferedReader for the location indicated using the encoding indicated.
	 */
	public static final BufferedReader getBufferedReader(String location, String encoding) throws IOException {
		try {
			InputStream in = getInputStream(location);
			return getBufferedReader(in, encoding);
		} catch (IOException e) {
			throw new IOException("Unexpected IO error", e);
		}
	}

	/**
	 * Return a BufferedReader that reads from s
	 */
	public static final BufferedReader getBufferedReaderFromString(String s) {
		return new BufferedReader(new StringReader(s));
	}

	/**
	 * Return a Writer that writes to out using the indicated encoding. null means use the platform's default encoding.
	 */
	public static final Writer getWriter(OutputStream out, String encoding) throws IOException {
		if (encoding == null) {
			return new BufferedWriter(new OutputStreamWriter(out));
		} else {
			return new BufferedWriter(new OutputStreamWriter(out, encoding));
		}
	}

	/**
	 * Return a BufferedReader that reads from file using the indicated encoding. null means use the platform's default encoding.
	 */
	public static final BufferedReader getBufferedReader(File file, String encoding) throws IOException {
		return getBufferedReader(FileUtils.openInputStream(file), encoding);
	}

	/**
	 * Return a BufferedReader that reads from in using the indicated encoding. null means use the platform's default encoding.
	 */
	public static final BufferedReader getBufferedReader(InputStream in, String encoding) throws IOException {
		if (encoding == null) {
			return new BufferedReader(new InputStreamReader(in));
		} else {
			return new BufferedReader(new InputStreamReader(in, encoding));
		}
	}

	/**
	 * Null safe method for determining if location is an existing file.
	 */
	public static final boolean isExistingFile(String location) {
		if (location == null) {
			return false;
		}
		File file = new File(location);
		return file.exists();
	}

	/**
	 * Null safe method for determining if location exists.
	 */
	public static final boolean exists(File file) {
		if (file == null) {
			return false;
		}
		String location = getCanonicalPath(file);
		if (isExistingFile(location)) {
			return true;
		} else {
			Resource resource = getResource(location);
			return resource.exists();
		}
	}

	public static void validateExists(List locations) {
		StringBuilder sb = new StringBuilder();
		for (String location : locations) {
			if (!LocationUtils.exists(location)) {
				sb.append("Location [" + location + "] does not exist\n");
			}
		}
		if (sb.length() > 0) {
			throw new IllegalArgumentException(sb.toString());
		}
	}

	/**
	 * Null safe method for determining if location exists.
	 */
	public static final boolean exists(String location) {
		if (location == null) {
			return false;
		}
		if (isExistingFile(location)) {
			return true;
		} else {
			Resource resource = getResource(location);
			return resource.exists();
		}
	}

	/**
	 * Open an InputStream to location. If location is the path to an existing File on the local file system, a
	 * FileInputStream is returned. Otherwise Spring's resource loading framework is used to open an InputStream to location.
	 */
	public static final InputStream getInputStream(String location) throws IOException {
		if (isExistingFile(location)) {
			return FileUtils.openInputStream(new File(location));
		}
		Resource resource = getResource(location);
		return resource.getInputStream();
	}

	public static final Resource getResource(String location) {
		if (location == null) {
			return null;
		}
		ResourceLoader loader = new DefaultResourceLoader();
		return loader.getResource(location);
	}

	public static final String getFilename(String location) {
		if (location == null) {
			return null;
		}
		if (isExistingFile(location)) {
			return getFileQuietly(location).getName();
		} else {
			Resource resource = getResource(location);
			return resource.getFilename();
		}
	}

	public static final List getAbsolutePaths(List files) {
		List results = new ArrayList(files.size());

		for (File f : files) {
			results.add(f.getAbsolutePath());
		}

		return results;
	}

	public static final ComparisonResults getLocationListComparison(List newLocations, List originalLocations) {
		ComparisonResults result = new ComparisonResults();

		result.setAdded(new ArrayList());
		result.setSame(new ArrayList());
		result.setDeleted(new ArrayList());

		for (String newLocation : newLocations) {
			if (originalLocations.contains(newLocation)) {
				// if a location is in both lists, add it to the "same" list
				result.getSame().add(newLocation);
			} else {
				// if a location is only in the new list, add it to the "added" list
				result.getAdded().add(newLocation);
			}
		}

		// the "deleted" list will contain all locations from the original list that are NOT in the new list
		result.getDeleted().addAll(originalLocations);
		result.getDeleted().removeAll(newLocations);

		return result;
	}

	public static String getChecksum(String location, String algorithm) {
		byte[] bytes = createChecksum(location, algorithm);
		return getChecksum(bytes, algorithm);
	}

	public static String getChecksum(byte[] bytes, String algorithm) {
		StringBuilder sb = new StringBuilder();
		for (int i = 0; i < bytes.length; i++) {
			sb.append(Integer.toString((bytes[i] & 0xff) + 0x100, 16).substring(1));
		}
		return sb.toString();
	}

	public static byte[] createChecksum(String location, String algorithm) {
		InputStream in = null;
		try {
			in = LocationUtils.getInputStream(location);
			return createChecksum(in, algorithm);
		} catch (IOException e) {
			throw new IllegalStateException("Unexpected IO error", e);
		} finally {
			IOUtils.closeQuietly(in);
		}
	}

	public static byte[] createChecksum(InputStream in, String algorithm) throws IOException {
		try {
			byte[] buffer = new byte[1024];
			MessageDigest complete = MessageDigest.getInstance(algorithm);
			int numRead;
			do {
				numRead = in.read(buffer);
				if (numRead > 0) {
					complete.update(buffer, 0, numRead);
				}
			} while (numRead != -1);
			IOUtils.closeQuietly(in);
			return complete.digest();
		} catch (NoSuchAlgorithmException e) {
			throw new IllegalStateException("Unexpected message digest error", e);
		} catch (IOException e) {
			throw e;
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy