GTMSessionFetcherLogging.m 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965
  1. /* Copyright 2014 Google Inc. All rights reserved.
  2. *
  3. * Licensed under the Apache License, Version 2.0 (the "License");
  4. * you may not use this file except in compliance with the License.
  5. * You may obtain a copy of the License at
  6. *
  7. * http://www.apache.org/licenses/LICENSE-2.0
  8. *
  9. * Unless required by applicable law or agreed to in writing, software
  10. * distributed under the License is distributed on an "AS IS" BASIS,
  11. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. * See the License for the specific language governing permissions and
  13. * limitations under the License.
  14. */
  15. #if !defined(__has_feature) || !__has_feature(objc_arc)
  16. #error "This file requires ARC support."
  17. #endif
  18. #include <sys/stat.h>
  19. #include <unistd.h>
  20. #import "GTMSessionFetcherLogging.h"
  21. #ifndef STRIP_GTM_FETCH_LOGGING
  22. #error GTMSessionFetcher headers should have defaulted this if it wasn't already defined.
  23. #endif
  24. #if !STRIP_GTM_FETCH_LOGGING
  25. // Sensitive credential strings are replaced in logs with _snip_
  26. //
  27. // Apps that must see the contents of sensitive tokens can set this to 1
  28. #ifndef SKIP_GTM_FETCH_LOGGING_SNIPPING
  29. #define SKIP_GTM_FETCH_LOGGING_SNIPPING 0
  30. #endif
  31. // If GTMReadMonitorInputStream is available, it can be used for
  32. // capturing uploaded streams of data
  33. //
  34. // We locally declare methods of GTMReadMonitorInputStream so we
  35. // do not need to import the header, as some projects may not have it available
  36. #if !GTMSESSION_BUILD_COMBINED_SOURCES
  37. @interface GTMReadMonitorInputStream : NSInputStream
  38. + (instancetype)inputStreamWithStream:(NSInputStream *)input;
  39. @property(assign) id readDelegate;
  40. @property(assign) SEL readSelector;
  41. @end
  42. #else
  43. @class GTMReadMonitorInputStream;
  44. #endif // !GTMSESSION_BUILD_COMBINED_SOURCES
  45. @interface GTMSessionFetcher (GTMSessionFetcherLoggingUtilities)
  46. + (NSString *)headersStringForDictionary:(NSDictionary *)dict;
  47. + (NSString *)snipSubstringOfString:(NSString *)originalStr
  48. betweenStartString:(NSString *)startStr
  49. endString:(NSString *)endStr;
  50. - (void)inputStream:(GTMReadMonitorInputStream *)stream
  51. readIntoBuffer:(void *)buffer
  52. length:(int64_t)length;
  53. @end
  54. @implementation GTMSessionFetcher (GTMSessionFetcherLogging)
  55. // fetchers come and fetchers go, but statics are forever
  56. static BOOL gIsLoggingEnabled = NO;
  57. static BOOL gIsLoggingToFile = YES;
  58. static NSString *gLoggingDirectoryPath = nil;
  59. static NSString *gLogDirectoryForCurrentRun = nil;
  60. static NSString *gLoggingDateStamp = nil;
  61. static NSString *gLoggingProcessName = nil;
  62. + (void)setLoggingDirectory:(NSString *)path {
  63. gLoggingDirectoryPath = [path copy];
  64. }
  65. + (NSString *)loggingDirectory {
  66. if (!gLoggingDirectoryPath) {
  67. NSArray *paths = nil;
  68. #if TARGET_IPHONE_SIMULATOR
  69. // default to a directory called GTMHTTPDebugLogs into a sandbox-safe
  70. // directory that a developer can find easily, the application home
  71. paths = @[ NSHomeDirectory() ];
  72. #elif TARGET_OS_IPHONE
  73. // Neither ~/Desktop nor ~/Home is writable on an actual iOS, watchOS, or tvOS device.
  74. // Put it in ~/Documents.
  75. paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
  76. #else
  77. // default to a directory called GTMHTTPDebugLogs in the desktop folder
  78. paths = NSSearchPathForDirectoriesInDomains(NSDesktopDirectory, NSUserDomainMask, YES);
  79. #endif
  80. NSString *desktopPath = paths.firstObject;
  81. if (desktopPath) {
  82. NSString *const kGTMLogFolderName = @"GTMHTTPDebugLogs";
  83. NSString *logsFolderPath = [desktopPath stringByAppendingPathComponent:kGTMLogFolderName];
  84. NSFileManager *fileMgr = [NSFileManager defaultManager];
  85. BOOL isDir;
  86. BOOL doesFolderExist = [fileMgr fileExistsAtPath:logsFolderPath isDirectory:&isDir];
  87. if (!doesFolderExist) {
  88. // make the directory
  89. doesFolderExist = [fileMgr createDirectoryAtPath:logsFolderPath
  90. withIntermediateDirectories:YES
  91. attributes:nil
  92. error:NULL];
  93. if (doesFolderExist) {
  94. // The directory has been created. Exclude it from backups.
  95. NSURL *pathURL = [NSURL fileURLWithPath:logsFolderPath isDirectory:YES];
  96. [pathURL setResourceValue:@YES forKey:NSURLIsExcludedFromBackupKey error:NULL];
  97. }
  98. }
  99. if (doesFolderExist) {
  100. // it's there; store it in the global
  101. gLoggingDirectoryPath = [logsFolderPath copy];
  102. }
  103. }
  104. }
  105. return gLoggingDirectoryPath;
  106. }
  107. + (void)setLogDirectoryForCurrentRun:(NSString *)logDirectoryForCurrentRun {
  108. // Set the path for this run's logs.
  109. gLogDirectoryForCurrentRun = [logDirectoryForCurrentRun copy];
  110. }
  111. + (NSString *)logDirectoryForCurrentRun {
  112. // make a directory for this run's logs, like SyncProto_logs_10-16_01-56-58PM
  113. if (gLogDirectoryForCurrentRun) return gLogDirectoryForCurrentRun;
  114. NSString *parentDir = [self loggingDirectory];
  115. NSString *logNamePrefix = [self processNameLogPrefix];
  116. NSString *dateStamp = [self loggingDateStamp];
  117. NSString *dirName = [NSString stringWithFormat:@"%@%@", logNamePrefix, dateStamp];
  118. NSString *logDirectory = [parentDir stringByAppendingPathComponent:dirName];
  119. if (gIsLoggingToFile) {
  120. NSFileManager *fileMgr = [NSFileManager defaultManager];
  121. // Be sure that the first time this app runs, it's not writing to a preexisting folder
  122. static BOOL gShouldReuseFolder = NO;
  123. if (!gShouldReuseFolder) {
  124. gShouldReuseFolder = YES;
  125. NSString *origLogDir = logDirectory;
  126. for (int ctr = 2; ctr < 20; ++ctr) {
  127. if (![fileMgr fileExistsAtPath:logDirectory]) break;
  128. // append a digit
  129. logDirectory = [origLogDir stringByAppendingFormat:@"_%d", ctr];
  130. }
  131. }
  132. if (![fileMgr createDirectoryAtPath:logDirectory
  133. withIntermediateDirectories:YES
  134. attributes:nil
  135. error:NULL])
  136. return nil;
  137. }
  138. gLogDirectoryForCurrentRun = logDirectory;
  139. return gLogDirectoryForCurrentRun;
  140. }
  141. + (void)setLoggingEnabled:(BOOL)isLoggingEnabled {
  142. gIsLoggingEnabled = isLoggingEnabled;
  143. }
  144. + (BOOL)isLoggingEnabled {
  145. return gIsLoggingEnabled;
  146. }
  147. + (void)setLoggingToFileEnabled:(BOOL)isLoggingToFileEnabled {
  148. gIsLoggingToFile = isLoggingToFileEnabled;
  149. }
  150. + (BOOL)isLoggingToFileEnabled {
  151. return gIsLoggingToFile;
  152. }
  153. + (void)setLoggingProcessName:(NSString *)processName {
  154. gLoggingProcessName = [processName copy];
  155. }
  156. + (NSString *)loggingProcessName {
  157. // get the process name (once per run) replacing spaces with underscores
  158. if (!gLoggingProcessName) {
  159. NSString *procName = [[NSProcessInfo processInfo] processName];
  160. gLoggingProcessName = [procName stringByReplacingOccurrencesOfString:@" " withString:@"_"];
  161. }
  162. return gLoggingProcessName;
  163. }
  164. + (void)setLoggingDateStamp:(NSString *)dateStamp {
  165. gLoggingDateStamp = [dateStamp copy];
  166. }
  167. + (NSString *)loggingDateStamp {
  168. // We'll pick one date stamp per run, so a run that starts at a later second
  169. // will get a unique results html file
  170. if (!gLoggingDateStamp) {
  171. // produce a string like 08-21_01-41-23PM
  172. NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
  173. [formatter setFormatterBehavior:NSDateFormatterBehavior10_4];
  174. [formatter setDateFormat:@"M-dd_hh-mm-ssa"];
  175. gLoggingDateStamp = [formatter stringFromDate:[NSDate date]];
  176. }
  177. return gLoggingDateStamp;
  178. }
  179. + (NSString *)processNameLogPrefix {
  180. static NSString *gPrefix = nil;
  181. if (!gPrefix) {
  182. NSString *processName = [self loggingProcessName];
  183. gPrefix = [[NSString alloc] initWithFormat:@"%@_log_", processName];
  184. }
  185. return gPrefix;
  186. }
  187. + (NSString *)symlinkNameSuffix {
  188. return @"_log_newest.html";
  189. }
  190. + (NSString *)htmlFileName {
  191. return @"aperçu_http_log.html";
  192. }
  193. + (void)deleteLogDirectoriesOlderThanDate:(NSDate *)cutoffDate {
  194. NSFileManager *fileMgr = [NSFileManager defaultManager];
  195. NSURL *parentDir = [NSURL fileURLWithPath:[[self class] loggingDirectory]];
  196. NSURL *logDirectoryForCurrentRun =
  197. [NSURL fileURLWithPath:[[self class] logDirectoryForCurrentRun]];
  198. NSError *error;
  199. NSArray *contents = [fileMgr contentsOfDirectoryAtURL:parentDir
  200. includingPropertiesForKeys:@[ NSURLContentModificationDateKey ]
  201. options:0
  202. error:&error];
  203. for (NSURL *itemURL in contents) {
  204. if ([itemURL isEqual:logDirectoryForCurrentRun]) continue;
  205. NSDate *modDate;
  206. if ([itemURL getResourceValue:&modDate forKey:NSURLContentModificationDateKey error:&error]) {
  207. if ([modDate compare:cutoffDate] == NSOrderedAscending) {
  208. if (![fileMgr removeItemAtURL:itemURL error:&error]) {
  209. NSLog(@"deleteLogDirectoriesOlderThanDate failed to delete %@: %@", itemURL.path, error);
  210. }
  211. }
  212. } else {
  213. NSLog(@"deleteLogDirectoriesOlderThanDate failed to get mod date of %@: %@", itemURL.path,
  214. error);
  215. }
  216. }
  217. }
  218. // formattedStringFromData returns a prettyprinted string for XML or JSON input,
  219. // and a plain string for other input data
  220. - (NSString *)formattedStringFromData:(NSData *)inputData
  221. contentType:(NSString *)contentType
  222. JSON:(NSDictionary **)outJSON {
  223. if (!inputData) return nil;
  224. // if the content type is JSON and we have the parsing class available, use that
  225. if ([contentType hasPrefix:@"application/json"] && inputData.length > 5) {
  226. // convert from JSON string to NSObjects and back to a formatted string
  227. NSMutableDictionary *obj =
  228. [NSJSONSerialization JSONObjectWithData:inputData
  229. options:NSJSONReadingMutableContainers
  230. error:NULL];
  231. if (obj) {
  232. if (outJSON) *outJSON = obj;
  233. if ([obj isKindOfClass:[NSMutableDictionary class]]) {
  234. // for security and privacy, omit OAuth 2 response access and refresh tokens
  235. if ([obj valueForKey:@"refresh_token"] != nil) {
  236. [obj setObject:@"_snip_" forKey:@"refresh_token"];
  237. }
  238. if ([obj valueForKey:@"access_token"] != nil) {
  239. [obj setObject:@"_snip_" forKey:@"access_token"];
  240. }
  241. }
  242. NSData *data = [NSJSONSerialization dataWithJSONObject:obj
  243. options:NSJSONWritingPrettyPrinted
  244. error:NULL];
  245. if (data) {
  246. NSString *jsonStr = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
  247. return jsonStr;
  248. }
  249. }
  250. }
  251. #if !TARGET_OS_IPHONE && !GTM_SKIP_LOG_XMLFORMAT
  252. // verify that this data starts with the bytes indicating XML
  253. NSString *const kXMLLintPath = @"/usr/bin/xmllint";
  254. static BOOL gHasCheckedAvailability = NO;
  255. static BOOL gIsXMLLintAvailable = NO;
  256. if (!gHasCheckedAvailability) {
  257. gIsXMLLintAvailable = [[NSFileManager defaultManager] fileExistsAtPath:kXMLLintPath];
  258. gHasCheckedAvailability = YES;
  259. }
  260. if (gIsXMLLintAvailable && inputData.length > 5 && strncmp(inputData.bytes, "<?xml", 5) == 0) {
  261. // call xmllint to format the data
  262. NSTask *task = [[NSTask alloc] init];
  263. [task setLaunchPath:kXMLLintPath];
  264. // use the dash argument to specify stdin as the source file
  265. [task setArguments:@[ @"--format", @"-" ]];
  266. [task setEnvironment:@{}];
  267. NSPipe *inputPipe = [NSPipe pipe];
  268. NSPipe *outputPipe = [NSPipe pipe];
  269. [task setStandardInput:inputPipe];
  270. [task setStandardOutput:outputPipe];
  271. [task launch];
  272. [[inputPipe fileHandleForWriting] writeData:inputData];
  273. [[inputPipe fileHandleForWriting] closeFile];
  274. // drain the stdout before waiting for the task to exit
  275. NSData *formattedData = [[outputPipe fileHandleForReading] readDataToEndOfFile];
  276. [task waitUntilExit];
  277. int status = [task terminationStatus];
  278. if (status == 0 && formattedData.length > 0) {
  279. // success
  280. inputData = formattedData;
  281. }
  282. }
  283. #else
  284. // we can't call external tasks on the iPhone; leave the XML unformatted
  285. #endif
  286. NSString *dataStr = [[NSString alloc] initWithData:inputData encoding:NSUTF8StringEncoding];
  287. return dataStr;
  288. }
  289. // stringFromStreamData creates a string given the supplied data
  290. //
  291. // If NSString can create a UTF-8 string from the data, then that is returned.
  292. //
  293. // Otherwise, this routine tries to find a MIME boundary at the beginning of the data block, and
  294. // uses that to break up the data into parts. Each part will be used to try to make a UTF-8 string.
  295. // For parts that fail, a replacement string showing the part header and <<n bytes>> is supplied
  296. // in place of the binary data.
  297. - (NSString *)stringFromStreamData:(NSData *)data contentType:(NSString *)contentType {
  298. if (!data) return nil;
  299. // optimistically, see if the whole data block is UTF-8
  300. NSString *streamDataStr = [self formattedStringFromData:data contentType:contentType JSON:NULL];
  301. if (streamDataStr) return streamDataStr;
  302. // Munge a buffer by replacing non-ASCII bytes with underscores, and turn that munged buffer an
  303. // NSString. That gives us a string we can use with NSScanner.
  304. NSMutableData *mutableData = [NSMutableData dataWithData:data];
  305. unsigned char *bytes = (unsigned char *)mutableData.mutableBytes;
  306. for (unsigned int idx = 0; idx < mutableData.length; ++idx) {
  307. if (bytes[idx] > 0x7F || bytes[idx] == 0) {
  308. bytes[idx] = '_';
  309. }
  310. }
  311. NSString *mungedStr = [[NSString alloc] initWithData:mutableData encoding:NSUTF8StringEncoding];
  312. if (mungedStr) {
  313. // scan for the boundary string
  314. NSString *boundary = nil;
  315. NSScanner *scanner = [NSScanner scannerWithString:mungedStr];
  316. if ([scanner scanUpToString:@"\r\n" intoString:&boundary] && [boundary hasPrefix:@"--"]) {
  317. // we found a boundary string; use it to divide the string into parts
  318. NSArray *mungedParts = [mungedStr componentsSeparatedByString:boundary];
  319. // look at each munged part in the original string, and try to convert those into UTF-8
  320. NSMutableArray *origParts = [NSMutableArray array];
  321. NSUInteger offset = 0;
  322. for (NSString *mungedPart in mungedParts) {
  323. NSUInteger partSize = mungedPart.length;
  324. NSData *origPartData = [data subdataWithRange:NSMakeRange(offset, partSize)];
  325. NSString *origPartStr = [[NSString alloc] initWithData:origPartData
  326. encoding:NSUTF8StringEncoding];
  327. if (origPartStr) {
  328. // we could make this original part into UTF-8; use the string
  329. [origParts addObject:origPartStr];
  330. } else {
  331. // this part can't be made into UTF-8; scan the header, if we can
  332. NSString *header = nil;
  333. NSScanner *headerScanner = [NSScanner scannerWithString:mungedPart];
  334. if (![headerScanner scanUpToString:@"\r\n\r\n" intoString:&header]) {
  335. // we couldn't find a header
  336. header = @"";
  337. }
  338. // make a part string with the header and <<n bytes>>
  339. NSString *binStr = [NSString
  340. stringWithFormat:@"\r%@\r<<%lu bytes>>\r", header, (long)(partSize - header.length)];
  341. [origParts addObject:binStr];
  342. }
  343. offset += partSize + boundary.length;
  344. }
  345. // rejoin the original parts
  346. streamDataStr = [origParts componentsJoinedByString:boundary];
  347. }
  348. }
  349. if (!streamDataStr) {
  350. // give up; just make a string showing the uploaded bytes
  351. streamDataStr = [NSString stringWithFormat:@"<<%u bytes>>", (unsigned int)data.length];
  352. }
  353. return streamDataStr;
  354. }
  355. // logFetchWithError is called following a successful or failed fetch attempt
  356. //
  357. // This method does all the work for appending to and creating log files
  358. - (void)logFetchWithError:(NSError *)error {
  359. if (![[self class] isLoggingEnabled]) return;
  360. NSString *logDirectory = [[self class] logDirectoryForCurrentRun];
  361. if (!logDirectory) return;
  362. NSString *processName = [[self class] loggingProcessName];
  363. // TODO: add Javascript to display response data formatted in hex
  364. // each response's NSData goes into its own xml or txt file, though all responses for this run of
  365. // the app share a main html file. This counter tracks all fetch responses for this app run.
  366. //
  367. // we'll use a local variable since this routine may be reentered while waiting for XML formatting
  368. // to be completed by an external task
  369. static int gResponseCounter = 0;
  370. int responseCounter = ++gResponseCounter;
  371. NSURLResponse *response = [self response];
  372. NSDictionary *responseHeaders = [self responseHeaders];
  373. NSString *responseDataStr = nil;
  374. NSDictionary *responseJSON = nil;
  375. // if there's response data, decide what kind of file to put it in based on the first bytes of the
  376. // file or on the mime type supplied by the server
  377. NSString *responseMIMEType = [response MIMEType];
  378. BOOL isResponseImage = NO;
  379. // file name for an image data file
  380. NSString *responseDataFileName = nil;
  381. int64_t responseDataLength = self.downloadedLength;
  382. if (responseDataLength > 0) {
  383. NSData *downloadedData = self.downloadedData;
  384. if (downloadedData == nil && responseDataLength > 0 && responseDataLength < 20000 &&
  385. self.destinationFileURL) {
  386. // There's a download file that's not too big, so get the data to display from the downloaded
  387. // file.
  388. NSURL *destinationURL = self.destinationFileURL;
  389. downloadedData = [NSData dataWithContentsOfURL:destinationURL];
  390. }
  391. NSString *responseType = [responseHeaders valueForKey:@"Content-Type"];
  392. responseDataStr = [self formattedStringFromData:downloadedData
  393. contentType:responseType
  394. JSON:&responseJSON];
  395. NSString *responseDataExtn = nil;
  396. NSData *dataToWrite = nil;
  397. if (responseDataStr) {
  398. // we were able to make a UTF-8 string from the response data
  399. if ([responseMIMEType isEqual:@"application/atom+xml"] ||
  400. [responseMIMEType hasSuffix:@"/xml"]) {
  401. responseDataExtn = @"xml";
  402. dataToWrite = [responseDataStr dataUsingEncoding:NSUTF8StringEncoding];
  403. }
  404. } else if ([responseMIMEType isEqual:@"image/jpeg"]) {
  405. responseDataExtn = @"jpg";
  406. dataToWrite = downloadedData;
  407. isResponseImage = YES;
  408. } else if ([responseMIMEType isEqual:@"image/gif"]) {
  409. responseDataExtn = @"gif";
  410. dataToWrite = downloadedData;
  411. isResponseImage = YES;
  412. } else if ([responseMIMEType isEqual:@"image/png"]) {
  413. responseDataExtn = @"png";
  414. dataToWrite = downloadedData;
  415. isResponseImage = YES;
  416. } else {
  417. // add more non-text types here
  418. }
  419. // if we have an extension, save the raw data in a file with that extension
  420. if (responseDataExtn && dataToWrite) {
  421. // generate a response file base name like
  422. NSString *responseBaseName =
  423. [NSString stringWithFormat:@"fetch_%d_response", responseCounter];
  424. responseDataFileName = [responseBaseName stringByAppendingPathExtension:responseDataExtn];
  425. NSString *responseDataFilePath =
  426. [logDirectory stringByAppendingPathComponent:responseDataFileName];
  427. NSError *downloadedError = nil;
  428. if (gIsLoggingToFile && ![dataToWrite writeToFile:responseDataFilePath
  429. options:0
  430. error:&downloadedError]) {
  431. NSLog(@"%@ logging write error:%@ (%@)", [self class], downloadedError,
  432. responseDataFileName);
  433. }
  434. }
  435. }
  436. // we'll have one main html file per run of the app
  437. NSString *htmlName = [[self class] htmlFileName];
  438. NSString *htmlPath = [logDirectory stringByAppendingPathComponent:htmlName];
  439. // if the html file exists (from logging previous fetches) we don't need
  440. // to re-write the header or the scripts
  441. NSFileManager *fileMgr = [NSFileManager defaultManager];
  442. BOOL didFileExist = [fileMgr fileExistsAtPath:htmlPath];
  443. NSMutableString *outputHTML = [NSMutableString string];
  444. // we need a header to say we'll have UTF-8 text
  445. if (!didFileExist) {
  446. [outputHTML
  447. appendFormat:@"<html><head><meta http-equiv=\"content-type\" "
  448. "content=\"text/html; charset=UTF-8\"><title>%@ HTTP fetch log %@</title>",
  449. processName, [[self class] loggingDateStamp]];
  450. }
  451. // now write the visible html elements
  452. NSString *copyableFileName = [NSString stringWithFormat:@"fetch_%d.txt", responseCounter];
  453. NSDate *now = [NSDate date];
  454. // write the date & time, the comment, and the link to the plain-text (copyable) log
  455. [outputHTML appendFormat:@"<b>%@ &nbsp;&nbsp;&nbsp;&nbsp; ", now];
  456. NSString *comment = [self comment];
  457. if (comment.length > 0) {
  458. [outputHTML appendFormat:@"%@ &nbsp;&nbsp;&nbsp;&nbsp; ", comment];
  459. }
  460. [outputHTML
  461. appendFormat:@"</b><a href='%@'><i>request/response log</i></a><br>", copyableFileName];
  462. NSTimeInterval elapsed = -self.initialBeginFetchDate.timeIntervalSinceNow;
  463. [outputHTML appendFormat:@"elapsed: %5.3fsec<br>", elapsed];
  464. // write the request URL
  465. NSURLRequest *request = self.request;
  466. NSString *requestMethod = request.HTTPMethod;
  467. NSURL *requestURL = request.URL;
  468. // Save the request URL for next time in case this redirects.
  469. NSString *redirectedFromURLString = [self.redirectedFromURL absoluteString];
  470. self.redirectedFromURL = [requestURL copy];
  471. if (redirectedFromURLString) {
  472. [outputHTML appendFormat:@"<FONT COLOR='#990066'><i>redirected from %@</i></FONT><br>",
  473. redirectedFromURLString];
  474. }
  475. [outputHTML appendFormat:@"<b>request:</b> %@ <code>%@</code><br>\n", requestMethod, requestURL];
  476. // write the request headers
  477. NSDictionary *requestHeaders = request.allHTTPHeaderFields;
  478. NSUInteger numberOfRequestHeaders = requestHeaders.count;
  479. if (numberOfRequestHeaders > 0) {
  480. // Indicate if the request is authorized; warn if the request is authorized but non-SSL
  481. NSString *auth = [requestHeaders objectForKey:@"Authorization"];
  482. NSString *headerDetails = @"";
  483. if (auth) {
  484. BOOL isInsecure = [[requestURL scheme] isEqual:@"http"];
  485. if (isInsecure) {
  486. // 26A0 = ⚠
  487. headerDetails =
  488. @"&nbsp;&nbsp;&nbsp;<i>authorized, non-SSL</i><FONT COLOR='#FF00FF'> &#x26A0;</FONT> ";
  489. } else {
  490. headerDetails = @"&nbsp;&nbsp;&nbsp;<i>authorized</i>";
  491. }
  492. }
  493. NSString *cookiesHdr = [requestHeaders objectForKey:@"Cookie"];
  494. if (cookiesHdr) {
  495. headerDetails = [headerDetails stringByAppendingString:@"&nbsp;&nbsp;&nbsp;<i>cookies</i>"];
  496. }
  497. NSString *matchHdr = [requestHeaders objectForKey:@"If-Match"];
  498. if (matchHdr) {
  499. headerDetails = [headerDetails stringByAppendingString:@"&nbsp;&nbsp;&nbsp;<i>if-match</i>"];
  500. }
  501. matchHdr = [requestHeaders objectForKey:@"If-None-Match"];
  502. if (matchHdr) {
  503. headerDetails =
  504. [headerDetails stringByAppendingString:@"&nbsp;&nbsp;&nbsp;<i>if-none-match</i>"];
  505. }
  506. [outputHTML appendFormat:@"&nbsp;&nbsp; headers: %d %@<br>", (int)numberOfRequestHeaders,
  507. headerDetails];
  508. } else {
  509. [outputHTML appendFormat:@"&nbsp;&nbsp; headers: none<br>"];
  510. }
  511. // write the request post data
  512. NSData *bodyData = nil;
  513. NSData *loggedStreamData = self.loggedStreamData;
  514. if (loggedStreamData) {
  515. bodyData = loggedStreamData;
  516. } else {
  517. bodyData = self.bodyData;
  518. if (bodyData == nil) {
  519. bodyData = self.request.HTTPBody;
  520. }
  521. }
  522. uint64_t bodyDataLength = bodyData.length;
  523. if (bodyData.length == 0) {
  524. // If the data is in a body upload file URL, read that in if it's not huge.
  525. NSURL *bodyFileURL = self.bodyFileURL;
  526. if (bodyFileURL) {
  527. NSNumber *fileSizeNum = nil;
  528. NSError *fileSizeError = nil;
  529. if ([bodyFileURL getResourceValue:&fileSizeNum
  530. forKey:NSURLFileSizeKey
  531. error:&fileSizeError]) {
  532. bodyDataLength = [fileSizeNum unsignedLongLongValue];
  533. if (bodyDataLength > 0 && bodyDataLength < 50000) {
  534. bodyData = [NSData dataWithContentsOfURL:bodyFileURL
  535. options:NSDataReadingUncached
  536. error:&fileSizeError];
  537. }
  538. }
  539. }
  540. }
  541. NSString *bodyDataStr = nil;
  542. NSString *postType = [requestHeaders valueForKey:@"Content-Type"];
  543. if (bodyDataLength > 0) {
  544. [outputHTML appendFormat:@"&nbsp;&nbsp; data: %llu bytes, <code>%@</code><br>\n",
  545. bodyDataLength, postType ? postType : @"(no type)"];
  546. NSString *logRequestBody = self.logRequestBody;
  547. if (logRequestBody) {
  548. bodyDataStr = [logRequestBody copy];
  549. self.logRequestBody = nil;
  550. } else {
  551. bodyDataStr = [self stringFromStreamData:bodyData contentType:postType];
  552. if (bodyDataStr) {
  553. // remove OAuth 2 client secret and refresh token
  554. bodyDataStr = [[self class] snipSubstringOfString:bodyDataStr
  555. betweenStartString:@"client_secret="
  556. endString:@"&"];
  557. bodyDataStr = [[self class] snipSubstringOfString:bodyDataStr
  558. betweenStartString:@"refresh_token="
  559. endString:@"&"];
  560. // remove ClientLogin password
  561. bodyDataStr = [[self class] snipSubstringOfString:bodyDataStr
  562. betweenStartString:@"&Passwd="
  563. endString:@"&"];
  564. }
  565. }
  566. } else {
  567. // no post data
  568. }
  569. // write the response status, MIME type, URL
  570. NSInteger status = [self statusCode];
  571. if (response) {
  572. NSString *statusString = @"";
  573. if (status != 0) {
  574. if (status == 200 || status == 201) {
  575. statusString = [NSString stringWithFormat:@"%ld", (long)status];
  576. // report any JSON-RPC error
  577. if ([responseJSON isKindOfClass:[NSDictionary class]]) {
  578. NSDictionary *jsonError = [responseJSON objectForKey:@"error"];
  579. if ([jsonError isKindOfClass:[NSDictionary class]]) {
  580. NSString *jsonCode = [[jsonError valueForKey:@"code"] description];
  581. NSString *jsonMessage = [jsonError valueForKey:@"message"];
  582. if (jsonCode || jsonMessage) {
  583. // 2691 = ⚑
  584. NSString *const jsonErrFmt = @"&nbsp;&nbsp;&nbsp;<i>JSON error:</i> <FONT "
  585. @"COLOR='#FF00FF'>%@ %@ &nbsp;&#x2691;</FONT>";
  586. statusString =
  587. [statusString stringByAppendingFormat:jsonErrFmt, jsonCode ? jsonCode : @"",
  588. jsonMessage ? jsonMessage : @""];
  589. }
  590. }
  591. }
  592. } else {
  593. // purple for anything other than 200 or 201
  594. NSString *flag = status >= 400 ? @"&nbsp;&#x2691;" : @""; // 2691 = ⚑
  595. NSString *explanation = [NSHTTPURLResponse localizedStringForStatusCode:status];
  596. NSString *const statusFormat = @"<FONT COLOR='#FF00FF'>%ld %@ %@</FONT>";
  597. statusString = [NSString stringWithFormat:statusFormat, (long)status, explanation, flag];
  598. }
  599. }
  600. // show the response URL only if it's different from the request URL
  601. NSString *responseURLStr = @"";
  602. NSURL *responseURL = response.URL;
  603. if (responseURL && ![responseURL isEqual:request.URL]) {
  604. NSString *const responseURLFormat =
  605. @"<FONT COLOR='#FF00FF'>response URL:</FONT> <code>%@</code><br>\n";
  606. responseURLStr = [NSString stringWithFormat:responseURLFormat, [responseURL absoluteString]];
  607. }
  608. [outputHTML appendFormat:@"<b>response:</b>&nbsp;&nbsp;status %@<br>\n%@", statusString,
  609. responseURLStr];
  610. // Write the response headers
  611. NSUInteger numberOfResponseHeaders = responseHeaders.count;
  612. if (numberOfResponseHeaders > 0) {
  613. // Indicate if the server is setting cookies
  614. NSString *cookiesSet = [responseHeaders valueForKey:@"Set-Cookie"];
  615. NSString *cookiesStr =
  616. cookiesSet ? @"&nbsp;&nbsp;<FONT COLOR='#990066'><i>sets cookies</i></FONT>" : @"";
  617. // Indicate if the server is redirecting
  618. NSString *location = [responseHeaders valueForKey:@"Location"];
  619. BOOL isRedirect = status >= 300 && status <= 399 && location != nil;
  620. NSString *redirectsStr =
  621. isRedirect ? @"&nbsp;&nbsp;<FONT COLOR='#990066'><i>redirects</i></FONT>" : @"";
  622. [outputHTML appendFormat:@"&nbsp;&nbsp; headers: %d %@ %@<br>\n",
  623. (int)numberOfResponseHeaders, cookiesStr, redirectsStr];
  624. } else {
  625. [outputHTML appendString:@"&nbsp;&nbsp; headers: none<br>\n"];
  626. }
  627. }
  628. // error
  629. if (error) {
  630. [outputHTML appendFormat:@"<b>Error:</b> %@ <br>\n", error.description];
  631. }
  632. // Write the response data
  633. if (responseDataFileName) {
  634. if (isResponseImage) {
  635. // Make a small inline image that links to the full image file
  636. [outputHTML appendFormat:@"&nbsp;&nbsp; data: %lld bytes, <code>%@</code><br>",
  637. responseDataLength, responseMIMEType];
  638. NSString *const fmt = @"<a href=\"%@\"><img src='%@' alt='image' style='border:solid "
  639. @"thin;max-height:32'></a>\n";
  640. [outputHTML appendFormat:fmt, responseDataFileName, responseDataFileName];
  641. } else {
  642. // The response data was XML; link to the xml file
  643. NSString *const fmt = @"&nbsp;&nbsp; data: %lld bytes, "
  644. @"<code>%@</code>&nbsp;&nbsp;&nbsp;<i><a href=\"%@\">%@</a></i>\n";
  645. [outputHTML appendFormat:fmt, responseDataLength, responseMIMEType, responseDataFileName,
  646. [responseDataFileName pathExtension]];
  647. }
  648. } else {
  649. // The response data was not an image; just show the length and MIME type
  650. [outputHTML appendFormat:@"&nbsp;&nbsp; data: %lld bytes, <code>%@</code>\n",
  651. responseDataLength,
  652. responseMIMEType ? responseMIMEType : @"(no response type)"];
  653. }
  654. // Make a single string of the request and response, suitable for copying
  655. // to the clipboard and pasting into a bug report
  656. NSMutableString *copyable = [NSMutableString string];
  657. if (comment) {
  658. [copyable appendFormat:@"%@\n\n", comment];
  659. }
  660. [copyable appendFormat:@"%@ elapsed: %5.3fsec\n", now, elapsed];
  661. if (redirectedFromURLString) {
  662. [copyable appendFormat:@"Redirected from %@\n", redirectedFromURLString];
  663. }
  664. [copyable appendFormat:@"Request: %@ %@\n", requestMethod, requestURL];
  665. if (requestHeaders.count > 0) {
  666. [copyable appendFormat:@"Request headers:\n%@\n",
  667. [[self class] headersStringForDictionary:requestHeaders]];
  668. }
  669. if (bodyDataLength > 0) {
  670. [copyable appendFormat:@"Request body: (%llu bytes)\n", bodyDataLength];
  671. if (bodyDataStr) {
  672. [copyable appendFormat:@"%@\n", bodyDataStr];
  673. }
  674. [copyable appendString:@"\n"];
  675. }
  676. if (response) {
  677. [copyable appendFormat:@"Response: status %d\n", (int)status];
  678. [copyable appendFormat:@"Response headers:\n%@\n",
  679. [[self class] headersStringForDictionary:responseHeaders]];
  680. [copyable appendFormat:@"Response body: (%lld bytes)\n", responseDataLength];
  681. if (responseDataLength > 0) {
  682. NSString *logResponseBody = self.logResponseBody;
  683. if (logResponseBody) {
  684. // The user has provided the response body text.
  685. responseDataStr = [logResponseBody copy];
  686. self.logResponseBody = nil;
  687. }
  688. if (responseDataStr != nil) {
  689. [copyable appendFormat:@"%@\n", responseDataStr];
  690. } else {
  691. // Even though it's redundant, we'll put in text to indicate that all the bytes are binary.
  692. if (self.destinationFileURL) {
  693. [copyable appendFormat:@"<<%lld bytes>> to file %@\n", responseDataLength,
  694. self.destinationFileURL.path];
  695. } else {
  696. [copyable appendFormat:@"<<%lld bytes>>\n", responseDataLength];
  697. }
  698. }
  699. }
  700. }
  701. if (error) {
  702. [copyable appendFormat:@"Error: %@\n", error];
  703. }
  704. // Save to log property before adding the separator
  705. self.log = copyable;
  706. [copyable appendString:@"-----------------------------------------------------------\n"];
  707. // Write the copyable version to another file (linked to at the top of the html file, above)
  708. //
  709. // Ideally, something to just copy this to the clipboard like
  710. // <span onCopy='window.event.clipboardData.setData(\"Text\",
  711. // \"copyable stuff\");return false;'>Copy here.</span>"
  712. // would work everywhere, but it only works in Safari as of 8/2010
  713. if (gIsLoggingToFile) {
  714. NSString *parentDir = [[self class] loggingDirectory];
  715. NSString *copyablePath = [logDirectory stringByAppendingPathComponent:copyableFileName];
  716. NSError *copyableError = nil;
  717. if (![copyable writeToFile:copyablePath
  718. atomically:NO
  719. encoding:NSUTF8StringEncoding
  720. error:&copyableError]) {
  721. // Error writing to file
  722. NSLog(@"%@ logging write error:%@ (%@)", [self class], copyableError, copyablePath);
  723. }
  724. [outputHTML appendString:@"<br><hr><p>"];
  725. // Append the HTML to the main output file
  726. const char *htmlBytes = outputHTML.UTF8String;
  727. NSOutputStream *stream = [NSOutputStream outputStreamToFileAtPath:htmlPath append:YES];
  728. [stream open];
  729. [stream write:(const uint8_t *)htmlBytes maxLength:strlen(htmlBytes)];
  730. [stream close];
  731. // Make a symlink to the latest html
  732. NSString *const symlinkNameSuffix = [[self class] symlinkNameSuffix];
  733. NSString *symlinkName = [processName stringByAppendingString:symlinkNameSuffix];
  734. NSString *symlinkPath = [parentDir stringByAppendingPathComponent:symlinkName];
  735. [fileMgr removeItemAtPath:symlinkPath error:NULL];
  736. [fileMgr createSymbolicLinkAtPath:symlinkPath withDestinationPath:htmlPath error:NULL];
  737. #if TARGET_OS_IPHONE
  738. static BOOL gReportedLoggingPath = NO;
  739. if (!gReportedLoggingPath) {
  740. gReportedLoggingPath = YES;
  741. NSLog(@"GTMSessionFetcher logging to \"%@\"", parentDir);
  742. }
  743. #endif
  744. }
  745. }
  746. - (NSInputStream *)loggedInputStreamForInputStream:(NSInputStream *)inputStream {
  747. if (!inputStream) return nil;
  748. if (![GTMSessionFetcher isLoggingEnabled]) return inputStream;
  749. [self clearLoggedStreamData]; // Clear any previous data.
  750. Class monitorClass = NSClassFromString(@"GTMReadMonitorInputStream");
  751. if (!monitorClass) {
  752. NSString const *str = @"<<Uploaded stream log unavailable without GTMReadMonitorInputStream>>";
  753. NSData *stringData = [str dataUsingEncoding:NSUTF8StringEncoding];
  754. [self appendLoggedStreamData:stringData];
  755. return inputStream;
  756. }
  757. inputStream = [monitorClass inputStreamWithStream:inputStream];
  758. GTMReadMonitorInputStream *readMonitorInputStream = (GTMReadMonitorInputStream *)inputStream;
  759. [readMonitorInputStream setReadDelegate:self];
  760. SEL readSel = @selector(inputStream:readIntoBuffer:length:);
  761. [readMonitorInputStream setReadSelector:readSel];
  762. return inputStream;
  763. }
  764. - (GTMSessionFetcherBodyStreamProvider)loggedStreamProviderForStreamProvider:
  765. (GTMSessionFetcherBodyStreamProvider)streamProvider {
  766. if (!streamProvider) return nil;
  767. if (![GTMSessionFetcher isLoggingEnabled]) return streamProvider;
  768. [self clearLoggedStreamData]; // Clear any previous data.
  769. Class monitorClass = NSClassFromString(@"GTMReadMonitorInputStream");
  770. if (!monitorClass) {
  771. NSString const *str = @"<<Uploaded stream log unavailable without GTMReadMonitorInputStream>>";
  772. NSData *stringData = [str dataUsingEncoding:NSUTF8StringEncoding];
  773. [self appendLoggedStreamData:stringData];
  774. return streamProvider;
  775. }
  776. GTMSessionFetcherBodyStreamProvider loggedStreamProvider =
  777. ^(GTMSessionFetcherBodyStreamProviderResponse response) {
  778. streamProvider(^(NSInputStream *bodyStream) {
  779. bodyStream = [self loggedInputStreamForInputStream:bodyStream];
  780. response(bodyStream);
  781. });
  782. };
  783. return loggedStreamProvider;
  784. }
  785. @end
  786. @implementation GTMSessionFetcher (GTMSessionFetcherLoggingUtilities)
  787. - (void)inputStream:(GTMReadMonitorInputStream *)stream
  788. readIntoBuffer:(void *)buffer
  789. length:(int64_t)length {
  790. // append the captured data
  791. NSData *data = [NSData dataWithBytesNoCopy:buffer length:(NSUInteger)length freeWhenDone:NO];
  792. [self appendLoggedStreamData:data];
  793. }
  794. #pragma mark Fomatting Utilities
  795. + (NSString *)snipSubstringOfString:(NSString *)originalStr
  796. betweenStartString:(NSString *)startStr
  797. endString:(NSString *)endStr {
  798. #if SKIP_GTM_FETCH_LOGGING_SNIPPING
  799. return originalStr;
  800. #else
  801. if (!originalStr) return nil;
  802. // Find the start string, and replace everything between it
  803. // and the end string (or the end of the original string) with "_snip_"
  804. NSRange startRange = [originalStr rangeOfString:startStr];
  805. if (startRange.location == NSNotFound) return originalStr;
  806. // We found the start string
  807. NSUInteger originalLength = originalStr.length;
  808. NSUInteger startOfTarget = NSMaxRange(startRange);
  809. NSRange targetAndRest = NSMakeRange(startOfTarget, originalLength - startOfTarget);
  810. NSRange endRange = [originalStr rangeOfString:endStr options:0 range:targetAndRest];
  811. NSRange replaceRange;
  812. if (endRange.location == NSNotFound) {
  813. // Found no end marker so replace to end of string
  814. replaceRange = targetAndRest;
  815. } else {
  816. // Replace up to the endStr
  817. replaceRange = NSMakeRange(startOfTarget, endRange.location - startOfTarget);
  818. }
  819. NSString *result = [originalStr stringByReplacingCharactersInRange:replaceRange
  820. withString:@"_snip_"];
  821. return result;
  822. #endif // SKIP_GTM_FETCH_LOGGING_SNIPPING
  823. }
  824. + (NSString *)headersStringForDictionary:(NSDictionary *)dict {
  825. // Format the dictionary in http header style, like
  826. // Accept: application/json
  827. // Cache-Control: no-cache
  828. // Content-Type: application/json; charset=utf-8
  829. //
  830. // Pad the key names, but not beyond 16 chars, since long custom header
  831. // keys just create too much whitespace
  832. NSArray *keys = [dict.allKeys sortedArrayUsingSelector:@selector(compare:)];
  833. NSMutableString *str = [NSMutableString string];
  834. for (NSString *key in keys) {
  835. NSString *value = [dict valueForKey:key];
  836. if ([key isEqual:@"Authorization"]) {
  837. // Remove OAuth 1 token
  838. value = [[self class] snipSubstringOfString:value
  839. betweenStartString:@"oauth_token=\""
  840. endString:@"\""];
  841. // Remove OAuth 2 bearer token (draft 16, and older form)
  842. value = [[self class] snipSubstringOfString:value
  843. betweenStartString:@"Bearer "
  844. endString:@"\n"];
  845. value = [[self class] snipSubstringOfString:value
  846. betweenStartString:@"OAuth "
  847. endString:@"\n"];
  848. // Remove Google ClientLogin
  849. value = [[self class] snipSubstringOfString:value
  850. betweenStartString:@"GoogleLogin auth="
  851. endString:@"\n"];
  852. }
  853. [str appendFormat:@" %@: %@\n", key, value];
  854. }
  855. return str;
  856. }
  857. @end
  858. #endif // !STRIP_GTM_FETCH_LOGGING