File: System\Xml\XPath\Internal\XPathParser.cs
Web Access
Project: src\src\libraries\System.Private.Xml\src\System.Private.Xml.csproj (System.Private.Xml)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Xml;
using System.Xml.XPath;
 
namespace MS.Internal.Xml.XPath
{
    internal struct XPathParser
    {
        private XPathScanner _scanner;
 
        private XPathParser(string xpathExpr)
        {
            _scanner = new XPathScanner(xpathExpr);
            _parseDepth = 0;
        }
 
        public static AstNode ParseXPathExpression(string xpathExpression)
        {
            XPathParser parser = new XPathParser(xpathExpression);
            AstNode result = parser.ParseExpression(null);
            if (parser._scanner.Kind != XPathScanner.LexKind.Eof)
            {
                throw XPathException.Create(SR.Xp_InvalidToken, parser._scanner.SourceText);
            }
            return result;
        }
 
        public static AstNode ParseXPathPattern(string xpathPattern)
        {
            XPathParser parser = new XPathParser(xpathPattern);
            AstNode result = parser.ParsePattern();
            if (parser._scanner.Kind != XPathScanner.LexKind.Eof)
            {
                throw XPathException.Create(SR.Xp_InvalidToken, parser._scanner.SourceText);
            }
            return result;
        }
 
        // --------------- Expression Parsing ----------------------
 
 
        //The recursive is like
        //ParseOrExpr->ParseAndExpr->ParseEqualityExpr->ParseRelationalExpr...->ParseFilterExpr->ParsePredicate->ParseExpression
        //So put 200 limitation here will max cause about 2000~3000 depth stack.
        private int _parseDepth;
        private const int MaxParseDepth = 200;
 
        private AstNode ParseExpression(AstNode? qyInput)
        {
            if (++_parseDepth > MaxParseDepth)
            {
                throw XPathException.Create(SR.Xp_QueryTooComplex);
            }
            AstNode result = ParseOrExpr(qyInput);
            --_parseDepth;
            return result;
        }
 
        //>> OrExpr ::= ( OrExpr 'or' )? AndExpr
        private AstNode ParseOrExpr(AstNode? qyInput)
        {
            AstNode opnd = ParseAndExpr(qyInput);
 
            do
            {
                if (!TestOp("or"))
                {
                    return opnd;
                }
                NextLex();
                opnd = new Operator(Operator.Op.OR, opnd, ParseAndExpr(qyInput));
            } while (true);
        }
 
        //>> AndExpr ::= ( AndExpr 'and' )? EqualityExpr
        private AstNode ParseAndExpr(AstNode? qyInput)
        {
            AstNode opnd = ParseEqualityExpr(qyInput);
 
            do
            {
                if (!TestOp("and"))
                {
                    return opnd;
                }
                NextLex();
                opnd = new Operator(Operator.Op.AND, opnd, ParseEqualityExpr(qyInput));
            } while (true);
        }
 
        //>> EqualityOp ::= '=' | '!='
        //>> EqualityExpr    ::= ( EqualityExpr EqualityOp )? RelationalExpr
        private AstNode ParseEqualityExpr(AstNode? qyInput)
        {
            AstNode opnd = ParseRelationalExpr(qyInput);
 
            do
            {
                Operator.Op op = (
                    _scanner.Kind == XPathScanner.LexKind.Eq ? Operator.Op.EQ :
                    _scanner.Kind == XPathScanner.LexKind.Ne ? Operator.Op.NE :
                    /*default :*/                                  Operator.Op.INVALID
                );
                if (op == Operator.Op.INVALID)
                {
                    return opnd;
                }
                NextLex();
                opnd = new Operator(op, opnd, ParseRelationalExpr(qyInput));
            } while (true);
        }
 
