harvesterplugins/bookmarks/src/bookmarksplugin.cpp
author Pat Downey <patd@symbian.org>
Wed, 23 Jun 2010 17:22:18 +0100
changeset 8 2f67eb14d003
parent 5 3bc31ad99ee7
child 9 4a2987baf8f7
permissions -rw-r--r--
Merge heads.

/*
* Copyright (c) 2010 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:  Bookmarks harvester plugin implementation
*
*/

#include "bookmarksplugin.h"

#include <FavouritesItemList.h>
#include <ActiveFavouritesDbNotifier.h>
#include <e32base.h> 
#include <uri8.h> //For parsing URL names.

#include "harvesterserverlogger.h"
#include "common.h"
#include "csearchdocument.h"
#include "ccpixindexer.h"
#include "OstTraceDefinitions.h"
#ifdef OST_TRACE_COMPILER_IN_USE
#include "bookmarkspluginTraces.h"
#endif


/** The milliseconds delay between harvesting chunks. */
const TInt KHarvestingDelay = 1000;
/** Number of bookmarks to process in one active scheduler cycle */
const TInt KBookmarksPerRunL = 1;
//_LIT(KExcerptDelimiter, " ");
/** Bookmark list size for dynamic array*/
const TInt KBookmarkListSize = 100;

// -----------------------------------------------------------------------------
CBookmarksPlugin* CBookmarksPlugin::NewL()
	{
	CBookmarksPlugin* instance = CBookmarksPlugin::NewLC();
    CleanupStack::Pop(instance);
    return instance;
	}

// -----------------------------------------------------------------------------
CBookmarksPlugin* CBookmarksPlugin::NewLC()
	{
	CBookmarksPlugin* instance = new (ELeave) CBookmarksPlugin();
    CleanupStack::PushL(instance);
    instance->ConstructL();
    return instance;
	}

// -----------------------------------------------------------------------------
CBookmarksPlugin::CBookmarksPlugin()
	{
	//No implementation required. 
	//Necessary things done in ConstructL()
	}

// -----------------------------------------------------------------------------
CBookmarksPlugin::~CBookmarksPlugin()
	{
	if (iAsynchronizer)
        iAsynchronizer->CancelCallback();
	iFavoritesNotifier->Cancel();
	delete iFavoritesNotifier;
	delete iAsynchronizer;
	delete iIndexer;
	delete iArrUidsCurrentBookmarkList;
	iFavouritesDb.Close();
	iFavouritesSession.Close();
	}
	
// -----------------------------------------------------------------------------
void CBookmarksPlugin::ConstructL()
	{
	iArrUidsCurrentBookmarkList = new(ELeave) CArrayFixFlat<TInt>( KBookmarkListSize );
	User::LeaveIfError( iFavouritesSession.Connect() );
	User::LeaveIfError( iFavouritesDb.Open( iFavouritesSession, KBrowserBookmarks ) );
	iFavoritesNotifier = new ( ELeave ) CActiveFavouritesDbNotifier( iFavouritesDb, *this );
    iAsynchronizer = CDelayedCallback::NewL( CActive::EPriorityIdle );
	}

// -----------------------------------------------------------------------------
void CBookmarksPlugin::StartPluginL()
	{
    iFavoritesNotifier->Start(); //Start the notifier. Guarenteed to be the first call to Start. So no check for IsActive().
	// Define this base application class, use default location
	User::LeaveIfError(iSearchSession.DefineVolume( _L(BOOKMARK_QBASEAPPCLASS), KNullDesC ));
	// Open database
	iIndexer = CCPixIndexer::NewL( iSearchSession );
	iIndexer->OpenDatabaseL( _L(BOOKMARK_QBASEAPPCLASS) ); 
	// Start harvester for this plugin
	iObserver->AddHarvestingQueue(this, iIndexer->GetBaseAppClass() );
	}

