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

net.sf.saxon.option.expath.zip.library.SaxonElement Maven / Gradle / Ivy

There is a newer version: 10.5
Show newest version
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2013 Saxonica Limited.
// This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
// If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
// This Source Code Form is "Incompatible With Secondary Licenses", as defined by the Mozilla Public License, v. 2.0.
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

/****************************************************************************/
/*  File:       SaxonElement.java                                           */
/*  Author:     F. Georges - H2O Consulting                                 */
/*  Date:       2011-02-21                                                  */
/*  Tags:                                                                   */
/*      Copyright (c) 2011 Florent Georges (see end of file.)               */
/* ------------------------------------------------------------------------ */


package net.sf.saxon.option.expath.zip.library;

import net.sf.saxon.Configuration;
import net.sf.saxon.event.PipelineConfiguration;
import net.sf.saxon.event.ProxyReceiver;
import net.sf.saxon.event.Receiver;
import net.sf.saxon.event.TreeReceiver;
import net.sf.saxon.lib.SerializerFactory;
import net.sf.saxon.om.AxisInfo;
import net.sf.saxon.om.Item;
import net.sf.saxon.om.NamespaceBinding;
import net.sf.saxon.om.NodeInfo;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.tree.iter.AxisIterator;
import net.sf.saxon.type.Type;
import net.sf.saxon.value.Base64BinaryValue;
import net.sf.saxon.value.HexBinaryValue;
import net.sf.saxon.value.Whitespace;

import javax.xml.transform.Result;
import javax.xml.transform.stream.StreamResult;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Iterator;
import java.util.Properties;

/**
 * The implementation for Saxon of an abstract element.
 *
 * @author Florent Georges
 * @date   2011-02-21
 */
