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

javanet.staxutils.XMLStreamUtils Maven / Gradle / Ivy

The newest version!
/*
 * $Id: XMLStreamUtils.java,v 1.8 2004/07/09 17:30:50 cniles Exp $
 * 
 * Copyright (c) 2004, Christian Niles, Unit12
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 * 
 *		*   Redistributions of source code must retain the above copyright
 *          notice, this list of conditions and the following disclaimer.
 * 
 *	    *	Redistributions in binary form must reproduce the above copyright
 *          notice, this list of conditions and the following disclaimer in the
 *          documentation and/or other materials provided with the distribution.
 * 
 *      *   Neither the name of Christian Niles, Unit12, nor the names of its
 *          contributors may be used to endorse or promote products derived from
 *          this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 * 
 */
package javanet.staxutils;

import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

import javax.xml.namespace.QName;
import javax.xml.stream.XMLEventFactory;
import javax.xml.stream.XMLEventReader;
import javax.xml.stream.XMLEventWriter;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLOutputFactory;
import javax.xml.stream.XMLStreamConstants;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.XMLStreamReader;
import javax.xml.stream.XMLStreamWriter;
import javax.xml.stream.events.Attribute;
import javax.xml.stream.events.StartElement;
import javax.xml.stream.events.XMLEvent;
import javax.xml.stream.util.XMLEventConsumer;
import javax.xml.transform.Result;
import javax.xml.transform.Source;

/**
 * Static utility methods useful when handling XML Streams.
 * 
 * @author Christian Niles
 * @version $Revision: 1.8 $
 */
public class XMLStreamUtils {

    private static XMLInputFactory inputFactory = XMLInputFactory.newInstance();

    private static XMLOutputFactory outputFactory = XMLOutputFactory.newInstance();

    private static final String[] EVENT_NAMES = new String[16];
    static {

        EVENT_NAMES[0] = ""; // no event has 0 index
        EVENT_NAMES[XMLStreamConstants.ATTRIBUTE] = "ATTRIBUTE";
        EVENT_NAMES[XMLStreamConstants.CDATA] = "CDATA";
        EVENT_NAMES[XMLStreamConstants.CHARACTERS] = "CHARACTERS";
        EVENT_NAMES[XMLStreamConstants.COMMENT] = "COMMENT";
        EVENT_NAMES[XMLStreamConstants.DTD] = "DTD";
        EVENT_NAMES[XMLStreamConstants.END_DOCUMENT] = "END_DOCUMENT";
        EVENT_NAMES[XMLStreamConstants.END_ELEMENT] = "END_ELEMENT";
        EVENT_NAMES[XMLStreamConstants.ENTITY_DECLARATION] = "ENTITY_DECLARATION";
        EVENT_NAMES[XMLStreamConstants.ENTITY_REFERENCE] = "ENTITY_REFERENCE";
        EVENT_NAMES[XMLStreamConstants.NAMESPACE] = "NAMESPACE";
        EVENT_NAMES[XMLStreamConstants.NOTATION_DECLARATION] = "NOTATION_DECLARATION";
        EVENT_NAMES[XMLStreamConstants.PROCESSING_INSTRUCTION] = "PROCESSING_INSTRUCTION";
        EVENT_NAMES[XMLStreamConstants.SPACE] = "SPACE";
        EVENT_NAMES[XMLStreamConstants.START_DOCUMENT] = "START_DOCUMENT";
        EVENT_NAMES[XMLStreamConstants.START_ELEMENT] = "START_ELEMENT";

    }

    /**
     * Returns the name of the specified stream event constant.
     * 
     * @param eventType The event constant, such as
     *     {@link XMLStreamConstants#START_DOCUMENT}.
     * @return The name of the specified event, or "UNKNOWN" if the
     *     constant isn't valid.
     */
    public static final String getEventTypeName(int eventType) {

        if (eventType > 0 || eventType < EVENT_NAMES.length) {

            return EVENT_NAMES[eventType];

        } else {

            return "UNKNOWN";

        }

    }

