pimprotocols/phonebooksync/plugin/phbksyncplugin.cpp
changeset 0 e686773b3f54
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/pimprotocols/phonebooksync/plugin/phbksyncplugin.cpp	Tue Feb 02 10:12:17 2010 +0200
@@ -0,0 +1,584 @@
+// Copyright (c) 2002-2009 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:
+// Contains the Phonebook Synchroniser Plug-in implementation. 
+// 
+//
+
+#include <e32base.h>
+#include <cntdb.h>
+#include <cntitem.h>
+#include <cntdef.h>
+#include <ecom/implementationproxy.h>
+
+#include "phbksyncplugin.h"
+#include "phbksynclog.h"
+#include "SyncContactICCEntry.h"
+#include "PhonebookManager.h"
+#include "SyncEngineServer.h"
+
+
+
+/** 
+ *  Validate the contact data specified by aId can be read, edited or searched. 
+ * 
+ *  @param aOp  Validate operation requested (read/edit/search)
+ *  @param aId  Contact item ID
+ */
+TInt CPhoneBookSyncPlugin::ValidateContact(TValidateOperation aOp, TContactItemId aId)
+	{
+	LOGPLUGIN3(_L8("ValidateContact(%d, 0x%08x)"), aOp, aId);
+
+#ifdef SYMBIAN_CNTMODEL_V2
+	return iSession.ValidateContact(aOp, aId);
+#else
+	if (!iInServerThread)
+		{
+		return iSession.ValidateContact(aOp, aId);
+		}
+	else
+		{
+		return KErrNone;
+		}
+#endif // SYMBIAN_CNTMODEL_V2
+	} // CPhoneBookSyncPlugin::ValidateContact
+
+
+/** 
+ *  Write the contact data specified by aContactItem to the ICC. The slot 
+ *  number used to write the data is specified by the KUidContactFieldICCSlot
+ *  field. Similarly the phonebook Uid is specified by the KUidContactFieldICCPhonebook 
+ *  field. 
+ * 
+ *  @param aContactItem  Contact item to write.
+ *
+ *  @return KErrNone if write to ICC was successful, otherwise return error
+ */
+TInt CPhoneBookSyncPlugin::ValidateWriteContact(CContactICCEntry& aContactItem)
+	{
+	LOGPLUGIN1(_L8("ValidateWriteContact()"));
+
+#ifndef SYMBIAN_CNTMODEL_V2
+	if (!iInServerThread)
+		{
+#endif // SYMBIAN_CNTMODEL_V2
+
+	TRequestStatus status(0);
+	TInt slotNum = ICCSlotNumber(aContactItem);
+
+	TUid phonebookUid(TUid::Null()); // First check whether phonebook field is supplied
+	TRAPD(error, phonebookUid = ICCPhonebookL(aContactItem)); 
+	if (error != KErrNone)
+		{
+		return error; 
+		}
+
+	TBuf<KTemplateFieldLength> buf;
+
+	if (phonebookUid == KUidIccPhonebookNotSpecified) // phonebook field not set within aContactItem 
+		{
+		iSession.WriteContact(status, aContactItem, slotNum);
+		}
+	else
+		{
+		iSession.WriteContact(status, aContactItem, slotNum, phonebookUid);
+		}
+	User::WaitForRequest(status);
+	TInt ret(status.Int());
+
+	if (ret == KErrNone)
+		{
+		// Server has added slot number, so add it to aContactItem
+		buf.Zero();
+		buf.AppendNum(slotNum);
+		TRAP(ret, AddFieldValueL(aContactItem, KUidContactFieldICCSlot, buf));
+		if(phonebookUid != KUidIccPhonebookNotSpecified) 
+		// Server has added phonebook Uid, so add it to aContactItem
+			{
+			buf.Zero();
+			buf.AppendNum(TUint(phonebookUid.iUid)); // Append integer part of UID as actual field value
+			TRAP(ret, AddFieldValueL(aContactItem, KUidContactFieldICCPhonebook, buf));
+			}
+		}
+
+	return ret;
+
+#ifndef SYMBIAN_CNTMODEL_V2
+		}
+	else
+		{
+		return KErrNone;
+		}
+#endif // SYMBIAN_CNTMODEL_V2
+	} // CPhoneBookSyncPlugin::ValidateWriteContact
+
+
+/** 
+ *  Delete the contact data from the ICC at the slot number specified 
+ *  in the CContactICCEntry item ID specified.
+ * 
+ *  @param aId  Contact Item ID
+ *
+ *  @return KErrNone if ICC deletion was successful, otherwise return error
+ */
+TInt CPhoneBookSyncPlugin::DeleteContact(TContactItemId aId)
+	{
+	LOGPLUGIN2(_L8("DeleteContact(0x%08x)"), aId);
+	
+#ifndef SYMBIAN_CNTMODEL_V2
+	if (!iInServerThread)
+		{
+#endif // SYMBIAN_CNTMODEL_V2
+
+	TRequestStatus status;
+	iSession.DeleteContact(status, aId);
+	User::WaitForRequest(status);
+	return status.Int();
+
+#ifndef SYMBIAN_CNTMODEL_V2
+		}
+	else
+		{
+		return KErrNone;
+		}
+#endif // SYMBIAN_CNTMODEL_V2
+	} // CPhoneBookSyncPlugin::DeleteContact
+
+
+/** 
+ *  Release any resources allocated by the plug-in and delete itself.
+ * 
+ *  This method is called by the destructor of Contact model's 
+ *  CContactDatabase class - ensuring it will be called when the plug-in
+ *  is unloaded.
+ */
+void CPhoneBookSyncPlugin::Release()
+	{
+	LOGPLUGIN1(_L8("Release()"));
+	
+	iSession.Close();
+	delete this;
+	} // CPhoneBookSyncPlugin::Release
+
+
+/** 
+ *  Update the lookup tables in the server.
+ *  This method is called by the Contacts model when a new CContactICCEntry
+ *  has been added to the database.
+ *
+ *  This is only important when it is done by a client other than the 
+ *  phonebook synchronisation server because the sync server can do the 
+ *  update when it knows what the UID of the newly added contact was.
+ * 
+ *  The plug-in should extract the appropriate details from the item 
+ *  (eg. item UID, slot number and ICC phonebook type) and call the sync server to 
+ *  update the lookup tables.
+ *
+ *  @param aContactItem  Contact Entry to update in the look up table.
+ */
+void CPhoneBookSyncPlugin::UpdatePostWriteL(const CContactICCEntry& aContactItem)
+	{
+	LOGPLUGIN1(_L8("UpdatePostWriteL()"));
+
+#ifndef SYMBIAN_CNTMODEL_V2
+	if (iInServerThread)
+		{
+		return;
+		}
+#endif // SYMBIAN_CNTMODEL_V2
+
+	TContactItemId id(aContactItem.Id());
+	TInt slotNum(ICCSlotNumber(aContactItem));
+	// Extract phonebook type from CContactICCEntry
+	TUid phonebook(ICCPhonebookL(const_cast<CContactICCEntry&>(aContactItem)));
+
+	if(phonebook == KUidIccPhonebookNotSpecified) // Phonebook not specified so assume
+		{
+		iSession.UpdateLookuptable(id, slotNum);  // Global/GSM ADN phonebook is used
+		}
+	else
+		{
+		iSession.UpdateLookuptable(id, slotNum, phonebook); // phonebook supplied 
+		}
+	} // CPhoneBookSyncPlugin::UpdatePostWriteL
+
+
+/** 
+ *  Return the template that should be used with CContactICCEntry items residing on the 
+ *  phonebook specified by the aPhonebookUid parameter.
+ *
+ *  @param aPhonebookUid  UID of ICC phonebook for which the template is needed
+ *
+ *  @return Template item ID 
+ */
+TContactItemId CPhoneBookSyncPlugin::ICCTemplateIdL(TUid aPhonebookUid)
+	{
+	LOGPLUGIN2(_L8("ICCTemplateIdL(0x%08x)"), aPhonebookUid);
+
+	TContactItemId templateId(KNullContactId);
+
+#ifdef SYMBIAN_CNTMODEL_V2
+	User::LeaveIfError(iSession.GetPhoneBookId(templateId, RPhoneBookSession::ESyncTemplateId, aPhonebookUid));
+#else
+	if (!iInServerThread)
+		{
+		User::LeaveIfError(iSession.GetPhoneBookId(templateId, RPhoneBookSession::ESyncTemplateId, aPhonebookUid));
+		}
+#endif // SYMBIAN_CNTMODEL_V2
+
+	return templateId;
+	} // CPhoneBookSyncPlugin::ICCTemplateIdL
+
+
+/** 
+ *  Return the group to be used for the phonebook specified by the aPhonebookUid parameter.
+ * 
+ *  @param aPhonebookUid  UID of ICC phonebook for which the group UID is required 
+ *
+ *  @return Group item ID 
+ */
+TContactItemId CPhoneBookSyncPlugin::GroupIdL(TUid aPhonebookUid)
+	{
+	LOGPLUGIN2(_L8("GroupIdL(0x%08x)"), aPhonebookUid);
+
+	TContactItemId groupId(KNullContactId);
+
+#ifdef SYMBIAN_CNTMODEL_V2
+	User::LeaveIfError(iSession.GetPhoneBookId(groupId, RPhoneBookSession::ESyncGroupId, aPhonebookUid));
+#else
+	if (!iInServerThread)
+		{
+		User::LeaveIfError(iSession.GetPhoneBookId(groupId, RPhoneBookSession::ESyncGroupId, aPhonebookUid));
+		}
+#endif // SYMBIAN_CNTMODEL_V2
+
+	return groupId;
+	} // CPhoneBookSyncPlugin::GroupIdL
+
+
+/** 
+ *  Request notification of a phbksync cache state change.
+ *
+ *  @param aStatus Request to be completed.
+ */
+void CPhoneBookSyncPlugin::NotifySyncStateChange(TRequestStatus& aStatus, TUid aPhonebookUid)
+	{
+	LOGPLUGIN2(_L8("NotifySyncStateChange(0x%08x)"), aPhonebookUid);
+
+#ifndef SYMBIAN_CNTMODEL_V2
+	__ASSERT_ALWAYS(!iInServerThread, User::Invariant()); 
+#endif // SYMBIAN_CNTMODEL_V2
+
+	iSession.NotifyPhBkCacheStateChange(aStatus, aPhonebookUid);
+	} // CPhoneBookSyncPlugin::NotifySyncStateChange
+
+
+/** 
+ *  Cancel previous request for notification of phbksync cache state change.
+ */
+void CPhoneBookSyncPlugin::CancelNotifyRequest(TUid aPhonebookUid)
+	{
+	LOGPLUGIN2(_L8("CancelNotifyRequest(0x%08x)"), aPhonebookUid);
+
+#ifndef SYMBIAN_CNTMODEL_V2
+	__ASSERT_ALWAYS(!iInServerThread, User::Invariant()); 
+#endif // SYMBIAN_CNTMODEL_V2
+
+	iSession.CancelRequest(ESyncNotifyCacheStateChangeCancelClient, aPhonebookUid);
+	} // CPhoneBookSyncPlugin::CancelNotifyRequest
+
+
+/** 
+ *  Return the current synchronisation state of the synchroniser.
+ *  If the state is EErrorDuringSync, this method leaves with the last sync
+ *  error recorded.
+ *
+ *  @return ETrue if cache valid. EFalse, otherwise
+ */
+TBool CPhoneBookSyncPlugin::IsSynchronisedL(TUid aPhonebookUid)
+	{
+	LOGPLUGIN2(_L8("IsSynchronisedL(0x%08x)"), aPhonebookUid);
+
+#ifndef SYMBIAN_CNTMODEL_V2
+	__ASSERT_ALWAYS(!iInServerThread, User::Invariant()); 
+#endif // SYMBIAN_CNTMODEL_V2
+
+	RPhoneBookSession::TSyncState state;
+
+	User::LeaveIfError(iSession.GetPhoneBookCacheState(state, aPhonebookUid));
+	
+	if (state==RPhoneBookSession::EErrorDuringSync)
+		{
+		TInt syncError(KErrNone);
+		User::LeaveIfError(iSession.GetLastSyncError(syncError, aPhonebookUid));
+		User::Leave(syncError);
+		}
+	
+	return (state==RPhoneBookSession::ECacheValid);
+	} // CPhoneBookSyncPlugin::IsSynchronisedL
+
+/** 
+ * Returns the list of Phonebooks that may be present on the ICC.
+ *
+ * @param aPhonebookList List of the phonebooks. If successful this updated with current list of phonebooks, otherwise it is reset.
+ * @return KErrNone or a system wide error such as KErrNoMemory.
+ */
+TInt CPhoneBookSyncPlugin::PhonebookList(RArray<TUid>& aPhonebookList)
+	{
+	LOGPLUGIN1(_L8("PhonebookList()"));
+
+	//
+	// Empty the list to start with...
+	//
+	aPhonebookList.Reset();
+
+#ifndef SYMBIAN_CNTMODEL_V2
+	if (iInServerThread)
+		{
+		return KErrNone;
+		}
+#endif // SYMBIAN_CNTMODEL_V2
+
+	//
+	// Return the list of phonebooks...
+	//	
+	RPhoneBookSession::TSyncState state;
+	TInt  result;
+		
+	result = iSession.GetPhoneBookCacheState(state, KUidIccGlobalAdnPhonebook);
+	if (result != KErrNotSupported  &&  result != KErrNotFound)
+		{
+		result = aPhonebookList.Append(KUidIccGlobalAdnPhonebook);
+		if (result != KErrNone)
+			{
+			aPhonebookList.Reset();
+			return result;
+			}
+		}
+
+	result = iSession.GetPhoneBookCacheState(state, KUidIccGlobalSdnPhonebook);
+	if (result != KErrNotSupported  &&  result != KErrNotFound)
+		{
+		result = aPhonebookList.Append(KUidIccGlobalSdnPhonebook);
+		if (result != KErrNone)
+			{
+			aPhonebookList.Reset();
+			return result;
+			}
+		}
+
+	result = iSession.GetPhoneBookCacheState(state, KUidIccGlobalLndPhonebook);
+	if (result != KErrNotSupported  &&  result != KErrNotFound)
+		{
+		result = aPhonebookList.Append(KUidIccGlobalLndPhonebook);
+		if (result != KErrNone)
+			{
+			aPhonebookList.Reset();
+			return result;
+			}
+		}
+
+	result = iSession.GetPhoneBookCacheState(state, KUidUsimAppAdnPhonebook);
+	if (result != KErrNotSupported  &&  result != KErrNotFound)
+		{
+		result = aPhonebookList.Append(KUidUsimAppAdnPhonebook);
+		if (result != KErrNone)
+			{
+			aPhonebookList.Reset();
+			return result;
+			}
+		}
+
+	result = iSession.GetPhoneBookCacheState(state, KUidIccGlobalFdnPhonebook);
+	if (result != KErrNotSupported  &&  result != KErrNotFound)
+		{
+		result = aPhonebookList.Append(KUidIccGlobalFdnPhonebook);
+		if (result != KErrNone)
+			{
+			aPhonebookList.Reset();
+			return result;
+			}
+		}
+
+	return KErrNone;
+	}
+
+
+/**
+ *  Extract the ICC slot number from the ICC slot field. 
+ * 
+ *  @param aContactItem CContactICCEntry item
+ *
+ *  @return ICC Slot number 
+ */
+TInt CPhoneBookSyncPlugin::ICCSlotNumber(const CContactICCEntry& aContactItem) const
+	{
+	LOGPLUGIN1(_L8("ICCSlotNumber()"));
+
+	CContactItemFieldSet&  fieldset = aContactItem.CardFields();
+	TInt  position = fieldset.Find(KUidContactFieldICCSlot);
+	TInt  slotNum(KErrNotFound);
+	if (position != KErrNotFound)
+		{
+		const CContactItemField&  field = fieldset[position];
+		TPtrC  iccSlotField = field.TextStorage()->Text();
+		TLex  input(iccSlotField);
+		/*TInt ignore  = */ input.Val(slotNum);
+		}
+
+	return slotNum;
+	} // CPhoneBookSyncPlugin::ICCSlotNumber
+
+
+/**
+ *  Extract the ICC phonebook Uid from the ICC phonebook field. It also creates the phonebook
+ *  field if it is not supplied within aContactItem. 
+ * 
+ *  @param aContactItem CContactICCEntry item
+ *
+ *  @return TUid that represents the ICC Phonebook type
+ */
+TUid CPhoneBookSyncPlugin::ICCPhonebookL(CContactICCEntry& aContactItem) const
+	{
+	LOGPLUGIN1(_L8("ICCPhonebookL()"));
+
+	TUid  phonebookUid;
+	CContactItemFieldSet&  fieldset = aContactItem.CardFields();
+	TInt  position = fieldset.Find(KUidContactFieldICCPhonebook);
+
+	if (position != KErrNotFound)
+		{
+		const CContactItemField& field = fieldset[position];
+		TPtrC phonebookField = field.TextStorage()->Text();
+		TLex input(phonebookField);
+		input.Val(phonebookUid.iUid);
+		}
+	else // phonebook field not supplied within aContactItem, so create one 
+		{
+		// Create phonebook type field within aContactItem 
+		TBuf<KTemplateFieldLength> buf;
+		phonebookUid = KUidIccPhonebookNotSpecified; // default phonebook Uid value 
+		buf.AppendNum(TUint(phonebookUid.iUid));
+		TPtrC phonebookPtrC(buf.Ptr());
+		CContactItemField* field=CContactItemField::NewLC(KStorageTypeText, KUidContactFieldICCPhonebook);
+		field->SetMapping(KUidContactFieldVCardMapNotRequired);
+		field->AddFieldTypeL(KUidContactFieldICCPhonebook); // Appends a field type to the field's content type
+		field->TextStorage()->SetTextL(phonebookPtrC);
+		aContactItem.AddFieldL(*field); // Add this phonebook field to ICC contact item 
+		CleanupStack::Pop(field); // No need to destroy it since contact item takes ownership of field
+		}
+
+	return phonebookUid;
+	} // CPhoneBookSyncPlugin::ICCPhonebookL
+
+
+/**
+ *  Add the field value aField supplied by the phonebook synchroniser to the 
+ *  item's field specified by the aFieldType parameter. 
+ * 
+ *  @param aContactItem  CContactICCEntry item
+ *  @param aFieldType    Field type
+ *  @param aField        Field data
+ */
+void CPhoneBookSyncPlugin::AddFieldValueL(CContactICCEntry& aContactItem, TFieldType aFieldType,
+										  TDesC& aField)
+	{
+	LOGPLUGIN1(_L8("AddFieldValueL()"));
+
+	CContactItemFieldSet&  fieldset = aContactItem.CardFields();
+	TInt  position = fieldset.Find(aFieldType);
+	CContactItemField&  field = fieldset[position];
+	CContactTextField*  textField = field.TextStorage();
+
+	if (!textField->IsFull())
+		{
+		textField->SetTextL(aField);
+		}
+	} // CPhoneBookSyncPlugin::AddFieldValueL
+
+
+/** 
+ *  Create a connection with the phonebook synchroniser server.
+ */
+void CPhoneBookSyncPlugin::ConstructL()
+	{
+	LOGPLUGIN1(_L8("ConstructL()"));
+
+#ifdef SYMBIAN_CNTMODEL_V2
+	User::LeaveIfError(iSession.Connect());
+#else
+	TName threadName = RThread().Name();
+	iInServerThread = (threadName.CompareF(PHBKSYNC_ENGINE_NAME)==0);
+	
+	if (!iInServerThread)
+		{
+		User::LeaveIfError(iSession.Connect());
+		}
+#endif // SYMBIAN_CNTMODEL_V2
+	} // CPhoneBookSyncPlugin::ConstructL
+
+
+/** 
+ *  Non-leaving factory construction method.
+ *
+ *  @return Pointer to plug-in or NULL if plug-in construction unsuccessful
+ */
+CPhoneBookSyncPlugin* CPhoneBookSyncPlugin::New()
+	{
+	CPhoneBookSyncPlugin* self = NULL;
+
+	TRAPD(err,self = NewL());
+
+	if (err != KErrNone)
+		{
+		return NULL;
+		}
+	else
+		{
+		return self;
+		}
+	} // CPhoneBookSyncPlugin::New
+
+
+/** 
+ *  Standard factory construction method
+ */
+CPhoneBookSyncPlugin* CPhoneBookSyncPlugin::NewL()
+	{
+	CPhoneBookSyncPlugin* self = new (ELeave) CPhoneBookSyncPlugin();
+	CleanupStack::PushL(self);
+	self->ConstructL();
+	CleanupStack::Pop(self);
+
+	return self;
+	} // CPhoneBookSyncPlugin::NewL
+
+
+/**
+ *  Implementation collection function.
+ */
+const TImplementationProxy ImplementationTable[] = 
+    {
+    IMPLEMENTATION_PROXY_ENTRY(0x1020428D, CPhoneBookSyncPlugin::NewL)
+    };
+
+
+/**
+ *  Return the implementation table size.
+ */
+EXPORT_C const TImplementationProxy* ImplementationGroupProxy(TInt& aTableCount)
+    {
+    aTableCount = sizeof(ImplementationTable) / sizeof(TImplementationProxy);
+
+    return ImplementationTable;
+    } // ImplementationGroupProxy