
com.marklogic.client.io.XMLEventReaderHandle Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of marklogic-client-api Show documentation
Show all versions of marklogic-client-api Show documentation
The official MarkLogic Java client API.
The newest version!
/*
* Copyright © 2024 MarkLogic Corporation. All Rights Reserved.
*/
package com.marklogic.client.io;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.Closeable;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets;
import javax.xml.stream.FactoryConfigurationError;
import javax.xml.stream.XMLEventReader;
import javax.xml.stream.XMLEventWriter;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLOutputFactory;
import javax.xml.stream.XMLResolver;
import javax.xml.stream.XMLStreamException;
import com.marklogic.client.io.marker.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.marklogic.client.MarkLogicIOException;
import com.marklogic.client.MarkLogicInternalException;
/**
* An XML Event Reader Handle represents XML content as an XML event reader
* for reading as a series of StAX events.
*
* Either call {@link #close} or {@link #get}.close() when finished with this handle
* to release the resources.
*/
public class XMLEventReaderHandle
extends BaseHandle
implements OutputStreamSender, StreamingContentHandle,
XMLReadHandle, XMLWriteHandle,
StructureReadHandle, StructureWriteHandle, CtsQueryWriteHandle,
Closeable
{
static final private Logger logger = LoggerFactory.getLogger(XMLEventReaderHandle.class);
private XMLResolver resolver;
private XMLEventReader content;
private XMLInputFactory factory;
private InputStream underlyingStream;
/**
* Creates a factory to create an XMLEventReaderHandle instance for a StAX event reader.
* @return the factory
*/
static public ContentHandleFactory newFactory() {
return new ContentHandleFactory() {
@Override
public Class>[] getHandledClasses() {
return new Class>[]{ XMLEventReader.class };
}
@Override
public boolean isHandled(Class> type) {
return XMLEventReader.class.isAssignableFrom(type);
}
@Override
public ContentHandle newHandle(Class type) {
@SuppressWarnings("unchecked")
ContentHandle handle = isHandled(type) ?
(ContentHandle) new XMLEventReaderHandle() : null;
return handle;
}
};
}
/**
* Zero-argument constructor.
*/
public XMLEventReaderHandle() {
super();
super.setFormat(Format.XML);
setResendable(false);
}
/**
* Initializes the handle with a StAX event reader for the content.
* @param content a StAX event reader
*/
public XMLEventReaderHandle(XMLEventReader content) {
this();
set(content);
}
/**
* Returns the resolver for resolving references while parsing
* the event reader source.
* @return the resolver
*/
public XMLResolver getResolver() {
return resolver;
}
/**
* Specifies the resolver for resolving references while parsing
* the event reader source.
* @param resolver the reference resolver
*/
public void setResolver(XMLResolver resolver) {
this.resolver = resolver;
}
/**
* Returns an XML Event Reader reading a resource from the database
* as a series of StAX events.
*
* When finished with the event reader, close the event reader to release
* the response.
*
* @return the XML event reader
*/
@Override
public XMLEventReader get() {
return content;
}
/**
* Assigns the event reader for the content.
* @param content a StAX event reader
*/
@Override
public void set(XMLEventReader content) {
this.content = content;
}
/**
* Assigns an event reader for the content and returns the handle
* as a fluent convenience.
* @param content a StAX event reader
* @return this handle
*/
public XMLEventReaderHandle with(XMLEventReader content) {
set(content);
return this;
}
@Override
public Class getContentClass() {
return XMLEventReader.class;
}
@Override
public XMLEventReaderHandle newHandle() {
return new XMLEventReaderHandle().withMimetype(getMimetype());
}
@Override
public XMLEventReaderHandle[] newHandleArray(int length) {
if (length < 0) throw new IllegalArgumentException("array length less than zero: "+length);
return new XMLEventReaderHandle[length];
}
@Override
public XMLEventReader[] newArray(int length) {
if (length < 0) throw new IllegalArgumentException("array length less than zero: "+length);
return new XMLEventReader[length];
}
/**
* Restricts the format to XML.
*/
@Override
public void setFormat(Format format) {
if (format != Format.XML)
throw new IllegalArgumentException("XMLEventReaderHandle supports the XML format only");
}
/**
* Specifies the mime type of the content and returns the handle
* as a fluent convenience.
* @param mimetype the mime type of the content
* @return this handle
*/
public XMLEventReaderHandle withMimetype(String mimetype) {
setMimetype(mimetype);
return this;
}
@Override
public void fromBuffer(byte[] buffer) {
if (buffer == null || buffer.length == 0)
content = null;
else
receiveContent(new ByteArrayInputStream(buffer));
}
@Override
public byte[] toBuffer() {
try {
if (content == null)
return null;
ByteArrayOutputStream buffer = new ByteArrayOutputStream();
write(buffer);
byte[] b = buffer.toByteArray();
fromBuffer(b);
return b;
} catch (IOException e) {
throw new MarkLogicIOException(e);
}
}
@Override
public XMLEventReader toContent(InputStream serialization) {
if (serialization == null) return null;
try {
XMLInputFactory factory = getFactory();
if (factory == null) {
throw new MarkLogicInternalException("Failed to make StAX input factory");
}
if (resolver != null)
factory.setXMLResolver(resolver);
return factory.createXMLEventReader(serialization, "UTF-8");
} catch (XMLStreamException e) {
logger.error("Failed to parse StAX events from input stream",e);
throw new MarkLogicInternalException(e);
} catch (FactoryConfigurationError e) {
logger.error("Failed to parse StAX events from input stream",e);
throw new MarkLogicInternalException(e);
}
}
@Override
public XMLEventReader bytesToContent(byte[] buffer) {
return (buffer == null || buffer.length == 0) ? null :
toContent(new ByteArrayInputStream(buffer));
}
@Override
public byte[] contentToBytes(XMLEventReader content) {
if (content == null) return null;
ByteArrayOutputStream buffer = new ByteArrayOutputStream();
write(content, buffer);
return buffer.toByteArray();
}
/**
* Buffers the StAX event source and returns the buffer
* as an XML string.
*/
@Override
public String toString() {
byte[] buffer = toBuffer();
return (buffer == null) ? null : new String(buffer, StandardCharsets.UTF_8);
}
/**
* Returns the factory for parsing StAX events.
* @return the StAX factory
*/
public XMLInputFactory getFactory() {
if (factory == null)
factory = makeXMLInputFactory();
return factory;
}
/**
* Specifies the factory for parsing StAX events.
* @param factory the StAX factory
*/
public void setFactory(XMLInputFactory factory) {
this.factory = factory;
}
protected XMLInputFactory makeXMLInputFactory() {
XMLInputFactory factory = XMLInputFactory.newFactory();
// default to best practices for conservative security including recommendations per
// https://github.com/OWASP/CheatSheetSeries/blob/master/cheatsheets/XML_External_Entity_Prevention_Cheat_Sheet.md
try {
factory.setProperty(XMLInputFactory.SUPPORT_DTD, false);
} catch (IllegalArgumentException e) {}
try {
factory.setProperty(XMLInputFactory.IS_SUPPORTING_EXTERNAL_ENTITIES, false);
} catch (IllegalArgumentException e) {}
try {
factory.setProperty(XMLInputFactory.IS_REPLACING_ENTITY_REFERENCES, false);
} catch (IllegalArgumentException e) {}
try {
factory.setProperty(XMLInputFactory.IS_NAMESPACE_AWARE, true);
} catch (IllegalArgumentException e) {}
try {
factory.setProperty(XMLInputFactory.IS_VALIDATING, false);
} catch (IllegalArgumentException e) {}
return factory;
}
@Override
protected Class receiveAs() {
return InputStream.class;
}
@Override
protected void receiveContent(InputStream content) {
if (content == null) {
this.content = null;
return;
}
this.underlyingStream = content;
if (logger.isInfoEnabled())
logger.info("Parsing StAX events from input stream");
this.content = toContent(content);
}
@Override
protected OutputStreamSender sendContent() {
if (content == null) {
throw new IllegalStateException("No input source to write");
}
return this;
}
@Override
public void write(OutputStream out) throws IOException {
write(this.content, out);
}
private void write(XMLEventReader content, OutputStream out) {
try {
XMLOutputFactory factory = XMLOutputFactory.newFactory();
XMLEventWriter writer = factory.createXMLEventWriter(out, "UTF-8");
writer.add(content);
writer.flush();
writer.close();
content.close();
} catch (XMLStreamException e) {
logger.error("Failed to parse StAX events from input stream",e);
throw new MarkLogicInternalException(e);
}
}
/** Either call close() or get().close() when finished with this handle to close the underlying InputStream.
*/
@Override
public void close() {
if ( underlyingStream != null ) {
try {
underlyingStream.close();
} catch (IOException e) {
logger.error("Failed to close underlying InputStream",e);
throw new MarkLogicIOException(e);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy