os/persistentdata/loggingservices/rfilelogger/Logger/Src/Client.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 2005-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
// Source file for the client api
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
/**
sl@0
    19
 @file Client.cpp
sl@0
    20
*/
sl@0
    21
#include <test/rfilelogger.h>
sl@0
    22
sl@0
    23
sl@0
    24
//_LIT8(KxmlHeader,"<?xml version=\"1.0\" encoding=\"ISO-8859-1\"?>\r\n");
sl@0
    25
sl@0
    26
sl@0
    27
// EKA1 requires DLL entry point
sl@0
    28
//
sl@0
    29
// RFileLogger class definition
sl@0
    30
//
sl@0
    31
sl@0
    32
EXPORT_C RFileFlogger::RFileFlogger() : ilogbody(NULL)
sl@0
    33
/**
sl@0
    34
 * Create a new flogger client interface object with an empty body.
sl@0
    35
 * @internalTechnology 
sl@0
    36
 */
sl@0
    37
	{
sl@0
    38
	iLogfileTag=FALSE;
sl@0
    39
	}
sl@0
    40
sl@0
    41
EXPORT_C RFileFlogger::~RFileFlogger()
sl@0
    42
/**
sl@0
    43
 * Destructor
sl@0
    44
 * @internalTechnology 
sl@0
    45
 */
sl@0
    46
	{}
sl@0
    47
sl@0
    48
sl@0
    49
sl@0
    50
EXPORT_C TInt RFileFlogger::Connect()
sl@0
    51
/**
sl@0
    52
 * @return int - Standard error codes
sl@0
    53
 * EKA2 all variants and EKA1 target.
sl@0
    54
 * Server is an exe
sl@0
    55
 */
sl@0
    56
	{
sl@0
    57
	// Sanity check to make sure it's not been called multiple times
sl@0
    58
	if (ilogbody)
sl@0
    59
		{
sl@0
    60
		return KErrAlreadyExists;
sl@0
    61
		}
sl@0
    62
	ilogbody = new RFileLoggerBody;
sl@0
    63
	if(!ilogbody)
sl@0
    64
		return KErrNoMemory;
sl@0
    65
sl@0
    66
	TVersion version(KRFileLoggerMajorVersion,KRFileLoggerMinorVersion,KRFileLoggerBuildVersion);
sl@0
    67
	// Assume the server is already running and attempt to create a session
sl@0
    68
	
sl@0
    69
	TInt err = ilogbody->DoCreateSession(KFileLogrerServerName,version,8);
sl@0
    70
	if(err== KErrServerTerminated)
sl@0
    71
	{
sl@0
    72
		User::After(1000000); // OS need time to close previous server properly
sl@0
    73
		err = ilogbody->DoCreateSession(KFileLogrerServerName,version,8);
sl@0
    74
	}
sl@0
    75
sl@0
    76
	if(err == KErrNotFound)
sl@0
    77
		{
sl@0
    78
		// Server not running
sl@0
    79
		// Construct the server binary name
sl@0
    80
		_LIT(KEmpty,"");
sl@0
    81
		// EKA2 is simple
sl@0
    82
		// No path required
sl@0
    83
		TBuf<32> serverFile;
sl@0
    84
		serverFile.Copy(KFileLogrerServerName);
sl@0
    85
		_LIT(KExe,".exe");
sl@0
    86
		serverFile.Append(KExe);
sl@0
    87
		RProcess server;
sl@0
    88
		err = server.Create(serverFile,KEmpty);
sl@0
    89
		if(err != KErrNone)
sl@0
    90
			return err;
sl@0
    91
		// Synchronise with the server
sl@0
    92
		TRequestStatus reqStatus;
sl@0
    93
		server.Rendezvous(reqStatus);
sl@0
    94
		server.Resume();
sl@0
    95
		// Server will call the reciprocal static synchronise call
sl@0
    96
		User::WaitForRequest(reqStatus);
sl@0
    97
		server.Close();
sl@0
    98
		if(reqStatus.Int() != KErrNone)
sl@0
    99
			return reqStatus.Int();
sl@0
   100
		// Create the root server session
sl@0
   101
		err = ilogbody->DoCreateSession(KFileLogrerServerName,version,8);
sl@0
   102
		}
sl@0
   103
		if (err != KErrNone)
sl@0
   104
		{  // some other problem, kill the logbody and clean the mamory
sl@0
   105
			delete ilogbody;
sl@0
   106
			ilogbody=NULL;
sl@0
   107
		}
sl@0
   108
		else
sl@0
   109
		{   // Makes the session shared among all threads in the process
sl@0
   110
			err = ilogbody->ShareAuto();
sl@0
   111
		}
sl@0
   112
sl@0
   113
sl@0
   114
	return err;
sl@0
   115
	}
