messagingappbase/obexmtms/obexmtm/obexserver/include/obexSendOp.h
author hgs
Tue, 05 Oct 2010 13:58:47 +0530
changeset 73 ecf6a73a9186
parent 0 72b543305e3a
permissions -rw-r--r--
201039_03

// Copyright (c) 2001-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:
//


#ifndef __OBEXSENDOP_H__
#define __OBEXSENDOP_H__



//#define __OBEX_SEND_OP_FILE_DEBUG_MODE__

#include <msvapi.h>
#include <badesca.h> // CDesCArray
#include <obex.h>
#include <obexclientmtm.h>	//TObexMtmProgress


class CObexHeaderList;


//
// Panic Code
//

//Obex Send Operation Panics
enum TObexSendOperationPanic
	{
	EObexSendOperationAlreadyActive, //< The current Obex send operation is already active
	EObexSendOperationUnknownJob, //<
	EObexSendOperationUnknownSendState, //< The current value of the Obex send state of the send operation is not found in TObexMtmProgress::TSendState
	EObexSendOperationUnexpectedTimeout //< Panic if send has timeout unexpectedly
	};

GLDEF_C void Panic(TObexSendOperationPanic aPanic);



//
// CObexServerSendOperation 
//
class CObexSendOpTimeout;
class CObexPasswordGetter;

