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

net.sf.saxon.sxpath.XPathEvaluator Maven / Gradle / Ivy

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// 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.
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

package net.sf.saxon.sxpath;

import net.sf.saxon.Configuration;
import net.sf.saxon.expr.Expression;
import net.sf.saxon.expr.instruct.Executable;
import net.sf.saxon.expr.instruct.SlotManager;
import net.sf.saxon.expr.parser.ExpressionTool;
import net.sf.saxon.expr.parser.ExpressionVisitor;
import net.sf.saxon.functions.*;
import net.sf.saxon.om.NamespaceResolver;
import net.sf.saxon.pattern.Pattern;
import net.sf.saxon.pattern.PatternSponsor;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.type.ItemType;
import net.sf.saxon.type.Type;
import net.sf.saxon.value.DecimalValue;

/**
 * This class provides a native Saxon API for free-standing evaluation of XPath expressions. Unlike the
 * JAXP API offered by {@link net.sf.saxon.xpath.XPathEvaluator} it exposes Saxon classes and interfaces
 * and thus provides a more strongly-typed interface with greater control over the detailed behaviour.
 *
 * 

However, the preferred API for user applications calling XPath is the s9api {@link net.sf.saxon.s9api.XPathCompiler} * interface.

* * @author Michael H. Kay * @since 8.4 */ public class XPathEvaluator { private XPathStaticContext staticContext; /** * Default constructor. Creates an XPathEvaluator with a default configuration and name pool. * Note that any source documents used by an XPath expression under this XPathEvaluator must * be built using the {@link Configuration} that is implicitly created by this constructor, * which is accessible using the {@link #getConfiguration} method. */ public XPathEvaluator() { this(new Configuration()); } /** * Construct an XPathEvaluator with a specified configuration. * @param config the configuration to be used. If the XPathEvaluator is * to be used to run schema-aware XPath expressions this must be an instance * of {@link com.saxonica.config.EnterpriseConfiguration} */ public XPathEvaluator(Configuration config) { staticContext = new IndependentContext(config); } /** * Get the Configuration in use. * @return the Saxon configuration */ public Configuration getConfiguration() { return staticContext.getConfiguration(); } /** * Declare a variable, making it available for use within a subsequently-compiled XPath expression. * The variable may subsequently be assigned a value using the method * {@link XPathDynamicContext#setVariable(XPathVariable, net.sf.saxon.om.Sequence)}. * Any external variable used in an XPath expression must be declared before the XPath expression * is compiled. * @param uri The namespace URI of the variable name. Use "" for the null namespace. * @param localName The local part of the variable name. * @return an object representing the variable */ public XPathVariable declareVariable(String uri, String localName) { return staticContext.declareVariable(uri, localName); } /** * Set the static context for compiling XPath expressions. This provides more detailed control over the * environment in which the expression is compiled, for example it allows namespace prefixes to * be declared, variables to be bound and functions to be defined. For most purposes, the static * context can be defined by providing and tailoring an instance of the {@link IndependentContext} class. * Until this method is called, a default static context is used, in which no namespaces are defined * other than the standard ones (xml, xslt, and saxon), and no variables or functions (other than the * core XPath functions) are available. * @param context the XPath static context *

Setting a new static context clears any variables and namespaces that have previously been declared.

*/ public void setStaticContext(XPathStaticContext context) { staticContext = context; } /** * Get the current static context. This will always return a value; if no static context has been * supplied by the user, the system will have created its own. A system-created static context * will always be an instance of {@link IndependentContext} * @return the static context object */ public XPathStaticContext getStaticContext() { return staticContext; } /** * Prepare (compile) an XPath expression for subsequent evaluation. * @param expression The XPath expression to be compiled, supplied as a string. * @return an XPathExpression object representing the prepared expression * @throws XPathException if the syntax of the expression is wrong, or if it references namespaces, * variables, or functions that have not been declared. */ public XPathExpression createExpression(String expression) throws XPathException { Configuration config = getConfiguration(); Executable exec = new Executable(config); SimpleContainer container = new SimpleContainer(exec); boolean allowXPath30 = staticContext.getXPathLanguageLevel().equals(DecimalValue.THREE); exec.setSchemaAware(staticContext.isSchemaAware()); exec.setHostLanguage(Configuration.XPATH, allowXPath30); // Make the function library that's available at run-time (e.g. for saxon:evaluate() and function-lookup()). // This includes all user-defined functions regardless of which module they are in FunctionLibrary userlib = exec.getFunctionLibrary(); FunctionLibraryList lib = new FunctionLibraryList(); int permittedFunctions = StandardFunction.CORE; if (allowXPath30) { permittedFunctions |= StandardFunction.XPATH30; } lib.addFunctionLibrary( SystemFunctionLibrary.getSystemFunctionLibrary(permittedFunctions)); lib.addFunctionLibrary(config.getVendorFunctionLibrary()); lib.addFunctionLibrary(new ConstructorFunctionLibrary(config)); lib.addFunctionLibrary(config.getIntegratedFunctionLibrary()); config.addExtensionBinders(lib); if(userlib != null) { lib.addFunctionLibrary(userlib); } exec.setFunctionLibrary(lib); Expression exp = ExpressionTool.make(expression, staticContext, container, 0, -1, 1, null); exp.setContainer(container); ExpressionVisitor visitor = ExpressionVisitor.make(staticContext, exec); visitor.setExecutable(exec); ItemType contextItemType = staticContext.getRequiredContextItemType(); ExpressionVisitor.ContextItemType cit = new ExpressionVisitor.ContextItemType(contextItemType, true); exp = visitor.typeCheck(exp, cit); exp = visitor.optimize(exp, cit); exp.setContainer(container); // might have lost its container during optimization SlotManager map = staticContext.getStackFrameMap(); int numberOfExternalVariables = map.getNumberOfVariables(); ExpressionTool.allocateSlots(exp, numberOfExternalVariables, map); XPathExpression xpe = new XPathExpression(this, exp); xpe.setStackFrameMap(map, numberOfExternalVariables); return xpe; } /** * Prepare (compile) an XSLT pattern for subsequent evaluation. The result is an XPathExpression * object representing a (pseudo-) expression that when evaluated returns a boolean result: true * if the context node matches the pattern, false if it does not. * @param pattern the XSLT pattern to be compiled, supplied as a string * @return an XPathExpression object representing the pattern, wrapped as an expression * @throws XPathException if the syntax of the expression is wrong, or if it references namespaces, * variables, or functions that have not been declared. * @since 9.1 */ /*@NotNull*/ public XPathExpression createPattern(String pattern) throws XPathException { Executable exec = new Executable(getConfiguration()); Pattern pat = Pattern.make(pattern, staticContext, exec); ExpressionVisitor visitor = ExpressionVisitor.make(staticContext, exec); pat.analyze(visitor, new ExpressionVisitor.ContextItemType(Type.NODE_TYPE, true)); SlotManager map = staticContext.getStackFrameMap(); int slots = map.getNumberOfVariables(); slots = pat.allocateSlots(map, slots); PatternSponsor sponsor = new PatternSponsor(pat); XPathExpression xpe = new XPathExpression(this, sponsor); xpe.setStackFrameMap(map, slots); return xpe; } /** * Set the external namespace resolver to be used. The NamespaceResolver is stored * as part of the static context. It overrides any namespaces declared directly * using declareNamespace on the staticContext object * @param namespaceResolver The namespace resolver, which maintains a mapping of prefixes to URIs. * Any namespace prefix used in the XPath expression is resolved using this namespaceResolver. */ public void setNamespaceResolver(NamespaceResolver namespaceResolver) { staticContext.setNamespaceResolver(namespaceResolver); } /** * Get the external namespace resolver, if one has been set using {@link #setNamespaceResolver} * @return the namespace resolver supplied by the user if set, or a system-defined namespace resolver * otherwise. By default, the {@link IndependentContext} object used as the {@link XPathStaticContext} * also acts as the {@link NamespaceResolver}. */ public NamespaceResolver getNamespaceResolver() { return staticContext.getNamespaceResolver(); } /** * Set the default namespace for elements and types * @param uri The namespace to be used to qualify unprefixed element names and type names appearing * in the XPath expression. */ public void setDefaultElementNamespace(String uri) { staticContext.setDefaultElementNamespace(uri); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy