FIRInstallationsIDController.m 18 KB

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