// -----------------------------------------------------------------------------
void CBookmarksPlugin::StartHarvestingL( const TDesC& /*aMedia*/ )
    {
    OstTrace0( TRACE_NORMAL, CBOOKMARKSPLUGIN_STARTHARVESTINGL, "StartHarvestingL: resetting database" );
    CPIXLOGSTRING("StartHarvestingL: resetting database");
    iIndexer->ResetL();//reset any indexes if exist already
    iCurrentIndex = 0; //Initialize to zero as it is started
    iFavouritesDb.Count( iCurrentCount );
    OstTrace1( TRACE_NORMAL, DUP1_CBOOKMARKSPLUGIN_STARTHARVESTINGL, "StartHarvestingL::Current count=%d", iCurrentCount );
    CPIXLOGSTRING2("StartHarvestingL::Current count = %d.", iCurrentCount);
    //Get all the UID list from current database
    //GetUids() appends items to the list. So, reset it first.
    //From the documentation of GetUids(): "Existing items remain (new ones appended)"
    iArrUidsCurrentBookmarkList->Reset();
    iFavouritesDb.GetUids( *iArrUidsCurrentBookmarkList );
#ifdef __PERFORMANCE_DATA
    iStartTime.UniversalTime();
#endif
  	iAsynchronizer->Start( 0, this, KHarvestingDelay ); //Guarenteed to be the first call to Start. So no checking for IsActive().
    }

// -----------------------------------------------------------------------------
void CBookmarksPlugin::DelayedCallbackL( TInt /*aCode*/ )
    {
    // Harvest items on each call

    // Read the next set of bookmark.
    for( TInt i = 0; i < KBookmarksPerRunL; i++ )
        {
        // Exit the loop if no more bookmarks
        if (iCurrentIndex >= iCurrentCount)
            break;
        
        //Create index item
        OstTraceExt2( TRACE_NORMAL, CBOOKMARKSPLUGIN_DELAYEDCALLBACKL, "CBookmarksPlugin::DelayedCallbackL;Harvesting id=%d;BookmarkUid=%d", iCurrentIndex, iArrUidsCurrentBookmarkList->At(iCurrentIndex) );
        CPIXLOGSTRING3("CBookmarksPlugin::DelayedCallbackL(): Harvesting id=%d, BookmarkUid = %d.", iCurrentIndex, iArrUidsCurrentBookmarkList->At(iCurrentIndex) );
        //Create new bookmark document and add
        CreateBookmarksIndexItemL(iArrUidsCurrentBookmarkList->At(iCurrentIndex),ECPixAddAction);
        iCurrentIndex++;
        }

    if( iAsynchronizer && (iCurrentIndex < iCurrentCount) )
        {
        // Launch the next RunL
        OstTrace1( TRACE_NORMAL, DUP1_CBOOKMARKSPLUGIN_DELAYEDCALLBACKL, "CBookmarksPlugin::DelayedCallbackL;scheduling item count=%d", iCurrentIndex );
        CPIXLOGSTRING2("CBookmarksPlugin::DelayedCallbackL(): scheduling item count: %d.", iCurrentIndex );
        iAsynchronizer->Start(0, this, KHarvestingDelay);
        OstTrace1( TRACE_NORMAL, DUP2_CBOOKMARKSPLUGIN_DELAYEDCALLBACKL, "CBookmarksPlugin::DelayedCallbackL;DONE scheduling item count=%d", iCurrentIndex );
        CPIXLOGSTRING2("CBookmarksPlugin::DelayedCallbackL(): DONE scheduling item count: %d.", iCurrentIndex );
        }
    else
        {
        // Harvesting was successfully completed
        Flush(*iIndexer);
#ifdef __PERFORMANCE_DATA
    UpdatePerformaceDataL();
#endif
        iObserver->HarvestingCompleted(this, iIndexer->GetBaseAppClass(), KErrNone);
        OstTrace0( TRACE_NORMAL, DUP3_CBOOKMARKSPLUGIN_DELAYEDCALLBACKL, "CBookmarksPlugin::DelayedCallbackL(): Harvesting complete" );
        CPIXLOGSTRING("CBookmarksPlugin::DelayedCallbackL(): Harvesting complete");
        }
	}

// -----------------------------------------------------------------------------
void CBookmarksPlugin::DelayedError( TInt aCode )
    {
    Flush(*iIndexer);
    iObserver->HarvestingCompleted(this, iIndexer->GetBaseAppClass(), aCode);
	}