class CObexServerSendOperation : public CActive, public MObexAuthChallengeHandler
/**
class CObexServerSendOperation

Obex Server Send Operation:

Base class for Send operations using Obex protocol.

Implements a state machine with the following states:
Initialise-->Connect-->ConnectAttemptComplete-->SendObject-->(SendNextObject-->)SendComplete-->MovedToSent-->Disconnected

The pure virtual function InitialiseObexClientL() must be overridden in the base class to initialise the
iObexClient member to use the desired Obex transport mechanism (e.g. infrared, Bluetooth).

In order to allow asynchronous transport initialisations (for example BT's SDP Query)
implementations of this function must set iAsyncInit and provide a mechanism to 
complete a request.  See the bluetooth server mtm code for implementation details.
 
@internalTechnology
@released
*/
	{
public:
	
	/**
	 * Destructor. Cancel()s, deletes owned objects and Close()s the connection to the FileServer.
	 */
	
	virtual IMPORT_C ~CObexServerSendOperation();
	
	// Must be implemented in the derived class to initialise the iObexClient member to use the desired Obex transport mechanism
	virtual void InitialiseObexClientL() =0;
	
	/**
	 * This is not required to do anything in the base implementation.
	 */
		
	IMPORT_C virtual void SecondPhaseObexClientInitL();		// may be overridden to do anything required in a second phase.  default implementation is blank.

	/**
	 * Operations to perform before attempting a connection.
	 * As multiple connection attempts can be made, it is necessary for this
	 * routine to ensure it can handle being called multiple times.
	 * May be overridden. Default implementation is blank.
	 */
	IMPORT_C virtual void PreConnectOperations();

	/**
	 * Operations to perform after attempting a connection.
	 * As multiple connection attempts can be made, it is necessary for this
	 * routine to ensure it can handle being called multiple times.
	 * May be overridden. Default implementation is blank.
	 */
	IMPORT_C virtual void PostConnectOperations();

	/**
	 * Operations to perform before attempting to send a set of objects.
	 * May be overridden. Default implementation is blank.
	 */
	IMPORT_C virtual void PreSendOperations();

	/**
	 * Operations to perform after attempting to send a set of objects.
	 * May be overridden. Default implementation is blank.
	 */
	IMPORT_C virtual void PostSendOperations();


protected: // for use by derived classes
	/**
	 * Constructor.
	 *
	 * @param aMsgTypeUid UID of message type
	 * @param aSendObj Reference to the object to send.
	 * @param aConnectTimeoutMicroSeconds Timeout period for Connect operation in microseconds.
	 * @param aPutTimeoutMicroseconds Timeout period for Put operation in microseconds.
	 * @param aObserverRequestStatus TRequestStatus of owning active object.
	 */
	
	IMPORT_C CObexServerSendOperation(TUid aMsgTypeUid, CMsvServerEntry& aSendObj, TInt aConnectTimeoutMicroSeconds, TInt aPutTimeoutMicroSeconds, TRequestStatus& aObserverRequestStatus);
	
	/**
	 * Second phase constructor. Sets up connection to the FileServer, initialises attachments or filename list then 
	 * starts sending process by initialising.
	 *
	 * @param aConnectPassword Pointer to the password to be used for authentication.
	 * @leave Leaves if insufficient memory.
	 * @leave Leaves if cannot connect to FileServer.
	 */
	
	IMPORT_C void ConstructL(const TDesC* aConnectPassword);

	/**
	 * Cancels the current operation, deletes the client and Cancel()s the timeout timer. Only completes the observer
	 * (by a call to CompleteObserver) if an external entity (i.e. the owner) has called Cancel().
	 * Otherwise the observer is not completed.
	 */
	
	IMPORT_C virtual void DoCancel();

	 /* Constructor, Alternative version
	 *
	 * @param aMsgTypeUid UID of message type
	 * @param aSendObj Reference to the object to send.
	 * @param aConnectTimeoutMicroSeconds Timeout period for Connect operation in microseconds.
	 * @param aPutTimeoutMicroseconds Timeout period for Put operation in microseconds.
	 * @param aObserverRequestStatus TRequestStatus of owning active object.
	 * @param aLastSendAttempt TBool flag to check for the second send attempt and also control header sending.  EFalse sends full headers, ETrue only sends name and size.
	 */
	
	IMPORT_C CObexServerSendOperation(TUid aMsgTypeUid, CMsvServerEntry& aSendObj, TInt aConnectTimeoutMicroSeconds,
	                                  TInt aPutTimeoutMicroSeconds, TRequestStatus& aObserverRequestStatus, 
	                                  TBool aLastSendAttempt);
	
	/**
	* Tells the derived class that the base class is about to complete the observer.
	* This is the first thing called when CompleteObserver is called.
	* Since the behaviour of CompleteObserver is to clean up the message that it was trying to send,
	* this calls gives the derived class an opportunity to either stop this deletion or recover any information
	* synchronously from the message.
	* If the derived class has no need to use this functionality, the default implementation allows deletion.
	* @param aErrorCode The last error code encountered
	* @return TBool True delete the message
	* @return TBool False DO NOT delete the message
	*/
	IMPORT_C virtual TBool CompletingObserver(TInt aErrorCode);

public: // called by CObexSendOpTimeout
	
	/**
	 * Called when the current operation times out. Causes the current operation to be cancelled, then reactivates with
	 * the appropriate error code (KErrIrObexClientNoDevicesFound or KErrIrObexClientPutPeerAborted).
	 */
	
	void TimeOut();

public:
	
	/**
	 * Returns current progress information.
	 *
	 * @return A reference to a TPckgC<TObexMtmProgress> package pointer descriptor containing progress information on this send operation.
	 * @leave KErrXXX system wide error codes
	 */
	IMPORT_C const TDesC8& ProgressL();
	
	TBool iAsyncInit;

private: // From MObexAuthChallengeHandler

	/**
	 * Called by the Obex Client when authentication is requested to pass the password back. If the password is invalid, this
	 * call should succeed but the send operation as a whole will inevitably fail.
	 *
	 * @param aRelm ignored, but could be used to indicate which password to use.
	 * @leave KErrXXX system wide error codes. Shouldn't leave just because the password is invalid.
	 */
	
	IMPORT_C virtual void GetUserPasswordL(const TDesC& aUserID);

private: // From CActive
	
	
	/**
	 * Calls RealRunL(), and traps errors
	 *
	 * @leave Leaves with errors from RealRunL()
	 */
	
	IMPORT_C virtual void RunL();

private:

	/**
	 * Destructor. Cancel()s, deletes owned objects and Close()s the connection to the FileServer.
	 */
	
	void BuildSpecificDestructor();
	
	/**
	 * Normal second phase constructor.
	 */
	
	void BuildSpecificConstructL();
	
	/**
	 * Cancels the current operation, then reactivates with the given error code.
	 *
	 * @param aError Error code to be passed to CompleteSelf.
	 */
	
	void ActivateRunLWithError(TInt aError);
	
	/**
	 * Cancel any pending obex operation without completing the observer. 
	 */

	void ExplicitCancel();  // Must call this instead of just Cancel(), otherwise the owner of this op will be completed.
	
	/**
	 * Complete the observer, reporting any error via the progress. THIS METHOD MUST BE CALLED ONCE ONLY.
	 *
	 */
	
	void CompleteObserverL();
	
	/**
	 * This causes this active object's request to complete which means
	 * RunL() will be called again if we are active (immediately if there 
	 * are no higher priority active objects in the active scheduler).
	 *
	 * @param aError Error to be passed forward to the next step of the state machine
	 */
	
	void CompleteSelf(TInt aError);

	/**
	 * Implementation of the send operation state machine. Progresses as:
	 * Initialise-->Connect-->ConnectAttemptComplete-->SendObject-->(SendNextObject-->)SendComplete-->Disconnected
	 * The SendNextObject state is repeated for each attachment in excess of one. 
	 * Also handles UserCancelled and SendError states by CompleteObserver()ing with appropriate error codes.
	 * Leaves will be passed back to RunL and handled there. 
	 *
	 * @leave KErrXXX system wide error codes
	 */
	
	void RealRunL();
	
	/**
	 * Delete the outbox entry as operation has 'completed'.
	 * Will be invisible&InPreparation anyway (MS delete will delete it the next 
	 * time it starts).
	 */
	
	TInt SynchronousEntryDelete();
	
	/**
	 * Load an attachment into the obex sending buffer, and create a new Obex object of name TMsvEntry::iDetails.
	 *
	 * @param aParent Reference to CMsvServerEntry to be sent.
	 * @param aWhichAttachment Zero-based index of attachment to send.
	 * @leave KErrXXX system wide error codes
	 */
	
	void InitialiseAttachmentL(CMsvServerEntry& aParent, TInt aWhichAttachment);
	
	void LoadFileIntoObjectL(const TDesC& aFileName, const TDesC& aObexName, const TDesC8& aMimeType);


	/**
	 * Checks the last object was sent correctly, and tries to action appropriate error feedback if not. Only to be called
	 * from ESendObject/ESendNextObject or ESendComplete states. 
	 *
	 * @param aStatus Status of last object
	 * @return ETrue if message was OK--EFalse if message failed and this function has taken the necessary action
	 */

	TBool CheckStatusOfLastObject(TInt aStatus, TObexMtmProgress::TSendState aSendState);
	
	/**
	 * Loads the next object to be sent, whether an attachment or a file in the file list.
	 *
	 * @return KErrXXX standard error code
	 * @return KErrNotFound if there were neither attachments nor files in the file list
	 * @leave KErrXXX system wide error codes
	 */
	
	TInt PrepareCurrentObjectAndSetStateL();
	
	/**
	 * Moves the newly sent message to the global sent items folder, and sets active ready for its completion.
	 *
	 * @leave KErrXXX system wide error codes
	 */
	
	void MoveToSentAndSetActiveL();
	
	/**
	 * Restores after the message has been moved to the inbox, and marks the message as visible.
	 */

	void CleanupAfterMovedToSent();
	
	/**
	 * Returns a reference to the file session (RFs) of the message
	 *
	 * @return A reference to the file session of the the message 
	 */

	RFs& FileSession();


#ifdef __OBEX_SEND_OP_FILE_DEBUG_MODE__
	/**
	 * Output the obex object to a file in the service folder
	 *
	 * @leave KErrXXX system wide error codes
	 */
	
	TInt PutObexObjectToServiceFileL();
#endif	//__OBEX_SEND_OP_FILE_DEBUG_MODE__

protected: 
	CObexClient* iObexClient; //<The Obex client memeber
	TBool        iLastSendAttempt;	// check for second send attempt and also to control header sending

private:
	//From member initialisation list
	TRequestStatus&					iObserverRequestStatus;		//<TRequestStatus for notifying observer (eventually  ClientMTM) of completion
	CMsvServerEntry&				iMsvSendParent;				//<The message being sent (i.e. the parent of any attachments being sent)
	const TInt						iConnectTimeout;			//<Connection attempt timeout in microseconds
	const TInt						iPutTimeout;				//<Put attempt timeout in microseconds
	TBool							iCancelWithoutCompleting;	//<Flag to allow cancellation without completing observer
	TPckgBuf<TObexMtmProgress>		iProgressPckg;				//<Progress package buffer
	const TUid						iMtm;						//<UID of this MTM


	TInt							iNextAttachment;			//<Index of next attachment to be sent
	CObexFileObject*				iObexObject;				//<Obex object currently being sent
	TFileName						iSendFile;					//<Filename of the object to be sent--necessary since CObexFileObject does not export its copy
	CObexSendOpTimeout*				iTimeoutTimer;				//<Timeout timer used for various operations
	HBufC*							iConnectPassword;			//<Authentication password 
	TObexMtmProgress::TSendState	iSendState;					//<State machine state
	TInt							iAttachmentEntryCount;		//<Number of attachments to send (==0 if iFileNameEntryCount!=0)


	CMsvOperation*					iMoveOperation;				//<Operation to govern the movement of the sccessfully sent message to the sent folder
	CMsvEntrySelection*				iMoveEntrySelection;		//<Entry selection containing the entry to rename

	TFileName						iServicePath;				//<Path of service folder
	};



//
// CObexSendOpTimeout 
//

class CObexSendOpTimeout : public CTimer
/**
class CObexSendOpTimeout

Obex Send Operation Timeout active object:

Straightforward active object used for timeout operations by CObexServerSendOperation.

@internalComponent
@released
*/
  	{
public:
	
	/**
	 *Canonical NewL function, which also sets the owner operation.
	 *
	 *@param aSendOperation Obex send operation which will be "timed out" when the timer expires
	 */

	static CObexSendOpTimeout* NewL(CObexServerSendOperation* aSendOperation);
private:
	
	/**
	 * Constructor. Calls CTimer's constructor with priority EPriorityStandard
	 */

	CObexSendOpTimeout();
	
	/**
	 * Second phase constructor. Calls CTimer::ConstructL(), and adds itself to the active scheduler
	 *
	 * @leave KErrXXX system wide error codes
	 */
	
	void ConstructL();
	
	/**
	 * Calls the TimeOut method of the associated CObexServerSendOperation when the timer expires
	 *
	 * @leave KErrXXX system wide error codes
	 */
	
	void RunL();
private:
	CObexServerSendOperation* iSendOperation; //<The Obex server send operation
	};


#endif // __OBEXSENDOP_H__