FIRInstallationsIDController.m 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481
  1. /*
  2. * Copyright 2019 Google
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #import "FIRInstallationsIDController.h"
  17. #if __has_include(<FBLPromises/FBLPromises.h>)
  18. #import <FBLPromises/FBLPromises.h>
  19. #else
  20. #import "FBLPromises.h"
  21. #endif
  22. #import <FirebaseCore/FIRAppInternal.h>
  23. #import "FIRInstallationsAPIService.h"
  24. #import "FIRInstallationsErrorUtil.h"
  25. #import "FIRInstallationsIIDStore.h"
  26. #import "FIRInstallationsItem.h"
  27. #import "FIRInstallationsLogger.h"
  28. #import "FIRInstallationsSingleOperationPromiseCache.h"
  29. #import "FIRInstallationsStore.h"
  30. #import "FIRSecureStorage.h"
  31. #import "FIRInstallationsHTTPError.h"
  32. #import "FIRInstallationsStoredAuthToken.h"
  33. #import "FIRInstallationsStoredRegistrationError.h"
  34. #import "FIRInstallationsStoredRegistrationParameters.h"
  35. const NSNotificationName FIRInstallationIDDidChangeNotification =
  36. @"FIRInstallationIDDidChangeNotification";
  37. NSString *const kFIRInstallationIDDidChangeNotificationAppNameKey =
  38. @"FIRInstallationIDDidChangeNotification";
  39. NSTimeInterval const kFIRInstallationsTokenExpirationThreshold = 60 * 60; // 1 hour.
  40. NSTimeInterval const kFIRInstallationsRegistrationErrorTimeout = 24 * 60 * 60; // 1 day.
  41. @interface FIRInstallationsIDController ()
  42. @property(nonatomic, readonly) NSString *appID;
  43. @property(nonatomic, readonly) NSString *appName;
  44. @property(nonatomic, readonly) FIRInstallationsStore *installationsStore;
  45. @property(nonatomic, readonly) FIRInstallationsIIDStore *IIDStore;
  46. @property(nonatomic, readonly) FIRInstallationsAPIService *APIService;
  47. @property(nonatomic, readonly) FIRInstallationsSingleOperationPromiseCache<FIRInstallationsItem *>
  48. *getInstallationPromiseCache;
  49. @property(nonatomic, readonly)
  50. FIRInstallationsSingleOperationPromiseCache<FIRInstallationsItem *> *authTokenPromiseCache;
  51. @property(nonatomic, readonly) FIRInstallationsSingleOperationPromiseCache<FIRInstallationsItem *>
  52. *authTokenForcingRefreshPromiseCache;
  53. @property(nonatomic, readonly)
  54. FIRInstallationsSingleOperationPromiseCache<NSNull *> *deleteInstallationPromiseCache;
  55. @end
  56. @implementation FIRInstallationsIDController
  57. - (instancetype)initWithGoogleAppID:(NSString *)appID
  58. appName:(NSString *)appName
  59. APIKey:(NSString *)APIKey
  60. projectID:(NSString *)projectID {
  61. FIRSecureStorage *secureStorage = [[FIRSecureStorage alloc] init];
  62. FIRInstallationsStore *installationsStore =
  63. [[FIRInstallationsStore alloc] initWithSecureStorage:secureStorage accessGroup:nil];
  64. FIRInstallationsAPIService *apiService =
  65. [[FIRInstallationsAPIService alloc] initWithAPIKey:APIKey projectID:projectID];
  66. FIRInstallationsIIDStore *IIDStore = [[FIRInstallationsIIDStore alloc] init];
  67. return [self initWithGoogleAppID:appID
  68. appName:appName
  69. installationsStore:installationsStore
  70. APIService:apiService
  71. IIDStore:IIDStore];
  72. }
  73. /// The initializer is supposed to be used by tests to inject `installationsStore`.
  74. - (instancetype)initWithGoogleAppID:(NSString *)appID
  75. appName:(NSString *)appName
  76. installationsStore:(FIRInstallationsStore *)installationsStore
  77. APIService:(FIRInstallationsAPIService *)APIService
  78. IIDStore:(FIRInstallationsIIDStore *)IIDStore {
  79. self = [super init];
  80. if (self) {
  81. _appID = appID;
  82. _appName = appName;
  83. _installationsStore = installationsStore;
  84. _APIService = APIService;
  85. _IIDStore = IIDStore;
  86. __weak FIRInstallationsIDController *weakSelf = self;
  87. _getInstallationPromiseCache = [[FIRInstallationsSingleOperationPromiseCache alloc]
  88. initWithNewOperationHandler:^FBLPromise *_Nonnull {
  89. FIRInstallationsIDController *strongSelf = weakSelf;
  90. return [strongSelf createGetInstallationItemPromise];
  91. }];
  92. _authTokenPromiseCache = [[FIRInstallationsSingleOperationPromiseCache alloc]
  93. initWithNewOperationHandler:^FBLPromise *_Nonnull {
  94. FIRInstallationsIDController *strongSelf = weakSelf;
  95. return [strongSelf installationWithValidAuthTokenForcingRefresh:NO];
  96. }];
  97. _authTokenForcingRefreshPromiseCache = [[FIRInstallationsSingleOperationPromiseCache alloc]
  98. initWithNewOperationHandler:^FBLPromise *_Nonnull {
  99. FIRInstallationsIDController *strongSelf = weakSelf;
  100. return [strongSelf installationWithValidAuthTokenForcingRefresh:YES];
  101. }];
  102. _deleteInstallationPromiseCache = [[FIRInstallationsSingleOperationPromiseCache alloc]
  103. initWithNewOperationHandler:^FBLPromise *_Nonnull {
  104. FIRInstallationsIDController *strongSelf = weakSelf;
  105. return [strongSelf createDeleteInstallationPromise];
  106. }];
  107. }
  108. return self;
  109. }
  110. #pragma mark - Get Installation.
  111. - (FBLPromise<FIRInstallationsItem *> *)getInstallationItem {
  112. return [self.getInstallationPromiseCache getExistingPendingOrCreateNewPromise];
  113. }
  114. - (FBLPromise<FIRInstallationsItem *> *)createGetInstallationItemPromise {
  115. FIRLogDebug(kFIRLoggerInstallations,
  116. kFIRInstallationsMessageCodeNewGetInstallationOperationCreated, @"%s, appName: %@",
  117. __PRETTY_FUNCTION__, self.appName);
  118. FBLPromise<FIRInstallationsItem *> *installationItemPromise =
  119. [self getStoredInstallation]
  120. .recover(^id(NSError *error) {
  121. return [self createAndSaveFID];
  122. })
  123. .then(^id(FIRInstallationsItem *installation) {
  124. // Validate if a previous registration attempt failed with an error requiring Firebase
  125. // configuration changes.
  126. if (installation.registrationStatus == FIRInstallationStatusRegistrationFailed &&
  127. [self isRegistrationErrorWithDateUpToDate:installation.registrationError.date] &&
  128. [self areInstallationRegistrationParametersEqualToCurrent:
  129. installation.registrationError.registrationParameters]) {
  130. return installation.registrationError.APIError;
  131. }
  132. return installation;
  133. });
  134. // Initiate registration process on success if needed, but return the installation without waiting
  135. // for it.
  136. installationItemPromise.then(^id(FIRInstallationsItem *installation) {
  137. [self getAuthTokenForcingRefresh:NO];
  138. return nil;
  139. });
  140. return installationItemPromise;
  141. }
  142. - (FBLPromise<FIRInstallationsItem *> *)getStoredInstallation {
  143. return [self.installationsStore installationForAppID:self.appID appName:self.appName].validate(
  144. ^BOOL(FIRInstallationsItem *installation) {
  145. BOOL isValid = NO;
  146. switch (installation.registrationStatus) {
  147. case FIRInstallationStatusUnregistered:
  148. case FIRInstallationStatusRegistered:
  149. case FIRInstallationStatusRegistrationFailed:
  150. isValid = YES;
  151. break;
  152. case FIRInstallationStatusUnknown:
  153. isValid = NO;
  154. break;
  155. }
  156. return isValid;
  157. });
  158. }
  159. - (FBLPromise<FIRInstallationsItem *> *)createAndSaveFID {
  160. return [self migrateOrGenerateFID]
  161. .then(^FBLPromise<FIRInstallationsItem *> *(NSString *FID) {
  162. return [self createAndSaveInstallationWithFID:FID];
  163. })
  164. .then(^FIRInstallationsItem *(FIRInstallationsItem *installation) {
  165. [self postFIDDidChangeNotification];
  166. return installation;
  167. });
  168. }
  169. - (FBLPromise<FIRInstallationsItem *> *)createAndSaveInstallationWithFID:(NSString *)FID {
  170. FIRInstallationsItem *installation = [[FIRInstallationsItem alloc] initWithAppID:self.appID
  171. firebaseAppName:self.appName];
  172. installation.firebaseInstallationID = FID;
  173. installation.registrationStatus = FIRInstallationStatusUnregistered;
  174. return [self.installationsStore saveInstallation:installation].then(^id(NSNull *result) {
  175. return installation;
  176. });
  177. }
  178. - (FBLPromise<NSString *> *)migrateOrGenerateFID {
  179. if (![self isDefaultApp]) {
  180. // Existing IID should be used only for default FirebaseApp.
  181. return [FBLPromise resolvedWith:[FIRInstallationsItem generateFID]];
  182. }
  183. return [self.IIDStore existingIID].recover(^NSString *(NSError *error) {
  184. return [FIRInstallationsItem generateFID];
  185. });
  186. }
  187. - (BOOL)areInstallationRegistrationParametersEqualToCurrent:
  188. (FIRInstallationsStoredRegistrationParameters *)parameters {
  189. NSString *APIKey = self.APIService.APIKey;
  190. NSString *projectID = self.APIService.projectID;
  191. return (parameters.APIKey == APIKey || [parameters.APIKey isEqual:APIKey]) &&
  192. (parameters.projectID == projectID || [parameters.projectID isEqual:projectID]);
  193. }
  194. - (BOOL)isRegistrationErrorWithDateUpToDate:(NSDate *)errorDate {
  195. return errorDate != nil &&
  196. -[errorDate timeIntervalSinceNow] <= kFIRInstallationsRegistrationErrorTimeout;
  197. }
  198. #pragma mark - FID registration
  199. - (FBLPromise<FIRInstallationsItem *> *)registerInstallationIfNeeded:
  200. (FIRInstallationsItem *)installation {
  201. switch (installation.registrationStatus) {
  202. case FIRInstallationStatusRegistered:
  203. // Already registered. Do nothing.
  204. return [FBLPromise resolvedWith:installation];
  205. case FIRInstallationStatusUnknown:
  206. case FIRInstallationStatusUnregistered:
  207. case FIRInstallationStatusRegistrationFailed:
  208. // Registration required. Proceed.
  209. break;
  210. }
  211. return [self.APIService registerInstallation:installation]
  212. .recover(^id(NSError *error) {
  213. return [self handleRegistrationRequestError:error installation:installation];
  214. })
  215. .then(^id(FIRInstallationsItem *registeredInstallation) {
  216. // Expected successful result: @[FIRInstallationsItem *registeredInstallation, NSNull]
  217. return [FBLPromise all:@[
  218. registeredInstallation, [self.installationsStore saveInstallation:registeredInstallation]
  219. ]];
  220. })
  221. .then(^FIRInstallationsItem *(NSArray *result) {
  222. FIRInstallationsItem *registeredInstallation = result.firstObject;
  223. // Server may respond with a different FID if the sent one cannot be accepted.
  224. if (![registeredInstallation.firebaseInstallationID
  225. isEqualToString:installation.firebaseInstallationID]) {
  226. [self postFIDDidChangeNotification];
  227. }
  228. return registeredInstallation;
  229. });
  230. }
  231. - (FBLPromise<FIRInstallationsItem *> *)handleRegistrationRequestError:(NSError *)error
  232. installation:
  233. (FIRInstallationsItem *)installation {
  234. if ([self doesRegistrationErrorRequireConfigChange:error]) {
  235. FIRLogError(kFIRLoggerInstallations, kFIRInstallationsMessageCodeInvalidFirebaseConfiguration,
  236. @"Firebase Installation registration failed for app with name: %@, error: "
  237. @"%@\nPlease make sure you use valid GoogleService-Info.plist",
  238. self.appName, error);
  239. FIRInstallationsItem *failedInstallation = [installation copy];
  240. [failedInstallation updateWithRegistrationError:error
  241. date:[NSDate date]
  242. registrationParameters:[self currentRegistrationParameters]];
  243. // Save the error and then fail with the API error.
  244. return
  245. [self.installationsStore saveInstallation:failedInstallation].then(^NSError *(id result) {
  246. return error;
  247. });
  248. }
  249. FBLPromise *errorPromise = [FBLPromise pendingPromise];
  250. [errorPromise reject:error];
  251. return errorPromise;
  252. }
  253. - (BOOL)doesRegistrationErrorRequireConfigChange:(NSError *)error {
  254. FIRInstallationsHTTPError *HTTPError = (FIRInstallationsHTTPError *)error;
  255. if (![HTTPError isKindOfClass:[FIRInstallationsHTTPError class]]) {
  256. return NO;
  257. }
  258. switch (HTTPError.HTTPResponse.statusCode) {
  259. // These are the errors that require Firebase configuration change.
  260. case FIRInstallationsRegistrationHTTPCodeInvalidArgument:
  261. case FIRInstallationsRegistrationHTTPCodeInvalidAPIKey:
  262. case FIRInstallationsRegistrationHTTPCodeAPIKeyToProjectIDMismatch:
  263. case FIRInstallationsRegistrationHTTPCodeProjectNotFound:
  264. return YES;
  265. default:
  266. return NO;
  267. }
  268. }
  269. - (FIRInstallationsStoredRegistrationParameters *)currentRegistrationParameters {
  270. return [[FIRInstallationsStoredRegistrationParameters alloc]
  271. initWithAPIKey:self.APIService.APIKey
  272. projectID:self.APIService.projectID];
  273. }
  274. #pragma mark - Auth Token
  275. - (FBLPromise<FIRInstallationsItem *> *)getAuthTokenForcingRefresh:(BOOL)forceRefresh {
  276. if (forceRefresh || [self.authTokenForcingRefreshPromiseCache getExistingPendingPromise] != nil) {
  277. return [self.authTokenForcingRefreshPromiseCache getExistingPendingOrCreateNewPromise];
  278. } else {
  279. return [self.authTokenPromiseCache getExistingPendingOrCreateNewPromise];
  280. }
  281. }
  282. - (FBLPromise<FIRInstallationsItem *> *)installationWithValidAuthTokenForcingRefresh:
  283. (BOOL)forceRefresh {
  284. FIRLogDebug(kFIRLoggerInstallations, kFIRInstallationsMessageCodeNewGetAuthTokenOperationCreated,
  285. @"-[FIRInstallationsIDController installationWithValidAuthTokenForcingRefresh:%@], "
  286. @"appName: %@",
  287. @(forceRefresh), self.appName);
  288. return [self getInstallationItem]
  289. .then(^FBLPromise<FIRInstallationsItem *> *(FIRInstallationsItem *installation) {
  290. return [self registerInstallationIfNeeded:installation];
  291. })
  292. .then(^id(FIRInstallationsItem *registeredInstallation) {
  293. BOOL isTokenExpiredOrExpiresSoon =
  294. [registeredInstallation.authToken.expirationDate timeIntervalSinceDate:[NSDate date]] <
  295. kFIRInstallationsTokenExpirationThreshold;
  296. if (forceRefresh || isTokenExpiredOrExpiresSoon) {
  297. return [self.APIService refreshAuthTokenForInstallation:registeredInstallation];
  298. } else {
  299. return registeredInstallation;
  300. }
  301. })
  302. .recover(^id(NSError *error) {
  303. return [self regenerateFIDOnRefreshTokenErrorIfNeeded:error];
  304. });
  305. }
  306. - (id)regenerateFIDOnRefreshTokenErrorIfNeeded:(NSError *)error {
  307. if (![error isKindOfClass:[FIRInstallationsHTTPError class]]) {
  308. // No recovery possible. Return the same error.
  309. return error;
  310. }
  311. FIRInstallationsHTTPError *HTTPError = (FIRInstallationsHTTPError *)error;
  312. switch (HTTPError.HTTPResponse.statusCode) {
  313. case FIRInstallationsAuthTokenHTTPCodeInvalidAuthentication:
  314. case FIRInstallationsAuthTokenHTTPCodeFIDNotFound:
  315. // The stored installation was damaged or blocked by the server.
  316. // Delete the stored installation then generate and register a new one.
  317. return [self getInstallationItem]
  318. .then(^FBLPromise<NSNull *> *(FIRInstallationsItem *installation) {
  319. return [self deleteInstallationLocally:installation];
  320. })
  321. .then(^FBLPromise<FIRInstallationsItem *> *(id result) {
  322. return [self installationWithValidAuthTokenForcingRefresh:NO];
  323. });
  324. default:
  325. // No recovery possible. Return the same error.
  326. return error;
  327. }
  328. }
  329. #pragma mark - Delete FID
  330. - (FBLPromise<NSNull *> *)deleteInstallation {
  331. return [self.deleteInstallationPromiseCache getExistingPendingOrCreateNewPromise];
  332. }
  333. - (FBLPromise<NSNull *> *)createDeleteInstallationPromise {
  334. FIRLogDebug(kFIRLoggerInstallations,
  335. kFIRInstallationsMessageCodeNewDeleteInstallationOperationCreated, @"%s, appName: %@",
  336. __PRETTY_FUNCTION__, self.appName);
  337. // Check for ongoing requests first, if there is no a request, then check local storage for
  338. // existing installation.
  339. FBLPromise<FIRInstallationsItem *> *currentInstallationPromise =
  340. [self mostRecentInstallationOperation] ?: [self getStoredInstallation];
  341. return currentInstallationPromise
  342. .then(^id(FIRInstallationsItem *installation) {
  343. return [self sendDeleteInstallationRequestIfNeeded:installation];
  344. })
  345. .then(^id(FIRInstallationsItem *installation) {
  346. // Remove the installation from the local storage.
  347. return [self deleteInstallationLocally:installation];
  348. });
  349. }
  350. - (FBLPromise<NSNull *> *)deleteInstallationLocally:(FIRInstallationsItem *)installation {
  351. return [self.installationsStore removeInstallationForAppID:installation.appID
  352. appName:installation.firebaseAppName]
  353. .then(^FBLPromise<NSNull *> *(NSNull *result) {
  354. return [self deleteExistingIIDIfNeeded];
  355. })
  356. .then(^NSNull *(NSNull *result) {
  357. [self postFIDDidChangeNotification];
  358. return result;
  359. });
  360. }
  361. - (FBLPromise<FIRInstallationsItem *> *)sendDeleteInstallationRequestIfNeeded:
  362. (FIRInstallationsItem *)installation {
  363. switch (installation.registrationStatus) {
  364. case FIRInstallationStatusUnknown:
  365. case FIRInstallationStatusUnregistered:
  366. case FIRInstallationStatusRegistrationFailed:
  367. // The installation is not registered, so it is safe to be deleted as is, so return early.
  368. return [FBLPromise resolvedWith:installation];
  369. break;
  370. case FIRInstallationStatusRegistered:
  371. // Proceed to de-register the installation on the server.
  372. break;
  373. }
  374. return [self.APIService deleteInstallation:installation].recover(^id(NSError *APIError) {
  375. if ([FIRInstallationsErrorUtil isAPIError:APIError withHTTPCode:404]) {
  376. // The installation was not found on the server.
  377. // Return success.
  378. return installation;
  379. } else {
  380. // Re-throw the error otherwise.
  381. return APIError;
  382. }
  383. });
  384. }
  385. - (FBLPromise<NSNull *> *)deleteExistingIIDIfNeeded {
  386. if ([self isDefaultApp]) {
  387. return [self.IIDStore deleteExistingIID];
  388. } else {
  389. return [FBLPromise resolvedWith:[NSNull null]];
  390. }
  391. }
  392. - (nullable FBLPromise<FIRInstallationsItem *> *)mostRecentInstallationOperation {
  393. return [self.authTokenForcingRefreshPromiseCache getExistingPendingPromise]
  394. ?: [self.authTokenPromiseCache getExistingPendingPromise]
  395. ?: [self.getInstallationPromiseCache getExistingPendingPromise];
  396. }
  397. #pragma mark - Notifications
  398. - (void)postFIDDidChangeNotification {
  399. [[NSNotificationCenter defaultCenter]
  400. postNotificationName:FIRInstallationIDDidChangeNotification
  401. object:nil
  402. userInfo:@{kFIRInstallationIDDidChangeNotificationAppNameKey : self.appName}];
  403. }
  404. #pragma mark - Default App
  405. - (BOOL)isDefaultApp {
  406. return [self.appName isEqualToString:kFIRDefaultAppName];
  407. }
  408. @end