localisation/apparchitecture/apserv/apsnnappupdates.h
author Maciej Seroka <maciejs@symbian.org>
Fri, 07 May 2010 15:36:59 +0100
branchSymbian3
changeset 36 650fe78be9c6
parent 6 c108117318cb
child 57 b8d18c84f71c
permissions -rw-r--r--
Disabled tests that don't build for Syborg. Replaced c: with z: drive.

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

#ifndef APSNNAPPUPDATES_H
#define APSNNAPPUPDATES_H

#include <e32base.h>
#include <f32file.h>
	


// classes referenced
class CApaAppData;
class CApaAppListServer;
class RApsUpdateLog;
class RWriteStream;

/**
This class keeps track of a resource file or icon file, 
whether it is in its final location or a temporary directory.

The NonNativeApps updates system makes heavy use of moving files
between temporary and final locations, in an attempt to provide
atomicity of update lists.

@internalComponent
*/
NONSHARABLE_CLASS(TFileDetails)
	{
public:
	TFileDetails();
	const TDesC& Path();
	void SetPath(const TDesC& aPath);
	TBool Exists() const;
	void CloseHandle();
	
	void OpenL(RFs& aFs, const TFileName& aFileName);
	void CreateTemporaryL(RFs& aFs, const TFileName& aDir);
	void GetDuplicateHandleL(RFile& aFile);

	void RenameToRealL(RFs& aFs);
	void RenameToTemporaryL(RFs& aFs, const TFileName& aDir);
	TInt Delete(RFs& aFs);
	TInt DeleteTemporary(RFs& aFs);
	TInt RestoreReal(RFs& aFs);

	void ExternalizeL(RWriteStream& aStream);
	void ExternalizeContinuationL(RWriteStream& aStream, TInt aTag);
	TInt ExternalizeContinuation(RWriteStream& aStream, TInt aTag);
	void InternalizeL(RReadStream& aStream, TInt& aPosition);
	void PostInternalizeL(RFs& aFs);

private:
	TInt RenameToReal(RFs& aFs);

private:
	enum TState
		{
		EStateInvalid = 0,
		EStateNull,
		EStateTemporary,
		EStateReal
		};
	TState iState;
	TFileName iPath;
	TFileName iTempPath;
	RFile iHandle;
	};

