FIRInstallations.m 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  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. return [self initWithAppOptions:appOptions
  76. appName:appName
  77. installationsIDController:IDController
  78. prefetchAuthToken:YES];
  79. }
  80. /// The initializer is supposed to be used by tests to inject `installationsStore`.
  81. - (instancetype)initWithAppOptions:(FIROptions *)appOptions
  82. appName:(NSString *)appName
  83. installationsIDController:(FIRInstallationsIDController *)installationsIDController
  84. prefetchAuthToken:(BOOL)prefetchAuthToken {
  85. self = [super init];
  86. if (self) {
  87. [[self class] validateAppOptions:appOptions appName:appName];
  88. [[self class] assertCompatibleIIDVersion];
  89. _appOptions = [appOptions copy];
  90. _appName = [appName copy];
  91. _installationsIDController = installationsIDController;
  92. // Pre-fetch auth token.
  93. if (prefetchAuthToken) {
  94. [self authTokenWithCompletion:^(FIRInstallationsAuthTokenResult *_Nullable tokenResult,
  95. NSError *_Nullable error){
  96. }];
  97. }
  98. }
  99. return self;
  100. }
  101. + (void)validateAppOptions:(FIROptions *)appOptions appName:(NSString *)appName {
  102. NSMutableArray *missingFields = [NSMutableArray array];
  103. if (appName.length < 1) {
  104. [missingFields addObject:@"`FirebaseApp.name`"];
  105. }
  106. if (appOptions.APIKey.length < 1) {
  107. [missingFields addObject:@"`FirebaseOptions.APIKey`"];
  108. }
  109. if (appOptions.googleAppID.length < 1) {
  110. [missingFields addObject:@"`FirebaseOptions.googleAppID`"];
  111. }
  112. // TODO(#4692): Check for `appOptions.projectID.length < 1` only.
  113. // We can use `GCMSenderID` instead of `projectID` temporary.
  114. if (appOptions.projectID.length < 1 && appOptions.GCMSenderID.length < 1) {
  115. [missingFields addObject:@"`FirebaseOptions.projectID`"];
  116. }
  117. if (missingFields.count > 0) {
  118. [NSException
  119. raise:kFirebaseInstallationsErrorDomain
  120. format:
  121. @"%@[%@] Could not configure Firebase Installations due to invalid FirebaseApp "
  122. @"options. The following parameters are nil or empty: %@. If you use "
  123. @"GoogleServices-Info.plist please download the most recent version from the Firebase "
  124. @"Console. If you configure Firebase in code, please make sure you specify all "
  125. @"required parameters.",
  126. kFIRLoggerInstallations, kFIRInstallationsMessageCodeInvalidFirebaseAppOptions,
  127. [missingFields componentsJoinedByString:@", "]];
  128. }
  129. }
  130. #pragma mark - Public
  131. + (FIRInstallations *)installations {
  132. FIRApp *defaultApp = [FIRApp defaultApp];
  133. if (!defaultApp) {
  134. [NSException raise:kFirebaseInstallationsErrorDomain
  135. format:@"The default FirebaseApp instance must be configured before the default"
  136. @"FirebaseApp instance can be initialized. One way to ensure that is to "
  137. @"call `[FIRApp configure];` (`FirebaseApp.configure()` in Swift) in the App"
  138. @" Delegate's `application:didFinishLaunchingWithOptions:` "
  139. @"(`application(_:didFinishLaunchingWithOptions:)` in Swift)."];
  140. }
  141. return [self installationsWithApp:defaultApp];
  142. }
  143. + (FIRInstallations *)installationsWithApp:(FIRApp *)app {
  144. id<FIRInstallationsInstanceProvider> installations =
  145. FIR_COMPONENT(FIRInstallationsInstanceProvider, app.container);
  146. return (FIRInstallations *)installations;
  147. }
  148. - (void)installationIDWithCompletion:(FIRInstallationsIDHandler)completion {
  149. [self.installationsIDController getInstallationItem]
  150. .then(^id(FIRInstallationsItem *installation) {
  151. completion(installation.firebaseInstallationID, nil);
  152. return nil;
  153. })
  154. .catch(^(NSError *error) {
  155. completion(nil, [FIRInstallationsErrorUtil publicDomainErrorWithError:error]);
  156. });
  157. }
  158. - (void)authTokenWithCompletion:(FIRInstallationsTokenHandler)completion {
  159. [self authTokenForcingRefresh:NO completion:completion];
  160. }
  161. - (void)authTokenForcingRefresh:(BOOL)forceRefresh
  162. completion:(FIRInstallationsTokenHandler)completion {
  163. [self.installationsIDController getAuthTokenForcingRefresh:forceRefresh]
  164. .then(^FIRInstallationsAuthTokenResult *(FIRInstallationsItem *installation) {
  165. FIRInstallationsAuthTokenResult *result = [[FIRInstallationsAuthTokenResult alloc]
  166. initWithToken:installation.authToken.token
  167. expirationDate:installation.authToken.expirationDate];
  168. return result;
  169. })
  170. .then(^id(FIRInstallationsAuthTokenResult *token) {
  171. completion(token, nil);
  172. return nil;
  173. })
  174. .catch(^void(NSError *error) {
  175. completion(nil, [FIRInstallationsErrorUtil publicDomainErrorWithError:error]);
  176. });
  177. }
  178. - (void)deleteWithCompletion:(void (^)(NSError *__nullable error))completion {
  179. [self.installationsIDController deleteInstallation]
  180. .then(^id(id result) {
  181. completion(nil);
  182. return nil;
  183. })
  184. .catch(^void(NSError *error) {
  185. completion([FIRInstallationsErrorUtil publicDomainErrorWithError:error]);
  186. });
  187. }
  188. #pragma mark - IID version compatibility
  189. + (void)assertCompatibleIIDVersion {
  190. // We use this flag to disable IID compatibility exception for unit tests.
  191. #ifdef FIR_INSTALLATIONS_ALLOWS_INCOMPATIBLE_IID_VERSION
  192. return;
  193. #else
  194. if (![self isIIDVersionCompatible]) {
  195. [NSException raise:kFirebaseInstallationsErrorDomain
  196. format:@"FirebaseInstallations will not work correctly with current version of "
  197. @"Firebase Instance ID. Please update your Firebase Instance ID version."];
  198. }
  199. #endif
  200. }
  201. + (BOOL)isIIDVersionCompatible {
  202. Class IIDClass = NSClassFromString(@"FIRInstanceID");
  203. if (IIDClass == nil) {
  204. // It is OK if there is no IID at all.
  205. return YES;
  206. }
  207. // We expect a compatible version having the method `+[FIRInstanceID usesFIS]` defined.
  208. BOOL isCompatibleVersion = [IIDClass respondsToSelector:NSSelectorFromString(@"usesFIS")];
  209. return isCompatibleVersion;
  210. }
  211. @end
  212. NS_ASSUME_NONNULL_END