sl@0
   116
sl@0
   117
///////
sl@0
   118
EXPORT_C TInt RFileFlogger::CreateLog(const TDesC& aLogFilePath, TLogMode aMode)
sl@0
   119
/**
sl@0
   120
 * @param aLogFilePath - Full path and filename of the log file
sl@0
   121
 * @param aMode - Overwrite or Append
sl@0
   122
 * Makes synchronous call to the log server to create a log session
sl@0
   123
 */
sl@0
   124
	{
sl@0
   125
	iloglevel = ESevrAll; //ELogNone;
sl@0
   126
sl@0
   127
	if(aLogFilePath.Length() > KMaxLoggerFilePath)
sl@0
   128
		return KErrTooBig;
sl@0
   129
sl@0
   130
	TIpcArgs args;
sl@0
   131
	args.Set(0,&aLogFilePath);
sl@0
   132
	args.Set(1,aMode);
sl@0
   133
    TInt err = ilogbody->DoSendReceive(ECreateLog,args);
sl@0
   134
    return err; 
sl@0
   135
sl@0
   136
	}
sl@0
   137
sl@0
   138
EXPORT_C void RFileFlogger::Log(const TText8* aFile, TInt aLine, TLogSeverity aSeverity, TRefByValue<const TDesC> aFmt,...)
sl@0
   139
/**
sl@0
   140
 * @param aFile - Source file name
sl@0
   141
 * @param aLine - Source file line number
sl@0
   142
 * @param aSeverity - ERR, WARN, INFO
sl@0
   143
 * @param aFmt - UNICODE format string
sl@0
   144
 */
sl@0
   145
	{
sl@0
   146
	// Set up a Variable argument list and call private method
sl@0
   147
	if (aSeverity>iloglevel && aSeverity != ESevrTEFUnit) 
sl@0
   148
		{
sl@0
   149
		return;
sl@0
   150
		}
sl@0
   151
	VA_LIST aList;
sl@0
   152
	VA_START(aList, aFmt);
sl@0
   153
	Log(aFile, aLine, aSeverity, aFmt, aList);
sl@0
   154
	VA_END(aList); 
sl@0
   155
sl@0
   156
	}
sl@0
   157
EXPORT_C void RFileFlogger::Log(const TText8* aFile, TInt aLine, TLogSeverity aSeverity,TInt arraylength, TExtraLogField* aLogFields, TRefByValue<const TDesC> aFmt,...)
sl@0
   158
/**
sl@0
   159
 * @param aFile - Source file name
sl@0
   160
 * @param aLine - Source file line number
sl@0
   161
 * @param aSeverity - ERR, WARN, INFO
sl@0
   162
 * @param aFmt - UNICODE format string
sl@0
   163
 */
sl@0
   164
	{
sl@0
   165
	// Set up a Variable argument list and call private method
sl@0
   166
	if (aSeverity>iloglevel && aSeverity != ESevrTEFUnit) 
sl@0
   167
		{
sl@0
   168
		return;
sl@0
   169
		}
sl@0
   170
	VA_LIST aList;
sl@0
   171
	VA_START(aList, aFmt);
sl@0
   172
	Log(aFile, aLine, aSeverity, arraylength, aLogFields, aFmt, aList);
sl@0
   173
	VA_END(aList); 
sl@0
   174
	}
sl@0
   175
sl@0
   176
sl@0
   177