    /**
     * Returns the value of the attribute with the given non-qualified name.
     * 
     * @param reader The xml stream reader
     * @param name The name of the attribute.
     * @return The value of the unqualified attribute, or null
     *         if the attribute wasn't present.
     */
    public static final String attributeValue(XMLStreamReader reader,
            String name) {

        return reader.getAttributeValue("", name);

    }

    /**
     * Returns the value of the attribute with the given name.
     * 
     * @param reader The xml stream reader
     * @param name The name of the attribute.
     * @return The value of the attribute, or null if the
     *         attribute wasn't present.
     */
    public static final String attributeValue(XMLStreamReader reader, QName name) {

        return reader.getAttributeValue(name.getNamespaceURI(),
                name.getLocalPart());

    }

    /**
     * Skips all events within a single element, including its start and end
     * tags. The provided reader must be positioned directly in front of a
     * StartElement event or it will have no effect. After this
     * method completes, the reader will be positioned before the event
     * following the end tag (the end tag will have been read).
     * 
     * @param reader The event stream to read.
     * @throws XMLStreamException If an error occurs reading events.
     */
    public static final void skipElement(XMLEventReader reader)
            throws XMLStreamException {

        copyElement(reader, null);

    }

    /**
     * Copies an element and all its content from the provided event reader, to
     * the provided event consumer. The event reader must be positioned before a
     * start element event, or this method has no effect.
     * 
     * @param reader The reader from which to read the events.
     * @param consumer The destination for read events, or null to
     * 		ignore all events.
     * @throws XMLStreamException If an error occurs reading or writing the
     *             events.
     */
    public static final void copyElement(XMLEventReader reader,
            XMLEventConsumer consumer) throws XMLStreamException {

        if (!reader.hasNext())
            return;

        XMLEvent event = reader.peek();
        if (!event.isStartElement())
            return;

        int depth = 0;
        do {

            XMLEvent currEvt = reader.nextEvent();
            if (currEvt.isStartElement()) {

                depth++;

            } else if (currEvt.isEndElement()) {

                depth--;

            }

            if (consumer != null) {

                consumer.add(currEvt);

            }

        } while (depth > 0 && reader.hasNext());

    }

    /**
     * Skips all events within a StartElement until the matching
     * EndElement is reached. This method assumes that the reader
     * is positioned after the StartElement event, and when the
     * method completes, the stream will be positioned before the
     * EndElement event, but it will not consume the end tag.
     * 
     * @param reader The event stream to read, positioned after the
     * 		StartElement
     * @throws XMLStreamException If an error occurs reading events.
     */
    public static final void skipElementContent(XMLEventReader reader)
            throws XMLStreamException {

        copyElementContent(reader, null);

    }

    /**
     * Copies all events within a StartElement until the matching
     * EndElement is reached. This method assumes that the reader
     * is positioned after the StartElement event, and when the
     * method completes, the stream will be positioned before the 
     * EndElement event, but it will not consume the end tag.
     * 
     * @param reader The event stream to read, positioned after the
     * 		StartElement
     * @param consumer The destination for events read from teh stream, or
     * 		null to ignore the events completely.
     * @throws XMLStreamException If an error occurs reading events.
     */
    public static final void copyElementContent(XMLEventReader reader,
            XMLEventConsumer consumer) throws XMLStreamException {

        if (!reader.hasNext())
            return;

        for (int depth = 1; true;) {

            // peek and see if we're at the end element
            XMLEvent currEvt = reader.peek();
            if (currEvt.isEndElement()) {

                depth--;
                if (depth == 0) {

                    break;

                }

            } else if (currEvt.isStartElement()) {

                depth++;

            }

            // consume the event
            currEvt = reader.nextEvent();

            if (consumer != null) {

                consumer.add(currEvt);

            }

        }

    }

