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

org.apache.axiom.om.util.StAXUtils Maven / Gradle / Ivy

There is a newer version: 1.4.0
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements. See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. The ASF licenses this file
 * to you 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 org.apache.axiom.om.util;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.axiom.om.OMConstants;

import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLOutputFactory;
import javax.xml.stream.XMLResolver;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.XMLStreamReader;
import javax.xml.stream.XMLStreamWriter;

import java.io.ByteArrayInputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.Reader;
import java.io.Writer;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.security.PrivilegedActionException;
import java.security.PrivilegedExceptionAction;
import java.util.Collections;
import java.util.Map;
import java.util.WeakHashMap;


public class StAXUtils {
    private static Log log = LogFactory.getLog(StAXUtils.class);
    private static boolean isDebugEnabled = log.isDebugEnabled();
    
    // If isFactoryPerClassLoader is true (default), then 
    // a separate singleton XMLInputFactory and XMLOutputFactory is maintained
    // for the each classloader.  The different classloaders may be using different
    // implementations of STAX.
    // 
    // If isFactoryPerClassLoader is false, then
    // a single XMLInputFactory and XMLOutputFactory is constructed using
    // the classloader that loaded StAXUtils. 
    private static boolean isFactoryPerClassLoader = true;
    
    // These static singletons are used when the XML*Factory is created with
    // the StAXUtils classloader.
    private static XMLInputFactory inputFactory = null;
    private static XMLInputFactory inputNDFactory = null;
    private static XMLOutputFactory outputFactory = null;
    
    // These maps are used for the isFactoryPerClassLoader==true case
    // The maps are synchronized and weak.
    private static Map inputFactoryPerCL = Collections.synchronizedMap(new WeakHashMap());
    private static Map inputNDFactoryPerCL = Collections.synchronizedMap(new WeakHashMap());
    private static Map outputFactoryPerCL = Collections.synchronizedMap(new WeakHashMap());
    
    /**
     * Gets an XMLInputFactory instance from pool.
     *
     * @return an XMLInputFactory instance.
     */
    public static XMLInputFactory getXMLInputFactory() {
        
        if (isFactoryPerClassLoader) {
            return getXMLInputFactory_perClassLoader(false);
        } else {
            return getXMLInputFactory_singleton(false);
        }
    }
    
    /**
     * Get XMLInputFactory
     * @param factoryPerClassLoaderPolicy 
     * (if true, then factory using current classloader.
     * if false, then factory using the classloader that loaded StAXUtils)
     * @return XMLInputFactory
     */
    public static XMLInputFactory getXMLInputFactory(boolean factoryPerClassLoaderPolicy) {
        if (factoryPerClassLoaderPolicy) {
            return getXMLInputFactory_perClassLoader(false);
        } else {
            return getXMLInputFactory_singleton(false);
        }
    }

    /**
     * @deprecated
     * Returns an XMLInputFactory instance for reuse.
     *
     * @param factory An XMLInputFactory instance that is available for reuse
     */
    public static void releaseXMLInputFactory(XMLInputFactory factory) {
    }

    public static XMLStreamReader createXMLStreamReader(final InputStream in, final String encoding)
            throws XMLStreamException {
        final XMLInputFactory inputFactory = getXMLInputFactory();
        try {
            XMLStreamReader reader = 
                (XMLStreamReader) 
                AccessController.doPrivileged(new PrivilegedExceptionAction() {
                    public Object run() throws XMLStreamException {
                        return inputFactory.createXMLStreamReader(in, encoding);
                    }
                }
                );
            if (isDebugEnabled) {
                log.debug("XMLStreamReader is " + reader.getClass().getName());
            }
            return reader;
        } catch (PrivilegedActionException pae) {
            throw (XMLStreamException) pae.getException();
        }
    }

    public static XMLStreamReader createXMLStreamReader(final InputStream in)
            throws XMLStreamException {
        final XMLInputFactory inputFactory = getXMLInputFactory();
        try {
            XMLStreamReader reader = 
                (XMLStreamReader)
                AccessController.doPrivileged(new PrivilegedExceptionAction() {
                    public Object run() throws XMLStreamException {
                        return inputFactory.createXMLStreamReader(in);
                    }
                }
                );
            
            if (isDebugEnabled) {
                log.debug("XMLStreamReader is " + reader.getClass().getName());
            }
            return reader;
        } catch (PrivilegedActionException pae) {
            throw (XMLStreamException) pae.getException();
        }
    }