/**
Abstract base class for objects that represent updates to the non-native applications
registry. Subclasses must implement the DoPerformUpdateL and DoRollbackUpdate methods.

- DoPerformUpdateL will only ever be called once or not at all
- DoPerformUpdateL will be called first, if at all
- DoRollbackUpdate will not be called before DoPerformUpdateL
- DoRollbackUpdate will be called if DoPerformUpdateL fails on this update or any subsequent one
- DoRollbackUpdate will not be called twice
- The object may be deleted at any time.

If the device is turned off / crashes partway through performing updates, 
the following will be called upon reboot:

- The subclass' NewL, with dummy values instead of class-specific data (if any)
- InternalizePerformLogL() will be called immediately afterwards
- InternalizeRollbackLogL() iff a rollback action for this update is in the log
- PostInternalizeL() will be called after the log has been read in entirely
- DoRollbackUpdaetL() may then be called if, after replaying the log, the class' state is
                      EPerforming, EPerformed or ERollingBack

@internalComponent
*/
NONSHARABLE_CLASS(CApsNonNativeApplicationsUpdate) : public CBase
	{
public: // Types
	enum TState
		{
		ENew,
		ENeedsInternalizing,
		EPerforming,
		EPerformed,
		ERollingBack,
		ERolledBack
		};
	
	enum TLogActionType
		{
		EInvalidAction = 0,
		ENewUpdate,				// Indicates the record of an update's initial state
		EPerformUpdate,			// Indicates the start of a Perform-update log
		ERollbackUpdate,        // Indicates the start of a Rollback-update log
		EContinuationOfUpdate,  // Specifies that the current update's log has not finished
		EChangeOfUpdateState,   // Similar to EContinuationOfUpdate, but used by this superclass
		EEndOfUpdate            // Specifies that the current update's log has finished
		};

	enum TLogUpdateType
		{
		EInvalidUpdate = 0,
		ERegisterApplication,   // Indicates the start of a RegisterApplication update
		EDeregisterApplication, // Indicates the start of a DeregisterApplication update
#ifdef _DEBUG
		EFail,                  // Indicates the start of a Forced-Fail update
		EPanic,                 // Indicates the start of a Forced-Panic update
		ERollbackPanic          // Indicates the start of a Forced-Panic-On-Rollback update
#endif
		};

public:
	~CApsNonNativeApplicationsUpdate();

protected:
	CApsNonNativeApplicationsUpdate(RFs& aFs, TUid aUid, TState aState, TLogUpdateType aType);

public: 
	CApsNonNativeApplicationsUpdate* Previous() const;
	CApsNonNativeApplicationsUpdate* Next() const;
	TUid Uid();
	void PerformUpdateL(RApsUpdateLog& aUpdateLog);
	void RollbackUpdateL(RApsUpdateLog& aUpdateLog);
	void InternalizeStateChangeL(RReadStream& aStream, TInt& aPosition);

	void InternalizeNewUpdateL(RReadStream& aStream, TInt& aPosition);
	void InternalizePerformUpdateL(RReadStream& aStream, TInt& aPosition);
	void InternalizeRollbackUpdateL(RReadStream& aStream, TInt& aPosition);

protected:
	TFileName TemporaryFilePathL(TDriveName& aDrive);
	void StateChangeL(TState aState, RWriteStream& aStream);

private:
	void SharedInternalizeLoopL(RReadStream& aStream, TInt& aPosition);

	virtual void DoPerformUpdateL(RApsUpdateLog& aUpdateLog) = 0;
	virtual void DoRollbackUpdate(RApsUpdateLog& aUpdateLog) = 0;

	virtual void ExternalizeL(RWriteStream& aStream);
	virtual void InternalizeL(RReadStream& aStream, TInt& aPosition);
	virtual void InternalizeUpdateContinuationL(RReadStream& aStream, TInt& aPosition);
	virtual void PostInternalizeL();


protected:
	TState iState;
	RFs& iFs;

private:
	friend class CApsNonNativeApplicationsUpdateList; // so it can link up iPrevious/iNext
	const TLogUpdateType iType;
	CApsNonNativeApplicationsUpdate* iPrevious;
	CApsNonNativeApplicationsUpdate* iNext;
	const TUid iUid;
	};

/**
Implementation of CApsNonNativeApplicationsUpdate that performs the registration of a
non-native application.

The Write/Copy methods of this class should be used to prepare the update,
they will write resource and icon files to a temporary directory.

When this update is executed, the files will be moved to the locations set with the
Set*FileTargetLocation functions.

@internalComponent
*/
NONSHARABLE_CLASS(CApsRegisterNonNativeApplication) : public CApsNonNativeApplicationsUpdate
	{
public:
	static CApsRegisterNonNativeApplication* NewL(RFs& aFs, TUid aUid, const TDriveName& aDrive, TState aState = ENew);
	~CApsRegisterNonNativeApplication();

private:
	CApsRegisterNonNativeApplication(RFs& aFs, TUid aUid, const TDriveName& aDrive, TState aState);

public:
	void SetResourceFileTargetLocation(const TDesC& aLocation);
	void WriteResourceFileL(const TDesC8& aData, const TDesC8* aDataPrefix);

	void SetLocalisableResourceFileTargetLocation(const TDesC& aLocation);
	void WriteLocalisableResourceFileL( const TDesC8& aData, const TDesC8* aDataPrefix);

	void SetIconFileTargetLocation(const TDesC& aLocation);
	void CopyIconFileL(RFile& aSourceFile);

private:	
	void NewTemporaryFileL(TFileDetails& aFile);
	void WriteResourceFileL(TFileDetails& aFile, const TDesC8& aData, const TDesC8* aDataPrefix);

	// from CApsNonNativeApplicationsUpdate
	void DoPerformUpdateL(RApsUpdateLog& aUpdateLog);
	void DoRollbackUpdate(RApsUpdateLog& aUpdateLog);
	void ExternalizeL(RWriteStream& aStream);
	void InternalizeL(RReadStream& aStream, TInt& aPosition);
	void InternalizeUpdateContinuationL(RReadStream& aStream, TInt& aPosition);
	void PostInternalizeL();


private:
	enum TLogContinuationType 
		{
		EInvalidContinuation = 0,
		EResourceFileUpdate,
		ELocalisableResourceFileUpdate,
		EIconFileUpdate
		};
	TDriveName iDrive;
	TFileDetails iResourceFile;
	TFileDetails iLocalisableResourceFile;
	TFileDetails iIconFile;
	};

