os/persistentdata/persistentstorage/dbms/tdbms/t_dbsql.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 // MSVC++ up to 5.0 has problems with expanding inline functions
    17 // This disables the mad warnings for the whole project
    18 #if defined(NDEBUG) && defined(__VC32__) && _MSC_VER<=1100
    19 #pragma warning(disable : 4710)			// function not expanded. MSVC 5.0 is stupid
    20 #endif
    21 
    22 #include <d32dbms.h>
    23 #include <f32file.h>
    24 #include <e32test.h>
    25 #include <e32math.h>
    26 
    27 // constructing literal TInt64 from compiler 64 bit integer rep
    28 
    29 #ifndef I64LIT
    30 #if defined __GCC32__ || defined __EABI__
    31 #define _LINT64(val) MAKE_TINT64(TUint(val##LL>>32),TUint(val##LL&0xffffffffu))
    32 #else
    33 #define _LINT64(val) MAKE_TINT64(TUint(__int64(val)>>32),TUint(__int64(val)&0xffffffffu))
    34 #endif
    35 #else // I64LIT
    36 #define _LINT64 I64LIT
    37 #endif // I64LIT
    38 
    39 LOCAL_D RTest TheTest(_L("t_dbsql: DBMS SQL parsing and execution tests"));
    40 
    41 LOCAL_D RFs TheFs;
    42 LOCAL_D CTrapCleanup* TheTrapCleanup;
    43 LOCAL_D RDbs TheDbs;
    44 LOCAL_D RDbNamedDatabase TheDatabase;
    45 LOCAL_D RDbTable TheTable;
    46 LOCAL_D RDbView TheView;
    47 LOCAL_D TBuf<1024> TheSql;
    48 
    49 const TInt KTestCleanupStack=0x20;
    50 const TPtrC KTestDatabase=_L("c:\\dbms-tst\\t_sql.db");
    51 
    52 #define elementsof(array) (sizeof(array)/sizeof(array[0]))
    53 
    54 LOCAL_C void TestCleanup()
    55 	{
    56 	TheTable.Close();
    57 	TheView.Close();
    58 	TheDatabase.Close();
    59 	TheDbs.Close();
    60 	TheFs.Close();
    61 	/////////////
    62 	RFs fsSession;
    63 	TInt err = fsSession.Connect();
    64 	if(err == KErrNone)
    65 		{
    66 		TEntry entry;
    67 		if(fsSession.Entry(KTestDatabase, entry) == KErrNone)
    68 			{
    69 			RDebug::Print(_L("Deleting \"%S\" file.\n"), &KTestDatabase);
    70 			err = fsSession.SetAtt(KTestDatabase, 0, KEntryAttReadOnly);
    71 			if(err != KErrNone)
    72 				{
    73 				RDebug::Print(_L("Error %d changing \"%S\" file attributes.\n"), err, &KTestDatabase);
    74 				}
    75 			err = fsSession.Delete(KTestDatabase);
    76 			if(err != KErrNone)
    77 				{
    78 				RDebug::Print(_L("Error %d deleting \"%S\" file.\n"), err, &KTestDatabase);
    79 				}
    80 			}
    81 		fsSession.Close();
    82 		}
    83 	else
    84 		{
    85 		RDebug::Print(_L("Error %d connecting file session. File: %S.\n"), err, &KTestDatabase);
    86 		}
    87 	}
    88 
    89 LOCAL_C void CloseDatabase()
    90 	{
    91 	TheDatabase.Close();
    92 	}
    93 
    94 LOCAL_C void Disconnect()
    95 	{
    96 	TheDbs.ResourceCheck();
    97 	TheDbs.Close();
    98 	}
    99 
   100 ///////////////////////////////////////////////////////////////////////////////////////
   101 ///////////////////////////////////////////////////////////////////////////////////////
   102 //Tests macros and functions.
   103 //If (!aValue) then the test will be panicked, the test data files will be deleted.
   104 static void Check(TInt aValue, TInt aLine)
   105 	{
   106 	if(!aValue)
   107 		{
   108         RDebug::Print(_L("*** Expression evaluated to false\r\n"));
   109 		::TestCleanup();
   110 		TheTest(EFalse, aLine);
   111 		}
   112 	}
   113 //If (aValue != aExpected) then the test will be panicked, the test data files will be deleted.
   114 static void Check(TInt aValue, TInt aExpected, TInt aLine)
   115 	{
   116 	if(aValue != aExpected)
   117 		{
   118 		RDebug::Print(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
   119 		::TestCleanup();
   120 		TheTest(EFalse, aLine);
   121 		}
   122 	}
   123 //Use these to test conditions.
   124 #define TEST(arg) ::Check((arg), __LINE__)
   125 #define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)
   126 
   127 ///////////////////////////////////////////////////////////////////////////////////////
   128 ///////////////////////////////////////////////////////////////////////////////////////
   129 
   130 // Create the database
   131 LOCAL_C void CreateDatabase()
   132 	{
   133 	TInt r=TheDatabase.Replace(TheFs,KTestDatabase);
   134 	TEST2(r, KErrNone);
   135 	}
   136 
   137 LOCAL_C void Connect()
   138 	{
   139 	TInt r=TheDbs.Connect();
   140 	TEST2(r, KErrNone);
   141 	TheDbs.ResourceMark();
   142 	}
   143 
   144 // Open the database through the server
   145 LOCAL_C void ShareDatabase()
   146 	{
   147 	TInt r=TheDatabase.Open(TheDbs,KTestDatabase);
   148 	TEST2(r, KErrNone);
   149 	}
   150 
   151 struct TSelectTest
   152 	{
   153 	const TText* iSearchCondition;
   154 	TUint iResultSet;
   155 	};
   156 #define ROWS(n,m) (((~0u)<<(n))&((2u<<(m))-1))
   157 #define ROW(n) ROWS(n,n)
   158 #define NO_ROWS 0
   159 #define ALL_ROWS (~1u)		// all rows which are not NULL
   160 
   161 class TestPredicateBase
   162 	{
   163 protected:
   164 	static void Create(const TText* aType);
   165 	static void Test(const TSelectTest* aTest,TInt aCount,TInt aRows, TBool aLog);
   166 	static void TestViewL(const TSelectTest* aTest,TInt aCount,TInt aRows, TBool aLog=EFalse);
   167 	};
   168 
   169 // Create the table for the predicate tests
   170 void TestPredicateBase::Create(const TText* aType)
   171 	{
   172 	TheTest.Next(TPtrC(aType));
   173 	TheDatabase.Begin();
   174 	TheSql.Format(_L("CREATE TABLE Compare (Id COUNTER,Test %s)"),aType);
   175 	TInt r=TheDatabase.Execute(TheSql);
   176 	TEST2(r, KErrNone);
   177 	r=TheTable.Open(TheDatabase,_L("Compare"),TheTable.EInsertOnly);
   178 	TEST2(r, KErrNone);
   179 	}
   180 
   181 // Test the predicate on the table, then on the indexed table
   182 void TestPredicateBase::Test(const TSelectTest* aTest,TInt aCount,TInt aRows, TBool aLog)
   183 	{
   184     if(aLog)
   185         {
   186         TheTest.Printf(_L("TestPredicateBase::Test\r\n"));
   187         }
   188 	TheTable.Close();
   189 	TInt r=TheDatabase.Commit();
   190 	if(aLog)
   191         {
   192         TheTest.Printf(_L("Commit %d\r\n"), r);
   193         }
   194 	TEST2(r, KErrNone);
   195 	TRAPD(errCode, TestViewL(aTest,aCount,aRows, aLog));
   196 	if(aLog)
   197         {
   198         TheTest.Printf(_L("TestViewL %d"), errCode);
   199         }
   200 
   201 	TEST2(errCode, KErrNone);
   202 	r=TheDatabase.Execute(_L("CREATE INDEX Key ON Compare (Test)"));
   203    if(aLog)
   204         {
   205         TheTest.Printf(_L("Execute %d"), r);
   206         }
   207 
   208 	TEST2(r, KErrNone);
   209 	TRAP(errCode,TestViewL(aTest,aCount,aRows, aLog));
   210 	if(aLog)
   211         {
   212         TheTest.Printf(_L("TestViewL %d"), errCode);
   213 	    }
   214 
   215 	TEST2(errCode, KErrNone);
   216 	r=TheDatabase.Execute(_L("DROP TABLE Compare"));
   217     if(aLog)
   218         {
   219         TheTest.Printf(_L("Execute %d"), r);
   220         }
   221 
   222 	TEST2(r, KErrNone);
   223 	}
   224 
   225 // Test the predicate on the table
   226 void TestPredicateBase::TestViewL(const TSelectTest* aTest,TInt aCount,TInt aRows, TBool /*aLog*/)
   227 	{
   228 	TUint rowMask=(2u<<aRows)-1;
   229 	for (;--aCount>=0;++aTest)
   230 		{
   231 		TheSql.Format(_L("SELECT Id FROM Compare WHERE Test %s"),aTest->iSearchCondition);
   232 		TInt r=TheView.Prepare(TheDatabase,TheSql,TheView.EReadOnly);
   233 		if(r!=KErrNone)
   234 		    {
   235             TheTest.Printf(_L("Prepare r= %d aCount= %d  statement %S\r\n"), r, aCount, &TheSql);
   236 		    }
   237 		TEST2(r, KErrNone);
   238 		TBool ignoreRow0=TheView.Unevaluated();
   239 		r=TheView.EvaluateAll();
   240  		TEST2(r, KErrNone);
   241 		TUint rows=0;
   242 		while (TheView.NextL())
   243 			{
   244  			TheView.GetL();
   245 			rows|=1u<<TheView.ColUint(1);
   246 			}
   247 		if (ignoreRow0)
   248 		    {
   249   			TEST((rows&~ROW(0))==(aTest->iResultSet&rowMask&~ROW(0)));
   250 		    }
   251 		else
   252 		    {
   253  			TEST(rows==(aTest->iResultSet&rowMask));
   254 		    }
   255 		TheView.Close();
   256 		}
   257 	}
   258 
   259 typedef void (*FSetColL)(TDbColNo aCol,const TAny* aVal);
   260 
   261 void WriteRowsL(const TAny* aValues,TInt aRows,TInt aSize,FSetColL aSetColL, TBool aLog)
   262 	{
   263 	for (TInt row=0;row<=aRows;++row)
   264 		{
   265         if(aLog)
   266             {
   267             TheTest.Printf(_L("row = %d"), row);
   268             }
   269 		TheTable.InsertL();
   270 		TEST(TheTable.ColUint(1)==TUint(row));
   271 		if (row>0)
   272 			{	// first row is always null
   273 			aSetColL(2,aValues);
   274 			aValues=PtrAdd(aValues,aSize);
   275 			}
   276 		TheTable.PutL();
   277 		}
   278 	}
   279 
   280 template <class T>
   281 struct SetCol
   282 	{
   283 	static void SetColL(TDbColNo aCol,const TAny* aVal)
   284 		{
   285 		TheTable.SetColL(aCol,*(const T*)aVal);
   286 		}
   287 	};
   288 
   289 template <class T>
   290 inline void WriteRowsL(const T* aValues,TUint aRows, TBool aLog)
   291 	{
   292 	WriteRowsL(aValues,aRows,sizeof(T),&SetCol<T>::SetColL, aLog);
   293 	}
   294 
   295 template <class T>
   296 class TestPredicate : public TestPredicateBase
   297 	{
   298 public:
   299 	static void RunL();
   300 	};
   301 
   302 // Test the Predicate operators for all types
   303 template <class T>
   304 void TestPredicate<T>::RunL()
   305 	{
   306 	Create(T::KType);
   307 	TBool log = EFalse;
   308 	if((TPtrC(T::KType)).CompareF(_L("TIME"))==0)
   309 	    {
   310         log = ETrue;
   311 	    }
   312 	WriteRowsL(T::KValues,elementsof(T::KValues), log);
   313 	Test(T::KTests,elementsof(T::KTests),elementsof(T::KValues),log);
   314 	}
   315 
   316 struct TypeBit
   317 	{
   318 public:
   319 	static const TText* const KType;
   320 	static const TUint KValues[];
   321 	static const TSelectTest KTests[];
   322 	};
   323 const TText* const TypeBit::KType=_S("BIT");
   324 const TUint TypeBit::KValues[]={0,1};
   325 const TSelectTest TypeBit::KTests[]=
   326 	{
   327 	{_S("IS NULL"),ROW(0)},
   328 	{_S("IS NOT NULL"),ALL_ROWS},
   329 	{_S("=0"),ROW(1)},
   330 	{_S("<>0"),ROW(2)},
   331 	{_S(">0"),ROW(2)},
   332 	{_S("<0"),NO_ROWS},
   333 	{_S("<=0"),ROW(1)},
   334 	{_S(">=0"),ALL_ROWS},
   335 	{_S("=1"),ROW(2)},
   336 	{_S("<>1"),ROW(1)},
   337 	{_S(">1"),NO_ROWS},
   338 	{_S("<1"),ROW(1)},
   339 	{_S("<=1"),ALL_ROWS},
   340 	{_S(">=1"),ROW(2)}
   341 	};
   342 
   343 struct TypeUint8
   344 	{
   345 public:
   346 	static const TText* const KType;
   347 	static const TUint KValues[];
   348 	static const TSelectTest KTests[];
   349 	};
   350 const TText* const TypeUint8::KType=_S("UNSIGNED TINYINT");
   351 const TUint TypeUint8::KValues[]={0,1,127,128,254,255};
   352 const TSelectTest TypeUint8::KTests[]=
   353 	{
   354 	{_S("IS NULL"),ROW(0)},
   355 	{_S("IS NOT NULL"),ALL_ROWS},
   356 	{_S("=0"),ROW(1)},
   357 	{_S("=3"),NO_ROWS},
   358 	{_S("<2"),ROWS(1,2)},
   359 	{_S(">=0"),ALL_ROWS},
   360 	{_S("<128"),ROWS(1,3)},
   361 	{_S("<>1"),ALL_ROWS-ROW(2)},
   362 	{_S(">255"),NO_ROWS}
   363 	};
   364 
   365 struct TypeUint16
   366 	{
   367 public:
   368 	static const TText* const KType;
   369 	static const TUint KValues[];
   370 	static const TSelectTest KTests[];
   371 	};
   372 const TText* const TypeUint16::KType=_S("UNSIGNED SMALLINT");
   373 const TUint TypeUint16::KValues[]={0,1,5000,32767,32768,65534,65535};
   374 const TSelectTest TypeUint16::KTests[]=
   375 	{
   376 	{_S("IS NULL"),ROW(0)},
   377 	{_S("IS NOT NULL"),ALL_ROWS},
   378 	{_S("=0"),ROW(1)},
   379 	{_S("=3"),NO_ROWS},
   380 	{_S("<2"),ROWS(1,2)},
   381 	{_S(">=32768"),ROWS(5,7)},
   382 	{_S("<32767"),ROWS(1,3)},
   383 	{_S("<>1"),ALL_ROWS-ROW(2)},
   384 	{_S(">65535"),NO_ROWS}
   385 	};
   386 
   387 struct TypeUint32
   388 	{
   389 public:
   390 	static const TText* const KType;
   391 	static const TUint KValues[];
   392 	static const TSelectTest KTests[];
   393 	};
   394 const TText* const TypeUint32::KType=_S("UNSIGNED INTEGER");
   395 const TUint TypeUint32::KValues[]={0,1,2147483647u,2147483648u,3000000000u,4294967294u,4294967295u};
   396 const TSelectTest TypeUint32::KTests[]=
   397 	{
   398 	{_S("IS NULL"),ROW(0)},
   399 	{_S("IS NOT NULL"),ALL_ROWS},
   400 	{_S("=0"),ROW(1)},
   401 	{_S("=3"),NO_ROWS},
   402 	{_S("<2"),ROWS(1,2)},
   403 	{_S(">=2147483648"),ROWS(4,7)},
   404 	{_S("<2147483647"),ROWS(1,2)},
   405 	{_S("<>3000000000"),ALL_ROWS-ROW(5)},
   406 	{_S(">4294967295"),NO_ROWS}
   407 	};
   408 
   409 struct TypeInt8
   410 	{
   411 public:
   412 	static const TText* const KType;
   413 	static const TInt KValues[];
   414 	static const TSelectTest KTests[];
   415 	};
   416 const TText* const TypeInt8::KType=_S("TINYINT");
   417 const TInt TypeInt8::KValues[]={-128,-1,0,1,127};
   418 const TSelectTest TypeInt8::KTests[]=
   419 	{
   420 	{_S("IS NULL"),ROW(0)},
   421 	{_S("IS NOT NULL"),ALL_ROWS},
   422 	{_S("=0"),ROW(3)},
   423 	{_S("=-1"),ROW(2)},
   424 	{_S("<2"),ROWS(1,4)},
   425 	{_S(">=-128"),ALL_ROWS},
   426 	{_S("<128"),ALL_ROWS},
   427 	{_S("<>1"),ALL_ROWS-ROW(4)}
   428 	};
   429 
   430 struct TypeInt16
   431 	{
   432 public:
   433 	static const TText* const KType;
   434 	static const TInt KValues[];
   435 	static const TSelectTest KTests[];
   436 	};
   437 const TText* const TypeInt16::KType=_S("SMALLINT");
   438 const TInt TypeInt16::KValues[]={-32768,-32767,-1,0,1,32766,32767};
   439 const TSelectTest TypeInt16::KTests[]=
   440 	{
   441 	{_S("IS NULL"),ROW(0)},
   442 	{_S("IS NOT NULL"),ALL_ROWS},
   443 	{_S("=0"),ROW(4)},
   444 	{_S("<>0"),ALL_ROWS-ROW(4)},
   445 	{_S("=-1"),ROW(3)},
   446 	{_S("<2"),ROWS(1,5)},
   447 	{_S(">=-40000"),ALL_ROWS},
   448 	{_S("<32766"),ROWS(1,5)},
   449 	{_S("=40"),NO_ROWS}
   450 	};
   451 
   452 struct TypeInt32
   453 	{
   454 public:
   455 	static const TText* const KType;
   456 	static const TInt KValues[];
   457 	static const TSelectTest KTests[];
   458 	};
   459 const TText* const TypeInt32::KType=_S("INTEGER");
   460 const TInt TypeInt32::KValues[]={0x80000000/*-2147483648*/,-2147483647,-1,0,1,2147483646,2147483647};
   461 const TSelectTest TypeInt32::KTests[]=
   462 	{
   463 	{_S("IS NULL"),ROW(0)},
   464 	{_S("IS NOT NULL"),ALL_ROWS},
   465 	{_S("=0"),ROW(4)},
   466 	{_S("<>0"),ALL_ROWS-ROW(4)},
   467 	{_S("=-1"),ROW(3)},
   468 	{_S("<2"),ROWS(1,5)},
   469 	{_S(">=-2147483648"),ALL_ROWS},
   470 	{_S("<2147483646"),ROWS(1,5)},
   471 	{_S(">2147483647"),NO_ROWS},
   472 	{_S("=40"),NO_ROWS}
   473 	};
   474 
   475 struct TypeInt64
   476 	{
   477 public:
   478 	static const TText* const KType;
   479 	static const TInt64 KValues[];
   480 	static const TSelectTest KTests[];
   481 	};
   482 const TText* const TypeInt64::KType=_S("BIGINT");
   483 const TInt64 TypeInt64::KValues[]=
   484 	{
   485 	MAKE_TINT64(0x80000000, 0x00000000), // min int64
   486 	_LINT64(-4294967296),
   487 	TInt(0x80000000),			// -2147483648!
   488 	-1,
   489 	0u,
   490 	1u,
   491 	2147483647u,
   492 	_LINT64(2147483648),
   493 	_LINT64(4294967295),
   494 	_LINT64(4294967296),
   495 	_LINT64(9223372036854775807)		// max int64
   496 	};
   497 const TSelectTest TypeInt64::KTests[]=
   498 	{
   499 	{_S("IS NULL"),ROW(0)},
   500 	{_S("IS NOT NULL"),ALL_ROWS},
   501 	{_S("=0"),ROW(5)},
   502 	{_S("<>0"),ALL_ROWS-ROW(5)},
   503 	{_S("=-1"),ROW(4)},
   504 	{_S("<2"),ROWS(1,6)},
   505 	{_S(">=-9223372036854775808"),ALL_ROWS},
   506 	{_S("<4294967296"),ROWS(1,9)},
   507 	{_S(">9223372036854775806"),ROW(11)},
   508 	{_S("=40"),NO_ROWS}
   509 	};
   510 
   511 struct TypeReal32
   512 	{
   513 public:
   514 	static const TText* const KType;
   515 	static const TReal32 KValues[];
   516 	static const TSelectTest KTests[];
   517 	};
   518 const TText* const TypeReal32::KType=_S("REAL");
   519 const TReal32 TypeReal32::KValues[]=
   520 	{
   521 	-KMaxTReal32,
   522 	-1.0f,
   523 	-KMinTReal32,
   524 	0.0f,
   525 	KMinTReal32,
   526 	1.0f,
   527 	KMaxTReal32
   528 	};
   529 const TSelectTest TypeReal32::KTests[]=
   530 	{
   531 	{_S("IS NULL"),ROW(0)},
   532 	{_S("IS NOT NULL"),ALL_ROWS},
   533 	{_S("=0"),ROW(4)},
   534 	{_S("<>0.0"),ALL_ROWS-ROW(4)},
   535 	{_S("=-1"),ROW(2)},
   536 	{_S("<2e1"),ROWS(1,6)},
   537 	{_S(">=-100000000000000"),ROWS(2,7)},
   538 	{_S("<1e-36"),ROWS(1,5)},
   539 	{_S(">1e15"),ROW(7)},
   540 	{_S("=.5"),NO_ROWS},
   541 	{_S("<=-.0"),ROWS(1,4)},
   542 	{_S("<1e40"),ALL_ROWS}
   543 	};
   544 
   545 struct TypeReal64
   546 	{
   547 public:
   548 	static const TText* const KType;
   549 	static const TReal64 KValues[];
   550 	static const TSelectTest KTests[];
   551 	};
   552 const TText* const TypeReal64::KType=_S("DOUBLE");
   553 const TReal64 TypeReal64::KValues[]=
   554 	{
   555 	-KMaxTReal64,
   556 	-1.0f,
   557 	-KMinTReal64,
   558 	0.0f,
   559 	KMinTReal64,
   560 	1.0f,
   561 	KMaxTReal64
   562 	};
   563 const TSelectTest TypeReal64::KTests[]=
   564 	{
   565 	{_S("IS NULL"),ROW(0)},
   566 	{_S("IS NOT NULL"),ALL_ROWS},
   567 	{_S("=0"),ROW(4)},
   568 	{_S("<>0"),ALL_ROWS-ROW(4)},
   569 	{_S("=-1"),ROW(2)},
   570 	{_S("<2"),ROWS(1,6)},
   571 	{_S(">=-100000000000000"),ROWS(2,7)},
   572 	{_S("<1e-300"),ROWS(1,5)},
   573 	{_S(">1e15"),ROW(7)},
   574 	{_S("=.5"),NO_ROWS},
   575 	{_S("<=0"),ROWS(1,4)},
   576 	{_S("<1e40"),ROWS(1,6)}
   577 	};
   578 
   579 struct TypeTime
   580 	{
   581 public:
   582 	static const TText* const KType;
   583 	static const TTime KValues[];
   584 	static const TSelectTest KTests[];
   585 	};
   586 const TText* const TypeTime::KType=_S("TIME");
   587 const TTime TypeTime::KValues[]=
   588 	{
   589 	TInt64(0u),				// zero date/time
   590 	_L(":085815"),			// 8:58:15am
   591 	_L("19181010:110000"),	// 11am, 11 Nov 1918
   592 	_L("19750226:"),		// midnight, 27 Mar 1975
   593 	_L("19961130:235959"),	// 11:59:59pm, 31 Dec 1996
   594 	_L("19970000:"),		// midnight, 1 Jan 1997
   595 	_L("19970611:210000"),	// 9pm, 12 July 1997
   596 	_L("19980309:214500"),	// 9:45pm, 10 April 1998
   597 	_L("20700608:")			// midnight, 9 July 2070
   598 	};
   599 const TSelectTest TypeTime::KTests[]=
   600 	{
   601 	{_S("IS NULL"),ROW(0)},
   602 	{_S("IS NOT NULL"),ALL_ROWS},
   603 	{_S("=#12am#"),ROW(1)},
   604 	{_S("<#Jan 1 2100#"),ALL_ROWS},
   605 	{_S("<>#31/12/1996 23:59:59#"),ALL_ROWS-ROW(5)},
   606 	{_S("<#9a#"),ROWS(1,2)},
   607 	{_S(">=#11:59:59pm, 31 Dec 1996#"),ROWS(5,9)},
   608 	{_S("=#9:45pm 10 April, 1998#"),ROW(8)},
   609 	{_S("=#8:58:15#"),ROW(2)}
   610 	};
   611 
   612 struct TypeText
   613 	{
   614 public:
   615 	static const TText* const KType;
   616 	static const TText* const KValues[];
   617 	static TSelectTest KTests[];
   618 	};
   619 const TText* const TypeText::KType=_S("VARCHAR(100)");
   620 const TText* const TypeText::KValues[]=
   621 	{
   622 	_S(""),					// this should be equivalent to NULL
   623 	_S("a"),
   624 	_S("aa"),
   625 	_S("aba"),
   626 	_S("like"),
   627 	_S("abcdefghijklmnopqrstuvwxyzlike"),
   628 	_S("likeabcdefghijklmnopqrstuvwxyz"),
   629 	_S("abcdefghijklmnopqrstuvwxyzlikeabcdefghijklmnopqrstuvwxyz"),
   630 	_S("abcdefghijklmnopqrstuvwxyzliveabcdefghijklmnopqrstuvwxyz"),
   631 	_S("l'ke"),
   632 	_S("'Tis")
   633 	};
   634 TSelectTest TypeText::KTests[]=
   635 	{
   636 	{_S("IS NULL"),ROWS(0,1)},
   637 	{_S("IS NOT NULL"),ALL_ROWS-ROW(1)},
   638 //
   639 	{_S("=''"),ROWS(0,1)},					// equivalent to IS NULL
   640 	{_S("<>''"),ALL_ROWS-ROW(1)},			// equivalent to IS NOT NULL
   641 	{_S(">''"),ALL_ROWS-ROW(1)},			// equivalent to IS NOT NULL
   642 	{_S("<''"),NO_ROWS},					// expression is trivially false
   643 	{_S("<=''"),ROWS(0,1)},
   644 	{_S(">=''"),ALL_ROWS+ROW(0)},			// expression is trivially true
   645 //
   646 	{_S("LIKE ''"),ROWS(0,1)},			// synonomous with IS NULL
   647 	{_S("NOT LIKE ''"),ALL_ROWS-ROW(1)},
   648 	{_S("LIKE '?*'"),ALL_ROWS-ROW(1)},	// synonomous with IS NOT NULL
   649 	{_S("NOT LIKE '?*'"),ROWS(0,1)},
   650 	{_S("LIKE '*'"),ALL_ROWS+ROW(0)},	// trivially true
   651 	{_S("NOT LIKE '*'"),NO_ROWS},
   652 //
   653 	{_S("='a'"),ROW(2)},
   654 	{_S("<'ab'"),ROWS(0,3)+ROW(11)},
   655 	{_S("<'abc'"),ROWS(0,4)+ROW(11)},
   656 	{_S("<'b'"),ROWS(0,4)+ROW(6)+ROWS(8,9)+ROW(11)},
   657 	{_S(">'abc'"),ROWS(5,10)},
   658 	{_S("='like'"),ROW(5)},
   659 	{_S("='l''ke'"),ROW(10)},
   660 	{_S("='''Tis'"),ROW(11)},
   661 //
   662 	{_S("LIKE 'a'"),ROW(2)},
   663 	{_S("LIKE 'a*'"),ROWS(2,4)+ROW(6)+ROWS(8,9)},
   664 	{_S("LIKE '*a'"),ROWS(2,4)},
   665 	{_S("LIKE 'a*a'"),ROWS(3,4)},
   666 	{_S("LIKE '*a*'"),ROWS(2,4)+ROWS(6,9)},
   667 //
   668 	{_S("LIKE 'like'"),ROW(5)},
   669 	{_S("LIKE 'l?ke'"),ROW(5)+ROW(10)},
   670 	{_S("LIKE 'like*'"),ROW(5)+ROW(7)},
   671 	{_S("LIKE '*like'"),ROWS(5,6)},
   672 	{_S("LIKE '*like*'"),ROWS(5,8)},
   673 	{_S("LIKE '*likeit*'"),NO_ROWS},
   674 	{_S("LIKE '*li?e*'"),ROWS(5,9)},
   675 	{_S("LIKE '?*li?e*'"),ROW(6)+ROWS(8,9)},
   676 	{_S("LIKE '*li?e*?'"),ROWS(7,9)},
   677 	{_S("LIKE '?*li?e*?'"),ROWS(8,9)},
   678 	{_S("LIKE '*?k?*'"),ROWS(5,10)},
   679 	{_S("LIKE '*?i?e*'"),ROWS(5,9)},
   680 //
   681 	{_S("LIKE '*e*'"),ROWS(5,10)},
   682 	{_S("LIKE '*z*k*e*'"),ROW(6)+ROW(8)},
   683 	{_S("LIKE '\?\?k?'"),ROW(5)+ROW(10)},
   684 	{_S("LIKE '\?\?k*'"),ROW(5)+ROW(7)+ROW(10)},
   685 	{_S("LIKE '*''*'"),ROWS(10,11)},
   686 	{_S("LIKE '?''\?\?'"),ROW(10)},
   687 	{_S("LIKE '?'"),ROW(2)},
   688 	{_S("LIKE '\?\?\?\?'"),ROW(5)+ROWS(10,11)},
   689 	{_S("LIKE '\?\?*\?\?'"),ROWS(5,11)},
   690 	{_S("LIKE '''*'"),ROW(11)}
   691 	};
   692 
   693 TEMPLATE_SPECIALIZATION struct SetCol<const TText* const>
   694 	{
   695 	static void SetColL(TDbColNo aCol,const TAny* aVal)
   696 		{
   697 		TheTable.SetColL(aCol,TPtrC(*(const TText* const*)aVal));
   698 		}
   699 	};
   700 TEMPLATE_SPECIALIZATION struct SetCol<const TText*>
   701 	{
   702 	static void SetColL(TDbColNo aCol,const TAny* aVal)
   703 		{
   704 		TheTable.SetColL(aCol,TPtrC(*(const TText* const*)aVal));
   705 		}
   706 	};
   707 
   708 struct TypeLongText
   709 	{
   710 public:
   711 	static const TText* const KType;
   712 	};
   713 const TText* const TypeLongText::KType=_S("LONG VARCHAR");
   714 
   715 class TestPredicate2 : public TestPredicateBase
   716 	{
   717 public:
   718 	static void RunL();
   719 	};
   720 
   721 // write rows equivalent to TypeText and use its tests
   722 void TestPredicate2::RunL()
   723 	{
   724 	Create(TypeLongText::KType);
   725 	TBuf<1022> fill=_S("abcdefghijklmnopqrstuvqxyz");
   726 	fill.AppendFill('.',fill.MaxLength()-fill.Length());
   727 
   728 	for (TInt row=0;row<=TInt(elementsof(TypeText::KValues));++row)
   729 		{
   730 		TheTable.InsertL();
   731 		TheTable.SetColL(1,row);
   732 		if (row>0)
   733 			{
   734 			RDbColWriteStream blob;
   735 			blob.OpenLC(TheTable,2);
   736 			switch (row)
   737 				{
   738 			case 0:
   739 				break;
   740 			case 1: case 2: case 3: case 4: case 5: case 10: case 11:
   741 				blob.WriteL(TPtrC(TypeText::KValues[row-1]));
   742 				break;
   743 			case 6:
   744 				blob.WriteL(fill);
   745 				blob.WriteL(_L("like"));
   746 				break;
   747 			case 7:
   748 				blob.WriteL(_L("like"));
   749 				blob.WriteL(fill);
   750 				break;
   751 			case 8:
   752 				blob.WriteL(fill);
   753 				blob.WriteL(_L("like"));
   754 				blob.WriteL(fill);
   755 				break;
   756 			case 9:
   757 				blob.WriteL(fill);
   758 				blob.WriteL(_L("live"));
   759 				blob.WriteL(fill);
   760 				break;
   761 				}
   762 			blob.CommitL();
   763 			CleanupStack::PopAndDestroy();
   764 			}
   765 		TheTable.PutL();
   766 		}
   767 	TheTable.Close();
   768 	TInt r=TheDatabase.Commit();
   769 	TEST2(r, KErrNone);
   770 	TestViewL(TypeText::KTests,elementsof(TypeText::KTests),elementsof(TypeText::KValues));
   771 	CDbKey& key=*CDbKey::NewLC();
   772 	key.AddL(TDbKeyCol(_L("Test"),120));
   773 	r=TheDatabase.CreateIndex(_L("Key"),_L("Compare"),key);
   774 	TEST2(r, KErrNone);
   775 	CleanupStack::PopAndDestroy();
   776 	TestViewL(TypeText::KTests,elementsof(TypeText::KTests),elementsof(TypeText::KValues));
   777 	r=TheDatabase.Execute(_L("DROP TABLE Compare"));
   778 	TEST2(r, KErrNone);
   779 	}
   780 
   781 /**
   782 * Utility for DEF063276 fix.
   783 */
   784 
   785 _LIT(KTypeTextKTests44, "Z:\\test\\TypeTextKTests44.dat");
   786 _LIT(KTypeTextKTests46, "Z:\\test\\TypeTextKTests46.dat");
   787 _LIT(KTypeTextKTests47, "Z:\\test\\TypeTextKTests47.dat");
   788 
   789 static void ReadDesc(TDes& aDes, const TDesC& aFilename, RFs& aFs)
   790 	{
   791     TheTest.Printf(_L("---ReadDesc(), aFilename=%S\r\n"), &aFilename);
   792 	RFile file;
   793 	TInt err = file.Open(aFs, aFilename, EFileRead);
   794 	TheTest.Printf(_L("Open file aFilename=%S err = %d\r\n"), &aFilename, err);
   795 	TEST2(err, KErrNone);
   796 
   797 	TPtr8 ptr(reinterpret_cast<TUint8*>(const_cast<TUint16*>(aDes.Ptr())), aDes.MaxSize());
   798 	err = file.Read(ptr);
   799 	TheTest.Printf(_L("Read file aFilename=%S err = %d\r\n"), &aFilename, err);
   800 	TEST2(err, KErrNone);
   801 	aDes.SetLength(ptr.Length() / sizeof(TText));
   802 	file.Close();
   803 	}
   804 
   805 /**
   806 @SYMTestCaseID          SYSLIB-DBMS-CT-0633
   807 @SYMTestCaseDesc        Tests the Predicate operators for all types
   808 @SYMTestPriority        Medium
   809 @SYMTestActions         Attempt to check with different types
   810 @SYMTestExpectedResults Test must not fail
   811 @SYMREQ                 REQ0000
   812 */
   813 LOCAL_C void TestPredicatesL()
   814 	{
   815 	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0633 init "));
   816 	CreateDatabase();
   817 	TestPredicate<TypeBit>::RunL();
   818 	TestPredicate<TypeUint8>::RunL();
   819 	TestPredicate<TypeUint16>::RunL();
   820 	TestPredicate<TypeUint32>::RunL();
   821 	TestPredicate<TypeInt8>::RunL();
   822 	TestPredicate<TypeInt16>::RunL();
   823 	TestPredicate<TypeInt32>::RunL();
   824 	TestPredicate<TypeInt64>::RunL();
   825 	TestPredicate<TypeReal32>::RunL();
   826 	TestPredicate<TypeReal64>::RunL();
   827 	TestPredicate<TypeTime>::RunL();
   828 
   829 	/**
   830 	* Work around for DEF063276.
   831 	* These literals are now loaded from Z:\test\TypeTextKTests44.dat,
   832 	* Z:\test\data\TypeTextKTests44.dat and Z:\test\TypeTextKTests44.dat respectively.
   833 	* Bullseye Coverage corrupts these literals to avoid this they are stored in files as to not be touched by Bullseye Coverage.
   834 	*/
   835 
   836 	TBuf<16> buf44;
   837 	ReadDesc(buf44, KTypeTextKTests44, TheFs);
   838 	TypeText::KTests[44].iSearchCondition = const_cast<TText*>(buf44.PtrZ());
   839 
   840 	TBuf<32> buf46(TypeText::KTests[46].iSearchCondition);
   841 	ReadDesc(buf46, KTypeTextKTests46, TheFs);
   842 	TypeText::KTests[46].iSearchCondition = const_cast<TText*>(buf46.PtrZ());
   843 
   844 	TBuf<32> buf47(TypeText::KTests[47].iSearchCondition);
   845 	ReadDesc(buf47, KTypeTextKTests47, TheFs);
   846 	TypeText::KTests[47].iSearchCondition = const_cast<TText*>(buf47.PtrZ());
   847 
   848 	// End fix.
   849 
   850 	TestPredicate<TypeText>::RunL();
   851 	TestPredicate2::RunL();
   852 	CloseDatabase();
   853 	TheTest.End();
   854 	}
   855 
   856 /**
   857 @SYMTestCaseID          SYSLIB-DBMS-CT-0634
   858 @SYMTestCaseDesc        DML Query test
   859                         Test for RDbNamedDatabase::Execute() function
   860 @SYMTestPriority        Medium
   861 @SYMTestActions         Tests for CREATE TABLE,CREATE UNIQUE INDEX,INSET INTO,UPDATE,DELETE,DROP queries
   862 @SYMTestExpectedResults Test must not fail
   863 @SYMREQ                 REQ0000
   864 */
   865 LOCAL_C void TestDataModificationlanguage()
   866 	{
   867 	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0634 init "));
   868 	Connect();
   869 	ShareDatabase();
   870 	TInt r=TheDatabase.Execute(_L("CREATE TABLE test (ID INTEGER NOT NULL,SALARY DOUBLE)"));
   871 	TEST2(r, KErrNone);
   872 	r=TheDatabase.Execute(_L("CREATE UNIQUE INDEX key ON test (ID)"));
   873 	TEST2(r, KErrNone);
   874 
   875 	TheTest.Next(_L("insert-statements"));
   876 	r=TheDatabase.Execute(_L("INSERT INTO test VALUES (0,0)"));
   877 	TEST(r==1);
   878 	r=TheDatabase.Execute(_L("INSERT INTO test (ID) VALUES (1)"));
   879 	TEST(r==1);
   880 	r=TheDatabase.Execute(_L("INSERT INTO test (SALARY,ID) VALUES (20.4,2)"));
   881 	TEST(r==1);
   882 
   883 	TheTest.Next(_L("update-statements"));
   884 	r=TheDatabase.Execute(_L("UPDATE test SET SALARY=30000 WHERE ID=1"));
   885 	TEST(r==1);
   886 
   887 	TheTest.Next(_L("delete-statements"));
   888 	r=TheDatabase.Execute(_L("DELETE FROM test WHERE SALARY<40"));
   889 	TEST(r==2);
   890 	r=TheDatabase.Execute(_L("DELETE FROM test"));
   891 	TEST(r==1);
   892 	r=TheDatabase.Execute(_L("DROP TABLE test"));
   893 	TEST2(r, KErrNone);
   894 //
   895 	TheTest.Next(_L("larger table"));
   896 	r=TheDatabase.Execute(_L("CREATE TABLE test (ID COUNTER,DATA INTEGER)"));
   897 	TEST2(r, KErrNone);
   898 
   899 	TheTest.Next(_L("insert"));
   900 	r=TheDatabase.Begin();
   901 	TEST2(r, KErrNone);
   902 	TBuf<256> sql;
   903 	for (TInt ii=0;ii<100;++ii)
   904 		{
   905 		sql.Format(_L("INSERT INTO test (DATA) VALUES (%D)"),100-ii);
   906 		r=TheDatabase.Execute(sql);
   907 		TEST(r==1);
   908 		}
   909 	r=TheDatabase.Commit();
   910 	TEST2(r, KErrNone);
   911 
   912 	TheTest.Next(_L("update"));
   913 	r=TheDatabase.Execute(_L("UPDATE test SET DATA=200 WHERE ID>=40 AND ID<60"));
   914 	TEST(r==20);
   915 
   916 	TheTest.Next(_L("delete"));
   917 	r=TheDatabase.Execute(_L("DELETE FROM test WHERE DATA>90"));
   918 	TEST(r==30);
   919 
   920 	TheTest.Next(_L("update"));
   921 	r=TheDatabase.Execute(_L("UPDATE test SET DATA=-1"));
   922 	TEST(r==70);
   923 
   924 	TheTest.Next(_L("delete"));
   925 	r=TheDatabase.Execute(_L("DELETE FROM test"));
   926 	TEST(r==70);
   927 	r=TheDatabase.Execute(_L("DROP TABLE test"));
   928 	TEST2(r, KErrNone);
   929 	CloseDatabase();
   930 	Disconnect();
   931 	TheTest.End();
   932 	}
   933 
   934 /**
   935 @SYMTestCaseID          SYSLIB-DBMS-CT-0635
   936 @SYMTestCaseDesc        DBMS SQL parsing and execution tests.Tests for database index order
   937 						Test for RDbNamedDatabase::Execute() function
   938 @SYMTestPriority        Medium
   939 @SYMTestActions         Tests for order by index
   940 @SYMTestExpectedResults Test must not fail
   941 @SYMREQ                 REQ0000
   942 */
   943 LOCAL_C void TestOrderByL()
   944 	{
   945 	static const TReal TestDataSalary[]={10,34000,15000,53200,17800,240000};
   946 	static const TText* const TestDataNames[]={_S("gopher"),_S("james '007' bond"),_S("moneypenny"),_S("Q"),_S("james '007' bond"),_S("M")};
   947 //
   948 	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0635 init "));
   949 	Connect();
   950 	ShareDatabase();
   951 	TheDatabase.Begin();
   952 	TInt r=TheDatabase.Execute(_L("CREATE TABLE test (ID INTEGER NOT NULL,SALARY DOUBLE,NAME VARCHAR)"));
   953 	TEST2(r, KErrNone);
   954 	r=TheDatabase.Execute(_L("CREATE UNIQUE INDEX key ON test (ID)"));
   955 	TEST2(r, KErrNone);
   956 
   957 	TheTest.Next(_L("insert data"));
   958 	r=TheView.Prepare(TheDatabase,_L("SELECT ID,SALARY,NAME FROM test"),TheView.EInsertOnly);
   959 	TEST2(r, KErrNone);
   960 	TInt ii;
   961 	for (ii=0;ii<6;++ii)
   962 		{
   963 		TheView.InsertL();
   964 		TheView.SetColL(1,6-ii);
   965 		TheView.SetColL(2,TestDataSalary[ii]);
   966 		TheView.SetColL(3,TPtrC(TestDataNames[ii]));
   967 		TheView.PutL();
   968 		}
   969 	r=TheDatabase.Commit();
   970 	TEST2(r, KErrNone);
   971 	TheView.Close();
   972 
   973 	TheTest.Next(_L("Order by <index>"));
   974 	// test the index is used here
   975 	r=TheView.Prepare(TheDatabase,_L("SELECT ID FROM test ORDER BY ID"));
   976 	TEST2(r, KErrNone);
   977 	TEST(!TheView.Unevaluated());
   978 	TInt c=0;
   979 	if (TheView.FirstL())
   980 		{
   981 		++c;
   982 		TheView.GetL();
   983 		TInt last=TheView.ColInt(1);
   984 		while (TheView.NextL())
   985 			{
   986 			++c;
   987 			TheView.GetL();
   988 			TInt v=TheView.ColInt(1);
   989 			TEST(v>last);
   990 			last=v;
   991 			}
   992 		}
   993 	TEST(c==6);
   994 	TEST(c==TheView.CountL());
   995 	TheView.Close();
   996 
   997 	TheTest.Next(_L("Order by <no-index> 1"));
   998 	// test that no index is used here
   999 	r=TheView.Prepare(TheDatabase,_L("SELECT SALARY FROM test ORDER BY SALARY"));
  1000 	TEST2(r, KErrNone);
  1001 	TEST(TheView.Unevaluated());
  1002 	r=TheView.EvaluateAll();
  1003 	TEST2(r, KErrNone);
  1004 	c=0;
  1005 	if (TheView.FirstL())
  1006 		{
  1007 		++c;
  1008 		TheView.GetL();
  1009 		TReal last=TheView.ColReal(1);
  1010 		while (TheView.NextL())
  1011 			{
  1012 			++c;
  1013 			TheView.GetL();
  1014 			TReal v=TheView.ColReal(1);
  1015 			TEST(v>=last);
  1016 			last=v;
  1017 			}
  1018 		}
  1019 	TEST(c==6);
  1020 	TEST(c==TheView.CountL());
  1021 	TheView.Close();
  1022 
  1023 	TheTest.Next(_L("Order by <no-index> 2"));
  1024 	// test that no index is used here
  1025 	r=TheView.Prepare(TheDatabase,_L("SELECT SALARY FROM test ORDER BY SALARY,NAME"));
  1026 	TEST2(r, KErrNone);
  1027 	TEST(TheView.Unevaluated());
  1028 	r=TheView.EvaluateAll();
  1029 	TEST2(r, KErrNone);
  1030 	c=0;
  1031 	if (TheView.FirstL())
  1032 		{
  1033 		++c;
  1034 		TheView.GetL();
  1035 		TReal last=TheView.ColReal(1);
  1036 		while (TheView.NextL())
  1037 			{
  1038 			++c;
  1039 			TheView.GetL();
  1040 			TReal v=TheView.ColReal(1);
  1041 			TEST(v>=last);
  1042 			last=v;
  1043 			}
  1044 		}
  1045 	TEST(c==6);
  1046 	TEST(c==TheView.CountL());
  1047 	TheView.Close();
  1048 
  1049 	TheTest.Next(_L("Order by <no-index> 3"));
  1050 	// test that no index is used here
  1051 	r=TheView.Prepare(TheDatabase,_L("SELECT NAME FROM test ORDER BY NAME"));
  1052 	TEST2(r, KErrNone);
  1053 	TEST(TheView.Unevaluated());
  1054 	r=TheView.EvaluateAll();
  1055 	TEST2(r, KErrNone);
  1056 	c=0;
  1057 	if (TheView.FirstL())
  1058 		{
  1059 		++c;
  1060 		TheView.GetL();
  1061 		TBuf<50> last=TheView.ColDes(1);
  1062 		while (TheView.NextL())
  1063 			{
  1064 			++c;
  1065 			TheView.GetL();
  1066 			TPtrC v=TheView.ColDes(1);
  1067 			TEST(v>=last);
  1068 			last=v;
  1069 			}
  1070 		}
  1071 	TEST(c==6);
  1072 	TEST(c==TheView.CountL());
  1073 	TheView.Close();
  1074 
  1075 	TheTest.Next(_L("Order by <no-index> 4"));
  1076 	// test that no index is used here
  1077 	r=TheView.Prepare(TheDatabase,_L("SELECT NAME FROM test ORDER BY NAME,SALARY"));
  1078 	TEST2(r, KErrNone);
  1079 	TEST(TheView.Unevaluated());
  1080 	r=TheView.EvaluateAll();
  1081 	TEST2(r, KErrNone);
  1082 	c=0;
  1083 	if (TheView.FirstL())
  1084 		{
  1085 		++c;
  1086 		TheView.GetL();
  1087 		TBuf<50> last=TheView.ColDes(1);
  1088 		while (TheView.NextL())
  1089 			{
  1090 			++c;
  1091 			TheView.GetL();
  1092 			TPtrC v=TheView.ColDes(1);
  1093 			TEST(v>=last);
  1094 			last=v;
  1095 			}
  1096 		}
  1097 	TEST(c==6);
  1098 	TEST(c==TheView.CountL());
  1099 	TheView.Close();
  1100 
  1101 	TheTest.Next(_L("Order by + search <no-index>"));
  1102 	// test that no index is used here
  1103 	r=TheView.Prepare(TheDatabase,_L("SELECT ID,SALARY FROM test WHERE SALARY>30000 ORDER BY SALARY DESC"));
  1104 	TEST2(r, KErrNone);
  1105 	TEST(TheView.Unevaluated());
  1106 	r=TheView.EvaluateAll();
  1107 	TEST2(r, KErrNone);
  1108 	c=0;
  1109 	if (TheView.FirstL())
  1110 		{
  1111 		++c;
  1112 		TheView.GetL();
  1113 		TReal last=TheView.ColReal(2);
  1114 		while (TheView.NextL())
  1115 			{
  1116 			++c;
  1117 			TheView.GetL();
  1118 			TReal v=TheView.ColReal(2);
  1119 			TEST(v<=last);
  1120 			last=v;
  1121 			}
  1122 		}
  1123 	TEST(c==3);
  1124 	TEST(c==TheView.CountL());
  1125 	TheView.Close();
  1126 //
  1127 	CloseDatabase();
  1128 	Disconnect();
  1129 
  1130 	TheTest.Next(_L("Order by <index> Finished"));
  1131 	TheTest.End();
  1132 	}
  1133 
  1134 LOCAL_C void doMain()
  1135 	{
  1136 	TheTest.Start(_L("Predicate tests"));
  1137 	TRAPD(errCode, TestPredicatesL());
  1138 	TEST2(errCode, KErrNone);
  1139 
  1140 	TheTest.Next(_L("DML execution"));
  1141 	TestDataModificationlanguage();
  1142 
  1143 	TheTest.Next(_L("ORDER BY clause"));
  1144 	TRAP(errCode, TestOrderByL());
  1145 	TEST2(errCode, KErrNone);
  1146 	TheTest.End();
  1147 	}
  1148 
  1149 // Prepare the test directory.
  1150 LOCAL_C void setupTestDirectory()
  1151     {
  1152 	TInt r=TheFs.Connect();
  1153 	TEST2(r, KErrNone);
  1154 //
  1155 	r=TheFs.MkDir(KTestDatabase);
  1156 	TEST(r==KErrNone || r==KErrAlreadyExists);
  1157 	}
  1158 
  1159 // Initialise the cleanup stack.
  1160 LOCAL_C void setupCleanup()
  1161     {
  1162 	TheTrapCleanup=CTrapCleanup::New();
  1163 	TEST(TheTrapCleanup!=NULL);
  1164 	TRAPD(r,\
  1165 		{\
  1166 		for (TInt i=KTestCleanupStack;i>0;i--)\
  1167 			CleanupStack::PushL((TAny*)0);\
  1168 		CleanupStack::Pop(KTestCleanupStack);\
  1169 		});
  1170 	TEST2(r, KErrNone);
  1171 	}
  1172 
  1173 // Test streaming conversions.
  1174 GLDEF_C TInt E32Main()
  1175     {
  1176 	TheTest.Title();
  1177 	setupTestDirectory();
  1178 	setupCleanup();
  1179 	__UHEAP_MARK;
  1180 //
  1181 	TheTest.Start(_L("Standard database"));
  1182 	doMain();
  1183 
  1184 	// clean up data file used by this test - must be done before call to End() - DEF047652
  1185 	::TestCleanup();
  1186 	TheTest.End();
  1187 //
  1188 	__UHEAP_MARKEND;
  1189 
  1190 
  1191 	delete TheTrapCleanup;
  1192 	TheTest.Close();
  1193 	return 0;
  1194     }