        //>> RelationalOp ::= '<' | '>' | '<=' | '>='
        //>> RelationalExpr    ::= ( RelationalExpr RelationalOp )? AdditiveExpr
        private AstNode ParseRelationalExpr(AstNode? qyInput)
        {
            AstNode opnd = ParseAdditiveExpr(qyInput);
 
            do
            {
                Operator.Op op = (
                    _scanner.Kind == XPathScanner.LexKind.Lt ? Operator.Op.LT :
                    _scanner.Kind == XPathScanner.LexKind.Le ? Operator.Op.LE :
                    _scanner.Kind == XPathScanner.LexKind.Gt ? Operator.Op.GT :
                    _scanner.Kind == XPathScanner.LexKind.Ge ? Operator.Op.GE :
                    /*default :*/                                  Operator.Op.INVALID
                );
                if (op == Operator.Op.INVALID)
                {
                    return opnd;
                }
                NextLex();
                opnd = new Operator(op, opnd, ParseAdditiveExpr(qyInput));
            } while (true);
        }
 
        //>> AdditiveOp   ::= '+' | '-'
        //>> AdditiveExpr ::= ( AdditiveExpr AdditiveOp )? MultiplicativeExpr
        private AstNode ParseAdditiveExpr(AstNode? qyInput)
        {
            AstNode opnd = ParseMultiplicativeExpr(qyInput);
 
            do
            {
                Operator.Op op = (
                    _scanner.Kind == XPathScanner.LexKind.Plus ? Operator.Op.PLUS :
                    _scanner.Kind == XPathScanner.LexKind.Minus ? Operator.Op.MINUS :
                    /*default :*/                                     Operator.Op.INVALID
                );
                if (op == Operator.Op.INVALID)
                {
                    return opnd;
                }
                NextLex();
                opnd = new Operator(op, opnd, ParseMultiplicativeExpr(qyInput));
            } while (true);
        }
 
        //>> MultiplicativeOp   ::= '*' | 'div' | 'mod'
        //>> MultiplicativeExpr ::= ( MultiplicativeExpr MultiplicativeOp )? UnaryExpr
        private AstNode ParseMultiplicativeExpr(AstNode? qyInput)
        {
            AstNode opnd = ParseUnaryExpr(qyInput);
 
            do
            {
                Operator.Op op = (
                    _scanner.Kind == XPathScanner.LexKind.Star ? Operator.Op.MUL :
                    TestOp("div") ? Operator.Op.DIV :
                    TestOp("mod") ? Operator.Op.MOD :
                    /*default :*/                                     Operator.Op.INVALID
                );
                if (op == Operator.Op.INVALID)
                {
                    return opnd;
                }
                NextLex();
                opnd = new Operator(op, opnd, ParseUnaryExpr(qyInput));
            } while (true);
        }
 
        //>> UnaryExpr    ::= UnionExpr | '-' UnaryExpr
        private AstNode ParseUnaryExpr(AstNode? qyInput)
        {
            bool minus = false;
            while (_scanner.Kind == XPathScanner.LexKind.Minus)
            {
                NextLex();
                minus = !minus;
            }
 
            if (minus)
            {
                return new Operator(Operator.Op.MUL, ParseUnionExpr(qyInput), new Operand(-1));
            }
            else
            {
                return ParseUnionExpr(qyInput);
            }
        }
 
        //>> UnionExpr ::= ( UnionExpr '|' )? PathExpr
        private AstNode ParseUnionExpr(AstNode? qyInput)
        {
            AstNode opnd = ParsePathExpr(qyInput);
 
            do
            {
                if (_scanner.Kind != XPathScanner.LexKind.Union)
                {
                    return opnd;
                }
                NextLex();
                AstNode opnd2 = ParsePathExpr(qyInput);
                CheckNodeSet(opnd.ReturnType);
                CheckNodeSet(opnd2.ReturnType);
                opnd = new Operator(Operator.Op.UNION, opnd, opnd2);
            } while (true);
        }
 
        private bool IsNodeType =>
            _scanner.Prefix.Length == 0 &&
            (_scanner.Name == "node" ||
             _scanner.Name == "text" ||
             _scanner.Name == "processing-instruction" ||
             _scanner.Name == "comment");
 
