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

net.sf.saxon.s9api.XPathSelector Maven / Gradle / Ivy

There is a newer version: 12.5
Show newest version
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2018-2022 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.s9api;

import net.sf.saxon.lib.ErrorReporter;
import net.sf.saxon.lib.ResourceResolver;
import net.sf.saxon.lib.ResourceResolverWrappingURIResolver;
import net.sf.saxon.lib.UnparsedTextURIResolver;
import net.sf.saxon.om.*;
import net.sf.saxon.s9api.streams.XdmStream;
import net.sf.saxon.sxpath.XPathDynamicContext;
import net.sf.saxon.sxpath.XPathExpression;
import net.sf.saxon.sxpath.XPathVariable;
import net.sf.saxon.trans.UncheckedXPathException;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.transpile.CSharpModifiers;

import javax.xml.transform.URIResolver;
import java.util.Map;
import java.util.Objects;

/**
 * An XPathSelector represents a compiled and loaded XPath expression ready for execution.
 * The XPathSelector holds details of the dynamic evaluation context for the XPath expression.
 */
//@CSharpInjectMembers(code = {
//        "    public void setErrorReporter(System.Action reporter) {"
//                + "        setErrorReporter(new Saxon.Impl.Helpers.ErrorReportingAction(reporter));"
//                + "    }"
//})
@SuppressWarnings({"ForeachStatement"})
@CSharpModifiers(code = {"internal"})
public class XPathSelector implements Iterable {

    private final XPathExpression exp;
    private final XPathDynamicContext dynamicContext;
    private final Map declaredVariables;

    // protected constructor

    protected XPathSelector(XPathExpression exp,
                            Map declaredVariables) {
        this.exp = exp;
        this.declaredVariables = declaredVariables;
        dynamicContext = exp.createDynamicContext();
    }

    /**
     * Set the context item for evaluating the XPath expression.
     * This may be either a node or an atomic value. Most commonly it will be a document node,
     * which might be constructed using the {@link DocumentBuilder#build} method.
     *
     * @param item The context item for evaluating the expression. Must not be null.
     * @throws SaxonApiException if an error occurs, for example because the type
     *                           of item supplied does not match the required item type
     */

    public void setContextItem(XdmItem item) throws SaxonApiException {
        if (item == null) {
            throw new NullPointerException("contextItem");
        }
        if (!exp.getInternalExpression().getPackageData().isSchemaAware()) {
            Item it = item.getUnderlyingValue().head();
            if (it instanceof NodeInfo && ((NodeInfo) it).getTreeInfo().isTyped()) {
                throw new SaxonApiException(
                    "The supplied node has been schema-validated, but the XPath expression was compiled without schema-awareness");
            }
        }
        try {
            dynamicContext.setContextItem(item.getUnderlyingValue());
        } catch (XPathException e) {
            throw new SaxonApiException(e);
        }
    }

    /**
     * Get the context item used for evaluating the XPath expression.
     * This may be either a node or an atomic value. Most commonly it will be a document node,
     * which might be constructed using the Build method of the DocumentBuilder object.
     *
     * @return The context item for evaluating the expression, or null if no context item
     *         has been set.
     */

    public XdmItem getContextItem() {
        return XdmItem.wrapItem(dynamicContext.getContextItem());
    }

    /**
     * Set the value of a variable
     *
     * @param name  The name of the variable. This must match the name of a variable
     *              that was declared to the XPathCompiler. No error occurs if the expression does not
     *              actually reference a variable with this name.
     * @param value The value to be given to the variable.
     * @throws SaxonApiException if the variable has not been declared or if the type of the value
     *                           supplied does not conform to the required type that was specified when the variable was declared
     */

    public void setVariable(QName name, XdmValue value) throws SaxonApiException {
        Objects.requireNonNull(name, "name");
        Objects.requireNonNull(value, "value");
        StructuredQName qn = name.getStructuredQName();
        XPathVariable var = declaredVariables.get(qn);
        if (var == null) {
            throw new SaxonApiException(
                    new XPathException("Variable has not been declared: " + name));
        }
        try {
            dynamicContext.setVariable(var, value.getUnderlyingValue());
        } catch (XPathException | UncheckedXPathException e) {
            throw new SaxonApiException(e);
        }
    }

    /**
     * Set an object that will be used to resolve URIs used in
     * fn:doc() and related functions.
     *
     * @param resolver An object that implements the URIResolver interface, or null.
     * @since 9.4
     */

    public void setResourceResolver(ResourceResolver resolver) {
        dynamicContext.setResourceResolver(resolver);
    }

    /**
     * Get the URI resolver.
     *
     * @return the user-supplied URI resolver if there is one, or the
     * system-defined one otherwise
     * @since 9.4
     */

    public ResourceResolver getResourceResolver() {
        return dynamicContext.getResourceResolver();
    }


    /**
     * Set an object that will be used to resolve URIs used in
     * fn:doc() and related functions.
     *
     * @param resolver An object that implements the URIResolver interface, or null.
     * @since 9.4
     */

    public void setURIResolver(URIResolver resolver) {
        dynamicContext.setResourceResolver(new ResourceResolverWrappingURIResolver(resolver));
    }

    /**
     * Get the URI resolver.
     *
     * @return the user-supplied URI resolver if there is one, or the
     *         system-defined one otherwise
     * @since 9.4
     */

