FIRLoggerTest.m 10 KB

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