        //>> PathOp   ::= '/' | '//'
        //>> PathExpr ::= LocationPath |
        //>>              FilterExpr ( PathOp  RelativeLocationPath )?
        private AstNode ParsePathExpr(AstNode? qyInput)
        {
            AstNode opnd;
            if (IsPrimaryExpr)
            { // in this moment we should distinct LocationPas vs FilterExpr (which starts from is PrimaryExpr)
                opnd = ParseFilterExpr(qyInput);
                if (_scanner.Kind == XPathScanner.LexKind.Slash)
                {
                    NextLex();
                    opnd = ParseRelativeLocationPath(opnd);
                }
                else if (_scanner.Kind == XPathScanner.LexKind.SlashSlash)
                {
                    NextLex();
                    opnd = ParseRelativeLocationPath(new Axis(Axis.AxisType.DescendantOrSelf, opnd));
                }
            }
            else
            {
                opnd = ParseLocationPath(null);
            }
 
            return opnd;
        }
 
        //>> FilterExpr ::= PrimaryExpr | FilterExpr Predicate
        private AstNode ParseFilterExpr(AstNode? qyInput)
        {
            AstNode opnd = ParsePrimaryExpr(qyInput);
            while (_scanner.Kind == XPathScanner.LexKind.LBracket)
            {
                // opnd must be a query
                opnd = new Filter(opnd, ParsePredicate(opnd));
            }
            return opnd;
        }
 
        //>> Predicate ::= '[' Expr ']'
        private AstNode ParsePredicate(AstNode qyInput)
        {
            AstNode opnd;
 
            // we have predicates. Check that input type is NodeSet
            CheckNodeSet(qyInput.ReturnType);
 
            PassToken(XPathScanner.LexKind.LBracket);
            opnd = ParseExpression(qyInput);
            PassToken(XPathScanner.LexKind.RBracket);
 
            return opnd;
        }
 
        //>> LocationPath ::= RelativeLocationPath | AbsoluteLocationPath
        private AstNode ParseLocationPath(AstNode? qyInput)
        {
            if (_scanner.Kind == XPathScanner.LexKind.Slash)
            {
                NextLex();
                AstNode opnd = new Root();
 
                if (IsStep(_scanner.Kind))
                {
                    opnd = ParseRelativeLocationPath(opnd);
                }
                return opnd;
            }
            else if (_scanner.Kind == XPathScanner.LexKind.SlashSlash)
            {
                NextLex();
                return ParseRelativeLocationPath(new Axis(Axis.AxisType.DescendantOrSelf, new Root()));
            }
            else
            {
                return ParseRelativeLocationPath(qyInput);
            }
        } // ParseLocationPath
 
        //>> PathOp   ::= '/' | '//'
        //>> RelativeLocationPath ::= ( RelativeLocationPath PathOp )? Step
        private AstNode ParseRelativeLocationPath(AstNode? qyInput)
        {
            AstNode? opnd = qyInput;
            do
            {
                opnd = ParseStep(opnd);
                if (XPathScanner.LexKind.SlashSlash == _scanner.Kind)
                {
                    NextLex();
                    opnd = new Axis(Axis.AxisType.DescendantOrSelf, opnd);
                }
                else if (XPathScanner.LexKind.Slash == _scanner.Kind)
                {
                    NextLex();
                }
                else
                {
                    break;
                }
            }
            while (true);
 
            return opnd;
        }
 
        private static bool IsStep(XPathScanner.LexKind lexKind)
        {
            return (
                lexKind == XPathScanner.LexKind.Dot ||
                lexKind == XPathScanner.LexKind.DotDot ||
                lexKind == XPathScanner.LexKind.At ||
                lexKind == XPathScanner.LexKind.Axe ||
                lexKind == XPathScanner.LexKind.Star ||
                lexKind == XPathScanner.LexKind.Name          // NodeTest is also Name
            );
        }
 