    /**
     * Skips the complete content of the element at the specified reader's
     * cursor. The reader's current event type must be START_ELEMENT, otherwise
     * this method will have no effect. Upon completion, the reader's cursor
     * will be at the END_ELEMENT event for the skipped element.
     * 
     * @param reader An XML stream reader currently in the START_ELEMENT event.
     */
    public static final void skipElement(XMLStreamReader reader)
            throws XMLStreamException {

        if (reader.isStartElement()) {

            skipElementContent(reader);

        }

    }

    /**
     * Skips an element's complete content. This method assumes that the
     * START_ELEMENT has already be passed, and when it terminates,
     * the stream will be positioned at the END_ELEMENT.
     * 
     * @param reader The stream reader to read.
     * @throws XMLStreamException If an error occurs reading the stream.
     */
    public static final void skipElementContent(XMLStreamReader reader)
            throws XMLStreamException {

        int depth = 0;
        while (depth >= 0) {

            reader.next();
            if (reader.isStartElement()) {

                depth++;

            } else if (reader.isEndElement()) {

                depth--;

            }

        }

    }

    /**
     * Static utility method that throws an exception if the supplied reader's
     * cursor doesn't point to a START_ELEMENT with the given name.
     * 
     * @param reader The reader to test.
     * @param name The name of the element to require.
     * @throws XMLStreamException If the reader state is an element with the
     *             specified name.
     */
    public static final void requireElement(XMLStreamReader reader, QName name)
            throws XMLStreamException {

        reader.require(XMLStreamReader.START_ELEMENT, name.getNamespaceURI(),
                name.getLocalPart());

    }

    /**
     * Copies the content read from the specified source stream to the provided
     * result stream. This method is exactly the same as calling
     * {@link XMLEventWriter#add(XMLEventReader)}, and is provided only for
     * completeness.
     * 
     * @param reader The source stream.
     * @param consumer The destination stream.
     * @throws XMLStreamException If an error occurs copying the stream
     *             contents.
     */
    public static final void copy(XMLEventReader reader,
            XMLEventConsumer consumer) throws XMLStreamException {

        if (consumer instanceof XMLEventWriter) {

            copy(reader, (XMLEventWriter) consumer);

        } else {

            while (reader.hasNext()) {

                consumer.add(reader.nextEvent());

            }

        }

    }

    /**
     * Copies the content read from the specified source stream to the provided
     * result stream. This method is exactly the same as calling
     * {@link XMLEventWriter#add(XMLEventReader)}, and is provided only for
     * completeness.
     * 
     * @param reader The source stream.
     * @param writer The destination stream.
     * @throws XMLStreamException If an error occurs copying the stream
     *             contents.
     */
    public static final void copy(XMLEventReader reader, XMLEventWriter writer)
            throws XMLStreamException {

        writer.add(reader);

    }

    /**
     * Copies the content read from the specified source stream to the provided
     * result stream.
     * 
     * @param reader The source stream.
     * @param writer The destination stream.
     * @throws XMLStreamException If an error occurs copying the stream
     *             contents.
     */
    public static final void copy(XMLStreamReader reader, XMLStreamWriter writer)
            throws XMLStreamException {

        XMLEventReader r = inputFactory.createXMLEventReader(reader);
        XMLEventWriter w = new XMLStreamEventWriter(writer);

        try {

            w.add(r);

        } finally {

            // force any cached events to the underlying writer
            w.flush();

        }

    }

    /**
     * Copies the content read from the specified source stream to the provided
     * result stream.
     * 
     * @param reader The source stream.
     * @param writer The destination stream.
     * @param factory An optional input factory used to create any intermediate
     *            streams.
     * @throws XMLStreamException If an error occurs copying the stream
     *             contents.
     */
    public static final void copy(XMLStreamReader reader,
            XMLStreamWriter writer, XMLInputFactory factory)
            throws XMLStreamException {

        if (factory == null) {

            factory = inputFactory;

        }

        XMLEventReader r = factory.createXMLEventReader(reader);
        XMLEventWriter w = new XMLStreamEventWriter(writer);

        try {

            w.add(r);

        } finally {

            // force any cached events to the underlying writer
            w.flush();

        }

    }

