FIRInstallations.m 9.4 KB

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