void RFileFlogger::AddTime(TDes8& aLogBuffer)
sl@0
   178
	{
sl@0
   179
	TTime now;
sl@0
   180
	now.UniversalTime();
sl@0
   181
	TDateTime dateTime = now.DateTime();
sl@0
   182
	_LIT8(KFormat,"%02d:%02d:%02d:%03d");
sl@0
   183
	// add the current time. 
sl@0
   184
/*--------- Maintaince Warning for aLogBuffer -----------------------------------
sl@0
   185
******* the fomat of below string is sensible to server.  
sl@0
   186
******* Adding any string to the aLogBuffer has to be checked 
sl@0
   187
******* in code on server side 
sl@0
   188
--------------------------------------------------------------------------------*/
sl@0
   189
	aLogBuffer.AppendFormat(KFormat,dateTime.Hour(),dateTime.Minute(),dateTime.Second(),(dateTime.MicroSecond()/1000)); 
sl@0
   190
/*--------------- End of Maintaince Warning  ----------------*/
sl@0
   191
	}
sl@0
   192
sl@0
   193
EXPORT_C void RFileFlogger::Log(const TText8* aFile, TInt aLine, TLogSeverity aSeverity, TRefByValue<const TDesC> aFmt, VA_LIST aList)
sl@0
   194
	{
sl@0
   195
	if (aSeverity>iloglevel && aSeverity != ESevrTEFUnit) 
sl@0
   196
		{
sl@0
   197
		return;
sl@0
   198
		}
sl@0
   199
sl@0
   200
	TInt arraylength = 0;
sl@0
   201
	TExtraLogField* aLogFields =NULL;	
sl@0
   202
	Log(aFile, aLine, aSeverity, arraylength, aLogFields, aFmt, aList);
sl@0
   203
	}
sl@0
   204
sl@0
   205
EXPORT_C void RFileFlogger::Log(const TText8* aFile, TInt aLine, TLogSeverity aSeverity,TInt arraylength, TExtraLogField* aLogFields, TRefByValue<const TDesC> aFmt, VA_LIST aList)
sl@0
   206
/**
sl@0
   207
 * @param aFile - Source file name
sl@0
   208
 * @param aLine - Source file line number
sl@0
   209
 * @param aSeverity - ERR, WARN, INFO
sl@0
   210
 * @param arraylength
sl@0
   211
 * @param aLogFields
sl@0
   212
 * @param aFmt - UNICODE format string
sl@0
   213
 * @param aList - Variable argument list
sl@0
   214
 *
sl@0
   215
 * Format a log output line
sl@0
   216
 */
