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

edu.internet2.middleware.grouperInstaller.util.GrouperInstallerUtils Maven / Gradle / Ivy

The newest version!
/*******************************************************************************
 * Copyright 2012 Internet2
 * 
 * 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 edu.internet2.middleware.grouperInstaller.util;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileFilter;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.io.PushbackInputStream;
import java.io.Reader;
import java.io.StringWriter;
import java.io.UnsupportedEncodingException;
import java.io.Writer;
import java.lang.annotation.Annotation;
import java.lang.reflect.Array;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.math.BigDecimal;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.URL;
import java.net.URLClassLoader;
import java.net.URLDecoder;
import java.net.URLEncoder;
import java.security.CodeSource;
import java.security.MessageDigest;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.Timestamp;
import java.text.DateFormat;
import java.text.DecimalFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Calendar;
import java.util.Collection;
import java.util.Collections;
import java.util.Date;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.jar.Attributes;
import java.util.jar.Attributes.Name;
import java.util.jar.JarInputStream;
import java.util.jar.Manifest;
import java.util.logging.ConsoleHandler;
import java.util.logging.FileHandler;
import java.util.logging.Handler;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.logging.SimpleFormatter;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.zip.GZIPOutputStream;
import java.util.zip.ZipFile;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;
import javax.xml.xpath.XPath;
import javax.xml.xpath.XPathConstants;
import javax.xml.xpath.XPathExpression;
import javax.xml.xpath.XPathFactory;

import org.w3c.dom.Document;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

import edu.internet2.middleware.grouperInstaller.GiGrouperVersion;
import org.apache.commons.compress.archivers.tar.TarArchiveEntry;
import org.apache.commons.compress.archivers.tar.TarArchiveOutputStream;
import org.apache.commons.httpclient.HttpMethodBase;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.commons.logging.impl.Jdk14Logger;



/**
 * utility methods for grouper.
 * @author mchyzer
 *
 */
@SuppressWarnings({ "serial", "unchecked" })
public class GrouperInstallerUtils  {

  /**
   * 
   * @param args
   */
  public static void main(String[] args) {
//    tar(new File("C:\\app\\grouperInstallerTarballDir\\grouper_v2_2_1_ui_patch_17"),
//        new File("C:\\app\\grouperInstallerTarballDir\\grouper_v2_2_1_ui_patch_17.tar"));
    //gzip(new File("c:\\temp\\test.tar"), new File("c:\\temp\\test.tar.gz"));

    System.out.println(toStringForLog(GrouperInstallerUtils.jarFileBaseNames("aws-java-sdk-core-1.11.529.jar")));
  
  }
  
  /**
   * delete a file
   * @param file
   * @return true if delete, false if not exist
   */
  public static boolean fileDelete(File file) {
    if (!file.exists()) {
      return false;
    }
    if (!file.delete()) {
      throw new RuntimeException("Couldnt delete file: " + file);
    }
    return true;
  }

