stash
This commit is contained in:
parent
2eb50d9d7d
commit
25ad4f8438
|
@ -203,26 +203,11 @@ android {
|
|||
dependencies {
|
||||
addUnimodulesDependencies()
|
||||
implementation "org.webkit:android-jsc:r241213"
|
||||
implementation project(':rn-extensions-share')
|
||||
implementation project(':rn-fetch-blob')
|
||||
implementation project(':react-native-document-picker')
|
||||
implementation project(':react-native-firebase')
|
||||
implementation project(':react-native-webview')
|
||||
implementation project(':react-native-orientation-locker')
|
||||
implementation project(':react-native-splash-screen')
|
||||
implementation project(':react-native-screens')
|
||||
implementation project(':react-native-action-sheet')
|
||||
implementation(project(":react-native-device-info"), {
|
||||
exclude group: "com.google.android.gms"
|
||||
})
|
||||
implementation project(':react-native-gesture-handler')
|
||||
implementation project(':react-native-image-crop-picker')
|
||||
implementation project(':react-native-localize')
|
||||
implementation project(':react-native-audio')
|
||||
implementation project(":reactnativekeyboardinput")
|
||||
implementation project(':react-native-video')
|
||||
implementation project(':react-native-vector-icons')
|
||||
implementation project(':react-native-fast-image')
|
||||
implementation project(':realm')
|
||||
implementation project(':reactnativenotifications')
|
||||
implementation fileTree(dir: "libs", include: ["*.jar"])
|
||||
|
|
|
@ -8,25 +8,14 @@ import com.facebook.hermes.reactexecutor.HermesExecutorFactory;
|
|||
import com.facebook.react.bridge.JavaScriptExecutorFactory;
|
||||
|
||||
import com.facebook.react.ReactApplication;
|
||||
import io.github.elyx0.reactnativedocumentpicker.DocumentPickerPackage;
|
||||
import io.invertase.firebase.RNFirebasePackage;
|
||||
import io.invertase.firebase.fabric.crashlytics.RNFirebaseCrashlyticsPackage;
|
||||
import io.invertase.firebase.analytics.RNFirebaseAnalyticsPackage;
|
||||
import io.invertase.firebase.perf.RNFirebasePerformancePackage;
|
||||
import com.reactnativecommunity.webview.RNCWebViewPackage;
|
||||
import org.wonday.orientation.OrientationPackage;
|
||||
import org.devio.rn.splashscreen.SplashScreenReactPackage;
|
||||
import com.facebook.react.ReactNativeHost;
|
||||
import com.facebook.react.ReactPackage;
|
||||
import com.facebook.react.shell.MainReactPackage;
|
||||
import com.facebook.soloader.SoLoader;
|
||||
|
||||
import com.reactcommunity.rnlocalize.RNLocalizePackage;
|
||||
import com.reactnative.ivpusic.imagepicker.PickerPackage;
|
||||
import com.brentvatne.react.ReactVideoPackage;
|
||||
import com.dylanvann.fastimage.FastImageViewPackage;
|
||||
import com.oblador.vectoricons.VectorIconsPackage;
|
||||
import com.rnim.rn.audio.ReactNativeAudioPackage;
|
||||
import com.wix.reactnativekeyboardinput.KeyboardInputPackage;
|
||||
import com.wix.reactnativenotifications.RNNotificationsPackage;
|
||||
import com.wix.reactnativenotifications.core.AppLaunchHelper;
|
||||
|
@ -34,13 +23,8 @@ import com.wix.reactnativenotifications.core.AppLifecycleFacade;
|
|||
import com.wix.reactnativenotifications.core.JsIOHelper;
|
||||
import com.wix.reactnativenotifications.core.notification.INotificationsApplication;
|
||||
import com.wix.reactnativenotifications.core.notification.IPushNotification;
|
||||
import com.swmansion.gesturehandler.react.RNGestureHandlerPackage;
|
||||
import com.learnium.RNDeviceInfo.RNDeviceInfo;
|
||||
import com.actionsheet.ActionSheetPackage;
|
||||
import io.realm.react.RealmReactPackage;
|
||||
import com.swmansion.rnscreens.RNScreensPackage;
|
||||
import chat.rocket.SharePackage;
|
||||
import com.RNFetchBlob.RNFetchBlobPackage;
|
||||
|
||||
import chat.rocket.reactnative.generated.BasePackageList;
|
||||
|
||||
|
|
|
@ -57,6 +57,4 @@ target 'ShareRocketChatRN' do
|
|||
pod 'DoubleConversion', :podspec => '../node_modules/react-native/third-party-podspecs/DoubleConversion.podspec'
|
||||
pod 'glog', :podspec => '../node_modules/react-native/third-party-podspecs/glog.podspec'
|
||||
pod 'Folly', :podspec => '../node_modules/react-native/third-party-podspecs/Folly.podspec'
|
||||
|
||||
use_native_modules!
|
||||
end
|
||||
end
|
||||
|
|
540
ios/Podfile.lock
540
ios/Podfile.lock
|
@ -1,178 +1,253 @@
|
|||
PODS:
|
||||
- boost-for-react-native (1.63.0)
|
||||
- Crashlytics (3.12.0):
|
||||
- Fabric (~> 1.9.0)
|
||||
- Crashlytics (3.13.2):
|
||||
- Fabric (~> 1.10.2)
|
||||
- DoubleConversion (1.1.6)
|
||||
- EXAppLoaderProvider (5.0.1)
|
||||
- EXConstants (5.0.1):
|
||||
- EXAppLoaderProvider (6.0.0)
|
||||
- EXConstants (6.0.0):
|
||||
- UMConstantsInterface
|
||||
- UMCore
|
||||
- EXFileSystem (5.0.1):
|
||||
- EXFileSystem (6.0.0):
|
||||
- UMCore
|
||||
- UMFileSystemInterface
|
||||
- EXHaptics (5.0.1):
|
||||
- EXHaptics (6.0.0):
|
||||
- UMCore
|
||||
- EXPermissions (5.0.1):
|
||||
- EXPermissions (6.0.0):
|
||||
- UMCore
|
||||
- UMPermissionsInterface
|
||||
- EXWebBrowser (5.0.3):
|
||||
- EXWebBrowser (6.0.0):
|
||||
- UMCore
|
||||
- Fabric (1.9.0)
|
||||
- Firebase/Core (5.20.2):
|
||||
- Fabric (1.10.2)
|
||||
- Firebase/Core (6.3.0):
|
||||
- Firebase/CoreOnly
|
||||
- FirebaseAnalytics (= 5.8.1)
|
||||
- Firebase/CoreOnly (5.20.2):
|
||||
- FirebaseCore (= 5.4.1)
|
||||
- Firebase/Performance (5.20.2):
|
||||
- Firebase/Core
|
||||
- FirebasePerformance (= 2.2.4)
|
||||
- FirebaseABTesting (2.0.0):
|
||||
- FirebaseCore (~> 5.0)
|
||||
- Protobuf (~> 3.5)
|
||||
- FirebaseAnalytics (5.8.1):
|
||||
- FirebaseCore (~> 5.4)
|
||||
- FirebaseInstanceID (~> 3.8)
|
||||
- GoogleAppMeasurement (= 5.8.1)
|
||||
- GoogleUtilities/AppDelegateSwizzler (~> 5.2)
|
||||
- GoogleUtilities/MethodSwizzler (~> 5.2)
|
||||
- GoogleUtilities/Network (~> 5.2)
|
||||
- "GoogleUtilities/NSData+zlib (~> 5.2)"
|
||||
- FirebaseAnalytics (= 6.0.2)
|
||||
- Firebase/CoreOnly (6.3.0):
|
||||
- FirebaseCore (= 6.0.3)
|
||||
- FirebaseAnalytics (6.0.2):
|
||||
- FirebaseCore (~> 6.0)
|
||||
- FirebaseInstanceID (~> 4.2)
|
||||
- GoogleAppMeasurement (= 6.0.2)
|
||||
- GoogleUtilities/AppDelegateSwizzler (~> 6.0)
|
||||
- GoogleUtilities/MethodSwizzler (~> 6.0)
|
||||
- GoogleUtilities/Network (~> 6.0)
|
||||
- "GoogleUtilities/NSData+zlib (~> 6.0)"
|
||||
- nanopb (~> 0.3)
|
||||
- FirebaseCore (5.4.1):
|
||||
- GoogleUtilities/Environment (~> 5.2)
|
||||
- GoogleUtilities/Logger (~> 5.2)
|
||||
- FirebaseInstanceID (3.8.1):
|
||||
- FirebaseCore (~> 5.2)
|
||||
- GoogleUtilities/Environment (~> 5.2)
|
||||
- GoogleUtilities/UserDefaults (~> 5.2)
|
||||
- FirebasePerformance (2.2.4):
|
||||
- FirebaseAnalytics (~> 5.8)
|
||||
- FirebaseInstanceID (~> 3.8)
|
||||
- FirebaseRemoteConfig (~> 3.1)
|
||||
- GoogleToolboxForMac/Logger (~> 2.1)
|
||||
- "GoogleToolboxForMac/NSData+zlib (~> 2.1)"
|
||||
- GoogleUtilities/ISASwizzler (~> 5.2)
|
||||
- GoogleUtilities/MethodSwizzler (~> 5.2)
|
||||
- GTMSessionFetcher/Core (~> 1.1)
|
||||
- Protobuf (~> 3.5)
|
||||
- FirebaseRemoteConfig (3.1.0):
|
||||
- FirebaseABTesting (~> 2.0)
|
||||
- FirebaseAnalytics (~> 5.3)
|
||||
- FirebaseCore (~> 5.1)
|
||||
- FirebaseInstanceID (~> 3.3)
|
||||
- GoogleUtilities/Environment (~> 5.2)
|
||||
- "GoogleUtilities/NSData+zlib (~> 5.2)"
|
||||
- Protobuf (~> 3.5)
|
||||
- FirebaseCore (6.0.3):
|
||||
- GoogleUtilities/Environment (~> 6.0)
|
||||
- GoogleUtilities/Logger (~> 6.0)
|
||||
- FirebaseInstanceID (4.2.0):
|
||||
- FirebaseCore (~> 6.0)
|
||||
- GoogleUtilities/Environment (~> 6.0)
|
||||
- GoogleUtilities/UserDefaults (~> 6.0)
|
||||
- Folly (2018.10.22.00):
|
||||
- boost-for-react-native
|
||||
- DoubleConversion
|
||||
- glog
|
||||
- glog (0.3.5)
|
||||
- GoogleAppMeasurement (5.8.1):
|
||||
- GoogleUtilities/AppDelegateSwizzler (~> 5.2)
|
||||
- GoogleUtilities/MethodSwizzler (~> 5.2)
|
||||
- GoogleUtilities/Network (~> 5.2)
|
||||
- "GoogleUtilities/NSData+zlib (~> 5.2)"
|
||||
- GoogleAppMeasurement (6.0.2):
|
||||
- GoogleUtilities/AppDelegateSwizzler (~> 6.0)
|
||||
- GoogleUtilities/MethodSwizzler (~> 6.0)
|
||||
- GoogleUtilities/Network (~> 6.0)
|
||||
- "GoogleUtilities/NSData+zlib (~> 6.0)"
|
||||
- nanopb (~> 0.3)
|
||||
- GoogleIDFASupport (3.14.0)
|
||||
- GoogleToolboxForMac/Defines (2.2.1)
|
||||
- GoogleToolboxForMac/Logger (2.2.1):
|
||||
- GoogleToolboxForMac/Defines (= 2.2.1)
|
||||
- "GoogleToolboxForMac/NSData+zlib (2.2.1)":
|
||||
- GoogleToolboxForMac/Defines (= 2.2.1)
|
||||
- GoogleUtilities/AppDelegateSwizzler (5.8.0):
|
||||
- GoogleUtilities/AppDelegateSwizzler (6.2.1):
|
||||
- GoogleUtilities/Environment
|
||||
- GoogleUtilities/Logger
|
||||
- GoogleUtilities/Network
|
||||
- GoogleUtilities/Environment (5.8.0)
|
||||
- GoogleUtilities/ISASwizzler (5.8.0)
|
||||
- GoogleUtilities/Logger (5.8.0):
|
||||
- GoogleUtilities/Environment (6.2.1)
|
||||
- GoogleUtilities/Logger (6.2.1):
|
||||
- GoogleUtilities/Environment
|
||||
- GoogleUtilities/MethodSwizzler (5.8.0):
|
||||
- GoogleUtilities/MethodSwizzler (6.2.1):
|
||||
- GoogleUtilities/Logger
|
||||
- GoogleUtilities/Network (5.8.0):
|
||||
- GoogleUtilities/Network (6.2.1):
|
||||
- GoogleUtilities/Logger
|
||||
- "GoogleUtilities/NSData+zlib"
|
||||
- GoogleUtilities/Reachability
|
||||
- "GoogleUtilities/NSData+zlib (5.8.0)"
|
||||
- GoogleUtilities/Reachability (5.8.0):
|
||||
- "GoogleUtilities/NSData+zlib (6.2.1)"
|
||||
- GoogleUtilities/Reachability (6.2.1):
|
||||
- GoogleUtilities/Logger
|
||||
- GoogleUtilities/UserDefaults (5.8.0):
|
||||
- GoogleUtilities/UserDefaults (6.2.1):
|
||||
- GoogleUtilities/Logger
|
||||
- GTMSessionFetcher/Core (1.2.2)
|
||||
- libwebp (1.0.2):
|
||||
- libwebp/core (= 1.0.2)
|
||||
- libwebp/dec (= 1.0.2)
|
||||
- libwebp/demux (= 1.0.2)
|
||||
- libwebp/dsp (= 1.0.2)
|
||||
- libwebp/enc (= 1.0.2)
|
||||
- libwebp/mux (= 1.0.2)
|
||||
- libwebp/utils (= 1.0.2)
|
||||
- libwebp/webp (= 1.0.2)
|
||||
- libwebp/core (1.0.2):
|
||||
- libwebp/webp
|
||||
- libwebp/dec (1.0.2):
|
||||
- libwebp/core
|
||||
- libwebp/demux (1.0.2):
|
||||
- libwebp/core
|
||||
- libwebp/dsp (1.0.2):
|
||||
- libwebp/core
|
||||
- libwebp/enc (1.0.2):
|
||||
- libwebp/core
|
||||
- libwebp/mux (1.0.2):
|
||||
- libwebp/core
|
||||
- libwebp/utils (1.0.2):
|
||||
- libwebp/core
|
||||
- libwebp/webp (1.0.2)
|
||||
- nanopb (0.3.901):
|
||||
- nanopb/decode (= 0.3.901)
|
||||
- nanopb/encode (= 0.3.901)
|
||||
- nanopb/decode (0.3.901)
|
||||
- nanopb/encode (0.3.901)
|
||||
- Protobuf (3.7.0)
|
||||
- QBImagePickerController (3.4.0)
|
||||
- React (0.59.8):
|
||||
- React/Core (= 0.59.8)
|
||||
- React (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-DevSupport (= 0.60.4)
|
||||
- React-RCTActionSheet (= 0.60.4)
|
||||
- React-RCTAnimation (= 0.60.4)
|
||||
- React-RCTBlob (= 0.60.4)
|
||||
- React-RCTImage (= 0.60.4)
|
||||
- React-RCTLinking (= 0.60.4)
|
||||
- React-RCTNetwork (= 0.60.4)
|
||||
- React-RCTSettings (= 0.60.4)
|
||||
- React-RCTText (= 0.60.4)
|
||||
- React-RCTVibration (= 0.60.4)
|
||||
- React-RCTWebSocket (= 0.60.4)
|
||||
- React-Core (0.60.4):
|
||||
- Folly (= 2018.10.22.00)
|
||||
- React-cxxreact (= 0.60.4)
|
||||
- React-jsiexecutor (= 0.60.4)
|
||||
- yoga (= 0.60.4.React)
|
||||
- React-cxxreact (0.60.4):
|
||||
- boost-for-react-native (= 1.63.0)
|
||||
- DoubleConversion
|
||||
- Folly (= 2018.10.22.00)
|
||||
- glog
|
||||
- React-jsinspector (= 0.60.4)
|
||||
- React-DevSupport (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTWebSocket (= 0.60.4)
|
||||
- React-fishhook (0.60.4)
|
||||
- React-jsi (0.60.4):
|
||||
- boost-for-react-native (= 1.63.0)
|
||||
- DoubleConversion
|
||||
- Folly (= 2018.10.22.00)
|
||||
- glog
|
||||
- React-jsi/Default (= 0.60.4)
|
||||
- React-jsi/Default (0.60.4):
|
||||
- boost-for-react-native (= 1.63.0)
|
||||
- DoubleConversion
|
||||
- Folly (= 2018.10.22.00)
|
||||
- glog
|
||||
- React-jsiexecutor (0.60.4):
|
||||
- DoubleConversion
|
||||
- Folly (= 2018.10.22.00)
|
||||
- glog
|
||||
- React-cxxreact (= 0.60.4)
|
||||
- React-jsi (= 0.60.4)
|
||||
- React-jsinspector (0.60.4)
|
||||
- react-native-document-picker (3.2.2):
|
||||
- React
|
||||
- react-native-orientation-locker (1.1.5):
|
||||
- react-native-keyboard-input (5.3.1):
|
||||
- React
|
||||
- react-native-notifications (1.2.6):
|
||||
- React
|
||||
- react-native-orientation-locker (1.1.6):
|
||||
- React
|
||||
- react-native-realm-path (1.2.11):
|
||||
- React
|
||||
- react-native-splash-screen (3.2.0):
|
||||
- React
|
||||
- react-native-webview (5.8.1):
|
||||
- react-native-video (5.0.0):
|
||||
- React
|
||||
- React/Core (0.59.8):
|
||||
- yoga (= 0.59.8.React)
|
||||
- React/fishhook (0.59.8)
|
||||
- React/RCTActionSheet (0.59.8):
|
||||
- React/Core
|
||||
- React/RCTAnimation (0.59.8):
|
||||
- React/Core
|
||||
- React/RCTBlob (0.59.8):
|
||||
- React/Core
|
||||
- React/RCTImage (0.59.8):
|
||||
- React/Core
|
||||
- React/RCTNetwork
|
||||
- React/RCTLinkingIOS (0.59.8):
|
||||
- React/Core
|
||||
- React/RCTNetwork (0.59.8):
|
||||
- React/Core
|
||||
- React/RCTSettings (0.59.8):
|
||||
- React/Core
|
||||
- React/RCTText (0.59.8):
|
||||
- React/Core
|
||||
- React/RCTVibration (0.59.8):
|
||||
- React/Core
|
||||
- React/RCTWebSocket (0.59.8):
|
||||
- React/Core
|
||||
- React/fishhook
|
||||
- React/RCTBlob
|
||||
- RNDeviceInfo (1.6.1):
|
||||
- react-native-video/Video (= 5.0.0)
|
||||
- react-native-video/Video (5.0.0):
|
||||
- React
|
||||
- RNImageCropPicker (0.24.1):
|
||||
- react-native-webview (5.12.1):
|
||||
- React
|
||||
- React-RCTActionSheet (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTAnimation (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTBlob (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTNetwork (= 0.60.4)
|
||||
- React-RCTWebSocket (= 0.60.4)
|
||||
- React-RCTImage (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTNetwork (= 0.60.4)
|
||||
- React-RCTLinking (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTNetwork (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTSettings (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTText (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTVibration (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-RCTWebSocket (0.60.4):
|
||||
- React-Core (= 0.60.4)
|
||||
- React-fishhook (= 0.60.4)
|
||||
- rn-extensions-share (2.3.10):
|
||||
- React
|
||||
- rn-fetch-blob (0.10.16):
|
||||
- React-Core
|
||||
- RNAudio (4.3.0):
|
||||
- React
|
||||
- RNDeviceInfo (2.3.2):
|
||||
- React
|
||||
- RNFastImage (7.0.2):
|
||||
- React
|
||||
- SDWebImage (~> 5.0)
|
||||
- SDWebImageWebPCoder (~> 0.2.3)
|
||||
- RNFirebase (5.5.6):
|
||||
- Firebase/Core
|
||||
- React
|
||||
- RNFirebase/Crashlytics (= 5.5.6)
|
||||
- RNFirebase/Crashlytics (5.5.6):
|
||||
- Crashlytics
|
||||
- Fabric
|
||||
- Firebase/Core
|
||||
- React
|
||||
- RNGestureHandler (1.3.0):
|
||||
- React
|
||||
- RNImageCropPicker (0.25.0):
|
||||
- QBImagePickerController
|
||||
- React/Core
|
||||
- React-Core
|
||||
- React-RCTImage
|
||||
- RSKImageCropper
|
||||
- RNLocalize (1.1.4):
|
||||
- React
|
||||
- RNScreens (1.0.0-alpha.22):
|
||||
- RNScreens (1.0.0-alpha.23):
|
||||
- React
|
||||
- RNUserDefaults (1.3.5):
|
||||
- React
|
||||
- RNVectorIcons (6.4.2):
|
||||
- React
|
||||
- RSKImageCropper (2.2.1)
|
||||
- UMBarCodeScannerInterface (2.0.1)
|
||||
- UMCameraInterface (2.0.1)
|
||||
- UMConstantsInterface (2.0.1)
|
||||
- UMCore (2.0.1)
|
||||
- UMFaceDetectorInterface (2.0.1)
|
||||
- UMFileSystemInterface (2.0.1)
|
||||
- UMFontInterface (2.0.1)
|
||||
- UMImageLoaderInterface (2.0.1)
|
||||
- UMPermissionsInterface (2.0.1)
|
||||
- UMReactNativeAdapter (2.0.1):
|
||||
- SDWebImage (5.0.6):
|
||||
- SDWebImage/Core (= 5.0.6)
|
||||
- SDWebImage/Core (5.0.6)
|
||||
- SDWebImageWebPCoder (0.2.3):
|
||||
- libwebp (~> 1.0)
|
||||
- SDWebImage/Core (~> 5.0)
|
||||
- UMBarCodeScannerInterface (3.0.0)
|
||||
- UMCameraInterface (3.0.0)
|
||||
- UMConstantsInterface (3.0.0)
|
||||
- UMCore (3.0.2)
|
||||
- UMFaceDetectorInterface (3.0.0)
|
||||
- UMFileSystemInterface (3.0.0)
|
||||
- UMFontInterface (3.0.0)
|
||||
- UMImageLoaderInterface (3.0.0)
|
||||
- UMPermissionsInterface (3.0.0)
|
||||
- UMReactNativeAdapter (3.0.0):
|
||||
- React
|
||||
- UMCore
|
||||
- UMFontInterface
|
||||
- UMSensorsInterface (2.0.1)
|
||||
- UMTaskManagerInterface (2.0.1)
|
||||
- yoga (0.59.8.React)
|
||||
- UMSensorsInterface (3.0.0)
|
||||
- UMTaskManagerInterface (3.0.0)
|
||||
- yoga (0.60.4.React)
|
||||
|
||||
DEPENDENCIES:
|
||||
- Crashlytics (~> 3.12.0)
|
||||
- DoubleConversion (from `../node_modules/react-native/third-party-podspecs/DoubleConversion.podspec`)
|
||||
- EXAppLoaderProvider (from `../node_modules/expo-app-loader-provider/ios`)
|
||||
- EXConstants (from `../node_modules/expo-constants/ios`)
|
||||
|
@ -180,30 +255,46 @@ DEPENDENCIES:
|
|||
- EXHaptics (from `../node_modules/expo-haptics/ios`)
|
||||
- EXPermissions (from `../node_modules/expo-permissions/ios`)
|
||||
- EXWebBrowser (from `../node_modules/expo-web-browser/ios`)
|
||||
- Fabric (~> 1.9.0)
|
||||
- Firebase/Core (~> 5.20.1)
|
||||
- Firebase/Performance (~> 5.20.1)
|
||||
- Folly (from `../node_modules/react-native/third-party-podspecs/Folly.podspec`)
|
||||
- glog (from `../node_modules/react-native/third-party-podspecs/glog.podspec`)
|
||||
- GoogleIDFASupport (~> 3.14.0)
|
||||
- React (from `../node_modules/react-native/`)
|
||||
- React-Core (from `../node_modules/react-native/React`)
|
||||
- React-cxxreact (from `../node_modules/react-native/ReactCommon/cxxreact`)
|
||||
- React-DevSupport (from `../node_modules/react-native/React`)
|
||||
- React-fishhook (from `../node_modules/react-native/Libraries/fishhook`)
|
||||
- React-jsi (from `../node_modules/react-native/ReactCommon/jsi`)
|
||||
- React-jsiexecutor (from `../node_modules/react-native/ReactCommon/jsiexecutor`)
|
||||
- React-jsinspector (from `../node_modules/react-native/ReactCommon/jsinspector`)
|
||||
- react-native-document-picker (from `../node_modules/react-native-document-picker`)
|
||||
- react-native-keyboard-input (from `../node_modules/react-native-keyboard-input`)
|
||||
- react-native-notifications (from `../node_modules/react-native-notifications`)
|
||||
- react-native-orientation-locker (from `../node_modules/react-native-orientation-locker`)
|
||||
- react-native-realm-path (from `../node_modules/react-native-realm-path`)
|
||||
- react-native-splash-screen (from `../node_modules/react-native-splash-screen`)
|
||||
- react-native-video (from `../node_modules/react-native-video`)
|
||||
- react-native-webview (from `../node_modules/react-native-webview`)
|
||||
- React/Core (from `../node_modules/react-native`)
|
||||
- React/RCTActionSheet (from `../node_modules/react-native`)
|
||||
- React/RCTAnimation (from `../node_modules/react-native`)
|
||||
- React/RCTImage (from `../node_modules/react-native`)
|
||||
- React/RCTLinkingIOS (from `../node_modules/react-native`)
|
||||
- React/RCTNetwork (from `../node_modules/react-native`)
|
||||
- React/RCTSettings (from `../node_modules/react-native`)
|
||||
- React/RCTText (from `../node_modules/react-native`)
|
||||
- React/RCTVibration (from `../node_modules/react-native`)
|
||||
- React/RCTWebSocket (from `../node_modules/react-native`)
|
||||
- React-RCTActionSheet (from `../node_modules/react-native/Libraries/ActionSheetIOS`)
|
||||
- React-RCTAnimation (from `../node_modules/react-native/Libraries/NativeAnimation`)
|
||||
- React-RCTBlob (from `../node_modules/react-native/Libraries/Blob`)
|
||||
- React-RCTImage (from `../node_modules/react-native/Libraries/Image`)
|
||||
- React-RCTLinking (from `../node_modules/react-native/Libraries/LinkingIOS`)
|
||||
- React-RCTNetwork (from `../node_modules/react-native/Libraries/Network`)
|
||||
- React-RCTSettings (from `../node_modules/react-native/Libraries/Settings`)
|
||||
- React-RCTText (from `../node_modules/react-native/Libraries/Text`)
|
||||
- React-RCTVibration (from `../node_modules/react-native/Libraries/Vibration`)
|
||||
- React-RCTWebSocket (from `../node_modules/react-native/Libraries/WebSocket`)
|
||||
- rn-extensions-share (from `../node_modules/rn-extensions-share`)
|
||||
- rn-fetch-blob (from `../node_modules/rn-fetch-blob`)
|
||||
- RNAudio (from `../node_modules/react-native-audio`)
|
||||
- RNDeviceInfo (from `../node_modules/react-native-device-info`)
|
||||
- RNFastImage (from `../node_modules/react-native-fast-image`)
|
||||
- RNFirebase (from `../node_modules/react-native-firebase/ios`)
|
||||
- RNGestureHandler (from `../node_modules/react-native-gesture-handler`)
|
||||
- RNImageCropPicker (from `../node_modules/react-native-image-crop-picker`)
|
||||
- RNLocalize (from `../node_modules/react-native-localize`)
|
||||
- RNScreens (from `../node_modules/react-native-screens`)
|
||||
- RNUserDefaults (from `../node_modules/rn-user-defaults`)
|
||||
- RNVectorIcons (from `../node_modules/react-native-vector-icons`)
|
||||
- UMBarCodeScannerInterface (from `../node_modules/unimodules-barcode-scanner-interface/ios`)
|
||||
- UMCameraInterface (from `../node_modules/unimodules-camera-interface/ios`)
|
||||
- UMConstantsInterface (from `../node_modules/unimodules-constants-interface/ios`)
|
||||
|
@ -224,21 +315,17 @@ SPEC REPOS:
|
|||
- Crashlytics
|
||||
- Fabric
|
||||
- Firebase
|
||||
- FirebaseABTesting
|
||||
- FirebaseAnalytics
|
||||
- FirebaseCore
|
||||
- FirebaseInstanceID
|
||||
- FirebasePerformance
|
||||
- FirebaseRemoteConfig
|
||||
- GoogleAppMeasurement
|
||||
- GoogleIDFASupport
|
||||
- GoogleToolboxForMac
|
||||
- GoogleUtilities
|
||||
- GTMSessionFetcher
|
||||
- libwebp
|
||||
- nanopb
|
||||
- Protobuf
|
||||
- QBImagePickerController
|
||||
- RSKImageCropper
|
||||
- SDWebImage
|
||||
- SDWebImageWebPCoder
|
||||
|
||||
EXTERNAL SOURCES:
|
||||
DoubleConversion:
|
||||
|
@ -266,23 +353,81 @@ EXTERNAL SOURCES:
|
|||
glog:
|
||||
:podspec: "../node_modules/react-native/third-party-podspecs/glog.podspec"
|
||||
React:
|
||||
:path: "../node_modules/react-native"
|
||||
:path: "../node_modules/react-native/"
|
||||
React-Core:
|
||||
:path: "../node_modules/react-native/React"
|
||||
React-cxxreact:
|
||||
:path: "../node_modules/react-native/ReactCommon/cxxreact"
|
||||
React-DevSupport:
|
||||
:path: "../node_modules/react-native/React"
|
||||
React-fishhook:
|
||||
:path: "../node_modules/react-native/Libraries/fishhook"
|
||||
React-jsi:
|
||||
:path: "../node_modules/react-native/ReactCommon/jsi"
|
||||
React-jsiexecutor:
|
||||
:path: "../node_modules/react-native/ReactCommon/jsiexecutor"
|
||||
React-jsinspector:
|
||||
:path: "../node_modules/react-native/ReactCommon/jsinspector"
|
||||
react-native-document-picker:
|
||||
:path: "../node_modules/react-native-document-picker"
|
||||
react-native-keyboard-input:
|
||||
:path: "../node_modules/react-native-keyboard-input"
|
||||
react-native-notifications:
|
||||
:path: "../node_modules/react-native-notifications"
|
||||
react-native-orientation-locker:
|
||||
:path: "../node_modules/react-native-orientation-locker"
|
||||
react-native-realm-path:
|
||||
:path: "../node_modules/react-native-realm-path"
|
||||
react-native-splash-screen:
|
||||
:path: "../node_modules/react-native-splash-screen"
|
||||
react-native-video:
|
||||
:path: "../node_modules/react-native-video"
|
||||
react-native-webview:
|
||||
:path: "../node_modules/react-native-webview"
|
||||
React-RCTActionSheet:
|
||||
:path: "../node_modules/react-native/Libraries/ActionSheetIOS"
|
||||
React-RCTAnimation:
|
||||
:path: "../node_modules/react-native/Libraries/NativeAnimation"
|
||||
React-RCTBlob:
|
||||
:path: "../node_modules/react-native/Libraries/Blob"
|
||||
React-RCTImage:
|
||||
:path: "../node_modules/react-native/Libraries/Image"
|
||||
React-RCTLinking:
|
||||
:path: "../node_modules/react-native/Libraries/LinkingIOS"
|
||||
React-RCTNetwork:
|
||||
:path: "../node_modules/react-native/Libraries/Network"
|
||||
React-RCTSettings:
|
||||
:path: "../node_modules/react-native/Libraries/Settings"
|
||||
React-RCTText:
|
||||
:path: "../node_modules/react-native/Libraries/Text"
|
||||
React-RCTVibration:
|
||||
:path: "../node_modules/react-native/Libraries/Vibration"
|
||||
React-RCTWebSocket:
|
||||
:path: "../node_modules/react-native/Libraries/WebSocket"
|
||||
rn-extensions-share:
|
||||
:path: "../node_modules/rn-extensions-share"
|
||||
rn-fetch-blob:
|
||||
:path: "../node_modules/rn-fetch-blob"
|
||||
RNAudio:
|
||||
:path: "../node_modules/react-native-audio"
|
||||
RNDeviceInfo:
|
||||
:path: "../node_modules/react-native-device-info"
|
||||
RNFastImage:
|
||||
:path: "../node_modules/react-native-fast-image"
|
||||
RNFirebase:
|
||||
:path: "../node_modules/react-native-firebase/ios"
|
||||
RNGestureHandler:
|
||||
:path: "../node_modules/react-native-gesture-handler"
|
||||
RNImageCropPicker:
|
||||
:path: "../node_modules/react-native-image-crop-picker"
|
||||
RNLocalize:
|
||||
:path: "../node_modules/react-native-localize"
|
||||
RNScreens:
|
||||
:path: "../node_modules/react-native-screens"
|
||||
RNUserDefaults:
|
||||
:path: "../node_modules/rn-user-defaults"
|
||||
RNVectorIcons:
|
||||
:path: "../node_modules/react-native-vector-icons"
|
||||
UMBarCodeScannerInterface:
|
||||
:path: !ruby/object:Pathname
|
||||
path: "../node_modules/unimodules-barcode-scanner-interface/ios"
|
||||
|
@ -324,56 +469,81 @@ EXTERNAL SOURCES:
|
|||
|
||||
SPEC CHECKSUMS:
|
||||
boost-for-react-native: 39c7adb57c4e60d6c5479dd8623128eb5b3f0f2c
|
||||
Crashlytics: 07fb167b1694128c1c9a5a5cc319b0e9c3ca0933
|
||||
Crashlytics: 611738c7847f8291a1a51084e35987b86ba6b3ee
|
||||
DoubleConversion: bb338842f62ab1d708ceb63ec3d999f0f3d98ecd
|
||||
EXAppLoaderProvider: 8f2c04a0a8d9be91f7c37c2b8824077ee5d4bc01
|
||||
EXConstants: 9fe56eec8bf0a3ee9beb8f3381fa91340a5b1e57
|
||||
EXFileSystem: 96624bd4b93a0684335c421a6567a92b25bd7ddb
|
||||
EXHaptics: 6d594612afa321bfc8e167bdfb0b47f940f8f537
|
||||
EXPermissions: 8e05008ed4fc8c9be6c17ea95301fc3f3f005a7b
|
||||
EXWebBrowser: e03894b4583bb726e5ea05d01b341ba00134c2b5
|
||||
Fabric: f988e33c97f08930a413e08123064d2e5f68d655
|
||||
Firebase: 0c8cf33f266410c61ab3e2265cfa412200351d9c
|
||||
FirebaseABTesting: 1f50b8d50f5e3469eea54e7463a7b7fe221d1f5e
|
||||
FirebaseAnalytics: ece1aa57a4f43c64d53a648b5a5e05151aae947b
|
||||
FirebaseCore: f1a9a8be1aee4bf71a2fc0f4096df6788bdfda61
|
||||
FirebaseInstanceID: a122b0c258720cf250551bb2bedf48c699f80d90
|
||||
FirebasePerformance: 25ecee2a260bcf398d7f32d6f4804438df953100
|
||||
FirebaseRemoteConfig: 7e11c65f0769c09bff6947997c209515058c5318
|
||||
EXAppLoaderProvider: 7a8185228d8ba9e689a0e2d6d957fe9bdd49c8a0
|
||||
EXConstants: 5d81e84ca71b9a552529889cc798b4a04e9e22b3
|
||||
EXFileSystem: a5a7b8409e7d11f3948804e674ea91720fa84cd5
|
||||
EXHaptics: f84c93d605e0905c47654e4a6e5dfbff78ed6906
|
||||
EXPermissions: 99e52dc3e5f8e55153f1958004f6df2a30a1f2f5
|
||||
EXWebBrowser: def838b95aa9d396f9ce71ace4e614ee16e7ee30
|
||||
Fabric: 706c8b8098fff96c33c0db69cbf81f9c551d0d74
|
||||
Firebase: 8432d732974498afd5987e9001a05f90f1a3d625
|
||||
FirebaseAnalytics: 470ddab7253b21ad5a40bebd4a9903d7ae19386a
|
||||
FirebaseCore: 68f8a7f50cdae542715d4e86afa37c4067217dcb
|
||||
FirebaseInstanceID: f20243a1d828e0e9a3798b995174dedc16f1b32a
|
||||
Folly: de497beb10f102453a1afa9edbf8cf8a251890de
|
||||
glog: aefd1eb5dda2ab95ba0938556f34b98e2da3a60d
|
||||
GoogleAppMeasurement: ffe513e90551844a739e7bcbb1d2aca1c28a4338
|
||||
GoogleIDFASupport: aaf8c10bd429abb1c15349d5252244f5eda8ead1
|
||||
GoogleToolboxForMac: b3553629623a3b1bff17f555e736cd5a6d95ad55
|
||||
GoogleUtilities: 04fce34bcd5620c1ee76fb79172105c74a4df335
|
||||
GTMSessionFetcher: 61bb0f61a4cb560030f1222021178008a5727a23
|
||||
GoogleAppMeasurement: a35a645835bae31b6bdc0576396bc23908f12a22
|
||||
GoogleUtilities: c7a0b08bda3bf808be823ed151f0e28ac6866e71
|
||||
libwebp: b068a3bd7c45f7460f6715be7bed1a18fd5d6b48
|
||||
nanopb: 2901f78ea1b7b4015c860c2fdd1ea2fee1a18d48
|
||||
Protobuf: 7a877b7f3e5964e3fce995e2eb323dbc6831bb5a
|
||||
QBImagePickerController: d54cf93db6decf26baf6ed3472f336ef35cae022
|
||||
React: 76e6aa2b87d05eb6cccb6926d72685c9a07df152
|
||||
React: ff7ee2ae5ee1c1d9ae2183b4111045b25294bb01
|
||||
React-Core: 8e0ea421cae5609d2562850f98421b15030476fa
|
||||
React-cxxreact: 326880209990151a7182a813311054e9772ba510
|
||||
React-DevSupport: e9f10e6721e78e87622fc985db695c0c0168db8a
|
||||
React-fishhook: 1f0e5b08449403fa75c3fb3881a0beefbada14af
|
||||
React-jsi: 21d3153b1153fbf6510a92b6b11e33e725cb7432
|
||||
React-jsiexecutor: 7549641e48bafae7bfee3f3ea19bf4901639c5de
|
||||
React-jsinspector: 73f24a02fa684ed6a2b828ba116874a2191ded88
|
||||
react-native-document-picker: 94a07ce0494c559e2ae9fa86621d6c624d810fec
|
||||
react-native-orientation-locker: 132a63bab4dddd2a5709f6f7935ad9676b0af7c5
|
||||
react-native-keyboard-input: 2a01e0aceac330592bbe9b3101761bb9d8e6d1fb
|
||||
react-native-notifications: dddb6bee55ca4ab303c17c27c75fb5955784c4b3
|
||||
react-native-orientation-locker: 23918c400376a7043e752c639c122fcf6bce8f1c
|
||||
react-native-realm-path: 868473ea0bc4629850f1ec51a70d81055c06d091
|
||||
react-native-splash-screen: 200d11d188e2e78cea3ad319964f6142b6384865
|
||||
react-native-webview: f3e28b48461c78db833f727feec08b13285e7b61
|
||||
RNDeviceInfo: 958a1ed6f94e04557b865b8ef848cfc83db0ebba
|
||||
RNImageCropPicker: 6134b66a3d5bc13e2895a97c630a4254006902b4
|
||||
react-native-video: 6555881252c8ca039760e1cd6df28ac28ffb2baf
|
||||
react-native-webview: 5036ed4a973ad1deeeff118262d2df7b60b3419d
|
||||
React-RCTActionSheet: 9f71d7ae3e8fb10e08d162cbf14c621349dbfab3
|
||||
React-RCTAnimation: 981d8c95b0e30918a9832ccac32af83562a27fae
|
||||
React-RCTBlob: 21e73d1020a302a75fed30dbaee9f15287b80baa
|
||||
React-RCTImage: c0bc6ac0926517b6fb7e4c279b04843113e99d1d
|
||||
React-RCTLinking: 1af3f3c59114bed3deec0107c62e7efad0932ee5
|
||||
React-RCTNetwork: 35df9de46e19cda5c56380be1a7759b9b8cb2fcd
|
||||
React-RCTSettings: f580504c2cd1f44e25add10fb9ed3954f67f8ac5
|
||||
React-RCTText: e0f224898b13af9aa036ea7cb3d438daa68c1044
|
||||
React-RCTVibration: 0bea40cd51bd089bd591a8f74c86e91fdf2666c5
|
||||
React-RCTWebSocket: 163873f4cdd5f1058a9483443404fc3801581cb6
|
||||
rn-extensions-share: 4bfee75806ad54aadeff1dfa535697a6345a50b8
|
||||
rn-fetch-blob: 651b8d076b43d0d7aa294a3d9ec16c00aab8bef9
|
||||
RNAudio: cae2991f2dccb75163f260b60da8051717b959fa
|
||||
RNDeviceInfo: 17e34f6dd902f08d88cbe2c0b7a01be948d43641
|
||||
RNFastImage: 9b0c22643872bb7494c8d87bbbb66cc4c0d9e7a2
|
||||
RNFirebase: ac0de8b24c6f91ae9459575491ed6a77327619c6
|
||||
RNGestureHandler: 5329a942fce3d41c68b84c2c2276ce06a696d8b0
|
||||
RNImageCropPicker: 0a731d984e64ee4c28bddaa7ce52262e4b80979f
|
||||
RNLocalize: 62a949d2ec5bee0eb8f39a80a48f01e2f4f67080
|
||||
RNScreens: 720a9e6968beb73e8196239801e887d8401f86ed
|
||||
RNScreens: f28b48b8345f2f5f39ed6195518291515032a788
|
||||
RNUserDefaults: 8a4928443510aa99e4ccb3b53f1bf186593d690b
|
||||
RNVectorIcons: 6607bd3a30291d0edb56f9bbe7ae411ee2b928b0
|
||||
RSKImageCropper: 98296ad26b41753f796b6898d015509598f13d97
|
||||
UMBarCodeScannerInterface: d5602e23de37f95bb4ee49ee3b2711e128058ae9
|
||||
UMCameraInterface: dde8491778ed062348e569bad33a890e60c32c9d
|
||||
UMConstantsInterface: de48a63a5af572fc4dcc0e68051b00503b83e301
|
||||
UMCore: 047dc01ae4ccdd0c993f2c190f2489e5409c3ad0
|
||||
UMFaceDetectorInterface: badd9e3d206f5ba254c85a26afa43da06638575f
|
||||
UMFileSystemInterface: ff9a18c26ee6321dc21a3f9663efe3a55313d4db
|
||||
UMFontInterface: 0575f33184974a38f3528a4750729c7f5256b848
|
||||
UMImageLoaderInterface: ee8642347161d66272e841377a888957feb1f48e
|
||||
UMPermissionsInterface: 2238fe9d7f99457a5cfe7f3140c2521c5bf453a6
|
||||
UMReactNativeAdapter: 110be971ff044f8cfd37cbf565a264cd79858391
|
||||
UMSensorsInterface: cda3ec177c7ff0a138e3135414b4a29013389358
|
||||
UMTaskManagerInterface: 296793ab2a7e181fe5ebe2ba9b40ae208ab4b8fa
|
||||
yoga: 92b2102c3d373d1a790db4ab761d2b0ffc634f64
|
||||
SDWebImage: 920f1a2ff1ca8296ad34f6e0510a1ef1d70ac965
|
||||
SDWebImageWebPCoder: 7568737603c50f6237850afedd7e9e28e5917e6b
|
||||
UMBarCodeScannerInterface: 84ea2d6b58ff0dc27ef9b68bab71286be18ee020
|
||||
UMCameraInterface: 26b26005d1756a0d5f4f04f1e168e39ea9154535
|
||||
UMConstantsInterface: 038bacb19de12b6fd328c589122c8dc977cccf61
|
||||
UMCore: 733094f43f7244c60ce1f0592d00013ed68fa52c
|
||||
UMFaceDetectorInterface: c9c3ae4cb045421283667a1698c2f31331f55e3f
|
||||
UMFileSystemInterface: e9adc71027017de38eaf7d05fa58b2848ecb3797
|
||||
UMFontInterface: f0c5846977ee8a93d7cfa8ae7e666772c727d195
|
||||
UMImageLoaderInterface: 36e54e570acc4d720856f03ceebc441f73ea472c
|
||||
UMPermissionsInterface: 938d010c74c43fcefc9bb990633a7c5a1631267e
|
||||
UMReactNativeAdapter: 131ea2b944ade8035f0b54c6570c405f6000548d
|
||||
UMSensorsInterface: 0ed023ce9b96f2ca6fada7bda05b7760da60b293
|
||||
UMTaskManagerInterface: 8664abd37a00715727e60df9ecd65e42ba47b548
|
||||
yoga: c2c050f6ae6e222534760cc82f559b89214b67e2
|
||||
|
||||
PODFILE CHECKSUM: 7875ef440f1c2fb6583ee808c642c97d1410140d
|
||||
PODFILE CHECKSUM: 350314aa14736c453f620aed1b4ab2eefbde5b81
|
||||
|
||||
COCOAPODS: 1.6.2
|
||||
|
|
|
@ -1,39 +1,23 @@
|
|||

|
||||
|
||||
Part of [Google Fabric](https://get.fabric.io), [Crashlytics](http://try.crashlytics.com/) offers the most powerful, yet lightest weight crash reporting solution for iOS. Crashlytics also provides real-time analytics through [Answers](https://answers.io/) and app distributions to testers using [Beta](http://try.crashlytics.com/beta/).
|
||||
# Crashlytics
|
||||
|
||||
## Overview
|
||||
|
||||
[Crashlytics](https://firebase.google.com/docs/crashlytics/get-started?platform=ios) offers the most powerful, yet lightest weight crash reporting solution for iOS.
|
||||
|
||||
|
||||
## Setup
|
||||
|
||||
1. Visit [https://fabric.io/sign_up](https://fabric.io/sign_up) to create your Fabric account and to download Fabric.app.
|
||||
To start using Crashlytics, there are two options:
|
||||
|
||||
1. Open Fabric.app, login and select the Crashlytics SDK.
|
||||
1) The recommended way is to go to the [Firebase Crashlytics Docs](https://firebase.google.com/docs/crashlytics/get-started?platform=ios) and follow the directions there.
|
||||
|
||||

|
||||
2) If you aren't using Firebase yet, go to [Fabric Kits](https://fabric.io/kits), and follow the directions for Crashlytics.
|
||||
|
||||
1. The Fabric app automatically detects when a project uses CocoaPods and gives you the option to install via the Podfile or Xcode.
|
||||
|
||||

|
||||
|
||||
1. Select the Podfile option and follow the installation instructions to update your Podfile. **Note:** the Crashlytics Pod includes Answers. If you have Answers included as a separate Pod it should be removed from your Podfile to avoid duplicate symbol errors.
|
||||
|
||||
```
|
||||
pod 'Fabric'
|
||||
pod 'Crashlytics'
|
||||
```
|
||||
|
||||
1. Run `pod install`
|
||||
|
||||
1. Add a Run Script Build Phase and build your app.
|
||||
|
||||

|
||||
|
||||
1. Initialize the SDK by inserting code outlined in the Fabric.app.
|
||||
|
||||
1. Run your app to finish the installation.
|
||||
|
||||
## Resources
|
||||
|
||||
* [Documentation](https://docs.fabric.io/apple/crashlytics/overview.html)
|
||||
* [API Reference](https://firebase.google.com/docs/reference/ios/crashlytics/api/reference/Classes)
|
||||
* [Forums](https://stackoverflow.com/questions/tagged/google-fabric)
|
||||
* [Website](http://try.crashlytics.com/)
|
||||
* Follow us on Twitter: [@fabric](https://twitter.com/fabric) and [@crashlytics](https://twitter.com/crashlytics)
|
||||
* [Website](https://firebase.google.com/docs/crashlytics)
|
||||
* Follow us on Twitter: [@crashlytics](https://twitter.com/crashlytics)
|
||||
|
|
Binary file not shown.
Binary file not shown.
|
@ -3,26 +3,71 @@
|
|||
# run
|
||||
#
|
||||
# Copyright (c) 2015 Crashlytics. All rights reserved.
|
||||
#
|
||||
#
|
||||
# This script is meant to be run as a Run Script in the "Build Phases" section
|
||||
# of your Xcode project. It sends debug symbols to symbolicate stacktraces,
|
||||
# sends build events to track versions, and onboard apps for Crashlytics.
|
||||
#
|
||||
# This script calls upload-symbols twice:
|
||||
#
|
||||
# 1) First it calls upload-symbols synchronously in "validation" mode. If the
|
||||
# script finds issues with the build environment, it will report errors to Xcode.
|
||||
# In validation mode it exits before doing any time consuming work.
|
||||
#
|
||||
# 2) Then it calls upload-symbols in the background to actually send the build
|
||||
# event and upload symbols. It does this in the background so that it doesn't
|
||||
# slow down your builds. If an error happens here, you won't see it in Xcode.
|
||||
#
|
||||
# You can find the output for the background execution in Console.app, by
|
||||
# searching for "upload-symbols".
|
||||
#
|
||||
# If you want verbose output, you can pass the --debug flag to this script
|
||||
#
|
||||
|
||||
# Figure out where we're being called from
|
||||
DIR=$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )
|
||||
|
||||
# Quote path in case of spaces or special chars
|
||||
DIR="\"${DIR}"
|
||||
# If the first argument is specified without a dash, treat it as the Fabric API
|
||||
# Key and add it as an argument
|
||||
if [ -z "$1" ] || [[ $1 == -* ]]; then
|
||||
API_KEY_ARG=""
|
||||
else
|
||||
API_KEY_ARG="-a $1"; shift
|
||||
fi
|
||||
|
||||
PATH_SEP="/"
|
||||
VALIDATE_COMMAND="uploadDSYM\" $@ validate run-script"
|
||||
UPLOAD_COMMAND="uploadDSYM\" $@ run-script"
|
||||
# If a second argument is specified without a dash, treat it as the Build Secret
|
||||
# and add it as an argument
|
||||
if [ -z "$1" ] || [[ $1 == -* ]]; then
|
||||
BUILD_SECRET_ARG=""
|
||||
else
|
||||
BUILD_SECRET_ARG="-bs $1"; shift
|
||||
fi
|
||||
|
||||
# Ensure params are as expected, run in sync mode to validate
|
||||
eval $DIR$PATH_SEP$VALIDATE_COMMAND
|
||||
# Build up the arguments list, passing through any flags added after the
|
||||
# API Key and Build Secret
|
||||
ARGUMENTS="$API_KEY_ARG $BUILD_SECRET_ARG $@"
|
||||
VALIDATE_ARGUMENTS="$ARGUMENTS --build-phase --validate"
|
||||
UPLOAD_ARGUMENTS="$ARGUMENTS --build-phase"
|
||||
|
||||
# Quote the path to handle folders with special characters
|
||||
COMMAND_PATH="\"$DIR/upload-symbols\" "
|
||||
|
||||
# Ensure params are as expected, run in sync mode to validate,
|
||||
# and cause a build error if validation fails
|
||||
eval $COMMAND_PATH$VALIDATE_ARGUMENTS
|
||||
return_code=$?
|
||||
|
||||
if [[ $return_code != 0 ]]; then
|
||||
exit $return_code
|
||||
fi
|
||||
|
||||
# Verification passed, upload dSYM in background to prevent Xcode from waiting
|
||||
# Note: Validation is performed again before upload.
|
||||
# Output can still be found in Console.app
|
||||
eval $DIR$PATH_SEP$UPLOAD_COMMAND > /dev/null 2>&1 &
|
||||
# Verification passed, convert and upload cSYMs in the background to prevent
|
||||
# build delays
|
||||
#
|
||||
# Note: Validation is performed again at this step before upload
|
||||
#
|
||||
# Note: Output can still be found in Console.app, by searching for
|
||||
# "upload-symbols"
|
||||
#
|
||||
eval $COMMAND_PATH$UPLOAD_ARGUMENTS > /dev/null 2>&1 &
|
||||
|
|
BIN
ios/Pods/Crashlytics/iOS/Crashlytics.framework/upload-symbols
generated
vendored
Executable file
BIN
ios/Pods/Crashlytics/iOS/Crashlytics.framework/upload-symbols
generated
vendored
Executable file
Binary file not shown.
Binary file not shown.
|
@ -1,38 +1,23 @@
|
|||

|
||||
|
||||
# Fabric
|
||||
|
||||
## Overview
|
||||
|
||||
[Fabric](https://get.fabric.io) provides developers with the tools they need to build the best apps. Developed and maintained by Google and the team that built Crashlytics, Fabric provides an easy way to manage all your SDKs so that you’ll never have to worry about tedious configurations or juggling different accounts. We let you get right into coding and building the next big app.
|
||||
[Fabric](https://get.fabric.io) provides developers with the tools they need to build the best apps. Developed and maintained by Google and the team that built Crashlytics.
|
||||
|
||||
For a full list of SDKs provided through Fabric visit [https://fabric.io/kits](https://fabric.io/kits).
|
||||
|
||||
To follow the migration to Firebase, check out the [Fabric Roadmap](https://get.fabric.io/roadmap).
|
||||
|
||||
For a full list of SDK provided through Fabric visit [https://fabric.io/kits](https://fabric.io/kits).
|
||||
|
||||
## Setup
|
||||
|
||||
The Fabric Pod is a dependency for all Fabric SDKs and is included when installing any Fabric related Pods. General setup instructions are shown below; however, these vary depending on the selected SDK.
|
||||
Fabric is a dependency for the Crashlytics SDK. To start using Crashlytics, there are two options:
|
||||
|
||||
1. Visit [https://fabric.io/sign_up](https://fabric.io/sign_up) to create your Fabric account and to download Fabric.app.
|
||||
1) The recommended way is to go to the [Firebase Crashlytics Docs](https://firebase.google.com/docs/crashlytics/get-started?platform=ios) and follow the directions there.
|
||||
|
||||
1. Open Fabric.app, login and select an SDK to install.
|
||||
2) If you aren't using Firebase yet, go to [Fabric Kits](https://fabric.io/kits), and follow the directions for Crashlytics.
|
||||
|
||||

|
||||
|
||||
1. The Fabric app automatically detects when a project uses CocoaPods and gives you the option to install via the Podfile or Xcode.
|
||||
|
||||

|
||||
|
||||
1. Select the Podfile option and follow the installation instructions to update your Podfile. Note: the example below is for the Crashlytics SDK. The instructions will vary based on the selected SDK.
|
||||
|
||||

|
||||
|
||||
1. Add a Run Script Build Phase and build your app.
|
||||
|
||||

|
||||
|
||||
1. Initialize the SDK by inserting code outlined in Fabric.app.
|
||||
|
||||
1. Run your app to finish the installation.
|
||||
|
||||
## Resources
|
||||
|
||||
|
|
Binary file not shown.
Binary file not shown.
|
@ -3,26 +3,71 @@
|
|||
# run
|
||||
#
|
||||
# Copyright (c) 2015 Crashlytics. All rights reserved.
|
||||
#
|
||||
#
|
||||
# This script is meant to be run as a Run Script in the "Build Phases" section
|
||||
# of your Xcode project. It sends debug symbols to symbolicate stacktraces,
|
||||
# sends build events to track versions, and onboard apps for Crashlytics.
|
||||
#
|
||||
# This script calls upload-symbols twice:
|
||||
#
|
||||
# 1) First it calls upload-symbols synchronously in "validation" mode. If the
|
||||
# script finds issues with the build environment, it will report errors to Xcode.
|
||||
# In validation mode it exits before doing any time consuming work.
|
||||
#
|
||||
# 2) Then it calls upload-symbols in the background to actually send the build
|
||||
# event and upload symbols. It does this in the background so that it doesn't
|
||||
# slow down your builds. If an error happens here, you won't see it in Xcode.
|
||||
#
|
||||
# You can find the output for the background execution in Console.app, by
|
||||
# searching for "upload-symbols".
|
||||
#
|
||||
# If you want verbose output, you can pass the --debug flag to this script
|
||||
#
|
||||
|
||||
# Figure out where we're being called from
|
||||
DIR=$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )
|
||||
|
||||
# Quote path in case of spaces or special chars
|
||||
DIR="\"${DIR}"
|
||||
# If the first argument is specified without a dash, treat it as the Fabric API
|
||||
# Key and add it as an argument
|
||||
if [ -z "$1" ] || [[ $1 == -* ]]; then
|
||||
API_KEY_ARG=""
|
||||
else
|
||||
API_KEY_ARG="-a $1"; shift
|
||||
fi
|
||||
|
||||
PATH_SEP="/"
|
||||
VALIDATE_COMMAND="uploadDSYM\" $@ validate run-script"
|
||||
UPLOAD_COMMAND="uploadDSYM\" $@ run-script"
|
||||
# If a second argument is specified without a dash, treat it as the Build Secret
|
||||
# and add it as an argument
|
||||
if [ -z "$1" ] || [[ $1 == -* ]]; then
|
||||
BUILD_SECRET_ARG=""
|
||||
else
|
||||
BUILD_SECRET_ARG="-bs $1"; shift
|
||||
fi
|
||||
|
||||
# Ensure params are as expected, run in sync mode to validate
|
||||
eval $DIR$PATH_SEP$VALIDATE_COMMAND
|
||||
# Build up the arguments list, passing through any flags added after the
|
||||
# API Key and Build Secret
|
||||
ARGUMENTS="$API_KEY_ARG $BUILD_SECRET_ARG $@"
|
||||
VALIDATE_ARGUMENTS="$ARGUMENTS --build-phase --validate"
|
||||
UPLOAD_ARGUMENTS="$ARGUMENTS --build-phase"
|
||||
|
||||
# Quote the path to handle folders with special characters
|
||||
COMMAND_PATH="\"$DIR/upload-symbols\" "
|
||||
|
||||
# Ensure params are as expected, run in sync mode to validate,
|
||||
# and cause a build error if validation fails
|
||||
eval $COMMAND_PATH$VALIDATE_ARGUMENTS
|
||||
return_code=$?
|
||||
|
||||
if [[ $return_code != 0 ]]; then
|
||||
exit $return_code
|
||||
fi
|
||||
|
||||
# Verification passed, upload dSYM in background to prevent Xcode from waiting
|
||||
# Note: Validation is performed again before upload.
|
||||
# Output can still be found in Console.app
|
||||
eval $DIR$PATH_SEP$UPLOAD_COMMAND > /dev/null 2>&1 &
|
||||
# Verification passed, convert and upload cSYMs in the background to prevent
|
||||
# build delays
|
||||
#
|
||||
# Note: Validation is performed again at this step before upload
|
||||
#
|
||||
# Note: Output can still be found in Console.app, by searching for
|
||||
# "upload-symbols"
|
||||
#
|
||||
eval $COMMAND_PATH$UPLOAD_ARGUMENTS > /dev/null 2>&1 &
|
||||
|
|
Binary file not shown.
Binary file not shown.
|
@ -3,26 +3,71 @@
|
|||
# run
|
||||
#
|
||||
# Copyright (c) 2015 Crashlytics. All rights reserved.
|
||||
#
|
||||
#
|
||||
# This script is meant to be run as a Run Script in the "Build Phases" section
|
||||
# of your Xcode project. It sends debug symbols to symbolicate stacktraces,
|
||||
# sends build events to track versions, and onboard apps for Crashlytics.
|
||||
#
|
||||
# This script calls upload-symbols twice:
|
||||
#
|
||||
# 1) First it calls upload-symbols synchronously in "validation" mode. If the
|
||||
# script finds issues with the build environment, it will report errors to Xcode.
|
||||
# In validation mode it exits before doing any time consuming work.
|
||||
#
|
||||
# 2) Then it calls upload-symbols in the background to actually send the build
|
||||
# event and upload symbols. It does this in the background so that it doesn't
|
||||
# slow down your builds. If an error happens here, you won't see it in Xcode.
|
||||
#
|
||||
# You can find the output for the background execution in Console.app, by
|
||||
# searching for "upload-symbols".
|
||||
#
|
||||
# If you want verbose output, you can pass the --debug flag to this script
|
||||
#
|
||||
|
||||
# Figure out where we're being called from
|
||||
DIR=$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )
|
||||
|
||||
# Quote path in case of spaces or special chars
|
||||
DIR="\"${DIR}"
|
||||
# If the first argument is specified without a dash, treat it as the Fabric API
|
||||
# Key and add it as an argument
|
||||
if [ -z "$1" ] || [[ $1 == -* ]]; then
|
||||
API_KEY_ARG=""
|
||||
else
|
||||
API_KEY_ARG="-a $1"; shift
|
||||
fi
|
||||
|
||||
PATH_SEP="/"
|
||||
VALIDATE_COMMAND="uploadDSYM\" $@ validate run-script"
|
||||
UPLOAD_COMMAND="uploadDSYM\" $@ run-script"
|
||||
# If a second argument is specified without a dash, treat it as the Build Secret
|
||||
# and add it as an argument
|
||||
if [ -z "$1" ] || [[ $1 == -* ]]; then
|
||||
BUILD_SECRET_ARG=""
|
||||
else
|
||||
BUILD_SECRET_ARG="-bs $1"; shift
|
||||
fi
|
||||
|
||||
# Ensure params are as expected, run in sync mode to validate
|
||||
eval $DIR$PATH_SEP$VALIDATE_COMMAND
|
||||
# Build up the arguments list, passing through any flags added after the
|
||||
# API Key and Build Secret
|
||||
ARGUMENTS="$API_KEY_ARG $BUILD_SECRET_ARG $@"
|
||||
VALIDATE_ARGUMENTS="$ARGUMENTS --build-phase --validate"
|
||||
UPLOAD_ARGUMENTS="$ARGUMENTS --build-phase"
|
||||
|
||||
# Quote the path to handle folders with special characters
|
||||
COMMAND_PATH="\"$DIR/upload-symbols\" "
|
||||
|
||||
# Ensure params are as expected, run in sync mode to validate,
|
||||
# and cause a build error if validation fails
|
||||
eval $COMMAND_PATH$VALIDATE_ARGUMENTS
|
||||
return_code=$?
|
||||
|
||||
if [[ $return_code != 0 ]]; then
|
||||
exit $return_code
|
||||
fi
|
||||
|
||||
# Verification passed, upload dSYM in background to prevent Xcode from waiting
|
||||
# Note: Validation is performed again before upload.
|
||||
# Output can still be found in Console.app
|
||||
eval $DIR$PATH_SEP$UPLOAD_COMMAND > /dev/null 2>&1 &
|
||||
# Verification passed, convert and upload cSYMs in the background to prevent
|
||||
# build delays
|
||||
#
|
||||
# Note: Validation is performed again at this step before upload
|
||||
#
|
||||
# Note: Output can still be found in Console.app, by searching for
|
||||
# "upload-symbols"
|
||||
#
|
||||
eval $COMMAND_PATH$UPLOAD_ARGUMENTS > /dev/null 2>&1 &
|
||||
|
|
Binary file not shown.
Binary file not shown.
|
@ -6,28 +6,25 @@
|
|||
#else
|
||||
#if __has_include(<FirebaseAnalytics/FirebaseAnalytics.h>)
|
||||
#import <FirebaseAnalytics/FirebaseAnalytics.h>
|
||||
#else
|
||||
#ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#warning "FirebaseAnalytics.framework is not included in your target. Please add \
|
||||
`Firebase/Core` to your Podfile or add FirebaseAnalytics.framework to your project to ensure \
|
||||
Firebase services work as intended."
|
||||
#endif // #ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseAuth/FirebaseAuth.h>)
|
||||
#import <FirebaseAuth/FirebaseAuth.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseCrash/FirebaseCrash.h>)
|
||||
#import <FirebaseCrash/FirebaseCrash.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseDatabase/FirebaseDatabase.h>)
|
||||
#import <FirebaseDatabase/FirebaseDatabase.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseDynamicLinks/FirebaseDynamicLinks.h>)
|
||||
#import <FirebaseDynamicLinks/FirebaseDynamicLinks.h>
|
||||
#if !__has_include(<FirebaseAnalytics/FirebaseAnalytics.h>)
|
||||
#ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#warning "FirebaseAnalytics.framework is not included in your target. Please add \
|
||||
`Firebase/Analytics` to your Podfile or add FirebaseAnalytics.framework to your project to ensure \
|
||||
Firebase Dynamic Links works as intended."
|
||||
#endif // #ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#endif
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseFirestore/FirebaseFirestore.h>)
|
||||
|
@ -40,19 +37,29 @@ Firebase services work as intended."
|
|||
|
||||
#if __has_include(<FirebaseInAppMessaging/FirebaseInAppMessaging.h>)
|
||||
#import <FirebaseInAppMessaging/FirebaseInAppMessaging.h>
|
||||
#if !__has_include(<FirebaseAnalytics/FirebaseAnalytics.h>)
|
||||
#ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#warning "FirebaseAnalytics.framework is not included in your target. Please add \
|
||||
`Firebase/Analytics` to your Podfile or add FirebaseAnalytics.framework to your project to ensure \
|
||||
Firebase In App Messaging works as intended."
|
||||
#endif // #ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#endif
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseInstanceID/FirebaseInstanceID.h>)
|
||||
#import <FirebaseInstanceID/FirebaseInstanceID.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseInvites/FirebaseInvites.h>)
|
||||
#import <FirebaseInvites/FirebaseInvites.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseMessaging/FirebaseMessaging.h>)
|
||||
#import <FirebaseMessaging/FirebaseMessaging.h>
|
||||
#endif
|
||||
#if !__has_include(<FirebaseAnalytics/FirebaseAnalytics.h>)
|
||||
#ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#warning "FirebaseAnalytics.framework is not included in your target. Please add \
|
||||
`Firebase/Analytics` to your Podfile or add FirebaseAnalytics.framework to your project to ensure \
|
||||
Firebase Messaging works as intended."
|
||||
#endif // #ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#endif
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseMLModelInterpreter/FirebaseMLModelInterpreter.h>)
|
||||
#import <FirebaseMLModelInterpreter/FirebaseMLModelInterpreter.h>
|
||||
|
@ -66,6 +73,10 @@ Firebase services work as intended."
|
|||
#import <FirebaseMLNLSmartReply/FirebaseMLNLSmartReply.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseMLNLTranslate/FirebaseMLNLTranslate.h>)
|
||||
#import <FirebaseMLNLTranslate/FirebaseMLNLTranslate.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseMLNaturalLanguage/FirebaseMLNaturalLanguage.h>)
|
||||
#import <FirebaseMLNaturalLanguage/FirebaseMLNaturalLanguage.h>
|
||||
#endif
|
||||
|
@ -74,6 +85,10 @@ Firebase services work as intended."
|
|||
#import <FirebaseMLVision/FirebaseMLVision.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseMLVisionAutoML/FirebaseMLVisionAutoML.h>)
|
||||
#import <FirebaseMLVisionAutoML/FirebaseMLVisionAutoML.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseMLVisionBarcodeModel/FirebaseMLVisionBarcodeModel.h>)
|
||||
#import <FirebaseMLVisionBarcodeModel/FirebaseMLVisionBarcodeModel.h>
|
||||
#endif
|
||||
|
@ -86,16 +101,34 @@ Firebase services work as intended."
|
|||
#import <FirebaseMLVisionLabelModel/FirebaseMLVisionLabelModel.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseMLVisionObjectDetection/FirebaseMLVisionObjectDetection.h>)
|
||||
#import <FirebaseMLVisionObjectDetection/FirebaseMLVisionObjectDetection.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseMLVisionTextModel/FirebaseMLVisionTextModel.h>)
|
||||
#import <FirebaseMLVisionTextModel/FirebaseMLVisionTextModel.h>
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebasePerformance/FirebasePerformance.h>)
|
||||
#import <FirebasePerformance/FirebasePerformance.h>
|
||||
#if !__has_include(<FirebaseAnalytics/FirebaseAnalytics.h>)
|
||||
#ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#warning "FirebaseAnalytics.framework is not included in your target. Please add \
|
||||
`Firebase/Analytics` to your Podfile or add FirebaseAnalytics.framework to your project to ensure \
|
||||
Firebase Performance works as intended."
|
||||
#endif // #ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#endif
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseRemoteConfig/FirebaseRemoteConfig.h>)
|
||||
#import <FirebaseRemoteConfig/FirebaseRemoteConfig.h>
|
||||
#if !__has_include(<FirebaseAnalytics/FirebaseAnalytics.h>)
|
||||
#ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#warning "FirebaseAnalytics.framework is not included in your target. Please add \
|
||||
`Firebase/Analytics` to your Podfile or add FirebaseAnalytics.framework to your project to ensure \
|
||||
Firebase Remote Config works as intended."
|
||||
#endif // #ifndef FIREBASE_ANALYTICS_SUPPRESS_WARNING
|
||||
#endif
|
||||
#endif
|
||||
|
||||
#if __has_include(<FirebaseStorage/FirebaseStorage.h>)
|
||||
|
|
|
@ -78,11 +78,14 @@ CocoaPods is used to install and manage dependencies in existing Xcode projects.
|
|||
pod 'Firebase/MLModelInterpreter'
|
||||
pod 'Firebase/MLNLLanguageID'
|
||||
pod 'Firebase/MLNLSmartReply'
|
||||
pod 'Firebase/MLNLTranslate'
|
||||
pod 'Firebase/MLNaturalLanguage'
|
||||
pod 'Firebase/MLVision'
|
||||
pod 'Firebase/MLVisionAutoML'
|
||||
pod 'Firebase/MLVisionBarcodeModel'
|
||||
pod 'Firebase/MLVisionFaceModel'
|
||||
pod 'Firebase/MLVisionLabelModel'
|
||||
pod 'Firebase/MLVisionObjectDetection'
|
||||
pod 'Firebase/MLVisionTextModel'
|
||||
pod 'Firebase/Performance'
|
||||
pod 'Firebase/RemoteConfig'
|
||||
|
|
|
@ -1,4 +0,0 @@
|
|||
Version 0.3.0
|
||||
======================================
|
||||
- Initial public beta release.
|
||||
|
Binary file not shown.
|
@ -1,48 +0,0 @@
|
|||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "developers/mobile/abt/proto/ExperimentPayload.pbobjc.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@class FIRLifecycleEvents;
|
||||
|
||||
/// The default experiment overflow policy, that is to discard the experiment with the oldest start
|
||||
/// time when users start the experiment on the web console.
|
||||
extern const ABTExperimentPayload_ExperimentOverflowPolicy FIRDefaultExperimentOverflowPolicy;
|
||||
|
||||
/// This class is for Firebase services to handle experiments updates to Firebase Analytics.
|
||||
/// Experiments can be set, cleared and updated through this controller.
|
||||
@interface FIRExperimentController : NSObject
|
||||
|
||||
/// Returns the FIRExperimentController singleton.
|
||||
+ (FIRExperimentController *)sharedInstance;
|
||||
|
||||
/// Updates the list of experiments. Experiments already existing in payloads are not affected,
|
||||
/// whose state and payload is preserved. This method compares whether the experiments have changed
|
||||
/// or not by their variant ID. This runs in a background queue.
|
||||
/// @param origin The originating service affected by the experiment, it is defined at
|
||||
/// Firebase Analytics FIREventOrigins.h.
|
||||
/// @param events A list of event names to be used for logging experiment lifecycle events,
|
||||
/// if they are not defined in the payload.
|
||||
/// @param policy The policy to handle new experiments when slots are full.
|
||||
/// @param lastStartTime The last known experiment start timestamp for this affected service.
|
||||
/// (Timestamps are specified by the number of seconds from 00:00:00 UTC on 1
|
||||
/// January 1970.).
|
||||
/// @param payloads List of experiment metadata.
|
||||
- (void)updateExperimentsWithServiceOrigin:(NSString *)origin
|
||||
events:(FIRLifecycleEvents *)events
|
||||
policy:(ABTExperimentPayload_ExperimentOverflowPolicy)policy
|
||||
lastStartTime:(NSTimeInterval)lastStartTime
|
||||
payloads:(NSArray<NSData *> *)payloads;
|
||||
|
||||
/// Returns the latest experiment start timestamp given a current latest timestamp and a list of
|
||||
/// experiment payloads. Timestamps are specified by the number of seconds from 00:00:00 UTC on 1
|
||||
/// January 1970.
|
||||
/// @param timestamp Current latest experiment start timestamp. If not known, affected service
|
||||
/// should specify -1;
|
||||
/// @param payloads List of experiment metadata.
|
||||
- (NSTimeInterval)latestExperimentStartTimestampBetweenTimestamp:(NSTimeInterval)timestamp
|
||||
andPayloads:(NSArray<NSData *> *)payloads;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
|
@ -1,46 +0,0 @@
|
|||
#import <Foundation/Foundation.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
/// Default event name for when an experiment is set.
|
||||
extern NSString *const FIRSetExperimentEventName;
|
||||
/// Default event name for when an experiment is activated.
|
||||
extern NSString *const FIRActivateExperimentEventName;
|
||||
/// Default event name for when an experiment is cleared.
|
||||
extern NSString *const FIRClearExperimentEventName;
|
||||
/// Default event name for when an experiment times out for being activated.
|
||||
extern NSString *const FIRTimeoutExperimentEventName;
|
||||
/// Default event name for when an experiment is expired as it reaches the end of TTL.
|
||||
extern NSString *const FIRExpireExperimentEventName;
|
||||
|
||||
/// An Experiment Lifecycle Event Object that specifies the name of the experiment event to be
|
||||
/// logged by Firebase Analytics.
|
||||
@interface FIRLifecycleEvents : NSObject
|
||||
|
||||
/// Event name for when an experiment is set. It is default to FIRSetExperimentEventName and can be
|
||||
/// overriden. If experiment payload has a valid string of this field, always use experiment
|
||||
/// payload.
|
||||
@property(nonatomic, copy) NSString *setExperimentEventName;
|
||||
|
||||
/// Event name for when an experiment is activated. It is default to FIRActivateExperimentEventName
|
||||
/// and can be overriden. If experiment payload has a valid string of this field, always use
|
||||
/// experiment payload.
|
||||
@property(nonatomic, copy) NSString *activateExperimentEventName;
|
||||
|
||||
/// Event name for when an experiment is clearred. It is default to FIRClearExperimentEventName and
|
||||
/// can be overriden. If experiment payload has a valid string of this field, always use experiment
|
||||
/// payload.
|
||||
@property(nonatomic, copy) NSString *clearExperimentEventName;
|
||||
/// Event name for when an experiment is timeout from being STANDBY. It is default to
|
||||
/// FIRTimeoutExperimentEventName and can be overriden. If experiment payload has a valid string
|
||||
/// of this field, always use experiment payload.
|
||||
@property(nonatomic, copy) NSString *timeoutExperimentEventName;
|
||||
|
||||
/// Event name when an experiment is expired when it reaches the end of its TTL.
|
||||
/// It is default to FIRExpireExperimentEventName and can be overriden. If experiment payload has a
|
||||
/// valid string of this field, always use experiment payload.
|
||||
@property(nonatomic, copy) NSString *expireExperimentEventName;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
|
@ -1,2 +0,0 @@
|
|||
#import "FIRExperimentController.h"
|
||||
#import "FIRLifecycleEvents.h"
|
|
@ -1,7 +0,0 @@
|
|||
framework module FirebaseABTesting {
|
||||
umbrella header "FirebaseABTesting.h"
|
||||
export *
|
||||
module * { export *}
|
||||
link "z"
|
||||
link framework "Security"
|
||||
link framework "SystemConfiguration"}
|
|
@ -1,11 +0,0 @@
|
|||
# Firebase ABTesting SDK for iOS
|
||||
|
||||
A/B testing is a Firebase service that lets you run experiments across users of
|
||||
your iOS and Android apps. It lets you learn how well one or more changes to
|
||||
your app work with a smaller set of users before you roll out changes to all
|
||||
users. You run experiments to find the most effective ways to use the
|
||||
Notifications composer and Firebase Remote Config in your app.
|
||||
|
||||
Please visit [our developer site]
|
||||
(https://firebase.google.com/docs/ab-testing/) for integration instructions,
|
||||
documentations, support information, and terms of service.
|
Binary file not shown.
Binary file not shown.
|
@ -15,3 +15,15 @@
|
|||
/// The method used to sign in. For example, "google", "facebook" or "twitter".
|
||||
static NSString *const kFIRUserPropertySignUpMethod
|
||||
NS_SWIFT_NAME(AnalyticsUserPropertySignUpMethod) = @"sign_up_method";
|
||||
|
||||
/// Indicates whether events logged by Google Analytics can be used to personalize ads for the user.
|
||||
/// Set to "YES" to enable, or "NO" to disable. Default is enabled. See the
|
||||
/// <a href="https://firebase.google.com/support/guides/disable-analytics">documentation</a> for
|
||||
/// more details and information about related settings.
|
||||
///
|
||||
/// <pre>
|
||||
/// [FIRAnalytics setUserPropertyString:@"NO"
|
||||
/// forName:kFIRUserPropertyAllowAdPersonalizationSignals];
|
||||
/// </pre>
|
||||
static NSString *const kFIRUserPropertyAllowAdPersonalizationSignals
|
||||
NS_SWIFT_NAME(AnalyticsUserPropertyAllowAdPersonalizationSignals) = @"allow_personalized_ads";
|
||||
|
|
Binary file not shown.
|
@ -2,6 +2,8 @@ framework module FirebaseCoreDiagnostics {
|
|||
export *
|
||||
module * { export * }
|
||||
link "z"
|
||||
link framework "Foundation"
|
||||
link framework "Security"
|
||||
link framework "SystemConfiguration"
|
||||
link framework "UIKit"
|
||||
}
|
||||
|
|
|
@ -12,9 +12,9 @@
|
|||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "FIRAnalyticsConfiguration.h"
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "Private/FIRAnalyticsConfiguration+Internal.h"
|
||||
#import "Private/FIRAnalyticsConfiguration.h"
|
||||
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-implementations"
|
||||
|
@ -39,16 +39,6 @@
|
|||
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];
|
||||
}
|
||||
|
|
|
@ -15,11 +15,11 @@
|
|||
#include <sys/utsname.h>
|
||||
|
||||
#import "FIRApp.h"
|
||||
#import "FIRConfiguration.h"
|
||||
#import "Private/FIRAnalyticsConfiguration+Internal.h"
|
||||
#import "Private/FIRAnalyticsConfiguration.h"
|
||||
#import "Private/FIRAppInternal.h"
|
||||
#import "Private/FIRBundleUtil.h"
|
||||
#import "Private/FIRComponentContainerInternal.h"
|
||||
#import "Private/FIRConfigurationInternal.h"
|
||||
#import "Private/FIRLibrary.h"
|
||||
#import "Private/FIRLogger.h"
|
||||
#import "Private/FIROptionsInternal.h"
|
||||
|
@ -139,6 +139,17 @@ static NSMutableDictionary *sLibraryVersions;
|
|||
[FIRApp configureWithName:kFIRDefaultAppName options:options];
|
||||
}
|
||||
|
||||
+ (NSCharacterSet *)applicationNameAllowedCharacters {
|
||||
static NSCharacterSet *applicationNameAllowedCharacters;
|
||||
static dispatch_once_t onceToken;
|
||||
dispatch_once(&onceToken, ^{
|
||||
NSMutableCharacterSet *allowedNameCharacters = [NSMutableCharacterSet alphanumericCharacterSet];
|
||||
[allowedNameCharacters addCharactersInString:@"-_"];
|
||||
applicationNameAllowedCharacters = [allowedNameCharacters copy];
|
||||
});
|
||||
return applicationNameAllowedCharacters;
|
||||
}
|
||||
|
||||
+ (void)configureWithName:(NSString *)name options:(FIROptions *)options {
|
||||
if (!name || !options) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain format:@"Neither name nor options can be nil."];
|
||||
|
@ -156,14 +167,12 @@ static NSMutableDictionary *sLibraryVersions;
|
|||
FIRLogDebug(kFIRLoggerCore, @"I-COR000001", @"Configuring the default app.");
|
||||
} else {
|
||||
// Validate the app name and ensure it hasn't been configured already.
|
||||
for (NSUInteger 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 can only contain alphanumeric (A-Z,a-z,0-9), "
|
||||
@"hyphen (-), and underscore (_) characters"];
|
||||
}
|
||||
NSCharacterSet *nameCharacters = [NSCharacterSet characterSetWithCharactersInString:name];
|
||||
|
||||
if (![[self applicationNameAllowedCharacters] isSupersetOfSet:nameCharacters]) {
|
||||
[NSException raise:kFirebaseCoreErrorDomain
|
||||
format:@"App name can only contain alphanumeric, "
|
||||
@"hyphen (-), and underscore (_) characters"];
|
||||
}
|
||||
|
||||
@synchronized(self) {
|
||||
|
@ -311,11 +320,7 @@ static NSMutableDictionary *sLibraryVersions;
|
|||
// always initialize first by itself before the other SDKs.
|
||||
if ([self.name isEqualToString:kFIRDefaultAppName]) {
|
||||
Class firAnalyticsClass = NSClassFromString(@"FIRAnalytics");
|
||||
if (!firAnalyticsClass) {
|
||||
FIRLogWarning(kFIRLoggerCore, @"I-COR000022",
|
||||
@"Firebase Analytics is not available. To add it, include Firebase/Core in the "
|
||||
@"Podfile or add FirebaseAnalytics.framework to the Link Build Phase");
|
||||
} else {
|
||||
if (firAnalyticsClass) {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wundeclared-selector"
|
||||
SEL startWithConfigurationSelector = @selector(startWithConfiguration:options:);
|
||||
|
|
|
@ -52,10 +52,10 @@
|
|||
// This allows app extensions that have the app's bundle as their prefix to pass this test.
|
||||
NSString *applicationBundleIdentifier =
|
||||
[GULAppEnvironmentUtil isAppExtension]
|
||||
? [self bundleIdentifierByRemovingLastPartFrom:bundleIdentifier]
|
||||
: bundleIdentifier;
|
||||
? [self bundleIdentifierByRemovingLastPartFrom:bundle.bundleIdentifier]
|
||||
: bundle.bundleIdentifier;
|
||||
|
||||
if ([applicationBundleIdentifier isEqualToString:bundle.bundleIdentifier]) {
|
||||
if ([applicationBundleIdentifier isEqualToString:bundleIdentifier]) {
|
||||
return YES;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -12,7 +12,9 @@
|
|||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#import "FIRConfiguration.h"
|
||||
#import "Private/FIRConfigurationInternal.h"
|
||||
|
||||
#import "Private/FIRAnalyticsConfiguration.h"
|
||||
|
||||
extern void FIRSetLoggerLevel(FIRLoggerLevel loggerLevel);
|
||||
|
||||
|
@ -30,10 +32,7 @@ extern void FIRSetLoggerLevel(FIRLoggerLevel loggerLevel);
|
|||
- (instancetype)init {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
|
||||
_analyticsConfiguration = [FIRAnalyticsConfiguration sharedInstance];
|
||||
#pragma clang diagnostic pop
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
|
|
@ -15,15 +15,7 @@
|
|||
#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";
|
||||
|
|
|
@ -20,23 +20,17 @@
|
|||
|
||||
#import "Private/FIRVersion.h"
|
||||
|
||||
FIRLoggerService kFIRLoggerCore = @"[Firebase/Core]";
|
||||
|
||||
// All the FIRLoggerService definitions should be migrated to clients. Do not add new ones!
|
||||
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";
|
||||
|
|
|
@ -17,6 +17,7 @@
|
|||
#import "Private/FIRErrors.h"
|
||||
#import "Private/FIRLogger.h"
|
||||
#import "Private/FIROptionsInternal.h"
|
||||
#import "Private/FIRVersion.h"
|
||||
|
||||
// Keys for the strings in the plist file.
|
||||
NSString *const kFIRAPIKey = @"API_KEY";
|
||||
|
@ -39,11 +40,13 @@ NSString *const kFIRIsAnalyticsCollectionDeactivated = @"FIREBASE_ANALYTICS_COLL
|
|||
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)
|
||||
@"04" // Minor version (exactly 2 digits)
|
||||
@"01" // Build number (exactly 2 digits)
|
||||
@"000"; // Fixed "000"
|
||||
// Library version ID formatted like:
|
||||
// @"5" // Major version (one or more digits)
|
||||
// @"04" // Minor version (exactly 2 digits)
|
||||
// @"01" // Build number (exactly 2 digits)
|
||||
// @"000"; // Fixed "000"
|
||||
NSString *kFIRLibraryVersionID;
|
||||
|
||||
// Plist file name.
|
||||
NSString *const kServiceInfoFileName = @"GoogleService-Info";
|
||||
// Plist file type.
|
||||
|
@ -109,17 +112,9 @@ static NSDictionary *sDefaultOptionsDictionary = nil;
|
|||
|
||||
+ (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]]];
|
||||
[FIRApp registerLibrary:@"fire-ios"
|
||||
withVersion:[NSString stringWithUTF8String:FIRCoreVersionString]];
|
||||
|
||||
NSDictionary<NSString *, id> *info = [[NSBundle mainBundle] infoDictionary];
|
||||
NSString *xcodeVersion = info[@"DTXcodeBuild"];
|
||||
NSString *sdkVersion = info[@"DTSDKBuild"];
|
||||
|
@ -295,6 +290,16 @@ static NSDictionary *sDefaultOptionsDictionary = nil;
|
|||
}
|
||||
|
||||
- (NSString *)libraryVersionID {
|
||||
static dispatch_once_t onceToken;
|
||||
dispatch_once(&onceToken, ^{
|
||||
// The unit tests are set up to catch anything that does not properly convert.
|
||||
NSString *version = [NSString stringWithUTF8String:FIRCoreVersionString];
|
||||
NSArray *components = [version componentsSeparatedByString:@"."];
|
||||
NSString *major = [components objectAtIndex:0];
|
||||
NSString *minor = [NSString stringWithFormat:@"%02d", [[components objectAtIndex:1] intValue]];
|
||||
NSString *patch = [NSString stringWithFormat:@"%02d", [[components objectAtIndex:2] intValue]];
|
||||
kFIRLibraryVersionID = [NSString stringWithFormat:@"%@%@%@000", major, minor, patch];
|
||||
});
|
||||
return kFIRLibraryVersionID;
|
||||
}
|
||||
|
||||
|
|
|
@ -14,7 +14,7 @@
|
|||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import "FIRAnalyticsConfiguration.h"
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
/// Values stored in analyticsEnabledState. Never alter these constants since they must match with
|
||||
/// values persisted to disk.
|
||||
|
@ -38,7 +38,14 @@ static NSString *const kFIRAnalyticsConfigurationSetMinimumSessionIntervalNotifi
|
|||
static NSString *const kFIRAnalyticsConfigurationSetSessionTimeoutIntervalNotification =
|
||||
@"FIRAnalyticsConfigurationSetSessionTimeoutIntervalNotification";
|
||||
|
||||
@interface FIRAnalyticsConfiguration (Internal)
|
||||
@interface FIRAnalyticsConfiguration : NSObject
|
||||
|
||||
/// Returns the shared instance of FIRAnalyticsConfiguration.
|
||||
+ (FIRAnalyticsConfiguration *)sharedInstance;
|
||||
|
||||
// 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;
|
||||
|
||||
/// 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
|
|
@ -14,8 +14,8 @@
|
|||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import "FIRApp.h"
|
||||
#import "FIRErrors.h"
|
||||
#import <FirebaseCore/FIRApp.h>
|
||||
#import <FirebaseCore/FIRErrors.h>
|
||||
|
||||
@class FIRComponentContainer;
|
||||
@protocol FIRLibrary;
|
||||
|
@ -40,11 +40,9 @@ typedef NS_ENUM(NSInteger, FIRConfigType) {
|
|||
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;
|
||||
|
|
|
@ -0,0 +1,29 @@
|
|||
/*
|
||||
* Copyright 2019 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"
|
||||
|
||||
@class FIRAnalyticsConfiguration;
|
||||
|
||||
@interface FIRConfiguration ()
|
||||
|
||||
/**
|
||||
* The configuration class for Firebase Analytics. This should be removed once the logic for
|
||||
* enabling and disabling Analytics is moved to Analytics.
|
||||
*/
|
||||
@property(nonatomic, readwrite) FIRAnalyticsConfiguration *analyticsConfiguration;
|
||||
|
||||
@end
|
|
@ -34,22 +34,5 @@ typedef NS_ENUM(NSInteger, FIRErrorCode) {
|
|||
/**
|
||||
* 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,
|
||||
};
|
||||
|
|
|
@ -19,15 +19,6 @@
|
|||
#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;
|
||||
|
|
|
@ -16,7 +16,7 @@
|
|||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRLoggerLevel.h"
|
||||
#import <FirebaseCore/FIRLoggerLevel.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
|
@ -29,19 +29,11 @@ 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 kFIRLoggerCore;
|
||||
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.
|
||||
|
|
|
@ -14,7 +14,7 @@
|
|||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import "FIROptions.h"
|
||||
#import <FirebaseCore/FIROptions.h>
|
||||
|
||||
/**
|
||||
* Keys for the strings in the plist file.
|
||||
|
|
|
@ -1,56 +0,0 @@
|
|||
/*
|
||||
* 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)
|
||||
DEPRECATED_MSG_ATTRIBUTE("Use these methods directly on the `Analytics` class.")
|
||||
@interface FIRAnalyticsConfiguration : NSObject
|
||||
|
||||
/**
|
||||
* Returns the shared instance of FIRAnalyticsConfiguration.
|
||||
*/
|
||||
+ (FIRAnalyticsConfiguration *)sharedInstance NS_SWIFT_NAME(shared());
|
||||
|
||||
/**
|
||||
* Deprecated.
|
||||
* Sets the minimum engagement time in seconds required to start a new session. The default value
|
||||
* is 10 seconds.
|
||||
*/
|
||||
- (void)setMinimumSessionInterval:(NSTimeInterval)minimumSessionInterval
|
||||
DEPRECATED_MSG_ATTRIBUTE(
|
||||
"Sessions are started immediately. More information at https://bit.ly/2FU46av");
|
||||
|
||||
/**
|
||||
* 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
|
|
@ -16,14 +16,12 @@
|
|||
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRAnalyticsConfiguration.h"
|
||||
#import "FIRLoggerLevel.h"
|
||||
#import <FirebaseCore/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.
|
||||
* This interface provides global level properties that the developer can tweak.
|
||||
*/
|
||||
NS_SWIFT_NAME(FirebaseConfiguration)
|
||||
@interface FIRConfiguration : NSObject
|
||||
|
@ -31,11 +29,6 @@ NS_SWIFT_NAME(FirebaseConfiguration)
|
|||
/** 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 DEPRECATED_MSG_ATTRIBUTE(
|
||||
"Use the methods available here directly on the `Analytics` class.");
|
||||
|
||||
/**
|
||||
* 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
|
||||
|
|
|
@ -14,7 +14,6 @@
|
|||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#import "FIRAnalyticsConfiguration.h"
|
||||
#import "FIRApp.h"
|
||||
#import "FIRConfiguration.h"
|
||||
#import "FIRLoggerLevel.h"
|
||||
|
|
|
@ -169,9 +169,9 @@ very grateful! We'd like to empower as many developers as we can to be able to
|
|||
participate in the Firebase community.
|
||||
|
||||
### macOS and tvOS
|
||||
Thanks to contributions from the community, FirebaseAuth, FirebaseCore, FirebaseDatabase,
|
||||
FirebaseFunctions and FirebaseStorage now compile, run unit tests, and work on macOS and tvOS.
|
||||
FirebaseFirestore is availiable for macOS and FirebaseMessaging for tvOS.
|
||||
Thanks to contributions from the community, FirebaseAuth, FirebaseCore, FirebaseDatabase, FirebaseMessaging,
|
||||
FirebaseFirestore, FirebaseFunctions and FirebaseStorage now compile, run unit tests, and work on
|
||||
macOS and tvOS.
|
||||
|
||||
For tvOS, checkout the [Sample](Example/tvOSSample).
|
||||
|
||||
|
|
|
@ -65,12 +65,15 @@ typedef NS_ENUM(NSInteger, FIRInstanceIDMessageCode) {
|
|||
kFIRInstanceIDMessageCodeService004 = 7004,
|
||||
kFIRInstanceIDMessageCodeService005 = 7005,
|
||||
kFIRInstanceIDMessageCodeService006 = 7006,
|
||||
kFIRIntsanceIDInvalidNetworkSession = 7007,
|
||||
kFIRInstanceIDInvalidNetworkSession = 7007,
|
||||
kFIRInstanceIDInvalidSettingResponse = 7008,
|
||||
// FIRInstanceIDCheckinStore.m
|
||||
// DO NOT USE 8002, 8004 - 8008
|
||||
kFIRInstanceIDMessageCodeCheckinStore000 = 8000,
|
||||
kFIRInstanceIDMessageCodeCheckinStore001 = 8001,
|
||||
kFIRInstanceIDMessageCodeCheckinStore003 = 8003,
|
||||
kFIRInstanceIDMessageCodeCheckinStoreCheckinPlistDeleted = 8009,
|
||||
kFIRInstanceIDMessageCodeCheckinStoreCheckinPlistSaved = 8010,
|
||||
// FIRInstanceIDKeyPair.m
|
||||
// DO NOT USE 9001, 9003
|
||||
kFIRInstanceIDMessageCodeKeyPair000 = 9000,
|
||||
|
@ -155,4 +158,8 @@ typedef NS_ENUM(NSInteger, FIRInstanceIDMessageCode) {
|
|||
kFIRInstanceIDKeychainCreateKeyPairError = 23003,
|
||||
kFIRInstanceIDKeychainUpdateItemError = 23004,
|
||||
|
||||
// FIRInstanceIDStringEncoding.m
|
||||
kFIRInstanceIDStringEncodingBufferUnderflow = 24000,
|
||||
kFIRInstanceIDStringEncodingBufferOverflow = 24001,
|
||||
|
||||
};
|
||||
|
|
|
@ -19,17 +19,11 @@
|
|||
#import "FIRInstanceIDCheckinService.h"
|
||||
|
||||
/**
|
||||
* Internal API used by other Firebase SDK teams, including Messaging, Analytics and Remote config.
|
||||
* Internal API used by Firebase SDK teams by calling in reflection or internal teams.
|
||||
*/
|
||||
// TODO(chliangGoogle) Rename this to Internal.
|
||||
@interface FIRInstanceID (Private)
|
||||
|
||||
/**
|
||||
* Return the cached checkin preferences on the disk. This is used internally only by Messaging.
|
||||
*
|
||||
* @return The cached checkin preferences on the client.
|
||||
*/
|
||||
- (nullable FIRInstanceIDCheckinPreferences *)cachedCheckinPreferences;
|
||||
|
||||
/**
|
||||
* Fetches checkin info for the app. If the app has valid cached checkin preferences
|
||||
* they are returned instead of making a network request.
|
||||
|
|
|
@ -29,10 +29,6 @@
|
|||
|
||||
@implementation FIRInstanceID (Private)
|
||||
|
||||
- (FIRInstanceIDCheckinPreferences *)cachedCheckinPreferences {
|
||||
return [self.tokenManager.authService checkinPreferences];
|
||||
}
|
||||
|
||||
// This method just wraps our pre-configured auth service to make the request.
|
||||
// This method is only needed by first-party users, like Remote Config.
|
||||
- (void)fetchCheckinInfoWithHandler:(FIRInstanceIDDeviceCheckinCompletion)handler {
|
||||
|
|
|
@ -1,42 +0,0 @@
|
|||
/*
|
||||
* Copyright 2019 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 "FIRInstanceID+Private.h"
|
||||
#import "FIRInstanceID.h"
|
||||
#import "FIRInstanceIDKeyPairStore.h"
|
||||
#import "FIRInstanceIDTokenManager.h"
|
||||
|
||||
@interface FIRInstanceID (Testing)
|
||||
|
||||
@property(nonatomic, readwrite, strong) FIRInstanceIDTokenManager *tokenManager;
|
||||
@property(nonatomic, readwrite, strong) FIRInstanceIDKeyPairStore *keyPairStore;
|
||||
@property(nonatomic, readwrite, copy) NSString *fcmSenderID;
|
||||
|
||||
/**
|
||||
* Private initializer.
|
||||
*/
|
||||
- (instancetype)initPrivately;
|
||||
|
||||
/**
|
||||
* Actually makes InstanceID instantiate both the IID and Token-related subsystems.
|
||||
*/
|
||||
- (void)start;
|
||||
|
||||
+ (int64_t)maxRetryCountForDefaultToken;
|
||||
+ (int64_t)minIntervalForDefaultTokenRetry;
|
||||
+ (int64_t)maxRetryIntervalForDefaultTokenInSeconds;
|
||||
|
||||
@end
|
|
@ -24,6 +24,7 @@
|
|||
#import <GoogleUtilities/GULAppEnvironmentUtil.h>
|
||||
#import "FIRInstanceID+Private.h"
|
||||
#import "FIRInstanceIDAuthService.h"
|
||||
#import "FIRInstanceIDCheckinPreferences.h"
|
||||
#import "FIRInstanceIDCombinedHandler.h"
|
||||
#import "FIRInstanceIDConstants.h"
|
||||
#import "FIRInstanceIDDefines.h"
|
||||
|
@ -57,7 +58,12 @@ int64_t const kMinRetryIntervalForDefaultTokenInSeconds = 10; // 10 second
|
|||
// change.
|
||||
NSInteger const kMaxRetryCountForDefaultToken = 5;
|
||||
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
static NSString *const kEntitlementsAPSEnvironmentKey = @"Entitlements.aps-environment";
|
||||
#else
|
||||
static NSString *const kEntitlementsAPSEnvironmentKey = @"com.apple.developer.aps-environment";
|
||||
#endif
|
||||
static NSString *const kEntitlementsKeyForMac = @"Entitlements";
|
||||
static NSString *const kAPSEnvironmentDevelopmentValue = @"development";
|
||||
/// FIRMessaging selector that returns the current FIRMessaging auto init
|
||||
/// enabled flag.
|
||||
|
@ -71,7 +77,6 @@ static NSString *const kFIRIIDAppNameKey = @"FIRAppNameKey";
|
|||
static NSString *const kFIRIIDErrorDomain = @"com.firebase.instanceid";
|
||||
static NSString *const kFIRIIDServiceInstanceID = @"InstanceID";
|
||||
|
||||
// This should be the same value as FIRErrorCodeInstanceIDFailed, which we can't import directly
|
||||
static NSInteger const kFIRIIDErrorCodeInstanceIDFailed = -121;
|
||||
|
||||
typedef void (^FIRInstanceIDKeyPairHandler)(FIRInstanceIDKeyPair *keyPair, NSError *error);
|
||||
|
@ -258,8 +263,6 @@ static FIRInstanceID *gInstanceID;
|
|||
scope:(NSString *)scope
|
||||
options:(NSDictionary *)options
|
||||
handler:(FIRInstanceIDTokenHandler)handler {
|
||||
_FIRInstanceIDDevAssert(handler != nil && [authorizedEntity length] && [scope length],
|
||||
@"Invalid authorizedEntity or scope to new token");
|
||||
if (!handler) {
|
||||
FIRInstanceIDLoggerError(kFIRInstanceIDMessageCodeInstanceID000,
|
||||
kFIRInstanceIDInvalidNilHandlerError);
|
||||
|
@ -367,9 +370,6 @@ static FIRInstanceID *gInstanceID;
|
|||
- (void)deleteTokenWithAuthorizedEntity:(NSString *)authorizedEntity
|
||||
scope:(NSString *)scope
|
||||
handler:(FIRInstanceIDDeleteTokenHandler)handler {
|
||||
_FIRInstanceIDDevAssert(handler != nil && [authorizedEntity length] && [scope length],
|
||||
@"Invalid authorizedEntity or scope to delete token");
|
||||
|
||||
if (!handler) {
|
||||
FIRInstanceIDLoggerError(kFIRInstanceIDMessageCodeInstanceID001,
|
||||
kFIRInstanceIDInvalidNilHandlerError);
|
||||
|
@ -462,8 +462,6 @@ static FIRInstanceID *gInstanceID;
|
|||
#pragma mark - Identity
|
||||
|
||||
- (void)getIDWithHandler:(FIRInstanceIDHandler)handler {
|
||||
_FIRInstanceIDDevAssert(handler, @"Invalid nil handler to getIdentity");
|
||||
|
||||
if (!handler) {
|
||||
FIRInstanceIDLoggerError(kFIRInstanceIDMessageCodeInstanceID003,
|
||||
kFIRInstanceIDInvalidNilHandlerError);
|
||||
|
@ -490,18 +488,13 @@ static FIRInstanceID *gInstanceID;
|
|||
// When getID is explicitly called, trigger getToken to make sure token always exists.
|
||||
// This is to avoid ID conflict (ID is not checked for conflict until we generate a token)
|
||||
if (appIdentity) {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
|
||||
[self token];
|
||||
#pragma clang diagnostic pop
|
||||
}
|
||||
callHandlerOnMainThread(appIdentity, error);
|
||||
});
|
||||
}
|
||||
|
||||
- (void)deleteIDWithHandler:(FIRInstanceIDDeleteHandler)handler {
|
||||
_FIRInstanceIDDevAssert(handler, @"Invalid nil handler to delete Identity");
|
||||
|
||||
if (!handler) {
|
||||
FIRInstanceIDLoggerError(kFIRInstanceIDMessageCodeInstanceID004,
|
||||
kFIRInstanceIDInvalidNilHandlerError);
|
||||
|
@ -615,6 +608,26 @@ static FIRInstanceID *gInstanceID;
|
|||
}];
|
||||
}
|
||||
|
||||
#pragma mark - Checkin
|
||||
|
||||
- (BOOL)tryToLoadValidCheckinInfo {
|
||||
FIRInstanceIDCheckinPreferences *checkinPreferences =
|
||||
[self.tokenManager.authService checkinPreferences];
|
||||
return [checkinPreferences hasValidCheckinInfo];
|
||||
}
|
||||
|
||||
- (NSString *)deviceAuthID {
|
||||
return [self.tokenManager.authService checkinPreferences].deviceID;
|
||||
}
|
||||
|
||||
- (NSString *)secretToken {
|
||||
return [self.tokenManager.authService checkinPreferences].secretToken;
|
||||
}
|
||||
|
||||
- (NSString *)versionInfo {
|
||||
return [self.tokenManager.authService checkinPreferences].versionInfo;
|
||||
}
|
||||
|
||||
#pragma mark - Config
|
||||
|
||||
+ (void)load {
|
||||
|
@ -707,20 +720,14 @@ static FIRInstanceID *gInstanceID;
|
|||
[self defaultTokenWithHandler:nil];
|
||||
}
|
||||
// Notify FCM with the default token.
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
|
||||
self.defaultFCMToken = [self token];
|
||||
#pragma clang diagnostic pop
|
||||
} else if ([self isFCMAutoInitEnabled]) {
|
||||
// When there is no cached token, must check auto init is enabled.
|
||||
// If it's disabled, don't initiate token generation/refresh.
|
||||
// If no cache token and auto init is enabled, fetch a token from server.
|
||||
[self defaultTokenWithHandler:nil];
|
||||
// Notify FCM with the default token.
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
|
||||
self.defaultFCMToken = [self token];
|
||||
#pragma clang diagnostic pop
|
||||
}
|
||||
// ONLY checkin when auto data collection is turned on.
|
||||
if ([self isFCMAutoInitEnabled]) {
|
||||
|
@ -1053,43 +1060,27 @@ static FIRInstanceID *gInstanceID;
|
|||
const BOOL defaultAppTypeProd = YES;
|
||||
|
||||
NSError *error = nil;
|
||||
|
||||
Class envClass = NSClassFromString(@"FIRAppEnvironmentUtil");
|
||||
SEL isSimulatorSelector = NSSelectorFromString(@"isSimulator");
|
||||
if ([envClass respondsToSelector:isSimulatorSelector]) {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
|
||||
if ([envClass performSelector:isSimulatorSelector]) {
|
||||
#pragma clang diagnostic pop
|
||||
[self logAPNSConfigurationError:@"Running InstanceID on a simulator doesn't have APNS. "
|
||||
@"Use prod profile by default."];
|
||||
return defaultAppTypeProd;
|
||||
}
|
||||
if ([GULAppEnvironmentUtil isSimulator]) {
|
||||
[self logAPNSConfigurationError:@"Running InstanceID on a simulator doesn't have APNS. "
|
||||
@"Use prod profile by default."];
|
||||
return defaultAppTypeProd;
|
||||
}
|
||||
|
||||
if ([GULAppEnvironmentUtil isFromAppStore]) {
|
||||
// Apps distributed via AppStore or TestFlight use the Production APNS certificates.
|
||||
return defaultAppTypeProd;
|
||||
}
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
NSString *path = [[[NSBundle mainBundle] bundlePath]
|
||||
stringByAppendingPathComponent:@"embedded.mobileprovision"];
|
||||
#elif TARGET_OS_OSX
|
||||
NSString *path = [[[[NSBundle mainBundle] resourcePath] stringByDeletingLastPathComponent]
|
||||
stringByAppendingPathComponent:@"embedded.provisionprofile"];
|
||||
#endif
|
||||
|
||||
// Apps distributed via AppStore or TestFlight use the Production APNS certificates.
|
||||
SEL isFromAppStoreSelector = NSSelectorFromString(@"isFromAppStore");
|
||||
if ([envClass respondsToSelector:isFromAppStoreSelector]) {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
|
||||
if ([envClass performSelector:isFromAppStoreSelector]) {
|
||||
#pragma clang diagnostic pop
|
||||
return defaultAppTypeProd;
|
||||
}
|
||||
}
|
||||
|
||||
SEL isAppStoreReceiptSandboxSelector = NSSelectorFromString(@"isAppStoreReceiptSandbox");
|
||||
if ([envClass respondsToSelector:isAppStoreReceiptSandboxSelector]) {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
|
||||
if ([envClass performSelector:isAppStoreReceiptSandboxSelector] && !path.length) {
|
||||
#pragma clang diagnostic pop
|
||||
// Distributed via TestFlight
|
||||
return defaultAppTypeProd;
|
||||
}
|
||||
if ([GULAppEnvironmentUtil isAppStoreReceiptSandbox] && !path.length) {
|
||||
// Distributed via TestFlight
|
||||
return defaultAppTypeProd;
|
||||
}
|
||||
|
||||
NSMutableData *profileData = [NSMutableData dataWithContentsOfFile:path options:0 error:&error];
|
||||
|
@ -1162,7 +1153,13 @@ static FIRInstanceID *gInstanceID;
|
|||
@"most likely a Dev profile.");
|
||||
}
|
||||
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
NSString *apsEnvironment = [plistMap valueForKeyPath:kEntitlementsAPSEnvironmentKey];
|
||||
#elif TARGET_OS_OSX
|
||||
NSDictionary *entitlements = [plistMap valueForKey:kEntitlementsKeyForMac];
|
||||
NSString *apsEnvironment = [entitlements valueForKey:kEntitlementsAPSEnvironmentKey];
|
||||
#endif
|
||||
|
||||
NSString *debugString __unused =
|
||||
[NSString stringWithFormat:@"APNS Environment in profile: %@", apsEnvironment];
|
||||
FIRInstanceIDLoggerDebug(kFIRInstanceIDMessageCodeInstanceID013, @"%@", debugString);
|
||||
|
|
|
@ -83,14 +83,19 @@ NSString *const kFIRInstanceIDKeychainWildcardIdentifier = @"*";
|
|||
}
|
||||
|
||||
NSMutableDictionary *keychainQuery = [self keychainQueryForService:service account:account];
|
||||
|
||||
NSMutableArray<NSData *> *results;
|
||||
keychainQuery[(__bridge id)kSecReturnData] = (__bridge id)kCFBooleanTrue;
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
keychainQuery[(__bridge id)kSecReturnAttributes] = (__bridge id)kCFBooleanTrue;
|
||||
keychainQuery[(__bridge id)kSecMatchLimit] = (__bridge id)kSecMatchLimitAll;
|
||||
// FIRInstanceIDKeychain should only take a query and return a result, will handle the query here.
|
||||
NSArray *passwordInfos =
|
||||
CFBridgingRelease([[FIRInstanceIDKeychain sharedInstance] itemWithQuery:keychainQuery]);
|
||||
#elif TARGET_OS_OSX
|
||||
keychainQuery[(__bridge id)kSecMatchLimit] = (__bridge id)kSecMatchLimitOne;
|
||||
NSData *passwordInfos =
|
||||
CFBridgingRelease([[FIRInstanceIDKeychain sharedInstance] itemWithQuery:keychainQuery]);
|
||||
#endif
|
||||
|
||||
if (!passwordInfos) {
|
||||
// Nothing was found, simply return from this sync block.
|
||||
|
@ -106,15 +111,18 @@ NSString *const kFIRInstanceIDKeychainWildcardIdentifier = @"*";
|
|||
}
|
||||
return @[];
|
||||
}
|
||||
NSInteger numPasswords = passwordInfos.count;
|
||||
results = [[NSMutableArray alloc] init];
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
NSInteger numPasswords = passwordInfos.count;
|
||||
for (NSUInteger i = 0; i < numPasswords; i++) {
|
||||
NSDictionary *passwordInfo = [passwordInfos objectAtIndex:i];
|
||||
if (passwordInfo[(__bridge id)kSecValueData]) {
|
||||
[results addObject:passwordInfo[(__bridge id)kSecValueData]];
|
||||
}
|
||||
}
|
||||
|
||||
#elif TARGET_OS_OSX
|
||||
[results addObject:passwordInfos];
|
||||
#endif
|
||||
// We query the keychain because it didn't exist in cache, now query is done, update the result in
|
||||
// the cache.
|
||||
if ([service isEqualToString:kFIRInstanceIDKeychainWildcardIdentifier] ||
|
||||
|
|
|
@ -16,7 +16,6 @@
|
|||
|
||||
#import "FIRInstanceIDBackupExcludedPlist.h"
|
||||
|
||||
#import "FIRInstanceIDDefines.h"
|
||||
#import "FIRInstanceIDLogger.h"
|
||||
|
||||
typedef enum : NSUInteger {
|
||||
|
@ -64,9 +63,6 @@ typedef enum : NSUInteger {
|
|||
// Successfully wrote contents -- change the in-memory contents
|
||||
self.cachedPlistContents = [dict copy];
|
||||
|
||||
_FIRInstanceIDDevAssert([[NSFileManager defaultManager] fileExistsAtPath:path],
|
||||
@"Error writing data to non-backed up plist %@.plist", self.fileName);
|
||||
|
||||
NSURL *URL = [NSURL fileURLWithPath:path];
|
||||
if (error) {
|
||||
*error = nil;
|
||||
|
|
|
@ -36,7 +36,6 @@
|
|||
@property(nonatomic, readonly, copy) NSString *deviceDataVersion;
|
||||
@property(nonatomic, readonly, copy) NSString *digest;
|
||||
@property(nonatomic, readonly, copy) NSString *versionInfo;
|
||||
@property(nonatomic, readonly, strong) NSMutableDictionary *gServicesData;
|
||||
@property(nonatomic, readonly, assign) int64_t lastCheckinTimestampMillis;
|
||||
|
||||
/**
|
||||
|
|
|
@ -18,7 +18,6 @@
|
|||
|
||||
#import <GoogleUtilities/GULUserDefaults.h>
|
||||
#import "FIRInstanceIDCheckinService.h"
|
||||
#import "FIRInstanceIDDefines.h"
|
||||
#import "FIRInstanceIDUtilities.h"
|
||||
|
||||
const NSTimeInterval kFIRInstanceIDDefaultCheckinInterval = 7 * 24 * 60 * 60; // 7 days.
|
||||
|
@ -62,8 +61,7 @@ const NSTimeInterval kFIRInstanceIDDefaultCheckinInterval = 7 * 24 * 60 * 60; /
|
|||
- (BOOL)hasValidCheckinInfo {
|
||||
int64_t currentTimestampInMillis = FIRInstanceIDCurrentTimestampInMilliseconds();
|
||||
int64_t timeSinceLastCheckinInMillis = currentTimestampInMillis - self.lastCheckinTimestampMillis;
|
||||
_FIRInstanceIDDevAssert(timeSinceLastCheckinInMillis >= 0,
|
||||
@"FCM error: cannot have last checkin timestamp in future");
|
||||
|
||||
BOOL hasCheckinInfo = [self hasCheckinInfo];
|
||||
NSString *lastLocale =
|
||||
[[GULUserDefaults standardUserDefaults] stringForKey:kFIRInstanceIDUserDefaultsKeyLocale];
|
||||
|
|
|
@ -28,7 +28,6 @@ FOUNDATION_EXPORT NSString *const kFIRInstanceIDLastCheckinTimeKey;
|
|||
FOUNDATION_EXPORT NSString *const kFIRInstanceIDVersionInfoStringKey;
|
||||
FOUNDATION_EXPORT NSString *const kFIRInstanceIDGServicesDictionaryKey;
|
||||
FOUNDATION_EXPORT NSString *const kFIRInstanceIDDeviceDataVersionKey;
|
||||
FOUNDATION_EXPORT NSString *const kFIRInstanceIDFirebaseUserAgentKey;
|
||||
|
||||
@class FIRInstanceIDCheckinPreferences;
|
||||
|
||||
|
|
|
@ -16,7 +16,6 @@
|
|||
|
||||
#import "FIRInstanceIDCheckinService.h"
|
||||
|
||||
#import <FirebaseCore/FIRAppInternal.h>
|
||||
#import "FIRInstanceIDCheckinPreferences+Internal.h"
|
||||
#import "FIRInstanceIDCheckinPreferences_Private.h"
|
||||
#import "FIRInstanceIDDefines.h"
|
||||
|
@ -35,7 +34,6 @@ NSString *const kFIRInstanceIDLastCheckinTimeKey = @"GMSInstanceIDLastCheckinTim
|
|||
NSString *const kFIRInstanceIDVersionInfoStringKey = @"GMSInstanceIDVersionInfo";
|
||||
NSString *const kFIRInstanceIDGServicesDictionaryKey = @"GMSInstanceIDGServicesData";
|
||||
NSString *const kFIRInstanceIDDeviceDataVersionKey = @"GMSInstanceIDDeviceDataVersion";
|
||||
NSString *const kFIRInstanceIDFirebaseUserAgentKey = @"X-firebase-client";
|
||||
|
||||
static NSUInteger const kCheckinType = 2; // DeviceType IOS in l/w/a/_checkin.proto
|
||||
static NSUInteger const kCheckinVersion = 2;
|
||||
|
@ -72,24 +70,20 @@ static FIRInstanceIDURLRequestTestBlock testBlock;
|
|||
|
||||
- (void)checkinWithExistingCheckin:(FIRInstanceIDCheckinPreferences *)existingCheckin
|
||||
completion:(FIRInstanceIDDeviceCheckinCompletion)completion {
|
||||
_FIRInstanceIDDevAssert(completion != nil, @"completion required");
|
||||
|
||||
if (self.session == nil) {
|
||||
FIRInstanceIDLoggerError(kFIRIntsanceIDInvalidNetworkSession,
|
||||
FIRInstanceIDLoggerError(kFIRInstanceIDInvalidNetworkSession,
|
||||
@"Inconsistent state: NSURLSession has been invalidated");
|
||||
NSError *error =
|
||||
[NSError errorWithFIRInstanceIDErrorCode:kFIRInstanceIDErrorCodeRegistrarFailedToCheckIn];
|
||||
completion(nil, error);
|
||||
if (completion) {
|
||||
completion(nil, error);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
NSURL *url = [NSURL URLWithString:kDeviceCheckinURL];
|
||||
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
|
||||
|
||||
[request setValue:@"application/json" forHTTPHeaderField:@"content-type"];
|
||||
[request setValue:[FIRApp firebaseUserAgent]
|
||||
forHTTPHeaderField:kFIRInstanceIDFirebaseUserAgentKey];
|
||||
|
||||
NSDictionary *checkinParameters = [self checkinParametersWithExistingCheckin:existingCheckin];
|
||||
NSData *checkinData = [NSJSONSerialization dataWithJSONObject:checkinParameters
|
||||
options:0
|
||||
|
@ -103,7 +97,9 @@ static FIRInstanceIDURLRequestTestBlock testBlock;
|
|||
FIRInstanceIDLoggerDebug(kFIRInstanceIDMessageCodeService000,
|
||||
@"Device checkin HTTP fetch error. Error Code: %ld",
|
||||
(long)error.code);
|
||||
completion(nil, error);
|
||||
if (completion) {
|
||||
completion(nil, error);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -115,7 +111,9 @@ static FIRInstanceIDURLRequestTestBlock testBlock;
|
|||
FIRInstanceIDLoggerDebug(kFIRInstanceIDMessageCodeService001,
|
||||
@"Error serializing json object. Error Code: %ld",
|
||||
_FIRInstanceID_L(serializationError.code));
|
||||
completion(nil, serializationError);
|
||||
if (completion) {
|
||||
completion(nil, serializationError);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -124,7 +122,9 @@ static FIRInstanceIDURLRequestTestBlock testBlock;
|
|||
if ([deviceAuthID length] == 0) {
|
||||
NSError *error =
|
||||
[NSError errorWithFIRInstanceIDErrorCode:kFIRInstanceIDErrorCodeInvalidRequest];
|
||||
completion(nil, error);
|
||||
if (completion) {
|
||||
completion(nil, error);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -157,8 +157,9 @@ static FIRInstanceIDURLRequestTestBlock testBlock;
|
|||
if (dict[@"name"] && dict[@"value"]) {
|
||||
gservicesData[dict[@"name"]] = dict[@"value"];
|
||||
} else {
|
||||
_FIRInstanceIDDevAssert(NO, @"Invalid setting in checkin response: (%@: %@)",
|
||||
dict[@"name"], dict[@"value"]);
|
||||
FIRInstanceIDLoggerDebug(kFIRInstanceIDInvalidSettingResponse,
|
||||
@"Invalid setting in checkin response: (%@: %@)",
|
||||
dict[@"name"], dict[@"value"]);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -173,7 +174,9 @@ static FIRInstanceIDURLRequestTestBlock testBlock;
|
|||
kFIRInstanceIDDeviceDataVersionKey : deviceDataVersionInfo,
|
||||
};
|
||||
[checkinPreferences updateWithCheckinPlistContents:preferences];
|
||||
completion(checkinPreferences, nil);
|
||||
if (completion) {
|
||||
completion(checkinPreferences, nil);
|
||||
}
|
||||
};
|
||||
// Test block
|
||||
if (testBlock) {
|
||||
|
@ -223,7 +226,7 @@ static FIRInstanceIDURLRequestTestBlock testBlock;
|
|||
@"locale" : locale,
|
||||
@"version" : @(kCheckinVersion),
|
||||
@"digest" : checkinPreferences.digest ?: @"",
|
||||
@"timezone" : timeZone,
|
||||
@"time_zone" : timeZone,
|
||||
@"user_serial_number" : @(userSerialNumber),
|
||||
@"id" : @([checkinPreferences.deviceID longLongValue]),
|
||||
@"security_token" : @([checkinPreferences.secretToken longLongValue]),
|
||||
|
|
|
@ -122,6 +122,9 @@ static const NSInteger kOldCheckinPlistCount = 6;
|
|||
}
|
||||
return;
|
||||
}
|
||||
FIRInstanceIDLoggerDebug(kFIRInstanceIDMessageCodeCheckinStoreCheckinPlistSaved,
|
||||
@"Checkin plist file is saved");
|
||||
|
||||
// Save the deviceID and secret in the Keychain
|
||||
if (!preferences.hasPreCachedAuthCredentials) {
|
||||
NSData *data = [checkinKeychainContent dataUsingEncoding:NSUTF8StringEncoding];
|
||||
|
@ -146,21 +149,19 @@ static const NSInteger kOldCheckinPlistCount = 6;
|
|||
}
|
||||
|
||||
- (void)removeCheckinPreferencesWithHandler:(void (^)(NSError *error))handler {
|
||||
// Delete the checkin preferences plist first to avoid delay.
|
||||
NSError *deletePlistError;
|
||||
if (![self.plist deleteFile:&deletePlistError]) {
|
||||
handler(deletePlistError);
|
||||
return;
|
||||
}
|
||||
FIRInstanceIDLoggerDebug(kFIRInstanceIDMessageCodeCheckinStoreCheckinPlistDeleted,
|
||||
@"Deleted checkin plist file.");
|
||||
// Remove deviceID and secret from Keychain
|
||||
[self.keychain
|
||||
removeItemsMatchingService:kFIRInstanceIDCheckinKeychainService
|
||||
account:self.bundleIdentifierForKeychainAccount
|
||||
handler:^(NSError *error) {
|
||||
if (error) {
|
||||
if (handler) {
|
||||
handler(error);
|
||||
}
|
||||
return;
|
||||
}
|
||||
// Delete the checkin preferences plist
|
||||
NSError *deletePlistError;
|
||||
[self.plist deleteFile:&deletePlistError];
|
||||
|
||||
// Try to remove from old location as well because migration
|
||||
// is no longer needed. Consider this is either a fresh install
|
||||
// or an identity wipe.
|
||||
|
@ -168,7 +169,7 @@ static const NSInteger kOldCheckinPlistCount = 6;
|
|||
removeItemsMatchingService:kFIRInstanceIDLegacyCheckinKeychainService
|
||||
account:kFIRInstanceIDLegacyCheckinKeychainAccount
|
||||
handler:nil];
|
||||
handler(deletePlistError);
|
||||
handler(error);
|
||||
}];
|
||||
}
|
||||
|
||||
|
|
|
@ -45,26 +45,3 @@
|
|||
#endif
|
||||
|
||||
#endif
|
||||
|
||||
// Debug Assert
|
||||
#ifndef _FIRInstanceIDDevAssert
|
||||
// we directly invoke the NSAssert handler so we can pass on the varargs
|
||||
// (NSAssert doesn't have a macro we can use that takes varargs)
|
||||
#if !defined(NS_BLOCK_ASSERTIONS)
|
||||
#define _FIRInstanceIDDevAssert(condition, ...) \
|
||||
do { \
|
||||
if (!(condition)) { \
|
||||
[[NSAssertionHandler currentHandler] \
|
||||
handleFailureInFunction:(NSString *)[NSString stringWithUTF8String:__PRETTY_FUNCTION__] \
|
||||
file:(NSString *)[NSString stringWithUTF8String:__FILE__] \
|
||||
lineNumber:__LINE__ \
|
||||
description:__VA_ARGS__]; \
|
||||
} \
|
||||
} while (0)
|
||||
#else // !defined(NS_BLOCK_ASSERTIONS)
|
||||
#define _FIRInstanceIDDevAssert(condition, ...) \
|
||||
do { \
|
||||
} while (0)
|
||||
#endif // !defined(NS_BLOCK_ASSERTIONS)
|
||||
|
||||
#endif // _FIRInstanceIDDevAssert
|
||||
|
|
|
@ -18,7 +18,6 @@
|
|||
|
||||
#import "FIRInstanceIDBackupExcludedPlist.h"
|
||||
#import "FIRInstanceIDConstants.h"
|
||||
#import "FIRInstanceIDDefines.h"
|
||||
#import "FIRInstanceIDKeyPair.h"
|
||||
#import "FIRInstanceIDKeyPairUtilities.h"
|
||||
#import "FIRInstanceIDKeychain.h"
|
||||
|
@ -45,7 +44,6 @@ NSString *const kFIRInstanceIDKeyPairSubType = @"";
|
|||
|
||||
// Query the key with NSData format
|
||||
NSData *FIRInstanceIDKeyDataWithTag(NSString *tag) {
|
||||
_FIRInstanceIDDevAssert([tag length], @"Invalid tag for keychain specified");
|
||||
if (![tag length]) {
|
||||
return NULL;
|
||||
}
|
||||
|
@ -59,7 +57,6 @@ NSData *FIRInstanceIDKeyDataWithTag(NSString *tag) {
|
|||
|
||||
// Query the key given a tag
|
||||
SecKeyRef FIRInstanceIDCachedKeyRefWithTag(NSString *tag) {
|
||||
_FIRInstanceIDDevAssert([tag length], @"Invalid tag for keychain specified");
|
||||
if (!tag.length) {
|
||||
return NULL;
|
||||
}
|
||||
|
@ -119,7 +116,7 @@ NSString *FIRInstanceIDCreationTimeKeyWithSubtype(NSString *subtype) {
|
|||
@interface FIRInstanceIDKeyPairStore ()
|
||||
|
||||
@property(nonatomic, readwrite, strong) FIRInstanceIDBackupExcludedPlist *plist;
|
||||
@property(nonatomic, readwrite, strong) FIRInstanceIDKeyPair *keyPair;
|
||||
@property(atomic, readwrite, strong) FIRInstanceIDKeyPair *keyPair;
|
||||
@property(nonatomic, readwrite, assign) NSInteger keychainEntitlementsErrorCount;
|
||||
|
||||
@end
|
||||
|
@ -299,8 +296,6 @@ NSString *FIRInstanceIDCreationTimeKeyWithSubtype(NSString *subtype) {
|
|||
+ (FIRInstanceIDKeyPair *)keyPairForPrivateKeyTag:(NSString *)privateKeyTag
|
||||
publicKeyTag:(NSString *)publicKeyTag
|
||||
error:(NSError *__autoreleasing *)error {
|
||||
_FIRInstanceIDDevAssert([privateKeyTag length] && [publicKeyTag length],
|
||||
@"Invalid tags for keypair");
|
||||
if (![privateKeyTag length] || ![publicKeyTag length]) {
|
||||
if (error) {
|
||||
*error = [NSError errorWithFIRInstanceIDErrorCode:kFIRInstanceIDErrorCodeInvalidKeyPairTags];
|
||||
|
@ -365,6 +360,8 @@ NSString *FIRInstanceIDCreationTimeKeyWithSubtype(NSString *subtype) {
|
|||
self.keyPair = keyPair;
|
||||
|
||||
// Either new key pair doesn't exist or it's different than legacy key pair, start the migration.
|
||||
__block NSError *updateKeyRefError;
|
||||
|
||||
NSString *privateKeyTag = FIRInstanceIDPrivateTagWithSubtype(kFIRInstanceIDKeyPairSubType);
|
||||
[self updateKeyRef:keyPair.publicKey
|
||||
withTag:publicKeyTag
|
||||
|
@ -372,23 +369,22 @@ NSString *FIRInstanceIDCreationTimeKeyWithSubtype(NSString *subtype) {
|
|||
if (error) {
|
||||
FIRInstanceIDLoggerError(kFIRInstanceIDMessageCodeKeyPairMigrationError,
|
||||
@"Unable to migrate key pair from legacy ones.");
|
||||
updateKeyRefError = error;
|
||||
}
|
||||
}];
|
||||
|
||||
[self updateKeyRef:keyPair.privateKey
|
||||
withTag:privateKeyTag
|
||||
handler:^(NSError *error) {
|
||||
if (error) {
|
||||
FIRInstanceIDLoggerError(kFIRInstanceIDMessageCodeKeyPairMigrationError,
|
||||
@"Unable to migrate key pair from legacy ones.");
|
||||
updateKeyRefError = error;
|
||||
}
|
||||
|
||||
if (handler) {
|
||||
handler(updateKeyRefError);
|
||||
}
|
||||
[self updateKeyRef:keyPair.privateKey
|
||||
withTag:privateKeyTag
|
||||
handler:^(NSError *error) {
|
||||
if (error) {
|
||||
FIRInstanceIDLoggerError(
|
||||
kFIRInstanceIDMessageCodeKeyPairMigrationError,
|
||||
@"Unable to migrate key pair from legacy ones.");
|
||||
return;
|
||||
}
|
||||
FIRInstanceIDLoggerDebug(
|
||||
kFIRInstanceIDMessageCodeKeyPairMigrationSuccess,
|
||||
@"Successfully migrated the key pair from legacy ones.");
|
||||
if (handler) {
|
||||
handler(error);
|
||||
}
|
||||
}];
|
||||
}];
|
||||
}
|
||||
|
||||
|
@ -400,6 +396,8 @@ NSString *FIRInstanceIDCreationTimeKeyWithSubtype(NSString *subtype) {
|
|||
handler:(void (^)(NSError *error))handler {
|
||||
NSData *updatedTagData = [tag dataUsingEncoding:NSUTF8StringEncoding];
|
||||
|
||||
__block NSError *keychainError;
|
||||
|
||||
// Always delete the old keychain before adding a new one to avoid conflicts.
|
||||
NSDictionary *deleteQuery = @{
|
||||
(__bridge id)kSecAttrApplicationTag : updatedTagData,
|
||||
|
@ -407,30 +405,29 @@ NSString *FIRInstanceIDCreationTimeKeyWithSubtype(NSString *subtype) {
|
|||
(__bridge id)kSecAttrKeyType : (__bridge id)kSecAttrKeyTypeRSA,
|
||||
(__bridge id)kSecReturnRef : @(YES),
|
||||
};
|
||||
[[FIRInstanceIDKeychain sharedInstance] removeItemWithQuery:deleteQuery
|
||||
handler:^(NSError *error) {
|
||||
if (error) {
|
||||
keychainError = error;
|
||||
}
|
||||
}];
|
||||
|
||||
[[FIRInstanceIDKeychain sharedInstance]
|
||||
removeItemWithQuery:deleteQuery
|
||||
handler:^(NSError *error) {
|
||||
if (error) {
|
||||
if (handler) {
|
||||
handler(error);
|
||||
}
|
||||
return;
|
||||
}
|
||||
NSDictionary *addQuery = @{
|
||||
(__bridge id)kSecAttrApplicationTag : updatedTagData,
|
||||
(__bridge id)kSecClass : (__bridge id)kSecClassKey,
|
||||
(__bridge id)kSecValueRef : (__bridge id)keyRef,
|
||||
(__bridge id)
|
||||
kSecAttrAccessible : (__bridge id)kSecAttrAccessibleAlwaysThisDeviceOnly,
|
||||
};
|
||||
[[FIRInstanceIDKeychain sharedInstance] addItemWithQuery:addQuery
|
||||
handler:^(NSError *addError) {
|
||||
if (handler) {
|
||||
handler(addError);
|
||||
}
|
||||
}];
|
||||
}];
|
||||
NSDictionary *addQuery = @{
|
||||
(__bridge id)kSecAttrApplicationTag : updatedTagData,
|
||||
(__bridge id)kSecClass : (__bridge id)kSecClassKey,
|
||||
(__bridge id)kSecValueRef : (__bridge id)keyRef,
|
||||
(__bridge id)kSecAttrAccessible : (__bridge id)kSecAttrAccessibleAlwaysThisDeviceOnly,
|
||||
};
|
||||
[[FIRInstanceIDKeychain sharedInstance] addItemWithQuery:addQuery
|
||||
handler:^(NSError *addError) {
|
||||
if (addError) {
|
||||
keychainError = addError;
|
||||
}
|
||||
|
||||
if (handler) {
|
||||
handler(keychainError);
|
||||
}
|
||||
}];
|
||||
}
|
||||
|
||||
- (void)deleteSavedKeyPairWithSubtype:(NSString *)subtype
|
||||
|
@ -453,6 +450,8 @@ NSString *FIRInstanceIDCreationTimeKeyWithSubtype(NSString *subtype) {
|
|||
}
|
||||
}
|
||||
|
||||
self.keyPair = nil;
|
||||
|
||||
[FIRInstanceIDKeyPairStore
|
||||
deleteKeyPairWithPrivateTag:privateKeyTag
|
||||
publicTag:publicKeyTag
|
||||
|
@ -475,7 +474,6 @@ NSString *FIRInstanceIDCreationTimeKeyWithSubtype(NSString *subtype) {
|
|||
handler(error);
|
||||
}
|
||||
} else {
|
||||
self.keyPair = nil;
|
||||
if (handler) {
|
||||
handler(nil);
|
||||
}
|
||||
|
@ -489,28 +487,25 @@ NSString *FIRInstanceIDCreationTimeKeyWithSubtype(NSString *subtype) {
|
|||
NSDictionary *queryPublicKey = FIRInstanceIDKeyPairQuery(publicTag, NO, NO);
|
||||
NSDictionary *queryPrivateKey = FIRInstanceIDKeyPairQuery(privateTag, NO, NO);
|
||||
|
||||
__block NSError *keychainError;
|
||||
|
||||
// Always remove public key first because it is the key we generate IID.
|
||||
[[FIRInstanceIDKeychain sharedInstance] removeItemWithQuery:queryPublicKey
|
||||
handler:^(NSError *error) {
|
||||
if (error) {
|
||||
if (handler) {
|
||||
handler(error);
|
||||
}
|
||||
return;
|
||||
keychainError = error;
|
||||
}
|
||||
}];
|
||||
|
||||
[[FIRInstanceIDKeychain sharedInstance] removeItemWithQuery:queryPrivateKey
|
||||
handler:^(NSError *error) {
|
||||
if (error) {
|
||||
keychainError = error;
|
||||
}
|
||||
|
||||
if (handler) {
|
||||
handler(keychainError);
|
||||
}
|
||||
[[FIRInstanceIDKeychain sharedInstance]
|
||||
removeItemWithQuery:queryPrivateKey
|
||||
handler:^(NSError *error) {
|
||||
if (error) {
|
||||
if (handler) {
|
||||
handler(error);
|
||||
}
|
||||
return;
|
||||
}
|
||||
if (handler) {
|
||||
handler(nil);
|
||||
}
|
||||
}];
|
||||
}];
|
||||
}
|
||||
|
||||
|
|
|
@ -18,7 +18,6 @@
|
|||
|
||||
#import <CommonCrypto/CommonDigest.h>
|
||||
|
||||
#import "FIRInstanceIDDefines.h"
|
||||
#import "FIRInstanceIDKeyPair.h"
|
||||
#import "FIRInstanceIDLogger.h"
|
||||
#import "FIRInstanceIDStringEncoding.h"
|
||||
|
|
|
@ -1,5 +1,3 @@
|
|||
#import "FIRInstanceIDDefines.h"
|
||||
|
||||
//
|
||||
// FIRInstanceIDStringEncoding.m
|
||||
//
|
||||
|
@ -23,6 +21,8 @@
|
|||
|
||||
#import "FIRInstanceIDStringEncoding.h"
|
||||
|
||||
#import "FIRInstanceIDLogger.h"
|
||||
|
||||
enum { kUnknownChar = -1, kPaddingChar = -2, kIgnoreChar = -3 };
|
||||
|
||||
@implementation FIRInstanceIDStringEncoding
|
||||
|
@ -138,7 +138,11 @@ static inline int lcm(int a, int b) {
|
|||
while (outPos < outLen) outBuf[outPos++] = paddingChar_;
|
||||
}
|
||||
|
||||
_FIRInstanceIDDevAssert(outPos == outLen, @"Underflowed output buffer");
|
||||
if (outPos != outLen) {
|
||||
FIRInstanceIDLoggerError(kFIRInstanceIDStringEncodingBufferUnderflow,
|
||||
@"Underflowed output buffer");
|
||||
return nil;
|
||||
}
|
||||
[outData setLength:outPos];
|
||||
|
||||
return [[NSString alloc] initWithData:outData encoding:NSASCIIStringEncoding];
|
||||
|
@ -193,7 +197,9 @@ static inline int lcm(int a, int b) {
|
|||
}
|
||||
|
||||
// Shorten buffer if needed due to padding chars
|
||||
_FIRInstanceIDDevAssert(outPos <= outLen, @"Overflowed buffer");
|
||||
if (outPos > outLen) {
|
||||
FIRInstanceIDLoggerError(kFIRInstanceIDStringEncodingBufferOverflow, @"Overflowed buffer");
|
||||
}
|
||||
[outData setLength:outPos];
|
||||
|
||||
return outData;
|
||||
|
|
|
@ -17,6 +17,9 @@
|
|||
#import "FIRInstanceIDTokenOperation.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
FOUNDATION_EXPORT NSString *const kFIRInstanceIDFirebaseUserAgentKey;
|
||||
|
||||
@interface FIRInstanceIDTokenFetchOperation : FIRInstanceIDTokenOperation
|
||||
|
||||
- (instancetype)initWithAuthorizedEntity:(NSString *)authorizedEntity
|
||||
|
|
|
@ -25,11 +25,14 @@
|
|||
#import "FIRInstanceIDUtilities.h"
|
||||
#import "NSError+FIRInstanceID.h"
|
||||
|
||||
#import <FirebaseCore/FIRAppInternal.h>
|
||||
|
||||
// We can have a static int since this error should theoretically only
|
||||
// happen once (for the first time). If it repeats there is something
|
||||
// else that is wrong.
|
||||
static int phoneRegistrationErrorRetryCount = 0;
|
||||
static const int kMaxPhoneRegistrationErrorRetryCount = 10;
|
||||
NSString *const kFIRInstanceIDFirebaseUserAgentKey = @"X-firebase-client";
|
||||
|
||||
@implementation FIRInstanceIDTokenFetchOperation
|
||||
|
||||
|
@ -55,6 +58,8 @@ static const int kMaxPhoneRegistrationErrorRetryCount = 10;
|
|||
NSMutableURLRequest *request = [[self class] requestWithAuthHeader:authHeader];
|
||||
NSString *checkinVersionInfo = self.checkinPreferences.versionInfo;
|
||||
[request setValue:checkinVersionInfo forHTTPHeaderField:@"info"];
|
||||
[request setValue:[FIRApp firebaseUserAgent]
|
||||
forHTTPHeaderField:kFIRInstanceIDFirebaseUserAgentKey];
|
||||
|
||||
// Build form-encoded body
|
||||
NSString *deviceAuthID = self.checkinPreferences.deviceID;
|
||||
|
@ -132,7 +137,6 @@ static const int kMaxPhoneRegistrationErrorRetryCount = 10;
|
|||
return;
|
||||
}
|
||||
NSDictionary *parsedResponse = [self parseFetchTokenResponse:dataResponse];
|
||||
_FIRInstanceIDDevAssert(parsedResponse.count, @"Invalid registration response");
|
||||
|
||||
if ([parsedResponse[@"token"] length]) {
|
||||
[self finishWithResult:FIRInstanceIDTokenOperationSucceeded
|
||||
|
|
|
@ -144,8 +144,12 @@ const NSTimeInterval kDefaultFetchTokenInterval = 7 * 24 * 60 * 60; // 7 days.
|
|||
|
||||
FIRInstanceIDAPNSInfo *APNSInfo = nil;
|
||||
if (rawAPNSInfo) {
|
||||
// TODO(chliangGoogle: Use the new API and secureCoding protocol.
|
||||
@try {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
|
||||
APNSInfo = [NSKeyedUnarchiver unarchiveObjectWithData:rawAPNSInfo];
|
||||
#pragma clang diagnostic pop
|
||||
} @catch (NSException *exception) {
|
||||
FIRInstanceIDLoggerInfo(kFIRInstanceIDMessageCodeTokenInfoBadAPNSInfo,
|
||||
@"Could not parse raw APNS Info while parsing archived token info.");
|
||||
|
@ -178,8 +182,14 @@ const NSTimeInterval kDefaultFetchTokenInterval = 7 * 24 * 60 * 60; // 7 days.
|
|||
[aCoder encodeObject:self.token forKey:kFIRInstanceIDTokenKey];
|
||||
[aCoder encodeObject:self.appVersion forKey:kFIRInstanceIDAppVersionKey];
|
||||
[aCoder encodeObject:self.firebaseAppID forKey:kFIRInstanceIDFirebaseAppIDKey];
|
||||
NSData *rawAPNSInfo;
|
||||
if (self.APNSInfo) {
|
||||
NSData *rawAPNSInfo = [NSKeyedArchiver archivedDataWithRootObject:self.APNSInfo];
|
||||
// TODO(chliangGoogle: Use the new API and secureCoding protocol.
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
|
||||
rawAPNSInfo = [NSKeyedArchiver archivedDataWithRootObject:self.APNSInfo];
|
||||
#pragma clang diagnostic pop
|
||||
|
||||
[aCoder encodeObject:rawAPNSInfo forKey:kFIRInstanceIDAPNSInfoKey];
|
||||
}
|
||||
[aCoder encodeObject:self.cacheTime forKey:kFIRInstanceIDCacheTimeKey];
|
||||
|
|
|
@ -17,7 +17,6 @@
|
|||
#import "FIRInstanceIDTokenOperation.h"
|
||||
|
||||
#import "FIRInstanceIDCheckinPreferences.h"
|
||||
#import "FIRInstanceIDDefines.h"
|
||||
#import "FIRInstanceIDKeyPair.h"
|
||||
#import "FIRInstanceIDKeyPairUtilities.h"
|
||||
#import "FIRInstanceIDLogger.h"
|
||||
|
@ -133,7 +132,6 @@ static NSString *const kFIRInstanceIDParamFCMLibVersion = @"X-cliv";
|
|||
|
||||
// Quickly validate whether or not the operation has all it needs to begin
|
||||
BOOL checkinfoAvailable = [self.checkinPreferences hasCheckinInfo];
|
||||
_FIRInstanceIDDevAssert(checkinfoAvailable, @"Cannot fetch token invalid checkin state");
|
||||
if (!checkinfoAvailable) {
|
||||
FIRInstanceIDErrorCode errorCode = kFIRInstanceIDErrorCodeRegistrarFailedToCheckIn;
|
||||
[self finishWithResult:FIRInstanceIDTokenOperationError
|
||||
|
|
|
@ -86,8 +86,13 @@ static NSString *const kFIRInstanceIDTokenKeychainId = @"com.google.iid-tokens";
|
|||
// NOTE: Passing in nil to unarchiveObjectWithData will result in an iOS error logged
|
||||
// in the console on iOS 10 and below. Avoid by checking item.data's existence.
|
||||
if (item) {
|
||||
// TODO(chliangGoogle: Use the new API and secureCoding protocol.
|
||||
@try {
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
|
||||
tokenInfo = [NSKeyedUnarchiver unarchiveObjectWithData:item];
|
||||
#pragma clang diagnostic pop
|
||||
|
||||
} @catch (NSException *exception) {
|
||||
FIRInstanceIDLoggerDebug(kFIRInstanceIDMessageCodeTokenStoreExceptionUnarchivingTokenInfo,
|
||||
@"Unable to parse token info from Keychain item; item was in an "
|
||||
|
@ -107,7 +112,12 @@ static NSString *const kFIRInstanceIDTokenKeychainId = @"com.google.iid-tokens";
|
|||
handler:(void (^)(NSError *))handler { // Keep the cachetime up-to-date.
|
||||
tokenInfo.cacheTime = [NSDate date];
|
||||
// Always write to the Keychain, so that the cacheTime is up-to-date.
|
||||
NSData *tokenInfoData = [NSKeyedArchiver archivedDataWithRootObject:tokenInfo];
|
||||
NSData *tokenInfoData;
|
||||
// TODO(chliangGoogle: Use the new API and secureCoding protocol.
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
|
||||
tokenInfoData = [NSKeyedArchiver archivedDataWithRootObject:tokenInfo];
|
||||
#pragma clang diagnostic pop
|
||||
NSString *account = FIRInstanceIDAppIdentifier();
|
||||
NSString *service = [[self class] serviceKeyForAuthorizedEntity:tokenInfo.authorizedEntity
|
||||
scope:tokenInfo.scope];
|
||||
|
|
|
@ -16,14 +16,15 @@
|
|||
|
||||
#import "FIRInstanceIDUtilities.h"
|
||||
|
||||
#if TARGET_OS_IOS || TARGET_OS_TV
|
||||
#import <UIKit/UIKit.h>
|
||||
#endif
|
||||
#import <sys/utsname.h>
|
||||
|
||||
#import <FirebaseCore/FIROptions.h>
|
||||
#import <GoogleUtilities/GULUserDefaults.h>
|
||||
#import "FIRInstanceID.h"
|
||||
#import "FIRInstanceIDConstants.h"
|
||||
#import "FIRInstanceIDDefines.h"
|
||||
#import "FIRInstanceIDLogger.h"
|
||||
|
||||
// Convert the macro to a string
|
||||
|
@ -186,7 +187,7 @@ NSString *FIRInstanceIDCurrentLocale() {
|
|||
return systemLanguage;
|
||||
}
|
||||
|
||||
if (@available(iOS 10.0, *)) {
|
||||
if (@available(macOS 10.12, iOS 10.0, *)) {
|
||||
return [NSLocale currentLocale].languageCode;
|
||||
} else {
|
||||
return nil;
|
||||
|
|
|
@ -16,8 +16,6 @@
|
|||
|
||||
#import "FIRInstanceIDVersionUtilities.h"
|
||||
|
||||
#import "FIRInstanceIDDefines.h"
|
||||
|
||||
// Convert the macro to a string
|
||||
#define STR(x) STR_EXPAND(x)
|
||||
#define STR_EXPAND(x) #x
|
||||
|
@ -41,7 +39,6 @@ void FIRInstanceIDParseCurrentLibraryVersion() {
|
|||
// Parse versions
|
||||
// major, minor, patch[-beta#]
|
||||
allVersions = [daylightVersion componentsSeparatedByString:kSemanticVersioningSeparator];
|
||||
_FIRInstanceIDDevAssert(allVersions.count == 3, @"Invalid versioning of FIRInstanceID library");
|
||||
if (allVersions.count == 3) {
|
||||
majorVersion = [allVersions[0] intValue];
|
||||
minorVersion = [allVersions[1] intValue];
|
||||
|
@ -49,9 +46,6 @@ void FIRInstanceIDParseCurrentLibraryVersion() {
|
|||
// Parse patch and beta versions
|
||||
NSArray *patchAndBetaVersion =
|
||||
[allVersions[2] componentsSeparatedByString:kBetaVersionPrefix];
|
||||
_FIRInstanceIDDevAssert(patchAndBetaVersion.count <= 2,
|
||||
@"Invalid versioning of FIRInstanceID library");
|
||||
|
||||
if (patchAndBetaVersion.count == 2) {
|
||||
patchVersion = [patchAndBetaVersion[0] intValue];
|
||||
betaVersion = [patchAndBetaVersion[1] intValue];
|
||||
|
|
67
ios/Pods/FirebaseInstanceID/Firebase/InstanceID/Private/FIRInstanceID_Private.h
generated
Normal file
67
ios/Pods/FirebaseInstanceID/Firebase/InstanceID/Private/FIRInstanceID_Private.h
generated
Normal file
|
@ -0,0 +1,67 @@
|
|||
/*
|
||||
* Copyright 2019 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 <FirebaseInstanceID/FIRInstanceID.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@class FIRInstanceIDCheckinPreferences;
|
||||
/**
|
||||
* Private API used by other Firebase SDKs.
|
||||
*/
|
||||
@interface FIRInstanceID ()
|
||||
|
||||
@property(nonatomic, readonly, strong) NSString *deviceAuthID;
|
||||
@property(nonatomic, readonly, strong) NSString *secretToken;
|
||||
@property(nonatomic, readonly, strong) NSString *versionInfo;
|
||||
|
||||
/**
|
||||
* Private initializer.
|
||||
*/
|
||||
- (instancetype)initPrivately;
|
||||
|
||||
/**
|
||||
* Returns a Firebase Messaging scoped token for the firebase app.
|
||||
*
|
||||
* @return Returns the stored token if the device has registered with Firebase Messaging, otherwise
|
||||
* returns nil.
|
||||
*/
|
||||
- (nullable NSString *)token;
|
||||
|
||||
/**
|
||||
* Verify if valid checkin preferences have been loaded in memory.
|
||||
*
|
||||
* @return YES if valid checkin preferences exist in memory else NO.
|
||||
*/
|
||||
- (BOOL)hasValidCheckinInfo;
|
||||
|
||||
/**
|
||||
* Try to load prefetched checkin preferences from the cache. This supports the use case where
|
||||
* InstanceID library has already obtained a valid checkin and we should be using that.
|
||||
*
|
||||
* This should be used as a last gasp effort to retreive any cached checkin preferences before
|
||||
* hitting the FIRMessaging backend to retrieve new preferences.
|
||||
*
|
||||
* Note this is only required because InstanceID and FIRMessaging both require checkin preferences
|
||||
* which need to be synced with each other.
|
||||
*
|
||||
* @return YES if successfully loaded cached checkin preferences into memory else NO.
|
||||
*/
|
||||
- (BOOL)tryToLoadValidCheckinInfo;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
|
@ -208,14 +208,6 @@ NS_SWIFT_NAME(InstanceID)
|
|||
*/
|
||||
- (void)instanceIDWithHandler:(FIRInstanceIDResultHandler)handler;
|
||||
|
||||
/**
|
||||
* Returns a Firebase Messaging scoped token for the firebase app.
|
||||
*
|
||||
* @return Returns the stored token if the device has registered with Firebase Messaging, otherwise
|
||||
* returns nil.
|
||||
*/
|
||||
- (nullable NSString *)token __deprecated_msg("Use instanceIDWithHandler: instead.");
|
||||
|
||||
/**
|
||||
* Returns a token that authorizes an Entity (example: cloud service) to perform
|
||||
* an action on behalf of the application identified by Instance ID.
|
||||
|
@ -244,7 +236,7 @@ NS_SWIFT_NAME(InstanceID)
|
|||
* passed to the UIApplicationDelegate's
|
||||
* `didRegisterForRemoteNotificationsWithDeviceToken` method.
|
||||
* The value for `apns_sandbox` should be a boolean (or an
|
||||
* NSNumber representing a BOOL in Objective C) set to true if
|
||||
* NSNumber representing a BOOL in Objective-C) set to true if
|
||||
* your app is a debug build, which means that the APNs
|
||||
* device token is for the sandbox environment. It should be
|
||||
* set to false otherwise. If the `apns_sandbox` key is not
|
||||
|
|
|
@ -169,9 +169,9 @@ very grateful! We'd like to empower as many developers as we can to be able to
|
|||
participate in the Firebase community.
|
||||
|
||||
### macOS and tvOS
|
||||
Thanks to contributions from the community, FirebaseAuth, FirebaseCore, FirebaseDatabase,
|
||||
FirebaseFunctions and FirebaseStorage now compile, run unit tests, and work on macOS and tvOS.
|
||||
FirebaseFirestore is availiable for macOS and FirebaseMessaging for tvOS.
|
||||
Thanks to contributions from the community, FirebaseAuth, FirebaseCore, FirebaseDatabase, FirebaseMessaging,
|
||||
FirebaseFirestore, FirebaseFunctions and FirebaseStorage now compile, run unit tests, and work on
|
||||
macOS and tvOS.
|
||||
|
||||
For tvOS, checkout the [Sample](Example/tvOSSample).
|
||||
|
||||
|
|
Binary file not shown.
|
@ -1,68 +0,0 @@
|
|||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRPerformanceAttributable.h"
|
||||
|
||||
/* Different HTTP methods. */
|
||||
typedef NS_ENUM(NSInteger, FIRHTTPMethod) {
|
||||
FIRHTTPMethodGET NS_SWIFT_NAME(get),
|
||||
FIRHTTPMethodPUT NS_SWIFT_NAME(put),
|
||||
FIRHTTPMethodPOST NS_SWIFT_NAME(post),
|
||||
FIRHTTPMethodDELETE NS_SWIFT_NAME(delete),
|
||||
FIRHTTPMethodHEAD NS_SWIFT_NAME(head),
|
||||
FIRHTTPMethodPATCH NS_SWIFT_NAME(patch),
|
||||
FIRHTTPMethodOPTIONS NS_SWIFT_NAME(options),
|
||||
FIRHTTPMethodTRACE NS_SWIFT_NAME(trace),
|
||||
FIRHTTPMethodCONNECT NS_SWIFT_NAME(connect)
|
||||
} NS_SWIFT_NAME(HTTPMethod);
|
||||
|
||||
/**
|
||||
* FIRHTTPMetric object can be used to make the SDK record information about a HTTP network request.
|
||||
*/
|
||||
NS_SWIFT_NAME(HTTPMetric)
|
||||
@interface FIRHTTPMetric : NSObject <FIRPerformanceAttributable>
|
||||
|
||||
/**
|
||||
* Creates HTTPMetric object for a network request.
|
||||
* @param URL The URL for which the metrics are recorded.
|
||||
* @param httpMethod HTTP method used by the request.
|
||||
*/
|
||||
- (nullable instancetype)initWithURL:(nonnull NSURL *)URL HTTPMethod:(FIRHTTPMethod)httpMethod
|
||||
NS_SWIFT_NAME(init(url:httpMethod:));
|
||||
|
||||
/**
|
||||
* Use `initWithURL:HTTPMethod:` for Objective-C and `init(url:httpMethod:)` for Swift.
|
||||
*/
|
||||
- (nonnull instancetype)init NS_UNAVAILABLE;
|
||||
|
||||
/**
|
||||
* @brief HTTP Response code. Values are greater than 0.
|
||||
*/
|
||||
@property(nonatomic, assign) NSInteger responseCode;
|
||||
|
||||
/**
|
||||
* @brief Size of the request payload.
|
||||
*/
|
||||
@property(nonatomic, assign) long requestPayloadSize;
|
||||
|
||||
/**
|
||||
* @brief Size of the response payload.
|
||||
*/
|
||||
@property(nonatomic, assign) long responsePayloadSize;
|
||||
|
||||
/**
|
||||
* @brief HTTP Response content type.
|
||||
*/
|
||||
@property(nonatomic, nullable, copy) NSString *responseContentType;
|
||||
|
||||
/**
|
||||
* Marks the start time of the request.
|
||||
*/
|
||||
- (void)start;
|
||||
|
||||
/**
|
||||
* Marks the end time of the response and queues the network request metric on the device for
|
||||
* transmission. Check the logs if the metric is valid.
|
||||
*/
|
||||
- (void)stop;
|
||||
|
||||
@end
|
|
@ -1,62 +0,0 @@
|
|||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRTrace.h"
|
||||
|
||||
/** This class allows you to configure the Firebase Performance Reporting SDK. It also provides the
|
||||
* interfaces to create timers and enable or disable automatic metrics capture.
|
||||
*
|
||||
* This SDK uses a Firebase Instance ID token to identify the app instance and periodically sends
|
||||
* data to the Firebase backend. (see `[FIRInstanceID getIDWithHandler:]`).
|
||||
* To stop the periodic sync, call `[FIRInstanceID deleteIDWithHandler:]` and
|
||||
* either disable this SDK or set FIRPerformance.dataCollectionEnabled to NO.
|
||||
*/
|
||||
NS_EXTENSION_UNAVAILABLE("FirebasePerformance does not support app extensions at this time.")
|
||||
NS_SWIFT_NAME(Performance)
|
||||
@interface FIRPerformance : NSObject
|
||||
|
||||
/**
|
||||
* Controls the capture of performance data. When this value is set to NO, none of the performance
|
||||
* data will sent to the server. Default is YES.
|
||||
*
|
||||
* This setting is persisted, and is applied on future invocations of your application. Once
|
||||
* explicitly set, it overrides any settings in your Info.plist.
|
||||
*/
|
||||
@property(nonatomic, assign, getter=isDataCollectionEnabled) BOOL dataCollectionEnabled;
|
||||
|
||||
/**
|
||||
* Controls the instrumentation of the app to capture performance data. Setting this value to NO has
|
||||
* immediate effect only if it is done so before calling [FIRApp configure]. Otherwise it takes
|
||||
* effect after the app starts again the next time.
|
||||
*
|
||||
* If set to NO, the app will not be instrumented to collect performance
|
||||
* data (in scenarios like app_start, networking monitoring). Default is YES.
|
||||
*
|
||||
* This setting is persisted, and is applied on future invocations of your application. Once
|
||||
* explicitly set, it overrides any settings in your Info.plist.
|
||||
*/
|
||||
@property(nonatomic, assign, getter=isInstrumentationEnabled) BOOL instrumentationEnabled;
|
||||
|
||||
/** @return The shared instance. */
|
||||
+ (nonnull instancetype)sharedInstance NS_SWIFT_NAME(sharedInstance());
|
||||
|
||||
/**
|
||||
* Creates an instance of FIRTrace after creating the shared instance of FIRPerformance. The trace
|
||||
* will automatically be started on a successful creation of the instance. The |name| of the trace
|
||||
* cannot be an empty string.
|
||||
*
|
||||
* @param name The name of the Trace.
|
||||
* @return The FIRTrace object.
|
||||
*/
|
||||
+ (nullable FIRTrace *)startTraceWithName:(nonnull NSString *)name
|
||||
NS_SWIFT_NAME(startTrace(name:));
|
||||
|
||||
/**
|
||||
* Creates an instance of FIRTrace. This API does not start the trace. To start the trace, use the
|
||||
* -start API on the returned |FIRTrace| object. The |name| cannot be an empty string.
|
||||
*
|
||||
* @param name The name of the Trace.
|
||||
* @return The FIRTrace object.
|
||||
*/
|
||||
- (nullable FIRTrace *)traceWithName:(nonnull NSString *)name NS_SWIFT_NAME(trace(name:));
|
||||
|
||||
@end
|
|
@ -1,38 +0,0 @@
|
|||
#import <Foundation/Foundation.h>
|
||||
|
||||
/** Defines the interface that allows adding/removing attributes to any object.
|
||||
*/
|
||||
NS_SWIFT_NAME(PerformanceAttributable)
|
||||
@protocol FIRPerformanceAttributable <NSObject>
|
||||
|
||||
/** List of attributes. */
|
||||
@property(nonatomic, nonnull, readonly) NSDictionary<NSString *, NSString *> *attributes;
|
||||
|
||||
/**
|
||||
* Sets a value as a string for the specified attribute. Updates the value of the attribute if a
|
||||
* value had already existed.
|
||||
*
|
||||
* @param value The value that needs to be set/updated for an attribute. If the length of the value
|
||||
* exceeds the maximum allowed, the value will be truncated to the maximum allowed.
|
||||
* @param attribute The name of the attribute. If the length of the value exceeds the maximum
|
||||
* allowed, the value will be truncated to the maximum allowed.
|
||||
*/
|
||||
- (void)setValue:(nonnull NSString *)value forAttribute:(nonnull NSString *)attribute;
|
||||
|
||||
/**
|
||||
* Reads the value for the specified attribute. If the attribute does not exist, returns nil.
|
||||
*
|
||||
* @param attribute The name of the attribute.
|
||||
* @return The value for the attribute. Returns nil if the attribute does not exist.
|
||||
*/
|
||||
- (nullable NSString *)valueForAttribute:(nonnull NSString *)attribute;
|
||||
|
||||
/**
|
||||
* Removes an attribute from the list. Does nothing if the attribute does not exist.
|
||||
*
|
||||
* @param attribute The name of the attribute.
|
||||
*/
|
||||
- (void)removeAttribute:(nonnull NSString *)attribute;
|
||||
|
||||
@end
|
||||
|
|
@ -1,91 +0,0 @@
|
|||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "FIRPerformanceAttributable.h"
|
||||
|
||||
/**
|
||||
* FIRTrace objects contain information about a "Trace", which is a sequence of steps. Traces can be
|
||||
* used to measure the time taken for a sequence of steps.
|
||||
* Traces also include "Counters". Counters are used to track information which is cumulative in
|
||||
* nature (e.g., Bytes downloaded). Counters are scoped to an FIRTrace object.
|
||||
*/
|
||||
NS_EXTENSION_UNAVAILABLE("FirebasePerformance does not support app extensions at this time.")
|
||||
NS_SWIFT_NAME(Trace)
|
||||
@interface FIRTrace : NSObject <FIRPerformanceAttributable>
|
||||
|
||||
/** @brief Name of the trace. */
|
||||
@property(nonatomic, copy, readonly, nonnull) NSString *name;
|
||||
|
||||
/** @brief Not a valid initializer. */
|
||||
- (nonnull instancetype)init NS_UNAVAILABLE;
|
||||
|
||||
/**
|
||||
* Starts the trace.
|
||||
*/
|
||||
- (void)start;
|
||||
|
||||
/**
|
||||
* Stops the trace if the trace is active.
|
||||
*/
|
||||
- (void)stop;
|
||||
|
||||
/**
|
||||
* Increments the counter for the provided counter name by 1. If it is a new counter name, the
|
||||
* counter value will be initialized to 1. Does nothing if the trace has not been started or has
|
||||
* already been stopped.
|
||||
*
|
||||
* Note: This API has been deprecated. Please use -incrementMetric:byInt: instead.
|
||||
*
|
||||
* @param counterName The name of the counter to increment.
|
||||
*/
|
||||
- (void)incrementCounterNamed:(nonnull NSString *)counterName
|
||||
NS_SWIFT_NAME(incrementCounter(named:))
|
||||
DEPRECATED_MSG_ATTRIBUTE("Please use -incrementMetric:byInt: instead.");
|
||||
|
||||
/**
|
||||
* Increments the counter for the provided counter name with the provided value. If it is a new
|
||||
* counter name, the counter value will be initialized to the value. Does nothing if the trace has
|
||||
* not been started or has already been stopped.
|
||||
*
|
||||
* Note: This API has been deprecated. Please use -incrementMetric:byInt: instead.
|
||||
*
|
||||
* @param counterName The name of the counter to increment.
|
||||
* @param incrementValue The value the counter would be incremented with.
|
||||
*/
|
||||
- (void)incrementCounterNamed:(nonnull NSString *)counterName by:(NSInteger)incrementValue
|
||||
NS_SWIFT_NAME(incrementCounter(named:by:))
|
||||
DEPRECATED_MSG_ATTRIBUTE("Please use -incrementMetric:byInt: instead.");
|
||||
|
||||
#pragma mark - Metrics API
|
||||
|
||||
/**
|
||||
* Atomically increments the metric for the provided metric name with the provided value. If it is a
|
||||
* new metric name, the metric value will be initialized to the value. Does nothing if the trace
|
||||
* has not been started or has already been stopped.
|
||||
*
|
||||
* @param metricName The name of the metric to increment.
|
||||
* @param incrementValue The value to increment the metric by.
|
||||
*/
|
||||
- (void)incrementMetric:(nonnull NSString *)metricName byInt:(int64_t)incrementValue
|
||||
NS_SWIFT_NAME(incrementMetric(_:by:));
|
||||
|
||||
/**
|
||||
* Gets the value of the metric for the provided metric name. If the metric doesn't exist, a 0 is
|
||||
* returned.
|
||||
*
|
||||
* @param metricName The name of metric whose value to get.
|
||||
* @return The value of the given metric or 0 if it hasn't yet been set.
|
||||
*/
|
||||
- (int64_t)valueForIntMetric:(nonnull NSString *)metricName
|
||||
NS_SWIFT_NAME(valueForMetric(_:));
|
||||
|
||||
/**
|
||||
* Sets the value of the metric for the provided metric name to the provided value. Does nothing if
|
||||
* the trace has not been started or has already been stopped.
|
||||
*
|
||||
* @param metricName The name of the metric to set.
|
||||
* @param value The value to set the metric to.
|
||||
*/
|
||||
- (void)setIntValue:(int64_t)value forMetric:(nonnull NSString *)metricName
|
||||
NS_SWIFT_NAME(setValue(_:forMetric:));
|
||||
|
||||
@end
|
|
@ -1,4 +0,0 @@
|
|||
#import "FIRHTTPMetric.h"
|
||||
#import "FIRPerformance.h"
|
||||
#import "FIRPerformanceAttributable.h"
|
||||
#import "FIRTrace.h"
|
|
@ -1,14 +0,0 @@
|
|||
framework module FirebasePerformance {
|
||||
umbrella header "FirebasePerformance.h"
|
||||
export *
|
||||
module * { export * }
|
||||
link "c++"
|
||||
link "sqlite3"
|
||||
link "z"
|
||||
link framework "CoreTelephony"
|
||||
link framework "QuartzCore"
|
||||
link framework "Security"
|
||||
link framework "StoreKit"
|
||||
link framework "SystemConfiguration"
|
||||
link framework "UIKit"
|
||||
}
|
|
@ -1,17 +0,0 @@
|
|||
# Firebase Performance
|
||||
|
||||
Firebase Performance is a free mobile app performance analytics service. It
|
||||
provides detailed information about the performance of your apps automatically,
|
||||
but works at its best with Timers set by you. For more information about app
|
||||
performance and many other cool mobile services, check out [Firebase]
|
||||
(https://firebase.google.com).
|
||||
|
||||
## Getting Started with Cocoapods
|
||||
|
||||
1. Follow the instructions for
|
||||
[setting up Firebase](https://developers.google.com/firebase/docs/ios/)
|
||||
2. Add the following to your Podfile
|
||||
|
||||
```
|
||||
pod 'Firebase/Performance'
|
||||
```
|
|
@ -1,93 +0,0 @@
|
|||
Version 3.1.0
|
||||
==================================
|
||||
- Internal changes to support the new version of Firebase Performance SDK.
|
||||
|
||||
Version 3.0.2
|
||||
==================================
|
||||
- Bug fixes.
|
||||
|
||||
Version 3.0.1
|
||||
==================================
|
||||
- Bug fix for a memory leak bug. (#488)
|
||||
|
||||
|
||||
Version 3.0.0
|
||||
==================================
|
||||
- Change the designated initializer for FIRRemoteConfigSettings to return a nonnull FIRRemoteConfigSettings object.
|
||||
|
||||
Version 2.1.3
|
||||
==================================
|
||||
- Improve documentation on GDPR usage.
|
||||
|
||||
Version 2.1.2
|
||||
==================================
|
||||
- Improve language targeting. Simplied Chinese (zh_hans) and Traditional Chinese (Taiwan) (zh_TW) language targeting should also be more accurate.
|
||||
|
||||
Version 2.1.1
|
||||
==================================
|
||||
- Fix an issue that throttle rate drops during developer mode.
|
||||
- Replaced FIR_SWIFT_NAME with NS_SWIFT_NAME.
|
||||
|
||||
Version 2.1.0
|
||||
==================================
|
||||
- Add ABTesting feature to allow developers to run experiments using Remote Config.
|
||||
|
||||
Version 2.0.3
|
||||
==================================
|
||||
- Resolved an issue that config values are not updating correctly when targeted by a user property condition.
|
||||
|
||||
Version 2.0.2
|
||||
==================================
|
||||
- Fix an issue that prevent app from crashing when main bundle ID is missing. Also notify developers remote config might not work if main bundle ID is missing.
|
||||
|
||||
Version 2.0.1
|
||||
==================================
|
||||
- Add a warning message if a plist file can't be found when setting default values from it.
|
||||
- Internal clean up removing code for testing that is no longer used.
|
||||
|
||||
Version 2.0.0
|
||||
==================================
|
||||
- Change Swift API names to better align with Swift convention.
|
||||
- Change Error message to debug message when getting InstanceID operation is in progress as this is an expected behavior.
|
||||
|
||||
Version 1.3.4
|
||||
==================================
|
||||
- Fix the issue with Remote Config getting an incorrect configuration when user configured multiple projects.
|
||||
- Fix the issue with existing users getting empty config results.
|
||||
|
||||
Version 1.3.3
|
||||
==================================
|
||||
- Switches to the new Protobuf from ProtocolBuffers2.
|
||||
|
||||
Version 1.3.2
|
||||
==================================
|
||||
Resolved Issues:
|
||||
- Fix an issue that activateFetched called when app starts will remove cached results.
|
||||
- Fix an issue that multiple fetches without activateFetched will not get recent changes.
|
||||
|
||||
Version 1.3.1
|
||||
==================================
|
||||
Resolved Issues:
|
||||
- Better documentation on the public headers.
|
||||
|
||||
Version 1.3.0
|
||||
==================================
|
||||
Features:
|
||||
- Support user property targeting for analytics abilities.
|
||||
|
||||
Resolved Issues:
|
||||
- Fix critical crashes due to concurrent fetches, make it more thread safe.
|
||||
|
||||
Version 1.2.0
|
||||
==================================
|
||||
Features:
|
||||
- Add two new API methods to allow developers to get all the keys based on a key prefix.
|
||||
|
||||
Resolved Issues:
|
||||
- Fix a crash issue during fetching config.
|
||||
- Clarify the confusion on the documents of activateFetched method.
|
||||
- Correct the cast error in the comment of remoteConfig method.
|
||||
|
||||
Version 1.1.1
|
||||
==================================
|
||||
Initial release in Google I/O 2016.
|
Binary file not shown.
|
@ -1,246 +0,0 @@
|
|||
//
|
||||
// FIRRemoteConfig.h
|
||||
// Firebase Remote Config service SDK
|
||||
// Copyright 2016 Google Inc. All rights reserved.
|
||||
//
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
/// The Firebase Remote Config service default namespace, to be used if the API method does not
|
||||
/// specify a different namespace. Use the default namespace if configuring from the Google Firebase
|
||||
/// service.
|
||||
extern NSString *const __nonnull FIRNamespaceGoogleMobilePlatform
|
||||
NS_SWIFT_NAME(NamespaceGoogleMobilePlatform);
|
||||
|
||||
/// Key used to manage throttling in NSError user info when the refreshing of Remote Config
|
||||
/// parameter values (data) is throttled. The value of this key is the elapsed time since 1970,
|
||||
/// measured in seconds.
|
||||
extern NSString *const __nonnull FIRRemoteConfigThrottledEndTimeInSecondsKey
|
||||
NS_SWIFT_NAME(RemoteConfigThrottledEndTimeInSecondsKey);
|
||||
|
||||
/// Indicates whether updated data was successfully fetched.
|
||||
typedef NS_ENUM(NSInteger, FIRRemoteConfigFetchStatus) {
|
||||
/// Config has never been fetched.
|
||||
FIRRemoteConfigFetchStatusNoFetchYet,
|
||||
/// Config fetch succeeded.
|
||||
FIRRemoteConfigFetchStatusSuccess,
|
||||
/// Config fetch failed.
|
||||
FIRRemoteConfigFetchStatusFailure,
|
||||
/// Config fetch was throttled.
|
||||
FIRRemoteConfigFetchStatusThrottled,
|
||||
} NS_SWIFT_NAME(RemoteConfigFetchStatus);
|
||||
|
||||
/// Remote Config error domain that handles errors when fetching data from the service.
|
||||
extern NSString *const __nonnull FIRRemoteConfigErrorDomain NS_SWIFT_NAME(RemoteConfigErrorDomain);
|
||||
/// Firebase Remote Config service fetch error.
|
||||
typedef NS_ENUM(NSInteger, FIRRemoteConfigError) {
|
||||
/// Unknown or no error.
|
||||
FIRRemoteConfigErrorUnknown = 8001,
|
||||
/// Frequency of fetch requests exceeds throttled limit.
|
||||
FIRRemoteConfigErrorThrottled = 8002,
|
||||
/// Internal error that covers all internal HTTP errors.
|
||||
FIRRemoteConfigErrorInternalError = 8003,
|
||||
} NS_SWIFT_NAME(RemoteConfigError);
|
||||
|
||||
/// Enumerated value that indicates the source of Remote Config data. Data can come from
|
||||
/// the Remote Config service, the DefaultConfig that is available when the app is first installed,
|
||||
/// or a static initialized value if data is not available from the service or DefaultConfig.
|
||||
typedef NS_ENUM(NSInteger, FIRRemoteConfigSource) {
|
||||
FIRRemoteConfigSourceRemote, ///< The data source is the Remote Config service.
|
||||
FIRRemoteConfigSourceDefault, ///< The data source is the DefaultConfig defined for this app.
|
||||
FIRRemoteConfigSourceStatic, ///< The data doesn't exist, return a static initialized value.
|
||||
} NS_SWIFT_NAME(RemoteConfigSource);
|
||||
|
||||
/// Completion handler invoked by fetch methods when they get a response from the server.
|
||||
///
|
||||
/// @param status Config fetching status.
|
||||
/// @param error Error message on failure.
|
||||
typedef void (^FIRRemoteConfigFetchCompletion)(FIRRemoteConfigFetchStatus status,
|
||||
NSError *__nullable error)
|
||||
NS_SWIFT_NAME(RemoteConfigFetchCompletion);
|
||||
|
||||
#pragma mark - FIRRemoteConfigValue
|
||||
/// This class provides a wrapper for Remote Config parameter values, with methods to get parameter
|
||||
/// values as different data types.
|
||||
NS_SWIFT_NAME(RemoteConfigValue)
|
||||
@interface FIRRemoteConfigValue : NSObject<NSCopying>
|
||||
/// Gets the value as a string.
|
||||
@property(nonatomic, readonly, nullable) NSString *stringValue;
|
||||
/// Gets the value as a number value.
|
||||
@property(nonatomic, readonly, nullable) NSNumber *numberValue;
|
||||
/// Gets the value as a NSData object.
|
||||
@property(nonatomic, readonly, nonnull) NSData *dataValue;
|
||||
/// Gets the value as a boolean.
|
||||
@property(nonatomic, readonly) BOOL boolValue;
|
||||
/// Identifies the source of the fetched value.
|
||||
@property(nonatomic, readonly) FIRRemoteConfigSource source;
|
||||
@end
|
||||
|
||||
#pragma mark - FIRRemoteConfigSettings
|
||||
/// Firebase Remote Config settings.
|
||||
NS_SWIFT_NAME(RemoteConfigSettings)
|
||||
@interface FIRRemoteConfigSettings : NSObject
|
||||
/// Indicates whether Developer Mode is enabled.
|
||||
@property(nonatomic, readonly) BOOL isDeveloperModeEnabled;
|
||||
/// Initializes FIRRemoteConfigSettings, which is used to set properties for custom settings. To
|
||||
/// make custom settings take effect, pass the FIRRemoteConfigSettings instance to the
|
||||
/// configSettings property of FIRRemoteConfig.
|
||||
- (nonnull FIRRemoteConfigSettings *)initWithDeveloperModeEnabled:(BOOL)developerModeEnabled
|
||||
NS_DESIGNATED_INITIALIZER;
|
||||
@end
|
||||
|
||||
#pragma mark - FIRRemoteConfig
|
||||
/// Firebase Remote Config class. The shared instance method +remoteConfig can be created and used
|
||||
/// to fetch, activate and read config results and set default config results.
|
||||
NS_SWIFT_NAME(RemoteConfig)
|
||||
@interface FIRRemoteConfig : NSObject<NSFastEnumeration>
|
||||
/// Last successful fetch completion time.
|
||||
@property(nonatomic, readonly, strong, nullable) NSDate *lastFetchTime;
|
||||
/// Last fetch status. The status can be any enumerated value from FIRRemoteConfigFetchStatus.
|
||||
@property(nonatomic, readonly, assign) FIRRemoteConfigFetchStatus lastFetchStatus;
|
||||
/// Config settings are custom settings.
|
||||
@property(nonatomic, readwrite, strong, nonnull) FIRRemoteConfigSettings *configSettings;
|
||||
|
||||
/// Returns the FIRRemoteConfig instance shared throughout your app. This singleton object contains
|
||||
/// the complete set of Remote Config parameter values available to the app, including the Active
|
||||
/// Config and Default Config. This object also caches values fetched from the Remote Config Server
|
||||
/// until they are copied to the Active Config by calling activateFetched.
|
||||
/// When you fetch values from the Remote Config Server using the default Firebase namespace
|
||||
/// service, you should use this class method to create a shared instance of the FIRRemoteConfig
|
||||
/// object to ensure that your app will function properly with the Remote Config Server and the
|
||||
/// Firebase service.
|
||||
+ (nonnull FIRRemoteConfig *)remoteConfig NS_SWIFT_NAME(remoteConfig());
|
||||
|
||||
/// Unavailable. Use +remoteConfig instead.
|
||||
- (nonnull instancetype)init __attribute__((unavailable("Use +remoteConfig instead.")));
|
||||
|
||||
#pragma mark - Fetch
|
||||
/// Fetches Remote Config data with a callback. Call activateFetched to make fetched data available
|
||||
/// to your app.
|
||||
///
|
||||
/// Note: This method uses a Firebase Instance ID token to identify the app instance, and once it's
|
||||
/// called, it periodically sends data to the Firebase backend. (see
|
||||
/// `[FIRInstanceID getIDWithHandler:]`).
|
||||
/// To stop the periodic sync, developers need to call `[FIRInstanceID deleteIDWithHandler:]` and
|
||||
/// avoid calling this method again.
|
||||
///
|
||||
/// @param completionHandler Fetch operation callback.
|
||||
- (void)fetchWithCompletionHandler:(nullable FIRRemoteConfigFetchCompletion)completionHandler;
|
||||
|
||||
/// Fetches Remote Config data and sets a duration that specifies how long config data lasts.
|
||||
/// Call activateFetched to make fetched data available to your app.
|
||||
///
|
||||
/// Note: This method uses a Firebase Instance ID token to identify the app instance, and once it's
|
||||
/// called, it periodically sends data to the Firebase backend. (see
|
||||
/// `[FIRInstanceID getIDWithHandler:]`).
|
||||
/// To stop the periodic sync, developers need to call `[FIRInstanceID deleteIDWithHandler:]` and
|
||||
/// avoid calling this method again.
|
||||
///
|
||||
/// @param expirationDuration Duration that defines how long fetched config data is available, in
|
||||
/// seconds. When the config data expires, a new fetch is required.
|
||||
/// @param completionHandler Fetch operation callback.
|
||||
- (void)fetchWithExpirationDuration:(NSTimeInterval)expirationDuration
|
||||
completionHandler:(nullable FIRRemoteConfigFetchCompletion)completionHandler;
|
||||
|
||||
#pragma mark - Apply
|
||||
/// Applies Fetched Config data to the Active Config, causing updates to the behavior and appearance
|
||||
/// of the app to take effect (depending on how config data is used in the app).
|
||||
/// Returns true if there was a Fetched Config, and it was activated.
|
||||
/// Returns false if no Fetched Config was found, or the Fetched Config was already activated.
|
||||
- (BOOL)activateFetched;
|
||||
|
||||
#pragma mark - Get Config
|
||||
/// Enables access to configuration values by using object subscripting syntax.
|
||||
/// This is used to get the config value of the default namespace.
|
||||
/// <pre>
|
||||
/// // Example:
|
||||
/// FIRRemoteConfig *config = [FIRRemoteConfig remoteConfig];
|
||||
/// FIRRemoteConfigValue *value = config[@"yourKey"];
|
||||
/// BOOL b = value.boolValue;
|
||||
/// NSNumber *number = config[@"yourKey"].numberValue;
|
||||
/// </pre>
|
||||
- (nonnull FIRRemoteConfigValue *)objectForKeyedSubscript:(nonnull NSString *)key;
|
||||
|
||||
/// Gets the config value of the default namespace.
|
||||
/// @param key Config key.
|
||||
- (nonnull FIRRemoteConfigValue *)configValueForKey:(nullable NSString *)key;
|
||||
|
||||
/// Gets the config value of a given namespace.
|
||||
/// @param key Config key.
|
||||
/// @param aNamespace Config results under a given namespace.
|
||||
- (nonnull FIRRemoteConfigValue *)configValueForKey:(nullable NSString *)key
|
||||
namespace:(nullable NSString *)aNamespace;
|
||||
|
||||
/// Gets the config value of a given namespace and a given source.
|
||||
/// @param key Config key.
|
||||
/// @param aNamespace Config results under a given namespace.
|
||||
/// @param source Config value source.
|
||||
- (nonnull FIRRemoteConfigValue *)configValueForKey:(nullable NSString *)key
|
||||
namespace:(nullable NSString *)aNamespace
|
||||
source:(FIRRemoteConfigSource)source;
|
||||
|
||||
/// Gets all the parameter keys from a given source and a given namespace.
|
||||
///
|
||||
/// @param source The config data source.
|
||||
/// @param aNamespace The config data namespace.
|
||||
/// @return An array of keys under the given source and namespace.
|
||||
- (nonnull NSArray<NSString *> *)allKeysFromSource:(FIRRemoteConfigSource)source
|
||||
namespace:(nullable NSString *)aNamespace;
|
||||
|
||||
/// Returns the set of parameter keys that start with the given prefix, from the default namespace
|
||||
/// in the active config.
|
||||
///
|
||||
/// @param prefix The key prefix to look for. If prefix is nil or empty, returns all the
|
||||
/// keys.
|
||||
/// @return The set of parameter keys that start with the specified prefix.
|
||||
- (nonnull NSSet<NSString *> *)keysWithPrefix:(nullable NSString *)prefix;
|
||||
|
||||
/// Returns the set of parameter keys that start with the given prefix, from the given namespace in
|
||||
/// the active config.
|
||||
///
|
||||
/// @param prefix The key prefix to look for. If prefix is nil or empty, returns all the
|
||||
/// keys in the given namespace.
|
||||
/// @param aNamespace The namespace in which to look up the keys. If the namespace is invalid,
|
||||
/// returns an empty set.
|
||||
/// @return The set of parameter keys that start with the specified prefix.
|
||||
- (nonnull NSSet<NSString *> *)keysWithPrefix:(nullable NSString *)prefix
|
||||
namespace:(nullable NSString *)aNamespace;
|
||||
|
||||
#pragma mark - Defaults
|
||||
/// Sets config defaults for parameter keys and values in the default namespace config.
|
||||
///
|
||||
/// @param defaults A dictionary mapping a NSString * key to a NSObject * value.
|
||||
- (void)setDefaults:(nullable NSDictionary<NSString *, NSObject *> *)defaults;
|
||||
|
||||
/// Sets config defaults for parameter keys and values in the default namespace config.
|
||||
///
|
||||
/// @param defaults A dictionary mapping a NSString * key to a NSObject * value.
|
||||
/// @param aNamespace Config under a given namespace.
|
||||
- (void)setDefaults:(nullable NSDictionary<NSString *, NSObject *> *)defaults
|
||||
namespace:(nullable NSString *)aNamespace;
|
||||
|
||||
/// Sets default configs from plist for default namespace;
|
||||
/// @param fileName The plist file name, with no file name extension. For example, if the plist file
|
||||
/// is defaultSamples.plist, call:
|
||||
/// [[FIRRemoteConfig remoteConfig] setDefaultsFromPlistFileName:@"defaultSamples"];
|
||||
- (void)setDefaultsFromPlistFileName:(nullable NSString *)fileName
|
||||
NS_SWIFT_NAME(setDefaults(fromPlist:));
|
||||
|
||||
/// Sets default configs from plist for a given namespace;
|
||||
/// @param fileName The plist file name, with no file name extension. For example, if the plist file
|
||||
/// is defaultSamples.plist, call:
|
||||
/// [[FIRRemoteConfig remoteConfig] setDefaultsFromPlistFileName:@"defaultSamples"];
|
||||
/// @param aNamespace The namespace where the default config is set.
|
||||
- (void)setDefaultsFromPlistFileName:(nullable NSString *)fileName
|
||||
namespace:(nullable NSString *)aNamespace
|
||||
NS_SWIFT_NAME(setDefaults(fromPlist:namespace:));
|
||||
|
||||
/// Returns the default value of a given key and a given namespace from the default config.
|
||||
///
|
||||
/// @param key The parameter key of default config.
|
||||
/// @param aNamespace The namespace of default config.
|
||||
/// @return Returns the default value of the specified key and namespace. Returns
|
||||
/// nil if the key or namespace doesn't exist in the default config.
|
||||
- (nullable FIRRemoteConfigValue *)defaultValueForKey:(nullable NSString *)key
|
||||
namespace:(nullable NSString *)aNamespace;
|
||||
|
||||
@end
|
|
@ -1 +0,0 @@
|
|||
#import "FIRRemoteConfig.h"
|
|
@ -1,10 +0,0 @@
|
|||
framework module FirebaseRemoteConfig {
|
||||
umbrella header "FirebaseRemoteConfig.h"
|
||||
export *
|
||||
module * { export *}
|
||||
link "sqlite3"
|
||||
link "z"
|
||||
link framework "Security"
|
||||
link framework "StoreKit"
|
||||
link framework "SystemConfiguration"
|
||||
link framework "UIKit"}
|
|
@ -1,11 +0,0 @@
|
|||
# Firebase Remote Config SDK for iOS
|
||||
|
||||
This pod contains the Firebase Remote Config SDK for iOS, supporting both
|
||||
Objective-C and Swift.
|
||||
|
||||
Firebase Remote Config is a cloud service that lets you change the appearance
|
||||
and behavior of your app without requiring users to download an app update.
|
||||
|
||||
Please visit [our developer site]
|
||||
(https://firebase.google.com/docs/remote-config/) for integration instructions,
|
||||
documentation, support information, and terms of service.
|
|
@ -1,202 +0,0 @@
|
|||
|
||||
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.
|
|
@ -1,23 +0,0 @@
|
|||
# Google Toolbox for Mac - Session Fetcher #
|
||||
|
||||
**Project site** <https://github.com/google/gtm-session-fetcher><br>
|
||||
**Discussion group** <http://groups.google.com/group/google-toolbox-for-mac>
|
||||
|
||||
[](https://travis-ci.org/google/gtm-session-fetcher)
|
||||
|
||||
`GTMSessionFetcher` makes it easy for Cocoa applications to perform http
|
||||
operations. The fetcher is implemented as a wrapper on `NSURLSession`, so its
|
||||
behavior is asynchronous and uses operating-system settings on iOS and Mac OS X.
|
||||
|
||||
Features include:
|
||||
- Simple to build; only one source/header file pair is required
|
||||
- Simple to use: takes just two lines of code to fetch a request
|
||||
- Supports upload and download sessions
|
||||
- Flexible cookie storage
|
||||
- Automatic retry on errors, with exponential backoff
|
||||
- Support for generating multipart MIME upload streams
|
||||
- Easy, convenient logging of http requests and responses
|
||||
- Supports plug-in authentication such as with GTMAppAuth
|
||||
- Easily testable; self-mocking
|
||||
- Automatic rate limiting when created by the `GTMSessionFetcherService` factory class
|
||||
- Fully independent of other projects
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -1,112 +0,0 @@
|
|||
/* Copyright 2014 Google Inc. All rights reserved.
|
||||
*
|
||||
* 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 "GTMSessionFetcher.h"
|
||||
|
||||
// GTM HTTP Logging
|
||||
//
|
||||
// All traffic using GTMSessionFetcher can be easily logged. Call
|
||||
//
|
||||
// [GTMSessionFetcher setLoggingEnabled:YES];
|
||||
//
|
||||
// to begin generating log files.
|
||||
//
|
||||
// Unless explicitly set by the application using +setLoggingDirectory:,
|
||||
// logs are put into a default directory, located at:
|
||||
// * macOS: ~/Desktop/GTMHTTPDebugLogs
|
||||
// * iOS simulator: ~/GTMHTTPDebugLogs (in application sandbox)
|
||||
// * iOS device: ~/Documents/GTMHTTPDebugLogs (in application sandbox)
|
||||
//
|
||||
// Tip: use the Finder's "Sort By Date" to find the most recent logs.
|
||||
//
|
||||
// Each run of an application gets a separate set of log files. An html
|
||||
// file is generated to simplify browsing the run's http transactions.
|
||||
// The html file includes javascript links for inline viewing of uploaded
|
||||
// and downloaded data.
|
||||
//
|
||||
// A symlink is created in the logs folder to simplify finding the html file
|
||||
// for the latest run of the application; the symlink is called
|
||||
//
|
||||
// AppName_http_log_newest.html
|
||||
//
|
||||
// For better viewing of XML logs, use Camino or Firefox rather than Safari.
|
||||
//
|
||||
// Each fetcher may be given a comment to be inserted as a label in the logs,
|
||||
// such as
|
||||
// [fetcher setCommentWithFormat:@"retrieve item %@", itemName];
|
||||
//
|
||||
// Projects may define STRIP_GTM_FETCH_LOGGING to remove logging code.
|
||||
|
||||
#if !STRIP_GTM_FETCH_LOGGING
|
||||
|
||||
@interface GTMSessionFetcher (GTMSessionFetcherLogging)
|
||||
|
||||
// Note: on macOS the default logs directory is ~/Desktop/GTMHTTPDebugLogs; on
|
||||
// iOS simulators it will be the ~/GTMHTTPDebugLogs (in the app sandbox); on
|
||||
// iOS devices it will be in ~/Documents/GTMHTTPDebugLogs (in the app sandbox).
|
||||
// These directories will be created as needed, and are excluded from backups
|
||||
// to iCloud and iTunes.
|
||||
//
|
||||
// If a custom directory is set, the directory should already exist. It is
|
||||
// the application's responsibility to exclude any custom directory from
|
||||
// backups, if desired.
|
||||
+ (void)setLoggingDirectory:(NSString *)path;
|
||||
+ (NSString *)loggingDirectory;
|
||||
|
||||
// client apps can turn logging on and off
|
||||
+ (void)setLoggingEnabled:(BOOL)isLoggingEnabled;
|
||||
+ (BOOL)isLoggingEnabled;
|
||||
|
||||
// client apps can turn off logging to a file if they want to only check
|
||||
// the fetcher's log property
|
||||
+ (void)setLoggingToFileEnabled:(BOOL)isLoggingToFileEnabled;
|
||||
+ (BOOL)isLoggingToFileEnabled;
|
||||
|
||||
// client apps can optionally specify process name and date string used in
|
||||
// log file names
|
||||
+ (void)setLoggingProcessName:(NSString *)processName;
|
||||
+ (NSString *)loggingProcessName;
|
||||
|
||||
+ (void)setLoggingDateStamp:(NSString *)dateStamp;
|
||||
+ (NSString *)loggingDateStamp;
|
||||
|
||||
// client apps can specify the directory for the log for this specific run,
|
||||
// typically to match the directory used by another fetcher class, like:
|
||||
//
|
||||
// [GTMSessionFetcher setLogDirectoryForCurrentRun:[GTMHTTPFetcher logDirectoryForCurrentRun]];
|
||||
//
|
||||
// Setting this overrides the logging directory, process name, and date stamp when writing
|
||||
// the log file.
|
||||
+ (void)setLogDirectoryForCurrentRun:(NSString *)logDirectoryForCurrentRun;
|
||||
+ (NSString *)logDirectoryForCurrentRun;
|
||||
|
||||
// Prunes old log directories that have not been modified since the provided date.
|
||||
// This will not delete the current run's log directory.
|
||||
+ (void)deleteLogDirectoriesOlderThanDate:(NSDate *)date;
|
||||
|
||||
// internal; called by fetcher
|
||||
- (void)logFetchWithError:(NSError *)error;
|
||||
- (NSInputStream *)loggedInputStreamForInputStream:(NSInputStream *)inputStream;
|
||||
- (GTMSessionFetcherBodyStreamProvider)loggedStreamProviderForStreamProvider:
|
||||
(GTMSessionFetcherBodyStreamProvider)streamProvider;
|
||||
|
||||
// internal; accessors useful for viewing logs
|
||||
+ (NSString *)processNameLogPrefix;
|
||||
+ (NSString *)symlinkNameSuffix;
|
||||
+ (NSString *)htmlFileName;
|
||||
|
||||
@end
|
||||
|
||||
#endif // !STRIP_GTM_FETCH_LOGGING
|
|
@ -1,982 +0,0 @@
|
|||
/* Copyright 2014 Google Inc. All rights reserved.
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
#if !defined(__has_feature) || !__has_feature(objc_arc)
|
||||
#error "This file requires ARC support."
|
||||
#endif
|
||||
|
||||
#include <sys/stat.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#import "GTMSessionFetcherLogging.h"
|
||||
|
||||
#ifndef STRIP_GTM_FETCH_LOGGING
|
||||
#error GTMSessionFetcher headers should have defaulted this if it wasn't already defined.
|
||||
#endif
|
||||
|
||||
#if !STRIP_GTM_FETCH_LOGGING
|
||||
|
||||
// Sensitive credential strings are replaced in logs with _snip_
|
||||
//
|
||||
// Apps that must see the contents of sensitive tokens can set this to 1
|
||||
#ifndef SKIP_GTM_FETCH_LOGGING_SNIPPING
|
||||
#define SKIP_GTM_FETCH_LOGGING_SNIPPING 0
|
||||
#endif
|
||||
|
||||
// If GTMReadMonitorInputStream is available, it can be used for
|
||||
// capturing uploaded streams of data
|
||||
//
|
||||
// We locally declare methods of GTMReadMonitorInputStream so we
|
||||
// do not need to import the header, as some projects may not have it available
|
||||
#if !GTMSESSION_BUILD_COMBINED_SOURCES
|
||||
@interface GTMReadMonitorInputStream : NSInputStream
|
||||
|
||||
+ (instancetype)inputStreamWithStream:(NSInputStream *)input;
|
||||
|
||||
@property (assign) id readDelegate;
|
||||
@property (assign) SEL readSelector;
|
||||
|
||||
@end
|
||||
#else
|
||||
@class GTMReadMonitorInputStream;
|
||||
#endif // !GTMSESSION_BUILD_COMBINED_SOURCES
|
||||
|
||||
@interface GTMSessionFetcher (GTMHTTPFetcherLoggingUtilities)
|
||||
|
||||
+ (NSString *)headersStringForDictionary:(NSDictionary *)dict;
|
||||
+ (NSString *)snipSubstringOfString:(NSString *)originalStr
|
||||
betweenStartString:(NSString *)startStr
|
||||
endString:(NSString *)endStr;
|
||||
- (void)inputStream:(GTMReadMonitorInputStream *)stream
|
||||
readIntoBuffer:(void *)buffer
|
||||
length:(int64_t)length;
|
||||
|
||||
@end
|
||||
|
||||
@implementation GTMSessionFetcher (GTMSessionFetcherLogging)
|
||||
|
||||
// fetchers come and fetchers go, but statics are forever
|
||||
static BOOL gIsLoggingEnabled = NO;
|
||||
static BOOL gIsLoggingToFile = YES;
|
||||
static NSString *gLoggingDirectoryPath = nil;
|
||||
static NSString *gLogDirectoryForCurrentRun = nil;
|
||||
static NSString *gLoggingDateStamp = nil;
|
||||
static NSString *gLoggingProcessName = nil;
|
||||
|
||||
+ (void)setLoggingDirectory:(NSString *)path {
|
||||
gLoggingDirectoryPath = [path copy];
|
||||
}
|
||||
|
||||
+ (NSString *)loggingDirectory {
|
||||
if (!gLoggingDirectoryPath) {
|
||||
NSArray *paths = nil;
|
||||
#if TARGET_IPHONE_SIMULATOR
|
||||
// default to a directory called GTMHTTPDebugLogs into a sandbox-safe
|
||||
// directory that a developer can find easily, the application home
|
||||
paths = @[ NSHomeDirectory() ];
|
||||
#elif TARGET_OS_IPHONE
|
||||
// Neither ~/Desktop nor ~/Home is writable on an actual iOS, watchOS, or tvOS device.
|
||||
// Put it in ~/Documents.
|
||||
paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
|
||||
#else
|
||||
// default to a directory called GTMHTTPDebugLogs in the desktop folder
|
||||
paths = NSSearchPathForDirectoriesInDomains(NSDesktopDirectory, NSUserDomainMask, YES);
|
||||
#endif
|
||||
|
||||
NSString *desktopPath = paths.firstObject;
|
||||
if (desktopPath) {
|
||||
NSString *const kGTMLogFolderName = @"GTMHTTPDebugLogs";
|
||||
NSString *logsFolderPath = [desktopPath stringByAppendingPathComponent:kGTMLogFolderName];
|
||||
|
||||
NSFileManager *fileMgr = [NSFileManager defaultManager];
|
||||
BOOL isDir;
|
||||
BOOL doesFolderExist = [fileMgr fileExistsAtPath:logsFolderPath isDirectory:&isDir];
|
||||
if (!doesFolderExist) {
|
||||
// make the directory
|
||||
doesFolderExist = [fileMgr createDirectoryAtPath:logsFolderPath
|
||||
withIntermediateDirectories:YES
|
||||
attributes:nil
|
||||
error:NULL];
|
||||
if (doesFolderExist) {
|
||||
// The directory has been created. Exclude it from backups.
|
||||
NSURL *pathURL = [NSURL fileURLWithPath:logsFolderPath isDirectory:YES];
|
||||
[pathURL setResourceValue:@YES forKey:NSURLIsExcludedFromBackupKey error:NULL];
|
||||
}
|
||||
}
|
||||
|
||||
if (doesFolderExist) {
|
||||
// it's there; store it in the global
|
||||
gLoggingDirectoryPath = [logsFolderPath copy];
|
||||
}
|
||||
}
|
||||
}
|
||||
return gLoggingDirectoryPath;
|
||||
}
|
||||
|
||||
+ (void)setLogDirectoryForCurrentRun:(NSString *)logDirectoryForCurrentRun {
|
||||
// Set the path for this run's logs.
|
||||
gLogDirectoryForCurrentRun = [logDirectoryForCurrentRun copy];
|
||||
}
|
||||
|
||||
+ (NSString *)logDirectoryForCurrentRun {
|
||||
// make a directory for this run's logs, like SyncProto_logs_10-16_01-56-58PM
|
||||
if (gLogDirectoryForCurrentRun) return gLogDirectoryForCurrentRun;
|
||||
|
||||
NSString *parentDir = [self loggingDirectory];
|
||||
NSString *logNamePrefix = [self processNameLogPrefix];
|
||||
NSString *dateStamp = [self loggingDateStamp];
|
||||
NSString *dirName = [NSString stringWithFormat:@"%@%@", logNamePrefix, dateStamp];
|
||||
NSString *logDirectory = [parentDir stringByAppendingPathComponent:dirName];
|
||||
|
||||
if (gIsLoggingToFile) {
|
||||
NSFileManager *fileMgr = [NSFileManager defaultManager];
|
||||
// Be sure that the first time this app runs, it's not writing to a preexisting folder
|
||||
static BOOL gShouldReuseFolder = NO;
|
||||
if (!gShouldReuseFolder) {
|
||||
gShouldReuseFolder = YES;
|
||||
NSString *origLogDir = logDirectory;
|
||||
for (int ctr = 2; ctr < 20; ++ctr) {
|
||||
if (![fileMgr fileExistsAtPath:logDirectory]) break;
|
||||
|
||||
// append a digit
|
||||
logDirectory = [origLogDir stringByAppendingFormat:@"_%d", ctr];
|
||||
}
|
||||
}
|
||||
if (![fileMgr createDirectoryAtPath:logDirectory
|
||||
withIntermediateDirectories:YES
|
||||
attributes:nil
|
||||
error:NULL]) return nil;
|
||||
}
|
||||
gLogDirectoryForCurrentRun = logDirectory;
|
||||
|
||||
return gLogDirectoryForCurrentRun;
|
||||
}
|
||||
|
||||
+ (void)setLoggingEnabled:(BOOL)isLoggingEnabled {
|
||||
gIsLoggingEnabled = isLoggingEnabled;
|
||||
}
|
||||
|
||||
+ (BOOL)isLoggingEnabled {
|
||||
return gIsLoggingEnabled;
|
||||
}
|
||||
|
||||
+ (void)setLoggingToFileEnabled:(BOOL)isLoggingToFileEnabled {
|
||||
gIsLoggingToFile = isLoggingToFileEnabled;
|
||||
}
|
||||
|
||||
+ (BOOL)isLoggingToFileEnabled {
|
||||
return gIsLoggingToFile;
|
||||
}
|
||||
|
||||
+ (void)setLoggingProcessName:(NSString *)processName {
|
||||
gLoggingProcessName = [processName copy];
|
||||
}
|
||||
|
||||
+ (NSString *)loggingProcessName {
|
||||
// get the process name (once per run) replacing spaces with underscores
|
||||
if (!gLoggingProcessName) {
|
||||
NSString *procName = [[NSProcessInfo processInfo] processName];
|
||||
gLoggingProcessName = [procName stringByReplacingOccurrencesOfString:@" " withString:@"_"];
|
||||
}
|
||||
return gLoggingProcessName;
|
||||
}
|
||||
|
||||
+ (void)setLoggingDateStamp:(NSString *)dateStamp {
|
||||
gLoggingDateStamp = [dateStamp copy];
|
||||
}
|
||||
|
||||
+ (NSString *)loggingDateStamp {
|
||||
// We'll pick one date stamp per run, so a run that starts at a later second
|
||||
// will get a unique results html file
|
||||
if (!gLoggingDateStamp) {
|
||||
// produce a string like 08-21_01-41-23PM
|
||||
|
||||
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
|
||||
[formatter setFormatterBehavior:NSDateFormatterBehavior10_4];
|
||||
[formatter setDateFormat:@"M-dd_hh-mm-ssa"];
|
||||
|
||||
gLoggingDateStamp = [formatter stringFromDate:[NSDate date]];
|
||||
}
|
||||
return gLoggingDateStamp;
|
||||
}
|
||||
|
||||
+ (NSString *)processNameLogPrefix {
|
||||
static NSString *gPrefix = nil;
|
||||
if (!gPrefix) {
|
||||
NSString *processName = [self loggingProcessName];
|
||||
gPrefix = [[NSString alloc] initWithFormat:@"%@_log_", processName];
|
||||
}
|
||||
return gPrefix;
|
||||
}
|
||||
|
||||
+ (NSString *)symlinkNameSuffix {
|
||||
return @"_log_newest.html";
|
||||
}
|
||||
|
||||
+ (NSString *)htmlFileName {
|
||||
return @"aperçu_http_log.html";
|
||||
}
|
||||
|
||||
+ (void)deleteLogDirectoriesOlderThanDate:(NSDate *)cutoffDate {
|
||||
NSFileManager *fileMgr = [NSFileManager defaultManager];
|
||||
NSURL *parentDir = [NSURL fileURLWithPath:[[self class] loggingDirectory]];
|
||||
NSURL *logDirectoryForCurrentRun =
|
||||
[NSURL fileURLWithPath:[[self class] logDirectoryForCurrentRun]];
|
||||
NSError *error;
|
||||
NSArray *contents = [fileMgr contentsOfDirectoryAtURL:parentDir
|
||||
includingPropertiesForKeys:@[ NSURLContentModificationDateKey ]
|
||||
options:0
|
||||
error:&error];
|
||||
for (NSURL *itemURL in contents) {
|
||||
if ([itemURL isEqual:logDirectoryForCurrentRun]) continue;
|
||||
|
||||
NSDate *modDate;
|
||||
if ([itemURL getResourceValue:&modDate
|
||||
forKey:NSURLContentModificationDateKey
|
||||
error:&error]) {
|
||||
if ([modDate compare:cutoffDate] == NSOrderedAscending) {
|
||||
if (![fileMgr removeItemAtURL:itemURL error:&error]) {
|
||||
NSLog(@"deleteLogDirectoriesOlderThanDate failed to delete %@: %@",
|
||||
itemURL.path, error);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
NSLog(@"deleteLogDirectoriesOlderThanDate failed to get mod date of %@: %@",
|
||||
itemURL.path, error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// formattedStringFromData returns a prettyprinted string for XML or JSON input,
|
||||
// and a plain string for other input data
|
||||
- (NSString *)formattedStringFromData:(NSData *)inputData
|
||||
contentType:(NSString *)contentType
|
||||
JSON:(NSDictionary **)outJSON {
|
||||
if (!inputData) return nil;
|
||||
|
||||
// if the content type is JSON and we have the parsing class available, use that
|
||||
if ([contentType hasPrefix:@"application/json"] && inputData.length > 5) {
|
||||
// convert from JSON string to NSObjects and back to a formatted string
|
||||
NSMutableDictionary *obj = [NSJSONSerialization JSONObjectWithData:inputData
|
||||
options:NSJSONReadingMutableContainers
|
||||
error:NULL];
|
||||
if (obj) {
|
||||
if (outJSON) *outJSON = obj;
|
||||
if ([obj isKindOfClass:[NSMutableDictionary class]]) {
|
||||
// for security and privacy, omit OAuth 2 response access and refresh tokens
|
||||
if ([obj valueForKey:@"refresh_token"] != nil) {
|
||||
[obj setObject:@"_snip_" forKey:@"refresh_token"];
|
||||
}
|
||||
if ([obj valueForKey:@"access_token"] != nil) {
|
||||
[obj setObject:@"_snip_" forKey:@"access_token"];
|
||||
}
|
||||
}
|
||||
NSData *data = [NSJSONSerialization dataWithJSONObject:obj
|
||||
options:NSJSONWritingPrettyPrinted
|
||||
error:NULL];
|
||||
if (data) {
|
||||
NSString *jsonStr = [[NSString alloc] initWithData:data
|
||||
encoding:NSUTF8StringEncoding];
|
||||
return jsonStr;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#if !TARGET_OS_IPHONE && !GTM_SKIP_LOG_XMLFORMAT
|
||||
// verify that this data starts with the bytes indicating XML
|
||||
|
||||
NSString *const kXMLLintPath = @"/usr/bin/xmllint";
|
||||
static BOOL gHasCheckedAvailability = NO;
|
||||
static BOOL gIsXMLLintAvailable = NO;
|
||||
|
||||
if (!gHasCheckedAvailability) {
|
||||
gIsXMLLintAvailable = [[NSFileManager defaultManager] fileExistsAtPath:kXMLLintPath];
|
||||
gHasCheckedAvailability = YES;
|
||||
}
|
||||
if (gIsXMLLintAvailable
|
||||
&& inputData.length > 5
|
||||
&& strncmp(inputData.bytes, "<?xml", 5) == 0) {
|
||||
|
||||
// call xmllint to format the data
|
||||
NSTask *task = [[NSTask alloc] init];
|
||||
[task setLaunchPath:kXMLLintPath];
|
||||
|
||||
// use the dash argument to specify stdin as the source file
|
||||
[task setArguments:@[ @"--format", @"-" ]];
|
||||
[task setEnvironment:@{}];
|
||||
|
||||
NSPipe *inputPipe = [NSPipe pipe];
|
||||
NSPipe *outputPipe = [NSPipe pipe];
|
||||
[task setStandardInput:inputPipe];
|
||||
[task setStandardOutput:outputPipe];
|
||||
|
||||
[task launch];
|
||||
|
||||
[[inputPipe fileHandleForWriting] writeData:inputData];
|
||||
[[inputPipe fileHandleForWriting] closeFile];
|
||||
|
||||
// drain the stdout before waiting for the task to exit
|
||||
NSData *formattedData = [[outputPipe fileHandleForReading] readDataToEndOfFile];
|
||||
|
||||
[task waitUntilExit];
|
||||
|
||||
int status = [task terminationStatus];
|
||||
if (status == 0 && formattedData.length > 0) {
|
||||
// success
|
||||
inputData = formattedData;
|
||||
}
|
||||
}
|
||||
#else
|
||||
// we can't call external tasks on the iPhone; leave the XML unformatted
|
||||
#endif
|
||||
|
||||
NSString *dataStr = [[NSString alloc] initWithData:inputData
|
||||
encoding:NSUTF8StringEncoding];
|
||||
return dataStr;
|
||||
}
|
||||
|
||||
// stringFromStreamData creates a string given the supplied data
|
||||
//
|
||||
// If NSString can create a UTF-8 string from the data, then that is returned.
|
||||
//
|
||||
// Otherwise, this routine tries to find a MIME boundary at the beginning of the data block, and
|
||||
// uses that to break up the data into parts. Each part will be used to try to make a UTF-8 string.
|
||||
// For parts that fail, a replacement string showing the part header and <<n bytes>> is supplied
|
||||
// in place of the binary data.
|
||||
|
||||
- (NSString *)stringFromStreamData:(NSData *)data
|
||||
contentType:(NSString *)contentType {
|
||||
|
||||
if (!data) return nil;
|
||||
|
||||
// optimistically, see if the whole data block is UTF-8
|
||||
NSString *streamDataStr = [self formattedStringFromData:data
|
||||
contentType:contentType
|
||||
JSON:NULL];
|
||||
if (streamDataStr) return streamDataStr;
|
||||
|
||||
// Munge a buffer by replacing non-ASCII bytes with underscores, and turn that munged buffer an
|
||||
// NSString. That gives us a string we can use with NSScanner.
|
||||
NSMutableData *mutableData = [NSMutableData dataWithData:data];
|
||||
unsigned char *bytes = (unsigned char *)mutableData.mutableBytes;
|
||||
|
||||
for (unsigned int idx = 0; idx < mutableData.length; ++idx) {
|
||||
if (bytes[idx] > 0x7F || bytes[idx] == 0) {
|
||||
bytes[idx] = '_';
|
||||
}
|
||||
}
|
||||
|
||||
NSString *mungedStr = [[NSString alloc] initWithData:mutableData
|
||||
encoding:NSUTF8StringEncoding];
|
||||
if (mungedStr) {
|
||||
|
||||
// scan for the boundary string
|
||||
NSString *boundary = nil;
|
||||
NSScanner *scanner = [NSScanner scannerWithString:mungedStr];
|
||||
|
||||
if ([scanner scanUpToString:@"\r\n" intoString:&boundary]
|
||||
&& [boundary hasPrefix:@"--"]) {
|
||||
|
||||
// we found a boundary string; use it to divide the string into parts
|
||||
NSArray *mungedParts = [mungedStr componentsSeparatedByString:boundary];
|
||||
|
||||
// look at each munged part in the original string, and try to convert those into UTF-8
|
||||
NSMutableArray *origParts = [NSMutableArray array];
|
||||
NSUInteger offset = 0;
|
||||
for (NSString *mungedPart in mungedParts) {
|
||||
NSUInteger partSize = mungedPart.length;
|
||||
NSData *origPartData = [data subdataWithRange:NSMakeRange(offset, partSize)];
|
||||
NSString *origPartStr = [[NSString alloc] initWithData:origPartData
|
||||
encoding:NSUTF8StringEncoding];
|
||||
if (origPartStr) {
|
||||
// we could make this original part into UTF-8; use the string
|
||||
[origParts addObject:origPartStr];
|
||||
} else {
|
||||
// this part can't be made into UTF-8; scan the header, if we can
|
||||
NSString *header = nil;
|
||||
NSScanner *headerScanner = [NSScanner scannerWithString:mungedPart];
|
||||
if (![headerScanner scanUpToString:@"\r\n\r\n" intoString:&header]) {
|
||||
// we couldn't find a header
|
||||
header = @"";
|
||||
}
|
||||
// make a part string with the header and <<n bytes>>
|
||||
NSString *binStr = [NSString stringWithFormat:@"\r%@\r<<%lu bytes>>\r",
|
||||
header, (long)(partSize - header.length)];
|
||||
[origParts addObject:binStr];
|
||||
}
|
||||
offset += partSize + boundary.length;
|
||||
}
|
||||
// rejoin the original parts
|
||||
streamDataStr = [origParts componentsJoinedByString:boundary];
|
||||
}
|
||||
}
|
||||
if (!streamDataStr) {
|
||||
// give up; just make a string showing the uploaded bytes
|
||||
streamDataStr = [NSString stringWithFormat:@"<<%u bytes>>", (unsigned int)data.length];
|
||||
}
|
||||
return streamDataStr;
|
||||
}
|
||||
|
||||
// logFetchWithError is called following a successful or failed fetch attempt
|
||||
//
|
||||
// This method does all the work for appending to and creating log files
|
||||
|
||||
- (void)logFetchWithError:(NSError *)error {
|
||||
if (![[self class] isLoggingEnabled]) return;
|
||||
NSString *logDirectory = [[self class] logDirectoryForCurrentRun];
|
||||
if (!logDirectory) return;
|
||||
NSString *processName = [[self class] loggingProcessName];
|
||||
|
||||
// TODO: add Javascript to display response data formatted in hex
|
||||
|
||||
// each response's NSData goes into its own xml or txt file, though all responses for this run of
|
||||
// the app share a main html file. This counter tracks all fetch responses for this app run.
|
||||
//
|
||||
// we'll use a local variable since this routine may be reentered while waiting for XML formatting
|
||||
// to be completed by an external task
|
||||
static int gResponseCounter = 0;
|
||||
int responseCounter = ++gResponseCounter;
|
||||
|
||||
NSURLResponse *response = [self response];
|
||||
NSDictionary *responseHeaders = [self responseHeaders];
|
||||
NSString *responseDataStr = nil;
|
||||
NSDictionary *responseJSON = nil;
|
||||
|
||||
// if there's response data, decide what kind of file to put it in based on the first bytes of the
|
||||
// file or on the mime type supplied by the server
|
||||
NSString *responseMIMEType = [response MIMEType];
|
||||
BOOL isResponseImage = NO;
|
||||
|
||||
// file name for an image data file
|
||||
NSString *responseDataFileName = nil;
|
||||
|
||||
int64_t responseDataLength = self.downloadedLength;
|
||||
if (responseDataLength > 0) {
|
||||
NSData *downloadedData = self.downloadedData;
|
||||
if (downloadedData == nil
|
||||
&& responseDataLength > 0
|
||||
&& responseDataLength < 20000
|
||||
&& self.destinationFileURL) {
|
||||
// There's a download file that's not too big, so get the data to display from the downloaded
|
||||
// file.
|
||||
NSURL *destinationURL = self.destinationFileURL;
|
||||
downloadedData = [NSData dataWithContentsOfURL:destinationURL];
|
||||
}
|
||||
NSString *responseType = [responseHeaders valueForKey:@"Content-Type"];
|
||||
responseDataStr = [self formattedStringFromData:downloadedData
|
||||
contentType:responseType
|
||||
JSON:&responseJSON];
|
||||
NSString *responseDataExtn = nil;
|
||||
NSData *dataToWrite = nil;
|
||||
if (responseDataStr) {
|
||||
// we were able to make a UTF-8 string from the response data
|
||||
if ([responseMIMEType isEqual:@"application/atom+xml"]
|
||||
|| [responseMIMEType hasSuffix:@"/xml"]) {
|
||||
responseDataExtn = @"xml";
|
||||
dataToWrite = [responseDataStr dataUsingEncoding:NSUTF8StringEncoding];
|
||||
}
|
||||
} else if ([responseMIMEType isEqual:@"image/jpeg"]) {
|
||||
responseDataExtn = @"jpg";
|
||||
dataToWrite = downloadedData;
|
||||
isResponseImage = YES;
|
||||
} else if ([responseMIMEType isEqual:@"image/gif"]) {
|
||||
responseDataExtn = @"gif";
|
||||
dataToWrite = downloadedData;
|
||||
isResponseImage = YES;
|
||||
} else if ([responseMIMEType isEqual:@"image/png"]) {
|
||||
responseDataExtn = @"png";
|
||||
dataToWrite = downloadedData;
|
||||
isResponseImage = YES;
|
||||
} else {
|
||||
// add more non-text types here
|
||||
}
|
||||
// if we have an extension, save the raw data in a file with that extension
|
||||
if (responseDataExtn && dataToWrite) {
|
||||
// generate a response file base name like
|
||||
NSString *responseBaseName = [NSString stringWithFormat:@"fetch_%d_response", responseCounter];
|
||||
responseDataFileName = [responseBaseName stringByAppendingPathExtension:responseDataExtn];
|
||||
NSString *responseDataFilePath = [logDirectory stringByAppendingPathComponent:responseDataFileName];
|
||||
|
||||
NSError *downloadedError = nil;
|
||||
if (gIsLoggingToFile && ![dataToWrite writeToFile:responseDataFilePath
|
||||
options:0
|
||||
error:&downloadedError]) {
|
||||
NSLog(@"%@ logging write error:%@ (%@)", [self class], downloadedError, responseDataFileName);
|
||||
}
|
||||
}
|
||||
}
|
||||
// we'll have one main html file per run of the app
|
||||
NSString *htmlName = [[self class] htmlFileName];
|
||||
NSString *htmlPath =[logDirectory stringByAppendingPathComponent:htmlName];
|
||||
|
||||
// if the html file exists (from logging previous fetches) we don't need
|
||||
// to re-write the header or the scripts
|
||||
NSFileManager *fileMgr = [NSFileManager defaultManager];
|
||||
BOOL didFileExist = [fileMgr fileExistsAtPath:htmlPath];
|
||||
|
||||
NSMutableString* outputHTML = [NSMutableString string];
|
||||
|
||||
// we need a header to say we'll have UTF-8 text
|
||||
if (!didFileExist) {
|
||||
[outputHTML appendFormat:@"<html><head><meta http-equiv=\"content-type\" "
|
||||
"content=\"text/html; charset=UTF-8\"><title>%@ HTTP fetch log %@</title>",
|
||||
processName, [[self class] loggingDateStamp]];
|
||||
}
|
||||
// now write the visible html elements
|
||||
NSString *copyableFileName = [NSString stringWithFormat:@"fetch_%d.txt", responseCounter];
|
||||
|
||||
NSDate *now = [NSDate date];
|
||||
// write the date & time, the comment, and the link to the plain-text (copyable) log
|
||||
[outputHTML appendFormat:@"<b>%@ ", now];
|
||||
|
||||
NSString *comment = [self comment];
|
||||
if (comment.length > 0) {
|
||||
[outputHTML appendFormat:@"%@ ", comment];
|
||||
}
|
||||
[outputHTML appendFormat:@"</b><a href='%@'><i>request/response log</i></a><br>", copyableFileName];
|
||||
NSTimeInterval elapsed = -self.initialBeginFetchDate.timeIntervalSinceNow;
|
||||
[outputHTML appendFormat:@"elapsed: %5.3fsec<br>", elapsed];
|
||||
|
||||
// write the request URL
|
||||
NSURLRequest *request = self.request;
|
||||
NSString *requestMethod = request.HTTPMethod;
|
||||
NSURL *requestURL = request.URL;
|
||||
|
||||
// Save the request URL for next time in case this redirects.
|
||||
NSString *redirectedFromURLString = [self.redirectedFromURL absoluteString];
|
||||
self.redirectedFromURL = [requestURL copy];
|
||||
if (redirectedFromURLString) {
|
||||
[outputHTML appendFormat:@"<FONT COLOR='#990066'><i>redirected from %@</i></FONT><br>",
|
||||
redirectedFromURLString];
|
||||
}
|
||||
[outputHTML appendFormat:@"<b>request:</b> %@ <code>%@</code><br>\n", requestMethod, requestURL];
|
||||
|
||||
// write the request headers
|
||||
NSDictionary *requestHeaders = request.allHTTPHeaderFields;
|
||||
NSUInteger numberOfRequestHeaders = requestHeaders.count;
|
||||
if (numberOfRequestHeaders > 0) {
|
||||
// Indicate if the request is authorized; warn if the request is authorized but non-SSL
|
||||
NSString *auth = [requestHeaders objectForKey:@"Authorization"];
|
||||
NSString *headerDetails = @"";
|
||||
if (auth) {
|
||||
BOOL isInsecure = [[requestURL scheme] isEqual:@"http"];
|
||||
if (isInsecure) {
|
||||
// 26A0 = ⚠
|
||||
headerDetails =
|
||||
@" <i>authorized, non-SSL</i><FONT COLOR='#FF00FF'> ⚠</FONT> ";
|
||||
} else {
|
||||
headerDetails = @" <i>authorized</i>";
|
||||
}
|
||||
}
|
||||
NSString *cookiesHdr = [requestHeaders objectForKey:@"Cookie"];
|
||||
if (cookiesHdr) {
|
||||
headerDetails = [headerDetails stringByAppendingString:@" <i>cookies</i>"];
|
||||
}
|
||||
NSString *matchHdr = [requestHeaders objectForKey:@"If-Match"];
|
||||
if (matchHdr) {
|
||||
headerDetails = [headerDetails stringByAppendingString:@" <i>if-match</i>"];
|
||||
}
|
||||
matchHdr = [requestHeaders objectForKey:@"If-None-Match"];
|
||||
if (matchHdr) {
|
||||
headerDetails = [headerDetails stringByAppendingString:@" <i>if-none-match</i>"];
|
||||
}
|
||||
[outputHTML appendFormat:@" headers: %d %@<br>",
|
||||
(int)numberOfRequestHeaders, headerDetails];
|
||||
} else {
|
||||
[outputHTML appendFormat:@" headers: none<br>"];
|
||||
}
|
||||
// write the request post data
|
||||
NSData *bodyData = nil;
|
||||
NSData *loggedStreamData = self.loggedStreamData;
|
||||
if (loggedStreamData) {
|
||||
bodyData = loggedStreamData;
|
||||
} else {
|
||||
bodyData = self.bodyData;
|
||||
if (bodyData == nil) {
|
||||
bodyData = self.request.HTTPBody;
|
||||
}
|
||||
}
|
||||
uint64_t bodyDataLength = bodyData.length;
|
||||
|
||||
if (bodyData.length == 0) {
|
||||
// If the data is in a body upload file URL, read that in if it's not huge.
|
||||
NSURL *bodyFileURL = self.bodyFileURL;
|
||||
if (bodyFileURL) {
|
||||
NSNumber *fileSizeNum = nil;
|
||||
NSError *fileSizeError = nil;
|
||||
if ([bodyFileURL getResourceValue:&fileSizeNum
|
||||
forKey:NSURLFileSizeKey
|
||||
error:&fileSizeError]) {
|
||||
bodyDataLength = [fileSizeNum unsignedLongLongValue];
|
||||
if (bodyDataLength > 0 && bodyDataLength < 50000) {
|
||||
bodyData = [NSData dataWithContentsOfURL:bodyFileURL
|
||||
options:NSDataReadingUncached
|
||||
error:&fileSizeError];
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
NSString *bodyDataStr = nil;
|
||||
NSString *postType = [requestHeaders valueForKey:@"Content-Type"];
|
||||
|
||||
if (bodyDataLength > 0) {
|
||||
[outputHTML appendFormat:@" data: %llu bytes, <code>%@</code><br>\n",
|
||||
bodyDataLength, postType ? postType : @"(no type)"];
|
||||
NSString *logRequestBody = self.logRequestBody;
|
||||
if (logRequestBody) {
|
||||
bodyDataStr = [logRequestBody copy];
|
||||
self.logRequestBody = nil;
|
||||
} else {
|
||||
bodyDataStr = [self stringFromStreamData:bodyData
|
||||
contentType:postType];
|
||||
if (bodyDataStr) {
|
||||
// remove OAuth 2 client secret and refresh token
|
||||
bodyDataStr = [[self class] snipSubstringOfString:bodyDataStr
|
||||
betweenStartString:@"client_secret="
|
||||
endString:@"&"];
|
||||
bodyDataStr = [[self class] snipSubstringOfString:bodyDataStr
|
||||
betweenStartString:@"refresh_token="
|
||||
endString:@"&"];
|
||||
// remove ClientLogin password
|
||||
bodyDataStr = [[self class] snipSubstringOfString:bodyDataStr
|
||||
betweenStartString:@"&Passwd="
|
||||
endString:@"&"];
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// no post data
|
||||
}
|
||||
// write the response status, MIME type, URL
|
||||
NSInteger status = [self statusCode];
|
||||
if (response) {
|
||||
NSString *statusString = @"";
|
||||
if (status != 0) {
|
||||
if (status == 200 || status == 201) {
|
||||
statusString = [NSString stringWithFormat:@"%ld", (long)status];
|
||||
|
||||
// report any JSON-RPC error
|
||||
if ([responseJSON isKindOfClass:[NSDictionary class]]) {
|
||||
NSDictionary *jsonError = [responseJSON objectForKey:@"error"];
|
||||
if ([jsonError isKindOfClass:[NSDictionary class]]) {
|
||||
NSString *jsonCode = [[jsonError valueForKey:@"code"] description];
|
||||
NSString *jsonMessage = [jsonError valueForKey:@"message"];
|
||||
if (jsonCode || jsonMessage) {
|
||||
// 2691 = ⚑
|
||||
NSString *const jsonErrFmt =
|
||||
@" <i>JSON error:</i> <FONT COLOR='#FF00FF'>%@ %@ ⚑</FONT>";
|
||||
statusString = [statusString stringByAppendingFormat:jsonErrFmt,
|
||||
jsonCode ? jsonCode : @"",
|
||||
jsonMessage ? jsonMessage : @""];
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// purple for anything other than 200 or 201
|
||||
NSString *flag = status >= 400 ? @" ⚑" : @""; // 2691 = ⚑
|
||||
NSString *explanation = [NSHTTPURLResponse localizedStringForStatusCode:status];
|
||||
NSString *const statusFormat = @"<FONT COLOR='#FF00FF'>%ld %@ %@</FONT>";
|
||||
statusString = [NSString stringWithFormat:statusFormat, (long)status, explanation, flag];
|
||||
}
|
||||
}
|
||||
// show the response URL only if it's different from the request URL
|
||||
NSString *responseURLStr = @"";
|
||||
NSURL *responseURL = response.URL;
|
||||
|
||||
if (responseURL && ![responseURL isEqual:request.URL]) {
|
||||
NSString *const responseURLFormat =
|
||||
@"<FONT COLOR='#FF00FF'>response URL:</FONT> <code>%@</code><br>\n";
|
||||
responseURLStr = [NSString stringWithFormat:responseURLFormat, [responseURL absoluteString]];
|
||||
}
|
||||
[outputHTML appendFormat:@"<b>response:</b> status %@<br>\n%@",
|
||||
statusString, responseURLStr];
|
||||
// Write the response headers
|
||||
NSUInteger numberOfResponseHeaders = responseHeaders.count;
|
||||
if (numberOfResponseHeaders > 0) {
|
||||
// Indicate if the server is setting cookies
|
||||
NSString *cookiesSet = [responseHeaders valueForKey:@"Set-Cookie"];
|
||||
NSString *cookiesStr =
|
||||
cookiesSet ? @" <FONT COLOR='#990066'><i>sets cookies</i></FONT>" : @"";
|
||||
// Indicate if the server is redirecting
|
||||
NSString *location = [responseHeaders valueForKey:@"Location"];
|
||||
BOOL isRedirect = status >= 300 && status <= 399 && location != nil;
|
||||
NSString *redirectsStr =
|
||||
isRedirect ? @" <FONT COLOR='#990066'><i>redirects</i></FONT>" : @"";
|
||||
[outputHTML appendFormat:@" headers: %d %@ %@<br>\n",
|
||||
(int)numberOfResponseHeaders, cookiesStr, redirectsStr];
|
||||
} else {
|
||||
[outputHTML appendString:@" headers: none<br>\n"];
|
||||
}
|
||||
}
|
||||
// error
|
||||
if (error) {
|
||||
[outputHTML appendFormat:@"<b>Error:</b> %@ <br>\n", error.description];
|
||||
}
|
||||
// Write the response data
|
||||
if (responseDataFileName) {
|
||||
if (isResponseImage) {
|
||||
// Make a small inline image that links to the full image file
|
||||
[outputHTML appendFormat:@" data: %lld bytes, <code>%@</code><br>",
|
||||
responseDataLength, responseMIMEType];
|
||||
NSString *const fmt =
|
||||
@"<a href=\"%@\"><img src='%@' alt='image' style='border:solid thin;max-height:32'></a>\n";
|
||||
[outputHTML appendFormat:fmt, responseDataFileName, responseDataFileName];
|
||||
} else {
|
||||
// The response data was XML; link to the xml file
|
||||
NSString *const fmt =
|
||||
@" data: %lld bytes, <code>%@</code> <i><a href=\"%@\">%@</a></i>\n";
|
||||
[outputHTML appendFormat:fmt, responseDataLength, responseMIMEType,
|
||||
responseDataFileName, [responseDataFileName pathExtension]];
|
||||
}
|
||||
} else {
|
||||
// The response data was not an image; just show the length and MIME type
|
||||
[outputHTML appendFormat:@" data: %lld bytes, <code>%@</code>\n",
|
||||
responseDataLength, responseMIMEType ? responseMIMEType : @"(no response type)"];
|
||||
}
|
||||
// Make a single string of the request and response, suitable for copying
|
||||
// to the clipboard and pasting into a bug report
|
||||
NSMutableString *copyable = [NSMutableString string];
|
||||
if (comment) {
|
||||
[copyable appendFormat:@"%@\n\n", comment];
|
||||
}
|
||||
[copyable appendFormat:@"%@ elapsed: %5.3fsec\n", now, elapsed];
|
||||
if (redirectedFromURLString) {
|
||||
[copyable appendFormat:@"Redirected from %@\n", redirectedFromURLString];
|
||||
}
|
||||
[copyable appendFormat:@"Request: %@ %@\n", requestMethod, requestURL];
|
||||
if (requestHeaders.count > 0) {
|
||||
[copyable appendFormat:@"Request headers:\n%@\n",
|
||||
[[self class] headersStringForDictionary:requestHeaders]];
|
||||
}
|
||||
if (bodyDataLength > 0) {
|
||||
[copyable appendFormat:@"Request body: (%llu bytes)\n", bodyDataLength];
|
||||
if (bodyDataStr) {
|
||||
[copyable appendFormat:@"%@\n", bodyDataStr];
|
||||
}
|
||||
[copyable appendString:@"\n"];
|
||||
}
|
||||
if (response) {
|
||||
[copyable appendFormat:@"Response: status %d\n", (int) status];
|
||||
[copyable appendFormat:@"Response headers:\n%@\n",
|
||||
[[self class] headersStringForDictionary:responseHeaders]];
|
||||
[copyable appendFormat:@"Response body: (%lld bytes)\n", responseDataLength];
|
||||
if (responseDataLength > 0) {
|
||||
NSString *logResponseBody = self.logResponseBody;
|
||||
if (logResponseBody) {
|
||||
// The user has provided the response body text.
|
||||
responseDataStr = [logResponseBody copy];
|
||||
self.logResponseBody = nil;
|
||||
}
|
||||
if (responseDataStr != nil) {
|
||||
[copyable appendFormat:@"%@\n", responseDataStr];
|
||||
} else {
|
||||
// Even though it's redundant, we'll put in text to indicate that all the bytes are binary.
|
||||
if (self.destinationFileURL) {
|
||||
[copyable appendFormat:@"<<%lld bytes>> to file %@\n",
|
||||
responseDataLength, self.destinationFileURL.path];
|
||||
} else {
|
||||
[copyable appendFormat:@"<<%lld bytes>>\n", responseDataLength];
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (error) {
|
||||
[copyable appendFormat:@"Error: %@\n", error];
|
||||
}
|
||||
// Save to log property before adding the separator
|
||||
self.log = copyable;
|
||||
|
||||
[copyable appendString:@"-----------------------------------------------------------\n"];
|
||||
|
||||
// Write the copyable version to another file (linked to at the top of the html file, above)
|
||||
//
|
||||
// Ideally, something to just copy this to the clipboard like
|
||||
// <span onCopy='window.event.clipboardData.setData(\"Text\",
|
||||
// \"copyable stuff\");return false;'>Copy here.</span>"
|
||||
// would work everywhere, but it only works in Safari as of 8/2010
|
||||
if (gIsLoggingToFile) {
|
||||
NSString *parentDir = [[self class] loggingDirectory];
|
||||
NSString *copyablePath = [logDirectory stringByAppendingPathComponent:copyableFileName];
|
||||
NSError *copyableError = nil;
|
||||
if (![copyable writeToFile:copyablePath
|
||||
atomically:NO
|
||||
encoding:NSUTF8StringEncoding
|
||||
error:©ableError]) {
|
||||
// Error writing to file
|
||||
NSLog(@"%@ logging write error:%@ (%@)", [self class], copyableError, copyablePath);
|
||||
}
|
||||
[outputHTML appendString:@"<br><hr><p>"];
|
||||
|
||||
// Append the HTML to the main output file
|
||||
const char* htmlBytes = outputHTML.UTF8String;
|
||||
NSOutputStream *stream = [NSOutputStream outputStreamToFileAtPath:htmlPath
|
||||
append:YES];
|
||||
[stream open];
|
||||
[stream write:(const uint8_t *) htmlBytes maxLength:strlen(htmlBytes)];
|
||||
[stream close];
|
||||
|
||||
// Make a symlink to the latest html
|
||||
NSString *const symlinkNameSuffix = [[self class] symlinkNameSuffix];
|
||||
NSString *symlinkName = [processName stringByAppendingString:symlinkNameSuffix];
|
||||
NSString *symlinkPath = [parentDir stringByAppendingPathComponent:symlinkName];
|
||||
|
||||
[fileMgr removeItemAtPath:symlinkPath error:NULL];
|
||||
[fileMgr createSymbolicLinkAtPath:symlinkPath
|
||||
withDestinationPath:htmlPath
|
||||
error:NULL];
|
||||
#if TARGET_OS_IPHONE
|
||||
static BOOL gReportedLoggingPath = NO;
|
||||
if (!gReportedLoggingPath) {
|
||||
gReportedLoggingPath = YES;
|
||||
NSLog(@"GTMSessionFetcher logging to \"%@\"", parentDir);
|
||||
}
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
||||
- (NSInputStream *)loggedInputStreamForInputStream:(NSInputStream *)inputStream {
|
||||
if (!inputStream) return nil;
|
||||
if (![GTMSessionFetcher isLoggingEnabled]) return inputStream;
|
||||
|
||||
[self clearLoggedStreamData]; // Clear any previous data.
|
||||
Class monitorClass = NSClassFromString(@"GTMReadMonitorInputStream");
|
||||
if (!monitorClass) {
|
||||
NSString const *str = @"<<Uploaded stream log unavailable without GTMReadMonitorInputStream>>";
|
||||
NSData *stringData = [str dataUsingEncoding:NSUTF8StringEncoding];
|
||||
[self appendLoggedStreamData:stringData];
|
||||
return inputStream;
|
||||
}
|
||||
inputStream = [monitorClass inputStreamWithStream:inputStream];
|
||||
|
||||
GTMReadMonitorInputStream *readMonitorInputStream = (GTMReadMonitorInputStream *)inputStream;
|
||||
[readMonitorInputStream setReadDelegate:self];
|
||||
SEL readSel = @selector(inputStream:readIntoBuffer:length:);
|
||||
[readMonitorInputStream setReadSelector:readSel];
|
||||
|
||||
return inputStream;
|
||||
}
|
||||
|
||||
- (GTMSessionFetcherBodyStreamProvider)loggedStreamProviderForStreamProvider:
|
||||
(GTMSessionFetcherBodyStreamProvider)streamProvider {
|
||||
if (!streamProvider) return nil;
|
||||
if (![GTMSessionFetcher isLoggingEnabled]) return streamProvider;
|
||||
|
||||
[self clearLoggedStreamData]; // Clear any previous data.
|
||||
Class monitorClass = NSClassFromString(@"GTMReadMonitorInputStream");
|
||||
if (!monitorClass) {
|
||||
NSString const *str = @"<<Uploaded stream log unavailable without GTMReadMonitorInputStream>>";
|
||||
NSData *stringData = [str dataUsingEncoding:NSUTF8StringEncoding];
|
||||
[self appendLoggedStreamData:stringData];
|
||||
return streamProvider;
|
||||
}
|
||||
GTMSessionFetcherBodyStreamProvider loggedStreamProvider =
|
||||
^(GTMSessionFetcherBodyStreamProviderResponse response) {
|
||||
streamProvider(^(NSInputStream *bodyStream) {
|
||||
bodyStream = [self loggedInputStreamForInputStream:bodyStream];
|
||||
response(bodyStream);
|
||||
});
|
||||
};
|
||||
return loggedStreamProvider;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation GTMSessionFetcher (GTMSessionFetcherLoggingUtilities)
|
||||
|
||||
- (void)inputStream:(GTMReadMonitorInputStream *)stream
|
||||
readIntoBuffer:(void *)buffer
|
||||
length:(int64_t)length {
|
||||
// append the captured data
|
||||
NSData *data = [NSData dataWithBytesNoCopy:buffer
|
||||
length:(NSUInteger)length
|
||||
freeWhenDone:NO];
|
||||
[self appendLoggedStreamData:data];
|
||||
}
|
||||
|
||||
#pragma mark Fomatting Utilities
|
||||
|
||||
+ (NSString *)snipSubstringOfString:(NSString *)originalStr
|
||||
betweenStartString:(NSString *)startStr
|
||||
endString:(NSString *)endStr {
|
||||
#if SKIP_GTM_FETCH_LOGGING_SNIPPING
|
||||
return originalStr;
|
||||
#else
|
||||
if (!originalStr) return nil;
|
||||
|
||||
// Find the start string, and replace everything between it
|
||||
// and the end string (or the end of the original string) with "_snip_"
|
||||
NSRange startRange = [originalStr rangeOfString:startStr];
|
||||
if (startRange.location == NSNotFound) return originalStr;
|
||||
|
||||
// We found the start string
|
||||
NSUInteger originalLength = originalStr.length;
|
||||
NSUInteger startOfTarget = NSMaxRange(startRange);
|
||||
NSRange targetAndRest = NSMakeRange(startOfTarget, originalLength - startOfTarget);
|
||||
NSRange endRange = [originalStr rangeOfString:endStr
|
||||
options:0
|
||||
range:targetAndRest];
|
||||
NSRange replaceRange;
|
||||
if (endRange.location == NSNotFound) {
|
||||
// Found no end marker so replace to end of string
|
||||
replaceRange = targetAndRest;
|
||||
} else {
|
||||
// Replace up to the endStr
|
||||
replaceRange = NSMakeRange(startOfTarget, endRange.location - startOfTarget);
|
||||
}
|
||||
NSString *result = [originalStr stringByReplacingCharactersInRange:replaceRange
|
||||
withString:@"_snip_"];
|
||||
return result;
|
||||
#endif // SKIP_GTM_FETCH_LOGGING_SNIPPING
|
||||
}
|
||||
|
||||
+ (NSString *)headersStringForDictionary:(NSDictionary *)dict {
|
||||
// Format the dictionary in http header style, like
|
||||
// Accept: application/json
|
||||
// Cache-Control: no-cache
|
||||
// Content-Type: application/json; charset=utf-8
|
||||
//
|
||||
// Pad the key names, but not beyond 16 chars, since long custom header
|
||||
// keys just create too much whitespace
|
||||
NSArray *keys = [dict.allKeys sortedArrayUsingSelector:@selector(compare:)];
|
||||
|
||||
NSMutableString *str = [NSMutableString string];
|
||||
for (NSString *key in keys) {
|
||||
NSString *value = [dict valueForKey:key];
|
||||
if ([key isEqual:@"Authorization"]) {
|
||||
// Remove OAuth 1 token
|
||||
value = [[self class] snipSubstringOfString:value
|
||||
betweenStartString:@"oauth_token=\""
|
||||
endString:@"\""];
|
||||
|
||||
// Remove OAuth 2 bearer token (draft 16, and older form)
|
||||
value = [[self class] snipSubstringOfString:value
|
||||
betweenStartString:@"Bearer "
|
||||
endString:@"\n"];
|
||||
value = [[self class] snipSubstringOfString:value
|
||||
betweenStartString:@"OAuth "
|
||||
endString:@"\n"];
|
||||
|
||||
// Remove Google ClientLogin
|
||||
value = [[self class] snipSubstringOfString:value
|
||||
betweenStartString:@"GoogleLogin auth="
|
||||
endString:@"\n"];
|
||||
}
|
||||
[str appendFormat:@" %@: %@\n", key, value];
|
||||
}
|
||||
return str;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
#endif // !STRIP_GTM_FETCH_LOGGING
|
|
@ -1,193 +0,0 @@
|
|||
/* Copyright 2014 Google Inc. All rights reserved.
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
// For best performance and convenient usage, fetchers should be generated by a common
|
||||
// GTMSessionFetcherService instance, like
|
||||
//
|
||||
// _fetcherService = [[GTMSessionFetcherService alloc] init];
|
||||
// GTMSessionFetcher* myFirstFetcher = [_fetcherService fetcherWithRequest:request1];
|
||||
// GTMSessionFetcher* mySecondFetcher = [_fetcherService fetcherWithRequest:request2];
|
||||
|
||||
#import "GTMSessionFetcher.h"
|
||||
|
||||
GTM_ASSUME_NONNULL_BEGIN
|
||||
|
||||
// Notifications.
|
||||
|
||||
// This notification indicates a reusable session has become invalid. It is intended mainly for the
|
||||
// service's unit tests.
|
||||
//
|
||||
// The notification object is the fetcher service.
|
||||
// The invalid session is provided via the userInfo kGTMSessionFetcherServiceSessionKey key.
|
||||
extern NSString *const kGTMSessionFetcherServiceSessionBecameInvalidNotification;
|
||||
extern NSString *const kGTMSessionFetcherServiceSessionKey;
|
||||
|
||||
@interface GTMSessionFetcherService : NSObject<GTMSessionFetcherServiceProtocol>
|
||||
|
||||
// Queues of delayed and running fetchers. Each dictionary contains arrays
|
||||
// of GTMSessionFetcher *fetchers, keyed by NSString *host
|
||||
@property(atomic, strong, readonly, GTM_NULLABLE) GTM_NSDictionaryOf(NSString *, NSArray *) *delayedFetchersByHost;
|
||||
@property(atomic, strong, readonly, GTM_NULLABLE) GTM_NSDictionaryOf(NSString *, NSArray *) *runningFetchersByHost;
|
||||
|
||||
// A max value of 0 means no fetchers should be delayed.
|
||||
// The default limit is 10 simultaneous fetchers targeting each host.
|
||||
// This does not apply to fetchers whose useBackgroundSession property is YES. Since services are
|
||||
// not resurrected on an app relaunch, delayed fetchers would effectively be abandoned.
|
||||
@property(atomic, assign) NSUInteger maxRunningFetchersPerHost;
|
||||
|
||||
// Properties to be applied to each fetcher; see GTMSessionFetcher.h for descriptions
|
||||
@property(atomic, strong, GTM_NULLABLE) NSURLSessionConfiguration *configuration;
|
||||
@property(atomic, copy, GTM_NULLABLE) GTMSessionFetcherConfigurationBlock configurationBlock;
|
||||
@property(atomic, strong, GTM_NULLABLE) NSHTTPCookieStorage *cookieStorage;
|
||||
@property(atomic, strong, GTM_NULL_RESETTABLE) dispatch_queue_t callbackQueue;
|
||||
@property(atomic, copy, GTM_NULLABLE) GTMSessionFetcherChallengeBlock challengeBlock;
|
||||
@property(atomic, strong, GTM_NULLABLE) NSURLCredential *credential;
|
||||
@property(atomic, strong) NSURLCredential *proxyCredential;
|
||||
@property(atomic, copy, GTM_NULLABLE) GTM_NSArrayOf(NSString *) *allowedInsecureSchemes;
|
||||
@property(atomic, assign) BOOL allowLocalhostRequest;
|
||||
@property(atomic, assign) BOOL allowInvalidServerCertificates;
|
||||
@property(atomic, assign, getter=isRetryEnabled) BOOL retryEnabled;
|
||||
@property(atomic, copy, GTM_NULLABLE) GTMSessionFetcherRetryBlock retryBlock;
|
||||
@property(atomic, assign) NSTimeInterval maxRetryInterval;
|
||||
@property(atomic, assign) NSTimeInterval minRetryInterval;
|
||||
@property(atomic, copy, GTM_NULLABLE) GTM_NSDictionaryOf(NSString *, id) *properties;
|
||||
|
||||
#if GTM_BACKGROUND_TASK_FETCHING
|
||||
@property(atomic, assign) BOOL skipBackgroundTask;
|
||||
#endif
|
||||
|
||||
// A default useragent of GTMFetcherStandardUserAgentString(nil) will be given to each fetcher
|
||||
// created by this service unless the request already has a user-agent header set.
|
||||
// This default will be added starting with builds with the SDKs for OS X 10.11 and iOS 9.
|
||||
//
|
||||
// To use the configuration's default user agent, set this property to nil.
|
||||
@property(atomic, copy, GTM_NULLABLE) NSString *userAgent;
|
||||
|
||||
// The authorizer to attach to the created fetchers. If a specific fetcher should
|
||||
// not authorize its requests, the fetcher's authorizer property may be set to nil
|
||||
// before the fetch begins.
|
||||
@property(atomic, strong, GTM_NULLABLE) id<GTMFetcherAuthorizationProtocol> authorizer;
|
||||
|
||||
// Delegate queue used by the session when calling back to the fetcher. The default
|
||||
// is the main queue. Changing this does not affect the queue used to call back to the
|
||||
// application; that is specified by the callbackQueue property above.
|
||||
@property(atomic, strong, GTM_NULL_RESETTABLE) NSOperationQueue *sessionDelegateQueue;
|
||||
|
||||
// When enabled, indicates the same session should be used by subsequent fetchers.
|
||||
//
|
||||
// This is enabled by default.
|
||||
@property(atomic, assign) BOOL reuseSession;
|
||||
|
||||
// Sets the delay until an unused session is invalidated.
|
||||
// The default interval is 60 seconds.
|
||||
//
|
||||
// If the interval is set to 0, then any reused session is not invalidated except by
|
||||
// explicitly invoking -resetSession. Be aware that setting the interval to 0 thus
|
||||
// causes the session's delegate to be retained until the session is explicitly reset.
|
||||
@property(atomic, assign) NSTimeInterval unusedSessionTimeout;
|
||||
|
||||
// If shouldReuseSession is enabled, this will force creation of a new session when future
|
||||
// fetchers begin.
|
||||
- (void)resetSession;
|
||||
|
||||
// Create a fetcher
|
||||
//
|
||||
// These methods will return a fetcher. If successfully created, the connection
|
||||
// will hold a strong reference to it for the life of the connection as well.
|
||||
// So the caller doesn't have to hold onto the fetcher explicitly unless they
|
||||
// want to be able to monitor or cancel it.
|
||||
- (GTMSessionFetcher *)fetcherWithRequest:(NSURLRequest *)request;
|
||||
- (GTMSessionFetcher *)fetcherWithURL:(NSURL *)requestURL;
|
||||
- (GTMSessionFetcher *)fetcherWithURLString:(NSString *)requestURLString;
|
||||
|
||||
// Common method for fetcher creation.
|
||||
//
|
||||
// -fetcherWithRequest:fetcherClass: may be overridden to customize creation of
|
||||
// fetchers. This is the ONLY method in the GTMSessionFetcher library intended to
|
||||
// be overridden.
|
||||
- (id)fetcherWithRequest:(NSURLRequest *)request
|
||||
fetcherClass:(Class)fetcherClass;
|
||||
|
||||
- (BOOL)isDelayingFetcher:(GTMSessionFetcher *)fetcher;
|
||||
|
||||
- (NSUInteger)numberOfFetchers; // running + delayed fetchers
|
||||
- (NSUInteger)numberOfRunningFetchers;
|
||||
- (NSUInteger)numberOfDelayedFetchers;
|
||||
|
||||
// Return a list of all running or delayed fetchers. This includes fetchers created
|
||||
// by the service which have been started and have not yet stopped.
|
||||
//
|
||||
// Returns an array of fetcher objects, or nil if none.
|
||||
- (GTM_NULLABLE GTM_NSArrayOf(GTMSessionFetcher *) *)issuedFetchers;
|
||||
|
||||
// Search for running or delayed fetchers with the specified URL.
|
||||
//
|
||||
// Returns an array of fetcher objects found, or nil if none found.
|
||||
- (GTM_NULLABLE GTM_NSArrayOf(GTMSessionFetcher *) *)issuedFetchersWithRequestURL:(NSURL *)requestURL;
|
||||
|
||||
- (void)stopAllFetchers;
|
||||
|
||||
// Methods for use by the fetcher class only.
|
||||
- (GTM_NULLABLE NSURLSession *)session;
|
||||
- (GTM_NULLABLE NSURLSession *)sessionForFetcherCreation;
|
||||
- (GTM_NULLABLE id<NSURLSessionDelegate>)sessionDelegate;
|
||||
- (GTM_NULLABLE NSDate *)stoppedAllFetchersDate;
|
||||
|
||||
// The testBlock can inspect its fetcher parameter's request property to
|
||||
// determine which fetcher is being faked.
|
||||
@property(atomic, copy, GTM_NULLABLE) GTMSessionFetcherTestBlock testBlock;
|
||||
|
||||
@end
|
||||
|
||||
@interface GTMSessionFetcherService (TestingSupport)
|
||||
|
||||
// Convenience methods to create a fetcher service for testing.
|
||||
//
|
||||
// Fetchers generated by this mock fetcher service will not perform any
|
||||
// network operation, but will invoke callbacks and provide the supplied data
|
||||
// or error to the completion handler.
|
||||
//
|
||||
// You can make more customized mocks by setting the test block property of the service
|
||||
// or fetcher; the test block can inspect the fetcher's request or other properties.
|
||||
//
|
||||
// See the description of the testBlock property below.
|
||||
+ (instancetype)mockFetcherServiceWithFakedData:(GTM_NULLABLE NSData *)fakedDataOrNil
|
||||
fakedError:(GTM_NULLABLE NSError *)fakedErrorOrNil;
|
||||
+ (instancetype)mockFetcherServiceWithFakedData:(GTM_NULLABLE NSData *)fakedDataOrNil
|
||||
fakedResponse:(NSHTTPURLResponse *)fakedResponse
|
||||
fakedError:(GTM_NULLABLE NSError *)fakedErrorOrNil;
|
||||
|
||||
// Spin the run loop and discard events (or, if not on the main thread, just sleep the thread)
|
||||
// until all running and delayed fetchers have completed.
|
||||
//
|
||||
// This is only for use in testing or in tools without a user interface.
|
||||
//
|
||||
// Synchronous fetches should never be done by shipping apps; they are
|
||||
// sufficient reason for rejection from the app store.
|
||||
//
|
||||
// Returns NO if timed out.
|
||||
- (BOOL)waitForCompletionOfAllFetchersWithTimeout:(NSTimeInterval)timeoutInSeconds;
|
||||
|
||||
@end
|
||||
|
||||
@interface GTMSessionFetcherService (BackwardsCompatibilityOnly)
|
||||
|
||||
// Clients using GTMSessionFetcher should set the cookie storage explicitly themselves.
|
||||
// This method is just for compatibility with the old fetcher.
|
||||
@property(atomic, assign) NSInteger cookieStorageMethod;
|
||||
|
||||
@end
|
||||
|
||||
GTM_ASSUME_NONNULL_END
|
File diff suppressed because it is too large
Load Diff
|
@ -1,166 +0,0 @@
|
|||
/* Copyright 2014 Google Inc. All rights reserved.
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
// GTMSessionUploadFetcher implements Google's resumable upload protocol.
|
||||
|
||||
//
|
||||
// This subclass of GTMSessionFetcher simulates the series of fetches
|
||||
// needed for chunked upload as a single fetch operation.
|
||||
//
|
||||
// Protocol document: TBD
|
||||
//
|
||||
// To the client, the only fetcher that exists is this class; the subsidiary
|
||||
// fetchers needed for uploading chunks are not visible (though the most recent
|
||||
// chunk fetcher may be accessed via the -activeFetcher or -chunkFetcher methods, and
|
||||
// -responseHeaders and -statusCode reflect results from the most recent chunk
|
||||
// fetcher.)
|
||||
//
|
||||
// Chunk fetchers are discarded as soon as they have completed.
|
||||
//
|
||||
// The protocol also allows for a cancellation notification request to be sent to the
|
||||
// server to allow discarding of the currently uploaded data and this will be sent
|
||||
// automatically upon calling stopFetching if the upload has already started.
|
||||
//
|
||||
// Note: Unlike the fetcher superclass, the methods of GTMSessionUploadFetcher should
|
||||
// only be used from the main thread until further work is done to make this subclass
|
||||
// thread-safe.
|
||||
|
||||
#import "GTMSessionFetcher.h"
|
||||
#import "GTMSessionFetcherService.h"
|
||||
|
||||
GTM_ASSUME_NONNULL_BEGIN
|
||||
|
||||
// The value to use for file size parameters when the file size is not yet known.
|
||||
extern int64_t const kGTMSessionUploadFetcherUnknownFileSize;
|
||||
|
||||
// Unless an application knows it needs a smaller chunk size, it should use the standard
|
||||
// chunk size, which sends the entire file as a single chunk to minimize upload overhead.
|
||||
// Setting an explicit chunk size that comfortably fits in memory is advisable for large
|
||||
// uploads.
|
||||
extern int64_t const kGTMSessionUploadFetcherStandardChunkSize;
|
||||
|
||||
// When uploading requires data buffer allocations (such as uploading from an NSData or
|
||||
// an NSFileHandle) this is the maximum buffer size that will be created by the fetcher.
|
||||
extern int64_t const kGTMSessionUploadFetcherMaximumDemandBufferSize;
|
||||
|
||||
// Notification that the upload location URL was provided by the server.
|
||||
extern NSString *const kGTMSessionFetcherUploadLocationObtainedNotification;
|
||||
|
||||
// Block to provide data during uploads.
|
||||
//
|
||||
// Response data may be allocated with dataWithBytesNoCopy:length:freeWhenDone: for efficiency,
|
||||
// and released after the response block returns.
|
||||
//
|
||||
// If the length of the file being uploaded is unknown or already set, send
|
||||
// kGTMSessionUploadFetcherUnknownFileSize for |fullUploadLength|. Otherwise, set |fullUploadLength|
|
||||
// to its proper value.
|
||||
//
|
||||
// Pass nil as the data (and optionally an NSError) for a failure.
|
||||
typedef void (^GTMSessionUploadFetcherDataProviderResponse)(NSData * GTM_NULLABLE_TYPE data,
|
||||
int64_t fullUploadLength,
|
||||
NSError * GTM_NULLABLE_TYPE error);
|
||||
// Do not call the response with an NSData object with less data than the requested length unless
|
||||
// you are passing the fullUploadLength to the fetcher for the first time and it is the last chunk
|
||||
// of data in the file being uploaded.
|
||||
typedef void (^GTMSessionUploadFetcherDataProvider)(int64_t offset, int64_t length,
|
||||
GTMSessionUploadFetcherDataProviderResponse response);
|
||||
|
||||
// Block to be notified about the final status of the cancellation request started in stopFetching.
|
||||
//
|
||||
// |fetcher| will be the cancel request that was sent to the server, or nil if stopFetching is not
|
||||
// going to send a cancel request. If |fetcher| is provided, the other parameters correspond to the
|
||||
// completion handler of the cancellation request fetcher.
|
||||
typedef void (^GTMSessionUploadFetcherCancellationHandler)(
|
||||
GTMSessionFetcher * GTM_NULLABLE_TYPE fetcher,
|
||||
NSData * GTM_NULLABLE_TYPE data,
|
||||
NSError * GTM_NULLABLE_TYPE error);
|
||||
|
||||
@interface GTMSessionUploadFetcher : GTMSessionFetcher
|
||||
|
||||
// Create an upload fetcher specifying either the request or the resume location URL,
|
||||
// then set an upload data source using one of these:
|
||||
//
|
||||
// setUploadFileURL:
|
||||
// setUploadDataLength:provider:
|
||||
// setUploadFileHandle:
|
||||
// setUploadData:
|
||||
|
||||
+ (instancetype)uploadFetcherWithRequest:(NSURLRequest *)request
|
||||
uploadMIMEType:(NSString *)uploadMIMEType
|
||||
chunkSize:(int64_t)chunkSize
|
||||
fetcherService:(GTM_NULLABLE GTMSessionFetcherService *)fetcherServiceOrNil;
|
||||
|
||||
+ (instancetype)uploadFetcherWithLocation:(NSURL * GTM_NULLABLE_TYPE)uploadLocationURL
|
||||
uploadMIMEType:(NSString *)uploadMIMEType
|
||||
chunkSize:(int64_t)chunkSize
|
||||
fetcherService:(GTM_NULLABLE GTMSessionFetcherService *)fetcherServiceOrNil;
|
||||
|
||||
// Allows dataProviders for files of unknown length. Pass kGTMSessionUploadFetcherUnknownFileSize as
|
||||
// |fullLength| if the length is unknown.
|
||||
- (void)setUploadDataLength:(int64_t)fullLength
|
||||
provider:(GTM_NULLABLE GTMSessionUploadFetcherDataProvider)block;
|
||||
|
||||
+ (NSArray *)uploadFetchersForBackgroundSessions;
|
||||
+ (GTM_NULLABLE instancetype)uploadFetcherForSessionIdentifier:(NSString *)sessionIdentifier;
|
||||
|
||||
- (void)pauseFetching;
|
||||
- (void)resumeFetching;
|
||||
- (BOOL)isPaused;
|
||||
|
||||
@property(atomic, strong, GTM_NULLABLE) NSURL *uploadLocationURL;
|
||||
@property(atomic, strong, GTM_NULLABLE) NSData *uploadData;
|
||||
@property(atomic, strong, GTM_NULLABLE) NSURL *uploadFileURL;
|
||||
@property(atomic, strong, GTM_NULLABLE) NSFileHandle *uploadFileHandle;
|
||||
@property(atomic, copy, readonly, GTM_NULLABLE) GTMSessionUploadFetcherDataProvider uploadDataProvider;
|
||||
@property(atomic, copy) NSString *uploadMIMEType;
|
||||
@property(atomic, assign) int64_t chunkSize;
|
||||
@property(atomic, readonly, assign) int64_t currentOffset;
|
||||
|
||||
// The fetcher for the current data chunk, if any
|
||||
@property(atomic, strong, GTM_NULLABLE) GTMSessionFetcher *chunkFetcher;
|
||||
|
||||
// The active fetcher is the current chunk fetcher, or the upload fetcher itself
|
||||
// if no chunk fetcher has yet been created.
|
||||
@property(atomic, readonly) GTMSessionFetcher *activeFetcher;
|
||||
|
||||
// The last request made by an active fetcher. Useful for testing.
|
||||
@property(atomic, readonly, GTM_NULLABLE) NSURLRequest *lastChunkRequest;
|
||||
|
||||
// The status code from the most recently-completed fetch.
|
||||
@property(atomic, assign) NSInteger statusCode;
|
||||
|
||||
// Invoked as part of the stop fetching process. Invoked immediately if there is no upload in
|
||||
// progress, otherwise invoked with the results of the attempt to notify the server that the
|
||||
// upload will not continue.
|
||||
//
|
||||
// Unlike other callbacks, since this is related specifically to the stopFetching flow it is not
|
||||
// cleared by stopFetching. It will instead clear itself after it is invoked or if the completion
|
||||
// has occured before stopFetching is called.
|
||||
@property(atomic, copy, GTM_NULLABLE) GTMSessionUploadFetcherCancellationHandler
|
||||
cancellationHandler;
|
||||
|
||||
// Exposed for testing only.
|
||||
@property(atomic, readonly, GTM_NULLABLE) dispatch_queue_t delegateCallbackQueue;
|
||||
@property(atomic, readonly, GTM_NULLABLE) GTMSessionFetcherCompletionHandler delegateCompletionHandler;
|
||||
|
||||
@end
|
||||
|
||||
@interface GTMSessionFetcher (GTMSessionUploadFetcherMethods)
|
||||
|
||||
@property(readonly, GTM_NULLABLE) GTMSessionUploadFetcher *parentUploadFetcher;
|
||||
|
||||
@end
|
||||
|
||||
GTM_ASSUME_NONNULL_END
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue