FIRInstallationsIDController.m 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  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 "FIRInstallationsIIDCheckinStore.h"
  26. #import "FIRInstallationsIIDStore.h"
  27. #import "FIRInstallationsItem.h"
  28. #import "FIRInstallationsLogger.h"
  29. #import "FIRInstallationsSingleOperationPromiseCache.h"
  30. #import "FIRInstallationsStore.h"
  31. #import "FIRSecureStorage.h"
  32. #import "FIRInstallationsHTTPError.h"
  33. #import "FIRInstallationsStoredAuthToken.h"
  34. #import "FIRInstallationsStoredIIDCheckin.h"
  35. const NSNotificationName FIRInstallationIDDidChangeNotification =
  36. @"FIRInstallationIDDidChangeNotification";
  37. NSString *const kFIRInstallationIDDidChangeNotificationAppNameKey =
  38. @"FIRInstallationIDDidChangeNotification";
  39. NSTimeInterval const kFIRInstallationsTokenExpirationThreshold = 60 * 60; // 1 hour.
  40. @interface FIRInstallationsIDController ()
  41. @property(nonatomic, readonly) NSString *appID;
  42. @property(nonatomic, readonly) NSString *appName;
  43. @property(nonatomic, readonly) FIRInstallationsStore *installationsStore;
  44. @property(nonatomic, readonly) FIRInstallationsIIDStore *IIDStore;
  45. @property(nonatomic, readonly) FIRInstallationsIIDCheckinStore *IIDCheckingStore;
  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. FIRInstallationsIIDCheckinStore *IIDCheckingStore =
  68. [[FIRInstallationsIIDCheckinStore alloc] init];
  69. return [self initWithGoogleAppID:appID
  70. appName:appName
  71. installationsStore:installationsStore
  72. APIService:apiService
  73. IIDStore:IIDStore
  74. IIDCheckingStore:IIDCheckingStore];
  75. }
  76. /// The initializer is supposed to be used by tests to inject `installationsStore`.
  77. - (instancetype)initWithGoogleAppID:(NSString *)appID
  78. appName:(NSString *)appName
  79. installationsStore:(FIRInstallationsStore *)installationsStore
  80. APIService:(FIRInstallationsAPIService *)APIService
  81. IIDStore:(FIRInstallationsIIDStore *)IIDStore
  82. IIDCheckingStore:(FIRInstallationsIIDCheckinStore *)IIDCheckingStore {
  83. self = [super init];
  84. if (self) {
  85. _appID = appID;
  86. _appName = appName;
  87. _installationsStore = installationsStore;
  88. _APIService = APIService;
  89. _IIDStore = IIDStore;
  90. _IIDCheckingStore = IIDCheckingStore;
  91. __weak FIRInstallationsIDController *weakSelf = self;
  92. _getInstallationPromiseCache = [[FIRInstallationsSingleOperationPromiseCache alloc]
  93. initWithNewOperationHandler:^FBLPromise *_Nonnull {
  94. FIRInstallationsIDController *strongSelf = weakSelf;
  95. return [strongSelf createGetInstallationItemPromise];
  96. }];
  97. _authTokenPromiseCache = [[FIRInstallationsSingleOperationPromiseCache alloc]
  98. initWithNewOperationHandler:^FBLPromise *_Nonnull {
  99. FIRInstallationsIDController *strongSelf = weakSelf;
  100. return [strongSelf installationWithValidAuthTokenForcingRefresh:NO];
  101. }];
  102. _authTokenForcingRefreshPromiseCache = [[FIRInstallationsSingleOperationPromiseCache alloc]
  103. initWithNewOperationHandler:^FBLPromise *_Nonnull {
  104. FIRInstallationsIDController *strongSelf = weakSelf;
  105. return [strongSelf installationWithValidAuthTokenForcingRefresh:YES];
  106. }];
  107. _deleteInstallationPromiseCache = [[FIRInstallationsSingleOperationPromiseCache alloc]
  108. initWithNewOperationHandler:^FBLPromise *_Nonnull {
  109. FIRInstallationsIDController *strongSelf = weakSelf;
  110. return [strongSelf createDeleteInstallationPromise];
  111. }];
  112. }
  113. return self;
  114. }
  115. #pragma mark - Get Installation.
  116. - (FBLPromise<FIRInstallationsItem *> *)getInstallationItem {
  117. return [self.getInstallationPromiseCache getExistingPendingOrCreateNewPromise];
  118. }
  119. - (FBLPromise<FIRInstallationsItem *> *)createGetInstallationItemPromise {
  120. FIRLogDebug(kFIRLoggerInstallations,
  121. kFIRInstallationsMessageCodeNewGetInstallationOperationCreated, @"%s, appName: %@",
  122. __PRETTY_FUNCTION__, self.appName);
  123. FBLPromise<FIRInstallationsItem *> *installationItemPromise =
  124. [self getStoredInstallation].recover(^id(NSError *error) {
  125. return [self createAndSaveFID];
  126. });
  127. // Initiate registration process on success if needed, but return the installation without waiting
  128. // for it.
  129. installationItemPromise.then(^id(FIRInstallationsItem *installation) {
  130. [self getAuthTokenForcingRefresh:NO];
  131. return nil;
  132. });
  133. return installationItemPromise;
  134. }
  135. - (FBLPromise<FIRInstallationsItem *> *)getStoredInstallation {
  136. return [self.installationsStore installationForAppID:self.appID appName:self.appName].validate(
  137. ^BOOL(FIRInstallationsItem *installation) {
  138. BOOL isValid = NO;
  139. switch (installation.registrationStatus) {
  140. case FIRInstallationStatusUnregistered:
  141. case FIRInstallationStatusRegistered:
  142. isValid = YES;
  143. break;
  144. case FIRInstallationStatusUnknown:
  145. isValid = NO;
  146. break;
  147. }
  148. return isValid;
  149. });
  150. }
  151. - (FBLPromise<FIRInstallationsItem *> *)createAndSaveFID {
  152. return [self migrateOrGenerateInstallation]
  153. .then(^FBLPromise<FIRInstallationsItem *> *(FIRInstallationsItem *installation) {
  154. return [self saveInstallation:installation];
  155. })
  156. .then(^FIRInstallationsItem *(FIRInstallationsItem *installation) {
  157. [self postFIDDidChangeNotification];
  158. return installation;
  159. });
  160. }
  161. - (FBLPromise<FIRInstallationsItem *> *)saveInstallation:(FIRInstallationsItem *)installation {
  162. return [self.installationsStore saveInstallation:installation].then(
  163. ^FIRInstallationsItem *(NSNull *result) {
  164. return installation;
  165. });
  166. }
  167. /**
  168. * Tries to migrate IID data stored by FirebaseInstanceID SDK or generates a new Installation ID if
  169. * not found.
  170. */
  171. - (FBLPromise<FIRInstallationsItem *> *)migrateOrGenerateInstallation {
  172. if (![self isDefaultApp]) {
  173. // Existing IID should be used only for default FirebaseApp.
  174. FIRInstallationsItem *installation =
  175. [self createInstallationWithFID:[FIRInstallationsItem generateFID] IIDCheckin:nil];
  176. return [FBLPromise resolvedWith:installation];
  177. }
  178. return
  179. [[[FBLPromise all:@[ [self.IIDStore existingIID], [self.IIDCheckingStore existingCheckin] ]]
  180. then:^id _Nullable(NSArray *_Nullable results) {
  181. NSString *existingIID = results[0];
  182. FIRInstallationsStoredIIDCheckin *IIDCheckin = results[1];
  183. return [self createInstallationWithFID:existingIID IIDCheckin:IIDCheckin];
  184. }] recover:^id _Nullable(NSError *_Nonnull error) {
  185. return [self createInstallationWithFID:[FIRInstallationsItem generateFID] IIDCheckin:nil];
  186. }];
  187. }
  188. - (FIRInstallationsItem *)createInstallationWithFID:(NSString *)FID
  189. IIDCheckin:(nullable FIRInstallationsStoredIIDCheckin *)
  190. IIDCheckin {
  191. FIRInstallationsItem *installation = [[FIRInstallationsItem alloc] initWithAppID:self.appID
  192. firebaseAppName:self.appName];
  193. installation.firebaseInstallationID = FID;
  194. installation.IIDCheckin = IIDCheckin;
  195. installation.registrationStatus = FIRInstallationStatusUnregistered;
  196. return installation;
  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. // Registration required. Proceed.
  208. break;
  209. }
  210. return [self.APIService registerInstallation:installation]
  211. .catch(^(NSError *_Nonnull error) {
  212. if ([self doesRegistrationErrorRequireConfigChange:error]) {
  213. FIRLogError(kFIRLoggerInstallations,
  214. kFIRInstallationsMessageCodeInvalidFirebaseConfiguration,
  215. @"Firebase Installation registration failed for app with name: %@, error: "
  216. @"%@\nPlease make sure you use valid GoogleService-Info.plist",
  217. self.appName, error);
  218. }
  219. })
  220. .then(^id(FIRInstallationsItem *registeredInstallation) {
  221. return [self saveInstallation:registeredInstallation];
  222. })
  223. .then(^FIRInstallationsItem *(FIRInstallationsItem *registeredInstallation) {
  224. // Server may respond with a different FID if the sent one cannot be accepted.
  225. if (![registeredInstallation.firebaseInstallationID
  226. isEqualToString:installation.firebaseInstallationID]) {
  227. [self postFIDDidChangeNotification];
  228. }
  229. return registeredInstallation;
  230. });
  231. }
  232. - (BOOL)doesRegistrationErrorRequireConfigChange:(NSError *)error {
  233. FIRInstallationsHTTPError *HTTPError = (FIRInstallationsHTTPError *)error;
  234. if (![HTTPError isKindOfClass:[FIRInstallationsHTTPError class]]) {
  235. return NO;
  236. }
  237. switch (HTTPError.HTTPResponse.statusCode) {
  238. // These are the errors that require Firebase configuration change.
  239. case FIRInstallationsRegistrationHTTPCodeInvalidArgument:
  240. case FIRInstallationsRegistrationHTTPCodeInvalidAPIKey:
  241. case FIRInstallationsRegistrationHTTPCodeAPIKeyToProjectIDMismatch:
  242. case FIRInstallationsRegistrationHTTPCodeProjectNotFound:
  243. return YES;
  244. default:
  245. return NO;
  246. }
  247. }
  248. #pragma mark - Auth Token
  249. - (FBLPromise<FIRInstallationsItem *> *)getAuthTokenForcingRefresh:(BOOL)forceRefresh {
  250. if (forceRefresh || [self.authTokenForcingRefreshPromiseCache getExistingPendingPromise] != nil) {
  251. return [self.authTokenForcingRefreshPromiseCache getExistingPendingOrCreateNewPromise];
  252. } else {
  253. return [self.authTokenPromiseCache getExistingPendingOrCreateNewPromise];
  254. }
  255. }
  256. - (FBLPromise<FIRInstallationsItem *> *)installationWithValidAuthTokenForcingRefresh:
  257. (BOOL)forceRefresh {
  258. FIRLogDebug(kFIRLoggerInstallations, kFIRInstallationsMessageCodeNewGetAuthTokenOperationCreated,
  259. @"-[FIRInstallationsIDController installationWithValidAuthTokenForcingRefresh:%@], "
  260. @"appName: %@",
  261. @(forceRefresh), self.appName);
  262. return [self getInstallationItem]
  263. .then(^FBLPromise<FIRInstallationsItem *> *(FIRInstallationsItem *installation) {
  264. return [self registerInstallationIfNeeded:installation];
  265. })
  266. .then(^id(FIRInstallationsItem *registeredInstallation) {
  267. BOOL isTokenExpiredOrExpiresSoon =
  268. [registeredInstallation.authToken.expirationDate timeIntervalSinceDate:[NSDate date]] <
  269. kFIRInstallationsTokenExpirationThreshold;
  270. if (forceRefresh || isTokenExpiredOrExpiresSoon) {
  271. return [self refreshAuthTokenForInstallation:registeredInstallation];
  272. } else {
  273. return registeredInstallation;
  274. }
  275. })
  276. .recover(^id(NSError *error) {
  277. return [self regenerateFIDOnRefreshTokenErrorIfNeeded:error];
  278. });
  279. }
  280. - (FBLPromise<FIRInstallationsItem *> *)refreshAuthTokenForInstallation:
  281. (FIRInstallationsItem *)installation {
  282. return [[self.APIService refreshAuthTokenForInstallation:installation]
  283. then:^id _Nullable(FIRInstallationsItem *_Nullable refreshedInstallation) {
  284. return [self saveInstallation:refreshedInstallation];
  285. }];
  286. }
  287. - (id)regenerateFIDOnRefreshTokenErrorIfNeeded:(NSError *)error {
  288. if (![error isKindOfClass:[FIRInstallationsHTTPError class]]) {
  289. // No recovery possible. Return the same error.
  290. return error;
  291. }
  292. FIRInstallationsHTTPError *HTTPError = (FIRInstallationsHTTPError *)error;
  293. switch (HTTPError.HTTPResponse.statusCode) {
  294. case FIRInstallationsAuthTokenHTTPCodeInvalidAuthentication:
  295. case FIRInstallationsAuthTokenHTTPCodeFIDNotFound:
  296. // The stored installation was damaged or blocked by the server.
  297. // Delete the stored installation then generate and register a new one.
  298. return [self getInstallationItem]
  299. .then(^FBLPromise<NSNull *> *(FIRInstallationsItem *installation) {
  300. return [self deleteInstallationLocally:installation];
  301. })
  302. .then(^FBLPromise<FIRInstallationsItem *> *(id result) {
  303. return [self installationWithValidAuthTokenForcingRefresh:NO];
  304. });
  305. default:
  306. // No recovery possible. Return the same error.
  307. return error;
  308. }
  309. }
  310. #pragma mark - Delete FID
  311. - (FBLPromise<NSNull *> *)deleteInstallation {
  312. return [self.deleteInstallationPromiseCache getExistingPendingOrCreateNewPromise];
  313. }
  314. - (FBLPromise<NSNull *> *)createDeleteInstallationPromise {
  315. FIRLogDebug(kFIRLoggerInstallations,
  316. kFIRInstallationsMessageCodeNewDeleteInstallationOperationCreated, @"%s, appName: %@",
  317. __PRETTY_FUNCTION__, self.appName);
  318. // Check for ongoing requests first, if there is no a request, then check local storage for
  319. // existing installation.
  320. FBLPromise<FIRInstallationsItem *> *currentInstallationPromise =
  321. [self mostRecentInstallationOperation] ?: [self getStoredInstallation];
  322. return currentInstallationPromise
  323. .then(^id(FIRInstallationsItem *installation) {
  324. return [self sendDeleteInstallationRequestIfNeeded:installation];
  325. })
  326. .then(^id(FIRInstallationsItem *installation) {
  327. // Remove the installation from the local storage.
  328. return [self deleteInstallationLocally:installation];
  329. });
  330. }
  331. - (FBLPromise<NSNull *> *)deleteInstallationLocally:(FIRInstallationsItem *)installation {
  332. return [self.installationsStore removeInstallationForAppID:installation.appID
  333. appName:installation.firebaseAppName]
  334. .then(^FBLPromise<NSNull *> *(NSNull *result) {
  335. return [self deleteExistingIIDIfNeeded];
  336. })
  337. .then(^NSNull *(NSNull *result) {
  338. [self postFIDDidChangeNotification];
  339. return result;
  340. });
  341. }
  342. - (FBLPromise<FIRInstallationsItem *> *)sendDeleteInstallationRequestIfNeeded:
  343. (FIRInstallationsItem *)installation {
  344. switch (installation.registrationStatus) {
  345. case FIRInstallationStatusUnknown:
  346. case FIRInstallationStatusUnregistered:
  347. // The installation is not registered, so it is safe to be deleted as is, so return early.
  348. return [FBLPromise resolvedWith:installation];
  349. break;
  350. case FIRInstallationStatusRegistered:
  351. // Proceed to de-register the installation on the server.
  352. break;
  353. }
  354. return [self.APIService deleteInstallation:installation].recover(^id(NSError *APIError) {
  355. if ([FIRInstallationsErrorUtil isAPIError:APIError withHTTPCode:404]) {
  356. // The installation was not found on the server.
  357. // Return success.
  358. return installation;
  359. } else {
  360. // Re-throw the error otherwise.
  361. return APIError;
  362. }
  363. });
  364. }
  365. - (FBLPromise<NSNull *> *)deleteExistingIIDIfNeeded {
  366. if ([self isDefaultApp]) {
  367. return [self.IIDStore deleteExistingIID];
  368. } else {
  369. return [FBLPromise resolvedWith:[NSNull null]];
  370. }
  371. }
  372. - (nullable FBLPromise<FIRInstallationsItem *> *)mostRecentInstallationOperation {
  373. return [self.authTokenForcingRefreshPromiseCache getExistingPendingPromise]
  374. ?: [self.authTokenPromiseCache getExistingPendingPromise]
  375. ?: [self.getInstallationPromiseCache getExistingPendingPromise];
  376. }
  377. #pragma mark - Notifications
  378. - (void)postFIDDidChangeNotification {
  379. [[NSNotificationCenter defaultCenter]
  380. postNotificationName:FIRInstallationIDDidChangeNotification
  381. object:nil
  382. userInfo:@{kFIRInstallationIDDidChangeNotificationAppNameKey : self.appName}];
  383. }
  384. #pragma mark - Default App
  385. - (BOOL)isDefaultApp {
  386. return [self.appName isEqualToString:kFIRDefaultAppName];
  387. }
  388. @end