        //>> Step ::= '.' | '..' | ( AxisName '::' | '@' )? NodeTest Predicate*
        private AstNode ParseStep(AstNode? qyInput)
        {
            AstNode opnd;
            if (XPathScanner.LexKind.Dot == _scanner.Kind)
            {         //>> '.'
                NextLex();
                opnd = new Axis(Axis.AxisType.Self, qyInput);
            }
            else if (XPathScanner.LexKind.DotDot == _scanner.Kind)
            { //>> '..'
                NextLex();
                opnd = new Axis(Axis.AxisType.Parent, qyInput);
            }
            else
            {                                                          //>> ( AxisName '::' | '@' )? NodeTest Predicate*
                Axis.AxisType axisType = Axis.AxisType.Child;
                switch (_scanner.Kind)
                {
                    case XPathScanner.LexKind.At:                               //>> '@'
                        axisType = Axis.AxisType.Attribute;
                        NextLex();
                        break;
                    case XPathScanner.LexKind.Axe:                              //>> AxisName '::'
                        axisType = GetAxis();
                        NextLex();
                        break;
                }
                XPathNodeType nodeType = (
                    axisType == Axis.AxisType.Attribute ? XPathNodeType.Attribute :
                    //                    axisType == Axis.AxisType.Namespace ? XPathNodeType.Namespace : // No Idea why it's this way but otherwise Axes doesn't work
                    /* default: */                        XPathNodeType.Element
                );
 
                opnd = ParseNodeTest(qyInput, axisType, nodeType);
 
                while (XPathScanner.LexKind.LBracket == _scanner.Kind)
                {
                    opnd = new Filter(opnd, ParsePredicate(opnd));
                }
            }
            return opnd;
        }
 
        //>> NodeTest ::= NameTest | 'comment ()' | 'text ()' | 'node ()' | 'processing-instruction ('  Literal ? ')'
        private Axis ParseNodeTest(AstNode? qyInput, Axis.AxisType axisType, XPathNodeType nodeType)
        {
            string nodeName, nodePrefix;
 
            switch (_scanner.Kind)
            {
                case XPathScanner.LexKind.Name:
                    if (_scanner.CanBeFunction && IsNodeType)
                    {
                        nodePrefix = string.Empty;
                        nodeName = string.Empty;
                        nodeType = (
                            _scanner.Name == "comment" ? XPathNodeType.Comment :
                            _scanner.Name == "text" ? XPathNodeType.Text :
                            _scanner.Name == "node" ? XPathNodeType.All :
                            _scanner.Name == "processing-instruction" ? XPathNodeType.ProcessingInstruction :
                            /* default: */ XPathNodeType.Root
                        );
                        Debug.Assert(nodeType != XPathNodeType.Root);
                        NextLex();
 
                        PassToken(XPathScanner.LexKind.LParens);
 
                        if (nodeType == XPathNodeType.ProcessingInstruction)
                        {
                            if (_scanner.Kind != XPathScanner.LexKind.RParens)
                            { //>> 'processing-instruction (' Literal ')'
                                CheckToken(XPathScanner.LexKind.String);
                                nodeName = _scanner.StringValue;
                                NextLex();
                            }
                        }
 
                        PassToken(XPathScanner.LexKind.RParens);
                    }
                    else
                    {
                        nodePrefix = _scanner.Prefix;
                        nodeName = _scanner.Name;
                        NextLex();
                        if (nodeName == "*")
                        {
                            nodeName = string.Empty;
                        }
                    }
                    break;
                case XPathScanner.LexKind.Star:
                    nodePrefix = string.Empty;
                    nodeName = string.Empty;
                    NextLex();
                    break;
                default:
                    throw XPathException.Create(SR.Xp_NodeSetExpected, _scanner.SourceText);
            }
            return new Axis(axisType, qyInput, nodePrefix, nodeName, nodeType);
        }
 
        private bool IsPrimaryExpr =>
            _scanner.Kind == XPathScanner.LexKind.String ||
            _scanner.Kind == XPathScanner.LexKind.Number ||
            _scanner.Kind == XPathScanner.LexKind.Dollar ||
            _scanner.Kind == XPathScanner.LexKind.LParens ||
            _scanner.Kind == XPathScanner.LexKind.Name && _scanner.CanBeFunction && !IsNodeType;
 
        //>> PrimaryExpr ::= Literal | Number | VariableReference | '(' Expr ')' | FunctionCall
        private AstNode ParsePrimaryExpr(AstNode? qyInput)
        {
            Debug.Assert(IsPrimaryExpr);
            AstNode? opnd = null;
            switch (_scanner.Kind)
            {
                case XPathScanner.LexKind.String:
                    opnd = new Operand(_scanner.StringValue);
                    NextLex();
                    break;
                case XPathScanner.LexKind.Number:
                    opnd = new Operand(_scanner.NumberValue);
                    NextLex();
                    break;
                case XPathScanner.LexKind.Dollar:
                    NextLex();
                    CheckToken(XPathScanner.LexKind.Name);
                    opnd = new Variable(_scanner.Name, _scanner.Prefix);
                    NextLex();
                    break;
                case XPathScanner.LexKind.LParens:
                    NextLex();
                    opnd = ParseExpression(qyInput);
                    if (opnd.Type != AstNode.AstType.ConstantOperand)
                    {
                        opnd = new Group(opnd);
                    }
                    PassToken(XPathScanner.LexKind.RParens);
                    break;
                case XPathScanner.LexKind.Name:
                    if (_scanner.CanBeFunction && !IsNodeType)
                    {
                        opnd = ParseMethod(null);
                    }
                    break;
            }
            Debug.Assert(opnd != null, "IsPrimaryExpr() was true. We should recognize this lex.");
            return opnd;
        }
 
