Adds firebase live data
Also no longer crashes when touching something while loading
This commit is contained in:
69
Pods/FirebaseCore/Firebase/Core/FIRAnalyticsConfiguration.m
generated
Normal file
69
Pods/FirebaseCore/Firebase/Core/FIRAnalyticsConfiguration.m
generated
Normal file
@@ -0,0 +1,69 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "FIRAnalyticsConfiguration.h"
|
||||
|
||||
#import "Private/FIRAnalyticsConfiguration+Internal.h"
|
||||
|
||||
@implementation FIRAnalyticsConfiguration
|
||||
|
||||
+ (FIRAnalyticsConfiguration *)sharedInstance {
|
||||
static FIRAnalyticsConfiguration *sharedInstance = nil;
|
||||
static dispatch_once_t onceToken;
|
||||
dispatch_once(&onceToken, ^{
|
||||
sharedInstance = [[FIRAnalyticsConfiguration alloc] init];
|
||||
});
|
||||
return sharedInstance;
|
||||
}
|
||||
|
||||
- (void)postNotificationName:(NSString *)name value:(id)value {
|
||||
if (!name.length || !value) {
|
||||
return;
|
||||
}
|
||||
[[NSNotificationCenter defaultCenter] postNotificationName:name
|
||||
object:self
|
||||
userInfo:@{name : value}];
|
||||
}
|
||||
|
||||
- (void)setMinimumSessionInterval:(NSTimeInterval)minimumSessionInterval {
|
||||
[self postNotificationName:kFIRAnalyticsConfigurationSetMinimumSessionIntervalNotification
|
||||
value:@(minimumSessionInterval)];
|
||||
}
|
||||
|
||||
- (void)setSessionTimeoutInterval:(NSTimeInterval)sessionTimeoutInterval {
|
||||
[self postNotificationName:kFIRAnalyticsConfigurationSetSessionTimeoutIntervalNotification
|
||||
value:@(sessionTimeoutInterval)];
|
||||
}
|
||||
|
||||
- (void)setAnalyticsCollectionEnabled:(BOOL)analyticsCollectionEnabled {
|
||||
[self setAnalyticsCollectionEnabled:analyticsCollectionEnabled persistSetting:YES];
|
||||
}
|
||||
|
||||
- (void)setAnalyticsCollectionEnabled:(BOOL)analyticsCollectionEnabled
|
||||
persistSetting:(BOOL)shouldPersist {
|
||||
// Persist the measurementEnabledState. Use FIRAnalyticsEnabledState values instead of YES/NO.
|
||||
FIRAnalyticsEnabledState analyticsEnabledState =
|
||||
analyticsCollectionEnabled ? kFIRAnalyticsEnabledStateSetYes : kFIRAnalyticsEnabledStateSetNo;
|
||||
if (shouldPersist) {
|
||||
NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
|
||||
[userDefaults setObject:@(analyticsEnabledState)
|
||||
forKey:kFIRAPersistedConfigMeasurementEnabledStateKey];
|
||||
[userDefaults synchronize];
|
||||
}
|
||||
|
||||
[self postNotificationName:kFIRAnalyticsConfigurationSetEnabledNotification
|
||||
value:@(analyticsCollectionEnabled)];
|
||||
}
|
||||
|
||||
@end
|
||||
737
Pods/FirebaseCore/Firebase/Core/FIRApp.m
generated
Normal file
737
Pods/FirebaseCore/Firebase/Core/FIRApp.m
generated
Normal file
@@ -0,0 +1,737 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#include <sys/utsname.h>
|
||||
|
||||
#import "FIRApp.h"
|
||||
#import "FIRConfiguration.h"
|
||||
#import "Private/FIRAnalyticsConfiguration+Internal.h"
|
||||
#import "Private/FIRAppInternal.h"
|
||||
#import "Private/FIRBundleUtil.h"
|
||||
#import "Private/FIRLogger.h"
|
||||
#import "Private/FIROptionsInternal.h"
|
||||
#import "third_party/FIRAppEnvironmentUtil.h"
|
||||
|
||||
NSString *const kFIRServiceAdMob = @"AdMob";
|
||||
NSString *const kFIRServiceAuth = @"Auth";
|
||||
NSString *const kFIRServiceAuthUI = @"AuthUI";
|
||||
NSString *const kFIRServiceCrash = @"Crash";
|
||||
NSString *const kFIRServiceDatabase = @"Database";
|
||||
NSString *const kFIRServiceDynamicLinks = @"DynamicLinks";
|
||||
NSString *const kFIRServiceFirestore = @"Firestore";
|
||||
NSString *const kFIRServiceFunctions = @"Functions";
|
||||
NSString *const kFIRServiceInstanceID = @"InstanceID";
|
||||
NSString *const kFIRServiceInvites = @"Invites";
|
||||
NSString *const kFIRServiceMessaging = @"Messaging";
|
||||
NSString *const kFIRServiceMeasurement = @"Measurement";
|
||||
NSString *const kFIRServicePerformance = @"Performance";
|
||||
NSString *const kFIRServiceRemoteConfig = @"RemoteConfig";
|
||||
NSString *const kFIRServiceStorage = @"Storage";
|
||||
NSString *const kGGLServiceAnalytics = @"Analytics";
|
||||
NSString *const kGGLServiceSignIn = @"SignIn";
|
||||
|
||||
NSString *const kFIRDefaultAppName = @"__FIRAPP_DEFAULT";
|
||||
NSString *const kFIRAppReadyToConfigureSDKNotification = @"FIRAppReadyToConfigureSDKNotification";
|
||||
NSString *const kFIRAppDeleteNotification = @"FIRAppDeleteNotification";
|
||||
NSString *const kFIRAppIsDefaultAppKey = @"FIRAppIsDefaultAppKey";
|
||||
NSString *const kFIRAppNameKey = @"FIRAppNameKey";
|
||||
NSString *const kFIRGoogleAppIDKey = @"FIRGoogleAppIDKey";
|
||||
|
||||
NSString *const kFIRGlobalAppDataCollectionEnabledDefaultsKeyFormat =
|
||||
@"/google/firebase/global_data_collection_enabled:%@";
|
||||
NSString *const kFIRGlobalAppDataCollectionEnabledPlistKey =
|
||||
@"FirebaseAutomaticDataCollectionEnabled";
|
||||
|
||||
NSString *const kFIRAppDiagnosticsNotification = @"FIRAppDiagnosticsNotification";
|
||||
|
||||
NSString *const kFIRAppDiagnosticsConfigurationTypeKey = @"ConfigType";
|
||||
NSString *const kFIRAppDiagnosticsErrorKey = @"Error";
|
||||
NSString *const kFIRAppDiagnosticsFIRAppKey = @"FIRApp";
|
||||
NSString *const kFIRAppDiagnosticsSDKNameKey = @"SDKName";
|
||||
NSString *const kFIRAppDiagnosticsSDKVersionKey = @"SDKVersion";
|
||||
|
||||
// Auth internal notification notification and key.
|
||||
NSString *const FIRAuthStateDidChangeInternalNotification =
|
||||
@"FIRAuthStateDidChangeInternalNotification";
|
||||
NSString *const FIRAuthStateDidChangeInternalNotificationAppKey =
|
||||
@"FIRAuthStateDidChangeInternalNotificationAppKey";
|
||||
NSString *const FIRAuthStateDidChangeInternalNotificationTokenKey =
|
||||
@"FIRAuthStateDidChangeInternalNotificationTokenKey";
|
||||
NSString *const FIRAuthStateDidChangeInternalNotificationUIDKey =
|
||||
@"FIRAuthStateDidChangeInternalNotificationUIDKey";
|
||||
|
||||
/**
|
||||
* The URL to download plist files.
|
||||
*/
|
||||
static NSString *const kPlistURL = @"https://console.firebase.google.com/";
|
||||
|
||||
@interface FIRApp ()
|
||||
|
||||
@property(nonatomic) BOOL alreadySentConfigureNotification;
|
||||
|
||||
@property(nonatomic) BOOL alreadySentDeleteNotification;
|
||||
|
||||
@end
|
||||
|
||||
@implementation FIRApp
|
||||
|
||||
// This is necessary since our custom getter prevents `_options` from being created.
|
||||
@synthesize options = _options;
|
||||
|
||||
static NSMutableDictionary *sAllApps;
|
||||
static FIRApp *sDefaultApp;
|
||||
static NSMutableDictionary *sLibraryVersions;
|
||||
|
||||
+ (void)configure {
|
||||
FIROptions *options = [FIROptions defaultOptions];
|
||||
if (!options) {
|
||||
[[NSNotificationCenter defaultCenter]
|
||||
postNotificationName:kFIRAppDiagnosticsNotification
|
||||
object:nil
|
||||
userInfo:@{
|
||||
kFIRAppDiagnosticsConfigurationTypeKey : @(FIRConfigTypeCore),
|
||||
kFIRAppDiagnosticsErrorKey : [FIRApp errorForMissingOptions]
|
||||
}];
|
||||
[NSException raise:kFirebaseCoreErrorDomain
|
||||
format:
|
||||
@"`[FIRApp configure];` (`FirebaseApp.configure()` in Swift) could not find "
|
||||
@"a valid GoogleService-Info.plist in your project. Please download one "
|
||||
@"from %@.",
|
||||
kPlistURL];
|
||||
}
|
||||
[FIRApp configureDefaultAppWithOptions:options sendingNotifications:YES];
|
||||
#if TARGET_OS_OSX || TARGET_OS_TV
|
||||
FIRLogNotice(kFIRLoggerCore, @"I-COR000028",
|
||||
@"tvOS and macOS SDK support is not part of the official Firebase product. "
|
||||
@"Instead they are community supported. Details at "
|
||||
@"https://github.com/firebase/firebase-ios-sdk/blob/master/README.md.");
|
||||
#endif
|
||||
}
|
||||
|
||||
+ (void)configureWithOptions:(FIROptions *)options {
|
||||
if (!options) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain
|
||||
format:@"Options is nil. Please pass a valid options."];
|
||||
}
|
||||
[FIRApp configureDefaultAppWithOptions:options sendingNotifications:YES];
|
||||
}
|
||||
|
||||
+ (void)configureDefaultAppWithOptions:(FIROptions *)options
|
||||
sendingNotifications:(BOOL)sendNotifications {
|
||||
if (sDefaultApp) {
|
||||
// FIRApp sets up FirebaseAnalytics and does plist validation, but does not cause it
|
||||
// to fire notifications. So, if the default app already exists, but has not sent out
|
||||
// configuration notifications, then continue re-initializing it.
|
||||
if (!sendNotifications || sDefaultApp.alreadySentConfigureNotification) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain
|
||||
format:@"Default app has already been configured."];
|
||||
}
|
||||
}
|
||||
@synchronized(self) {
|
||||
FIRLogDebug(kFIRLoggerCore, @"I-COR000001", @"Configuring the default app.");
|
||||
sDefaultApp = [[FIRApp alloc] initInstanceWithName:kFIRDefaultAppName options:options];
|
||||
[FIRApp addAppToAppDictionary:sDefaultApp];
|
||||
if (!sDefaultApp.alreadySentConfigureNotification && sendNotifications) {
|
||||
[FIRApp sendNotificationsToSDKs:sDefaultApp];
|
||||
sDefaultApp.alreadySentConfigureNotification = YES;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
+ (void)configureWithName:(NSString *)name options:(FIROptions *)options {
|
||||
if (!name || !options) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain format:@"Neither name nor options can be nil."];
|
||||
}
|
||||
if (name.length == 0) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain format:@"Name cannot be empty."];
|
||||
}
|
||||
if ([name isEqualToString:kFIRDefaultAppName]) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain format:@"Name cannot be __FIRAPP_DEFAULT."];
|
||||
}
|
||||
for (NSInteger charIndex = 0; charIndex < name.length; charIndex++) {
|
||||
char character = [name characterAtIndex:charIndex];
|
||||
if (!((character >= 'a' && character <= 'z') || (character >= 'A' && character <= 'Z') ||
|
||||
(character >= '0' && character <= '9') || character == '_' || character == '-')) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain
|
||||
format:
|
||||
@"App name should only contain Letters, "
|
||||
@"Numbers, Underscores, and Dashes."];
|
||||
}
|
||||
}
|
||||
|
||||
if (sAllApps && sAllApps[name]) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain
|
||||
format:@"App named %@ has already been configured.", name];
|
||||
}
|
||||
|
||||
@synchronized(self) {
|
||||
FIRLogDebug(kFIRLoggerCore, @"I-COR000002", @"Configuring app named %@", name);
|
||||
FIRApp *app = [[FIRApp alloc] initInstanceWithName:name options:options];
|
||||
[FIRApp addAppToAppDictionary:app];
|
||||
if (!app.alreadySentConfigureNotification) {
|
||||
[FIRApp sendNotificationsToSDKs:app];
|
||||
app.alreadySentConfigureNotification = YES;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
+ (FIRApp *)defaultApp {
|
||||
if (sDefaultApp) {
|
||||
return sDefaultApp;
|
||||
}
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000003",
|
||||
@"The default Firebase app has not yet been "
|
||||
@"configured. Add `[FIRApp configure];` (`FirebaseApp.configure()` in Swift) to your "
|
||||
@"application initialization. Read more: https://goo.gl/ctyzm8.");
|
||||
return nil;
|
||||
}
|
||||
|
||||
+ (FIRApp *)appNamed:(NSString *)name {
|
||||
@synchronized(self) {
|
||||
if (sAllApps) {
|
||||
FIRApp *app = sAllApps[name];
|
||||
if (app) {
|
||||
return app;
|
||||
}
|
||||
}
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000004", @"App with name %@ does not exist.", name);
|
||||
return nil;
|
||||
}
|
||||
}
|
||||
|
||||
+ (NSDictionary *)allApps {
|
||||
@synchronized(self) {
|
||||
if (!sAllApps) {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000005", @"No app has been configured yet.");
|
||||
}
|
||||
NSDictionary *dict = [NSDictionary dictionaryWithDictionary:sAllApps];
|
||||
return dict;
|
||||
}
|
||||
}
|
||||
|
||||
// Public only for tests
|
||||
+ (void)resetApps {
|
||||
sDefaultApp = nil;
|
||||
[sAllApps removeAllObjects];
|
||||
sAllApps = nil;
|
||||
[sLibraryVersions removeAllObjects];
|
||||
sLibraryVersions = nil;
|
||||
}
|
||||
|
||||
- (void)deleteApp:(FIRAppVoidBoolCallback)completion {
|
||||
@synchronized([self class]) {
|
||||
if (sAllApps && sAllApps[self.name]) {
|
||||
FIRLogDebug(kFIRLoggerCore, @"I-COR000006", @"Deleting app named %@", self.name);
|
||||
[sAllApps removeObjectForKey:self.name];
|
||||
[self clearDataCollectionSwitchFromUserDefaults];
|
||||
if ([self.name isEqualToString:kFIRDefaultAppName]) {
|
||||
sDefaultApp = nil;
|
||||
}
|
||||
if (!self.alreadySentDeleteNotification) {
|
||||
NSDictionary *appInfoDict = @{kFIRAppNameKey : self.name};
|
||||
[[NSNotificationCenter defaultCenter] postNotificationName:kFIRAppDeleteNotification
|
||||
object:[self class]
|
||||
userInfo:appInfoDict];
|
||||
self.alreadySentDeleteNotification = YES;
|
||||
}
|
||||
completion(YES);
|
||||
} else {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000007", @"App does not exist.");
|
||||
completion(NO);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
+ (void)addAppToAppDictionary:(FIRApp *)app {
|
||||
if (!sAllApps) {
|
||||
sAllApps = [NSMutableDictionary dictionary];
|
||||
}
|
||||
if ([app configureCore]) {
|
||||
sAllApps[app.name] = app;
|
||||
[[NSNotificationCenter defaultCenter]
|
||||
postNotificationName:kFIRAppDiagnosticsNotification
|
||||
object:nil
|
||||
userInfo:@{
|
||||
kFIRAppDiagnosticsConfigurationTypeKey : @(FIRConfigTypeCore),
|
||||
kFIRAppDiagnosticsFIRAppKey : app
|
||||
}];
|
||||
} else {
|
||||
[NSException raise:kFirebaseCoreErrorDomain
|
||||
format:
|
||||
@"Configuration fails. It may be caused by an invalid GOOGLE_APP_ID in "
|
||||
@"GoogleService-Info.plist or set in the customized options."];
|
||||
}
|
||||
}
|
||||
|
||||
- (instancetype)initInstanceWithName:(NSString *)name options:(FIROptions *)options {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_name = [name copy];
|
||||
_options = [options copy];
|
||||
_options.editingLocked = YES;
|
||||
|
||||
FIRApp *app = sAllApps[name];
|
||||
_alreadySentConfigureNotification = app.alreadySentConfigureNotification;
|
||||
_alreadySentDeleteNotification = app.alreadySentDeleteNotification;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)getTokenForcingRefresh:(BOOL)forceRefresh withCallback:(FIRTokenCallback)callback {
|
||||
if (!_getTokenImplementation) {
|
||||
callback(nil, nil);
|
||||
return;
|
||||
}
|
||||
|
||||
_getTokenImplementation(forceRefresh, callback);
|
||||
}
|
||||
|
||||
- (BOOL)configureCore {
|
||||
[self checkExpectedBundleID];
|
||||
if (![self isAppIDValid]) {
|
||||
if (_options.usingOptionsFromDefaultPlist) {
|
||||
[[NSNotificationCenter defaultCenter]
|
||||
postNotificationName:kFIRAppDiagnosticsNotification
|
||||
object:nil
|
||||
userInfo:@{
|
||||
kFIRAppDiagnosticsConfigurationTypeKey : @(FIRConfigTypeCore),
|
||||
kFIRAppDiagnosticsErrorKey : [FIRApp errorForInvalidAppID],
|
||||
}];
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
|
||||
// Initialize the Analytics once there is a valid options under default app. Analytics should
|
||||
// always initialize first by itself before the other SDKs.
|
||||
if ([self.name isEqualToString:kFIRDefaultAppName]) {
|
||||
Class firAnalyticsClass = NSClassFromString(@"FIRAnalytics");
|
||||
if (!firAnalyticsClass) {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000022", @"Firebase Analytics is not available.");
|
||||
} else {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wundeclared-selector"
|
||||
SEL startWithConfigurationSelector = @selector(startWithConfiguration:options:);
|
||||
#pragma clang diagnostic pop
|
||||
if ([firAnalyticsClass respondsToSelector:startWithConfigurationSelector]) {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
|
||||
[firAnalyticsClass performSelector:startWithConfigurationSelector
|
||||
withObject:[FIRConfiguration sharedInstance].analyticsConfiguration
|
||||
withObject:_options];
|
||||
#pragma clang diagnostic pop
|
||||
}
|
||||
}
|
||||
}
|
||||
return YES;
|
||||
}
|
||||
|
||||
- (FIROptions *)options {
|
||||
return [_options copy];
|
||||
}
|
||||
|
||||
- (void)setAutomaticDataCollectionEnabled:(BOOL)automaticDataCollectionEnabled {
|
||||
NSString *key =
|
||||
[NSString stringWithFormat:kFIRGlobalAppDataCollectionEnabledDefaultsKeyFormat, self.name];
|
||||
[[NSUserDefaults standardUserDefaults] setBool:automaticDataCollectionEnabled forKey:key];
|
||||
|
||||
// Core also controls the FirebaseAnalytics flag, so check if the Analytics flags are set
|
||||
// within FIROptions and change the Analytics value if necessary. Analytics only works with the
|
||||
// default app, so return if this isn't the default app.
|
||||
if (self != sDefaultApp) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Check if the Analytics flag is explicitly set. If so, no further actions are necessary.
|
||||
if ([self.options isAnalyticsCollectionExpicitlySet]) {
|
||||
return;
|
||||
}
|
||||
|
||||
// The Analytics flag has not been explicitly set, so update with the value being set.
|
||||
[[FIRAnalyticsConfiguration sharedInstance]
|
||||
setAnalyticsCollectionEnabled:automaticDataCollectionEnabled
|
||||
persistSetting:NO];
|
||||
}
|
||||
|
||||
- (BOOL)isAutomaticDataCollectionEnabled {
|
||||
// Check if it's been manually set before in code, and use that as the higher priority value.
|
||||
NSNumber *defaultsObject = [[self class] readDataCollectionSwitchFromUserDefaultsForApp:self];
|
||||
if (defaultsObject) {
|
||||
return [defaultsObject boolValue];
|
||||
}
|
||||
|
||||
// Read the Info.plist to see if the flag is set. If it's not set, it should default to `YES`.
|
||||
// As per the implementation of `readDataCollectionSwitchFromPlist`, it's a cached value and has
|
||||
// no performance impact calling multiple times.
|
||||
NSNumber *collectionEnabledPlistValue = [[self class] readDataCollectionSwitchFromPlist];
|
||||
if (collectionEnabledPlistValue) {
|
||||
return [collectionEnabledPlistValue boolValue];
|
||||
}
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
#pragma mark - private
|
||||
|
||||
+ (void)sendNotificationsToSDKs:(FIRApp *)app {
|
||||
NSNumber *isDefaultApp = [NSNumber numberWithBool:(app == sDefaultApp)];
|
||||
NSDictionary *appInfoDict = @{
|
||||
kFIRAppNameKey : app.name,
|
||||
kFIRAppIsDefaultAppKey : isDefaultApp,
|
||||
kFIRGoogleAppIDKey : app.options.googleAppID
|
||||
};
|
||||
[[NSNotificationCenter defaultCenter] postNotificationName:kFIRAppReadyToConfigureSDKNotification
|
||||
object:self
|
||||
userInfo:appInfoDict];
|
||||
}
|
||||
|
||||
+ (NSError *)errorForMissingOptions {
|
||||
NSDictionary *errorDict = @{
|
||||
NSLocalizedDescriptionKey :
|
||||
@"Unable to parse GoogleService-Info.plist in order to configure services.",
|
||||
NSLocalizedRecoverySuggestionErrorKey :
|
||||
@"Check formatting and location of GoogleService-Info.plist."
|
||||
};
|
||||
return [NSError errorWithDomain:kFirebaseCoreErrorDomain
|
||||
code:FIRErrorCodeInvalidPlistFile
|
||||
userInfo:errorDict];
|
||||
}
|
||||
|
||||
+ (NSError *)errorForSubspecConfigurationFailureWithDomain:(NSString *)domain
|
||||
errorCode:(FIRErrorCode)code
|
||||
service:(NSString *)service
|
||||
reason:(NSString *)reason {
|
||||
NSString *description =
|
||||
[NSString stringWithFormat:@"Configuration failed for service %@.", service];
|
||||
NSDictionary *errorDict =
|
||||
@{NSLocalizedDescriptionKey : description, NSLocalizedFailureReasonErrorKey : reason};
|
||||
return [NSError errorWithDomain:domain code:code userInfo:errorDict];
|
||||
}
|
||||
|
||||
+ (NSError *)errorForInvalidAppID {
|
||||
NSDictionary *errorDict = @{
|
||||
NSLocalizedDescriptionKey : @"Unable to validate Google App ID",
|
||||
NSLocalizedRecoverySuggestionErrorKey :
|
||||
@"Check formatting and location of GoogleService-Info.plist or GoogleAppID set in the "
|
||||
@"customized options."
|
||||
};
|
||||
return [NSError errorWithDomain:kFirebaseCoreErrorDomain
|
||||
code:FIRErrorCodeInvalidAppID
|
||||
userInfo:errorDict];
|
||||
}
|
||||
|
||||
+ (BOOL)isDefaultAppConfigured {
|
||||
return (sDefaultApp != nil);
|
||||
}
|
||||
|
||||
+ (void)registerLibrary:(nonnull NSString *)library withVersion:(nonnull NSString *)version {
|
||||
// Create the set of characters which aren't allowed, only if this feature is used.
|
||||
NSMutableCharacterSet *allowedSet = [NSMutableCharacterSet alphanumericCharacterSet];
|
||||
[allowedSet addCharactersInString:@"-_."];
|
||||
NSCharacterSet *disallowedSet = [allowedSet invertedSet];
|
||||
// Make sure the library name and version strings do not contain unexpected characters, and
|
||||
// add the name/version pair to the dictionary.
|
||||
if ([library rangeOfCharacterFromSet:disallowedSet].location == NSNotFound &&
|
||||
[version rangeOfCharacterFromSet:disallowedSet].location == NSNotFound) {
|
||||
if (!sLibraryVersions) {
|
||||
sLibraryVersions = [[NSMutableDictionary alloc] init];
|
||||
}
|
||||
sLibraryVersions[library] = version;
|
||||
} else {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000027",
|
||||
@"The library name (%@) or version number (%@) contain illegal characters. "
|
||||
@"Only alphanumeric, dash, underscore and period characters are allowed.",
|
||||
library, version);
|
||||
}
|
||||
}
|
||||
|
||||
+ (NSString *)firebaseUserAgent {
|
||||
NSMutableArray<NSString *> *libraries =
|
||||
[[NSMutableArray<NSString *> alloc] initWithCapacity:sLibraryVersions.count];
|
||||
for (NSString *libraryName in sLibraryVersions) {
|
||||
[libraries
|
||||
addObject:[NSString stringWithFormat:@"%@/%@", libraryName, sLibraryVersions[libraryName]]];
|
||||
}
|
||||
[libraries sortUsingSelector:@selector(localizedCaseInsensitiveCompare:)];
|
||||
return [libraries componentsJoinedByString:@" "];
|
||||
}
|
||||
|
||||
- (void)checkExpectedBundleID {
|
||||
NSArray *bundles = [FIRBundleUtil relevantBundles];
|
||||
NSString *expectedBundleID = [self expectedBundleID];
|
||||
// The checking is only done when the bundle ID is provided in the serviceInfo dictionary for
|
||||
// backward compatibility.
|
||||
if (expectedBundleID != nil &&
|
||||
![FIRBundleUtil hasBundleIdentifier:expectedBundleID inBundles:bundles]) {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000008",
|
||||
@"The project's Bundle ID is inconsistent with "
|
||||
@"either the Bundle ID in '%@.%@', or the Bundle ID in the options if you are "
|
||||
@"using a customized options. To ensure that everything can be configured "
|
||||
@"correctly, you may need to make the Bundle IDs consistent. To continue with this "
|
||||
@"plist file, you may change your app's bundle identifier to '%@'. Or you can "
|
||||
@"download a new configuration file that matches your bundle identifier from %@ "
|
||||
@"and replace the current one.",
|
||||
kServiceInfoFileName, kServiceInfoFileType, expectedBundleID, kPlistURL);
|
||||
}
|
||||
}
|
||||
|
||||
- (nullable NSString *)getUID {
|
||||
if (!_getUIDImplementation) {
|
||||
FIRLogWarning(kFIRLoggerCore, @"I-COR000025", @"FIRAuth getUID implementation wasn't set.");
|
||||
return nil;
|
||||
}
|
||||
return _getUIDImplementation();
|
||||
}
|
||||
|
||||
#pragma mark - private - App ID Validation
|
||||
|
||||
/**
|
||||
* Validates the format and fingerprint of the app ID contained in GOOGLE_APP_ID in the plist file.
|
||||
* This is the main method for validating app ID.
|
||||
*
|
||||
* @return YES if the app ID fulfills the expected format and fingerprint, NO otherwise.
|
||||
*/
|
||||
- (BOOL)isAppIDValid {
|
||||
NSString *appID = _options.googleAppID;
|
||||
BOOL isValid = [FIRApp validateAppID:appID];
|
||||
if (!isValid) {
|
||||
NSString *expectedBundleID = [self expectedBundleID];
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000009",
|
||||
@"The GOOGLE_APP_ID either in the plist file "
|
||||
@"'%@.%@' or the one set in the customized options is invalid. If you are using "
|
||||
@"the plist file, use the iOS version of bundle identifier to download the file, "
|
||||
@"and do not manually edit the GOOGLE_APP_ID. You may change your app's bundle "
|
||||
@"identifier to '%@'. Or you can download a new configuration file that matches "
|
||||
@"your bundle identifier from %@ and replace the current one.",
|
||||
kServiceInfoFileName, kServiceInfoFileType, expectedBundleID, kPlistURL);
|
||||
};
|
||||
return isValid;
|
||||
}
|
||||
|
||||
+ (BOOL)validateAppID:(NSString *)appID {
|
||||
// Failing validation only occurs when we are sure we are looking at a V2 app ID and it does not
|
||||
// have a valid fingerprint, otherwise we just warn about the potential issue.
|
||||
if (!appID.length) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
// All app IDs must start with at least "<version number>:".
|
||||
NSString *const versionPattern = @"^\\d+:";
|
||||
NSRegularExpression *versionRegex =
|
||||
[NSRegularExpression regularExpressionWithPattern:versionPattern options:0 error:NULL];
|
||||
if (!versionRegex) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
NSRange appIDRange = NSMakeRange(0, appID.length);
|
||||
NSArray *versionMatches = [versionRegex matchesInString:appID options:0 range:appIDRange];
|
||||
if (versionMatches.count != 1) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
NSRange versionRange = [(NSTextCheckingResult *)versionMatches.firstObject range];
|
||||
NSString *appIDVersion = [appID substringWithRange:versionRange];
|
||||
NSArray *knownVersions = @[ @"1:" ];
|
||||
if (![knownVersions containsObject:appIDVersion]) {
|
||||
// Permit unknown yet properly formatted app ID versions.
|
||||
return YES;
|
||||
}
|
||||
|
||||
if (![FIRApp validateAppIDFormat:appID withVersion:appIDVersion]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![FIRApp validateAppIDFingerprint:appID withVersion:appIDVersion]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
+ (NSString *)actualBundleID {
|
||||
return [[NSBundle mainBundle] bundleIdentifier];
|
||||
}
|
||||
|
||||
/**
|
||||
* Validates that the format of the app ID string is what is expected based on the supplied version.
|
||||
* The version must end in ":".
|
||||
*
|
||||
* For v1 app ids the format is expected to be
|
||||
* '<version #>:<project number>:ios:<fingerprint of bundle id>'.
|
||||
*
|
||||
* This method does not verify that the contents of the app id are correct, just that they fulfill
|
||||
* the expected format.
|
||||
*
|
||||
* @param appID Contents of GOOGLE_APP_ID from the plist file.
|
||||
* @param version Indicates what version of the app id format this string should be.
|
||||
* @return YES if provided string fufills the expected format, NO otherwise.
|
||||
*/
|
||||
+ (BOOL)validateAppIDFormat:(NSString *)appID withVersion:(NSString *)version {
|
||||
if (!appID.length || !version.length) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![version hasSuffix:@":"]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![appID hasPrefix:version]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
NSString *const pattern = @"^\\d+:ios:[a-f0-9]+$";
|
||||
NSRegularExpression *regex =
|
||||
[NSRegularExpression regularExpressionWithPattern:pattern options:0 error:NULL];
|
||||
if (!regex) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
NSRange localRange = NSMakeRange(version.length, appID.length - version.length);
|
||||
NSUInteger numberOfMatches = [regex numberOfMatchesInString:appID options:0 range:localRange];
|
||||
if (numberOfMatches != 1) {
|
||||
return NO;
|
||||
}
|
||||
return YES;
|
||||
}
|
||||
|
||||
/**
|
||||
* Validates that the fingerprint of the app ID string is what is expected based on the supplied
|
||||
* version. The version must end in ":".
|
||||
*
|
||||
* Note that the v1 hash algorithm is not permitted on the client and cannot be fully validated.
|
||||
*
|
||||
* @param appID Contents of GOOGLE_APP_ID from the plist file.
|
||||
* @param version Indicates what version of the app id format this string should be.
|
||||
* @return YES if provided string fufills the expected fingerprint and the version is known, NO
|
||||
* otherwise.
|
||||
*/
|
||||
+ (BOOL)validateAppIDFingerprint:(NSString *)appID withVersion:(NSString *)version {
|
||||
if (!appID.length || !version.length) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![version hasSuffix:@":"]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if (![appID hasPrefix:version]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
// Extract the supplied fingerprint from the supplied app ID.
|
||||
// This assumes the app ID format is the same for all known versions below. If the app ID format
|
||||
// changes in future versions, the tokenizing of the app ID format will need to take into account
|
||||
// the version of the app ID.
|
||||
NSArray *components = [appID componentsSeparatedByString:@":"];
|
||||
if (components.count != 4) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
NSString *suppliedFingerprintString = components[3];
|
||||
if (!suppliedFingerprintString.length) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
uint64_t suppliedFingerprint;
|
||||
NSScanner *scanner = [NSScanner scannerWithString:suppliedFingerprintString];
|
||||
if (![scanner scanHexLongLong:&suppliedFingerprint]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if ([version isEqual:@"1:"]) {
|
||||
// The v1 hash algorithm is not permitted on the client so the actual hash cannot be validated.
|
||||
return YES;
|
||||
}
|
||||
|
||||
// Unknown version.
|
||||
return NO;
|
||||
}
|
||||
|
||||
- (NSString *)expectedBundleID {
|
||||
return _options.bundleID;
|
||||
}
|
||||
|
||||
// end App ID validation
|
||||
|
||||
#pragma mark - Reading From Plist & User Defaults
|
||||
|
||||
/**
|
||||
* Clears the data collection switch from the standard NSUserDefaults for easier testing and
|
||||
* readability.
|
||||
*/
|
||||
- (void)clearDataCollectionSwitchFromUserDefaults {
|
||||
NSString *key =
|
||||
[NSString stringWithFormat:kFIRGlobalAppDataCollectionEnabledDefaultsKeyFormat, self.name];
|
||||
[[NSUserDefaults standardUserDefaults] removeObjectForKey:key];
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads the data collection switch from the standard NSUserDefaults for easier testing and
|
||||
* readability.
|
||||
*/
|
||||
+ (nullable NSNumber *)readDataCollectionSwitchFromUserDefaultsForApp:(FIRApp *)app {
|
||||
// Read the object in user defaults, and only return if it's an NSNumber.
|
||||
NSString *key =
|
||||
[NSString stringWithFormat:kFIRGlobalAppDataCollectionEnabledDefaultsKeyFormat, app.name];
|
||||
id collectionEnabledDefaultsObject = [[NSUserDefaults standardUserDefaults] objectForKey:key];
|
||||
if ([collectionEnabledDefaultsObject isKindOfClass:[NSNumber class]]) {
|
||||
return collectionEnabledDefaultsObject;
|
||||
}
|
||||
|
||||
return nil;
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads the data collection switch from the Info.plist for easier testing and readability. Will
|
||||
* only read once from the plist and return the cached value.
|
||||
*/
|
||||
+ (nullable NSNumber *)readDataCollectionSwitchFromPlist {
|
||||
static NSNumber *collectionEnabledPlistObject;
|
||||
static dispatch_once_t onceToken;
|
||||
dispatch_once(&onceToken, ^{
|
||||
// Read the data from the `Info.plist`, only assign it if it's there and an NSNumber.
|
||||
id plistValue = [[NSBundle mainBundle]
|
||||
objectForInfoDictionaryKey:kFIRGlobalAppDataCollectionEnabledPlistKey];
|
||||
if (plistValue && [plistValue isKindOfClass:[NSNumber class]]) {
|
||||
collectionEnabledPlistObject = (NSNumber *)plistValue;
|
||||
}
|
||||
});
|
||||
|
||||
return collectionEnabledPlistObject;
|
||||
}
|
||||
|
||||
#pragma mark - Sending Logs
|
||||
|
||||
- (void)sendLogsWithServiceName:(NSString *)serviceName
|
||||
version:(NSString *)version
|
||||
error:(NSError *)error {
|
||||
// If the user has manually turned off data collection, return and don't send logs.
|
||||
if (![self isAutomaticDataCollectionEnabled]) {
|
||||
return;
|
||||
}
|
||||
|
||||
NSMutableDictionary *userInfo = [[NSMutableDictionary alloc] initWithDictionary:@{
|
||||
kFIRAppDiagnosticsConfigurationTypeKey : @(FIRConfigTypeSDK),
|
||||
kFIRAppDiagnosticsSDKNameKey : serviceName,
|
||||
kFIRAppDiagnosticsSDKVersionKey : version,
|
||||
kFIRAppDiagnosticsFIRAppKey : self
|
||||
}];
|
||||
if (error) {
|
||||
userInfo[kFIRAppDiagnosticsErrorKey] = error;
|
||||
}
|
||||
[[NSNotificationCenter defaultCenter] postNotificationName:kFIRAppDiagnosticsNotification
|
||||
object:nil
|
||||
userInfo:userInfo];
|
||||
}
|
||||
|
||||
@end
|
||||
47
Pods/FirebaseCore/Firebase/Core/FIRAppAssociationRegistration.m
generated
Normal file
47
Pods/FirebaseCore/Firebase/Core/FIRAppAssociationRegistration.m
generated
Normal file
@@ -0,0 +1,47 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "Private/FIRAppAssociationRegistration.h"
|
||||
|
||||
#import <objc/runtime.h>
|
||||
|
||||
@implementation FIRAppAssociationRegistration
|
||||
|
||||
+ (nullable id)registeredObjectWithHost:(id)host
|
||||
key:(NSString *)key
|
||||
creationBlock:(id _Nullable (^)(void))creationBlock {
|
||||
@synchronized(self) {
|
||||
SEL dictKey = @selector(registeredObjectWithHost:key:creationBlock:);
|
||||
NSMutableDictionary<NSString *, id> *objectsByKey = objc_getAssociatedObject(host, dictKey);
|
||||
if (!objectsByKey) {
|
||||
objectsByKey = [[NSMutableDictionary alloc] init];
|
||||
objc_setAssociatedObject(host, dictKey, objectsByKey, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
|
||||
}
|
||||
id obj = objectsByKey[key];
|
||||
NSValue *creationBlockBeingCalled = [NSValue valueWithPointer:dictKey];
|
||||
if (obj) {
|
||||
if ([creationBlockBeingCalled isEqual:obj]) {
|
||||
[NSException raise:@"Reentering registeredObjectWithHost:key:creationBlock: not allowed"
|
||||
format:@"host: %@ key: %@", host, key];
|
||||
}
|
||||
return obj;
|
||||
}
|
||||
objectsByKey[key] = creationBlockBeingCalled;
|
||||
obj = creationBlock();
|
||||
objectsByKey[key] = obj;
|
||||
return obj;
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
||||
57
Pods/FirebaseCore/Firebase/Core/FIRBundleUtil.m
generated
Normal file
57
Pods/FirebaseCore/Firebase/Core/FIRBundleUtil.m
generated
Normal file
@@ -0,0 +1,57 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "Private/FIRBundleUtil.h"
|
||||
|
||||
@implementation FIRBundleUtil
|
||||
|
||||
+ (NSArray *)relevantBundles {
|
||||
return @[ [NSBundle mainBundle], [NSBundle bundleForClass:[self class]] ];
|
||||
}
|
||||
|
||||
+ (NSString *)optionsDictionaryPathWithResourceName:(NSString *)resourceName
|
||||
andFileType:(NSString *)fileType
|
||||
inBundles:(NSArray *)bundles {
|
||||
// Loop through all bundles to find the config dict.
|
||||
for (NSBundle *bundle in bundles) {
|
||||
NSString *path = [bundle pathForResource:resourceName ofType:fileType];
|
||||
// Use the first one we find.
|
||||
if (path) {
|
||||
return path;
|
||||
}
|
||||
}
|
||||
return nil;
|
||||
}
|
||||
|
||||
+ (NSArray *)relevantURLSchemes {
|
||||
NSMutableArray *result = [[NSMutableArray alloc] init];
|
||||
for (NSBundle *bundle in [[self class] relevantBundles]) {
|
||||
NSArray *urlTypes = [bundle objectForInfoDictionaryKey:@"CFBundleURLTypes"];
|
||||
for (NSDictionary *urlType in urlTypes) {
|
||||
[result addObjectsFromArray:urlType[@"CFBundleURLSchemes"]];
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
+ (BOOL)hasBundleIdentifier:(NSString *)bundleIdentifier inBundles:(NSArray *)bundles {
|
||||
for (NSBundle *bundle in bundles) {
|
||||
if ([bundle.bundleIdentifier isEqualToString:bundleIdentifier]) {
|
||||
return YES;
|
||||
}
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
|
||||
@end
|
||||
44
Pods/FirebaseCore/Firebase/Core/FIRConfiguration.m
generated
Normal file
44
Pods/FirebaseCore/Firebase/Core/FIRConfiguration.m
generated
Normal file
@@ -0,0 +1,44 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "FIRConfiguration.h"
|
||||
|
||||
extern void FIRSetLoggerLevel(FIRLoggerLevel loggerLevel);
|
||||
|
||||
@implementation FIRConfiguration
|
||||
|
||||
+ (instancetype)sharedInstance {
|
||||
static FIRConfiguration *sharedInstance = nil;
|
||||
static dispatch_once_t onceToken;
|
||||
dispatch_once(&onceToken, ^{
|
||||
sharedInstance = [[FIRConfiguration alloc] init];
|
||||
});
|
||||
return sharedInstance;
|
||||
}
|
||||
|
||||
- (instancetype)init {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_analyticsConfiguration = [FIRAnalyticsConfiguration sharedInstance];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)setLoggerLevel:(FIRLoggerLevel)loggerLevel {
|
||||
NSAssert(loggerLevel <= FIRLoggerLevelMax && loggerLevel >= FIRLoggerLevelMin,
|
||||
@"Invalid logger level, %ld", (long)loggerLevel);
|
||||
FIRSetLoggerLevel(loggerLevel);
|
||||
}
|
||||
|
||||
@end
|
||||
29
Pods/FirebaseCore/Firebase/Core/FIRErrors.m
generated
Normal file
29
Pods/FirebaseCore/Firebase/Core/FIRErrors.m
generated
Normal file
@@ -0,0 +1,29 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "Private/FIRErrors.h"
|
||||
|
||||
NSString *const kFirebaseErrorDomain = @"com.firebase";
|
||||
NSString *const kFirebaseAdMobErrorDomain = @"com.firebase.admob";
|
||||
NSString *const kFirebaseAppInviteErrorDomain = @"com.firebase.appinvite";
|
||||
NSString *const kFirebaseAuthErrorDomain = @"com.firebase.auth";
|
||||
NSString *const kFirebaseCloudMessagingErrorDomain = @"com.firebase.cloudmessaging";
|
||||
NSString *const kFirebaseConfigErrorDomain = @"com.firebase.config";
|
||||
NSString *const kFirebaseCoreErrorDomain = @"com.firebase.core";
|
||||
NSString *const kFirebaseCrashReportingErrorDomain = @"com.firebase.crashreporting";
|
||||
NSString *const kFirebaseDatabaseErrorDomain = @"com.firebase.database";
|
||||
NSString *const kFirebaseDurableDeepLinkErrorDomain = @"com.firebase.durabledeeplink";
|
||||
NSString *const kFirebaseInstanceIDErrorDomain = @"com.firebase.instanceid";
|
||||
NSString *const kFirebasePerfErrorDomain = @"com.firebase.perf";
|
||||
NSString *const kFirebaseStorageErrorDomain = @"com.firebase.storage";
|
||||
275
Pods/FirebaseCore/Firebase/Core/FIRLogger.m
generated
Normal file
275
Pods/FirebaseCore/Firebase/Core/FIRLogger.m
generated
Normal file
@@ -0,0 +1,275 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "Private/FIRLogger.h"
|
||||
|
||||
#import "FIRLoggerLevel.h"
|
||||
#import "Private/FIRVersion.h"
|
||||
#import "third_party/FIRAppEnvironmentUtil.h"
|
||||
|
||||
#include <asl.h>
|
||||
#include <assert.h>
|
||||
#include <stdbool.h>
|
||||
#include <sys/sysctl.h>
|
||||
#include <sys/types.h>
|
||||
#include <unistd.h>
|
||||
|
||||
FIRLoggerService kFIRLoggerABTesting = @"[Firebase/ABTesting]";
|
||||
FIRLoggerService kFIRLoggerAdMob = @"[Firebase/AdMob]";
|
||||
FIRLoggerService kFIRLoggerAnalytics = @"[Firebase/Analytics]";
|
||||
FIRLoggerService kFIRLoggerAuth = @"[Firebase/Auth]";
|
||||
FIRLoggerService kFIRLoggerCore = @"[Firebase/Core]";
|
||||
FIRLoggerService kFIRLoggerCrash = @"[Firebase/Crash]";
|
||||
FIRLoggerService kFIRLoggerDatabase = @"[Firebase/Database]";
|
||||
FIRLoggerService kFIRLoggerDynamicLinks = @"[Firebase/DynamicLinks]";
|
||||
FIRLoggerService kFIRLoggerFirestore = @"[Firebase/Firestore]";
|
||||
FIRLoggerService kFIRLoggerInstanceID = @"[Firebase/InstanceID]";
|
||||
FIRLoggerService kFIRLoggerInvites = @"[Firebase/Invites]";
|
||||
FIRLoggerService kFIRLoggerMLKit = @"[Firebase/MLKit]";
|
||||
FIRLoggerService kFIRLoggerMessaging = @"[Firebase/Messaging]";
|
||||
FIRLoggerService kFIRLoggerPerf = @"[Firebase/Performance]";
|
||||
FIRLoggerService kFIRLoggerRemoteConfig = @"[Firebase/RemoteConfig]";
|
||||
FIRLoggerService kFIRLoggerStorage = @"[Firebase/Storage]";
|
||||
FIRLoggerService kFIRLoggerSwizzler = @"[FirebaseSwizzlingUtilities]";
|
||||
|
||||
/// Arguments passed on launch.
|
||||
NSString *const kFIRDisableDebugModeApplicationArgument = @"-FIRDebugDisabled";
|
||||
NSString *const kFIREnableDebugModeApplicationArgument = @"-FIRDebugEnabled";
|
||||
NSString *const kFIRLoggerForceSDTERRApplicationArgument = @"-FIRLoggerForceSTDERR";
|
||||
|
||||
/// Key for the debug mode bit in NSUserDefaults.
|
||||
NSString *const kFIRPersistedDebugModeKey = @"/google/firebase/debug_mode";
|
||||
|
||||
/// ASL client facility name used by FIRLogger.
|
||||
const char *kFIRLoggerASLClientFacilityName = "com.firebase.app.logger";
|
||||
|
||||
/// Message format used by ASL client that matches format of NSLog.
|
||||
const char *kFIRLoggerCustomASLMessageFormat =
|
||||
"$((Time)(J.3)) $(Sender)[$(PID)] <$((Level)(str))> $Message";
|
||||
|
||||
/// Keys for the number of errors and warnings logged.
|
||||
NSString *const kFIRLoggerErrorCountKey = @"/google/firebase/count_of_errors_logged";
|
||||
NSString *const kFIRLoggerWarningCountKey = @"/google/firebase/count_of_warnings_logged";
|
||||
|
||||
static dispatch_once_t sFIRLoggerOnceToken;
|
||||
|
||||
static aslclient sFIRLoggerClient;
|
||||
|
||||
static dispatch_queue_t sFIRClientQueue;
|
||||
|
||||
static BOOL sFIRLoggerDebugMode;
|
||||
|
||||
// The sFIRAnalyticsDebugMode flag is here to support the -FIRDebugEnabled/-FIRDebugDisabled
|
||||
// flags used by Analytics. Users who use those flags expect Analytics to log verbosely,
|
||||
// while the rest of Firebase logs at the default level. This flag is introduced to support
|
||||
// that behavior.
|
||||
static BOOL sFIRAnalyticsDebugMode;
|
||||
|
||||
static FIRLoggerLevel sFIRLoggerMaximumLevel;
|
||||
|
||||
#ifdef DEBUG
|
||||
/// The regex pattern for the message code.
|
||||
static NSString *const kMessageCodePattern = @"^I-[A-Z]{3}[0-9]{6}$";
|
||||
static NSRegularExpression *sMessageCodeRegex;
|
||||
#endif
|
||||
|
||||
void FIRLoggerInitializeASL() {
|
||||
dispatch_once(&sFIRLoggerOnceToken, ^{
|
||||
NSInteger majorOSVersion = [[FIRAppEnvironmentUtil systemVersion] integerValue];
|
||||
uint32_t aslOptions = ASL_OPT_STDERR;
|
||||
#if TARGET_OS_SIMULATOR
|
||||
// The iOS 11 simulator doesn't need the ASL_OPT_STDERR flag.
|
||||
if (majorOSVersion >= 11) {
|
||||
aslOptions = 0;
|
||||
}
|
||||
#else
|
||||
// Devices running iOS 10 or higher don't need the ASL_OPT_STDERR flag.
|
||||
if (majorOSVersion >= 10) {
|
||||
aslOptions = 0;
|
||||
}
|
||||
#endif // TARGET_OS_SIMULATOR
|
||||
|
||||
// Override the aslOptions to ASL_OPT_STDERR if the override argument is passed in.
|
||||
NSArray *arguments = [NSProcessInfo processInfo].arguments;
|
||||
if ([arguments containsObject:kFIRLoggerForceSDTERRApplicationArgument]) {
|
||||
aslOptions = ASL_OPT_STDERR;
|
||||
}
|
||||
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-declarations" // asl is deprecated
|
||||
// Initialize the ASL client handle.
|
||||
sFIRLoggerClient = asl_open(NULL, kFIRLoggerASLClientFacilityName, aslOptions);
|
||||
|
||||
// Set the filter used by system/device log. Initialize in default mode.
|
||||
asl_set_filter(sFIRLoggerClient, ASL_FILTER_MASK_UPTO(ASL_LEVEL_NOTICE));
|
||||
sFIRLoggerDebugMode = NO;
|
||||
sFIRAnalyticsDebugMode = NO;
|
||||
sFIRLoggerMaximumLevel = FIRLoggerLevelNotice;
|
||||
|
||||
NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
|
||||
BOOL debugMode = [userDefaults boolForKey:kFIRPersistedDebugModeKey];
|
||||
|
||||
if ([arguments containsObject:kFIRDisableDebugModeApplicationArgument]) { // Default mode
|
||||
[userDefaults removeObjectForKey:kFIRPersistedDebugModeKey];
|
||||
} else if ([arguments containsObject:kFIREnableDebugModeApplicationArgument] ||
|
||||
debugMode) { // Debug mode
|
||||
[userDefaults setBool:YES forKey:kFIRPersistedDebugModeKey];
|
||||
asl_set_filter(sFIRLoggerClient, ASL_FILTER_MASK_UPTO(ASL_LEVEL_DEBUG));
|
||||
sFIRLoggerDebugMode = YES;
|
||||
}
|
||||
|
||||
// We should disable debug mode if we are running from App Store.
|
||||
if (sFIRLoggerDebugMode && [FIRAppEnvironmentUtil isFromAppStore]) {
|
||||
sFIRLoggerDebugMode = NO;
|
||||
}
|
||||
|
||||
sFIRClientQueue = dispatch_queue_create("FIRLoggingClientQueue", DISPATCH_QUEUE_SERIAL);
|
||||
dispatch_set_target_queue(sFIRClientQueue,
|
||||
dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0));
|
||||
|
||||
#ifdef DEBUG
|
||||
sMessageCodeRegex =
|
||||
[NSRegularExpression regularExpressionWithPattern:kMessageCodePattern options:0 error:NULL];
|
||||
#endif
|
||||
});
|
||||
}
|
||||
|
||||
void FIRSetAnalyticsDebugMode(BOOL analyticsDebugMode) {
|
||||
FIRLoggerInitializeASL();
|
||||
dispatch_async(sFIRClientQueue, ^{
|
||||
// We should not enable debug mode if we are running from App Store.
|
||||
if (analyticsDebugMode && [FIRAppEnvironmentUtil isFromAppStore]) {
|
||||
return;
|
||||
}
|
||||
sFIRAnalyticsDebugMode = analyticsDebugMode;
|
||||
asl_set_filter(sFIRLoggerClient, ASL_FILTER_MASK_UPTO(ASL_LEVEL_DEBUG));
|
||||
});
|
||||
}
|
||||
|
||||
void FIRSetLoggerLevel(FIRLoggerLevel loggerLevel) {
|
||||
if (loggerLevel < FIRLoggerLevelMin || loggerLevel > FIRLoggerLevelMax) {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000023", @"Invalid logger level, %ld", (long)loggerLevel);
|
||||
return;
|
||||
}
|
||||
FIRLoggerInitializeASL();
|
||||
// We should not raise the logger level if we are running from App Store.
|
||||
if (loggerLevel >= FIRLoggerLevelNotice && [FIRAppEnvironmentUtil isFromAppStore]) {
|
||||
return;
|
||||
}
|
||||
|
||||
sFIRLoggerMaximumLevel = loggerLevel;
|
||||
dispatch_async(sFIRClientQueue, ^{
|
||||
asl_set_filter(sFIRLoggerClient, ASL_FILTER_MASK_UPTO(loggerLevel));
|
||||
});
|
||||
}
|
||||
|
||||
BOOL FIRIsLoggableLevel(FIRLoggerLevel loggerLevel, BOOL analyticsComponent) {
|
||||
FIRLoggerInitializeASL();
|
||||
if (sFIRLoggerDebugMode) {
|
||||
return YES;
|
||||
} else if (sFIRAnalyticsDebugMode && analyticsComponent) {
|
||||
return YES;
|
||||
}
|
||||
return (BOOL)(loggerLevel <= sFIRLoggerMaximumLevel);
|
||||
}
|
||||
|
||||
#ifdef DEBUG
|
||||
void FIRResetLogger() {
|
||||
sFIRLoggerOnceToken = 0;
|
||||
[[NSUserDefaults standardUserDefaults] removeObjectForKey:kFIRPersistedDebugModeKey];
|
||||
}
|
||||
|
||||
aslclient getFIRLoggerClient() {
|
||||
return sFIRLoggerClient;
|
||||
}
|
||||
|
||||
dispatch_queue_t getFIRClientQueue() {
|
||||
return sFIRClientQueue;
|
||||
}
|
||||
|
||||
BOOL getFIRLoggerDebugMode() {
|
||||
return sFIRLoggerDebugMode;
|
||||
}
|
||||
#endif
|
||||
|
||||
void FIRLogBasic(FIRLoggerLevel level,
|
||||
FIRLoggerService service,
|
||||
NSString *messageCode,
|
||||
NSString *message,
|
||||
va_list args_ptr) {
|
||||
FIRLoggerInitializeASL();
|
||||
BOOL canLog = level <= sFIRLoggerMaximumLevel;
|
||||
|
||||
if (sFIRLoggerDebugMode) {
|
||||
canLog = YES;
|
||||
} else if (sFIRAnalyticsDebugMode && [kFIRLoggerAnalytics isEqualToString:service]) {
|
||||
canLog = YES;
|
||||
}
|
||||
|
||||
if (!canLog) {
|
||||
return;
|
||||
}
|
||||
#ifdef DEBUG
|
||||
NSCAssert(messageCode.length == 11, @"Incorrect message code length.");
|
||||
NSRange messageCodeRange = NSMakeRange(0, messageCode.length);
|
||||
NSUInteger numberOfMatches =
|
||||
[sMessageCodeRegex numberOfMatchesInString:messageCode options:0 range:messageCodeRange];
|
||||
NSCAssert(numberOfMatches == 1, @"Incorrect message code format.");
|
||||
#endif
|
||||
NSString *logMsg = [[NSString alloc] initWithFormat:message arguments:args_ptr];
|
||||
logMsg =
|
||||
[NSString stringWithFormat:@"%s - %@[%@] %@", FIRVersionString, service, messageCode, logMsg];
|
||||
dispatch_async(sFIRClientQueue, ^{
|
||||
asl_log(sFIRLoggerClient, NULL, level, "%s", logMsg.UTF8String);
|
||||
});
|
||||
}
|
||||
#pragma clang diagnostic pop
|
||||
|
||||
/**
|
||||
* Generates the logging functions using macros.
|
||||
*
|
||||
* Calling FIRLogError(kFIRLoggerCore, @"I-COR000001", @"Configure %@ failed.", @"blah") shows:
|
||||
* yyyy-mm-dd hh:mm:ss.SSS sender[PID] <Error> [Firebase/Core][I-COR000001] Configure blah failed.
|
||||
* Calling FIRLogDebug(kFIRLoggerCore, @"I-COR000001", @"Configure succeed.") shows:
|
||||
* yyyy-mm-dd hh:mm:ss.SSS sender[PID] <Debug> [Firebase/Core][I-COR000001] Configure succeed.
|
||||
*/
|
||||
#define FIR_LOGGING_FUNCTION(level) \
|
||||
void FIRLog##level(FIRLoggerService service, NSString *messageCode, NSString *message, ...) { \
|
||||
va_list args_ptr; \
|
||||
va_start(args_ptr, message); \
|
||||
FIRLogBasic(FIRLoggerLevel##level, service, messageCode, message, args_ptr); \
|
||||
va_end(args_ptr); \
|
||||
}
|
||||
|
||||
FIR_LOGGING_FUNCTION(Error)
|
||||
FIR_LOGGING_FUNCTION(Warning)
|
||||
FIR_LOGGING_FUNCTION(Notice)
|
||||
FIR_LOGGING_FUNCTION(Info)
|
||||
FIR_LOGGING_FUNCTION(Debug)
|
||||
|
||||
#undef FIR_MAKE_LOGGER
|
||||
|
||||
#pragma mark - FIRLoggerWrapper
|
||||
|
||||
@implementation FIRLoggerWrapper
|
||||
|
||||
+ (void)logWithLevel:(FIRLoggerLevel)level
|
||||
withService:(FIRLoggerService)service
|
||||
withCode:(NSString *)messageCode
|
||||
withMessage:(NSString *)message
|
||||
withArgs:(va_list)args {
|
||||
FIRLogBasic(level, service, messageCode, message, args);
|
||||
}
|
||||
|
||||
@end
|
||||
97
Pods/FirebaseCore/Firebase/Core/FIRMutableDictionary.m
generated
Normal file
97
Pods/FirebaseCore/Firebase/Core/FIRMutableDictionary.m
generated
Normal file
@@ -0,0 +1,97 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "Private/FIRMutableDictionary.h"
|
||||
|
||||
@implementation FIRMutableDictionary {
|
||||
/// The mutable dictionary.
|
||||
NSMutableDictionary *_objects;
|
||||
|
||||
/// Serial synchronization queue. All reads should use dispatch_sync, while writes use
|
||||
/// dispatch_async.
|
||||
dispatch_queue_t _queue;
|
||||
}
|
||||
|
||||
- (instancetype)init {
|
||||
self = [super init];
|
||||
|
||||
if (self) {
|
||||
_objects = [[NSMutableDictionary alloc] init];
|
||||
_queue = dispatch_queue_create("FIRMutableDictionary", DISPATCH_QUEUE_SERIAL);
|
||||
}
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
- (NSString *)description {
|
||||
__block NSString *description;
|
||||
dispatch_sync(_queue, ^{
|
||||
description = self->_objects.description;
|
||||
});
|
||||
return description;
|
||||
}
|
||||
|
||||
- (id)objectForKey:(id)key {
|
||||
__block id object;
|
||||
dispatch_sync(_queue, ^{
|
||||
object = self->_objects[key];
|
||||
});
|
||||
return object;
|
||||
}
|
||||
|
||||
- (void)setObject:(id)object forKey:(id<NSCopying>)key {
|
||||
dispatch_async(_queue, ^{
|
||||
self->_objects[key] = object;
|
||||
});
|
||||
}
|
||||
|
||||
- (void)removeObjectForKey:(id)key {
|
||||
dispatch_async(_queue, ^{
|
||||
[self->_objects removeObjectForKey:key];
|
||||
});
|
||||
}
|
||||
|
||||
- (void)removeAllObjects {
|
||||
dispatch_async(_queue, ^{
|
||||
[self->_objects removeAllObjects];
|
||||
});
|
||||
}
|
||||
|
||||
- (NSUInteger)count {
|
||||
__block NSUInteger count;
|
||||
dispatch_sync(_queue, ^{
|
||||
count = self->_objects.count;
|
||||
});
|
||||
return count;
|
||||
}
|
||||
|
||||
- (id)objectForKeyedSubscript:(id<NSCopying>)key {
|
||||
// The method this calls is already synchronized.
|
||||
return [self objectForKey:key];
|
||||
}
|
||||
|
||||
- (void)setObject:(id)obj forKeyedSubscript:(id<NSCopying>)key {
|
||||
// The method this calls is already synchronized.
|
||||
[self setObject:obj forKey:key];
|
||||
}
|
||||
|
||||
- (NSDictionary *)dictionary {
|
||||
__block NSDictionary *dictionary;
|
||||
dispatch_sync(_queue, ^{
|
||||
dictionary = [self->_objects copy];
|
||||
});
|
||||
return dictionary;
|
||||
}
|
||||
|
||||
@end
|
||||
390
Pods/FirebaseCore/Firebase/Core/FIRNetwork.m
generated
Normal file
390
Pods/FirebaseCore/Firebase/Core/FIRNetwork.m
generated
Normal file
@@ -0,0 +1,390 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "Private/FIRNetwork.h"
|
||||
#import "Private/FIRNetworkMessageCode.h"
|
||||
|
||||
#import "Private/FIRLogger.h"
|
||||
#import "Private/FIRMutableDictionary.h"
|
||||
#import "Private/FIRNetworkConstants.h"
|
||||
#import "Private/FIRReachabilityChecker.h"
|
||||
|
||||
#import <GoogleToolboxForMac/GTMNSData+zlib.h>
|
||||
|
||||
/// Constant string for request header Content-Encoding.
|
||||
static NSString *const kFIRNetworkContentCompressionKey = @"Content-Encoding";
|
||||
|
||||
/// Constant string for request header Content-Encoding value.
|
||||
static NSString *const kFIRNetworkContentCompressionValue = @"gzip";
|
||||
|
||||
/// Constant string for request header Content-Length.
|
||||
static NSString *const kFIRNetworkContentLengthKey = @"Content-Length";
|
||||
|
||||
/// Constant string for request header Content-Type.
|
||||
static NSString *const kFIRNetworkContentTypeKey = @"Content-Type";
|
||||
|
||||
/// Constant string for request header Content-Type value.
|
||||
static NSString *const kFIRNetworkContentTypeValue = @"application/x-www-form-urlencoded";
|
||||
|
||||
/// Constant string for GET request method.
|
||||
static NSString *const kFIRNetworkGETRequestMethod = @"GET";
|
||||
|
||||
/// Constant string for POST request method.
|
||||
static NSString *const kFIRNetworkPOSTRequestMethod = @"POST";
|
||||
|
||||
/// Default constant string as a prefix for network logger.
|
||||
static NSString *const kFIRNetworkLogTag = @"Firebase/Network";
|
||||
|
||||
@interface FIRNetwork () <FIRReachabilityDelegate, FIRNetworkLoggerDelegate>
|
||||
@end
|
||||
|
||||
@implementation FIRNetwork {
|
||||
/// Network reachability.
|
||||
FIRReachabilityChecker *_reachability;
|
||||
|
||||
/// The dictionary of requests by session IDs { NSString : id }.
|
||||
FIRMutableDictionary *_requests;
|
||||
}
|
||||
|
||||
- (instancetype)init {
|
||||
return [self initWithReachabilityHost:kFIRNetworkReachabilityHost];
|
||||
}
|
||||
|
||||
- (instancetype)initWithReachabilityHost:(NSString *)reachabilityHost {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
// Setup reachability.
|
||||
_reachability = [[FIRReachabilityChecker alloc] initWithReachabilityDelegate:self
|
||||
loggerDelegate:self
|
||||
withHost:reachabilityHost];
|
||||
if (![_reachability start]) {
|
||||
return nil;
|
||||
}
|
||||
|
||||
_requests = [[FIRMutableDictionary alloc] init];
|
||||
_timeoutInterval = kFIRNetworkTimeOutInterval;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)dealloc {
|
||||
_reachability.reachabilityDelegate = nil;
|
||||
[_reachability stop];
|
||||
}
|
||||
|
||||
#pragma mark - External Methods
|
||||
|
||||
+ (void)handleEventsForBackgroundURLSessionID:(NSString *)sessionID
|
||||
completionHandler:(FIRNetworkSystemCompletionHandler)completionHandler {
|
||||
[FIRNetworkURLSession handleEventsForBackgroundURLSessionID:sessionID
|
||||
completionHandler:completionHandler];
|
||||
}
|
||||
|
||||
- (NSString *)postURL:(NSURL *)url
|
||||
payload:(NSData *)payload
|
||||
queue:(dispatch_queue_t)queue
|
||||
usingBackgroundSession:(BOOL)usingBackgroundSession
|
||||
completionHandler:(FIRNetworkCompletionHandler)handler {
|
||||
if (!url.absoluteString.length) {
|
||||
[self handleErrorWithCode:FIRErrorCodeNetworkInvalidURL queue:queue withHandler:handler];
|
||||
return nil;
|
||||
}
|
||||
|
||||
NSTimeInterval timeOutInterval = _timeoutInterval ?: kFIRNetworkTimeOutInterval;
|
||||
|
||||
NSMutableURLRequest *request =
|
||||
[[NSMutableURLRequest alloc] initWithURL:url
|
||||
cachePolicy:NSURLRequestReloadIgnoringLocalCacheData
|
||||
timeoutInterval:timeOutInterval];
|
||||
|
||||
if (!request) {
|
||||
[self handleErrorWithCode:FIRErrorCodeNetworkSessionTaskCreation
|
||||
queue:queue
|
||||
withHandler:handler];
|
||||
return nil;
|
||||
}
|
||||
|
||||
NSError *compressError = nil;
|
||||
NSData *compressedData = [NSData gtm_dataByGzippingData:payload error:&compressError];
|
||||
if (!compressedData || compressError) {
|
||||
if (compressError || payload.length > 0) {
|
||||
// If the payload is not empty but it fails to compress the payload, something has been wrong.
|
||||
[self handleErrorWithCode:FIRErrorCodeNetworkPayloadCompression
|
||||
queue:queue
|
||||
withHandler:handler];
|
||||
return nil;
|
||||
}
|
||||
compressedData = [[NSData alloc] init];
|
||||
}
|
||||
|
||||
NSString *postLength = @(compressedData.length).stringValue;
|
||||
|
||||
// Set up the request with the compressed data.
|
||||
[request setValue:postLength forHTTPHeaderField:kFIRNetworkContentLengthKey];
|
||||
request.HTTPBody = compressedData;
|
||||
request.HTTPMethod = kFIRNetworkPOSTRequestMethod;
|
||||
[request setValue:kFIRNetworkContentTypeValue forHTTPHeaderField:kFIRNetworkContentTypeKey];
|
||||
[request setValue:kFIRNetworkContentCompressionValue
|
||||
forHTTPHeaderField:kFIRNetworkContentCompressionKey];
|
||||
|
||||
FIRNetworkURLSession *fetcher = [[FIRNetworkURLSession alloc] initWithNetworkLoggerDelegate:self];
|
||||
fetcher.backgroundNetworkEnabled = usingBackgroundSession;
|
||||
|
||||
__weak FIRNetwork *weakSelf = self;
|
||||
NSString *requestID = [fetcher
|
||||
sessionIDFromAsyncPOSTRequest:request
|
||||
completionHandler:^(NSHTTPURLResponse *response, NSData *data,
|
||||
NSString *sessionID, NSError *error) {
|
||||
FIRNetwork *strongSelf = weakSelf;
|
||||
if (!strongSelf) {
|
||||
return;
|
||||
}
|
||||
dispatch_queue_t queueToDispatch = queue ? queue : dispatch_get_main_queue();
|
||||
dispatch_async(queueToDispatch, ^{
|
||||
if (sessionID.length) {
|
||||
[strongSelf->_requests removeObjectForKey:sessionID];
|
||||
}
|
||||
if (handler) {
|
||||
handler(response, data, error);
|
||||
}
|
||||
});
|
||||
}];
|
||||
if (!requestID) {
|
||||
[self handleErrorWithCode:FIRErrorCodeNetworkSessionTaskCreation
|
||||
queue:queue
|
||||
withHandler:handler];
|
||||
return nil;
|
||||
}
|
||||
|
||||
[self firNetwork_logWithLevel:kFIRNetworkLogLevelDebug
|
||||
messageCode:kFIRNetworkMessageCodeNetwork000
|
||||
message:@"Uploading data. Host"
|
||||
context:url];
|
||||
_requests[requestID] = fetcher;
|
||||
return requestID;
|
||||
}
|
||||
|
||||
- (NSString *)getURL:(NSURL *)url
|
||||
headers:(NSDictionary *)headers
|
||||
queue:(dispatch_queue_t)queue
|
||||
usingBackgroundSession:(BOOL)usingBackgroundSession
|
||||
completionHandler:(FIRNetworkCompletionHandler)handler {
|
||||
if (!url.absoluteString.length) {
|
||||
[self handleErrorWithCode:FIRErrorCodeNetworkInvalidURL queue:queue withHandler:handler];
|
||||
return nil;
|
||||
}
|
||||
|
||||
NSTimeInterval timeOutInterval = _timeoutInterval ?: kFIRNetworkTimeOutInterval;
|
||||
NSMutableURLRequest *request =
|
||||
[[NSMutableURLRequest alloc] initWithURL:url
|
||||
cachePolicy:NSURLRequestReloadIgnoringLocalCacheData
|
||||
timeoutInterval:timeOutInterval];
|
||||
|
||||
if (!request) {
|
||||
[self handleErrorWithCode:FIRErrorCodeNetworkSessionTaskCreation
|
||||
queue:queue
|
||||
withHandler:handler];
|
||||
return nil;
|
||||
}
|
||||
|
||||
request.HTTPMethod = kFIRNetworkGETRequestMethod;
|
||||
request.allHTTPHeaderFields = headers;
|
||||
|
||||
FIRNetworkURLSession *fetcher = [[FIRNetworkURLSession alloc] initWithNetworkLoggerDelegate:self];
|
||||
fetcher.backgroundNetworkEnabled = usingBackgroundSession;
|
||||
|
||||
__weak FIRNetwork *weakSelf = self;
|
||||
NSString *requestID = [fetcher
|
||||
sessionIDFromAsyncGETRequest:request
|
||||
completionHandler:^(NSHTTPURLResponse *response, NSData *data, NSString *sessionID,
|
||||
NSError *error) {
|
||||
FIRNetwork *strongSelf = weakSelf;
|
||||
if (!strongSelf) {
|
||||
return;
|
||||
}
|
||||
dispatch_queue_t queueToDispatch = queue ? queue : dispatch_get_main_queue();
|
||||
dispatch_async(queueToDispatch, ^{
|
||||
if (sessionID.length) {
|
||||
[strongSelf->_requests removeObjectForKey:sessionID];
|
||||
}
|
||||
if (handler) {
|
||||
handler(response, data, error);
|
||||
}
|
||||
});
|
||||
}];
|
||||
|
||||
if (!requestID) {
|
||||
[self handleErrorWithCode:FIRErrorCodeNetworkSessionTaskCreation
|
||||
queue:queue
|
||||
withHandler:handler];
|
||||
return nil;
|
||||
}
|
||||
|
||||
[self firNetwork_logWithLevel:kFIRNetworkLogLevelDebug
|
||||
messageCode:kFIRNetworkMessageCodeNetwork001
|
||||
message:@"Downloading data. Host"
|
||||
context:url];
|
||||
_requests[requestID] = fetcher;
|
||||
return requestID;
|
||||
}
|
||||
|
||||
- (BOOL)hasUploadInProgress {
|
||||
return _requests.count > 0;
|
||||
}
|
||||
|
||||
#pragma mark - Network Reachability
|
||||
|
||||
/// Tells reachability delegate to call reachabilityDidChangeToStatus: to notify the network
|
||||
/// reachability has changed.
|
||||
- (void)reachability:(FIRReachabilityChecker *)reachability
|
||||
statusChanged:(FIRReachabilityStatus)status {
|
||||
_networkConnected = (status == kFIRReachabilityViaCellular || status == kFIRReachabilityViaWifi);
|
||||
[_reachabilityDelegate reachabilityDidChange];
|
||||
}
|
||||
|
||||
#pragma mark - Network logger delegate
|
||||
|
||||
- (void)setLoggerDelegate:(id<FIRNetworkLoggerDelegate>)loggerDelegate {
|
||||
// Explicitly check whether the delegate responds to the methods because conformsToProtocol does
|
||||
// not work correctly even though the delegate does respond to the methods.
|
||||
if (!loggerDelegate ||
|
||||
![loggerDelegate
|
||||
respondsToSelector:@selector(firNetwork_logWithLevel:messageCode:message:contexts:)] ||
|
||||
![loggerDelegate
|
||||
respondsToSelector:@selector(firNetwork_logWithLevel:messageCode:message:context:)] ||
|
||||
!
|
||||
[loggerDelegate respondsToSelector:@selector(firNetwork_logWithLevel:messageCode:message:)]) {
|
||||
FIRLogError(kFIRLoggerAnalytics,
|
||||
[NSString stringWithFormat:@"I-NET%06ld", (long)kFIRNetworkMessageCodeNetwork002],
|
||||
@"Cannot set the network logger delegate: delegate does not conform to the network "
|
||||
"logger protocol.");
|
||||
return;
|
||||
}
|
||||
_loggerDelegate = loggerDelegate;
|
||||
}
|
||||
|
||||
#pragma mark - Private methods
|
||||
|
||||
/// Handles network error and calls completion handler with the error.
|
||||
- (void)handleErrorWithCode:(NSInteger)code
|
||||
queue:(dispatch_queue_t)queue
|
||||
withHandler:(FIRNetworkCompletionHandler)handler {
|
||||
NSDictionary *userInfo = @{kFIRNetworkErrorContext : @"Failed to create network request"};
|
||||
NSError *error =
|
||||
[[NSError alloc] initWithDomain:kFIRNetworkErrorDomain code:code userInfo:userInfo];
|
||||
[self firNetwork_logWithLevel:kFIRNetworkLogLevelWarning
|
||||
messageCode:kFIRNetworkMessageCodeNetwork002
|
||||
message:@"Failed to create network request. Code, error"
|
||||
contexts:@[ @(code), error ]];
|
||||
if (handler) {
|
||||
dispatch_queue_t queueToDispatch = queue ? queue : dispatch_get_main_queue();
|
||||
dispatch_async(queueToDispatch, ^{
|
||||
handler(nil, nil, error);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - Network logger
|
||||
|
||||
- (void)firNetwork_logWithLevel:(FIRNetworkLogLevel)logLevel
|
||||
messageCode:(FIRNetworkMessageCode)messageCode
|
||||
message:(NSString *)message
|
||||
contexts:(NSArray *)contexts {
|
||||
// Let the delegate log the message if there is a valid logger delegate. Otherwise, just log
|
||||
// errors/warnings/info messages to the console log.
|
||||
if (_loggerDelegate) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:logLevel
|
||||
messageCode:messageCode
|
||||
message:message
|
||||
contexts:contexts];
|
||||
return;
|
||||
}
|
||||
if (_isDebugModeEnabled || logLevel == kFIRNetworkLogLevelError ||
|
||||
logLevel == kFIRNetworkLogLevelWarning || logLevel == kFIRNetworkLogLevelInfo) {
|
||||
NSString *formattedMessage = FIRStringWithLogMessage(message, logLevel, contexts);
|
||||
NSLog(@"%@", formattedMessage);
|
||||
FIRLogBasic((FIRLoggerLevel)logLevel, kFIRLoggerCore,
|
||||
[NSString stringWithFormat:@"I-NET%06ld", (long)messageCode], formattedMessage,
|
||||
NULL);
|
||||
}
|
||||
}
|
||||
|
||||
- (void)firNetwork_logWithLevel:(FIRNetworkLogLevel)logLevel
|
||||
messageCode:(FIRNetworkMessageCode)messageCode
|
||||
message:(NSString *)message
|
||||
context:(id)context {
|
||||
if (_loggerDelegate) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:logLevel
|
||||
messageCode:messageCode
|
||||
message:message
|
||||
context:context];
|
||||
return;
|
||||
}
|
||||
NSArray *contexts = context ? @[ context ] : @[];
|
||||
[self firNetwork_logWithLevel:logLevel messageCode:messageCode message:message contexts:contexts];
|
||||
}
|
||||
|
||||
- (void)firNetwork_logWithLevel:(FIRNetworkLogLevel)logLevel
|
||||
messageCode:(FIRNetworkMessageCode)messageCode
|
||||
message:(NSString *)message {
|
||||
if (_loggerDelegate) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:logLevel messageCode:messageCode message:message];
|
||||
return;
|
||||
}
|
||||
[self firNetwork_logWithLevel:logLevel messageCode:messageCode message:message contexts:@[]];
|
||||
}
|
||||
|
||||
/// Returns a string for the given log level (e.g. kFIRNetworkLogLevelError -> @"ERROR").
|
||||
static NSString *FIRLogLevelDescriptionFromLogLevel(FIRNetworkLogLevel logLevel) {
|
||||
static NSDictionary *levelNames = nil;
|
||||
static dispatch_once_t onceToken;
|
||||
dispatch_once(&onceToken, ^{
|
||||
levelNames = @{
|
||||
@(kFIRNetworkLogLevelError) : @"ERROR",
|
||||
@(kFIRNetworkLogLevelWarning) : @"WARNING",
|
||||
@(kFIRNetworkLogLevelInfo) : @"INFO",
|
||||
@(kFIRNetworkLogLevelDebug) : @"DEBUG"
|
||||
};
|
||||
});
|
||||
return levelNames[@(logLevel)];
|
||||
}
|
||||
|
||||
/// Returns a formatted string to be used for console logging.
|
||||
static NSString *FIRStringWithLogMessage(NSString *message,
|
||||
FIRNetworkLogLevel logLevel,
|
||||
NSArray *contexts) {
|
||||
if (!message) {
|
||||
message = @"(Message was nil)";
|
||||
} else if (!message.length) {
|
||||
message = @"(Message was empty)";
|
||||
}
|
||||
NSMutableString *result = [[NSMutableString alloc]
|
||||
initWithFormat:@"<%@/%@> %@", kFIRNetworkLogTag, FIRLogLevelDescriptionFromLogLevel(logLevel),
|
||||
message];
|
||||
|
||||
if (!contexts.count) {
|
||||
return result;
|
||||
}
|
||||
|
||||
NSMutableArray *formattedContexts = [[NSMutableArray alloc] init];
|
||||
for (id item in contexts) {
|
||||
[formattedContexts addObject:(item != [NSNull null] ? item : @"(nil)")];
|
||||
}
|
||||
|
||||
[result appendString:@": "];
|
||||
[result appendString:[formattedContexts componentsJoinedByString:@", "]];
|
||||
return result;
|
||||
}
|
||||
|
||||
@end
|
||||
39
Pods/FirebaseCore/Firebase/Core/FIRNetworkConstants.m
generated
Normal file
39
Pods/FirebaseCore/Firebase/Core/FIRNetworkConstants.m
generated
Normal file
@@ -0,0 +1,39 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "Private/FIRNetworkConstants.h"
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
NSString *const kFIRNetworkBackgroundSessionConfigIDPrefix =
|
||||
@"com.firebase.network.background-upload";
|
||||
NSString *const kFIRNetworkApplicationSupportSubdirectory = @"Firebase/Network";
|
||||
NSString *const kFIRNetworkTempDirectoryName = @"FIRNetworkTemporaryDirectory";
|
||||
const NSTimeInterval kFIRNetworkTempFolderExpireTime = 60 * 60; // 1 hour
|
||||
const NSTimeInterval kFIRNetworkTimeOutInterval = 60; // 1 minute.
|
||||
NSString *const kFIRNetworkReachabilityHost = @"app-measurement.com";
|
||||
NSString *const kFIRNetworkErrorContext = @"Context";
|
||||
|
||||
const int kFIRNetworkHTTPStatusOK = 200;
|
||||
const int kFIRNetworkHTTPStatusNoContent = 204;
|
||||
const int kFIRNetworkHTTPStatusCodeMultipleChoices = 300;
|
||||
const int kFIRNetworkHTTPStatusCodeMovedPermanently = 301;
|
||||
const int kFIRNetworkHTTPStatusCodeFound = 302;
|
||||
const int kFIRNetworkHTTPStatusCodeNotModified = 304;
|
||||
const int kFIRNetworkHTTPStatusCodeMovedTemporarily = 307;
|
||||
const int kFIRNetworkHTTPStatusCodeNotFound = 404;
|
||||
const int kFIRNetworkHTTPStatusCodeCannotAcceptTraffic = 429;
|
||||
const int kFIRNetworkHTTPStatusCodeUnavailable = 503;
|
||||
|
||||
NSString *const kFIRNetworkErrorDomain = @"com.firebase.network.ErrorDomain";
|
||||
669
Pods/FirebaseCore/Firebase/Core/FIRNetworkURLSession.m
generated
Normal file
669
Pods/FirebaseCore/Firebase/Core/FIRNetworkURLSession.m
generated
Normal file
@@ -0,0 +1,669 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "Private/FIRNetworkURLSession.h"
|
||||
|
||||
#import "Private/FIRLogger.h"
|
||||
#import "Private/FIRMutableDictionary.h"
|
||||
#import "Private/FIRNetworkConstants.h"
|
||||
#import "Private/FIRNetworkMessageCode.h"
|
||||
|
||||
@implementation FIRNetworkURLSession {
|
||||
/// The handler to be called when the request completes or error has occurs.
|
||||
FIRNetworkURLSessionCompletionHandler _completionHandler;
|
||||
|
||||
/// Session ID generated randomly with a fixed prefix.
|
||||
NSString *_sessionID;
|
||||
|
||||
/// The session configuration.
|
||||
NSURLSessionConfiguration *_sessionConfig;
|
||||
|
||||
/// The path to the directory where all temporary files are stored before uploading.
|
||||
NSURL *_networkDirectoryURL;
|
||||
|
||||
/// The downloaded data from fetching.
|
||||
NSData *_downloadedData;
|
||||
|
||||
/// The path to the temporary file which stores the uploading data.
|
||||
NSURL *_uploadingFileURL;
|
||||
|
||||
/// The current request.
|
||||
NSURLRequest *_request;
|
||||
}
|
||||
|
||||
#pragma mark - Init
|
||||
|
||||
- (instancetype)initWithNetworkLoggerDelegate:(id<FIRNetworkLoggerDelegate>)networkLoggerDelegate {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
// Create URL to the directory where all temporary files to upload have to be stored.
|
||||
NSArray *paths =
|
||||
NSSearchPathForDirectoriesInDomains(NSApplicationSupportDirectory, NSUserDomainMask, YES);
|
||||
NSString *applicationSupportDirectory = paths.firstObject;
|
||||
NSArray *tempPathComponents = @[
|
||||
applicationSupportDirectory, kFIRNetworkApplicationSupportSubdirectory,
|
||||
kFIRNetworkTempDirectoryName
|
||||
];
|
||||
_networkDirectoryURL = [NSURL fileURLWithPathComponents:tempPathComponents];
|
||||
_sessionID = [NSString stringWithFormat:@"%@-%@", kFIRNetworkBackgroundSessionConfigIDPrefix,
|
||||
[[NSUUID UUID] UUIDString]];
|
||||
_loggerDelegate = networkLoggerDelegate;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
#pragma mark - External Methods
|
||||
|
||||
#pragma mark - To be called from AppDelegate
|
||||
|
||||
+ (void)handleEventsForBackgroundURLSessionID:(NSString *)sessionID
|
||||
completionHandler:
|
||||
(FIRNetworkSystemCompletionHandler)systemCompletionHandler {
|
||||
// The session may not be FIRAnalytics background. Ignore those that do not have the prefix.
|
||||
if (![sessionID hasPrefix:kFIRNetworkBackgroundSessionConfigIDPrefix]) {
|
||||
return;
|
||||
}
|
||||
FIRNetworkURLSession *fetcher = [self fetcherWithSessionIdentifier:sessionID];
|
||||
if (fetcher != nil) {
|
||||
[fetcher addSystemCompletionHandler:systemCompletionHandler forSession:sessionID];
|
||||
} else {
|
||||
FIRLogError(kFIRLoggerCore,
|
||||
[NSString stringWithFormat:@"I-NET%06ld", (long)kFIRNetworkMessageCodeNetwork003],
|
||||
@"Failed to retrieve background session with ID %@ after app is relaunched.",
|
||||
sessionID);
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - External Methods
|
||||
|
||||
/// Sends an async POST request using NSURLSession for iOS >= 7.0, and returns an ID of the
|
||||
/// connection.
|
||||
- (NSString *)sessionIDFromAsyncPOSTRequest:(NSURLRequest *)request
|
||||
completionHandler:(FIRNetworkURLSessionCompletionHandler)handler {
|
||||
// NSURLSessionUploadTask does not work with NSData in the background.
|
||||
// To avoid this issue, write the data to a temporary file to upload it.
|
||||
// Make a temporary file with the data subset.
|
||||
_uploadingFileURL = [self temporaryFilePathWithSessionID:_sessionID];
|
||||
NSError *writeError;
|
||||
NSURLSessionUploadTask *postRequestTask;
|
||||
NSURLSession *session;
|
||||
BOOL didWriteFile = NO;
|
||||
|
||||
// Clean up the entire temp folder to avoid temp files that remain in case the previous session
|
||||
// crashed and did not clean up.
|
||||
[self maybeRemoveTempFilesAtURL:_networkDirectoryURL
|
||||
expiringTime:kFIRNetworkTempFolderExpireTime];
|
||||
|
||||
// If there is no background network enabled, no need to write to file. This will allow default
|
||||
// network session which runs on the foreground.
|
||||
if (_backgroundNetworkEnabled && [self ensureTemporaryDirectoryExists]) {
|
||||
didWriteFile = [request.HTTPBody writeToFile:_uploadingFileURL.path
|
||||
options:NSDataWritingAtomic
|
||||
error:&writeError];
|
||||
|
||||
if (writeError) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession000
|
||||
message:@"Failed to write request data to file"
|
||||
context:writeError];
|
||||
}
|
||||
}
|
||||
|
||||
if (didWriteFile) {
|
||||
// Exclude this file from backing up to iTunes. There are conflicting reports that excluding
|
||||
// directory from backing up does not excluding files of that directory from backing up.
|
||||
[self excludeFromBackupForURL:_uploadingFileURL];
|
||||
|
||||
_sessionConfig = [self backgroundSessionConfigWithSessionID:_sessionID];
|
||||
[self populateSessionConfig:_sessionConfig withRequest:request];
|
||||
session = [NSURLSession sessionWithConfiguration:_sessionConfig
|
||||
delegate:self
|
||||
delegateQueue:[NSOperationQueue mainQueue]];
|
||||
postRequestTask = [session uploadTaskWithRequest:request fromFile:_uploadingFileURL];
|
||||
} else {
|
||||
// If we cannot write to file, just send it in the foreground.
|
||||
_sessionConfig = [NSURLSessionConfiguration defaultSessionConfiguration];
|
||||
[self populateSessionConfig:_sessionConfig withRequest:request];
|
||||
_sessionConfig.URLCache = nil;
|
||||
session = [NSURLSession sessionWithConfiguration:_sessionConfig
|
||||
delegate:self
|
||||
delegateQueue:[NSOperationQueue mainQueue]];
|
||||
postRequestTask = [session uploadTaskWithRequest:request fromData:request.HTTPBody];
|
||||
}
|
||||
|
||||
if (!session || !postRequestTask) {
|
||||
NSError *error = [[NSError alloc]
|
||||
initWithDomain:kFIRNetworkErrorDomain
|
||||
code:FIRErrorCodeNetworkRequestCreation
|
||||
userInfo:@{kFIRNetworkErrorContext : @"Cannot create network session"}];
|
||||
[self callCompletionHandler:handler withResponse:nil data:nil error:error];
|
||||
return nil;
|
||||
}
|
||||
|
||||
// Save the session into memory.
|
||||
NSMapTable *sessionIdentifierToFetcherMap = [[self class] sessionIDToFetcherMap];
|
||||
[sessionIdentifierToFetcherMap setObject:self forKey:_sessionID];
|
||||
|
||||
_request = [request copy];
|
||||
|
||||
// Store completion handler because background session does not accept handler block but custom
|
||||
// delegate.
|
||||
_completionHandler = [handler copy];
|
||||
[postRequestTask resume];
|
||||
|
||||
return _sessionID;
|
||||
}
|
||||
|
||||
/// Sends an async GET request using NSURLSession for iOS >= 7.0, and returns an ID of the session.
|
||||
- (NSString *)sessionIDFromAsyncGETRequest:(NSURLRequest *)request
|
||||
completionHandler:(FIRNetworkURLSessionCompletionHandler)handler {
|
||||
if (_backgroundNetworkEnabled) {
|
||||
_sessionConfig = [self backgroundSessionConfigWithSessionID:_sessionID];
|
||||
} else {
|
||||
_sessionConfig = [NSURLSessionConfiguration defaultSessionConfiguration];
|
||||
}
|
||||
|
||||
[self populateSessionConfig:_sessionConfig withRequest:request];
|
||||
|
||||
// Do not cache the GET request.
|
||||
_sessionConfig.URLCache = nil;
|
||||
|
||||
NSURLSession *session = [NSURLSession sessionWithConfiguration:_sessionConfig
|
||||
delegate:self
|
||||
delegateQueue:[NSOperationQueue mainQueue]];
|
||||
NSURLSessionDownloadTask *downloadTask = [session downloadTaskWithRequest:request];
|
||||
|
||||
if (!session || !downloadTask) {
|
||||
NSError *error = [[NSError alloc]
|
||||
initWithDomain:kFIRNetworkErrorDomain
|
||||
code:FIRErrorCodeNetworkRequestCreation
|
||||
userInfo:@{kFIRNetworkErrorContext : @"Cannot create network session"}];
|
||||
[self callCompletionHandler:handler withResponse:nil data:nil error:error];
|
||||
return nil;
|
||||
}
|
||||
|
||||
// Save the session into memory.
|
||||
NSMapTable *sessionIdentifierToFetcherMap = [[self class] sessionIDToFetcherMap];
|
||||
[sessionIdentifierToFetcherMap setObject:self forKey:_sessionID];
|
||||
|
||||
_request = [request copy];
|
||||
|
||||
_completionHandler = [handler copy];
|
||||
[downloadTask resume];
|
||||
|
||||
return _sessionID;
|
||||
}
|
||||
|
||||
#pragma mark - NSURLSessionTaskDelegate
|
||||
|
||||
/// Called by the NSURLSession once the download task is completed. The file is saved in the
|
||||
/// provided URL so we need to read the data and store into _downloadedData. Once the session is
|
||||
/// completed, URLSession:task:didCompleteWithError will be called and the completion handler will
|
||||
/// be called with the downloaded data.
|
||||
- (void)URLSession:(NSURLSession *)session
|
||||
downloadTask:(NSURLSessionDownloadTask *)task
|
||||
didFinishDownloadingToURL:(NSURL *)url {
|
||||
if (!url.path) {
|
||||
[_loggerDelegate
|
||||
firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession001
|
||||
message:@"Unable to read downloaded data from empty temp path"];
|
||||
_downloadedData = nil;
|
||||
return;
|
||||
}
|
||||
|
||||
NSError *error;
|
||||
_downloadedData = [NSData dataWithContentsOfFile:url.path options:0 error:&error];
|
||||
|
||||
if (error) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession002
|
||||
message:@"Cannot read the content of downloaded data"
|
||||
context:error];
|
||||
_downloadedData = nil;
|
||||
}
|
||||
}
|
||||
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
- (void)URLSessionDidFinishEventsForBackgroundURLSession:(NSURLSession *)session {
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelDebug
|
||||
messageCode:kFIRNetworkMessageCodeURLSession003
|
||||
message:@"Background session finished"
|
||||
context:session.configuration.identifier];
|
||||
[self callSystemCompletionHandler:session.configuration.identifier];
|
||||
}
|
||||
#endif
|
||||
|
||||
- (void)URLSession:(NSURLSession *)session
|
||||
task:(NSURLSessionTask *)task
|
||||
didCompleteWithError:(NSError *)error {
|
||||
// Avoid any chance of recursive behavior leading to it being used repeatedly.
|
||||
FIRNetworkURLSessionCompletionHandler handler = _completionHandler;
|
||||
_completionHandler = nil;
|
||||
|
||||
if (task.response) {
|
||||
// The following assertion should always be true for HTTP requests, see https://goo.gl/gVLxT7.
|
||||
NSAssert([task.response isKindOfClass:[NSHTTPURLResponse class]], @"URL response must be HTTP");
|
||||
|
||||
// The server responded so ignore the error created by the system.
|
||||
error = nil;
|
||||
} else if (!error) {
|
||||
error = [[NSError alloc]
|
||||
initWithDomain:kFIRNetworkErrorDomain
|
||||
code:FIRErrorCodeNetworkInvalidResponse
|
||||
userInfo:@{kFIRNetworkErrorContext : @"Network Error: Empty network response"}];
|
||||
}
|
||||
|
||||
[self callCompletionHandler:handler
|
||||
withResponse:(NSHTTPURLResponse *)task.response
|
||||
data:_downloadedData
|
||||
error:error];
|
||||
|
||||
// Remove the temp file to avoid trashing devices with lots of temp files.
|
||||
[self removeTempItemAtURL:_uploadingFileURL];
|
||||
|
||||
// Try to clean up stale files again.
|
||||
[self maybeRemoveTempFilesAtURL:_networkDirectoryURL
|
||||
expiringTime:kFIRNetworkTempFolderExpireTime];
|
||||
}
|
||||
|
||||
- (void)URLSession:(NSURLSession *)session
|
||||
task:(NSURLSessionTask *)task
|
||||
didReceiveChallenge:(NSURLAuthenticationChallenge *)challenge
|
||||
completionHandler:(void (^)(NSURLSessionAuthChallengeDisposition disposition,
|
||||
NSURLCredential *credential))completionHandler {
|
||||
// The handling is modeled after GTMSessionFetcher.
|
||||
if ([challenge.protectionSpace.authenticationMethod
|
||||
isEqualToString:NSURLAuthenticationMethodServerTrust]) {
|
||||
SecTrustRef serverTrust = challenge.protectionSpace.serverTrust;
|
||||
if (serverTrust == NULL) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelDebug
|
||||
messageCode:kFIRNetworkMessageCodeURLSession004
|
||||
message:@"Received empty server trust for host. Host"
|
||||
context:_request.URL];
|
||||
completionHandler(NSURLSessionAuthChallengePerformDefaultHandling, nil);
|
||||
return;
|
||||
}
|
||||
NSURLCredential *credential = [NSURLCredential credentialForTrust:serverTrust];
|
||||
if (!credential) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelWarning
|
||||
messageCode:kFIRNetworkMessageCodeURLSession005
|
||||
message:@"Unable to verify server identity. Host"
|
||||
context:_request.URL];
|
||||
completionHandler(NSURLSessionAuthChallengeCancelAuthenticationChallenge, nil);
|
||||
return;
|
||||
}
|
||||
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelDebug
|
||||
messageCode:kFIRNetworkMessageCodeURLSession006
|
||||
message:@"Received SSL challenge for host. Host"
|
||||
context:_request.URL];
|
||||
|
||||
void (^callback)(BOOL) = ^(BOOL allow) {
|
||||
if (allow) {
|
||||
completionHandler(NSURLSessionAuthChallengeUseCredential, credential);
|
||||
} else {
|
||||
[self->_loggerDelegate
|
||||
firNetwork_logWithLevel:kFIRNetworkLogLevelDebug
|
||||
messageCode:kFIRNetworkMessageCodeURLSession007
|
||||
message:@"Cancelling authentication challenge for host. Host"
|
||||
context:self->_request.URL];
|
||||
completionHandler(NSURLSessionAuthChallengeCancelAuthenticationChallenge, nil);
|
||||
}
|
||||
};
|
||||
|
||||
// Retain the trust object to avoid a SecTrustEvaluate() crash on iOS 7.
|
||||
CFRetain(serverTrust);
|
||||
|
||||
// Evaluate the certificate chain.
|
||||
//
|
||||
// The delegate queue may be the main thread. Trust evaluation could cause some
|
||||
// blocking network activity, so we must evaluate async, as documented at
|
||||
// https://developer.apple.com/library/ios/technotes/tn2232/
|
||||
dispatch_queue_t evaluateBackgroundQueue =
|
||||
dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
|
||||
|
||||
dispatch_async(evaluateBackgroundQueue, ^{
|
||||
SecTrustResultType trustEval = kSecTrustResultInvalid;
|
||||
BOOL shouldAllow;
|
||||
OSStatus trustError;
|
||||
|
||||
@synchronized([FIRNetworkURLSession class]) {
|
||||
trustError = SecTrustEvaluate(serverTrust, &trustEval);
|
||||
}
|
||||
|
||||
if (trustError != errSecSuccess) {
|
||||
[self->_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession008
|
||||
message:@"Cannot evaluate server trust. Error, host"
|
||||
contexts:@[ @(trustError), self->_request.URL ]];
|
||||
shouldAllow = NO;
|
||||
} else {
|
||||
// Having a trust level "unspecified" by the user is the usual result, described at
|
||||
// https://developer.apple.com/library/mac/qa/qa1360
|
||||
shouldAllow =
|
||||
(trustEval == kSecTrustResultUnspecified || trustEval == kSecTrustResultProceed);
|
||||
}
|
||||
|
||||
// Call the call back with the permission.
|
||||
callback(shouldAllow);
|
||||
|
||||
CFRelease(serverTrust);
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
// Default handling for other Auth Challenges.
|
||||
completionHandler(NSURLSessionAuthChallengePerformDefaultHandling, nil);
|
||||
}
|
||||
|
||||
#pragma mark - Internal Methods
|
||||
|
||||
/// Stores system completion handler with session ID as key.
|
||||
- (void)addSystemCompletionHandler:(FIRNetworkSystemCompletionHandler)handler
|
||||
forSession:(NSString *)identifier {
|
||||
if (!handler) {
|
||||
[_loggerDelegate
|
||||
firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession009
|
||||
message:@"Cannot store nil system completion handler in network"];
|
||||
return;
|
||||
}
|
||||
|
||||
if (!identifier.length) {
|
||||
[_loggerDelegate
|
||||
firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession010
|
||||
message:
|
||||
@"Cannot store system completion handler with empty network "
|
||||
"session identifier"];
|
||||
return;
|
||||
}
|
||||
|
||||
FIRMutableDictionary *systemCompletionHandlers =
|
||||
[[self class] sessionIDToSystemCompletionHandlerDictionary];
|
||||
if (systemCompletionHandlers[identifier]) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelWarning
|
||||
messageCode:kFIRNetworkMessageCodeURLSession011
|
||||
message:@"Got multiple system handlers for a single session ID"
|
||||
context:identifier];
|
||||
}
|
||||
|
||||
systemCompletionHandlers[identifier] = handler;
|
||||
}
|
||||
|
||||
/// Calls the system provided completion handler with the session ID stored in the dictionary.
|
||||
/// The handler will be removed from the dictionary after being called.
|
||||
- (void)callSystemCompletionHandler:(NSString *)identifier {
|
||||
FIRMutableDictionary *systemCompletionHandlers =
|
||||
[[self class] sessionIDToSystemCompletionHandlerDictionary];
|
||||
FIRNetworkSystemCompletionHandler handler = [systemCompletionHandlers objectForKey:identifier];
|
||||
|
||||
if (handler) {
|
||||
[systemCompletionHandlers removeObjectForKey:identifier];
|
||||
|
||||
dispatch_async(dispatch_get_main_queue(), ^{
|
||||
handler();
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
/// Sets or updates the session ID of this session.
|
||||
- (void)setSessionID:(NSString *)sessionID {
|
||||
_sessionID = [sessionID copy];
|
||||
}
|
||||
|
||||
/// Creates a background session configuration with the session ID using the supported method.
|
||||
- (NSURLSessionConfiguration *)backgroundSessionConfigWithSessionID:(NSString *)sessionID {
|
||||
#if (TARGET_OS_OSX && defined(MAC_OS_X_VERSION_10_10) && \
|
||||
MAC_OS_X_VERSION_MIN_REQUIRED >= MAC_OS_X_VERSION_10_10) || \
|
||||
TARGET_OS_TV || \
|
||||
(TARGET_OS_IOS && defined(__IPHONE_8_0) && __IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_8_0)
|
||||
|
||||
// iOS 8/10.10 builds require the new backgroundSessionConfiguration method name.
|
||||
return [NSURLSessionConfiguration backgroundSessionConfigurationWithIdentifier:sessionID];
|
||||
|
||||
#elif (TARGET_OS_OSX && defined(MAC_OS_X_VERSION_10_10) && \
|
||||
MAC_OS_X_VERSION_MIN_REQUIRED < MAC_OS_X_VERSION_10_10) || \
|
||||
(TARGET_OS_IOS && defined(__IPHONE_8_0) && __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_8_0)
|
||||
|
||||
// Do a runtime check to avoid a deprecation warning about using
|
||||
// +backgroundSessionConfiguration: on iOS 8.
|
||||
if ([NSURLSessionConfiguration
|
||||
respondsToSelector:@selector(backgroundSessionConfigurationWithIdentifier:)]) {
|
||||
// Running on iOS 8+/OS X 10.10+.
|
||||
return [NSURLSessionConfiguration backgroundSessionConfigurationWithIdentifier:sessionID];
|
||||
} else {
|
||||
// Running on iOS 7/OS X 10.9.
|
||||
return [NSURLSessionConfiguration backgroundSessionConfiguration:sessionID];
|
||||
}
|
||||
|
||||
#else
|
||||
// Building with an SDK earlier than iOS 8/OS X 10.10.
|
||||
return [NSURLSessionConfiguration backgroundSessionConfiguration:sessionID];
|
||||
#endif
|
||||
}
|
||||
|
||||
- (void)maybeRemoveTempFilesAtURL:(NSURL *)folderURL expiringTime:(NSTimeInterval)staleTime {
|
||||
if (!folderURL.absoluteString.length) {
|
||||
return;
|
||||
}
|
||||
|
||||
NSFileManager *fileManager = [NSFileManager defaultManager];
|
||||
NSError *error = nil;
|
||||
|
||||
NSArray *properties = @[ NSURLCreationDateKey ];
|
||||
NSArray *directoryContent =
|
||||
[fileManager contentsOfDirectoryAtURL:folderURL
|
||||
includingPropertiesForKeys:properties
|
||||
options:NSDirectoryEnumerationSkipsSubdirectoryDescendants
|
||||
error:&error];
|
||||
if (error && error.code != NSFileReadNoSuchFileError) {
|
||||
[_loggerDelegate
|
||||
firNetwork_logWithLevel:kFIRNetworkLogLevelDebug
|
||||
messageCode:kFIRNetworkMessageCodeURLSession012
|
||||
message:@"Cannot get files from the temporary network folder. Error"
|
||||
context:error];
|
||||
return;
|
||||
}
|
||||
|
||||
if (!directoryContent.count) {
|
||||
return;
|
||||
}
|
||||
|
||||
NSTimeInterval now = [NSDate date].timeIntervalSince1970;
|
||||
for (NSURL *tempFile in directoryContent) {
|
||||
NSDate *creationDate;
|
||||
BOOL getCreationDate =
|
||||
[tempFile getResourceValue:&creationDate forKey:NSURLCreationDateKey error:NULL];
|
||||
if (!getCreationDate) {
|
||||
continue;
|
||||
}
|
||||
NSTimeInterval creationTimeInterval = creationDate.timeIntervalSince1970;
|
||||
if (fabs(now - creationTimeInterval) > staleTime) {
|
||||
[self removeTempItemAtURL:tempFile];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Removes the temporary file written to disk for sending the request. It has to be cleaned up
|
||||
/// after the session is done.
|
||||
- (void)removeTempItemAtURL:(NSURL *)fileURL {
|
||||
if (!fileURL.absoluteString.length) {
|
||||
return;
|
||||
}
|
||||
|
||||
NSFileManager *fileManager = [NSFileManager defaultManager];
|
||||
NSError *error = nil;
|
||||
|
||||
if (![fileManager removeItemAtURL:fileURL error:&error] && error.code != NSFileNoSuchFileError) {
|
||||
[_loggerDelegate
|
||||
firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession013
|
||||
message:@"Failed to remove temporary uploading data file. Error"
|
||||
context:error.localizedDescription];
|
||||
}
|
||||
}
|
||||
|
||||
/// Gets the fetcher with the session ID.
|
||||
+ (instancetype)fetcherWithSessionIdentifier:(NSString *)sessionIdentifier {
|
||||
NSMapTable *sessionIdentifierToFetcherMap = [self sessionIDToFetcherMap];
|
||||
FIRNetworkURLSession *session = [sessionIdentifierToFetcherMap objectForKey:sessionIdentifier];
|
||||
if (!session && [sessionIdentifier hasPrefix:kFIRNetworkBackgroundSessionConfigIDPrefix]) {
|
||||
session = [[FIRNetworkURLSession alloc] initWithNetworkLoggerDelegate:nil];
|
||||
[session setSessionID:sessionIdentifier];
|
||||
[sessionIdentifierToFetcherMap setObject:session forKey:sessionIdentifier];
|
||||
}
|
||||
return session;
|
||||
}
|
||||
|
||||
/// Returns a map of the fetcher by session ID. Creates a map if it is not created.
|
||||
+ (NSMapTable *)sessionIDToFetcherMap {
|
||||
static NSMapTable *sessionIDToFetcherMap;
|
||||
|
||||
static dispatch_once_t sessionMapOnceToken;
|
||||
dispatch_once(&sessionMapOnceToken, ^{
|
||||
sessionIDToFetcherMap = [NSMapTable strongToWeakObjectsMapTable];
|
||||
});
|
||||
return sessionIDToFetcherMap;
|
||||
}
|
||||
|
||||
/// Returns a map of system provided completion handler by session ID. Creates a map if it is not
|
||||
/// created.
|
||||
+ (FIRMutableDictionary *)sessionIDToSystemCompletionHandlerDictionary {
|
||||
static FIRMutableDictionary *systemCompletionHandlers;
|
||||
|
||||
static dispatch_once_t systemCompletionHandlerOnceToken;
|
||||
dispatch_once(&systemCompletionHandlerOnceToken, ^{
|
||||
systemCompletionHandlers = [[FIRMutableDictionary alloc] init];
|
||||
});
|
||||
return systemCompletionHandlers;
|
||||
}
|
||||
|
||||
- (NSURL *)temporaryFilePathWithSessionID:(NSString *)sessionID {
|
||||
NSString *tempName = [NSString stringWithFormat:@"FIRUpload_temp_%@", sessionID];
|
||||
return [_networkDirectoryURL URLByAppendingPathComponent:tempName];
|
||||
}
|
||||
|
||||
/// Makes sure that the directory to store temp files exists. If not, tries to create it and returns
|
||||
/// YES. If there is anything wrong, returns NO.
|
||||
- (BOOL)ensureTemporaryDirectoryExists {
|
||||
NSFileManager *fileManager = [NSFileManager defaultManager];
|
||||
NSError *error = nil;
|
||||
|
||||
// Create a temporary directory if it does not exist or was deleted.
|
||||
if ([_networkDirectoryURL checkResourceIsReachableAndReturnError:&error]) {
|
||||
return YES;
|
||||
}
|
||||
|
||||
if (error && error.code != NSFileReadNoSuchFileError) {
|
||||
[_loggerDelegate
|
||||
firNetwork_logWithLevel:kFIRNetworkLogLevelWarning
|
||||
messageCode:kFIRNetworkMessageCodeURLSession014
|
||||
message:@"Error while trying to access Network temp folder. Error"
|
||||
context:error];
|
||||
}
|
||||
|
||||
NSError *writeError = nil;
|
||||
|
||||
[fileManager createDirectoryAtURL:_networkDirectoryURL
|
||||
withIntermediateDirectories:YES
|
||||
attributes:nil
|
||||
error:&writeError];
|
||||
if (writeError) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession015
|
||||
message:@"Cannot create temporary directory. Error"
|
||||
context:writeError];
|
||||
return NO;
|
||||
}
|
||||
|
||||
// Set the iCloud exclusion attribute on the Documents URL.
|
||||
[self excludeFromBackupForURL:_networkDirectoryURL];
|
||||
|
||||
return YES;
|
||||
}
|
||||
|
||||
- (void)excludeFromBackupForURL:(NSURL *)url {
|
||||
if (!url.path) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Set the iCloud exclusion attribute on the Documents URL.
|
||||
NSError *preventBackupError = nil;
|
||||
[url setResourceValue:@YES forKey:NSURLIsExcludedFromBackupKey error:&preventBackupError];
|
||||
if (preventBackupError) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession016
|
||||
message:@"Cannot exclude temporary folder from iTunes backup"];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)URLSession:(NSURLSession *)session
|
||||
task:(NSURLSessionTask *)task
|
||||
willPerformHTTPRedirection:(NSHTTPURLResponse *)response
|
||||
newRequest:(NSURLRequest *)request
|
||||
completionHandler:(void (^)(NSURLRequest *))completionHandler {
|
||||
NSArray *nonAllowedRedirectionCodes = @[
|
||||
@(kFIRNetworkHTTPStatusCodeFound), @(kFIRNetworkHTTPStatusCodeMovedPermanently),
|
||||
@(kFIRNetworkHTTPStatusCodeMovedTemporarily), @(kFIRNetworkHTTPStatusCodeMultipleChoices)
|
||||
];
|
||||
|
||||
// Allow those not in the non allowed list to be followed.
|
||||
if (![nonAllowedRedirectionCodes containsObject:@(response.statusCode)]) {
|
||||
completionHandler(request);
|
||||
return;
|
||||
}
|
||||
|
||||
// Do not allow redirection if the response code is in the non-allowed list.
|
||||
NSURLRequest *newRequest = request;
|
||||
|
||||
if (response) {
|
||||
newRequest = nil;
|
||||
}
|
||||
|
||||
completionHandler(newRequest);
|
||||
}
|
||||
|
||||
#pragma mark - Helper Methods
|
||||
|
||||
- (void)callCompletionHandler:(FIRNetworkURLSessionCompletionHandler)handler
|
||||
withResponse:(NSHTTPURLResponse *)response
|
||||
data:(NSData *)data
|
||||
error:(NSError *)error {
|
||||
if (error) {
|
||||
[_loggerDelegate firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeURLSession017
|
||||
message:@"Encounter network error. Code, error"
|
||||
contexts:@[ @(error.code), error ]];
|
||||
}
|
||||
|
||||
if (handler) {
|
||||
dispatch_async(dispatch_get_main_queue(), ^{
|
||||
handler(response, data, self->_sessionID, error);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
- (void)populateSessionConfig:(NSURLSessionConfiguration *)sessionConfig
|
||||
withRequest:(NSURLRequest *)request {
|
||||
sessionConfig.HTTPAdditionalHeaders = request.allHTTPHeaderFields;
|
||||
sessionConfig.timeoutIntervalForRequest = request.timeoutInterval;
|
||||
sessionConfig.timeoutIntervalForResource = request.timeoutInterval;
|
||||
sessionConfig.requestCachePolicy = request.cachePolicy;
|
||||
}
|
||||
|
||||
@end
|
||||
445
Pods/FirebaseCore/Firebase/Core/FIROptions.m
generated
Normal file
445
Pods/FirebaseCore/Firebase/Core/FIROptions.m
generated
Normal file
@@ -0,0 +1,445 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "Private/FIRAppInternal.h"
|
||||
#import "Private/FIRBundleUtil.h"
|
||||
#import "Private/FIRErrors.h"
|
||||
#import "Private/FIRLogger.h"
|
||||
#import "Private/FIROptionsInternal.h"
|
||||
|
||||
// Keys for the strings in the plist file.
|
||||
NSString *const kFIRAPIKey = @"API_KEY";
|
||||
NSString *const kFIRTrackingID = @"TRACKING_ID";
|
||||
NSString *const kFIRGoogleAppID = @"GOOGLE_APP_ID";
|
||||
NSString *const kFIRClientID = @"CLIENT_ID";
|
||||
NSString *const kFIRGCMSenderID = @"GCM_SENDER_ID";
|
||||
NSString *const kFIRAndroidClientID = @"ANDROID_CLIENT_ID";
|
||||
NSString *const kFIRDatabaseURL = @"DATABASE_URL";
|
||||
NSString *const kFIRStorageBucket = @"STORAGE_BUCKET";
|
||||
// The key to locate the expected bundle identifier in the plist file.
|
||||
NSString *const kFIRBundleID = @"BUNDLE_ID";
|
||||
// The key to locate the project identifier in the plist file.
|
||||
NSString *const kFIRProjectID = @"PROJECT_ID";
|
||||
|
||||
NSString *const kFIRIsMeasurementEnabled = @"IS_MEASUREMENT_ENABLED";
|
||||
NSString *const kFIRIsAnalyticsCollectionEnabled = @"FIREBASE_ANALYTICS_COLLECTION_ENABLED";
|
||||
NSString *const kFIRIsAnalyticsCollectionDeactivated = @"FIREBASE_ANALYTICS_COLLECTION_DEACTIVATED";
|
||||
|
||||
NSString *const kFIRIsAnalyticsEnabled = @"IS_ANALYTICS_ENABLED";
|
||||
NSString *const kFIRIsSignInEnabled = @"IS_SIGNIN_ENABLED";
|
||||
|
||||
// Library version ID.
|
||||
NSString *const kFIRLibraryVersionID =
|
||||
@"5" // Major version (one or more digits)
|
||||
@"00" // Minor version (exactly 2 digits)
|
||||
@"02" // Build number (exactly 2 digits)
|
||||
@"000"; // Fixed "000"
|
||||
// Plist file name.
|
||||
NSString *const kServiceInfoFileName = @"GoogleService-Info";
|
||||
// Plist file type.
|
||||
NSString *const kServiceInfoFileType = @"plist";
|
||||
|
||||
// Exception raised from attempting to modify a FIROptions after it's been copied to a FIRApp.
|
||||
NSString *const kFIRExceptionBadModification =
|
||||
@"Attempted to modify options after it's set on FIRApp. Please modify all properties before "
|
||||
@"initializing FIRApp.";
|
||||
|
||||
@interface FIROptions ()
|
||||
|
||||
/**
|
||||
* This property maintains the actual configuration key-value pairs.
|
||||
*/
|
||||
@property(nonatomic, readwrite) NSMutableDictionary *optionsDictionary;
|
||||
|
||||
/**
|
||||
* Calls `analyticsOptionsDictionaryWithInfoDictionary:` using [NSBundle mainBundle].infoDictionary.
|
||||
* It combines analytics options from both the infoDictionary and the GoogleService-Info.plist.
|
||||
* Values which are present in the main plist override values from the GoogleService-Info.plist.
|
||||
*/
|
||||
@property(nonatomic, readonly) NSDictionary *analyticsOptionsDictionary;
|
||||
|
||||
/**
|
||||
* Combination of analytics options from both the infoDictionary and the GoogleService-Info.plist.
|
||||
* Values which are present in the infoDictionary override values from the GoogleService-Info.plist.
|
||||
*/
|
||||
- (NSDictionary *)analyticsOptionsDictionaryWithInfoDictionary:(NSDictionary *)infoDictionary;
|
||||
|
||||
/**
|
||||
* Throw exception if editing is locked when attempting to modify an option.
|
||||
*/
|
||||
- (void)checkEditingLocked;
|
||||
|
||||
@end
|
||||
|
||||
@implementation FIROptions {
|
||||
/// Backing variable for self.analyticsOptionsDictionary.
|
||||
NSDictionary *_analyticsOptionsDictionary;
|
||||
dispatch_once_t _createAnalyticsOptionsDictionaryOnce;
|
||||
}
|
||||
|
||||
static FIROptions *sDefaultOptions = nil;
|
||||
static NSDictionary *sDefaultOptionsDictionary = nil;
|
||||
|
||||
#pragma mark - Public only for internal class methods
|
||||
|
||||
+ (FIROptions *)defaultOptions {
|
||||
if (sDefaultOptions != nil) {
|
||||
return sDefaultOptions;
|
||||
}
|
||||
|
||||
NSDictionary *defaultOptionsDictionary = [self defaultOptionsDictionary];
|
||||
if (defaultOptionsDictionary == nil) {
|
||||
return nil;
|
||||
}
|
||||
|
||||
sDefaultOptions = [[FIROptions alloc] initInternalWithOptionsDictionary:defaultOptionsDictionary];
|
||||
return sDefaultOptions;
|
||||
}
|
||||
|
||||
#pragma mark - Private class methods
|
||||
|
||||
+ (void)initialize {
|
||||
// Report FirebaseCore version for useragent string
|
||||
NSRange major = NSMakeRange(0, 1);
|
||||
NSRange minor = NSMakeRange(1, 2);
|
||||
NSRange patch = NSMakeRange(3, 2);
|
||||
[FIRApp
|
||||
registerLibrary:@"fire-ios"
|
||||
withVersion:[NSString stringWithFormat:@"%@.%d.%d",
|
||||
[kFIRLibraryVersionID substringWithRange:major],
|
||||
[[kFIRLibraryVersionID substringWithRange:minor]
|
||||
intValue],
|
||||
[[kFIRLibraryVersionID substringWithRange:patch]
|
||||
intValue]]];
|
||||
NSDictionary<NSString *, id> *info = [[NSBundle mainBundle] infoDictionary];
|
||||
NSString *xcodeVersion = info[@"DTXcodeBuild"];
|
||||
NSString *sdkVersion = info[@"DTSDKBuild"];
|
||||
if (xcodeVersion) {
|
||||
[FIRApp registerLibrary:@"xcode" withVersion:xcodeVersion];
|
||||
}
|
||||
if (sdkVersion) {
|
||||
[FIRApp registerLibrary:@"apple-sdk" withVersion:sdkVersion];
|
||||
}
|
||||
}
|
||||
|
||||
+ (NSDictionary *)defaultOptionsDictionary {
|
||||
if (sDefaultOptionsDictionary != nil) {
|
||||
return sDefaultOptionsDictionary;
|
||||
}
|
||||
NSString *plistFilePath = [FIROptions plistFilePathWithName:kServiceInfoFileName];
|
||||
if (plistFilePath == nil) {
|
||||
return nil;
|
||||
}
|
||||
sDefaultOptionsDictionary = [NSDictionary dictionaryWithContentsOfFile:plistFilePath];
|
||||
if (sDefaultOptionsDictionary == nil) {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000011",
|
||||
@"The configuration file is not a dictionary: "
|
||||
@"'%@.%@'.",
|
||||
kServiceInfoFileName, kServiceInfoFileType);
|
||||
}
|
||||
return sDefaultOptionsDictionary;
|
||||
}
|
||||
|
||||
// Returns the path of the plist file with a given file name.
|
||||
+ (NSString *)plistFilePathWithName:(NSString *)fileName {
|
||||
NSArray *bundles = [FIRBundleUtil relevantBundles];
|
||||
NSString *plistFilePath =
|
||||
[FIRBundleUtil optionsDictionaryPathWithResourceName:fileName
|
||||
andFileType:kServiceInfoFileType
|
||||
inBundles:bundles];
|
||||
if (plistFilePath == nil) {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000012", @"Could not locate configuration file: '%@.%@'.",
|
||||
fileName, kServiceInfoFileType);
|
||||
}
|
||||
return plistFilePath;
|
||||
}
|
||||
|
||||
+ (void)resetDefaultOptions {
|
||||
sDefaultOptions = nil;
|
||||
sDefaultOptionsDictionary = nil;
|
||||
}
|
||||
|
||||
#pragma mark - Private instance methods
|
||||
|
||||
- (instancetype)initInternalWithOptionsDictionary:(NSDictionary *)optionsDictionary {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_optionsDictionary = [optionsDictionary mutableCopy];
|
||||
_usingOptionsFromDefaultPlist = YES;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (id)copyWithZone:(NSZone *)zone {
|
||||
FIROptions *newOptions = [[[self class] allocWithZone:zone] init];
|
||||
if (newOptions) {
|
||||
newOptions.optionsDictionary = self.optionsDictionary;
|
||||
newOptions.deepLinkURLScheme = self.deepLinkURLScheme;
|
||||
newOptions.editingLocked = self.isEditingLocked;
|
||||
newOptions.usingOptionsFromDefaultPlist = self.usingOptionsFromDefaultPlist;
|
||||
}
|
||||
return newOptions;
|
||||
}
|
||||
|
||||
#pragma mark - Public instance methods
|
||||
|
||||
- (instancetype)initWithContentsOfFile:(NSString *)plistPath {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
if (plistPath == nil) {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000013", @"The plist file path is nil.");
|
||||
return nil;
|
||||
}
|
||||
_optionsDictionary = [[NSDictionary dictionaryWithContentsOfFile:plistPath] mutableCopy];
|
||||
if (_optionsDictionary == nil) {
|
||||
FIRLogError(kFIRLoggerCore, @"I-COR000014",
|
||||
@"The configuration file at %@ does not exist or "
|
||||
@"is not a well-formed plist file.",
|
||||
plistPath);
|
||||
return nil;
|
||||
}
|
||||
// TODO: Do we want to validate the dictionary here? It says we do that already in
|
||||
// the public header.
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (instancetype)initWithGoogleAppID:(NSString *)googleAppID GCMSenderID:(NSString *)GCMSenderID {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
NSMutableDictionary *mutableOptionsDict = [NSMutableDictionary dictionary];
|
||||
[mutableOptionsDict setValue:googleAppID forKey:kFIRGoogleAppID];
|
||||
[mutableOptionsDict setValue:GCMSenderID forKey:kFIRGCMSenderID];
|
||||
[mutableOptionsDict setValue:[[NSBundle mainBundle] bundleIdentifier] forKey:kFIRBundleID];
|
||||
self.optionsDictionary = mutableOptionsDict;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (NSString *)APIKey {
|
||||
return self.optionsDictionary[kFIRAPIKey];
|
||||
}
|
||||
|
||||
- (void)checkEditingLocked {
|
||||
if (self.isEditingLocked) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain format:kFIRExceptionBadModification];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)setAPIKey:(NSString *)APIKey {
|
||||
[self checkEditingLocked];
|
||||
_optionsDictionary[kFIRAPIKey] = [APIKey copy];
|
||||
}
|
||||
|
||||
- (NSString *)clientID {
|
||||
return self.optionsDictionary[kFIRClientID];
|
||||
}
|
||||
|
||||
- (void)setClientID:(NSString *)clientID {
|
||||
[self checkEditingLocked];
|
||||
_optionsDictionary[kFIRClientID] = [clientID copy];
|
||||
}
|
||||
|
||||
- (NSString *)trackingID {
|
||||
return self.optionsDictionary[kFIRTrackingID];
|
||||
}
|
||||
|
||||
- (void)setTrackingID:(NSString *)trackingID {
|
||||
[self checkEditingLocked];
|
||||
_optionsDictionary[kFIRTrackingID] = [trackingID copy];
|
||||
}
|
||||
|
||||
- (NSString *)GCMSenderID {
|
||||
return self.optionsDictionary[kFIRGCMSenderID];
|
||||
}
|
||||
|
||||
- (void)setGCMSenderID:(NSString *)GCMSenderID {
|
||||
[self checkEditingLocked];
|
||||
_optionsDictionary[kFIRGCMSenderID] = [GCMSenderID copy];
|
||||
}
|
||||
|
||||
- (NSString *)projectID {
|
||||
return self.optionsDictionary[kFIRProjectID];
|
||||
}
|
||||
|
||||
- (void)setProjectID:(NSString *)projectID {
|
||||
[self checkEditingLocked];
|
||||
_optionsDictionary[kFIRProjectID] = [projectID copy];
|
||||
}
|
||||
|
||||
- (NSString *)androidClientID {
|
||||
return self.optionsDictionary[kFIRAndroidClientID];
|
||||
}
|
||||
|
||||
- (void)setAndroidClientID:(NSString *)androidClientID {
|
||||
[self checkEditingLocked];
|
||||
_optionsDictionary[kFIRAndroidClientID] = [androidClientID copy];
|
||||
}
|
||||
|
||||
- (NSString *)googleAppID {
|
||||
return self.optionsDictionary[kFIRGoogleAppID];
|
||||
}
|
||||
|
||||
- (void)setGoogleAppID:(NSString *)googleAppID {
|
||||
[self checkEditingLocked];
|
||||
_optionsDictionary[kFIRGoogleAppID] = [googleAppID copy];
|
||||
}
|
||||
|
||||
- (NSString *)libraryVersionID {
|
||||
return kFIRLibraryVersionID;
|
||||
}
|
||||
|
||||
- (void)setLibraryVersionID:(NSString *)libraryVersionID {
|
||||
_optionsDictionary[kFIRLibraryVersionID] = [libraryVersionID copy];
|
||||
}
|
||||
|
||||
- (NSString *)databaseURL {
|
||||
return self.optionsDictionary[kFIRDatabaseURL];
|
||||
}
|
||||
|
||||
- (void)setDatabaseURL:(NSString *)databaseURL {
|
||||
[self checkEditingLocked];
|
||||
|
||||
_optionsDictionary[kFIRDatabaseURL] = [databaseURL copy];
|
||||
}
|
||||
|
||||
- (NSString *)storageBucket {
|
||||
return self.optionsDictionary[kFIRStorageBucket];
|
||||
}
|
||||
|
||||
- (void)setStorageBucket:(NSString *)storageBucket {
|
||||
[self checkEditingLocked];
|
||||
_optionsDictionary[kFIRStorageBucket] = [storageBucket copy];
|
||||
}
|
||||
|
||||
- (void)setDeepLinkURLScheme:(NSString *)deepLinkURLScheme {
|
||||
[self checkEditingLocked];
|
||||
_deepLinkURLScheme = [deepLinkURLScheme copy];
|
||||
}
|
||||
|
||||
- (NSString *)bundleID {
|
||||
return self.optionsDictionary[kFIRBundleID];
|
||||
}
|
||||
|
||||
- (void)setBundleID:(NSString *)bundleID {
|
||||
[self checkEditingLocked];
|
||||
_optionsDictionary[kFIRBundleID] = [bundleID copy];
|
||||
}
|
||||
|
||||
#pragma mark - Internal instance methods
|
||||
|
||||
- (NSDictionary *)analyticsOptionsDictionaryWithInfoDictionary:(NSDictionary *)infoDictionary {
|
||||
dispatch_once(&_createAnalyticsOptionsDictionaryOnce, ^{
|
||||
NSMutableDictionary *tempAnalyticsOptions = [[NSMutableDictionary alloc] init];
|
||||
NSArray *measurementKeys = @[
|
||||
kFIRIsMeasurementEnabled, kFIRIsAnalyticsCollectionEnabled,
|
||||
kFIRIsAnalyticsCollectionDeactivated
|
||||
];
|
||||
for (NSString *key in measurementKeys) {
|
||||
id value = infoDictionary[key] ?: self.optionsDictionary[key] ?: nil;
|
||||
if (!value) {
|
||||
continue;
|
||||
}
|
||||
tempAnalyticsOptions[key] = value;
|
||||
}
|
||||
self->_analyticsOptionsDictionary = tempAnalyticsOptions;
|
||||
});
|
||||
return _analyticsOptionsDictionary;
|
||||
}
|
||||
|
||||
- (NSDictionary *)analyticsOptionsDictionary {
|
||||
return [self analyticsOptionsDictionaryWithInfoDictionary:[NSBundle mainBundle].infoDictionary];
|
||||
}
|
||||
|
||||
/**
|
||||
* Whether or not Measurement was enabled. Measurement is enabled unless explicitly disabled in
|
||||
* GoogleService-Info.plist. This uses the old plist flag IS_MEASUREMENT_ENABLED, which should still
|
||||
* be supported.
|
||||
*/
|
||||
- (BOOL)isMeasurementEnabled {
|
||||
if (self.isAnalyticsCollectionDeactivated) {
|
||||
return NO;
|
||||
}
|
||||
NSNumber *value = self.analyticsOptionsDictionary[kFIRIsMeasurementEnabled];
|
||||
if (value == nil) {
|
||||
// TODO: This could probably be cleaned up since FIROptions shouldn't know about FIRApp or have
|
||||
// to check if it's the default app. The FIROptions instance can't be modified after
|
||||
// `+configure` is called, so it's not a good place to copy it either in case the flag is
|
||||
// changed at runtime.
|
||||
|
||||
// If no values are set for Analytics, fall back to the global collection switch in FIRApp.
|
||||
// Analytics only supports the default FIRApp, so check that first.
|
||||
if (![FIRApp isDefaultAppConfigured]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
// Fall back to the default app's collection switch when the key is not in the dictionary.
|
||||
return [FIRApp defaultApp].automaticDataCollectionEnabled;
|
||||
}
|
||||
return [value boolValue];
|
||||
}
|
||||
|
||||
- (BOOL)isAnalyticsCollectionExpicitlySet {
|
||||
// If it's de-activated, it classifies as explicity set. If not, it's not a good enough indication
|
||||
// that the developer wants FirebaseAnalytics enabled so continue checking.
|
||||
if (self.isAnalyticsCollectionDeactivated) {
|
||||
return YES;
|
||||
}
|
||||
|
||||
// Check if the current Analytics flag is set.
|
||||
id collectionEnabledObject = self.analyticsOptionsDictionary[kFIRIsAnalyticsCollectionEnabled];
|
||||
if (collectionEnabledObject && [collectionEnabledObject isKindOfClass:[NSNumber class]]) {
|
||||
// It doesn't matter what the value is, it's explicitly set.
|
||||
return YES;
|
||||
}
|
||||
|
||||
// Check if the old measurement flag is set.
|
||||
id measurementEnabledObject = self.analyticsOptionsDictionary[kFIRIsMeasurementEnabled];
|
||||
if (measurementEnabledObject && [measurementEnabledObject isKindOfClass:[NSNumber class]]) {
|
||||
// It doesn't matter what the value is, it's explicitly set.
|
||||
return YES;
|
||||
}
|
||||
|
||||
// No flags are set to explicitly enable or disable FirebaseAnalytics.
|
||||
return NO;
|
||||
}
|
||||
|
||||
- (BOOL)isAnalyticsCollectionEnabled {
|
||||
if (self.isAnalyticsCollectionDeactivated) {
|
||||
return NO;
|
||||
}
|
||||
NSNumber *value = self.analyticsOptionsDictionary[kFIRIsAnalyticsCollectionEnabled];
|
||||
if (value == nil) {
|
||||
return self.isMeasurementEnabled; // Fall back to older plist flag.
|
||||
}
|
||||
return [value boolValue];
|
||||
}
|
||||
|
||||
- (BOOL)isAnalyticsCollectionDeactivated {
|
||||
NSNumber *value = self.analyticsOptionsDictionary[kFIRIsAnalyticsCollectionDeactivated];
|
||||
if (value == nil) {
|
||||
return NO; // Analytics Collection is not deactivated when the key is not in the dictionary.
|
||||
}
|
||||
return [value boolValue];
|
||||
}
|
||||
|
||||
- (BOOL)isAnalyticsEnabled {
|
||||
return [self.optionsDictionary[kFIRIsAnalyticsEnabled] boolValue];
|
||||
}
|
||||
|
||||
- (BOOL)isSignInEnabled {
|
||||
return [self.optionsDictionary[kFIRIsSignInEnabled] boolValue];
|
||||
}
|
||||
|
||||
@end
|
||||
256
Pods/FirebaseCore/Firebase/Core/FIRReachabilityChecker.m
generated
Normal file
256
Pods/FirebaseCore/Firebase/Core/FIRReachabilityChecker.m
generated
Normal file
@@ -0,0 +1,256 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "Private/FIRReachabilityChecker+Internal.h"
|
||||
#import "Private/FIRReachabilityChecker.h"
|
||||
|
||||
#import "Private/FIRLogger.h"
|
||||
#import "Private/FIRNetwork.h"
|
||||
#import "Private/FIRNetworkMessageCode.h"
|
||||
|
||||
static void ReachabilityCallback(SCNetworkReachabilityRef reachability,
|
||||
SCNetworkReachabilityFlags flags,
|
||||
void *info);
|
||||
|
||||
static const struct FIRReachabilityApi kFIRDefaultReachabilityApi = {
|
||||
SCNetworkReachabilityCreateWithName,
|
||||
SCNetworkReachabilitySetCallback,
|
||||
SCNetworkReachabilityScheduleWithRunLoop,
|
||||
SCNetworkReachabilityUnscheduleFromRunLoop,
|
||||
CFRelease,
|
||||
};
|
||||
|
||||
static NSString *const kFIRReachabilityUnknownStatus = @"Unknown";
|
||||
static NSString *const kFIRReachabilityConnectedStatus = @"Connected";
|
||||
static NSString *const kFIRReachabilityDisconnectedStatus = @"Disconnected";
|
||||
|
||||
@interface FIRReachabilityChecker ()
|
||||
|
||||
@property(nonatomic, assign) const struct FIRReachabilityApi *reachabilityApi;
|
||||
@property(nonatomic, assign) FIRReachabilityStatus reachabilityStatus;
|
||||
@property(nonatomic, copy) NSString *host;
|
||||
@property(nonatomic, assign) SCNetworkReachabilityRef reachability;
|
||||
|
||||
@end
|
||||
|
||||
@implementation FIRReachabilityChecker
|
||||
|
||||
@synthesize reachabilityApi = reachabilityApi_;
|
||||
@synthesize reachability = reachability_;
|
||||
|
||||
- (const struct FIRReachabilityApi *)reachabilityApi {
|
||||
return reachabilityApi_;
|
||||
}
|
||||
|
||||
- (void)setReachabilityApi:(const struct FIRReachabilityApi *)reachabilityApi {
|
||||
if (reachability_) {
|
||||
NSString *message =
|
||||
@"Cannot change reachability API while reachability is running. "
|
||||
@"Call stop first.";
|
||||
[loggerDelegate_ firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeReachabilityChecker000
|
||||
message:message];
|
||||
return;
|
||||
}
|
||||
reachabilityApi_ = reachabilityApi;
|
||||
}
|
||||
|
||||
@synthesize reachabilityStatus = reachabilityStatus_;
|
||||
@synthesize host = host_;
|
||||
@synthesize reachabilityDelegate = reachabilityDelegate_;
|
||||
@synthesize loggerDelegate = loggerDelegate_;
|
||||
|
||||
- (BOOL)isActive {
|
||||
return reachability_ != nil;
|
||||
}
|
||||
|
||||
- (void)setReachabilityDelegate:(id<FIRReachabilityDelegate>)reachabilityDelegate {
|
||||
if (reachabilityDelegate &&
|
||||
(![(NSObject *)reachabilityDelegate conformsToProtocol:@protocol(FIRReachabilityDelegate)])) {
|
||||
FIRLogError(kFIRLoggerCore,
|
||||
[NSString stringWithFormat:@"I-NET%06ld",
|
||||
(long)kFIRNetworkMessageCodeReachabilityChecker005],
|
||||
@"Reachability delegate doesn't conform to Reachability protocol.");
|
||||
return;
|
||||
}
|
||||
reachabilityDelegate_ = reachabilityDelegate;
|
||||
}
|
||||
|
||||
- (void)setLoggerDelegate:(id<FIRNetworkLoggerDelegate>)loggerDelegate {
|
||||
if (loggerDelegate &&
|
||||
(![(NSObject *)loggerDelegate conformsToProtocol:@protocol(FIRNetworkLoggerDelegate)])) {
|
||||
FIRLogError(kFIRLoggerCore,
|
||||
[NSString stringWithFormat:@"I-NET%06ld",
|
||||
(long)kFIRNetworkMessageCodeReachabilityChecker006],
|
||||
@"Reachability delegate doesn't conform to Logger protocol.");
|
||||
return;
|
||||
}
|
||||
loggerDelegate_ = loggerDelegate;
|
||||
}
|
||||
|
||||
- (instancetype)initWithReachabilityDelegate:(id<FIRReachabilityDelegate>)reachabilityDelegate
|
||||
loggerDelegate:(id<FIRNetworkLoggerDelegate>)loggerDelegate
|
||||
withHost:(NSString *)host {
|
||||
self = [super init];
|
||||
|
||||
[self setLoggerDelegate:loggerDelegate];
|
||||
|
||||
if (!host || !host.length) {
|
||||
[loggerDelegate_ firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeReachabilityChecker001
|
||||
message:@"Invalid host specified"];
|
||||
return nil;
|
||||
}
|
||||
if (self) {
|
||||
[self setReachabilityDelegate:reachabilityDelegate];
|
||||
reachabilityApi_ = &kFIRDefaultReachabilityApi;
|
||||
reachabilityStatus_ = kFIRReachabilityUnknown;
|
||||
host_ = [host copy];
|
||||
reachability_ = nil;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)dealloc {
|
||||
reachabilityDelegate_ = nil;
|
||||
loggerDelegate_ = nil;
|
||||
[self stop];
|
||||
}
|
||||
|
||||
- (BOOL)start {
|
||||
if (!reachability_) {
|
||||
reachability_ = reachabilityApi_->createWithNameFn(kCFAllocatorDefault, [host_ UTF8String]);
|
||||
if (!reachability_) {
|
||||
return NO;
|
||||
}
|
||||
SCNetworkReachabilityContext context = {
|
||||
0, /* version */
|
||||
(__bridge void *)(self), /* info (passed as last parameter to reachability callback) */
|
||||
NULL, /* retain */
|
||||
NULL, /* release */
|
||||
NULL /* copyDescription */
|
||||
};
|
||||
if (!reachabilityApi_->setCallbackFn(reachability_, ReachabilityCallback, &context) ||
|
||||
!reachabilityApi_->scheduleWithRunLoopFn(reachability_, CFRunLoopGetMain(),
|
||||
kCFRunLoopCommonModes)) {
|
||||
reachabilityApi_->releaseFn(reachability_);
|
||||
reachability_ = nil;
|
||||
[loggerDelegate_ firNetwork_logWithLevel:kFIRNetworkLogLevelError
|
||||
messageCode:kFIRNetworkMessageCodeReachabilityChecker002
|
||||
message:@"Failed to start reachability handle"];
|
||||
return NO;
|
||||
}
|
||||
}
|
||||
[loggerDelegate_ firNetwork_logWithLevel:kFIRNetworkLogLevelDebug
|
||||
messageCode:kFIRNetworkMessageCodeReachabilityChecker003
|
||||
message:@"Monitoring the network status"];
|
||||
return YES;
|
||||
}
|
||||
|
||||
- (void)stop {
|
||||
if (reachability_) {
|
||||
reachabilityStatus_ = kFIRReachabilityUnknown;
|
||||
reachabilityApi_->unscheduleFromRunLoopFn(reachability_, CFRunLoopGetMain(),
|
||||
kCFRunLoopCommonModes);
|
||||
reachabilityApi_->releaseFn(reachability_);
|
||||
reachability_ = nil;
|
||||
}
|
||||
}
|
||||
|
||||
- (FIRReachabilityStatus)statusForFlags:(SCNetworkReachabilityFlags)flags {
|
||||
FIRReachabilityStatus status = kFIRReachabilityNotReachable;
|
||||
// If the Reachable flag is not set, we definitely don't have connectivity.
|
||||
if (flags & kSCNetworkReachabilityFlagsReachable) {
|
||||
// Reachable flag is set. Check further flags.
|
||||
if (!(flags & kSCNetworkReachabilityFlagsConnectionRequired)) {
|
||||
// Connection required flag is not set, so we have connectivity.
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
status = (flags & kSCNetworkReachabilityFlagsIsWWAN) ? kFIRReachabilityViaCellular
|
||||
: kFIRReachabilityViaWifi;
|
||||
#elif TARGET_OS_OSX
|
||||
status = kFIRReachabilityViaWifi;
|
||||
#endif
|
||||
} else if ((flags & (kSCNetworkReachabilityFlagsConnectionOnDemand |
|
||||
kSCNetworkReachabilityFlagsConnectionOnTraffic)) &&
|
||||
!(flags & kSCNetworkReachabilityFlagsInterventionRequired)) {
|
||||
// If the connection on demand or connection on traffic flag is set, and user intervention
|
||||
// is not required, we have connectivity.
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
status = (flags & kSCNetworkReachabilityFlagsIsWWAN) ? kFIRReachabilityViaCellular
|
||||
: kFIRReachabilityViaWifi;
|
||||
#elif TARGET_OS_OSX
|
||||
status = kFIRReachabilityViaWifi;
|
||||
#endif
|
||||
}
|
||||
}
|
||||
return status;
|
||||
}
|
||||
|
||||
- (void)reachabilityFlagsChanged:(SCNetworkReachabilityFlags)flags {
|
||||
FIRReachabilityStatus status = [self statusForFlags:flags];
|
||||
if (reachabilityStatus_ != status) {
|
||||
NSString *reachabilityStatusString;
|
||||
if (status == kFIRReachabilityUnknown) {
|
||||
reachabilityStatusString = kFIRReachabilityUnknownStatus;
|
||||
} else {
|
||||
reachabilityStatusString = (status == kFIRReachabilityNotReachable)
|
||||
? kFIRReachabilityDisconnectedStatus
|
||||
: kFIRReachabilityConnectedStatus;
|
||||
}
|
||||
[loggerDelegate_ firNetwork_logWithLevel:kFIRNetworkLogLevelDebug
|
||||
messageCode:kFIRNetworkMessageCodeReachabilityChecker004
|
||||
message:@"Network status has changed. Code, status"
|
||||
contexts:@[ @(status), reachabilityStatusString ]];
|
||||
reachabilityStatus_ = status;
|
||||
[reachabilityDelegate_ reachability:self statusChanged:reachabilityStatus_];
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
static void ReachabilityCallback(SCNetworkReachabilityRef reachability,
|
||||
SCNetworkReachabilityFlags flags,
|
||||
void *info) {
|
||||
FIRReachabilityChecker *checker = (__bridge FIRReachabilityChecker *)info;
|
||||
[checker reachabilityFlagsChanged:flags];
|
||||
}
|
||||
|
||||
// This function used to be at the top of the file, but it was moved here
|
||||
// as a workaround for a suspected compiler bug. When compiled in Release mode
|
||||
// and run on an iOS device with WiFi disabled, the reachability code crashed
|
||||
// when calling SCNetworkReachabilityScheduleWithRunLoop, or shortly thereafter.
|
||||
// After unsuccessfully trying to diagnose the cause of the crash, it was
|
||||
// discovered that moving this function to the end of the file magically fixed
|
||||
// the crash. If you are going to edit this file, exercise caution and make sure
|
||||
// to test thoroughly with an iOS device under various network conditions.
|
||||
const NSString *FIRReachabilityStatusString(FIRReachabilityStatus status) {
|
||||
switch (status) {
|
||||
case kFIRReachabilityUnknown:
|
||||
return @"Reachability Unknown";
|
||||
|
||||
case kFIRReachabilityNotReachable:
|
||||
return @"Not reachable";
|
||||
|
||||
case kFIRReachabilityViaWifi:
|
||||
return @"Reachable via Wifi";
|
||||
|
||||
case kFIRReachabilityViaCellular:
|
||||
return @"Reachable via Cellular Data";
|
||||
|
||||
default:
|
||||
return [NSString stringWithFormat:@"Invalid reachability status %d", (int)status];
|
||||
}
|
||||
}
|
||||
33
Pods/FirebaseCore/Firebase/Core/FIRVersion.m
generated
Normal file
33
Pods/FirebaseCore/Firebase/Core/FIRVersion.m
generated
Normal file
@@ -0,0 +1,33 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#ifndef Firebase_VERSION
|
||||
#error "Firebase_VERSION is not defined: add -DFirebase_VERSION=... to the build invocation"
|
||||
#endif
|
||||
|
||||
#ifndef FIRCore_VERSION
|
||||
#error "FIRCore_VERSION is not defined: add -DFIRCore_VERSION=... to the build invocation"
|
||||
#endif
|
||||
|
||||
// The following two macros supply the incantation so that the C
|
||||
// preprocessor does not try to parse the version as a floating
|
||||
// point number. See
|
||||
// https://www.guyrutenberg.com/2008/12/20/expanding-macros-into-string-constants-in-c/
|
||||
#define STR(x) STR_EXPAND(x)
|
||||
#define STR_EXPAND(x) #x
|
||||
|
||||
const unsigned char *const FIRVersionString = (const unsigned char *const)STR(Firebase_VERSION);
|
||||
const unsigned char *const FIRCoreVersionString = (const unsigned char *const)STR(FIRCore_VERSION);
|
||||
49
Pods/FirebaseCore/Firebase/Core/Private/FIRAnalyticsConfiguration+Internal.h
generated
Normal file
49
Pods/FirebaseCore/Firebase/Core/Private/FIRAnalyticsConfiguration+Internal.h
generated
Normal file
@@ -0,0 +1,49 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import "FIRAnalyticsConfiguration.h"
|
||||
|
||||
/// Values stored in analyticsEnabledState. Never alter these constants since they must match with
|
||||
/// values persisted to disk.
|
||||
typedef NS_ENUM(int64_t, FIRAnalyticsEnabledState) {
|
||||
// 0 is the default value for keys not found stored in persisted config, so it cannot represent
|
||||
// kFIRAnalyticsEnabledStateSetNo. It must represent kFIRAnalyticsEnabledStateNotSet.
|
||||
kFIRAnalyticsEnabledStateNotSet = 0,
|
||||
kFIRAnalyticsEnabledStateSetYes = 1,
|
||||
kFIRAnalyticsEnabledStateSetNo = 2,
|
||||
};
|
||||
|
||||
/// The user defaults key for the persisted measurementEnabledState value. FIRAPersistedConfig reads
|
||||
/// measurementEnabledState using this same key.
|
||||
static NSString *const kFIRAPersistedConfigMeasurementEnabledStateKey =
|
||||
@"/google/measurement/measurement_enabled_state";
|
||||
|
||||
static NSString *const kFIRAnalyticsConfigurationSetEnabledNotification =
|
||||
@"FIRAnalyticsConfigurationSetEnabledNotification";
|
||||
static NSString *const kFIRAnalyticsConfigurationSetMinimumSessionIntervalNotification =
|
||||
@"FIRAnalyticsConfigurationSetMinimumSessionIntervalNotification";
|
||||
static NSString *const kFIRAnalyticsConfigurationSetSessionTimeoutIntervalNotification =
|
||||
@"FIRAnalyticsConfigurationSetSessionTimeoutIntervalNotification";
|
||||
|
||||
@interface FIRAnalyticsConfiguration (Internal)
|
||||
|
||||
/// Sets whether analytics collection is enabled for this app on this device, and a flag to persist
|
||||
/// the value or not. The setting should not be persisted if being set by the global data collection
|
||||
/// flag.
|
||||
- (void)setAnalyticsCollectionEnabled:(BOOL)analyticsCollectionEnabled
|
||||
persistSetting:(BOOL)shouldPersist;
|
||||
|
||||
@end
|
||||
48
Pods/FirebaseCore/Firebase/Core/Private/FIRAppAssociationRegistration.h
generated
Normal file
48
Pods/FirebaseCore/Firebase/Core/Private/FIRAppAssociationRegistration.h
generated
Normal file
@@ -0,0 +1,48 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
/** @class FIRAppAssociationRegistration
|
||||
@brief Manages object associations as a singleton-dependent: At most one object is
|
||||
registered for any given host/key pair, and the object shall be created on-the-fly when
|
||||
asked for.
|
||||
*/
|
||||
@interface FIRAppAssociationRegistration <ObjectType> : NSObject
|
||||
|
||||
/** @fn registeredObjectWithHost:key:creationBlock:
|
||||
@brief Retrieves the registered object with a particular host and key.
|
||||
@param host The host object.
|
||||
@param key The key to specify the registered object on the host.
|
||||
@param creationBlock The block to return the object to be registered if not already.
|
||||
The block is executed immediately before this method returns if it is executed at all.
|
||||
It can also be executed multiple times across different method invocations if previous
|
||||
execution of the block returns @c nil.
|
||||
@return The registered object for the host/key pair, or @c nil if no object is registered
|
||||
and @c creationBlock returns @c nil.
|
||||
@remarks The method is thread-safe but non-reentrant in the sense that attempting to call this
|
||||
method again within the @c creationBlock with the same host/key pair raises an exception.
|
||||
The registered object is retained by the host.
|
||||
*/
|
||||
+ (nullable ObjectType)registeredObjectWithHost:(id)host
|
||||
key:(NSString *)key
|
||||
creationBlock:(ObjectType _Nullable (^)(void))creationBlock;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
214
Pods/FirebaseCore/Firebase/Core/Private/FIRAppInternal.h
generated
Normal file
214
Pods/FirebaseCore/Firebase/Core/Private/FIRAppInternal.h
generated
Normal file
@@ -0,0 +1,214 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import "FIRApp.h"
|
||||
#import "FIRErrors.h"
|
||||
|
||||
/**
|
||||
* The internal interface to FIRApp. This is meant for first-party integrators, who need to receive
|
||||
* FIRApp notifications, log info about the success or failure of their configuration, and access
|
||||
* other internal functionality of FIRApp.
|
||||
*
|
||||
* TODO(b/28296561): Restructure this header.
|
||||
*/
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
typedef NS_ENUM(NSInteger, FIRConfigType) {
|
||||
FIRConfigTypeCore = 1,
|
||||
FIRConfigTypeSDK = 2,
|
||||
};
|
||||
|
||||
/**
|
||||
* Names of services provided by Firebase.
|
||||
*/
|
||||
extern NSString *const kFIRServiceAdMob;
|
||||
extern NSString *const kFIRServiceAuth;
|
||||
extern NSString *const kFIRServiceAuthUI;
|
||||
extern NSString *const kFIRServiceCrash;
|
||||
extern NSString *const kFIRServiceDatabase;
|
||||
extern NSString *const kFIRServiceDynamicLinks;
|
||||
extern NSString *const kFIRServiceInstanceID;
|
||||
extern NSString *const kFIRServiceInvites;
|
||||
extern NSString *const kFIRServiceMessaging;
|
||||
extern NSString *const kFIRServiceMeasurement;
|
||||
extern NSString *const kFIRServiceRemoteConfig;
|
||||
extern NSString *const kFIRServiceStorage;
|
||||
|
||||
/**
|
||||
* Names of services provided by the Google pod, but logged by the Firebase pod.
|
||||
*/
|
||||
extern NSString *const kGGLServiceAnalytics;
|
||||
extern NSString *const kGGLServiceSignIn;
|
||||
|
||||
extern NSString *const kFIRDefaultAppName;
|
||||
extern NSString *const kFIRAppReadyToConfigureSDKNotification;
|
||||
extern NSString *const kFIRAppDeleteNotification;
|
||||
extern NSString *const kFIRAppIsDefaultAppKey;
|
||||
extern NSString *const kFIRAppNameKey;
|
||||
extern NSString *const kFIRGoogleAppIDKey;
|
||||
|
||||
/**
|
||||
* The format string for the User Defaults key used for storing the data collection enabled flag.
|
||||
* This includes formatting to append the Firebase App's name.
|
||||
*/
|
||||
extern NSString *const kFIRGlobalAppDataCollectionEnabledDefaultsKeyFormat;
|
||||
|
||||
/**
|
||||
* The plist key used for storing the data collection enabled flag.
|
||||
*/
|
||||
extern NSString *const kFIRGlobalAppDataCollectionEnabledPlistKey;
|
||||
|
||||
/**
|
||||
* A notification fired containing diagnostic information when SDK errors occur.
|
||||
*/
|
||||
extern NSString *const kFIRAppDiagnosticsNotification;
|
||||
|
||||
/** @var FIRAuthStateDidChangeInternalNotification
|
||||
@brief The name of the @c NSNotificationCenter notification which is posted when the auth state
|
||||
changes (e.g. a new token has been produced, a user logs in or out). The object parameter of
|
||||
the notification is a dictionary possibly containing the key:
|
||||
@c FIRAuthStateDidChangeInternalNotificationTokenKey (the new access token.) If it does not
|
||||
contain this key it indicates a sign-out event took place.
|
||||
*/
|
||||
extern NSString *const FIRAuthStateDidChangeInternalNotification;
|
||||
|
||||
/** @var FIRAuthStateDidChangeInternalNotificationTokenKey
|
||||
@brief A key present in the dictionary object parameter of the
|
||||
@c FIRAuthStateDidChangeInternalNotification notification. The value associated with this
|
||||
key will contain the new access token.
|
||||
*/
|
||||
extern NSString *const FIRAuthStateDidChangeInternalNotificationTokenKey;
|
||||
|
||||
/** @var FIRAuthStateDidChangeInternalNotificationAppKey
|
||||
@brief A key present in the dictionary object parameter of the
|
||||
@c FIRAuthStateDidChangeInternalNotification notification. The value associated with this
|
||||
key will contain the FIRApp associated with the auth instance.
|
||||
*/
|
||||
extern NSString *const FIRAuthStateDidChangeInternalNotificationAppKey;
|
||||
|
||||
/** @var FIRAuthStateDidChangeInternalNotificationUIDKey
|
||||
@brief A key present in the dictionary object parameter of the
|
||||
@c FIRAuthStateDidChangeInternalNotification notification. The value associated with this
|
||||
key will contain the new user's UID (or nil if there is no longer a user signed in).
|
||||
*/
|
||||
extern NSString *const FIRAuthStateDidChangeInternalNotificationUIDKey;
|
||||
|
||||
/** @typedef FIRTokenCallback
|
||||
@brief The type of block which gets called when a token is ready.
|
||||
*/
|
||||
typedef void (^FIRTokenCallback)(NSString *_Nullable token, NSError *_Nullable error);
|
||||
|
||||
/** @typedef FIRAppGetTokenImplementation
|
||||
@brief The type of block which can provide an implementation for the @c getTokenWithCallback:
|
||||
method.
|
||||
@param forceRefresh Forces the token to be refreshed.
|
||||
@param callback The block which should be invoked when the async call completes.
|
||||
*/
|
||||
typedef void (^FIRAppGetTokenImplementation)(BOOL forceRefresh, FIRTokenCallback callback);
|
||||
|
||||
/** @typedef FIRAppGetUID
|
||||
@brief The type of block which can provide an implementation for the @c getUID method.
|
||||
*/
|
||||
typedef NSString *_Nullable (^FIRAppGetUIDImplementation)(void);
|
||||
|
||||
@interface FIRApp ()
|
||||
|
||||
/** @property getTokenImplementation
|
||||
@brief Gets or sets the block to use for the implementation of
|
||||
@c getTokenForcingRefresh:withCallback:
|
||||
*/
|
||||
@property(nonatomic, copy) FIRAppGetTokenImplementation getTokenImplementation;
|
||||
|
||||
/** @property getUIDImplementation
|
||||
@brief Gets or sets the block to use for the implementation of @c getUID.
|
||||
*/
|
||||
@property(nonatomic, copy) FIRAppGetUIDImplementation getUIDImplementation;
|
||||
|
||||
/**
|
||||
* Creates an error for failing to configure a subspec service. This method is called by each
|
||||
* FIRApp notification listener.
|
||||
*/
|
||||
+ (NSError *)errorForSubspecConfigurationFailureWithDomain:(NSString *)domain
|
||||
errorCode:(FIRErrorCode)code
|
||||
service:(NSString *)service
|
||||
reason:(NSString *)reason;
|
||||
/**
|
||||
* Checks if the default app is configured without trying to configure it.
|
||||
*/
|
||||
+ (BOOL)isDefaultAppConfigured;
|
||||
|
||||
/**
|
||||
* Registers a given third-party library with the given version number to be reported for
|
||||
* analyitcs.
|
||||
*
|
||||
* @param library Name of the library
|
||||
* @param version Version of the library
|
||||
*/
|
||||
// clang-format off
|
||||
+ (void)registerLibrary:(NSString *)library
|
||||
withVersion:(NSString *)version NS_SWIFT_NAME(registerLibrary(_:version:));
|
||||
// clang-format on
|
||||
|
||||
/**
|
||||
* A concatenated string representing all the third-party libraries and version numbers.
|
||||
*/
|
||||
+ (NSString *)firebaseUserAgent;
|
||||
|
||||
/**
|
||||
* Used by each SDK to send logs about SDK configuration status to Clearcut.
|
||||
*/
|
||||
- (void)sendLogsWithServiceName:(NSString *)serviceName
|
||||
version:(NSString *)version
|
||||
error:(NSError *)error;
|
||||
|
||||
/**
|
||||
* Can be used by the unit tests in eack SDK to reset FIRApp. This method is thread unsafe.
|
||||
*/
|
||||
+ (void)resetApps;
|
||||
|
||||
/**
|
||||
* Can be used by the unit tests in each SDK to set customized options.
|
||||
*/
|
||||
- (instancetype)initInstanceWithName:(NSString *)name options:(FIROptions *)options;
|
||||
|
||||
/** @fn getTokenForcingRefresh:withCallback:
|
||||
@brief Retrieves the Firebase authentication token, possibly refreshing it.
|
||||
@param forceRefresh Forces a token refresh. Useful if the token becomes invalid for some reason
|
||||
other than an expiration.
|
||||
@param callback The block to invoke when the token is available.
|
||||
*/
|
||||
- (void)getTokenForcingRefresh:(BOOL)forceRefresh withCallback:(FIRTokenCallback)callback;
|
||||
|
||||
/**
|
||||
* Expose the UID of the current user for Firestore.
|
||||
*/
|
||||
- (nullable NSString *)getUID;
|
||||
|
||||
/**
|
||||
* WARNING: THIS SETTING DOES NOT WORK YET. IT WILL BE MOVED TO THE PUBLIC HEADER ONCE ALL SDKS
|
||||
* CONFORM TO THIS PREFERENCE. DO NOT RELY ON IT.
|
||||
*
|
||||
* Gets or sets whether automatic data collection is enabled for all products. Defaults to `YES`
|
||||
* unless `FirebaseAutomaticDataCollectionEnabled` is set to `NO` in your app's Info.plist. This
|
||||
* value is persisted across runs of the app so that it can be set once when users have consented to
|
||||
* collection.
|
||||
*/
|
||||
@property(nonatomic, readwrite, getter=isAutomaticDataCollectionEnabled)
|
||||
BOOL automaticDataCollectionEnabled;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
52
Pods/FirebaseCore/Firebase/Core/Private/FIRBundleUtil.h
generated
Normal file
52
Pods/FirebaseCore/Firebase/Core/Private/FIRBundleUtil.h
generated
Normal file
@@ -0,0 +1,52 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
/**
|
||||
* This class provides utilities for accessing resources in bundles.
|
||||
*/
|
||||
@interface FIRBundleUtil : NSObject
|
||||
|
||||
/**
|
||||
* Finds all relevant bundles, starting with [NSBundle mainBundle].
|
||||
*/
|
||||
+ (NSArray *)relevantBundles;
|
||||
|
||||
/**
|
||||
* Reads the options dictionary from one of the provided bundles.
|
||||
*
|
||||
* @param resourceName The resource name, e.g. @"GoogleService-Info".
|
||||
* @param fileType The file type (extension), e.g. @"plist".
|
||||
* @param bundles The bundles to expect, in priority order. See also
|
||||
* +[FIRBundleUtil relevantBundles].
|
||||
*/
|
||||
+ (NSString *)optionsDictionaryPathWithResourceName:(NSString *)resourceName
|
||||
andFileType:(NSString *)fileType
|
||||
inBundles:(NSArray *)bundles;
|
||||
|
||||
/**
|
||||
* Finds URL schemes defined in all relevant bundles, starting with those from
|
||||
* [NSBundle mainBundle].
|
||||
*/
|
||||
+ (NSArray *)relevantURLSchemes;
|
||||
|
||||
/**
|
||||
* Checks if the bundle identifier exists in the given bundles.
|
||||
*/
|
||||
+ (BOOL)hasBundleIdentifier:(NSString *)bundleIdentifier inBundles:(NSArray *)bundles;
|
||||
|
||||
@end
|
||||
55
Pods/FirebaseCore/Firebase/Core/Private/FIRErrorCode.h
generated
Normal file
55
Pods/FirebaseCore/Firebase/Core/Private/FIRErrorCode.h
generated
Normal file
@@ -0,0 +1,55 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
/** Error codes in Firebase error domain. */
|
||||
typedef NS_ENUM(NSInteger, FIRErrorCode) {
|
||||
/**
|
||||
* Unknown error.
|
||||
*/
|
||||
FIRErrorCodeUnknown = 0,
|
||||
/**
|
||||
* Loading data from the GoogleService-Info.plist file failed. This is a fatal error and should
|
||||
* not be ignored. Further calls to the API will fail and/or possibly cause crashes.
|
||||
*/
|
||||
FIRErrorCodeInvalidPlistFile = -100,
|
||||
|
||||
/**
|
||||
* Validating the Google App ID format failed.
|
||||
*/
|
||||
FIRErrorCodeInvalidAppID = -101,
|
||||
|
||||
/**
|
||||
* Error code for failing to configure a specific service.
|
||||
*/
|
||||
FIRErrorCodeAdMobFailed = -110,
|
||||
FIRErrorCodeAppInviteFailed = -112,
|
||||
FIRErrorCodeCloudMessagingFailed = -113,
|
||||
FIRErrorCodeConfigFailed = -114,
|
||||
FIRErrorCodeDatabaseFailed = -115,
|
||||
FIRErrorCodeCrashReportingFailed = -118,
|
||||
FIRErrorCodeDurableDeepLinkFailed = -119,
|
||||
FIRErrorCodeAuthFailed = -120,
|
||||
FIRErrorCodeInstanceIDFailed = -121,
|
||||
FIRErrorCodeStorageFailed = -123,
|
||||
|
||||
/**
|
||||
* Error codes returned by Dynamic Links
|
||||
*/
|
||||
FIRErrorCodeDynamicLinksStrongMatchNotAvailable = -124,
|
||||
FIRErrorCodeDynamicLinksManualRetrievalNotEnabled = -125,
|
||||
FIRErrorCodeDynamicLinksPendingLinkOnlyAvailableAtFirstLaunch = -126,
|
||||
FIRErrorCodeDynamicLinksPendingLinkRetrievalAlreadyRunning = -127,
|
||||
};
|
||||
33
Pods/FirebaseCore/Firebase/Core/Private/FIRErrors.h
generated
Normal file
33
Pods/FirebaseCore/Firebase/Core/Private/FIRErrors.h
generated
Normal file
@@ -0,0 +1,33 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#include "FIRErrorCode.h"
|
||||
|
||||
extern NSString *const kFirebaseErrorDomain;
|
||||
extern NSString *const kFirebaseAdMobErrorDomain;
|
||||
extern NSString *const kFirebaseAppInviteErrorDomain;
|
||||
extern NSString *const kFirebaseAuthErrorDomain;
|
||||
extern NSString *const kFirebaseCloudMessagingErrorDomain;
|
||||
extern NSString *const kFirebaseConfigErrorDomain;
|
||||
extern NSString *const kFirebaseCoreErrorDomain;
|
||||
extern NSString *const kFirebaseCrashReportingErrorDomain;
|
||||
extern NSString *const kFirebaseDatabaseErrorDomain;
|
||||
extern NSString *const kFirebaseDurableDeepLinkErrorDomain;
|
||||
extern NSString *const kFirebaseInstanceIDErrorDomain;
|
||||
extern NSString *const kFirebasePerfErrorDomain;
|
||||
extern NSString *const kFirebaseStorageErrorDomain;
|
||||
159
Pods/FirebaseCore/Firebase/Core/Private/FIRLogger.h
generated
Normal file
159
Pods/FirebaseCore/Firebase/Core/Private/FIRLogger.h
generated
Normal file
@@ -0,0 +1,159 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRLoggerLevel.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
/**
|
||||
* The Firebase services used in Firebase logger.
|
||||
*/
|
||||
typedef NSString *const FIRLoggerService;
|
||||
|
||||
extern FIRLoggerService kFIRLoggerABTesting;
|
||||
extern FIRLoggerService kFIRLoggerAdMob;
|
||||
extern FIRLoggerService kFIRLoggerAnalytics;
|
||||
extern FIRLoggerService kFIRLoggerAuth;
|
||||
extern FIRLoggerService kFIRLoggerCore;
|
||||
extern FIRLoggerService kFIRLoggerCrash;
|
||||
extern FIRLoggerService kFIRLoggerDatabase;
|
||||
extern FIRLoggerService kFIRLoggerDynamicLinks;
|
||||
extern FIRLoggerService kFIRLoggerFirestore;
|
||||
extern FIRLoggerService kFIRLoggerInstanceID;
|
||||
extern FIRLoggerService kFIRLoggerInvites;
|
||||
extern FIRLoggerService kFIRLoggerMLKit;
|
||||
extern FIRLoggerService kFIRLoggerMessaging;
|
||||
extern FIRLoggerService kFIRLoggerPerf;
|
||||
extern FIRLoggerService kFIRLoggerRemoteConfig;
|
||||
extern FIRLoggerService kFIRLoggerStorage;
|
||||
extern FIRLoggerService kFIRLoggerSwizzler;
|
||||
|
||||
/**
|
||||
* The key used to store the logger's error count.
|
||||
*/
|
||||
extern NSString *const kFIRLoggerErrorCountKey;
|
||||
|
||||
/**
|
||||
* The key used to store the logger's warning count.
|
||||
*/
|
||||
extern NSString *const kFIRLoggerWarningCountKey;
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif // __cplusplus
|
||||
|
||||
/**
|
||||
* Enables or disables Analytics debug mode.
|
||||
* If set to YES, the logging level for Analytics will be set to FIRLoggerLevelDebug.
|
||||
* Enabling the debug mode has no effect if the app is running from App Store.
|
||||
* (required) analytics debug mode flag.
|
||||
*/
|
||||
void FIRSetAnalyticsDebugMode(BOOL analyticsDebugMode);
|
||||
|
||||
/**
|
||||
* Changes the default logging level of FIRLoggerLevelNotice to a user-specified level.
|
||||
* The default level cannot be set above FIRLoggerLevelNotice if the app is running from App Store.
|
||||
* (required) log level (one of the FIRLoggerLevel enum values).
|
||||
*/
|
||||
void FIRSetLoggerLevel(FIRLoggerLevel loggerLevel);
|
||||
|
||||
/**
|
||||
* Checks if the specified logger level is loggable given the current settings.
|
||||
* (required) log level (one of the FIRLoggerLevel enum values).
|
||||
* (required) whether or not this function is called from the Analytics component.
|
||||
*/
|
||||
BOOL FIRIsLoggableLevel(FIRLoggerLevel loggerLevel, BOOL analyticsComponent);
|
||||
|
||||
/**
|
||||
* Logs a message to the Xcode console and the device log. If running from AppStore, will
|
||||
* not log any messages with a level higher than FIRLoggerLevelNotice to avoid log spamming.
|
||||
* (required) log level (one of the FIRLoggerLevel enum values).
|
||||
* (required) service name of type FIRLoggerService.
|
||||
* (required) message code starting with "I-" which means iOS, followed by a capitalized
|
||||
* three-character service identifier and a six digit integer message ID that is unique
|
||||
* within the service.
|
||||
* An example of the message code is @"I-COR000001".
|
||||
* (required) message string which can be a format string.
|
||||
* (optional) variable arguments list obtained from calling va_start, used when message is a format
|
||||
* string.
|
||||
*/
|
||||
extern void FIRLogBasic(FIRLoggerLevel level,
|
||||
FIRLoggerService service,
|
||||
NSString *messageCode,
|
||||
NSString *message,
|
||||
// On 64-bit simulators, va_list is not a pointer, so cannot be marked nullable
|
||||
// See: http://stackoverflow.com/q/29095469
|
||||
#if __LP64__ && TARGET_OS_SIMULATOR || TARGET_OS_OSX
|
||||
va_list args_ptr
|
||||
#else
|
||||
va_list _Nullable args_ptr
|
||||
#endif
|
||||
);
|
||||
|
||||
/**
|
||||
* The following functions accept the following parameters in order:
|
||||
* (required) service name of type FIRLoggerService.
|
||||
* (required) message code starting from "I-" which means iOS, followed by a capitalized
|
||||
* three-character service identifier and a six digit integer message ID that is unique
|
||||
* within the service.
|
||||
* An example of the message code is @"I-COR000001".
|
||||
* See go/firebase-log-proposal for details.
|
||||
* (required) message string which can be a format string.
|
||||
* (optional) the list of arguments to substitute into the format string.
|
||||
* Example usage:
|
||||
* FIRLogError(kFIRLoggerCore, @"I-COR000001", @"Configuration of %@ failed.", app.name);
|
||||
*/
|
||||
extern void FIRLogError(FIRLoggerService service, NSString *messageCode, NSString *message, ...)
|
||||
NS_FORMAT_FUNCTION(3, 4);
|
||||
extern void FIRLogWarning(FIRLoggerService service, NSString *messageCode, NSString *message, ...)
|
||||
NS_FORMAT_FUNCTION(3, 4);
|
||||
extern void FIRLogNotice(FIRLoggerService service, NSString *messageCode, NSString *message, ...)
|
||||
NS_FORMAT_FUNCTION(3, 4);
|
||||
extern void FIRLogInfo(FIRLoggerService service, NSString *messageCode, NSString *message, ...)
|
||||
NS_FORMAT_FUNCTION(3, 4);
|
||||
extern void FIRLogDebug(FIRLoggerService service, NSString *messageCode, NSString *message, ...)
|
||||
NS_FORMAT_FUNCTION(3, 4);
|
||||
|
||||
#ifdef __cplusplus
|
||||
} // extern "C"
|
||||
#endif // __cplusplus
|
||||
|
||||
@interface FIRLoggerWrapper : NSObject
|
||||
|
||||
/**
|
||||
* Objective-C wrapper for FIRLogBasic to allow weak linking to FIRLogger
|
||||
* (required) log level (one of the FIRLoggerLevel enum values).
|
||||
* (required) service name of type FIRLoggerService.
|
||||
* (required) message code starting with "I-" which means iOS, followed by a capitalized
|
||||
* three-character service identifier and a six digit integer message ID that is unique
|
||||
* within the service.
|
||||
* An example of the message code is @"I-COR000001".
|
||||
* (required) message string which can be a format string.
|
||||
* (optional) variable arguments list obtained from calling va_start, used when message is a format
|
||||
* string.
|
||||
*/
|
||||
|
||||
+ (void)logWithLevel:(FIRLoggerLevel)level
|
||||
withService:(FIRLoggerService)service
|
||||
withCode:(NSString *)messageCode
|
||||
withMessage:(NSString *)message
|
||||
withArgs:(va_list)args;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
46
Pods/FirebaseCore/Firebase/Core/Private/FIRMutableDictionary.h
generated
Normal file
46
Pods/FirebaseCore/Firebase/Core/Private/FIRMutableDictionary.h
generated
Normal file
@@ -0,0 +1,46 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
/// A mutable dictionary that provides atomic accessor and mutators.
|
||||
@interface FIRMutableDictionary : NSObject
|
||||
|
||||
/// Returns an object given a key in the dictionary or nil if not found.
|
||||
- (id)objectForKey:(id)key;
|
||||
|
||||
/// Updates the object given its key or adds it to the dictionary if it is not in the dictionary.
|
||||
- (void)setObject:(id)object forKey:(id<NSCopying>)key;
|
||||
|
||||
/// Removes the object given its session ID from the dictionary.
|
||||
- (void)removeObjectForKey:(id)key;
|
||||
|
||||
/// Removes all objects.
|
||||
- (void)removeAllObjects;
|
||||
|
||||
/// Returns the number of current objects in the dictionary.
|
||||
- (NSUInteger)count;
|
||||
|
||||
/// Returns an object given a key in the dictionary or nil if not found.
|
||||
- (id)objectForKeyedSubscript:(id<NSCopying>)key;
|
||||
|
||||
/// Updates the object given its key or adds it to the dictionary if it is not in the dictionary.
|
||||
- (void)setObject:(id)obj forKeyedSubscript:(id<NSCopying>)key;
|
||||
|
||||
/// Returns the immutable dictionary.
|
||||
- (NSDictionary *)dictionary;
|
||||
|
||||
@end
|
||||
87
Pods/FirebaseCore/Firebase/Core/Private/FIRNetwork.h
generated
Normal file
87
Pods/FirebaseCore/Firebase/Core/Private/FIRNetwork.h
generated
Normal file
@@ -0,0 +1,87 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRNetworkConstants.h"
|
||||
#import "FIRNetworkLoggerProtocol.h"
|
||||
#import "FIRNetworkURLSession.h"
|
||||
|
||||
/// Delegate protocol for FIRNetwork events.
|
||||
@protocol FIRNetworkReachabilityDelegate
|
||||
|
||||
/// Tells the delegate to handle events when the network reachability changes to connected or not
|
||||
/// connected.
|
||||
- (void)reachabilityDidChange;
|
||||
|
||||
@end
|
||||
|
||||
/// The Network component that provides network status and handles network requests and responses.
|
||||
/// This is not thread safe.
|
||||
///
|
||||
/// NOTE:
|
||||
/// User must add FIRAnalytics handleEventsForBackgroundURLSessionID:completionHandler to the
|
||||
/// AppDelegate application:handleEventsForBackgroundURLSession:completionHandler:
|
||||
@interface FIRNetwork : NSObject
|
||||
|
||||
/// Indicates if network connectivity is available.
|
||||
@property(nonatomic, readonly, getter=isNetworkConnected) BOOL networkConnected;
|
||||
|
||||
/// Indicates if there are any uploads in progress.
|
||||
@property(nonatomic, readonly, getter=hasUploadInProgress) BOOL uploadInProgress;
|
||||
|
||||
/// An optional delegate that can be used in the event when network reachability changes.
|
||||
@property(nonatomic, weak) id<FIRNetworkReachabilityDelegate> reachabilityDelegate;
|
||||
|
||||
/// An optional delegate that can be used to log messages, warnings or errors that occur in the
|
||||
/// network operations.
|
||||
@property(nonatomic, weak) id<FIRNetworkLoggerDelegate> loggerDelegate;
|
||||
|
||||
/// Indicates whether the logger should display debug messages.
|
||||
@property(nonatomic, assign) BOOL isDebugModeEnabled;
|
||||
|
||||
/// The time interval in seconds for the network request to timeout.
|
||||
@property(nonatomic, assign) NSTimeInterval timeoutInterval;
|
||||
|
||||
/// Initializes with the default reachability host.
|
||||
- (instancetype)init;
|
||||
|
||||
/// Initializes with a custom reachability host.
|
||||
- (instancetype)initWithReachabilityHost:(NSString *)reachabilityHost;
|
||||
|
||||
/// Handles events when background session with the given ID has finished.
|
||||
+ (void)handleEventsForBackgroundURLSessionID:(NSString *)sessionID
|
||||
completionHandler:(FIRNetworkSystemCompletionHandler)completionHandler;
|
||||
|
||||
/// Compresses and sends a POST request with the provided data to the URL. The session will be
|
||||
/// background session if usingBackgroundSession is YES. Otherwise, the POST session is default
|
||||
/// session. Returns a session ID or nil if an error occurs.
|
||||
- (NSString *)postURL:(NSURL *)url
|
||||
payload:(NSData *)payload
|
||||
queue:(dispatch_queue_t)queue
|
||||
usingBackgroundSession:(BOOL)usingBackgroundSession
|
||||
completionHandler:(FIRNetworkCompletionHandler)handler;
|
||||
|
||||
/// Sends a GET request with the provided data to the URL. The session will be background session
|
||||
/// if usingBackgroundSession is YES. Otherwise, the GET session is default session. Returns a
|
||||
/// session ID or nil if an error occurs.
|
||||
- (NSString *)getURL:(NSURL *)url
|
||||
headers:(NSDictionary *)headers
|
||||
queue:(dispatch_queue_t)queue
|
||||
usingBackgroundSession:(BOOL)usingBackgroundSession
|
||||
completionHandler:(FIRNetworkCompletionHandler)handler;
|
||||
|
||||
@end
|
||||
75
Pods/FirebaseCore/Firebase/Core/Private/FIRNetworkConstants.h
generated
Normal file
75
Pods/FirebaseCore/Firebase/Core/Private/FIRNetworkConstants.h
generated
Normal file
@@ -0,0 +1,75 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
/// Error codes in Firebase Network error domain.
|
||||
/// Note: these error codes should never change. It would make it harder to decode the errors if
|
||||
/// we inadvertently altered any of these codes in a future SDK version.
|
||||
typedef NS_ENUM(NSInteger, FIRNetworkErrorCode) {
|
||||
/// Unknown error.
|
||||
FIRNetworkErrorCodeUnknown = 0,
|
||||
/// Error occurs when the request URL is invalid.
|
||||
FIRErrorCodeNetworkInvalidURL = 1,
|
||||
/// Error occurs when request cannot be constructed.
|
||||
FIRErrorCodeNetworkRequestCreation = 2,
|
||||
/// Error occurs when payload cannot be compressed.
|
||||
FIRErrorCodeNetworkPayloadCompression = 3,
|
||||
/// Error occurs when session task cannot be created.
|
||||
FIRErrorCodeNetworkSessionTaskCreation = 4,
|
||||
/// Error occurs when there is no response.
|
||||
FIRErrorCodeNetworkInvalidResponse = 5
|
||||
};
|
||||
|
||||
#pragma mark - Network constants
|
||||
|
||||
/// The prefix of the ID of the background session.
|
||||
extern NSString *const kFIRNetworkBackgroundSessionConfigIDPrefix;
|
||||
|
||||
/// The sub directory to store the files of data that is being uploaded in the background.
|
||||
extern NSString *const kFIRNetworkApplicationSupportSubdirectory;
|
||||
|
||||
/// Name of the temporary directory that stores files for background uploading.
|
||||
extern NSString *const kFIRNetworkTempDirectoryName;
|
||||
|
||||
/// The period when the temporary uploading file can stay.
|
||||
extern const NSTimeInterval kFIRNetworkTempFolderExpireTime;
|
||||
|
||||
/// The default network request timeout interval.
|
||||
extern const NSTimeInterval kFIRNetworkTimeOutInterval;
|
||||
|
||||
/// The host to check the reachability of the network.
|
||||
extern NSString *const kFIRNetworkReachabilityHost;
|
||||
|
||||
/// The key to get the error context of the UserInfo.
|
||||
extern NSString *const kFIRNetworkErrorContext;
|
||||
|
||||
#pragma mark - Network Status Code
|
||||
|
||||
extern const int kFIRNetworkHTTPStatusOK;
|
||||
extern const int kFIRNetworkHTTPStatusNoContent;
|
||||
extern const int kFIRNetworkHTTPStatusCodeMultipleChoices;
|
||||
extern const int kFIRNetworkHTTPStatusCodeMovedPermanently;
|
||||
extern const int kFIRNetworkHTTPStatusCodeFound;
|
||||
extern const int kFIRNetworkHTTPStatusCodeNotModified;
|
||||
extern const int kFIRNetworkHTTPStatusCodeMovedTemporarily;
|
||||
extern const int kFIRNetworkHTTPStatusCodeNotFound;
|
||||
extern const int kFIRNetworkHTTPStatusCodeCannotAcceptTraffic;
|
||||
extern const int kFIRNetworkHTTPStatusCodeUnavailable;
|
||||
|
||||
#pragma mark - Error Domain
|
||||
|
||||
extern NSString *const kFIRNetworkErrorDomain;
|
||||
50
Pods/FirebaseCore/Firebase/Core/Private/FIRNetworkLoggerProtocol.h
generated
Normal file
50
Pods/FirebaseCore/Firebase/Core/Private/FIRNetworkLoggerProtocol.h
generated
Normal file
@@ -0,0 +1,50 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRLoggerLevel.h"
|
||||
#import "FIRNetworkMessageCode.h"
|
||||
|
||||
/// The log levels used by FIRNetworkLogger.
|
||||
typedef NS_ENUM(NSInteger, FIRNetworkLogLevel) {
|
||||
kFIRNetworkLogLevelError = FIRLoggerLevelError,
|
||||
kFIRNetworkLogLevelWarning = FIRLoggerLevelWarning,
|
||||
kFIRNetworkLogLevelInfo = FIRLoggerLevelInfo,
|
||||
kFIRNetworkLogLevelDebug = FIRLoggerLevelDebug,
|
||||
};
|
||||
|
||||
@protocol FIRNetworkLoggerDelegate <NSObject>
|
||||
|
||||
@required
|
||||
/// Tells the delegate to log a message with an array of contexts and the log level.
|
||||
- (void)firNetwork_logWithLevel:(FIRNetworkLogLevel)logLevel
|
||||
messageCode:(FIRNetworkMessageCode)messageCode
|
||||
message:(NSString *)message
|
||||
contexts:(NSArray *)contexts;
|
||||
|
||||
/// Tells the delegate to log a message with a context and the log level.
|
||||
- (void)firNetwork_logWithLevel:(FIRNetworkLogLevel)logLevel
|
||||
messageCode:(FIRNetworkMessageCode)messageCode
|
||||
message:(NSString *)message
|
||||
context:(id)context;
|
||||
|
||||
/// Tells the delegate to log a message with the log level.
|
||||
- (void)firNetwork_logWithLevel:(FIRNetworkLogLevel)logLevel
|
||||
messageCode:(FIRNetworkMessageCode)messageCode
|
||||
message:(NSString *)message;
|
||||
|
||||
@end
|
||||
52
Pods/FirebaseCore/Firebase/Core/Private/FIRNetworkMessageCode.h
generated
Normal file
52
Pods/FirebaseCore/Firebase/Core/Private/FIRNetworkMessageCode.h
generated
Normal file
@@ -0,0 +1,52 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// Make sure these codes do not overlap with any contained in the FIRAMessageCode enum.
|
||||
typedef NS_ENUM(NSInteger, FIRNetworkMessageCode) {
|
||||
// FIRNetwork.m
|
||||
kFIRNetworkMessageCodeNetwork000 = 900000, // I-NET900000
|
||||
kFIRNetworkMessageCodeNetwork001 = 900001, // I-NET900001
|
||||
kFIRNetworkMessageCodeNetwork002 = 900002, // I-NET900002
|
||||
kFIRNetworkMessageCodeNetwork003 = 900003, // I-NET900003
|
||||
// FIRNetworkURLSession.m
|
||||
kFIRNetworkMessageCodeURLSession000 = 901000, // I-NET901000
|
||||
kFIRNetworkMessageCodeURLSession001 = 901001, // I-NET901001
|
||||
kFIRNetworkMessageCodeURLSession002 = 901002, // I-NET901002
|
||||
kFIRNetworkMessageCodeURLSession003 = 901003, // I-NET901003
|
||||
kFIRNetworkMessageCodeURLSession004 = 901004, // I-NET901004
|
||||
kFIRNetworkMessageCodeURLSession005 = 901005, // I-NET901005
|
||||
kFIRNetworkMessageCodeURLSession006 = 901006, // I-NET901006
|
||||
kFIRNetworkMessageCodeURLSession007 = 901007, // I-NET901007
|
||||
kFIRNetworkMessageCodeURLSession008 = 901008, // I-NET901008
|
||||
kFIRNetworkMessageCodeURLSession009 = 901009, // I-NET901009
|
||||
kFIRNetworkMessageCodeURLSession010 = 901010, // I-NET901010
|
||||
kFIRNetworkMessageCodeURLSession011 = 901011, // I-NET901011
|
||||
kFIRNetworkMessageCodeURLSession012 = 901012, // I-NET901012
|
||||
kFIRNetworkMessageCodeURLSession013 = 901013, // I-NET901013
|
||||
kFIRNetworkMessageCodeURLSession014 = 901014, // I-NET901014
|
||||
kFIRNetworkMessageCodeURLSession015 = 901015, // I-NET901015
|
||||
kFIRNetworkMessageCodeURLSession016 = 901016, // I-NET901016
|
||||
kFIRNetworkMessageCodeURLSession017 = 901017, // I-NET901017
|
||||
kFIRNetworkMessageCodeURLSession018 = 901018, // I-NET901018
|
||||
// FIRReachabilityChecker.m
|
||||
kFIRNetworkMessageCodeReachabilityChecker000 = 902000, // I-NET902000
|
||||
kFIRNetworkMessageCodeReachabilityChecker001 = 902001, // I-NET902001
|
||||
kFIRNetworkMessageCodeReachabilityChecker002 = 902002, // I-NET902002
|
||||
kFIRNetworkMessageCodeReachabilityChecker003 = 902003, // I-NET902003
|
||||
kFIRNetworkMessageCodeReachabilityChecker004 = 902004, // I-NET902004
|
||||
kFIRNetworkMessageCodeReachabilityChecker005 = 902005, // I-NET902005
|
||||
kFIRNetworkMessageCodeReachabilityChecker006 = 902006, // I-NET902006
|
||||
};
|
||||
60
Pods/FirebaseCore/Firebase/Core/Private/FIRNetworkURLSession.h
generated
Normal file
60
Pods/FirebaseCore/Firebase/Core/Private/FIRNetworkURLSession.h
generated
Normal file
@@ -0,0 +1,60 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRNetworkLoggerProtocol.h"
|
||||
|
||||
typedef void (^FIRNetworkCompletionHandler)(NSHTTPURLResponse *response,
|
||||
NSData *data,
|
||||
NSError *error);
|
||||
typedef void (^FIRNetworkURLSessionCompletionHandler)(NSHTTPURLResponse *response,
|
||||
NSData *data,
|
||||
NSString *sessionID,
|
||||
NSError *error);
|
||||
typedef void (^FIRNetworkSystemCompletionHandler)(void);
|
||||
|
||||
/// The protocol that uses NSURLSession for iOS >= 7.0 to handle requests and responses.
|
||||
@interface FIRNetworkURLSession
|
||||
: NSObject <NSURLSessionDelegate, NSURLSessionTaskDelegate, NSURLSessionDownloadDelegate>
|
||||
|
||||
/// Indicates whether the background network is enabled. Default value is NO.
|
||||
@property(nonatomic, getter=isBackgroundNetworkEnabled) BOOL backgroundNetworkEnabled;
|
||||
|
||||
/// The logger delegate to log message, errors or warnings that occur during the network operations.
|
||||
@property(nonatomic, weak) id<FIRNetworkLoggerDelegate> loggerDelegate;
|
||||
|
||||
/// Calls the system provided completion handler after the background session is finished.
|
||||
+ (void)handleEventsForBackgroundURLSessionID:(NSString *)sessionID
|
||||
completionHandler:(FIRNetworkSystemCompletionHandler)completionHandler;
|
||||
|
||||
/// Initializes with logger delegate.
|
||||
- (instancetype)initWithNetworkLoggerDelegate:(id<FIRNetworkLoggerDelegate>)networkLoggerDelegate
|
||||
NS_DESIGNATED_INITIALIZER;
|
||||
|
||||
- (instancetype)init NS_UNAVAILABLE;
|
||||
|
||||
/// Sends an asynchronous POST request and calls the provided completion handler when the request
|
||||
/// completes or when errors occur, and returns an ID of the session/connection.
|
||||
- (NSString *)sessionIDFromAsyncPOSTRequest:(NSURLRequest *)request
|
||||
completionHandler:(FIRNetworkURLSessionCompletionHandler)handler;
|
||||
|
||||
/// Sends an asynchronous GET request and calls the provided completion handler when the request
|
||||
/// completes or when errors occur, and returns an ID of the session.
|
||||
- (NSString *)sessionIDFromAsyncGETRequest:(NSURLRequest *)request
|
||||
completionHandler:(FIRNetworkURLSessionCompletionHandler)handler;
|
||||
|
||||
@end
|
||||
114
Pods/FirebaseCore/Firebase/Core/Private/FIROptionsInternal.h
generated
Normal file
114
Pods/FirebaseCore/Firebase/Core/Private/FIROptionsInternal.h
generated
Normal file
@@ -0,0 +1,114 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import "FIROptions.h"
|
||||
|
||||
/**
|
||||
* Keys for the strings in the plist file.
|
||||
*/
|
||||
extern NSString *const kFIRAPIKey;
|
||||
extern NSString *const kFIRTrackingID;
|
||||
extern NSString *const kFIRGoogleAppID;
|
||||
extern NSString *const kFIRClientID;
|
||||
extern NSString *const kFIRGCMSenderID;
|
||||
extern NSString *const kFIRAndroidClientID;
|
||||
extern NSString *const kFIRDatabaseURL;
|
||||
extern NSString *const kFIRStorageBucket;
|
||||
extern NSString *const kFIRBundleID;
|
||||
extern NSString *const kFIRProjectID;
|
||||
|
||||
/**
|
||||
* Keys for the plist file name
|
||||
*/
|
||||
extern NSString *const kServiceInfoFileName;
|
||||
|
||||
extern NSString *const kServiceInfoFileType;
|
||||
|
||||
/**
|
||||
* This header file exposes the initialization of FIROptions to internal use.
|
||||
*/
|
||||
@interface FIROptions ()
|
||||
|
||||
/**
|
||||
* resetDefaultOptions and initInternalWithOptionsDictionary: are exposed only for unit tests.
|
||||
*/
|
||||
+ (void)resetDefaultOptions;
|
||||
|
||||
/**
|
||||
* Initializes the options with dictionary. The above strings are the keys of the dictionary.
|
||||
* This is the designated initializer.
|
||||
*/
|
||||
- (instancetype)initInternalWithOptionsDictionary:(NSDictionary *)serviceInfoDictionary;
|
||||
|
||||
/**
|
||||
* defaultOptions and defaultOptionsDictionary are exposed in order to be used in FIRApp and
|
||||
* other first party services.
|
||||
*/
|
||||
+ (FIROptions *)defaultOptions;
|
||||
|
||||
+ (NSDictionary *)defaultOptionsDictionary;
|
||||
|
||||
/**
|
||||
* Indicates whether or not Analytics collection was explicitly enabled via a plist flag or at
|
||||
* runtime.
|
||||
*/
|
||||
@property(nonatomic, readonly) BOOL isAnalyticsCollectionExpicitlySet;
|
||||
|
||||
/**
|
||||
* Whether or not Analytics Collection was enabled. Analytics Collection is enabled unless
|
||||
* explicitly disabled in GoogleService-Info.plist.
|
||||
*/
|
||||
@property(nonatomic, readonly) BOOL isAnalyticsCollectionEnabled;
|
||||
|
||||
/**
|
||||
* Whether or not Analytics Collection was completely disabled. If YES, then
|
||||
* isAnalyticsCollectionEnabled will be NO.
|
||||
*/
|
||||
@property(nonatomic, readonly) BOOL isAnalyticsCollectionDeactivated;
|
||||
|
||||
/**
|
||||
* The version ID of the client library, e.g. @"1100000".
|
||||
*/
|
||||
@property(nonatomic, readonly, copy) NSString *libraryVersionID;
|
||||
|
||||
/**
|
||||
* The flag indicating whether this object was constructed with the values in the default plist
|
||||
* file.
|
||||
*/
|
||||
@property(nonatomic) BOOL usingOptionsFromDefaultPlist;
|
||||
|
||||
/**
|
||||
* Whether or not Measurement was enabled. Measurement is enabled unless explicitly disabled in
|
||||
* GoogleService-Info.plist.
|
||||
*/
|
||||
@property(nonatomic, readonly) BOOL isMeasurementEnabled;
|
||||
|
||||
/**
|
||||
* Whether or not Analytics was enabled in the developer console.
|
||||
*/
|
||||
@property(nonatomic, readonly) BOOL isAnalyticsEnabled;
|
||||
|
||||
/**
|
||||
* Whether or not SignIn was enabled in the developer console.
|
||||
*/
|
||||
@property(nonatomic, readonly) BOOL isSignInEnabled;
|
||||
|
||||
/**
|
||||
* Whether or not editing is locked. This should occur after FIROptions has been set on a FIRApp.
|
||||
*/
|
||||
@property(nonatomic, getter=isEditingLocked) BOOL editingLocked;
|
||||
|
||||
@end
|
||||
47
Pods/FirebaseCore/Firebase/Core/Private/FIRReachabilityChecker+Internal.h
generated
Normal file
47
Pods/FirebaseCore/Firebase/Core/Private/FIRReachabilityChecker+Internal.h
generated
Normal file
@@ -0,0 +1,47 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import "FIRReachabilityChecker.h"
|
||||
|
||||
typedef SCNetworkReachabilityRef (*FIRReachabilityCreateWithNameFn)(CFAllocatorRef allocator,
|
||||
const char *host);
|
||||
|
||||
typedef Boolean (*FIRReachabilitySetCallbackFn)(SCNetworkReachabilityRef target,
|
||||
SCNetworkReachabilityCallBack callback,
|
||||
SCNetworkReachabilityContext *context);
|
||||
typedef Boolean (*FIRReachabilityScheduleWithRunLoopFn)(SCNetworkReachabilityRef target,
|
||||
CFRunLoopRef runLoop,
|
||||
CFStringRef runLoopMode);
|
||||
typedef Boolean (*FIRReachabilityUnscheduleFromRunLoopFn)(SCNetworkReachabilityRef target,
|
||||
CFRunLoopRef runLoop,
|
||||
CFStringRef runLoopMode);
|
||||
|
||||
typedef void (*FIRReachabilityReleaseFn)(CFTypeRef cf);
|
||||
|
||||
struct FIRReachabilityApi {
|
||||
FIRReachabilityCreateWithNameFn createWithNameFn;
|
||||
FIRReachabilitySetCallbackFn setCallbackFn;
|
||||
FIRReachabilityScheduleWithRunLoopFn scheduleWithRunLoopFn;
|
||||
FIRReachabilityUnscheduleFromRunLoopFn unscheduleFromRunLoopFn;
|
||||
FIRReachabilityReleaseFn releaseFn;
|
||||
};
|
||||
|
||||
@interface FIRReachabilityChecker (Internal)
|
||||
|
||||
- (const struct FIRReachabilityApi *)reachabilityApi;
|
||||
- (void)setReachabilityApi:(const struct FIRReachabilityApi *)reachabilityApi;
|
||||
|
||||
@end
|
||||
83
Pods/FirebaseCore/Firebase/Core/Private/FIRReachabilityChecker.h
generated
Normal file
83
Pods/FirebaseCore/Firebase/Core/Private/FIRReachabilityChecker.h
generated
Normal file
@@ -0,0 +1,83 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
#import <SystemConfiguration/SystemConfiguration.h>
|
||||
|
||||
/// Reachability Status
|
||||
typedef enum {
|
||||
kFIRReachabilityUnknown, ///< Have not yet checked or been notified whether host is reachable.
|
||||
kFIRReachabilityNotReachable, ///< Host is not reachable.
|
||||
kFIRReachabilityViaWifi, ///< Host is reachable via Wifi.
|
||||
kFIRReachabilityViaCellular, ///< Host is reachable via cellular.
|
||||
} FIRReachabilityStatus;
|
||||
|
||||
const NSString *FIRReachabilityStatusString(FIRReachabilityStatus status);
|
||||
|
||||
@class FIRReachabilityChecker;
|
||||
@protocol FIRNetworkLoggerDelegate;
|
||||
|
||||
/// Google Analytics iOS Reachability Checker.
|
||||
@protocol FIRReachabilityDelegate
|
||||
@required
|
||||
/// Called when network status has changed.
|
||||
- (void)reachability:(FIRReachabilityChecker *)reachability
|
||||
statusChanged:(FIRReachabilityStatus)status;
|
||||
@end
|
||||
|
||||
/// Google Analytics iOS Network Status Checker.
|
||||
@interface FIRReachabilityChecker : NSObject
|
||||
|
||||
/// The last known reachability status, or FIRReachabilityStatusUnknown if the
|
||||
/// checker is not active.
|
||||
@property(nonatomic, readonly) FIRReachabilityStatus reachabilityStatus;
|
||||
/// The host to which reachability status is to be checked.
|
||||
@property(nonatomic, copy, readonly) NSString *host;
|
||||
/// The delegate to be notified of reachability status changes.
|
||||
@property(nonatomic, weak) id<FIRReachabilityDelegate> reachabilityDelegate;
|
||||
/// The delegate to be notified to log messages.
|
||||
@property(nonatomic, weak) id<FIRNetworkLoggerDelegate> loggerDelegate;
|
||||
/// `YES` if the reachability checker is active, `NO` otherwise.
|
||||
@property(nonatomic, readonly) BOOL isActive;
|
||||
|
||||
/// Initialize the reachability checker. Note that you must call start to begin checking for and
|
||||
/// receiving notifications about network status changes.
|
||||
///
|
||||
/// @param reachabilityDelegate The delegate to be notified when reachability status to host
|
||||
/// changes.
|
||||
///
|
||||
/// @param loggerDelegate The delegate to send log messages to.
|
||||
///
|
||||
/// @param host The name of the host.
|
||||
///
|
||||
- (instancetype)initWithReachabilityDelegate:(id<FIRReachabilityDelegate>)reachabilityDelegate
|
||||
loggerDelegate:(id<FIRNetworkLoggerDelegate>)loggerDelegate
|
||||
withHost:(NSString *)host;
|
||||
|
||||
- (instancetype)init NS_UNAVAILABLE;
|
||||
|
||||
/// Start checking for reachability to the specified host. This has no effect if the status
|
||||
/// checker is already checking for connectivity.
|
||||
///
|
||||
/// @return `YES` if initiating status checking was successful or the status checking has already
|
||||
/// been initiated, `NO` otherwise.
|
||||
- (BOOL)start;
|
||||
|
||||
/// Stop checking for reachability to the specified host. This has no effect if the status
|
||||
/// checker is not checking for connectivity.
|
||||
- (void)stop;
|
||||
|
||||
@end
|
||||
23
Pods/FirebaseCore/Firebase/Core/Private/FIRVersion.h
generated
Normal file
23
Pods/FirebaseCore/Firebase/Core/Private/FIRVersion.h
generated
Normal file
@@ -0,0 +1,23 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
/** The version of the Firebase SDK. */
|
||||
FOUNDATION_EXPORT const unsigned char *const FIRVersionString;
|
||||
|
||||
/** The version of the FirebaseCore Component. */
|
||||
FOUNDATION_EXPORT const unsigned char *const FIRCoreVersionString;
|
||||
52
Pods/FirebaseCore/Firebase/Core/Public/FIRAnalyticsConfiguration.h
generated
Normal file
52
Pods/FirebaseCore/Firebase/Core/Public/FIRAnalyticsConfiguration.h
generated
Normal file
@@ -0,0 +1,52 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
/**
|
||||
* This class provides configuration fields for Firebase Analytics.
|
||||
*/
|
||||
NS_SWIFT_NAME(AnalyticsConfiguration)
|
||||
@interface FIRAnalyticsConfiguration : NSObject
|
||||
|
||||
/**
|
||||
* Returns the shared instance of FIRAnalyticsConfiguration.
|
||||
*/
|
||||
+ (FIRAnalyticsConfiguration *)sharedInstance NS_SWIFT_NAME(shared());
|
||||
|
||||
/**
|
||||
* Sets the minimum engagement time in seconds required to start a new session. The default value
|
||||
* is 10 seconds.
|
||||
*/
|
||||
- (void)setMinimumSessionInterval:(NSTimeInterval)minimumSessionInterval;
|
||||
|
||||
/**
|
||||
* Sets the interval of inactivity in seconds that terminates the current session. The default
|
||||
* value is 1800 seconds (30 minutes).
|
||||
*/
|
||||
- (void)setSessionTimeoutInterval:(NSTimeInterval)sessionTimeoutInterval;
|
||||
|
||||
/**
|
||||
* Sets whether analytics collection is enabled for this app on this device. This setting is
|
||||
* persisted across app sessions. By default it is enabled.
|
||||
*/
|
||||
- (void)setAnalyticsCollectionEnabled:(BOOL)analyticsCollectionEnabled;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
118
Pods/FirebaseCore/Firebase/Core/Public/FIRApp.h
generated
Normal file
118
Pods/FirebaseCore/Firebase/Core/Public/FIRApp.h
generated
Normal file
@@ -0,0 +1,118 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
@class FIROptions;
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
/** A block that takes a BOOL and has no return value. */
|
||||
typedef void (^FIRAppVoidBoolCallback)(BOOL success) NS_SWIFT_NAME(FirebaseAppVoidBoolCallback);
|
||||
|
||||
/**
|
||||
* The entry point of Firebase SDKs.
|
||||
*
|
||||
* Initialize and configure FIRApp using +[FIRApp configure]
|
||||
* or other customized ways as shown below.
|
||||
*
|
||||
* The logging system has two modes: default mode and debug mode. In default mode, only logs with
|
||||
* log level Notice, Warning and Error will be sent to device. In debug mode, all logs will be sent
|
||||
* to device. The log levels that Firebase uses are consistent with the ASL log levels.
|
||||
*
|
||||
* Enable debug mode by passing the -FIRDebugEnabled argument to the application. You can add this
|
||||
* argument in the application's Xcode scheme. When debug mode is enabled via -FIRDebugEnabled,
|
||||
* further executions of the application will also be in debug mode. In order to return to default
|
||||
* mode, you must explicitly disable the debug mode with the application argument -FIRDebugDisabled.
|
||||
*
|
||||
* It is also possible to change the default logging level in code by calling setLoggerLevel: on
|
||||
* the FIRConfiguration interface.
|
||||
*/
|
||||
NS_SWIFT_NAME(FirebaseApp)
|
||||
@interface FIRApp : NSObject
|
||||
|
||||
/**
|
||||
* Configures a default Firebase app. Raises an exception if any configuration step fails. The
|
||||
* default app is named "__FIRAPP_DEFAULT". This method should be called after the app is launched
|
||||
* and before using Firebase services. This method is thread safe and contains synchronous file I/O
|
||||
* (reading GoogleService-Info.plist from disk).
|
||||
*/
|
||||
+ (void)configure;
|
||||
|
||||
/**
|
||||
* Configures the default Firebase app with the provided options. The default app is named
|
||||
* "__FIRAPP_DEFAULT". Raises an exception if any configuration step fails. This method is thread
|
||||
* safe.
|
||||
*
|
||||
* @param options The Firebase application options used to configure the service.
|
||||
*/
|
||||
+ (void)configureWithOptions:(FIROptions *)options NS_SWIFT_NAME(configure(options:));
|
||||
|
||||
/**
|
||||
* Configures a Firebase app with the given name and options. Raises an exception if any
|
||||
* configuration step fails. This method is thread safe.
|
||||
*
|
||||
* @param name The application's name given by the developer. The name should should only contain
|
||||
Letters, Numbers and Underscore.
|
||||
* @param options The Firebase application options used to configure the services.
|
||||
*/
|
||||
// clang-format off
|
||||
+ (void)configureWithName:(NSString *)name
|
||||
options:(FIROptions *)options NS_SWIFT_NAME(configure(name:options:));
|
||||
// clang-format on
|
||||
|
||||
/**
|
||||
* Returns the default app, or nil if the default app does not exist.
|
||||
*/
|
||||
+ (nullable FIRApp *)defaultApp NS_SWIFT_NAME(app());
|
||||
|
||||
/**
|
||||
* Returns a previously created FIRApp instance with the given name, or nil if no such app exists.
|
||||
* This method is thread safe.
|
||||
*/
|
||||
+ (nullable FIRApp *)appNamed:(NSString *)name NS_SWIFT_NAME(app(name:));
|
||||
|
||||
/**
|
||||
* Returns the set of all extant FIRApp instances, or nil if there are no FIRApp instances. This
|
||||
* method is thread safe.
|
||||
*/
|
||||
@property(class, readonly, nullable) NSDictionary<NSString *, FIRApp *> *allApps;
|
||||
|
||||
/**
|
||||
* Cleans up the current FIRApp, freeing associated data and returning its name to the pool for
|
||||
* future use. This method is thread safe.
|
||||
*/
|
||||
- (void)deleteApp:(FIRAppVoidBoolCallback)completion;
|
||||
|
||||
/**
|
||||
* FIRApp instances should not be initialized directly. Call +[FIRApp configure],
|
||||
* +[FIRApp configureWithOptions:], or +[FIRApp configureWithNames:options:] directly.
|
||||
*/
|
||||
- (instancetype)init NS_UNAVAILABLE;
|
||||
|
||||
/**
|
||||
* Gets the name of this app.
|
||||
*/
|
||||
@property(nonatomic, copy, readonly) NSString *name;
|
||||
|
||||
/**
|
||||
* Gets a copy of the options for this app. These are non-modifiable.
|
||||
*/
|
||||
@property(nonatomic, copy, readonly) FIROptions *options;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
50
Pods/FirebaseCore/Firebase/Core/Public/FIRConfiguration.h
generated
Normal file
50
Pods/FirebaseCore/Firebase/Core/Public/FIRConfiguration.h
generated
Normal file
@@ -0,0 +1,50 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRAnalyticsConfiguration.h"
|
||||
#import "FIRLoggerLevel.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
/**
|
||||
* This interface provides global level properties that the developer can tweak, and the singleton
|
||||
* of the Firebase Analytics configuration class.
|
||||
*/
|
||||
NS_SWIFT_NAME(FirebaseConfiguration)
|
||||
@interface FIRConfiguration : NSObject
|
||||
|
||||
/** Returns the shared configuration object. */
|
||||
@property(class, nonatomic, readonly) FIRConfiguration *sharedInstance NS_SWIFT_NAME(shared);
|
||||
|
||||
/** The configuration class for Firebase Analytics. */
|
||||
@property(nonatomic, readwrite) FIRAnalyticsConfiguration *analyticsConfiguration;
|
||||
|
||||
/**
|
||||
* Sets the logging level for internal Firebase logging. Firebase will only log messages
|
||||
* that are logged at or below loggerLevel. The messages are logged both to the Xcode
|
||||
* console and to the device's log. Note that if an app is running from AppStore, it will
|
||||
* never log above FIRLoggerLevelNotice even if loggerLevel is set to a higher (more verbose)
|
||||
* setting.
|
||||
*
|
||||
* @param loggerLevel The maximum logging level. The default level is set to FIRLoggerLevelNotice.
|
||||
*/
|
||||
- (void)setLoggerLevel:(FIRLoggerLevel)loggerLevel;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
35
Pods/FirebaseCore/Firebase/Core/Public/FIRLoggerLevel.h
generated
Normal file
35
Pods/FirebaseCore/Firebase/Core/Public/FIRLoggerLevel.h
generated
Normal file
@@ -0,0 +1,35 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
/**
|
||||
* The log levels used by internal logging.
|
||||
*/
|
||||
typedef NS_ENUM(NSInteger, FIRLoggerLevel) {
|
||||
/** Error level, matches ASL_LEVEL_ERR. */
|
||||
FIRLoggerLevelError = 3,
|
||||
/** Warning level, matches ASL_LEVEL_WARNING. */
|
||||
FIRLoggerLevelWarning = 4,
|
||||
/** Notice level, matches ASL_LEVEL_NOTICE. */
|
||||
FIRLoggerLevelNotice = 5,
|
||||
/** Info level, matches ASL_LEVEL_NOTICE. */
|
||||
FIRLoggerLevelInfo = 6,
|
||||
/** Debug level, matches ASL_LEVEL_DEBUG. */
|
||||
FIRLoggerLevelDebug = 7,
|
||||
/** Minimum log level. */
|
||||
FIRLoggerLevelMin = FIRLoggerLevelError,
|
||||
/** Maximum log level. */
|
||||
FIRLoggerLevelMax = FIRLoggerLevelDebug
|
||||
} NS_SWIFT_NAME(FirebaseLoggerLevel);
|
||||
116
Pods/FirebaseCore/Firebase/Core/Public/FIROptions.h
generated
Normal file
116
Pods/FirebaseCore/Firebase/Core/Public/FIROptions.h
generated
Normal file
@@ -0,0 +1,116 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
/**
|
||||
* This class provides constant fields of Google APIs.
|
||||
*/
|
||||
NS_SWIFT_NAME(FirebaseOptions)
|
||||
@interface FIROptions : NSObject <NSCopying>
|
||||
|
||||
/**
|
||||
* Returns the default options. The first time this is called it synchronously reads
|
||||
* GoogleService-Info.plist from disk.
|
||||
*/
|
||||
+ (nullable FIROptions *)defaultOptions NS_SWIFT_NAME(defaultOptions());
|
||||
|
||||
/**
|
||||
* An iOS API key used for authenticating requests from your app, e.g.
|
||||
* @"AIzaSyDdVgKwhZl0sTTTLZ7iTmt1r3N2cJLnaDk", used to identify your app to Google servers.
|
||||
*/
|
||||
@property(nonatomic, copy, nullable) NSString *APIKey NS_SWIFT_NAME(apiKey);
|
||||
|
||||
/**
|
||||
* The bundle ID for the application. Defaults to `[[NSBundle mainBundle] bundleID]` when not set
|
||||
* manually or in a plist.
|
||||
*/
|
||||
@property(nonatomic, copy) NSString *bundleID;
|
||||
|
||||
/**
|
||||
* The OAuth2 client ID for iOS application used to authenticate Google users, for example
|
||||
* @"12345.apps.googleusercontent.com", used for signing in with Google.
|
||||
*/
|
||||
@property(nonatomic, copy, nullable) NSString *clientID;
|
||||
|
||||
/**
|
||||
* The tracking ID for Google Analytics, e.g. @"UA-12345678-1", used to configure Google Analytics.
|
||||
*/
|
||||
@property(nonatomic, copy, nullable) NSString *trackingID;
|
||||
|
||||
/**
|
||||
* The Project Number from the Google Developer's console, for example @"012345678901", used to
|
||||
* configure Google Cloud Messaging.
|
||||
*/
|
||||
@property(nonatomic, copy) NSString *GCMSenderID NS_SWIFT_NAME(gcmSenderID);
|
||||
|
||||
/**
|
||||
* The Project ID from the Firebase console, for example @"abc-xyz-123".
|
||||
*/
|
||||
@property(nonatomic, copy, nullable) NSString *projectID;
|
||||
|
||||
/**
|
||||
* The Android client ID used in Google AppInvite when an iOS app has its Android version, for
|
||||
* example @"12345.apps.googleusercontent.com".
|
||||
*/
|
||||
@property(nonatomic, copy, nullable) NSString *androidClientID;
|
||||
|
||||
/**
|
||||
* The Google App ID that is used to uniquely identify an instance of an app.
|
||||
*/
|
||||
@property(nonatomic, copy) NSString *googleAppID;
|
||||
|
||||
/**
|
||||
* The database root URL, e.g. @"http://abc-xyz-123.firebaseio.com".
|
||||
*/
|
||||
@property(nonatomic, copy, nullable) NSString *databaseURL;
|
||||
|
||||
/**
|
||||
* The URL scheme used to set up Durable Deep Link service.
|
||||
*/
|
||||
@property(nonatomic, copy, nullable) NSString *deepLinkURLScheme;
|
||||
|
||||
/**
|
||||
* The Google Cloud Storage bucket name, e.g. @"abc-xyz-123.storage.firebase.com".
|
||||
*/
|
||||
@property(nonatomic, copy, nullable) NSString *storageBucket;
|
||||
|
||||
/**
|
||||
* Initializes a customized instance of FIROptions from the file at the given plist file path. This
|
||||
* will read the file synchronously from disk.
|
||||
* For example,
|
||||
* NSString *filePath =
|
||||
* [[NSBundle mainBundle] pathForResource:@"GoogleService-Info" ofType:@"plist"];
|
||||
* FIROptions *options = [[FIROptions alloc] initWithContentsOfFile:filePath];
|
||||
* Returns nil if the plist file does not exist or is invalid.
|
||||
*/
|
||||
- (nullable instancetype)initWithContentsOfFile:(NSString *)plistPath;
|
||||
|
||||
/**
|
||||
* Initializes a customized instance of FIROptions with required fields. Use the mutable properties
|
||||
* to modify fields for configuring specific services.
|
||||
*/
|
||||
// clang-format off
|
||||
- (instancetype)initWithGoogleAppID:(NSString *)googleAppID
|
||||
GCMSenderID:(NSString *)GCMSenderID
|
||||
NS_SWIFT_NAME(init(googleAppID:gcmSenderID:));
|
||||
// clang-format on
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
21
Pods/FirebaseCore/Firebase/Core/Public/FirebaseCore.h
generated
Normal file
21
Pods/FirebaseCore/Firebase/Core/Public/FirebaseCore.h
generated
Normal file
@@ -0,0 +1,21 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import "FIRAnalyticsConfiguration.h"
|
||||
#import "FIRApp.h"
|
||||
#import "FIRConfiguration.h"
|
||||
#import "FIRLoggerLevel.h"
|
||||
#import "FIROptions.h"
|
||||
43
Pods/FirebaseCore/Firebase/Core/third_party/FIRAppEnvironmentUtil.h
generated
vendored
Normal file
43
Pods/FirebaseCore/Firebase/Core/third_party/FIRAppEnvironmentUtil.h
generated
vendored
Normal file
@@ -0,0 +1,43 @@
|
||||
/*
|
||||
* Copyright 2017 Google
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
@interface FIRAppEnvironmentUtil : NSObject
|
||||
|
||||
/// Indicates whether the app is from Apple Store or not. Returns NO if the app is on simulator,
|
||||
/// development environment or sideloaded.
|
||||
+ (BOOL)isFromAppStore;
|
||||
|
||||
/// Indicates whether the app is a Testflight app. Returns YES if the app has sandbox receipt.
|
||||
/// Returns NO otherwise.
|
||||
+ (BOOL)isAppStoreReceiptSandbox;
|
||||
|
||||
/// Indicates whether the app is on simulator or not at runtime depending on the device
|
||||
/// architecture.
|
||||
+ (BOOL)isSimulator;
|
||||
|
||||
/// The current device model. Returns an empty string if device model cannot be retrieved.
|
||||
+ (NSString *)deviceModel;
|
||||
|
||||
/// The current operating system version. Returns an empty string if the system version cannot be
|
||||
/// retrieved.
|
||||
+ (NSString *)systemVersion;
|
||||
|
||||
/// Indicates whether it is running inside an extension or an app.
|
||||
+ (BOOL)isAppExtension;
|
||||
|
||||
@end
|
||||
239
Pods/FirebaseCore/Firebase/Core/third_party/FIRAppEnvironmentUtil.m
generated
vendored
Normal file
239
Pods/FirebaseCore/Firebase/Core/third_party/FIRAppEnvironmentUtil.m
generated
vendored
Normal file
@@ -0,0 +1,239 @@
|
||||
// Copyright 2017 Google
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRAppEnvironmentUtil.h"
|
||||
|
||||
#import <dlfcn.h>
|
||||
#import <mach-o/dyld.h>
|
||||
#import <sys/utsname.h>
|
||||
|
||||
/// The encryption info struct and constants are missing from the iPhoneSimulator SDK, but not from
|
||||
/// the iPhoneOS or Mac OS X SDKs. Since one doesn't ever ship a Simulator binary, we'll just
|
||||
/// provide the definitions here.
|
||||
#if TARGET_OS_SIMULATOR && !defined(LC_ENCRYPTION_INFO)
|
||||
#define LC_ENCRYPTION_INFO 0x21
|
||||
struct encryption_info_command {
|
||||
uint32_t cmd;
|
||||
uint32_t cmdsize;
|
||||
uint32_t cryptoff;
|
||||
uint32_t cryptsize;
|
||||
uint32_t cryptid;
|
||||
};
|
||||
#endif
|
||||
|
||||
@implementation FIRAppEnvironmentUtil
|
||||
|
||||
/// A key for the Info.plist to enable or disable checking if the App Store is running in a sandbox.
|
||||
/// This will affect your data integrity when using Firebase Analytics, as it will disable some
|
||||
/// necessary checks.
|
||||
static NSString *const kFIRAppStoreReceiptURLCheckEnabledKey =
|
||||
@"FirebaseAppStoreReceiptURLCheckEnabled";
|
||||
|
||||
/// The file name of the sandbox receipt. This is available on iOS >= 8.0
|
||||
static NSString *const kFIRAIdentitySandboxReceiptFileName = @"sandboxReceipt";
|
||||
|
||||
/// The following copyright from Landon J. Fuller applies to the isAppEncrypted function.
|
||||
///
|
||||
/// Copyright (c) 2017 Landon J. Fuller <landon@landonf.org>
|
||||
/// All rights reserved.
|
||||
///
|
||||
/// Permission is hereby granted, free of charge, to any person obtaining a copy of this software
|
||||
/// and associated documentation files (the "Software"), to deal in the Software without
|
||||
/// restriction, including without limitation the rights to use, copy, modify, merge, publish,
|
||||
/// distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the
|
||||
/// Software is furnished to do so, subject to the following conditions:
|
||||
///
|
||||
/// The above copyright notice and this permission notice shall be included in all copies or
|
||||
/// substantial portions of the Software.
|
||||
///
|
||||
/// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
|
||||
/// BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||
/// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
||||
/// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
/// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
///
|
||||
/// Comment from <a href="http://iphonedevwiki.net/index.php/Crack_prevention">iPhone Dev Wiki
|
||||
/// Crack Prevention</a>:
|
||||
/// App Store binaries are signed by both their developer and Apple. This encrypts the binary so
|
||||
/// that decryption keys are needed in order to make the binary readable. When iOS executes the
|
||||
/// binary, the decryption keys are used to decrypt the binary into a readable state where it is
|
||||
/// then loaded into memory and executed. iOS can tell the encryption status of a binary via the
|
||||
/// cryptid structure member of LC_ENCRYPTION_INFO MachO load command. If cryptid is a non-zero
|
||||
/// value then the binary is encrypted.
|
||||
///
|
||||
/// 'Cracking' works by letting the kernel decrypt the binary then siphoning the decrypted data into
|
||||
/// a new binary file, resigning, and repackaging. This will only work on jailbroken devices as
|
||||
/// codesignature validation has been removed. Resigning takes place because while the codesignature
|
||||
/// doesn't have to be valid thanks to the jailbreak, it does have to be in place unless you have
|
||||
/// AppSync or similar to disable codesignature checks.
|
||||
///
|
||||
/// More information at <a href="http://landonf.org/2009/02/index.html">Landon Fuller's blog</a>
|
||||
static BOOL isAppEncrypted() {
|
||||
const struct mach_header *executableHeader = NULL;
|
||||
for (uint32_t i = 0; i < _dyld_image_count(); i++) {
|
||||
const struct mach_header *header = _dyld_get_image_header(i);
|
||||
if (header && header->filetype == MH_EXECUTE) {
|
||||
executableHeader = header;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!executableHeader) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
BOOL is64bit = (executableHeader->magic == MH_MAGIC_64);
|
||||
uintptr_t cursor = (uintptr_t)executableHeader +
|
||||
(is64bit ? sizeof(struct mach_header_64) : sizeof(struct mach_header));
|
||||
const struct segment_command *segmentCommand = NULL;
|
||||
uint32_t i = 0;
|
||||
|
||||
while (i++ < executableHeader->ncmds) {
|
||||
segmentCommand = (struct segment_command *)cursor;
|
||||
|
||||
if (!segmentCommand) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if ((!is64bit && segmentCommand->cmd == LC_ENCRYPTION_INFO) ||
|
||||
(is64bit && segmentCommand->cmd == LC_ENCRYPTION_INFO_64)) {
|
||||
if (is64bit) {
|
||||
struct encryption_info_command_64 *cryptCmd =
|
||||
(struct encryption_info_command_64 *)segmentCommand;
|
||||
return cryptCmd && cryptCmd->cryptid != 0;
|
||||
} else {
|
||||
struct encryption_info_command *cryptCmd = (struct encryption_info_command *)segmentCommand;
|
||||
return cryptCmd && cryptCmd->cryptid != 0;
|
||||
}
|
||||
}
|
||||
cursor += segmentCommand->cmdsize;
|
||||
}
|
||||
|
||||
return NO;
|
||||
}
|
||||
|
||||
+ (BOOL)isFromAppStore {
|
||||
static dispatch_once_t isEncryptedOnce;
|
||||
static BOOL isEncrypted = NO;
|
||||
|
||||
dispatch_once(&isEncryptedOnce, ^{
|
||||
isEncrypted = isAppEncrypted();
|
||||
});
|
||||
|
||||
if ([FIRAppEnvironmentUtil isSimulator]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
// If an app contain the sandboxReceipt file, it means its coming from TestFlight
|
||||
// This must be checked before the SCInfo Folder check below since TestFlight apps may
|
||||
// also have an SCInfo folder.
|
||||
if ([FIRAppEnvironmentUtil isAppStoreReceiptSandbox]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
if ([FIRAppEnvironmentUtil hasSCInfoFolder]) {
|
||||
// When iTunes downloads a .ipa, it also gets a customized .sinf file which is added to the
|
||||
// main SC_Info directory.
|
||||
return YES;
|
||||
}
|
||||
|
||||
// For iOS >= 8.0, iTunesMetadata.plist is moved outside of the sandbox. Any attempt to read
|
||||
// the iTunesMetadata.plist outside of the sandbox will be rejected by Apple.
|
||||
// If the app does not contain the embedded.mobileprovision which is stripped out by Apple when
|
||||
// the app is submitted to store, then it is highly likely that it is from Apple Store.
|
||||
return isEncrypted && ![FIRAppEnvironmentUtil hasEmbeddedMobileProvision];
|
||||
}
|
||||
|
||||
+ (BOOL)isAppStoreReceiptSandbox {
|
||||
// Since checking the App Store's receipt URL can be memory intensive, check the option in the
|
||||
// Info.plist if developers opted out of this check.
|
||||
id enableSandboxCheck =
|
||||
[[NSBundle mainBundle] objectForInfoDictionaryKey:kFIRAppStoreReceiptURLCheckEnabledKey];
|
||||
if (enableSandboxCheck && [enableSandboxCheck isKindOfClass:[NSNumber class]] &&
|
||||
![enableSandboxCheck boolValue]) {
|
||||
return NO;
|
||||
}
|
||||
|
||||
NSURL *appStoreReceiptURL = [NSBundle mainBundle].appStoreReceiptURL;
|
||||
NSString *appStoreReceiptFileName = appStoreReceiptURL.lastPathComponent;
|
||||
return [appStoreReceiptFileName isEqualToString:kFIRAIdentitySandboxReceiptFileName];
|
||||
}
|
||||
|
||||
+ (BOOL)hasEmbeddedMobileProvision {
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
return [[NSBundle mainBundle] pathForResource:@"embedded" ofType:@"mobileprovision"].length > 0;
|
||||
#elif TARGET_OS_OSX
|
||||
return NO;
|
||||
#endif
|
||||
}
|
||||
|
||||
+ (BOOL)isSimulator {
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
NSString *platform = [FIRAppEnvironmentUtil deviceModel];
|
||||
return [platform isEqual:@"x86_64"] || [platform isEqual:@"i386"];
|
||||
#elif TARGET_OS_OSX
|
||||
return NO;
|
||||
#endif
|
||||
}
|
||||
|
||||
+ (NSString *)deviceModel {
|
||||
static dispatch_once_t once;
|
||||
static NSString *deviceModel;
|
||||
|
||||
dispatch_once(&once, ^{
|
||||
struct utsname systemInfo;
|
||||
if (uname(&systemInfo) == 0) {
|
||||
deviceModel = [NSString stringWithUTF8String:systemInfo.machine];
|
||||
}
|
||||
});
|
||||
return deviceModel;
|
||||
}
|
||||
|
||||
+ (NSString *)systemVersion {
|
||||
// Assemble the systemVersion, excluding the patch version if it's 0.
|
||||
NSOperatingSystemVersion osVersion = [NSProcessInfo processInfo].operatingSystemVersion;
|
||||
NSMutableString *versionString = [[NSMutableString alloc]
|
||||
initWithFormat:@"%ld.%ld", (long)osVersion.majorVersion, (long)osVersion.minorVersion];
|
||||
if (osVersion.patchVersion != 0) {
|
||||
[versionString appendFormat:@".%ld", (long)osVersion.patchVersion];
|
||||
}
|
||||
|
||||
return versionString;
|
||||
}
|
||||
|
||||
+ (BOOL)isAppExtension {
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
// Documented by <a href="https://goo.gl/RRB2Up">Apple</a>
|
||||
BOOL appExtension = [[[NSBundle mainBundle] bundlePath] hasSuffix:@".appex"];
|
||||
return appExtension;
|
||||
#elif TARGET_OS_OSX
|
||||
return NO;
|
||||
#endif
|
||||
}
|
||||
|
||||
#pragma mark - Helper methods
|
||||
|
||||
+ (BOOL)hasSCInfoFolder {
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
NSString *bundlePath = [NSBundle mainBundle].bundlePath;
|
||||
NSString *scInfoPath = [bundlePath stringByAppendingPathComponent:@"SC_Info"];
|
||||
return [[NSFileManager defaultManager] fileExistsAtPath:scInfoPath];
|
||||
#elif TARGET_OS_OSX
|
||||
return NO;
|
||||
#endif
|
||||
}
|
||||
|
||||
@end
|
||||
202
Pods/FirebaseCore/LICENSE
generated
Normal file
202
Pods/FirebaseCore/LICENSE
generated
Normal file
@@ -0,0 +1,202 @@
|
||||
|
||||
Apache License
|
||||
Version 2.0, January 2004
|
||||
http://www.apache.org/licenses/
|
||||
|
||||
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||
|
||||
1. Definitions.
|
||||
|
||||
"License" shall mean the terms and conditions for use, reproduction,
|
||||
and distribution as defined by Sections 1 through 9 of this document.
|
||||
|
||||
"Licensor" shall mean the copyright owner or entity authorized by
|
||||
the copyright owner that is granting the License.
|
||||
|
||||
"Legal Entity" shall mean the union of the acting entity and all
|
||||
other entities that control, are controlled by, or are under common
|
||||
control with that entity. For the purposes of this definition,
|
||||
"control" means (i) the power, direct or indirect, to cause the
|
||||
direction or management of such entity, whether by contract or
|
||||
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||
|
||||
"You" (or "Your") shall mean an individual or Legal Entity
|
||||
exercising permissions granted by this License.
|
||||
|
||||
"Source" form shall mean the preferred form for making modifications,
|
||||
including but not limited to software source code, documentation
|
||||
source, and configuration files.
|
||||
|
||||
"Object" form shall mean any form resulting from mechanical
|
||||
transformation or translation of a Source form, including but
|
||||
not limited to compiled object code, generated documentation,
|
||||
and conversions to other media types.
|
||||
|
||||
"Work" shall mean the work of authorship, whether in Source or
|
||||
Object form, made available under the License, as indicated by a
|
||||
copyright notice that is included in or attached to the work
|
||||
(an example is provided in the Appendix below).
|
||||
|
||||
"Derivative Works" shall mean any work, whether in Source or Object
|
||||
form, that is based on (or derived from) the Work and for which the
|
||||
editorial revisions, annotations, elaborations, or other modifications
|
||||
represent, as a whole, an original work of authorship. For the purposes
|
||||
of this License, Derivative Works shall not include works that remain
|
||||
separable from, or merely link (or bind by name) to the interfaces of,
|
||||
the Work and Derivative Works thereof.
|
||||
|
||||
"Contribution" shall mean any work of authorship, including
|
||||
the original version of the Work and any modifications or additions
|
||||
to that Work or Derivative Works thereof, that is intentionally
|
||||
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||
or by an individual or Legal Entity authorized to submit on behalf of
|
||||
the copyright owner. For the purposes of this definition, "submitted"
|
||||
means any form of electronic, verbal, or written communication sent
|
||||
to the Licensor or its representatives, including but not limited to
|
||||
communication on electronic mailing lists, source code control systems,
|
||||
and issue tracking systems that are managed by, or on behalf of, the
|
||||
Licensor for the purpose of discussing and improving the Work, but
|
||||
excluding communication that is conspicuously marked or otherwise
|
||||
designated in writing by the copyright owner as "Not a Contribution."
|
||||
|
||||
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||
on behalf of whom a Contribution has been received by Licensor and
|
||||
subsequently incorporated within the Work.
|
||||
|
||||
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||
this License, each Contributor hereby grants to You a perpetual,
|
||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||
copyright license to reproduce, prepare Derivative Works of,
|
||||
publicly display, publicly perform, sublicense, and distribute the
|
||||
Work and such Derivative Works in Source or Object form.
|
||||
|
||||
3. Grant of Patent License. Subject to the terms and conditions of
|
||||
this License, each Contributor hereby grants to You a perpetual,
|
||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||
(except as stated in this section) patent license to make, have made,
|
||||
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||
where such license applies only to those patent claims licensable
|
||||
by such Contributor that are necessarily infringed by their
|
||||
Contribution(s) alone or by combination of their Contribution(s)
|
||||
with the Work to which such Contribution(s) was submitted. If You
|
||||
institute patent litigation against any entity (including a
|
||||
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||
or a Contribution incorporated within the Work constitutes direct
|
||||
or contributory patent infringement, then any patent licenses
|
||||
granted to You under this License for that Work shall terminate
|
||||
as of the date such litigation is filed.
|
||||
|
||||
4. Redistribution. You may reproduce and distribute copies of the
|
||||
Work or Derivative Works thereof in any medium, with or without
|
||||
modifications, and in Source or Object form, provided that You
|
||||
meet the following conditions:
|
||||
|
||||
(a) You must give any other recipients of the Work or
|
||||
Derivative Works a copy of this License; and
|
||||
|
||||
(b) You must cause any modified files to carry prominent notices
|
||||
stating that You changed the files; and
|
||||
|
||||
(c) You must retain, in the Source form of any Derivative Works
|
||||
that You distribute, all copyright, patent, trademark, and
|
||||
attribution notices from the Source form of the Work,
|
||||
excluding those notices that do not pertain to any part of
|
||||
the Derivative Works; and
|
||||
|
||||
(d) If the Work includes a "NOTICE" text file as part of its
|
||||
distribution, then any Derivative Works that You distribute must
|
||||
include a readable copy of the attribution notices contained
|
||||
within such NOTICE file, excluding those notices that do not
|
||||
pertain to any part of the Derivative Works, in at least one
|
||||
of the following places: within a NOTICE text file distributed
|
||||
as part of the Derivative Works; within the Source form or
|
||||
documentation, if provided along with the Derivative Works; or,
|
||||
within a display generated by the Derivative Works, if and
|
||||
wherever such third-party notices normally appear. The contents
|
||||
of the NOTICE file are for informational purposes only and
|
||||
do not modify the License. You may add Your own attribution
|
||||
notices within Derivative Works that You distribute, alongside
|
||||
or as an addendum to the NOTICE text from the Work, provided
|
||||
that such additional attribution notices cannot be construed
|
||||
as modifying the License.
|
||||
|
||||
You may add Your own copyright statement to Your modifications and
|
||||
may provide additional or different license terms and conditions
|
||||
for use, reproduction, or distribution of Your modifications, or
|
||||
for any such Derivative Works as a whole, provided Your use,
|
||||
reproduction, and distribution of the Work otherwise complies with
|
||||
the conditions stated in this License.
|
||||
|
||||
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||
any Contribution intentionally submitted for inclusion in the Work
|
||||
by You to the Licensor shall be under the terms and conditions of
|
||||
this License, without any additional terms or conditions.
|
||||
Notwithstanding the above, nothing herein shall supersede or modify
|
||||
the terms of any separate license agreement you may have executed
|
||||
with Licensor regarding such Contributions.
|
||||
|
||||
6. Trademarks. This License does not grant permission to use the trade
|
||||
names, trademarks, service marks, or product names of the Licensor,
|
||||
except as required for reasonable and customary use in describing the
|
||||
origin of the Work and reproducing the content of the NOTICE file.
|
||||
|
||||
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||
agreed to in writing, Licensor provides the Work (and each
|
||||
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||
implied, including, without limitation, any warranties or conditions
|
||||
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||
appropriateness of using or redistributing the Work and assume any
|
||||
risks associated with Your exercise of permissions under this License.
|
||||
|
||||
8. Limitation of Liability. In no event and under no legal theory,
|
||||
whether in tort (including negligence), contract, or otherwise,
|
||||
unless required by applicable law (such as deliberate and grossly
|
||||
negligent acts) or agreed to in writing, shall any Contributor be
|
||||
liable to You for damages, including any direct, indirect, special,
|
||||
incidental, or consequential damages of any character arising as a
|
||||
result of this License or out of the use or inability to use the
|
||||
Work (including but not limited to damages for loss of goodwill,
|
||||
work stoppage, computer failure or malfunction, or any and all
|
||||
other commercial damages or losses), even if such Contributor
|
||||
has been advised of the possibility of such damages.
|
||||
|
||||
9. Accepting Warranty or Additional Liability. While redistributing
|
||||
the Work or Derivative Works thereof, You may choose to offer,
|
||||
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||
or other liability obligations and/or rights consistent with this
|
||||
License. However, in accepting such obligations, You may act only
|
||||
on Your own behalf and on Your sole responsibility, not on behalf
|
||||
of any other Contributor, and only if You agree to indemnify,
|
||||
defend, and hold each Contributor harmless for any liability
|
||||
incurred by, or claims asserted against, such Contributor by reason
|
||||
of your accepting any such warranty or additional liability.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
APPENDIX: How to apply the Apache License to your work.
|
||||
|
||||
To apply the Apache License to your work, attach the following
|
||||
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||
replaced with your own identifying information. (Don't include
|
||||
the brackets!) The text should be enclosed in the appropriate
|
||||
comment syntax for the file format. We also recommend that a
|
||||
file or class name and description of purpose be included on the
|
||||
same "printed page" as the copyright notice for easier
|
||||
identification within third-party archives.
|
||||
|
||||
Copyright [yyyy] [name of copyright owner]
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
198
Pods/FirebaseCore/README.md
generated
Normal file
198
Pods/FirebaseCore/README.md
generated
Normal file
@@ -0,0 +1,198 @@
|
||||
# Firebase iOS Open Source Development [](https://travis-ci.org/firebase/firebase-ios-sdk)
|
||||
|
||||
This repository contains a subset of the Firebase iOS SDK source. It currently
|
||||
includes FirebaseCore, FirebaseAuth, FirebaseDatabase, FirebaseFirestore,
|
||||
FirebaseFunctions, FirebaseMessaging and FirebaseStorage.
|
||||
|
||||
Firebase is an app development platform with tools to help you build, grow and
|
||||
monetize your app. More information about Firebase can be found at
|
||||
[https://firebase.google.com](https://firebase.google.com).
|
||||
|
||||
## Installation
|
||||
|
||||
See the three subsections for details about three different installation methods.
|
||||
1. [Standard pod install](README.md#standard-pod-install)
|
||||
1. [Installing from the GitHub repo](README.md#installing-from-github)
|
||||
1. [Experimental Carthage](README.md#carthage-ios-only)
|
||||
|
||||
### Standard pod install
|
||||
|
||||
Go to
|
||||
[https://firebase.google.com/docs/ios/setup](https://firebase.google.com/docs/ios/setup).
|
||||
|
||||
### Installing from GitHub
|
||||
|
||||
For releases starting with 5.0.0, the source for each release is also deployed
|
||||
to CocoaPods master and available via standard
|
||||
[CocoaPods Podfile syntax](https://guides.cocoapods.org/syntax/podfile.html#pod).
|
||||
|
||||
These instructions can be used to access the Firebase repo at other branches,
|
||||
tags, or commits.
|
||||
|
||||
#### Background
|
||||
|
||||
See
|
||||
[the Podfile Syntax Reference](https://guides.cocoapods.org/syntax/podfile.html#pod)
|
||||
for instructions and options about overriding pod source locations.
|
||||
|
||||
#### Step-by-step Source Pod Installation Instructions
|
||||
|
||||
For iOS, copy a subset of the following lines to your Podfile:
|
||||
|
||||
```
|
||||
pod 'Firebase' # To enable Firebase module, with `@import Firebase` support
|
||||
pod 'FirebaseCore', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
pod 'FirebaseAuth', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
pod 'FirebaseDatabase', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
pod 'FirebaseFirestore', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
pod 'FirebaseFunctions', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
pod 'FirebaseMessaging', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
pod 'FirebaseStorage', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
```
|
||||
|
||||
For macOS and tvOS, copy a subset of the following:
|
||||
|
||||
```
|
||||
pod 'FirebaseCore', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
pod 'FirebaseAuth', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
pod 'FirebaseDatabase', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
pod 'FirebaseStorage', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :tag => '5.0.0'
|
||||
```
|
||||
|
||||
1. Make sure you have at least CocoaPods version 1.4.0 - `pod --version`.
|
||||
1. Delete pods for any components you don't need, except `FirebaseCore` must always be included.
|
||||
1. Update the tags to the latest Firebase release. See the
|
||||
[release notes](https://firebase.google.com/support/release-notes/ios).
|
||||
1. Run `pod update`.
|
||||
|
||||
#### Examples
|
||||
|
||||
To access FirebaseMessaging via a checked out version of the firebase-ios-sdk repo do:
|
||||
|
||||
```
|
||||
pod 'FirebaseMessaging', :path => '/path/to/firebase-ios-sdk'
|
||||
pod 'FirebaseCore', :path => '/path/to/firebase-ios-sdk'
|
||||
```
|
||||
To access via a branch:
|
||||
```
|
||||
pod 'FirebaseFirestore', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :branch => 'master'
|
||||
pod 'FirebaseCore', :git => 'https://github.com/firebase/firebase-ios-sdk.git', :branch => 'master'
|
||||
```
|
||||
|
||||
### Carthage (iOS only)
|
||||
|
||||
An experimental Carthage distribution is now available. See
|
||||
[Carthage](Carthage.md).
|
||||
|
||||
## Development
|
||||
|
||||
Follow the subsequent instructions to develop, debug, unit test, run integration
|
||||
tests, and try out reference samples:
|
||||
|
||||
```
|
||||
$ git clone git@github.com:firebase/firebase-ios-sdk.git
|
||||
$ cd firebase-ios-sdk/Example
|
||||
$ pod update
|
||||
$ open Firebase.xcworkspace
|
||||
```
|
||||
|
||||
Firestore and Functions have self contained Xcode projects. See
|
||||
[Firestore/README.md](Firestore/README.md) and
|
||||
[Functions/README.md](Functions/README.md).
|
||||
|
||||
### Running Unit Tests
|
||||
|
||||
Select a scheme and press Command-u to build a component and run its unit tests.
|
||||
|
||||
### Running Sample Apps
|
||||
In order to run the sample apps and integration tests, you'll need valid
|
||||
`GoogleService-Info.plist` files for those samples. The Firebase Xcode project contains dummy plist
|
||||
files without real values, but can be replaced with real plist files. To get your own
|
||||
`GoogleService-Info.plist` files:
|
||||
|
||||
1. Go to the [Firebase Console](https://console.firebase.google.com/)
|
||||
2. Create a new Firebase project, if you don't already have one
|
||||
3. For each sample app you want to test, create a new Firebase app with the sample app's bundle
|
||||
identifier (e.g. `com.google.Database-Example`)
|
||||
4. Download the resulting `GoogleService-Info.plist` and replace the appropriate dummy plist file
|
||||
(e.g. in [Example/Database/App/](Example/Database/App/));
|
||||
|
||||
Some sample apps like Firebase Messaging ([Example/Messaging/App](Example/Messaging/App)) require
|
||||
special Apple capabilities, and you will have to change the sample app to use a unique bundle
|
||||
identifier that you can control in your own Apple Developer account.
|
||||
|
||||
## Specific Component Instructions
|
||||
See the sections below for any special instructions for those components.
|
||||
|
||||
### Firebase Auth
|
||||
|
||||
If you're doing specific Firebase Auth development, see
|
||||
[AuthSamples/README.md](AuthSamples/README.md) for instructions about
|
||||
building and running the FirebaseAuth pod along with various samples and tests.
|
||||
|
||||
### Firebase Database
|
||||
|
||||
To run the Database Integration tests, make your database authentication rules
|
||||
[public](https://firebase.google.com/docs/database/security/quickstart).
|
||||
|
||||
### Firebase Storage
|
||||
|
||||
To run the Storage Integration tests, follow the instructions in
|
||||
[FIRStorageIntegrationTests.m](Example/Storage/Tests/Integration/FIRStorageIntegrationTests.m).
|
||||
|
||||
#### Push Notifications
|
||||
|
||||
Push notifications can only be delivered to specially provisioned App IDs in the developer portal.
|
||||
In order to actually test receiving push notifications, you will need to:
|
||||
|
||||
1. Change the bundle identifier of the sample app to something you own in your Apple Developer
|
||||
account, and enable that App ID for push notifications.
|
||||
2. You'll also need to
|
||||
[upload your APNs Provider Authentication Key or certificate to the Firebase Console](https://firebase.google.com/docs/cloud-messaging/ios/certs)
|
||||
at **Project Settings > Cloud Messaging > [Your Firebase App]**.
|
||||
3. Ensure your iOS device is added to your Apple Developer portal as a test device.
|
||||
|
||||
#### iOS Simulator
|
||||
|
||||
The iOS Simulator cannot register for remote notifications, and will not receive push notifications.
|
||||
In order to receive push notifications, you'll have to follow the steps above and run the app on a
|
||||
physical device.
|
||||
|
||||
## Community Supported Efforts
|
||||
|
||||
We've seen an amazing amount of interest and contributions to improve the Firebase SDKs, and we are
|
||||
very grateful! We'd like to empower as many developers as we can to be able to use Firebase and
|
||||
participate in the Firebase community.
|
||||
|
||||
### macOS and tvOS
|
||||
FirebaseAuth, FirebaseCore, FirebaseDatabase and FirebaseStorage now compile, run unit tests, and
|
||||
work on macOS and tvOS, thanks to contributions from the community. There are a few tweaks needed,
|
||||
like ensuring iOS-only, macOS-only, or tvOS-only code is correctly guarded with checks for
|
||||
`TARGET_OS_IOS`, `TARGET_OS_OSX` and `TARGET_OS_TV`.
|
||||
|
||||
For tvOS, checkout the [Sample](Example/tvOSSample).
|
||||
|
||||
Keep in mind that macOS and tvOS are not officially supported by Firebase, and this repository is
|
||||
actively developed primarily for iOS. While we can catch basic unit test issues with Travis, there
|
||||
may be some changes where the SDK no longer works as expected on macOS or tvOS. If you encounter
|
||||
this, please [file an issue](https://github.com/firebase/firebase-ios-sdk/issues).
|
||||
|
||||
For installation instructions, see [above](README.md#step-by-step-source-pod-installation-instructions).
|
||||
|
||||
## Roadmap
|
||||
|
||||
See [Roadmap](ROADMAP.md) for more about the Firebase iOS SDK Open Source
|
||||
plans and directions.
|
||||
|
||||
## Contributing
|
||||
|
||||
See [Contributing](CONTRIBUTING.md) for more information on contributing to the Firebase
|
||||
iOS SDK.
|
||||
|
||||
## License
|
||||
|
||||
The contents of this repository is licensed under the
|
||||
[Apache License, version 2.0](http://www.apache.org/licenses/LICENSE-2.0).
|
||||
|
||||
Your use of Firebase is governed by the
|
||||
[Terms of Service for Firebase Services](https://firebase.google.com/terms/).
|
||||
Reference in New Issue
Block a user