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

org.apache.jackrabbit.commons.xml.ToXmlContentHandler Maven / Gradle / Ivy

/*
 * 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.jackrabbit.commons.xml;

import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.StringWriter;
import java.io.UnsupportedEncodingException;
import java.io.Writer;
import java.nio.charset.StandardCharsets;

import org.xml.sax.Attributes;
import org.xml.sax.ContentHandler;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;

/**
 * Simple XML serializer. This content handler serializes the received
 * SAX events as XML to a given {@link Writer} or {@link OutputStream}.
 * The serialization assumes that the incoming SAX events are well-formed,
 * i.e. that all elements are properly nested, that element and attribute
 * names are valid and that no invalid XML characters are included. Assuming
 * these preconditions are met, the result will be a well-formed XML stream.
 * 

* This serializer does not have any special support for namespaces. For * example, namespace prefixes are declared in the resulting XML stream * if and only if the corresponding "xmlns" attributes are explicitly * included in the {@link Attributes} instances passed in * {@link #startElement(String, String, String, Attributes)} calls. *

* As a convenience this class inherits the {@link DefaultHandler} class * instead of just the {@link ContentHandler} interface. This makes it * easier to pass instances of this class to methods like * {@link javax.xml.parsers.SAXParser#parse(String, DefaultHandler)} that * expect a DefaultHandler instance instead of a ContentHandler. */ public class ToXmlContentHandler extends DefaultHandler { /** * The XML stream. */ private final Writer writer; /** * The data part of the <?xml?> processing instruction included * at the beginning of the XML stream. */ private final String declaration; /** * Flag variable that is used to track whether a start tag has had it's * closing ">" appended. Set to true by the * {@link #startElement(String, String, String, Attributes)} method that * does not output the closing ">". If this flag is still set * when the {#link {@link #endElement(String, String, String)}} method * is called, then the method knows that the element is empty and can * close it with "/>". Any other SAX event will cause the open start tag * to be closed with a normal ">". * * @see #closeStartTagIfOpen() */ private boolean startTagIsOpen = false; //--------------------------------------------------------< constructors > /** * Creates an XML serializer that writes the serialized XML stream * to the given output stream using the given character encoding. * * @param stream XML output stream * @param encoding character encoding * @throws UnsupportedEncodingException if the encoding is not supported */ public ToXmlContentHandler(OutputStream stream, String encoding) throws UnsupportedEncodingException { this.writer = new OutputStreamWriter(stream, encoding); this.declaration = "version=\"1.0\" encoding=\"" + encoding + "\""; } /** * Creates an XML serializer that writes the serialized XML stream * to the given output stream using the UTF-8 character encoding. * * @param stream XML output stream */ public ToXmlContentHandler(OutputStream stream) { this.writer = new OutputStreamWriter(stream, StandardCharsets.UTF_8); this.declaration = "version=\"1.0\" encoding=\"UTF-8\""; } /** * Creates an XML serializer that writes the serialized XML stream * to the given writer. * * @param writer XML output stream */ public ToXmlContentHandler(Writer writer) { this.writer = writer; this.declaration = "version=\"1.0\""; } /** * Creates an XML serializer that writes the serialized XML stream * to an internal buffer. Use the {@link #toString()} method to access * the serialized XML document. */ public ToXmlContentHandler() { this(new StringWriter()); } //-------------------------------------------------------------< private > private void write(char[] ch, int start, int length, boolean attribute) throws SAXException { for (int i = start; i < start + length; i++) { try { if (ch[i] == '>') { writer.write(">"); } else if (ch[i] == '<') { writer.write("<"); } else if (ch[i] == '&') { writer.write("&"); } else if (attribute && ch[i] == '"') { writer.write("""); } else if (attribute && ch[i] == '\'') { writer.write("'"); } else { writer.write(ch[i]); } } catch (IOException e) { throw new SAXException( "Failed to output XML character: " + ch[i], e); } } } private void closeStartTagIfOpen() throws SAXException { if (startTagIsOpen) { try { writer.write(">"); } catch (IOException e) { throw new SAXException( "Failed to output XML bracket: >", e); } startTagIsOpen = false; } } //------------------------------------------------------< ContentHandler > /** * Starts the XML serialization by outputting the <?xml?> header. */ public void startDocument() throws SAXException { processingInstruction("xml", declaration); } /** * Ends the XML serialization by flushing the output stream. */ public void endDocument() throws SAXException { try { writer.flush(); } catch (IOException e) { throw new SAXException("Failed to flush XML output", e); } } /** * Serializes a processing instruction. */ public void processingInstruction(String target, String data) throws SAXException { closeStartTagIfOpen(); try { writer.write(""); } catch (IOException e) { throw new SAXException( "Failed to output XML processing instruction: " + target, e); } } /** * Outputs the specified start tag with the given attributes. */ public void startElement( String namespaceURI, String localName, String qName, Attributes atts) throws SAXException { closeStartTagIfOpen(); try { writer.write("<"); writer.write(qName); for (int i = 0; i < atts.getLength(); i++) { writer.write(" "); writer.write(atts.getQName(i)); writer.write("=\""); char[] ch = atts.getValue(i).toCharArray(); write(ch, 0, ch.length, true); writer.write("\""); } startTagIsOpen = true; } catch (IOException e) { throw new SAXException( "Failed to output XML end tag: " + qName, e); } } /** * Escapes and outputs the given characters. */ public void characters(char[] ch, int start, int length) throws SAXException { closeStartTagIfOpen(); write(ch, start, length, false); } /** * Escapes and outputs the given characters. */ public void ignorableWhitespace(char[] ch, int start, int length) throws SAXException { characters(ch, start, length); } /** * Outputs the specified end tag. */ public void endElement( String namespaceURI, String localName, String qName) throws SAXException { try { if (startTagIsOpen) { writer.write("/>"); startTagIsOpen = false; } else { writer.write(""); } } catch (IOException e) { throw new SAXException( "Failed to output XML end tag: " + qName, e); } } //--------------------------------------------------------------< Object > /** * Returns the serialized XML document (assuming the default no-argument * constructor was used). * * @return serialized XML document */ public String toString() { return writer.toString(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy