creator/src/creator_fileelement.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 14 Sep 2010 21:20:12 +0300
branchRCL_3
changeset 49 f8280f3bfeb7
parent 0 d6fe6244b863
child 50 9b2cffad4b5e
permissions -rw-r--r--
Revision: 201033 Kit: 201035

/*
* Copyright (c) 2008 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 "creator_fileelement.h"
#include <DrmPermission.h>
#include <DrmConstraint.h>
#include "creator_traces.h"
#include "creator_file.h"

using namespace creatorfile;

// 15 minutes per time zone, 60 seconds per minute
const TInt KSecondsPerTimeZone = 900;
const TInt KMinuteInMicroseconds = 60000000;
const TInt KTimeZoneIncrement = 15;

/*
 * 
 */
CCreatorFileElement* CCreatorFileElement::NewL(CCreatorEngine* aEngine, const TDesC& aName, const TDesC& aContext )
    {
    CCreatorFileElement* self = new (ELeave) CCreatorFileElement(aEngine);
    CleanupStack::PushL(self);
    self->ConstructL(aName, aContext);
    CleanupStack::Pop(self);
    return self;
    }
/*
 * 
 */
CCreatorFileElement::CCreatorFileElement(CCreatorEngine* aEngine) 
: 
CCreatorScriptElement(aEngine)
    {
    iIsCommandElement = ETrue;

    TTime currUniversal;
    TTime currLocal;
    TInt64 result( 0 );
    currUniversal.UniversalTime();
    currLocal.HomeTime();
    result = currLocal.Int64() - currUniversal.Int64();
    result /= KMinuteInMicroseconds;
    result /= KTimeZoneIncrement;
    iTimeZone = I64INT( result );
    }
/*
 * 
 */
void CCreatorFileElement::ExecuteCommandL()
    {
    const CCreatorScriptAttribute* amountAttr = FindAttributeByName(KAmount);
    TInt fileAmount = 1;    
    if( amountAttr )
        {
        fileAmount = ConvertStrToIntL(amountAttr->Value());
        }
    // Get 'fields' element 
    CCreatorScriptElement* fieldsElement = FindSubElement(KFields);
    if( fieldsElement && fieldsElement->SubElements().Count() > 0)
        {
        // Get sub-elements
        const RPointerArray<CCreatorScriptElement>& fields = fieldsElement->SubElements();
        // Create files, the amount of files is defined by fileAmount:
        for( TInt cI = 0; cI < fileAmount; ++cI )
            {
            CFilesParameters* param = new (ELeave) CFilesParameters;
            CleanupStack::PushL( param );
            
            for( TInt i = 0; i < fields.Count(); ++i )
                {
                CCreatorScriptElement* field = fields[i];
                TPtrC elemName = field->Name();
                TPtrC elemContent = field->Content();

                const CCreatorScriptAttribute* randomAttr = field->FindAttributeByName( KRandomLength );
                
                if( elemName == KType )
                    {
                    param->iFileCommand = GetFileCommandL(elemContent, randomAttr || elemContent.Length() == 0);
                    }
                else if( elemName == KDirectory )
                    {
                    delete param->iFullFilePath;
                    param->iFullFilePath = 0;
                    if( randomAttr == 0 && elemContent.Length() > 0 )
                    	{
                    	param->iFullFilePath = elemContent.AllocL();
                    	}
                    else
                        {
                        TFileName defaultPath;
                        iEngine->SetDefaultPathForFileCommandL( param->iFileCommand, defaultPath );
                        param->iFullFilePath = defaultPath.AllocL();
                        }
                    }
                else if ( elemName == KEncryption )
                    {
                    ParseDRMElementsL( field, param );
                    }
                }

            if ( !param->iFileCommand )
                {
                param->iFileCommand = ECmdCreateFileEntryEmptyFolder;
                }
            iEngine->AppendToCommandArrayL( param->iFileCommand, param);
            CleanupStack::Pop( param );
            }
        }
    else
    	{
    	_LIT(KDummy, "");
    	for( TInt cI = 0; cI < fileAmount; ++cI )
    		{
    		CFilesParameters* param = new (ELeave) CFilesParameters;
    		CleanupStack::PushL(param);
    		delete param->iFullFilePath;
    		param->iFullFilePath = 0;
    		TFileName defaultPath;
    		param->iFileCommand = GetFileCommandL(KDummy, ETrue);
    		iEngine->SetDefaultPathForFileCommandL(param->iFileCommand, defaultPath);                
    		param->iFullFilePath = defaultPath.AllocL();
    		
    		iEngine->AppendToCommandArrayL(param->iFileCommand, param);
    		CleanupStack::Pop(param);
    		}
    	}
    }

TInt CCreatorFileElement::GetFileCommandL( const TDesC& aFileIdStr, TBool aRandom ) const
    {
    TInt mapCount = sizeof(FileMap) / sizeof(FileMapping);
    
    if( aRandom )
        {
        return FileMap[iEngine->RandomNumber(0, mapCount-1)].CommandId();
        }
    
    for( TInt i = 0; i < mapCount; ++i )
        {
        if( CompareIgnoreCase(FileMap[i].FileName(), aFileIdStr) == 0 )
            return FileMap[i].CommandId();
        }
    LOGSTRING2("CCreatorFileElement::GetFileCommandL: Unknown file id: %S", &aFileIdStr);
    User::Leave(KErrGeneral);
    return 0; // Not reached, but disables compiler warning...
    }

