FIRLoggerTest.m 9.9 KB

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