/* * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. * * Copyright (c) 2009-2014 Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2009-2014 Jason Mehrens. All rights reserved. * * The contents of this file are subject to the terms of either the GNU * General Public License Version 2 only ("GPL") or the Common Development * and Distribution License("CDDL") (collectively, the "License"). You * may not use this file except in compliance with the License. You can * obtain a copy of the License at * https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html * or packager/legal/LICENSE.txt. See the License for the specific * language governing permissions and limitations under the License. * * When distributing the software, include this License Header Notice in each * file and include the License file at packager/legal/LICENSE.txt. * * GPL Classpath Exception: * Oracle designates this particular file as subject to the "Classpath" * exception as provided by Oracle in the GPL Version 2 section of the License * file that accompanied this code. * * Modifications: * If applicable, add the following below the License Header, with the fields * enclosed by brackets [] replaced by your own identifying information: * "Portions Copyright [year] [name of copyright owner]" * * Contributor(s): * If you wish your version of this file to be governed by only the CDDL or * only the GPL Version 2, indicate your decision by adding "[Contributor] * elects to include this software in this distribution under the [CDDL or GPL * Version 2] license." If you don't indicate a single choice of license, a * recipient has the option to distribute your version of this file under * either the CDDL, the GPL Version 2 or to extend the choice of license to * its licensees as provided above. However, if you add GPL Version 2 code * and therefore, elected the GPL Version 2 license, then the option applies * only if the new code is made subject to such option by the copyright * holder. */ package com.sun.mail.util.logging; import java.io.ObjectStreamException; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.lang.reflect.Modifier; import java.security.AccessController; import java.security.PrivilegedAction; import java.util.*; import java.util.logging.*; import java.util.logging.Formatter; import javax.mail.Authenticator; /** * An adapter class to allow the Mail API to access the LogManager properties. * The LogManager properties are treated as the root of all properties. First, * the parent properties are searched. If no value is found, then, the * LogManager is searched with prefix value. If not found, then, just the key * itself is searched in the LogManager. If a value is found in the LogManager * it is then copied to this properties object with no key prefix. If no value * is found in the LogManager or the parent properties, then this properties * object is searched only by passing the key value. * * <p> * This class also emulates the LogManager functions for creating new objects * from string class names. This is to support initial setup of objects such as * log filters, formatters, error managers, etc. * * <p> * This class should never be exposed outside of this package. Keep this class * package private (default access). * * @author Jason Mehrens * @since JavaMail 1.4.3 */ final class LogManagerProperties extends Properties { /** * Generated serial id. */ private static final long serialVersionUID = -2239983349056806252L; /** * Caches the LogManager so we only read the config once. */ private static final LogManager LOG_MANAGER = LogManager.getLogManager(); /** * Caches the read only reflection class names string array. Declared * volatile for safe publishing only. The VO_VOLATILE_REFERENCE_TO_ARRAY * warning is a false positive. */ @SuppressWarnings("VolatileArrayField") private static volatile String[] REFLECT_NAMES; /** * Gets the LogManger for the running JVM. * * @return the LogManager. * @since JavaMail 1.4.5 */ static LogManager getLogManager() { return LOG_MANAGER; } /** * Converts a locale to a language tag. * * @param locale the locale to convert. * @return the language tag. * @throws NullPointerException if the given locale is null. * @since JavaMail 1.4.5 */ static String toLanguageTag(final Locale locale) { final String l = locale.getLanguage(); final String c = locale.getCountry(); final String v = locale.getVariant(); final char[] b = new char[l.length() + c.length() + v.length() + 2]; int count = l.length(); l.getChars(0, count, b, 0); if (c.length() != 0 || (l.length() != 0 && v.length() != 0)) { b[count] = '-'; ++count; //be nice to the client compiler. c.getChars(0, c.length(), b, count); count += c.length(); } if (v.length() != 0 && (l.length() != 0 || c.length() != 0)) { b[count] = '-'; ++count; //be nice to the client compiler. v.getChars(0, v.length(), b, count); count += v.length(); } return String.valueOf(b, 0, count); } /** * Creates a new filter from the given class name. * * @param name the fully qualified class name. * @return a new filter. * @throws ClassCastException if class name does not match the type. * @throws ClassNotFoundException if the class name was not found. * @throws IllegalAccessException if the constructor is inaccessible. * @throws InstantiationException if the given class name is abstract. * @throws InvocationTargetException if the constructor throws an exception. * @throws LinkageError if the linkage fails. * @throws ExceptionInInitializerError if the static initializer fails. * @throws Exception to match the error method of the ErrorManager. * @throws NoSuchMethodException if the class name does not have a no * argument constructor. * @since JavaMail 1.4.5 */ static Filter newFilter(String name) throws Exception { return newObjectFrom(name, Filter.class); } /** * Creates a new formatter from the given class name. * * @param name the fully qualified class name. * @return a new formatter. * @throws ClassCastException if class name does not match the type. * @throws ClassNotFoundException if the class name was not found. * @throws IllegalAccessException if the constructor is inaccessible. * @throws InstantiationException if the given class name is abstract. * @throws InvocationTargetException if the constructor throws an exception. * @throws LinkageError if the linkage fails. * @throws ExceptionInInitializerError if the static initializer fails. * @throws Exception to match the error method of the ErrorManager. * @throws NoSuchMethodException if the class name does not have a no * argument constructor. * @since JavaMail 1.4.5 */ static Formatter newFormatter(String name) throws Exception { return newObjectFrom(name, Formatter.class); } /** * Creates a new log record comparator from the given class name. * * @param name the fully qualified class name. * @return a new comparator. * @throws ClassCastException if class name does not match the type. * @throws ClassNotFoundException if the class name was not found. * @throws IllegalAccessException if the constructor is inaccessible. * @throws InstantiationException if the given class name is abstract. * @throws InvocationTargetException if the constructor throws an exception. * @throws LinkageError if the linkage fails. * @throws ExceptionInInitializerError if the static initializer fails. * @throws Exception to match the error method of the ErrorManager. * @throws NoSuchMethodException if the class name does not have a no * argument constructor. * @since JavaMail 1.4.5 * @see java.util.logging.LogRecord */ @SuppressWarnings("unchecked") static Comparator<? super LogRecord> newComparator(String name) throws Exception { return newObjectFrom(name, Comparator.class); } /** * Returns a comparator that imposes the reverse ordering of the specified * {@link Comparator}. If the given comparator declares a public * reverseOrder method that method is called first and the return value is * used. If that method is not declared or the caller does not have access * then a comparator wrapping the given comparator is returned. * * @param <T> the element type to be compared * @param c a comparator whose ordering is to be reversed by the returned * comparator * @return A comparator that imposes the reverse ordering of the specified * comparator. * @throws NullPointerException if the given comparator is null. * @since JavaMail 1.5.0 */ @SuppressWarnings("unchecked") static <T> Comparator<T> reverseOrder(final Comparator<T> c) { if (c == null) { throw new NullPointerException(); } Comparator<T> reverse = null; //Comparator in Java 1.8 has 'reversed' as a default method. //This code calls that method first to allow custom //code to define what reverse order means. try { //assert Modifier.isPublic(c.getClass().getModifiers()) : // Modifier.toString(c.getClass().getModifiers()); final Method m = c.getClass().getMethod("reversed"); if (!Modifier.isStatic(m.getModifiers()) && Comparator.class.isAssignableFrom(m.getReturnType())) { try { reverse = (Comparator<T>) m.invoke(c); } catch (final ExceptionInInitializerError eiie) { throw wrapOrThrow(eiie); } } } catch (final NoSuchMethodException ignore) { } catch (final IllegalAccessException ignore) { } catch (final RuntimeException ignore) { } catch (final InvocationTargetException ite) { paramOrError(ite); //Ignore invocation bugs. } if (reverse == null) { reverse = Collections.reverseOrder(c); } return reverse; } /** * Creates a new error manager from the given class name. * * @param name the fully qualified class name. * @return a new error manager. * @throws ClassCastException if class name does not match the type. * @throws ClassNotFoundException if the class name was not found. * @throws IllegalAccessException if the constructor is inaccessible. * @throws InstantiationException if the given class name is abstract. * @throws InvocationTargetException if the constructor throws an exception. * @throws LinkageError if the linkage fails. * @throws ExceptionInInitializerError if the static initializer fails. * @throws Exception to match the error method of the ErrorManager. * @throws NoSuchMethodException if the class name does not have a no * argument constructor. * @since JavaMail 1.4.5 */ static ErrorManager newErrorManager(String name) throws Exception { return newObjectFrom(name, ErrorManager.class); } /** * Creates a new authenticator from the given class name. * * @param name the fully qualified class name. * @return a new authenticator. * @throws ClassCastException if class name does not match the type. * @throws ClassNotFoundException if the class name was not found. * @throws IllegalAccessException if the constructor is inaccessible. * @throws InstantiationException if the given class name is abstract. * @throws InvocationTargetException if the constructor throws an exception. * @throws LinkageError if the linkage fails. * @throws ExceptionInInitializerError if the static initializer fails. * @throws Exception to match the error method of the ErrorManager. * @throws NoSuchMethodException if the class name does not have a no * argument constructor. * @since JavaMail 1.4.5 */ static Authenticator newAuthenticator(String name) throws Exception { return newObjectFrom(name, Authenticator.class); } /** * Determines if the given class name identifies a utility class. * * @param name the fully qualified class name. * @return true if the given class name * @throws ClassNotFoundException if the class name was not found. * @throws IllegalAccessException if the constructor is inaccessible. * @throws LinkageError if the linkage fails. * @throws ExceptionInInitializerError if the static initializer fails. * @throws Exception to match the error method of the ErrorManager. * @throws SecurityException if unable to inspect properties of class. * @since JavaMail 1.5.2 */ static boolean isStaticUtilityClass(String name) throws Exception { final Class<?> c = findClass(name); final Class<?> obj = Object.class; Method[] methods = c.getMethods(); boolean util; if (c != obj && methods.length != 0) { util = true; for (Method m : methods) { if (m.getDeclaringClass() != obj && !Modifier.isStatic(m.getModifiers())) { util = false; break; } } } else { util = false; } return util; } /** * Determines if the given class name is a reflection class name responsible * for invoking methods and or constructors. * * @param name the fully qualified class name. * @return true if the given class name * @throws ClassNotFoundException if the class name was not found. * @throws IllegalAccessException if the constructor is inaccessible. * @throws LinkageError if the linkage fails. * @throws ExceptionInInitializerError if the static initializer fails. * @throws Exception to match the error method of the ErrorManager. * @throws SecurityException if unable to inspect properties of class. * @since JavaMail 1.5.2 */ static boolean isReflectionClass(String name) throws Exception { String[] names; if ((names = REFLECT_NAMES) == null) { //Benign data race. REFLECT_NAMES = names = reflectionClassNames(); } for (String rf : names) { //The set of names is small. if (name.equals(rf)) { return true; } } findClass(name); //Fail late instead of normal return. return false; } /** * Determines all of the reflection class names used to invoke methods. * * This method performs indirect and direct calls on a throwable to capture * the standard class names and the implementation class names. * * @return a string array containing the fully qualified class names. * @throws Exception if there is a problem. */ private static String[] reflectionClassNames() throws Exception { final Class<?> thisClass = LogManagerProperties.class; assert Modifier.isFinal(thisClass.getModifiers()) : thisClass; try { final HashSet<String> traces = new HashSet<String>(); Throwable t = Throwable.class.getConstructor().newInstance(); for (StackTraceElement ste : t.getStackTrace()) { if (!thisClass.getName().equals(ste.getClassName())) { traces.add(ste.getClassName()); } else { break; } } Throwable.class.getMethod("fillInStackTrace").invoke(t); for (StackTraceElement ste : t.getStackTrace()) { if (!thisClass.getName().equals(ste.getClassName())) { traces.add(ste.getClassName()); } else { break; } } return traces.toArray(new String[traces.size()]); } catch (final InvocationTargetException ITE) { throw paramOrError(ITE); } } /** * Creates a new object from the given class name. * * @param <T> The generic class type. * @param name the fully qualified class name. * @param type the assignable type for the given name. * @return a new object assignable to the given type. * @throws ClassCastException if class name does not match the type. * @throws ClassNotFoundException if the class name was not found. * @throws IllegalAccessException if the constructor is inaccessible. * @throws InstantiationException if the given class name is abstract. * @throws InvocationTargetException if the constructor throws an exception. * @throws LinkageError if the linkage fails. * @throws ExceptionInInitializerError if the static initializer fails. * @throws Exception to match the error method of the ErrorManager. * @throws NoSuchMethodException if the class name does not have a no * argument constructor. * @since JavaMail 1.4.5 */ private static <T> T newObjectFrom(String name, Class<T> type) throws Exception { try { final Class<?> clazz = LogManagerProperties.findClass(name); //This check avoids additional side effects when the name parameter //is a literal name and not a class name. if (type.isAssignableFrom(clazz)) { try { return type.cast(clazz.getConstructor().newInstance()); } catch (final InvocationTargetException ITE) { throw paramOrError(ITE); } } else { throw new ClassCastException(clazz.getName() + " cannot be cast to " + type.getName()); } } catch (final NoClassDefFoundError NCDFE) { //No class def found can occur on filesystems that are //case insensitive (BUG ID 6196068). In some cases, we allow class //names or literal names, this code guards against the case where a //literal name happens to match a class name in a different case. //This is also a nice way to adap this error for the error manager. throw new ClassNotFoundException(NCDFE.toString(), NCDFE); } catch (final ExceptionInInitializerError EIIE) { throw wrapOrThrow(EIIE); } } /** * Returns the given exception or throws the escaping cause. * * @param ite any invocation target. * @return the exception. * @throws VirtualMachineError if present as cause. * @throws ThreadDeath if present as cause. * @since JavaMail 1.4.5 */ private static Exception paramOrError(InvocationTargetException ite) { final Throwable cause = ite.getCause(); if (cause != null) { if (cause instanceof VirtualMachineError || cause instanceof ThreadDeath) { throw (Error) cause; } } return ite; } /** * Throws the given error if the cause is an error otherwise the given error * is wrapped. * * @param eiie the error. * @return an InvocationTargetException. * @since JavaMail 1.5.0 */ private static InvocationTargetException wrapOrThrow( ExceptionInInitializerError eiie) { //This linkage error will escape the constructor new instance call. //If the cause is an error, rethrow to skip any error manager. if (eiie.getCause() instanceof Error) { throw eiie; } else { //Considered a bug in the code, wrap the error so it can be //reported to the error manager. return new InvocationTargetException(eiie); } } /** * This code is modified from the LogManager, which explictly states * searching the system class loader first, then the context class loader. * There is resistance (compatibility) to change this behavior to simply * searching the context class loader. * * @param name full class name * @return the class. * @throws LinkageError if the linkage fails. * @throws ClassNotFoundException if the class name was not found. * @throws ExceptionInInitializerError if static initializer fails. */ private static Class<?> findClass(String name) throws ClassNotFoundException { ClassLoader[] loaders = getClassLoaders(); assert loaders.length == 2 : loaders.length; Class<?> clazz; if (loaders[0] != null) { try { clazz = Class.forName(name, false, loaders[0]); } catch (ClassNotFoundException tryContext) { clazz = tryLoad(name, loaders[1]); } } else { clazz = tryLoad(name, loaders[1]); } return clazz; } /** * Loads a class using the given loader or the class loader of this class. * * @param name the class name. * @param l any class loader or null. * @return the raw class. * @throws ClassNotFoundException if not found. */ private static Class<?> tryLoad(String name, ClassLoader l) throws ClassNotFoundException { if (l != null) { return Class.forName(name, false, l); } else { return Class.forName(name); } } /** * Gets the class loaders using elevated privileges. * * @return any array of class loaders. Indexes may be null. */ private static ClassLoader[] getClassLoaders() { return AccessController.doPrivileged(new PrivilegedAction<ClassLoader[]>() { public ClassLoader[] run() { final ClassLoader[] loaders = new ClassLoader[2]; try { loaders[0] = ClassLoader.getSystemClassLoader(); } catch (SecurityException ignore) { loaders[0] = null; } try { loaders[1] = Thread.currentThread().getContextClassLoader(); } catch (SecurityException ignore) { loaders[1] = null; } return loaders; } }); } /** * The namespace prefix to search LogManager and defaults. */ private final String prefix; /** * Creates a log manager properties object. * * @param parent the parent properties. * @param prefix the namespace prefix. * @throws NullPointerException if <tt>prefix</tt> or <tt>parent</tt> is * <tt>null</tt>. */ LogManagerProperties(final Properties parent, final String prefix) { super(parent); parent.isEmpty(); //null check, happens-before if (prefix == null) { throw new NullPointerException(); } this.prefix = prefix; //'defaults' is not decalared final. super.isEmpty(); //happens-before. } /** * Returns a properties object that contains a snapshot of the current * state. This method violates the clone contract so that no instances of * LogManagerProperties is exported for public use. * * @return the snapshot. * @since JavaMail 1.4.4 */ @Override @SuppressWarnings("CloneDoesntCallSuperClone") public synchronized Object clone() { return exportCopy(defaults); } /** * Searches defaults, then searches the log manager by the prefix property, * and then by the key itself. * * @param key a non null key. * @return the value for that key. */ @Override public synchronized String getProperty(final String key) { String value = defaults.getProperty(key); if (value == null) { final LogManager manager = getLogManager(); if (key.length() > 0) { value = manager.getProperty(prefix + '.' + key); } if (value == null) { value = manager.getProperty(key); } /** * Copy the log manager properties as we read them. If a value is no * longer present in the LogManager read it from here. The reason * this works is because LogManager.reset() closes all attached * handlers therefore, stale values only exist in closed handlers. */ if (value != null) { super.put(key, value); } else { Object v = super.get(key); //defaults are not used. value = v instanceof String ? (String) v : null; } } return value; } /** * Calls getProperty directly. If getProperty returns null the default value * is returned. * * @param key a key to search for. * @param def the default value to use if not found. * @return the value for the key. * @since JavaMail 1.4.4 */ @Override public String getProperty(final String key, final String def) { final String value = this.getProperty(key); return value == null ? def : value; } /** * Required to work with PropUtil. Calls getProperty directly if the given * key is a string. Otherwise, performs a normal get operation. * * @param key any key. * @return the value for the key or null. * @since JavaMail 1.4.5 */ @Override public Object get(final Object key) { if (key instanceof String) { return this.getProperty((String) key); } else { return super.get(key); } } /** * Required to work with PropUtil. An updated copy of the key is fetched * from the log manager if the key doesn't exist in this properties. * * @param key any key. * @return the value for the key or the default value for the key. * @since JavaMail 1.4.5 */ @Override public synchronized Object put(final Object key, final Object value) { final Object def = preWrite(key); final Object man = super.put(key, value); return man == null ? def : man; } /** * Calls the put method directly. * * @param key any key. * @return the value for the key or the default value for the key. * @since JavaMail 1.4.5 */ @Override public Object setProperty(String key, String value) { return this.put(key, value); } /** * Required to work with PropUtil. An updated copy of the key is fetched * from the log manager prior to returning. * * @param key any key. * @return the value for the key or null. * @since JavaMail 1.4.5 */ @Override public boolean containsKey(final Object key) { if (key instanceof String) { return this.getProperty((String) key) != null; } else { return super.containsKey(key); } } /** * Required to work with PropUtil. An updated copy of the key is fetched * from the log manager if the key doesn't exist in this properties. * * @param key any key. * @return the value for the key or the default value for the key. * @since JavaMail 1.4.5 */ @Override public synchronized Object remove(final Object key) { final Object def = preWrite(key); final Object man = super.remove(key); return man == null ? def : man; } /** * It is assumed that this method will never be called. No way to get the * property names from LogManager. * * @return the property names */ @Override public Enumeration<?> propertyNames() { assert false; return super.propertyNames(); } /** * It is assumed that this method will never be called. The prefix value is * not used for the equals method. * * @param o any object or null. * @return true if equal, otherwise false. */ @Override public boolean equals(final Object o) { if (o == null) { return false; } if (o == this) { return true; } if (o instanceof Properties == false) { return false; } assert false : prefix; return super.equals(o); } /** * It is assumed that this method will never be called. See equals. * * @return the hash code. */ @Override public int hashCode() { assert false : prefix.hashCode(); return super.hashCode(); } /** * Called before a write operation of a key. Caches a key read from the log * manager in this properties object. The key is only cached if it is an * instance of a String and this properties doesn't contain a copy of the * key. * * @param key the key to search. * @return the default value for the key. */ private Object preWrite(final Object key) { assert Thread.holdsLock(this); Object value; if (key instanceof String && !super.containsKey(key)) { value = this.getProperty((String) key); //fetch and cache. } else { value = null; } return value; } /** * Creates a public snapshot of this properties object using the given * parent properties. * * @param parent the defaults to use with the snapshot. * @return the safe snapshot. */ private Properties exportCopy(final Properties parent) { Thread.holdsLock(this); final Properties child = new Properties(parent); child.putAll(this); return child; } /** * It is assumed that this method will never be called. We return a safe * copy for export to avoid locking this properties object or the defaults * during write. * * @return the parent properties. * @throws ObjectStreamException if there is a problem. */ private synchronized Object writeReplace() throws ObjectStreamException { assert false; return exportCopy((Properties) defaults.clone()); } }