org.apache.xmlbeans.impl.xpath.saxon.XBeansXPath Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of commons-xmlbeans Show documentation
Show all versions of commons-xmlbeans Show documentation
The Apache Commons Codec package contains simple encoder and decoders for
various formats such as Base64 and Hexadecimal. In addition to these
widely used encoders and decoders, the codec package also maintains a
collection of phonetic encoding utilities.
The newest version!
/* Copyright 2004 The Apache Software Foundation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.xmlbeans.impl.xpath.saxon;
import java.util.List;
import java.util.Map;
import java.util.ListIterator;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.TransformerException;
import org.w3c.dom.Node;
import net.sf.saxon.Configuration;
import net.sf.saxon.dom.NodeWrapper;
import net.sf.saxon.om.NodeInfo;
import net.sf.saxon.om.VirtualNode;
import net.sf.saxon.om.Item;
import net.sf.saxon.value.Value;
import net.sf.saxon.sxpath.XPathEvaluator;
import net.sf.saxon.sxpath.XPathExpression;
import net.sf.saxon.sxpath.IndependentContext;
import net.sf.saxon.sxpath.XPathDynamicContext;
import net.sf.saxon.sxpath.XPathVariable;
import org.apache.xmlbeans.impl.store.PathDelegate;
public class XBeansXPath
implements PathDelegate.SelectPathInterface
{
private Object[] namespaceMap;
private String path;
private String contextVar;
private String defaultNS;
/**
* Construct given an XPath expression string.
* @param path The XPath expression
* @param contextVar The name of the context variable
* @param namespaceMap a map of prefix/uri bindings for NS support
* @param defaultNS the uri for the default element NS, if any
*/
public XBeansXPath(String path, String contextVar,
Map namespaceMap, String defaultNS)
{
this.path = path;
this.contextVar = contextVar;
this.defaultNS = defaultNS;
this.namespaceMap = namespaceMap.entrySet().toArray();
}
/**
* Select all nodes that are selectable by this XPath
* expression. If multiple nodes match, multiple nodes
* will be returned.
*
*
* NOTE: In most cases, nodes will be returned
* in document-order, as defined by the XML Canonicalization
* specification. The exception occurs when using XPath
* expressions involving the union
operator
* (denoted with the pipe '|' character).
*
*
*
* NOTE: Param node must be a DOM node which will be used
* during the xpath execution and iteration through the results.
* A call of node.dispose() must be done after reading all results.
*
*
* @param node The node, nodeset or Context object for evaluation.
* This value can be null.
* @return The List
of all items selected
* by this XPath expression.
*/
public List selectNodes(Object node)
{
try
{
Node contextNode = (Node)node;
XPathEvaluator xpe = new XPathEvaluator();
Configuration config = new Configuration();
config.setDOMLevel(2);
config.setTreeModel(net.sf.saxon.event.Builder.STANDARD_TREE);
IndependentContext sc = new IndependentContext(config);
// Declare ns bindings
if (defaultNS != null)
sc.setDefaultElementNamespace(defaultNS);
for (int i = 0; i < namespaceMap.length; i++)
{
Map.Entry entry = (Map.Entry) namespaceMap[i];
sc.declareNamespace((String) entry.getKey(),
(String) entry.getValue());
}
xpe.setStaticContext(sc);
XPathVariable thisVar = xpe.declareVariable("", contextVar);
XPathExpression xpath = xpe.createExpression(path);
NodeInfo contextItem =
//config.buildDocument(new DOMSource(contextNode));
config.unravel(new DOMSource(contextNode));
XPathDynamicContext dc = xpath.createDynamicContext(null);
dc.setContextItem(contextItem);
dc.setVariable(thisVar, contextItem);
List saxonNodes = xpath.evaluate(dc);
for (ListIterator it = saxonNodes.listIterator(); it.hasNext(); )
{
Object o = it.next();
if (o instanceof NodeInfo)
{
if (o instanceof NodeWrapper)
{
Node n = getUnderlyingNode((NodeWrapper)o);
it.set(n);
}
else
{
it.set(((NodeInfo)o).getStringValue());
}
}
else if (o instanceof Item)
it.set(Value.convertToJava((Item)o));
}
return saxonNodes;
}
catch (TransformerException e)
{
throw new RuntimeException(e);
}
}
public List selectPath(Object node)
{
return selectNodes(node);
}
/**
* According to the Saxon javadoc:
* getUnderlyingNode
in NodeWrapper
implements
* the method specified in the interface VirtualNode
, and
* the specification of the latter says that it may return another
* VirtualNode
, and you may have to drill down through
* several layers of wrapping.
* To be safe, this method is provided to drill down through multiple
* layers of wrapping.
* @param v The VirtualNode
* @return The underlying node
*/
private static Node getUnderlyingNode(VirtualNode v)
{
Object o = v;
while (o instanceof VirtualNode)
{
o = ((VirtualNode)o).getUnderlyingNode();
}
return (Node)o;
}
}