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

org.jdom2.xpath.XPathFactory Maven / Gradle / Ivy

Go to download

A complete, Java-based solution for accessing, manipulating, and outputting XML data

There is a newer version: 2.0.2
Show newest version
/*--

 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 compile(String expression) { return compile(expression, Filters.fpassthrough(), null, EMPTYNS); } }