    public static XMLStreamReader createXMLStreamReader(final Reader in)
            throws XMLStreamException {
        final XMLInputFactory inputFactory = getXMLInputFactory();
        try {
            XMLStreamReader reader = 
                (XMLStreamReader)
                AccessController.doPrivileged(new PrivilegedExceptionAction() {
                    public Object run() throws XMLStreamException {
                        return inputFactory.createXMLStreamReader(in);
                    }
                }
                );
            if (isDebugEnabled) {
                log.debug("XMLStreamReader is " + reader.getClass().getName());
            }
            return reader;
        } catch (PrivilegedActionException pae) {
            throw (XMLStreamException) pae.getException();
        }
    }

    /**
     * Gets an XMLOutputFactory instance from pool.
     *
     * @return an XMLOutputFactory instance.
     */
    public static XMLOutputFactory getXMLOutputFactory() {
        if (isFactoryPerClassLoader) {
            return getXMLOutputFactory_perClassLoader();
        } else {
            return getXMLOutputFactory_singleton();
        }
    }
    
    /**
     * Get XMLOutputFactory
     * @param factoryPerClassLoaderPolicy 
     * (if true, then factory using current classloader.
     * if false, then factory using the classloader that loaded StAXUtils)
     * @return XMLInputFactory
     */
    public static XMLOutputFactory getXMLOutputFactory(boolean factoryPerClassLoaderPolicy) {
        if (factoryPerClassLoaderPolicy) {
            return getXMLOutputFactory_perClassLoader();
        } else {
            return getXMLOutputFactory_singleton();
        }
    }
    
    /**
     * Set the policy for how to maintain the XMLInputFactory and XMLOutputFactory
     * @param value (if false, then one singleton...if true...then singleton per class loader 
     *  (default is true)
     */
    public static void setFactoryPerClassLoader(boolean value) {
        isFactoryPerClassLoader = value;
    }

    /**
     * @deprecated
     * Returns an XMLOutputFactory instance for reuse.
     *
     * @param factory An XMLOutputFactory instance that is available for reuse.
     */
    public static void releaseXMLOutputFactory(XMLOutputFactory factory) {
    }

    public static XMLStreamWriter createXMLStreamWriter(final OutputStream out)
            throws XMLStreamException {
        final XMLOutputFactory outputFactory = getXMLOutputFactory();
        try {
            XMLStreamWriter writer = 
                (XMLStreamWriter)
                AccessController.doPrivileged(new PrivilegedExceptionAction() {
                    public Object run() throws XMLStreamException {
                        return outputFactory.createXMLStreamWriter(out, OMConstants.DEFAULT_CHAR_SET_ENCODING);
                    }
                }
                );
                
            if (isDebugEnabled) {
                log.debug("XMLStreamWriter is " + writer.getClass().getName());
            }
            return writer;
        } catch (PrivilegedActionException pae) {
            throw (XMLStreamException) pae.getException();
        }
    }

    public static XMLStreamWriter createXMLStreamWriter(final OutputStream out, final String encoding)
            throws XMLStreamException {
        final XMLOutputFactory outputFactory = getXMLOutputFactory();
        try {
            XMLStreamWriter writer = 
                (XMLStreamWriter)
                AccessController.doPrivileged(new PrivilegedExceptionAction() {
                    public Object run() throws XMLStreamException {
                        return outputFactory.createXMLStreamWriter(out, encoding);
                    }
                }
                );
            
            if (isDebugEnabled) {
                log.debug("XMLStreamWriter is " + writer.getClass().getName());
            }
            return writer;
        } catch (PrivilegedActionException pae) {
            throw (XMLStreamException) pae.getException();
        }
    }

    public static XMLStreamWriter createXMLStreamWriter(final Writer out)
            throws XMLStreamException {
        final XMLOutputFactory outputFactory = getXMLOutputFactory();
        try {
            XMLStreamWriter writer = 
                (XMLStreamWriter)
                AccessController.doPrivileged(new PrivilegedExceptionAction() {
                    public Object run() throws XMLStreamException {
                        return outputFactory.createXMLStreamWriter(out);
                    }
                }
                );
            if (isDebugEnabled) {
                log.debug("XMLStreamWriter is " + writer.getClass().getName());
            }
            return writer;
        } catch (PrivilegedActionException pae) {
            throw (XMLStreamException) pae.getException();
        }
    }