// -----------------------------------------------------------------------------
/**
 * If input is www2.google.com, this return google.com.
 * If input is www2.google.com/someting, this returns google.com.
 *
 * @param aUrl URL from which to extract domain name.
 * @param aDomain descriptor pointer in which domain will be returned.
 *
 * @note Leaves in case of error with KErrArgument
 * @example
 * TPtrC domain;
 * GetDomainNameLC( myUrl, domain );
 */
void GetDomainNameL( const TDesC& aUrl, TPtrC& aDomain )
    {
    OstTraceExt1( TRACE_NORMAL, CBOOKMARKSPLUGIN_GETDOMAINNAMEL, "CBookmarksPlugin::GetDomainNameL();URL=%S", aUrl );
    CPIXLOGSTRING2("CBookmarksPlugin::GetDomainNameL(): URL = %S", &aUrl );
    //Convert to 8-bit descriptors.
    HBufC8* url8 = HBufC8::NewLC( aUrl.Length() );
    url8->Des().Copy( aUrl );
    
    TUriParser8  uriParser;
    User::LeaveIfError( uriParser.Parse( *url8 ) );

    //Get the host name
    //This returns www.google.com if URL is http://www.google.com/something...
    const TDesC8& host = uriParser.Extract( EUriHost );
    if( host.Length() == 0  )
        {
        aDomain.Set( aUrl );
        CleanupStack::PopAndDestroy( url8 );
        return;
        }
    TLex8 lex( host );
    //Check if starts with www.
    TChar currentChar = 0;
    int wwwCount = 0;
    do{
        if( ( currentChar = lex.Get() ) != 0 ) wwwCount++;
    }while( currentChar == TChar('w') || currentChar == TChar('W') );
    
    if( currentChar != 0 )
        {
        lex.UnGet();
        wwwCount--;
        }

    HBufC* domain16 = NULL; 
    //if www34.google.com, skip anything after www, till '.'
    if( 3 == wwwCount )
        {
        while( lex.Peek() != '.' ) lex.Inc();
        //Now stopped at .google.com. So remove '.'
        lex.Inc();
        //The rest is the domain!
        const TDesC8& domain = lex.Remainder();
        if ( domain.Length() == 0 )
            {
            User::Leave( KErrArgument );
            }
        //Convert to 16-bit descriptors.
        domain16 = HBufC::NewLC( domain.Length() );
        domain16->Des().Copy( domain );
        }
    else 
        {
        domain16 = HBufC::NewLC( host.Length() );
        domain16->Des().Copy( host );
        }
    aDomain.Set( *domain16 );
    CleanupStack::Pop( domain16 );
    CleanupStack::PopAndDestroy( url8 );
    OstTraceExt1( TRACE_NORMAL, DUP1_CBOOKMARKSPLUGIN_GETDOMAINNAMEL, "CBookmarksPlugin::GetDomainNameL();domain=%S", aDomain );
    CPIXLOGSTRING2("CBookmarksPlugin::GetDomainNameL(): domain = %S", &aDomain );
    }

