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

com.helger.xml.serialize.read.XMLReaderFactoryCommons Maven / Gradle / Ivy

/**
 * Copyright (C) 2014-2016 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.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.ClassLoaderHelper;
import com.helger.commons.system.SystemProperties;

/**
 * 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 { ClassLoader cl = ClassLoaderHelper.getContextClassLoader (); if (cl == null) cl = ClassLoaderHelper.getSystemClassLoader (); return cl; } @SuppressWarnings ("resource") InputStream getResourceAsStream (final ClassLoader cl, final String name) { return AccessController.doPrivileged ((PrivilegedAction ) () -> { 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 = SystemProperties.getPropertyValueOrNull (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); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy