| author | teknolog | 
| Mon, 05 Apr 2010 11:33:41 +0100 | |
| changeset 83 | a2e43aa1ad11 | 
| parent 82 | d87e984bd8b8 | 
| child 84 | 3b59b88b089e | 
| permissions | -rw-r--r-- | 
| 2 | 1 | /* | 
| 2 | * Copyright (c) 2007-2010 Sebastian Brannstrom, Lars Persson, EmbedDev AB | |
| 3 | * | |
| 4 | * All rights reserved. | |
| 5 | * This component and the accompanying materials are made available | |
| 6 | * under the terms of the License "Eclipse Public License v1.0" | |
| 7 | * which accompanies this distribution, and is available | |
| 8 | * at the URL "http://www.eclipse.org/legal/epl-v10.html". | |
| 9 | * | |
| 10 | * Initial Contributors: | |
| 11 | * EmbedDev AB - initial contribution. | |
| 12 | * | |
| 13 | * Contributors: | |
| 14 | * | |
| 15 | * Description: | |
| 16 | * | |
| 17 | */ | |
| 18 | ||
| 19 | // HttpEventHandler.cpp | |
| 20 | #include <e32debug.h> | |
| 21 | #include <httperr.h> | |
| 22 | ||
| 23 | #include "HttpEventHandler.h" | |
| 24 | #include "bautils.h" | |
| 25 | #include "Httpclient.h" | |
| 26 | ||
| 27 | void CHttpEventHandler::ConstructL() | |
| 28 | 	{
 | |
| 29 | //iVerbose = ETrue; | |
| 30 | } | |
| 31 | ||
| 32 | ||
| 33 | CHttpEventHandler::CHttpEventHandler(CHttpClient* aClient, MHttpClientObserver &aCallbacks, RFs& aFs): | |
| 34 | iFileServ(aFs), iHttpClient(aClient), iCallbacks(aCallbacks) | |
| 35 | 	{
 | |
| 36 | } | |
| 37 | ||
| 38 | ||
| 39 | CHttpEventHandler::~CHttpEventHandler() | |
| 40 | 	{	
 | |
| 41 | } | |
| 42 | ||
| 43 | ||
| 44 | CHttpEventHandler* CHttpEventHandler::NewLC(CHttpClient* aClient, MHttpClientObserver &aCallbacks, RFs& aFs) | |
| 45 | 	{
 | |
| 46 | CHttpEventHandler* me = new(ELeave)CHttpEventHandler(aClient, aCallbacks, aFs); | |
| 47 | CleanupStack::PushL(me); | |
| 48 | me->ConstructL(); | |
| 49 | return me; | |
| 50 | } | |
| 51 | ||
| 52 | ||
| 53 | CHttpEventHandler* CHttpEventHandler::NewL(CHttpClient* aClient, MHttpClientObserver &aCallbacks, RFs& aFs) | |
| 54 | 	{
 | |
| 55 | CHttpEventHandler* me = NewLC(aClient, aCallbacks, aFs); | |
| 56 | CleanupStack::Pop(me); | |
| 57 | return me; | |
| 58 | } | |
| 59 | ||
| 60 | void CHttpEventHandler::MHFRunL(RHTTPTransaction aTransaction, const THTTPEvent& aEvent) | |
| 61 | 	{
 | |
| 62 | switch (aEvent.iStatus) | |
| 63 | 		{
 | |
| 64 | case THTTPEvent::EGotResponseHeaders: | |
| 65 | 			{
 | |
| 66 | // HTTP response headers have been received. We can determine now if there is | |
| 67 | // going to be a response body to save. | |
| 68 | RHTTPResponse resp = aTransaction.Response(); | |
| 69 | iLastStatusCode = resp.StatusCode(); | |
| 70 | RStringF statusStr = resp.StatusText(); | |
| 71 | TBuf<32> statusStr16; | |
| 72 | statusStr16.Copy(statusStr.DesC()); | |
| 73 | 			DP2("Status: %d (%S)", iLastStatusCode, &statusStr16);
 | |
| 74 | ||
| 75 | // Dump the headers if we're being verbose | |
| 76 | //DumpRespHeadersL(aTransaction); | |
| 77 | ||
| 78 | // Determine if the body will be saved to disk | |
| 79 | iSavingResponseBody = ETrue; | |
| 80 | TBool cancelling = EFalse; | |
| 81 | if (resp.HasBody() && (iLastStatusCode >= 200) && (iLastStatusCode < 300) && (iLastStatusCode != 204)) | |
| 82 | 				{
 | |
| 83 | //iBytesDownloaded = 0; | |
| 84 | TInt dataSize = resp.Body()->OverallDataSize(); | |
| 85 | 				if (dataSize >= 0) {
 | |
| 86 | 					DP1("Response body size is %d", dataSize);
 | |
| 87 | iBytesTotal = dataSize; | |
| 88 | 				} else {
 | |
| 89 | 					DP("Response body size is unknown");
 | |
| 90 | iBytesTotal = -1; | |
| 91 | } | |
| 92 | iCallbacks.DownloadInfo(iHttpClient, dataSize); | |
| 93 | ||
| 94 | cancelling = EFalse; | |
| 95 | } | |
| 96 | ||
| 97 | // If we're cancelling, must do it now.. | |
| 98 | if (cancelling) | |
| 99 | 				{
 | |
| 100 | 				DP("Transaction Cancelled");
 | |
| 101 | aTransaction.Close(); | |
| 102 | iHttpClient->ClientRequestCompleteL(KErrCancel); | |
| 103 | } | |
| 104 | else if (iSavingResponseBody) // If we're saving, then open a file handle for the new file | |
| 105 | 				{
 | |
| 106 | iFileServ.Parse(iFileName, iParsedFileName); | |
| 107 | TInt valid = iFileServ.IsValidName(iFileName); | |
| 108 | if (!valid) | |
| 109 | 					{
 | |
| 110 | 					DP("The specified filename is not valid!.");
 | |
| 111 | iSavingResponseBody = EFalse; | |
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 112 | iHttpClient->ClientRequestCompleteL(KErrBadName); | 
| 2 | 113 | } | 
| 114 | else | |
| 115 | 					{
 | |
| 116 | 					if (iContinue) {
 | |
| 117 | TInt err = iRespBodyFile.Open(iFileServ, iParsedFileName.FullName(),EFileWrite); | |
| 118 | if (err) | |
| 119 | 							{
 | |
| 83 | 120 | 							DP2("There was an error opening file '%S', err=%d", &iParsedFileName.FullName(), err);
 | 
| 2 | 121 | iSavingResponseBody = EFalse; | 
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 122 | iHttpClient->ClientRequestCompleteL(KErrInUse); | 
| 2 | 123 | User::Leave(err); | 
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 124 | } | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 125 | else | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 126 | 							{
 | 
| 2 | 127 | int pos = -KByteOverlap; | 
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 128 | if((err=iRespBodyFile.Seek(ESeekEnd, pos)) != KErrNone) | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 129 | 								{
 | 
| 2 | 130 | 								DP("Failed to set position!");
 | 
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 131 | iHttpClient->ClientRequestCompleteL(KErrGeneral); | 
| 2 | 132 | User::Leave(err); | 
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 133 | } | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 134 | iBytesDownloaded = (pos > 0) ? pos : 0; | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 135 | iBytesTotal += iBytesDownloaded; | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 136 | 						DP1("Total bytes is now %u", iBytesTotal);
 | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 137 | 						DP1("Seeking end: %d", pos);
 | 
| 2 | 138 | } | 
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 139 | } | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 140 | else | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 141 | 						{
 | 
| 2 | 142 | TInt err = iRespBodyFile.Replace(iFileServ, | 
| 143 | iParsedFileName.FullName(), | |
| 144 | EFileWrite); | |
| 145 | if (err) | |
| 146 | 							{
 | |
| 147 | 							DP("There was an error replacing file");
 | |
| 148 | iSavingResponseBody = EFalse; | |
| 149 | User::Leave(err); | |
| 150 | } | |
| 151 | } | |
| 152 | } | |
| 153 | } | |
| 154 | ||
| 155 | } break; | |
| 156 | case THTTPEvent::EGotResponseBodyData: | |
| 157 | 			{
 | |
| 158 | // Get the body data supplier | |
| 159 | iRespBody = aTransaction.Response().Body(); | |
| 160 | ||
| 161 | // Some (more) body data has been received (in the HTTP response) | |
| 162 | //DumpRespBody(aTransaction); | |
| 163 | 			//DP1("Saving: %d", iSavingResponseBody);
 | |
| 164 | // Append to the output file if we're saving responses | |
| 165 | if (iSavingResponseBody) | |
| 166 | 				{
 | |
| 167 | TPtrC8 bodyData; | |
| 168 | iRespBody->GetNextDataPart(bodyData); | |
| 169 | iBytesDownloaded += bodyData.Length(); | |
| 170 | TInt error = iRespBodyFile.Write(bodyData); | |
| 171 | // on writing error we close connection | |
| 172 | 				if (error != KErrNone) {
 | |
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 173 | iRespBodyFile.Close(); | 
| 2 | 174 | iCallbacks.FileError(error); | 
| 175 | iHttpClient->ClientRequestCompleteL(error); | |
| 176 | return; | |
| 177 | } | |
| 178 | ||
| 179 | 				if (!iSilent) {
 | |
| 180 | iCallbacks.Progress(iHttpClient, iBytesDownloaded, iBytesTotal); | |
| 181 | } | |
| 182 | } | |
| 183 | ||
| 184 | // Done with that bit of body data | |
| 185 | iRespBody->ReleaseData(); | |
| 186 | } break; | |
| 187 | case THTTPEvent::EResponseComplete: | |
| 188 | 			{
 | |
| 189 | // The transaction's response is complete | |
| 190 | ||
| 191 | 			DP("Transaction Complete");
 | |
| 192 | 			DP("Closing file");
 | |
| 193 | iRespBodyFile.Close(); | |
| 194 | } break; | |
| 195 | case THTTPEvent::ESucceeded: | |
| 196 | 			{
 | |
| 197 | 			DP("Transaction Successful");
 | |
| 198 | aTransaction.Close(); | |
| 199 | iHttpClient->ClientRequestCompleteL(KErrNone); | |
| 200 | } break; | |
| 201 | case THTTPEvent::EFailed: | |
| 202 | 			{
 | |
| 203 | 			DP("Transaction Failed");
 | |
| 204 | aTransaction.Close(); | |
| 205 | ||
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 206 | // if(iLastStatusCode == HTTPStatus::EOk || iLastStatusCode == HTTPStatus::ECreated || iLastStatusCode == HTTPStatus::EAccepted) | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 207 | //				{
 | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 208 | // iLastStatusCode = KErrNone; | 
| 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 209 | // } | 
| 2 | 210 | |
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 211 | iHttpClient->ClientRequestCompleteL(KErrGeneral); | 
| 2 | 212 | } break; | 
| 213 | case THTTPEvent::ERedirectedPermanently: | |
| 214 | 			{
 | |
| 215 | 			DP("Permanent Redirection");
 | |
| 216 | } break; | |
| 217 | case THTTPEvent::ERedirectedTemporarily: | |
| 218 | 			{
 | |
| 219 | 			DP("Temporary Redirection");
 | |
| 220 | } break; | |
| 221 | default: | |
| 222 | 			{
 | |
| 223 | 			DP1("<unrecognised event: %d>", aEvent.iStatus);
 | |
| 224 | // close off the transaction if it's an error | |
| 225 | if (aEvent.iStatus < 0) | |
| 226 | 				{
 | |
| 227 | aTransaction.Close(); | |
| 228 | iHttpClient->ClientRequestCompleteL(aEvent.iStatus); | |
| 229 | } | |
| 230 | } break; | |
| 231 | } | |
| 232 | } | |
| 233 | ||
| 234 | TInt CHttpEventHandler::MHFRunError(TInt aError, RHTTPTransaction /*aTransaction*/, const THTTPEvent& /*aEvent*/) | |
| 235 | 	{
 | |
| 236 | 	DP1("MHFRunError fired with error code %d", aError);
 | |
| 82 
d87e984bd8b8
Even more robustness improvements for HTTP client - specifically disk full problems
 teknolog parents: 
2diff
changeset | 237 | iHttpClient->ClientRequestCompleteL(aError); | 
| 2 | 238 | return KErrNone; | 
| 239 | } | |
| 240 | ||
| 241 | void CHttpEventHandler::SetSaveFileName(const TDesC &fName, TBool aContinue) | |
| 242 | 	{
 | |
| 243 | iFileName.Copy(fName); | |
| 244 | iContinue = aContinue; | |
| 245 | } | |
| 246 | ||
| 247 | void CHttpEventHandler::DumpRespHeadersL(RHTTPTransaction& aTrans) | |
| 248 | 	{
 | |
| 249 | RHTTPResponse resp = aTrans.Response(); | |
| 250 | RStringPool strP = aTrans.Session().StringPool(); | |
| 251 | RHTTPHeaders hdr = resp.GetHeaderCollection(); | |
| 252 | THTTPHdrFieldIter it = hdr.Fields(); | |
| 253 | ||
| 254 | TBuf<KMaxHeaderNameLen> fieldName16; | |
| 255 | TBuf<KMaxHeaderValueLen> fieldVal16; | |
| 256 | ||
| 257 | while (it.AtEnd() == EFalse) | |
| 258 | 		{
 | |
| 259 | RStringTokenF fieldName = it(); | |
| 260 | RStringF fieldNameStr = strP.StringF(fieldName); | |
| 261 | THTTPHdrVal fieldVal; | |
| 262 | if (hdr.GetField(fieldNameStr,0,fieldVal) == KErrNone) | |
| 263 | 			{
 | |
| 264 | const TDesC8& fieldNameDesC = fieldNameStr.DesC(); | |
| 265 | fieldName16.Copy(fieldNameDesC.Left(KMaxHeaderNameLen)); | |
| 266 | switch (fieldVal.Type()) | |
| 267 | 				{
 | |
| 268 | case THTTPHdrVal::KTIntVal: | |
| 269 | 				DP2("%S: %d", &fieldName16, fieldVal.Int());
 | |
| 270 | break; | |
| 271 | case THTTPHdrVal::KStrFVal: | |
| 272 | 				{
 | |
| 273 | RStringF fieldValStr = strP.StringF(fieldVal.StrF()); | |
| 274 | const TDesC8& fieldValDesC = fieldValStr.DesC(); | |
| 275 | fieldVal16.Copy(fieldValDesC.Left(KMaxHeaderValueLen)); | |
| 276 | 				DP2("%S: %S", &fieldName16, &fieldVal16);
 | |
| 277 | } | |
| 278 | break; | |
| 279 | case THTTPHdrVal::KStrVal: | |
| 280 | 				{
 | |
| 281 | RString fieldValStr = strP.String(fieldVal.Str()); | |
| 282 | const TDesC8& fieldValDesC = fieldValStr.DesC(); | |
| 283 | fieldVal16.Copy(fieldValDesC.Left(KMaxHeaderValueLen)); | |
| 284 | 				DP2("%S: %S", &fieldName16, &fieldVal16);
 | |
| 285 | } | |
| 286 | break; | |
| 287 | case THTTPHdrVal::KDateVal: | |
| 288 | 				{
 | |
| 289 | TDateTime date = fieldVal.DateTime(); | |
| 290 | } | |
| 291 | break; | |
| 292 | default: | |
| 293 | 				DP1("%S: <unrecognised value type>", &fieldName16);
 | |
| 294 | break; | |
| 295 | } | |
| 296 | ||
| 297 | // Display realm for WWW-Authenticate header | |
| 298 | RStringF wwwAuth = strP.StringF(HTTP::EWWWAuthenticate,RHTTPSession::GetTable()); | |
| 299 | if (fieldNameStr == wwwAuth) | |
| 300 | 				{
 | |
| 301 | // check the auth scheme is 'basic' | |
| 302 | RStringF basic = strP.StringF(HTTP::EBasic,RHTTPSession::GetTable()); | |
| 303 | RStringF realm = strP.StringF(HTTP::ERealm,RHTTPSession::GetTable()); | |
| 304 | THTTPHdrVal realmVal; | |
| 305 | if ((fieldVal.StrF() == basic) && | |
| 306 | (!hdr.GetParam(wwwAuth, realm, realmVal))) | |
| 307 | 					{
 | |
| 308 | RStringF realmValStr = strP.StringF(realmVal.StrF()); | |
| 309 | fieldVal16.Copy(realmValStr.DesC()); | |
| 310 | 					DP1("Realm is: %S", &fieldVal16);
 | |
| 311 | } | |
| 312 | } | |
| 313 | } | |
| 314 | ++it; | |
| 315 | } | |
| 316 | } | |
| 317 | ||
| 318 | void CHttpEventHandler::DumpRespBody(RHTTPTransaction& aTrans) | |
| 319 | 	{
 | |
| 320 | MHTTPDataSupplier* body = aTrans.Response().Body(); | |
| 321 | TPtrC8 dataChunk; | |
| 322 | TBool isLast = body->GetNextDataPart(dataChunk); | |
| 323 | DumpIt(dataChunk); | |
| 324 | if (isLast) | |
| 325 | 		DP("Got last data chunk.");
 | |
| 326 | } | |
| 327 | ||
| 328 | ||
| 329 | void CHttpEventHandler::DumpIt(const TDesC8& aData) | |
| 330 | //Do a formatted dump of binary data | |
| 331 | 	{
 | |
| 332 | // Iterate the supplied block of data in blocks of cols=80 bytes | |
| 333 | const TInt cols=16; | |
| 334 | TInt pos = 0; | |
| 335 | TBuf<KMaxFileName - 2> logLine; | |
| 336 | TBuf<KMaxFileName - 2> anEntry; | |
| 337 | const TInt dataLength = aData.Length(); | |
| 338 | ||
| 339 | while (pos < dataLength) | |
| 340 | 		{
 | |
| 341 | //start-line hexadecimal( a 4 digit number) | |
| 342 | 		anEntry.Format(TRefByValue<const TDesC>_L("%04x : "), pos);
 | |
| 343 | logLine.Append(anEntry); | |
| 344 | ||
| 345 | // Hex output | |
| 346 | TInt offset; | |
| 347 | for (offset = 0; offset < cols; ++offset) | |
| 348 | 			{
 | |
| 349 | if (pos + offset < aData.Length()) | |
| 350 | 				{
 | |
| 351 | TInt nextByte = aData[pos + offset]; | |
| 352 | 				anEntry.Format(TRefByValue<const TDesC>_L("%02x "), nextByte);
 | |
| 353 | logLine.Append(anEntry); | |
| 354 | } | |
| 355 | else | |
| 356 | 				{
 | |
| 357 | //fill the remaining spaces with blanks untill the cols-th Hex number | |
| 358 | 				anEntry.Format(TRefByValue<const TDesC>_L("   "));
 | |
| 359 | logLine.Append(anEntry); | |
| 360 | } | |
| 361 | } | |
| 362 | 			anEntry.Format(TRefByValue<const TDesC>_L(": "));
 | |
| 363 | logLine.Append(anEntry); | |
| 364 | ||
| 365 | // Char output | |
| 366 | for (offset = 0; offset < cols; ++offset) | |
| 367 | 			{
 | |
| 368 | if (pos + offset < aData.Length()) | |
| 369 | 				{
 | |
| 370 | TInt nextByte = aData[pos + offset]; | |
| 371 | if ((nextByte >= ' ') && (nextByte <= '~')) | |
| 372 | 					{
 | |
| 373 | 					anEntry.Format(TRefByValue<const TDesC>_L("%c"), nextByte);
 | |
| 374 | logLine.Append(anEntry); | |
| 375 | } | |
| 376 | else | |
| 377 | 					{
 | |
| 378 | 					anEntry.Format(TRefByValue<const TDesC>_L("."));
 | |
| 379 | logLine.Append(anEntry); | |
| 380 | } | |
| 381 | } | |
| 382 | else | |
| 383 | 				{
 | |
| 384 | 				anEntry.Format(TRefByValue<const TDesC>_L(" "));
 | |
| 385 | logLine.Append(anEntry); | |
| 386 | } | |
| 387 | } | |
| 388 | logLine.Zero(); | |
| 389 | ||
| 390 | // Advance to next byte segment (1 seg= cols) | |
| 391 | pos += cols; | |
| 392 | } | |
| 393 | } | |
| 394 | ||
| 395 | void CHttpEventHandler::SetSilent(TBool aSilent) | |
| 396 | 	{
 | |
| 397 | iSilent = aSilent; | |
| 398 | } | |
| 399 | ||
| 400 | void CHttpEventHandler::CloseSaveFile() | |
| 401 | {
 | |
| 402 | if(iRespBody != NULL) | |
| 403 | 	{		
 | |
| 404 | if(iRespBodyFile.SubSessionHandle() != 0) | |
| 405 | 			{
 | |
| 406 | TInt size; | |
| 407 | iRespBodyFile.Size(size); | |
| 408 | 			DP2("Closing file at size %d, bytes downloaded %d", size, iBytesDownloaded);
 | |
| 409 | iRespBodyFile.Close(); | |
| 410 | } | |
| 411 | } | |
| 412 | } | |
| 413 |