// -----------------------------------------------------------------------------
void CBookmarksPlugin::DoIndexingL(CFavouritesItem*& aItem, const TDesC& aDocidStr, TCPixActionType& aActionType)
    {
    OstTraceFunctionEntry0( CBOOKMARKSPLUGIN_DOINDEXINGL_ENTRY );
    CPIXLOGSTRING("CBookmarksPlugin::DoIndexingL(): Entering");
    if(aItem->Type() == CFavouritesItem::EItem ) //Store only Items not folders
        {
        OstTraceExt2( TRACE_NORMAL, CBOOKMARKSPLUGIN_DOINDEXINGL, "CBookmarksPlugin::DoIndexingL();url=%S;name=%S", (aItem->Url()), (aItem->Name()) );
        CPIXLOGSTRING3("CBookmarksPlugin::DoIndexingL(): url = %S ,name = %S", &(aItem->Url()), &(aItem->Name()));
        CSearchDocument* index_item = CSearchDocument::NewLC(aDocidStr, _L(BOOKMARKAPPCLASS));
        index_item->AddFieldL(KMimeTypeField, KMimeTypeBookmark, CDocumentField::EStoreYes | CDocumentField::EIndexUnTokenized);
        if(aItem->Name() != KNullDesC)
            index_item->AddFieldL(KBookMarkFieldName,aItem->Name(),CDocumentField::EStoreYes | CDocumentField::EIndexTokenized);

        TPtrC domain;
        if(aItem->Url() != KNullDesC)
            {
            index_item->AddFieldL(KBookMarkUrl, aItem->Url(), CDocumentField::EStoreYes | CDocumentField::EIndexTokenized);
            GetDomainNameL( aItem->Url(), domain );
            index_item->AddFieldL(KBookMarkDomain, domain , CDocumentField::EStoreYes | CDocumentField::EIndexTokenized);
            OstTraceExt1( TRACE_NORMAL, DUP1_CBOOKMARKSPLUGIN_DOINDEXINGL, "CBookmarksPlugin::DoIndexingL();domain=%S", domain );
            CPIXLOGSTRING2("CBookmarksPlugin::DoIndexingL(): domain = %S", &domain );
            }
        
        //For bookmarks, only URL should go into exceprt field.
        //For more info, check the appclass-hierarchy.txt

        //Add Excerpt as it is must have field. What should be excerpt in bookmarks ?
        HBufC* excerpt = HBufC::NewLC( aItem->Url().Length() + 1 );
        TPtr ptr = excerpt->Des();
        ptr.Append( aItem->Url() );
        index_item->AddExcerptL( *excerpt );
        CleanupStack::PopAndDestroy( excerpt );

        // Send for indexing
        TRAPD(err, iIndexer->AddL(*index_item));
        if (err == KErrNone)
            {
                OstTrace0( TRACE_NORMAL, DUP2_CBOOKMARKSPLUGIN_DOINDEXINGL, "CBookmarksPlugin::DoIndexingL(): Added." );
                CPIXLOGSTRING("CBookmarksPlugin::DoIndexingL(): Added.");
                }
            else
                {
                OstTrace1( TRACE_NORMAL, DUP3_CBOOKMARKSPLUGIN_DOINDEXINGL, "CBookmarksPlugin::DoIndexingL();Error %d in adding", err );
                CPIXLOGSTRING2("CBookmarksPlugin::DoIndexingL(): Error %d in adding.", err);
                }           
        OstTrace0( TRACE_NORMAL, DUP4_CBOOKMARKSPLUGIN_DOINDEXINGL, "CBookmarksPlugin::DoIndexingL(): Logic complete" );
        CPIXLOGSTRING("CBookmarksPlugin::DoIndexingL(): Logic complete");
        CleanupStack::PopAndDestroy( index_item );
        OstTrace0( TRACE_NORMAL, DUP5_CBOOKMARKSPLUGIN_DOINDEXINGL, "CBookmarksPlugin::DoIndexingL(): Pop complete complete" );
        CPIXLOGSTRING("CBookmarksPlugin::DoIndexingL(): Pop complete complete");
        }
    CPIXLOGSTRING("CBookmarksPlugin::DoIndexingL(): Returning");
    OstTraceFunctionExit0( CBOOKMARKSPLUGIN_DOINDEXINGL_EXIT );
    }

// -----------------------------------------------------------------------------
void CBookmarksPlugin::CreateBookmarksIndexItemL(TInt aBookMarkUid, TCPixActionType aActionType)
    {
    //@sai: CTC: Will never be null in normal usecase. Needs to be present for UTs.
    if( !iIndexer )
        return;
    OstTrace1( TRACE_NORMAL, CBOOKMARKSPLUGIN_CREATEBOOKMARKSINDEXITEML, "CBookmarksPlugin::CreateBookmarkIndexItemL();aBookMarkUid=%d", aBookMarkUid );
    CPIXLOGSTRING2("CBookmarksPlugin::CreateBookmarkIndexItemL(): aBookMarkUid = %d ", aBookMarkUid );
    // creating CSearchDocument object with unique ID for this application
    TBuf<20> docid_str;
    docid_str.AppendNum(aBookMarkUid);
    //Add or update actions.
    CFavouritesItem* item = CFavouritesItem::NewLC();
    TInt err = iFavouritesDb.Get(aBookMarkUid,*item);
    OstTrace1( TRACE_NORMAL, DUP1_CBOOKMARKSPLUGIN_CREATEBOOKMARKSINDEXITEML, "CBookmarksPlugin::CreateBookmarkIndexItemL();DB Get error=%d", err );
    CPIXLOGSTRING2("CBookmarksPlugin::CreateBookmarkIndexItemL(): DB Get error = %d ", err );
    //@sai: CTC: did not ever get 'false' for this check. Since this is a private function,
    // we cannot UT this with an invalid bookmarkUid.
    if(KErrNone != err)
        {
        CleanupStack::PopAndDestroy(item);
        return;
        }
    OstTrace1( TRACE_NORMAL, DUP2_CBOOKMARKSPLUGIN_CREATEBOOKMARKSINDEXITEML, "CBookmarksPlugin::CreateBookmarkIndexItemL();item->Type()=%d", item->Type() );
    CPIXLOGSTRING2("CBookmarksPlugin::CreateBookmarkIndexItemL(): item->Type() = %d ", item->Type() );
    DoIndexingL(item, docid_str, aActionType); //add to / update index
    CleanupStack::PopAndDestroy(item);
    }

