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

net.sf.saxon.functions.Doc Maven / Gradle / Ivy

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

import net.sf.saxon.Configuration;
import net.sf.saxon.Controller;
import net.sf.saxon.expr.*;
import net.sf.saxon.expr.parser.ContextItemStaticInfo;
import net.sf.saxon.expr.parser.ExpressionTool;
import net.sf.saxon.expr.parser.ExpressionVisitor;
import net.sf.saxon.functions.registry.BuiltInFunctionSet;
import net.sf.saxon.lib.Feature;
import net.sf.saxon.lib.ParseOptions;
import net.sf.saxon.om.GroundedValue;
import net.sf.saxon.om.NodeInfo;
import net.sf.saxon.om.Sequence;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.trans.XsltController;
import net.sf.saxon.transpile.CSharpInnerClass;
import net.sf.saxon.value.AtomicValue;
import net.sf.saxon.value.EmptySequence;

/**
 * Implement the fn:doc() function - a simplified form of the Document function
 */

public class Doc extends SystemFunction implements Callable {

    private ParseOptions parseOptions;

    /**
     * Get the parsing options set via setParseOptions()
     *
     * @return the parsing options
     */

    public ParseOptions getParseOptions() {
        return parseOptions;
    }

    /**
     * Set options to be used for the parsing operation. Defaults to the parsing options set in the Configuration
     *
     * @param parseOptions the parsing options to be used. Currently only affects the behaviour of the sendDocument()
     *                     method (used in streamed merging)
     */

    public void setParseOptions(ParseOptions parseOptions) {
        this.parseOptions = parseOptions;
    }

    @Override
    public int getCardinality(Expression[] arguments) {
        return arguments[0].getCardinality() & ~StaticProperty.ALLOWS_MANY;
    }

    @Override
    public Expression makeFunctionCall(Expression... arguments) {
        Expression expr = maybePreEvaluate(this, arguments);
        return expr == null ? super.makeFunctionCall(arguments) : expr;
    }

    @CSharpInnerClass(outer=false, extra={"Saxon.Hej.functions.SystemFunction sf"})
    public static Expression maybePreEvaluate(final SystemFunction sf, final Expression[] arguments) {
        if (arguments.length > 1 ||
                !sf.getRetainedStaticContext().getConfiguration().getBooleanProperty(Feature.PRE_EVALUATE_DOC_FUNCTION)) {
            sf.getDetails().properties = sf.getDetails().properties | BuiltInFunctionSet.LATE;
            return null;

        } else {
            // allow early evaluation
            return new SystemFunctionCall(sf, arguments) {
                @Override
                public Expression preEvaluate(ExpressionVisitor visitor) {
                    Configuration config = visitor.getConfiguration();
                    try {
                        GroundedValue firstArg = ((Literal) this.getArg(0)).getGroundedValue();
                        if (firstArg.getLength() == 0) {
                            return null;
                        } else if (firstArg.getLength() > 1) {
                            return this;
                        }
                        String href = firstArg.head().getStringValue();
                        if (href.indexOf('#') >= 0) {
                            return this;
                        }
                        NodeInfo item = DocumentFn.preLoadDoc(href, sf.getStaticBaseUriString(),
                                                              sf.getRetainedStaticContext().getPackageData(),
                                                              config, getLocation());
                        if (item != null) {
                            Expression constant = Literal.makeLiteral(item);
                            ExpressionTool.copyLocationInfo(this.getArg(0), constant);
                            return constant;
                        }
                    } catch (Exception err) {
                        // ignore the exception and try again at run-time
                        return this;
                    }
                    return this;
                }

                @Override
                public Expression optimize(ExpressionVisitor visitor, ContextItemStaticInfo contextItemType) throws XPathException {
                    optimizeChildren(visitor, contextItemType);
                    if (getArg(0) instanceof StringLiteral) {
                        return preEvaluate(visitor);
                    }
                    return this;
                }

            };

        }

    }



    /**
     * Evaluate the expression
     *
     * @param context   the dynamic evaluation context
     * @param arguments the values of the arguments, supplied as SequenceIterators
     * @return the result of the evaluation, in the form of a SequenceIterator
     * @throws net.sf.saxon.trans.XPathException if a dynamic error occurs during the evaluation of the expression
     */
    @Override
    public Sequence call(XPathContext context, Sequence[] arguments) throws XPathException {
        AtomicValue hrefVal = (AtomicValue) arguments[0].head();
        if (hrefVal == null) {
            return EmptySequence.getInstance();
        }
        String href = hrefVal.getStringValue();
        PackageData packageData = getRetainedStaticContext().getPackageData();
        NodeInfo item = DocumentFn.makeDoc(href, getRetainedStaticContext().getStaticBaseUriString(), packageData, getParseOptions(), context, null, false);
        if (item == null) {
            // we failed to read the document
            throw new XPathException("Failed to load document " + href, "FODC0002", context);
        }
        Controller controller = context.getController();
        if (parseOptions != null && controller instanceof XsltController) {
            ((XsltController) controller).getAccumulatorManager().setApplicableAccumulators(
                    item.getTreeInfo(), parseOptions.getApplicableAccumulators()
            );
        }
        return item;
    }

    /**
     * Get the static properties of this expression (other than its type). The result is
     * bit-signficant. These properties are used for optimizations. In general, if
     * property bit is set, it is true, but if it is unset, the value is unknown.
     *
     * @param arguments the expressions supplied as arguments to the function
     */

    @Override
    public int getSpecialProperties(Expression[] arguments) {
        return StaticProperty.ORDERED_NODESET |
                StaticProperty.PEER_NODESET |
                StaticProperty.NO_NODES_NEWLY_CREATED |
                StaticProperty.SINGLE_DOCUMENT_NODESET;
        // Declaring it as a peer node-set expression avoids sorting of expressions such as
        // doc(XXX)/a/b/c
        // The doc() function might appear to be creative: but it isn't, because multiple calls
        // with the same arguments will produce identical results.
    }


}





© 2015 - 2024 Weber Informatics LLC | Privacy Policy