public class SaxonElement
        implements Element
{
    public SaxonElement(NodeInfo node)
    {
        // TODO: Should we perform some checks (is it non-null?, is it an
        // element node?, etc.)  Or is it guaranteed by construction?
        myNode = node;
    }

    public String getBaseUri()
    {
        return myNode.getBaseURI();
    }

    public String getLocalName()
    {
        return myNode.getLocalPart();
    }

    public String formatName()
    {
        return myNode.getDisplayName();
    }

    public Iterable attributes()
    {
        AxisIterator it = myNode.iterateAxis(AxisInfo.ATTRIBUTE);
        return new AttributeIterable(it);
    }

    public Iterable entries()
    {
        AxisIterator it = myNode.iterateAxis(AxisInfo.CHILD);
        return new EntryIterable(it);
    }

    public Serialization makeSerialization()
            throws ZipException
    {
        return new SaxonSerialization();
    }

    public void serialize(OutputStream out, Serialization serial)
            throws ZipException
                 , IOException
    {
        AxisIterator children = myNode.iterateAxis(AxisInfo.CHILD);
        try {
            if ( "base64".equals(serial.getMethod()) ) {
                Base64BinaryValue val = new Base64BinaryValue(children.getStringValue());
                out.write(val.getBinaryValue());
            }
            else if ( "hex".equals(serial.getMethod()) ) {
                HexBinaryValue val = new HexBinaryValue(children.getStringValue());
                out.write(val.getBinaryValue());
            }
            else {
                Configuration         config   = myNode.getConfiguration();
                SerializerFactory     factory  = config.getSerializerFactory();
                PipelineConfiguration pipe     = config.makePipelineConfiguration();
                Result                result   = new StreamResult(out);
                Properties            options  = serial.getOutputOptions();
                Receiver              base     = factory.getReceiver(result, pipe, options);
                Receiver              receiver = new ZipNamespaceFilter(base);
                TreeReceiver          tr       = new TreeReceiver(receiver);
                tr.open();
                while ( true ) {
                    Item item = children.next();
                    if ( item == null ) {
                        break;
                    }
                    tr.append(item, 0, NodeInfo.ALL_NAMESPACES);
                }
                tr.close();
            }
        }
        catch ( XPathException ex ) {
            throw new ZipException("Error serializing the source", ex);
        }
    }

    private NodeInfo myNode;

    /**
     * Filters out the ZIP namespace binding.
     */
    private static class ZipNamespaceFilter
            extends ProxyReceiver
    {
        public ZipNamespaceFilter(Receiver base)
        {
            super(base);
        }

        @Override
        public void namespace(NamespaceBinding ns, int props)
                throws XPathException
        {
            String uri = ns.getURI();
            // forward the namespace binding event only if it is not the ZIP namespace
            if ( ! ZipConstants.ZIP_NS_URI.equals(uri) ) {
                super.namespace(ns, props);
            }
        }
    }

    private static class AttributeIterable
            implements Iterable
    {
        public AttributeIterable(AxisIterator it)
        {
            myIter = new AttributeIteratorWrapper(it);
        }

        public Iterator iterator()
        {
            return myIter;
        }

        private Iterator myIter;
    }

    private static class AttributeIteratorWrapper
            implements Iterator
    {
        public AttributeIteratorWrapper(AxisIterator it)
        {
            myIter = it;
            myNext = (NodeInfo) it.next();
        }

        public boolean hasNext()
        {
            return myNext != null;
        }

        public Attribute next()
        {
            if ( myNext == null ) {
                // TODO: Throw an exception instead?
                return null;
            }
            Attribute a = new SaxonAttribute(myNext);
            myNext = (NodeInfo) myIter.next();
            return a;
        }

        public void remove()
        {
            throw new UnsupportedOperationException("remove() is not supported");
        }

        private AxisIterator myIter;
        private NodeInfo myNext;
    }

    private static class EntryIterable
            implements Iterable
    {
        public EntryIterable(AxisIterator it)
        {
            myIter = new EntryIteratorWrapper(it);
        }

        public Iterator iterator()
        {
            return myIter;
        }

        private Iterator myIter;
    }

    private static class EntryIteratorWrapper
            implements Iterator
    {
        public EntryIteratorWrapper(AxisIterator it)
        {
            myIter = it;
            myNext = (NodeInfo) it.next();
            ensureNextEntry();
        }

        public boolean hasNext()
        {
            return myNext != null;
        }

        public Element next()
        {
            if ( myNext == null ) {
                // TODO: Throw an exception instead?
                return null;
            }
            Element e = new SaxonElement(myNext);
            myNext = (NodeInfo) myIter.next();
            ensureNextEntry();
            return e;
        }

        public void remove()
        {
            throw new UnsupportedOperationException("remove() is not supported");
        }

        private void ensureNextEntry()
                throws ZipRuntimeException
        {
            while ( myNext != null && ! isEntry(myNext) ) {
                myNext = (NodeInfo) myIter.next();
            }
        }

        // if unknown child of zip:file, throw an error, if not then return true if
        // it is an entry element, or false if it is an element to ignore (in another
        // namespace for instance)
        private boolean isEntry(NodeInfo node)
                throws ZipRuntimeException
        {
            String local = node.getLocalPart();
            int kind = node.getNodeKind();
            if ( kind == Type.TEXT ) {
                if ( ! Whitespace.isWhite(node.getStringValueCS()) ) {
                    throw new ZipRuntimeException("Non-whitespace text nodes are not allowed");
                }
                return false;
            }
            else if ( kind == Type.COMMENT || kind == Type.PROCESSING_INSTRUCTION ) {
                // ignore comment and PI nodes
                return false;
            }
            else if ( kind != Type.ELEMENT ) {
                // namespace, document and attribute nodes cannot be no the child axis
                throw new ZipRuntimeException("Not an element: could not happen");
            }
            else if ( "".equals(node.getURI()) ) {
                // elements in no namespace are an error
                throw new ZipRuntimeException("Element in no namespace: " + local);
            }
            else if ( ! ZipConstants.ZIP_NS_URI.equals(node.getURI()) ) {
                // ignore elements in other namespaces
                return false;
            }
            else if ( "entry".equals(local) || "dir".equals(local) ) {
                return true;
            }
            else {
                throw new ZipRuntimeException("Unknown element: " + local);
            }
        }

        private AxisIterator myIter;
        private NodeInfo myNext;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy