001/** 002 * Logback: the reliable, generic, fast and flexible logging framework. 003 * Copyright (C) 1999-2015, QOS.ch. All rights reserved. 004 * 005 * This program and the accompanying materials are dual-licensed under 006 * either the terms of the Eclipse Public License v1.0 as published by 007 * the Eclipse Foundation 008 * 009 * or (per the licensee's choosing) 010 * 011 * under the terms of the GNU Lesser General Public License version 2.1 012 * as published by the Free Software Foundation. 013 */ 014package ch.qos.logback.core.util; 015 016import java.lang.reflect.Constructor; 017import java.util.Properties; 018 019import ch.qos.logback.core.Context; 020import ch.qos.logback.core.spi.ContextAware; 021import ch.qos.logback.core.spi.PropertyContainer; 022import ch.qos.logback.core.spi.ScanException; 023import ch.qos.logback.core.subst.NodeToStringTransformer; 024 025/** 026 * @author Ceki Gulcu 027 */ 028public class OptionHelper { 029 030 public static Object instantiateByClassName(String className, Class<?> superClass, Context context) 031 throws IncompatibleClassException, DynamicClassLoadingException { 032 ClassLoader classLoader = Loader.getClassLoaderOfObject(context); 033 return instantiateByClassName(className, superClass, classLoader); 034 } 035 036 public static Object instantiateByClassNameAndParameter(String className, Class<?> superClass, Context context, 037 Class<?> type, Object param) throws IncompatibleClassException, DynamicClassLoadingException { 038 ClassLoader classLoader = Loader.getClassLoaderOfObject(context); 039 return instantiateByClassNameAndParameter(className, superClass, classLoader, type, param); 040 } 041 042 public static Object instantiateByClassName(String className, Class<?> superClass, ClassLoader classLoader) 043 throws IncompatibleClassException, DynamicClassLoadingException { 044 return instantiateByClassNameAndParameter(className, superClass, classLoader, null, null); 045 } 046 047 public static Object instantiateByClassNameAndParameter(String className, Class<?> superClass, 048 ClassLoader classLoader, Class<?> type, Object parameter) 049 throws IncompatibleClassException, DynamicClassLoadingException { 050 051 if (className == null) { 052 throw new NullPointerException(); 053 } 054 try { 055 Class<?> classObj = null; 056 classObj = classLoader.loadClass(className); 057 if (!superClass.isAssignableFrom(classObj)) { 058 throw new IncompatibleClassException(superClass, classObj); 059 } 060 if (type == null) { 061 return classObj.getConstructor().newInstance(); 062 } else { 063 Constructor<?> constructor = classObj.getConstructor(type); 064 return constructor.newInstance(parameter); 065 } 066 } catch (IncompatibleClassException ice) { 067 throw ice; 068 } catch (Throwable t) { 069 throw new DynamicClassLoadingException("Failed to instantiate type " + className, t); 070 } 071 } 072 073 /** 074 * Find the value corresponding to <code>key</code> in <code>props</code>. Then 075 * perform variable substitution on the found value. 076 */ 077 // public static String findAndSubst(String key, Properties props) { 078 // String value = props.getProperty(key); 079 // 080 // if (value == null) { 081 // return null; 082 // } 083 // 084 // try { 085 // return substVars(value, props); 086 // } catch (IllegalArgumentException e) { 087 // return value; 088 // } 089 // } 090 final static String DELIM_START = "${"; 091 final static char DELIM_STOP = '}'; 092 final static String DELIM_DEFAULT = ":-"; 093 094 final static int DELIM_START_LEN = 2; 095 final static int DELIM_STOP_LEN = 1; 096 final static int DELIM_DEFAULT_LEN = 2; 097 098 final static String _IS_UNDEFINED = "_IS_UNDEFINED"; 099 100 /** 101 * @see #substVars(String, PropertyContainer, PropertyContainer) 102 */ 103 public static String substVars(String val, PropertyContainer pc1) throws ScanException { 104 return substVars(val, pc1, null); 105 } 106 107 /** 108 * See http://logback.qos.ch/manual/configuration.html#variableSubstitution 109 */ 110 public static String substVars(String input, PropertyContainer pc0, PropertyContainer pc1) throws ScanException { 111 // may throw IllegalArgumentException or ScanException 112 return NodeToStringTransformer.substituteVariable(input, pc0, pc1); 113 114 } 115 116 public static String propertyLookup(String key, PropertyContainer pc1, PropertyContainer pc2) { 117 String value = null; 118 // first try the props passed as parameter 119 value = pc1.getProperty(key); 120 121 // then try the pc2 122 if (value == null && pc2 != null) { 123 value = pc2.getProperty(key); 124 } 125 // then try in System properties 126 if (value == null) { 127 value = getSystemProperty(key, null); 128 } 129 if (value == null) { 130 value = getEnv(key); 131 } 132 return value; 133 } 134 135 /** 136 * Very similar to <code>System.getProperty</code> except that the 137 * {@link SecurityException} is absorbed. 138 * 139 * @param key The key to search for. 140 * @param def The default value to return. 141 * @return the string value of the system property, or the default value if 142 * there is no property with that key. 143 */ 144 public static String getSystemProperty(String key, String def) { 145 try { 146 return System.getProperty(key, def); 147 } catch (SecurityException e) { 148 return def; 149 } 150 } 151 152 /** 153 * Lookup a key from the environment. 154 * 155 * @param key 156 * @return value corresponding to key from the OS environment 157 */ 158 public static String getEnv(String key) { 159 try { 160 return System.getenv(key); 161 } catch (SecurityException e) { 162 return null; 163 } 164 } 165 166 /** 167 * Very similar to <code>System.getProperty</code> except that the 168 * {@link SecurityException} is absorbed. 169 * 170 * @param key The key to search for. 171 * @return the string value of the system property. 172 */ 173 public static String getSystemProperty(String key) { 174 try { 175 return System.getProperty(key); 176 } catch (SecurityException e) { 177 return null; 178 } 179 } 180 181 public static void setSystemProperties(ContextAware contextAware, Properties props) { 182 for (Object o : props.keySet()) { 183 String key = (String) o; 184 String value = props.getProperty(key); 185 setSystemProperty(contextAware, key, value); 186 } 187 } 188 189 public static void setSystemProperty(ContextAware contextAware, String key, String value) { 190 try { 191 System.setProperty(key, value); 192 } catch (SecurityException e) { 193 contextAware.addError("Failed to set system property [" + key + "]", e); 194 } 195 } 196 197 /** 198 * Very similar to {@link System#getProperties()} except that the 199 * {@link SecurityException} is absorbed. 200 * 201 * @return the system properties 202 */ 203 public static Properties getSystemProperties() { 204 try { 205 return System.getProperties(); 206 } catch (SecurityException e) { 207 return new Properties(); 208 } 209 } 210 211 /** 212 * Return a String[] of size two. The first item containing the key part and the 213 * second item containing a default value specified by the user. The second item 214 * will be null if no default value is specified. 215 * 216 * @param key 217 * @return 218 */ 219 static public String[] extractDefaultReplacement(String key) { 220 String[] result = new String[2]; 221 if (key == null) 222 return result; 223 224 result[0] = key; 225 int d = key.indexOf(DELIM_DEFAULT); 226 if (d != -1) { 227 result[0] = key.substring(0, d); 228 result[1] = key.substring(d + DELIM_DEFAULT_LEN); 229 } 230 return result; 231 } 232 233 /** 234 * If <code>value</code> is "true", then <code>true</code> is returned. If 235 * <code>value</code> is "false", then <code>true</code> is returned. Otherwise, 236 * <code>default</code> is returned. 237 * <p> 238 * Case of value is unimportant. 239 */ 240 public static boolean toBoolean(String value, boolean dEfault) { 241 if (value == null) { 242 return dEfault; 243 } 244 245 String trimmedVal = value.trim(); 246 247 if ("true".equalsIgnoreCase(trimmedVal)) { 248 return true; 249 } 250 251 if ("false".equalsIgnoreCase(trimmedVal)) { 252 return false; 253 } 254 255 return dEfault; 256 } 257 258 /** 259 * @deprecated 260 * @since 1.3 261 */ 262 public static boolean isEmpty(String str) { 263 return isNullOrEmpty(str); 264 } 265 266 /** 267 * Returns true if input str is null or empty. 268 * 269 * @param str 270 * @return 271 */ 272 public static boolean isNullOrEmpty(String str) { 273 return ((str == null) || str.trim().length() == 0); 274 } 275 276 final public static boolean isNullOrEmpty(Object[] array) { 277 if(array == null || array.length == 0) 278 return true; 279 else 280 return false; 281 } 282 283 final public static boolean isNotEmtpy(Object[] array) { 284 return !isNullOrEmpty(array); 285 } 286}