    /**
     * Copies the content read from a TrAX {@link Source} to a StAX
     * {@link XMLStreamWriter}.
     * 
     * @param source The content source.
     * @param writer The destination stream.
     * @throws XMLStreamException If an error occurs copying the content to the
     * 		stream.
     */
    public static final void copy(Source source, XMLStreamWriter writer)
            throws XMLStreamException {

        XMLStreamReader reader = inputFactory.createXMLStreamReader(source);
        copy(reader, writer);

    }

    /**
     * Copies the content read from a TrAX {@link Source} to a StAX
     * {@link XMLEventWriter}.
     * 
     * @param source The content source.
     * @param writer The destination event stream.
     * @throws XMLStreamException If an error occurs copying the content to the
     * 		event stream.
     */
    public static final void copy(Source source, XMLEventWriter writer)
            throws XMLStreamException {

        XMLEventReader reader = inputFactory.createXMLEventReader(source);
        copy(reader, writer);

    }

    /**
     * Copies the content read from a StAX {@link XMLEventReader} to a TrAX
     * {@link Result}.
     * 
     * @param reader The source event stream.
     * @param result The destination {@link Result}.
     * @throws XMLStreamException If an error occurs copying the content to the
     * 		result.
     */
    public static final void copy(XMLEventReader reader, Result result)
            throws XMLStreamException {

        XMLEventWriter writer = outputFactory.createXMLEventWriter(result);

        copy(reader, writer);

        // force any cached events to the result
        writer.flush();

    }

    /**
     * Copies the content read from a StAX {@link XMLStreamReader} to a TrAX
     * {@link Result}.
     * 
     * @param reader The source stream.
     * @param result The destination {@link Result}.
     * @throws XMLStreamException If an error occurs copying the content to the
     * 		result.
     */
    public static final void copy(XMLStreamReader reader, Result result)
            throws XMLStreamException {

        XMLStreamWriter writer = outputFactory.createXMLStreamWriter(result);

        copy(reader, writer);

        // force any cached content to the result
        writer.flush();

    }

    /**
     * Utility method that throws an exception if the provided reader is not
     * positioned before a StartElement event with the specified tag name.
     * 
     * @param reader The reader to test.
     * @param qname The required name of the start-tag. If null,
     *            any start tag is accepted.
     * @throws XMLStreamException If an error occurs reading from the stream.
     */
    public static final void requireStartElement(XMLEventReader reader,
            QName qname) throws XMLStreamException {

        if (reader.hasNext()) {

            XMLEvent nextEvent = reader.peek();
            if (nextEvent.isStartElement()) {

                if (qname != null) {

                    StartElement start = nextEvent.asStartElement();
                    QName name = start.getName();
                    if (!name.equals(qname)) {

                        throw new XMLStreamException(
                                "Encountered unexpected element; expected "
                                        + qname + ", but found " + name);

                    }

                }

            } else {

                throw new XMLStreamException(
                        "Encountered unexpected event; expected " + qname
                                + " start-tag, but found event " + nextEvent);

            }

        } else {

            throw new XMLStreamException(
                    "Encountered unexpected end of stream; expected element "
                            + qname);

        }

    }

    /**
     * Constructs a new StartElement that merges the attributes and namespaces
     * found in the specified StartElement, with the provided attributes. The
     * returned StartElement will contain all the attributes and namespaces of
     * the original, plus those defined in the map.
     * 
     * @param tag The original StartElement
     * @param attrs An iterator of Atributes to add to the element.
     * @return A new StartElement that contains all the original attributes and
     *         namespaces, plus the provided attributes.
     */
    public static StartElement mergeAttributes(StartElement tag,
            Iterator attrs, XMLEventFactory factory) {

        // create Attribute map
        Map attributes = new HashMap();

        // iterate through start tag's attributes
        for (Iterator i = tag.getAttributes(); i.hasNext();) {

            Attribute attr = (Attribute) i.next();
            attributes.put(attr.getName(), attr);

        }

        // iterate through new attributes
        while (attrs.hasNext()) {

            Attribute attr = (Attribute) attrs.next();
            attributes.put(attr.getName(), attr);

        }

        factory.setLocation(tag.getLocation());

        QName tagName = tag.getName();
        return factory.createStartElement(tagName.getPrefix(),
                tagName.getNamespaceURI(), tagName.getLocalPart(),
                attributes.values().iterator(), tag.getNamespaces(),
                tag.getNamespaceContext());

    }