  /**
   * Deletes a file, never throwing an exception.
   * 

* The difference between File.delete() and this method are: *

    *
  • No exceptions are thrown when a file or directory cannot be deleted.
  • *
* * @param file file to delete, can be null * @return true if the file was deleted, otherwise * false * * @since Commons IO 1.4 */ public static boolean deleteQuietly(File file) { if (file == null) { return false; } try { return file.delete(); } catch (Exception e) { return false; } } /** * move a file * @param file * @param newFile move to new file */ public static void fileMove(File file, File newFile) { fileMove(file, newFile, true); } /** * move a file * @param file * @param newFile move to new file * @param exceptionIfError * @return if success */ public static boolean fileMove(File file, File newFile, boolean exceptionIfError) { fileDelete(newFile); if (!file.renameTo(newFile)) { copyFile( file, newFile ); if (!file.delete()) { if (!exceptionIfError) { return false; } deleteQuietly(newFile); throw new RuntimeException("Could not native Java rename, and failed to delete original file '" + file + "' after copy to '" + newFile + "'"); } } return true; } /** override map for properties in thread local to be used in a web server or the like */ private static ThreadLocal>> propertiesThreadLocalOverrideMap = new ThreadLocal>>(); /** * return the arg after the argBefore, or null if not there, or exception * if argBefore is not found * @param args * @param argBefore * @return the arg */ public static String argAfter(String[] args, String argBefore) { if (length(args) <= 1) { return null; } int argBeforeIndex = -1; for (int i=0;i 0) { displaySize = String.valueOf(size / ONE_GB) + " GB"; } else if (size / ONE_MB > 0) { displaySize = String.valueOf(size / ONE_MB) + " MB"; } else if (size / ONE_KB > 0) { displaySize = String.valueOf(size / ONE_KB) + " KB"; } else { displaySize = String.valueOf(size) + " bytes"; } return displaySize; } /** * see if options have a specific option by int bits * @param options * @param option * @return if the option is there */ public static boolean hasOption(int options, int option) { return (options & option) > 0; } /** * get canonical path of file * @param file * @return the path */ public static String fileCanonicalPath(File file) { try { return file.getCanonicalPath(); } catch (IOException ioe) { throw new RuntimeException(ioe); } } /** * return the suffix after a char. If the char doesnt exist, just return the string * @param input string * @param theChar char * @return new string */ public static String suffixAfterChar(String input, char theChar) { if (input == null) { return null; } //get the real type off the end int lastIndex = input.lastIndexOf(theChar); if (lastIndex > -1) { input = input.substring(lastIndex + 1, input.length()); } return input; } /** * get the oracle underscore name e.g. javaNameHere -> JAVA_NAME_HERE * * @param javaName * the java convention name * * @return the oracle underscore name based on the java name */ public static String oracleStandardNameFromJava(String javaName) { StringBuilder result = new StringBuilder(); if ((javaName == null) || (0 == "".compareTo(javaName))) { return javaName; } //if package is specified, only look at class name javaName = suffixAfterChar(javaName, '.'); //dont check the first char result.append(javaName.charAt(0)); char currChar; boolean previousCap = false; //loop through the string, looking for uppercase for (int i = 1; i < javaName.length(); i++) { currChar = javaName.charAt(i); //if uppcase append an underscore if (!previousCap && (currChar >= 'A') && (currChar <= 'Z')) { result.append("_"); } result.append(currChar); if ((currChar >= 'A') && (currChar <= 'Z')) { previousCap = true; } else { previousCap = false; } } //this is in upper-case return result.toString().toUpperCase(); } /** * see if two maps are the equivalent (based on number of entries, * and the equals() method of the keys and values) * @param * @param * @param first * @param second * @return true if equal */ public static boolean mapEquals(Map first, Map second) { Set keysMismatch = new HashSet(); mapDifferences(first, second, keysMismatch, null); //if any keys mismatch, then not equal return keysMismatch.size() == 0; } /** * empty map */ private static final Map EMPTY_MAP = Collections.unmodifiableMap(new HashMap()); /** * see if two maps are the equivalent (based on number of entries, * and the equals() method of the keys and values) * @param * @param * @param first map to check diffs * @param second map to check diffs * @param differences set of keys (with prefix) of the diffs * @param prefix for the entries in the diffs (e.g. "attribute__" */ @SuppressWarnings("unchecked") public static void mapDifferences(Map first, Map second, Set differences, String prefix) { if (first == second) { return; } //put the collections in new collections so we can remove and keep track if (first == null) { first = EMPTY_MAP; } if (second == null) { second = EMPTY_MAP; } else { //make linked so the results are ordered second = new LinkedHashMap(second); } int firstSize = first == null ? 0 : first.size(); int secondSize = second == null ? 0 : second.size(); //if both empty then all good if (firstSize == 0 && secondSize == 0) { return; } for (K key : first.keySet()) { if (second.containsKey(key)) { V firstValue = first.get(key); V secondValue = second.get(key); //keep track by removing from second second.remove(key); if (equals(firstValue, secondValue)) { continue; } } differences.add(isNotBlank(prefix) ? (K)(prefix + key) : key); } //add the ones left over in the second map which are not in the first map for (K key : second.keySet()) { differences.add(isNotBlank(prefix) ? (K)(prefix + key) : key); } } /** * sleep, if interrupted, throw runtime * @param millis */ public static void sleep(long millis) { try { Thread.sleep(millis); } catch (InterruptedException ie) { throw new RuntimeException(ie); } } /** * If we can, inject this into the exception, else return false * @param t * @param message * @return true if success, false if not */ public static boolean injectInException(Throwable t, String message) { String throwableFieldName = "detailMessage"; try { String currentValue = t.getMessage(); if (!isBlank(currentValue)) { currentValue += ",\n" + message; } else { currentValue = message; } assignField(t, throwableFieldName, currentValue); return true; } catch (Throwable t2) { //dont worry about what the problem is, return false so the caller can log return false; } } /** * get a unique string identifier based on the current time, * this is not globally unique, just unique for as long as this * server is running... * * @return String */ public static String uniqueId() { //this needs to be threadsafe since we are using a static field synchronized (GrouperInstallerUtils.class) { lastId = incrementStringInt(lastId); } return String.valueOf(lastId); } /** * get a file name from a resource name * * @param resourceName * is the classpath location * * @return the file path on the system */ public static File fileFromResourceName(String resourceName) { URL url = computeUrl(resourceName, true); if (url == null) { return null; } try { String fileName = URLDecoder.decode(url.getFile(), "UTF-8"); File configFile = new File(fileName); return configFile; } catch (UnsupportedEncodingException uee) { throw new RuntimeException(uee); } } /** * compute a url of a resource * @param resourceName * @param canBeNull if cant be null, throw runtime * @return the URL */ public static URL computeUrl(String resourceName, boolean canBeNull) { //get the url of the navigation file ClassLoader cl = classLoader(); URL url = null; try { //CH 20081012: sometimes it starts with slash and it shouldnt... String newResourceName = resourceName.startsWith("/") ? resourceName.substring(1) : resourceName; url = cl.getResource(newResourceName); } catch (NullPointerException npe) { String error = "computeUrl() Could not find resource file: " + resourceName; throw new RuntimeException(error, npe); } if (!canBeNull && url == null) { throw new RuntimeException("Cant find resource: " + resourceName); } return url; } /** * fast class loader * @return the class loader */ public static ClassLoader classLoader() { return GrouperInstallerUtils.class.getClassLoader(); } /** * make sure a array is non null. If null, then return an empty array. * Note: this will probably not work for primitive arrays (e.g. int[]) * @param * @param array * @param theClass to make array from * @return the list or empty list if null */ @SuppressWarnings("unchecked") public static T[] nonNull(T[] array, Class theClass) { return array == null ? ((T[])Array.newInstance(theClass, 0)) : array; } /** * get the prefix or suffix of a string based on a separator * * @param startString * is the string to start with * @param separator * is the separator to split on * @param isPrefix * if thre prefix or suffix should be returned * * @return the prefix or suffix, if the separator isnt there, return the * original string */ public static String prefixOrSuffix(String startString, String separator, boolean isPrefix) { String prefixOrSuffix = null; //no nulls if (startString == null) { return startString; } //where is the separator int separatorIndex = startString.indexOf(separator); //if none exists, dont proceed if (separatorIndex == -1) { return startString; } //maybe the separator isnt on character int separatorLength = separator.length(); if (isPrefix) { prefixOrSuffix = startString.substring(0, separatorIndex); } else { prefixOrSuffix = startString.substring(separatorIndex + separatorLength, startString.length()); } return prefixOrSuffix; } /** *
   * this method will indent xml or json.
   * this is for logging or documentations purposes only and should
   * not be used for a production use (since it is not 100% tested
   * or compliant with all constructs like xml CDATA
   * 
   * For xml, assumes elements either have text or sub elements, not both.
   * No cdata, nothing fancy.
   * 
   * If the input is <a><b><c>hey</c><d><e>there</e></d></b></a>
   * It would output:
   * <a>
   *   <b>
   *     <c>hey</c>
   *     <d>
   *       <e>there</e>
   *     </d>
   *   </b>
   * </a>
   * 
   * For json, if the input is: {"a":{"b\"b":{"c\\":"d"},"e":"f","g":["h":"i"]}}
   * It would output:
   * {
   *   "a":{
   *     "b\"b":{
   *       "c\\":"d"
   *     },
   *     "e":"f",
   *     "g":[
   *       "h":"i"
   *     ]
   *   }
   * }
   * 
   * 
   * 
   * @param string
   * @param failIfTypeNotFound
   * @return the indented string, 2 spaces per line
   */
  public static String indent(String string, boolean failIfTypeNotFound) {
    if (string == null) {
      return null;
    }
    string = trim(string);
    if (string.startsWith("<")) {
      //this is xml
      return new XmlIndenter(string).result();
    }
    if (!failIfTypeNotFound) {
      //just return if cant indent
      return string;
    }
    throw new RuntimeException("Cant find type of string: " + string);
    
    
  }

  /**
   * get the extension from name.  if name is a:b:c, name is c
   * @param name
   * @return the name
   */
  public static String extensionFromName(String name) {
    if (isBlank(name)) {
      return name;
    }
    int lastColonIndex = name.lastIndexOf(':');
    if (lastColonIndex == -1) {
      return name;
    }
    String extension = name.substring(lastColonIndex+1);
    return extension;
  }
  
  /**
   * 
Returns the class object.
* @param origClassName is fully qualified * @return the class */ public static Class forName(String origClassName) { try { return Class.forName(origClassName); } catch (Throwable t) { throw new RuntimeException("Problem loading class: " + origClassName, t); } } /** * Construct a class * @param template type * @param theClass * @return the instance */ public static T newInstance(Class theClass) { try { return theClass.newInstance(); } catch (Throwable e) { if (theClass != null && Modifier.isAbstract(theClass.getModifiers())) { throw new RuntimeException("Problem with class: " + theClass + ", maybe because it is abstract!", e); } throw new RuntimeException("Problem with class: " + theClass, e); } } /** * get the parent stem name from name. if already a root stem * then just return null. e.g. if the name is a:b:c then * the return value is a:b * @param name * @return the parent stem name or null if none */ public static String parentStemNameFromName(String name) { int lastColonIndex = name.lastIndexOf(':'); if (lastColonIndex == -1) { return null; } String parentStemName = name.substring(0,lastColonIndex); return parentStemName; } /** * return the string or the other if the first is blank * @param string * @param defaultStringIfBlank * @return the string or the default one */ public static String defaultIfBlank(String string, String defaultStringIfBlank) { return isBlank(string) ? defaultStringIfBlank : string; } /** * genericized method to see if first is null, if so then return second, else first. * @param * @param theValue first input * @param defaultIfTheValueIsNull second input * @return the first if not null, second if no */ public static T defaultIfNull(T theValue, T defaultIfTheValueIsNull) { return theValue != null ? theValue : defaultIfTheValueIsNull; } /** * add each element of listToAdd if it is not already in list * @param * @param list to add to * @param listToAdd each element will be added to list, or null if none */ public static void addIfNotThere(Collection list, Collection listToAdd) { //maybe nothing to do if (listToAdd == null) { return; } for (T t : listToAdd) { if (!list.contains(t)) { list.add(t); } } } /** * print out various types of objects * * @param object * @param maxChars is where it should stop when figuring out object. note, result might be longer than max... * need to abbreviate when back * @param result is where to append to */ @SuppressWarnings("unchecked") private static void toStringForLogHelper(Object object, int maxChars, StringBuilder result) { try { if (object == null) { result.append("null"); } else if (object.getClass().isArray()) { // handle arrays int length = Array.getLength(object); if (length == 0) { result.append("Empty array"); } else { result.append("Array size: ").append(length).append(": "); for (int i = 0; i < length; i++) { result.append("[").append(i).append("]: ").append( Array.get(object, i)).append("\n"); if (maxChars != -1 && result.length() > maxChars) { return; } } } } else if (object instanceof Collection) { //give size and type if collection Collection collection = (Collection) object; int collectionSize = collection.size(); if (collectionSize == 0) { result.append("Empty ").append(object.getClass().getSimpleName()); } else { result.append(object.getClass().getSimpleName()).append(" size: ").append(collectionSize).append(": "); int i=0; for (Object collectionObject : collection) { result.append("[").append(i).append("]: ").append( collectionObject).append("\n"); if (maxChars != -1 && result.length() > maxChars) { return; } i++; } } } else { result.append(object.toString()); } } catch (Exception e) { result.append("<> ").append(object.getClass()).append(":\n") .append(getFullStackTrace(e)).append("\n"); } } /** * convert a set to a string (comma separate) * @param set * @return the String */ public static String setToString(Set set) { if (set == null) { return "null"; } if (set.size() == 0) { return "empty"; } StringBuilder result = new StringBuilder(); boolean first = true; for (Object object : set) { if (!first) { result.append(", "); } first = false; result.append(object); } return result.toString(); } /** * convert a set to a string (comma separate) * @param map * @return the String * @deprecated use mapToString(map) */ @Deprecated public static String MapToString(Map map) { return mapToString(map); } /** * convert a set to a string (comma separate) * @param map * @return the String */ public static String mapToString(Map map) { if (map == null) { return "null"; } if (map.size() == 0) { return "empty"; } StringBuilder result = new StringBuilder(); boolean first = true; for (Object object : map.keySet()) { if (!first) { result.append(", "); } first = false; result.append(object).append(": ").append(map.get(object)); } return result.toString(); } /** * print out various types of objects * * @param object * @return the string value */ public static String toStringForLog(Object object) { StringBuilder result = new StringBuilder(); toStringForLogHelper(object, -1, result); return result.toString(); } /** * print out various types of objects * * @param object * @param maxChars is the max chars that should be returned (abbreviate if longer), or -1 for any amount * @return the string value */ public static String toStringForLog(Object object, int maxChars) { StringBuilder result = new StringBuilder(); toStringForLogHelper(object, -1, result); String resultString = result.toString(); if (maxChars != -1) { return abbreviate(resultString, maxChars); } return resultString; } /** * If batching this is the number of batches * @param count is size of set * @param batchSize * @return the number of batches */ public static int batchNumberOfBatches(int count, int batchSize) { int batches = 1 + ((count - 1) / batchSize); return batches; } /** * If batching this is the number of batches * @param collection * @param batchSize * @return the number of batches */ public static int batchNumberOfBatches(Collection collection, int batchSize) { int arrraySize = length(collection); return batchNumberOfBatches(arrraySize, batchSize); } /** * retrieve a batch by 0 index. Will return an array of size batchSize or * the remainder. the array will be full of elements. Note, this requires an * ordered input (so use linkedhashset not hashset if doing sets) * @param template type * @param collection * @param batchSize * @param batchIndex * @return the list * This never returns null, only empty list */ @SuppressWarnings("unchecked") public static List batchList(Collection collection, int batchSize, int batchIndex) { int numberOfBatches = batchNumberOfBatches(collection, batchSize); int arraySize = length(collection); // short circuit if (arraySize == 0) { return new ArrayList(); } List theBatchObjects = new ArrayList(); // lets get the type of the first element if possible // Object first = get(arrayOrCollection, 0); // // Class theType = first == null ? Object.class : first.getClass(); // if last batch if (batchIndex == numberOfBatches - 1) { // needs to work to 1-n //int thisBatchSize = 1 + ((arraySize - 1) % batchSize); int collectionIndex = 0; for (T t : collection) { if (collectionIndex++ < batchIndex * batchSize) { continue; } //just copy the rest //if (collectionIndex >= (batchIndex * batchSize) + arraySize) { // break; //} //we are in the copy mode theBatchObjects.add(t); } } else { // if non-last batch //int newIndex = 0; int collectionIndex = 0; for (T t : collection) { if (collectionIndex < batchIndex * batchSize) { collectionIndex++; continue; } //done with batch if (collectionIndex >= (batchIndex + 1) * batchSize) { break; } theBatchObjects.add(t); collectionIndex++; } } return theBatchObjects; } /** * trim the end of a string * @param text * @return the string */ public static String trimEnd(String text) { if (text == null) { return null; } //replace any whitespace at the end of the string return text.replaceFirst("\\s+$", ""); } /** * split a string based on a separator into an array, and trim each entry (see * the Commons Util trim() for more details) * * @param input * is the delimited input to split and trim * @param separator * is what to split on * * @return the array of items after split and trimmed, or null if input is null. will be trimmed to empty */ public static String[] splitTrim(String input, String separator) { return splitTrim(input, separator, true); } /** * split a string based on a separator into an array, and trim each entry (see * the Commons Util trim() for more details) * * @param input * is the delimited input to split and trim * @param separator * is what to split on * * @return the list of items after split and trimmed, or null if input is null. will be trimmed to empty */ public static List splitTrimToList(String input, String separator) { if (isBlank(input)) { return null; } String[] array = splitTrim(input, separator); return toList(array); } /** * split a string based on a separator into an array, and trim each entry (see * the Commons Util trim() for more details) * * @param input * is the delimited input to split and trim * @param separator * is what to split on * @param treatAdjacentSeparatorsAsOne * @return the array of items after split and trimmed, or null if input is null. will be trimmed to empty */ public static String[] splitTrim(String input, String separator, boolean treatAdjacentSeparatorsAsOne) { if (isBlank(input)) { return null; } //first split String[] items = treatAdjacentSeparatorsAsOne ? split(input, separator) : splitPreserveAllTokens(input, separator); //then trim for (int i = 0; (items != null) && (i < items.length); i++) { items[i] = trim(items[i]); } //return the array return items; } /** * escape url chars (e.g. a # is %23) * @param string input * @return the encoded string */ public static String escapeUrlEncode(String string) { String result = null; try { result = URLEncoder.encode(string, "UTF-8"); } catch (UnsupportedEncodingException ex) { throw new RuntimeException("UTF-8 not supported", ex); } return result; } /** * unescape url chars (e.g. a space is %20) * @param string input * @return the encoded string */ public static String escapeUrlDecode(String string) { String result = null; try { result = URLDecoder.decode(string, "UTF-8"); } catch (UnsupportedEncodingException ex) { throw new RuntimeException("UTF-8 not supported", ex); } return result; } /** * make sure a list is non null. If null, then return an empty list * @param * @param list * @return the list or empty list if null */ public static List nonNull(List list) { return list == null ? new ArrayList() : list; } /** * make sure a list is non null. If null, then return an empty set * @param * @param set * @return the set or empty set if null */ public static Set nonNull(Set set) { return set == null ? new HashSet() : set; } /** * make sure it is non null, if null, then give new map * * @param key of map * @param value of map * @param map is map * @return set non null */ public static Map nonNull(Map map) { return map == null ? new HashMap() : map; } /** * return a list of objects from varargs. Though if there is one * object, and it is a list, return it. * * @param * template type of the objects * @param objects * @return the list or null if objects is null */ @SuppressWarnings("unchecked") public static List toList(T... objects) { if (objects == null) { return null; } if (objects.length == 1 && objects[0] instanceof List) { return (List)objects[0]; } List result = new ArrayList(); for (T object : objects) { result.add(object); } return result; } /** * convert classes to a list * @param classes * @return list of classes */ public static List> toListClasses(Class... classes) { return toList(classes); } /** * return a set of objects from varargs. * * @param template type of the objects * @param objects * @return the set */ public static Set toSet(T... objects) { Set result = new LinkedHashSet(); for (T object : objects) { result.add(object); } return result; } /** * cache separator */ private static final String CACHE_SEPARATOR = "__"; /** * string format of dates */ public static final String DATE_FORMAT = "yyyyMMdd"; /** * format including minutes and seconds: yyyy/MM/dd HH:mm:ss */ public static final String DATE_MINUTES_SECONDS_FORMAT = "yyyy/MM/dd HH:mm:ss"; /** * format including minutes and seconds: yyyyMMdd HH:mm:ss */ public static final String DATE_MINUTES_SECONDS_NO_SLASH_FORMAT = "yyyyMMdd HH:mm:ss"; /** * format on screen of config for milestone: yyyy/MM/dd HH:mm:ss.SSS */ public static final String TIMESTAMP_FORMAT = "yyyy/MM/dd HH:mm:ss.SSS"; /** * format on screen of config for milestone: yyyyMMdd HH:mm:ss.SSS */ public static final String TIMESTAMP_NO_SLASH_FORMAT = "yyyyMMdd HH:mm:ss.SSS"; /** * date format, make sure to synchronize */ final static SimpleDateFormat dateFormat = new SimpleDateFormat(DATE_FORMAT); /** * synchronize code that uses this standard formatter for dates with minutes and seconds */ public final static SimpleDateFormat dateMinutesSecondsFormat = new SimpleDateFormat( DATE_MINUTES_SECONDS_FORMAT); /** * synchronize code that uses this standard formatter for dates with minutes and seconds */ final static SimpleDateFormat dateMinutesSecondsNoSlashFormat = new SimpleDateFormat( DATE_MINUTES_SECONDS_NO_SLASH_FORMAT); /** *
 format: yyyy/MM/dd HH:mm:ss.SSS synchronize code that uses this standard formatter for timestamps 
*/ final static SimpleDateFormat timestampFormat = new SimpleDateFormat(TIMESTAMP_FORMAT); /** * synchronize code that uses this standard formatter for timestamps */ final static SimpleDateFormat timestampNoSlashFormat = new SimpleDateFormat( TIMESTAMP_NO_SLASH_FORMAT); /** * If false, throw an assertException, and give a reason * * @param isTrue * @param reason */ public static void assertion(boolean isTrue, String reason) { if (!isTrue) { throw new RuntimeException(reason); } } /** * use the field cache, expire every day (just to be sure no leaks) */ private static ExpirableCache> fieldSetCache = null; /** * lazy load * @return field set cache */ private static ExpirableCache> fieldSetCache() { if (fieldSetCache == null) { fieldSetCache = new ExpirableCache>(60*24); } return fieldSetCache; } /** * make a cache with max size to cache declared methods */ private static ExpirableCache declaredMethodsCache = null; /** * lazy load * @return declared method cache */ private static ExpirableCache declaredMethodsCache() { if (declaredMethodsCache == null) { declaredMethodsCache = new ExpirableCache(60*24); } return declaredMethodsCache; } /** * use the field cache, expire every day (just to be sure no leaks) */ private static ExpirableCache> getterSetCache = null; /** * lazy load * @return getter cache */ private static ExpirableCache> getterSetCache() { if (getterSetCache == null) { getterSetCache = new ExpirableCache>(60*24); } return getterSetCache; } /** * use the field cache, expire every day (just to be sure no leaks) */ private static ExpirableCache> setterSetCache = null; /** * lazy load * @return setter cache */ private static ExpirableCache> setterSetCache() { if (setterSetCache == null) { setterSetCache = new ExpirableCache>(60*24); } return setterSetCache; } /** * Field lastId. */ private static char[] lastId = convertLongToStringSmall(new Date().getTime()) .toCharArray(); /** * cache the properties read from resource */ private static Map resourcePropertiesCache = new HashMap(); /** * assign data to a field * * @param theClass * the class which has the method * @param invokeOn * to call on or null for static * @param fieldName * method name to call * @param dataToAssign * data * @param callOnSupers * if static and method not exists, try on supers * @param overrideSecurity * true to call on protected or private etc methods * @param typeCast * true if we should typecast * @param annotationWithValueOverride * annotation with value of override */ public static void assignField(Class theClass, Object invokeOn, String fieldName, Object dataToAssign, boolean callOnSupers, boolean overrideSecurity, boolean typeCast, Class annotationWithValueOverride) { if (theClass == null && invokeOn != null) { theClass = invokeOn.getClass(); } Field field = field(theClass, fieldName, callOnSupers, true); assignField(field, invokeOn, dataToAssign, overrideSecurity, typeCast, annotationWithValueOverride); } /** * assign data to a field. Will find the field in superclasses, will * typecast, and will override security (private, protected, etc) * * @param theClass * the class which has the method * @param invokeOn * to call on or null for static * @param fieldName * method name to call * @param dataToAssign * data * @param annotationWithValueOverride * annotation with value of override */ public static void assignField(Class theClass, Object invokeOn, String fieldName, Object dataToAssign, Class annotationWithValueOverride) { assignField(theClass, invokeOn, fieldName, dataToAssign, true, true, true, annotationWithValueOverride); } /** * assign data to a field * * @param field * is the field to assign to * @param invokeOn * to call on or null for static * @param dataToAssign * data * @param overrideSecurity * true to call on protected or private etc methods * @param typeCast * true if we should typecast */ @SuppressWarnings("unchecked") public static void assignField(Field field, Object invokeOn, Object dataToAssign, boolean overrideSecurity, boolean typeCast) { try { Class fieldType = field.getType(); // typecast if (typeCast) { dataToAssign = typeCast(dataToAssign, fieldType, true, true); } if (overrideSecurity) { field.setAccessible(true); } field.set(invokeOn, dataToAssign); } catch (Exception e) { throw new RuntimeException("Cant assign reflection field: " + (field == null ? null : field.getName()) + ", on: " + className(invokeOn) + ", with args: " + classNameCollection(dataToAssign), e); } } /** * null safe iterator getter if the type if collection * * @param collection * @return the iterator */ public static Iterator iterator(Object collection) { if (collection == null) { return null; } // array list doesnt need an iterator if (collection instanceof Collection && !(collection instanceof ArrayList)) { return ((Collection) collection).iterator(); } return null; } /** * Null safe array length or map * * @param arrayOrCollection * @return the length of the array (0 for null) */ public static int length(Object arrayOrCollection) { if (arrayOrCollection == null) { return 0; } if (arrayOrCollection.getClass().isArray()) { return Array.getLength(arrayOrCollection); } if (arrayOrCollection instanceof Collection) { return ((Collection) arrayOrCollection).size(); } if (arrayOrCollection instanceof Map) { return ((Map) arrayOrCollection).size(); } // simple non array non collection object return 1; } /** * If array, get the element based on index, if Collection, get it based on * iterator. * * @param arrayOrCollection * @param iterator * @param index * @return the object */ public static Object next(Object arrayOrCollection, Iterator iterator, int index) { if (arrayOrCollection.getClass().isArray()) { return Array.get(arrayOrCollection, index); } if (arrayOrCollection instanceof ArrayList) { return ((ArrayList) arrayOrCollection).get(index); } if (arrayOrCollection instanceof Collection) { return iterator.next(); } // simple object if (0 == index) { return arrayOrCollection; } throw new RuntimeException("Invalid class type: " + arrayOrCollection.getClass().getName()); } /** * Remove the iterator or index * * @param arrayOrCollection * @param index * @return the object list or new array */ public static Object remove(Object arrayOrCollection, int index) { return remove(arrayOrCollection, null, index); } /** * Remove the iterator or index * * @param arrayOrCollection * @param iterator * @param index * @return the object list or new array */ public static Object remove(Object arrayOrCollection, Iterator iterator, int index) { //if theres an iterator, just use that if (iterator != null) { iterator.remove(); return arrayOrCollection; } if (arrayOrCollection.getClass().isArray()) { int newLength = Array.getLength(arrayOrCollection) - 1; Object newArray = Array.newInstance(arrayOrCollection.getClass().getComponentType(), newLength); if (newLength == 0) { return newArray; } if (index > 0) { System.arraycopy(arrayOrCollection, 0, newArray, 0, index); } if (index < newLength) { System.arraycopy(arrayOrCollection, index+1, newArray, index, newLength - index); } return newArray; } if (arrayOrCollection instanceof List) { ((List)arrayOrCollection).remove(index); return arrayOrCollection; } else if (arrayOrCollection instanceof Collection) { //this should work unless there are duplicates or something weird ((Collection)arrayOrCollection).remove(get(arrayOrCollection, index)); return arrayOrCollection; } throw new RuntimeException("Invalid class type: " + arrayOrCollection.getClass().getName()); } /** * print the simple names of a list of classes * @param object * @return the simple names */ public static String classesString(Object object) { StringBuilder result = new StringBuilder(); if (object.getClass().isArray()) { int length = Array.getLength(object); for (int i=0;i annotationWithValueOverride) { if (annotationWithValueOverride != null) { // see if in annotation Annotation annotation = field .getAnnotation(annotationWithValueOverride); if (annotation != null) { // type of the value, or String if not specific Class // typeOfAnnotationValue = typeCast ? field.getType() : // String.class; dataToAssign = // AnnotationUtils.retrieveAnnotationValue( // typeOfAnnotationValue, annotation, "value"); throw new RuntimeException("Not supported"); } } assignField(field, invokeOn, dataToAssign, overrideSecurity, typeCast); } /** * assign data to a field. Will find the field in superclasses, will * typecast, and will override security (private, protected, etc) * * @param invokeOn * to call on or null for static * @param fieldName * method name to call * @param dataToAssign * data */ public static void assignField(Object invokeOn, String fieldName, Object dataToAssign) { assignField(null, invokeOn, fieldName, dataToAssign, true, true, true, null); } /** * get a field object for a class, potentially in superclasses * * @param theClass * @param fieldName * @param callOnSupers * true if superclasses should be looked in for the field * @param throwExceptionIfNotFound * will throw runtime exception if not found * @return the field object or null if not found (or exception if param is * set) */ public static Field field(Class theClass, String fieldName, boolean callOnSupers, boolean throwExceptionIfNotFound) { try { Field field = theClass.getDeclaredField(fieldName); // found it return field; } catch (NoSuchFieldException e) { // if method not found // if traversing up, and not Object, and not instance method if (callOnSupers && !theClass.equals(Object.class)) { return field(theClass.getSuperclass(), fieldName, callOnSupers, throwExceptionIfNotFound); } } // maybe throw an exception if (throwExceptionIfNotFound) { throw new RuntimeException("Cant find field: " + fieldName + ", in: " + theClass + ", callOnSupers: " + callOnSupers); } return null; } /** * return a set of Strings for a class and type. This is not for any * supertypes, only for the type at hand. includes final fields * * @param theClass * @param fieldType * or null for all * @param includeStaticFields * @return the set of strings, or the empty Set if none */ @SuppressWarnings("unchecked") public static Set fieldNames(Class theClass, Class fieldType, boolean includeStaticFields) { return fieldNamesHelper(theClass, theClass, fieldType, true, true, includeStaticFields, null, true); } /** * get all field names from a class, including superclasses (if specified) * * @param theClass * to look for fields in * @param superclassToStopAt * to go up to or null to go up to Object * @param fieldType * is the type of the field to get * @param includeSuperclassToStopAt * if we should include the superclass * @param includeStaticFields * if include static fields * @param includeFinalFields * if final fields should be included * @return the set of field names or empty set if none */ public static Set fieldNames(Class theClass, Class superclassToStopAt, Class fieldType, boolean includeSuperclassToStopAt, boolean includeStaticFields, boolean includeFinalFields) { return fieldNamesHelper(theClass, superclassToStopAt, fieldType, includeSuperclassToStopAt, includeStaticFields, includeFinalFields, null, true); } /** * get all field names from a class, including superclasses (if specified). * ignore a certain marker annotation * * @param theClass * to look for fields in * @param superclassToStopAt * to go up to or null to go up to Object * @param fieldType * is the type of the field to get * @param includeSuperclassToStopAt * if we should include the superclass * @param includeStaticFields * if include static fields * @param includeFinalFields * if final fields should be included * @param markerAnnotationToIngore * if this is not null, then if the field has this annotation, * then do not include in list * @return the set of field names */ public static Set fieldNames(Class theClass, Class superclassToStopAt, Class fieldType, boolean includeSuperclassToStopAt, boolean includeStaticFields, boolean includeFinalFields, Class markerAnnotationToIngore) { return fieldNamesHelper(theClass, superclassToStopAt, fieldType, includeSuperclassToStopAt, includeStaticFields, includeFinalFields, markerAnnotationToIngore, false); } /** * get all field names from a class, including superclasses (if specified) * (up to and including the specified superclass). ignore a certain marker * annotation. Dont get static or final field, and get fields of all types * * @param theClass * to look for fields in * @param superclassToStopAt * to go up to or null to go up to Object * @param markerAnnotationToIngore * if this is not null, then if the field has this annotation, * then do not include in list * @return the set of field names or empty set if none */ public static Set fieldNames(Class theClass, Class superclassToStopAt, Class markerAnnotationToIngore) { return fieldNamesHelper(theClass, superclassToStopAt, null, true, false, false, markerAnnotationToIngore, false); } /** * get all field names from a class, including superclasses (if specified) * * @param theClass * to look for fields in * @param superclassToStopAt * to go up to or null to go up to Object * @param fieldType * is the type of the field to get * @param includeSuperclassToStopAt * if we should include the superclass * @param includeStaticFields * if include static fields * @param includeFinalFields * true to include finals * @param markerAnnotation * if this is not null, then if the field has this annotation, * then do not include in list (if includeAnnotation is false) * @param includeAnnotation * true if the attribute should be included if annotation is * present, false if exclude * @return the set of field names or empty set if none */ @SuppressWarnings("unchecked") static Set fieldNamesHelper(Class theClass, Class superclassToStopAt, Class fieldType, boolean includeSuperclassToStopAt, boolean includeStaticFields, boolean includeFinalFields, Class markerAnnotation, boolean includeAnnotation) { Set fieldSet = fieldsHelper(theClass, superclassToStopAt, fieldType, includeSuperclassToStopAt, includeStaticFields, includeFinalFields, markerAnnotation, includeAnnotation); Set fieldNameSet = new LinkedHashSet(); for (Field field : fieldSet) { fieldNameSet.add(field.getName()); } return fieldNameSet; } /** * get all fields from a class, including superclasses (if specified) * * @param theClass * to look for fields in * @param superclassToStopAt * to go up to or null to go up to Object * @param fieldType * is the type of the field to get * @param includeSuperclassToStopAt * if we should include the superclass * @param includeStaticFields * if include static fields * @param includeFinalFields * if final fields should be included * @param markerAnnotation * if this is not null, then if the field has this annotation, * then do not include in list (if includeAnnotation is false) * @param includeAnnotation * true if the attribute should be included if annotation is * present, false if exclude * @return the set of fields (wont return null) */ @SuppressWarnings("unchecked") public static Set fields(Class theClass, Class superclassToStopAt, Class fieldType, boolean includeSuperclassToStopAt, boolean includeStaticFields, boolean includeFinalFields, Class markerAnnotation, boolean includeAnnotation) { return fieldsHelper(theClass, superclassToStopAt, fieldType, includeSuperclassToStopAt, includeStaticFields, includeFinalFields, markerAnnotation, includeAnnotation); } /** * get all fields from a class, including superclasses (if specified) (up to * and including the specified superclass). ignore a certain marker * annotation, or only include it. Dont get static or final field, and get * fields of all types * * @param theClass * to look for fields in * @param superclassToStopAt * to go up to or null to go up to Object * @param markerAnnotation * if this is not null, then if the field has this annotation, * then do not include in list (if includeAnnotation is false) * @param includeAnnotation * true if the attribute should be included if annotation is * present, false if exclude * @return the set of field names or empty set if none */ @SuppressWarnings("unchecked") public static Set fields(Class theClass, Class superclassToStopAt, Class markerAnnotation, boolean includeAnnotation) { return fieldsHelper(theClass, superclassToStopAt, null, true, false, false, markerAnnotation, includeAnnotation); } /** * get all fields from a class, including superclasses (if specified) * * @param theClass * to look for fields in * @param superclassToStopAt * to go up to or null to go up to Object * @param fieldType * is the type of the field to get * @param includeSuperclassToStopAt * if we should include the superclass * @param includeStaticFields * if include static fields * @param includeFinalFields * if final fields should be included * @param markerAnnotation * if this is not null, then if the field has this annotation, * then do not include in list (if includeAnnotation is false) * @param includeAnnotation * true if the attribute should be included if annotation is * present, false if exclude * @return the set of fields (wont return null) */ @SuppressWarnings("unchecked") static Set fieldsHelper(Class theClass, Class superclassToStopAt, Class fieldType, boolean includeSuperclassToStopAt, boolean includeStaticFields, boolean includeFinalFields, Class markerAnnotation, boolean includeAnnotation) { // MAKE SURE IF ANY MORE PARAMS ARE ADDED, THE CACHE KEY IS CHANGED! Set fieldNameSet = null; String cacheKey = theClass + CACHE_SEPARATOR + superclassToStopAt + CACHE_SEPARATOR + fieldType + CACHE_SEPARATOR + includeSuperclassToStopAt + CACHE_SEPARATOR + includeStaticFields + CACHE_SEPARATOR + includeFinalFields + CACHE_SEPARATOR + markerAnnotation + CACHE_SEPARATOR + includeAnnotation; fieldNameSet = fieldSetCache().get(cacheKey); if (fieldNameSet != null) { return fieldNameSet; } fieldNameSet = new LinkedHashSet(); fieldsHelper(theClass, superclassToStopAt, fieldType, includeSuperclassToStopAt, includeStaticFields, includeFinalFields, markerAnnotation, fieldNameSet, includeAnnotation); // add to cache fieldSetCache().put(cacheKey, fieldNameSet); return fieldNameSet; } /** * compare two objects, compare primitives, Strings, maps of string attributes. * if both objects equal each others references, then return empty set. * then, if not, then if either is null, return all fields * @param first * @param second * @param fieldsToCompare * @param mapPrefix is the prefix for maps which are compared (e.g. attribute__) * @return the set of fields which are different. never returns null */ @SuppressWarnings("unchecked") public static Set compareObjectFields(Object first, Object second, Set fieldsToCompare, String mapPrefix) { Set differentFields = new LinkedHashSet(); if (first == second) { return differentFields; } //if either null, then all fields are different if (first == null || second == null) { differentFields.addAll(fieldsToCompare); } for (String fieldName : fieldsToCompare) { try { Object firstValue = fieldValue(first, fieldName); Object secondValue = fieldValue(second, fieldName); if (firstValue == secondValue) { continue; } if (firstValue instanceof Map || secondValue instanceof Map) { mapDifferences((Map)firstValue, (Map)secondValue, differentFields, mapPrefix); continue; } //compare things... //for strings, null is equal to empty if (firstValue instanceof String || secondValue instanceof String) { if (!equals(defaultString((String)firstValue), defaultString((String)secondValue))) { differentFields.add(fieldName); } continue; } //if one is null, that is not good if (firstValue == null || secondValue == null) { differentFields.add(fieldName); continue; } //everything (numbers, dates, etc) should work with equals method... if (!firstValue.equals(secondValue)) { differentFields.add(fieldName); continue; } } catch (RuntimeException re) { throw new RuntimeException("Problem comparing field " + fieldName + " on objects: " + className(first) + ", " + className(second)); } } return differentFields; } /** * clone an object, assign primitives, Strings, maps of string attributes. Clone GrouperCloneable fields. * @param template * @param object * @param fieldsToClone * @return the cloned object or null if input is null */ @SuppressWarnings("unchecked") public static T clone(T object, Set fieldsToClone) { //make a return object T result = (T)newInstance(object.getClass()); cloneFields(object, result, fieldsToClone); return result; } /** * clone an object, assign primitives, Strings, maps of string attributes. Clone GrouperCloneable fields. * @param template * @param object * @param result * @param fieldsToClone */ public static void cloneFields(T object, T result, Set fieldsToClone) { if (object == result) { return; } //if either null, then all fields are different if (object == null || result == null) { throw new RuntimeException("Cant copy from or to null: " + className(object) + ", " + className(result)); } Class fieldValueClass = null; for (String fieldName : nonNull(fieldsToClone)) { try { Object fieldValue = fieldValue(object, fieldName); fieldValueClass = fieldValue == null ? null : fieldValue.getClass(); Object fieldValueToAssign = cloneValue(fieldValue); //assign the field to the clone assignField(result, fieldName, fieldValueToAssign); } catch (RuntimeException re) { throw new RuntimeException("Problem cloning field: " + object.getClass() + ", " + fieldName + ", " + fieldValueClass, re); } } } /** * helper method to clone the value of a field. just returns the same * reference for primitives and immutables. Will subclone GrouperCloneables, * and will throw exception if not expecting the type. Will clone sets, lists, maps. * @param template * * @param value * @return the cloned value */ @SuppressWarnings("unchecked") public static T cloneValue(T value) { Object clonedValue = value; if (value == null || value instanceof String || value.getClass().isPrimitive() || value instanceof Number || value instanceof Boolean || value instanceof Date) { //clone things //for strings, and immutable classes, just assign //nothing to do, just assign the value } else if (value instanceof Map) { clonedValue = new LinkedHashMap(); Map mapValue = (Map)value; Map clonedMapValue = (Map)clonedValue; for (Object key : mapValue.keySet()) { clonedMapValue.put(cloneValue(key), cloneValue(mapValue.get(key))); } } else if (value instanceof Set) { clonedValue = new LinkedHashSet(); Set setValue = (Set)value; Set clonedSetValue = (Set)clonedValue; for (Object each : setValue) { clonedSetValue.add(cloneValue(each)); } } else if (value instanceof List) { clonedValue = new ArrayList(); List listValue = (List)value; List clonedListValue = (List)clonedValue; for (Object each : listValue) { clonedListValue.add(cloneValue(each)); } } else if (value.getClass().isArray()) { clonedValue = Array.newInstance(value.getClass().getComponentType(), Array.getLength(value)); for (int i=0;i methodNames(Class theClass, Class superclassToStopAt, boolean includeSuperclassToStopAt, boolean includeStaticMethods) { Set methods = new LinkedHashSet(); methodsHelper(theClass, superclassToStopAt, includeSuperclassToStopAt, includeStaticMethods, null, false, methods); Set methodNames = new HashSet(); for (Method method : methods) { methodNames.add(method.getName()); } return methodNames; } /** * get the set of methods * @param theClass * @param superclassToStopAt * @param includeSuperclassToStopAt * @param includeStaticMethods * @param markerAnnotation * @param includeAnnotation * @param methodSet */ public static void methodsHelper(Class theClass, Class superclassToStopAt, boolean includeSuperclassToStopAt, boolean includeStaticMethods, Class markerAnnotation, boolean includeAnnotation, Set methodSet) { Method[] methods = theClass.getDeclaredMethods(); if (length(methods) != 0) { for (Method method : methods) { // if not static, then continue if (!includeStaticMethods && Modifier.isStatic(method.getModifiers())) { continue; } // if checking for annotation if (markerAnnotation != null && (includeAnnotation != method .isAnnotationPresent(markerAnnotation))) { continue; } // go for it methodSet.add(method); } } // see if done recursing (if superclassToStopAt is null, then stop at // Object if (theClass.equals(superclassToStopAt) || theClass.equals(Object.class)) { return; } Class superclass = theClass.getSuperclass(); if (!includeSuperclassToStopAt && superclass.equals(superclassToStopAt)) { return; } // recurse methodsHelper(superclass, superclassToStopAt, includeSuperclassToStopAt, includeStaticMethods, markerAnnotation, includeAnnotation, methodSet); } /** * get the set of methods * @param theClass * @param methodName * @param paramTypesOrArrayOrList * types of the params * @param superclassToStopAt * @param includeSuperclassToStopAt * @param isStaticOrInstance true if static * @param markerAnnotation * @param includeAnnotation * @return the method or null if not found * */ public static Method method(Class theClass, String methodName, Object paramTypesOrArrayOrList, Class superclassToStopAt, boolean includeSuperclassToStopAt, boolean isStaticOrInstance, Class markerAnnotation, boolean includeAnnotation) { Class[] paramTypesArray = (Class[]) toArray(paramTypesOrArrayOrList); Method method = null; try { method = theClass.getDeclaredMethod(methodName, paramTypesArray); } catch (NoSuchMethodException nsme) { //this is ok } catch (Exception e) { throw new RuntimeException("Problem retrieving method: " + theClass.getSimpleName() + ", " + methodName, e); } if (method != null) { //we found a method, make sure it is valid // if not static, then return null (dont worry about superclass) if (!isStaticOrInstance && Modifier.isStatic(method.getModifiers())) { return null; } // if checking for annotation, if not there, then recurse if (markerAnnotation == null || (includeAnnotation == method .isAnnotationPresent(markerAnnotation))) { return method; } } // see if done recursing (if superclassToStopAt is null, then stop at // Object if (theClass.equals(superclassToStopAt) || theClass.equals(Object.class)) { return null; } Class superclass = theClass.getSuperclass(); if (!includeSuperclassToStopAt && superclass.equals(superclassToStopAt)) { return null; } // recurse return method(superclass, methodName, paramTypesArray, superclassToStopAt, includeSuperclassToStopAt, isStaticOrInstance, markerAnnotation, includeAnnotation); } /** * get all field names from a class, including superclasses (if specified) * * @param theClass * to look for fields in * @param superclassToStopAt * to go up to or null to go up to Object * @param fieldType * is the type of the field to get * @param includeSuperclassToStopAt * if we should include the superclass * @param includeStaticFields * if include static fields * @param includeFinalFields * if final fields should be included * @param markerAnnotation * if this is not null, then if the field has this annotation, * then do not include in list * @param fieldSet * set to add fields to * @param includeAnnotation * if include or exclude */ @SuppressWarnings("unchecked") private static void fieldsHelper(Class theClass, Class superclassToStopAt, Class fieldType, boolean includeSuperclassToStopAt, boolean includeStaticFields, boolean includeFinalFields, Class markerAnnotation, Set fieldSet, boolean includeAnnotation) { Field[] fields = theClass.getDeclaredFields(); if (length(fields) != 0) { for (Field field : fields) { // if checking for type, and not right type, continue if (fieldType != null && !fieldType.isAssignableFrom(field.getType())) { continue; } // if not static, then continue if (!includeStaticFields && Modifier.isStatic(field.getModifiers())) { continue; } // if not final constinue if (!includeFinalFields && Modifier.isFinal(field.getModifiers())) { continue; } // if checking for annotation if (markerAnnotation != null && (includeAnnotation != field .isAnnotationPresent(markerAnnotation))) { continue; } // go for it fieldSet.add(field); } } // see if done recursing (if superclassToStopAt is null, then stop at // Object if (theClass.equals(superclassToStopAt) || theClass.equals(Object.class)) { return; } Class superclass = theClass.getSuperclass(); if (!includeSuperclassToStopAt && superclass.equals(superclassToStopAt)) { return; } // recurse fieldsHelper(superclass, superclassToStopAt, fieldType, includeSuperclassToStopAt, includeStaticFields, includeFinalFields, markerAnnotation, fieldSet, includeAnnotation); } /** * find out a field value * * @param theClass * the class which has the method * @param invokeOn * to call on or null for static * @param fieldName * method name to call * @param callOnSupers * if static and method not exists, try on supers * @param overrideSecurity * true to call on protected or private etc methods * @return the current value */ public static Object fieldValue(Class theClass, Object invokeOn, String fieldName, boolean callOnSupers, boolean overrideSecurity) { Field field = null; // only if the method exists, try to execute try { // ok if null if (theClass == null) { theClass = invokeOn.getClass(); } field = field(theClass, fieldName, callOnSupers, true); return fieldValue(field, invokeOn, overrideSecurity); } catch (Exception e) { throw new RuntimeException("Cant execute reflection field: " + fieldName + ", on: " + className(invokeOn), e); } } /** * get the value of a field, override security if needbe * * @param field * @param invokeOn * @return the value of the field */ public static Object fieldValue(Field field, Object invokeOn) { return fieldValue(field, invokeOn, true); } /** * get the value of a field * * @param field * @param invokeOn * @param overrideSecurity * @return the value of the field */ public static Object fieldValue(Field field, Object invokeOn, boolean overrideSecurity) { if (overrideSecurity) { field.setAccessible(true); } try { return field.get(invokeOn); } catch (Exception e) { throw new RuntimeException("Cant execute reflection field: " + field.getName() + ", on: " + className(invokeOn), e); } } /** * find out a field value (invoke on supers, override security) * * @param invokeOn * to call on or null for static * @param fieldName * method name to call * @return the current value */ public static Object fieldValue(Object invokeOn, String fieldName) { return fieldValue(null, invokeOn, fieldName, true, true); } /** * get the decalred methods for a class, perhaps from cache * * @param theClass * @return the declared methods */ @SuppressWarnings("unused") private static Method[] retrieveDeclaredMethods(Class theClass) { Method[] methods = declaredMethodsCache().get(theClass); // get from cache if we can if (methods == null) { methods = theClass.getDeclaredMethods(); declaredMethodsCache().put(theClass, methods); } return methods; } /** * helper method for calling a method with no params (could be in * superclass) * * @param theClass * the class which has the method * @param invokeOn * to call on or null for static * @param methodName * method name to call * @return the data */ public static Object callMethod(Class theClass, Object invokeOn, String methodName) { return callMethod(theClass, invokeOn, methodName, null, null); } /** * helper method for calling a method (could be in superclass) * * @param theClass * the class which has the method * @param invokeOn * to call on or null for static * @param methodName * method name to call * @param paramTypesOrArrayOrList * types of the params * @param paramsOrListOrArray * data * @return the data */ public static Object callMethod(Class theClass, Object invokeOn, String methodName, Object paramTypesOrArrayOrList, Object paramsOrListOrArray) { return callMethod(theClass, invokeOn, methodName, paramTypesOrArrayOrList, paramsOrListOrArray, true); } /** * helper method for calling a method * * @param theClass * the class which has the method * @param invokeOn * to call on or null for static * @param methodName * method name to call * @param paramTypesOrArrayOrList * types of the params * @param paramsOrListOrArray * data * @param callOnSupers * if static and method not exists, try on supers * @return the data */ public static Object callMethod(Class theClass, Object invokeOn, String methodName, Object paramTypesOrArrayOrList, Object paramsOrListOrArray, boolean callOnSupers) { return callMethod(theClass, invokeOn, methodName, paramTypesOrArrayOrList, paramsOrListOrArray, callOnSupers, false); } /** * construct an instance by reflection * @param * @param theClass * @param args * @param types * @return the instance */ public static T construct(Class theClass, Class[] types, Object[] args) { try { Constructor constructor = theClass.getConstructor(types); return constructor.newInstance(args); } catch (Exception e) { throw new RuntimeException("Having trouble with constructor for class: " + theClass.getSimpleName() + " and args: " + classesString(types), e); } } /** * helper method for calling a method * * @param theClass * the class which has the method * @param invokeOn * to call on or null for static * @param methodName * method name to call * @param paramTypesOrArrayOrList * types of the params * @param paramsOrListOrArray * data * @param callOnSupers * if static and method not exists, try on supers * @param overrideSecurity * true to call on protected or private etc methods * @return the data */ public static Object callMethod(Class theClass, Object invokeOn, String methodName, Object paramTypesOrArrayOrList, Object paramsOrListOrArray, boolean callOnSupers, boolean overrideSecurity) { Method method = null; Class[] paramTypesArray = (Class[]) toArray(paramTypesOrArrayOrList); try { method = theClass.getDeclaredMethod(methodName, paramTypesArray); if (overrideSecurity) { method.setAccessible(true); } } catch (Exception e) { // if method not found if (e instanceof NoSuchMethodException) { // if traversing up, and not Object, and not instance method // CH 070425 not sure why invokeOn needs to be null, removing // this if (callOnSupers /* && invokeOn == null */ && !theClass.equals(Object.class)) { return callMethod(theClass.getSuperclass(), invokeOn, methodName, paramTypesOrArrayOrList, paramsOrListOrArray, callOnSupers, overrideSecurity); } } throw new RuntimeException("Problem calling method " + methodName + " on " + theClass.getName(), e); } return invokeMethod(method, invokeOn, paramsOrListOrArray); } /** pass this in the invokeOn to signify no params */ private static final Object NO_PARAMS = new Object(); /** * Safely invoke a reflection method that takes no args * * @param method * to invoke * @param invokeOn * if NO_PARAMS then will not pass in params. * @return the result */ public static Object invokeMethod(Method method, Object invokeOn) { return invokeMethod(method, invokeOn, NO_PARAMS); } /** * Safely invoke a reflection method * * @param method * to invoke * @param invokeOn * @param paramsOrListOrArray must be an arg. If null, will pass null. * if NO_PARAMS then will not pass in params. * @return the result */ public static Object invokeMethod(Method method, Object invokeOn, Object paramsOrListOrArray) { Object[] args = paramsOrListOrArray == NO_PARAMS ? null : (Object[]) toArray(paramsOrListOrArray); //we want to make sure things are accessible method.setAccessible(true); //only if the method exists, try to execute Object result = null; Exception e = null; try { result = method.invoke(invokeOn, args); } catch (IllegalAccessException iae) { e = iae; } catch (IllegalArgumentException iae) { e = iae; } catch (InvocationTargetException ite) { //this means the underlying call caused exception... its ok if runtime if (ite.getCause() instanceof RuntimeException) { throw (RuntimeException)ite.getCause(); } //else throw as invocation target... e = ite; } if (e != null) { throw new RuntimeException("Cant execute reflection method: " + method.getName() + ", on: " + className(invokeOn) + ", with args: " + classNameCollection(args), e); } return result; } /** * Convert a list to an array with the type of the first element e.g. if it * is a list of Person objects, then the array is Person[] * * @param objectOrArrayOrCollection * is a list * @return the array of objects with type of the first element in the list */ public static Object toArray(Object objectOrArrayOrCollection) { // do this before length since if array with null in it, we want ti get // it back if (objectOrArrayOrCollection != null && objectOrArrayOrCollection.getClass().isArray()) { return objectOrArrayOrCollection; } int length = length(objectOrArrayOrCollection); if (length == 0) { return null; } if (objectOrArrayOrCollection instanceof Collection) { Collection collection = (Collection) objectOrArrayOrCollection; Object first = collection.iterator().next(); return toArray(collection, (Class)(first == null ? Object.class : first .getClass())); } // make an array of the type of object passed in, size one Object array = Array.newInstance(objectOrArrayOrCollection.getClass(), 1); Array.set(array, 0, objectOrArrayOrCollection); return array; } /** * convert a list into an array of type of theClass * @param is the type of the array * @param collection list to convert * @param theClass type of array to return * @return array of type theClass[] filled with the objects from list */ @SuppressWarnings("unchecked") public static T[] toArray(Collection collection, Class theClass) { if (collection == null || collection.size() == 0) { return null; } return (T[])collection.toArray((Object[]) Array.newInstance(theClass, collection.size())); } /** * helper method for calling a static method up the stack. method takes no * args (could be in superclass) * * @param theClass * the class which has the method * @param methodName * method name to call * @return the data */ public static Object callMethod(Class theClass, String methodName) { return callMethod(theClass, null, methodName, null, null); } /** * helper method for calling a static method with no params * * @param theClass * the class which has the method * @param methodName * method name to call * @param callOnSupers * if we should try the super classes if not exists in this class * @return the data */ public static Object callMethod(Class theClass, String methodName, boolean callOnSupers) { return callMethod(theClass, null, methodName, null, null, callOnSupers); } /** * helper method for calling a static method up the stack * * @param theClass * the class which has the method * @param methodName * method name to call * @param paramTypesOrArrayOrList * types of the params * @param paramsOrListOrArray * data * @return the data */ public static Object callMethod(Class theClass, String methodName, Object paramTypesOrArrayOrList, Object paramsOrListOrArray) { return callMethod(theClass, null, methodName, paramTypesOrArrayOrList, paramsOrListOrArray); } /** * helper method for calling a method with no params (could be in * superclass), will override security * * @param invokeOn * instance to invoke on * @param methodName * method name to call not exists in this class * @return the data */ public static Object callMethod(Object invokeOn, String methodName) { if (invokeOn == null) { throw new NullPointerException("invokeOn is null: " + methodName); } return callMethod(invokeOn.getClass(), invokeOn, methodName, null, null, true, true); } /** * replace a string or strings from a string, and put the output in a string * buffer. This does not recurse * * @param text * string to look in * @param searchFor * string array to search for * @param replaceWith * string array to replace with * @return the string */ public static String replace(String text, Object searchFor, Object replaceWith) { return replace(null, null, text, searchFor, replaceWith, false, 0, false); } /** * replace a string or strings from a string, and put the output in a string * buffer * * @param text * string to look in * @param searchFor * string array to search for * @param replaceWith * string array to replace with * @param recurse * if true then do multiple replaces (on the replacements) * @return the string */ public static String replace(String text, Object searchFor, Object replaceWith, boolean recurse) { return replace(null, null, text, searchFor, replaceWith, recurse, recurse ? length(searchFor) : 0, false); } /** * replace a string or strings from a string, and put the output in a string * buffer * * @param text * string to look in * @param searchFor * string array to search for * @param replaceWith * string array to replace with * @param recurse * if true then do multiple replaces (on the replacements) * @param removeIfFound * true if removing from searchFor and replaceWith if found * @return the string */ public static String replace(String text, Object searchFor, Object replaceWith, boolean recurse, boolean removeIfFound) { return replace(null, null, text, searchFor, replaceWith, recurse, recurse ? length(searchFor) : 0, removeIfFound); } /** *

* Replaces all occurrences of a String within another String. *

* *

* A null reference passed to this method is a no-op. *

* *
   * replace(null, *, *)        = null
   * replace("", *, *)          = ""
   * replace("any", null, *)    = "any"
   * replace("any", *, null)    = "any"
   * replace("any", "", *)      = "any"
   * replace("aba", "a", null)  = "aba"
   * replace("aba", "a", "")    = "b"
   * replace("aba", "a", "z")   = "zbz"
   * 
* * @see #replace(String text, String repl, String with, int max) * @param text * text to search and replace in, may be null * @param repl * the String to search for, may be null * @param with * the String to replace with, may be null * @return the text with any replacements processed, null if * null String input */ public static String replace(String text, String repl, String with) { return replace(text, repl, with, -1); } /** *

* Replaces a String with another String inside a larger String, for the * first max values of the search String. *

* *

* A null reference passed to this method is a no-op. *

* *
   * replace(null, *, *, *)         = null
   * replace("", *, *, *)           = ""
   * replace("any", null, *, *)     = "any"
   * replace("any", *, null, *)     = "any"
   * replace("any", "", *, *)       = "any"
   * replace("any", *, *, 0)        = "any"
   * replace("abaa", "a", null, -1) = "abaa"
   * replace("abaa", "a", "", -1)   = "b"
   * replace("abaa", "a", "z", 0)   = "abaa"
   * replace("abaa", "a", "z", 1)   = "zbaa"
   * replace("abaa", "a", "z", 2)   = "zbza"
   * replace("abaa", "a", "z", -1)  = "zbzz"
   * 
* * @param text * text to search and replace in, may be null * @param repl * the String to search for, may be null * @param with * the String to replace with, may be null * @param max * maximum number of values to replace, or -1 if * no maximum * @return the text with any replacements processed, null if * null String input */ public static String replace(String text, String repl, String with, int max) { if (text == null || isEmpty(repl) || with == null || max == 0) { return text; } StringBuffer buf = new StringBuffer(text.length()); int start = 0, end = 0; while ((end = text.indexOf(repl, start)) != -1) { buf.append(text.substring(start, end)).append(with); start = end + repl.length(); if (--max == 0) { break; } } buf.append(text.substring(start)); return buf.toString(); } /** *

* Checks if a String is empty ("") or null. *

* *
   * isEmpty(null)      = true
   * isEmpty("")        = true
   * isEmpty(" ")       = false
   * isEmpty("bob")     = false
   * isEmpty("  bob  ") = false
   * 
* *

* NOTE: This method changed in Lang version 2.0. It no longer trims the * String. That functionality is available in isBlank(). *

* * @param str * the String to check, may be null * @return true if the String is empty or null */ public static boolean isEmpty(String str) { return str == null || str.length() == 0; } /** * replace a string or strings from a string, and put the output in a string * buffer. This does not recurse * * @param outBuffer * stringbuffer to write to * @param text * string to look in * @param searchFor * string array to search for * @param replaceWith * string array to replace with */ public static void replace(StringBuffer outBuffer, String text, Object searchFor, Object replaceWith) { replace(outBuffer, null, text, searchFor, replaceWith, false, 0, false); } /** * replace a string or strings from a string, and put the output in a string * buffer * * @param outBuffer * stringbuffer to write to * @param text * string to look in * @param searchFor * string array to search for * @param replaceWith * string array to replace with * @param recurse * if true then do multiple replaces (on the replacements) */ public static void replace(StringBuffer outBuffer, String text, Object searchFor, Object replaceWith, boolean recurse) { replace(outBuffer, null, text, searchFor, replaceWith, recurse, recurse ? length(searchFor) : 0, false); } /** * replace a string with other strings, and either write to outWriter, or * StringBuffer, and if StringBuffer potentially return a string. If * outBuffer and outWriter are null, then return the String * * @param outBuffer * stringbuffer to write to, or null to not * @param outWriter * Writer to write to, or null to not. * @param text * string to look in * @param searchFor * string array to search for, or string, or list * @param replaceWith * string array to replace with, or string, or list * @param recurse * if true then do multiple replaces (on the replacements) * @param timeToLive * if recursing, prevent endless loops * @param removeIfFound * true if removing from searchFor and replaceWith if found * @return the String if outBuffer and outWriter are null * @throws IndexOutOfBoundsException * if the lengths of the arrays are not the same (null is ok, * and/or size 0) * @throws IllegalArgumentException * if the search is recursive and there is an endless loop due * to outputs of one being inputs to another */ private static String replace(StringBuffer outBuffer, Writer outWriter, String text, Object searchFor, Object replaceWith, boolean recurse, int timeToLive, boolean removeIfFound) { // if recursing, we need to get the string, then print to buffer (since // we need multiple passes) if (!recurse) { return replaceHelper(outBuffer, outWriter, text, searchFor, replaceWith, recurse, timeToLive, removeIfFound); } // get the string String result = replaceHelper(null, null, text, searchFor, replaceWith, recurse, timeToLive, removeIfFound); if (outBuffer != null) { outBuffer.append(result); return null; } if (outWriter != null) { try { outWriter.write(result); } catch (IOException ioe) { throw new RuntimeException(ioe); } return null; } return result; } /** * replace a string or strings from a string, and put the output in a string * buffer. This does not recurse * * @param outWriter * writer to write to * @param text * string to look in * @param searchFor * string array to search for * @param replaceWith * string array to replace with */ public static void replace(Writer outWriter, String text, Object searchFor, Object replaceWith) { replace(null, outWriter, text, searchFor, replaceWith, false, 0, false); } /** * replace a string or strings from a string, and put the output in a string * buffer * * @param outWriter * writer to write to * @param text * string to look in * @param searchFor * string array to search for * @param replaceWith * string array to replace with * @param recurse * if true then do multiple replaces (on the replacements) */ public static void replace(Writer outWriter, String text, Object searchFor, Object replaceWith, boolean recurse) { replace(null, outWriter, text, searchFor, replaceWith, recurse, recurse ? length(searchFor) : 0, false); } /** * replace a string with other strings, and either write to outWriter, or * StringBuffer, and if StringBuffer potentially return a string. If * outBuffer and outWriter are null, then return the String * * @param outBuffer * stringbuffer to write to, or null to not * @param outWriter * Writer to write to, or null to not. * @param text * string to look in * @param searchFor * string array to search for, or string, or list * @param replaceWith * string array to replace with, or string, or list * @param recurse * if true then do multiple replaces (on the replacements) * @param timeToLive * if recursing, prevent endless loops * @param removeIfFound * true if removing from searchFor and replaceWith if found * @return the String if outBuffer and outWriter are null * @throws IllegalArgumentException * if the search is recursive and there is an endless loop due * to outputs of one being inputs to another * @throws IndexOutOfBoundsException * if the lengths of the arrays are not the same (null is ok, * and/or size 0) */ private static String replaceHelper(StringBuffer outBuffer, Writer outWriter, String text, Object searchFor, Object replaceWith, boolean recurse, int timeToLive, boolean removeIfFound) { try { // if recursing, this shouldnt be less than 0 if (timeToLive < 0) { throw new IllegalArgumentException("TimeToLive under 0: " + timeToLive + ", " + text); } int searchForLength = length(searchFor); boolean done = false; // no need to do anything if (isEmpty(text)) { return text; } // need to write the input to output, later if (searchForLength == 0) { done = true; } boolean[] noMoreMatchesForReplIndex = null; int inputIndex = -1; int replaceIndex = -1; long resultPacked = -1; if (!done) { // make sure lengths are ok, these need to be equal if (searchForLength != length(replaceWith)) { throw new IndexOutOfBoundsException("Lengths dont match: " + searchForLength + ", " + length(replaceWith)); } // keep track of which still have matches noMoreMatchesForReplIndex = new boolean[searchForLength]; // index of replace array that will replace the search string // found resultPacked = findNextIndexHelper(searchForLength, searchFor, replaceWith, noMoreMatchesForReplIndex, text, 0); inputIndex = unpackInt(resultPacked, true); replaceIndex = unpackInt(resultPacked, false); } // get a good guess on the size of the result buffer so it doesnt // have to double if it // goes over a bit boolean writeToWriter = outWriter != null; // no search strings found, we are done if (done || inputIndex == -1) { if (writeToWriter) { outWriter.write(text, 0, text.length()); return null; } if (outBuffer != null) { appendSubstring(outBuffer, text, 0, text.length()); return null; } return text; } // no buffer if writing to writer StringBuffer bufferToWriteTo = outBuffer != null ? outBuffer : (writeToWriter ? null : new StringBuffer(text.length() + replaceStringsBufferIncrease(text, searchFor, replaceWith))); String searchString = null; String replaceString = null; int start = 0; while (inputIndex != -1) { searchString = (String) get(searchFor, replaceIndex); replaceString = (String) get(replaceWith, replaceIndex); if (writeToWriter) { outWriter.write(text, start, inputIndex - start); outWriter.write(replaceString); } else { appendSubstring(bufferToWriteTo, text, start, inputIndex) .append(replaceString); } if (removeIfFound) { // better be an iterator based find replace searchFor = remove(searchFor, replaceIndex); replaceWith = remove(replaceWith, replaceIndex); noMoreMatchesForReplIndex = (boolean[])remove(noMoreMatchesForReplIndex, replaceIndex); // we now have a lesser size if we removed one searchForLength--; } start = inputIndex + searchString.length(); resultPacked = findNextIndexHelper(searchForLength, searchFor, replaceWith, noMoreMatchesForReplIndex, text, start); inputIndex = unpackInt(resultPacked, true); replaceIndex = unpackInt(resultPacked, false); } if (writeToWriter) { outWriter.write(text, start, text.length() - start); } else { appendSubstring(bufferToWriteTo, text, start, text.length()); } // no need to convert to string if incoming buffer or writer if (writeToWriter || outBuffer != null) { if (recurse) { throw new IllegalArgumentException( "Cannot recurse and write to existing buffer or writer!"); } return null; } String resultString = bufferToWriteTo.toString(); if (recurse) { return replaceHelper(outBuffer, outWriter, resultString, searchFor, replaceWith, recurse, timeToLive - 1, false); } // this might be null for writer return resultString; } catch (IOException ioe) { throw new RuntimeException(ioe); } } /** * give a best guess on buffer increase for String[] replace get a good * guess on the size of the result buffer so it doesnt have to double if it * goes over a bit * * @param text * @param repl * @param with * @return the increase, with 20% cap */ static int replaceStringsBufferIncrease(String text, Object repl, Object with) { // count the greaters int increase = 0; Iterator iteratorReplace = iterator(repl); Iterator iteratorWith = iterator(with); int replLength = length(repl); String currentRepl = null; String currentWith = null; for (int i = 0; i < replLength; i++) { currentRepl = (String) next(repl, iteratorReplace, i); currentWith = (String) next(with, iteratorWith, i); if (currentRepl == null || currentWith == null) { throw new NullPointerException("Replace string is null: " + text + ", " + currentRepl + ", " + currentWith); } int greater = currentWith.length() - currentRepl.length(); increase += greater > 0 ? 3 * greater : 0; // assume 3 matches } // have upper-bound at 20% increase, then let Java take over increase = Math.min(increase, text.length() / 5); return increase; } /** * Helper method to find the next match in an array of strings replace * * @param searchForLength * @param searchFor * @param replaceWith * @param noMoreMatchesForReplIndex * @param input * @param start * is where to start looking * @return result packed into a long, inputIndex first, then replaceIndex */ private static long findNextIndexHelper(int searchForLength, Object searchFor, Object replaceWith, boolean[] noMoreMatchesForReplIndex, String input, int start) { int inputIndex = -1; int replaceIndex = -1; Iterator iteratorSearchFor = iterator(searchFor); Iterator iteratorReplaceWith = iterator(replaceWith); String currentSearchFor = null; String currentReplaceWith = null; int tempIndex = -1; for (int i = 0; i < searchForLength; i++) { currentSearchFor = (String) next(searchFor, iteratorSearchFor, i); currentReplaceWith = (String) next(replaceWith, iteratorReplaceWith, i); if (noMoreMatchesForReplIndex[i] || isEmpty(currentSearchFor) || currentReplaceWith == null) { continue; } tempIndex = input.indexOf(currentSearchFor, start); // see if we need to keep searching for this noMoreMatchesForReplIndex[i] = tempIndex == -1; if (tempIndex != -1 && (inputIndex == -1 || tempIndex < inputIndex)) { inputIndex = tempIndex; replaceIndex = i; } } // dont create an array, no more objects long resultPacked = packInts(inputIndex, replaceIndex); return resultPacked; } /** * pack two ints into a long. Note: the first is held in the left bits, the * second is held in the right bits * * @param first * is first int * @param second * is second int * @return the long which has two ints in there */ public static long packInts(int first, int second) { long result = first; result <<= 32; result |= second; return result; } /** * take a long * * @param theLong * to unpack * @param isFirst * true for first, false for second * @return one of the packed ints, first or second */ public static int unpackInt(long theLong, boolean isFirst) { int result = 0; // put this in the position of the second one if (isFirst) { theLong >>= 32; } // only look at right part result = (int) (theLong & 0xffffffff); return result; } /** * append a substring to a stringbuffer. removes dependency on substring * which creates objects * * @param buf * stringbuffer * @param string * source string * @param start * start index of source string * @param end * end index of source string * @return the string buffer for chaining */ private static StringBuffer appendSubstring(StringBuffer buf, String string, int start, int end) { for (int i = start; i < end; i++) { buf.append(string.charAt(i)); } return buf; } /** * Get a specific index of an array or collection (note for collections and * iterating, it is more efficient to get an iterator and iterate * * @param arrayOrCollection * @param index * @return the object at that index */ public static Object get(Object arrayOrCollection, int index) { if (arrayOrCollection == null) { if (index == 0) { return null; } throw new RuntimeException("Trying to access index " + index + " of null"); } // no need to iterator on list (e.g. FastProxyList has no iterator if (arrayOrCollection instanceof List) { return ((List) arrayOrCollection).get(index); } if (arrayOrCollection instanceof Collection) { Iterator iterator = iterator(arrayOrCollection); for (int i = 0; i < index; i++) { next(arrayOrCollection, iterator, i); } return next(arrayOrCollection, iterator, index); } if (arrayOrCollection.getClass().isArray()) { return Array.get(arrayOrCollection, index); } if (index == 0) { return arrayOrCollection; } throw new RuntimeException("Trying to access index " + index + " of and object: " + arrayOrCollection); } /** * fail safe toString for Exception blocks, and include the stack * if there is a problem with toString() * @param object * @return the toStringSafe string */ @SuppressWarnings("unchecked") public static String toStringSafe(Object object) { if (object == null) { return null; } try { //give size and type if collection if (object instanceof Collection) { Collection collection = (Collection) object; int collectionSize = collection.size(); if (collectionSize == 0) { return "Empty " + object.getClass().getSimpleName(); } Object first = collection.iterator().next(); return object.getClass().getSimpleName() + " of size " + collectionSize + " with first type: " + (first == null ? null : first.getClass()); } return object.toString(); } catch (Exception e) { return "<> " + object.getClass() + ":\n" + getFullStackTrace(e) + "\n"; } } /** * get the boolean value for an object, cant be null or blank * * @param object * @return the boolean */ public static boolean booleanValue(Object object) { // first handle blanks if (nullOrBlank(object)) { throw new RuntimeException( "Expecting something which can be converted to boolean, but is null or blank: '" + object + "'"); } // its not blank, just convert if (object instanceof Boolean) { return (Boolean) object; } if (object instanceof String) { String string = (String) object; if (equalsIgnoreCase(string, "true") || equalsIgnoreCase(string, "t") || equalsIgnoreCase(string, "yes") || equalsIgnoreCase(string, "y")) { return true; } if (equalsIgnoreCase(string, "false") || equalsIgnoreCase(string, "f") || equalsIgnoreCase(string, "no") || equalsIgnoreCase(string, "n")) { return false; } throw new RuntimeException( "Invalid string to boolean conversion: '" + string + "' expecting true|false or t|f or yes|no or y|n case insensitive"); } throw new RuntimeException("Cant convert object to boolean: " + object.getClass()); } /** * get the boolean value for an object * * @param object * @param defaultBoolean * if object is null or empty * @return the boolean */ public static boolean booleanValue(Object object, boolean defaultBoolean) { if (nullOrBlank(object)) { return defaultBoolean; } return booleanValue(object); } /** * get the Boolean value for an object * * @param object * @return the Boolean or null if null or empty */ public static Boolean booleanObjectValue(Object object) { if (nullOrBlank(object)) { return null; } return booleanValue(object); } /** * is an object null or blank * * @param object * @return true if null or blank */ public static boolean nullOrBlank(Object object) { // first handle blanks and nulls if (object == null) { return true; } if (object instanceof String && isBlank(((String) object))) { return true; } return false; } /** * get a getter method object for a class, potentially in superclasses * @param theClass * @param fieldName * @param callOnSupers true if superclasses should be looked in for the getter * @param throwExceptionIfNotFound will throw runtime exception if not found * @return the getter object or null if not found (or exception if param is set) */ public static Method getter(Class theClass, String fieldName, boolean callOnSupers, boolean throwExceptionIfNotFound) { String getterName = getterNameFromPropertyName(fieldName); return getterHelper(theClass, fieldName, getterName, callOnSupers, throwExceptionIfNotFound); } /** * get a setter method object for a class, potentially in superclasses * @param theClass * @param fieldName * @param getterName name of setter * @param callOnSupers true if superclasses should be looked in for the setter * @param throwExceptionIfNotFound will throw runtime exception if not found * @return the setter object or null if not found (or exception if param is set) */ public static Method getterHelper(Class theClass, String fieldName, String getterName, boolean callOnSupers, boolean throwExceptionIfNotFound) { Method[] methods = retrieveDeclaredMethods(theClass); if (methods != null) { for (Method method : methods) { if (equals(getterName, method.getName()) && isGetter(method)) { return method; } } } //if method not found //if traversing up, and not Object, and not instance method if (callOnSupers && !theClass.equals(Object.class)) { return getterHelper(theClass.getSuperclass(), fieldName, getterName, callOnSupers, throwExceptionIfNotFound); } //maybe throw an exception if (throwExceptionIfNotFound) { throw new RuntimeException("Cant find getter: " + getterName + ", in: " + theClass + ", callOnSupers: " + callOnSupers); } return null; } /** * generate getBb from bb * @param propertyName * @return the getter */ public static String getterNameFromPropertyName(String propertyName) { return "get" + capitalize(propertyName); } /** * get all getters from a class, including superclasses (if specified) (up to and including the specified superclass). * ignore a certain marker annotation, or only include it. * Dont get static or final getters, and get getters of all types * @param theClass to look for fields in * @param superclassToStopAt to go up to or null to go up to Object * @param markerAnnotation if this is not null, then if the field has this annotation, then do not * include in list (if includeAnnotation is false) * @param includeAnnotation true if the attribute should be included if annotation is present, false if exclude * @return the set of field names or empty set if none */ @SuppressWarnings("unchecked") public static Set getters(Class theClass, Class superclassToStopAt, Class markerAnnotation, Boolean includeAnnotation) { return gettersHelper(theClass, superclassToStopAt, null, true, markerAnnotation, includeAnnotation); } /** * get all getters from a class, including superclasses (if specified) * @param theClass to look for fields in * @param superclassToStopAt to go up to or null to go up to Object * @param fieldType is the type of the field to get * @param includeSuperclassToStopAt if we should include the superclass * @param markerAnnotation if this is not null, then if the field has this annotation, then do not * include in list (if includeAnnotation is false) * @param includeAnnotation true if the attribute should be included if annotation is present, false if exclude * @return the set of fields (wont return null) */ @SuppressWarnings("unchecked") static Set gettersHelper(Class theClass, Class superclassToStopAt, Class fieldType, boolean includeSuperclassToStopAt, Class markerAnnotation, Boolean includeAnnotation) { //MAKE SURE IF ANY MORE PARAMS ARE ADDED, THE CACHE KEY IS CHANGED! Set getterSet = null; String cacheKey = theClass + CACHE_SEPARATOR + superclassToStopAt + CACHE_SEPARATOR + fieldType + CACHE_SEPARATOR + includeSuperclassToStopAt + CACHE_SEPARATOR + markerAnnotation + CACHE_SEPARATOR + includeAnnotation; getterSet = getterSetCache().get(cacheKey); if (getterSet != null) { return getterSet; } getterSet = new LinkedHashSet(); gettersHelper(theClass, superclassToStopAt, fieldType, includeSuperclassToStopAt, markerAnnotation, getterSet, includeAnnotation); //add to cache getterSetCache().put(cacheKey, getterSet); return getterSet; } /** * get all getters from a class, including superclasses (if specified) * @param theClass to look for fields in * @param superclassToStopAt to go up to or null to go up to Object * @param propertyType is the type of the field to get * @param includeSuperclassToStopAt if we should include the superclass * @param markerAnnotation if this is not null, then if the field has this annotation, then do not * include in list * @param getterSet set to add fields to * @param includeAnnotation if include or exclude */ @SuppressWarnings("unchecked") private static void gettersHelper(Class theClass, Class superclassToStopAt, Class propertyType, boolean includeSuperclassToStopAt, Class markerAnnotation, Set getterSet, Boolean includeAnnotation) { Method[] methods = retrieveDeclaredMethods(theClass); if (length(methods) != 0) { for (Method method: methods) { //must be a getter if (!isGetter(method)) { continue; } //if checking for annotation if (markerAnnotation != null && (includeAnnotation != method.isAnnotationPresent(markerAnnotation))) { continue; } //if checking for type, and not right type, continue if (propertyType != null && !propertyType.isAssignableFrom(method.getReturnType())) { continue; } //go for it getterSet.add(method); } } //see if done recursing (if superclassToStopAt is null, then stop at Object if (theClass.equals(superclassToStopAt) || theClass.equals(Object.class)) { return; } Class superclass = theClass.getSuperclass(); if (!includeSuperclassToStopAt && superclass.equals(superclassToStopAt)) { return; } //recurse gettersHelper(superclass, superclassToStopAt, propertyType, includeSuperclassToStopAt, markerAnnotation, getterSet, includeAnnotation); } /** * if the method name starts with get, and takes no args, and returns something, * then getter * @param method * @return true if getter */ public static boolean isGetter(Method method) { //must start with get String methodName = method.getName(); if (!methodName.startsWith("get") && !methodName.startsWith("is")) { return false; } //must not be void if (method.getReturnType() == Void.TYPE) { return false; } //must not take args if (length(method.getParameterTypes()) != 0) { return false; } //must not be static if (Modifier.isStatic(method.getModifiers())) { return false; } return true; } /** * assign data to a setter. Will find the field in superclasses, will typecast, * and will override security (private, protected, etc) * @param invokeOn to call on or null for static * @param fieldName method name to call * @param dataToAssign data * @param typeCast will typecast if true * @throws RuntimeException if not there */ public static void assignSetter(Object invokeOn, String fieldName, Object dataToAssign, boolean typeCast) { Class invokeOnClass = invokeOn.getClass(); try { Method setter = setter(invokeOnClass, fieldName, true, true); setter.setAccessible(true); if (typeCast) { dataToAssign = typeCast(dataToAssign, setter.getParameterTypes()[0]); } setter.invoke(invokeOn, new Object[]{dataToAssign}); } catch (Exception e) { throw new RuntimeException("Problem assigning setter: " + fieldName + " on class: " + invokeOnClass + ", type of data is: " + className(dataToAssign), e); } } /** * if the method name starts with get, and takes no args, and returns something, * then getter * @param method * @return true if getter */ public static boolean isSetter(Method method) { //must start with get if (!method.getName().startsWith("set")) { return false; } //must be void if (method.getReturnType() != Void.TYPE) { return false; } //must take one arg if (length(method.getParameterTypes()) != 1) { return false; } //must not be static if (Modifier.isStatic(method.getModifiers())) { return false; } return true; } /** * get a setter method object for a class, potentially in superclasses * @param theClass * @param fieldName * @param callOnSupers true if superclasses should be looked in for the setter * @param throwExceptionIfNotFound will throw runtime exception if not found * @return the setter object or null if not found (or exception if param is set) */ public static Method setter(Class theClass, String fieldName, boolean callOnSupers, boolean throwExceptionIfNotFound) { String setterName = setterNameFromPropertyName(fieldName); return setterHelper(theClass, fieldName, setterName, callOnSupers, throwExceptionIfNotFound); } /** * get a setter method object for a class, potentially in superclasses * @param theClass * @param fieldName * @param setterName name of setter * @param callOnSupers true if superclasses should be looked in for the setter * @param throwExceptionIfNotFound will throw runtime exception if not found * @return the setter object or null if not found (or exception if param is set) */ public static Method setterHelper(Class theClass, String fieldName, String setterName, boolean callOnSupers, boolean throwExceptionIfNotFound) { Method[] methods = retrieveDeclaredMethods(theClass); if (methods != null) { for (Method method : methods) { if (equals(setterName, method.getName()) && isSetter(method)) { return method; } } } //if method not found //if traversing up, and not Object, and not instance method if (callOnSupers && !theClass.equals(Object.class)) { return setterHelper(theClass.getSuperclass(), fieldName, setterName, callOnSupers, throwExceptionIfNotFound); } //maybe throw an exception if (throwExceptionIfNotFound) { throw new RuntimeException("Cant find setter: " + setterName + ", in: " + theClass + ", callOnSupers: " + callOnSupers); } return null; } /** * generate setBb from bb * @param propertyName * @return the setter */ public static String setterNameFromPropertyName(String propertyName) { return "set" + capitalize(propertyName); } /** * get all setters from a class, including superclasses (if specified) * @param theClass to look for fields in * @param superclassToStopAt to go up to or null to go up to Object * @param fieldType is the type of the field to get * @param includeSuperclassToStopAt if we should include the superclass * @param markerAnnotation if this is not null, then if the field has this annotation, then do not * include in list (if includeAnnotation is false) * @param includeAnnotation true if the attribute should be included if annotation is present, false if exclude * @return the set of fields (wont return null) */ @SuppressWarnings("unchecked") public static Set setters(Class theClass, Class superclassToStopAt, Class fieldType, boolean includeSuperclassToStopAt, Class markerAnnotation, boolean includeAnnotation) { return settersHelper(theClass, superclassToStopAt, fieldType, includeSuperclassToStopAt, markerAnnotation, includeAnnotation); } /** * get all setters from a class, including superclasses (if specified) * @param theClass to look for fields in * @param superclassToStopAt to go up to or null to go up to Object * @param fieldType is the type of the field to get * @param includeSuperclassToStopAt if we should include the superclass * @param markerAnnotation if this is not null, then if the field has this annotation, then do not * include in list (if includeAnnotation is false) * @param includeAnnotation true if the attribute should be included if annotation is present, false if exclude * @return the set of fields (wont return null) */ @SuppressWarnings("unchecked") static Set settersHelper(Class theClass, Class superclassToStopAt, Class fieldType, boolean includeSuperclassToStopAt, Class markerAnnotation, boolean includeAnnotation) { //MAKE SURE IF ANY MORE PARAMS ARE ADDED, THE CACHE KEY IS CHANGED! Set setterSet = null; String cacheKey = theClass + CACHE_SEPARATOR + superclassToStopAt + CACHE_SEPARATOR + fieldType + CACHE_SEPARATOR + includeSuperclassToStopAt + CACHE_SEPARATOR + markerAnnotation + CACHE_SEPARATOR + includeAnnotation; setterSet = setterSetCache().get(cacheKey); if (setterSet != null) { return setterSet; } setterSet = new LinkedHashSet(); settersHelper(theClass, superclassToStopAt, fieldType, includeSuperclassToStopAt, markerAnnotation, setterSet, includeAnnotation); //add to cache setterSetCache().put(cacheKey, setterSet); return setterSet; } /** * get all setters from a class, including superclasses (if specified) * @param theClass to look for fields in * @param superclassToStopAt to go up to or null to go up to Object * @param propertyType is the type of the field to get * @param includeSuperclassToStopAt if we should include the superclass * @param markerAnnotation if this is not null, then if the field has this annotation, then do not * include in list * @param setterSet set to add fields to * @param includeAnnotation if include or exclude (or null if not looking for annotations) */ @SuppressWarnings("unchecked") private static void settersHelper(Class theClass, Class superclassToStopAt, Class propertyType, boolean includeSuperclassToStopAt, Class markerAnnotation, Set setterSet, Boolean includeAnnotation) { Method[] methods = retrieveDeclaredMethods(theClass); if (length(methods) != 0) { for (Method method: methods) { //must be a getter if (!isSetter(method)) { continue; } //if checking for annotation if (markerAnnotation != null && (includeAnnotation != method.isAnnotationPresent(markerAnnotation))) { continue; } //if checking for type, and not right type, continue if (propertyType != null && !propertyType.isAssignableFrom(method.getParameterTypes()[0])) { continue; } //go for it setterSet.add(method); } } //see if done recursing (if superclassToStopAt is null, then stop at Object if (theClass.equals(superclassToStopAt) || theClass.equals(Object.class)) { return; } Class superclass = theClass.getSuperclass(); if (!includeSuperclassToStopAt && superclass.equals(superclassToStopAt)) { return; } //recurse settersHelper(superclass, superclassToStopAt, propertyType, includeSuperclassToStopAt, markerAnnotation, setterSet, includeAnnotation); } /** * If this is a getter or setter, then get the property name * @param method * @return the property name */ public static String propertyName(Method method) { String methodName = method.getName(); boolean isGetter = methodName.startsWith("get"); boolean isSetter = methodName.startsWith("set"); boolean isIsser = methodName.startsWith("is"); int expectedLength = isIsser ? 2 : 3; int length = methodName.length(); if ((!(isGetter || isSetter || isIsser)) || (length <= expectedLength)) { throw new RuntimeException("Not a getter or setter: " + methodName); } char fourthCharLower = Character.toLowerCase(methodName.charAt(expectedLength)); //if size 4, then return the string if (length == expectedLength +1) { return Character.toString(fourthCharLower); } //return the lower appended with the rest return fourthCharLower + methodName.substring(expectedLength+1, length); } /** * use reflection to get a property type based on getter or setter or field * @param theClass * @param propertyName * @return the property type */ public static Class propertyType(Class theClass, String propertyName) { Method method = getter(theClass, propertyName, true, false); if (method != null) { return method.getReturnType(); } //use setter method = setter(theClass, propertyName, true, false); if (method != null) { return method.getParameterTypes()[0]; } //no setter or getter, use field Field field = field(theClass, propertyName, true, true); return field.getType(); } /** * If necessary, convert an object to another type. if type is Object.class, just return the input. * Do not convert null to an empty primitive * @param is template type * @param value * @param theClass * @return the object of that instance converted into something else */ public static T typeCast(Object value, Class theClass) { //default behavior is not to convert null to empty primitive return typeCast(value, theClass, false, false); } /** *
   * make a new file in the name prefix dir.  If parent dir name is c:\temp
   * and namePrefix is grouperDdl and nameSuffix is sql, then the file will be:
   * 
   * c:\temp\grouperDdl_20080721_13_45_43_123.sql
   *  
   * If the file exists, it will make a new filename, and create the empty file, and return it
   * 
* * @param parentDirName can be blank for current dir * @param namePrefix the part before the date part * @param nameSuffix the last part of file name (can contain dot or will be the extension * @param createFile true to create the file * @return the created file */ public static File newFileUniqueName(String parentDirName, String namePrefix, String nameSuffix, boolean createFile) { DateFormat fileNameFormat = new SimpleDateFormat("yyyyMMdd_HH_mm_ss_SSS"); if (!isBlank(parentDirName)) { if (!parentDirName.endsWith("/") && !parentDirName.endsWith("\\")) { parentDirName += File.separator; } //make sure it exists and is a dir File parentDir = new File(parentDirName); if (!parentDir.exists()) { if (!parentDir.mkdirs()) { throw new RuntimeException("Cant make dir: " + parentDir.getAbsolutePath()); } } else { if (!parentDir.isDirectory()) { throw new RuntimeException("Parent dir is not a directory: " + parentDir.getAbsolutePath()); } } } else { //make it empty string so it will concatenate well parentDirName = ""; } //make sure suffix has a dot in it if (!nameSuffix.contains(".")) { nameSuffix = "." + nameSuffix; } String fileName = parentDirName + namePrefix + "_" + fileNameFormat.format(new Date()) + nameSuffix; int dotLocation = fileName.lastIndexOf('.'); String fileNamePre = fileName.substring(0,dotLocation); String fileNamePost = fileName.substring(dotLocation); File theFile = new File(fileName); int i; for (i=0;i<1000;i++) { if (!theFile.exists()) { break; } fileName = fileNamePre + "_" + i + fileNamePost; theFile = new File(fileName); } if (i>=1000) { throw new RuntimeException("Cant find filename to create: " + fileName); } if (createFile) { try { if (!theFile.createNewFile()) { throw new RuntimeException("Cant create file, it returned false"); } } catch (Exception e) { throw new RuntimeException("Cant create file: " + fileName + ", make sure " + "permissions and such are ok, or change file location in grouper.properties if applicable", e); } } return theFile; } /** *
   * Convert an object to a java.util.Date.  allows, dates, null, blank, 
   * yyyymmdd or yyyymmdd hh24:mm:ss
   * or yyyy/MM/dd HH:mm:ss.SSS
   * 
* @param inputObject * is the String or Date to convert * * @return the Date */ public static Date dateValue(Object inputObject) { if (inputObject == null) { return null; } if (inputObject instanceof java.util.Date) { return (Date)inputObject; } if (inputObject instanceof String) { String input = (String)inputObject; //trim and handle null and empty if (isBlank(input)) { return null; } try { if (input.length() == 8) { return dateFormat().parse(input); } if (!contains(input, '.')) { if (contains(input, '/')) { return dateMinutesSecondsFormat.parse(input); } //else no slash return dateMinutesSecondsNoSlashFormat.parse(input); } if (contains(input, '/')) { //see if the period is 6 back int lastDotIndex = input.lastIndexOf('.'); if (lastDotIndex == input.length() - 7) { String nonNanoInput = input.substring(0,input.length()-3); Date date = timestampFormat.parse(nonNanoInput); //get the last 3 String lastThree = input.substring(input.length()-3,input.length()); int lastThreeInt = Integer.parseInt(lastThree); Timestamp timestamp = new Timestamp(date.getTime()); timestamp.setNanos(timestamp.getNanos() + (lastThreeInt * 1000)); return timestamp; } return timestampFormat.parse(input); } //else no slash return timestampNoSlashFormat.parse(input); } catch (ParseException pe) { throw new RuntimeException(errorStart + toStringForLog(input)); } } throw new RuntimeException("Cannot convert Object to date : " + toStringForLog(inputObject)); } /** * See if the input is null or if string, if it is empty or blank (whitespace) * @param input * @return true if blank */ public static boolean isBlank(Object input) { if (null == input) { return true; } return (input instanceof String && isBlank((String)input)); } /** * If necessary, convert an object to another type. if type is Object.class, just return the input * @param is the type to return * @param value * @param theClass * @param convertNullToDefaultPrimitive if the value is null, and theClass is primitive, should we * convert the null to a primitive default value * @param useNewInstanceHooks if theClass is not recognized, then honor the string "null", "newInstance", * or get a constructor with one param, and call it * @return the object of that instance converted into something else */ @SuppressWarnings({ "unchecked", "cast" }) public static T typeCast(Object value, Class theClass, boolean convertNullToDefaultPrimitive, boolean useNewInstanceHooks) { if (Object.class.equals(theClass)) { return (T)value; } if (value==null) { if (convertNullToDefaultPrimitive && theClass.isPrimitive()) { if ( theClass == boolean.class ) { return (T)Boolean.FALSE; } if ( theClass == char.class ) { return (T)(Object)0; } //convert 0 to the type return typeCast(0, theClass, false, false); } return null; } if (theClass.isInstance(value)) { return (T)value; } //if array, get the base class if (theClass.isArray() && theClass.getComponentType() != null) { theClass = (Class)theClass.getComponentType(); } Object resultValue = null; //loop through and see the primitive types etc if (theClass.equals(Date.class)) { resultValue = dateValue(value); } else if (theClass.equals(String.class)) { resultValue = stringValue(value); } else if (theClass.equals(Timestamp.class)) { resultValue = toTimestamp(value); } else if (theClass.equals(Boolean.class) || theClass.equals(boolean.class)) { resultValue = booleanObjectValue(value); } else if (theClass.equals(Integer.class) || theClass.equals(int.class)) { resultValue = intObjectValue(value, true); } else if (theClass.equals(Double.class) || theClass.equals(double.class)) { resultValue = doubleObjectValue(value, true); } else if (theClass.equals(Float.class) || theClass.equals(float.class)) { resultValue = floatObjectValue(value, true); } else if (theClass.equals(Long.class) || theClass.equals(long.class)) { resultValue = longObjectValue(value, true); } else if (theClass.equals(Byte.class) || theClass.equals(byte.class)) { resultValue = byteObjectValue(value); } else if (theClass.equals(Character.class) || theClass.equals(char.class)) { resultValue = charObjectValue(value); } else if (theClass.equals(Short.class) || theClass.equals(short.class)) { resultValue = shortObjectValue(value); } else if ( theClass.isEnum() && (value instanceof String) ) { resultValue = Enum.valueOf((Class)theClass, (String) value); } else if ( theClass.equals(Class.class) && (value instanceof String) ) { resultValue = forName((String)value); } else if (useNewInstanceHooks && value instanceof String) { String stringValue = (String)value; if ( equals("null", stringValue)) { resultValue = null; } else if (equals("newInstance", stringValue)) { resultValue = newInstance(theClass); } else { // instantiate using string //note, we could typecast this to fit whatever is there... right now this is used for annotation try { Constructor constructor = theClass.getConstructor(new Class[] {String.class} ); resultValue = constructor.newInstance(new Object[] {stringValue} ); } catch (Exception e) { throw new RuntimeException("Cant find constructor with string for class: " + theClass); } } } else { throw new RuntimeException("Cannot convert from type: " + value.getClass() + " to type: " + theClass); } return (T)resultValue; } /** * see if a class is a scalar (not bean, not array or list, etc) * @param type * @return true if scalar */ public static boolean isScalar(Class type) { if (type.isArray()) { return false; } //definitely all primitives if (type.isPrimitive()) { return true; } //Integer, Float, etc if (Number.class.isAssignableFrom(type)) { return true; } //Date, Timestamp if (Date.class.isAssignableFrom(type)) { return true; } if (Character.class.equals(type)) { return true; } //handles strings and string builders if (CharSequence.class.equals(type) || CharSequence.class.isAssignableFrom(type)) { return true; } if (Class.class == type || Boolean.class == type || type.isEnum()) { return true; } //appears not to be a scalar return false; } /** *
   * Convert a string or object to a timestamp (could be string, date, timestamp, etc)
   * yyyymmdd
   * or
   * yyyy/MM/dd HH:mm:ss
   * or
   * yyyy/MM/dd HH:mm:ss.SSS
   * or
   * yyyy/MM/dd HH:mm:ss.SSSSSS
   * 
   * 
* * @param input * @return the timestamp * @throws RuntimeException if invalid format */ public static Timestamp toTimestamp(Object input) { if (null == input) { return null; } else if (input instanceof java.sql.Timestamp) { return (Timestamp) input; } else if (input instanceof String) { return stringToTimestamp((String) input); } else if (input instanceof Date) { return new Timestamp(((Date)input).getTime()); } else if (input instanceof java.sql.Date) { return new Timestamp(((java.sql.Date)input).getTime()); } else { throw new RuntimeException("Cannot convert Object to timestamp : " + input); } } /** * convert an object to a string * * @param input * is the object to convert * * @return the String conversion of the object */ public static String stringValue(Object input) { //this isnt needed if (input == null) { return (String) input; } if (input instanceof Timestamp) { //convert to yyyy/MM/dd HH:mm:ss.SSS return timestampToString((Timestamp) input); } if (input instanceof Date) { //convert to yyyymmdd return stringValue((Date) input); } if (input instanceof Number) { DecimalFormat decimalFormat = new DecimalFormat( "###################.###############"); return decimalFormat.format(((Number) input).doubleValue()); } return input.toString(); } /** * Convert a timestamp into a string: yyyy/MM/dd HH:mm:ss.SSS * @param timestamp * @return the string representation */ public synchronized static String timestampToString(Date timestamp) { if (timestamp == null) { return null; } return timestampFormat.format(timestamp); } /** * get the timestamp format for this thread * if you call this make sure to synchronize on FastDateUtils.class * @return the timestamp format */ synchronized static SimpleDateFormat dateFormat() { return dateFormat; } /** * convert a date to the standard string yyyymmdd * @param date * @return the string value */ public static String stringValue(java.util.Date date) { synchronized (GrouperInstallerUtils.class) { if (date == null) { return null; } String theString = dateFormat().format(date); return theString; } } /** *
convert a string to timestamp based on the following formats:
   * yyyyMMdd
   * yyyy/MM/dd HH:mm:ss
   * yyyy/MM/dd HH:mm:ss.SSS
   * yyyy/MM/dd HH:mm:ss.SSSSSS
   * 
* @param input * @return the timestamp object */ public static Timestamp stringToTimestamp(String input) { Date date = stringToTimestampHelper(input); if (date == null) { return null; } //maybe already a timestamp if (date instanceof Timestamp) { return (Timestamp)date; } return new Timestamp(date.getTime()); } /** * return a date based on input, null safe. Allow any of the three * formats: * yyyyMMdd * yyyy/MM/dd HH:mm:ss * yyyy/MM/dd HH:mm:ss.SSS * yyyy/MM/dd HH:mm:ss.SSSSSS * * @param input * @return the millis, -1 for null */ synchronized static Date stringToTimestampHelper(String input) { //trim and handle null and empty if (isBlank(input)) { return null; } try { //convert mainframe if (equals("99999999", input) || equals("999999", input)) { input = "20991231"; } if (input.length() == 8) { return dateFormat().parse(input); } if (!contains(input, '.')) { if (contains(input, '/')) { return dateMinutesSecondsFormat.parse(input); } //else no slash return dateMinutesSecondsNoSlashFormat.parse(input); } if (contains(input, '/')) { //see if the period is 6 back int lastDotIndex = input.lastIndexOf('.'); if (lastDotIndex == input.length() - 7) { String nonNanoInput = input.substring(0,input.length()-3); Date date = timestampFormat.parse(nonNanoInput); //get the last 3 String lastThree = input.substring(input.length()-3,input.length()); int lastThreeInt = Integer.parseInt(lastThree); Timestamp timestamp = new Timestamp(date.getTime()); timestamp.setNanos(timestamp.getNanos() + (lastThreeInt * 1000)); return timestamp; } return timestampFormat.parse(input); } //else no slash return timestampNoSlashFormat.parse(input); } catch (ParseException pe) { throw new RuntimeException(errorStart + input); } } /** * start of error parsing messages */ private static final String errorStart = "Invalid timestamp, please use any of the formats: " + DATE_FORMAT + ", " + TIMESTAMP_FORMAT + ", " + DATE_MINUTES_SECONDS_FORMAT + ": "; /** * Convert an object to a byte, allow nulls * @param input * @return the boolean object value */ public static BigDecimal bigDecimalObjectValue(Object input) { if (input instanceof BigDecimal) { return (BigDecimal)input; } if (isBlank(input)) { return null; } return BigDecimal.valueOf(doubleValue(input)); } /** * Convert an object to a byte, allow nulls * @param input * @return the boolean object value */ public static Byte byteObjectValue(Object input) { if (input instanceof Byte) { return (Byte)input; } if (isBlank(input)) { return null; } return Byte.valueOf(byteValue(input)); } /** * convert an object to a byte * @param input * @return the byte */ public static byte byteValue(Object input) { if (input instanceof String) { String string = (String)input; return Byte.parseByte(string); } if (input instanceof Number) { return ((Number)input).byteValue(); } throw new RuntimeException("Cannot convert to byte: " + className(input)); } /** * get the Double value of an object * * @param input * is a number or String * @param allowNullBlank used to default to false, if true, return null if nul inputted * * @return the Double equivalent */ public static Double doubleObjectValue(Object input, boolean allowNullBlank) { if (input instanceof Double) { return (Double) input; } if (allowNullBlank && isBlank(input)) { return null; } return Double.valueOf(doubleValue(input)); } /** * get the double value of an object * * @param input * is a number or String * * @return the double equivalent */ public static double doubleValue(Object input) { if (input instanceof String) { String string = (String)input; return Double.parseDouble(string); } if (input instanceof Number) { return ((Number)input).doubleValue(); } throw new RuntimeException("Cannot convert to double: " + className(input)); } /** * get the double value of an object, do not throw an * exception if there is an * error * * @param input * is a number or String * * @return the double equivalent */ public static double doubleValueNoError(Object input) { if (input == null || (input instanceof String && isBlank((String)input))) { return NOT_FOUND; } try { return doubleValue(input); } catch (Exception e) { //no need to log here } return NOT_FOUND; } /** * get the Float value of an object * * @param input * is a number or String * @param allowNullBlank true if allow null or blank * * @return the Float equivalent */ public static Float floatObjectValue(Object input, boolean allowNullBlank) { if (input instanceof Float) { return (Float) input; } if (allowNullBlank && isBlank(input)) { return null; } return Float.valueOf(floatValue(input)); } /** * get the float value of an object * * @param input * is a number or String * * @return the float equivalent */ public static float floatValue(Object input) { if (input instanceof String) { String string = (String)input; return Float.parseFloat(string); } if (input instanceof Number) { return ((Number)input).floatValue(); } throw new RuntimeException("Cannot convert to float: " + className(input)); } /** * get the float value of an object, do not throw an exception if there is an * error * * @param input * is a number or String * * @return the float equivalent */ public static float floatValueNoError(Object input) { if (input == null || (input instanceof String && isBlank((String)input))) { return NOT_FOUND; } try { return floatValue(input); } catch (Exception e) { e.printStackTrace(); } return NOT_FOUND; } /** * get the Integer value of an object * * @param input * is a number or String * @param allowNullBlank true if convert null or blank to null * * @return the Integer equivalent */ public static Integer intObjectValue(Object input, boolean allowNullBlank) { if (input instanceof Integer) { return (Integer) input; } if (allowNullBlank && isBlank(input)) { return null; } return Integer.valueOf(intValue(input)); } /** * convert an object to a int * @param input * @return the number */ public static int intValue(Object input) { if (input instanceof String) { String string = (String)input; return Integer.parseInt(string); } if (input instanceof Number) { return ((Number)input).intValue(); } if (false) { if (input == null) { return 0; } if (input instanceof String || isBlank((String)input)) { return 0; } } throw new RuntimeException("Cannot convert to int: " + className(input)); } /** * convert an object to a int * @param input * @param valueIfNull is if the input is null or empty, return this value * @return the number */ public static int intValue(Object input, int valueIfNull) { if (input == null || "".equals(input)) { return valueIfNull; } return intObjectValue(input, false); } /** * get the int value of an object, do not throw an exception if there is an * error * * @param input * is a number or String * * @return the int equivalent */ public static int intValueNoError(Object input) { if (input == null || (input instanceof String && isBlank((String)input))) { return NOT_FOUND; } try { return intValue(input); } catch (Exception e) { //no need to log here } return NOT_FOUND; } /** special number when a number is not found */ public static final int NOT_FOUND = -999999999; /** * The name says it all. */ public static final int DEFAULT_BUFFER_SIZE = 1024 * 4; /** * get the Long value of an object * * @param input * is a number or String * @param allowNullBlank true if null or blank converts to null * * @return the Long equivalent */ public static Long longObjectValue(Object input, boolean allowNullBlank) { if (input instanceof Long) { return (Long) input; } if (allowNullBlank && isBlank(input)) { return null; } return Long.valueOf(longValue(input)); } /** * convert an object to a long * @param input * @return the number */ public static long longValue(Object input) { if (input instanceof String) { String string = (String)input; return Long.parseLong(string); } if (input instanceof Number) { return ((Number)input).longValue(); } throw new RuntimeException("Cannot convert to long: " + className(input)); } /** * convert an object to a long * @param input * @param valueIfNull is if the input is null or empty, return this value * @return the number */ public static long longValue(Object input, long valueIfNull) { if (input == null || "".equals(input)) { return valueIfNull; } return longObjectValue(input, false); } /** * get the long value of an object, do not throw an exception if there is an * error * * @param input * is a number or String * * @return the long equivalent */ public static long longValueNoError(Object input) { if (input == null || (input instanceof String && isBlank((String)input))) { return NOT_FOUND; } try { return longValue(input); } catch (Exception e) { //no need to log here } return NOT_FOUND; } /** * get the Short value of an object. converts null or blank to null * * @param input * is a number or String * * @return the Long equivalent */ public static Short shortObjectValue(Object input) { if (input instanceof Short) { return (Short) input; } if (isBlank(input)) { return null; } return Short.valueOf(shortValue(input)); } /** * convert an object to a short * @param input * @return the number */ public static short shortValue(Object input) { if (input instanceof String) { String string = (String)input; return Short.parseShort(string); } if (input instanceof Number) { return ((Number)input).shortValue(); } throw new RuntimeException("Cannot convert to short: " + className(input)); } /** * get the Character wrapper value for the input * @param input allow null, return null * @return the Character object wrapper */ public static Character charObjectValue(Object input) { if (input instanceof Character) { return (Character) input; } if (isBlank(input)) { return null; } return new Character(charValue(input)); } /** * convert an object to a char * @param input * @return the number */ public static char charValue(Object input) { if (input instanceof Character) { return ((Character) input).charValue(); } //if string length 1, thats ok if (input instanceof String) { String inputString = (String) input; if (inputString.length() == 1) { return inputString.charAt(0); } } throw new RuntimeException("Cannot convert to char: " + (input == null ? null : (input.getClass() + ", " + input))); } /** * Create the parent directories for a file if they do not already exist * @param file */ public static void createParentDirectories(File file) { if (!file.getParentFile().exists()) { if (!file.getParentFile().mkdirs()) { throw new RuntimeException("Could not create directory : " + file.getParentFile()); } } } /** * save a string into a file, file does not have to exist * * @param file * is the file to save to * @param contents * is the contents of the file */ public static void saveStringIntoFile(File file, String contents) { try { writeStringToFile(file, contents, "UTF-8"); } catch (IOException ioe) { throw new RuntimeException(ioe); } } /** * save a string into a file, file does not have to exist * * @param file * is the file to save to * @param contents * is the contents of the file * @param onlyIfDifferentContents true if only saving due to different contents * @param ignoreWhitespace true to ignore whitespace * @return true if contents were saved (thus different if param set) */ public static boolean saveStringIntoFile(File file, String contents, boolean onlyIfDifferentContents, boolean ignoreWhitespace) { if (onlyIfDifferentContents && file.exists()) { String fileContents = readFileIntoString(file); String compressedContents = contents; if (ignoreWhitespace) { compressedContents = replaceWhitespaceWithSpace(compressedContents); fileContents = replaceWhitespaceWithSpace(fileContents); } //they are the same, dont worry about it if (equals(fileContents, compressedContents)) { return false; } } saveStringIntoFile(file, contents); return true; } /** *

* Writes data to a file. The file will be created if it does not exist. *

*

* There is no readFileToString method without encoding parameter because * the default encoding can differ between platforms and therefore results * in inconsistent results. *

* * @param file the file to write. * @param data The content to write to the file. * @param encoding encoding to use * @throws IOException in case of an I/O error * @throws UnsupportedEncodingException if the encoding is not supported * by the VM */ public static void writeStringToFile(File file, String data, String encoding) throws IOException { OutputStream out = new java.io.FileOutputStream(file); try { out.write(data.getBytes(encoding)); } finally { closeQuietly(out); } } /** *

* Writes data to a file. The file will be created if it does not exist. *

*

* There is no readFileToString method without encoding parameter because * the default encoding can differ between platforms and therefore results * in inconsistent results. *

* * @param file the file to write. * @param data The content to write to the file. */ public static void writeStringToFile(File file, String data) { try { writeStringToFile(file, data, "UTF-8"); } catch (IOException ioe) { throw new RuntimeException(ioe); } } /** * @param file * is the file to read into a string * * @return String */ public static String readFileIntoString(File file) { if (file == null) { return null; } try { return readFileToString(file, "UTF-8"); } catch (IOException ioe) { throw new RuntimeException(ioe); } } /** * @param resourceName is the string resource from classpath to read (e.g. grouper.properties) * @param allowNull is true if its ok if the resource is null or if it is not found or blank or whatever. * * @return String or null if allowed or RuntimeException if not allowed */ public static String readResourceIntoString(String resourceName, boolean allowNull) { if (isBlank(resourceName)) { if (allowNull) { return null; } throw new RuntimeException("Resource name is blank"); } URL url = computeUrl(resourceName, allowNull); //this is ok if (url == null && allowNull) { return null; } InputStream inputStream = null; StringWriter stringWriter = new StringWriter(); try { inputStream = url.openStream(); copy(inputStream, stringWriter, "UTF-8"); } catch (IOException ioe) { throw new RuntimeException("Error reading resource: '" + resourceName + "'", ioe); } finally { closeQuietly(inputStream); closeQuietly(stringWriter); } return stringWriter.toString(); } /** * read resource into string * @param resourceName * @param classInJar if not null, then look for the jar where this file is, and look in the same dir * @return the properties or null if not exist */ public static String readResourceIntoString(String resourceName, Class classInJar) { try { return readResourceIntoString(resourceName, false); } catch (Exception e) { //try from jar location } //lets look next to jar File jarFile = classInJar == null ? null : jarFile(classInJar); File parentDir = jarFile == null ? null : jarFile.getParentFile(); String fileName = parentDir == null ? null : (stripLastSlashIfExists(fileCanonicalPath(parentDir)) + File.separator + resourceName); File configFile = fileName == null ? null : new File(fileName); return readFileIntoString(configFile); } /** *

* Reads the contents of a file into a String. *

*

* There is no readFileToString method without encoding parameter because * the default encoding can differ between platforms and therefore results * in inconsistent results. *

* * @param file the file to read. * @param encoding the encoding to use * @return The file contents or null if read failed. * @throws IOException in case of an I/O error */ public static String readFileToString(File file, String encoding) throws IOException { InputStream in = new java.io.FileInputStream(file); try { return toString(in, encoding); } finally { closeQuietly(in); } } /** * replace all whitespace with space * @param input * @return the string */ public static String replaceWhitespaceWithSpace(String input) { if (input == null) { return input; } return input.replaceAll("\\s+", " "); } /** * Unconditionally close an InputStream. * Equivalent to {@link InputStream#close()}, except any exceptions will be ignored. * @param input A (possibly null) InputStream */ public static void closeQuietly(InputStream input) { if (input == null) { return; } try { input.close(); } catch (IOException ioe) { } } /** * Unconditionally close a ZipFile. * Equivalent to {@link ZipFile#close()}, except any exceptions will be ignored. * @param input A (possibly null) ZipFile */ public static void closeQuietly(ZipFile input) { if (input == null) { return; } try { input.close(); } catch (IOException ioe) { } } /** * Unconditionally close an OutputStream. * Equivalent to {@link OutputStream#close()}, except any exceptions will be ignored. * @param output A (possibly null) OutputStream */ public static void closeQuietly(OutputStream output) { if (output == null) { return; } try { output.close(); } catch (IOException ioe) { } } /** * Unconditionally close an Reader. * Equivalent to {@link Reader#close()}, except any exceptions will be ignored. * * @param input A (possibly null) Reader */ public static void closeQuietly(Reader input) { if (input == null) { return; } try { input.close(); } catch (IOException ioe) { } } /** * close a writer quietly * @param writer */ public static void closeQuietly(Writer writer) { if (writer != null) { try { writer.close(); } catch (IOException e) { //swallow, its ok } } } /** * Get the contents of an InputStream as a String. * @param input the InputStream to read from * @param encoding The name of a supported character encoding. See the * IANA * Charset Registry for a list of valid encoding types. * @return the requested String * @throws IOException In case of an I/O problem */ public static String toString(InputStream input, String encoding) throws IOException { StringWriter sw = new StringWriter(); copy(input, sw, encoding); return sw.toString(); } /** * Copy and convert bytes from an InputStream to chars on a * Writer, using the specified encoding. * @param input the InputStream to read from * @param output the Writer to write to * @param encoding The name of a supported character encoding. See the * IANA * Charset Registry for a list of valid encoding types. * @throws IOException In case of an I/O problem */ public static void copy(InputStream input, Writer output, String encoding) throws IOException { InputStreamReader in = new InputStreamReader(input, encoding); copy(in, output); } /** * Copy chars from a Reader to a Writer. * @param input the Reader to read from * @param output the Writer to write to * @return the number of characters copied * @throws IOException In case of an I/O problem */ public static int copy(Reader input, Writer output) throws IOException { char[] buffer = new char[DEFAULT_BUFFER_SIZE]; int count = 0; int n = 0; while (-1 != (n = input.read(buffer))) { output.write(buffer, 0, n); count += n; } return count; } /** * this method takes a long (less than 62) and converts it to a 1 character * string (a-z, A-Z, 0-9) * * @param theLong * is the long (less than 62) to convert to a 1 character string * * @return a one character string */ public static String convertLongToChar(long theLong) { if ((theLong < 0) || (theLong >= 62)) { throw new RuntimeException("convertLongToChar() " + " invalid input (not >=0 && <62: " + theLong); } else if (theLong < 26) { return "" + (char) ('a' + theLong); } else if (theLong < 52) { return "" + (char) ('A' + (theLong - 26)); } else { return "" + (char) ('0' + (theLong - 52)); } } /** * this method takes a long (less than 36) and converts it to a 1 character * string (A-Z, 0-9) * * @param theLong * is the long (less than 36) to convert to a 1 character string * * @return a one character string */ public static String convertLongToCharSmall(long theLong) { if ((theLong < 0) || (theLong >= 36)) { throw new RuntimeException("convertLongToCharSmall() " + " invalid input (not >=0 && <36: " + theLong); } else if (theLong < 26) { return "" + (char) ('A' + theLong); } else { return "" + (char) ('0' + (theLong - 26)); } } /** * convert a long to a string by converting it to base 62 (26 lower, 26 upper, * 10 digits) * * @param theLong * is the long to convert * * @return the String conversion of this */ public static String convertLongToString(long theLong) { long quotient = theLong / 62; long remainder = theLong % 62; if (quotient == 0) { return convertLongToChar(remainder); } StringBuffer result = new StringBuffer(); result.append(convertLongToString(quotient)); result.append(convertLongToChar(remainder)); return result.toString(); } /** * convert a long to a string by converting it to base 36 (26 upper, 10 * digits) * * @param theLong * is the long to convert * * @return the String conversion of this */ public static String convertLongToStringSmall(long theLong) { long quotient = theLong / 36; long remainder = theLong % 36; if (quotient == 0) { return convertLongToCharSmall(remainder); } StringBuffer result = new StringBuffer(); result.append(convertLongToStringSmall(quotient)); result.append(convertLongToCharSmall(remainder)); return result.toString(); } /** * increment a character (A-Z then 0-9) * * @param theChar * * @return the value */ public static char incrementChar(char theChar) { if (theChar == 'Z') { return '0'; } if (theChar == '9') { return 'A'; } return ++theChar; } /** * see if we are running on windows * @return true if windows */ public static boolean isWindows() { String osname = defaultString(System.getProperty("os.name")); if (contains(osname.toLowerCase(), "windows")) { return true; } return false; } /** * Increment a string with A-Z and 0-9 (no lower case so case insensitive apps * like windows IE will still work) * * @param string * * @return the value */ public static char[] incrementStringInt(char[] string) { if (string == null) { return string; } //loop through the string backwards int i = 0; for (i = string.length - 1; i >= 0; i--) { char inc = string[i]; inc = incrementChar(inc); string[i] = inc; if (inc != 'A') { break; } } //if we are at 0, then it means we hit AAAAAAA (or more) if (i < 0) { return ("A" + new String(string)).toCharArray(); } return string; } /** * read properties from a resource, dont modify the properties returned since they are cached * @param resourceName * @return the properties */ public synchronized static Properties propertiesFromResourceName(String resourceName) { return propertiesFromResourceName(resourceName, true, true, null, null); } /** * logger */ private static Log LOG = GrouperInstallerUtils.retrieveLog(GrouperInstallerUtils.class); /** * clear properties cache (e.g. for testing) */ public static void propertiesCacheClear() { resourcePropertiesCache.clear(); } /** * properties from file * @param propertiesFile * @return properties */ public static Properties propertiesFromFile(File propertiesFile) { Properties properties = new Properties(); FileInputStream fileInputStream = null; try { fileInputStream = new FileInputStream(propertiesFile); properties.load(fileInputStream); return properties; } catch (IOException ioe) { throw new RuntimeException("Probably reading properties from file: " + (propertiesFile == null ? null : propertiesFile.getAbsolutePath()), ioe); } finally { closeQuietly(fileInputStream); } } /** * read properties from a resource, dont modify the properties returned since they are cached * @param resourceName * @param useCache * @param exceptionIfNotExist * @param classInJar if not null, then look for the jar where this file is, and look in the same dir * @param callingLog * @return the properties or null if not exist */ public synchronized static Properties propertiesFromResourceName(String resourceName, boolean useCache, boolean exceptionIfNotExist, Class classInJar, StringBuilder callingLog) { Properties properties = resourcePropertiesCache.get(resourceName); if (!useCache || !resourcePropertiesCache.containsKey(resourceName)) { properties = new Properties(); boolean success = false; URL url = computeUrl(resourceName, true); InputStream inputStream = null; try { inputStream = url.openStream(); properties.load(inputStream); success = true; String theLog = "Reading resource: " + resourceName + ", from: " + url.toURI(); if (LOG != null) { LOG.debug(theLog); } if (callingLog != null) { callingLog.append(theLog); } } catch (Exception e) { //clear out just in case properties.clear(); //lets look next to jar File jarFile = classInJar == null ? null : jarFile(classInJar); File parentDir = jarFile == null ? null : jarFile.getParentFile(); String fileName = parentDir == null ? null : (stripLastSlashIfExists(fileCanonicalPath(parentDir)) + File.separator + resourceName); File configFile = fileName == null ? null : new File(fileName); try { //looks like we have a match if (configFile != null && configFile.exists() && configFile.isFile()) { inputStream = new FileInputStream(configFile); properties.load(inputStream); success = true; String theLog = "Reading resource: " + resourceName + ", from: " + fileCanonicalPath(configFile); if (LOG != null) { LOG.debug(theLog); } if (callingLog != null) { callingLog.append(theLog); } } } catch (Exception e2) { if (LOG != null) { LOG.debug("Error reading from file for resource: " + resourceName + ", file: " + fileName, e2); } } if (!success) { properties = null; if (exceptionIfNotExist) { throw new RuntimeException("Problem with resource: '" + resourceName + "'", e); } } } finally { closeQuietly(inputStream); if (useCache && properties != null && properties.size() > 0) { resourcePropertiesCache.put(resourceName, properties); } } } return properties; } /** * do a case-insensitive matching * @param theEnumClass class of the enum * @param generic type * * @param string * @param exceptionOnNotFound true if exception should be thrown on not found * @return the enum or null or exception if not found * @throws RuntimeException if there is a problem */ public static > E enumValueOfIgnoreCase(Class theEnumClass, String string, boolean exceptionOnNotFound) throws RuntimeException { return enumValueOfIgnoreCase(theEnumClass, string, exceptionOnNotFound, true); } /** * do a case-insensitive matching * @param theEnumClass class of the enum * @param generic type * * @param string * @param exceptionOnNotFound true if exception should be thrown on not found * @param exceptionIfInvalid if there is a string, but it is invalid, if should throw exception * @return the enum or null or exception if not found * @throws RuntimeException if there is a problem */ public static > E enumValueOfIgnoreCase(Class theEnumClass, String string, boolean exceptionOnNotFound, boolean exceptionIfInvalid) throws RuntimeException { if (!exceptionOnNotFound && isBlank(string)) { return null; } for (E e : theEnumClass.getEnumConstants()) { if (equalsIgnoreCase(string, e.name())) { return e; } } if (!exceptionIfInvalid) { return null; } StringBuilder error = new StringBuilder( "Cant find " + theEnumClass.getSimpleName() + " from string: '").append(string); error.append("', expecting one of: "); for (E e : theEnumClass.getEnumConstants()) { error.append(e.name()).append(", "); } throw new RuntimeException(error.toString()); } /** * this assumes the property exists, and is a simple property * @param object * @param property * @return the value */ public static Object propertyValue(Object object, String property) { Method getter = getter(object.getClass(), property, true, true); Object result = invokeMethod(getter, object); return result; } /** * get a value (trimmed to e) from a property file * @param properties * @param key * @return the property value */ public static String propertiesValue(Properties properties, String key) { return propertiesValue(properties, null, key); } /** * get a value (trimmed to e) from a property file * @param properties * @param overrideMap for testing, to override some properties values * @param key * @return the property value */ public static String propertiesValue(Properties properties, Map overrideMap, String key) { return propertiesValue(properties, overrideMap, null, key); } /** * get a value (trimmed to e) from a property file * @param properties * @param overrideMap for testing or threadlocal, to override some properties values * @param overrideMap2 for testing, to provide some properties values * @param key * @return the property value */ public static String propertiesValue(Properties properties, Map overrideMap, Map overrideMap2, String key) { String value = overrideMap == null ? null : overrideMap.get(key); if (isBlank(value)) { value = overrideMap2 == null ? null : overrideMap2.get(key); } if (isBlank(value)) { value = properties.getProperty(key); } value = trim(value); value = substituteCommonVars(value); return value; } /** * substitute common vars like $space$ and $newline$ * @param string * @return the string */ public static String substituteCommonVars(String string) { if (string == null) { return string; } //short circuit if (string.indexOf('$') < 0) { return string; } //might have $space$ string = GrouperInstallerUtils.replace(string, "$space$", " "); //note, at some point we could be OS specific string = GrouperInstallerUtils.replace(string, "$newline$", "\n"); return string; } /** * get a boolean property, or the default if cant find * @param properties * @param propertyName * @param defaultValue * @return the boolean */ public static boolean propertiesValueBoolean(Properties properties, String propertyName, boolean defaultValue) { return propertiesValueBoolean(properties, null, propertyName, defaultValue); } /** * get a boolean property, or the default if cant find. Validate also with a descriptive exception if problem * @param resourceName * @param properties * @param overrideMap for testing to override properties * @param propertyName * @param defaultValue * @param required * @return the boolean */ public static boolean propertiesValueBoolean(String resourceName, Properties properties, Map overrideMap, String propertyName, boolean defaultValue, boolean required) { propertyValidateValueBoolean(resourceName, properties, overrideMap, propertyName, required, true); Map threadLocalMap = propertiesThreadLocalOverrideMap(resourceName); return propertiesValueBoolean(properties, threadLocalMap, overrideMap, propertyName, defaultValue); } /** * get an int property, or the default if cant find. Validate also with a descriptive exception if problem * @param resourceName * @param properties * @param overrideMap for testing to override properties * @param propertyName * @param defaultValue * @param required * @return the int */ public static int propertiesValueInt(String resourceName, Properties properties, Map overrideMap, String propertyName, int defaultValue, boolean required) { propertyValidateValueInt(resourceName, properties, overrideMap, propertyName, required, true); Map threadLocalMap = propertiesThreadLocalOverrideMap(resourceName); return propertiesValueInt(properties, threadLocalMap, overrideMap, propertyName, defaultValue); } /** * get a boolean property, or the default if cant find. Validate also with a descriptive exception if problem * @param resourceName * @param properties * @param overrideMap for threadlocal or testing to override properties * @param propertyName * @param required * @return the string */ public static String propertiesValue(String resourceName, Properties properties, Map overrideMap, String propertyName, boolean required) { if (required) { propertyValidateValueRequired(resourceName, properties, overrideMap, propertyName, true); } Map threadLocalMap = propertiesThreadLocalOverrideMap(resourceName); return propertiesValue(properties, threadLocalMap, overrideMap, propertyName); } /** * get a int property, or the default if cant find * @param properties * @param overrideMap for testing to override properties * @param propertyName * @param defaultValue * @return the int */ public static int propertiesValueInt(Properties properties, Map overrideMap, String propertyName, int defaultValue) { return propertiesValueInt(properties, overrideMap, null, propertyName, defaultValue); } /** * get a int property, or the default if cant find * @param properties * @param overrideMap for testing to override properties * @param overrideMap2 * @param propertyName * @param defaultValue * @return the int */ public static int propertiesValueInt(Properties properties, Map overrideMap, Map overrideMap2, String propertyName, int defaultValue) { String value = propertiesValue(properties, overrideMap, overrideMap2, propertyName); if (isBlank(value)) { return defaultValue; } try { return intValue(value); } catch (Exception e) {} throw new RuntimeException("Invalid int value: '" + value + "' for property: " + propertyName + " in grouper.properties"); } /** * get a boolean property, or the default if cant find * @param properties * @param overrideMap for testing to override properties * @param propertyName * @param defaultValue * @return the boolean */ public static boolean propertiesValueBoolean(Properties properties, Map overrideMap, String propertyName, boolean defaultValue) { return propertiesValueBoolean(properties, overrideMap, null, propertyName, defaultValue); } /** * get a boolean property, or the default if cant find * @param properties * @param overrideMap for testing or threadlocal to override properties * @param overrideMap2 for testing or threadlocal to override properties * @param propertyName * @param defaultValue * @return the boolean */ public static boolean propertiesValueBoolean(Properties properties, Map overrideMap, Map overrideMap2, String propertyName, boolean defaultValue) { String value = propertiesValue(properties, overrideMap, overrideMap2, propertyName); if (isBlank(value)) { return defaultValue; } if ("true".equalsIgnoreCase(value)) { return true; } if ("false".equalsIgnoreCase(value)) { return false; } if ("t".equalsIgnoreCase(value)) { return true; } if ("f".equalsIgnoreCase(value)) { return false; } throw new RuntimeException("Invalid boolean value: '" + value + "' for property: " + propertyName + " in properties file"); } /** * close a connection null safe and dont throw exception * @param connection */ public static void closeQuietly(Connection connection) { if (connection != null) { try { connection.close(); } catch (Exception e) { //ignore } } } /** * close a statement null safe and dont throw exception * @param statement */ public static void closeQuietly(Statement statement) { if (statement != null) { try { statement.close(); } catch (Exception e) { //ignore } } } /** * close a resultSet null safe and dont throw exception * @param resultSet */ public static void closeQuietly(ResultSet resultSet) { if (resultSet != null) { try { resultSet.close(); } catch (Exception e) { //ignore } } } /** cache the hostname, it wont change */ private static String hostname = null; /** * get the hostname of this machine * @return the hostname */ public static String hostname() { if (isBlank(hostname)) { //get the hostname hostname = "unknown"; try { InetAddress addr = InetAddress.getLocalHost(); // Get hostname hostname = addr.getHostName(); } catch (Exception e) { System.err.println("Cant find servers hostname: "); e.printStackTrace(); } } return hostname; } /** * is ascii char * @param input * @return true if ascii */ public static boolean isAscii(char input) { return input < 128; } /** * find the length of ascii chars (non ascii are counted as two) * @param input * @return the length of ascii chars */ public static int lengthAscii(String input) { if (input == null) { return 0; } //see what real length is int utfLength = input.length(); //count how many non asciis int extras = 0; for (int i=0;i requiredLength) { //do not include the current char return input.substring(0,i); } } //must have fit return input; } /** * if the input is a file, read string from file. if not, or if disabled from grouper.properties, return the input * @param in * @param disableExternalFileLookup * @return the result */ public static String readFromFileIfFile(String in, boolean disableExternalFileLookup) { String theIn = in; //convert both slashes to file slashes if (File.separatorChar == '/') { theIn = replace(theIn, "\\", "/"); } else { theIn = replace(theIn, "/", "\\"); } //see if it is a file reference if (theIn.indexOf(File.separatorChar) != -1 && !disableExternalFileLookup) { //read the contents of the file into a string theIn = readFileIntoString(new File(theIn)); return theIn; } return in; } /** * Create directories, throw exception if not possible. * This is will be ok if the directory already exists (will not delete it) * @param dir */ public static void mkdirs(File dir) { if (!dir.exists()) { if (!dir.mkdirs()) { throw new RuntimeException("Could not create directory : " + dir.getParentFile()); } return; } if (!dir.isDirectory()) { throw new RuntimeException("Should be a directory but is not: " + dir); } } /** * null safe string compare * @param first * @param second * @return true if equal */ public static boolean equals(String first, String second) { if (first == second) { return true; } if (first == null || second == null) { return false; } return first.equals(second); } /** *

Checks if a String is whitespace, empty ("") or null.

* *
   * isBlank(null)      = true
   * isBlank("")        = true
   * isBlank(" ")       = true
   * isBlank("bob")     = false
   * isBlank("  bob  ") = false
   * 
* * @param str the String to check, may be null * @return true if the String is null, empty or whitespace * @since 2.0 */ public static boolean isBlank(String str) { int strLen; if (str == null || (strLen = str.length()) == 0) { return true; } for (int i = 0; i < strLen; i++) { if ((Character.isWhitespace(str.charAt(i)) == false)) { return false; } } return true; } /** * * @param str * @return true if not blank */ public static boolean isNotBlank(String str) { return !isBlank(str); } /** * trim whitespace from string * @param str * @return trimmed string */ public static String trim(String str) { return str == null ? null : str.trim(); } /** * equalsignorecase * @param str1 * @param str2 * @return true if the strings are equal ignore case */ public static boolean equalsIgnoreCase(String str1, String str2) { return str1 == null ? str2 == null : str1.equalsIgnoreCase(str2); } /** * trim to empty, convert null to empty * @param str * @return trimmed */ public static String trimToEmpty(String str) { return str == null ? "" : str.trim(); } /** *

Abbreviates a String using ellipses. This will turn * "Now is the time for all good men" into "Now is the time for..."

* *

Specifically: *

    *
  • If str is less than maxWidth characters * long, return it.
  • *
  • Else abbreviate it to (substring(str, 0, max-3) + "...").
  • *
  • If maxWidth is less than 4, throw an * IllegalArgumentException.
  • *
  • In no case will it return a String of length greater than * maxWidth.
  • *
*

* *
   * StringUtils.abbreviate(null, *)      = null
   * StringUtils.abbreviate("", 4)        = ""
   * StringUtils.abbreviate("abcdefg", 6) = "abc..."
   * StringUtils.abbreviate("abcdefg", 7) = "abcdefg"
   * StringUtils.abbreviate("abcdefg", 8) = "abcdefg"
   * StringUtils.abbreviate("abcdefg", 4) = "a..."
   * StringUtils.abbreviate("abcdefg", 3) = IllegalArgumentException
   * 
* * @param str the String to check, may be null * @param maxWidth maximum length of result String, must be at least 4 * @return abbreviated String, null if null String input * @throws IllegalArgumentException if the width is too small * @since 2.0 */ public static String abbreviate(String str, int maxWidth) { return abbreviate(str, 0, maxWidth); } /** *

Abbreviates a String using ellipses. This will turn * "Now is the time for all good men" into "...is the time for..."

* *

Works like abbreviate(String, int), but allows you to specify * a "left edge" offset. Note that this left edge is not necessarily going to * be the leftmost character in the result, or the first character following the * ellipses, but it will appear somewhere in the result. * *

In no case will it return a String of length greater than * maxWidth.

* *
   * StringUtils.abbreviate(null, *, *)                = null
   * StringUtils.abbreviate("", 0, 4)                  = ""
   * StringUtils.abbreviate("abcdefghijklmno", -1, 10) = "abcdefg..."
   * StringUtils.abbreviate("abcdefghijklmno", 0, 10)  = "abcdefg..."
   * StringUtils.abbreviate("abcdefghijklmno", 1, 10)  = "abcdefg..."
   * StringUtils.abbreviate("abcdefghijklmno", 4, 10)  = "abcdefg..."
   * StringUtils.abbreviate("abcdefghijklmno", 5, 10)  = "...fghi..."
   * StringUtils.abbreviate("abcdefghijklmno", 6, 10)  = "...ghij..."
   * StringUtils.abbreviate("abcdefghijklmno", 8, 10)  = "...ijklmno"
   * StringUtils.abbreviate("abcdefghijklmno", 10, 10) = "...ijklmno"
   * StringUtils.abbreviate("abcdefghijklmno", 12, 10) = "...ijklmno"
   * StringUtils.abbreviate("abcdefghij", 0, 3)        = IllegalArgumentException
   * StringUtils.abbreviate("abcdefghij", 5, 6)        = IllegalArgumentException
   * 
* * @param str the String to check, may be null * @param offset left edge of source String * @param maxWidth maximum length of result String, must be at least 4 * @return abbreviated String, null if null String input * @throws IllegalArgumentException if the width is too small * @since 2.0 */ public static String abbreviate(String str, int offset, int maxWidth) { if (str == null) { return null; } if (maxWidth < 4) { throw new IllegalArgumentException("Minimum abbreviation width is 4"); } if (str.length() <= maxWidth) { return str; } if (offset > str.length()) { offset = str.length(); } if ((str.length() - offset) < (maxWidth - 3)) { offset = str.length() - (maxWidth - 3); } if (offset <= 4) { return str.substring(0, maxWidth - 3) + "..."; } if (maxWidth < 7) { throw new IllegalArgumentException("Minimum abbreviation width with offset is 7"); } if ((offset + (maxWidth - 3)) < str.length()) { return "..." + abbreviate(str.substring(offset), maxWidth - 3); } return "..." + str.substring(str.length() - (maxWidth - 3)); } // Splitting //----------------------------------------------------------------------- /** *

Splits the provided text into an array, using whitespace as the * separator. * Whitespace is defined by {@link Character#isWhitespace(char)}.

* *

The separator is not included in the returned String array. * Adjacent separators are treated as one separator. * For more control over the split use the StrTokenizer class.

* *

A null input String returns null.

* *
   * StringUtils.split(null)       = null
   * StringUtils.split("")         = []
   * StringUtils.split("abc def")  = ["abc", "def"]
   * StringUtils.split("abc  def") = ["abc", "def"]
   * StringUtils.split(" abc ")    = ["abc"]
   * 
* * @param str the String to parse, may be null * @return an array of parsed Strings, null if null String input */ public static String[] split(String str) { return split(str, null, -1); } /** *

Splits the provided text into an array, separator specified. * This is an alternative to using StringTokenizer.

* *

The separator is not included in the returned String array. * Adjacent separators are treated as one separator. * For more control over the split use the StrTokenizer class.

* *

A null input String returns null.

* *
   * StringUtils.split(null, *)         = null
   * StringUtils.split("", *)           = []
   * StringUtils.split("a.b.c", '.')    = ["a", "b", "c"]
   * StringUtils.split("a..b.c", '.')   = ["a", "b", "c"]
   * StringUtils.split("a:b:c", '.')    = ["a:b:c"]
   * StringUtils.split("a\tb\nc", null) = ["a", "b", "c"]
   * StringUtils.split("a b c", ' ')    = ["a", "b", "c"]
   * 
* * @param str the String to parse, may be null * @param separatorChar the character used as the delimiter, * null splits on whitespace * @return an array of parsed Strings, null if null String input * @since 2.0 */ public static String[] split(String str, char separatorChar) { return splitWorker(str, separatorChar, false); } /** *

Splits the provided text into an array, separators specified. * This is an alternative to using StringTokenizer.

* *

The separator is not included in the returned String array. * Adjacent separators are treated as one separator. * For more control over the split use the StrTokenizer class.

* *

A null input String returns null. * A null separatorChars splits on whitespace.

* *
   * StringUtils.split(null, *)         = null
   * StringUtils.split("", *)           = []
   * StringUtils.split("abc def", null) = ["abc", "def"]
   * StringUtils.split("abc def", " ")  = ["abc", "def"]
   * StringUtils.split("abc  def", " ") = ["abc", "def"]
   * StringUtils.split("ab:cd:ef", ":") = ["ab", "cd", "ef"]
   * 
* * @param str the String to parse, may be null * @param separatorChars the characters used as the delimiters, * null splits on whitespace * @return an array of parsed Strings, null if null String input */ public static String[] split(String str, String separatorChars) { return splitWorker(str, separatorChars, -1, false); } /** *

Splits the provided text into an array with a maximum length, * separators specified.

* *

The separator is not included in the returned String array. * Adjacent separators are treated as one separator.

* *

A null input String returns null. * A null separatorChars splits on whitespace.

* *

If more than max delimited substrings are found, the last * returned string includes all characters after the first max - 1 * returned strings (including separator characters).

* *
   * StringUtils.split(null, *, *)            = null
   * StringUtils.split("", *, *)              = []
   * StringUtils.split("ab de fg", null, 0)   = ["ab", "cd", "ef"]
   * StringUtils.split("ab   de fg", null, 0) = ["ab", "cd", "ef"]
   * StringUtils.split("ab:cd:ef", ":", 0)    = ["ab", "cd", "ef"]
   * StringUtils.split("ab:cd:ef", ":", 2)    = ["ab", "cd:ef"]
   * 
* * @param str the String to parse, may be null * @param separatorChars the characters used as the delimiters, * null splits on whitespace * @param max the maximum number of elements to include in the * array. A zero or negative value implies no limit * @return an array of parsed Strings, null if null String input */ public static String[] split(String str, String separatorChars, int max) { return splitWorker(str, separatorChars, max, false); } /** *

Splits the provided text into an array, separator string specified.

* *

The separator(s) will not be included in the returned String array. * Adjacent separators are treated as one separator.

* *

A null input String returns null. * A null separator splits on whitespace.

* *
   * StringUtils.split(null, *)            = null
   * StringUtils.split("", *)              = []
   * StringUtils.split("ab de fg", null)   = ["ab", "de", "fg"]
   * StringUtils.split("ab   de fg", null) = ["ab", "de", "fg"]
   * StringUtils.split("ab:cd:ef", ":")    = ["ab", "cd", "ef"]
   * StringUtils.split("abstemiouslyaeiouyabstemiously", "aeiouy")  = ["bst", "m", "sl", "bst", "m", "sl"]
   * StringUtils.split("abstemiouslyaeiouyabstemiously", "aeiouy")  = ["abstemiously", "abstemiously"]
   * 
* * @param str the String to parse, may be null * @param separator String containing the String to be used as a delimiter, * null splits on whitespace * @return an array of parsed Strings, null if null String was input */ public static String[] splitByWholeSeparator(String str, String separator) { return splitByWholeSeparator(str, separator, -1); } /** *

Splits the provided text into an array, separator string specified. * Returns a maximum of max substrings.

* *

The separator(s) will not be included in the returned String array. * Adjacent separators are treated as one separator.

* *

A null input String returns null. * A null separator splits on whitespace.

* *
   * StringUtils.splitByWholeSeparator(null, *, *)               = null
   * StringUtils.splitByWholeSeparator("", *, *)                 = []
   * StringUtils.splitByWholeSeparator("ab de fg", null, 0)      = ["ab", "de", "fg"]
   * StringUtils.splitByWholeSeparator("ab   de fg", null, 0)    = ["ab", "de", "fg"]
   * StringUtils.splitByWholeSeparator("ab:cd:ef", ":", 2)       = ["ab", "cd"]
   * StringUtils.splitByWholeSeparator("abstemiouslyaeiouyabstemiously", "aeiouy", 2) = ["bst", "m"]
   * StringUtils.splitByWholeSeparator("abstemiouslyaeiouyabstemiously", "aeiouy", 2)  = ["abstemiously", "abstemiously"]
   * 
* * @param str the String to parse, may be null * @param separator String containing the String to be used as a delimiter, * null splits on whitespace * @param max the maximum number of elements to include in the returned * array. A zero or negative value implies no limit. * @return an array of parsed Strings, null if null String was input */ @SuppressWarnings("unchecked") public static String[] splitByWholeSeparator(String str, String separator, int max) { if (str == null) { return null; } int len = str.length(); if (len == 0) { return EMPTY_STRING_ARRAY; } if ((separator == null) || ("".equals(separator))) { // Split on whitespace. return split(str, null, max); } int separatorLength = separator.length(); ArrayList substrings = new ArrayList(); int numberOfSubstrings = 0; int beg = 0; int end = 0; while (end < len) { end = str.indexOf(separator, beg); if (end > -1) { if (end > beg) { numberOfSubstrings += 1; if (numberOfSubstrings == max) { end = len; substrings.add(str.substring(beg)); } else { // The following is OK, because String.substring( beg, end ) excludes // the character at the position 'end'. substrings.add(str.substring(beg, end)); // Set the starting point for the next search. // The following is equivalent to beg = end + (separatorLength - 1) + 1, // which is the right calculation: beg = end + separatorLength; } } else { // We found a consecutive occurrence of the separator, so skip it. beg = end + separatorLength; } } else { // String.substring( beg ) goes from 'beg' to the end of the String. substrings.add(str.substring(beg)); end = len; } } return (String[]) substrings.toArray(new String[substrings.size()]); } //----------------------------------------------------------------------- /** *

Splits the provided text into an array, using whitespace as the * separator, preserving all tokens, including empty tokens created by * adjacent separators. This is an alternative to using StringTokenizer. * Whitespace is defined by {@link Character#isWhitespace(char)}.

* *

The separator is not included in the returned String array. * Adjacent separators are treated as separators for empty tokens. * For more control over the split use the StrTokenizer class.

* *

A null input String returns null.

* *
   * StringUtils.splitPreserveAllTokens(null)       = null
   * StringUtils.splitPreserveAllTokens("")         = []
   * StringUtils.splitPreserveAllTokens("abc def")  = ["abc", "def"]
   * StringUtils.splitPreserveAllTokens("abc  def") = ["abc", "", "def"]
   * StringUtils.splitPreserveAllTokens(" abc ")    = ["", "abc", ""]
   * 
* * @param str the String to parse, may be null * @return an array of parsed Strings, null if null String input * @since 2.1 */ public static String[] splitPreserveAllTokens(String str) { return splitWorker(str, null, -1, true); } /** *

Splits the provided text into an array, separator specified, * preserving all tokens, including empty tokens created by adjacent * separators. This is an alternative to using StringTokenizer.

* *

The separator is not included in the returned String array. * Adjacent separators are treated as separators for empty tokens. * For more control over the split use the StrTokenizer class.

* *

A null input String returns null.

* *
   * StringUtils.splitPreserveAllTokens(null, *)         = null
   * StringUtils.splitPreserveAllTokens("", *)           = []
   * StringUtils.splitPreserveAllTokens("a.b.c", '.')    = ["a", "b", "c"]
   * StringUtils.splitPreserveAllTokens("a..b.c", '.')   = ["a", "b", "c"]
   * StringUtils.splitPreserveAllTokens("a:b:c", '.')    = ["a:b:c"]
   * StringUtils.splitPreserveAllTokens("a\tb\nc", null) = ["a", "b", "c"]
   * StringUtils.splitPreserveAllTokens("a b c", ' ')    = ["a", "b", "c"]
   * StringUtils.splitPreserveAllTokens("a b c ", ' ')   = ["a", "b", "c", ""]
   * StringUtils.splitPreserveAllTokens("a b c ", ' ')   = ["a", "b", "c", "", ""]
   * StringUtils.splitPreserveAllTokens(" a b c", ' ')   = ["", a", "b", "c"]
   * StringUtils.splitPreserveAllTokens("  a b c", ' ')  = ["", "", a", "b", "c"]
   * StringUtils.splitPreserveAllTokens(" a b c ", ' ')  = ["", a", "b", "c", ""]
   * 
* * @param str the String to parse, may be null * @param separatorChar the character used as the delimiter, * null splits on whitespace * @return an array of parsed Strings, null if null String input * @since 2.1 */ public static String[] splitPreserveAllTokens(String str, char separatorChar) { return splitWorker(str, separatorChar, true); } /** * Performs the logic for the split and * splitPreserveAllTokens methods that do not return a * maximum array length. * * @param str the String to parse, may be null * @param separatorChar the separate character * @param preserveAllTokens if true, adjacent separators are * treated as empty token separators; if false, adjacent * separators are treated as one separator. * @return an array of parsed Strings, null if null String input */ @SuppressWarnings("unchecked") private static String[] splitWorker(String str, char separatorChar, boolean preserveAllTokens) { // Performance tuned for 2.0 (JDK1.4) if (str == null) { return null; } int len = str.length(); if (len == 0) { return EMPTY_STRING_ARRAY; } List list = new ArrayList(); int i = 0, start = 0; boolean match = false; boolean lastMatch = false; while (i < len) { if (str.charAt(i) == separatorChar) { if (match || preserveAllTokens) { list.add(str.substring(start, i)); match = false; lastMatch = true; } start = ++i; continue; } lastMatch = false; match = true; i++; } if (match || (preserveAllTokens && lastMatch)) { list.add(str.substring(start, i)); } return (String[]) list.toArray(new String[list.size()]); } /** *

Splits the provided text into an array, separators specified, * preserving all tokens, including empty tokens created by adjacent * separators. This is an alternative to using StringTokenizer.

* *

The separator is not included in the returned String array. * Adjacent separators are treated as separators for empty tokens. * For more control over the split use the StrTokenizer class.

* *

A null input String returns null. * A null separatorChars splits on whitespace.

* *
   * StringUtils.splitPreserveAllTokens(null, *)           = null
   * StringUtils.splitPreserveAllTokens("", *)             = []
   * StringUtils.splitPreserveAllTokens("abc def", null)   = ["abc", "def"]
   * StringUtils.splitPreserveAllTokens("abc def", " ")    = ["abc", "def"]
   * StringUtils.splitPreserveAllTokens("abc  def", " ")   = ["abc", "", def"]
   * StringUtils.splitPreserveAllTokens("ab:cd:ef", ":")   = ["ab", "cd", "ef"]
   * StringUtils.splitPreserveAllTokens("ab:cd:ef:", ":")  = ["ab", "cd", "ef", ""]
   * StringUtils.splitPreserveAllTokens("ab:cd:ef::", ":") = ["ab", "cd", "ef", "", ""]
   * StringUtils.splitPreserveAllTokens("ab::cd:ef", ":")  = ["ab", "", cd", "ef"]
   * StringUtils.splitPreserveAllTokens(":cd:ef", ":")     = ["", cd", "ef"]
   * StringUtils.splitPreserveAllTokens("::cd:ef", ":")    = ["", "", cd", "ef"]
   * StringUtils.splitPreserveAllTokens(":cd:ef:", ":")    = ["", cd", "ef", ""]
   * 
* * @param str the String to parse, may be null * @param separatorChars the characters used as the delimiters, * null splits on whitespace * @return an array of parsed Strings, null if null String input * @since 2.1 */ public static String[] splitPreserveAllTokens(String str, String separatorChars) { return splitWorker(str, separatorChars, -1, true); } /** *

Splits the provided text into an array with a maximum length, * separators specified, preserving all tokens, including empty tokens * created by adjacent separators.

* *

The separator is not included in the returned String array. * Adjacent separators are treated as separators for empty tokens. * Adjacent separators are treated as one separator.

* *

A null input String returns null. * A null separatorChars splits on whitespace.

* *

If more than max delimited substrings are found, the last * returned string includes all characters after the first max - 1 * returned strings (including separator characters).

* *
   * StringUtils.splitPreserveAllTokens(null, *, *)            = null
   * StringUtils.splitPreserveAllTokens("", *, *)              = []
   * StringUtils.splitPreserveAllTokens("ab de fg", null, 0)   = ["ab", "cd", "ef"]
   * StringUtils.splitPreserveAllTokens("ab   de fg", null, 0) = ["ab", "cd", "ef"]
   * StringUtils.splitPreserveAllTokens("ab:cd:ef", ":", 0)    = ["ab", "cd", "ef"]
   * StringUtils.splitPreserveAllTokens("ab:cd:ef", ":", 2)    = ["ab", "cd:ef"]
   * StringUtils.splitPreserveAllTokens("ab   de fg", null, 2) = ["ab", "  de fg"]
   * StringUtils.splitPreserveAllTokens("ab   de fg", null, 3) = ["ab", "", " de fg"]
   * StringUtils.splitPreserveAllTokens("ab   de fg", null, 4) = ["ab", "", "", "de fg"]
   * 
* * @param str the String to parse, may be null * @param separatorChars the characters used as the delimiters, * null splits on whitespace * @param max the maximum number of elements to include in the * array. A zero or negative value implies no limit * @return an array of parsed Strings, null if null String input * @since 2.1 */ public static String[] splitPreserveAllTokens(String str, String separatorChars, int max) { return splitWorker(str, separatorChars, max, true); } /** * Performs the logic for the split and * splitPreserveAllTokens methods that return a maximum array * length. * * @param str the String to parse, may be null * @param separatorChars the separate character * @param max the maximum number of elements to include in the * array. A zero or negative value implies no limit. * @param preserveAllTokens if true, adjacent separators are * treated as empty token separators; if false, adjacent * separators are treated as one separator. * @return an array of parsed Strings, null if null String input */ @SuppressWarnings("unchecked") private static String[] splitWorker(String str, String separatorChars, int max, boolean preserveAllTokens) { // Performance tuned for 2.0 (JDK1.4) // Direct code is quicker than StringTokenizer. // Also, StringTokenizer uses isSpace() not isWhitespace() if (str == null) { return null; } int len = str.length(); if (len == 0) { return EMPTY_STRING_ARRAY; } List list = new ArrayList(); int sizePlus1 = 1; int i = 0, start = 0; boolean match = false; boolean lastMatch = false; if (separatorChars == null) { // Null separator means use whitespace while (i < len) { if (Character.isWhitespace(str.charAt(i))) { if (match || preserveAllTokens) { lastMatch = true; if (sizePlus1++ == max) { i = len; lastMatch = false; } list.add(str.substring(start, i)); match = false; } start = ++i; continue; } lastMatch = false; match = true; i++; } } else if (separatorChars.length() == 1) { // Optimise 1 character case char sep = separatorChars.charAt(0); while (i < len) { if (str.charAt(i) == sep) { if (match || preserveAllTokens) { lastMatch = true; if (sizePlus1++ == max) { i = len; lastMatch = false; } list.add(str.substring(start, i)); match = false; } start = ++i; continue; } lastMatch = false; match = true; i++; } } else { // standard case while (i < len) { if (separatorChars.indexOf(str.charAt(i)) >= 0) { if (match || preserveAllTokens) { lastMatch = true; if (sizePlus1++ == max) { i = len; lastMatch = false; } list.add(str.substring(start, i)); match = false; } start = ++i; continue; } lastMatch = false; match = true; i++; } } if (match || (preserveAllTokens && lastMatch)) { list.add(str.substring(start, i)); } return (String[]) list.toArray(new String[list.size()]); } // Joining //----------------------------------------------------------------------- /** *

Joins the elements of the provided array into a single String * containing the provided list of elements.

* *

No separator is added to the joined String. * Null objects or empty strings within the array are represented by * empty strings.

* *
   * StringUtils.join(null)            = null
   * StringUtils.join([])              = ""
   * StringUtils.join([null])          = ""
   * StringUtils.join(["a", "b", "c"]) = "abc"
   * StringUtils.join([null, "", "a"]) = "a"
   * 
* * @param array the array of values to join together, may be null * @return the joined String, null if null array input * @since 2.0 */ public static String join(Object[] array) { return join(array, null); } /** *

Joins the elements of the provided array into a single String * containing the provided list of elements.

* *

No delimiter is added before or after the list. * Null objects or empty strings within the array are represented by * empty strings.

* *
   * StringUtils.join(null, *)               = null
   * StringUtils.join([], *)                 = ""
   * StringUtils.join([null], *)             = ""
   * StringUtils.join(["a", "b", "c"], ';')  = "a;b;c"
   * StringUtils.join(["a", "b", "c"], null) = "abc"
   * StringUtils.join([null, "", "a"], ';')  = ";;a"
   * 
* * @param array the array of values to join together, may be null * @param separator the separator character to use * @return the joined String, null if null array input * @since 2.0 */ public static String join(Object[] array, char separator) { if (array == null) { return null; } int arraySize = array.length; int bufSize = (arraySize == 0 ? 0 : ((array[0] == null ? 16 : array[0].toString() .length()) + 1) * arraySize); StringBuffer buf = new StringBuffer(bufSize); for (int i = 0; i < arraySize; i++) { if (i > 0) { buf.append(separator); } if (array[i] != null) { buf.append(array[i]); } } return buf.toString(); } /** *

Joins the elements of the provided array into a single String * containing the provided list of elements.

* *

No delimiter is added before or after the list. * A null separator is the same as an empty String (""). * Null objects or empty strings within the array are represented by * empty strings.

* *
   * StringUtils.join(null, *)                = null
   * StringUtils.join([], *)                  = ""
   * StringUtils.join([null], *)              = ""
   * StringUtils.join(["a", "b", "c"], "--")  = "a--b--c"
   * StringUtils.join(["a", "b", "c"], null)  = "abc"
   * StringUtils.join(["a", "b", "c"], "")    = "abc"
   * StringUtils.join([null, "", "a"], ',')   = ",,a"
   * 
* * @param array the array of values to join together, may be null * @param separator the separator character to use, null treated as "" * @return the joined String, null if null array input */ public static String join(Object[] array, String separator) { if (array == null) { return null; } if (separator == null) { separator = ""; } int arraySize = array.length; // ArraySize == 0: Len = 0 // ArraySize > 0: Len = NofStrings *(len(firstString) + len(separator)) // (Assuming that all Strings are roughly equally long) int bufSize = ((arraySize == 0) ? 0 : arraySize * ((array[0] == null ? 16 : array[0].toString().length()) + separator.length())); StringBuffer buf = new StringBuffer(bufSize); for (int i = 0; i < arraySize; i++) { if (i > 0) { buf.append(separator); } if (array[i] != null) { buf.append(array[i]); } } return buf.toString(); } /** *

Joins the elements of the provided Iterator into * a single String containing the provided elements.

* *

No delimiter is added before or after the list. Null objects or empty * strings within the iteration are represented by empty strings.

* *

See the examples here: {@link #join(Object[],char)}.

* * @param iterator the Iterator of values to join together, may be null * @param separator the separator character to use * @return the joined String, null if null iterator input * @since 2.0 */ public static String join(Iterator iterator, char separator) { if (iterator == null) { return null; } StringBuffer buf = new StringBuffer(256); // Java default is 16, probably too small while (iterator.hasNext()) { Object obj = iterator.next(); if (obj != null) { buf.append(obj); } if (iterator.hasNext()) { buf.append(separator); } } return buf.toString(); } /** *

Joins the elements of the provided Iterator into * a single String containing the provided elements.

* *

No delimiter is added before or after the list. * A null separator is the same as an empty String ("").

* *

See the examples here: {@link #join(Object[],String)}.

* * @param iterator the Iterator of values to join together, may be null * @param separator the separator character to use, null treated as "" * @return the joined String, null if null iterator input */ public static String join(Iterator iterator, String separator) { if (iterator == null) { return null; } StringBuffer buf = new StringBuffer(256); // Java default is 16, probably too small while (iterator.hasNext()) { Object obj = iterator.next(); if (obj != null) { buf.append(obj); } if ((separator != null) && iterator.hasNext()) { buf.append(separator); } } return buf.toString(); } /** *

Returns either the passed in String, * or if the String is null, an empty String ("").

* *
   * StringUtils.defaultString(null)  = ""
   * StringUtils.defaultString("")    = ""
   * StringUtils.defaultString("bat") = "bat"
   * 
* * @see String#valueOf(Object) * @param str the String to check, may be null * @return the passed in String, or the empty String if it * was null */ public static String defaultString(String str) { return str == null ? "" : str; } /** *

Returns either the passed in String, or if the String is * null, the value of defaultStr.

* *
   * StringUtils.defaultString(null, "NULL")  = "NULL"
   * StringUtils.defaultString("", "NULL")    = ""
   * StringUtils.defaultString("bat", "NULL") = "bat"
   * 
* * @see String#valueOf(Object) * @param str the String to check, may be null * @param defaultStr the default String to return * if the input is null, may be null * @return the passed in String, or the default if it was null */ public static String defaultString(String str, String defaultStr) { return str == null ? defaultStr : str; } /** *

Returns either the passed in String, or if the String is * empty or null, the value of defaultStr.

* *
   * StringUtils.defaultIfEmpty(null, "NULL")  = "NULL"
   * StringUtils.defaultIfEmpty("", "NULL")    = "NULL"
   * StringUtils.defaultIfEmpty("bat", "NULL") = "bat"
   * 
* * @param str the String to check, may be null * @param defaultStr the default String to return * if the input is empty ("") or null, may be null * @return the passed in String, or the default */ public static String defaultIfEmpty(String str, String defaultStr) { return isEmpty(str) ? defaultStr : str; } /** *

Capitalizes a String changing the first letter to title case as * per {@link Character#toTitleCase(char)}. No other letters are changed.

* * A null input String returns null.

* *
   * StringUtils.capitalize(null)  = null
   * StringUtils.capitalize("")    = ""
   * StringUtils.capitalize("cat") = "Cat"
   * StringUtils.capitalize("cAt") = "CAt"
   * 
* * @param str the String to capitalize, may be null * @return the capitalized String, null if null String input * @since 2.0 */ public static String capitalize(String str) { int strLen; if (str == null || (strLen = str.length()) == 0) { return str; } return new StringBuffer(strLen).append(Character.toTitleCase(str.charAt(0))).append( str.substring(1)).toString(); } /** *

Checks if String contains a search character, handling null. * This method uses {@link String#indexOf(int)}.

* *

A null or empty ("") String will return false.

* *
   * StringUtils.contains(null, *)    = false
   * StringUtils.contains("", *)      = false
   * StringUtils.contains("abc", 'a') = true
   * StringUtils.contains("abc", 'z') = false
   * 
* * @param str the String to check, may be null * @param searchChar the character to find * @return true if the String contains the search character, * false if not or null string input * @since 2.0 */ public static boolean contains(String str, char searchChar) { if (isEmpty(str)) { return false; } return str.indexOf(searchChar) >= 0; } /** *

Checks if String contains a search String, handling null. * This method uses {@link String#indexOf(int)}.

* *

A null String will return false.

* *
   * StringUtils.contains(null, *)     = false
   * StringUtils.contains(*, null)     = false
   * StringUtils.contains("", "")      = true
   * StringUtils.contains("abc", "")   = true
   * StringUtils.contains("abc", "a")  = true
   * StringUtils.contains("abc", "z")  = false
   * 
* * @param str the String to check, may be null * @param searchStr the String to find, may be null * @return true if the String contains the search String, * false if not or null string input * @since 2.0 */ public static boolean contains(String str, String searchStr) { if (str == null || searchStr == null) { return false; } return str.indexOf(searchStr) >= 0; } /** * An empty immutable String array. */ public static final String[] EMPTY_STRING_ARRAY = new String[0]; /** *

Compares two objects for equality, where either one or both * objects may be null.

* *
   * ObjectUtils.equals(null, null)                  = true
   * ObjectUtils.equals(null, "")                    = false
   * ObjectUtils.equals("", null)                    = false
   * ObjectUtils.equals("", "")                      = true
   * ObjectUtils.equals(Boolean.TRUE, null)          = false
   * ObjectUtils.equals(Boolean.TRUE, "true")        = false
   * ObjectUtils.equals(Boolean.TRUE, Boolean.TRUE)  = true
   * ObjectUtils.equals(Boolean.TRUE, Boolean.FALSE) = false
   * 
* * @param object1 the first object, may be null * @param object2 the second object, may be null * @return true if the values of both objects are the same */ public static boolean equals(Object object1, Object object2) { if (object1 == object2) { return true; } if ((object1 == null) || (object2 == null)) { return false; } return object1.equals(object2); } /** *

A way to get the entire nested stack-trace of an throwable.

* * @param throwable the Throwable to be examined * @return the nested stack trace, with the root cause first * @since 2.0 */ public static String getFullStackTrace(Throwable throwable) { StringWriter sw = new StringWriter(); PrintWriter pw = new PrintWriter(sw, true); Throwable[] ts = getThrowables(throwable); for (int i = 0; i < ts.length; i++) { ts[i].printStackTrace(pw); if (isNestedThrowable(ts[i])) { break; } } return sw.getBuffer().toString(); } /** *

Returns the list of Throwable objects in the * exception chain.

* *

A throwable without cause will return an array containing * one element - the input throwable. * A throwable with one cause will return an array containing * two elements. - the input throwable and the cause throwable. * A null throwable will return an array size zero.

* * @param throwable the throwable to inspect, may be null * @return the array of throwables, never null */ @SuppressWarnings("unchecked") public static Throwable[] getThrowables(Throwable throwable) { List list = new ArrayList(); while (throwable != null) { list.add(throwable); throwable = getCause(throwable); } return (Throwable[]) list.toArray(new Throwable[list.size()]); } /** *

The names of methods commonly used to access a wrapped exception.

*/ private static String[] CAUSE_METHOD_NAMES = { "getCause", "getNextException", "getTargetException", "getException", "getSourceException", "getRootCause", "getCausedByException", "getNested", "getLinkedException", "getNestedException", "getLinkedCause", "getThrowable", }; /** *

Checks whether this Throwable class can store a cause.

* *

This method does not check whether it actually does store a cause.

* * @param throwable the Throwable to examine, may be null * @return boolean true if nested otherwise false * @since 2.0 */ public static boolean isNestedThrowable(Throwable throwable) { if (throwable == null) { return false; } if (throwable instanceof SQLException) { return true; } else if (throwable instanceof InvocationTargetException) { return true; } else if (isThrowableNested()) { return true; } Class cls = throwable.getClass(); for (int i = 0, isize = CAUSE_METHOD_NAMES.length; i < isize; i++) { try { Method method = cls.getMethod(CAUSE_METHOD_NAMES[i], (Class[])null); if (method != null && Throwable.class.isAssignableFrom(method.getReturnType())) { return true; } } catch (NoSuchMethodException ignored) { } catch (SecurityException ignored) { } } try { Field field = cls.getField("detail"); if (field != null) { return true; } } catch (NoSuchFieldException ignored) { } catch (SecurityException ignored) { } return false; } /** *

The Method object for JDK1.4 getCause.

*/ private static final Method THROWABLE_CAUSE_METHOD; static { Method getCauseMethod; try { getCauseMethod = Throwable.class.getMethod("getCause", (Class[])null); } catch (Exception e) { getCauseMethod = null; } THROWABLE_CAUSE_METHOD = getCauseMethod; } /** *

Checks if the Throwable class has a getCause method.

* *

This is true for JDK 1.4 and above.

* * @return true if Throwable is nestable * @since 2.0 */ public static boolean isThrowableNested() { return THROWABLE_CAUSE_METHOD != null; } /** *

Introspects the Throwable to obtain the cause.

* *

The method searches for methods with specific names that return a * Throwable object. This will pick up most wrapping exceptions, * including those from JDK 1.4, and

* *

The default list searched for are:

*
    *
  • getCause()
  • *
  • getNextException()
  • *
  • getTargetException()
  • *
  • getException()
  • *
  • getSourceException()
  • *
  • getRootCause()
  • *
  • getCausedByException()
  • *
  • getNested()
  • *
* *

In the absence of any such method, the object is inspected for a * detail field assignable to a Throwable.

* *

If none of the above is found, returns null.

* * @param throwable the throwable to introspect for a cause, may be null * @return the cause of the Throwable, * null if none found or null throwable input * @since 1.0 */ public static Throwable getCause(Throwable throwable) { return getCause(throwable, CAUSE_METHOD_NAMES); } /** *

Introspects the Throwable to obtain the cause.

* *
    *
  1. Try known exception types.
  2. *
  3. Try the supplied array of method names.
  4. *
  5. Try the field 'detail'.
  6. *
* *

A null set of method names means use the default set. * A null in the set of method names will be ignored.

* * @param throwable the throwable to introspect for a cause, may be null * @param methodNames the method names, null treated as default set * @return the cause of the Throwable, * null if none found or null throwable input * @since 1.0 */ public static Throwable getCause(Throwable throwable, String[] methodNames) { if (throwable == null) { return null; } Throwable cause = getCauseUsingWellKnownTypes(throwable); if (cause == null) { if (methodNames == null) { methodNames = CAUSE_METHOD_NAMES; } for (int i = 0; i < methodNames.length; i++) { String methodName = methodNames[i]; if (methodName != null) { cause = getCauseUsingMethodName(throwable, methodName); if (cause != null) { break; } } } if (cause == null) { cause = getCauseUsingFieldName(throwable, "detail"); } } return cause; } /** *

Finds a Throwable by method name.

* * @param throwable the exception to examine * @param methodName the name of the method to find and invoke * @return the wrapped exception, or null if not found */ private static Throwable getCauseUsingMethodName(Throwable throwable, String methodName) { Method method = null; try { method = throwable.getClass().getMethod(methodName, (Class[])null); } catch (NoSuchMethodException ignored) { } catch (SecurityException ignored) { } if (method != null && Throwable.class.isAssignableFrom(method.getReturnType())) { try { return (Throwable) method.invoke(throwable, EMPTY_OBJECT_ARRAY); } catch (IllegalAccessException ignored) { } catch (IllegalArgumentException ignored) { } catch (InvocationTargetException ignored) { } } return null; } /** *

Finds a Throwable by field name.

* * @param throwable the exception to examine * @param fieldName the name of the attribute to examine * @return the wrapped exception, or null if not found */ private static Throwable getCauseUsingFieldName(Throwable throwable, String fieldName) { Field field = null; try { field = throwable.getClass().getField(fieldName); } catch (NoSuchFieldException ignored) { } catch (SecurityException ignored) { } if (field != null && Throwable.class.isAssignableFrom(field.getType())) { try { return (Throwable) field.get(throwable); } catch (IllegalAccessException ignored) { } catch (IllegalArgumentException ignored) { } } return null; } /** *

Finds a Throwable for known types.

* *

Uses instanceof checks to examine the exception, * looking for well known types which could contain chained or * wrapped exceptions.

* * @param throwable the exception to examine * @return the wrapped exception, or null if not found */ private static Throwable getCauseUsingWellKnownTypes(Throwable throwable) { if (throwable instanceof SQLException) { return ((SQLException) throwable).getNextException(); } else if (throwable instanceof InvocationTargetException) { return ((InvocationTargetException) throwable).getTargetException(); } else { return null; } } /** * An empty immutable Object array. */ public static final Object[] EMPTY_OBJECT_ARRAY = new Object[0]; /** * from a command line arg, get the key. e.g. if input is --whatever=something * then key is whatever, value is something * @param option * @return the key */ public static String argKey(String option) { int equalsIndex = option.indexOf("="); if (equalsIndex == -1) { throw new RuntimeException("Invalid option: " + option + ", it should look like: --someOption=someValue"); } String key = option.substring(0,equalsIndex); if (!key.startsWith("--")) { throw new RuntimeException("Invalid option: " + option + ", it should look like: --someOption=someValue"); } key = key.substring(2); return key; } /** * from a command line arg, get the key. e.g. if input is --whatever=something * then key is whatever, value is something * @param option * @return the value */ public static String argValue(String option) { int equalsIndex = option.indexOf("="); if (equalsIndex == -1) { throw new RuntimeException("Invalid option: " + option + ", it should look like: --someOption=someValue"); } String value = option.substring(equalsIndex+1, option.length()); return value; } /** add an option: --whatever=val to a map of options where --whatever is key, and val is value * @param args * @return the map */ public static Map argMap(String[] args) { Map result = new LinkedHashMap(); for (String arg : nonNull(args,String.class)) { String key = argKey(arg); String value = argValue(arg); if (result.containsKey(key)) { throw new RuntimeException("Passing key twice: " + key); } result.put(key, value); } return result; } /** * create a file, throw exception if doesnt work (unless already exists) * @param fileToCreate * @return if created */ public static boolean fileCreate(File fileToCreate) { if (fileToCreate.exists() && fileToCreate.isFile()) { return false; } if (fileToCreate.exists() && !fileToCreate.isFile()) { throw new RuntimeException("Trying to create file and it doesnt exist: " + fileToCreate.getAbsolutePath()); } try { if (!fileToCreate.createNewFile()) { throw new RuntimeException("Cant create file: " + fileToCreate); } } catch (IOException ioe) { throw new RuntimeException("Cant create file: " + fileToCreate.getAbsolutePath(), ioe); } return true; } /** * get the value from the argMap, throw exception if not there and required * @param argMap * @param argMapNotUsed * @param key * @param required * @return the value or null or exception */ public static String argMapString(Map argMap, Map argMapNotUsed, String key, boolean required) { if (argMap.containsKey(key)) { //keep track that this is gone argMapNotUsed.remove(key); return argMap.get(key); } if (required) { throw new RuntimeException("Argument '--" + key + "' is required, but not specified. e.g. --" + key + "=value"); } return null; } /** * Copies a directory to within another directory preserving the file dates. *

* This method copies the source directory and all its contents to a * directory of the same name in the specified destination directory. *

* The destination directory is created if it does not exist. * If the destination directory did exist, then this method merges * the source with the destination, with the source taking precedence. * * @param srcDir an existing directory to copy, must not be null * @param destDir the directory to place the copy in, must not be null * * @throws NullPointerException if source or destination is null * @throws IOException if source or destination is invalid * @throws IOException if an IO error occurs during copying * @since Commons IO 1.2 */ public static void copyDirectoryToDirectory(File srcDir, File destDir) throws IOException { if (srcDir == null) { throw new NullPointerException("Source must not be null"); } if (srcDir.exists() && srcDir.isDirectory() == false) { throw new IllegalArgumentException("Source '" + destDir + "' is not a directory"); } if (destDir == null) { throw new NullPointerException("Destination must not be null"); } if (destDir.exists() && destDir.isDirectory() == false) { throw new IllegalArgumentException("Destination '" + destDir + "' is not a directory"); } copyDirectory(srcDir, new File(destDir, srcDir.getName()), true); } /** * Copies a whole directory to a new location preserving the file dates. *

* This method copies the specified directory and all its child * directories and files to the specified destination. * The destination is the new location and name of the directory. *

* The destination directory is created if it does not exist. * If the destination directory did exist, then this method merges * the source with the destination, with the source taking precedence. * * @param srcDir an existing directory to copy, must not be null * @param destDir the new directory, must not be null * * @throws NullPointerException if source or destination is null * @since Commons IO 1.1 */ public static void copyDirectory(File srcDir, File destDir) { try { copyDirectory(srcDir, destDir, true); } catch (IOException ioe) { throw new RuntimeException("Problem with sourceDir: " + srcDir + ", destDir: " + destDir, ioe); } } /** * find a jar * @param allJars * @param fileName * @return the list that matches */ public static List jarFindJar(List allJars, String fileName) { Set result = new HashSet(); //find the passed in base file name Set baseFileNames = jarFileBaseNames(fileName); if (GrouperInstallerUtils.length(baseFileNames) == 0) { throw new RuntimeException("Why is base file name null? " + fileName); } //loop through and find matchers for (File file : allJars) { if (!file.getName().endsWith(".jar")) { continue; } Set fileBaseFileNames = jarFileBaseNames(file.getName()); for (String fileBaseFileName : GrouperInstallerUtils.nonNull(fileBaseFileNames)) { if (baseFileNames.contains(fileBaseFileName)) { result.add(file); } } } return new ArrayList(result); } /** * find a jar * @param dir * @param fileName * @return the list that matches */ public static List jarFindJar(File dir, String fileName) { if (dir.getName().equals("grouper") || dir.getName().equals("custom") || dir.getName().equals("jdbcSamples")) { return jarFindJar(dir.getParentFile(), fileName); } return jarFindJar(GrouperInstallerUtils.fileListRecursive(dir), fileName); } /** * if jarfile is someThing-1.2.3.jar, return something * @param fileName * @return the base file name for jar or null */ public static Set jarFileBaseNames(String fileName) { Set result = new HashSet(); if (fileName.toLowerCase().startsWith("log4j-1.2-api")) { result.add("log4j-1.2-api"); return result; } Pattern pattern = null; Matcher matcher = null; pattern = Pattern.compile("^(.*?)-[0-9].*-tests.jar$"); matcher = pattern.matcher(fileName); String baseName = null; if (matcher.matches()) { baseName = matcher.group(1) + "-tests"; } else { pattern = Pattern.compile("^(.*?)-[0-9].*.jar$"); matcher = pattern.matcher(fileName); if (matcher.matches()) { baseName = matcher.group(1); } else if (fileName.endsWith(".jar")) { baseName = fileName.substring(0, fileName.length() - ".jar".length()); } else { return result; } } if (fileName.toLowerCase().startsWith("okhttp-2")) { result.add("okhttp-2"); } else if (fileName.toLowerCase().startsWith("okhttp-3")) { result.add("okhttp-3"); } else { result.add(baseName.toLowerCase()); } // we should remove this and add cores to handle instead of remove ones which shouldnt... if (baseName.endsWith("-core") && !baseName.toLowerCase().contains("aws") && !baseName.toLowerCase().contains("jersey")) { baseName = baseName.substring(0, baseName.length() - "-core".length()); result.add(baseName.toLowerCase()); } if (baseName.toLowerCase().equals("mysql-connector-java") || baseName.toLowerCase().equals("mysql-connector-java-bin")) { result.add("mysql-connector-java"); result.add("mysql-connector-java-bin"); } if (baseName.toLowerCase().equals("mail") || baseName.toLowerCase().equals("mailapi")) { result.add("mail"); result.add("mailapi"); } return result; } /** * if a collection contains any element in another collection * @param * @param a * @param b * @return true if contains */ public static boolean containsAny(Collection a, Collection b) { if (a == null || b == null) { return false; } for (T t : a) { if (b.contains(t)) { return true; } } return false; } /** * if jarfile is something-1.2.3.jar, return something * @param fileName * @return the base file name for jar or null */ public static String jarFileBaseName(String fileName) { Pattern pattern = Pattern.compile("^(.*?)-[0-9].*.jar$"); Matcher matcher = pattern.matcher(fileName); String baseName = null; if (matcher.matches()) { baseName = matcher.group(1); } else if (fileName.endsWith(".jar")) { baseName = fileName.substring(0, fileName.length() - ".jar".length()); } else { return null; } if (baseName.endsWith("-core")) { baseName = baseName.substring(0, baseName.length() - "-core".length()); } return baseName; } /** * Copies a whole directory to a new location. *

* This method copies the contents of the specified source directory * to within the specified destination directory. *

* The destination directory is created if it does not exist. * If the destination directory did exist, then this method merges * the source with the destination, with the source taking precedence. * * @param srcDir an existing directory to copy, must not be null * @param destDir the new directory, must not be null * @param preserveFileDate true if the file date of the copy * should be the same as the original * * @throws NullPointerException if source or destination is null * @throws IOException if source or destination is invalid * @throws IOException if an IO error occurs during copying * @since Commons IO 1.1 */ public static void copyDirectory(File srcDir, File destDir, boolean preserveFileDate) throws IOException { copyDirectory(srcDir, destDir, null, preserveFileDate); } /** * Copies a filtered directory to a new location preserving the file dates. *

* This method copies the contents of the specified source directory * to within the specified destination directory. *

* The destination directory is created if it does not exist. * If the destination directory did exist, then this method merges * the source with the destination, with the source taking precedence. * *

Example: Copy directories only

*
   *  // only copy the directory structure
   *  FileUtils.copyDirectory(srcDir, destDir, DirectoryFileFilter.DIRECTORY);
   *  
* *

Example: Copy directories and txt files

*
   *  // Create a filter for ".txt" files
   *  IOFileFilter txtSuffixFilter = FileFilterUtils.suffixFileFilter(".txt");
   *  IOFileFilter txtFiles = FileFilterUtils.andFileFilter(FileFileFilter.FILE, txtSuffixFilter);
   *
   *  // Create a filter for either directories or ".txt" files
   *  FileFilter filter = FileFilterUtils.orFileFilter(DirectoryFileFilter.DIRECTORY, txtFiles);
   *
   *  // Copy using the filter
   *  FileUtils.copyDirectory(srcDir, destDir, filter);
   *  
* * @param srcDir an existing directory to copy, must not be null * @param destDir the new directory, must not be null * @param filter the filter to apply, null means copy all directories and files * should be the same as the original * * @throws NullPointerException if source or destination is null * @throws IOException if source or destination is invalid * @throws IOException if an IO error occurs during copying * @since Commons IO 1.4 */ public static void copyDirectory(File srcDir, File destDir, FileFilter filter) throws IOException { copyDirectory(srcDir, destDir, filter, true); } /** * Copies a filtered directory to a new location. *

* This method copies the contents of the specified source directory * to within the specified destination directory. *

* The destination directory is created if it does not exist. * If the destination directory did exist, then this method merges * the source with the destination, with the source taking precedence. * *

Example: Copy directories only

*
   *  // only copy the directory structure
   *  FileUtils.copyDirectory(srcDir, destDir, DirectoryFileFilter.DIRECTORY, false);
   *  
* *

Example: Copy directories and txt files

*
   *  // Create a filter for ".txt" files
   *  IOFileFilter txtSuffixFilter = FileFilterUtils.suffixFileFilter(".txt");
   *  IOFileFilter txtFiles = FileFilterUtils.andFileFilter(FileFileFilter.FILE, txtSuffixFilter);
   *
   *  // Create a filter for either directories or ".txt" files
   *  FileFilter filter = FileFilterUtils.orFileFilter(DirectoryFileFilter.DIRECTORY, txtFiles);
   *
   *  // Copy using the filter
   *  FileUtils.copyDirectory(srcDir, destDir, filter, false);
   *  
* * @param srcDir an existing directory to copy, must not be null * @param destDir the new directory, must not be null * @param filter the filter to apply, null means copy all directories and files * @param preserveFileDate true if the file date of the copy * should be the same as the original * * @throws NullPointerException if source or destination is null * @throws IOException if source or destination is invalid * @throws IOException if an IO error occurs during copying * @since Commons IO 1.4 */ public static void copyDirectory(File srcDir, File destDir, FileFilter filter, boolean preserveFileDate) throws IOException { if (srcDir == null) { throw new NullPointerException("Source must not be null"); } if (destDir == null) { throw new NullPointerException("Destination must not be null"); } if (srcDir.exists() == false) { throw new FileNotFoundException("Source '" + srcDir + "' does not exist"); } if (srcDir.isDirectory() == false) { throw new IOException("Source '" + srcDir + "' exists but is not a directory"); } if (srcDir.getCanonicalPath().equals(destDir.getCanonicalPath())) { throw new IOException("Source '" + srcDir + "' and destination '" + destDir + "' are the same"); } // Cater for destination being directory within the source directory (see IO-141) List exclusionList = null; if (destDir.getCanonicalPath().startsWith(srcDir.getCanonicalPath())) { File[] srcFiles = filter == null ? srcDir.listFiles() : srcDir.listFiles(filter); if (srcFiles != null && srcFiles.length > 0) { exclusionList = new ArrayList(srcFiles.length); for (int i = 0; i < srcFiles.length; i++) { File copiedFile = new File(destDir, srcFiles[i].getName()); exclusionList.add(copiedFile.getCanonicalPath()); } } } doCopyDirectory(srcDir, destDir, filter, preserveFileDate, exclusionList); } /** * Internal copy directory method. * * @param srcDir the validated source directory, must not be null * @param destDir the validated destination directory, must not be null * @param filter the filter to apply, null means copy all directories and files * @param preserveFileDate whether to preserve the file date * @param exclusionList List of files and directories to exclude from the copy, may be null * @throws IOException if an error occurs * @since Commons IO 1.1 */ private static void doCopyDirectory(File srcDir, File destDir, FileFilter filter, boolean preserveFileDate, List exclusionList) throws IOException { if (destDir.exists()) { if (destDir.isDirectory() == false) { throw new IOException("Destination '" + destDir + "' exists but is not a directory"); } } else { if (destDir.mkdirs() == false) { throw new IOException("Destination '" + destDir + "' directory cannot be created"); } if (preserveFileDate) { destDir.setLastModified(srcDir.lastModified()); } } if (destDir.canWrite() == false) { throw new IOException("Destination '" + destDir + "' cannot be written to"); } // recurse File[] files = filter == null ? srcDir.listFiles() : srcDir.listFiles(filter); if (files == null) { // null if security restricted throw new IOException("Failed to list contents of " + srcDir); } for (int i = 0; i < files.length; i++) { File copiedFile = new File(destDir, files[i].getName()); if (exclusionList == null || !exclusionList.contains(files[i].getCanonicalPath())) { if (files[i].isDirectory()) { doCopyDirectory(files[i], copiedFile, filter, preserveFileDate, exclusionList); } else { doCopyFile(files[i], copiedFile, preserveFileDate); } } } } /** * get the value from the argMap, throw exception if not there and required * @param argMap * @param argMapNotUsed * @param key * @param required * @param defaultValue * @return the value or null or exception */ public static boolean argMapBoolean(Map argMap, Map argMapNotUsed, String key, boolean required, boolean defaultValue) { String argString = argMapString(argMap, argMapNotUsed, key, required); if (isBlank(argString) && required) { throw new RuntimeException("Argument '--" + key + "' is required, but not specified. e.g. --" + key + "=true"); } return booleanValue(argString, defaultValue); } /** * get the value from the argMap, throw exception if not there and required * @param argMap * @param argMapNotUsed * @param key * @return the value or null or exception */ public static Timestamp argMapTimestamp(Map argMap, Map argMapNotUsed, String key) { String argString = argMapString(argMap, argMapNotUsed, key, false); if (isBlank(argString)) { return null; } Date date = stringToDate2(argString); return new Timestamp(date.getTime()); } /** * get the value from the argMap * @param argMap * @param argMapNotUsed * @param key * @return the value or null or exception */ public static Boolean argMapBoolean(Map argMap, Map argMapNotUsed, String key) { String argString = argMapString(argMap, argMapNotUsed, key, false); return booleanObjectValue(argString); } /** * get the set from comma separated from the argMap, throw exception if not there and required * @param argMap * @param argMapNotUsed * @param key * @param required * @return the value or null or exception */ public static Set argMapSet(Map argMap, Map argMapNotUsed, String key, boolean required) { List list = argMapList(argMap, argMapNotUsed, key, required); return list == null ? null : new LinkedHashSet(list); } /** * get the list from comma separated from the argMap, throw exception if not there and required * @param argMap * @param argMapNotUsed * @param key * @param required * @return the value or null or exception */ public static List argMapList(Map argMap, Map argMapNotUsed, String key, boolean required) { String argString = argMapString(argMap, argMapNotUsed, key, required); if (isBlank(argString)) { return null; } return splitTrimToList(argString, ","); } /** * get the list from comma separated from the argMap, throw exception if not there and required * @param argMap * @param argMapNotUsed * @param key * @param required * @return the value or null or exception */ public static List argMapFileList(Map argMap, Map argMapNotUsed, String key, boolean required) { String argString = argMapString(argMap, argMapNotUsed, key, required); if (isBlank(argString)) { return null; } //read from file File file = new File(argString); try { //do this by regex, since we dont know what platform we are on String listString = GrouperInstallerUtils.readFileIntoString(file); String[] array = listString.split("\\s+"); List list = new ArrayList(); for (String string : array) { //dont know if any here will be blank or whatnot if (!GrouperInstallerUtils.isBlank(string)) { //should already be trimmed, but just in case list.add(trim(string)); } } return list; } catch (Exception e) { throw new RuntimeException("Error reading file: '" + GrouperInstallerUtils.fileCanonicalPath(file) + "' from command line arg: " + key, e ); } } /** * for testing, get the response body as a string * @param method * @return the string of response body */ public static String responseBodyAsString(HttpMethodBase method) { InputStream inputStream = null; try { StringWriter writer = new StringWriter(); inputStream = method.getResponseBodyAsStream(); copy(inputStream, writer); return writer.toString(); } catch (Exception e) { throw new RuntimeException(e); } finally { closeQuietly(inputStream); } } /** * Copy bytes from an InputStream to chars on a * Writer using the default character encoding of the platform. *

* This method buffers the input internally, so there is no need to use a * BufferedInputStream. *

* This method uses {@link InputStreamReader}. * * @param input the InputStream to read from * @param output the Writer to write to * @throws NullPointerException if the input or output is null * @throws IOException if an I/O error occurs * @since Commons IO 1.1 */ public static void copy(InputStream input, Writer output) throws IOException { String charsetName = "UTF-8"; InputStreamReader in = new InputStreamReader(input, charsetName); copy(in, output); } /** * get a jar file from a sample class * @param sampleClass * @return the jar file */ public static File jarFile(Class sampleClass) { try { CodeSource codeSource = sampleClass.getProtectionDomain().getCodeSource(); if (codeSource != null && codeSource.getLocation() != null) { String fileName = URLDecoder.decode(codeSource.getLocation().getFile(), "UTF-8"); return new File(fileName); } String resourcePath = sampleClass.getName(); resourcePath = resourcePath.replace('.', '/') + ".class"; URL url = computeUrl(resourcePath, true); String urlPath = url.toString(); if (urlPath.startsWith("jar:")) { urlPath = urlPath.substring(4); } if (urlPath.startsWith("file:")) { urlPath = urlPath.substring(5); } urlPath = prefixOrSuffix(urlPath, "!", true); urlPath = URLDecoder.decode(urlPath, "UTF-8"); File file = new File(urlPath); if (urlPath.endsWith(".jar") && file.exists() && file.isFile()) { return file; } } catch (Exception e) { LOG.warn("Cant find jar for class: " + sampleClass + ", " + e.getMessage(), e); } return null; } /** * strip the last slash (/ or \) from a string if it exists * * @param input * * @return input - the last / or \ */ public static String stripLastSlashIfExists(String input) { if ((input == null) || (input.length() == 0)) { return null; } char lastChar = input.charAt(input.length() - 1); if ((lastChar == '\\') || (lastChar == '/')) { return input.substring(0, input.length() - 1); } return input; } /** * retrieve a password from stdin * @param dontMask * @param prompt to print to user * @return the password */ public static String retrievePasswordFromStdin(boolean dontMask, String prompt) { String passwordString = null; if (dontMask) { System.out.print(prompt); // open up standard input BufferedReader br = new BufferedReader(new InputStreamReader(System.in)); // read the username from the command-line; need to use try/catch with the // readLine() method try { passwordString = br.readLine(); } catch (IOException ioe) { System.out.println("IO error! " + getFullStackTrace(ioe)); System.exit(1); } } else { char password[] = null; try { password = retrievePasswordFromStdin(System.in, prompt); } catch (IOException ioe) { ioe.printStackTrace(); } passwordString = String.valueOf(password); } return passwordString; } /** * @param in stream to be used (e.g. System.in) * @param prompt The prompt to display to the user. * @return The password as entered by the user. * @throws IOException */ public static final char[] retrievePasswordFromStdin(InputStream in, String prompt) throws IOException { MaskingThread maskingthread = new MaskingThread(prompt); Thread thread = new Thread(maskingthread); thread.start(); char[] lineBuffer; char[] buf; buf = lineBuffer = new char[128]; int room = buf.length; int offset = 0; int c; loop: while (true) { switch (c = in.read()) { case -1: case '\n': break loop; case '\r': int c2 = in.read(); if ((c2 != '\n') && (c2 != -1)) { if (!(in instanceof PushbackInputStream)) { in = new PushbackInputStream(in); } ((PushbackInputStream) in).unread(c2); } else { break loop; } default: if (--room < 0) { buf = new char[offset + 128]; room = buf.length - offset - 1; System.arraycopy(lineBuffer, 0, buf, 0, offset); Arrays.fill(lineBuffer, ' '); lineBuffer = buf; } buf[offset++] = (char) c; break; } } maskingthread.stopMasking(); if (offset == 0) { return null; } char[] ret = new char[offset]; System.arraycopy(buf, 0, ret, 0, offset); Arrays.fill(buf, ' '); return ret; } /** * thread to mask input */ static class MaskingThread extends Thread { /** stop */ private volatile boolean stop; /** echo char, this doesnt work correctly, so make a space so people dont notice... * prints out too many */ private char echochar = ' '; /** *@param prompt The prompt displayed to the user */ public MaskingThread(String prompt) { System.out.print(prompt); } /** * Begin masking until asked to stop. */ @Override public void run() { int priority = Thread.currentThread().getPriority(); Thread.currentThread().setPriority(Thread.MAX_PRIORITY); try { this.stop = true; while (this.stop) { System.out.print("\010" + this.echochar); try { // attempt masking at this rate Thread.sleep(1); } catch (InterruptedException iex) { Thread.currentThread().interrupt(); return; } } } finally { // restore the original priority Thread.currentThread().setPriority(priority); } } /** * Instruct the thread to stop masking. */ public void stopMasking() { this.stop = false; } } /** * make sure a value exists in properties * @param resourceName * @param properties * @param overrideMap * @param key * @param exceptionOnError * @return true if ok, false if not */ public static boolean propertyValidateValueRequired(String resourceName, Properties properties, Map overrideMap, String key, boolean exceptionOnError) { Map threadLocalMap = propertiesThreadLocalOverrideMap(resourceName); String value = propertiesValue(properties, threadLocalMap, overrideMap, key); if (!GrouperInstallerUtils.isBlank(value)) { return true; } String error = "Cant find property " + key + " in resource: " + resourceName + ", it is required"; if (exceptionOnError) { throw new RuntimeException(error); } System.err.println("Grouper error: " + error); LOG.error(error); return false; } /** * make sure a value is boolean in properties * @param resourceName * @param properties * @param overrideMap * @param key * @param required * @param exceptionOnError * @return true if ok, false if not */ public static boolean propertyValidateValueBoolean(String resourceName, Properties properties, Map overrideMap, String key, boolean required, boolean exceptionOnError) { if (required && !propertyValidateValueRequired(resourceName, properties, overrideMap, key, exceptionOnError)) { return false; } Map threadLocalMap = propertiesThreadLocalOverrideMap(resourceName); String value = propertiesValue(properties, threadLocalMap, overrideMap, key); //maybe ok not there if (!required && GrouperInstallerUtils.isBlank(value)) { return true; } try { booleanValue(value); return true; } catch (Exception e) { } String error = "Expecting true or false property " + key + " in resource: " + resourceName + ", but is '" + value + "'"; if (exceptionOnError) { throw new RuntimeException(error); } System.err.println("Grouper error: " + error); LOG.error(error); return false; } /** * every 5 seconds print a status dot to system out, and newline at end. After 20, newline * @param runnable runnable to run * @param printProgress if print progress during thread */ public static void threadRunWithStatusDots(final Runnable runnable, boolean printProgress) { threadRunWithStatusDots(runnable, printProgress, true); } /** * every 5 seconds print a status dot to system out, and newline at end. After 20, newline * @param runnable runnable to run * @param printProgress if print progress during thread * @param injectStackInException */ public static void threadRunWithStatusDots(final Runnable runnable, boolean printProgress, boolean injectStackInException) { if (!printProgress) { runnable.run(); return; } try { final Exception[] theException = new Exception[1]; Runnable wrappedRunnable = new Runnable() { public void run() { try { runnable.run(); } catch (Exception e) { theException[0] = e; } } }; Thread thread = new Thread(wrappedRunnable); thread.start(); long start = System.currentTimeMillis(); boolean wroteProgress = false; int dotCount = 0; while (true) { if (thread.isAlive()) { if (System.currentTimeMillis() - start > 5000) { wroteProgress = true; System.out.print("."); dotCount++; start = System.currentTimeMillis(); if (dotCount % 40 == 0) { System.out.println(""); } } else { GrouperInstallerUtils.sleep(500); } } else { if (wroteProgress) { //start with a newline System.out.println(""); } break; } } //probably dont need this thread.join(); if (theException[0] != null) { if (injectStackInException) { //append this stack injectInException(theException[0], getFullStackTrace(new RuntimeException("caller stack"))); } throw theException[0]; } } catch (Exception exception) { if (exception instanceof RuntimeException) { throw (RuntimeException)exception; } throw new RuntimeException(exception); } } /** * make sure a value is int in properties * @param resourceName * @param properties * @param overrideMap * @param key * @param required * @param exceptionOnError * @return true if ok, false if not */ public static boolean propertyValidateValueInt(String resourceName, Properties properties, Map overrideMap, String key, boolean required, boolean exceptionOnError) { if (required && !propertyValidateValueRequired(resourceName, properties, overrideMap, key, exceptionOnError)) { return false; } Map threadLocalMap = propertiesThreadLocalOverrideMap(resourceName); String value = propertiesValue(properties, threadLocalMap, overrideMap, key); //maybe ok not there if (!required && GrouperInstallerUtils.isBlank(value)) { return true; } try { intValue(value); return true; } catch (Exception e) { } String error = "Expecting integer property " + key + " in resource: " + resourceName + ", but is '" + value + "'"; if (exceptionOnError) { throw new RuntimeException(error); } System.err.println("Grouper error: " + error); LOG.error(error); return false; } /** * make sure a property is a class of a certain type * @param resourceName * @param properties * @param overrideMap * @param key * @param classType * @param required * @param exceptionOnError * @return true if ok */ public static boolean propertyValidateValueClass(String resourceName, Properties properties, Map overrideMap, String key, Class classType, boolean required, boolean exceptionOnError) { if (required && !propertyValidateValueRequired(resourceName, properties, overrideMap, key, exceptionOnError)) { return false; } String value = propertiesValue(properties, overrideMap, key); //maybe ok not there if (!required && GrouperInstallerUtils.isBlank(value)) { return true; } String extraError = ""; try { Class theClass = forName(value); if (classType.isAssignableFrom(theClass)) { return true; } extraError = " does not derive from class: " + classType.getSimpleName(); } catch (Exception e) { extraError = ", " + getFullStackTrace(e); } String error = "Cant process property " + key + " in resource: " + resourceName + ", the current" + " value is '" + value + "', which should be of type: " + classType.getName() + extraError; if (exceptionOnError) { throw new RuntimeException(error); } System.err.println("Grouper error: " + error); LOG.error(error); return false; } /** *

Strips any of a set of characters from the start of a String.

* *

A null input String returns null. * An empty string ("") input returns the empty string.

* *

If the stripChars String is null, whitespace is * stripped as defined by {@link Character#isWhitespace(char)}.

* *
   * StringUtils.stripStart(null, *)          = null
   * StringUtils.stripStart("", *)            = ""
   * StringUtils.stripStart("abc", "")        = "abc"
   * StringUtils.stripStart("abc", null)      = "abc"
   * StringUtils.stripStart("  abc", null)    = "abc"
   * StringUtils.stripStart("abc  ", null)    = "abc  "
   * StringUtils.stripStart(" abc ", null)    = "abc "
   * StringUtils.stripStart("yxabc  ", "xyz") = "abc  "
   * 
* * @param str the String to remove characters from, may be null * @param stripChars the characters to remove, null treated as whitespace * @return the stripped String, null if null String input */ public static String stripStart(String str, String stripChars) { int strLen; if (str == null || (strLen = str.length()) == 0) { return str; } int start = 0; if (stripChars == null) { while ((start != strLen) && Character.isWhitespace(str.charAt(start))) { start++; } } else if (stripChars.length() == 0) { return str; } else { while ((start != strLen) && (stripChars.indexOf(str.charAt(start)) != -1)) { start++; } } return str.substring(start); } /** *

Strips any of a set of characters from the end of a String.

* *

A null input String returns null. * An empty string ("") input returns the empty string.

* *

If the stripChars String is null, whitespace is * stripped as defined by {@link Character#isWhitespace(char)}.

* *
   * StringUtils.stripEnd(null, *)          = null
   * StringUtils.stripEnd("", *)            = ""
   * StringUtils.stripEnd("abc", "")        = "abc"
   * StringUtils.stripEnd("abc", null)      = "abc"
   * StringUtils.stripEnd("  abc", null)    = "  abc"
   * StringUtils.stripEnd("abc  ", null)    = "abc"
   * StringUtils.stripEnd(" abc ", null)    = " abc"
   * StringUtils.stripEnd("  abcyx", "xyz") = "  abc"
   * 
* * @param str the String to remove characters from, may be null * @param stripChars the characters to remove, null treated as whitespace * @return the stripped String, null if null String input */ public static String stripEnd(String str, String stripChars) { int end; if (str == null || (end = str.length()) == 0) { return str; } if (stripChars == null) { while ((end != 0) && Character.isWhitespace(str.charAt(end - 1))) { end--; } } else if (stripChars.length() == 0) { return str; } else { while ((end != 0) && (stripChars.indexOf(str.charAt(end - 1)) != -1)) { end--; } } return str.substring(0, end); } /** * The empty String "". * @since 2.0 */ public static final String EMPTY = ""; /** * Represents a failed index search. * @since 2.1 */ public static final int INDEX_NOT_FOUND = -1; /** *

The maximum size to which the padding constant(s) can expand.

*/ private static final int PAD_LIMIT = 8192; /** *

An array of Strings used for padding.

* *

Used for efficient space padding. The length of each String expands as needed.

*/ private static final String[] PADDING = new String[Character.MAX_VALUE]; static { // space padding is most common, start with 64 chars PADDING[32] = " "; } /** *

Repeat a String repeat times to form a * new String.

* *
   * StringUtils.repeat(null, 2) = null
   * StringUtils.repeat("", 0)   = ""
   * StringUtils.repeat("", 2)   = ""
   * StringUtils.repeat("a", 3)  = "aaa"
   * StringUtils.repeat("ab", 2) = "abab"
   * StringUtils.repeat("a", -2) = ""
   * 
* * @param str the String to repeat, may be null * @param repeat number of times to repeat str, negative treated as zero * @return a new String consisting of the original String repeated, * null if null String input */ public static String repeat(String str, int repeat) { // Performance tuned for 2.0 (JDK1.4) if (str == null) { return null; } if (repeat <= 0) { return EMPTY; } int inputLength = str.length(); if (repeat == 1 || inputLength == 0) { return str; } if (inputLength == 1 && repeat <= PAD_LIMIT) { return padding(repeat, str.charAt(0)); } int outputLength = inputLength * repeat; switch (inputLength) { case 1: char ch = str.charAt(0); char[] output1 = new char[outputLength]; for (int i = repeat - 1; i >= 0; i--) { output1[i] = ch; } return new String(output1); case 2: char ch0 = str.charAt(0); char ch1 = str.charAt(1); char[] output2 = new char[outputLength]; for (int i = repeat * 2 - 2; i >= 0; i--, i--) { output2[i] = ch0; output2[i + 1] = ch1; } return new String(output2); default: StringBuffer buf = new StringBuffer(outputLength); for (int i = 0; i < repeat; i++) { buf.append(str); } return buf.toString(); } } /** *

Returns padding using the specified delimiter repeated * to a given length.

* *
   * StringUtils.padding(0, 'e')  = ""
   * StringUtils.padding(3, 'e')  = "eee"
   * StringUtils.padding(-2, 'e') = IndexOutOfBoundsException
   * 
* * @param repeat number of times to repeat delim * @param padChar character to repeat * @return String with repeated character * @throws IndexOutOfBoundsException if repeat < 0 */ private static String padding(int repeat, char padChar) { // be careful of synchronization in this method // we are assuming that get and set from an array index is atomic String pad = PADDING[padChar]; if (pad == null) { pad = String.valueOf(padChar); } while (pad.length() < repeat) { pad = pad.concat(pad); } PADDING[padChar] = pad; return pad.substring(0, repeat); } /** *

Right pad a String with spaces (' ').

* *

The String is padded to the size of size.

* *
   * StringUtils.rightPad(null, *)   = null
   * StringUtils.rightPad("", 3)     = "   "
   * StringUtils.rightPad("bat", 3)  = "bat"
   * StringUtils.rightPad("bat", 5)  = "bat  "
   * StringUtils.rightPad("bat", 1)  = "bat"
   * StringUtils.rightPad("bat", -1) = "bat"
   * 
* * @param str the String to pad out, may be null * @param size the size to pad to * @return right padded String or original String if no padding is necessary, * null if null String input */ public static String rightPad(String str, int size) { return rightPad(str, size, ' '); } /** *

Right pad a String with a specified character.

* *

The String is padded to the size of size.

* *
   * StringUtils.rightPad(null, *, *)     = null
   * StringUtils.rightPad("", 3, 'z')     = "zzz"
   * StringUtils.rightPad("bat", 3, 'z')  = "bat"
   * StringUtils.rightPad("bat", 5, 'z')  = "batzz"
   * StringUtils.rightPad("bat", 1, 'z')  = "bat"
   * StringUtils.rightPad("bat", -1, 'z') = "bat"
   * 
* * @param str the String to pad out, may be null * @param size the size to pad to * @param padChar the character to pad with * @return right padded String or original String if no padding is necessary, * null if null String input * @since 2.0 */ public static String rightPad(String str, int size, char padChar) { if (str == null) { return null; } int pads = size - str.length(); if (pads <= 0) { return str; // returns original String when possible } if (pads > PAD_LIMIT) { return rightPad(str, size, String.valueOf(padChar)); } return str.concat(padding(pads, padChar)); } /** *

Right pad a String with a specified String.

* *

The String is padded to the size of size.

* *
   * StringUtils.rightPad(null, *, *)      = null
   * StringUtils.rightPad("", 3, "z")      = "zzz"
   * StringUtils.rightPad("bat", 3, "yz")  = "bat"
   * StringUtils.rightPad("bat", 5, "yz")  = "batyz"
   * StringUtils.rightPad("bat", 8, "yz")  = "batyzyzy"
   * StringUtils.rightPad("bat", 1, "yz")  = "bat"
   * StringUtils.rightPad("bat", -1, "yz") = "bat"
   * StringUtils.rightPad("bat", 5, null)  = "bat  "
   * StringUtils.rightPad("bat", 5, "")    = "bat  "
   * 
* * @param str the String to pad out, may be null * @param size the size to pad to * @param padStr the String to pad with, null or empty treated as single space * @return right padded String or original String if no padding is necessary, * null if null String input */ public static String rightPad(String str, int size, String padStr) { if (str == null) { return null; } if (isEmpty(padStr)) { padStr = " "; } int padLen = padStr.length(); int strLen = str.length(); int pads = size - strLen; if (pads <= 0) { return str; // returns original String when possible } if (padLen == 1 && pads <= PAD_LIMIT) { return rightPad(str, size, padStr.charAt(0)); } if (pads == padLen) { return str.concat(padStr); } else if (pads < padLen) { return str.concat(padStr.substring(0, pads)); } else { char[] padding = new char[pads]; char[] padChars = padStr.toCharArray(); for (int i = 0; i < pads; i++) { padding[i] = padChars[i % padLen]; } return str.concat(new String(padding)); } } /** *

Left pad a String with spaces (' ').

* *

The String is padded to the size of size.

* *
   * StringUtils.leftPad(null, *)   = null
   * StringUtils.leftPad("", 3)     = "   "
   * StringUtils.leftPad("bat", 3)  = "bat"
   * StringUtils.leftPad("bat", 5)  = "  bat"
   * StringUtils.leftPad("bat", 1)  = "bat"
   * StringUtils.leftPad("bat", -1) = "bat"
   * 
* * @param str the String to pad out, may be null * @param size the size to pad to * @return left padded String or original String if no padding is necessary, * null if null String input */ public static String leftPad(String str, int size) { return leftPad(str, size, ' '); } /** *

Left pad a String with a specified character.

* *

Pad to a size of size.

* *
   * StringUtils.leftPad(null, *, *)     = null
   * StringUtils.leftPad("", 3, 'z')     = "zzz"
   * StringUtils.leftPad("bat", 3, 'z')  = "bat"
   * StringUtils.leftPad("bat", 5, 'z')  = "zzbat"
   * StringUtils.leftPad("bat", 1, 'z')  = "bat"
   * StringUtils.leftPad("bat", -1, 'z') = "bat"
   * 
* * @param str the String to pad out, may be null * @param size the size to pad to * @param padChar the character to pad with * @return left padded String or original String if no padding is necessary, * null if null String input * @since 2.0 */ public static String leftPad(String str, int size, char padChar) { if (str == null) { return null; } int pads = size - str.length(); if (pads <= 0) { return str; // returns original String when possible } if (pads > PAD_LIMIT) { return leftPad(str, size, String.valueOf(padChar)); } return padding(pads, padChar).concat(str); } /** *

Left pad a String with a specified String.

* *

Pad to a size of size.

* *
   * StringUtils.leftPad(null, *, *)      = null
   * StringUtils.leftPad("", 3, "z")      = "zzz"
   * StringUtils.leftPad("bat", 3, "yz")  = "bat"
   * StringUtils.leftPad("bat", 5, "yz")  = "yzbat"
   * StringUtils.leftPad("bat", 8, "yz")  = "yzyzybat"
   * StringUtils.leftPad("bat", 1, "yz")  = "bat"
   * StringUtils.leftPad("bat", -1, "yz") = "bat"
   * StringUtils.leftPad("bat", 5, null)  = "  bat"
   * StringUtils.leftPad("bat", 5, "")    = "  bat"
   * 
* * @param str the String to pad out, may be null * @param size the size to pad to * @param padStr the String to pad with, null or empty treated as single space * @return left padded String or original String if no padding is necessary, * null if null String input */ public static String leftPad(String str, int size, String padStr) { if (str == null) { return null; } if (isEmpty(padStr)) { padStr = " "; } int padLen = padStr.length(); int strLen = str.length(); int pads = size - strLen; if (pads <= 0) { return str; // returns original String when possible } if (padLen == 1 && pads <= PAD_LIMIT) { return leftPad(str, size, padStr.charAt(0)); } if (pads == padLen) { return padStr.concat(str); } else if (pads < padLen) { return padStr.substring(0, pads).concat(str); } else { char[] padding = new char[pads]; char[] padChars = padStr.toCharArray(); for (int i = 0; i < pads; i++) { padding[i] = padChars[i % padLen]; } return new String(padding).concat(str); } } /** * convert an exception to a runtime exception * @param e */ public static void convertToRuntimeException(Exception e) { if (e instanceof RuntimeException) { throw (RuntimeException)e; } throw new RuntimeException(e.getMessage(), e); } /** *

Gets the substring before the first occurrence of a separator. * The separator is not returned.

* *

A null string input will return null. * An empty ("") string input will return the empty string. * A null separator will return the input string.

* *
   * StringUtils.substringBefore(null, *)      = null
   * StringUtils.substringBefore("", *)        = ""
   * StringUtils.substringBefore("abc", "a")   = ""
   * StringUtils.substringBefore("abcba", "b") = "a"
   * StringUtils.substringBefore("abc", "c")   = "ab"
   * StringUtils.substringBefore("abc", "d")   = "abc"
   * StringUtils.substringBefore("abc", "")    = ""
   * StringUtils.substringBefore("abc", null)  = "abc"
   * 
* * @param str the String to get a substring from, may be null * @param separator the String to search for, may be null * @return the substring before the first occurrence of the separator, * null if null String input * @since 2.0 */ public static String substringBefore(String str, String separator) { if (isEmpty(str) || separator == null) { return str; } if (separator.length() == 0) { return EMPTY; } int pos = str.indexOf(separator); if (pos == -1) { return str; } return str.substring(0, pos); } /** *

Gets the substring after the first occurrence of a separator. * The separator is not returned.

* *

A null string input will return null. * An empty ("") string input will return the empty string. * A null separator will return the empty string if the * input string is not null.

* *
   * StringUtils.substringAfter(null, *)      = null
   * StringUtils.substringAfter("", *)        = ""
   * StringUtils.substringAfter(*, null)      = ""
   * StringUtils.substringAfter("abc", "a")   = "bc"
   * StringUtils.substringAfter("abcba", "b") = "cba"
   * StringUtils.substringAfter("abc", "c")   = ""
   * StringUtils.substringAfter("abc", "d")   = ""
   * StringUtils.substringAfter("abc", "")    = "abc"
   * 
* * @param str the String to get a substring from, may be null * @param separator the String to search for, may be null * @return the substring after the first occurrence of the separator, * null if null String input * @since 2.0 */ public static String substringAfter(String str, String separator) { if (isEmpty(str)) { return str; } if (separator == null) { return EMPTY; } int pos = str.indexOf(separator); if (pos == -1) { return EMPTY; } return str.substring(pos + separator.length()); } /** *

Gets the substring before the last occurrence of a separator. * The separator is not returned.

* *

A null string input will return null. * An empty ("") string input will return the empty string. * An empty or null separator will return the input string.

* *
   * StringUtils.substringBeforeLast(null, *)      = null
   * StringUtils.substringBeforeLast("", *)        = ""
   * StringUtils.substringBeforeLast("abcba", "b") = "abc"
   * StringUtils.substringBeforeLast("abc", "c")   = "ab"
   * StringUtils.substringBeforeLast("a", "a")     = ""
   * StringUtils.substringBeforeLast("a", "z")     = "a"
   * StringUtils.substringBeforeLast("a", null)    = "a"
   * StringUtils.substringBeforeLast("a", "")      = "a"
   * 
* * @param str the String to get a substring from, may be null * @param separator the String to search for, may be null * @return the substring before the last occurrence of the separator, * null if null String input * @since 2.0 */ public static String substringBeforeLast(String str, String separator) { if (isEmpty(str) || isEmpty(separator)) { return str; } int pos = str.lastIndexOf(separator); if (pos == -1) { return str; } return str.substring(0, pos); } /** *

Gets the substring after the last occurrence of a separator. * The separator is not returned.

* *

A null string input will return null. * An empty ("") string input will return the empty string. * An empty or null separator will return the empty string if * the input string is not null.

* *
   * StringUtils.substringAfterLast(null, *)      = null
   * StringUtils.substringAfterLast("", *)        = ""
   * StringUtils.substringAfterLast(*, "")        = ""
   * StringUtils.substringAfterLast(*, null)      = ""
   * StringUtils.substringAfterLast("abc", "a")   = "bc"
   * StringUtils.substringAfterLast("abcba", "b") = "a"
   * StringUtils.substringAfterLast("abc", "c")   = ""
   * StringUtils.substringAfterLast("a", "a")     = ""
   * StringUtils.substringAfterLast("a", "z")     = ""
   * 
* * @param str the String to get a substring from, may be null * @param separator the String to search for, may be null * @return the substring after the last occurrence of the separator, * null if null String input * @since 2.0 */ public static String substringAfterLast(String str, String separator) { if (isEmpty(str)) { return str; } if (isEmpty(separator)) { return EMPTY; } int pos = str.lastIndexOf(separator); if (pos == -1 || pos == (str.length() - separator.length())) { return EMPTY; } return str.substring(pos + separator.length()); } /** * get the value from the argMap, throw exception if not there and required * @param argMap * @param argMapNotUsed * @param key * @param required * @param defaultValue * @return the value or null or exception */ public static Integer argMapInteger(Map argMap, Map argMapNotUsed, String key, boolean required, Integer defaultValue) { String argString = argMapString(argMap, argMapNotUsed, key, required); if (isBlank(argString) && required) { throw new RuntimeException("Argument '--" + key + "' is required, but not specified. e.g. --" + key + "=5"); } if (isBlank(argString)) { if (defaultValue != null) { return defaultValue; } return null; } return intValue(argString); } /** * null safe convert from util date to sql date * @param date * @return the sql date */ public static java.sql.Date toSqlDate(Date date) { if (date == null) { return null; } return new java.sql.Date(date.getTime()); } /** *

Find the index of the given object in the array.

* *

This method returns -1 if null array input.

* * @param array the array to search through for the object, may be null * @param objectToFind the object to find, may be null * @return the index of the object within the array, * -1 if not found or null array input */ public static int indexOf(Object[] array, Object objectToFind) { return indexOf(array, objectToFind, 0); } /** *

Checks if the object is in the given array.

* *

The method returns false if a null array is passed in.

* * @param array the array to search through * @param objectToFind the object to find * @return true if the array contains the object */ public static boolean contains(Object[] array, Object objectToFind) { return indexOf(array, objectToFind) != -1; } /** *

Find the index of the given object in the array starting at the given index.

* *

This method returns -1 if null array input.

* *

A negative startIndex is treated as zero. A startIndex larger than the array * length will return -1.

* * @param array the array to search through for the object, may be null * @param objectToFind the object to find, may be null * @param startIndex the index to start searching at * @return the index of the object within the array starting at the index, * -1 if not found or null array input */ public static int indexOf(Object[] array, Object objectToFind, int startIndex) { if (array == null) { return -1; } if (startIndex < 0) { startIndex = 0; } if (objectToFind == null) { for (int i = startIndex; i < array.length; i++) { if (array[i] == null) { return i; } } } else { for (int i = startIndex; i < array.length; i++) { if (objectToFind.equals(array[i])) { return i; } } } return -1; } /** * Note, this is * web service format string */ private static final String WS_DATE_FORMAT = "yyyy/MM/dd HH:mm:ss.SSS"; /** * Note, this is * web service format string */ private static final String WS_DATE_FORMAT2 = "yyyy/MM/dd_HH:mm:ss.SSS"; /** * convert a date to a string using the standard web service pattern * yyyy/MM/dd HH:mm:ss.SSS Note that HH is 0-23 * * @param date * @return the string, or null if the date is null */ public static String dateToString(Date date) { if (date == null) { return null; } SimpleDateFormat simpleDateFormat = new SimpleDateFormat(WS_DATE_FORMAT); return simpleDateFormat.format(date); } /** * convert a string to a date using the standard web service pattern Note * that HH is 0-23 * * @param dateString * @return the string, or null if the date was null */ public static Date stringToDate(String dateString) { if (isBlank(dateString)) { return null; } SimpleDateFormat simpleDateFormat = new SimpleDateFormat(WS_DATE_FORMAT); try { return simpleDateFormat.parse(dateString); } catch (ParseException e) { SimpleDateFormat simpleDateFormat2 = new SimpleDateFormat(WS_DATE_FORMAT2); try { return simpleDateFormat2.parse(dateString); } catch (ParseException e2) { throw new RuntimeException("Cannot convert '" + dateString + "' to a date based on format: " + WS_DATE_FORMAT, e); } } } /** * match regex pattern yyyy-mm-dd or yyyy/mm/dd */ private static Pattern datePattern_yyyy_mm_dd = Pattern.compile("^(\\d{4})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})$"); /** * match regex pattern dd-mon-yyyy or dd/mon/yyyy */ private static Pattern datePattern_dd_mon_yyyy = Pattern.compile("^(\\d{1,2})[^\\d]+([a-zA-Z]{3,15})[^\\d]+(\\d{4})$"); /** * match regex pattern yyyy-mm-dd hh:mm:ss or yyyy/mm/dd hh:mm:ss */ private static Pattern datePattern_yyyy_mm_dd_hhmmss = Pattern.compile("^(\\d{4})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})$"); /** * match regex pattern dd-mon-yyyy hh:mm:ss or dd/mon/yyyy hh:mm:ss */ private static Pattern datePattern_dd_mon_yyyy_hhmmss = Pattern.compile("^(\\d{1,2})[^\\d]+([a-zA-Z]{3,15})[^\\d]+(\\d{4})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})$"); /** * match regex pattern yyyy-mm-dd hh:mm:ss.SSS or yyyy/mm/dd hh:mm:ss.SSS */ private static Pattern datePattern_yyyy_mm_dd_hhmmss_SSS = Pattern.compile("^(\\d{4})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,3})$"); /** * match regex pattern dd-mon-yyyy hh:mm:ss.SSS or dd/mon/yyyy hh:mm:ss.SSS */ private static Pattern datePattern_dd_mon_yyyy_hhmmss_SSS = Pattern.compile("^(\\d{1,2})[^\\d]+([a-zA-Z]{3,15})[^\\d]+(\\d{4})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,2})[^\\d]+(\\d{1,3})$"); /** * take as input: * yyyy/mm/dd * yyyy-mm-dd * dd-mon-yyyy * yyyy/mm/dd hh:mm:ss * dd-mon-yyyy hh:mm:ss * yyyy/mm/dd hh:mm:ss.SSS * dd-mon-yyyy hh:mm:ss.SSS * @param input * @return the date */ public static Date stringToDate2(String input) { if (isBlank(input)) { return null; } input = input.trim(); Matcher matcher = null; int month = 0; int day = 0; int year = 0; int hour = 0; int minute = 0; int second = 0; int milli = 0; boolean foundMatch = false; //yyyy/mm/dd if (!foundMatch) { matcher = datePattern_yyyy_mm_dd.matcher(input); if (matcher.matches()) { year = intValue(matcher.group(1)); month = intValue(matcher.group(2)); day = intValue(matcher.group(3)); foundMatch = true; } } //dd-mon-yyyy if (!foundMatch) { matcher = datePattern_dd_mon_yyyy.matcher(input); if (matcher.matches()) { day = intValue(matcher.group(1)); month = monthInt(matcher.group(2)); year = intValue(matcher.group(3)); foundMatch = true; } } //yyyy/mm/dd hh:mm:ss if (!foundMatch) { matcher = datePattern_yyyy_mm_dd_hhmmss.matcher(input); if (matcher.matches()) { year = intValue(matcher.group(1)); month = intValue(matcher.group(2)); day = intValue(matcher.group(3)); hour = intValue(matcher.group(4)); minute = intValue(matcher.group(5)); second = intValue(matcher.group(6)); foundMatch = true; } } //dd-mon-yyyy hh:mm:ss if (!foundMatch) { matcher = datePattern_dd_mon_yyyy_hhmmss.matcher(input); if (matcher.matches()) { day = intValue(matcher.group(1)); month = monthInt(matcher.group(2)); year = intValue(matcher.group(3)); hour = intValue(matcher.group(4)); minute = intValue(matcher.group(5)); second = intValue(matcher.group(6)); foundMatch = true; } } //yyyy/mm/dd hh:mm:ss.SSS if (!foundMatch) { matcher = datePattern_yyyy_mm_dd_hhmmss_SSS.matcher(input); if (matcher.matches()) { year = intValue(matcher.group(1)); month = intValue(matcher.group(2)); day = intValue(matcher.group(3)); hour = intValue(matcher.group(4)); minute = intValue(matcher.group(5)); second = intValue(matcher.group(6)); milli = intValue(matcher.group(7)); foundMatch = true; } } //dd-mon-yyyy hh:mm:ss.SSS if (!foundMatch) { matcher = datePattern_dd_mon_yyyy_hhmmss_SSS.matcher(input); if (matcher.matches()) { day = intValue(matcher.group(1)); month = monthInt(matcher.group(2)); year = intValue(matcher.group(3)); hour = intValue(matcher.group(4)); minute = intValue(matcher.group(5)); second = intValue(matcher.group(6)); milli = intValue(matcher.group(7)); foundMatch = true; } } Calendar calendar = Calendar.getInstance(); calendar.set(Calendar.YEAR, year); calendar.set(Calendar.MONTH, month-1); calendar.set(Calendar.DAY_OF_MONTH, day); calendar.set(Calendar.HOUR_OF_DAY, hour); calendar.set(Calendar.MINUTE, minute); calendar.set(Calendar.SECOND, second); calendar.set(Calendar.MILLISECOND, milli); return calendar.getTime(); } /** * convert a month string to an int (1 indexed). * e.g. if input is feb or Feb or february or February return 2 * @param mon * @return the month */ public static int monthInt(String mon) { if (!isBlank(mon)) { mon = mon.toLowerCase(); if (equals(mon, "jan") || equals(mon, "january")) { return 1; } if (equals(mon, "feb") || equals(mon, "february")) { return 2; } if (equals(mon, "mar") || equals(mon, "march")) { return 3; } if (equals(mon, "apr") || equals(mon, "april")) { return 4; } if (equals(mon, "may")) { return 5; } if (equals(mon, "jun") || equals(mon, "june")) { return 6; } if (equals(mon, "jul") || equals(mon, "july")) { return 7; } if (equals(mon, "aug") || equals(mon, "august")) { return 8; } if (equals(mon, "sep") || equals(mon, "september")) { return 9; } if (equals(mon, "oct") || equals(mon, "october")) { return 10; } if (equals(mon, "nov") || equals(mon, "november")) { return 11; } if (equals(mon, "dec") || equals(mon, "december")) { return 12; } } throw new RuntimeException("Invalid month: " + mon); } /** * override map for properties in thread local to be used in a web server or the like, based on property file name * @param propertiesFileName * @return the override map */ public static Map propertiesThreadLocalOverrideMap(String propertiesFileName) { Map> overrideMap = propertiesThreadLocalOverrideMap.get(); if (overrideMap == null) { overrideMap = new HashMap>(); propertiesThreadLocalOverrideMap.set(overrideMap); } Map propertiesOverrideMap = overrideMap.get(propertiesFileName); if (propertiesOverrideMap == null) { propertiesOverrideMap = new HashMap(); overrideMap.put(propertiesFileName, propertiesOverrideMap); } return propertiesOverrideMap; } /** * configure jdk14 logs once */ private static boolean configuredLogs = false; /** log level */ public static String theLogLevel = "WARNING"; /** * replace separators which are wrong os, and add last slash if not exist * @param filePath * @return the file path */ public static String fileAddLastSlashIfNotExists(String filePath) { filePath = filePath.replace(File.separatorChar == '/' ? '\\' : '/', File.separatorChar); if (!filePath.endsWith(File.separator)) { filePath = filePath + File.separatorChar; } return filePath; } /** * @param theClass * @return the log */ public static Log retrieveLog(Class theClass) { Log theLog = LogFactory.getLog(theClass); if (!configuredLogs) { String logLevel = theLogLevel; String logFile = null; boolean hasLogLevel = !isBlank(logLevel); boolean hasLogFile = !isBlank(logFile); if (hasLogLevel || hasLogFile) { if (theLog instanceof Jdk14Logger) { Jdk14Logger jdkLogger = (Jdk14Logger) theLog; Logger logger = jdkLogger.getLogger(); long timeToLive = 60; while (logger.getParent() != null && timeToLive-- > 0) { //this should be root logger logger = logger.getParent(); } if (length(logger.getHandlers()) == 1) { //remove console appender if only one if (logger.getHandlers()[0].getClass() == ConsoleHandler.class) { logger.removeHandler(logger.getHandlers()[0]); } } if (length(logger.getHandlers()) == 0) { Handler handler = null; if (hasLogFile) { try { handler = new FileHandler(logFile, true); } catch (IOException ioe) { throw new RuntimeException(ioe); } } else { handler = new ConsoleHandler(); } handler.setFormatter(new SimpleFormatter()); handler.setLevel(Level.ALL); logger.addHandler(handler); logger.setUseParentHandlers(false); } if (hasLogLevel) { Level level = Level.parse(logLevel); logger.setLevel(level); } } } configuredLogs = true; } return new GrouperInstallerLog(theLog); } /** * turn a directory and contents into a tar file * @param directory * @param tarFile */ public static void tar(File directory, File tarFile) { tar(directory, tarFile, true); } /** * turn a directory and contents into a tar file * @param directory * @param includeDirectoryInTarPath true if should include the dirname in the tar file * @param tarFile */ public static void tar(File directory, File tarFile, boolean includeDirectoryInTarPath) { try { tarFile.createNewFile(); TarArchiveOutputStream tarArchiveOutputStream = new TarArchiveOutputStream(new FileOutputStream(tarFile)); //we need long file support tarArchiveOutputStream.setLongFileMode(TarArchiveOutputStream.LONGFILE_POSIX); for (File file : fileListRecursive(directory)) { if (file.isFile()) { String relativePath = (includeDirectoryInTarPath ? (directory.getName() + File.separator) : "") + fileRelativePath(directory, file); //lets do back slashes relativePath = replace(relativePath, "\\", "/"); TarArchiveEntry entry = new TarArchiveEntry(file, relativePath); tarArchiveOutputStream.putArchiveEntry(entry); copy(new FileInputStream(file), tarArchiveOutputStream); tarArchiveOutputStream.closeArchiveEntry(); } } tarArchiveOutputStream.close(); } catch (Exception e) { throw new RuntimeException("Error creating tar: " + tarFile.getAbsolutePath() + ", from dir: " + directory.getAbsolutePath(), e); } } /** * * @param inputFile * @param outputFile */ public static void gzip(File inputFile, File outputFile) { GZIPOutputStream gzipOutputStream = null; try { outputFile.createNewFile(); gzipOutputStream = new GZIPOutputStream( new BufferedOutputStream(new FileOutputStream(outputFile))); copy(new FileInputStream(inputFile), gzipOutputStream); } catch (Exception e) { throw new RuntimeException("Error creating gzip from " + inputFile.getAbsolutePath() + " to " + outputFile.getAbsolutePath()); } finally { closeQuietly(gzipOutputStream); } } /** * * @param file * @return the hex sha1 */ public static String fileSha1(File file) { FileInputStream fis = null; try { MessageDigest md = MessageDigest.getInstance("SHA1"); fis = new FileInputStream(file); byte[] dataBytes = new byte[1024]; int nread = 0; while ((nread = fis.read(dataBytes)) != -1) { md.update(dataBytes, 0, nread); }; byte[] mdbytes = md.digest(); //convert the byte to hex format StringBuffer sb = new StringBuffer(""); for (int i = 0; i < mdbytes.length; i++) { sb.append(Integer.toString((mdbytes[i] & 0xff) + 0x100, 16).substring(1)); } return sb.toString(); } catch (Exception e) { throw new RuntimeException("Problem getting checksum of file: " + file.getAbsolutePath(), e); } finally { closeQuietly(fis); } } /** override map for properties */ private static Map grouperInstallerOverrideMap = new LinkedHashMap(); /** * override map for properties for testing * @return the override map */ public static Map grouperInstallerOverrideMap() { return grouperInstallerOverrideMap; } /** * grouper installer properties * @return the properties */ public static Properties grouperInstallerProperties() { Properties properties = null; try { properties = propertiesFromResourceName( "grouper.installer.properties", true, false, GrouperInstallerUtils.class, null); if (properties == null) { properties = new Properties(); } } catch (Exception e) { throw new RuntimeException("Error accessing file: grouper.installer.properties " + "This properties file needs to be in the same directory as grouperInstaller.jar, or on your Java classpath", e); } return properties; } /** * if the properties contains a key * @param key * @return true or false */ public static boolean propertiesContainsKey(String key) { return grouperInstallerProperties().containsKey(key); } /** * get a property and validate required from grouper.installer.properties * @param key * @param required * @return the value */ public static String propertiesValue(String key, boolean required) { return GrouperInstallerUtils.propertiesValue("grouper.installer.properties", grouperInstallerProperties(), GrouperInstallerUtils.grouperInstallerOverrideMap(), key, required); } /** * get a boolean and validate from grouper.installer.properties * @param key * @param defaultValue * @param required * @return the string */ public static boolean propertiesValueBoolean(String key, boolean defaultValue, boolean required ) { return GrouperInstallerUtils.propertiesValueBoolean( "grouper.installer.properties", grouperInstallerProperties(), GrouperInstallerUtils.grouperInstallerOverrideMap(), key, defaultValue, required); } /** * get a boolean and validate from grouper.installer.properties * @param key * @param defaultValue * @param required * @return the string */ public static int propertiesValueInt(String key, int defaultValue, boolean required ) { return GrouperInstallerUtils.propertiesValueInt( "grouper.installer.properties", grouperInstallerProperties(), GrouperInstallerUtils.grouperInstallerOverrideMap(), key, defaultValue, required); } /** * Copy bytes from an InputStream to an * OutputStream. *

* This method buffers the input internally, so there is no need to use a * BufferedInputStream. *

* Large streams (over 2GB) will return a bytes copied value of * -1 after the copy has completed since the correct * number of bytes cannot be returned as an int. For large streams * use the copyLarge(InputStream, OutputStream) method. * * @param input the InputStream to read from * @param output the OutputStream to write to * @return the number of bytes copied * @throws NullPointerException if the input or output is null * @throws IOException if an I/O error occurs * @throws ArithmeticException if the byte count is too large * @since Commons IO 1.1 */ public static int copy(InputStream input, OutputStream output) throws IOException { long count = copyLarge(input, output); if (count > Integer.MAX_VALUE) { return -1; } return (int) count; } /** * Copy bytes from a large (over 2GB) InputStream to an * OutputStream. *

* This method buffers the input internally, so there is no need to use a * BufferedInputStream. * * @param input the InputStream to read from * @param output the OutputStream to write to * @return the number of bytes copied * @throws NullPointerException if the input or output is null * @throws IOException if an I/O error occurs * @since Commons IO 1.3 */ public static long copyLarge(InputStream input, OutputStream output) throws IOException { byte[] buffer = new byte[DEFAULT_BUFFER_SIZE]; long count = 0; int n = 0; while (-1 != (n = input.read(buffer))) { output.write(buffer, 0, n); count += n; } return count; } /** * clear out all files recursively in a directory, including the directory * itself * @param dirName * * @throws RuntimeException * when something goes wrong */ public static void deleteRecursiveDirectory(String dirName) { //delete all files in the directory File dir = new File(dirName); //if it doesnt exist then we are done if (!dir.exists()) { return; } //see if its a directory if (!dir.isDirectory()) { throw new RuntimeException("The directory: " + dirName + " is not a directory"); } //get the files into a vector File[] allFiles = dir.listFiles(); //loop through the array for (int i = 0; i < allFiles.length; i++) { if (-1 < allFiles[i].getName().indexOf("..")) { continue; //dont go to the parent directory } if (allFiles[i].isFile()) { //delete the file if (!allFiles[i].delete()) { throw new RuntimeException("Could not delete file: " + allFiles[i].getPath()); } } else { //its a directory deleteRecursiveDirectory(allFiles[i].getPath()); } } //delete the directory itself if (!dir.delete()) { throw new RuntimeException("Could not delete directory: " + dir.getPath()); } } /** * This will execute a command, and split spaces for args (might not be what * you want if you are using quotes) * * @param command * @param printProgress if dots for progress should be used * @return the result of the command */ public static CommandResult execCommand(String command, boolean printProgress) { String[] args = splitTrim(command, " "); return execCommand(args, printProgress); } /** * Gobble up a stream from a runtime * @author mchyzer */ private static class StreamGobbler implements Runnable { /** stream to read */ private InputStream inputStream; /** where to put the result */ private String resultString; /** type of the output for logging purposes */ private String type; /** command to log */ private String command; /** if print to stdout */ private File printToFile; /** if this is out or error */ private boolean outOrErr; /** * if should print stdout and stderr as received */ private boolean printOutputErrorAsReceived; /** * construct * @param is * @param theType * @param theCommand * @param thePrintToFile * @param thePrintOutputErrorAsReceived * @param theOutOrErr */ private StreamGobbler(InputStream is, String theType, String theCommand, File thePrintToFile, boolean thePrintOutputErrorAsReceived, boolean theOutOrErr) { this.inputStream = is; this.type = theType; this.command = theCommand; this.printToFile = thePrintToFile; this.printOutputErrorAsReceived = thePrintOutputErrorAsReceived; this.outOrErr = theOutOrErr; } /** * get the string result * @return the result */ public String getResultString() { return this.resultString; } /** * * @see java.lang.Runnable#run() */ @Override public void run() { FileOutputStream fileOutputStream = null; try { fileOutputStream = this.printToFile == null ? null : new FileOutputStream(this.printToFile); } catch (IOException ioe) { throw new RuntimeException(ioe); } try { if (this.printOutputErrorAsReceived) { if (this.outOrErr) { copy(this.inputStream, System.out); } else { copy(this.inputStream, System.err); } } else if (this.printToFile != null) { copy(this.inputStream, fileOutputStream); } else { StringWriter stringWriter = new StringWriter(); copy(this.inputStream, stringWriter); this.resultString = stringWriter.toString(); } } catch (Exception e) { LOG.warn("Error saving output of executable: " + (this.resultString) + ", " + this.type + ", " + this.command, e); throw new RuntimeException(e); } finally { closeQuietly(fileOutputStream); } } } /** *

This will execute a command (with args). In this method, 
   * if the exit code of the command is not zero, an exception will be thrown.
   * Example call: execCommand(new String[]{"/bin/bash", "-c", "cd /someFolder; runSomeScript.sh"}, true);
   * 
* @param arguments are the commands * @param printProgress if dots for progress should be used * @return the output text of the command. */ public static CommandResult execCommand(String[] arguments, boolean printProgress) { return execCommand(arguments, true, printProgress); } /** *
This will execute a command (with args). In this method, 
   * if the exit code of the command is not zero, an exception will be thrown.
   * Example call: execCommand(new String[]{"/bin/bash", "-c", "cd /someFolder; runSomeScript.sh"}, true);
   * 
* @param command to execute * @param arguments are the commands * @param printProgress if dots for progress should be used * @return the output text of the command. */ public static CommandResult execCommand(String command, String[] arguments, boolean printProgress) { List args = new ArrayList(); args.add(command); for (String argument : nonNull(arguments, String.class)) { args.add(argument); } return execCommand(toArray(args, String.class), true, printProgress); } /** * threadpool */ private static ExecutorService executorService = Executors.newCachedThreadPool(); /** * * @return executor service */ public static ExecutorService retrieveExecutorService() { return executorService; } /** *
This will execute a command (with args). Under normal operation, 
   * if the exit code of the command is not zero, an exception will be thrown.
   * If the parameter exceptionOnExitValueNeZero is set to true, the 
   * results of the call will be returned regardless of the exit status.
   * Example call: execCommand(new String[]{"/bin/bash", "-c", "cd /someFolder; runSomeScript.sh"}, true);
   * 
* @param arguments are the commands * @param exceptionOnExitValueNeZero if this is set to false, the * results of the call will be returned regardless of the exit status * @param printProgress if dots for progress should be used * @return the output text of the command, and the error and return code if exceptionOnExitValueNeZero is false. */ public static CommandResult execCommand(String[] arguments, boolean exceptionOnExitValueNeZero, boolean printProgress) { return execCommand(arguments, exceptionOnExitValueNeZero, true, printProgress); } /** *
This will execute a command (with args). Under normal operation, 
   * if the exit code of the command is not zero, an exception will be thrown.
   * If the parameter exceptionOnExitValueNeZero is set to true, the 
   * results of the call will be returned regardless of the exit status.
   * Example call: execCommand(new String[]{"/bin/bash", "-c", "cd /someFolder; runSomeScript.sh"}, true);
   * 
* @param arguments are the commands * @param exceptionOnExitValueNeZero if this is set to false, the * results of the call will be returned regardless of the exit status * @param waitFor if we should wait for this process to end * @param printProgress if dots for progress should be used * @return the output text of the command, and the error and return code if exceptionOnExitValueNeZero is false. */ public static CommandResult execCommand(String[] arguments, boolean exceptionOnExitValueNeZero, boolean waitFor, boolean printProgress) { return execCommand(arguments, exceptionOnExitValueNeZero, waitFor, null, null, null, printProgress); } /** *
This will execute a command (with args). Under normal operation, 
   * if the exit code of the command is not zero, an exception will be thrown.
   * If the parameter exceptionOnExitValueNeZero is set to true, the 
   * results of the call will be returned regardless of the exit status.
   * Example call: execCommand(new String[]{"/bin/bash", "-c", "cd /someFolder; runSomeScript.sh"}, true);
   * 
* @param arguments are the commands * @param exceptionOnExitValueNeZero if this is set to false, the * results of the call will be returned regardless of the exit status * @param waitFor if we should wait for this process to end * @param workingDirectory * @param envVariables are env vars with name=val * @param outputFilePrefix will be the file prefix and Out.log and Err.log will be added to them * @param printProgress if dots for progress should be used * @return the output text of the command, and the error and return code if exceptionOnExitValueNeZero is false. */ public static CommandResult execCommand(String[] arguments, boolean exceptionOnExitValueNeZero, boolean waitFor, String[] envVariables, File workingDirectory, String outputFilePrefix, boolean printProgress) { return execCommand(arguments, exceptionOnExitValueNeZero, waitFor, envVariables, workingDirectory, outputFilePrefix, false, printProgress); } /** *
This will execute a command (with args). Under normal operation, 
   * if the exit code of the command is not zero, an exception will be thrown.
   * If the parameter exceptionOnExitValueNeZero is set to true, the 
   * results of the call will be returned regardless of the exit status.
   * Example call: execCommand(new String[]{"/bin/bash", "-c", "cd /someFolder; runSomeScript.sh"}, true);
   * 
* @param arguments are the commands * @param exceptionOnExitValueNeZero if this is set to false, the * results of the call will be returned regardless of the exit status * @param waitFor if we should wait for this process to end * @param workingDirectory * @param envVariables are env vars with name=val * @param outputFilePrefix will be the file prefix and Out.log and Err.log will be added to them * @param printOutputErrorAsReceived if should print output error as received * @param printProgress if dots for progress should be used * @return the output text of the command, and the error and return code if exceptionOnExitValueNeZero is false. */ public static CommandResult execCommand(final String[] arguments, final boolean exceptionOnExitValueNeZero, final boolean waitFor, final String[] envVariables, final File workingDirectory, final String outputFilePrefix, final boolean printOutputErrorAsReceived, boolean printProgress) { return execCommand(arguments, exceptionOnExitValueNeZero, waitFor, envVariables, workingDirectory, outputFilePrefix, printOutputErrorAsReceived, printProgress, true); } /** *
This will execute a command (with args). Under normal operation, 
   * if the exit code of the command is not zero, an exception will be thrown.
   * If the parameter exceptionOnExitValueNeZero is set to true, the 
   * results of the call will be returned regardless of the exit status.
   * Example call: execCommand(new String[]{"/bin/bash", "-c", "cd /someFolder; runSomeScript.sh"}, true);
   * 
* @param arguments are the commands * @param exceptionOnExitValueNeZero if this is set to false, the * results of the call will be returned regardless of the exit status * @param waitFor if we should wait for this process to end * @param workingDirectory * @param envVariables are env vars with name=val * @param outputFilePrefix will be the file prefix and Out.log and Err.log will be added to them * @param printOutputErrorAsReceived if should print output error as received * @param printProgress if dots for progress should be used * @param logError true if errors should be logged. otherwise they will only be thrown * @return the output text of the command, and the error and return code if exceptionOnExitValueNeZero is false. */ public static CommandResult execCommand(final String[] arguments, final boolean exceptionOnExitValueNeZero, final boolean waitFor, final String[] envVariables, final File workingDirectory, final String outputFilePrefix, final boolean printOutputErrorAsReceived, boolean printProgress, final boolean logError) { final CommandResult[] result = new CommandResult[1]; Runnable runnable = new Runnable() { public void run() { result[0] = execCommandHelper(arguments, exceptionOnExitValueNeZero, waitFor, envVariables, workingDirectory, outputFilePrefix, printOutputErrorAsReceived, logError); } }; GrouperInstallerUtils.threadRunWithStatusDots(runnable, printProgress, logError); return result[0]; } /** *
This will execute a command (with args). Under normal operation, 
   * if the exit code of the command is not zero, an exception will be thrown.
   * If the parameter exceptionOnExitValueNeZero is set to true, the 
   * results of the call will be returned regardless of the exit status.
   * Example call: execCommand(new String[]{"/bin/bash", "-c", "cd /someFolder; runSomeScript.sh"}, true);
   * 
* @param arguments are the commands * @param exceptionOnExitValueNeZero if this is set to false, the * results of the call will be returned regardless of the exit status * @param waitFor if we should wait for this process to end * @param workingDirectory * @param envVariables are env vars with name=val * @param outputFilePrefix will be the file prefix and Out.log and Err.log will be added to them * @param printOutputErrorAsReceived if should print output error as received * @param logError if error should be logged, otherwise it will only be thrown * @return the output text of the command, and the error and return code if exceptionOnExitValueNeZero is false. */ private static CommandResult execCommandHelper(String[] arguments, boolean exceptionOnExitValueNeZero, boolean waitFor, String[] envVariables, File workingDirectory, String outputFilePrefix, boolean printOutputErrorAsReceived, boolean logError) { if (printOutputErrorAsReceived && !isBlank(outputFilePrefix)) { throw new RuntimeException("Cant print as received and have output file prefix"); } Process process = null; StringBuilder commandBuilder = new StringBuilder(); for (int i = 0; i < arguments.length; i++) { commandBuilder.append(arguments[i]).append(" "); } String command = commandBuilder.toString(); if (LOG.isDebugEnabled()) { LOG.debug("Running command: " + command); } StreamGobbler outputGobbler = null; StreamGobbler errorGobbler = null; try { process = Runtime.getRuntime().exec(arguments, envVariables, workingDirectory); if (!waitFor) { return new CommandResult(null, null, -1); } outputGobbler = new StreamGobbler(process.getInputStream(), ".out", command, outputFilePrefix == null ? null : new File(outputFilePrefix + "Out.log"), printOutputErrorAsReceived, true); errorGobbler = new StreamGobbler(process.getErrorStream(), ".err", command, outputFilePrefix == null ? null : new File(outputFilePrefix + "Err.log"), printOutputErrorAsReceived, false); Thread outputThread = new Thread(outputGobbler); outputThread.setDaemon(true); outputThread.start(); Thread errorThread = new Thread(errorGobbler); errorThread.setDaemon(true); errorThread.start(); try { process.waitFor(); } finally { //finish running these threads try { outputThread.join(); } catch (Exception e) { } try { errorThread.join(); } catch (Exception e) { } } } catch (Exception e) { if (logError) { LOG.error("Error running command: " + command, e); } throw new RuntimeException("Error running command: " + command + ", " + e.getMessage(), e); } finally { try { process.destroy(); } catch (Exception e) { } } //was not successful??? if (process.exitValue() != 0 && exceptionOnExitValueNeZero) { String message = "Process exit status=" + process.exitValue() + ": out: " + (outputGobbler == null ? null : outputGobbler.getResultString()) + ", err: " + (errorGobbler == null ? null : errorGobbler.getResultString()); if (logError) { LOG.error(message + ", on command: " + command + (workingDirectory == null ? "" : (", workingDir: " + workingDirectory.getAbsolutePath()))); } throw new RuntimeException(message); } int exitValue = process.exitValue(); return new CommandResult(errorGobbler.getResultString(), outputGobbler.getResultString(), exitValue); } /** * The results of executing a command. */ public static class CommandResult{ /** * If any error text was generated by the call, it will be set here. */ private String errorText; /** * If any output text was generated by the call, it will be set here. */ private String outputText; /** * If any exit code was generated by the call, it will be set here. */ private int exitCode; /** * Create a container to hold the results of an execution. * @param _errorText * @param _outputText * @param _exitCode */ public CommandResult(String _errorText, String _outputText, int _exitCode){ this.errorText = _errorText; this.outputText = _outputText; this.exitCode = _exitCode; } /** * If any error text was generated by the call, it will be set here. * @return the errorText */ public String getErrorText() { return this.errorText; } /** * If any output text was generated by the call, it will be set here. * @return the outputText */ public String getOutputText() { return this.outputText; } /** * If any exit code was generated by the call, it will be set here. * @return the exitCode */ public int getExitCode() { return this.exitCode; } } /** * * @return java command */ @Deprecated public static String javaCommand() { return javaHome() + File.separator + "bin" + File.separator + "java"; } /** */ private static String JAVA_HOME = null; /** * The effective java home (System.getProperty("java.home")), based on the path to the currently running * java; deprecated since it may not be the same as the JAVA_HOME environment variable (e.g. in a Windows * install java may be set to the jre,even if the JDK is present * * @return the java home location without slash afterward */ @Deprecated public static String javaHome() { if (isBlank(JAVA_HOME)) { //try to get a jdk JAVA_HOME = System.getProperty("java.home"); if (JAVA_HOME.endsWith("jre")) { String newJavaHome = JAVA_HOME.substring(0,JAVA_HOME.length()-4); File javac = new File(newJavaHome + File.separator + "bin" + File.separator + "javac"); if (javac.exists()) { JAVA_HOME = newJavaHome; } javac = new File(newJavaHome + File.separator + "bin" + File.separator + "javac.exe"); if (javac.exists()) { JAVA_HOME = newJavaHome; } } } return JAVA_HOME; } /** * Checks to see if a specific port is available. * * @param port the port to check for availability * @return true if available */ public static boolean portAvailable(int port) { return portAvailable(port, null); } /** * Checks to see if a specific port is available. * * @param port the port to check for availability * @param ipAddress * @return true if available */ public static boolean portAvailable(int port, String ipAddress) { ServerSocket ss = null; try { if (isBlank(ipAddress) || "0.0.0.0".equals(ipAddress)) { ss = new ServerSocket(port); } else { Pattern pattern = Pattern.compile("(\\d+)\\.(\\d+)\\.(\\d+)\\.(\\d+)"); Matcher matcher = pattern.matcher(ipAddress); if (!matcher.matches()) { throw new RuntimeException("IP address not valid! '" + ipAddress + "'"); } byte[] b = new byte[4]; for (int i=0;i<4;i++) { int theInt = intValue(matcher.group(i+1)); if (theInt > 255 || theInt < 0) { System.out.println("IP address part must be between 0 and 255: '" + theInt + "'"); } b[i] = (byte)theInt; } //Returns an InetAddress object given the raw IP address . InetAddress inetAddress = InetAddress.getByAddress(b); ss = new ServerSocket(port, 50, inetAddress); } ss.setReuseAddress(true); return true; } catch (IOException e) { } finally { if (ss != null) { try { ss.close(); } catch (IOException e) { /* should not be thrown */ } } } return false; } /** * add a jar to classpath * @param file */ public static void classpathAddFile(File file) { try { classpathAddUrl(file.toURI().toURL()); } catch (IOException ioe) { throw new RuntimeException("Problem adding file to classpath: " + (file == null ? null : file.getAbsolutePath())); } } /** * keep track of urls added so we dont add twice */ private static Set urlsAddedToClasspath = new HashSet(); /** * use reflection to add a jar to the classpath * @param url */ public static void classpathAddUrl(URL url) { String urlString = url.toString(); if (urlsAddedToClasspath.contains(urlString)) { return; } URLClassLoader sysloader = (URLClassLoader) ClassLoader.getSystemClassLoader(); Class sysclass = URLClassLoader.class; try { Method method = sysclass.getDeclaredMethod("addURL", new Class[] { URL.class }); method.setAccessible(true); method.invoke(sysloader, new Object[] { url }); } catch (Throwable t) { throw new RuntimeException("Error, could not add URL to system classloader: " + urlString, t); } urlsAddedToClasspath.add(urlString); } /** * Return the zero element of the list, if it exists, null if the list is empty. * If there is more than one element in the list, an exception is thrown. * @param * @param list is the container of objects to get the first of. * @return the first object, null, or exception. */ public static T listPopOne(List list) { int size = length(list); if (size == 1) { return list.get(0); } else if (size == 0) { return null; } throw new RuntimeException("More than one object of type " + className(list.get(0)) + " was returned when only one was expected. (size:" + size +")" ); } /** * Copy a file to another file. this perserves the file date * * @param sourceFile * @param destinationFile */ public static void copyFile(File sourceFile, File destinationFile) { copyFile(sourceFile, destinationFile, true); } /** * Copy a file to another file. this perserves the file date * * @param sourceFile * @param destinationFile * @param onlyIfDifferentContents true if only saving due to different contents. Note, this is only for non-binary files! * @param ignoreWhitespace true to ignore whitespace in comparisons * @return true if contents were saved (thus different if param set) */ public static boolean copyFile(File sourceFile, File destinationFile, boolean onlyIfDifferentContents, boolean ignoreWhitespace) { if (onlyIfDifferentContents) { String sourceContents = readFileIntoString(sourceFile); return saveStringIntoFile(destinationFile, sourceContents, onlyIfDifferentContents, ignoreWhitespace); } copyFile(sourceFile, destinationFile); return true; } /** * Copies a file to a new location. *

* This method copies the contents of the specified source file * to the specified destination file. * The directory holding the destination file is created if it does not exist. * If the destination file exists, then this method will overwrite it. * * @param srcFile an existing file to copy, must not be null * @param destFile the new file, must not be null * @param preserveFileDate true if the file date of the copy * should be the same as the original * * @throws NullPointerException if source or destination is null */ public static void copyFile(File srcFile, File destFile, boolean preserveFileDate) { try { if (srcFile == null) { throw new NullPointerException("Source must not be null"); } if (destFile == null) { throw new NullPointerException("Destination must not be null"); } if (srcFile.exists() == false) { throw new FileNotFoundException("Source '" + srcFile + "' does not exist"); } if (srcFile.isDirectory()) { throw new IOException("Source '" + srcFile + "' exists but is a directory"); } if (srcFile.getCanonicalPath().equals(destFile.getCanonicalPath())) { throw new IOException("Source '" + srcFile + "' and destination '" + destFile + "' are the same"); } if (destFile.getParentFile() != null && destFile.getParentFile().exists() == false) { if (destFile.getParentFile().mkdirs() == false) { throw new IOException("Destination '" + destFile + "' directory cannot be created"); } } if (destFile.exists() && destFile.canWrite() == false) { throw new IOException("Destination '" + destFile + "' exists but is read-only"); } doCopyFile(srcFile, destFile, preserveFileDate); } catch (IOException ioe) { throw new RuntimeException(ioe); } } /** * Internal copy file method. * * @param srcFile the validated source file, must not be null * @param destFile the validated destination file, must not be null * @param preserveFileDate whether to preserve the file date * @throws IOException if an error occurs */ private static void doCopyFile(File srcFile, File destFile, boolean preserveFileDate) throws IOException { if (destFile.exists() && destFile.isDirectory()) { throw new IOException("Destination '" + destFile + "' exists but is a directory"); } FileInputStream input = new FileInputStream(srcFile); try { FileOutputStream output = new FileOutputStream(destFile); try { copy(input, output); } finally { closeQuietly(output); } } finally { closeQuietly(input); } if (srcFile.length() != destFile.length()) { throw new IOException("Failed to copy full contents from '" + srcFile + "' to '" + destFile + "'"); } if (preserveFileDate) { destFile.setLastModified(srcFile.lastModified()); } } /** * * @param xmlFile * @param xpathExpression * @return the nodelist */ public static NodeList xpathEvaluate(File xmlFile, String xpathExpression) { InputStream inputStream = null; try { inputStream = new FileInputStream(xmlFile); return xpathEvaluate(inputStream, xpathExpression); } catch (Exception e) { String errorMessage = "Problem with file: " + xmlFile == null ? null : xmlFile.getAbsolutePath(); if (e instanceof RuntimeException) { GrouperInstallerUtils.injectInException(e, errorMessage); throw (RuntimeException)e; } throw new RuntimeException(errorMessage, e); } finally { GrouperInstallerUtils.closeQuietly(inputStream); } } /** * @param url * @param xpathExpression * @return the nodelist */ public static NodeList xpathEvaluate(URL url, String xpathExpression) { InputStream inputStream = null; try { inputStream = url.openStream(); return xpathEvaluate(inputStream, xpathExpression); } catch (Exception e) { String errorMessage = "Problem with url: " + url == null ? null : url.toExternalForm(); if (e instanceof RuntimeException) { GrouperInstallerUtils.injectInException(e, errorMessage); throw (RuntimeException)e; } throw new RuntimeException(errorMessage, e); } finally { GrouperInstallerUtils.closeQuietly(inputStream); } } /** * @param inputStream * @param xpathExpression * @return the nodelist */ public static NodeList xpathEvaluate(InputStream inputStream, String xpathExpression) { try { DocumentBuilderFactory domFactory = xmlDocumentBuilderFactory(); DocumentBuilder builder = domFactory.newDocumentBuilder(); Document doc = builder.parse(inputStream); XPath xpath = XPathFactory.newInstance().newXPath(); // XPath Query for showing all nodes value XPathExpression expr = xpath.compile(xpathExpression); Object result = expr.evaluate(doc, XPathConstants.NODESET); NodeList nodes = (NodeList) result; return nodes; } catch (Exception e) { throw new RuntimeException("Problem evaluating xpath: " + ", expression: '" + xpathExpression + "'", e); } } /** * xml builder factory that doesnt go to internet to get dtd * @return the factory */ public static DocumentBuilderFactory xmlDocumentBuilderFactory() { DocumentBuilderFactory domFactory = DocumentBuilderFactory.newInstance(); domFactory.setNamespaceAware(true); domFactory.setValidating(false); try { domFactory.setFeature("http://apache.org/xml/features/nonvalidating/load-external-dtd", false); } catch (ParserConfigurationException pce) { throw new RuntimeException(pce); } return domFactory; } /** array for converting HTML to string */ private static final String[] XML_SEARCH_NO_SINGLE = new String[]{"&","<",">","\""}; /** array for converting HTML to string */ private static final String[] XML_REPLACE_NO_SINGLE = new String[]{"&","<",">","""}; /** * Convert an XML string to HTML to display on the screen * * @param input * is the XML to convert * * @return the HTML converted string */ public static String xmlEscape(String input) { return xmlEscape(input, true); } /** * Convert an XML string to HTML to display on the screen * * @param input * is the XML to convert * @param isEscape true to escape chars, false to unescape * * @return the HTML converted string */ public static String xmlEscape(String input, boolean isEscape) { if (isEscape) { return replace(input, XML_SEARCH_NO_SINGLE, XML_REPLACE_NO_SINGLE); } return replace(input, XML_REPLACE_NO_SINGLE, XML_SEARCH_NO_SINGLE); } /** * * @param xmlFile * @param xpathExpression * @param attributeName * @return the nodelist */ public static String xpathEvaluateAttribute(File xmlFile, String xpathExpression, String attributeName) { NodeList nodes = GrouperInstallerUtils.xpathEvaluate(xmlFile, xpathExpression); if (nodes == null || nodes.getLength() == 0) { return null; } if (nodes.getLength() != 1) { throw new RuntimeException("There is more than 1 xpath expression: '" + xpathExpression + "' element in server.xml: " + xmlFile.getAbsolutePath()); } NamedNodeMap attributes = nodes.item(0).getAttributes(); if (attributes == null || attributes.getLength() == 0 ) { return null; } Node attribute = attributes.getNamedItem(attributeName); if (attribute == null) { return null; } String nodeValue = attribute.getNodeValue(); return nodeValue; } /** * turn a map of attributes into an xml string * @param elementName * @param extraAttributes * @param attributes * @return the xml */ public static String xmlElementToXml(String elementName, String extraAttributes, Map attributes) { StringBuilder result = new StringBuilder(); result.append("<").append(elementName).append(" "); if (!isBlank(extraAttributes)) { result.append(extraAttributes); } for (String attributeName : attributes.keySet()) { result.append(" ").append(attributeName).append("=\""); String attributeValue = GrouperInstallerUtils.trimToEmpty(attributes.get(attributeName)); result.append(GrouperInstallerUtils.xmlEscape(attributeValue)).append("\""); } result.append(" />"); return result.toString(); } /** * convert XML to string * @param document * @return the string of the XML */ public static String xmlToString(Node document) { try { DOMSource domSource = new DOMSource(document); StringWriter writer = new StringWriter(); StreamResult result = new StreamResult(writer); TransformerFactory tf = TransformerFactory.newInstance(); Transformer transformer = tf.newTransformer(); transformer.transform(domSource, result); return writer.toString(); } catch (Exception exception) { throw new RuntimeException(exception); } } /** * * @param xmlFile * @param xpathExpression * @param attributeName * @param defaultValue * @return the nodelist */ public static Integer xpathEvaluateAttributeInt(File xmlFile, String xpathExpression, String attributeName, Integer defaultValue) { String nodeValue = xpathEvaluateAttribute(xmlFile, xpathExpression, attributeName); Integer intValue = GrouperInstallerUtils.intValue(nodeValue, defaultValue); return intValue; } /** * * @param jarFile * @return the version */ public static String jarVersion(File jarFile) { return jarVersion(jarFile, false); } /** * see which jar is newer or null if dont know * @param jar1 * @param jar2 * @return the jar which is newer or null if cant find */ public static File jarNewerVersion(File jar1, File jar2) { String version1 = jarVersion(jar1, false); String version2 = jarVersion(jar2, false); if (version1 == null && version2 == null) { return null; } if (version1 == null) { return jar2; } if (version2 == null) { return jar1; } int compare = compareVersions(version1, version2); return (compare >= 0 ? jar1 : jar2); } /** properties in manifest for version */ private static final String[] versionProperties = new String[]{ "Implementation-Version","Version"}; /** * get the version from the manifest of a jar * @param sampleClass * @return the version * @throws Exception */ public static String jarVersion(Class sampleClass) throws Exception { return manifestProperty(sampleClass, versionProperties); } private static String grouperVersionString = null; /** * get the version from jar e.g. 2.5.12 * @return the version */ public static String grouperInstallerVersion() { if (grouperVersionString == null) { try { grouperVersionString = GrouperInstallerUtils.jarVersion(GiGrouperVersion.class); } catch (Exception e) { if (LOG.isDebugEnabled()) { LOG.debug("Can't find version of grouper jar, using 2.5.0", e); } else { LOG.warn("Can't find version of grouper jar, using 2.5.0"); } } if (grouperVersionString == null) { grouperVersionString = "2.5.0"; } } return grouperVersionString; } /** * get the version from the manifest of a jar * @param sampleClass * @param propertyNames that we are looking for (usually just one) * @return the version * @throws Exception */ public static String manifestProperty(Class sampleClass, String[] propertyNames) throws Exception { File jarFile = jarFile(sampleClass); URL manifestUrl = new URL("jar:file:" + jarFile.getCanonicalPath() + "!/META-INF/MANIFEST.MF"); Manifest manifest = new Manifest(manifestUrl.openStream()); Map attributeMap = manifest.getEntries(); String value = null; for (String propertyName : propertyNames) { value = manifest.getMainAttributes().getValue(propertyName); if (!isBlank(value)) { break; } } if (value == null) { OUTER: for (Attributes attributes: attributeMap.values()) { for (String propertyName : propertyNames) { value = attributes.getValue(propertyName); if (!isBlank(value)) { break OUTER; } } } } if (value == null) { for (Attributes attributes: attributeMap.values()) { for (Object key : attributes.keySet()) { LOG.info(jarFile.getName() + ", " + key + ": " + attributes.getValue((Name)key)); } } Attributes attributes = manifest.getMainAttributes(); for (Object key : attributes.keySet()) { LOG.info(jarFile.getName() + ", " + key + ": " + attributes.getValue((Name)key)); } } return value; } /** * * @param jarFile * @param exceptionIfProblem true if should throw exception if problem, otherwise blank * @return the version */ public static String jarVersion(File jarFile, boolean exceptionIfProblem) { try { String version = jarVersion0(jarFile); if (isBlank(version)) { version = jarVersion1(jarFile); } if (isBlank(version)) { //hopefully this is set if (tempFilePathForJars != null) { String jarFilePath = jarFile.getAbsolutePath(); jarFilePath = replace(jarFilePath, ":", "_"); if (jarFilePath.startsWith("/") || jarFilePath.startsWith("\\")) { jarFilePath = jarFilePath.substring(1); } jarFilePath = tempFilePathForJars + jarFilePath; File bakJarFile = new File(jarFilePath); mkdirs(bakJarFile.getParentFile()); copyFile(jarFile, bakJarFile); version = jarVersion2(bakJarFile); } else { throw new RuntimeException("You need to set tempFileForJars"); } } return version; } catch (RuntimeException e) { injectInException(e, "Problem with jar: " + jarFile.getAbsolutePath()); if (exceptionIfProblem) { throw e; } System.out.println("Non-fatal issue with " + jarFile.getAbsolutePath() + ", " + e.getMessage() + ", assuming cant find version"); } return null; } /** * get the property value from version in the jar filename; e.g. jarfile-1.2.3.jar * @param jarFile * @return the version or null if cant find */ public static String jarVersion0(File jarFile) { Pattern versionPattern = Pattern.compile("^.*-((\\d+)\\.(\\d+)(\\.(\\d+))*)\\.jar*$"); String fileName = jarFile.getName(); Matcher matcher = versionPattern.matcher(fileName); if (matcher.matches()) { return matcher.group(1); } return null; } /** set this to copy jars if the jarinputstream doesnt work, must end in File.separator, set this at the beginning of the * program. */ public static String tempFilePathForJars = null; /** * * @param jarFile * @return the version */ public static String jarVersion2(File jarFile) { InputStream manifestStream = null; try { URL manifestUrl = new URL("jar:file:" + jarFile.getCanonicalPath() + "!/META-INF/MANIFEST.MF"); manifestStream = manifestUrl.openStream(); Manifest manifest = new Manifest(manifestStream); return manifest == null ? null : manifestVersion(jarFile, manifest); } catch (Exception e) { if (e instanceof RuntimeException) { throw (RuntimeException)e; } throw new RuntimeException(jarFile.getAbsolutePath() + ", " + e.getMessage(), e); } finally { closeQuietly(manifestStream); } } /** * split a string (e.g. file contents) into lines * @param string * @return the lines */ public static String[] splitLines(String string) { String newline = newlineFromFile(string); String[] lines = null; if ("\n".equals(newline)) { lines = string.split("[\\n]"); } else if ("\r".equals(newline)) { lines = string.split("[\\r]"); } else if ("\r\n".equals(newline)) { lines = string.split("[\\r\\n]"); } else { lines = string.split("[\\r\\n]+"); } return lines; } /** * replace newlines with space * @param input * @return the string */ public static String replaceNewlinesWithSpace(String input) { if (input == null) { return null; } input = GrouperInstallerUtils.replace(input, "\r\n", " "); input = GrouperInstallerUtils.replace(input, "\r", " "); input = GrouperInstallerUtils.replace(input, "\n", " "); return input; } /** * if printed cant find version, put the jar name here */ private static Set printedCantFindVersionJarName = new HashSet(); static { //we know about this one, signed, cant change //sqljdbc4.jar, Manifest-Version: 1.0 //sqljdbc4.jar, Created-By: 1.6.0_33 (Sun Microsystems Inc.) printedCantFindVersionJarName.add("sqljdbc4.jar"); } /** * @param jarFile * @param manifest * @return manifest version */ public static String manifestVersion(File jarFile, Manifest manifest) { boolean printJarVersionProblemsV1 = propertiesValueBoolean("grouperInstaller.printJarVersionIssuesV1", false, false); String[] propertyNames = new String[]{ "Implementation-Version","Version"}; Map attributeMap = manifest.getEntries(); String value = null; for (String propertyName : propertyNames) { value = manifest.getMainAttributes().getValue(propertyName); if (!isBlank(value)) { break; } } if (value == null) { OUTER: for (Attributes attributes: attributeMap.values()) { for (String propertyName : propertyNames) { value = attributes.getValue(propertyName); if (!isBlank(value)) { break OUTER; } } } } if (value == null) { if (!printedCantFindVersionJarName.contains(jarFile.getName())) { printedCantFindVersionJarName.add(jarFile.getName()); if (printJarVersionProblemsV1) { System.out.println("Error: cant find version for jar: " + jarFile.getName()); } if (printJarVersionProblemsV1) { for (Attributes attributes: attributeMap.values()) { for (Object key : attributes.keySet()) { System.out.println(jarFile.getName() + ", " + key + ": " + attributes.getValue((Name)key)); } } Attributes attributes = manifest.getMainAttributes(); for (Object key : attributes.keySet()) { System.out.println(jarFile.getName() + ", main " + key + ": " + attributes.getValue((Name)key)); } } } } return value; } /** * based on file contents see what the newline type is * @param fileContents * @return the newline */ public static String newlineFromFile(String fileContents) { String newline = "\n"; if (fileContents.contains("\r\n")) { newline = "\r\n"; } if (fileContents.contains("\n\r")) { newline = "\n\r"; } if (fileContents.contains("\r")) { newline = "\r"; } return newline; } /** * list files recursively from parent, dont include * @param parent * @param fileName * @return set of files wont return null */ public static List fileListRecursive(File parent, String fileName) { List allFiles = GrouperInstallerUtils.fileListRecursive(parent); List result = new ArrayList(); for (File file : allFiles) { if (equals(file.getName(), fileName)) { result.add(file); } } return result; } /** * list files recursively from parent, dont include * @param parent * @return set of files wont return null */ public static List fileListRecursive(File parent) { List results = new ArrayList(); fileListRecursiveHelper(parent, results); return results; } /** * helper to add child files to a parent ( * @param parent * @param fileList */ private static void fileListRecursiveHelper(File parent, List fileList) { if (parent == null || !parent.exists() || !parent.isDirectory()) { return; } List subFiles = nonNull(toList(parent.listFiles())); for (File subFile : subFiles) { if (subFile.isFile()) { fileList.add(subFile); } if (subFile.isDirectory()) { fileListRecursiveHelper(subFile, fileList); } } } /** * @param path * @return the new path */ public static String fileMassagePathsNoLeadingOrTrailing(String path) { path = path.replace(File.separatorChar == '/' ? '\\' : '/', File.separatorChar); if (path.startsWith(File.separator)) { path = path.substring(1); } if (path.endsWith(File.separator)) { path = path.substring(0, path.length()-1); } return path; } /** * Compares the contents of two files to determine if they are equal or not. *

* This method checks to see if the two files are different lengths * or if they point to the same file, before resorting to byte-by-byte * comparison of the contents. *

* Code origin: Avalon * * @param file1 the first file * @param file2 the second file * @return true if the content of the files are equal or they both don't * exist, false otherwise */ public static boolean contentEquals(File file1, File file2) { try { boolean file1Exists = file1 != null && file1.exists(); boolean file2Exists = file2 != null && file2.exists(); if (file1Exists != file2Exists) { return false; } if (!file1Exists) { // two not existing files are equal return true; } if (file1.isDirectory() || file2.isDirectory()) { // don't want to compare directory contents throw new IOException("Can't compare directories, only files: " + file1.getAbsolutePath() + ", " + file2.getAbsolutePath()); } if (file1.length() != file2.length()) { // lengths differ, cannot be equal return false; } if (file1.getCanonicalFile().equals(file2.getCanonicalFile())) { // same file return true; } InputStream input1 = null; InputStream input2 = null; try { input1 = new FileInputStream(file1); input2 = new FileInputStream(file2); return contentEquals(input1, input2); } finally { closeQuietly(input1); closeQuietly(input2); } } catch (IOException ioe) { throw new RuntimeException(ioe); } } /** * Compare the contents of two Streams to determine if they are equal or * not. *

* This method buffers the input internally using * BufferedInputStream if they are not already buffered. * * @param input1 the first stream * @param input2 the second stream * @return true if the content of the streams are equal or they both don't * exist, false otherwise * @throws NullPointerException if either input is null * @throws IOException if an I/O error occurs */ public static boolean contentEquals(InputStream input1, InputStream input2) throws IOException { if (!(input1 instanceof BufferedInputStream)) { input1 = new BufferedInputStream(input1); } if (!(input2 instanceof BufferedInputStream)) { input2 = new BufferedInputStream(input2); } int ch = input1.read(); while (-1 != ch) { int ch2 = input2.read(); if (ch != ch2) { return false; } ch = input1.read(); } int ch2 = input2.read(); return (ch2 == -1); } /** * Compare the contents of two Readers to determine if they are equal or * not. *

* This method buffers the input internally using * BufferedReader if they are not already buffered. * * @param input1 the first reader * @param input2 the second reader * @return true if the content of the readers are equal or they both don't * exist, false otherwise * @throws NullPointerException if either input is null * @throws IOException if an I/O error occurs * @since Commons IO 1.1 */ public static boolean contentEquals(Reader input1, Reader input2) throws IOException { if (!(input1 instanceof BufferedReader)) { input1 = new BufferedReader(input1); } if (!(input2 instanceof BufferedReader)) { input2 = new BufferedReader(input2); } int ch = input1.read(); while (-1 != ch) { int ch2 = input2.read(); if (ch != ch2) { return false; } ch = input1.read(); } int ch2 = input2.read(); return (ch2 == -1); } /** * get the relative paths of descendant files * @param parentDir * @return the relative paths of files underneath, dont start with slash */ public static Set fileDescendantRelativePaths(File parentDir) { Set result = new LinkedHashSet(); List descendants = fileListRecursive(parentDir); for (File file : GrouperInstallerUtils.nonNull(descendants)) { String descendantPath = file.getAbsolutePath(); String parentPath = parentDir.getAbsolutePath(); if (!descendantPath.startsWith(parentPath)) { throw new RuntimeException("Why doesnt descendantPath '" + descendantPath + "' start with parent path '" + parentPath + "'?"); } descendantPath = descendantPath.substring(parentPath.length()); if (descendantPath.startsWith("/") || descendantPath.startsWith("\\")) { descendantPath = descendantPath.substring(1); } result.add(descendantPath); } return result; } /** * get the relative path of descendant file * @param parentDir * @param file * @return the relative path of file underneath, dont start with slash */ public static String fileRelativePath(File parentDir, File file) { String descendantPath = file.getAbsolutePath(); String parentPath = parentDir.getAbsolutePath(); if (!descendantPath.startsWith(parentPath)) { throw new RuntimeException("Why doesnt descendantPath '" + descendantPath + "' start with parent path '" + parentPath + "'?"); } descendantPath = descendantPath.substring(parentPath.length()); if (descendantPath.startsWith("/") || descendantPath.startsWith("\\")) { descendantPath = descendantPath.substring(1); } return descendantPath; } /** * * @param bigPath * @param prefixPath * @return true if starts with */ public static boolean filePathStartsWith(String bigPath, String prefixPath) { bigPath = replace(bigPath, "\\\\", "\\"); bigPath = replace(bigPath, "\\", "/"); prefixPath = replace(prefixPath, "\\\\", "\\"); prefixPath = replace(prefixPath, "\\", "/"); return bigPath.startsWith(prefixPath); } /** * get the property value from version in the manifest of a jar * @param jarFile * @return the version */ public static String jarVersion1(File jarFile) { FileInputStream fileInputStream = null; try { fileInputStream = new FileInputStream(jarFile); JarInputStream jarInputStream = new JarInputStream(fileInputStream); Manifest manifest = jarInputStream.getManifest(); return manifest == null ? null : manifestVersion(jarFile, manifest); } catch (Exception e) { if (e instanceof RuntimeException) { throw (RuntimeException)e; } throw new RuntimeException(e.getMessage(), e); } finally { closeQuietly(fileInputStream); } } /** * Given two string of a.b.c..., return 0 if equal, -1 if first string is lower, and 1 if first string is higher (similar to Integer.compareTo()) * @param s1String first version string * @param s2String second version string * @return int value -1 if first is lower, 1 if first is higher, and 0 if equal */ public static int compareVersions(String s1String, String s2String) { if (s1String.equals(s2String)) { return 0; } String[] s1Tokens = s1String.split("\\."); String[] s2Tokens = s2String.split("\\."); int i = 0; while( i < s1Tokens.length && i < s2Tokens.length && s1Tokens[i].equals(s2Tokens[i]) ) { i++; } while (i < s1Tokens.length || i < s2Tokens.length) { Integer s1Value = 0; Integer s2Value = 0; if (i < s1Tokens.length) { String s1Token = s1Tokens[i]; try { s1Value = Integer.parseInt(s1Token); } catch (NumberFormatException e) { System.out.println("WARNING: Could not parse number from '" + s1Token + "' in version string '" + s1String + "'"); return -1; } } if (i < s2Tokens.length) { String s2Token = s2Tokens[i]; try { s2Value = Integer.parseInt(s2Token); } catch (NumberFormatException e) { System.out.println("WARNING: Could not parse number from '" + s2Token + "' in version string '" + s2String + "' -- assuming -1"); return 1; } } int cmp = s1Value.compareTo(s2Value); if (cmp != 0) { return cmp; } ++i; } // exhausted both strings with neither being greater; assume equal at this point return 0; } }