        private Function ParseMethod(AstNode? qyInput)
        {
            List<AstNode> argList = new List<AstNode>();
            string name = _scanner.Name;
            string prefix = _scanner.Prefix;
            PassToken(XPathScanner.LexKind.Name);
            PassToken(XPathScanner.LexKind.LParens);
            if (_scanner.Kind != XPathScanner.LexKind.RParens)
            {
                do
                {
                    argList.Add(ParseExpression(qyInput));
                    if (_scanner.Kind == XPathScanner.LexKind.RParens)
                    {
                        break;
                    }
                    PassToken(XPathScanner.LexKind.Comma);
                } while (true);
            }
            PassToken(XPathScanner.LexKind.RParens);
            if (prefix.Length == 0)
            {
                ParamInfo? pi;
                if (s_functionTable.TryGetValue(name, out pi))
                {
                    int argCount = argList.Count;
                    if (argCount < pi.Minargs)
                    {
                        throw XPathException.Create(SR.Xp_InvalidNumArgs, name, _scanner.SourceText);
                    }
                    if (pi.FType == Function.FunctionType.FuncConcat)
                    {
                        for (int i = 0; i < argCount; i++)
                        {
                            AstNode arg = (AstNode)argList[i];
                            if (arg.ReturnType != XPathResultType.String)
                            {
                                arg = new Function(Function.FunctionType.FuncString, arg);
                            }
                            argList[i] = arg;
                        }
                    }
                    else
                    {
                        if (pi.Maxargs < argCount)
                        {
                            throw XPathException.Create(SR.Xp_InvalidNumArgs, name, _scanner.SourceText);
                        }
                        if (pi.ArgTypes.Length < argCount)
                        {
                            argCount = pi.ArgTypes.Length;    // argument we have the type specified (can be < pi.Minargs)
                        }
                        for (int i = 0; i < argCount; i++)
                        {
                            AstNode arg = (AstNode)argList[i];
                            if (
                                pi.ArgTypes[i] != XPathResultType.Any &&
                                pi.ArgTypes[i] != arg.ReturnType
                            )
                            {
                                switch (pi.ArgTypes[i])
                                {
                                    case XPathResultType.NodeSet:
                                        if (!(arg is Variable) && !(arg is Function && arg.ReturnType == XPathResultType.Any))
                                        {
                                            throw XPathException.Create(SR.Xp_InvalidArgumentType, name, _scanner.SourceText);
                                        }
                                        break;
                                    case XPathResultType.String:
                                        arg = new Function(Function.FunctionType.FuncString, arg);
                                        break;
                                    case XPathResultType.Number:
                                        arg = new Function(Function.FunctionType.FuncNumber, arg);
                                        break;
                                    case XPathResultType.Boolean:
                                        arg = new Function(Function.FunctionType.FuncBoolean, arg);
                                        break;
                                }
                                argList[i] = arg;
                            }
                        }
                    }
                    return new Function(pi.FType, argList);
                }
            }
            return new Function(prefix, name, argList);
        }
 
        // --------------- Pattern Parsing ----------------------
 
        //>> Pattern ::= ( Pattern '|' )? LocationPathPattern
        private AstNode ParsePattern()
        {
            AstNode opnd = ParseLocationPathPattern();
 
            do
            {
                if (_scanner.Kind != XPathScanner.LexKind.Union)
                {
                    return opnd;
                }
                NextLex();
                opnd = new Operator(Operator.Op.UNION, opnd, ParseLocationPathPattern());
            } while (true);
        }
 