/**
Implementation of CApsNonNativeApplicationsUpdate that performs the deregistration of a
non-native application.

@internalComponent
*/
NONSHARABLE_CLASS(CApsDeregisterNonNativeApplication) : public CApsNonNativeApplicationsUpdate
	{
public:
	static CApsDeregisterNonNativeApplication* NewL(RFs& aFs, CApaAppListServer& aServ, TUid aUid, TState aState = ENew);
	~CApsDeregisterNonNativeApplication();

private:
	CApsDeregisterNonNativeApplication(RFs& aFs, CApaAppListServer& aServ, TUid aUid, TState aState);

private:
	void RenameToTemporaryL(TFileDetails& aFile, RApsUpdateLog& aUpdateLog);
	CApaAppData* FindAppDataLC(RApsUpdateLog& aUpdateLog);

	// from CApsNonNativeApplicationsUpdate
	void DoPerformUpdateL(RApsUpdateLog& aUpdateLog);
	void DoRollbackUpdate(RApsUpdateLog& aUpdateLog);
	void ExternalizeL(RWriteStream& aStream);
	void InternalizeL(RReadStream& aStream, TInt& aPosition);
	void InternalizeUpdateContinuationL(RReadStream& aStream, TInt& aPosition);
	void PostInternalizeL();

private:
	enum TLogContinuationType 
		{
		EInvalidContinuation = 0,
		EResourceFileUpdate,
		ELocalisableResourceFileUpdate,
		EIconFileUpdate
		};
	CApaAppListServer& iServ;
	TFileDetails iResourceFile;
	TFileDetails iLocalisableResourceFile;
	TFileDetails iIconFile;
	};


#ifdef _DEBUG
/**
CApsAlwaysFailUpdate

A dummy CNonNativeApplicationsUpdate for testing purpouses.
Always fails with KErrGeneral when performed

@internalComponent
*/
NONSHARABLE_CLASS(CApsAlwaysFailUpdate) : public CApsNonNativeApplicationsUpdate
	{
public:
	CApsAlwaysFailUpdate(RFs& aFs, TState aState = ENew);
private:
	// from CApsNonNativeApplicationsUpdate
	void DoPerformUpdateL(RApsUpdateLog& aUpdateLog);
	void DoRollbackUpdate(RApsUpdateLog&) {}
	};


/**
CApsAlwaysPanicUpdate

A dummy CNonNativeApplicationsUpdate for testing purpouses.
Always panics when performed

@internalComponent
*/
NONSHARABLE_CLASS(CApsAlwaysPanicUpdate) : public CApsNonNativeApplicationsUpdate
	{
public:
	CApsAlwaysPanicUpdate(RFs& aFs, TState aState = ENew);
private:
	// from CApsNonNativeApplicationsUpdate
	void DoPerformUpdateL(RApsUpdateLog& aUpdateLog);
	void DoRollbackUpdate(RApsUpdateLog&) {}
	};


/**
CApsAlwaysPanicOnRollbackUpdate

A dummy CNonNativeApplicationsUpdate for testing purpouses.
Always panics upon rollback

@internalComponent
*/
NONSHARABLE_CLASS(CApsAlwaysPanicOnRollbackUpdate) : public CApsNonNativeApplicationsUpdate
	{
public:
	CApsAlwaysPanicOnRollbackUpdate(RFs& aFs, TState aState = ENew);
private:
	// from CApsNonNativeApplicationsUpdate
	void DoPerformUpdateL(RApsUpdateLog&) {}
	void DoRollbackUpdate(RApsUpdateLog& aUpdateLog);
	};
#endif // _DEBUG


#endif // APSNNAPPUPDATES_H