Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.ibm.fhir.model.util.XMLSupport Maven / Gradle / Ivy
/*
* (C) Copyright IBM Corp. 2019, 2020
*
* SPDX-License-Identifier: Apache-2.0
*/
package com.ibm.fhir.model.util;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.Reader;
import java.io.StringWriter;
import java.io.Writer;
import javax.xml.XMLConstants;
import javax.xml.namespace.NamespaceContext;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLOutputFactory;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.XMLStreamReader;
import javax.xml.stream.XMLStreamWriter;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerConfigurationException;
import javax.xml.transform.TransformerFactory;
public final class XMLSupport {
public static final String FHIR_NS_URI = "http://hl7.org/fhir";
public static final String XHTML_NS_URI = "http://www.w3.org/1999/xhtml";
private static final String XML_INPUT_FACTORY_IMPL = "com.sun.xml.internal.stream.XMLInputFactoryImpl";
private static final String XML_OUTPUT_FACTORY_IMPL = "com.sun.xml.internal.stream.XMLOutputFactoryImpl";
private static final String TRANSFORMER_FACTORY_IMPL = "com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl";
private static final XMLInputFactory XML_INPUT_FACTORY = createXMLInputFactory();
private static final XMLOutputFactory XML_OUTPUT_FACTORY = createXMLOutputFactory();
private static final TransformerFactory TRANSFORMER_FACTORY = createTransformerFactory();
private XMLSupport() { }
/**
* Calling this method allows us to load/initialize this class during startup.
*/
public static void init() { }
/**
* Checks the order of the current element using its position relative to the position
* of the previous element
*
* @param elementName
* the name of the current element
* @param current
* the position of the current element in the sequence
* @param previous
* the position of the previous element in the sequence
* @param repeating
* true if the element is allowed to repeat
* @return
* the position of the current element
* @throws
* IllegalArgumentException if the element is out of order or if the parser has already
* seen the element and it is not allowed to repeat
*/
public static int checkElementOrder(String elementName, int current, int previous, boolean repeating) {
if (current > previous) {
return current;
} else if (current == previous) {
if (repeating) {
return current;
}
throw new IllegalArgumentException("Element: '" + elementName + "' is not allowed to repeat");
} else {
throw new IllegalArgumentException("Element: '" + elementName + "' is out of order");
}
}
public static StreamReaderDelegate createStreamReaderDelegate(InputStream in) throws XMLStreamException {
return new StreamReaderDelegate(createXMLStreamReader(in));
}
public static StreamReaderDelegate createStreamReaderDelegate(Reader reader) throws XMLStreamException {
return new StreamReaderDelegate(createXMLStreamReader(reader));
}
public static StreamWriterDelegate createStreamWriterDelegate(OutputStream out) throws XMLStreamException {
return createStreamWriterDelegate(createXMLStreamWriter(out));
}
public static StreamWriterDelegate createStreamWriterDelegate(Writer writer) throws XMLStreamException {
return createStreamWriterDelegate(createXMLStreamWriter(writer));
}
public static XMLStreamReader createXMLStreamReader(InputStream in) throws XMLStreamException {
return XML_INPUT_FACTORY.createXMLStreamReader(in, "UTF-8");
}
public static XMLStreamReader createXMLStreamReader(Reader reader) throws XMLStreamException {
return XML_INPUT_FACTORY.createXMLStreamReader(reader);
}
public static XMLStreamWriter createXMLStreamWriter(OutputStream out) throws XMLStreamException {
return XML_OUTPUT_FACTORY.createXMLStreamWriter(out, "UTF-8");
}
public static XMLStreamWriter createXMLStreamWriter(Writer writer) throws XMLStreamException {
return XML_OUTPUT_FACTORY.createXMLStreamWriter(writer);
}
public static boolean isResourceContainer(String elementName) {
return "contained".equals(elementName) ||
"resource".equals(elementName) ||
"outcome".equals(elementName);
}
public static String parseDiv(XMLStreamReader reader) throws XMLStreamException {
int depth = 0;
StringWriter sw = new StringWriter();
XMLStreamWriter writer = createStreamWriterDelegate(sw);
writeStartElement(reader, writer, depth);
depth++;
while (reader.hasNext()) {
int eventType = reader.next();
switch (eventType) {
case XMLStreamReader.START_ELEMENT:
requireNamespace(reader, XHTML_NS_URI);
writeStartElement(reader, writer, depth);
depth++;
break;
case XMLStreamReader.SPACE:
case XMLStreamReader.CHARACTERS:
writer.writeCharacters(reader.getTextCharacters(), reader.getTextStart(), reader.getTextLength());
break;
case XMLStreamReader.END_ELEMENT:
depth--;
writer.writeEndElement();
if ("div".equals(reader.getLocalName()) && depth == 0) {
writer.flush();
writer.close();
return sw.toString();
}
break;
}
}
throw new XMLStreamException("Unexpected end of stream");
}
public static void requireNamespace(XMLStreamReader reader, String namespaceURI) throws XMLStreamException {
reader.require(XMLStreamReader.START_ELEMENT, namespaceURI, null);
}
private static StreamWriterDelegate createStreamWriterDelegate(XMLStreamWriter writer) {
return new StreamWriterDelegate(writer) {
@Override
public void writeEndDocument() {
// do nothing
}
@Override
public void writeStartDocument() throws XMLStreamException {
// do nothing
}
@Override
public void writeStartDocument(String version) throws XMLStreamException {
// do nothing
}
@Override
public void writeStartDocument(String encoding, String version) throws XMLStreamException {
// do nothing
}
};
}
public static StreamWriterDelegate createNonClosingStreamWriterDelegate(XMLStreamWriter writer) {
return new StreamWriterDelegate(writer) {
@Override
public void close() throws XMLStreamException {
// do nothing
}
};
}
private static XMLInputFactory createXMLInputFactory() {
try {
XMLInputFactory factory = (XMLInputFactory) Class.forName(XML_INPUT_FACTORY_IMPL).getConstructor().newInstance();
factory.setProperty(XMLInputFactory.SUPPORT_DTD, false);
factory.setProperty(XMLInputFactory.IS_SUPPORTING_EXTERNAL_ENTITIES, false);
return factory;
} catch (Exception e) {
throw new Error(e);
}
}
private static XMLOutputFactory createXMLOutputFactory() {
try {
return (XMLOutputFactory) Class.forName(XML_OUTPUT_FACTORY_IMPL).getConstructor().newInstance();
} catch (Exception e) {
throw new Error(e);
}
}
private static TransformerFactory createTransformerFactory() {
try {
TransformerFactory factory = (TransformerFactory) Class.forName(TRANSFORMER_FACTORY_IMPL).getConstructor().newInstance();
factory.setFeature(XMLConstants.FEATURE_SECURE_PROCESSING, true);
return factory;
} catch (Exception e) {
throw new Error(e);
}
}
public static Transformer createTransformer() {
try {
return TRANSFORMER_FACTORY.newTransformer();
} catch (TransformerConfigurationException e) {
throw new Error(e);
}
}
private static void writeAttributes(XMLStreamReader reader, XMLStreamWriter writer) throws XMLStreamException {
for (int i = 0; i < reader.getAttributeCount(); i++) {
String prefix = reader.getAttributePrefix(i);
String namespaceURI = reader.getAttributeNamespace(i);
String localName = reader.getAttributeLocalName(i);
String value = reader.getAttributeValue(i);
if (namespaceURI != null) {
if (prefix != null) {
writer.writeAttribute(prefix, namespaceURI, localName, value);
} else {
writer.writeAttribute(namespaceURI, localName, value);
}
} else {
writer.writeAttribute(localName, value);
}
}
}
private static void writeNamespaces(XMLStreamReader reader, XMLStreamWriter writer, int depth) throws XMLStreamException {
if (depth == 0) {
writer.writeDefaultNamespace(XHTML_NS_URI);
}
for (int i = 0; i < reader.getNamespaceCount(); i++) {
if (XHTML_NS_URI.equals(reader.getNamespaceURI(i))) {
continue;
}
writer.writeNamespace(reader.getNamespacePrefix(i), reader.getNamespaceURI(i));
}
}
private static void writeStartElement(XMLStreamReader reader, XMLStreamWriter writer, int depth) throws XMLStreamException {
String localName = reader.getLocalName();
writer.setDefaultNamespace(XHTML_NS_URI);
writer.writeStartElement(XHTML_NS_URI, localName);
writeNamespaces(reader, writer, depth);
writeAttributes(reader, writer);
}
/**
* Simple subclass of {@link javax.xml.stream.util.StreamReaderDelegate} to make it AutoCloseable
*/
public static class StreamReaderDelegate extends javax.xml.stream.util.StreamReaderDelegate implements AutoCloseable {
public StreamReaderDelegate(XMLStreamReader reader) {
super(reader);
}
}
/**
* The corollary to {@link javax.xml.stream.util.StreamReaderDelegate}.
*
* This class is designed to sit between an XMLStreamWriter and an
* application's XMLStreamWriter.
* By default each method does nothing but call the corresponding method
* on the parent interface.
*/
public static class StreamWriterDelegate implements XMLStreamWriter, AutoCloseable {
protected final XMLStreamWriter writer;
public StreamWriterDelegate(XMLStreamWriter writer) {
this.writer = writer;
}
/**
* @return the underlying writer being delegated to
*/
public XMLStreamWriter getWriter() {
return writer;
}
@Override
public void close() throws XMLStreamException {
writer.close();
}
@Override
public void flush() throws XMLStreamException {
writer.flush();
}
@Override
public NamespaceContext getNamespaceContext() {
return writer.getNamespaceContext();
}
@Override
public java.lang.String getPrefix(java.lang.String uri) throws XMLStreamException {
return writer.getPrefix(uri);
}
@Override
public Object getProperty(java.lang.String name) throws IllegalArgumentException {
return writer.getProperty(name);
}
@Override
public void setDefaultNamespace(java.lang.String uri) throws XMLStreamException {
writer.setDefaultNamespace(uri);
}
@Override
public void setNamespaceContext(NamespaceContext context) throws XMLStreamException {
writer.setNamespaceContext(context);
}
@Override
public void setPrefix(java.lang.String prefix, java.lang.String uri) throws XMLStreamException {
writer.setPrefix(prefix, uri);
}
@Override
public void writeAttribute(java.lang.String localName, java.lang.String value) throws XMLStreamException {
writer.writeAttribute(localName, value);
}
@Override
public void writeAttribute(java.lang.String namespaceURI, java.lang.String localName, java.lang.String value) throws XMLStreamException {
writer.writeAttribute(namespaceURI, localName, value);
}
@Override
public void writeAttribute(java.lang.String prefix, java.lang.String namespaceURI, java.lang.String localName, java.lang.String value) throws XMLStreamException {
writer.writeAttribute(prefix, namespaceURI, localName, value);
}
@Override
public void writeCData(java.lang.String data) throws XMLStreamException {
writer.writeCData(data);
}
@Override
public void writeCharacters(char[] text, int start, int len) throws XMLStreamException {
writer.writeCharacters(text, start, len);
}
@Override
public void writeCharacters(java.lang.String text) throws XMLStreamException {
writer.writeCharacters(text);
}
@Override
public void writeComment(java.lang.String data) throws XMLStreamException {
writer.writeComment(data);
}
@Override
public void writeDefaultNamespace(java.lang.String namespaceURI) throws XMLStreamException {
writer.writeDefaultNamespace(namespaceURI);
}
@Override
public void writeDTD(java.lang.String dtd) throws XMLStreamException {
writer.writeDTD(dtd);
}
@Override
public void writeEmptyElement(java.lang.String localName) throws XMLStreamException {
writer.writeEmptyElement(localName);
}
@Override
public void writeEmptyElement(java.lang.String namespaceURI, java.lang.String localName) throws XMLStreamException {
writer.writeEmptyElement(namespaceURI, localName);
}
@Override
public void writeEmptyElement(java.lang.String prefix, java.lang.String localName, java.lang.String namespaceURI) throws XMLStreamException {
writer.writeEmptyElement(prefix, localName, namespaceURI);
}
@Override
public void writeEndDocument() throws XMLStreamException {
writer.writeEndDocument();
}
@Override
public void writeEndElement() throws XMLStreamException {
writer.writeEndElement();
}
@Override
public void writeEntityRef(java.lang.String name) throws XMLStreamException {
writer.writeEntityRef(name);
}
@Override
public void writeNamespace(java.lang.String prefix, java.lang.String namespaceURI) throws XMLStreamException {
writer.writeNamespace(prefix, namespaceURI);
}
@Override
public void writeProcessingInstruction(java.lang.String target) throws XMLStreamException {
writer.writeProcessingInstruction(target);
}
@Override
public void writeProcessingInstruction(java.lang.String target, java.lang.String data) throws XMLStreamException {
writer.writeProcessingInstruction(target, data);
}
@Override
public void writeStartDocument() throws XMLStreamException {
writer.writeStartDocument();
}
@Override
public void writeStartDocument(java.lang.String version) throws XMLStreamException {
writer.writeStartDocument(version);
}
@Override
public void writeStartDocument(java.lang.String encoding, java.lang.String version) throws XMLStreamException {
writer.writeStartDocument(encoding, version);
}
@Override
public void writeStartElement(java.lang.String localName) throws XMLStreamException {
writer.writeStartElement(localName);
}
@Override
public void writeStartElement(java.lang.String namespaceURI, java.lang.String localName) throws XMLStreamException {
writer.writeStartElement(namespaceURI, localName);
}
@Override
public void writeStartElement(java.lang.String prefix, java.lang.String localName, java.lang.String namespaceURI) throws XMLStreamException {
writer.writeStartElement(prefix, localName, namespaceURI);
}
}
}