d2f5c8e572
* use less xpath to speed things up a little bit
152 lines
3.5 KiB
Java
152 lines
3.5 KiB
Java
|
|
package net.sourceforge.tuned;
|
|
|
|
|
|
import java.util.AbstractList;
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
|
|
import javax.xml.xpath.XPathConstants;
|
|
import javax.xml.xpath.XPathExpression;
|
|
import javax.xml.xpath.XPathExpressionException;
|
|
import javax.xml.xpath.XPathFactory;
|
|
|
|
import org.w3c.dom.Node;
|
|
import org.w3c.dom.NodeList;
|
|
|
|
|
|
public final class XPathUtilities {
|
|
|
|
public static Node selectNode(String xpath, Object node) {
|
|
try {
|
|
return (Node) getXPath(xpath).evaluate(node, XPathConstants.NODE);
|
|
} catch (Exception e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
|
|
public static List<Node> selectNodes(String xpath, Object node) {
|
|
try {
|
|
return new NodeListDecorator((NodeList) getXPath(xpath).evaluate(node, XPathConstants.NODESET));
|
|
} catch (Exception e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
|
|
public static String selectString(String xpath, Object node) {
|
|
try {
|
|
return ((String) getXPath(xpath).evaluate(node, XPathConstants.STRING)).trim();
|
|
} catch (Exception e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
|
|
public static boolean exists(String xpath, Object node) {
|
|
return selectNode(xpath, node) != null;
|
|
}
|
|
|
|
|
|
/**
|
|
* @param nodeName search for nodes with this name
|
|
* @param parentNode search in the child nodes of this nodes
|
|
* @return text content of the child node or null if no child with the given name was found
|
|
*/
|
|
public static Node getChild(String nodeName, Node parentNode) {
|
|
for (Node child : new NodeListDecorator(parentNode.getChildNodes())) {
|
|
if (nodeName.equals(child.getNodeName()))
|
|
return child;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
|
|
public static List<Node> getChildren(String nodeName, Node parentNode) {
|
|
List<Node> children = new ArrayList<Node>();
|
|
|
|
for (Node child : new NodeListDecorator(parentNode.getChildNodes())) {
|
|
if (nodeName.equals(child.getNodeName()))
|
|
children.add(child);
|
|
}
|
|
|
|
return children;
|
|
}
|
|
|
|
|
|
public static String getAttribute(String attribute, Node node) {
|
|
return node.getAttributes().getNamedItem(attribute).getNodeValue().trim();
|
|
}
|
|
|
|
|
|
/**
|
|
* Get text content of the first child node matching the given node name. Use this method
|
|
* instead of {@link #selectString(String, Object)} whenever xpath support is not required,
|
|
* because it is much faster, especially for large documents.
|
|
*
|
|
* @param childName search for nodes with this name
|
|
* @param parentNode search in the child nodes of this nodes
|
|
* @return text content of the child node or null if no child with the given name was found
|
|
*/
|
|
public static String getTextContent(String childName, Node parentNode) {
|
|
Node child = getChild(childName, parentNode);
|
|
|
|
if (child == null) {
|
|
return null;
|
|
}
|
|
|
|
return getTextContent(child);
|
|
}
|
|
|
|
|
|
public static String getTextContent(Node node) {
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
for (Node textNode : getChildren("#text", node)) {
|
|
sb.append(textNode.getNodeValue());
|
|
}
|
|
|
|
return sb.toString().trim();
|
|
}
|
|
|
|
|
|
private static XPathExpression getXPath(String xpath) throws XPathExpressionException {
|
|
return XPathFactory.newInstance().newXPath().compile(xpath);
|
|
}
|
|
|
|
|
|
/**
|
|
* Dummy constructor to prevent instantiation.
|
|
*/
|
|
private XPathUtilities() {
|
|
throw new UnsupportedOperationException();
|
|
}
|
|
|
|
|
|
protected static class NodeListDecorator extends AbstractList<Node> {
|
|
|
|
private final NodeList nodes;
|
|
|
|
|
|
public NodeListDecorator(NodeList nodes) {
|
|
this.nodes = nodes;
|
|
}
|
|
|
|
|
|
@Override
|
|
public Node get(int index) {
|
|
return nodes.item(index);
|
|
}
|
|
|
|
|
|
@Override
|
|
public int size() {
|
|
return nodes.getLength();
|
|
}
|
|
|
|
}
|
|
|
|
}
|