FIRInstallations.m 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  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 "FIRInstallations.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 <FirebaseCore/FIRComponent.h>
  24. #import <FirebaseCore/FIRComponentContainer.h>
  25. #import <FirebaseCore/FIRLibrary.h>
  26. #import <FirebaseCore/FIRLogger.h>
  27. #import <FirebaseCore/FIROptions.h>
  28. #import "FIRInstallationsAuthTokenResultInternal.h"
  29. #import "FIRInstallationsErrorUtil.h"
  30. #import "FIRInstallationsIDController.h"
  31. #import "FIRInstallationsItem.h"
  32. #import "FIRInstallationsLogger.h"
  33. #import "FIRInstallationsStoredAuthToken.h"
  34. #import "FIRInstallationsVersion.h"
  35. NS_ASSUME_NONNULL_BEGIN
  36. @protocol FIRInstallationsInstanceProvider <FIRLibrary>
  37. @end
  38. @interface FIRInstallations () <FIRInstallationsInstanceProvider>
  39. @property(nonatomic, readonly) FIROptions *appOptions;
  40. @property(nonatomic, readonly) NSString *appName;
  41. @property(nonatomic, readonly) FIRInstallationsIDController *installationsIDController;
  42. @end
  43. @implementation FIRInstallations
  44. #pragma mark - Firebase component
  45. + (void)load {
  46. [FIRApp registerInternalLibrary:(Class<FIRLibrary>)self
  47. withName:@"fire-install"
  48. withVersion:[NSString stringWithUTF8String:FIRInstallationsVersionStr]];
  49. }
  50. + (nonnull NSArray<FIRComponent *> *)componentsToRegister {
  51. FIRComponentCreationBlock creationBlock =
  52. ^id _Nullable(FIRComponentContainer *container, BOOL *isCacheable) {
  53. *isCacheable = YES;
  54. FIRInstallations *installations = [[FIRInstallations alloc] initWithApp:container.app];
  55. return installations;
  56. };
  57. FIRComponent *installationsProvider =
  58. [FIRComponent componentWithProtocol:@protocol(FIRInstallationsInstanceProvider)
  59. instantiationTiming:FIRInstantiationTimingAlwaysEager
  60. dependencies:@[]
  61. creationBlock:creationBlock];
  62. return @[ installationsProvider ];
  63. }
  64. - (instancetype)initWithApp:(FIRApp *)app {
  65. return [self initWitAppOptions:app.options appName:app.name];
  66. }
  67. - (instancetype)initWitAppOptions:(FIROptions *)appOptions appName:(NSString *)appName {
  68. FIRInstallationsIDController *IDController =
  69. [[FIRInstallationsIDController alloc] initWithGoogleAppID:appOptions.googleAppID
  70. appName:appName
  71. APIKey:appOptions.APIKey
  72. projectID:appOptions.projectID
  73. GCMSenderID:appOptions.GCMSenderID
  74. accessGroup:appOptions.appGroupID];
  75. // `prefetchAuthToken` is disabled due to b/156746574.
  76. return [self initWithAppOptions:appOptions
  77. appName:appName
  78. installationsIDController:IDController
  79. prefetchAuthToken:NO];
  80. }
  81. /// The initializer is supposed to be used by tests to inject `installationsStore`.
  82. - (instancetype)initWithAppOptions:(FIROptions *)appOptions
  83. appName:(NSString *)appName
  84. installationsIDController:(FIRInstallationsIDController *)installationsIDController
  85. prefetchAuthToken:(BOOL)prefetchAuthToken {
  86. self = [super init];
  87. if (self) {
  88. [[self class] validateAppOptions:appOptions appName:appName];
  89. [[self class] assertCompatibleIIDVersion];
  90. _appOptions = [appOptions copy];
  91. _appName = [appName copy];
  92. _installationsIDController = installationsIDController;
  93. // Pre-fetch auth token.
  94. if (prefetchAuthToken) {
  95. [self authTokenWithCompletion:^(FIRInstallationsAuthTokenResult *_Nullable tokenResult,
  96. NSError *_Nullable error){
  97. }];
  98. }
  99. }
  100. return self;
  101. }
  102. + (void)validateAppOptions:(FIROptions *)appOptions appName:(NSString *)appName {
  103. NSMutableArray *missingFields = [NSMutableArray array];
  104. if (appName.length < 1) {
  105. [missingFields addObject:@"`FirebaseApp.name`"];
  106. }
  107. if (appOptions.APIKey.length < 1) {
  108. [missingFields addObject:@"`FirebaseOptions.APIKey`"];
  109. }
  110. if (appOptions.googleAppID.length < 1) {
  111. [missingFields addObject:@"`FirebaseOptions.googleAppID`"];
  112. }
  113. // TODO(#4692): Check for `appOptions.projectID.length < 1` only.
  114. // We can use `GCMSenderID` instead of `projectID` temporary.
  115. if (appOptions.projectID.length < 1 && appOptions.GCMSenderID.length < 1) {
  116. [missingFields addObject:@"`FirebaseOptions.projectID`"];
  117. }
  118. if (missingFields.count > 0) {
  119. [NSException
  120. raise:kFirebaseInstallationsErrorDomain
  121. format:
  122. @"%@[%@] Could not configure Firebase Installations due to invalid FirebaseApp "
  123. @"options. The following parameters are nil or empty: %@. If you use "
  124. @"GoogleServices-Info.plist please download the most recent version from the Firebase "
  125. @"Console. If you configure Firebase in code, please make sure you specify all "
  126. @"required parameters.",
  127. kFIRLoggerInstallations, kFIRInstallationsMessageCodeInvalidFirebaseAppOptions,
  128. [missingFields componentsJoinedByString:@", "]];
  129. }
  130. }
  131. #pragma mark - Public
  132. + (FIRInstallations *)installations {
  133. FIRApp *defaultApp = [FIRApp defaultApp];
  134. if (!defaultApp) {
  135. [NSException raise:kFirebaseInstallationsErrorDomain
  136. format:@"The default FirebaseApp instance must be configured before the default"
  137. @"FirebaseApp instance can be initialized. One way to ensure that is to "
  138. @"call `[FIRApp configure];` (`FirebaseApp.configure()` in Swift) in the App"
  139. @" Delegate's `application:didFinishLaunchingWithOptions:` "
  140. @"(`application(_:didFinishLaunchingWithOptions:)` in Swift)."];
  141. }
  142. return [self installationsWithApp:defaultApp];
  143. }
  144. + (FIRInstallations *)installationsWithApp:(FIRApp *)app {
  145. id<FIRInstallationsInstanceProvider> installations =
  146. FIR_COMPONENT(FIRInstallationsInstanceProvider, app.container);
  147. return (FIRInstallations *)installations;
  148. }
  149. - (void)installationIDWithCompletion:(FIRInstallationsIDHandler)completion {
  150. [self.installationsIDController getInstallationItem]
  151. .then(^id(FIRInstallationsItem *installation) {
  152. completion(installation.firebaseInstallationID, nil);
  153. return nil;
  154. })
  155. .catch(^(NSError *error) {
  156. completion(nil, [FIRInstallationsErrorUtil publicDomainErrorWithError:error]);
  157. });
  158. }
  159. - (void)authTokenWithCompletion:(FIRInstallationsTokenHandler)completion {
  160. [self authTokenForcingRefresh:NO completion:completion];
  161. }
  162. - (void)authTokenForcingRefresh:(BOOL)forceRefresh
  163. completion:(FIRInstallationsTokenHandler)completion {
  164. [self.installationsIDController getAuthTokenForcingRefresh:forceRefresh]
  165. .then(^FIRInstallationsAuthTokenResult *(FIRInstallationsItem *installation) {
  166. FIRInstallationsAuthTokenResult *result = [[FIRInstallationsAuthTokenResult alloc]
  167. initWithToken:installation.authToken.token
  168. expirationDate:installation.authToken.expirationDate];
  169. return result;
  170. })
  171. .then(^id(FIRInstallationsAuthTokenResult *token) {
  172. completion(token, nil);
  173. return nil;
  174. })
  175. .catch(^void(NSError *error) {
  176. completion(nil, [FIRInstallationsErrorUtil publicDomainErrorWithError:error]);
  177. });
  178. }
  179. - (void)deleteWithCompletion:(void (^)(NSError *__nullable error))completion {
  180. [self.installationsIDController deleteInstallation]
  181. .then(^id(id result) {
  182. completion(nil);
  183. return nil;
  184. })
  185. .catch(^void(NSError *error) {
  186. completion([FIRInstallationsErrorUtil publicDomainErrorWithError:error]);
  187. });
  188. }
  189. #pragma mark - IID version compatibility
  190. + (void)assertCompatibleIIDVersion {
  191. // We use this flag to disable IID compatibility exception for unit tests.
  192. #ifdef FIR_INSTALLATIONS_ALLOWS_INCOMPATIBLE_IID_VERSION
  193. return;
  194. #else
  195. if (![self isIIDVersionCompatible]) {
  196. [NSException raise:kFirebaseInstallationsErrorDomain
  197. format:@"FirebaseInstallations will not work correctly with current version of "
  198. @"Firebase Instance ID. Please update your Firebase Instance ID version."];
  199. }
  200. #endif
  201. }
  202. + (BOOL)isIIDVersionCompatible {
  203. Class IIDClass = NSClassFromString(@"FIRInstanceID");
  204. if (IIDClass == nil) {
  205. // It is OK if there is no IID at all.
  206. return YES;
  207. }
  208. // We expect a compatible version having the method `+[FIRInstanceID usesFIS]` defined.
  209. BOOL isCompatibleVersion = [IIDClass respondsToSelector:NSSelectorFromString(@"usesFIS")];
  210. return isCompatibleVersion;
  211. }
  212. @end
  213. NS_ASSUME_NONNULL_END