FIRCrashlyticsReportTests.m 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322
  1. // Copyright 2019 Google
  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. #import <Foundation/Foundation.h>
  15. #import <XCTest/XCTest.h>
  16. #import "Crashlytics/Crashlytics/Components/FIRCLSContext.h"
  17. #import "Crashlytics/Crashlytics/Components/FIRCLSGlobals.h"
  18. #import "Crashlytics/Crashlytics/Helpers/FIRCLSFile.h"
  19. #import "Crashlytics/Crashlytics/Models/FIRCLSInternalReport.h"
  20. #import "Crashlytics/Crashlytics/Private/FIRCrashlyticsReport_Private.h"
  21. #import "Crashlytics/Crashlytics/Public/FirebaseCrashlytics/FIRCrashlyticsReport.h"
  22. @interface FIRCrashlyticsReportTests : XCTestCase
  23. @end
  24. @implementation FIRCrashlyticsReportTests
  25. - (void)setUp {
  26. [super setUp];
  27. FIRCLSContextBaseInit();
  28. // these values must be set for the internals of logging to work
  29. _firclsContext.readonly->logging.userKVStorage.maxCount = 64;
  30. _firclsContext.readonly->logging.userKVStorage.maxIncrementalCount =
  31. FIRCLSUserLoggingMaxKVEntries;
  32. _firclsContext.readonly->logging.internalKVStorage.maxCount = 32;
  33. _firclsContext.readonly->logging.internalKVStorage.maxIncrementalCount = 16;
  34. _firclsContext.readonly->logging.logStorage.maxSize = 64 * 1000;
  35. _firclsContext.readonly->logging.logStorage.maxEntries = 0;
  36. _firclsContext.readonly->logging.logStorage.restrictBySize = true;
  37. _firclsContext.readonly->logging.logStorage.entryCount = NULL;
  38. _firclsContext.readonly->initialized = true;
  39. }
  40. - (void)tearDown {
  41. FIRCLSContextBaseDeinit();
  42. [super tearDown];
  43. }
  44. - (NSString *)resourcePath {
  45. return [[NSBundle bundleForClass:[self class]] resourcePath];
  46. }
  47. - (NSString *)pathForResource:(NSString *)name {
  48. return [[self resourcePath] stringByAppendingPathComponent:name];
  49. }
  50. - (FIRCLSInternalReport *)createTempCopyOfInternalReportWithName:(NSString *)name {
  51. NSString *tempPath = [NSTemporaryDirectory() stringByAppendingPathComponent:name];
  52. // make sure to remove anything that was there previously
  53. [[NSFileManager defaultManager] removeItemAtPath:tempPath error:nil];
  54. NSString *resourcePath = [self pathForResource:name];
  55. [[NSFileManager defaultManager] copyItemAtPath:resourcePath toPath:tempPath error:nil];
  56. return [[FIRCLSInternalReport alloc] initWithPath:tempPath];
  57. }
  58. - (FIRCrashlyticsReport *)createTempCopyOfReportWithName:(NSString *)name {
  59. FIRCLSInternalReport *internalReport = [self createTempCopyOfInternalReportWithName:name];
  60. return [[FIRCrashlyticsReport alloc] initWithInternalReport:internalReport];
  61. }
  62. #pragma mark - Public Getter Methods
  63. - (void)testPropertiesFromMetadatFile {
  64. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"metadata_only_report"];
  65. XCTAssertEqualObjects(@"772929a7f21f4ad293bb644668f257cd", report.reportID);
  66. XCTAssertEqualObjects([NSDate dateWithTimeIntervalSince1970:1423944888], report.dateCreated);
  67. }
  68. #pragma mark - Public Setter Methods
  69. - (void)testSetUserID {
  70. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"metadata_only_report"];
  71. [report setUserID:@"12345-6"];
  72. NSArray *entries = FIRCLSFileReadSections(
  73. [[report.internalReport pathForContentFile:FIRCLSReportInternalIncrementalKVFile]
  74. fileSystemRepresentation],
  75. false, nil);
  76. XCTAssertEqual([entries count], 1, @"");
  77. XCTAssertEqualObjects(entries[0][@"kv"][@"key"],
  78. FIRCLSFileHexEncodeString([FIRCLSUserIdentifierKey UTF8String]), @"");
  79. XCTAssertEqualObjects(entries[0][@"kv"][@"value"], FIRCLSFileHexEncodeString("12345-6"), @"");
  80. }
  81. - (void)testClearUserID {
  82. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"metadata_only_report"];
  83. // Add a user ID
  84. [report setUserID:@"12345-6"];
  85. NSArray *entries = FIRCLSFileReadSections(
  86. [[report.internalReport pathForContentFile:FIRCLSReportInternalIncrementalKVFile]
  87. fileSystemRepresentation],
  88. false, nil);
  89. XCTAssertEqual([entries count], 1, @"");
  90. // Now remove it
  91. [report setUserID:nil];
  92. entries = FIRCLSFileReadSections(
  93. [[report.internalReport pathForContentFile:FIRCLSReportInternalCompactedKVFile]
  94. fileSystemRepresentation],
  95. false, nil);
  96. XCTAssertEqual([entries count], 0, @"");
  97. }
  98. - (void)testCustomKeysNoExisting {
  99. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"metadata_only_report"];
  100. [report setCustomValue:@"hello" forKey:@"mykey"];
  101. [report setCustomValue:@"goodbye" forKey:@"anotherkey"];
  102. [report setCustomKeysAndValues:@{
  103. @"is_test" : @(YES),
  104. @"test_number" : @(10),
  105. }];
  106. NSArray *entries = FIRCLSFileReadSections(
  107. [[report.internalReport pathForContentFile:FIRCLSReportUserIncrementalKVFile]
  108. fileSystemRepresentation],
  109. false, nil);
  110. XCTAssertEqual([entries count], 4, @"");
  111. XCTAssertEqualObjects(entries[0][@"kv"][@"key"], FIRCLSFileHexEncodeString("mykey"), @"");
  112. XCTAssertEqualObjects(entries[0][@"kv"][@"value"], FIRCLSFileHexEncodeString("hello"), @"");
  113. XCTAssertEqualObjects(entries[1][@"kv"][@"key"], FIRCLSFileHexEncodeString("anotherkey"), @"");
  114. XCTAssertEqualObjects(entries[1][@"kv"][@"value"], FIRCLSFileHexEncodeString("goodbye"), @"");
  115. XCTAssertEqualObjects(entries[2][@"kv"][@"key"], FIRCLSFileHexEncodeString("is_test"), @"");
  116. XCTAssertEqualObjects(entries[2][@"kv"][@"value"], FIRCLSFileHexEncodeString("1"), @"");
  117. XCTAssertEqualObjects(entries[3][@"kv"][@"key"], FIRCLSFileHexEncodeString("test_number"), @"");
  118. XCTAssertEqualObjects(entries[3][@"kv"][@"value"], FIRCLSFileHexEncodeString("10"), @"");
  119. }
  120. - (void)testCustomKeysWithExisting {
  121. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"ios_all_files_crash"];
  122. [report setCustomValue:@"hello" forKey:@"mykey"];
  123. [report setCustomValue:@"goodbye" forKey:@"anotherkey"];
  124. [report setCustomKeysAndValues:@{
  125. @"is_test" : @(YES),
  126. @"test_number" : @(10),
  127. }];
  128. NSArray *entries = FIRCLSFileReadSections(
  129. [[report.internalReport pathForContentFile:FIRCLSReportUserIncrementalKVFile]
  130. fileSystemRepresentation],
  131. false, nil);
  132. XCTAssertEqual([entries count], 5, @"");
  133. XCTAssertEqualObjects(entries[1][@"kv"][@"key"], FIRCLSFileHexEncodeString("mykey"), @"");
  134. XCTAssertEqualObjects(entries[1][@"kv"][@"value"], FIRCLSFileHexEncodeString("hello"), @"");
  135. XCTAssertEqualObjects(entries[2][@"kv"][@"key"], FIRCLSFileHexEncodeString("anotherkey"), @"");
  136. XCTAssertEqualObjects(entries[2][@"kv"][@"value"], FIRCLSFileHexEncodeString("goodbye"), @"");
  137. XCTAssertEqualObjects(entries[3][@"kv"][@"key"], FIRCLSFileHexEncodeString("is_test"), @"");
  138. XCTAssertEqualObjects(entries[3][@"kv"][@"value"], FIRCLSFileHexEncodeString("1"), @"");
  139. XCTAssertEqualObjects(entries[4][@"kv"][@"key"], FIRCLSFileHexEncodeString("test_number"), @"");
  140. XCTAssertEqualObjects(entries[4][@"kv"][@"value"], FIRCLSFileHexEncodeString("10"), @"");
  141. }
  142. - (void)testClearCustomKeys {
  143. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"metadata_only_report"];
  144. // Add keys
  145. [report setCustomValue:@"hello" forKey:@"mykey"];
  146. [report setCustomValue:@"goodbye" forKey:@"anotherkey"];
  147. [report setCustomKeysAndValues:@{
  148. @"is_test" : @(YES),
  149. @"test_number" : @(10),
  150. }];
  151. NSArray *entries = FIRCLSFileReadSections(
  152. [[report.internalReport pathForContentFile:FIRCLSReportUserIncrementalKVFile]
  153. fileSystemRepresentation],
  154. false, nil);
  155. XCTAssertEqual([entries count], 4, @"");
  156. // Now remove them
  157. [report setCustomValue:nil forKey:@"mykey"];
  158. [report setCustomValue:nil forKey:@"anotherkey"];
  159. entries = FIRCLSFileReadSections(
  160. [[report.internalReport pathForContentFile:FIRCLSReportUserIncrementalKVFile]
  161. fileSystemRepresentation],
  162. false, nil);
  163. [report setCustomKeysAndValues:@{
  164. @"is_test" : [NSNull null],
  165. @"test_number" : [NSNull null],
  166. }];
  167. entries = FIRCLSFileReadSections(
  168. [[report.internalReport pathForContentFile:FIRCLSReportInternalCompactedKVFile]
  169. fileSystemRepresentation],
  170. false, nil);
  171. XCTAssertEqual([entries count], 0, @"");
  172. }
  173. - (void)testCustomKeysLimits {
  174. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"ios_all_files_crash"];
  175. // Write a bunch of keys and values
  176. for (int i = 0; i < 120; i++) {
  177. NSString *key = [NSString stringWithFormat:@"key_%i", i];
  178. [report setCustomValue:@"hello" forKey:key];
  179. }
  180. NSArray *entriesI = FIRCLSFileReadSections(
  181. [[report.internalReport pathForContentFile:FIRCLSReportUserIncrementalKVFile]
  182. fileSystemRepresentation],
  183. false, nil);
  184. NSArray *entriesC = FIRCLSFileReadSections(
  185. [[report.internalReport pathForContentFile:FIRCLSReportUserCompactedKVFile]
  186. fileSystemRepresentation],
  187. false, nil);
  188. // One of these should be the max (64), and one should be the number of written keys modulo 64
  189. // (eg. 56 == (120 mod 64))
  190. XCTAssertEqual(entriesI.count, 56, @"");
  191. XCTAssertEqual(entriesC.count, 64, @"");
  192. }
  193. - (void)testLogsNoExisting {
  194. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"metadata_only_report"];
  195. [report log:@"Normal log without formatting"];
  196. [report logWithFormat:@"%@, %@", @"First", @"Second"];
  197. NSArray *entries = FIRCLSFileReadSections(
  198. [[report.internalReport pathForContentFile:FIRCLSReportLogAFile] fileSystemRepresentation],
  199. false, nil);
  200. XCTAssertEqual([entries count], 2, @"");
  201. XCTAssertEqualObjects(entries[0][@"log"][@"msg"],
  202. FIRCLSFileHexEncodeString("Normal log without formatting"), @"");
  203. XCTAssertEqualObjects(entries[1][@"log"][@"msg"], FIRCLSFileHexEncodeString("First, Second"),
  204. @"");
  205. }
  206. - (void)testLogsWithExisting {
  207. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"ios_all_files_crash"];
  208. [report log:@"Normal log without formatting"];
  209. [report logWithFormat:@"%@, %@", @"First", @"Second"];
  210. NSArray *entries = FIRCLSFileReadSections(
  211. [[report.internalReport pathForContentFile:FIRCLSReportLogAFile] fileSystemRepresentation],
  212. false, nil);
  213. XCTAssertEqual([entries count], 8, @"");
  214. XCTAssertEqualObjects(entries[6][@"log"][@"msg"],
  215. FIRCLSFileHexEncodeString("Normal log without formatting"), @"");
  216. XCTAssertEqualObjects(entries[7][@"log"][@"msg"], FIRCLSFileHexEncodeString("First, Second"),
  217. @"");
  218. }
  219. - (void)testLogLimits {
  220. FIRCrashlyticsReport *report = [self createTempCopyOfReportWithName:@"metadata_only_report"];
  221. for (int i = 0; i < 2000; i++) {
  222. [report log:@"0123456789"];
  223. }
  224. unsigned long long sizeA = [[[NSFileManager defaultManager]
  225. attributesOfItemAtPath:[report.internalReport pathForContentFile:FIRCLSReportLogAFile]
  226. error:nil] fileSize];
  227. unsigned long long sizeB = [[[NSFileManager defaultManager]
  228. attributesOfItemAtPath:[report.internalReport pathForContentFile:FIRCLSReportLogBFile]
  229. error:nil] fileSize];
  230. NSArray *entriesA = FIRCLSFileReadSections(
  231. [[report.internalReport pathForContentFile:FIRCLSReportLogAFile] fileSystemRepresentation],
  232. false, nil);
  233. NSArray *entriesB = FIRCLSFileReadSections(
  234. [[report.internalReport pathForContentFile:FIRCLSReportLogBFile] fileSystemRepresentation],
  235. false, nil);
  236. // If these numbers have changed, the goal is to validate that the size of log_a and log_b are
  237. // under the limit, logStorage.maxSize (64 * 1000). These numbers don't need to be exact so if
  238. // they fluctuate then we might just need to accept a range in these tests.
  239. XCTAssertEqual(entriesB.count + entriesA.count, 2000, @"");
  240. XCTAssertEqual(sizeA, 64 * 1000 + 20, @"");
  241. XCTAssertEqual(sizeB, 55980, @"");
  242. }
  243. @end