package org.infinispan.configuration.serializing; import java.lang.reflect.UndeclaredThrowableException; import java.util.ArrayDeque; import java.util.Iterator; import javax.xml.namespace.NamespaceContext; import javax.xml.stream.XMLStreamConstants; import javax.xml.stream.XMLStreamException; import javax.xml.stream.XMLStreamWriter; /** * XMLExtendedStreamWriterImpl. * * @author David Lloyd * @since 9.0 */ public final class XMLExtendedStreamWriterImpl implements XMLExtendedStreamWriter, XMLStreamConstants { private static final String NO_NAMESPACE = new String(); private final XMLStreamWriter delegate; private final ArrayDeque<ArgRunnable> attrQueue = new ArrayDeque<>(); private int level; private int state = START_DOCUMENT; private boolean indentEndElement = false; private ArrayDeque<String> unspecifiedNamespaces = new ArrayDeque<>(); public XMLExtendedStreamWriterImpl(final XMLStreamWriter delegate) { this.delegate = delegate; unspecifiedNamespaces.push(NO_NAMESPACE); } private void nl() throws XMLStreamException { delegate.writeCharacters("\n"); } private void indent() throws XMLStreamException { int level = this.level; final XMLStreamWriter delegate = this.delegate; for (int i = 0; i < level; i++) { delegate.writeCharacters(" "); } } public interface ArgRunnable { public void run(int arg) throws XMLStreamException; } @Override public void setUnspecifiedElementNamespace(final String namespace) { ArrayDeque<String> namespaces = this.unspecifiedNamespaces; namespaces.pop(); namespaces.push(namespace == null ? NO_NAMESPACE : namespace); } private String nestUnspecifiedNamespace() { ArrayDeque<String> namespaces = unspecifiedNamespaces; String clone = namespaces.getFirst(); namespaces.push(clone); return clone; } @Override public void writeStartElement(final String localName) throws XMLStreamException { ArrayDeque<String> namespaces = unspecifiedNamespaces; String namespace = namespaces.getFirst(); if (namespace != NO_NAMESPACE) { writeStartElement(namespace, localName); return; } unspecifiedNamespaces.push(namespace); // If this is a nested element flush the outer runAttrQueue(); nl(); indent(); attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { if (arg == 0) { delegate.writeStartElement(localName); } else { delegate.writeEmptyElement(localName); } } }); level++; state = START_ELEMENT; indentEndElement = false; } @Override public void writeStartElement(final String namespaceURI, final String localName) throws XMLStreamException { nestUnspecifiedNamespace(); // If this is a nested element flush the outer runAttrQueue(); nl(); indent(); attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { if (arg == 0) { delegate.writeStartElement(namespaceURI, localName); } else { delegate.writeEmptyElement(namespaceURI, localName); } } }); level++; state = START_ELEMENT; indentEndElement = false; } @Override public void writeStartElement(Enum<?> localNameEnum) throws XMLStreamException { writeStartElement(localNameEnum.toString()); } @Override public void writeStartElement(final String prefix, final String localName, final String namespaceURI) throws XMLStreamException { nestUnspecifiedNamespace(); // If this is a nested element flush the outer runAttrQueue(); nl(); indent(); attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { if (arg == 0) { delegate.writeStartElement(prefix, namespaceURI, localName); } else { delegate.writeEmptyElement(prefix, namespaceURI, localName); } } }); level++; state = START_ELEMENT; indentEndElement = false; } @Override public void writeEmptyElement(final String namespaceURI, final String localName) throws XMLStreamException { runAttrQueue(); nl(); indent(); delegate.writeEmptyElement(namespaceURI, localName); state = END_ELEMENT; } @Override public void writeEmptyElement(final String prefix, final String localName, final String namespaceURI) throws XMLStreamException { runAttrQueue(); nl(); indent(); delegate.writeEmptyElement(prefix, namespaceURI, localName); state = END_ELEMENT; } @Override public void writeEmptyElement(final String localName) throws XMLStreamException { String namespace = unspecifiedNamespaces.getFirst(); if (namespace != NO_NAMESPACE) { writeEmptyElement(namespace, localName); return; } runAttrQueue(); nl(); indent(); delegate.writeEmptyElement(localName); state = END_ELEMENT; } @Override public void writeEndElement() throws XMLStreamException { level--; if (state != START_ELEMENT) { runAttrQueue(); if (state != CHARACTERS || indentEndElement) { nl(); indent(); indentEndElement = false; } delegate.writeEndElement(); } else { // Change the start element to an empty element ArgRunnable start = attrQueue.poll(); start.run(1); // Write everything else runAttrQueue(); } unspecifiedNamespaces.pop(); state = END_ELEMENT; } private void runAttrQueue() throws XMLStreamException { ArgRunnable attr; while ((attr = attrQueue.poll()) != null) { attr.run(0); } } @Override public void writeEndDocument() throws XMLStreamException { delegate.writeEndDocument(); state = END_DOCUMENT; } @Override public void close() throws XMLStreamException { delegate.close(); state = END_DOCUMENT; } @Override public void flush() throws XMLStreamException { delegate.flush(); } @Override public void writeAttribute(final String localName, final String value) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { try { delegate.writeAttribute(localName, value); } catch (XMLStreamException e) { throw new UndeclaredThrowableException(e); } } }); } @Override public void writeAttribute(Enum<?> name, String property) throws XMLStreamException { writeAttribute(name.toString(), property); } @Override public void writeAttribute(final String prefix, final String namespaceURI, final String localName, final String value) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeAttribute(prefix, namespaceURI, localName, value); } }); } @Override public void writeAttribute(final String namespaceURI, final String localName, final String value) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeAttribute(namespaceURI, localName, value); } }); } @Override public void writeAttribute(final String localName, final String[] values) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeAttribute(localName, join(values)); } }); } @Override public void writeAttribute(final String prefix, final String namespaceURI, final String localName, final String[] values) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeAttribute(prefix, namespaceURI, localName, join(values)); } }); } @Override public void writeAttribute(final String namespaceURI, final String localName, final String[] values) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeAttribute(namespaceURI, localName, join(values)); } }); } @Override public void writeAttribute(final String localName, final Iterable<String> values) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeAttribute(localName, join(values)); } }); } @Override public void writeAttribute(final String prefix, final String namespaceURI, final String localName, final Iterable<String> values) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeAttribute(prefix, namespaceURI, localName, join(values)); } }); } @Override public void writeAttribute(final String namespaceURI, final String localName, final Iterable<String> values) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeAttribute(namespaceURI, localName, join(values)); } }); } @Override public void writeNamespace(final String prefix, final String namespaceURI) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeNamespace(prefix, namespaceURI); } }); } @Override public void writeDefaultNamespace(final String namespaceURI) throws XMLStreamException { attrQueue.add(new ArgRunnable() { @Override public void run(int arg) throws XMLStreamException { delegate.writeDefaultNamespace(namespaceURI); } }); } @Override public void writeProcessingInstruction(final String target) throws XMLStreamException { runAttrQueue(); nl(); indent(); delegate.writeProcessingInstruction(target); state = PROCESSING_INSTRUCTION; } @Override public void writeProcessingInstruction(final String target, final String data) throws XMLStreamException { runAttrQueue(); nl(); indent(); delegate.writeProcessingInstruction(target, data); state = PROCESSING_INSTRUCTION; } @Override public void writeCData(final String data) throws XMLStreamException { runAttrQueue(); delegate.writeCData(data); state = CDATA; } @Override public void writeDTD(final String dtd) throws XMLStreamException { nl(); indent(); delegate.writeDTD(dtd); state = DTD; } @Override public void writeEntityRef(final String name) throws XMLStreamException { runAttrQueue(); delegate.writeEntityRef(name); state = ENTITY_REFERENCE; } @Override public void writeStartDocument() throws XMLStreamException { delegate.writeStartDocument(); nl(); state = START_DOCUMENT; } @Override public void writeStartDocument(final String version) throws XMLStreamException { delegate.writeStartDocument(version); nl(); state = START_DOCUMENT; } @Override public void writeStartDocument(final String encoding, final String version) throws XMLStreamException { delegate.writeStartDocument(encoding, version); nl(); state = START_DOCUMENT; } @Override public void writeCharacters(final String text) throws XMLStreamException { runAttrQueue(); if (state != CHARACTERS) { nl(); indent(); } String[] split = text.split("\\n"); for(int i = 0; i < split.length; i++) { delegate.writeCharacters(split[i]); if (i < split.length - 1) { nl(); indent(); } } state = CHARACTERS; indentEndElement = true; } @Override public void writeCharacters(final char[] text, final int start, final int len) throws XMLStreamException { runAttrQueue(); delegate.writeCharacters(text, start, len); state = CHARACTERS; } @Override public String getPrefix(final String uri) throws XMLStreamException { return delegate.getPrefix(uri); } @Override public void setPrefix(final String prefix, final String uri) throws XMLStreamException { delegate.setPrefix(prefix, uri); } @Override public void setDefaultNamespace(final String uri) throws XMLStreamException { runAttrQueue(); delegate.setDefaultNamespace(uri); } @Override public void setNamespaceContext(final NamespaceContext context) throws XMLStreamException { delegate.setNamespaceContext(context); } @Override public NamespaceContext getNamespaceContext() { return delegate.getNamespaceContext(); } @Override public Object getProperty(final String name) throws IllegalArgumentException { return delegate.getProperty(name); } private static String join(final String[] values) { final StringBuilder b = new StringBuilder(); for (int i = 0, valuesLength = values.length; i < valuesLength; i++) { final String s = values[i]; if (s != null) { if (i > 0) { b.append(' '); } b.append(s); } } return b.toString(); } private static String join(final Iterable<String> values) { final StringBuilder b = new StringBuilder(); Iterator<String> iterator = values.iterator(); while (iterator.hasNext()) { final String s = iterator.next(); if (s != null) { b.append(s); if (iterator.hasNext()) b.append(' '); } } return b.toString(); } @Override public void writeComment(String data) throws XMLStreamException { // No need to support comments at the moment } }