FIRLoggerTest.m 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. // Copyright 2017 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 "FIRTestCase.h"
  17. // TODO - FIRLoggerTest should be split into a separate FIRLoggerTest and GULLoggerTest.
  18. // No test should include both includes.
  19. #import <FirebaseCore/FIRLogger.h>
  20. #import <GoogleUtilities/GULLogger.h>
  21. #import <asl.h>
  22. // The following constants are exposed from FIRLogger for unit tests.
  23. extern NSString *const kFIRDisableDebugModeApplicationArgument;
  24. extern NSString *const kFIREnableDebugModeApplicationArgument;
  25. /// Key for the debug mode bit in NSUserDefaults.
  26. extern NSString *const kFIRPersistedDebugModeKey;
  27. extern const char *kGULLoggerASLClientFacilityName;
  28. extern void FIRResetLogger(void);
  29. extern void FIRSetLoggerUserDefaults(NSUserDefaults *defaults);
  30. extern aslclient getGULLoggerClient(void);
  31. extern dispatch_queue_t getGULClientQueue(void);
  32. extern BOOL getGULLoggerDebugMode(void);
  33. static NSString *const kMessageCode = @"I-COR000001";
  34. @interface FIRLoggerTest : FIRTestCase
  35. @property(nonatomic) NSString *randomLogString;
  36. @property(nonatomic, strong) NSUserDefaults *defaults;
  37. @end
  38. @implementation FIRLoggerTest
  39. - (void)setUp {
  40. [super setUp];
  41. FIRResetLogger();
  42. // Stub NSUserDefaults for cleaner testing.
  43. _defaults = [[NSUserDefaults alloc] initWithSuiteName:@"com.firebase.logger_test"];
  44. FIRSetLoggerUserDefaults(_defaults);
  45. }
  46. - (void)tearDown {
  47. [super tearDown];
  48. _defaults = nil;
  49. }
  50. // Test some stable variables to make sure they weren't accidently changed.
  51. - (void)testStableVariables {
  52. // Strings of type FIRLoggerServices.
  53. XCTAssertEqualObjects(kFIRLoggerABTesting, @"[Firebase/ABTesting]");
  54. XCTAssertEqualObjects(kFIRLoggerAdMob, @"[Firebase/AdMob]");
  55. XCTAssertEqualObjects(kFIRLoggerAnalytics, @"[Firebase/Analytics]");
  56. XCTAssertEqualObjects(kFIRLoggerCore, @"[Firebase/Core]");
  57. XCTAssertEqualObjects(kFIRLoggerMLKit, @"[Firebase/MLKit]");
  58. XCTAssertEqualObjects(kFIRLoggerRemoteConfig, @"[Firebase/RemoteConfig]");
  59. }
  60. - (void)testInitializeASLForNonDebugMode {
  61. // Stub.
  62. id processInfoMock = [OCMockObject partialMockForObject:[NSProcessInfo processInfo]];
  63. NSArray *arguments = @[ kFIRDisableDebugModeApplicationArgument ];
  64. [[[processInfoMock stub] andReturn:arguments] arguments];
  65. // Test.
  66. FIRLogError(kFIRLoggerCore, kMessageCode, @"Some error.");
  67. // Assert.
  68. #if MAKE_THREAD_SAFE
  69. NSNumber *debugMode = [self.defaults objectForKey:kFIRPersistedDebugModeKey];
  70. XCTAssertNil(debugMode);
  71. XCTAssertFalse(getGULLoggerDebugMode());
  72. #endif
  73. // Stop.
  74. [processInfoMock stopMocking];
  75. }
  76. - (void)testInitializeASLForDebugModeWithArgument {
  77. // Stub.
  78. id processInfoMock = [OCMockObject partialMockForObject:[NSProcessInfo processInfo]];
  79. NSArray *arguments = @[ kFIREnableDebugModeApplicationArgument ];
  80. [[[processInfoMock stub] andReturn:arguments] arguments];
  81. // Test.
  82. FIRLogError(kFIRLoggerCore, kMessageCode, @"Some error.");
  83. #ifdef MAKE_THREAD_SAFE
  84. // Assert.
  85. NSNumber *debugMode = [self.defaults objectForKey:kGULPersistedDebugModeKey];
  86. XCTAssertTrue(debugMode.boolValue);
  87. XCTAssertTrue(getGULLoggerDebugMode());
  88. #endif
  89. // Stop.
  90. [processInfoMock stopMocking];
  91. }
  92. - (void)testInitializeASLForDebugModeWithUserDefaults {
  93. // Stub.
  94. NSNumber *debugMode = @YES;
  95. [self.defaults setBool:debugMode.boolValue forKey:kFIRPersistedDebugModeKey];
  96. // Test.
  97. GULLogError(@"my service", NO, kMessageCode, @"Some error.");
  98. // Assert.
  99. debugMode = [self.defaults objectForKey:kFIRPersistedDebugModeKey];
  100. XCTAssertTrue(debugMode.boolValue);
  101. }
  102. - (void)testMessageCodeFormat {
  103. // Valid case.
  104. XCTAssertNoThrow(FIRLogError(kFIRLoggerCore, @"I-APP000001", @"Message."));
  105. // An extra dash or missing dash should fail.
  106. XCTAssertThrows(FIRLogError(kFIRLoggerCore, @"I-APP-000001", @"Message."));
  107. XCTAssertThrows(FIRLogError(kFIRLoggerCore, @"IAPP000001", @"Message."));
  108. // Wrong number of digits should fail.
  109. XCTAssertThrows(FIRLogError(kFIRLoggerCore, @"I-APP00001", @"Message."));
  110. XCTAssertThrows(FIRLogError(kFIRLoggerCore, @"I-APP0000001", @"Message."));
  111. // Lowercase should fail.
  112. XCTAssertThrows(FIRLogError(kFIRLoggerCore, @"I-app000001", @"Message."));
  113. // nil or empty message code should fail.
  114. #pragma clang diagnostic push
  115. #pragma clang diagnostic ignored "-Wnonnull"
  116. XCTAssertThrows(FIRLogError(kFIRLoggerCore, nil, @"Message."));
  117. #pragma clang diagnostic pop
  118. XCTAssertThrows(FIRLogError(kFIRLoggerCore, @"", @"Message."));
  119. // Android message code should fail.
  120. XCTAssertThrows(FIRLogError(kFIRLoggerCore, @"A-APP000001", @"Message."));
  121. }
  122. - (void)testLoggerInterface {
  123. XCTAssertNoThrow(FIRLogError(kFIRLoggerCore, kMessageCode, @"Message."));
  124. XCTAssertNoThrow(FIRLogError(kFIRLoggerCore, kMessageCode, @"Configure %@.", @"blah"));
  125. XCTAssertNoThrow(FIRLogWarning(kFIRLoggerCore, kMessageCode, @"Message."));
  126. XCTAssertNoThrow(FIRLogWarning(kFIRLoggerCore, kMessageCode, @"Configure %@.", @"blah"));
  127. XCTAssertNoThrow(FIRLogNotice(kFIRLoggerCore, kMessageCode, @"Message."));
  128. XCTAssertNoThrow(FIRLogNotice(kFIRLoggerCore, kMessageCode, @"Configure %@.", @"blah"));
  129. XCTAssertNoThrow(FIRLogInfo(kFIRLoggerCore, kMessageCode, @"Message."));
  130. XCTAssertNoThrow(FIRLogInfo(kFIRLoggerCore, kMessageCode, @"Configure %@.", @"blah"));
  131. XCTAssertNoThrow(FIRLogDebug(kFIRLoggerCore, kMessageCode, @"Message."));
  132. XCTAssertNoThrow(FIRLogDebug(kFIRLoggerCore, kMessageCode, @"Configure %@.", @"blah"));
  133. }
  134. // asl_set_filter does not perform as expected in unit test environment with simulator. The
  135. // following test only checks whether the logs have been sent to system with the default settings in
  136. // the unit test environment.
  137. - (void)testSystemLogWithDefaultStatus {
  138. #if !(BUG128) // Disable until https://github.com/firebase/firebase-ios-sdk/issues/128 is fixed
  139. // Test fails on device and iOS 9 simulators - b/38130372
  140. return;
  141. #else
  142. // Sets the time interval that we need to wait in order to fetch all the logs.
  143. NSTimeInterval timeInterval = 0.1f;
  144. // Generates a random string each time and check whether it has been logged.
  145. // Log messages with Notice level and below should be logged to system/device by default.
  146. self.randomLogString = [NSUUID UUID].UUIDString;
  147. FIRLogError(kFIRLoggerCore, kMessageCode, @"%@", self.randomLogString);
  148. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  149. XCTAssertTrue([self logExists]);
  150. self.randomLogString = [NSUUID UUID].UUIDString;
  151. FIRLogWarning(kFIRLoggerCore, kMessageCode, @"%@", self.randomLogString);
  152. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  153. XCTAssertTrue([self logExists]);
  154. self.randomLogString = [NSUUID UUID].UUIDString;
  155. FIRLogNotice(kFIRLoggerCore, kMessageCode, @"%@", self.randomLogString);
  156. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  157. XCTAssertTrue([self logExists]);
  158. // Log messages with Info level and above should NOT be logged to system/device by default.
  159. self.randomLogString = [NSUUID UUID].UUIDString;
  160. FIRLogInfo(kFIRLoggerCore, kMessageCode, @"%@", self.randomLogString);
  161. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  162. XCTAssertFalse([self logExists]);
  163. self.randomLogString = [NSUUID UUID].UUIDString;
  164. FIRLogDebug(kFIRLoggerCore, kMessageCode, @"%@", self.randomLogString);
  165. [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:timeInterval]];
  166. XCTAssertFalse([self logExists]);
  167. #endif
  168. }
  169. // The FIRLoggerLevel enum must match the ASL_LEVEL_* constants, but we manually redefine
  170. // them in FIRLoggerLevel.h since we cannot include <asl.h> (see b/34976089 for more details).
  171. // This test ensures the constants match.
  172. - (void)testFIRLoggerLevelValues {
  173. XCTAssertEqual(FIRLoggerLevelError, ASL_LEVEL_ERR);
  174. XCTAssertEqual(FIRLoggerLevelWarning, ASL_LEVEL_WARNING);
  175. XCTAssertEqual(FIRLoggerLevelNotice, ASL_LEVEL_NOTICE);
  176. XCTAssertEqual(FIRLoggerLevelInfo, ASL_LEVEL_INFO);
  177. XCTAssertEqual(FIRLoggerLevelDebug, ASL_LEVEL_DEBUG);
  178. }
  179. // Helper functions.
  180. - (BOOL)logExists {
  181. [self drainFIRClientQueue];
  182. NSString *correctMsg =
  183. [NSString stringWithFormat:@"%@[%@] %@", kFIRLoggerCore, kMessageCode, self.randomLogString];
  184. return [self messageWasLogged:correctMsg];
  185. }
  186. - (void)drainFIRClientQueue {
  187. dispatch_semaphore_t workerSemaphore = dispatch_semaphore_create(0);
  188. dispatch_async(getGULClientQueue(), ^{
  189. dispatch_semaphore_signal(workerSemaphore);
  190. });
  191. dispatch_semaphore_wait(workerSemaphore, DISPATCH_TIME_FOREVER);
  192. }
  193. - (BOOL)messageWasLogged:(NSString *)message {
  194. #pragma clang diagnostic push
  195. #pragma clang diagnostic ignored "-Wdeprecated-declarations"
  196. aslmsg query = asl_new(ASL_TYPE_QUERY);
  197. asl_set_query(query, ASL_KEY_FACILITY, kGULLoggerASLClientFacilityName, ASL_QUERY_OP_EQUAL);
  198. aslresponse r = asl_search(getGULLoggerClient(), query);
  199. asl_free(query);
  200. aslmsg m;
  201. const char *val;
  202. NSMutableArray *allMsg = [[NSMutableArray alloc] init];
  203. while ((m = asl_next(r)) != NULL) {
  204. val = asl_get(m, ASL_KEY_MSG);
  205. if (val) {
  206. [allMsg addObject:[NSString stringWithUTF8String:val]];
  207. }
  208. }
  209. asl_free(m);
  210. asl_release(r);
  211. return [allMsg containsObject:message];
  212. #pragma clang pop
  213. }
  214. @end
  215. #endif