sl@0: // Copyright (c) 2005-2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // SQL Client side API header sl@0: // sl@0: // sl@0: sl@0: /** sl@0: @file sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: #ifndef __SQLDB_H__ sl@0: #define __SQLDB_H__ sl@0: sl@0: #ifndef __S32STD_H__ sl@0: #include //RReadStream, RWriteStream sl@0: #endif sl@0: sl@0: #ifndef SYMBIAN_ENABLE_SPLIT_HEADERS sl@0: #include sl@0: #endif sl@0: sl@0: //Forward declarations sl@0: class CSqlSecurityPolicy; sl@0: class RSqlDatabase; sl@0: class CSqlDatabaseImpl; sl@0: class RSqlStatement; sl@0: class CSqlStatementImpl; sl@0: class RSqlColumnReadStream; sl@0: class RSqlParamWriteStream; sl@0: class TSqlScalarFullSelectQuery; sl@0: class RSqlBlob; sl@0: class RSqlBlobReadStream; sl@0: class RSqlBlobWriteStream; sl@0: class TSqlResourceProfiler; sl@0: sl@0: /** sl@0: Used to specify that the ROWID of the most recently inserted record sl@0: from the specified database connection should be used as the ROWID sl@0: in a call to directly access a blob. sl@0: sl@0: @see RSqlBlobReadStream sl@0: @see RSqlBlobWriteStream sl@0: @see TSqlBlob sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlLastInsertedRowId = -1; sl@0: sl@0: /** sl@0: A container for the security policies for a shared SQL database. sl@0: sl@0: The container can contain: sl@0: - security policies that apply to the database. sl@0: - security policies that apply to individual database objects, i.e. database tables. sl@0: sl@0: For the database, you use RSqlSecurityPolicy::SetDbPolicy() to apply a separate sl@0: security policy to: sl@0: - the database schema. sl@0: - read activity on the database. sl@0: - write activity on the database. sl@0: sl@0: For database tables, you use RSqlSecurityPolicy::SetPolicy() to apply a separate sl@0: security policy to: sl@0: - write activity on each named database table. sl@0: - read activity on each named database table. sl@0: sl@0: A client uses a RSqlSecurityPolicy object to create a secure database. It does this by: sl@0: - creating a RSqlSecurityPolicy object. sl@0: - setting all the appropriate security policies into it. sl@0: - passing the object as an argument to RSqlDatabase::Create(). sl@0: - closing the RSqlSecurityPolicy object on return from RSqlDatabase::Create(). sl@0: sl@0: Once a secure shared database has been created with specific security policies, sl@0: these policies are made persistent and cannot be changed during the life of sl@0: that database. sl@0: sl@0: Security policies are encapsulated by TSecurityPolicy objects. sl@0: The general usage pattern is to create the security policies container object sl@0: (RSqlSecurityPolicy) using a default security policy (TSecurityPolicy), and then sl@0: to assign more specific 'overriding' security policies. sl@0: sl@0: The following code fragment shows how you do this: sl@0: sl@0: @code sl@0: TSecurityPolicy defaultPolicy; sl@0: RSqlSecurityPolicy securityPolicy; sl@0: RSqlDatabase database; sl@0: TInt err; sl@0: sl@0: // Create security policies container object using a default security policy. sl@0: securityPolicy.Create(defaultPolicy); sl@0: sl@0: // Set up policy to apply to database schema sl@0: // and assign it sl@0: TSecurityPolicy schemaPolicy; sl@0: ... sl@0: err = securityPolicy.SetDbPolicy(RSqlSecurityPolicy::ESchemaPolicy, schemaPolicy); sl@0: sl@0: // Set up policy to apply to write activity on the database sl@0: // and assign it sl@0: TSecurityPolicy writePolicy; sl@0: ... sl@0: err = securityPolicy.SetDbPolicy(RSqlSecurityPolicy::EWritePolicy, writePolicy); sl@0: sl@0: // Set up policy to apply to write activity to the database table named "Table1" sl@0: // and assign it sl@0: TSecurityPolicy tablePolicy1; sl@0: ... sl@0: err = securityPolicy.SetPolicy(RSqlSecurityPolicy::ETable, _L("Table1"), RSqlSecurityPolicy::EWritePolicy, tablePolicy1); sl@0: sl@0: // Set up policy to apply to read activity to the database table named "Table2" sl@0: TSecurityPolicy tablePolicy2; sl@0: err = securityPolicy.SetPolicy(RSqlSecurityPolicy::ETable, _L("Table2"), RSqlSecurityPolicy::EReadPolicy, tablePolicy2); sl@0: sl@0: // Create the database, passing the security policies sl@0: err = database.Create(KDatabaseName, securityPolicy); sl@0: sl@0: // We can close the RSqlSecurityPolicy object. sl@0: securityPolicy.Close(); sl@0: @endcode sl@0: sl@0: Note that in this example code fragment, the client has not assigned specific sl@0: overriding policies for all possible cases; for example, no overriding policy sl@0: has been assigned to control read activity on the database, read activity sl@0: on "Table1", nor write activity on "Table2". sl@0: For these cases, the default security policy will apply. sl@0: sl@0: A client can also retrieve a database's security policies by calling sl@0: RSqlDatabase::GetSecurityPolicy(); this returns a RSqlSecurityPolicy object sl@0: containing the security policies. Note that it is the client's responsibility sl@0: to close the RSqlSecurityPolicy object when the client no longer needs it. The sl@0: following code fragment suggests how you might do this: sl@0: sl@0: @code sl@0: RSqlDatabase database; sl@0: RSqlSecurityPolicy securityPolicy; sl@0: sl@0: // Retrieve the security policies; on return from the call to sl@0: // GetSecurityPolicy(), the RSqlSecurityPolicy object passed sl@0: // to this function will contain the security policies. sl@0: database.GetSecurityPolicy(securityPolicy); sl@0: ... sl@0: // This is the security policy that applies to database schema sl@0: TSecurityPolicy schemaPolicy = securityPolicy.DbPolicy(RSqlSecurityPolicy::ESchemaPolicy); sl@0: ... sl@0: // This is the security policy that applies to write activity to the database sl@0: // table named "Table1". sl@0: TSecurityPolicy writePolicy = securityPolicy.Policy(RSqlSecurityPolicy::ETable, _L("Table1"), RSqlSecurityPolicy::EWritePolicy); sl@0: ... sl@0: // Close the RSqlSecurityPolicy object when no longer needed. sl@0: securityPolicy.Close(); sl@0: @endcode sl@0: sl@0: Note that in the cases where an 'overriding' security policy was not originally assigned, sl@0: then the security policy returned will simply be the default security policy. sl@0: sl@0: Note: The database security policies are used to control the access to the objects (tables, indexes, triggers, views) sl@0: in the main database. The access to the temporary tables, indexes, etc. is not a subject of any restrictions, e.g. sl@0: a client with "read" database security policy only can create and use temporary tables, views, indexes, triggers. sl@0: sl@0: @see TSecurityPolicy sl@0: @see RSqlDatabase sl@0: @see RSqlSecurityPolicy::SetDbPolicy() sl@0: @see RSqlSecurityPolicy::SetPolicy() sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: class RSqlSecurityPolicy sl@0: { sl@0: friend class RSqlDatabase; sl@0: sl@0: public: sl@0: /** sl@0: Defines a set of values that represents the database security policy types. sl@0: Each database security policy type refers to a set of capabilities encapsulated in sl@0: a TSecurityPolicy object. The TSecurityPolicy object defines what capabilities the calling sl@0: application must have in order to perform partiqular database operation. sl@0: @see TSecurityPolicy sl@0: */ sl@0: enum TPolicyType sl@0: { sl@0: /** sl@0: Schema database security policy. An application with schema database security policy can sl@0: modify the database schema, write to database, read from database. sl@0: */ sl@0: ESchemaPolicy, sl@0: /** sl@0: Read database security policy. An application with read database security policy can sl@0: read from database. sl@0: */ sl@0: EReadPolicy, sl@0: /** sl@0: Write database security policy. An application with write database security policy can sl@0: write to database. sl@0: */ sl@0: EWritePolicy sl@0: }; sl@0: /** sl@0: Not currently supported. sl@0: sl@0: Defines a set of values that represents the database objects which can be protected by sl@0: database security policy types. sl@0: */ sl@0: enum TObjectType sl@0: { sl@0: ETable sl@0: }; sl@0: IMPORT_C RSqlSecurityPolicy(); sl@0: IMPORT_C TInt Create(const TSecurityPolicy& aDefaultPolicy); sl@0: IMPORT_C void CreateL(const TSecurityPolicy& aDefaultPolicy); sl@0: IMPORT_C void Close(); sl@0: IMPORT_C TInt SetDbPolicy(TPolicyType aPolicyType, const TSecurityPolicy& aPolicy); sl@0: IMPORT_C TInt SetPolicy(TObjectType aObjectType, const TDesC& aObjectName, TPolicyType aPolicyType, const TSecurityPolicy& aPolicy); sl@0: IMPORT_C TSecurityPolicy DefaultPolicy() const; sl@0: IMPORT_C TSecurityPolicy DbPolicy(TPolicyType aPolicyType) const; sl@0: IMPORT_C TSecurityPolicy Policy(TObjectType aObjectType, const TDesC& aObjectName, TPolicyType aPolicyType) const; sl@0: sl@0: IMPORT_C void ExternalizeL(RWriteStream& aStream) const; sl@0: IMPORT_C void InternalizeL(RReadStream& aStream); sl@0: sl@0: private: sl@0: void Set(CSqlSecurityPolicy& aImpl); sl@0: CSqlSecurityPolicy& Impl() const; sl@0: sl@0: private: sl@0: CSqlSecurityPolicy* iImpl; sl@0: }; sl@0: sl@0: /** sl@0: A handle to a SQL database. sl@0: sl@0: A RSqlDatabase object is, in effect, a handle to the SQL database. A client can: sl@0: - create a SQL database by calling RSqlDatabase::Create(). sl@0: - open an existing SQL database by calling RSqlDatabase::Open(). sl@0: - close a SQL database by calling RSqlDatabase::Close(). sl@0: - copy a SQL database by calling RSqlDatabase::Copy(). sl@0: - delete a SQL database by calling RSqlDatabase::Delete(). sl@0: - attach a SQL database to current database connection by calling RSqlDatabase::Attach(). sl@0: - detach a SQL database from current database connection by calling RSqlDatabase::Detach(). sl@0: sl@0: The RSqlDatabase handles are not thread-safe. sl@0: sl@0: A client can create either a non-secure database or a secure database, sl@0: depending on the variant of RSqlDatabase::Create() that is used. sl@0: - a non-secure database is created if the RSqlDatabase::Create(const TDesC&) variant is used. sl@0: - a secure database is created if the RSqlDatabase::Create(const TDesC&, const RSqlSecurityPolicy&) sl@0: variant is used. In this case, a container containing a collection of security sl@0: policies needs to be set up first and passed to this Create() function. sl@0: See references to RSqlSecurityPolicy for more information on security policies. sl@0: sl@0: A client can also specify how it wants a transaction to interact with sl@0: other transactions that may be running concurrently. The various ways in which sl@0: transactions can interact (i.e. how one transaction can affect another) are sl@0: referred to as "transaction isolation levels", and are defined by the values sl@0: of the TIsolationLevel enum. A client specifies this by calling RSqlDatabase::SetIsolationLevel(). sl@0: sl@0: Each of the various flavours of Open and Create allows the optional provision of a sl@0: configuration string. It is acceptable for this string to be missing. sl@0: In the case where the string is missing, the config in the SqlServer.sql file sl@0: will be used. If that does not exist then the MMH macro definitions will be used. sl@0: sl@0: The config string is in the format PARAM=VALUE; PARAM=VALUE;... sl@0: sl@0: Allowed parameters are: sl@0: cache_size=nnnn sl@0: page_size=nnnn sl@0: encoding=UTF8|UTF16 sl@0: sl@0: Badly formed config strings are reported as KErrArgument sl@0: sl@0: The string may not exceed 255 characters. sl@0: sl@0: Please note that a database can only be accessed within the thread where it has been created. It is then not possible sl@0: to create a database from thread1 and access it from thread2. sl@0: sl@0: A client calls RSqlDatabase::Exec() to execute SQL statements. sl@0: @see RSqlDatabase::Create() sl@0: @see RSqlDatabase::Open() sl@0: @see RSqlDatabase::Close() sl@0: @see RSqlDatabase::Copy() sl@0: @see RSqlDatabase::Delete() sl@0: @see RSqlDatabase::Attach() sl@0: @see RSqlDatabase::Detach() sl@0: @see RSqlDatabase::SetIsolationLevel() sl@0: @see RSqlDatabase::Exec() sl@0: @see TIsolationLevel sl@0: @see RSqlSecurityPolicy sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: class RSqlDatabase sl@0: { sl@0: friend class RSqlStatement; sl@0: friend class TSqlScalarFullSelectQuery; sl@0: friend class RSqlBlob; sl@0: friend class RSqlBlobReadStream; sl@0: friend class RSqlBlobWriteStream; sl@0: friend class TSqlResourceProfiler; sl@0: sl@0: public: sl@0: /** sl@0: Defines a set of values that represents the transaction isolation level. sl@0: sl@0: A transaction isolation level defines the way in which a transaction sl@0: interacts with other transactions that may be in progress concurrently. sl@0: sl@0: A client sets the transaction isolation level by calling SetIsolationLevel() sl@0: sl@0: @see RSqlDatabase::SetIsolationLevel() sl@0: */ sl@0: enum TIsolationLevel sl@0: { sl@0: /** sl@0: A transaction can read uncommitted data, i.e. data that is being changed sl@0: by another transaction, which is still in progress. sl@0: sl@0: This means that sl@0: - a 'database read' transaction will not block 'database write' transactions sl@0: being performed by different database connections on the same shared database. sl@0: - a 'database read' transaction will not be blocked by 'database write' sl@0: transactions performed by the same database connection. sl@0: - concurrent 'database write' transactions are prevented. sl@0: sl@0: This transaction isolation level can be set at any time during sl@0: the lifetime of the database. sl@0: sl@0: @see TIsolationLevel sl@0: @see RSqlDatabase::SetIsolationLevel() sl@0: */ sl@0: EReadUncommitted, sl@0: sl@0: /** sl@0: Not currently supported. sl@0: sl@0: A transaction cannot read uncommitted data. "Dirty reads" are prevented. sl@0: sl@0: "Dirty read" is a data inconsistency type which can be described with the following example: sl@0: - Transaction A updates TableA.Column1 value from 1 to 2; sl@0: - Transaction B reads TableA.Column1 value; sl@0: - Transaction A rolls back and restores the original value of TableA.Column1 (1); sl@0: - Transaction B ends showing that TableA.Column1 value is 2, even though, logically and transactionally, sl@0: this data never really even existed in the database because Transaction A never committed that change sl@0: to the database; sl@0: sl@0: @see TIsolationLevel sl@0: @see RSqlDatabase::SetIsolationLevel() sl@0: */ sl@0: EReadCommitted, sl@0: sl@0: /** sl@0: Not currently supported. sl@0: sl@0: A transaction cannot change data that is being read by a different transaction. sl@0: "Dirty reads" and "non-repeatable reads" are prevented. sl@0: sl@0: "Non-repeatable reads" is a data inconsistency type which can be described with the following example: sl@0: - Transaction A reads TableA.Column1 value which is 1; sl@0: - Transaction B updates TableA.Column1 value from 1 to 2; sl@0: - Transaction B commits the chages; sl@0: - Transaction A reads TableA.Column1 value again. Transaction A has inconsistent data because TableA.Column1 sl@0: value now is 2 instead of 1, all within the scope of the same Transaction A; sl@0: sl@0: @see TIsolationLevel sl@0: @see RSqlDatabase::SetIsolationLevel() sl@0: */ sl@0: ERepeatableRead, sl@0: sl@0: /** sl@0: Any number of 'database read' transactions can be performed concurrently sl@0: by different database connections on the same shared database. sl@0: sl@0: Only one 'database write' transaction can be performed at any one time. If a sl@0: 'database write' transaction is in progress, then any attempt to start sl@0: another 'database read' or 'database write' transaction will be blocked sl@0: until the first 'database write' transaction has completed. sl@0: sl@0: This is the default isolation level, if no isolation level is sl@0: explicitly set. sl@0: sl@0: "Dirty reads", "non-repeatable" reads and "phantom reads" are prevented. sl@0: sl@0: "Phantom reads" is a data inconsistency type which can be described with the following example: sl@0: - Transaction A reads all rows that have Column1 = 1; sl@0: - Transaction B inserts a new row which has Column1 = 1; sl@0: - Transaction B commits; sl@0: - Transaction A updates all rows that have Column1 = 1. This will also update the row that sl@0: Transaction B inserted, because Transaction A must read the data again in order to update it. sl@0: - Transaction A commits; sl@0: sl@0: @see TIsolationLevel sl@0: @see RSqlDatabase::SetIsolationLevel() sl@0: */ sl@0: ESerializable sl@0: }; sl@0: /** sl@0: This structure is used for retrieving the database size and database free space. sl@0: @see RSqlDatabase::Size(TSize&) sl@0: */ sl@0: struct TSize sl@0: { sl@0: /** The database size in bytes*/ sl@0: TInt64 iSize; sl@0: /** The database free space in bytes*/ sl@0: TInt64 iFree; sl@0: }; sl@0: sl@0: /** If this value is used as an argument of RSqlDatabase::Compact() (aSize argument), then all free space will be removed */ sl@0: enum {EMaxCompaction = -1}; sl@0: sl@0: IMPORT_C RSqlDatabase(); sl@0: sl@0: IMPORT_C TInt Create(const TDesC& aDbFileName, const TDesC8* aConfig=NULL); sl@0: IMPORT_C TInt Create(const TDesC& aDbFileName, sl@0: const RSqlSecurityPolicy& aSecurityPolicy, const TDesC8* aConfig=NULL); sl@0: IMPORT_C TInt Open(const TDesC& aDbFileName, const TDesC8* aConfig=NULL); sl@0: IMPORT_C void CreateL(const TDesC& aDbFileName, const TDesC8* aConfig=NULL); sl@0: IMPORT_C void CreateL(const TDesC& aDbFileName, sl@0: const RSqlSecurityPolicy& aSecurityPolicy, const TDesC8* aConfig=NULL); sl@0: IMPORT_C void OpenL(const TDesC& aDbFileName, const TDesC8* aConfig=NULL); sl@0: sl@0: IMPORT_C void Close(); sl@0: sl@0: IMPORT_C TInt Attach(const TDesC& aDbFileName, const TDesC& aDbName); sl@0: IMPORT_C TInt Detach(const TDesC& aDbName); sl@0: sl@0: IMPORT_C static TInt Copy(const TDesC& aSrcDbFileName, const TDesC& aDestDbFileName); sl@0: IMPORT_C static TInt Delete(const TDesC& aDbFileName); sl@0: sl@0: IMPORT_C TInt GetSecurityPolicy(RSqlSecurityPolicy& aSecurityPolicy) const; sl@0: IMPORT_C void GetSecurityPolicyL(RSqlSecurityPolicy& aSecurityPolicy) const; sl@0: sl@0: IMPORT_C TInt SetIsolationLevel(TIsolationLevel aIsolationLevel); sl@0: sl@0: IMPORT_C TInt Exec(const TDesC& aSqlStmt); sl@0: IMPORT_C TInt Exec(const TDesC8& aSqlStmt); sl@0: sl@0: IMPORT_C void Exec(const TDesC& aSqlStmt, TRequestStatus& aStatus); sl@0: IMPORT_C void Exec(const TDesC8& aSqlStmt, TRequestStatus& aStatus); sl@0: sl@0: IMPORT_C TPtrC LastErrorMessage() const; sl@0: IMPORT_C TInt64 LastInsertedRowId() const; sl@0: sl@0: IMPORT_C TBool InTransaction() const; sl@0: IMPORT_C TInt Size() const; sl@0: IMPORT_C TInt Size(TSize& aSize, const TDesC& aDbName = KNullDesC) const; sl@0: sl@0: IMPORT_C TInt Compact(TInt64 aSize, const TDesC& aDbName = KNullDesC); sl@0: IMPORT_C void Compact(TInt64 aSize, TRequestStatus& aStatus, const TDesC& aDbName = KNullDesC); sl@0: sl@0: IMPORT_C TInt ReserveDriveSpace(TInt aSize); sl@0: IMPORT_C void FreeReservedSpace(); sl@0: IMPORT_C TInt GetReserveAccess(); sl@0: IMPORT_C void ReleaseReserveAccess(); sl@0: sl@0: private: sl@0: CSqlDatabaseImpl& Impl() const; sl@0: sl@0: private: sl@0: CSqlDatabaseImpl* iImpl; sl@0: }; sl@0: sl@0: /** sl@0: TSqlScalarFullSelectQuery interface is used for executing SELECT sql queries, which sl@0: return a single row consisting of a single column value. sl@0: sl@0: Examples. sl@0: sl@0: CASE 1 - retrieving records count of a table: sl@0: @code sl@0: RSqlDatabase db; sl@0: //initialize db object.... sl@0: ....... sl@0: TSqlScalarFullSelectQuery fullSelectQuery(db); sl@0: TInt recCnt = fullSelectQuery.SelectIntL(_L("SELECT COUNT(*) FROM PersonTbl")); sl@0: @endcode sl@0: sl@0: CASE 2 - retrieving specific column value using a condition in the SELECT statement: sl@0: @code sl@0: RSqlDatabase db; sl@0: //initialize db object.... sl@0: ....... sl@0: TSqlScalarFullSelectQuery fullSelectQuery(db); sl@0: TInt personId = fullSelectQuery.SelectIntL(_L("SELECT ID FROM PersonTbl WHERE Name = 'John'")); sl@0: @endcode sl@0: sl@0: CASE 3 - retrieving a text column value, the receiving buffer is not big enough: sl@0: @code sl@0: RSqlDatabase db; sl@0: //initialize db object.... sl@0: ....... sl@0: TSqlScalarFullSelectQuery fullSelectQuery(db); sl@0: HBufC* buf = HBufC::NewLC(20); sl@0: TPtr name = buf->Des(); sl@0: TInt rc = fullSelectQuery.SelectTextL(_L("SELECT Name FROM PersonTbl WHERE Id = 1"), name); sl@0: TEST(rc >= 0); //the function may return only non-negative values sl@0: if(rc > 0) sl@0: { sl@0: buf = buf->ReAllocL(rc); sl@0: CleanupStack::Pop(); sl@0: CleanupStack::PushL(buf); sl@0: name.Set(buf->Des()); sl@0: rc = fullSelectQuery.SelectTextL(_L("SELECT Name FROM PersonTbl WHERE Id = 1"), name); sl@0: TEST(rc == 0); sl@0: } sl@0: CleanupStack::PopAndDestroy();//buf sl@0: @endcode sl@0: sl@0: @see RSqlDatabase sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: class TSqlScalarFullSelectQuery sl@0: { sl@0: public: sl@0: IMPORT_C TSqlScalarFullSelectQuery(); sl@0: IMPORT_C TSqlScalarFullSelectQuery(RSqlDatabase& aDatabase); sl@0: IMPORT_C void SetDatabase(RSqlDatabase& aDatabase); sl@0: sl@0: IMPORT_C TInt SelectIntL(const TDesC& aSqlStmt); sl@0: IMPORT_C TInt64 SelectInt64L(const TDesC& aSqlStmt); sl@0: IMPORT_C TReal SelectRealL(const TDesC& aSqlStmt); sl@0: IMPORT_C TInt SelectTextL(const TDesC& aSqlStmt, TDes& aDest); sl@0: IMPORT_C TInt SelectBinaryL(const TDesC& aSqlStmt, TDes8& aDest); sl@0: sl@0: IMPORT_C TInt SelectIntL(const TDesC8& aSqlStmt); sl@0: IMPORT_C TInt64 SelectInt64L(const TDesC8& aSqlStmt); sl@0: IMPORT_C TReal SelectRealL(const TDesC8& aSqlStmt); sl@0: IMPORT_C TInt SelectTextL(const TDesC8& aSqlStmt, TDes& aDest); sl@0: IMPORT_C TInt SelectBinaryL(const TDesC8& aSqlStmt, TDes8& aDest); sl@0: sl@0: private: sl@0: inline CSqlDatabaseImpl& Impl() const; sl@0: sl@0: private: sl@0: CSqlDatabaseImpl* iDatabaseImpl; sl@0: }; sl@0: sl@0: /** sl@0: An enumeration whose values represent the supported database column types. sl@0: sl@0: sl@0: @see RSqlStatement::ColumnType() sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: enum TSqlColumnType sl@0: { sl@0: /** sl@0: Null column value. sl@0: */ sl@0: ESqlNull, sl@0: sl@0: /** sl@0: 32-bit integer column value. sl@0: */ sl@0: ESqlInt, sl@0: sl@0: /** sl@0: 64-bit integer column value. sl@0: */ sl@0: ESqlInt64, sl@0: sl@0: /** sl@0: 64-bit floating point column value. sl@0: */ sl@0: ESqlReal, sl@0: sl@0: /** sl@0: Unicode text, a sequence of 16-bit character codes. sl@0: */ sl@0: ESqlText, sl@0: sl@0: /** sl@0: Binary data, a sequence of bytes. sl@0: */ sl@0: ESqlBinary sl@0: }; sl@0: sl@0: /** sl@0: Represents an SQL statement. sl@0: sl@0: An object of this type can be used to execute all types of SQL statements; this sl@0: includes SQL statements with parameters. sl@0: sl@0: If a SELECT statament is passed to RSqlStatement::Prepare(), then the returned record set sl@0: is forward only, non-updateable. sl@0: sl@0: There are a number of ways that this object is used; here are some examples. sl@0: sl@0: CASE 1 - the execution of a SQL statement, which does not return record set: sl@0: sl@0: @code sl@0: RSqlDatabase database; sl@0: ......... sl@0: RSqlStatement stmt; sl@0: TInt err = stmt.Prepare(database, _L("INSERT INTO Tbl1(Fld1) VALUES(:Val)")); sl@0: TInt paramIndex = stmt.ParameterIndex(_L(":Val")); sl@0: for(TInt i=1;i<=10;++i) sl@0: { sl@0: err = stmt.BindInt(paramIndex, i); sl@0: err = stmt.Exec(); sl@0: err = stmt.Reset(); sl@0: } sl@0: stmt.Close(); sl@0: @endcode sl@0: sl@0: The following pseudo code shows the general pattern: sl@0: sl@0: @code sl@0: sl@0: [begin:] sl@0: ()> sl@0: sl@0: [] sl@0: [()>] sl@0: [] sl@0: @endcode sl@0: sl@0: CASE 2 - the execution of a SQL statement, which returns a record set: sl@0: sl@0: @code sl@0: RSqlDatabase database; sl@0: ......... sl@0: RSqlStatement stmt; sl@0: TInt err = stmt.Prepare(database, _L("SELECT Fld1 FROM Tbl1 WHERE Fld1 > :Val")); sl@0: TInt paramIndex = stmt.ParameterIndex(_L(":Val")); sl@0: err = stmt.BindInt(paramIndex, 5); sl@0: TInt columnIndex = stmt.ColumnIndex(_L("Fld1")); sl@0: while((err = stmt.Next()) == KSqlAtRow) sl@0: { sl@0: TInt val = stmt.ColumnInt(columnIndex); sl@0: RDebug::Print(_L("val=%d\n"), val); sl@0: } sl@0: if(err == KSqlAtEnd) sl@0: ; sl@0: else sl@0: ; sl@0: stmt.Close(); sl@0: @endcode sl@0: sl@0: The following pseudo code shows the general pattern: sl@0: sl@0: @code sl@0: sl@0: [begin:] sl@0: sl@0: sl@0: if(err == KSqlAtEnd) sl@0: ; sl@0: else sl@0: ; sl@0: [] sl@0: [()>] sl@0: [] sl@0: @endcode sl@0: sl@0: CASE 3.1 - SELECT statements: large column data processing, where the data is sl@0: copied into a buffer supplied by the client: sl@0: sl@0: @code sl@0: RSqlDatabase database; sl@0: ......... sl@0: RSqlStatement stmt; sl@0: TInt err = stmt.Prepare(database, _L("SELECT BinaryField FROM Tbl1")); sl@0: TInt columnIndex = stmt.ColumnIndex(_L("BinaryField")); sl@0: while((err = stmt.Next()) == KSqlAtRow) sl@0: { sl@0: TInt size = stmt. ColumnSize(columnIndex); sl@0: HBufC8* buf = HBufC8::NewL(size); sl@0: err = stmt.ColumnBinary(columnIndex, buf->Ptr()); sl@0: ; sl@0: delete buf; sl@0: } sl@0: if(err == KSqlAtEnd) sl@0: ; sl@0: else sl@0: ; sl@0: stmt.Close(); sl@0: @endcode sl@0: sl@0: CASE 3.2 - SELECT statements: large column data processing, where the data is sl@0: accessed by the client without copying: sl@0: sl@0: @code sl@0: RSqlDatabase database; sl@0: ......... sl@0: RSqlStatement stmt; sl@0: TInt err = stmt.Prepare(database, _L("SELECT BinaryField FROM Tbl1")); sl@0: TInt columnIndex = stmt.ColumnIndex(_L("BinaryField")); sl@0: while((err = stmt.Next()) == KSqlAtRow) sl@0: { sl@0: TPtrC8 data = stmt.ColumnBinaryL(columnIndex); sl@0: ; sl@0: } sl@0: if(err == KSqlAtEnd) sl@0: ; sl@0: else sl@0: ; sl@0: stmt.Close(); sl@0: @endcode sl@0: sl@0: CASE 3.3 - SELECT statements, large column data processing (the data is accessed by sl@0: the client without copying), leaving-safe processing: sl@0: sl@0: @code sl@0: RSqlDatabase database; sl@0: ......... sl@0: RSqlStatement stmt; sl@0: TInt err = stmt.Prepare(database, _L("SELECT BinaryField FROM Tbl1")); sl@0: TInt columnIndex = stmt.ColumnIndex(_L("BinaryField")); sl@0: while((err = stmt.Next()) == KSqlAtRow) sl@0: { sl@0: TPtrC8 data; sl@0: TInt err = stmt.ColumnBinary(columnIndex, data); sl@0: if(err == KErrNone) sl@0: { sl@0: ; sl@0: } sl@0: } sl@0: if(err == KSqlAtEnd) sl@0: ; sl@0: else sl@0: ; sl@0: stmt.Close(); sl@0: @endcode sl@0: sl@0: CASE 3.4 - SELECT statements: large column data processing, where the data is sl@0: accessed by the client using a stream: sl@0: sl@0: @code sl@0: RSqlDatabase database; sl@0: ......... sl@0: RSqlStatement stmt; sl@0: TInt err = stmt.Prepare(database, _L("SELECT BinaryField FROM Tbl1")); sl@0: TInt columnIndex = stmt.ColumnIndex(_L("BinaryField")); sl@0: while((err = stmt.Next()) == KSqlAtRow) sl@0: { sl@0: RSqlColumnReadStream stream; sl@0: err = stream.ColumnBinary(stmt, columnIndex); sl@0: ; sl@0: stream.Close(); sl@0: } sl@0: if(err == KSqlAtEnd) sl@0: ; sl@0: else sl@0: ; sl@0: stmt.Close(); sl@0: @endcode sl@0: sl@0: CASE 4 - the execution of a SQL statement with parameter(s), some of which may sl@0: be large text or binary values: sl@0: sl@0: @code sl@0: RSqlDatabase database; sl@0: ......... sl@0: RSqlStatement stmt; sl@0: TInt err = sl@0: stmt.Prepare(database, _L("UPDATE Tbl1 SET LargeTextField = :LargeTextVal WHERE IdxField = :KeyVal")); sl@0: TInt paramIndex1 = stmt.ParameterIndex(_L(":LargeTextVal")); sl@0: TInt paramIndex2 = stmt.ParameterIndex(_L(":KeyVal")); sl@0: for(TInt i=1;i<=10;++i) sl@0: { sl@0: RSqlParamWriteStream stream; sl@0: err = stream.BindText(stmt, paramIndex1); sl@0: ; sl@0: stream.Close(); sl@0: err = stmt.BindInt(paramIndex2, i); sl@0: err = stmt.Exec(); sl@0: stmt.Reset(); sl@0: } sl@0: stmt.Close(); sl@0: @endcode sl@0: sl@0: The following table shows what is returned when the caller uses a specific sl@0: column data retrieving function on a specific column type. sl@0: sl@0: @code sl@0: -------------------------------------------------------------------------------- sl@0: Column type | ColumnInt() ColumnInt64() ColumnReal() ColumnText() ColumnBinary() sl@0: -------------------------------------------------------------------------------- sl@0: Null........|.0...........0.............0.0..........KNullDesC....KNullDesC8 sl@0: Int.........|.Int.........Int64.........Real.........KNullDesC....KNullDesC8 sl@0: Int64.......|.clamp.......Int64.........Real.........KNullDesC....KNullDesC8 sl@0: Real........|.round.......round.........Real.........KNullDesC....KNullDesC8 sl@0: Text........|.0...........0.............0.0..........Text.........KNullDesC8 sl@0: Binary......|.0...........0.............0.0..........KNullDesC....Binary sl@0: -------------------------------------------------------------------------------- sl@0: @endcode sl@0: Note the following definitions: sl@0: - "clamp": return KMinTInt or KMaxTInt if the value is outside the range that can be sl@0: represented by the type returned by the accessor function. sl@0: - "round": the floating point value will be rounded up to the nearest integer. sl@0: If the result is outside the range that can be represented by the type returned sl@0: by the accessor function, then it will be clamped. sl@0: sl@0: Note that when handling blob and text data over 2Mb in size it is recommended that the sl@0: RSqlBlobReadStream and RSqlBlobWriteStream classes or the TSqlBlob class is used instead. sl@0: These classes provide a more RAM-efficient way of reading and writing large amounts of sl@0: blob or text data from a database. sl@0: sl@0: @see KMinTInt sl@0: @see KMaxTInt sl@0: @see KNullDesC sl@0: @see KNullDesC8 sl@0: @see RSqlBlobReadStream sl@0: @see RSqlBlobWriteStream sl@0: @see TSqlBlob sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: class RSqlStatement sl@0: { sl@0: friend class RSqlColumnReadStream; sl@0: friend class RSqlParamWriteStream; sl@0: sl@0: public: sl@0: IMPORT_C RSqlStatement(); sl@0: IMPORT_C TInt Prepare(RSqlDatabase& aDatabase, const TDesC& aSqlStmt); sl@0: IMPORT_C TInt Prepare(RSqlDatabase& aDatabase, const TDesC8& aSqlStmt); sl@0: IMPORT_C void PrepareL(RSqlDatabase& aDatabase, const TDesC& aSqlStmt); sl@0: IMPORT_C void PrepareL(RSqlDatabase& aDatabase, const TDesC8& aSqlStmt); sl@0: IMPORT_C void Close(); sl@0: IMPORT_C TBool AtRow() const; sl@0: IMPORT_C TInt Reset(); sl@0: IMPORT_C TInt Exec(); sl@0: IMPORT_C void Exec(TRequestStatus& aStatus); sl@0: IMPORT_C TInt Next(); sl@0: sl@0: IMPORT_C TInt ParameterIndex(const TDesC& aParameterName) const; sl@0: IMPORT_C TInt ColumnCount() const; sl@0: IMPORT_C TInt ColumnIndex(const TDesC& aColumnName) const; sl@0: IMPORT_C TSqlColumnType ColumnType(TInt aColumnIndex) const; sl@0: IMPORT_C TInt DeclaredColumnType(TInt aColumnIndex, TSqlColumnType& aColumnType) const; sl@0: IMPORT_C TInt ColumnSize(TInt aColumnIndex) const; sl@0: sl@0: IMPORT_C TInt BindNull(TInt aParameterIndex); sl@0: IMPORT_C TInt BindInt(TInt aParameterIndex, TInt aParameterValue); sl@0: IMPORT_C TInt BindInt64(TInt aParameterIndex, TInt64 aParameterValue); sl@0: IMPORT_C TInt BindReal(TInt aParameterIndex, TReal aParameterValue); sl@0: IMPORT_C TInt BindText(TInt aParameterIndex, const TDesC& aParameterText); sl@0: IMPORT_C TInt BindBinary(TInt aParameterIndex, const TDesC8& aParameterData); sl@0: IMPORT_C TInt BindZeroBlob(TInt aParameterIndex, TInt aBlobSize); sl@0: sl@0: IMPORT_C TBool IsNull(TInt aColumnIndex) const; sl@0: IMPORT_C TInt ColumnInt(TInt aColumnIndex) const; sl@0: IMPORT_C TInt64 ColumnInt64(TInt aColumnIndex) const; sl@0: IMPORT_C TReal ColumnReal(TInt aColumnIndex) const; sl@0: sl@0: IMPORT_C TPtrC ColumnTextL(TInt aColumnIndex) const; sl@0: IMPORT_C TInt ColumnText(TInt aColumnIndex, TPtrC& aPtr) const; sl@0: IMPORT_C TInt ColumnText(TInt aColumnIndex, TDes& aDest) const; sl@0: sl@0: IMPORT_C TPtrC8 ColumnBinaryL(TInt aColumnIndex) const; sl@0: IMPORT_C TInt ColumnBinary(TInt aColumnIndex, TPtrC8& aPtr) const; sl@0: IMPORT_C TInt ColumnBinary(TInt aColumnIndex, TDes8& aDest) const; sl@0: sl@0: IMPORT_C TInt ColumnName(TInt aColumnIndex, TPtrC& aNameDest); sl@0: IMPORT_C TInt ParameterName(TInt aParameterIndex, TPtrC& aNameDest); sl@0: IMPORT_C TInt ParamName(TInt aParameterIndex, TPtrC& aNameDest); sl@0: private: sl@0: CSqlStatementImpl& Impl() const; sl@0: sl@0: private: sl@0: CSqlStatementImpl* iImpl; sl@0: sl@0: }; sl@0: sl@0: /** sl@0: The read stream interface. sl@0: sl@0: The class is used for reading the content of a column containing either sl@0: binary data or text data. sl@0: sl@0: The class derives from RReadStream, which means that all RReadStream public sl@0: member functions and predefined stream operators \>\> can be used to deal sl@0: with column data. sl@0: sl@0: If the blob or text data is over 2Mb in size then it is recommended that the sl@0: RSqlBlobReadStream or TSqlBlob class is used instead. These classes provide sl@0: a more RAM-efficient way of reading large amounts of blob or text data from sl@0: a database. sl@0: sl@0: The following two cases are typical: sl@0: sl@0: CASE 1 - processing large binary column data. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: ; sl@0: RSqlStatement stmt; sl@0: ; sl@0: TInt rc = stmt.Next(); sl@0: if(rc == KSqlAtRow) sl@0: { sl@0: RSqlColumnReadStream colStream; sl@0: CleanupClosePushL(colStream); sl@0: User::LeaveIfError(colStream.ColumnBinary(stmt, )); sl@0: TInt size = stmt.ColumnSize(); sl@0: //read the column data in a buffer ("buf" variable). sl@0: //(or the column data can be retrieved in a smaller portions) sl@0: colStream.ReadL(buf, size); sl@0: //Close the stream sl@0: CleanupStack::PopAndDestroy(&colStream); sl@0: } sl@0: else sl@0: { sl@0: ... sl@0: } sl@0: @endcode sl@0: sl@0: CASE 2 - processing large text column data. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: ; sl@0: RSqlStatement stmt; sl@0: ; sl@0: TInt rc = stmt.Next(); sl@0: if(rc == KSqlAtRow) sl@0: { sl@0: RSqlColumnReadStream colStream; sl@0: CleanupClosePushL(colStream); sl@0: User::LeaveIfError(colStream.ColumnText(stmt, )); sl@0: TInt size = stmt.ColumnSize(); sl@0: //read the column data in a buffer ("buf" variable). sl@0: //(or the column data can be retrieved in a smaller portions) sl@0: colStream.ReadL(buf, size); sl@0: //Close the stream sl@0: CleanupStack::PopAndDestroy(&colStream); sl@0: } sl@0: else sl@0: { sl@0: ... sl@0: } sl@0: @endcode sl@0: sl@0: @see RSqlBlobReadStream sl@0: @see TSqlBlob sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: class RSqlColumnReadStream : public RReadStream sl@0: { sl@0: public: sl@0: IMPORT_C TInt ColumnText(RSqlStatement& aStmt, TInt aColumnIndex); sl@0: IMPORT_C TInt ColumnBinary(RSqlStatement& aStmt, TInt aColumnIndex); sl@0: IMPORT_C void ColumnTextL(RSqlStatement& aStmt, TInt aColumnIndex); sl@0: IMPORT_C void ColumnBinaryL(RSqlStatement& aStmt, TInt aColumnIndex); sl@0: sl@0: }; sl@0: sl@0: /** sl@0: The write stream interface. sl@0: sl@0: The class is used to set binary data or text data into a parameter. sl@0: This is a also known as binding a parameter. sl@0: sl@0: The class derives from RWriteStream, which means that all RWriteStream public sl@0: member functions and predefined stream operators \<\< can be used to deal with sl@0: the parameter data. sl@0: sl@0: If the blob or text data is over 2Mb in size then it is recommended that the sl@0: RSqlBlobWriteStream or TSqlBlob class is used instead. These classes provide sl@0: a more RAM-efficient way of writing large amounts of blob or text data to sl@0: a database. sl@0: sl@0: The following two cases are typical: sl@0: sl@0: CASE 1 - binding a large binary parameter. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: ; sl@0: RSqlStatement stmt; sl@0: ;//The SQL statement references large binary parameter sl@0: RSqlParamWriteStream paramStream; sl@0: CleanupClosePushL(paramStream); sl@0: User::LeaveIfError(paramStream.BindBinary(stmt, )); sl@0: //Write out the parameter data sl@0: paramStream.WriteL(..); sl@0: paramStream << ; sl@0: ... sl@0: //Commit the stream sl@0: paramStream.CommitL(); sl@0: //Continue with the statement processing issuing Next() or Exec(). sl@0: TInt rc = stmt.Next();//rc = stmt.Exec() sl@0: //Close the stream sl@0: CleanupStack::PopAndDestroy(¶mStream); sl@0: @endcode sl@0: sl@0: CASE 2 - binding a large text parameter. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: ; sl@0: RSqlStatement stmt; sl@0: ;//The SQL statement references large text parameter sl@0: RSqlParamWriteStream paramStream; sl@0: CleanupClosePushL(paramStream); sl@0: User::LeaveIfError(paramStream.BindText(stmt, )); sl@0: //Write out the parameter data sl@0: paramStream.WriteL(..); sl@0: paramStream << ; sl@0: ... sl@0: //Commit the stream sl@0: paramStream.CommitL(); sl@0: //Continue with the statement processing issuing Next() or Exec(). sl@0: TInt rc = stmt.Next();//rc = stmt.Exec() sl@0: //Close the stream sl@0: CleanupStack::PopAndDestroy(¶mStream); sl@0: @endcode sl@0: sl@0: @see RSqlBlobWriteStream sl@0: @see TSqlBlob sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: class RSqlParamWriteStream : public RWriteStream sl@0: { sl@0: public: sl@0: IMPORT_C TInt BindText(RSqlStatement& aStmt, TInt aParameterIndex); sl@0: IMPORT_C TInt BindBinary(RSqlStatement& aStmt, TInt aParameterIndex); sl@0: IMPORT_C void BindTextL(RSqlStatement& aStmt, TInt aParameterIndex); sl@0: IMPORT_C void BindBinaryL(RSqlStatement& aStmt, TInt aParameterIndex); sl@0: sl@0: }; sl@0: sl@0: /** sl@0: A direct handle to a blob, used for reading the content of the blob via a streaming interface. sl@0: sl@0: The target blob is identified using the relevant database connection, table name, sl@0: column name and ROWID of the record to which the blob belongs (also the attached sl@0: database name if the blob is contained in an attached database). sl@0: sl@0: A blob in this context refers to the content of a BLOB or TEXT column, sl@0: and a read handle can be opened on both types of column. sl@0: For TEXT columns it is important to note that no conversions are performed on sl@0: data retrieved using this class - the data is returned as a stream of bytes. sl@0: sl@0: The class derives from RReadStream and provides all of its streaming methods. sl@0: The SizeL() method can be used to check the total size of the blob, in bytes. sl@0: sl@0: It is strongly recommended to use this class for reading the content of large blobs sl@0: because it significantly reduces the amount of RAM that is used when compared to using the sl@0: RSqlColumnReadStream, RSqlStatement::ColumnBinary(L) or RSqlStatement::ColumnText(L) APIs. sl@0: sl@0: Specifically, it is recommended to use this class for blobs over 2Mb in size. sl@0: Indeed, in some circumstances where very large blobs are in use it may be impossible sl@0: to read the blob content using the legacy APIs (due to the server's finite RAM capacity), sl@0: and this class may provide the only way to access the data. sl@0: sl@0: The following code illustrates typical use cases of this class: sl@0: sl@0: CASE 1 - reading large blob data from the last inserted record. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: CleanupClosePushL(db); sl@0: ; sl@0: RSqlBlobReadStream rdStrm; sl@0: CleanupClosePushL(rdStrm); sl@0: rdStrm.OpenL(db, , ); sl@0: HBufC8* buffer = HBufC8::NewLC(KBlockSize); sl@0: TPtr8 bufPtr(buffer->Des()); sl@0: TInt size = rdStrm.SizeL(); sl@0: while(size) sl@0: { sl@0: TInt bytesToRead = (size >= KBlockSize) ? KBlockSize : size ; sl@0: rdStrm.ReadL(bufPtr, bytesToRead); // read the next block of data sl@0: sl@0: size =- bytesToRead; sl@0: } sl@0: CleanupStack::PopAndDestroy(3); // buffer, rdStrm, db sl@0: @endcode sl@0: sl@0: CASE 2 - reading large blob data from a selection of records. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: CleanupClosePushL(db); sl@0: ; sl@0: RSqlStatement stmt; sl@0: CleanupClosePushL(stmt); sl@0: ; sl@0: TInt rc = 0; sl@0: while((rc = stmt.Next()) == KSqlAtRow) sl@0: { sl@0: TInt64 rowid = stmt.ColumnInt64(0); sl@0: RSqlBlobReadStream rdStrm; sl@0: CleanupClosePushL(rdStrm); sl@0: rdStrm.OpenL(db, , , rowid); sl@0: sl@0: HBufC8* buffer = HBufC8::NewLC(KBlockSize); sl@0: TPtr8 bufPtr(buffer->Des()); sl@0: TInt size = rdStrm.SizeL(); sl@0: while(size) sl@0: { sl@0: TInt bytesToRead = (size >= KBlockSize) ? KBlockSize : size ; sl@0: rdStrm.ReadL(bufPtr, bytesToRead); // read the next block of data sl@0: sl@0: size =- bytesToRead; sl@0: } sl@0: CleanupStack::PopAndDestroy(2); // buffer, rdStrm sl@0: } sl@0: CleanupStack::PopAndDestroy(2); // stmt, db sl@0: @endcode sl@0: sl@0: @see RSqlBlobWriteStream sl@0: @see RSqlDatabase::LastInsertedRowId() sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: class RSqlBlobReadStream : public RReadStream sl@0: { sl@0: public: sl@0: IMPORT_C void OpenL(RSqlDatabase& aDb, const TDesC& aTableName, const TDesC& aColumnName, sl@0: TInt64 aRowId = KSqlLastInsertedRowId, const TDesC& aDbName = KNullDesC); sl@0: IMPORT_C TInt SizeL(); sl@0: }; sl@0: sl@0: /** sl@0: A direct handle to a blob, used for writing the content of the blob via a streaming interface. sl@0: sl@0: The target blob is identified using the relevant database connection, table name, sl@0: column name and ROWID of the record to which the blob belongs (also the attached sl@0: database name if the blob is contained in an attached database). sl@0: sl@0: A blob in this context refers to the content of a BLOB or TEXT column, sl@0: and a write handle can be opened on both types of column, except if the sl@0: column is indexed, in which case the open call will fail with KSqlErrGeneral. sl@0: For TEXT columns it is important to note that no conversions are performed on data sl@0: that is stored using this class - the data is simply stored as a stream of bytes. sl@0: sl@0: The class derives from RWriteStream and provides all of its streaming methods. sl@0: The SizeL() method can be used to check the total size of the blob, in bytes. sl@0: Note that this class cannot be used to increase the size of a blob, only to modify sl@0: the existing contents of a blob. An attempt to write beyond the end of a blob will sl@0: fail with KErrEof. sl@0: sl@0: It is strongly recommended to use this class for writing the content of large blobs sl@0: because it significantly reduces the amount of RAM that is used when compared to using sl@0: the RSqlParamWriteStream, RSqlStatement::BindBinary or RSqlStatement::BindText APIs. sl@0: sl@0: Specifically, it is recommended to use this class for blobs over 2Mb in size. sl@0: Indeed, in some circumstances where very large blobs are required it may be impossible sl@0: to create a blob or update its content using the legacy APIs (due to the server's finite sl@0: RAM capacity), and this class may provide the only way to achieve this. sl@0: sl@0: Using this class in combination with zeroblobs it is possible to create and manipulate sl@0: blobs that are gigabytes in size. A zeroblob acts as a place-holder for a blob whose sl@0: content is later written using this class and one can be created using an INSERT sl@0: statement that either contains the SQLite 'zeroblob()' function or on which sl@0: RSqlStatement::BindZeroBlob() has been executed. sl@0: Note that a zeroblob should be created in a column after which there are no columns sl@0: that contain anything other than zeroblobs or NULLs, otherwise the zeroblob must be sl@0: allocated in full in RAM. sl@0: sl@0: When creating a zeroblob it is recommended, where possible, to create the zeroblob and sl@0: then write the blob content within the same transaction. Otherwise the zeroblob will sl@0: have to be journalled before being written to. sl@0: sl@0: It is also strongly recommended to execute calls to WriteL() within a transaction. sl@0: If a leave occurs during a call to WriteL() then the current state of the blob object is sl@0: undefined and a ROLLBACK should be executed to return the blob object to its previous state. sl@0: Note that in order for a ROLLBACK to execute successfully all open RSqlBlobReadStream sl@0: and RSqlBlobWriteStream handles and all open RSqlStatement objects must be closed sl@0: before the ROLLBACK is executed. sl@0: sl@0: The following code illustrates typical use cases of this class: sl@0: sl@0: CASE 1 - creating a 5Mb blob. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: CleanupClosePushL(db); sl@0: ; sl@0: CleanupStack::PushL(TCleanupItem(&DoRollback, &db)); // rollback function sl@0: TInt err = db.Exec(_L("BEGIN")); sl@0: sl@0: err = db.Exec(_L("INSERT INTO table1 VALUES(35, zeroblob(5242880))")); sl@0: sl@0: RSqlBlobWriteStream wrStrm; sl@0: CleanupClosePushL(wrStrm); sl@0: wrStrm.OpenL(db, , ); sl@0: TInt size = wrStrm.SizeL(); sl@0: while(size) sl@0: { sl@0: TInt bytesToWrite = (size >= KBlockSize) ? KBlockSize : size ; sl@0: sl@0: wrStrm.WriteL(buf); // write the next block of data sl@0: size =- bytesToWrite; sl@0: } sl@0: CleanupStack::PopAndDestroy(&wrStrm); sl@0: CleanupStack::Pop(); // TCleanupItem sl@0: err = db.Exec(_L("COMMIT")); // blob data committed to disk sl@0: sl@0: CleanupStack::PopAndDestroy(&db); sl@0: @endcode sl@0: sl@0: CASE 2 - updating a large blob in the last inserted record. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: CleanupClosePushL(db); sl@0: ; sl@0: CleanupStack::PushL(TCleanupItem(&DoRollback, &db)); // rollback function sl@0: TInt err = db.Exec(_L("BEGIN")); sl@0: sl@0: RSqlBlobWriteStream wrStrm; sl@0: CleanupClosePushL(wrStrm); sl@0: wrStrm.OpenL(db, , ); sl@0: sl@0: wrStrm.WriteL(buf); // update the blob sl@0: CleanupStack::PopAndDestroy(&wrStrm); sl@0: CleanupStack::Pop(); // TCleanupItem sl@0: err = db.Exec(_L("COMMIT")); // blob data committed to disk sl@0: sl@0: CleanupStack::PopAndDestroy(&db); sl@0: @endcode sl@0: sl@0: @see RSqlBlobReadStream sl@0: @see RSqlDatabase::LastInsertedRowId() sl@0: @see RSqlStatement::BindZeroBlob() sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: class RSqlBlobWriteStream : public RWriteStream sl@0: { sl@0: public: sl@0: IMPORT_C void OpenL(RSqlDatabase& aDb, const TDesC& aTableName, const TDesC& aColumnName, sl@0: TInt64 aRowId = KSqlLastInsertedRowId, const TDesC& aDbName = KNullDesC); sl@0: IMPORT_C TInt SizeL(); sl@0: }; sl@0: sl@0: /** sl@0: Utility class that provides methods for reading and writing the entire content of sl@0: a blob in a single call. sl@0: sl@0: The target blob is identified using the relevant database connection, table name, sl@0: column name and ROWID of the record to which the blob belongs (also the attached sl@0: database name if the blob is contained in an attached database). sl@0: sl@0: The behaviour of the RSqlBlobReadStream class and the recommendations for using sl@0: it exist for the Get() and GetLC() methods of this class. Similarly, the behaviour sl@0: of the RSqlBlobWriteStream class and the recommendations for using it exist for the sl@0: SetL() method of this class. sl@0: sl@0: In particular, it is strongly recommended to use this class or the RSqlBlobReadStream sl@0: and RSqlBlobWriteStream classes for reading and writing the content of large blobs sl@0: because it significantly reduces the amount of RAM that is used when compared to using sl@0: the legacy streaming and RSqlStatement APIs. sl@0: sl@0: Specifically, it is recommended to use this class for blobs over 2Mb in size. sl@0: Indeed, in some circumstances where very large blobs are in use it may be impossible sl@0: to read or write to a blob using the legacy APIs (due to the server's finite sl@0: RAM capacity), and this class or the RSqlBlobReadStream and RSqlBlobWriteStream classes sl@0: may provide the only way to achieve this. sl@0: sl@0: It is strongly recommended to execute calls to the SetL() method within a transaction. sl@0: If a leave occurs during a call to SetL() then the current state of the blob object is sl@0: undefined and a ROLLBACK should be executed to return the blob object to its previous state. sl@0: Note that in order for a ROLLBACK to execute successfully all open RSqlBlobReadStream sl@0: and RSqlBlobWriteStream handles and all open RSqlStatement objects must be closed sl@0: before the ROLLBACK is executed. sl@0: sl@0: When using SetL() to update the content of a zeroblob it is recommended, where possible, sl@0: to create the zeroblob and then call SetL() within the same transaction. sl@0: Otherwise the zeroblob will have to be journalled before being written to. sl@0: sl@0: The following code illustrates typical use cases of this class: sl@0: sl@0: CASE 1 - retrieving the entire content of a large blob. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: CleanupClosePushL(db); sl@0: ; sl@0: HBufC8* wholeBlob = TSqlBlob::GetLC(db, , , ); sl@0: sl@0: CleanupStack::PopAndDestroy(2); // wholeBlob, db sl@0: @endcode sl@0: sl@0: sl@0: CASE 2 - creating a 4Mb blob. sl@0: sl@0: @code sl@0: RSqlDatabase db; sl@0: CleanupClosePushL(db); sl@0: ; sl@0: CleanupStack::PushL(TCleanupItem(&DoRollback, &db)); // rollback function sl@0: TInt err = db.Exec(_L("BEGIN")); sl@0: sl@0: err = db.Exec(_L("INSERT INTO table1 VALUES(99, zeroblob(4194304))")); sl@0: sl@0: sl@0: TSqlBlob::SetL(db, , , buf); sl@0: CleanupStack::Pop(); // TCleanupItem sl@0: err = db.Exec(_L("COMMIT")); // blob data committed to disk sl@0: sl@0: CleanupStack::PopAndDestroy(&db); sl@0: @endcode sl@0: sl@0: @see RSqlBlobReadStream sl@0: @see RSqlBlobWriteStream sl@0: @see RSqlDatabase::LastInsertedRowId() sl@0: @see RSqlStatement::BindZeroBlob() sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: class TSqlBlob sl@0: { sl@0: public: sl@0: IMPORT_C static HBufC8* GetLC(RSqlDatabase& aDb, sl@0: const TDesC& aTableName, sl@0: const TDesC& aColumnName, sl@0: TInt64 aRowId = KSqlLastInsertedRowId, sl@0: const TDesC& aDbName = KNullDesC); sl@0: sl@0: IMPORT_C static TInt Get(RSqlDatabase& aDb, sl@0: const TDesC& aTableName, sl@0: const TDesC& aColumnName, sl@0: TDes8& aBuffer, sl@0: TInt64 aRowId = KSqlLastInsertedRowId, sl@0: const TDesC& aDbName = KNullDesC); sl@0: sl@0: IMPORT_C static void SetL(RSqlDatabase& aDb, sl@0: const TDesC& aTableName, sl@0: const TDesC& aColumnName, sl@0: const TDesC8& aData, sl@0: TInt64 aRowId = KSqlLastInsertedRowId, sl@0: const TDesC& aDbName = KNullDesC); sl@0: }; sl@0: sl@0: /** sl@0: Defines a set of categories for the values returned by the SQL API. sl@0: sl@0: A call to an SQL API may complete with a non-zero return code indicating that some sl@0: unexpected behaviour has occurred. This can be categorised in a number of ways, sl@0: for example, as a Symbian OS error, or as a database error etc. sl@0: sl@0: Callers to the SQL API may not want to be concerned with the detailed meaning of sl@0: a specific return code value, and may find it sufficient just to know the category sl@0: of the error. sl@0: sl@0: The category associated with a specific return code can be found by passing the sl@0: return code value to the function SqlRetCodeClass(). sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: enum TSqlRetCodeClass sl@0: { sl@0: /** sl@0: Indicates that a return code is just for information. sl@0: sl@0: This category corresponds to the SQL API return codes: KSqlAtRow and KSqlAtEnd. sl@0: sl@0: @see SqlRetCodeClass() sl@0: @see TSqlRetCodeClass sl@0: @see KSqlAtRow sl@0: @see KSqlAtEnd sl@0: */ sl@0: ESqlInformation, sl@0: sl@0: /** sl@0: Indicates that a return code represents a database-specific error. sl@0: sl@0: This category corresponds to SQL API return codes in the range KSqlErrGeneral to KSqlErrStmtExpired. sl@0: sl@0: @see SqlRetCodeClass() sl@0: @see TSqlRetCodeClass sl@0: @see KSqlErrGeneral sl@0: @see KSqlErrStmtExpired sl@0: */ sl@0: ESqlDbError, sl@0: sl@0: /** sl@0: Indicates that a return code represents a Symbian OS error. sl@0: sl@0: This category corresponds to SQL API return codes in the range KErrPermissionDenied to KErrNone, sl@0: sl@0: @see SqlRetCodeClass() sl@0: @see TSqlRetCodeClass sl@0: @see KErrPermissionDenied sl@0: @see KErrNone sl@0: */ sl@0: ESqlOsError sl@0: }; sl@0: sl@0: /** sl@0: An information type return code from a call to RSqlStatement::Next(). sl@0: sl@0: It means that the RSqlStatement object points to a valid row, and that sl@0: the user can access the column data using the appropriate RSqlStatement sl@0: member functions. sl@0: sl@0: @see RSqlStatement::Next() sl@0: @see RSqlStatement sl@0: @see ESqlInformation sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlAtRow = 1; sl@0: sl@0: /** sl@0: An information type return code from a call to RSqlStatement::Next(). sl@0: sl@0: It means that the RSqlStatement object does not point to a valid row, sl@0: and that column data accessors cannot be used. sl@0: sl@0: @see RSqlStatement::Next() sl@0: @see RSqlStatement sl@0: @see ESqlInformation sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlAtEnd = 2; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates a general SQL error or a missing database. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrGeneral = -311; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates an internal logic error in the SQL database engine, and specifically sl@0: that an internal consistency check within the SQL database engine has failed. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrInternal = -312; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that access permission has been denied. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrPermission = -313; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates an internal logic error in the SQL database engine, and specifically sl@0: that a callback routine requested an abort. sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrAbort = -314; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that the database file is locked. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrBusy = -315; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that a table in the database is locked. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrLocked = -316; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates an attempt to write to a database that is read-only. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrReadOnly = -318; sl@0: sl@0: /** sl@0: SQL database-specific error type. Operation terminated. sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrInterrupt = -319; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that a disk I/O error has occurred. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrIO = -320; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that the database disk image is malformed. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrCorrupt = -321; sl@0: sl@0: /** sl@0: SQL database-specific error type. Table or record not found. sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrNotFound = -322; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that an insertion operation has failed because an autoincrement column used up sl@0: all awailable rowids. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrFull = -323; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates a failure to open the database file. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrCantOpen = -324; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates a database lock protocol error. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrProtocol = -325; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that the database is empty. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrEmpty = -326; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that a prepared SQL statement is no longer valid sl@0: and cannot be executed. sl@0: sl@0: The most common reason for this return code is that the database schema was modified after sl@0: the SQL statement was prepared. The SQL statement must be prepared again sl@0: using the RSqlStatement::Prepare() member functions. sl@0: sl@0: Another possible reason for this return code is a detached database. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrSchema = -327; sl@0: sl@0: /** sl@0: SQL database-specific error type. Too much data for one row. sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrTooBig = -328; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates an abort due to constraint violation. sl@0: sl@0: "Constraint violation" means violation of one or more column constraints ("NOT NULL", "PRIMARY KEY", sl@0: "UNIQUE", "CHECK", "DEFAULT", "COLLATE" SQL keywords) or table constraints ("PRIMARY KEY", "UNIQUE", sl@0: "CHECK" SQL keywords). sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrConstraint = -329; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates a data type mismatch. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrMismatch = -330; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates an internal logic error in the SQL database engine. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrMisuse = -331; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that a parameter index value is out of range. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrRange = -335; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that the file that has been opened is not a database file. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrNotDb = -336; sl@0: sl@0: /** sl@0: An SQL database-specific error type return code from a call to the SQL API. sl@0: sl@0: It indicates that an SQL statement has expired, and needs to be prepared again. sl@0: sl@0: @see RSqlStatement sl@0: @see ESqlDbError sl@0: @see TSqlRetCodeClass sl@0: sl@0: @publishedAll sl@0: @released sl@0: */ sl@0: const TInt KSqlErrStmtExpired = -360; sl@0: sl@0: IMPORT_C TSqlRetCodeClass SqlRetCodeClass(TInt aSqlRetCode); sl@0: sl@0: #endif //__SQLDB_H__