contentmgmt/contentaccessfwfordrm/source/caf/agentinfo.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 31 Aug 2010 17:00:08 +0300
branchRCL_3
changeset 61 641f389e9157
parent 8 35751d3474b7
permissions -rw-r--r--
Revision: 201035 Kit: 201035

/*
* Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of the License "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 <ecom/ecom.h>
#include "agentinfo.h"
#include <caf/agent.h>
#include <caf/agentfactory.h>
#include <caf/agentinterface.h>

using namespace ContentAccess;

// Default buffer size of zero
const TInt KCafApparcBufferSize = 0;

CAgentInfo* CAgentInfo::NewLC(const CImplementationInformation& aImplInfo)
	{
	CAgentInfo* self = new (ELeave) CAgentInfo;
	CleanupStack::PushL(self);
	self->ConstructL(aImplInfo);
	return self;
	}

CAgentInfo::CAgentInfo()
	{
	}

CAgentInfo::~CAgentInfo()
	{
	iSupplierMimeTypes.ResetAndDestroy();
	iSupplierMimeTypes.Close();
	iConsumerMimeTypes.ResetAndDestroy();
	iConsumerMimeTypes.Close();
	
	// Delete the agent manager and agent factory, this may unload the agent DLL
	delete iAgentManager;
	delete iAgentFactory;
	}


void CAgentInfo::ConstructL(const CImplementationInformation& aImplInfo)
	{
	// Set up the agent member with the name and Uid of the agent
	iAgent.SetValue(aImplInfo.DisplayName(), aImplInfo.ImplementationUid());

	// get the name of the agents private directory
	TInt length = aImplInfo.OpaqueData().Length();
	if(length > KMaxSIDLength)
		{
		User::Leave(KErrCorrupt);
		}
	else
		{
		iPrivateDirectoryName.Copy(aImplInfo.OpaqueData().Left(KMaxSIDLength));
		}
			
	// Extract from the data field from the info.
	// The data field should be of the form:
	// "<bufferlength>|<supplier1>,<supplier2>,....,<suppliern>:<consumer1>,<consumer2>"
	// bufferlength is the desired length of the buffer passed from apparc to DoRecognize
	TPtrC8 data(aImplInfo.DataType());
	TPtrC8 supplier(KNullDesC8());
	TPtrC8 consumer(KNullDesC8());

	// Search for the "|" field to delimit buffersize and supplier mime types
	TInt index = data.LocateF('|');
	if(index < 0)
		{
		// No default buffersize, ie. corrupt
		User::Leave(KErrCorrupt);
		}
	else
		{
		TPtrC8 buffersize(data.Left(index));
		TLex8 bufferLex(buffersize);
		bufferLex.Val(iPreferredBufferSize);

		// Make sure a buffer size was actually specified before the | character
		if(iPreferredBufferSize == 0)
			{
			iPreferredBufferSize = KCafApparcBufferSize;
			}

		// Make sure '|' is not the last character
		if (index + 1 < data.Length())
			{
			data.Set(aImplInfo.DataType().Mid(index + 1));
			}
		else 
			{
			User::Leave(KErrCorrupt);
			}
		}

	
	// Search for the ":" field to delimit supplier and consumer
	index = data.LocateF(':');
	
	// If the colon was present, then set the pointers appropriately
	if (index >= 0)
		{
		// Set supplier pointer
		supplier.Set(data.Left(index));

		// Check that ':' is not the last character before setting consumer pointer
		if (index + 1 < data.Length())
			{
			consumer.Set(data.Mid(index + 1));
			}
		}
	else
		{
		User::Leave(KErrCorrupt);
		}

	// Now parse the supplier mime types
	ParseMimeTypesL(supplier, iSupplierMimeTypes);
	
	// Do the same for the consumer mime types
	ParseMimeTypesL(consumer, iConsumerMimeTypes);
	}

CAgentFactory& CAgentInfo::AgentFactoryL()
	{
	// Create the agent factory if it hasn't been done already
	if (!iAgentFactory)
		{
		iAgentFactory = CAgentFactory::NewL(iAgent.ImplementationUid());
		}
	return *iAgentFactory;
	}

CAgentManager& CAgentInfo::AgentManagerL()
	{
	// Create the agent manager
	if (!iAgentManager)
		{
		iAgentManager = AgentFactoryL().CreateManagerL();
		}
	return *iAgentManager;
	}	

TBool CAgentInfo::IsSupportedSupplier(const TDesC8& aSupplierMime) const
	{
	for (TInt i = 0; i < iSupplierMimeTypes.Count(); ++i)
		{
		if (aSupplierMime == *iSupplierMimeTypes[i])
			{
			return ETrue;
			}
		}
	return EFalse;
	}

TBool CAgentInfo::IsSupportedConsumer(const TDesC8& aConsumerMime) const
	{

	for (TInt i = 0; i < iConsumerMimeTypes.Count(); ++i)
		{
		if (aConsumerMime == *iConsumerMimeTypes[i])
			{
			return ETrue;
			}
		}
	return EFalse;
	}
		
const RPointerArray<HBufC8>& CAgentInfo::SupplierMimeTypes() const
	{
	return iSupplierMimeTypes;
	}

const RPointerArray<HBufC8>& CAgentInfo::ConsumerMimeTypes() const
	{
	return iConsumerMimeTypes;
	}

void CAgentInfo::AddToArrayL(const TDesC8& aElement, 
							 RPointerArray<HBufC8>& aArray)
	{
	// Don't bother adding empty elements
	if (aElement.Length())
		{
		HBufC8* newElem = aElement.AllocLC();
		TPtr8 lowerCasePtr = newElem->Des();
		lowerCasePtr.LowerCase();
		User::LeaveIfError(aArray.Append(newElem));
		CleanupStack::Pop(newElem);
		}
	}

void CAgentInfo::ParseMimeTypesL(const TDesC8& aBuf, 
								 RPointerArray<HBufC8>& aMimeTypes)
	{
	TPtrC8 ptr(aBuf);
	TInt pos = 0;
	while ((pos = ptr.LocateF(',')) >= 0)
		{
		// Take into account possibility of ,,
		if (pos > 0)
			{
			AddToArrayL(ptr.Left(pos), aMimeTypes);
			}

		// Now, move the pointer to the position after the ','. BUT if the
		// ',' is the last position, then we are done, so return (a bit
		// dirty, but makes things easier
		if (pos + 1 < ptr.Length())
			{
			ptr.Set(ptr.Mid(pos + 1));
			}
		else
			{
			// The ',' is the last character, so quit
			return;
			}
		}

	// Now extract the last mime type.
	AddToArrayL(ptr, aMimeTypes);
	}

TInt CAgentInfo::PreferredBufferSize() const
	{
	return iPreferredBufferSize;
	}

const TDesC& CAgentInfo::PrivateDirectoryName() const
	{
	return iPrivateDirectoryName;
	}
	
TAgent& CAgentInfo::Agent()
	{
	return iAgent;
	}