| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285 |
- //
- // SqlWhereClauseTokenizer.cs
- //
- // Author:
- // Juraj Skripsky ([email protected])
- //
- // (C) 2004 HotFeet GmbH (http://www.hotfeet.ch)
- //
- using System;
- using System.Data;
- using System.IO;
- using System.Text;
- using System.Collections;
- namespace Mono.Data.SqlExpressions {
- internal class Tokenizer : yyParser.yyInput {
- private static readonly IDictionary tokenMap = new Hashtable ();
- private static readonly Object [] tokens = {
- Token.AND, "and",
- Token.OR, "or",
- Token.NOT, "not",
-
- Token.TRUE, "true",
- Token.FALSE, "false",
- Token.NULL, "null",
-
- Token.PARENT, "parent",
- Token.CHILD, "child",
-
- Token.IS, "is",
- Token.IN, "in",
- Token.LIKE, "like",
-
- Token.COUNT, "count",
- Token.SUM, "sum",
- Token.AVG, "avg",
- Token.MAX, "max",
- Token.MIN, "min",
- Token.STDEV, "stdev",
- Token.VAR, "var",
-
- Token.IIF, "iif",
- Token.SUBSTRING, "substring",
- Token.ISNULL, "isnull",
- Token.LEN, "len",
- Token.TRIM, "trim",
- Token.CONVERT, "convert"
- };
- private char[] input;
- private int pos;
- private int tok;
- private object val;
- static Tokenizer ()
- {
- for (int i = 0; i < tokens.Length; i += 2)
- tokenMap.Add (tokens [i + 1], tokens [i]);
- }
- public Tokenizer (string strInput)
- {
- input = strInput.ToCharArray ();
- pos = 0;
- }
- private char Current() {
- return input [pos];
- }
- private char Next() {
- if (pos + 1 >= input.Length)
- return (char)0;
- return input [pos + 1];
- }
- private void MoveNext() {
- pos++;
- }
-
- private void SkipWhiteSpace ()
- {
- while (Char.IsWhiteSpace (Current ()))
- MoveNext ();
- }
- private object ReadNumber ()
- {
- StringBuilder sb = new StringBuilder ();
- sb.Append (Current ());
- char next;
- while (Char.IsDigit (next = Next ())) {
- sb.Append (next);
- MoveNext ();
- }
- if (next == '.') {
- sb.Append (next);
- while (Char.IsDigit (next = Next ())) {
- sb.Append (next);
- MoveNext ();
- }
-
- return double.Parse (sb.ToString ());
- }
-
- return int.Parse (sb.ToString ());
- }
- private char ProcessEscapes(char c)
- {
- if (c == '\\') {
- MoveNext();
- c = Next();
- switch (c) {
- case 'n':
- c = '\n';
- break;
- case 'r':
- c = '\r';
- break;
- case 't':
- c = '\t';
- break;
- case '\\':
- c = '\\';
- break;
-
- default:
- throw new SyntaxErrorException (String.Format ("Invalid escape sequence: '\\{0}'.", c));
- }
- }
- return c;
- }
- private string ReadString (char terminator)
- {
- StringBuilder sb = new StringBuilder ();
- char next;
- while ((next = Next ()) != terminator) {
- sb.Append (ProcessEscapes (next));
- MoveNext ();
- }
- MoveNext ();
-
- return sb.ToString ();
- }
- private string ReadIdentifier ()
- {
- StringBuilder sb = new StringBuilder ();
- sb.Append (Current ());
- char next;
- while ((next = Next ()) == '_' || Char.IsLetterOrDigit (next) || next == '\\') {
- sb.Append (ProcessEscapes (next));
- MoveNext ();
- }
- return sb.ToString ();
- }
- private int ParseIdentifier ()
- {
- string strToken = ReadIdentifier ();
- object tokenObj = tokenMap[strToken.ToLower()];
- if(tokenObj != null)
- return (int)tokenObj;
-
- val = strToken;
- return Token.Identifier;
- }
- private int ParseToken ()
- {
- char cur;
- switch (cur = Current ()) {
- case '(':
- return Token.PAROPEN;
- case ')':
- return Token.PARCLOSE;
- case '.':
- return Token.DOT;
- case ',':
- return Token.COMMA;
- case '+':
- return Token.PLUS;
- case '-':
- return Token.MINUS;
- case '*':
- return Token.MUL;
- case '/':
- return Token.DIV;
- case '%':
- return Token.MOD;
-
- case '=':
- return Token.EQ;
- case '<':
- return Token.LT;
- case '>':
- return Token.GT;
- case '[':
- val = ReadString (']');
- return Token.Identifier;
- case '#':
- string date = ReadString ('#');
- val = DateTime.Parse (date);
- return Token.DateLiteral;
- case '\'':
- case '\"':
- val = ReadString (cur);
- return Token.StringLiteral;
- default:
- if (Char.IsDigit (cur)) {
- val = ReadNumber ();
- return Token.NumberLiteral;
- } else if (Char.IsLetter (cur) || cur == '_')
- return ParseIdentifier ();
- break;
- }
- throw new Exception ("invalid token: '" + cur + "'");
- }
- ///////////////////////////
- // yyParser.yyInput methods
- ///////////////////////////
- /** move on to next token.
- @return false if positioned beyond tokens.
- @throws IOException on input error.
- */
- public bool advance ()
- {
- val = null;
- tok = -1;
-
- try {
- SkipWhiteSpace();
- tok = ParseToken();
- MoveNext();
- return true;
- } catch(IndexOutOfRangeException) {
- return false;
- }
- }
- /** classifies current token.
- Should not be called if advance() returned false.
- @return current %token or single character.
- */
- public int token ()
- {
- return tok;
- }
- /** associated with current token.
- Should not be called if advance() returned false.
- @return value for token().
- */
- public Object value ()
- {
- return val;
- }
- }
- }
|