persistentstorage/sql/TEST/t_sqlood.cpp
changeset 0 08ec8eefde2f
child 8 fa9941cf3867
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/persistentstorage/sql/TEST/t_sqlood.cpp	Fri Jan 22 11:06:30 2010 +0200
@@ -0,0 +1,486 @@
+// Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
+// All rights reserved.
+// This component and the accompanying materials are made available
+// under the terms of "Eclipse Public License v1.0"
+// which accompanies this distribution, and is available
+// at the URL "http://www.eclipse.org/legal/epl-v10.html".
+//
+// Initial Contributors:
+// Nokia Corporation - initial contribution.
+//
+// Contributors:
+//
+// Description:
+//
+
+#include <e32test.h>
+#include <bautils.h>
+#include <sqldb.h>
+
+///////////////////////////////////////////////////////////////////////////////////////
+
+static RFs TheFs;
+RTest TheTest(_L("t_sqlood test"));
+
+#if  defined __WINSCW__ || defined __WINS__
+
+	//The C: drive may be too big and may be used concurently by other applications. 
+	//The T: drive is more suitable for the test if running on the emulator
+	const TInt KTestDrive = EDriveT;
+	_LIT(KTestDir, "t:\\test\\");
+	_LIT(KTestDatabase, "t:\\test\\t_sql_ood.db");
+	
+#elif defined __X86GCC__
+
+	const TInt KTestDrive = EDriveG;
+	_LIT(KTestDir, "g:\\test\\");
+	_LIT(KTestDatabase, "g:\\test\\t_sql_ood.db");
+	
+#else
+
+	const TInt KTestDrive = EDriveE;
+	_LIT(KTestDir, "e:\\test\\");
+	_LIT(KTestDatabase, "e:\\test\\t_sql_ood.db");
+	
+#endif
+
+//One or more files with KLargeFileName name and ".<n>" extension, where n is 
+//000, 001, 002, 003...
+//will be created and they will occupy almost all available disk space.
+//The idea is to perform after that one "delete"
+//transaction, which must to fail, because there won't be enough available disk space to complete the transaction.
+#if  defined __WINSCW__ || defined __WINS__
+
+	_LIT(KLargeFileName, "t:\\test\\DeleteMe");
+
+#elif defined __X86GCC__
+
+	_LIT(KLargeFileName, "g:\\test\\DeleteMe");
+
+#else
+
+	_LIT(KLargeFileName, "e:\\test\\DeleteMe");
+
+#endif
+
+const TInt KTestRecordsCount = 350;
+
+///////////////////////////////////////////////////////////////////////////////////////
+
+//Assemblesd a file name from "aFileName" and "aFileNumber" parameters and places the resulting string in "aResultPath".
+void AssembleLargeFileName(const TDesC& aFileName, TInt aFileNumber, TDes& aResultPath)
+	{
+	_LIT(KFormatStr, "%S.%03d");
+	aResultPath.Format(KFormatStr, &aFileName, aFileNumber);
+	}
+
+//Deletes all created large data files.
+void DeleteLargeDataFiles()
+	{
+	TInt err = KErrNone;
+	TInt i = -1;
+	while(err == KErrNone)
+		{
+		TBuf<KMaxFileName> filePath;
+		::AssembleLargeFileName(KLargeFileName, ++i, filePath);
+		err = TheFs.Delete(filePath);
+		}
+	}
+
+//Deletes all created test files.
+void DeleteTestFiles()
+	{
+	DeleteLargeDataFiles();
+	(void)RSqlDatabase::Delete(KTestDatabase);
+	}
+
+///////////////////////////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////////////////////////
+//Test macros and functions
+void Check1(TInt aValue, TInt aLine)
+	{
+	if(!aValue)
+		{
+		DeleteTestFiles();
+		RDebug::Print(_L("*** Line %d\r\n"), aLine);
+		TheTest(EFalse, aLine);
+		}
+	}
+void Check2(TInt aValue, TInt aExpected, TInt aLine)
+	{
+	if(aValue != aExpected)
+		{
+		DeleteTestFiles();
+		RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
+		TheTest(EFalse, aLine);
+		}
+	}
+#define TEST(arg) ::Check1((arg), __LINE__)
+#define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
+
+///////////////////////////////////////////////////////////////////////////////////////
+
+//Creates file session instance and the test directory
+void CreateTestEnv()
+    {
+	TInt err = TheFs.Connect();
+	TEST2(err, KErrNone);
+
+	err = TheFs.MkDir(KTestDir);
+	TEST(err == KErrNone || err == KErrAlreadyExists);
+	}
+
+//Creates one or more large files with the total size near to the size of the available disk space.
+//The idea is to cause an "out of disk space" condition.
+void FillLargeDataFile(RFile& aFile, TInt aSize)
+	{
+	TInt err = KErrDiskFull;
+	while(err == KErrDiskFull)
+		{
+		err = aFile.SetSize(aSize);
+		aSize -= 100;
+		if(aSize <= 0)
+			{
+			break;
+			}
+		}
+	TEST(err == KErrNone || err == KErrDiskFull);
+	}
+
+//Gets and returns the available disk space of the tested drive.
+TInt64 FreeDiskSpace()
+	{
+	TVolumeInfo volInfoBefore;
+	TInt err = TheFs.Volume(volInfoBefore, KTestDrive);
+	TEST2(err, KErrNone);
+	return volInfoBefore.iFree;
+	}
+
+//Creates a large data file with aSize size (in bytes).
+void DoCreateLargeFile(const TDesC& aPath, TInt aSize)
+	{
+	RFile file;
+	TInt err = file.Replace(TheFs, aPath, EFileRead | EFileWrite);
+	TEST2(err, KErrNone);
+	FillLargeDataFile(file, aSize);
+	err = file.Flush();
+	TEST2(err, KErrNone);
+	file.Close();
+	}
+
+//Creates enough number of large data files to fill the available disk space.
+void CreateLargeFile()
+	{
+	TInt fileNo = 0;
+	const TInt KLargeFileSize = 1000000000;
+	TInt64 diskSpace = ::FreeDiskSpace();
+	RDebug::Print(_L("CreateLargeFile: free space before = %ld\r\n"), diskSpace);
+	TBuf<KMaxFileName> filePath;
+	while(diskSpace > KLargeFileSize)
+		{
+		AssembleLargeFileName(KLargeFileName, fileNo++, filePath);
+		DoCreateLargeFile(filePath, KLargeFileSize);
+		diskSpace = ::FreeDiskSpace();
+		RDebug::Print(_L("----CreateLargeFile, step %d, free space = %ld\r\n"), fileNo, diskSpace);
+		}
+	//Reserve almost all disk space, except a small amount - 200 bytes.
+	if(diskSpace > 0)
+		{
+		::AssembleLargeFileName(KLargeFileName, fileNo++, filePath);
+		const TInt64 KSpaceLeft = 200;
+		TInt64 lastFileSize = diskSpace - KSpaceLeft;
+        TInt lastFileSize32 = I64LOW(lastFileSize);
+		RDebug::Print(_L("----file size32 = %d\r\n"), lastFileSize32);
+		::DoCreateLargeFile(filePath, lastFileSize32);
+		RDebug::Print(_L("----CreateLargeFile, last step (%d), file size = %ld\r\n"), fileNo, lastFileSize);
+		}
+	diskSpace = ::FreeDiskSpace();
+	RDebug::Print(_L("CreateLargeFile: free space after = %ld\r\n"), diskSpace);
+	}
+
+//Creates and fills with some records a test database
+void CreateAndFillTestDatabase(RSqlDatabase& aDb)
+	{
+	TInt err = aDb.Create(KTestDatabase);
+	TEST2(err, KErrNone);
+	err = aDb.Exec(_L("CREATE TABLE A(Id INTEGER, Data TEXT)"));
+	TEST(err >= 0);
+	err = aDb.Exec(_L("BEGIN TRANSACTION"));
+	TEST(err >= 0);
+	for(TInt i=0;i<KTestRecordsCount;++i)
+		{
+		TBuf<200> sql;
+		sql.Format(_L("INSERT INTO A(Id, Data) VALUES(%d, 'A0123456789B0123456789C0123456789D0123456789E0123456789F0123456789G0123456789H0123456789')"), i + 1);
+		err = aDb.Exec(sql);
+		TEST2(err, 1);
+		}
+	err = aDb.Exec(_L("COMMIT TRANSACTION"));
+	TEST(err >= 0);
+	}
+
+//Tries to delete test database records
+TInt DeleteTestRecords(RSqlDatabase& aDb)
+	{
+	TInt err = aDb.Exec(_L("BEGIN TRANSACTION"));
+	TEST(err >= 0);
+	for(TInt i=0;i<KTestRecordsCount;++i)
+		{
+		TBuf<100> sql;
+		sql.Format(_L("DELETE FROM A WHERE Id = %d"), i + 1);
+		err = aDb.Exec(sql);// May fail with KErrDiskFull
+		if(err < 0)
+			{
+			(void)aDb.Exec(_L("ROLLBACK TRANSACTION"));
+			return err;
+			}
+		}
+	err = aDb.Exec(_L("COMMIT TRANSACTION"));// May fail with KErrDiskFull
+	return err;
+	}
+
+///////////////////////////////////////////////////////////////////////////////////////
+
+//The function simply calls RSqlDatabase::ReserveDriveSpace(), RSqlDatabase::GetReserveAccess(),
+//RSqlDatabase::ReleaseReserveAccess() methods and checks the return values.
+//It might be usefull for debugging in case if something gets wrong.
+void SimpleCallsTest()
+	{
+	RSqlDatabase db, db2;
+	TInt err = db.Create(KTestDatabase);
+	TEST2(err, KErrNone);
+
+	err = db2.Open(KTestDatabase);
+	TEST2(err, KErrNone);
+	
+	//An attempt to get an access to the reserved space (which is not reserved yet).
+	err = db.GetReserveAccess();
+	TEST2(err, KErrNotFound);
+
+	//Reserve disk space
+	err = db.ReserveDriveSpace(0);
+	TEST2(err, KErrNone);
+
+	//An attempt to re-reserve it
+   	err = db.ReserveDriveSpace(0);
+	TEST2(err, KErrAlreadyExists);
+
+	//Get an access to the reserved disk space
+	err = db.GetReserveAccess();
+	TEST2(err, KErrNone);
+
+	//Reserve disk space from the second connection
+	err = db2.ReserveDriveSpace(0);
+	TEST2(err, KErrNone);
+
+	//An attempt to get an access to the reserved space twice.
+	err = db.GetReserveAccess();
+	TEST2(err, KErrInUse);
+
+	//An attempt to get an access to the reserved space from the second connection.
+	err = db2.GetReserveAccess();
+	TEST2(err, KErrNone);
+
+	db.ReleaseReserveAccess();
+
+	//An attempt to release the reserved space twice.
+	db.ReleaseReserveAccess();
+
+	//Free the reserved disk space
+	db.FreeReservedSpace();
+
+	//Free the reserved disk space twice.
+	db.FreeReservedSpace();
+
+	//Free the reserved disk space from the second connection.
+	db2.FreeReservedSpace();
+
+	db2.Close();
+	db.Close();
+	(void)RSqlDatabase::Delete(KTestDatabase);
+	}
+
+/**
+@SYMTestCaseID			SYSLIB-SQL-CT-1649
+@SYMTestCaseDesc		SQL database "out of disk space" tests.
+						The test creates and fills with some records a test database and then reserves a disk space.
+						The second step: the test fills almost all available disk space creting large data files.
+						The third step: the test attempts to delete all records from the test database and fails with
+						KErrDiskFull error.
+						The fourth step: the test gets an access to the reserved disk space and attempts to delete
+						records again. This time the test should not fail.
+@SYMTestPriority		High
+@SYMTestActions			SQL database "out of disk space" tests.
+@SYMTestExpectedResults Test must not fail
+@SYMREQ					REQ5792
+                        REQ5793
+*/
+void DeleteTransactionTest()
+	{
+	TVolumeIOParamInfo volIoPrm;
+	TInt err = TheFs.VolumeIOParam(KTestDrive, volIoPrm);
+    TEST2(err, KErrNone);
+    RDebug::Print(_L("--Drive %d. BlockSize=%d, ClusterSize=%d, RecReadBufSize=%d, RecWriteBufSize=%d\r\n"), KTestDrive, volIoPrm.iBlockSize, volIoPrm.iClusterSize, volIoPrm.iRecReadBufSize, volIoPrm.iRecWriteBufSize);
+	/////////////////////////////////////////////////////////
+    RDebug::Print(_L("--Create and fill database \"%S\".\r\n"), &KTestDatabase);
+	RSqlDatabase db;
+	CreateAndFillTestDatabase(db);
+	db.Close();//When the database gets closed, the persisted journal file will be deleted.
+    RDebug::Print(_L("--Close and reopen database \"%S\" (in order to get the persisted journal file deleted).\r\n"), &KTestDatabase);
+    err = db.Open(KTestDatabase);
+    TEST2(err, KErrNone);
+    RDebug::Print(_L("--Reserve disk space for database \"%S\".\r\n"), &KTestDatabase);
+    err = db.ReserveDriveSpace(0);
+	TEST2(err, KErrNone);
+    RDebug::Print(_L("--Simulate an \"out of disk space\" situation with creating a very large data file, which occupies almost the all the available disk space.\r\n"));
+	CreateLargeFile();
+	RDebug::Print(_L("--Attempt to delete test data records. The transaction must fail, because of \"out of disk space\".\r\n"));
+	err = DeleteTestRecords(db);
+	TEST2(err, KErrDiskFull);
+    RDebug::Print(_L("--Get an access to the reserved disk space.\r\n"));
+	err = db.GetReserveAccess();
+	TEST2(err, KErrNone);
+    TInt64 diskSpace = ::FreeDiskSpace();
+    RDebug::Print(_L("After GetReserveAccess(), free disk space = %ld. Try again \"Delete records\" transaction. The transaction must not fail.\r\n"), diskSpace);
+	err = DeleteTestRecords(db);
+	RDebug::Print(_L("--DeleteTestRecords() returned %d error.\r\n"), err);
+	TEST(err >= 0);
+	//Releases the access to the reserved disk space
+	db.ReleaseReserveAccess();
+	//Frees the reserved disk space
+	db.FreeReservedSpace();
+    //Free the resources, used in the test
+	DeleteLargeDataFiles();
+	//Verify that the records have been deleted
+	RSqlStatement stmt;
+	err = stmt.Prepare(db, _L("SELECT COUNT(*) FROM A"));
+	TEST2(err, KErrNone);
+	err = stmt.Next();
+	TEST2(err, KSqlAtRow);
+	TInt recCount = stmt.ColumnInt(0);
+	TEST2(recCount, 0);
+	stmt.Close();
+	db.Close();
+	(void)RSqlDatabase::Delete(KTestDatabase);
+	}
+
+//OOD API tests with more than one connection to the same SQL database.
+//The test calls ReserveDriveSpace/GetReserveAccess/ReleaseReserveAccess in a different
+//combinations on four RSqlDatabase objects, connected to the same database .
+//The test should not fail or panic.
+void MultiDbTest()
+    {
+    RSqlDatabase db1;
+	CreateAndFillTestDatabase(db1);
+
+    RSqlDatabase db2;
+    TInt err = db2.Open(KTestDatabase);
+    TEST2(err, KErrNone);
+
+    //Play with "ReserveDriveSpace" on both sessions
+    err = db1.ReserveDriveSpace(0);
+    TEST2(err, KErrNone);
+    err = db2.ReserveDriveSpace(0);
+    TEST2(err, KErrNone);
+    db2.FreeReservedSpace();
+    err = db2.ReserveDriveSpace(0);
+    TEST2(err, KErrNone);
+
+    //Get an access to the reserved space through db2
+	err = db2.GetReserveAccess();
+    TEST2(err, KErrNone);
+    //Free/re-reserve disk space for db1.
+    db1.FreeReservedSpace();
+    err = db1.ReserveDriveSpace(0);
+    TEST2(err, KErrNone);
+
+    RSqlDatabase db4;
+    err = db4.Open(KTestDatabase);
+    TEST2(err, KErrNone);
+
+    //Try to reserve space for db4.
+    err = db4.ReserveDriveSpace(0);
+    TEST2(err, KErrNone);
+
+    RSqlDatabase db3;
+    err = db3.Open(KTestDatabase);
+    TEST2(err, KErrNone);
+
+    //Try to reserve space for session db3.
+    err = db3.ReserveDriveSpace(0);
+    TEST2(err, KErrNone);
+
+    //Release and free db2 access to the reserved space.
+    db2.ReleaseReserveAccess();
+    db2.FreeReservedSpace();
+
+    db3.FreeReservedSpace();
+    db3.Close();
+
+    db4.FreeReservedSpace();
+    db4.Close();
+
+    //Get an access to the reserved space through db2.
+    //But it was freed, so the call will fail.
+	err = db2.GetReserveAccess();
+    TEST2(err, KErrNotFound);
+
+    //Free/re-reserve disk space for db1.
+    db1.FreeReservedSpace();
+    err = db1.ReserveDriveSpace(0);
+    TEST2(err, KErrNone);
+
+    //Get/release the access to the reserved space for db1.
+	err = db1.GetReserveAccess();
+    TEST2(err, KErrNone);
+    db1.ReleaseReserveAccess();
+
+    //Get an access to the reserved space for db2.
+    //The call will fail because there is no reserved disk space for db2.
+	err = db2.GetReserveAccess();
+    TEST2(err, KErrNotFound);
+
+    //Free the reserved space - db1
+    db1.FreeReservedSpace();
+
+	db2.Close();
+	db1.Close();
+
+	(void)RSqlDatabase::Delete(KTestDatabase);
+    }
+
+void DoTests()
+	{
+	TheTest.Start(_L(" \"Simple calls\" OOD test "));
+	SimpleCallsTest();
+
+	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1649 \"Delete transaction\" OOD test "));
+	DeleteTransactionTest();
+
+	TheTest.Next(_L(" Multi db OOD test "));
+	MultiDbTest();
+	}
+
+TInt E32Main()
+	{
+	TheTest.Title();
+
+	CTrapCleanup* tc = CTrapCleanup::New();
+
+	__UHEAP_MARK;
+
+	CreateTestEnv();
+	DeleteTestFiles();
+	DoTests();
+	DeleteTestFiles();
+	TheFs.Close();
+
+	__UHEAP_MARKEND;
+
+	TheTest.End();
+	TheTest.Close();
+
+	delete tc;
+
+	User::Heap().Check();
+	return KErrNone;
+	}