Update contrib.
1 // Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
18 // Class CDbTableDatabase::CInterface
21 // ctor. Add the initial reference to the database engine
23 CDbTableDatabase::CInterface::CInterface(CDbTableDatabase& aDatabase)
30 // dtor. remove our reference from the database engine
32 CDbTableDatabase::CInterface::~CInterface()
34 if (Database().Transaction().InTransaction(*this))
35 Rollback(); // release the lock if we have it
39 void CDbTableDatabase::CInterface::PrepareDDLL()
41 Database().Transaction().DDLPrepareL(*this);
45 // destroy the database (database must be empty)
47 TInt CDbTableDatabase::CInterface::Destroy()
49 __ASSERT(Database().Schema().IsEmpty());
50 TRAPD(r,Database().DestroyL());
55 // Initiate a transaction on the database
57 TInt CDbTableDatabase::CInterface::Begin()
59 TRAPD(r,Database().Transaction().BeginL(*this));
64 // Commit a transaction on the database
66 TInt CDbTableDatabase::CInterface::Commit()
68 TRAPD(r,Database().Transaction().CommitL(*this));
73 // Rollback a failed transaction on the database
75 void CDbTableDatabase::CInterface::Rollback()
77 Database().Transaction().Rollback(*this);
81 // Report a requested property
83 TInt CDbTableDatabase::CInterface::Property(CDbDatabase::TProperty aProperty)
85 if (aProperty==EInTransaction)
86 return Database().Transaction().InTransaction(*this) ? 1 : 0;
88 return Database().Property(aProperty);
91 void CDbTableDatabase::CInterface::CreateTableL(const TDesC& aName,const CDbColSet& aColSet,const CDbKey* aPrimaryKey)
94 Database().DoCreateTableL(aName,aColSet,aPrimaryKey);
97 CDbCursor* CDbTableDatabase::CInterface::PrepareViewL(const TDbQuery& aQuery,const TDbWindow& aWindow,RDbRowSet::TAccess anAccess)
99 return Database().PrepareViewL(aQuery,aWindow,anAccess);
102 CDbCursor* CDbTableDatabase::CInterface::OpenTableL(const TDesC& aName,RDbRowSet::TAccess anAccess)
104 return Database().PrepareTableL(aName,anAccess);
107 CDbIncremental* CDbTableDatabase::CInterface::OpenDropTableL(const TDesC& aTable,TInt& aStep)
110 return Database().OpenDropTableL(aTable,aStep);
113 CDbIncremental* CDbTableDatabase::CInterface::OpenAlterTableL(const TDesC& aTable,const CDbColSet& aNewDef,TInt& aStep)
116 return Database().OpenAlterTableL(aTable,aNewDef,aStep);
119 CDbIncremental* CDbTableDatabase::CInterface::OpenCreateIndexL(const TDesC& aName,const TDesC& aTable,const CDbKey& aKey,TInt& aStep)
122 return Database().OpenCreateIndexL(aName,aTable,aKey,aStep);
125 CDbIncremental* CDbTableDatabase::CInterface::OpenDropIndexL(const TDesC& aName,const TDesC& aTable,TInt& aStep)
128 return Database().OpenDropIndexL(aName,aTable,aStep);
132 // Create a database utility. Check the corresponding property to see if it is viable
134 CDbIncremental* CDbTableDatabase::CInterface::OpenUtilityL(CDbDatabase::TUtility aType,TInt& aStep)
136 if (!Property(TProperty(-aType)))
137 { // nothing to do, or not capable
141 RDbTransaction& t=Database().Transaction();
142 t.UtilityPrepareL(*this);
144 Database().Release(); // must not have any tables open during recovery
145 return CUtility::NewL(t,aType,aStep);
148 CDbIncremental* CDbTableDatabase::CInterface::OpenExecuteL(const TDesC& aSql,TDbTextComparison aComparison,TInt& aInit)
150 switch (Sql::Type(aSql))
155 __LEAVE(KErrArgument);
158 CDbIncremental* inc=Sql::ParseDDLStatementLC(aSql)->ExecuteL(*this,aComparison,aInit);
159 CleanupStack::PopAndDestroy(); // statement
164 CSqlDMLStatement* statement=Sql::ParseDMLStatementLC(aSql);
165 RDbTransaction& t=Database().Transaction();
166 t.DMLCheckL(); // ensure we can open a cursor
167 t.DMLPrepareL(*this);
168 CDbIncremental* inc=statement->ExecuteL(Database(),aComparison,aInit);
169 CleanupStack::PopAndDestroy(); // statement
176 // Open a client-side notifier object
178 CDbNotifier* CDbTableDatabase::CInterface::OpenNotifierL()
180 return Database().Transaction().NotifierL();
184 // list the tables on the database
186 void CDbTableDatabase::CInterface::TablesL(CDbTableNames& aNames)
188 TSglQueIterC<CDbTableDef> iter(Database().SchemaL());
189 for (const CDbTableDef* def;(def=iter++)!=0;)
190 aNames.AddL(def->Name());
194 // build a column set for the table
196 void CDbTableDatabase::CInterface::ColumnsL(CDbColSet& aColSet,const TDesC& aName)
199 const HDbColumnSet& set=Database().SchemaL().FindL(aName).Columns();
200 HDbColumnSet::TIteratorC iter=set.Begin();
201 const HDbColumnSet::TIteratorC end=set.End();
206 } while (++iter<end);
210 // List the indexes on a table
212 void CDbTableDatabase::CInterface::IndexesL(CDbIndexNames& aNames,const TDesC& aTable)
214 TSglQueIterC<CDbTableIndexDef> iter(Database().SchemaL().FindL(aTable).Indexes().AsQue());
215 for (const CDbTableIndexDef* def;(def=iter++)!=0;)
216 aNames.AddL(def->Name());
220 // build a key for the index
222 void CDbTableDatabase::CInterface::KeysL(CDbKey& aKey,const TDesC& aName,const TDesC& aTable)
224 const CDbKey& key=Database().SchemaL().FindL(aTable).Indexes().FindL(aName).Key();
225 TInt max=key.Count();
226 for (TInt ii=0;ii<max;++ii)
232 aKey.SetComparison(key.Comparison());
235 // Class CDbTableDatabase::CSource
238 // ctor. Add a reference to the database engine
240 CDbTableDatabase::CSource::CSource(CDbTableDatabase& aDatabase)
241 :iDatabase(aDatabase)
247 // dtor. remove our reference from the database engine
249 CDbTableDatabase::CSource::~CSource()
255 //SYMBIAN_REMOVE_TRIVIAL_ENCRYPTION version of the method
256 CDbDatabase* CDbTableDatabase::CSource::AuthenticateL()
258 Database().AuthenticateL();
259 return Database().InterfaceL();
263 CDbNotifier* CDbTableDatabase::CSource::OpenNotifierL()
265 return Database().Transaction().NotifierL();
268 // Class CDbTableDatabase
270 EXPORT_C CDbTableDatabase::CDbTableDatabase()
273 iTransaction.Open(*this);
276 EXPORT_C CDbTableDatabase::~CDbTableDatabase()
279 iTransaction.Close();
285 void CDbTableDatabase::Close()
293 // Construct the implementation interface for the database framework
294 // On success, the interface takes an owning reference to this
296 EXPORT_C CDbDatabase* CDbTableDatabase::InterfaceL()
298 return new(ELeave) CInterface(*this);
302 // Construct the implementation interface for the database framework
303 // On success, the interface takes an owning reference to this
305 EXPORT_C CDbSource* CDbTableDatabase::SourceL()
307 return new(ELeave) CSource(*this);
311 // Load the schema if required
313 EXPORT_C RDbTableSchema& CDbTableDatabase::SchemaL()
315 RDbTableSchema& schema=Schema();
316 if (!schema.IsLoaded())
326 // Find any tables which are now idle, and cache them
327 // Caching such a table can result in *any* table being deleted
328 // and so the iteration must restart every time an idle table is cached
330 void CDbTableDatabase::CheckIdle()
332 while (!iTables.IsEmpty())
334 for (TSglQueIter<CDbTable> iter(iTables);;)
336 CDbTable* table=iter++;
337 if (table==0) // no more idle tables
340 { // idle->cache transition can modify the collection
346 // If there are no tables left open, then the database can go idle
351 // Does nothing by default
353 EXPORT_C void CDbTableDatabase::Idle()
358 // default implementation
360 EXPORT_C TInt CDbTableDatabase::Property(CDbDatabase::TProperty aProperty)
363 return 0; // utilities are not available with table database
364 __ASSERT(aProperty!=CDbDatabase::EInTransaction);
365 return KErrNotSupported;
369 // default implementation, should never be invoked
371 EXPORT_C CDbTableDatabase::CStepper* CDbTableDatabase::UtilityL(CDbDatabase::TUtility,TInt&)
377 void CDbTableDatabase::FlushL(TDbLockType aLock)
379 if (aLock>=EDbWriteLock)
381 TSglQueIter<CDbTable> iter(iTables);
382 for (CDbTable* table;(table=iter++)!=0;)
387 void CDbTableDatabase::Abandon(TDbLockType aLock)
389 if (aLock>=EDbWriteLock)
391 TSglQueIter<CDbTable> iter(iTables);
392 for (CDbTable* table;(table=iter++)!=0;)
395 if (aLock==EDbSchemaLock)
397 Release(); // ensure no-one is using the old schema
403 // Discard a table if it is using this definition
405 void CDbTableDatabase::Release(const CDbTableDef& aDef)
407 TSglQueIter<CDbTable> iter(iTables);
408 for (CDbTable* table;(table=iter++)!=0;)
410 if (&table->Def()==&aDef)
419 // Discard all open tables as the schema has been modified or discarded
421 void CDbTableDatabase::Release()
423 TSglQueIter<CDbTable> iter(iTables);
424 for (CDbTable* table;(table=iter++)!=0;)
429 // Remove a table from the open set
431 void CDbTableDatabase::RemoveTable(CDbTable& aTable)
433 iTables.Remove(aTable);
434 if (!Transaction().IsLocked() && iTables.IsEmpty())
438 CDbTableSource* CDbTableDatabase::TableSourceL(const TDesC& aTableName)
440 CDbTableSource* rec=new(ELeave) CDbTableSource;
441 CDbTable* table=iTables.Find(aTableName);
443 table->Open(); // a new reference to it
446 CleanupStack::PushL(rec);
447 table=TableL(SchemaL().FindL(aTableName));
448 CleanupStack::Pop(); // table source
450 rec->Construct(table);
451 iCache.Flush(); // check-point for object cache membership
456 // Prepare the data pipeline
458 CDbCursor* CDbTableDatabase::PrepareViewL(const TDbQuery& aQuery,const TDbWindow& aWindow,RDbRowSet::TAccess aAccess)
460 Transaction().DMLCheckL(); // ensure we can open a cursor
461 CSqlQuery *query=Sql::ParseQueryLC(aQuery.Query());
462 RDbAccessPlan plan(query,aQuery.Comparison());
463 plan.BuildLC(*this,aAccess,aWindow);
464 CDbTableCursor* cursor=CDbTableCursor::NewL(plan,aAccess);
465 CleanupStack::Pop(); // the plan
466 CleanupStack::PopAndDestroy(); // the query
471 // Prepare the data pipeline
473 CDbCursor* CDbTableDatabase::PrepareTableL(const TDesC& aTable,RDbRowSet::TAccess aAccess)
475 Transaction().DMLCheckL(); // ensure we can open a cursor
477 plan.BuildLC(*this,aTable,aAccess);
478 CDbTableCursor* cursor=CDbTableCursor::NewL(plan,aAccess);
479 CleanupStack::Pop(); // the plan
483 void CDbTableDatabase::DoCreateTableL(const TDesC& aName,const CDbColSet& aColSet,const CDbKey* aPrimaryKey)
485 if (SchemaL().Find(aName))
486 __LEAVE(KErrAlreadyExists);
487 // validate the colset set passed in
488 Validate::NameL(aName);
489 Validate::ColSetL(aColSet);
490 Transaction().DDLBeginLC();
491 iSchema.Add(CreateTableL(aName,aColSet,aPrimaryKey));
492 Transaction().DDLCommitL();
493 CleanupStack::Pop(); // rollback not required
497 // Create index definition and prepare to build it
499 CDbTableDatabase::CIncremental* CDbTableDatabase::OpenCreateIndexL(const TDesC& aName,const TDesC& aTable,const CDbKey& aKey,TInt& aStep)
501 CDbTableDef& tDef=SchemaL().FindL(aTable);
502 RDbIndexes& indexes=tDef.Indexes();
503 if (indexes.Find(aName)!=NULL)
504 __LEAVE(KErrAlreadyExists);
505 if (indexes.Count()==KDbTableMaxIndexes)
506 __LEAVE(KErrNotSupported);
507 Validate::NameL(aName);
508 Validate::KeyL(aKey,tDef.Columns());
509 if (aKey.IsPrimary())
510 __LEAVE(KErrArgument); // cannot create a primary index
512 CCreateIndex* builder=CCreateIndex::NewLC(Transaction());
513 CDbTableIndexDef* xDef=CreateIndexL(tDef,aName,aKey);
514 tDef.Indexes().Add(xDef);
515 Release(tDef); // release any table using the old schema
516 aStep=builder->ConstructL(tDef,*xDef);
517 CleanupStack::Pop(); // builder
522 // Remove index from schema first, then remove index data
524 CDbTableDatabase::CIncremental* CDbTableDatabase::OpenDropIndexL(const TDesC& aName,const TDesC& aTable,TInt& aStep)
526 CDbTableDef& tDef=SchemaL().FindL(aTable);
527 CDbTableIndexDef& xDef=tDef.Indexes().FindL(aName);
529 Release(tDef); // release any table using the old schema
530 CDropIndex* drop=CDropIndex::NewL(Transaction(),tDef,&xDef,aStep);
531 tDef.Indexes().Remove(&xDef);
535 CDbTableDatabase::CIncremental* CDbTableDatabase::OpenDropTableL(const TDesC& aTable,TInt& aStep)
537 CDbTableDef& tDef=SchemaL().FindL(aTable);
539 Release(tDef); // release the tables using the old schema
540 CDropTable* drop=CDropTable::NewL(Transaction(),&tDef,aStep);
541 iSchema.Remove(&tDef);
546 // Create an incremental table altering object
548 CDbTableDatabase::CIncremental* CDbTableDatabase::OpenAlterTableL(const TDesC& aTable,const CDbColSet& aNewDef,TInt& aStep)
550 CDbTableDef& tDef=SchemaL().FindL(aTable);
551 Validate::ColSetL(aNewDef);
552 Release(tDef); // release the tables using the old schema
553 return CAlterTable::NewL(Transaction(),tDef,aNewDef,aStep);
557 // Reserved for future development
559 EXPORT_C void CDbTableDatabase::Reserved_1()
564 // Reserved for future development
566 EXPORT_C void CDbTableDatabase::Reserved_2()