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

org.exist.xmldb.RemoteXMLResource Maven / Gradle / Ivy

/*
 * eXist Open Source Native XML Database
 * Copyright (C) 2001-2019 The eXist Project
 * http://exist-db.org
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program; if not, write to the Free Software Foundation
 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 */
package org.exist.xmldb;

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.apache.xmlrpc.XmlRpcException;

import org.apache.xmlrpc.client.XmlRpcClient;
import org.exist.Namespaces;
import org.exist.dom.persistent.DocumentTypeImpl;
import org.exist.util.ExistSAXParserFactory;
import org.exist.util.Leasable;
import org.exist.util.MimeType;
import org.exist.util.io.TemporaryFileManager;
import org.exist.util.io.VirtualTempPath;
import org.exist.util.serializer.DOMSerializer;
import org.exist.util.serializer.SAXSerializer;
import org.exist.xquery.value.StringValue;

import org.w3c.dom.Document;
import org.w3c.dom.DocumentType;
import org.w3c.dom.Node;

import org.xml.sax.ContentHandler;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.SAXNotRecognizedException;
import org.xml.sax.SAXNotSupportedException;
import org.xml.sax.XMLReader;
import org.xml.sax.ext.LexicalHandler;
import org.xmldb.api.base.ErrorCodes;
import org.xmldb.api.base.XMLDBException;
import org.xmldb.api.modules.XMLResource;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import javax.xml.transform.TransformerException;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.StringReader;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;

import static java.nio.charset.StandardCharsets.UTF_8;
import static javax.xml.XMLConstants.FEATURE_SECURE_PROCESSING;

import java.util.Optional;

