org.jdom2.xpath.XPathFactory Maven / Gradle / Ivy
Show all versions of jdom Show documentation
/*--
Copyright (C) 2012 Jason Hunter & Brett McLaughlin.
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions, and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions, and the disclaimer that follows
these conditions in the documentation and/or other materials
provided with the distribution.
3. The name "JDOM" must not be used to endorse or promote products
derived from this software without prior written permission. For
written permission, please contact .
4. Products derived from this software may not be called "JDOM", nor
may "JDOM" appear in their name, without prior written permission
from the JDOM Project Management .
In addition, we request (but do not require) that you include in the
end-user documentation provided with the redistribution and/or in the
software itself an acknowledgement equivalent to the following:
"This product includes software developed by the
JDOM Project (http://www.jdom.org/)."
Alternatively, the acknowledgment may be graphical using the logos
available at http://www.jdom.org/images/logos.
THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE JDOM AUTHORS OR THE PROJECT
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
SUCH DAMAGE.
This software consists of voluntary contributions made by many
individuals on behalf of the JDOM Project and was originally
created by Jason Hunter and
Brett McLaughlin . For more information
on the JDOM Project, please see .
*/
package org.jdom2.xpath;
import java.util.Collection;
import java.util.Map;
import java.util.concurrent.atomic.AtomicReference;
import org.jdom2.JDOMConstants;
import org.jdom2.Namespace;
import org.jdom2.filter.Filter;
import org.jdom2.filter.Filters;
import org.jdom2.internal.ReflectionConstructor;
import org.jdom2.xpath.jaxen.JaxenXPathFactory;
/**
* XPathFactory allows JDOM users to configure which XPath implementation to use
* when evaluating XPath expressions.
*
* JDOM does not extend the core Java XPath API (javax.xml.xpath.XPath). Instead
* it creates a new API that is more JDOM and Java friendly leading to neater
* and more understandable code (in a JDOM context).
*
* A JDOM XPathFactory instance is able to create JDOM XPathExpression instances
* that can be used to evaluate XPath expressions against JDOM Content.
*
* The XPathFactory allows either default or custom XPathFactory instances to be
* created. If you use the {@link #newInstance(String)} method then an
* XPathFactory of that specific type will be created. If you use the
* {@link #instance()} method then a default XPathFactory instance will be
* returned.
*
* Instances of XPathFactory are specified to be thread-safe. You can reuse an
* XPathFactory in multiple threads. Instances of XPathExpression are
* NOT thread-safe.
*
* @since JDOM2
* @author Rolf Lear
*/
public abstract class XPathFactory {
private static final Namespace[] EMPTYNS = new Namespace[0];
/**
* An atomic reference storing an instance of the default XPathFactory.
*/
private static final AtomicReference defaultreference = new AtomicReference();
private static final String DEFAULTFACTORY = System.getProperty(
JDOMConstants.JDOM2_PROPERTY_XPATH_FACTORY, null);
/**
* Obtain an instance of an XPathFactory using the default mechanisms to
* determine what XPathFactory implementation to use.
*
* The exact same XPathFactory instance will be returned from each call.
*
* The default mechanism will inspect the system property (only once)
* {@link JDOMConstants#JDOM2_PROPERTY_XPATH_FACTORY} to determine what
* class should be used for the XPathFactory. If that property is not set
* then JDOM will use the {@link JaxenXPathFactory}.
*
* @return the default XPathFactory instance
*/
public static final XPathFactory instance() {
final XPathFactory ret = defaultreference.get();
if (ret != null) {
return ret;
}
XPathFactory fac = DEFAULTFACTORY == null ? new JaxenXPathFactory()
: newInstance(DEFAULTFACTORY);
if (defaultreference.compareAndSet(null, fac)) {
return fac;
}
// someone else installed a different instance before we added ours.
// return that other instance.
return defaultreference.get();
}
/**
* Create a new instance of an explicit XPathFactory. A new instance of the
* specified XPathFactory is created each time. The target XPathFactory
* needs to have a no-argument default constructor.
*
* This method is a convenience mechanism only, and JDOM users are free to
* create a custom XPathFactory instance and use a simple:
* XPathFactory fac = new MyXPathFactory(arg1, arg2, ...)
*
* @param factoryclass
* The name of the XPathFactory class to create.
* @return An XPathFactory of the specified class.
*/
public static final XPathFactory newInstance(String factoryclass) {
return ReflectionConstructor
.construct(factoryclass, XPathFactory.class);
}
/**
* Create a Compiled XPathExpression<> instance from this factory. This
* is the only abstract method on this class. All other compile and evaluate
* methods prepare the data in some way to call this compile method.
*
* XPathFactory implementations override this method to implement support
* for the JDOM/XPath API.
*
*
Namespace
XPath expressions are always namespace aware, and
* expect to be able to resolve prefixes to namespace URIs. In XPath
* expressions the prefix "" always resolves to the empty Namespace URI "".
* A prefix in an XPath query is expected to resolve to exactly one URI.
* Multiple different prefixes in the expression may resolve to the same
* URI.
*
* This compile method ensures that these XPath/Namespace rules are followed
* and thus this method will throw IllegalArgumentException if:
*
* - a namespace has the empty-string prefix but has a non-empty URI.
*
- more than one Namespace has any one prefix.
*
*
*
Variables
*
* Variables are referenced from XPath expressions using a
* $varname
syntax. The variable name may be a Namespace
* qualified variable name of the form $pfx:localname
.
* Variables $pa:var
and $pb:var
are the identical
* variables if the namespace URI for prefix 'pa' is the same URI as for
* prefix 'pb'.
*
* This compile method expects all variable names to be expressed in a
* prefix-qualified format, where all prefixes have to be available in one
* of the specified Namespace instances.
*
* e.g. if you specify a variable name "ns:var" with value "value", you also
* need to have some namespace provided with the prefix "ns" such as
* Namespace.getNamespace("ns", "http://example.com/nsuri");
*
* Some XPath libraries allow null variable values (Jaxen), some do not
* (native Java). This compile method will silently convert any null
* Variable value to an empty string ""
.
*
* Variables are provided in the form of a Map where the key is the variable
* name and the mapped value is the variable value. If the entire map is
* null then the compile Method assumes there are no variables.
*
* In light of the above, this compile method will throw an
* IllegalArgumentException if:
*
* - a variable name is not a valid XML QName.
*
- The prefix associated with a variable name is not available as a
* Namespace.
*
* A NullPointerException will be thrown if the map contains a null variable
* name
*
* @param
* The generic type of the results that the XPathExpression will
* produce.
* @param expression
* The XPath expression.
* @param filter
* The Filter that is used to coerce the xpath result data in to the
* generic-typed results.
* @param variables
* Any variable values that may be referenced from the query. A null
* value indicates that there are no variables.
* @param namespaces
* Any namespaces that may be referenced from the query
* @return an XPathExpression<> instance.
* @throws NullPointerException
* if the query, filter, any namespace, any variable name or any
* variable value is null (although the entire variables value may
* be null).
* @throws IllegalArgumentException
* if any two Namespace values share the same prefix, or if there is
* any other reason that the XPath query cannot be compiled.
*/
public abstract XPathExpression compile(String expression,
Filter filter, Map variables,
Namespace... namespaces);
/**
* Create a XPathExpression<> instance from this factory.
*
* @param
* The generic type of the results that the XPathExpression will
* produce.
* @param expression
* The XPath expression.
* @param filter
* The Filter that is used to coerce the xpath result data in to the
* generic-typed results.
* @param variables
* Any variable values that may be referenced from the query. A null
* value indicates that there are no variables.
* @param namespaces
* List of all namespaces that may be referenced from the query
* @return an XPathExpression<T> instance.
* @throws NullPointerException
* if the query, filter, namespaces, any variable name or any
* variable value is null (although the entire variables value may
* be null).
* @throws IllegalArgumentException
* if any two Namespace values share the same prefix, or if there is
* any other reason that the XPath query cannot be compiled.
*/
public XPathExpression compile(String expression, Filter filter,
Map variables, Collection namespaces) {
return compile(expression, filter, variables, namespaces.toArray(EMPTYNS));
}
/**
* Create a XPathExpression<T> instance from this factory.
*
* @param
* The generic type of the results that the XPathExpression will
* produce.
* @param expression
* The XPath expression.
* @param filter
* The Filter that is used to coerce the xpath result data in to the
* generic-typed results.
* @return an XPathExpression<T> instance.
* @throws NullPointerException
* if the query or filter is null
* @throws IllegalArgumentException
* if there is any reason that the XPath query cannot be compiled.
*/
public XPathExpression compile(String expression, Filter filter) {
return compile(expression, filter, null, EMPTYNS);
}
/**
* Create a XPathExpression<Object> instance from this factory.
*
* @param expression
* The XPath expression.
* @return an XPathExpression<Object> instance.
* @throws NullPointerException
* if the query or filter is null
* @throws IllegalArgumentException
* if there is any reason that the XPath query cannot be compiled.
*/
public XPathExpression