        //>> LocationPathPattern ::= '/' | RelativePathPattern | '//' RelativePathPattern  |  '/' RelativePathPattern
        //>>                       | IdKeyPattern (('/' | '//') RelativePathPattern)?
        private AstNode ParseLocationPathPattern()
        {
            AstNode? opnd = null;
            switch (_scanner.Kind)
            {
                case XPathScanner.LexKind.Slash:
                    NextLex();
                    opnd = new Root();
                    if (_scanner.Kind == XPathScanner.LexKind.Eof || _scanner.Kind == XPathScanner.LexKind.Union)
                    {
                        return opnd;
                    }
                    break;
                case XPathScanner.LexKind.SlashSlash:
                    NextLex();
                    opnd = new Axis(Axis.AxisType.DescendantOrSelf, new Root());
                    break;
                case XPathScanner.LexKind.Name:
                    if (_scanner.CanBeFunction)
                    {
                        opnd = ParseIdKeyPattern();
                        if (opnd != null)
                        {
                            switch (_scanner.Kind)
                            {
                                case XPathScanner.LexKind.Slash:
                                    NextLex();
                                    break;
                                case XPathScanner.LexKind.SlashSlash:
                                    NextLex();
                                    opnd = new Axis(Axis.AxisType.DescendantOrSelf, opnd);
                                    break;
                                default:
                                    return opnd;
                            }
                        }
                    }
                    break;
            }
            return ParseRelativePathPattern(opnd);
        }
 
        //>> IdKeyPattern ::= 'id' '(' Literal ')' | 'key' '(' Literal ',' Literal ')'
        private Function? ParseIdKeyPattern()
        {
            Debug.Assert(_scanner.CanBeFunction);
            List<AstNode> argList = new List<AstNode>();
            if (_scanner.Prefix.Length == 0)
            {
                if (_scanner.Name == "id")
                {
                    ParamInfo pi = (ParamInfo)s_functionTable["id"];
                    NextLex();
                    PassToken(XPathScanner.LexKind.LParens);
                    CheckToken(XPathScanner.LexKind.String);
                    argList.Add(new Operand(_scanner.StringValue));
                    NextLex();
                    PassToken(XPathScanner.LexKind.RParens);
                    return new Function(pi.FType, argList);
                }
                if (_scanner.Name == "key")
                {
                    NextLex();
                    PassToken(XPathScanner.LexKind.LParens);
                    CheckToken(XPathScanner.LexKind.String);
                    argList.Add(new Operand(_scanner.StringValue));
                    NextLex();
                    PassToken(XPathScanner.LexKind.Comma);
                    CheckToken(XPathScanner.LexKind.String);
                    argList.Add(new Operand(_scanner.StringValue));
                    NextLex();
                    PassToken(XPathScanner.LexKind.RParens);
                    return new Function("", "key", argList);
                }
            }
            return null;
        }
 
        //>> PathOp   ::= '/' | '//'
        //>> RelativePathPattern ::= ( RelativePathPattern PathOp )? StepPattern
        private AstNode ParseRelativePathPattern(AstNode? qyInput)
        {
            AstNode opnd = ParseStepPattern(qyInput);
            if (XPathScanner.LexKind.SlashSlash == _scanner.Kind)
            {
                NextLex();
                opnd = ParseRelativePathPattern(new Axis(Axis.AxisType.DescendantOrSelf, opnd));
            }
            else if (XPathScanner.LexKind.Slash == _scanner.Kind)
            {
                NextLex();
                opnd = ParseRelativePathPattern(opnd);
            }
            return opnd;
        }
 
