//------------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
// Microsoft
//------------------------------------------------------------------------------
using System.Collections.Generic;
using System.Diagnostics;
using System.Xml;
using System.Xml.Xsl.Qil;
using System.Xml.Xsl.XPath;
namespace System.Xml.Xsl.Xslt {
using XPathParser = XPathParser;
using XPathNodeType = System.Xml.XPath.XPathNodeType;
using Res = System.Xml.Utils.Res;
internal class XPathPatternParser {
public interface IPatternBuilder : IXPathBuilder {
IXPathBuilder GetPredicateBuilder(QilNode context);
}
XPathScanner scanner;
IPatternBuilder ptrnBuilder;
XPathParser predicateParser = new XPathParser();
public QilNode Parse(XPathScanner scanner, IPatternBuilder ptrnBuilder) {
Debug.Assert(this.scanner == null && this.ptrnBuilder == null);
Debug.Assert(scanner != null && ptrnBuilder != null);
QilNode result = null;
ptrnBuilder.StartBuild();
try {
this.scanner = scanner;
this.ptrnBuilder = ptrnBuilder;
result = this.ParsePattern();
this.scanner.CheckToken(LexKind.Eof);
} finally {
result = ptrnBuilder.EndBuild(result);
#if DEBUG
this.ptrnBuilder = null;
this.scanner = null;
#endif
}
return result;
}
/*
* Pattern ::= LocationPathPattern ('|' LocationPathPattern)*
*/
private QilNode ParsePattern() {
QilNode opnd = ParseLocationPathPattern();
while (scanner.Kind == LexKind.Union) {
scanner.NextLex();
opnd = ptrnBuilder.Operator(XPathOperator.Union, opnd, ParseLocationPathPattern());
}
return opnd;
}
/*
* LocationPathPattern ::= '/' RelativePathPattern? | '//'? RelativePathPattern | IdKeyPattern (('/' | '//') RelativePathPattern)?
*/
private QilNode ParseLocationPathPattern() {
QilNode opnd;
switch (scanner.Kind) {
case LexKind.Slash :
scanner.NextLex();
opnd = ptrnBuilder.Axis(XPathAxis.Root, XPathNodeType.All, null, null);
if (XPathParser.IsStep(scanner.Kind)) {
opnd = ptrnBuilder.JoinStep(opnd, ParseRelativePathPattern());
}
return opnd;
case LexKind.SlashSlash :
scanner.NextLex();
return ptrnBuilder.JoinStep(
ptrnBuilder.Axis(XPathAxis.Root, XPathNodeType.All, null, null),
ptrnBuilder.JoinStep(
ptrnBuilder.Axis(XPathAxis.DescendantOrSelf, XPathNodeType.All, null, null),
ParseRelativePathPattern()
)
);
case LexKind.Name :
if (scanner.CanBeFunction && scanner.Prefix.Length == 0 && (scanner.Name == "id" || scanner.Name == "key")) {
opnd = ParseIdKeyPattern();
switch (scanner.Kind) {
case LexKind.Slash :
scanner.NextLex();
opnd = ptrnBuilder.JoinStep(opnd, ParseRelativePathPattern());
break;
case LexKind.SlashSlash :
scanner.NextLex();
opnd = ptrnBuilder.JoinStep(opnd,
ptrnBuilder.JoinStep(
ptrnBuilder.Axis(XPathAxis.DescendantOrSelf, XPathNodeType.All, null, null),
ParseRelativePathPattern()
)
);
break;
}
return opnd;
}
break;
}
opnd = ParseRelativePathPattern();
return opnd;
}
/*
* IdKeyPattern ::= 'id' '(' Literal ')' | 'key' '(' Literal ',' Literal ')'
*/
private QilNode ParseIdKeyPattern() {
Debug.Assert(scanner.CanBeFunction);
Debug.Assert(scanner.Prefix.Length == 0);
Debug.Assert(scanner.Name == "id" || scanner.Name == "key");
List args = new List(2);
if (scanner.Name == "id") {
scanner.NextLex();
scanner.PassToken(LexKind.LParens);
scanner.CheckToken(LexKind.String);
args.Add(ptrnBuilder.String(scanner.StringValue));
scanner.NextLex();
scanner.PassToken(LexKind.RParens);
return ptrnBuilder.Function("", "id", args);
} else {
scanner.NextLex();
scanner.PassToken(LexKind.LParens);
scanner.CheckToken(LexKind.String);
args.Add(ptrnBuilder.String(scanner.StringValue));
scanner.NextLex();
scanner.PassToken(LexKind.Comma);
scanner.CheckToken(LexKind.String);
args.Add(ptrnBuilder.String(scanner.StringValue));
scanner.NextLex();
scanner.PassToken(LexKind.RParens);
return ptrnBuilder.Function("", "key", args);
}
}
/*
* RelativePathPattern ::= StepPattern (('/' | '//') StepPattern)*
*/
//Max depth to avoid StackOverflow
const int MaxParseRelativePathDepth = 1024;
private int parseRelativePath = 0;
private QilNode ParseRelativePathPattern() {
if (++parseRelativePath > MaxParseRelativePathDepth) {
if (System.Xml.XmlConfiguration.XsltConfigSection.LimitXPathComplexity) {
throw scanner.CreateException(System.Xml.Utils.Res.Xslt_InputTooComplex);
}
}
QilNode opnd = ParseStepPattern();
if (scanner.Kind == LexKind.Slash) {
scanner.NextLex();
opnd = ptrnBuilder.JoinStep(opnd, ParseRelativePathPattern());
} else if (scanner.Kind == LexKind.SlashSlash) {
scanner.NextLex();
opnd = ptrnBuilder.JoinStep(opnd,
ptrnBuilder.JoinStep(
ptrnBuilder.Axis(XPathAxis.DescendantOrSelf, XPathNodeType.All, null, null),
ParseRelativePathPattern()
)
);
}
--parseRelativePath;
return opnd;
}
/*
* StepPattern ::= ChildOrAttributeAxisSpecifier NodeTest Predicate*
* ChildOrAttributeAxisSpecifier ::= @ ? | ('child' | 'attribute') '::'
*/
private QilNode ParseStepPattern() {
QilNode opnd;
XPathAxis axis;
switch (scanner.Kind) {
case LexKind.Dot:
case LexKind.DotDot:
throw scanner.CreateException(Res.XPath_InvalidAxisInPattern);
case LexKind.At:
axis = XPathAxis.Attribute;
scanner.NextLex();
break;
case LexKind.Axis:
axis = scanner.Axis;
if (axis != XPathAxis.Child && axis != XPathAxis.Attribute) {
throw scanner.CreateException(Res.XPath_InvalidAxisInPattern);
}
scanner.NextLex(); // Skip '::'
scanner.NextLex();
break;
case LexKind.Name:
case LexKind.Star:
// NodeTest must start with Name or '*'
axis = XPathAxis.Child;
break;
default:
throw scanner.CreateException(Res.XPath_UnexpectedToken, scanner.RawValue);
}
XPathNodeType nodeType;
string nodePrefix, nodeName;
XPathParser.InternalParseNodeTest(scanner, axis, out nodeType, out nodePrefix, out nodeName);
opnd = ptrnBuilder.Axis(axis, nodeType, nodePrefix, nodeName);
XPathPatternBuilder xpathPatternBuilder = ptrnBuilder as XPathPatternBuilder;
if (xpathPatternBuilder != null) {
//for XPathPatternBuilder, get all predicates and then build them
List predicates = new List();
while (scanner.Kind == LexKind.LBracket) {
predicates.Add(ParsePredicate(opnd));
}
if (predicates.Count > 0)
opnd = xpathPatternBuilder.BuildPredicates(opnd, predicates);
}
else {
while (scanner.Kind == LexKind.LBracket) {
opnd = ptrnBuilder.Predicate(opnd, ParsePredicate(opnd), /*reverseStep:*/false);
}
}
return opnd;
}
/*
* Predicate ::= '[' Expr ']'
*/
private QilNode ParsePredicate(QilNode context) {
Debug.Assert(scanner.Kind == LexKind.LBracket);
scanner.NextLex();
QilNode result = predicateParser.Parse(scanner, ptrnBuilder.GetPredicateBuilder(context), LexKind.RBracket);
Debug.Assert(scanner.Kind == LexKind.RBracket);
scanner.NextLex();
return result;
}
}
}