GULLoggerTest.m 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. // Copyright 2018 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. #ifdef DEBUG
  15. // The tests depend upon library methods only built with #ifdef DEBUG
  16. #import <OCMock/OCMock.h>
  17. #import <XCTest/XCTest.h>
  18. #import <GoogleUtilities/GULLogger.h>
  19. #import <asl.h>
  20. extern const char *kGULLoggerASLClientFacilityName;
  21. extern void GULResetLogger(void);
  22. extern aslclient getGULLoggerClient(void);
  23. extern dispatch_queue_t getGULClientQueue(void);
  24. extern BOOL getGULLoggerDebugMode(void);
  25. static NSString *const kMessageCode = @"I-COR000001";
  26. @interface GULLoggerTest : XCTestCase
  27. @property(nonatomic) NSString *randomLogString;
  28. @property(nonatomic, strong) NSUserDefaults *defaults;
  29. @end
  30. @implementation GULLoggerTest
  31. - (void)setUp {
  32. [super setUp];
  33. GULResetLogger();
  34. // Stub NSUserDefaults for cleaner testing.
  35. _defaults = [[NSUserDefaults alloc] initWithSuiteName:@"com.google.logger_test"];
  36. }
  37. - (void)tearDown {
  38. [super tearDown];
  39. _defaults = nil;
  40. }
  41. - (void)testMessageCodeFormat {
  42. // Valid case.
  43. XCTAssertNoThrow(GULLogError(@"my service", NO, @"I-APP000001", @"Message."));
  44. // An extra dash or missing dash should fail.
  45. XCTAssertThrows(GULLogError(@"my service", NO, @"I-APP-000001", @"Message."));
  46. XCTAssertThrows(GULLogError(@"my service", NO, @"IAPP000001", @"Message."));
  47. // Wrong number of digits should fail.
  48. XCTAssertThrows(GULLogError(@"my service", NO, @"I-APP00001", @"Message."));
  49. XCTAssertThrows(GULLogError(@"my service", NO, @"I-APP0000001", @"Message."));
  50. // Lowercase should fail.
  51. XCTAssertThrows(GULLogError(@"my service", NO, @"I-app000001", @"Message."));
  52. // nil or empty message code should fail.
  53. #pragma clang diagnostic push
  54. #pragma clang diagnostic ignored "-Wnonnull"
  55. XCTAssertThrows(GULLogError(@"my service", NO, nil, @"Message."));
  56. #pragma clang diagnostic pop
  57. XCTAssertThrows(GULLogError(@"my service", NO, @"", @"Message."));
  58. // Android message code should fail.
  59. XCTAssertThrows(GULLogError(@"my service", NO, @"A-APP000001", @"Message."));
  60. }
  61. - (void)testLoggerInterface {
  62. XCTAssertNoThrow(GULLogError(@"my service", NO, kMessageCode, @"Message."));
  63. XCTAssertNoThrow(GULLogError(@"my service", NO, kMessageCode, @"Configure %@.", @"blah"));
  64. XCTAssertNoThrow(GULLogWarning(@"my service", NO, kMessageCode, @"Message."));
  65. XCTAssertNoThrow(GULLogWarning(@"my service", NO, kMessageCode, @"Configure %@.", @"blah"));
  66. XCTAssertNoThrow(GULLogNotice(@"my service", NO, kMessageCode, @"Message."));
  67. XCTAssertNoThrow(GULLogNotice(@"my service", NO, kMessageCode, @"Configure %@.", @"blah"));
  68. XCTAssertNoThrow(GULLogInfo(@"my service", NO, kMessageCode, @"Message."));
  69. XCTAssertNoThrow(GULLogInfo(@"my service", NO, kMessageCode, @"Configure %@.", @"blah"));
  70. XCTAssertNoThrow(GULLogDebug(@"my service", NO, kMessageCode, @"Message."));
  71. XCTAssertNoThrow(GULLogDebug(@"my service", NO, kMessageCode, @"Configure %@.", @"blah"));
  72. }
  73. // asl_set_filter does not perform as expected in unit test environment with simulator. The
  74. // following test only checks whether the logs have been sent to system with the default settings in
  75. // the unit test environment.
  76. - (void)testSystemLogWithDefaultStatus {
  77. #if !(BUG128) // Disable until https://github.com/firebase/firebase-ios-sdk/issues/128 is fixed
  78. // Test fails on device and iOS 9 simulators - b/38130372
  79. return;
  80. #else
  81. // Sets the time interval that we need to wait in order to fetch all the logs.
  82. NSTimeInterval timeInterval = 0.1f;
  83. // Generates a random string each time and check whether it has been logged.
  84. // Log messages with Notice level and below should be logged to system/device by default.
  85. self.randomLogString = [NSUUID UUID].UUIDString;
  86. GULLogError(@"my service", NO, kMessageCode, @"%@", self.randomLogString);
  87. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  88. XCTAssertTrue([self logExists]);
  89. self.randomLogString = [NSUUID UUID].UUIDString;
  90. GULLogWarning(@"my service", kMessageCode, @"%@", self.randomLogString);
  91. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  92. XCTAssertTrue([self logExists]);
  93. self.randomLogString = [NSUUID UUID].UUIDString;
  94. GULLogNotice(@"my service", kMessageCode, @"%@", self.randomLogString);
  95. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  96. XCTAssertTrue([self logExists]);
  97. // Log messages with Info level and above should NOT be logged to system/device by default.
  98. self.randomLogString = [NSUUID UUID].UUIDString;
  99. GULLogInfo(@"my service", kMessageCode, @"%@", self.randomLogString);
  100. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  101. XCTAssertFalse([self logExists]);
  102. self.randomLogString = [NSUUID UUID].UUIDString;
  103. GULLogDebug(@"my service", kMessageCode, @"%@", self.randomLogString);
  104. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  105. XCTAssertFalse([self logExists]);
  106. #endif
  107. }
  108. // The GULLoggerLevel enum must match the ASL_LEVEL_* constants, but we manually redefine
  109. // them in GULLoggerLevel.h since we cannot include <asl.h> (see b/34976089 for more details).
  110. // This test ensures the constants match.
  111. - (void)testGULLoggerLevelValues {
  112. XCTAssertEqual(GULLoggerLevelError, ASL_LEVEL_ERR);
  113. XCTAssertEqual(GULLoggerLevelWarning, ASL_LEVEL_WARNING);
  114. XCTAssertEqual(GULLoggerLevelNotice, ASL_LEVEL_NOTICE);
  115. XCTAssertEqual(GULLoggerLevelInfo, ASL_LEVEL_INFO);
  116. XCTAssertEqual(GULLoggerLevelDebug, ASL_LEVEL_DEBUG);
  117. }
  118. // Helper functions.
  119. - (BOOL)logExists {
  120. [self drainGULClientQueue];
  121. NSString *correctMsg =
  122. [NSString stringWithFormat:@"%@[%@] %@", @"my service", kMessageCode, self.randomLogString];
  123. return [self messageWasLogged:correctMsg];
  124. }
  125. - (void)drainGULClientQueue {
  126. dispatch_semaphore_t workerSemaphore = dispatch_semaphore_create(0);
  127. dispatch_async(getGULClientQueue(), ^{
  128. dispatch_semaphore_signal(workerSemaphore);
  129. });
  130. dispatch_semaphore_wait(workerSemaphore, DISPATCH_TIME_FOREVER);
  131. }
  132. - (BOOL)messageWasLogged:(NSString *)message {
  133. #pragma clang diagnostic push
  134. #pragma clang diagnostic ignored "-Wdeprecated-declarations"
  135. aslmsg query = asl_new(ASL_TYPE_QUERY);
  136. asl_set_query(query, ASL_KEY_FACILITY, kGULLoggerASLClientFacilityName, ASL_QUERY_OP_EQUAL);
  137. aslresponse r = asl_search(getGULLoggerClient(), query);
  138. asl_free(query);
  139. aslmsg m;
  140. const char *val;
  141. NSMutableArray *allMsg = [[NSMutableArray alloc] init];
  142. while ((m = asl_next(r)) != NULL) {
  143. val = asl_get(m, ASL_KEY_MSG);
  144. if (val) {
  145. [allMsg addObject:[NSString stringWithUTF8String:val]];
  146. }
  147. }
  148. asl_free(m);
  149. asl_release(r);
  150. return [allMsg containsObject:message];
  151. #pragma clang pop
  152. }
  153. @end
  154. #endif