        //>> StepPattern    ::=    ChildOrAttributeAxisSpecifier NodeTest Predicate*
        //>> ChildOrAttributeAxisSpecifier    ::=    @ ? | ('child' | 'attribute') '::'
        private AstNode ParseStepPattern(AstNode? qyInput)
        {
            AstNode opnd;
            Axis.AxisType axisType = Axis.AxisType.Child;
            switch (_scanner.Kind)
            {
                case XPathScanner.LexKind.At:                               //>> '@'
                    axisType = Axis.AxisType.Attribute;
                    NextLex();
                    break;
                case XPathScanner.LexKind.Axe:                              //>> AxisName '::'
                    axisType = GetAxis();
                    if (axisType != Axis.AxisType.Child && axisType != Axis.AxisType.Attribute)
                    {
                        throw XPathException.Create(SR.Xp_InvalidToken, _scanner.SourceText);
                    }
                    NextLex();
                    break;
            }
            XPathNodeType nodeType = (
                axisType == Axis.AxisType.Attribute ? XPathNodeType.Attribute :
                /* default: */                        XPathNodeType.Element
            );
 
            opnd = ParseNodeTest(qyInput, axisType, nodeType);
 
            while (XPathScanner.LexKind.LBracket == _scanner.Kind)
            {
                opnd = new Filter(opnd, ParsePredicate(opnd));
            }
            return opnd;
        }
 
        // --------------- Helper methods ----------------------
 
        private void CheckToken(XPathScanner.LexKind t)
        {
            if (_scanner.Kind != t)
            {
                throw XPathException.Create(SR.Xp_InvalidToken, _scanner.SourceText);
            }
        }
 
        private void PassToken(XPathScanner.LexKind t)
        {
            CheckToken(t);
            NextLex();
        }
 
        private void NextLex()
        {
            _scanner.NextLex();
        }
 
        private bool TestOp(string op)
        {
            return (
                _scanner.Kind == XPathScanner.LexKind.Name &&
                _scanner.Prefix.Length == 0 &&
                _scanner.Name.Equals(op)
            );
        }
 
        private void CheckNodeSet(XPathResultType t)
        {
            if (t != XPathResultType.NodeSet && t != XPathResultType.Any)
            {
                throw XPathException.Create(SR.Xp_NodeSetExpected, _scanner.SourceText);
            }
        }
 
        // ----------------------------------------------------------------
        private static readonly XPathResultType[] s_temparray1 = Array.Empty<XPathResultType>();
        private static readonly XPathResultType[] s_temparray2 = { XPathResultType.NodeSet };
        private static readonly XPathResultType[] s_temparray3 = { XPathResultType.Any };
        private static readonly XPathResultType[] s_temparray4 = { XPathResultType.String };
        private static readonly XPathResultType[] s_temparray5 = { XPathResultType.String, XPathResultType.String };
        private static readonly XPathResultType[] s_temparray6 = { XPathResultType.String, XPathResultType.Number, XPathResultType.Number };
        private static readonly XPathResultType[] s_temparray7 = { XPathResultType.String, XPathResultType.String, XPathResultType.String };
        private static readonly XPathResultType[] s_temparray8 = { XPathResultType.Boolean };
        private static readonly XPathResultType[] s_temparray9 = { XPathResultType.Number };
 
        private sealed class ParamInfo
        {
            private readonly Function.FunctionType _ftype;
            private readonly int _minargs;
            private readonly int _maxargs;
            private readonly XPathResultType[] _argTypes;
 
            public Function.FunctionType FType { get { return _ftype; } }
            public int Minargs { get { return _minargs; } }
            public int Maxargs { get { return _maxargs; } }
            public XPathResultType[] ArgTypes { get { return _argTypes; } }
 
            internal ParamInfo(Function.FunctionType ftype, int minargs, int maxargs, XPathResultType[] argTypes)
            {
                _ftype = ftype;
                _minargs = minargs;
                _maxargs = maxargs;
                _argTypes = argTypes;
            }
        } //ParamInfo
 