    /**
     * Reads the text content of an element. The reader should be positioned in
     * front of a StartElement event, and will be read up to and including the
     * end element tag.
     * 
     * @param reader The event stream from which to read the element text.
     * @param elemName The optional name of the element being read. If this
     *            paramter is non- null then an exception will
     *            be thrown if the element read doesn't have the same name.
     * @return The text read from the element.
     * @throws XMLStreamException If an error occurs reading the stream, or if
     *             the read element doesn't match the provided QName.
     */
    public static final String readTextElement(XMLEventReader reader,
            QName elemName) throws XMLStreamException {

        if (elemName != null) {

            requireStartElement(reader, elemName);

        }

        // read text
        String text = reader.getElementText();

        // consume the end tag
        reader.nextEvent();

        return text;

    }

    /**
     * Advances the event stream until it encounters a start or end tag, but
     * does not actaully read the event.
     * 
     * @param reader The reader to peek.
     * @return The next StartElement or EndElement event, retrieved using
     *         peek(), or null if the end of the
     *         stream was encountered before any tag event.
     * @throws XMLStreamException If an error occurs reading the stream.
     */
    public static final XMLEvent nextTag(XMLEventReader reader)
            throws XMLStreamException {

        while (reader.hasNext()) {

            XMLEvent nextEvent = reader.peek();
            if (nextEvent.isStartElement() || nextEvent.isEndElement()) {

                return nextEvent;

            } else {

                // eat the event.
                reader.nextEvent();

            }

        }

        return null;

    }

    /**
     * Reads the events from the provided event stream until either a start or
     * end tag is encountered. In the former case, the start tag will be
     * returned, but if an end tag is encountered, null will be
     * returned. After returning, the stream will be positioned just before the
     * returned start element. The start element will not be consumed by this
     * method.
     * 
     * @param reader The event stream from which to read.
     * @return The StartElement read from the stream, or null if
     *         an end tag was found first, or the stream ended before a start
     *         element was found.
     * @throws XMLStreamException If an error occurs reading the stream.
     */
    public static final StartElement nextElement(XMLEventReader reader)
            throws XMLStreamException {

        return nextElement(reader, null);

    }

    /**
     * Reads the events from the provided event stream until either a start or
     * end tag is encountered. In the former case, the start tag will be
     * returned if it matches the specified QName, but if it doesn't match, an
     * end tag is encountered, or the stream ends, null will be
     * returned. After returning, the stream will be positioned just before the
     * start element. The start element will not be consumed by this method.
     * 
     * @param reader The event stream from which to read.
     * @param name The name of the element to read, or null to
     *            read any start tag.
     * @return The StartElement read from the stream, or null if
     *         the encountered start tag didn't match the specified QName, an
     *         end tag was found first, or the stream ended before a start
     *         element was found.
     * @throws XMLStreamException If an error occurs reading the stream.
     */
    public static final StartElement nextElement(XMLEventReader reader,
            QName name) throws XMLStreamException {

        while (reader.hasNext()) {

            XMLEvent nextEvent = reader.peek();
            if (nextEvent.isStartElement()) {

                StartElement start = nextEvent.asStartElement();
                if (name == null || start.getName().equals(name)) {

                    return start;

                } else {

                    break;

                }

            } else if (nextEvent.isEndElement()) {

                break;

            } else {

                // consume the event.
                reader.nextEvent();

            }

        }

        return null;

    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy