Update contrib.
1 // Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
22 const TInt KSqlError = -1;
24 TSqlParser::TSqlParser(const TDesC& aSql)
27 NextToken(); // parse the first token
30 TSqlTokenType TSqlParser::NextToken()
32 return iSql.NextToken(iToken);
35 CSqlSearchCondition* TSqlParser::SqlError()
38 iToken.SetError(KErrArgument);
42 TSqlTokenType TSqlParser::SqlErrorL()
44 // Report a SQL syntax error
47 return TSqlTokenType(__LEAVE_IF_ERROR(KErrArgument));
50 TSqlTokenType TSqlParser::Parse(TSqlKeyword aKeyword)
52 // look for requested keyword, skip to the next token if found
53 // return the next token if found, else ESqlNoToken (==0)
56 return TSqlLexer::IsKeyword(aKeyword,iToken) ? NextToken() : ESqlNoToken;
59 TSqlTokenType TSqlParser::ParseL(TSqlTokenType aToken)
61 // parse the desired token
64 return iToken!=aToken ? SqlErrorL() : NextToken();
67 TSqlTokenType TSqlParser::ParseL(TSqlKeyword aKeyword)
69 // parse the desired keyword
72 TSqlTokenType t=Parse(aKeyword);
73 return t==ESqlNoToken ? SqlErrorL() : t;
76 TSqlKeyword TSqlParser::Keyword()
81 TSqlKeyword k=TSqlLexer::Keyword(iToken);
82 if (k!=ESqlNotKeyword)
87 TSqlTokenType TSqlParser::RightBracketL()
89 // parse a right bracket, and fail if not found
92 return ParseL(ESqlRightBracket);
95 void TSqlParser::EndL()
97 // Check that the SQL has been fully parsed
104 TSqlTokenType TSqlParser::IdentifierL(TPtrC& aIdentifier)
106 // parse an identifer, fail if not found
109 if (iToken!=ESqlIdentifier)
111 const TText* p=iToken.Literal().Ptr();
112 TInt len=iToken.Literal().End()-p;
113 aIdentifier.Set(p,len);
117 TPtrC TSqlParser::IdentifierL()
119 // parse an identifer, fail if not found
122 if (iToken!=ESqlIdentifier)
124 return iToken.Literal().DesC();
127 TSqlTokenType TSqlParser::ColumnNameL(RSqlColumnList& aList)
129 // Parse a column-identifier
132 __LEAVE_IF_ERROR(aList.Append(IdentifierL()));
136 TSqlTokenType TSqlParser::ColumnListL(RSqlColumnList& aList)
138 // Parse a column-identifier-comma-list
143 TSqlTokenType t=ColumnNameL(aList);
150 TSqlTokenType TSqlParser::AddColumnSpecL(TDbCol& aDef)
152 // Parse an add-column-spec
156 aDef.iName=IdentifierL();
162 case ESqlKeyword_bit:
165 case ESqlKeyword_tinyint:
168 case ESqlKeyword_smallint:
171 case ESqlKeyword_integer:
174 case ESqlKeyword_counter:
175 aDef.iAttributes=aDef.EAutoIncrement;
178 case ESqlKeyword_bigint:
181 case ESqlKeyword_real:
184 case ESqlKeyword_double:
185 //Return value is intentionaly not checked. Parse() checks for
186 //an optional SQL keyword and calls internal checks.
187 //coverity[check_return]
188 //coverity[unchecked_value]
189 Parse(ESqlKeyword_precision);
191 case ESqlKeyword_float:
194 case ESqlKeyword_date:
195 case ESqlKeyword_time:
196 case ESqlKeyword_timestamp:
199 case ESqlKeyword_char:
200 case ESqlKeyword_varchar:
203 case ESqlKeyword_binary:
204 case ESqlKeyword_varbinary:
207 case ESqlKeyword_unsigned: // unsigned tinyint, smallint or integer
210 case ESqlKeyword_tinyint:
213 case ESqlKeyword_smallint:
216 case ESqlKeyword_integer:
223 case ESqlKeyword_long: // varchar or varbinary
226 case ESqlKeyword_varchar:
229 case ESqlKeyword_varbinary:
230 type=EDbColLongBinary;
241 // get any optional length
242 aDef.iMaxLength=KDbUndefinedLength;
243 TSqlTokenType t=iToken.Type();
248 if (t==ESqlLeftBracket)
250 if (NextToken()==ESqlLiteralInt)
252 iToken.Literal().ToInt32L();
253 aDef.iMaxLength=iToken.Literal().Int32();
267 CSqlSearchCondition* TSqlParser::SearchCondition(TInt aNot)
269 // Parse a search-condition
272 CSqlSearchCondition* left=BooleanTerm(aNot);
273 if (left==0 || Parse(ESqlKeyword_or)==ESqlNoToken)
275 return CSqlMultiNode::New(aNot ? CSqlMultiNode::EAnd : CSqlMultiNode::EOr,left,SearchCondition(aNot));
278 CSqlSearchCondition* TSqlParser::BooleanTerm(TInt aNot)
280 // Parse a boolean-term
283 CSqlSearchCondition* left=BooleanFactor(aNot);
284 if (left==0 || Parse(ESqlKeyword_and)==ESqlNoToken)
286 return CSqlMultiNode::New(aNot ? CSqlMultiNode::EOr : CSqlMultiNode::EAnd,left,BooleanTerm(aNot));
289 CSqlSearchCondition* TSqlParser::BooleanFactor(TInt aNot)
291 // Parse a boolean-factor
294 while (Parse(ESqlKeyword_not))
296 return BooleanPrimary(aNot);
299 CSqlSearchCondition* TSqlParser::BooleanPrimary(TInt aNot)
301 // Parse a boolean-factor
304 // brackets only allowed in this element, so this ordering is valid
305 if (iToken!=ESqlLeftBracket)
306 return Predicate(aNot);
307 // bracketed search condition
309 CSqlSearchCondition* node=SearchCondition(aNot);
312 if (iToken==ESqlRightBracket)
321 CSqlSearchCondition* TSqlParser::Predicate(TInt aNot)
324 // On null return, error will already be set
327 if (iToken!=ESqlIdentifier) // column name
329 TPtrC column(iToken.Literal().DesC());
330 TSqlTokenType t=NextToken();
331 if (t==ESqlIdentifier)
332 { // like-predicate or null-predicate
335 case ESqlKeyword_is: // IS [NOT] NULL
336 if (Parse(ESqlKeyword_not))
338 if (Parse(ESqlKeyword_null)==ESqlNoToken)
340 return new CSqlNullPredicate(aNot ? CSqlNullPredicate::EIsNotNull : CSqlNullPredicate::EIsNull,column);
341 case ESqlKeyword_not: // NOT LIKE
342 if (Parse(ESqlKeyword_like)==ESqlNoToken)
345 // drop through to Like
346 case ESqlKeyword_like: // LIKE
348 if (iToken!=ESqlLiteralText)
351 //Following code is for the implementation of limited-ESCAPE-clause
352 const TText *next = iSql.Next(); //remember thecurrent position in the SQL statement
353 RSqlLiteral pattern = iToken.Literal(); //this is the LIKE pattern, remember it, because the next keyword might be ESCAPE
355 NextToken(); // Searching for ESCAPE keyword
357 if (Keyword() == ESqlKeyword_escape)
359 if (pattern.DesC().Length() <= 0 || pattern.DesC().Length() > (KMaxSegmentLength + 2 ))
361 TPtrC escapeChar(iToken.Literal().DesC());
362 if (escapeChar.Length() <= 0)
364 TChar escchar = escapeChar[0];
365 TText newPattern[KMaxSegmentLength + 2]; // '*' can come as first and last char
366 TInt count = PatternFilter(pattern.DesC(),escchar, newPattern);//remove the escape characters from the pattern and store it in "newPattern" variable
369 iToken.Literal().SetText(newPattern,(newPattern + count));
370 // copy the text to RSqlLiteral as "newPattern" is stack based variable and will go out of scope
371 if (iToken.Literal().CopyText() != KErrNone)
375 CSqlSearchCondition* node = new CSqlLikePredicate(aNot ? CSqlLikePredicate::ENotLike : CSqlLikePredicate::ELike,column,iToken.Literal());
376 //cleanup iToken.Literal(), because if there is another LIKE predicate, the allocated by CopyText() buffer will
377 //be shared between two RSqlLiteral instances and RSqlLiteral::Close() call will crash.
378 //RSqlLiteral::RSqlLiteral() will set the "iBuffer" data member to NULL.
379 iToken.Literal() = RSqlLiteral();
383 node->iIsEscape = ETrue;
389 //Set to the previous node
391 CSqlSearchCondition* node = new CSqlLikePredicate(aNot ? CSqlLikePredicate::ENotLike : CSqlLikePredicate::ELike,column,pattern);
402 // Comparison predicate...
403 CSqlSearchCondition::TType op;
406 case ESqlGreaterEqual:
408 // drop through to less
410 op=aNot ? CSqlSearchCondition::EGreaterEqual : CSqlSearchCondition::ELess;
414 // drop through to less equal
416 op=aNot ? CSqlSearchCondition::EGreater: CSqlSearchCondition::ELessEqual;
420 // drop through to equal
422 op=aNot ? CSqlSearchCondition::ENotEqual : CSqlSearchCondition::EEqual;
428 if (t!=ESqlLiteralInt && t!=ESqlLiteralReal && t!=ESqlLiteralTime && t!=ESqlLiteralText)
430 CSqlSearchCondition* node=new CSqlCompPredicate(op,column,iToken.Literal());
436 CSqlSearchCondition* TSqlParser::SearchConditionL()
438 // Parse a search-condition
441 CSqlSearchCondition* sc=SearchCondition(0);
444 __LEAVE_IF_ERROR(Error()); // syntax error
445 User::LeaveNoMemory(); // otherwise a OOM error
450 void TSqlParser::SortSpecificationL(CDbKey& aKey)
452 // Parse a sort-specification
457 TDbKeyCol col(IdentifierL());
459 if (Parse(ESqlKeyword_desc))
460 col.iOrder=col.EDesc;
463 //Return value is intentionaly not checked. Parse() checks for
464 //an optional SQL keyword and calls internal checks.
465 //coverity[check_return]
466 //coverity[unchecked_value]
467 Parse(ESqlKeyword_asc);
471 if (iToken!=ESqlComma)
477 CSqlQuery* TSqlParser::QueryLC()
479 // Generate a CSqlQuery
482 CSqlQuery* query=new(ELeave) CSqlQuery;
483 CleanupStack::PushL(query);
484 if (ParseL(ESqlKeyword_select)==ESqlAsterisk)
487 ColumnListL(query->iColumns);
488 ParseL(ESqlKeyword_from);
489 IdentifierL(query->iTable);
490 if (Parse(ESqlKeyword_where))
491 query->iSearchCondition=SearchConditionL();
492 if (Parse(ESqlKeyword_order))
494 ParseL(ESqlKeyword_by);
495 SortSpecificationL(query->SortSpecificationL());
501 CSqlSearchCondition* TSqlParser::SearchConditionLC()
503 // Parse a standalone search-condition
506 CSqlSearchCondition* sc=SearchConditionL();
507 CleanupStack::PushL(sc);
512 CSqlDDLStatement* TSqlParser::CreateTableLC()
514 // Parse a CREATE TABLE statement
517 CSqlCreateTableStatement* statement=new(ELeave) CSqlCreateTableStatement;
518 CleanupStack::PushL(statement);
519 IdentifierL(statement->iName);
520 ParseL(ESqlLeftBracket);
525 if (Parse(ESqlKeyword_not))
527 ParseL(ESqlKeyword_null);
528 def.iAttributes|=TDbCol::ENotNull;
530 statement->iColumns.AddL(def);
531 if (iToken!=ESqlComma)
539 CSqlDDLStatement* TSqlParser::DropTableLC()
541 // Parse a DROP TABLE statement
544 CSqlDropTableStatement* statement=new(ELeave) CSqlDropTableStatement;
545 CleanupStack::PushL(statement);
546 IdentifierL(statement->iName);
550 CSqlDDLStatement* TSqlParser::AlterTableLC()
552 // Parse a CREATE TABLE statement
555 CSqlAlterTableStatement* statement=new(ELeave) CSqlAlterTableStatement;
556 CleanupStack::PushL(statement);
557 IdentifierL(statement->iName);
558 TSqlTokenType t=Parse(ESqlKeyword_add);
562 if (t==ESqlLeftBracket)
567 t=AddColumnSpecL(def);
568 statement->iAdd.AddL(def);
578 statement->iAdd.AddL(def);
581 t=Parse(ESqlKeyword_drop);
584 if (t!=ESqlLeftBracket)
585 ColumnNameL(statement->iDrop);
589 ColumnListL(statement->iDrop);
596 CSqlDDLStatement* TSqlParser::CreateIndexLC(TBool aUnique)
598 // Parse a CREATE INDEX statement
601 CSqlCreateIndexStatement* statement=new(ELeave) CSqlCreateIndexStatement;
602 CleanupStack::PushL(statement);
604 statement->iKey.MakeUnique();
605 IdentifierL(statement->iName);
606 ParseL(ESqlKeyword_on);
607 IdentifierL(statement->iTable);
608 ParseL(ESqlLeftBracket);
609 SortSpecificationL(statement->iKey);
614 CSqlDDLStatement* TSqlParser::DropIndexLC()
616 // Parse a DROP INDEX statement
619 CSqlDropIndexStatement* statement=new(ELeave) CSqlDropIndexStatement;
620 CleanupStack::PushL(statement);
621 IdentifierL(statement->iName);
622 ParseL(ESqlKeyword_from);
623 IdentifierL(statement->iTable);
627 CSqlDDLStatement* TSqlParser::DDLStatementLC()
629 CSqlDDLStatement* statement;
630 if (Parse(ESqlKeyword_create))
632 if (Parse(ESqlKeyword_table))
633 statement=CreateTableLC();
636 TSqlTokenType unique=Parse(ESqlKeyword_unique);
637 ParseL(ESqlKeyword_index);
638 statement=CreateIndexLC(unique);
641 else if (Parse(ESqlKeyword_drop))
643 if (Parse(ESqlKeyword_table))
644 statement=DropTableLC();
647 ParseL(ESqlKeyword_index);
648 statement=DropIndexLC();
653 ParseL(ESqlKeyword_alter);
654 ParseL(ESqlKeyword_table);
655 statement=AlterTableLC();
661 TSqlTokenType TSqlParser::ColumnValueL(CSqlValues& aValues)
663 // parse a column-value and add to aValues
666 switch (iToken.Type())
669 case ESqlLiteralReal:
670 case ESqlLiteralTime:
671 case ESqlLiteralText:
672 aValues.AddL(iToken.Literal());
674 case ESqlIdentifier: // NULL
676 TSqlTokenType t=ParseL(ESqlKeyword_null);
677 aValues.AddL(RSqlLiteral()); // default c'ted RSqlLiteral is null
680 default: // SQL error
685 CSqlDMLStatement* TSqlParser::InsertStatementLC()
687 // parse an insert-statement
690 ParseL(ESqlKeyword_into);
691 CSqlInsertStatement* statement=CSqlInsertStatement::NewLC();
692 if (IdentifierL(statement->iQuery.iTable)==ESqlLeftBracket)
695 ColumnListL(statement->iQuery.iColumns);
698 ParseL(ESqlKeyword_values);
699 ParseL(ESqlLeftBracket);
700 CSqlValues& values=statement->ValuesL();
701 while (ColumnValueL(values)==ESqlComma)
707 CSqlDMLStatement* TSqlParser::UpdateStatementLC()
709 // parse an update-statement
712 CSqlModifyStatement* statement=CSqlModifyStatement::NewLC();
713 IdentifierL(statement->iQuery.iTable);
714 ParseL(ESqlKeyword_set);
715 CSqlValues& values=statement->ValuesL();
718 ColumnNameL(statement->iQuery.iColumns);
720 if (ColumnValueL(values)!=ESqlComma)
724 if (Parse(ESqlKeyword_where))
725 statement->iQuery.iSearchCondition=SearchConditionL();
729 CSqlDMLStatement* TSqlParser::DeleteStatementLC()
731 // parse a delete-statement
734 ParseL(ESqlKeyword_from);
735 CSqlModifyStatement* statement=CSqlModifyStatement::NewLC();
736 IdentifierL(statement->iQuery.iTable);
737 if (Parse(ESqlKeyword_where))
738 statement->iQuery.iSearchCondition=SearchConditionL();
742 CSqlDMLStatement* TSqlParser::DMLStatementLC()
744 CSqlDMLStatement* statement;
745 if (Parse(ESqlKeyword_insert))
746 statement=InsertStatementLC();
747 else if (Parse(ESqlKeyword_update))
748 statement=UpdateStatementLC();
751 ParseL(ESqlKeyword_delete);
752 statement=DeleteStatementLC();
758 Sql::TStatementType TSqlParser::Type()
760 TSqlKeyword k=TSqlLexer::Keyword(iToken);
761 if (k==ESqlKeyword_create || k==ESqlKeyword_alter || k==ESqlKeyword_drop)
763 if (k==ESqlKeyword_insert || k==ESqlKeyword_update || k==ESqlKeyword_delete)
768 TInt TSqlParser::PatternFilter(const TDesC& aPattern,const TChar aEscape,TText *aNewPatternBuffer )
770 TInt length = aPattern.Length();
773 // Ensure that the pattern begins and ends with an '*'
774 if ((length < 2) || (aPattern[0] != KMatchAny || aPattern[length-1] != KMatchAny))
779 for (TInt i = 1 ; i< length -1 ;++i)
781 if (aPattern[i]== (TUint)aEscape)
783 if ((aPattern[i + 1] == KMatchAny) || (aPattern[i + 1] == KMatchOne) || (aPattern[i + 1] == (TUint)aEscape))
786 aNewPatternBuffer[count++] = aPattern[i];
795 if ((aPattern[i] == KMatchAny || aPattern[i] == KMatchOne) )
801 aNewPatternBuffer[count++] = aPattern[i];