        private static readonly Dictionary<string, ParamInfo> s_functionTable = CreateFunctionTable();
        private static Dictionary<string, ParamInfo> CreateFunctionTable()
        {
            Dictionary<string, ParamInfo> table = new Dictionary<string, ParamInfo>(36);
            table.Add("last", new ParamInfo(Function.FunctionType.FuncLast, 0, 0, s_temparray1));
            table.Add("position", new ParamInfo(Function.FunctionType.FuncPosition, 0, 0, s_temparray1));
            table.Add("name", new ParamInfo(Function.FunctionType.FuncName, 0, 1, s_temparray2));
            table.Add("namespace-uri", new ParamInfo(Function.FunctionType.FuncNameSpaceUri, 0, 1, s_temparray2));
            table.Add("local-name", new ParamInfo(Function.FunctionType.FuncLocalName, 0, 1, s_temparray2));
            table.Add("count", new ParamInfo(Function.FunctionType.FuncCount, 1, 1, s_temparray2));
            table.Add("id", new ParamInfo(Function.FunctionType.FuncID, 1, 1, s_temparray3));
            table.Add("string", new ParamInfo(Function.FunctionType.FuncString, 0, 1, s_temparray3));
            table.Add("concat", new ParamInfo(Function.FunctionType.FuncConcat, 2, 100, s_temparray4));
            table.Add("starts-with", new ParamInfo(Function.FunctionType.FuncStartsWith, 2, 2, s_temparray5));
            table.Add("contains", new ParamInfo(Function.FunctionType.FuncContains, 2, 2, s_temparray5));
            table.Add("substring-before", new ParamInfo(Function.FunctionType.FuncSubstringBefore, 2, 2, s_temparray5));
            table.Add("substring-after", new ParamInfo(Function.FunctionType.FuncSubstringAfter, 2, 2, s_temparray5));
            table.Add("substring", new ParamInfo(Function.FunctionType.FuncSubstring, 2, 3, s_temparray6));
            table.Add("string-length", new ParamInfo(Function.FunctionType.FuncStringLength, 0, 1, s_temparray4));
            table.Add("normalize-space", new ParamInfo(Function.FunctionType.FuncNormalize, 0, 1, s_temparray4));
            table.Add("translate", new ParamInfo(Function.FunctionType.FuncTranslate, 3, 3, s_temparray7));
            table.Add("boolean", new ParamInfo(Function.FunctionType.FuncBoolean, 1, 1, s_temparray3));
            table.Add("not", new ParamInfo(Function.FunctionType.FuncNot, 1, 1, s_temparray8));
            table.Add("true", new ParamInfo(Function.FunctionType.FuncTrue, 0, 0, s_temparray8));
            table.Add("false", new ParamInfo(Function.FunctionType.FuncFalse, 0, 0, s_temparray8));
            table.Add("lang", new ParamInfo(Function.FunctionType.FuncLang, 1, 1, s_temparray4));
            table.Add("number", new ParamInfo(Function.FunctionType.FuncNumber, 0, 1, s_temparray3));
            table.Add("sum", new ParamInfo(Function.FunctionType.FuncSum, 1, 1, s_temparray2));
            table.Add("floor", new ParamInfo(Function.FunctionType.FuncFloor, 1, 1, s_temparray9));
            table.Add("ceiling", new ParamInfo(Function.FunctionType.FuncCeiling, 1, 1, s_temparray9));
            table.Add("round", new ParamInfo(Function.FunctionType.FuncRound, 1, 1, s_temparray9));
            return table;
        }
 
        private static readonly Dictionary<string, Axis.AxisType> s_AxesTable = CreateAxesTable();
        private static Dictionary<string, Axis.AxisType> CreateAxesTable()
        {
            Dictionary<string, Axis.AxisType> table = new Dictionary<string, Axis.AxisType>(13);
            table.Add("ancestor", Axis.AxisType.Ancestor);
            table.Add("ancestor-or-self", Axis.AxisType.AncestorOrSelf);
            table.Add("attribute", Axis.AxisType.Attribute);
            table.Add("child", Axis.AxisType.Child);
            table.Add("descendant", Axis.AxisType.Descendant);
            table.Add("descendant-or-self", Axis.AxisType.DescendantOrSelf);
            table.Add("following", Axis.AxisType.Following);
            table.Add("following-sibling", Axis.AxisType.FollowingSibling);
            table.Add("namespace", Axis.AxisType.Namespace);
            table.Add("parent", Axis.AxisType.Parent);
            table.Add("preceding", Axis.AxisType.Preceding);
            table.Add("preceding-sibling", Axis.AxisType.PrecedingSibling);
            table.Add("self", Axis.AxisType.Self);
            return table;
        }
 
        private Axis.AxisType GetAxis()
        {
            Debug.Assert(_scanner.Kind == XPathScanner.LexKind.Axe);
            Axis.AxisType axis;
            if (!s_AxesTable.TryGetValue(_scanner.Name, out axis))
            {
                throw XPathException.Create(SR.Xp_InvalidToken, _scanner.SourceText);
            }
            return axis;
        }
    }
}