    public URIResolver getURIResolver() {
        if (dynamicContext.getResourceResolver() instanceof ResourceResolverWrappingURIResolver) {
            return ((ResourceResolverWrappingURIResolver) dynamicContext.getResourceResolver()).getWrappedURIResolver();
        } else {
            return null;
        }
    }

    /**
     * Set an object that will be used to resolve URIs used in
     * fn:unparsed-text() and related functions.
     *
     * @param resolver An object that implements the UnparsedTextURIResolver interface, or
     *                 null.
     * @since 11
     */

    public void setUnparsedTextResolver(UnparsedTextURIResolver resolver) {
        dynamicContext.setUnparsedTextURIResolver(resolver);
    }

    /**
     * Get the URI resolver used for fn:unparsed-text() and related functions.
     *
     * @return the user-supplied URI resolver if there is one, or the
     * system-defined one otherwise
     * @since 11
     */

    public UnparsedTextURIResolver getUnparsedTextURIResolver() {
        return dynamicContext.getUnparsedTextURIResolver();
    }


    /**
     * Set a callback to be used for reporting of run-time errors and warnings.
     * Note that fatal run-time errors are always reported by throwing an exception,
     * so this is primarily useful for controlling the way that warnings are reported.
     * By default, warnings are reported to the {@link ErrorReporter} associated with the
     * Saxon Configuration.
     * @param reporter the user-supplied error reporter to be called when dynamic errors and warnings
     *                 arise
     * @since 10.0
     */

    public void setErrorReporter(ErrorReporter reporter) {
        dynamicContext.setErrorReporter(reporter);
    }

    /**
     * Evaluate the expression, returning the result as an XdmValue (that is,
     * a sequence of nodes and/or atomic values).
     * 

Note: Although a singleton result may be represented as an XdmItem, there is * no guarantee that this will always be the case. If you know that the expression will return at * most one node or atomic value, it is best to use the evaluateSingle method, which * does guarantee that an XdmItem (or null) will be returned.

* * @return An XdmValue representing the results of the expression. * @throws SaxonApiException if a dynamic error occurs during the expression evaluation. */ public XdmValue evaluate() throws SaxonApiException { Sequence value; try { value = SequenceTool.toGroundedValue(exp.iterate(dynamicContext)); } catch (UncheckedXPathException uxe) { throw new SaxonApiException(uxe); } catch (XPathException e) { throw new SaxonApiException(e); } return XdmValue.wrap(value); } /** * Evaluate the XPath expression, returning the result as an XdmItem (that is, * a single node or atomic value). * * @return an XdmItem representing the result of the expression, or null if the expression * returns an empty sequence. If the expression returns a sequence of more than one item, * any items after the first are ignored. * @throws SaxonApiException if a dynamic error occurs during the expression evaluation. */ public XdmItem evaluateSingle() throws SaxonApiException { try { Item i = exp.evaluateSingle(dynamicContext); if (i == null) { return null; } return (XdmItem) XdmValue.wrap(i); } catch (XPathException e) { throw new SaxonApiException(e); } } /** * Evaluate the expression, returning the result as an Iterator (that is, * an iterator over a sequence of nodes and/or atomic values). *

Because an XPathSelector is an {@link Iterable}, it is possible to * iterate over the result using a Java 5 "for each" expression, for example:

*
     * XPathCompiler compiler = processor.newXPathCompiler();
     * XPathSelector seq = compiler.compile("1 to 20").load();
     * for (XdmItem item : seq) {
     *   System.err.println(item);
     * }
     * 
* * @return An iterator over the sequence that represents the results of the expression. * Each object in this sequence will be an instance of XdmItem. Note * that the expression may be evaluated lazily, which means that a successful response * from this method does not imply that the expression has executed successfully: failures * may be reported later while retrieving items from the iterator. * @throws SaxonApiUncheckedException if a dynamic error occurs during XPath evaluation that * can be detected at this point. It is also possible that an SaxonApiUncheckedException will * be thrown by the hasNext() method of the returned iterator. */ @Override public XdmSequenceIterator iterator() throws SaxonApiUncheckedException { try { return new XdmSequenceIterator<>(exp.iterate(dynamicContext)); } catch (XPathException e) { throw new SaxonApiUncheckedException(e); } } /** * Evaluate the expression, returning the result as an Stream. * * @return A stream delivering the results of the expression. * Each object in this stream will be an instance of XdmItem. Note * that the expression may be evaluated lazily, which means that a successful response * from this method does not imply that the expression has executed successfully: failures * may be reported later while retrieving items from the iterator. * @throws SaxonApiUncheckedException if a dynamic error occurs during XPath evaluation that * can be detected at this point. */ public XdmStream stream() throws SaxonApiUncheckedException { return iterator().stream(); } /** * Evaluate the XPath expression, returning the effective boolean value of the result. * * @return a boolean representing the effective boolean value of the result of evaluating * the expression, as defined by the rules for the fn:boolean() function. * @throws SaxonApiException if a dynamic error occurs during the expression evaluation, or if the result * of the expression is a value whose effective boolean value is not defined (for example, a date or a * sequence of three integers) * @since 9.1 */ public boolean effectiveBooleanValue() throws SaxonApiException { try { return exp.effectiveBooleanValue(dynamicContext); } catch (XPathException e) { throw new SaxonApiException(e); } } /** * Get the underlying dynamic context object. This provides an escape hatch to the underlying * implementation classes, which contain methods that may change from one release to another. * * @return the underlying object representing the dynamic context for query execution */ public XPathDynamicContext getUnderlyingXPathContext() { return dynamicContext; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy