sl@0
|
1 |
// Copyright (c) 2006-2010 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
2 |
// All rights reserved.
|
sl@0
|
3 |
// This component and the accompanying materials are made available
|
sl@0
|
4 |
// under the terms of "Eclipse Public License v1.0"
|
sl@0
|
5 |
// which accompanies this distribution, and is available
|
sl@0
|
6 |
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
7 |
//
|
sl@0
|
8 |
// Initial Contributors:
|
sl@0
|
9 |
// Nokia Corporation - initial contribution.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Contributors:
|
sl@0
|
12 |
//
|
sl@0
|
13 |
// Description:
|
sl@0
|
14 |
//
|
sl@0
|
15 |
|
sl@0
|
16 |
#include <e32test.h>
|
sl@0
|
17 |
#include <e32math.h>
|
sl@0
|
18 |
#include <bautils.h>
|
sl@0
|
19 |
#include <sqldb.h>
|
sl@0
|
20 |
#include "SqlResourceTester.h"
|
sl@0
|
21 |
|
sl@0
|
22 |
///////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
23 |
|
sl@0
|
24 |
#define UNUSED_VAR(a) (a) = (a)
|
sl@0
|
25 |
|
sl@0
|
26 |
RTest TheTest(_L("t_sqlload test"));
|
sl@0
|
27 |
|
sl@0
|
28 |
TDriveNumber KTestDrive = EDriveC;
|
sl@0
|
29 |
|
sl@0
|
30 |
_LIT(KTestDir, "c:\\test\\");
|
sl@0
|
31 |
_LIT(KTestDbName1, "c:\\test\\t_sqlload_1.db");
|
sl@0
|
32 |
_LIT(KTestDbName2, "c:\\test\\t_sqlload_2.db");
|
sl@0
|
33 |
_LIT(KTestDbName3, "c:\\test\\t_sqlload_3.db");
|
sl@0
|
34 |
_LIT(KTestDbName4, "c:\\test\\t_sqlload_4.db");
|
sl@0
|
35 |
_LIT(KTestDbName5, "c:\\test\\t_sqlload_5.db");
|
sl@0
|
36 |
|
sl@0
|
37 |
//Test thread count
|
sl@0
|
38 |
const TInt KTestThreadCnt = 4;
|
sl@0
|
39 |
|
sl@0
|
40 |
//Test database names
|
sl@0
|
41 |
const TPtrC KTestDbNames[] =
|
sl@0
|
42 |
{
|
sl@0
|
43 |
KTestDbName1(),
|
sl@0
|
44 |
KTestDbName2(),
|
sl@0
|
45 |
KTestDbName3()
|
sl@0
|
46 |
};
|
sl@0
|
47 |
|
sl@0
|
48 |
//Test database count
|
sl@0
|
49 |
const TInt KTestDbCnt = sizeof(KTestDbNames) / sizeof(KTestDbNames[0]);
|
sl@0
|
50 |
|
sl@0
|
51 |
//Test duration
|
sl@0
|
52 |
const TInt KTestDuration = 120;//seconds
|
sl@0
|
53 |
|
sl@0
|
54 |
//Test record count
|
sl@0
|
55 |
const TInt KRecordCnt = 100;
|
sl@0
|
56 |
//Record count which will be used in the test SQL queries
|
sl@0
|
57 |
const TInt KQueriedRecordCnt = 40;
|
sl@0
|
58 |
//Every SQL query will be processed (stepped) in KTestStepCnt steps.
|
sl@0
|
59 |
const TInt KTestStepCnt = 4;
|
sl@0
|
60 |
//RSqlStatement object count which will be used in the tests
|
sl@0
|
61 |
const TInt KStatementCnt = 10;
|
sl@0
|
62 |
//Max allowed alive RSqlStatement objects per thread
|
sl@0
|
63 |
const TInt KMaxStatementPerThread = 30;
|
sl@0
|
64 |
//Binary data length
|
sl@0
|
65 |
const TInt KBinDataLen = 2003;
|
sl@0
|
66 |
|
sl@0
|
67 |
//StatementMaxNumberTest() time limit in seconds.
|
sl@0
|
68 |
const TInt KTestTimeLimit = 60;//seconds
|
sl@0
|
69 |
|
sl@0
|
70 |
///////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
71 |
|
sl@0
|
72 |
void DeleteTestFiles()
|
sl@0
|
73 |
{
|
sl@0
|
74 |
RSqlDatabase::Delete(KTestDbName5);
|
sl@0
|
75 |
RSqlDatabase::Delete(KTestDbName4);
|
sl@0
|
76 |
RSqlDatabase::Delete(KTestDbName3);
|
sl@0
|
77 |
RSqlDatabase::Delete(KTestDbName2);
|
sl@0
|
78 |
RSqlDatabase::Delete(KTestDbName1);
|
sl@0
|
79 |
}
|
sl@0
|
80 |
|
sl@0
|
81 |
void GetHomeTimeAsString(TDes& aStr)
|
sl@0
|
82 |
{
|
sl@0
|
83 |
TTime time;
|
sl@0
|
84 |
time.HomeTime();
|
sl@0
|
85 |
TDateTime dt = time.DateTime();
|
sl@0
|
86 |
aStr.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
|
sl@0
|
87 |
}
|
sl@0
|
88 |
|
sl@0
|
89 |
///////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
90 |
///////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
91 |
//Test macros and functions
|
sl@0
|
92 |
void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
|
sl@0
|
93 |
{
|
sl@0
|
94 |
if(!aValue)
|
sl@0
|
95 |
{
|
sl@0
|
96 |
DeleteTestFiles();
|
sl@0
|
97 |
if(aPrintThreadName)
|
sl@0
|
98 |
{
|
sl@0
|
99 |
RThread th;
|
sl@0
|
100 |
TName name = th.Name();
|
sl@0
|
101 |
RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
|
sl@0
|
102 |
}
|
sl@0
|
103 |
else
|
sl@0
|
104 |
{
|
sl@0
|
105 |
RDebug::Print(_L("*** Line %d\r\n"), aLine);
|
sl@0
|
106 |
}
|
sl@0
|
107 |
TheTest(EFalse, aLine);
|
sl@0
|
108 |
}
|
sl@0
|
109 |
}
|
sl@0
|
110 |
void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
|
sl@0
|
111 |
{
|
sl@0
|
112 |
if(aValue != aExpected)
|
sl@0
|
113 |
{
|
sl@0
|
114 |
DeleteTestFiles();
|
sl@0
|
115 |
if(aPrintThreadName)
|
sl@0
|
116 |
{
|
sl@0
|
117 |
RThread th;
|
sl@0
|
118 |
TName name = th.Name();
|
sl@0
|
119 |
RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
|
sl@0
|
120 |
}
|
sl@0
|
121 |
else
|
sl@0
|
122 |
{
|
sl@0
|
123 |
RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
|
sl@0
|
124 |
}
|
sl@0
|
125 |
TheTest(EFalse, aLine);
|
sl@0
|
126 |
}
|
sl@0
|
127 |
}
|
sl@0
|
128 |
#define TEST(arg) ::Check1((arg), __LINE__)
|
sl@0
|
129 |
#define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
|
sl@0
|
130 |
#define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
|
sl@0
|
131 |
#define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
|
sl@0
|
132 |
|
sl@0
|
133 |
///////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
134 |
|
sl@0
|
135 |
//StatementMaxNumberTest() timeouts in WDP builds.
|
sl@0
|
136 |
//This function is used to return the seconds passed from the start of the test case.
|
sl@0
|
137 |
TTimeIntervalSeconds ExecutionTimeSeconds(TTime& aStartTime)
|
sl@0
|
138 |
{
|
sl@0
|
139 |
TTime currTime;
|
sl@0
|
140 |
currTime.HomeTime();
|
sl@0
|
141 |
|
sl@0
|
142 |
TTimeIntervalSeconds s;
|
sl@0
|
143 |
TInt err = currTime.SecondsFrom(aStartTime, s);
|
sl@0
|
144 |
TEST2(err, KErrNone);
|
sl@0
|
145 |
return s;
|
sl@0
|
146 |
}
|
sl@0
|
147 |
|
sl@0
|
148 |
void CreateTestDir()
|
sl@0
|
149 |
{
|
sl@0
|
150 |
RFs fs;
|
sl@0
|
151 |
TInt err = fs.Connect();
|
sl@0
|
152 |
TEST2(err, KErrNone);
|
sl@0
|
153 |
|
sl@0
|
154 |
err = fs.MkDir(KTestDir);
|
sl@0
|
155 |
TEST(err == KErrNone || err == KErrAlreadyExists);
|
sl@0
|
156 |
|
sl@0
|
157 |
err = fs.CreatePrivatePath(KTestDrive);
|
sl@0
|
158 |
TEST(err == KErrNone || err == KErrAlreadyExists);
|
sl@0
|
159 |
|
sl@0
|
160 |
fs.Close();
|
sl@0
|
161 |
}
|
sl@0
|
162 |
|
sl@0
|
163 |
///////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
164 |
|
sl@0
|
165 |
void CreateTestDatabases()
|
sl@0
|
166 |
{
|
sl@0
|
167 |
HBufC8* recData = HBufC8::New(KBinDataLen * 2 + 50);//"* 2" - hex values for the INSERT SQL statement
|
sl@0
|
168 |
TEST(recData != NULL);
|
sl@0
|
169 |
TPtr8 sql = recData->Des();
|
sl@0
|
170 |
|
sl@0
|
171 |
for(TInt dbIdx=0;dbIdx<KTestDbCnt;++dbIdx)
|
sl@0
|
172 |
{
|
sl@0
|
173 |
//Create test database
|
sl@0
|
174 |
RSqlDatabase db;
|
sl@0
|
175 |
TInt err = db.Create(KTestDbNames[dbIdx]);
|
sl@0
|
176 |
TEST2(err, KErrNone);
|
sl@0
|
177 |
|
sl@0
|
178 |
//Create test table
|
sl@0
|
179 |
_LIT8(KCreateSql, "CREATE TABLE A(F1 INTEGER, F2 BLOB)");
|
sl@0
|
180 |
err = db.Exec(KCreateSql);
|
sl@0
|
181 |
TEST(err >= 0);
|
sl@0
|
182 |
|
sl@0
|
183 |
//Insert records in the test table
|
sl@0
|
184 |
for(TInt recIdx=1;recIdx<=KRecordCnt;++recIdx)
|
sl@0
|
185 |
{
|
sl@0
|
186 |
_LIT8(KInsertSql, "INSERT INTO A(F1, F2) VALUES(");
|
sl@0
|
187 |
sql.Copy(KInsertSql);
|
sl@0
|
188 |
sql.AppendNum((TInt64)recIdx);
|
sl@0
|
189 |
sql.Append(_L(", X'"));
|
sl@0
|
190 |
for(TInt k=0;k<KBinDataLen;++k)
|
sl@0
|
191 |
{
|
sl@0
|
192 |
sql.AppendFormat(_L8("%02X"), recIdx);
|
sl@0
|
193 |
}
|
sl@0
|
194 |
sql.Append(_L("')"));
|
sl@0
|
195 |
err = db.Exec(sql);
|
sl@0
|
196 |
TEST2(err, 1);
|
sl@0
|
197 |
}
|
sl@0
|
198 |
|
sl@0
|
199 |
db.Close();
|
sl@0
|
200 |
}
|
sl@0
|
201 |
|
sl@0
|
202 |
delete recData;
|
sl@0
|
203 |
}
|
sl@0
|
204 |
|
sl@0
|
205 |
//Structure used by the test thread function for orginizing its set of test data.
|
sl@0
|
206 |
struct TSqlStatement
|
sl@0
|
207 |
{
|
sl@0
|
208 |
RSqlStatement iObj;
|
sl@0
|
209 |
TBool iAlive; //Non-zero if iObj is alive
|
sl@0
|
210 |
TInt iCurIndex; //The number of the current record in the set controlled by iObj statement
|
sl@0
|
211 |
TInt iEndIndex; //The last record number in the set controlled by iObj statement
|
sl@0
|
212 |
TInt iCount; //Records count in the set controlled by iObj statement
|
sl@0
|
213 |
};
|
sl@0
|
214 |
|
sl@0
|
215 |
typedef RArray<TSqlStatement> RSqlStatementArray;
|
sl@0
|
216 |
|
sl@0
|
217 |
//Inits the random numbers generator.
|
sl@0
|
218 |
//Opens one of the test databases.
|
sl@0
|
219 |
void PreTest(RSqlDatabase& aDb, TInt64& aSeed, TName& aThreadName)
|
sl@0
|
220 |
{
|
sl@0
|
221 |
RThread currThread;
|
sl@0
|
222 |
|
sl@0
|
223 |
//Init the random numbers generator
|
sl@0
|
224 |
TTime now;
|
sl@0
|
225 |
now.UniversalTime();
|
sl@0
|
226 |
aSeed = now.Int64() + currThread.Id();
|
sl@0
|
227 |
|
sl@0
|
228 |
//Open one of the test databases
|
sl@0
|
229 |
const TInt KDbIndex = Math::Rand(aSeed) % KTestDbCnt;
|
sl@0
|
230 |
|
sl@0
|
231 |
aThreadName = currThread.Name();
|
sl@0
|
232 |
RDebug::Print(_L("=== Thread %S, database %S\r\n"), &aThreadName, &KTestDbNames[KDbIndex]);
|
sl@0
|
233 |
|
sl@0
|
234 |
TInt err = aDb.Open(KTestDbNames[KDbIndex]);
|
sl@0
|
235 |
TTEST2(err, KErrNone);
|
sl@0
|
236 |
}
|
sl@0
|
237 |
|
sl@0
|
238 |
//Creates N statements, where 0 < N < KStatementCnt
|
sl@0
|
239 |
TInt CreateStatements(RSqlDatabase& aDb, TInt64& aSeed, RSqlStatementArray& aStmtArray)
|
sl@0
|
240 |
{
|
sl@0
|
241 |
TInt stmtCount = Math::Rand(aSeed) % KStatementCnt;
|
sl@0
|
242 |
if(stmtCount == 0)
|
sl@0
|
243 |
{
|
sl@0
|
244 |
stmtCount = 1;
|
sl@0
|
245 |
}
|
sl@0
|
246 |
for(TInt i=0;i<stmtCount;++i)
|
sl@0
|
247 |
{
|
sl@0
|
248 |
TSqlStatement stmt;
|
sl@0
|
249 |
stmt.iAlive = EFalse;
|
sl@0
|
250 |
stmt.iCount = KQueriedRecordCnt;
|
sl@0
|
251 |
stmt.iCurIndex = Math::Rand(aSeed) % KRecordCnt;
|
sl@0
|
252 |
if(stmt.iCurIndex == 0)
|
sl@0
|
253 |
{
|
sl@0
|
254 |
stmt.iCurIndex = 1;
|
sl@0
|
255 |
}
|
sl@0
|
256 |
if(stmt.iCurIndex > (KRecordCnt - KQueriedRecordCnt))
|
sl@0
|
257 |
{
|
sl@0
|
258 |
stmt.iCurIndex = KRecordCnt - KQueriedRecordCnt;
|
sl@0
|
259 |
}
|
sl@0
|
260 |
stmt.iEndIndex = stmt.iCurIndex + KQueriedRecordCnt;
|
sl@0
|
261 |
TBuf8<100> sql;
|
sl@0
|
262 |
sql.Copy(_L8("SELECT * FROM A WHERE F1 >= "));
|
sl@0
|
263 |
sql.AppendNum(stmt.iCurIndex);
|
sl@0
|
264 |
sql.Append(_L8(" AND F1 < "));
|
sl@0
|
265 |
sql.AppendNum(stmt.iEndIndex);
|
sl@0
|
266 |
TInt err = stmt.iObj.Prepare(aDb, sql);
|
sl@0
|
267 |
TTEST2(err, KErrNone);
|
sl@0
|
268 |
stmt.iAlive = ETrue;
|
sl@0
|
269 |
err = aStmtArray.Append(stmt);
|
sl@0
|
270 |
TTEST2(err, KErrNone);
|
sl@0
|
271 |
}
|
sl@0
|
272 |
return stmtCount;
|
sl@0
|
273 |
}
|
sl@0
|
274 |
|
sl@0
|
275 |
//For each alive statement object - do (TSqlStatement::iCount / KTestStepCnt)
|
sl@0
|
276 |
//RSqlStatement::Next() calls. If the Next() call reaches the end - close the statement object.
|
sl@0
|
277 |
TInt ProcessStatements(RSqlStatementArray& aStmtArray)
|
sl@0
|
278 |
{
|
sl@0
|
279 |
const TInt KTotalStmtCount = aStmtArray.Count();
|
sl@0
|
280 |
TInt alive = 0;
|
sl@0
|
281 |
TInt completed = 0;
|
sl@0
|
282 |
for(TInt k=0;k<KTotalStmtCount;++k)
|
sl@0
|
283 |
{
|
sl@0
|
284 |
TSqlStatement& stmt = aStmtArray[k];
|
sl@0
|
285 |
if(stmt.iAlive)
|
sl@0
|
286 |
{
|
sl@0
|
287 |
++alive;
|
sl@0
|
288 |
TInt endIndex = stmt.iCurIndex + stmt.iCount / KTestStepCnt;
|
sl@0
|
289 |
if(endIndex <= stmt.iEndIndex)
|
sl@0
|
290 |
{
|
sl@0
|
291 |
while(stmt.iCurIndex < endIndex)
|
sl@0
|
292 |
{
|
sl@0
|
293 |
TInt err = stmt.iObj.Next();
|
sl@0
|
294 |
TTEST2(err, KSqlAtRow);
|
sl@0
|
295 |
//test column values
|
sl@0
|
296 |
TInt val1 = stmt.iObj.ColumnInt(0);
|
sl@0
|
297 |
TTEST(val1 == stmt.iCurIndex);
|
sl@0
|
298 |
RSqlColumnReadStream strm;
|
sl@0
|
299 |
err = strm.ColumnBinary(stmt.iObj, 1);
|
sl@0
|
300 |
TTEST2(err, KErrNone);
|
sl@0
|
301 |
for(TInt ii=0;ii<KBinDataLen;++ii)
|
sl@0
|
302 |
{
|
sl@0
|
303 |
TUint8 byte = 0;
|
sl@0
|
304 |
TRAP(err, byte = strm.ReadUint8L());
|
sl@0
|
305 |
TTEST2(err, KErrNone);
|
sl@0
|
306 |
TTEST(byte == (TUint8)val1);
|
sl@0
|
307 |
}
|
sl@0
|
308 |
strm.Close();
|
sl@0
|
309 |
++stmt.iCurIndex;
|
sl@0
|
310 |
}
|
sl@0
|
311 |
}
|
sl@0
|
312 |
if(stmt.iCurIndex >= stmt.iEndIndex)
|
sl@0
|
313 |
{
|
sl@0
|
314 |
stmt.iObj.Close();
|
sl@0
|
315 |
stmt.iAlive = EFalse;
|
sl@0
|
316 |
++completed;
|
sl@0
|
317 |
}
|
sl@0
|
318 |
}
|
sl@0
|
319 |
}
|
sl@0
|
320 |
return completed;
|
sl@0
|
321 |
}
|
sl@0
|
322 |
|
sl@0
|
323 |
//Close up to N statements, where 0 < N < KStatementCnt
|
sl@0
|
324 |
TInt CloseStatements(RSqlStatementArray& aStmtArray, TInt64& aSeed)
|
sl@0
|
325 |
{
|
sl@0
|
326 |
TInt stmtCount = Math::Rand(aSeed) % KStatementCnt;
|
sl@0
|
327 |
if(stmtCount == 0)
|
sl@0
|
328 |
{
|
sl@0
|
329 |
stmtCount = 1;
|
sl@0
|
330 |
}
|
sl@0
|
331 |
const TInt KTotalStmtCount = aStmtArray.Count();
|
sl@0
|
332 |
TInt closed = 0;
|
sl@0
|
333 |
for(TInt j=0;j<stmtCount;++j)
|
sl@0
|
334 |
{
|
sl@0
|
335 |
const TInt KIdx = Math::Rand(aSeed) % KTotalStmtCount;
|
sl@0
|
336 |
TInt idx = KIdx;
|
sl@0
|
337 |
while((idx = (++idx % KTotalStmtCount)) != KIdx)
|
sl@0
|
338 |
{
|
sl@0
|
339 |
if(aStmtArray[idx].iAlive)
|
sl@0
|
340 |
{
|
sl@0
|
341 |
aStmtArray[idx].iObj.Close();
|
sl@0
|
342 |
aStmtArray[idx].iAlive = EFalse;
|
sl@0
|
343 |
++closed;
|
sl@0
|
344 |
break;
|
sl@0
|
345 |
}
|
sl@0
|
346 |
}
|
sl@0
|
347 |
}
|
sl@0
|
348 |
return closed;
|
sl@0
|
349 |
}
|
sl@0
|
350 |
|
sl@0
|
351 |
//Counts the alive statements
|
sl@0
|
352 |
TInt AliveStatementsCount(RSqlStatementArray& aStmtArray)
|
sl@0
|
353 |
{
|
sl@0
|
354 |
TInt aliveCnt = 0;
|
sl@0
|
355 |
const TInt KTotalStmtCount = aStmtArray.Count();
|
sl@0
|
356 |
for(TInt l=0;l<KTotalStmtCount;++l)
|
sl@0
|
357 |
{
|
sl@0
|
358 |
if(aStmtArray[l].iAlive)
|
sl@0
|
359 |
{
|
sl@0
|
360 |
++aliveCnt;
|
sl@0
|
361 |
}
|
sl@0
|
362 |
}
|
sl@0
|
363 |
return aliveCnt;
|
sl@0
|
364 |
}
|
sl@0
|
365 |
|
sl@0
|
366 |
//Close all alive statements
|
sl@0
|
367 |
void CloseAllStatements(RSqlStatementArray& aStmtArray)
|
sl@0
|
368 |
{
|
sl@0
|
369 |
const TInt KTotalStmtCount = aStmtArray.Count();
|
sl@0
|
370 |
for(TInt i=0;i<KTotalStmtCount;++i)
|
sl@0
|
371 |
{
|
sl@0
|
372 |
if(aStmtArray[i].iAlive)
|
sl@0
|
373 |
{
|
sl@0
|
374 |
aStmtArray[i].iObj.Close();
|
sl@0
|
375 |
}
|
sl@0
|
376 |
}
|
sl@0
|
377 |
TTEST2(TSqlResourceTester::Count(), 0);
|
sl@0
|
378 |
}
|
sl@0
|
379 |
|
sl@0
|
380 |
//Removes the already closed statements and compresses the array
|
sl@0
|
381 |
void RemoveDeadStatements(RSqlStatementArray& aStmtArray)
|
sl@0
|
382 |
{
|
sl@0
|
383 |
for(TInt i=aStmtArray.Count()-1;i>=0;--i)
|
sl@0
|
384 |
{
|
sl@0
|
385 |
if(!aStmtArray[i].iAlive)
|
sl@0
|
386 |
{
|
sl@0
|
387 |
aStmtArray.Remove(i);
|
sl@0
|
388 |
}
|
sl@0
|
389 |
}
|
sl@0
|
390 |
aStmtArray.Compress();
|
sl@0
|
391 |
}
|
sl@0
|
392 |
|
sl@0
|
393 |
//Close statement objects, statements array and the database object
|
sl@0
|
394 |
TInt PostTest(RSqlDatabase& aDb, RSqlStatementArray& aStmtArray)
|
sl@0
|
395 |
{
|
sl@0
|
396 |
TInt statementsAlive = AliveStatementsCount(aStmtArray);
|
sl@0
|
397 |
CloseAllStatements(aStmtArray);
|
sl@0
|
398 |
aStmtArray.Close();
|
sl@0
|
399 |
aDb.Close();
|
sl@0
|
400 |
return statementsAlive;
|
sl@0
|
401 |
}
|
sl@0
|
402 |
|
sl@0
|
403 |
//Test thread function
|
sl@0
|
404 |
//The thread function works with a set of TSqlStatement objects
|
sl@0
|
405 |
//The test consists of 4 steps:
|
sl@0
|
406 |
//Step 1: the test thread creates m TSqlStatement objects, 0 < m < KStatementCnt.
|
sl@0
|
407 |
// With each of the created TSqlStatement objects the test thread prepares SELECT SQL query
|
sl@0
|
408 |
// "SELECT * FROM A WHERE F1 >= K1 AND F1 < K2", where K1 is random generated number, such that:
|
sl@0
|
409 |
// 0 < K1 < (KRecordCnt - KQueriedRecordCnt)
|
sl@0
|
410 |
// K2 = K1 + KQueriedRecordCnt
|
sl@0
|
411 |
// All just created TSqlStatement objects are marked as alive.
|
sl@0
|
412 |
//Step 2: For each alive TSqlStatement object the test thread calls iObj.Next() method KTestStepCnt times,
|
sl@0
|
413 |
// KTestStepCnt < KQueriedRecordCnt.
|
sl@0
|
414 |
// The column values are retrieved and checked.
|
sl@0
|
415 |
//Step 3: the test thread closes n TSqlStatement objects, 0 < n < KStatementCnt.
|
sl@0
|
416 |
//Step 4: the test thread counts how many alive TSqlStatement objects are there.
|
sl@0
|
417 |
// If this count > KMaxStatementPerThread then the test thread closes all alive TSqlStatement objects
|
sl@0
|
418 |
// to avoid OOM errors during the test.
|
sl@0
|
419 |
//
|
sl@0
|
420 |
// Each test thread does steps 1..4 for a period of KTestDuration seconds.
|
sl@0
|
421 |
// At the end all TSqlStatement objects are closed.
|
sl@0
|
422 |
//
|
sl@0
|
423 |
// The idea of the test is to load the SQL server creating several amount of statement and stream objects
|
sl@0
|
424 |
// and see that it is working stable and without problems.
|
sl@0
|
425 |
TInt ThreadFunc(void*)
|
sl@0
|
426 |
{
|
sl@0
|
427 |
__UHEAP_MARK;
|
sl@0
|
428 |
|
sl@0
|
429 |
CTrapCleanup* tc = CTrapCleanup::New();
|
sl@0
|
430 |
TTEST(tc != NULL);
|
sl@0
|
431 |
|
sl@0
|
432 |
TInt64 seed = 0;
|
sl@0
|
433 |
RSqlDatabase db;
|
sl@0
|
434 |
TName threadName;
|
sl@0
|
435 |
RSqlStatementArray statements;
|
sl@0
|
436 |
|
sl@0
|
437 |
//Init the random numbers generator, opens the database
|
sl@0
|
438 |
PreTest(db, seed, threadName);
|
sl@0
|
439 |
|
sl@0
|
440 |
//Main test loop
|
sl@0
|
441 |
TInt iteration = 0;
|
sl@0
|
442 |
TTime currTime;
|
sl@0
|
443 |
currTime.UniversalTime();
|
sl@0
|
444 |
TTime endTime = currTime + TTimeIntervalSeconds(KTestDuration);
|
sl@0
|
445 |
while(currTime < endTime)
|
sl@0
|
446 |
{
|
sl@0
|
447 |
++iteration;
|
sl@0
|
448 |
///////////////////////////////////////////////////////////////////////
|
sl@0
|
449 |
TInt statementsAliveBegin = statements.Count();
|
sl@0
|
450 |
//Step 1: Create N statements, where 0 < N < KStatementCnt
|
sl@0
|
451 |
TInt statementsCreated = CreateStatements(db, seed, statements);
|
sl@0
|
452 |
///////////////////////////////////////////////////////////////////////
|
sl@0
|
453 |
//Step 2: For each alive statement object - do (TSqlStatement::iCount / KTestStepCnt)
|
sl@0
|
454 |
// RSqlStatement::Next() calls. If the Next() call reaches the end - close the statement object.
|
sl@0
|
455 |
TInt statementsCompleted = ProcessStatements(statements);
|
sl@0
|
456 |
///////////////////////////////////////////////////////////////////////
|
sl@0
|
457 |
//Step 3: Close up to N statements, where 0 < N < KStatementCnt
|
sl@0
|
458 |
TInt statementsClosed = CloseStatements(statements, seed);
|
sl@0
|
459 |
///////////////////////////////////////////////////////////////////////
|
sl@0
|
460 |
//Step 4: If the alive statement count is more than KMaxStatementPerThread, then close them all
|
sl@0
|
461 |
TInt statementsAliveEnd = AliveStatementsCount(statements);
|
sl@0
|
462 |
if(statementsAliveEnd > KMaxStatementPerThread)
|
sl@0
|
463 |
{
|
sl@0
|
464 |
RDebug::Print(_L("!!! Thread %S, iteration %d, alive %d, close all\r\n"), &threadName, iteration, statementsAliveEnd);
|
sl@0
|
465 |
CloseAllStatements(statements);
|
sl@0
|
466 |
statementsAliveEnd = 0;
|
sl@0
|
467 |
}
|
sl@0
|
468 |
///////////////////////////////////////////////////////////////////////
|
sl@0
|
469 |
RemoveDeadStatements(statements);
|
sl@0
|
470 |
RDebug::Print(_L("=== Thread %S, iteration % 4d, begin: % 3d, created % 2d, closed % 2d, completed % 2d, end % 3d, \r\n"),
|
sl@0
|
471 |
&threadName, iteration, statementsAliveBegin,
|
sl@0
|
472 |
statementsCreated, statementsClosed, statementsCompleted,
|
sl@0
|
473 |
statementsAliveEnd);
|
sl@0
|
474 |
currTime.UniversalTime();
|
sl@0
|
475 |
}
|
sl@0
|
476 |
|
sl@0
|
477 |
//Close statement objects and the database object
|
sl@0
|
478 |
TInt statementsAlive = PostTest(db, statements);
|
sl@0
|
479 |
|
sl@0
|
480 |
delete tc;
|
sl@0
|
481 |
|
sl@0
|
482 |
__UHEAP_MARKEND;
|
sl@0
|
483 |
|
sl@0
|
484 |
RDebug::Print(_L("=== Thread %S exit, still alive %d\r\n"), &threadName, statementsAlive);
|
sl@0
|
485 |
|
sl@0
|
486 |
return KErrNone;
|
sl@0
|
487 |
}
|
sl@0
|
488 |
|
sl@0
|
489 |
void CreateTestThreads(RThread aThreads[], TRequestStatus aStatuses[], TInt aMaxCount)
|
sl@0
|
490 |
{
|
sl@0
|
491 |
_LIT(KThreadName, "TstThr");
|
sl@0
|
492 |
for(TInt i=0;i<aMaxCount;++i)
|
sl@0
|
493 |
{
|
sl@0
|
494 |
TBuf<20> threadName(KThreadName);
|
sl@0
|
495 |
threadName.AppendNum((TInt64)(i + 1));
|
sl@0
|
496 |
TEST2(aThreads[i].Create(threadName, &ThreadFunc, 0x2000, 0x1000, 0x10000, NULL, EOwnerProcess), KErrNone);
|
sl@0
|
497 |
aThreads[i].Logon(aStatuses[i]);
|
sl@0
|
498 |
TEST2(aStatuses[i].Int(), KRequestPending);
|
sl@0
|
499 |
}
|
sl@0
|
500 |
}
|
sl@0
|
501 |
|
sl@0
|
502 |
void ResumeTestThreads(RThread aThreads[], TInt aMaxCount)
|
sl@0
|
503 |
{
|
sl@0
|
504 |
for(TInt i=0;i<aMaxCount;++i)
|
sl@0
|
505 |
{
|
sl@0
|
506 |
aThreads[i].Resume();
|
sl@0
|
507 |
}
|
sl@0
|
508 |
}
|
sl@0
|
509 |
|
sl@0
|
510 |
|
sl@0
|
511 |
void CloseTestThreads(RThread aThreads[], TRequestStatus aStatuses[], TInt aMaxCount)
|
sl@0
|
512 |
{
|
sl@0
|
513 |
for(TInt i=0;i<aMaxCount;++i)
|
sl@0
|
514 |
{
|
sl@0
|
515 |
User::WaitForRequest(aStatuses[i]);
|
sl@0
|
516 |
TEST(aThreads[i].ExitType() != EExitPanic);
|
sl@0
|
517 |
aThreads[i].Close();
|
sl@0
|
518 |
}
|
sl@0
|
519 |
}
|
sl@0
|
520 |
|
sl@0
|
521 |
/**
|
sl@0
|
522 |
@SYMTestCaseID SYSLIB-SQL-CT-1627-0001
|
sl@0
|
523 |
@SYMTestCaseDesc SQL server load test. The test creates KTestThreadCnt threads, KTestDbCnt test databases and
|
sl@0
|
524 |
inserts in each of them KRecordCnt test records.
|
sl@0
|
525 |
Pre-test step: each test thread randomly chooses and opens one of the test databases.
|
sl@0
|
526 |
Then, each of the test threads is doing the following 4 test steps:
|
sl@0
|
527 |
Step 1: the test thread creates m TSqlStatement objects, 0 < m < KStatementCnt.
|
sl@0
|
528 |
With each of the created TSqlStatement objects the test thread prepares SELECT SQL query
|
sl@0
|
529 |
"SELECT * FROM A WHERE F1 >= K1 AND F1 < K2", where K1 is random generated number, such that:
|
sl@0
|
530 |
0 < K1 < (KRecordCnt - KQueriedRecordCnt)
|
sl@0
|
531 |
K2 = K1 + KQueriedRecordCnt
|
sl@0
|
532 |
All just created TSqlStatement objects are marked as alive.
|
sl@0
|
533 |
Step 2: For each alive TSqlStatement object the test thread calls iObj.Next() method KTestStepCnt times,
|
sl@0
|
534 |
KTestStepCnt < KQueriedRecordCnt.
|
sl@0
|
535 |
The column values are retrieved and checked.
|
sl@0
|
536 |
Step 3: the test thread closes n TSqlStatement objects, 0 < n < KStatementCnt.
|
sl@0
|
537 |
Step 4: the test thread counts how many alive TSqlStatement objects are there.
|
sl@0
|
538 |
If this count > KMaxStatementPerThread then the test thread closes all alive TSqlStatement objects
|
sl@0
|
539 |
to avoid OOM errors during the test.
|
sl@0
|
540 |
|
sl@0
|
541 |
Each test thread does steps 1..4 for a period of KTestDuration seconds.
|
sl@0
|
542 |
At the end all TSqlStatement objects are closed.
|
sl@0
|
543 |
|
sl@0
|
544 |
The idea of the test is to load the SQL server creating several amount of statement and stream objects
|
sl@0
|
545 |
and see that it is working stable and without problems.
|
sl@0
|
546 |
@SYMTestPriority High
|
sl@0
|
547 |
@SYMTestActions SQL server load test
|
sl@0
|
548 |
@SYMTestExpectedResults Test must not fail
|
sl@0
|
549 |
@SYMREQ REQ5792
|
sl@0
|
550 |
REQ5793
|
sl@0
|
551 |
*/
|
sl@0
|
552 |
void SqlLoadTest()
|
sl@0
|
553 |
{
|
sl@0
|
554 |
CreateTestDatabases();
|
sl@0
|
555 |
|
sl@0
|
556 |
RThread threads[KTestThreadCnt];
|
sl@0
|
557 |
TRequestStatus statuses[KTestThreadCnt];
|
sl@0
|
558 |
|
sl@0
|
559 |
CreateTestThreads(threads, statuses, KTestThreadCnt);
|
sl@0
|
560 |
|
sl@0
|
561 |
ResumeTestThreads(threads, KTestThreadCnt);
|
sl@0
|
562 |
|
sl@0
|
563 |
User::After(2000000);
|
sl@0
|
564 |
|
sl@0
|
565 |
CloseTestThreads(threads, statuses, KTestThreadCnt);
|
sl@0
|
566 |
}
|
sl@0
|
567 |
|
sl@0
|
568 |
/**
|
sl@0
|
569 |
@SYMTestCaseID PDS-SQL-CT-4201
|
sl@0
|
570 |
@SYMTestCaseDesc Max number of SQL statements test.
|
sl@0
|
571 |
@SYMTestPriority High
|
sl@0
|
572 |
@SYMTestActions The test creates a table with couple of records and then
|
sl@0
|
573 |
creates as many as possible SQL statements. The expected result is
|
sl@0
|
574 |
that either the statement creation process will fail with KErrNoMemory or
|
sl@0
|
575 |
the max number of statements to be created is reached (100000).
|
sl@0
|
576 |
Then the test deletes 1/2 of the created statements objects and
|
sl@0
|
577 |
after that attempts to execute Next() on the rest of them.
|
sl@0
|
578 |
Note that the test has a time limit of 120 seconds. Otherwise on some platforms
|
sl@0
|
579 |
with WDP feature switched on the test may timeout.
|
sl@0
|
580 |
@SYMTestExpectedResults Test must not fail
|
sl@0
|
581 |
@SYMDEF DEF145236
|
sl@0
|
582 |
*/
|
sl@0
|
583 |
void StatementMaxNumberTest()
|
sl@0
|
584 |
{
|
sl@0
|
585 |
TBuf<30> time;
|
sl@0
|
586 |
GetHomeTimeAsString(time);
|
sl@0
|
587 |
TheTest.Printf(_L("=== %S: Create database\r\n"), &time);
|
sl@0
|
588 |
|
sl@0
|
589 |
(void)RSqlDatabase::Delete(KTestDbName1);
|
sl@0
|
590 |
RSqlDatabase db;
|
sl@0
|
591 |
TInt err = db.Create(KTestDbName1);
|
sl@0
|
592 |
TEST2(err, KErrNone);
|
sl@0
|
593 |
err = db.Exec(_L("CREATE TABLE A(I INTEGER); INSERT INTO A(I) VALUES(1); INSERT INTO A(I) VALUES(2);"));
|
sl@0
|
594 |
TEST(err >= 0);
|
sl@0
|
595 |
|
sl@0
|
596 |
GetHomeTimeAsString(time);
|
sl@0
|
597 |
TheTest.Printf(_L("=== %S: Create statements array\r\n"), &time);
|
sl@0
|
598 |
|
sl@0
|
599 |
//Reserve memory for the statement objects
|
sl@0
|
600 |
const TInt KMaxStmtCount = 100000;
|
sl@0
|
601 |
RSqlStatement* stmt = new RSqlStatement[KMaxStmtCount];
|
sl@0
|
602 |
TEST(stmt != NULL);
|
sl@0
|
603 |
|
sl@0
|
604 |
TTime startTime;
|
sl@0
|
605 |
startTime.HomeTime();
|
sl@0
|
606 |
|
sl@0
|
607 |
//Create as many statement objects as possible
|
sl@0
|
608 |
TInt idx = 0;
|
sl@0
|
609 |
err = KErrNone;
|
sl@0
|
610 |
for(;idx<KMaxStmtCount;++idx)
|
sl@0
|
611 |
{
|
sl@0
|
612 |
err = stmt[idx].Prepare(db, _L("SELECT * FROM A WHERE I>=0 AND I<10"));
|
sl@0
|
613 |
if(err != KErrNone)
|
sl@0
|
614 |
{
|
sl@0
|
615 |
break;
|
sl@0
|
616 |
}
|
sl@0
|
617 |
TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
|
sl@0
|
618 |
if((idx % 100) == 0)
|
sl@0
|
619 |
{
|
sl@0
|
620 |
GetHomeTimeAsString(time);
|
sl@0
|
621 |
TheTest.Printf(_L("=== %S: Create % 5d statements. %d seconds.\r\n"), &time, idx + 1, s.Int());
|
sl@0
|
622 |
}
|
sl@0
|
623 |
if(s.Int() > KTestTimeLimit)
|
sl@0
|
624 |
{
|
sl@0
|
625 |
GetHomeTimeAsString(time);
|
sl@0
|
626 |
TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
|
sl@0
|
627 |
++idx;//The idx-th statement is valid, the statement count is idx + 1.
|
sl@0
|
628 |
break;
|
sl@0
|
629 |
}
|
sl@0
|
630 |
}
|
sl@0
|
631 |
|
sl@0
|
632 |
TInt stmtCnt = idx;
|
sl@0
|
633 |
TheTest.Printf(_L("%d created statement objects. Last error: %d.\r\n"), stmtCnt, err);
|
sl@0
|
634 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
635 |
|
sl@0
|
636 |
//Close 1/2 of the statements to free some memory
|
sl@0
|
637 |
idx = 0;
|
sl@0
|
638 |
for(;idx<(stmtCnt/2);++idx)
|
sl@0
|
639 |
{
|
sl@0
|
640 |
stmt[idx].Close();
|
sl@0
|
641 |
if((idx % 100) == 0)
|
sl@0
|
642 |
{
|
sl@0
|
643 |
GetHomeTimeAsString(time);
|
sl@0
|
644 |
TheTest.Printf(_L("=== %S: % 5d statements closed\r\n"), &time, idx + 1);
|
sl@0
|
645 |
}
|
sl@0
|
646 |
}
|
sl@0
|
647 |
|
sl@0
|
648 |
//Now, there should be enough memory to be able to execute Next() on the rest of the statements
|
sl@0
|
649 |
for(TInt j=0;idx<stmtCnt;++idx,++j)
|
sl@0
|
650 |
{
|
sl@0
|
651 |
err = stmt[idx].Next();
|
sl@0
|
652 |
TEST2(err, KSqlAtRow);
|
sl@0
|
653 |
err = stmt[idx].Next();
|
sl@0
|
654 |
TEST2(err, KSqlAtRow);
|
sl@0
|
655 |
err = stmt[idx].Next();
|
sl@0
|
656 |
TEST2(err, KSqlAtEnd);
|
sl@0
|
657 |
GetHomeTimeAsString(time);
|
sl@0
|
658 |
TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
|
sl@0
|
659 |
if((j % 100) == 0)
|
sl@0
|
660 |
{
|
sl@0
|
661 |
TheTest.Printf(_L("=== %S: % 5d statements processed. %d seconds.\r\n"), &time, j + 1, s.Int());
|
sl@0
|
662 |
}
|
sl@0
|
663 |
if(s.Int() > KTestTimeLimit)
|
sl@0
|
664 |
{
|
sl@0
|
665 |
TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
|
sl@0
|
666 |
break;
|
sl@0
|
667 |
}
|
sl@0
|
668 |
}
|
sl@0
|
669 |
|
sl@0
|
670 |
//Cleanup
|
sl@0
|
671 |
for(idx=0;idx<stmtCnt;++idx)
|
sl@0
|
672 |
{
|
sl@0
|
673 |
stmt[idx].Close();
|
sl@0
|
674 |
if((idx % 100) == 0)
|
sl@0
|
675 |
{
|
sl@0
|
676 |
GetHomeTimeAsString(time);
|
sl@0
|
677 |
TheTest.Printf(_L("=== %S: % 5d statements closed\r\n"), &time, idx + 1);
|
sl@0
|
678 |
}
|
sl@0
|
679 |
}
|
sl@0
|
680 |
delete [] stmt;
|
sl@0
|
681 |
db.Close();
|
sl@0
|
682 |
(void)RSqlDatabase::Delete(KTestDbName1);
|
sl@0
|
683 |
GetHomeTimeAsString(time);
|
sl@0
|
684 |
TheTest.Printf(_L("=== %S: Test case end\r\n"), &time);
|
sl@0
|
685 |
}
|
sl@0
|
686 |
|
sl@0
|
687 |
TInt CreateFileSessions(TInt& aIdx, RFs aFs[], TInt aMaxFsSessCount)
|
sl@0
|
688 |
{
|
sl@0
|
689 |
TBuf<30> time;
|
sl@0
|
690 |
TTime startTime;
|
sl@0
|
691 |
startTime.HomeTime();
|
sl@0
|
692 |
//Create as many file session objects as possible
|
sl@0
|
693 |
TInt err = KErrNone;
|
sl@0
|
694 |
for(;aIdx<aMaxFsSessCount;++aIdx)
|
sl@0
|
695 |
{
|
sl@0
|
696 |
err = aFs[aIdx].Connect();
|
sl@0
|
697 |
if(err != KErrNone)
|
sl@0
|
698 |
{
|
sl@0
|
699 |
break;
|
sl@0
|
700 |
}
|
sl@0
|
701 |
TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
|
sl@0
|
702 |
if((aIdx % 500) == 0)
|
sl@0
|
703 |
{
|
sl@0
|
704 |
GetHomeTimeAsString(time);
|
sl@0
|
705 |
TheTest.Printf(_L("=== %S: Create % 5d file sessions. %d seconds.\r\n"), &time, aIdx + 1, s.Int());
|
sl@0
|
706 |
}
|
sl@0
|
707 |
if(s.Int() > KTestTimeLimit)
|
sl@0
|
708 |
{
|
sl@0
|
709 |
GetHomeTimeAsString(time);
|
sl@0
|
710 |
TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
|
sl@0
|
711 |
++aIdx;//The idx-th file session object is valid, the file session count is idx + 1.
|
sl@0
|
712 |
break;
|
sl@0
|
713 |
}
|
sl@0
|
714 |
}
|
sl@0
|
715 |
return err;
|
sl@0
|
716 |
}
|
sl@0
|
717 |
|
sl@0
|
718 |
/**
|
sl@0
|
719 |
@SYMTestCaseID PDS-SQL-CT-4237
|
sl@0
|
720 |
@SYMTestCaseDesc Max file session number test.
|
sl@0
|
721 |
@SYMTestPriority High
|
sl@0
|
722 |
@SYMTestActions The test creates as many as possible file session objects. The expected result is
|
sl@0
|
723 |
that either the file session creation process will fail with KErrNoMemory or
|
sl@0
|
724 |
the max number of file sessions to be created is reached (100000).
|
sl@0
|
725 |
Then the test attempts to create a database. If there is no memory, the test
|
sl@0
|
726 |
closes some of the file session objects. The test also attempts to copy
|
sl@0
|
727 |
the created database and to delete it after that, both operations performed
|
sl@0
|
728 |
with all file session objects still open. The expectation is that the test
|
sl@0
|
729 |
will not crash the SQL server or the client side SQL dll.
|
sl@0
|
730 |
Note that the test has a time limit of 120 seconds. Otherwise on some platforms
|
sl@0
|
731 |
with WDP feature switched on the test may timeout.
|
sl@0
|
732 |
@SYMTestExpectedResults Test must not fail
|
sl@0
|
733 |
*/
|
sl@0
|
734 |
void FileSessionMaxNumberTest()
|
sl@0
|
735 |
{
|
sl@0
|
736 |
TBuf<30> time;
|
sl@0
|
737 |
GetHomeTimeAsString(time);
|
sl@0
|
738 |
TheTest.Printf(_L("=== %S: Create file sessions\r\n"), &time);
|
sl@0
|
739 |
|
sl@0
|
740 |
const TInt KMaxFsCount = 100000;
|
sl@0
|
741 |
RFs* fs = new RFs[KMaxFsCount];
|
sl@0
|
742 |
TEST(fs != NULL);
|
sl@0
|
743 |
|
sl@0
|
744 |
//Create as many file session objects as possible
|
sl@0
|
745 |
TInt idx = 0;
|
sl@0
|
746 |
TInt err = CreateFileSessions(idx, fs, KMaxFsCount);
|
sl@0
|
747 |
TheTest.Printf(_L("%d created file session objects. Last error: %d.\r\n"), idx, err);
|
sl@0
|
748 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
749 |
|
sl@0
|
750 |
TBool dbCreated = EFalse;
|
sl@0
|
751 |
RSqlDatabase db;
|
sl@0
|
752 |
|
sl@0
|
753 |
//An attempt to create a database
|
sl@0
|
754 |
while(idx > 0 && err == KErrNoMemory)
|
sl@0
|
755 |
{
|
sl@0
|
756 |
(void)RSqlDatabase::Delete(KTestDbName1);
|
sl@0
|
757 |
err = db.Create(KTestDbName1);
|
sl@0
|
758 |
if(err == KErrNone)
|
sl@0
|
759 |
{
|
sl@0
|
760 |
err = db.Exec(_L("CREATE TABLE A(I INTEGER); INSERT INTO A(I) VALUES(1); INSERT INTO A(I) VALUES(2);"));
|
sl@0
|
761 |
}
|
sl@0
|
762 |
TheTest.Printf(_L("Database creation. Last error: %d.\r\n"), err);
|
sl@0
|
763 |
TEST(err == KErrNoMemory || err >= 0);
|
sl@0
|
764 |
if(err == KErrNoMemory)
|
sl@0
|
765 |
{
|
sl@0
|
766 |
fs[--idx].Close();
|
sl@0
|
767 |
}
|
sl@0
|
768 |
else
|
sl@0
|
769 |
{
|
sl@0
|
770 |
dbCreated = ETrue;
|
sl@0
|
771 |
}
|
sl@0
|
772 |
db.Close();
|
sl@0
|
773 |
}
|
sl@0
|
774 |
|
sl@0
|
775 |
if(dbCreated)
|
sl@0
|
776 |
{
|
sl@0
|
777 |
//Create again file session objects - as many as possible
|
sl@0
|
778 |
err = CreateFileSessions(idx, fs, KMaxFsCount);
|
sl@0
|
779 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
780 |
//Try to copy the database
|
sl@0
|
781 |
err = RSqlDatabase::Copy(KTestDbName1, KTestDbName4);
|
sl@0
|
782 |
TheTest.Printf(_L("Copy database. Last error: %d.\r\n"), err);
|
sl@0
|
783 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
784 |
//Try to delete the databases
|
sl@0
|
785 |
if(err == KErrNone)
|
sl@0
|
786 |
{
|
sl@0
|
787 |
err = RSqlDatabase::Delete(KTestDbName4);
|
sl@0
|
788 |
TheTest.Printf(_L("Delete database copy. Last error: %d.\r\n"), err);
|
sl@0
|
789 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
790 |
}
|
sl@0
|
791 |
err = RSqlDatabase::Delete(KTestDbName1);
|
sl@0
|
792 |
TheTest.Printf(_L("Delete database. Last error: %d.\r\n"), err);
|
sl@0
|
793 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
794 |
}
|
sl@0
|
795 |
|
sl@0
|
796 |
//Cleanup
|
sl@0
|
797 |
for(TInt i=0;i<idx;++i)
|
sl@0
|
798 |
{
|
sl@0
|
799 |
fs[i].Close();
|
sl@0
|
800 |
if((i % 500) == 0)
|
sl@0
|
801 |
{
|
sl@0
|
802 |
GetHomeTimeAsString(time);
|
sl@0
|
803 |
TheTest.Printf(_L("=== %S: % 5d file sessions closed\r\n"), &time, i + 1);
|
sl@0
|
804 |
}
|
sl@0
|
805 |
}
|
sl@0
|
806 |
delete [] fs;
|
sl@0
|
807 |
err = RSqlDatabase::Delete(KTestDbName4);
|
sl@0
|
808 |
TEST(err == KErrNone || err == KErrNotFound);
|
sl@0
|
809 |
err = RSqlDatabase::Delete(KTestDbName1);
|
sl@0
|
810 |
TEST(err == KErrNone || err == KErrNotFound);
|
sl@0
|
811 |
}
|
sl@0
|
812 |
|
sl@0
|
813 |
TInt CreateSqlConnections(TInt& aIdx, RSqlDatabase aDb[], TInt aMaxSqlConnCount)
|
sl@0
|
814 |
{
|
sl@0
|
815 |
TBuf<30> time;
|
sl@0
|
816 |
TTime startTime;
|
sl@0
|
817 |
startTime.HomeTime();
|
sl@0
|
818 |
//Create as many file session objects as possible
|
sl@0
|
819 |
TInt err = KErrNone;
|
sl@0
|
820 |
for(;aIdx<aMaxSqlConnCount;++aIdx)
|
sl@0
|
821 |
{
|
sl@0
|
822 |
err = aDb[aIdx].Open(KTestDbName1);
|
sl@0
|
823 |
if(err != KErrNone)
|
sl@0
|
824 |
{
|
sl@0
|
825 |
break;
|
sl@0
|
826 |
}
|
sl@0
|
827 |
TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
|
sl@0
|
828 |
if((aIdx % 100) == 0)
|
sl@0
|
829 |
{
|
sl@0
|
830 |
GetHomeTimeAsString(time);
|
sl@0
|
831 |
TheTest.Printf(_L("=== %S: Create % 5d sql connections. %d seconds.\r\n"), &time, aIdx + 1, s.Int());
|
sl@0
|
832 |
}
|
sl@0
|
833 |
if(s.Int() > KTestTimeLimit)
|
sl@0
|
834 |
{
|
sl@0
|
835 |
GetHomeTimeAsString(time);
|
sl@0
|
836 |
TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
|
sl@0
|
837 |
++aIdx;//The idx-th sql connection is valid, the sql connection count is idx + 1.
|
sl@0
|
838 |
break;
|
sl@0
|
839 |
}
|
sl@0
|
840 |
}
|
sl@0
|
841 |
return err;
|
sl@0
|
842 |
}
|
sl@0
|
843 |
|
sl@0
|
844 |
/**
|
sl@0
|
845 |
@SYMTestCaseID PDS-SQL-CT-4238
|
sl@0
|
846 |
@SYMTestCaseDesc Max sql connection number test.
|
sl@0
|
847 |
@SYMTestPriority High
|
sl@0
|
848 |
@SYMTestActions The test creates as many as possible sql connection objects. The expected result is
|
sl@0
|
849 |
that either the sql connection creation process will fail with KErrNoMemory or
|
sl@0
|
850 |
the max number of sql connection to be created is reached (100000).
|
sl@0
|
851 |
Then the test attempts to create a database. If there is no memory, the test
|
sl@0
|
852 |
closes some of the sql connection objects. The test also attempts to copy
|
sl@0
|
853 |
the created database and to delete it after that, both operations performed
|
sl@0
|
854 |
with all sql connection objects still open. The expectation is that the test
|
sl@0
|
855 |
will not crash the SQL server or the client side SQL dll.
|
sl@0
|
856 |
Note that the test has a time limit of 120 seconds. Otherwise on some platforms
|
sl@0
|
857 |
with WDP feature switched on the test may timeout.
|
sl@0
|
858 |
@SYMTestExpectedResults Test must not fail
|
sl@0
|
859 |
*/
|
sl@0
|
860 |
void SqlConnectionMaxNumberTest()
|
sl@0
|
861 |
{
|
sl@0
|
862 |
TBuf<30> time;
|
sl@0
|
863 |
GetHomeTimeAsString(time);
|
sl@0
|
864 |
TheTest.Printf(_L("=== %S: Create sql connections\r\n"), &time);
|
sl@0
|
865 |
|
sl@0
|
866 |
(void)RSqlDatabase::Delete(KTestDbName1);
|
sl@0
|
867 |
RSqlDatabase db1;
|
sl@0
|
868 |
TInt err = db1.Create(KTestDbName1);//CreateSqlConnections() opens the already existing KTestDbName1 database
|
sl@0
|
869 |
TEST2(err, KErrNone);
|
sl@0
|
870 |
|
sl@0
|
871 |
const TInt KMaxConnCount = 100000;
|
sl@0
|
872 |
RSqlDatabase* db = new RSqlDatabase[KMaxConnCount];
|
sl@0
|
873 |
TEST(db != NULL);
|
sl@0
|
874 |
|
sl@0
|
875 |
//Create as many sql connection objects as possible
|
sl@0
|
876 |
TInt idx = 0;
|
sl@0
|
877 |
err = CreateSqlConnections(idx, db, KMaxConnCount);
|
sl@0
|
878 |
TheTest.Printf(_L("%d created sql connection objects. Last error: %d.\r\n"), idx, err);
|
sl@0
|
879 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
880 |
|
sl@0
|
881 |
TBool dbCreated = EFalse;
|
sl@0
|
882 |
RSqlDatabase db2;
|
sl@0
|
883 |
|
sl@0
|
884 |
//An attempt to create a database
|
sl@0
|
885 |
while(idx > 0 && err == KErrNoMemory)
|
sl@0
|
886 |
{
|
sl@0
|
887 |
(void)RSqlDatabase::Delete(KTestDbName4);
|
sl@0
|
888 |
err = db2.Create(KTestDbName4);
|
sl@0
|
889 |
if(err == KErrNone)
|
sl@0
|
890 |
{
|
sl@0
|
891 |
err = db2.Exec(_L("CREATE TABLE A(I INTEGER); INSERT INTO A(I) VALUES(1); INSERT INTO A(I) VALUES(2);"));
|
sl@0
|
892 |
}
|
sl@0
|
893 |
TheTest.Printf(_L("Database creation. Last error: %d.\r\n"), err);
|
sl@0
|
894 |
TEST(err == KErrNoMemory || err >= 0);
|
sl@0
|
895 |
if(err == KErrNoMemory)
|
sl@0
|
896 |
{
|
sl@0
|
897 |
db[--idx].Close();
|
sl@0
|
898 |
}
|
sl@0
|
899 |
else
|
sl@0
|
900 |
{
|
sl@0
|
901 |
dbCreated = ETrue;
|
sl@0
|
902 |
}
|
sl@0
|
903 |
db2.Close();
|
sl@0
|
904 |
}
|
sl@0
|
905 |
|
sl@0
|
906 |
if(dbCreated)
|
sl@0
|
907 |
{
|
sl@0
|
908 |
//Create again sql connection objects - as many as possible
|
sl@0
|
909 |
err = CreateSqlConnections(idx, db, KMaxConnCount);
|
sl@0
|
910 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
911 |
//Try to copy the database
|
sl@0
|
912 |
err = RSqlDatabase::Copy(KTestDbName4, KTestDbName5);
|
sl@0
|
913 |
TheTest.Printf(_L("Copy database. Last error: %d.\r\n"), err);
|
sl@0
|
914 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
915 |
//Try to delete the databases
|
sl@0
|
916 |
if(err == KErrNone)
|
sl@0
|
917 |
{
|
sl@0
|
918 |
err = RSqlDatabase::Delete(KTestDbName5);
|
sl@0
|
919 |
TheTest.Printf(_L("Delete database copy. Last error: %d.\r\n"), err);
|
sl@0
|
920 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
921 |
}
|
sl@0
|
922 |
err = RSqlDatabase::Delete(KTestDbName4);
|
sl@0
|
923 |
TheTest.Printf(_L("Delete database. Last error: %d.\r\n"), err);
|
sl@0
|
924 |
TEST(err == KErrNone || err == KErrNoMemory);
|
sl@0
|
925 |
}
|
sl@0
|
926 |
|
sl@0
|
927 |
//Cleanup
|
sl@0
|
928 |
for(TInt i=0;i<idx;++i)
|
sl@0
|
929 |
{
|
sl@0
|
930 |
db[i].Close();
|
sl@0
|
931 |
if((i % 100) == 0)
|
sl@0
|
932 |
{
|
sl@0
|
933 |
GetHomeTimeAsString(time);
|
sl@0
|
934 |
TheTest.Printf(_L("=== %S: % 5d sql connections closed\r\n"), &time, i + 1);
|
sl@0
|
935 |
}
|
sl@0
|
936 |
}
|
sl@0
|
937 |
delete [] db;
|
sl@0
|
938 |
db1.Close();
|
sl@0
|
939 |
err = RSqlDatabase::Delete(KTestDbName5);
|
sl@0
|
940 |
TEST(err == KErrNone || err == KErrNotFound);
|
sl@0
|
941 |
err = RSqlDatabase::Delete(KTestDbName4);
|
sl@0
|
942 |
TEST(err == KErrNone || err == KErrNotFound);
|
sl@0
|
943 |
err = RSqlDatabase::Delete(KTestDbName1);
|
sl@0
|
944 |
TEST(err == KErrNone || err == KErrNotFound);
|
sl@0
|
945 |
}
|
sl@0
|
946 |
|
sl@0
|
947 |
void DoTests()
|
sl@0
|
948 |
{
|
sl@0
|
949 |
TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1627-0001 SQL server load test "));
|
sl@0
|
950 |
SqlLoadTest();
|
sl@0
|
951 |
TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4201 Statement max number test"));
|
sl@0
|
952 |
StatementMaxNumberTest();
|
sl@0
|
953 |
#if defined __WINS__ || defined __WINSCW__
|
sl@0
|
954 |
//The next two tests are likely to timeout on hardware because they create a lot of file sessions and sql connections.
|
sl@0
|
955 |
//The SQL server heap is 32Mb on hardware but only 6Mb on the Emulator.
|
sl@0
|
956 |
TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4237 File session max number test"));
|
sl@0
|
957 |
FileSessionMaxNumberTest();
|
sl@0
|
958 |
TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4238 Sql connection max number test"));
|
sl@0
|
959 |
SqlConnectionMaxNumberTest();
|
sl@0
|
960 |
#endif
|
sl@0
|
961 |
}
|
sl@0
|
962 |
|
sl@0
|
963 |
TInt E32Main()
|
sl@0
|
964 |
{
|
sl@0
|
965 |
TheTest.Title();
|
sl@0
|
966 |
|
sl@0
|
967 |
CTrapCleanup* tc = CTrapCleanup::New();
|
sl@0
|
968 |
|
sl@0
|
969 |
__UHEAP_MARK;
|
sl@0
|
970 |
|
sl@0
|
971 |
CreateTestDir();
|
sl@0
|
972 |
DeleteTestFiles();
|
sl@0
|
973 |
DoTests();
|
sl@0
|
974 |
DeleteTestFiles();
|
sl@0
|
975 |
|
sl@0
|
976 |
__UHEAP_MARKEND;
|
sl@0
|
977 |
|
sl@0
|
978 |
TheTest.End();
|
sl@0
|
979 |
TheTest.Close();
|
sl@0
|
980 |
|
sl@0
|
981 |
delete tc;
|
sl@0
|
982 |
|
sl@0
|
983 |
User::Heap().Check();
|
sl@0
|
984 |
return KErrNone;
|
sl@0
|
985 |
}
|