    /**
     * @deprecated
     */
    public static void reset() {
    }
    
    // This has package access since it is used from within anonymous inner classes
    static XMLInputFactory newXMLInputFactory(boolean isNetworkDetached) {
        XMLInputFactory factory = XMLInputFactory.newInstance();
        if (isNetworkDetached) {
            factory.setProperty(XMLInputFactory.IS_SUPPORTING_EXTERNAL_ENTITIES, 
                      Boolean.FALSE);
            // Some StAX parser such as Woodstox still try to load the external DTD subset,
            // even if IS_SUPPORTING_EXTERNAL_ENTITIES is set to false. To work around this,
            // we add a custom XMLResolver that returns empty documents. See WSTX-117 for
            // an interesting discussion about this.
            factory.setXMLResolver(new XMLResolver() {
                public Object resolveEntity(String publicID, String systemID, String baseURI,
                        String namespace) throws XMLStreamException {
                    return new ByteArrayInputStream(new byte[0]);
                }
            });
        }
        return factory;
    }

    /**
     * @return XMLInputFactory for the current classloader
     */
    private static XMLInputFactory getXMLInputFactory_perClassLoader(final boolean isNetworkDetached) {
        
        ClassLoader cl = getContextClassLoader();
        XMLInputFactory factory;
        if (cl == null) {
            factory = getXMLInputFactory_singleton(isNetworkDetached);
        } else {
            // Check the cache
            if (isNetworkDetached) {
                factory = (XMLInputFactory) inputNDFactoryPerCL.get(cl);
            } else {
                factory = (XMLInputFactory) inputFactoryPerCL.get(cl);
            }
            
            // If not found in the cache map, crate a new factory
            if (factory == null) {

                if (log.isDebugEnabled()) {
                    log.debug("About to create XMLInputFactory implementation with " +
                                "classloader=" + cl);
                    log.debug("The classloader for javax.xml.stream.XMLInputFactory is: "
                              + XMLInputFactory.class.getClassLoader());
                }
                factory = null;
                try {
                    factory = (XMLInputFactory) 
                    AccessController.doPrivileged(
                        new PrivilegedAction() {
                            public Object run() {
                                return newXMLInputFactory(isNetworkDetached);
                            }
                        });
                } catch (ClassCastException cce) {
                    if (log.isDebugEnabled()) {
                        log.debug("Failed creation of XMLInputFactory implementation with " +
                                        "classloader=" + cl);
                        log.debug("Exception is=" + cce);
                        log.debug("Attempting with classloader: " + 
                                  XMLInputFactory.class.getClassLoader());
                    }
                    factory = (XMLInputFactory) 
                    AccessController.doPrivileged(
                        new PrivilegedAction() {
                            public Object run() {
                                ClassLoader saveCL = getContextClassLoader();
                                try {                              
                                    Thread.currentThread().
                                        setContextClassLoader(
                                            XMLInputFactory.class.getClassLoader());
                                    return newXMLInputFactory(isNetworkDetached);
                                } finally {
                                    Thread.currentThread().
                                        setContextClassLoader(saveCL);
                                }
                            }
                        });
                }
                    
                if (factory != null) {
                    // Cache the new factory
                    if (isNetworkDetached) {
                        inputNDFactoryPerCL.put(cl, factory);
                    } else {
                        inputFactoryPerCL.put(cl, factory);
                    }
                    
                    if (log.isDebugEnabled()) {
                        log.debug("Created XMLInputFactory = " + factory.getClass() + 
                                  " with classloader=" + cl);
                        log.debug("Size of XMLInputFactory map =" + inputFactoryPerCL.size());
                        log.debug("isNetworkDetached =" + isNetworkDetached);
                    }
                } else {
                    factory = getXMLInputFactory_singleton(isNetworkDetached);
                }
            }
            
        }
        return factory;
    }
    