public class RemoteXMLResource
        extends AbstractRemoteResource
        implements XMLResource {

    protected final static Logger LOG = LogManager.getLogger(RemoteXMLResource.class);

    /**
     * Use external XMLReader to parse XML.
     */
    private XMLReader xmlReader = null;

    private final Optional id;
    private final Optional type;
    private final int handle;
    private int pos = -1;
    private String content = null;

    private Properties outputProperties = null;
    private LexicalHandler lexicalHandler = null;

    /**
     * Construct a remote XML Resource.
     *
     * @param xmlRpcClientLease the XML-RPC Client lease
     * @param parent the parent collection
     * @param docId the document if of the remote resource
     * @param id the id of the remote resource
     *
     * @throws XMLDBException if an error occurs during construction
     *
     * @deprecated Use {@link #RemoteXMLResource(Leasable.Lease, RemoteCollection, XmldbURI, Optional, Optional)}.
     */
    @Deprecated
    public RemoteXMLResource(final Leasable.Lease xmlRpcClientLease, final RemoteCollection parent, final XmldbURI docId, final Optional id)
            throws XMLDBException {
        this(xmlRpcClientLease, parent, -1, -1, docId, id, Optional.empty());
    }

    /**
     * Construct a remote XML Resource.
     *
     * @param xmlRpcClientLease the XML-RPC Client lease
     * @param parent the parent collection
     * @param docId the document if of the remote resource
     * @param id the id of the remote resource
     * @param type the type of the remote resource
     *
     * @throws XMLDBException if an error occurs during construction
     *
     * @deprecated Use {@link #RemoteXMLResource(Leasable.Lease, RemoteCollection, int, int, XmldbURI, Optional, Optional)}.
     */
    public RemoteXMLResource(final Leasable.Lease xmlRpcClientLease, final RemoteCollection parent, final XmldbURI docId, final Optional id, final Optional type)
            throws XMLDBException {
        this(xmlRpcClientLease, parent, -1, -1, docId, id, type);
    }

    /**
     * Construct a remote XML Resource.
     *
     * @param xmlRpcClientLease the XML-RPC Client lease
     * @param parent the parent collection
     * @param handle the handle to the remote resource
     * @param pos the position of the remote resource
     * @param docId the document if of the remote resource
     * @param id the id of the remote resource
     *
     * @throws XMLDBException if an error occurs during construction
     *
     * @deprecated Use {@link #RemoteXMLResource(Leasable.Lease, RemoteCollection, int, int, XmldbURI, Optional, Optional)}.
     */
    @Deprecated
    public RemoteXMLResource(
            final Leasable.Lease xmlRpcClientLease,
            final RemoteCollection parent,
            final int handle,
            final int pos,
            final XmldbURI docId,
            final Optional id) throws XMLDBException {
        this(xmlRpcClientLease, parent, handle, pos, docId, id, Optional.empty());
    }

    public RemoteXMLResource(
            final Leasable.Lease xmlRpcClientLease,
            final RemoteCollection parent,
            final int handle,
            final int pos,
            final XmldbURI docId,
            final Optional id,
            final Optional type)
            throws XMLDBException {
        super(xmlRpcClientLease, parent, docId, MimeType.XML_TYPE.getName());
        this.handle = handle;
        this.pos = pos;
        this.id = id;
        this.type = type;
    }

    @Override
    public String getId() throws XMLDBException {
        return id.map(x -> x.equals("1") ? getDocumentId() : getDocumentId() + '_' + id).orElse(getDocumentId());
    }

    @Override
    public String getResourceType() throws XMLDBException {
        return XMLResource.RESOURCE_TYPE;
    }

    @Override
    public Properties getProperties() {
        return outputProperties == null ? super.getProperties() : outputProperties;
    }

    @Override
    public void setProperties(final Properties properties) {
        this.outputProperties = properties;
    }

    @Override
    public String getDocumentId() {
        return path.lastSegment().toString();
    }

    @Override
    public Object getContent() throws XMLDBException {
        if (content != null) {
            return new StringValue(content).getStringValue(true);
        }
        final Object res = super.getContent();
        if (res != null) {
            if (res instanceof byte[]) {
                return new String((byte[]) res, UTF_8);

            } else {
                return res;
            }
        }
        return null;
    }

    @Override
    public Node getContentAsDOM() throws XMLDBException {
        final InputSource is;
        InputStream cis = null;

        try {
            if (content != null) {
                is = new InputSource(new StringReader(content));
            } else {
                cis = getStreamContent();
                is = new InputSource(cis);
            }

            final DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
            factory.setNamespaceAware(true);
            factory.setValidating(false);
            final DocumentBuilder builder = factory.newDocumentBuilder();
            final Document doc = builder.parse(is);

            final boolean isDocumentNode = type.map(t -> t.equals("document-node()")).orElse(true);
            if (isDocumentNode) {
                return doc;
            } else {
                return doc.getFirstChild();
            }
        } catch (final SAXException | IOException | ParserConfigurationException e) {
            throw new XMLDBException(ErrorCodes.VENDOR_ERROR, e.getMessage(), e);
        } finally {
            if (cis != null) {
                try {
                    cis.close();
                } catch (final IOException ioe) {
                    LOG.warn(ioe.getMessage(), ioe);
                }
            }
        }
    }

    @Override
    public void getContentAsSAX(final ContentHandler handler) throws XMLDBException {
        final InputSource is;
        InputStream cis = null;

        try {
            if (content != null) {
                is = new InputSource(new StringReader(content));
            } else {
                cis = getStreamContent();
                is = new InputSource(cis);
            }

            XMLReader reader = xmlReader;
            if (reader == null) {
                final SAXParserFactory saxFactory = ExistSAXParserFactory.getSAXParserFactory();
                saxFactory.setNamespaceAware(true);
                saxFactory.setValidating(false);
                final SAXParser sax = saxFactory.newSAXParser();
                reader = sax.getXMLReader();
                reader.setFeature(FEATURE_SECURE_PROCESSING, true);
            }

            reader.setContentHandler(handler);
            if (lexicalHandler != null) {
                reader.setProperty(Namespaces.SAX_LEXICAL_HANDLER, lexicalHandler);
            }
            reader.parse(is);
        } catch (final ParserConfigurationException | SAXException | IOException e) {
            throw new XMLDBException(ErrorCodes.VENDOR_ERROR, e.getMessage(), e);
        } finally {
            if (cis != null) {
                try {
                    cis.close();
                } catch (final IOException ioe) {
                    LOG.warn(ioe.getMessage(), ioe);
                }
            }
        }
    }

    @Override
    public Object getExtendedContent() throws XMLDBException {
        return getExtendedContentInternal(content, idIsPresent(), handle, pos);
    }

    @Override
    public void setContent(final Object value) throws XMLDBException {
        content = null;
        if (!super.setContentInternal(value)) {
            if (value instanceof String) {
                content = (String) value;
            } else if (value instanceof byte[]) {
                content = new String((byte[]) value, UTF_8);

            } else {
                content = value.toString();
            }
        }
    }

    @Override
    public void setContentAsDOM(final Node root) throws XMLDBException {
        Properties properties = getProperties();
        try  {
            VirtualTempPath tempFile = new VirtualTempPath(getInMemorySize(properties), TemporaryFileManager.getInstance());
            try (OutputStream out = tempFile.newOutputStream(); OutputStreamWriter osw = new OutputStreamWriter(out, UTF_8)) {
                final DOMSerializer xmlout = new DOMSerializer(osw, properties);
                final short type = root.getNodeType();
                if (type == Node.ELEMENT_NODE || type == Node.DOCUMENT_FRAGMENT_NODE || type == Node.DOCUMENT_NODE) {
                    xmlout.serialize(root);
                } else {
                    throw new XMLDBException(ErrorCodes.VENDOR_ERROR, "invalid node type");
                }
            }
            setContent(tempFile);
        } catch (final TransformerException | IOException ioe) {
            freeResources();
            throw new XMLDBException(ErrorCodes.VENDOR_ERROR, ioe.getMessage(), ioe);
        }
    }

    @Override
    public ContentHandler setContentAsSAX() throws XMLDBException {
        freeResources();
        content = null;
        return new InternalXMLSerializer();
    }

    public boolean idIsPresent() {
        return id.isPresent();
    }
    
    public String getNodeId() {
        return id.orElse("1");
    }

    /**
     * Sets the external XMLReader to use.
     *
     * @param xmlReader the XMLReader
     */
    public void setXMLReader(final XMLReader xmlReader) {
        this.xmlReader = xmlReader;
    }

    private class InternalXMLSerializer extends SAXSerializer {
        private VirtualTempPath  tempFile = null;
        private OutputStreamWriter writer = null;

        public InternalXMLSerializer() {
            super();
        }

        @Override
        public void startDocument() throws SAXException {
            try {
                tempFile = new VirtualTempPath(getInMemorySize(getProperties()), TemporaryFileManager.getInstance());
                writer = new OutputStreamWriter(tempFile.newOutputStream(), UTF_8);
                setOutput(writer, new Properties());

            } catch (final IOException ioe) {
                throw new SAXException("Unable to create temp file for serialization data", ioe);
            }

            super.startDocument();
        }

        @Override
        public void endDocument() throws SAXException {
            super.endDocument();

            try {
                if (writer != null) {
                    writer.flush();
                    writer.close();
                }

                setContent(tempFile);
            } catch (final IOException | XMLDBException e) {
                throw new SAXException("Unable to set file content containing serialized data", e);
            }
        }
    }

    @Override
    public boolean getSAXFeature(final String name)
            throws SAXNotRecognizedException, SAXNotSupportedException {
        return false;
    }

    @Override
    public void setSAXFeature(final String name, final boolean value)
            throws SAXNotRecognizedException, SAXNotSupportedException {
    }

    @Override
    public void setLexicalHandler(final LexicalHandler handler) {
        this.lexicalHandler = handler;
    }

    @Override
    public DocumentType getDocType() throws XMLDBException {
        final List params = new ArrayList(1);
        params.add(path.toString());

        try {
            final Object[] request = (Object[]) xmlRpcClientLease.get().execute("getDocType", params);
            final DocumentType result;
            if (!"".equals(request[0])) {
                result = new DocumentTypeImpl((String) request[0], (String) request[1], (String) request[2]);
            } else {
                result = null;
            }
            return result;
        } catch (final XmlRpcException e) {
            throw new XMLDBException(ErrorCodes.UNKNOWN_ERROR, e.getMessage(), e);
        }
    }

    @Override
    public void setDocType(final DocumentType doctype) throws XMLDBException {
        if (doctype != null) {
            final List params = new ArrayList(4);
            params.add(path.toString());
            params.add(doctype.getName());
            params.add(doctype.getPublicId() == null ? "" : doctype.getPublicId());
            params.add(doctype.getSystemId() == null ? "" : doctype.getSystemId());

            try {
                xmlRpcClientLease.get().execute("setDocType", params);
            } catch (final XmlRpcException e) {
                throw new XMLDBException(ErrorCodes.UNKNOWN_ERROR, e.getMessage(), e);
            }
        }
    }

    @Override
    public void getContentIntoAStream(final OutputStream os)
            throws XMLDBException {
        getContentIntoAStreamInternal(os, content, idIsPresent(), handle, pos);
    }

    @Override
    public InputStream getStreamContent() throws XMLDBException {
        return getStreamContentInternal(content, idIsPresent(), handle, pos);
    }

    @Override
    public long getStreamLength()
            throws XMLDBException {
        return getStreamLengthInternal(content);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy