e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
226 lines
8.2 KiB
C#
226 lines
8.2 KiB
C#
//---------------------------------------------------------------------
|
|
// <copyright file="CqlParserHelpers.cs" company="Microsoft">
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
// </copyright>
|
|
//
|
|
// @owner [....]
|
|
// @backupOwner [....]
|
|
//---------------------------------------------------------------------
|
|
|
|
namespace System.Data.Common.EntitySql
|
|
{
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Data.Common.EntitySql.AST;
|
|
using System.Data.Entity;
|
|
using System.Globalization;
|
|
|
|
/// <summary>
|
|
/// Represents the Cql Parser engine. Also, implements helpers and util routines.
|
|
/// </summary>
|
|
internal sealed partial class CqlParser
|
|
{
|
|
private Node _parsedTree;
|
|
private CqlLexer _lexer;
|
|
private string _query;
|
|
private ParserOptions _parserOptions;
|
|
private const string _internalYaccSyntaxErrorMessage = "syntax error";
|
|
/// <summary>
|
|
/// Contains inclusive count of method expressions.
|
|
/// </summary>
|
|
private uint _methodExprCounter;
|
|
private Stack<uint> _methodExprCounterStack;
|
|
|
|
private string _version = YYMAJOR.ToString(NumberFormatInfo.InvariantInfo) + '.' + YYMINOR.ToString(NumberFormatInfo.InvariantInfo);
|
|
|
|
internal CqlParser(ParserOptions parserOptions, bool debug)
|
|
{
|
|
// The common practice is to make the null check at the public surface,
|
|
// however this method is a convergence zone from multiple public entry points and it makes sense to
|
|
// check for null once, here.
|
|
EntityUtil.CheckArgumentNull(parserOptions, "parserOptions");
|
|
|
|
_parserOptions = parserOptions;
|
|
yydebug = debug;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Main entry point for parsing cql.
|
|
/// </summary>
|
|
/// <param name="query">query text</param>
|
|
/// <exception cref="System.Data.EntityException">Thrown when Syntatic rules are violated and the query cannot be accepted</exception>
|
|
/// <returns>Abstract Syntax Tree</returns>
|
|
internal Node Parse(string query)
|
|
{
|
|
// The common practice is to make the null check at the public surface,
|
|
// however this method is a convergence zone from multiple public entry points and it makes sense to
|
|
// check for null once, here.
|
|
EntityUtil.CheckArgumentNull(query, "query");
|
|
if (String.IsNullOrEmpty(query) || query.Trim().Length == 0)
|
|
{
|
|
throw EntityUtil.Argument(System.Data.Entity.Strings.InvalidEmptyQueryTextArgument);
|
|
}
|
|
|
|
_query = query;
|
|
_parsedTree = null;
|
|
_methodExprCounter = 0;
|
|
_methodExprCounterStack = new Stack<uint>();
|
|
internalParseEntryPoint();
|
|
return _parsedTree;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns query string
|
|
/// </summary>
|
|
internal string Query
|
|
{
|
|
get { return _query; }
|
|
}
|
|
|
|
#if ENTITYSQL_PARSER_YYDEBUG
|
|
/// <summary>
|
|
/// Enables/Disables yacc debugging.
|
|
/// </summary>
|
|
internal bool EnableDebug
|
|
{
|
|
get { return yydebug; }
|
|
set { yydebug = value; }
|
|
}
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Returns ParserOptions used
|
|
/// </summary>
|
|
/// <remarks>Once parse has been invoked, ParserOptions are frozen and cannot be changed. otherwise a EntityException exception will be thrown</remarks>
|
|
internal ParserOptions ParserOptions
|
|
{
|
|
get { return _parserOptions; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Internal entry point
|
|
/// </summary>
|
|
private void internalParseEntryPoint()
|
|
{
|
|
_lexer = new CqlLexer(Query, ParserOptions);
|
|
#if ENTITYSQL_PARSER_YYDEBUG
|
|
CqlLexer.Token tk = lexer.yylex();
|
|
while (null != tk)
|
|
{
|
|
Console.WriteLine("{0} := {1}", tk.TokenId, lexer.yytext());
|
|
tk = lexer.yylex();
|
|
}
|
|
#endif
|
|
yyparse();
|
|
}
|
|
|
|
//
|
|
// Conversion/Cast/Helpers
|
|
//
|
|
private static Node AstNode(object o) { return ((Node)o); }
|
|
private static int AstNodePos( object o ) { return ((Node)o).ErrCtx.InputPosition; }
|
|
private static CqlLexer.TerminalToken Terminal( object o ) { return ((CqlLexer.TerminalToken)o); }
|
|
private static int TerminalPos( object o ) { return ((CqlLexer.TerminalToken)o).IPos; }
|
|
private static NodeList<T> ToNodeList<T>(object o) where T : Node { return ((NodeList<T>)o); }
|
|
|
|
private short yylex()
|
|
{
|
|
CqlLexer.Token token = null;
|
|
token = _lexer.yylex();
|
|
if (null == token)
|
|
{
|
|
return 0;
|
|
}
|
|
_lexer.AdvanceIPos();
|
|
yylval = token.Value;
|
|
return token.TokenId;
|
|
}
|
|
|
|
private void yyerror_stackoverflow()
|
|
{
|
|
yyerror(System.Data.Entity.Strings.StackOverflowInParser);
|
|
}
|
|
|
|
private void yyerror( string s )
|
|
{
|
|
if (s.Equals(_internalYaccSyntaxErrorMessage, StringComparison.Ordinal))
|
|
{
|
|
int errorPosition = _lexer.IPos;
|
|
string syntaxContextInfo = null;
|
|
string term = _lexer.YYText;
|
|
if (!String.IsNullOrEmpty(term))
|
|
{
|
|
syntaxContextInfo = System.Data.Entity.Strings.LocalizedTerm;
|
|
ErrorContext errCtx = null;
|
|
Node astNode = yylval as Node;
|
|
if (null != astNode && (null != astNode.ErrCtx) && (!String.IsNullOrEmpty(astNode.ErrCtx.ErrorContextInfo)))
|
|
{
|
|
errCtx = astNode.ErrCtx;
|
|
errorPosition = Math.Min(errorPosition, errorPosition - term.Length);
|
|
}
|
|
|
|
if ((yylval is CqlLexer.TerminalToken) && CqlLexer.IsReservedKeyword(term) && !(astNode is Identifier))
|
|
{
|
|
syntaxContextInfo = System.Data.Entity.Strings.LocalizedKeyword;
|
|
term = term.ToUpperInvariant();
|
|
errorPosition = Math.Min(errorPosition, errorPosition - term.Length);
|
|
}
|
|
else if (null != errCtx)
|
|
{
|
|
syntaxContextInfo = EntityRes.GetString(errCtx.ErrorContextInfo);
|
|
}
|
|
|
|
syntaxContextInfo = String.Format(CultureInfo.CurrentCulture, "{0} '{1}'", syntaxContextInfo, term);
|
|
}
|
|
|
|
throw EntityUtil.EntitySqlError(_query,
|
|
System.Data.Entity.Strings.GenericSyntaxError,
|
|
errorPosition,
|
|
syntaxContextInfo,
|
|
false /* loadErrorContextInfoFromResource */);
|
|
}
|
|
throw EntityUtil.EntitySqlError(_query, s, _lexer.IPos);
|
|
}
|
|
|
|
//
|
|
// Error tracking helpers
|
|
//
|
|
private void SetErrCtx(Node astExpr, CqlLexer.TerminalToken tokenValue, string info)
|
|
{
|
|
SetErrCtx(astExpr, tokenValue.IPos, info);
|
|
}
|
|
|
|
private void SetErrCtx(Node astExpr, int inputPos, string info)
|
|
{
|
|
astExpr.ErrCtx.InputPosition = inputPos;
|
|
astExpr.ErrCtx.ErrorContextInfo = info;
|
|
astExpr.ErrCtx.CommandText = _query;
|
|
}
|
|
|
|
private void StartMethodExprCounting()
|
|
{
|
|
// Save the current counter value.
|
|
_methodExprCounterStack.Push(_methodExprCounter);
|
|
|
|
// Reset the counter for the current level.
|
|
_methodExprCounter = 0;
|
|
}
|
|
|
|
private void IncrementMethodExprCount()
|
|
{
|
|
++_methodExprCounter;
|
|
}
|
|
|
|
private uint EndMethodExprCounting()
|
|
{
|
|
// Save number of method expressions on the current level.
|
|
uint count = _methodExprCounter;
|
|
|
|
// Restore upper level counter and adjust it with the number of method expressions on the current level.
|
|
_methodExprCounter += _methodExprCounterStack.Pop();
|
|
|
|
return count;
|
|
}
|
|
}
|
|
}
|