sl@0
   217
 	{
sl@0
   218
	if (aSeverity>iloglevel && aSeverity != ESevrTEFUnit) 
sl@0
   219
		{
sl@0
   220
		return;
sl@0
   221
		}
sl@0
   222
sl@0
   223
	if (aSeverity == ESevrTEFUnit)
sl@0
   224
		{
sl@0
   225
		aSeverity = ESevrInfo;
sl@0
   226
		}
sl@0
   227
/*----- Maintaince Warning for this section: -----------------------------------
sl@0
   228
******* the fomat of below string is very sensible to server Server 
sl@0
   229
******* defomating these string with the understanding of this 
sl@0
   230
******* perticular format. Any change made here should be checked 
sl@0
   231
******* in code on server side 
sl@0
   232
--------------------------------------------------------------------------------*/
sl@0
   233
	// Create a filename string
sl@0
   234
	TBuf16<KMaxFilename> fileName;
sl@0
   235
	GetCPPModuleName(fileName, aFile);  
sl@0
   236
	// Create a buffer for formatting
sl@0
   237
	HBufC* buffer = HBufC::New(KMaxLoggerLineLength*2);
sl@0
   238
	if(!buffer)
sl@0
   239
		return;  // no memory
sl@0
   240
	TPtr ptr(buffer->Des());
sl@0
   241
	_LIT(KEnd,"\r\n");
sl@0
   242
	_LIT(KErr,"ERROR");
sl@0
   243
	_LIT(KHigh,"HIGH");
sl@0
   244
	_LIT(KWarn,"WARN");
sl@0
   245
	_LIT(KMedium,"MEDIUM");
sl@0
   246
	_LIT(KInfo,"INFO");
sl@0
   247
	_LIT(KLow,"LOW");
sl@0
   248
//	ptr.Append(KTypeTagBeging);
sl@0
   249
	ptr.Append(KSeperation);
sl@0
   250
	if(aSeverity == ESevrErr)
sl@0
   251
		ptr.Append(KErr);
sl@0
   252
	else if(aSeverity == ESevrHigh)
sl@0
   253
		ptr.Append(KHigh);
sl@0
   254
	else if(aSeverity == ESevrWarn)
sl@0
   255
		ptr.Append(KWarn);
sl@0
   256
	else if(aSeverity == ESevrMedium)
sl@0
   257
		ptr.Append(KMedium);
sl@0
   258
	else if (aSeverity == ESevrInfo)
sl@0
   259
		ptr.Append(KInfo);
sl@0
   260
	else if(aSeverity == ESevrLow)
sl@0
   261
		ptr.Append(KLow);
sl@0
   262
	else //if(aSeverity == ESevrAll)
sl@0
   263
		ptr.Append(KInfo);
sl@0
   264
	// Add the thread id -------- read CIniData to decide the level of details
sl@0
   265
	ptr.AppendFormat(KMessageFormat,(TInt)(RThread().Id()),&fileName, aLine);
sl@0
   266
	ptr.AppendFormatList(aFmt, aList);
sl@0
   267
	if(arraylength>0)  // trust user providing correct number with actual arrary length
sl@0
   268
	{
sl@0
   269
		_LIT(KTab,"\t");
sl@0
   270
		// presuming the following string is hardly being apart of log message from users
sl@0
   271
		// and no carrige return and line feed in their log field name and field value
sl@0
   272
		ptr.Append(KEnd);
sl@0
   273
		ptr.Append(KTagSeperation);
sl@0
   274
		ptr.Append(KTab);
sl@0
   275
		ptr.AppendNum(TInt64(arraylength));
sl@0
   276
		TInt loopValue(arraylength);
sl@0
   277
		while(loopValue)
sl@0
   278
		{
sl@0
   279
			ptr.Append(KEnd);
sl@0
   280
			ptr.Append(aLogFields->iLogFieldName);
sl@0
   281
			ptr.Append(KTab);
sl@0
   282
			ptr.Append(aLogFields->iLogFieldValue);
sl@0
   283
			loopValue--; // Decrement the looping until all fields are exhausted
sl@0
   284
			aLogFields++; // Increment the pointer address to access sucessive array index values			
sl@0
   285
		}
sl@0
   286
		ptr.Append(KEnd);
sl@0
   287
		ptr.Append(KTagSeperationEnd);
sl@0
   288
sl@0
   289
	}
sl@0
   290
/*----------------- End of Maintaince warning section --------------------------*/
sl@0
   291
	TRAP_IGNORE(WriteL(ptr));
sl@0
   292
sl@0
   293
	delete buffer;
sl@0
   294
sl@0
   295
	}
sl@0
   296
sl@0
   297
sl@0
   298
void RFileFlogger::WriteL(const TDesC& aLogBuffer)
sl@0
   299
/**
sl@0
   300
 * @param aLogBuffer - UNICODE buffer
sl@0
   301
 */
sl@0
   302
	{
sl@0
   303
	HBufC8* buffer = HBufC8::NewLC(aLogBuffer.Length()+100);
sl@0
   304
	if(!buffer)
sl@0
   305
		return;  // no memory
sl@0
   306
	TPtr8 ptr(buffer->Des());
sl@0
   307
	AddTime(ptr);
sl@0
   308
	ptr.Append(aLogBuffer);
sl@0
   309
	TRAP_IGNORE(WriteL(ptr));
sl@0
   310
sl@0
   311
	CleanupStack::PopAndDestroy(buffer);
sl@0
   312
	}
sl@0
   313
sl@0
   314
void RFileFlogger::WriteL(TDes8& aLogBuffer)
sl@0
   315
/**
sl@0
   316
 * @param aLogBuffer - pre-formatted narrow buffer
sl@0
   317
 * 
sl@0
   318
 * Synchronous write to the server
sl@0
   319
 */