// -----------------------------------------------------------------------------
void CBookmarksPlugin::HandleFavouritesDbEventL( RDbNotifier::TEvent aEvent )
    {
    //Since we cannot efficiently identify updated item, it makes most
    //sense to simply re-index the bookmarks - their numbers are not 
    //prohibitively large.
    //@sai:CTC: the DB does not seem to be sending any other event.
    if( RDbNotifier::ECommit == aEvent )
        {
        if( iAsynchronizer->CallbackPending() )
            {
            OstTrace0( TRACE_NORMAL, CBOOKMARKSPLUGIN_HANDLEFAVOURITESDBEVENTL, "HandleFavouritesDbEventL: Cancelling callback" );
            CPIXLOGSTRING("HandleFavouritesDbEventL: Cancelling callback");
            iAsynchronizer->CancelCallback(); //first cancel any ongoing harvesting.
            }
        OstTrace0( TRACE_NORMAL, DUP1_CBOOKMARKSPLUGIN_HANDLEFAVOURITESDBEVENTL, "HandleFavouritesDbEventL: calling StartHarvestingL" );
        CPIXLOGSTRING("HandleFavouritesDbEventL: calling StartHarvestingL");
        StartHarvestingL( KNullDesC );
        }
    }
	
// -----------------------------------------------------------------------------
#ifdef __PERFORMANCE_DATA
void  CBookmarksPlugin::UpdatePerformaceDataL()
    {
    TTime now;
   
    
    iCompleteTime.UniversalTime();
    TTimeIntervalMicroSeconds timeDiff = iCompleteTime.MicroSecondsFrom(iStartTime);
    
    RFs fileSession;
    RFile perfFile;
    User::LeaveIfError( fileSession.Connect () );
    
    
    /* Open file if it exists, otherwise create it and write content in it */
    
        if(perfFile.Open(fileSession, _L("c:\\data\\BookmarksPerf.txt"), EFileWrite))
                   User::LeaveIfError(perfFile.Create (fileSession, _L("c:\\data\\BookmarksPerf.txt"), EFileWrite));
    
    HBufC8 *heap = HBufC8::NewL(100);
    TPtr8 ptr = heap->Des();
    now.HomeTime();
    TBuf<50> timeString;             
                
    _LIT(KOwnTimeFormat,"%:0%H%:1%T%:2%S");
    now.FormatL(timeString,KOwnTimeFormat);
    ptr.AppendNum(now.DateTime().Day());
    ptr.Append(_L("/"));
    ptr.AppendNum(now.DateTime().Month());
    ptr.Append(_L("/"));
    ptr.AppendNum(now.DateTime().Year());
    ptr.Append(_L(":"));
    ptr.Append(timeString);
    ptr.Append( _L(":Ani: Time took for Harvesting Bookmarks is : "));
    ptr.AppendNum(timeDiff.Int64()/1000) ;
    ptr.Append(_L(" MilliSeonds \n"));
    TInt myInt = 0;
    perfFile.Seek(ESeekEnd,myInt);
    perfFile.Write (ptr);
    perfFile.Close ();
    fileSession.Close ();
    delete heap;
    }
#endif
// End of file