Update contrib.
1 // Copyright (c) 2006-2010 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.
20 #include "SqlResourceTester.h"
22 ///////////////////////////////////////////////////////////////////////////////////////
24 #define UNUSED_VAR(a) (a) = (a)
26 RTest TheTest(_L("t_sqlload test"));
28 TDriveNumber KTestDrive = EDriveC;
30 _LIT(KTestDir, "c:\\test\\");
31 _LIT(KTestDbName1, "c:\\test\\t_sqlload_1.db");
32 _LIT(KTestDbName2, "c:\\test\\t_sqlload_2.db");
33 _LIT(KTestDbName3, "c:\\test\\t_sqlload_3.db");
34 _LIT(KTestDbName4, "c:\\test\\t_sqlload_4.db");
35 _LIT(KTestDbName5, "c:\\test\\t_sqlload_5.db");
38 const TInt KTestThreadCnt = 4;
41 const TPtrC KTestDbNames[] =
49 const TInt KTestDbCnt = sizeof(KTestDbNames) / sizeof(KTestDbNames[0]);
52 const TInt KTestDuration = 120;//seconds
55 const TInt KRecordCnt = 100;
56 //Record count which will be used in the test SQL queries
57 const TInt KQueriedRecordCnt = 40;
58 //Every SQL query will be processed (stepped) in KTestStepCnt steps.
59 const TInt KTestStepCnt = 4;
60 //RSqlStatement object count which will be used in the tests
61 const TInt KStatementCnt = 10;
62 //Max allowed alive RSqlStatement objects per thread
63 const TInt KMaxStatementPerThread = 30;
65 const TInt KBinDataLen = 2003;
67 //StatementMaxNumberTest() time limit in seconds.
68 const TInt KTestTimeLimit = 60;//seconds
70 ///////////////////////////////////////////////////////////////////////////////////////
72 void DeleteTestFiles()
74 RSqlDatabase::Delete(KTestDbName5);
75 RSqlDatabase::Delete(KTestDbName4);
76 RSqlDatabase::Delete(KTestDbName3);
77 RSqlDatabase::Delete(KTestDbName2);
78 RSqlDatabase::Delete(KTestDbName1);
81 void GetHomeTimeAsString(TDes& aStr)
85 TDateTime dt = time.DateTime();
86 aStr.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
89 ///////////////////////////////////////////////////////////////////////////////////////
90 ///////////////////////////////////////////////////////////////////////////////////////
91 //Test macros and functions
92 void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
100 TName name = th.Name();
101 RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
105 RDebug::Print(_L("*** Line %d\r\n"), aLine);
107 TheTest(EFalse, aLine);
110 void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
112 if(aValue != aExpected)
118 TName name = th.Name();
119 RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
123 RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
125 TheTest(EFalse, aLine);
128 #define TEST(arg) ::Check1((arg), __LINE__)
129 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
130 #define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
131 #define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
133 ///////////////////////////////////////////////////////////////////////////////////////
135 //StatementMaxNumberTest() timeouts in WDP builds.
136 //This function is used to return the seconds passed from the start of the test case.
137 TTimeIntervalSeconds ExecutionTimeSeconds(TTime& aStartTime)
142 TTimeIntervalSeconds s;
143 TInt err = currTime.SecondsFrom(aStartTime, s);
144 TEST2(err, KErrNone);
151 TInt err = fs.Connect();
152 TEST2(err, KErrNone);
154 err = fs.MkDir(KTestDir);
155 TEST(err == KErrNone || err == KErrAlreadyExists);
157 err = fs.CreatePrivatePath(KTestDrive);
158 TEST(err == KErrNone || err == KErrAlreadyExists);
163 ///////////////////////////////////////////////////////////////////////////////////////
165 void CreateTestDatabases()
167 HBufC8* recData = HBufC8::New(KBinDataLen * 2 + 50);//"* 2" - hex values for the INSERT SQL statement
168 TEST(recData != NULL);
169 TPtr8 sql = recData->Des();
171 for(TInt dbIdx=0;dbIdx<KTestDbCnt;++dbIdx)
173 //Create test database
175 TInt err = db.Create(KTestDbNames[dbIdx]);
176 TEST2(err, KErrNone);
179 _LIT8(KCreateSql, "CREATE TABLE A(F1 INTEGER, F2 BLOB)");
180 err = db.Exec(KCreateSql);
183 //Insert records in the test table
184 for(TInt recIdx=1;recIdx<=KRecordCnt;++recIdx)
186 _LIT8(KInsertSql, "INSERT INTO A(F1, F2) VALUES(");
187 sql.Copy(KInsertSql);
188 sql.AppendNum((TInt64)recIdx);
189 sql.Append(_L(", X'"));
190 for(TInt k=0;k<KBinDataLen;++k)
192 sql.AppendFormat(_L8("%02X"), recIdx);
194 sql.Append(_L("')"));
205 //Structure used by the test thread function for orginizing its set of test data.
209 TBool iAlive; //Non-zero if iObj is alive
210 TInt iCurIndex; //The number of the current record in the set controlled by iObj statement
211 TInt iEndIndex; //The last record number in the set controlled by iObj statement
212 TInt iCount; //Records count in the set controlled by iObj statement
215 typedef RArray<TSqlStatement> RSqlStatementArray;
217 //Inits the random numbers generator.
218 //Opens one of the test databases.
219 void PreTest(RSqlDatabase& aDb, TInt64& aSeed, TName& aThreadName)
223 //Init the random numbers generator
226 aSeed = now.Int64() + currThread.Id();
228 //Open one of the test databases
229 const TInt KDbIndex = Math::Rand(aSeed) % KTestDbCnt;
231 aThreadName = currThread.Name();
232 RDebug::Print(_L("=== Thread %S, database %S\r\n"), &aThreadName, &KTestDbNames[KDbIndex]);
234 TInt err = aDb.Open(KTestDbNames[KDbIndex]);
235 TTEST2(err, KErrNone);
238 //Creates N statements, where 0 < N < KStatementCnt
239 TInt CreateStatements(RSqlDatabase& aDb, TInt64& aSeed, RSqlStatementArray& aStmtArray)
241 TInt stmtCount = Math::Rand(aSeed) % KStatementCnt;
246 for(TInt i=0;i<stmtCount;++i)
249 stmt.iAlive = EFalse;
250 stmt.iCount = KQueriedRecordCnt;
251 stmt.iCurIndex = Math::Rand(aSeed) % KRecordCnt;
252 if(stmt.iCurIndex == 0)
256 if(stmt.iCurIndex > (KRecordCnt - KQueriedRecordCnt))
258 stmt.iCurIndex = KRecordCnt - KQueriedRecordCnt;
260 stmt.iEndIndex = stmt.iCurIndex + KQueriedRecordCnt;
262 sql.Copy(_L8("SELECT * FROM A WHERE F1 >= "));
263 sql.AppendNum(stmt.iCurIndex);
264 sql.Append(_L8(" AND F1 < "));
265 sql.AppendNum(stmt.iEndIndex);
266 TInt err = stmt.iObj.Prepare(aDb, sql);
267 TTEST2(err, KErrNone);
269 err = aStmtArray.Append(stmt);
270 TTEST2(err, KErrNone);
275 //For each alive statement object - do (TSqlStatement::iCount / KTestStepCnt)
276 //RSqlStatement::Next() calls. If the Next() call reaches the end - close the statement object.
277 TInt ProcessStatements(RSqlStatementArray& aStmtArray)
279 const TInt KTotalStmtCount = aStmtArray.Count();
282 for(TInt k=0;k<KTotalStmtCount;++k)
284 TSqlStatement& stmt = aStmtArray[k];
288 TInt endIndex = stmt.iCurIndex + stmt.iCount / KTestStepCnt;
289 if(endIndex <= stmt.iEndIndex)
291 while(stmt.iCurIndex < endIndex)
293 TInt err = stmt.iObj.Next();
294 TTEST2(err, KSqlAtRow);
296 TInt val1 = stmt.iObj.ColumnInt(0);
297 TTEST(val1 == stmt.iCurIndex);
298 RSqlColumnReadStream strm;
299 err = strm.ColumnBinary(stmt.iObj, 1);
300 TTEST2(err, KErrNone);
301 for(TInt ii=0;ii<KBinDataLen;++ii)
304 TRAP(err, byte = strm.ReadUint8L());
305 TTEST2(err, KErrNone);
306 TTEST(byte == (TUint8)val1);
312 if(stmt.iCurIndex >= stmt.iEndIndex)
315 stmt.iAlive = EFalse;
323 //Close up to N statements, where 0 < N < KStatementCnt
324 TInt CloseStatements(RSqlStatementArray& aStmtArray, TInt64& aSeed)
326 TInt stmtCount = Math::Rand(aSeed) % KStatementCnt;
331 const TInt KTotalStmtCount = aStmtArray.Count();
333 for(TInt j=0;j<stmtCount;++j)
335 const TInt KIdx = Math::Rand(aSeed) % KTotalStmtCount;
337 while((idx = (++idx % KTotalStmtCount)) != KIdx)
339 if(aStmtArray[idx].iAlive)
341 aStmtArray[idx].iObj.Close();
342 aStmtArray[idx].iAlive = EFalse;
351 //Counts the alive statements
352 TInt AliveStatementsCount(RSqlStatementArray& aStmtArray)
355 const TInt KTotalStmtCount = aStmtArray.Count();
356 for(TInt l=0;l<KTotalStmtCount;++l)
358 if(aStmtArray[l].iAlive)
366 //Close all alive statements
367 void CloseAllStatements(RSqlStatementArray& aStmtArray)
369 const TInt KTotalStmtCount = aStmtArray.Count();
370 for(TInt i=0;i<KTotalStmtCount;++i)
372 if(aStmtArray[i].iAlive)
374 aStmtArray[i].iObj.Close();
377 TTEST2(TSqlResourceTester::Count(), 0);
380 //Removes the already closed statements and compresses the array
381 void RemoveDeadStatements(RSqlStatementArray& aStmtArray)
383 for(TInt i=aStmtArray.Count()-1;i>=0;--i)
385 if(!aStmtArray[i].iAlive)
387 aStmtArray.Remove(i);
390 aStmtArray.Compress();
393 //Close statement objects, statements array and the database object
394 TInt PostTest(RSqlDatabase& aDb, RSqlStatementArray& aStmtArray)
396 TInt statementsAlive = AliveStatementsCount(aStmtArray);
397 CloseAllStatements(aStmtArray);
400 return statementsAlive;
403 //Test thread function
404 //The thread function works with a set of TSqlStatement objects
405 //The test consists of 4 steps:
406 //Step 1: the test thread creates m TSqlStatement objects, 0 < m < KStatementCnt.
407 // With each of the created TSqlStatement objects the test thread prepares SELECT SQL query
408 // "SELECT * FROM A WHERE F1 >= K1 AND F1 < K2", where K1 is random generated number, such that:
409 // 0 < K1 < (KRecordCnt - KQueriedRecordCnt)
410 // K2 = K1 + KQueriedRecordCnt
411 // All just created TSqlStatement objects are marked as alive.
412 //Step 2: For each alive TSqlStatement object the test thread calls iObj.Next() method KTestStepCnt times,
413 // KTestStepCnt < KQueriedRecordCnt.
414 // The column values are retrieved and checked.
415 //Step 3: the test thread closes n TSqlStatement objects, 0 < n < KStatementCnt.
416 //Step 4: the test thread counts how many alive TSqlStatement objects are there.
417 // If this count > KMaxStatementPerThread then the test thread closes all alive TSqlStatement objects
418 // to avoid OOM errors during the test.
420 // Each test thread does steps 1..4 for a period of KTestDuration seconds.
421 // At the end all TSqlStatement objects are closed.
423 // The idea of the test is to load the SQL server creating several amount of statement and stream objects
424 // and see that it is working stable and without problems.
425 TInt ThreadFunc(void*)
429 CTrapCleanup* tc = CTrapCleanup::New();
435 RSqlStatementArray statements;
437 //Init the random numbers generator, opens the database
438 PreTest(db, seed, threadName);
443 currTime.UniversalTime();
444 TTime endTime = currTime + TTimeIntervalSeconds(KTestDuration);
445 while(currTime < endTime)
448 ///////////////////////////////////////////////////////////////////////
449 TInt statementsAliveBegin = statements.Count();
450 //Step 1: Create N statements, where 0 < N < KStatementCnt
451 TInt statementsCreated = CreateStatements(db, seed, statements);
452 ///////////////////////////////////////////////////////////////////////
453 //Step 2: For each alive statement object - do (TSqlStatement::iCount / KTestStepCnt)
454 // RSqlStatement::Next() calls. If the Next() call reaches the end - close the statement object.
455 TInt statementsCompleted = ProcessStatements(statements);
456 ///////////////////////////////////////////////////////////////////////
457 //Step 3: Close up to N statements, where 0 < N < KStatementCnt
458 TInt statementsClosed = CloseStatements(statements, seed);
459 ///////////////////////////////////////////////////////////////////////
460 //Step 4: If the alive statement count is more than KMaxStatementPerThread, then close them all
461 TInt statementsAliveEnd = AliveStatementsCount(statements);
462 if(statementsAliveEnd > KMaxStatementPerThread)
464 RDebug::Print(_L("!!! Thread %S, iteration %d, alive %d, close all\r\n"), &threadName, iteration, statementsAliveEnd);
465 CloseAllStatements(statements);
466 statementsAliveEnd = 0;
468 ///////////////////////////////////////////////////////////////////////
469 RemoveDeadStatements(statements);
470 RDebug::Print(_L("=== Thread %S, iteration % 4d, begin: % 3d, created % 2d, closed % 2d, completed % 2d, end % 3d, \r\n"),
471 &threadName, iteration, statementsAliveBegin,
472 statementsCreated, statementsClosed, statementsCompleted,
474 currTime.UniversalTime();
477 //Close statement objects and the database object
478 TInt statementsAlive = PostTest(db, statements);
484 RDebug::Print(_L("=== Thread %S exit, still alive %d\r\n"), &threadName, statementsAlive);
489 void CreateTestThreads(RThread aThreads[], TRequestStatus aStatuses[], TInt aMaxCount)
491 _LIT(KThreadName, "TstThr");
492 for(TInt i=0;i<aMaxCount;++i)
494 TBuf<20> threadName(KThreadName);
495 threadName.AppendNum((TInt64)(i + 1));
496 TEST2(aThreads[i].Create(threadName, &ThreadFunc, 0x2000, 0x1000, 0x10000, NULL, EOwnerProcess), KErrNone);
497 aThreads[i].Logon(aStatuses[i]);
498 TEST2(aStatuses[i].Int(), KRequestPending);
502 void ResumeTestThreads(RThread aThreads[], TInt aMaxCount)
504 for(TInt i=0;i<aMaxCount;++i)
506 aThreads[i].Resume();
511 void CloseTestThreads(RThread aThreads[], TRequestStatus aStatuses[], TInt aMaxCount)
513 for(TInt i=0;i<aMaxCount;++i)
515 User::WaitForRequest(aStatuses[i]);
516 TEST(aThreads[i].ExitType() != EExitPanic);
522 @SYMTestCaseID SYSLIB-SQL-CT-1627-0001
523 @SYMTestCaseDesc SQL server load test. The test creates KTestThreadCnt threads, KTestDbCnt test databases and
524 inserts in each of them KRecordCnt test records.
525 Pre-test step: each test thread randomly chooses and opens one of the test databases.
526 Then, each of the test threads is doing the following 4 test steps:
527 Step 1: the test thread creates m TSqlStatement objects, 0 < m < KStatementCnt.
528 With each of the created TSqlStatement objects the test thread prepares SELECT SQL query
529 "SELECT * FROM A WHERE F1 >= K1 AND F1 < K2", where K1 is random generated number, such that:
530 0 < K1 < (KRecordCnt - KQueriedRecordCnt)
531 K2 = K1 + KQueriedRecordCnt
532 All just created TSqlStatement objects are marked as alive.
533 Step 2: For each alive TSqlStatement object the test thread calls iObj.Next() method KTestStepCnt times,
534 KTestStepCnt < KQueriedRecordCnt.
535 The column values are retrieved and checked.
536 Step 3: the test thread closes n TSqlStatement objects, 0 < n < KStatementCnt.
537 Step 4: the test thread counts how many alive TSqlStatement objects are there.
538 If this count > KMaxStatementPerThread then the test thread closes all alive TSqlStatement objects
539 to avoid OOM errors during the test.
541 Each test thread does steps 1..4 for a period of KTestDuration seconds.
542 At the end all TSqlStatement objects are closed.
544 The idea of the test is to load the SQL server creating several amount of statement and stream objects
545 and see that it is working stable and without problems.
546 @SYMTestPriority High
547 @SYMTestActions SQL server load test
548 @SYMTestExpectedResults Test must not fail
554 CreateTestDatabases();
556 RThread threads[KTestThreadCnt];
557 TRequestStatus statuses[KTestThreadCnt];
559 CreateTestThreads(threads, statuses, KTestThreadCnt);
561 ResumeTestThreads(threads, KTestThreadCnt);
563 User::After(2000000);
565 CloseTestThreads(threads, statuses, KTestThreadCnt);
569 @SYMTestCaseID PDS-SQL-CT-4201
570 @SYMTestCaseDesc Max number of SQL statements test.
571 @SYMTestPriority High
572 @SYMTestActions The test creates a table with couple of records and then
573 creates as many as possible SQL statements. The expected result is
574 that either the statement creation process will fail with KErrNoMemory or
575 the max number of statements to be created is reached (100000).
576 Then the test deletes 1/2 of the created statements objects and
577 after that attempts to execute Next() on the rest of them.
578 Note that the test has a time limit of 120 seconds. Otherwise on some platforms
579 with WDP feature switched on the test may timeout.
580 @SYMTestExpectedResults Test must not fail
583 void StatementMaxNumberTest()
586 GetHomeTimeAsString(time);
587 TheTest.Printf(_L("=== %S: Create database\r\n"), &time);
589 (void)RSqlDatabase::Delete(KTestDbName1);
591 TInt err = db.Create(KTestDbName1);
592 TEST2(err, KErrNone);
593 err = db.Exec(_L("CREATE TABLE A(I INTEGER); INSERT INTO A(I) VALUES(1); INSERT INTO A(I) VALUES(2);"));
596 GetHomeTimeAsString(time);
597 TheTest.Printf(_L("=== %S: Create statements array\r\n"), &time);
599 //Reserve memory for the statement objects
600 const TInt KMaxStmtCount = 100000;
601 RSqlStatement* stmt = new RSqlStatement[KMaxStmtCount];
605 startTime.HomeTime();
607 //Create as many statement objects as possible
610 for(;idx<KMaxStmtCount;++idx)
612 err = stmt[idx].Prepare(db, _L("SELECT * FROM A WHERE I>=0 AND I<10"));
617 TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
620 GetHomeTimeAsString(time);
621 TheTest.Printf(_L("=== %S: Create % 5d statements. %d seconds.\r\n"), &time, idx + 1, s.Int());
623 if(s.Int() > KTestTimeLimit)
625 GetHomeTimeAsString(time);
626 TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
627 ++idx;//The idx-th statement is valid, the statement count is idx + 1.
633 TheTest.Printf(_L("%d created statement objects. Last error: %d.\r\n"), stmtCnt, err);
634 TEST(err == KErrNone || err == KErrNoMemory);
636 //Close 1/2 of the statements to free some memory
638 for(;idx<(stmtCnt/2);++idx)
643 GetHomeTimeAsString(time);
644 TheTest.Printf(_L("=== %S: % 5d statements closed\r\n"), &time, idx + 1);
648 //Now, there should be enough memory to be able to execute Next() on the rest of the statements
649 for(TInt j=0;idx<stmtCnt;++idx,++j)
651 err = stmt[idx].Next();
652 TEST2(err, KSqlAtRow);
653 err = stmt[idx].Next();
654 TEST2(err, KSqlAtRow);
655 err = stmt[idx].Next();
656 TEST2(err, KSqlAtEnd);
657 GetHomeTimeAsString(time);
658 TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
661 TheTest.Printf(_L("=== %S: % 5d statements processed. %d seconds.\r\n"), &time, j + 1, s.Int());
663 if(s.Int() > KTestTimeLimit)
665 TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
671 for(idx=0;idx<stmtCnt;++idx)
676 GetHomeTimeAsString(time);
677 TheTest.Printf(_L("=== %S: % 5d statements closed\r\n"), &time, idx + 1);
682 (void)RSqlDatabase::Delete(KTestDbName1);
683 GetHomeTimeAsString(time);
684 TheTest.Printf(_L("=== %S: Test case end\r\n"), &time);
687 TInt CreateFileSessions(TInt& aIdx, RFs aFs[], TInt aMaxFsSessCount)
691 startTime.HomeTime();
692 //Create as many file session objects as possible
694 for(;aIdx<aMaxFsSessCount;++aIdx)
696 err = aFs[aIdx].Connect();
701 TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
702 if((aIdx % 500) == 0)
704 GetHomeTimeAsString(time);
705 TheTest.Printf(_L("=== %S: Create % 5d file sessions. %d seconds.\r\n"), &time, aIdx + 1, s.Int());
707 if(s.Int() > KTestTimeLimit)
709 GetHomeTimeAsString(time);
710 TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
711 ++aIdx;//The idx-th file session object is valid, the file session count is idx + 1.
719 @SYMTestCaseID PDS-SQL-CT-4237
720 @SYMTestCaseDesc Max file session number test.
721 @SYMTestPriority High
722 @SYMTestActions The test creates as many as possible file session objects. The expected result is
723 that either the file session creation process will fail with KErrNoMemory or
724 the max number of file sessions to be created is reached (100000).
725 Then the test attempts to create a database. If there is no memory, the test
726 closes some of the file session objects. The test also attempts to copy
727 the created database and to delete it after that, both operations performed
728 with all file session objects still open. The expectation is that the test
729 will not crash the SQL server or the client side SQL dll.
730 Note that the test has a time limit of 120 seconds. Otherwise on some platforms
731 with WDP feature switched on the test may timeout.
732 @SYMTestExpectedResults Test must not fail
734 void FileSessionMaxNumberTest()
737 GetHomeTimeAsString(time);
738 TheTest.Printf(_L("=== %S: Create file sessions\r\n"), &time);
740 const TInt KMaxFsCount = 100000;
741 RFs* fs = new RFs[KMaxFsCount];
744 //Create as many file session objects as possible
746 TInt err = CreateFileSessions(idx, fs, KMaxFsCount);
747 TheTest.Printf(_L("%d created file session objects. Last error: %d.\r\n"), idx, err);
748 TEST(err == KErrNone || err == KErrNoMemory);
750 TBool dbCreated = EFalse;
753 //An attempt to create a database
754 while(idx > 0 && err == KErrNoMemory)
756 (void)RSqlDatabase::Delete(KTestDbName1);
757 err = db.Create(KTestDbName1);
760 err = db.Exec(_L("CREATE TABLE A(I INTEGER); INSERT INTO A(I) VALUES(1); INSERT INTO A(I) VALUES(2);"));
762 TheTest.Printf(_L("Database creation. Last error: %d.\r\n"), err);
763 TEST(err == KErrNoMemory || err >= 0);
764 if(err == KErrNoMemory)
777 //Create again file session objects - as many as possible
778 err = CreateFileSessions(idx, fs, KMaxFsCount);
779 TEST(err == KErrNone || err == KErrNoMemory);
780 //Try to copy the database
781 err = RSqlDatabase::Copy(KTestDbName1, KTestDbName4);
782 TheTest.Printf(_L("Copy database. Last error: %d.\r\n"), err);
783 TEST(err == KErrNone || err == KErrNoMemory);
784 //Try to delete the databases
787 err = RSqlDatabase::Delete(KTestDbName4);
788 TheTest.Printf(_L("Delete database copy. Last error: %d.\r\n"), err);
789 TEST(err == KErrNone || err == KErrNoMemory);
791 err = RSqlDatabase::Delete(KTestDbName1);
792 TheTest.Printf(_L("Delete database. Last error: %d.\r\n"), err);
793 TEST(err == KErrNone || err == KErrNoMemory);
797 for(TInt i=0;i<idx;++i)
802 GetHomeTimeAsString(time);
803 TheTest.Printf(_L("=== %S: % 5d file sessions closed\r\n"), &time, i + 1);
807 err = RSqlDatabase::Delete(KTestDbName4);
808 TEST(err == KErrNone || err == KErrNotFound);
809 err = RSqlDatabase::Delete(KTestDbName1);
810 TEST(err == KErrNone || err == KErrNotFound);
813 TInt CreateSqlConnections(TInt& aIdx, RSqlDatabase aDb[], TInt aMaxSqlConnCount)
817 startTime.HomeTime();
818 //Create as many file session objects as possible
820 for(;aIdx<aMaxSqlConnCount;++aIdx)
822 err = aDb[aIdx].Open(KTestDbName1);
827 TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
828 if((aIdx % 100) == 0)
830 GetHomeTimeAsString(time);
831 TheTest.Printf(_L("=== %S: Create % 5d sql connections. %d seconds.\r\n"), &time, aIdx + 1, s.Int());
833 if(s.Int() > KTestTimeLimit)
835 GetHomeTimeAsString(time);
836 TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
837 ++aIdx;//The idx-th sql connection is valid, the sql connection count is idx + 1.
845 @SYMTestCaseID PDS-SQL-CT-4238
846 @SYMTestCaseDesc Max sql connection number test.
847 @SYMTestPriority High
848 @SYMTestActions The test creates as many as possible sql connection objects. The expected result is
849 that either the sql connection creation process will fail with KErrNoMemory or
850 the max number of sql connection to be created is reached (100000).
851 Then the test attempts to create a database. If there is no memory, the test
852 closes some of the sql connection objects. The test also attempts to copy
853 the created database and to delete it after that, both operations performed
854 with all sql connection objects still open. The expectation is that the test
855 will not crash the SQL server or the client side SQL dll.
856 Note that the test has a time limit of 120 seconds. Otherwise on some platforms
857 with WDP feature switched on the test may timeout.
858 @SYMTestExpectedResults Test must not fail
860 void SqlConnectionMaxNumberTest()
863 GetHomeTimeAsString(time);
864 TheTest.Printf(_L("=== %S: Create sql connections\r\n"), &time);
866 (void)RSqlDatabase::Delete(KTestDbName1);
868 TInt err = db1.Create(KTestDbName1);//CreateSqlConnections() opens the already existing KTestDbName1 database
869 TEST2(err, KErrNone);
871 const TInt KMaxConnCount = 100000;
872 RSqlDatabase* db = new RSqlDatabase[KMaxConnCount];
875 //Create as many sql connection objects as possible
877 err = CreateSqlConnections(idx, db, KMaxConnCount);
878 TheTest.Printf(_L("%d created sql connection objects. Last error: %d.\r\n"), idx, err);
879 TEST(err == KErrNone || err == KErrNoMemory);
881 TBool dbCreated = EFalse;
884 //An attempt to create a database
885 while(idx > 0 && err == KErrNoMemory)
887 (void)RSqlDatabase::Delete(KTestDbName4);
888 err = db2.Create(KTestDbName4);
891 err = db2.Exec(_L("CREATE TABLE A(I INTEGER); INSERT INTO A(I) VALUES(1); INSERT INTO A(I) VALUES(2);"));
893 TheTest.Printf(_L("Database creation. Last error: %d.\r\n"), err);
894 TEST(err == KErrNoMemory || err >= 0);
895 if(err == KErrNoMemory)
908 //Create again sql connection objects - as many as possible
909 err = CreateSqlConnections(idx, db, KMaxConnCount);
910 TEST(err == KErrNone || err == KErrNoMemory);
911 //Try to copy the database
912 err = RSqlDatabase::Copy(KTestDbName4, KTestDbName5);
913 TheTest.Printf(_L("Copy database. Last error: %d.\r\n"), err);
914 TEST(err == KErrNone || err == KErrNoMemory);
915 //Try to delete the databases
918 err = RSqlDatabase::Delete(KTestDbName5);
919 TheTest.Printf(_L("Delete database copy. Last error: %d.\r\n"), err);
920 TEST(err == KErrNone || err == KErrNoMemory);
922 err = RSqlDatabase::Delete(KTestDbName4);
923 TheTest.Printf(_L("Delete database. Last error: %d.\r\n"), err);
924 TEST(err == KErrNone || err == KErrNoMemory);
928 for(TInt i=0;i<idx;++i)
933 GetHomeTimeAsString(time);
934 TheTest.Printf(_L("=== %S: % 5d sql connections closed\r\n"), &time, i + 1);
939 err = RSqlDatabase::Delete(KTestDbName5);
940 TEST(err == KErrNone || err == KErrNotFound);
941 err = RSqlDatabase::Delete(KTestDbName4);
942 TEST(err == KErrNone || err == KErrNotFound);
943 err = RSqlDatabase::Delete(KTestDbName1);
944 TEST(err == KErrNone || err == KErrNotFound);
949 TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1627-0001 SQL server load test "));
951 TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4201 Statement max number test"));
952 StatementMaxNumberTest();
953 #if defined __WINS__ || defined __WINSCW__
954 //The next two tests are likely to timeout on hardware because they create a lot of file sessions and sql connections.
955 //The SQL server heap is 32Mb on hardware but only 6Mb on the Emulator.
956 TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4237 File session max number test"));
957 FileSessionMaxNumberTest();
958 TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4238 Sql connection max number test"));
959 SqlConnectionMaxNumberTest();
967 CTrapCleanup* tc = CTrapCleanup::New();
983 User::Heap().Check();