sl@0
   320
	{
sl@0
   321
	_LIT8(KEnd,"\r\n");
sl@0
   322
	// Check to see if there's room to add CRLF
sl@0
   323
	if(aLogBuffer.Length()+2 > aLogBuffer.MaxLength())
sl@0
   324
		{
sl@0
   325
		HBufC8* buffer = HBufC8::NewLC(aLogBuffer.Length()+2);
sl@0
   326
		if(!buffer)
sl@0
   327
			return;  // no memory
sl@0
   328
		TPtr8 ptr(buffer->Des());
sl@0
   329
		ptr.Copy(aLogBuffer);
sl@0
   330
		TBuf8<4> tempBuf(_L8("\r\n"));
sl@0
   331
		if (aLogBuffer.Mid(aLogBuffer.Length()-2,2).CompareF(tempBuf) != 0)
sl@0
   332
			ptr.Append(KEnd);
sl@0
   333
		TIpcArgs args;
sl@0
   334
		args.Set(0,&ptr);
sl@0
   335
		args.Set(1,ptr.Length());
sl@0
   336
		User::LeaveIfError(ilogbody->DoSendReceive(EWriteLog,args));
sl@0
   337
		CleanupStack::PopAndDestroy(buffer);
sl@0
   338
		}
sl@0
   339
	else
sl@0
   340
		{
sl@0
   341
		TBuf8<4> tempBuf(_L8("\r\n"));
sl@0
   342
		if (aLogBuffer.Mid(aLogBuffer.Length()-2,2).CompareF(tempBuf) != 0)
sl@0
   343
			aLogBuffer.Append(KEnd);
sl@0
   344
		TIpcArgs args;
sl@0
   345
		args.Set(0,&aLogBuffer);
sl@0
   346
		args.Set(1,aLogBuffer.Length());
sl@0
   347
		User::LeaveIfError(ilogbody->DoSendReceive(EWriteLog,args));
sl@0
   348
		}
sl@0
   349
	}
sl@0
   350
sl@0
   351
void RFileFlogger::GetCPPModuleName(TDes& aModuleName, const TText8* aCPPFileName)
sl@0
   352
/**
sl@0
   353
 * @return aModuleName - Filename in descriptor
sl@0
   354
 * @param aCppFileName - Filename
sl@0
   355
 * Borrowed from scheduletest
sl@0
   356
 */
sl@0
   357
	{
sl@0
   358
	TPtrC8 fileNamePtrC8(aCPPFileName);
sl@0
   359
	// We do our own filename munging here; TParse can't help us since that's
sl@0
   360
	// expressly for EPOC filepaths and here we've got whatever the build system is
sl@0
   361
	// At present Win32 and Unix directory delimiters are supported
sl@0
   362
	TInt lastDelimiter = Max(fileNamePtrC8.LocateReverse('\\'), fileNamePtrC8.LocateReverse('/'));
sl@0
   363
	if(lastDelimiter >= 0 && lastDelimiter < fileNamePtrC8.Length() - 1)
sl@0
   364
		{
sl@0
   365
		// Found a delimiter which isn't trailing; update the ptr to start at the next char
sl@0
   366
		TInt fileNameLen = Min(KMaxFilename, fileNamePtrC8.Length() - (lastDelimiter + 1));
sl@0
   367
		fileNamePtrC8.Set(aCPPFileName + lastDelimiter + 1, fileNameLen);
sl@0
   368
		}
sl@0
   369
	else
sl@0
   370
		{
sl@0
   371
		// Didn't find a delimiter; take as much of the right-end of the name as fits
sl@0
   372
		fileNamePtrC8.Set(aCPPFileName + Max(0, fileNamePtrC8.Length() - KMaxFilename), Min(fileNamePtrC8.Length(), KMaxFilename));
sl@0
   373
		}
sl@0
   374
	aModuleName.Copy(fileNamePtrC8);
sl@0
   375
	}
sl@0
   376
sl@0
   377
sl@0
   378
sl@0
   379
EXPORT_C void RFileFlogger::SetLogLevel(TLogSeverity aloglevel)
sl@0
   380
	{
sl@0
   381
	iloglevel=aloglevel;
sl@0
   382
	}
sl@0
   383
sl@0
   384
sl@0
   385
EXPORT_C void RFileFlogger::Close()
sl@0
   386
	{
sl@0
   387
  	if(ilogbody)
sl@0
   388
  		{
sl@0
   389
		ilogbody->Close();
sl@0
   390
		delete ilogbody;
sl@0
   391
		ilogbody = NULL;
sl@0
   392
  		}
sl@0
   393
	}
sl@0
   394
sl@0
   395