    /**
     * @return singleton XMLInputFactory loaded with the StAXUtils classloader
     */
    private static XMLInputFactory getXMLInputFactory_singleton(final boolean isNetworkDetached) {
        XMLInputFactory f;
        if (isNetworkDetached) {
            f = inputNDFactory;
        } else {
            f = inputFactory;
        }
        if (f == null) {
            f = (XMLInputFactory) AccessController.doPrivileged(
                    new PrivilegedAction() {
                        public Object run() {
                            Thread currentThread = Thread.currentThread();
                            ClassLoader savedClassLoader = currentThread.getContextClassLoader();
                            try {
                                currentThread.setContextClassLoader(StAXUtils.class.getClassLoader());
                                return newXMLInputFactory(isNetworkDetached);
                            }
                            finally {
                                currentThread.setContextClassLoader(savedClassLoader);
                            }
                        }
                    });
            if (isNetworkDetached) {
                inputNDFactory = f;
            } else {
                inputFactory = f;
            }
            if (log.isDebugEnabled()) {
                if (f != null) {
                    if (isNetworkDetached) {
                        log.debug("Created singleton network detached XMLInputFactory = " + f.getClass());
                    } else {
                        log.debug("Created singleton XMLInputFactory = " + f.getClass());
                    }
                }
            }
        }
        
        return f;
    }
    
    /**
     * @return XMLOutputFactory for the current classloader
     */
    public static XMLOutputFactory getXMLOutputFactory_perClassLoader() {
        ClassLoader cl = getContextClassLoader();
        XMLOutputFactory factory;
        if (cl == null) {
            factory = getXMLOutputFactory_singleton();
        } else {
            factory = (XMLOutputFactory) outputFactoryPerCL.get(cl);
            if (factory == null) {
                if (log.isDebugEnabled()) {
                    log.debug("About to create XMLOutputFactory implementation with " +
                                "classloader=" + cl);
                    log.debug("The classloader for javax.xml.stream.XMLOutputFactory is: " + 
                              XMLOutputFactory.class.getClassLoader());
                }
                try {
                    factory = (XMLOutputFactory) 
                    AccessController.doPrivileged(
                        new PrivilegedAction() {
                            public Object run() {
                                XMLOutputFactory factory = XMLOutputFactory.newInstance();
                                factory.setProperty(XMLOutputFactory.IS_REPAIRING_NAMESPACES, 
                                                    Boolean.FALSE);
                                return factory;
                            }
                        });
                } catch (ClassCastException cce) {
                    if (log.isDebugEnabled()) {
                        log.debug("Failed creation of XMLOutputFactory implementation with " +
                                        "classloader=" + cl);
                        log.debug("Exception is=" + cce);
                        log.debug("Attempting with classloader: " + 
                                  XMLOutputFactory.class.getClassLoader());
                    }
                    factory = (XMLOutputFactory) 
                    AccessController.doPrivileged(
                        new PrivilegedAction() {
                            public Object run() {
                                XMLOutputFactory f = null;
                                ClassLoader saveCL = getContextClassLoader();
                                try {                              
                                    Thread.currentThread().
                                        setContextClassLoader(
                                           XMLOutputFactory.class.getClassLoader());
                                    f =XMLOutputFactory.newInstance();
                                    f.setProperty(XMLOutputFactory.IS_REPAIRING_NAMESPACES, 
                                                        Boolean.FALSE);
                                } finally {
                                    Thread.currentThread().
                                        setContextClassLoader(saveCL);
                                }
                                return f;
                            }
                        });
                }
                if (factory != null) {
                    outputFactoryPerCL.put(cl, factory);
                    if (log.isDebugEnabled()) {
                        log.debug("Created XMLOutputFactory = " + factory.getClass() 
                                  + " for classloader=" + cl);
                        log.debug("Size of XMLOutputFactory map =" + outputFactoryPerCL.size());
                    }
                } else {
                    factory = getXMLOutputFactory_singleton();
                }
            }
            
        }
        return factory;
    }
    
    /**
     * @return XMLOutputFactory singleton loaded with the StAXUtils classloader
     */
    public static XMLOutputFactory getXMLOutputFactory_singleton() {
        if (outputFactory == null) {
            outputFactory = (XMLOutputFactory) AccessController.doPrivileged(
                    new PrivilegedAction() {
                        public Object run() {

                            Thread currentThread = Thread.currentThread();
                            ClassLoader savedClassLoader = currentThread.getContextClassLoader();
                            XMLOutputFactory factory = null;
                            try {
                                currentThread.setContextClassLoader(StAXUtils.class.getClassLoader());
                                factory = XMLOutputFactory.newInstance();
                                factory.setProperty(XMLOutputFactory.IS_REPAIRING_NAMESPACES, 
                                                    Boolean.FALSE);
                            }
                            finally {
                                currentThread.setContextClassLoader(savedClassLoader);
                            }
                            return factory;
                        }
                    });
            if (log.isDebugEnabled()) {
                if (outputFactory != null) {
                    log.debug("Created singleton XMLOutputFactory = " + outputFactory.getClass());
                }
            }
        }
        return outputFactory;
    }
    