void CCreatorFileElement::ParseDRMElementsL( CCreatorScriptElement* aField, CFilesParameters* aParam )
    {
    const CCreatorScriptAttribute* encryptionTypeAttr = aField->FindAttributeByName( KType );
    if ( encryptionTypeAttr && encryptionTypeAttr->Value().Length() > 0 )
        {
        aParam->iEncrypt = ETrue;
        if ( encryptionTypeAttr->Value() == KDRMCD ) 
            {
            delete aParam->iPermission;
            aParam->iPermission = NULL;
            // ownership transferred:
            aParam->iPermission = CDRMPermission::NewL();
            }
        else if ( encryptionTypeAttr->Value() != KDRMFL )
            {
            // must be either KDRMCD or KDRMFL
            User::Leave( KErrArgument );
            }
        }
    else
        {
        User::Leave( KErrArgument );
        }
    
    const RPointerArray<CCreatorScriptElement>& rights = aField->SubElements();
    
    if ( encryptionTypeAttr->Value() == KDRMFL && rights.Count() )
        {
        // Right definitions not allowed with DRM Forward Lock
        // User::Leave( KErrArgument ); // Do not leave here
        return; // Just omit rights definitions
        }
    
    // parse right elements
    CDRMPermission* drmPermission = aParam->iPermission;
    for ( TInt r = 0; r < rights.Count(); r++ )
        {
        const CCreatorScriptAttribute* rightAttr = rights[r]->FindAttributeByName( KType );
        if ( rightAttr->Value() == KDRMPlayRight &&
             !( drmPermission->iAvailableRights & ERightsPlay ) )
            {
            drmPermission->iAvailableRights |= ERightsPlay;
            ParseDRMConstraintsL( rights[r], drmPermission->iPlay );
            }
        else if ( rightAttr->Value() == KDRMDisplayRight  &&
                !( drmPermission->iAvailableRights & ERightsDisplay ))
            {
            drmPermission->iAvailableRights |= ERightsDisplay;
            ParseDRMConstraintsL( rights[r], drmPermission->iDisplay );
            }
        else if ( rightAttr->Value() == KDRMPrintRight  &&
                !( drmPermission->iAvailableRights & ERightsPrint ) )
            {
            drmPermission->iAvailableRights |= ERightsPrint;
            ParseDRMConstraintsL( rights[r], drmPermission->iPrint );
            }
        else if ( rightAttr->Value() == KDRMExecuteRight  &&
                !( drmPermission->iAvailableRights & ERightsExecute ) )
            {
            drmPermission->iAvailableRights |= ERightsExecute;
            ParseDRMConstraintsL( rights[r], drmPermission->iExecute );
            }
        }
    }

void CCreatorFileElement::ParseDRMConstraintsL( CCreatorScriptElement* aRight, CDRMConstraint* aConstraint )
    {
    const RPointerArray<CCreatorScriptElement>& constraints = aRight->SubElements();
    for ( TInt c = 0; c < constraints.Count(); c++ )
        {
        TPtrC elemName = constraints[c]->Name();
        TPtrC elemContent = constraints[c]->Content();
        if ( elemName == KDRMCount )
            {
            TInt count = ConvertStrToIntL( elemContent );
            // apply count constraint
            aConstraint->iActiveConstraints |= EConstraintCounter;
            aConstraint->iCounter = count;
            aConstraint->iOriginalCounter = count;
            }
        else if ( elemName == KDRMInterval )
            {
            // apply interval constraint           
            aConstraint->iActiveConstraints |= EConstraintInterval;
            aConstraint->iInterval = ParseTimeInterval( elemContent );
            aConstraint->iIntervalStart = Time::NullTTime();
            }
        else if ( elemName == KDRMStartTime )
            {
            // apply start time constraint           
            aConstraint->iActiveConstraints |= EConstraintStartTime;
            TTimeIntervalSeconds offset( iTimeZone * KSecondsPerTimeZone );
            aConstraint->iStartTime = ConvertToDateTimeL( elemContent ) - offset;
            }
        else if ( elemName == KDRMEndTime )
            {
            // apply end time constraint
            aConstraint->iActiveConstraints |= EConstraintEndTime;
            TTimeIntervalSeconds offset( iTimeZone * KSecondsPerTimeZone );
            aConstraint->iEndTime = ConvertToDateTimeL( elemContent ) - offset;
            }
        else if ( elemName == KDRMAccumulated )
            {
            // apply accumulated time constraint
            aConstraint->iActiveConstraints |= EConstraintAccumulated;
            if ( aConstraint->iEndTime == Time::NullTTime() )
                {
                aConstraint->iEndTime = Time::MaxTTime();
                }
            if ( aConstraint->iStartTime == Time::NullTTime() )
                {
                aConstraint->iStartTime = Time::MinTTime();
                }
            aConstraint->iAccumulatedTime = ParseTimeInterval( elemContent );
            }
        }
    }

TTimeIntervalSeconds CCreatorFileElement::ParseTimeInterval( TDesC& aTimeString )
    {
    // Iso8601 format (P1Y2M3DT4H5M6S) to TimeIntervalSeconds
    TLex lex;
    TInt year = 0;
    TInt month = 0;
    TInt day = 0;
    TInt hour = 0;
    TInt minute = 0;
    TInt second = 0;

    if (aTimeString.Length() > 0)
        {    
        lex = aTimeString;
        lex.Inc(); // skip 'P'
        lex.Val(year);
        lex.Inc();
        lex.Val(month);
        lex.Inc();
        lex.Val(day);
        lex.Inc(2);
        lex.Val(hour);
        lex.Inc();
        lex.Val(minute);
        lex.Inc();
        lex.Val(second);
        }
    TTimeIntervalSeconds result( ( ( ( year * 365 + month * 30 + day ) * 24 + hour )
                * 60 + minute ) * 60 + second );
    return result;
    }

// End of file