com.helger.commons.xml.serialize.read.XMLReaderFactoryCommons Maven / Gradle / Ivy
/**
* Copyright (C) 2014-2015 Philip Helger (www.helger.com)
* philip[at]helger[dot]com
*
* 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 com.helger.commons.xml.serialize.read;
import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.security.AccessController;
import java.security.PrivilegedAction;
import javax.annotation.Nonnull;
import org.xml.sax.SAXException;
import org.xml.sax.XMLReader;
import com.helger.commons.io.stream.StreamHelper;
import com.helger.commons.lang.priviledged.PrivilegedActionSystemGetProperty;
/**
* Factory for creating an XML reader.
* This module, both source code and documentation, is in the
* Public Domain, and comes with NO WARRANTY. See http://www.saxproject.org for further
* information.
*
* This class contains static methods for creating an XML reader from an
* explicit class name, or based on runtime defaults:
*
*
*
* try
* {
* XMLReader myReader = XMLReaderFactoryCommons.createXMLReader ();
* }
* catch (final SAXException e)
* {
* System.err.println (e.getMessage ());
* }
*
*
* Note to Distributions bundled with parsers: You should
* modify the implementation of the no-arguments createXMLReader to
* handle cases where the external configuration mechanisms aren't set up. That
* method should do its best to return a parser when one is in the class path,
* even when nothing bound its class name to org.xml.sax.driver
so
* those configuration mechanisms would see it.
*
*
* Copied from JDK 1.6.0_45 and modified by phloc to work around an error when
* the class name was read from a JAR file and this method is invoked a second
* time!
*
*
* @since SAX 2.0
* @author David Megginson, David Brownell
* @version 2.0.1 (sax2r2)
*/
public final class XMLReaderFactoryCommons
{
/**
* This class is duplicated for each JAXP subpackage so keep it in sync. It is
* package private and therefore is not exposed as part of the JAXP API.
* Security related methods that only work on J2SE 1.2 and newer.
*/
private static final class SecuritySupport
{
ClassLoader getContextClassLoader () throws SecurityException
{
return AccessController.doPrivileged (new PrivilegedAction ()
{
public ClassLoader run ()
{
ClassLoader cl = Thread.currentThread ().getContextClassLoader ();
if (cl == null)
cl = ClassLoader.getSystemClassLoader ();
return cl;
}
});
}
String getSystemProperty (final String propName)
{
return AccessController.doPrivileged (new PrivilegedActionSystemGetProperty (propName));
}
InputStream getResourceAsStream (final ClassLoader cl, final String name)
{
return AccessController.doPrivileged (new PrivilegedAction ()
{
public InputStream run ()
{
InputStream ris;
if (cl == null)
ris = Object.class.getResourceAsStream (name);
else
ris = cl.getResourceAsStream (name);
return ris;
}
});
}
}
/**
* Create a new instance of a class by name.
* This module, both source code and documentation, is in the
* Public Domain, and comes with NO WARRANTY. See http://www.saxproject.org for further
* information.
*
* This class contains a static method for creating an instance of a class
* from an explicit class name. It tries to use the thread's context
* ClassLoader if possible and falls back to using Class.forName(String).
*
*
* This code is designed to compile and run on JDK version 1.1 and later
* including versions of Java 2.
*
*
* @author Edwin Goei, David Brownell
* @version 2.0.1 (sax2r2)
*/
private static final class NewInstance
{
private static final String DEFAULT_PACKAGE = "com.sun.org.apache.xerces.internal";
/**
* Creates a new instance of the specified class name Package private so
* this code is not exposed at the API level.
*/
static Object newInstance (final ClassLoader classLoader, final String className) throws ClassNotFoundException,
IllegalAccessException,
InstantiationException
{
// make sure we have access to restricted packages
boolean internal = false;
if (System.getSecurityManager () != null)
if (className != null && className.startsWith (DEFAULT_PACKAGE))
internal = true;
Class > driverClass;
if (classLoader == null || internal)
driverClass = Class.forName (className);
else
driverClass = classLoader.loadClass (className);
return driverClass.newInstance ();
}
}
/**
* Private constructor.
*
* This constructor prevents the class from being instantiated.
*
*/
private XMLReaderFactoryCommons ()
{}
private static final String PROPERTY_NAME = "org.xml.sax.driver";
private static SecuritySupport s_aSecuritySupport = new SecuritySupport ();
private static boolean s_bJARRead = false;
private static String s_sPreviouslyReadClassname;
/**
*
* Attempt to create an XMLReader from system defaults. In environments which
* can support it, the name of the XMLReader class is determined by trying
* each these options in order, and using the first one which succeeds:
*
*
* - If the system property
org.xml.sax.driver
has a value,
* that is used as an XMLReader class name.
* - The JAR "Services API" is used to look for a class name in the
* META-INF/services/org.xml.sax.driver file in jarfiles available to
* the runtime.
* - SAX parser distributions are strongly encouraged to provide a default
* XMLReader class name that will take effect only when previous options (on
* this list) are not successful.
*
*
* In environments such as small embedded systems, which can not support that
* flexibility, other mechanisms to determine the default may be used.
*
*
* Note that many Java environments allow system properties to be initialized
* on a command line. This means that in most cases setting a good
* value for that property ensures that calls to this method will succeed,
* except when security policies intervene. This will also maximize
* application portability to older SAX environments, with less robust
* implementations of this method.
*
*
* @return A new XMLReader.
* @exception org.xml.sax.SAXException
* If no default XMLReader class can be identified and
* instantiated.
* @see #createXMLReader(java.lang.String)
*/
@Nonnull
public static XMLReader createXMLReader () throws SAXException
{
String className = s_sPreviouslyReadClassname;
ClassLoader cl = s_aSecuritySupport.getContextClassLoader ();
if (className == null)
{
// 1. try the JVM-instance-wide system property
try
{
className = s_aSecuritySupport.getSystemProperty (PROPERTY_NAME);
if (className != null)
s_sPreviouslyReadClassname = className;
}
catch (final RuntimeException e)
{
// continue searching
}
// 2. if that fails, try META-INF/services/
if (className == null)
{
if (!s_bJARRead)
{
s_bJARRead = true;
final String service = "META-INF/services/" + PROPERTY_NAME;
InputStream in;
BufferedReader reader;
try
{
if (cl != null)
{
in = s_aSecuritySupport.getResourceAsStream (cl, service);
// If no provider found then try the current ClassLoader
if (in == null)
{
cl = null;
in = s_aSecuritySupport.getResourceAsStream (cl, service);
}
}
else
{
// No Context ClassLoader, try the current ClassLoader
in = s_aSecuritySupport.getResourceAsStream (null, service);
}
if (in != null)
{
try
{
reader = new BufferedReader (new InputStreamReader (in, "UTF8"));
try
{
className = reader.readLine ();
}
finally
{
StreamHelper.close (reader);
}
// [ph] Remember the read class name
s_sPreviouslyReadClassname = className;
}
finally
{
StreamHelper.close (in);
}
}
}
catch (final Throwable t)
{}
}
}
}
// 3. Distro-specific fallback
if (className == null)
{
// EXAMPLE:
// className = "com.example.sax.XmlReader";
// or a $JAVA_HOME/jre/lib/*properties setting...
className = s_sPreviouslyReadClassname = "com.sun.org.apache.xerces.internal.parsers.SAXParser";
}
else
{
// [ph] added
// Try to speed things up for Apache Xerces
// Even though Xerces is only an optional dependency, this invocation
// should work since the check on the name is done previously
if ("org.apache.xerces.parsers.SAXParser".equals (className))
return new org.apache.xerces.parsers.SAXParser ();
}
// do we know the XMLReader implementation class yet?
return _loadClass (cl, className);
}
/**
* Attempt to create an XML reader from a class name.
*
* Given a class name, this method attempts to load and instantiate the class
* as an XML reader.
*
*
* Note that this method will not be usable in environments where the caller
* (perhaps an applet) is not permitted to load classes dynamically.
*
*
* @param sClassName
* Class name to be loaded
* @return A new XML reader.
* @exception org.xml.sax.SAXException
* If the class cannot be loaded, instantiated, and cast to
* XMLReader.
* @see #createXMLReader()
*/
@Nonnull
public static XMLReader createXMLReader (final String sClassName) throws SAXException
{
return _loadClass (s_aSecuritySupport.getContextClassLoader (), sClassName);
}
@Nonnull
private static XMLReader _loadClass (final ClassLoader aClassLoader, final String sClassName) throws SAXException
{
try
{
return (XMLReader) NewInstance.newInstance (aClassLoader, sClassName);
}
catch (final ClassNotFoundException e1)
{
throw new SAXException ("SAX2 driver class " + sClassName + " not found", e1);
}
catch (final IllegalAccessException e2)
{
throw new SAXException ("SAX2 driver class " + sClassName + " found but cannot be loaded", e2);
}
catch (final InstantiationException e3)
{
throw new SAXException ("SAX2 driver class " +
sClassName +
" loaded but cannot be instantiated (no empty public constructor?)", e3);
}
catch (final ClassCastException e4)
{
throw new SAXException ("SAX2 driver class " + sClassName + " does not implement XMLReader", e4);
}
}
}