    /**
     * @return Trhead Context ClassLoader
     */
    private static ClassLoader getContextClassLoader() {
        ClassLoader cl = (ClassLoader) AccessController.doPrivileged(
                    new PrivilegedAction() {
                        public Object run()  {
                            return Thread.currentThread().getContextClassLoader();
                        }
                    }
            );
        
        return cl;
    }

    /**
     * Create an XMLStreamReader that will operate when detached from a network.
     * The XMLStreamReader is created from a OMInputFactory that has external
     * entities disabled.  This kind of XMLStreamReader is useful for reading 
     * deployment information.
     * @param in
     * @param encoding
     * @return
     * @throws XMLStreamException
     */
    public static XMLStreamReader createNetworkDetachedXMLStreamReader(final InputStream in, final String encoding)
        throws XMLStreamException {
        final XMLInputFactory inputFactory = getNetworkDetachedXMLInputFactory();
        try {
            XMLStreamReader reader = 
                (XMLStreamReader) 
                AccessController.doPrivileged(new PrivilegedExceptionAction() {
                    public Object run() throws XMLStreamException {
                        return inputFactory.createXMLStreamReader(in, encoding);
                    }
                }
                );
            if (isDebugEnabled) {
                log.debug("XMLStreamReader is " + reader.getClass().getName());
            }
            return reader;
        } catch (PrivilegedActionException pae) {
            throw (XMLStreamException) pae.getException();
        }
    }

    /**
     * Gets an XMLInputFactory instance from pool.
     *
     * @return an XMLInputFactory instance.
     */
    public static XMLInputFactory getNetworkDetachedXMLInputFactory() {
        if (isFactoryPerClassLoader) {
            return getXMLInputFactory_perClassLoader(true);
        } else {
            return getXMLInputFactory_singleton(true);
        }
    }
    
    /**
     * Create an XMLStreamReader that will operate when detached from a network.
     * The XMLStreamReader is created from a OMInputFactory that has external
     * entities disabled.  This kind of XMLStreamReader is useful for reading 
     * deployment information.
     * 
     * @param in
     * @return
     * @throws XMLStreamException
     */
    public static XMLStreamReader createNetworkDetachedXMLStreamReader(final InputStream in)
    throws XMLStreamException {
        final XMLInputFactory inputFactory = getNetworkDetachedXMLInputFactory();
        try {
            XMLStreamReader reader = 
                (XMLStreamReader)
                AccessController.doPrivileged(new PrivilegedExceptionAction() {
                    public Object run() throws XMLStreamException {
                        return inputFactory.createXMLStreamReader(in);
                    }
                }
                );

            if (isDebugEnabled) {
                log.debug("XMLStreamReader is " + reader.getClass().getName());
            }
            return reader;
        } catch (PrivilegedActionException pae) {
            throw (XMLStreamException) pae.getException();
        }
    }

    /**
     * Create an XMLStreamReader that will operate when detached from a network.
     * The XMLStreamReader is created from a OMInputFactory that has external
     * entities disabled.  This kind of XMLStreamReader is useful for reading 
     * deployment information.
     * 
     * @param in
     * @return
     * @throws XMLStreamException
     */
    public static XMLStreamReader createNetworkDetachedXMLStreamReader(final Reader in)
    throws XMLStreamException {
        final XMLInputFactory inputFactory = getNetworkDetachedXMLInputFactory();
        try {
            XMLStreamReader reader = 
                (XMLStreamReader)
                AccessController.doPrivileged(new PrivilegedExceptionAction() {
                    public Object run() throws XMLStreamException {
                        return inputFactory.createXMLStreamReader(in);
                    }
                }
                );
            if (isDebugEnabled) {
                log.debug("XMLStreamReader is " + reader.getClass().getName());
            }
            return reader;
        } catch (PrivilegedActionException pae) {
            throw (XMLStreamException) pae.getException();
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy