invertase / react-native-firebase

πŸ”₯ A well-tested feature-rich modular Firebase implementation for React Native. Supports both iOS & Android platforms for all Firebase services.
https://rnfirebase.io
Other
11.53k stars 2.18k forks source link

πŸ”₯[Analytics][iOS/Android] Inconsistent tracking in Debug View #4010

Closed lucawidergy closed 3 years ago

lucawidergy commented 3 years ago

Issue

Hi! We are having a problem with Debug View in Firebase Console. What we are experiencing is that events are being tracked inconsistently. For example: If we test a payment flow, when pressing the "pay" button an event is fired, this event is sometimes shown in Debug View and sometimes not. We have checking how we using it and configuration and since the problem is with all events and not only one which is not being tracked (all events are inconsistent with showing up in Debug View) we can't figure what could be the issue, if it's something wrong we are doing or it's something out of our reach.

This is a custom utility for using this with redux

const LOG_EVENT = 'LOG_EVENT';
const SET_CURRENT_SCREEN = 'SET_CURRENT_SCREEN';
const SET_USER_ID = 'SET_USER_ID';

const createTracker = type => eventDefinition => (action, prevState, nextState) => ({
  type,
  ...eventDefinition(action, prevState, nextState)
});

const trackEvent = createTracker(LOG_EVENT);
const trackScreen = createTracker(SET_CURRENT_SCREEN);
const setUserTracking = createTracker(SET_USER_ID);

const sanitizeValue = (value, limit) => (value.length > limit ? `${value.slice(0, limit - 3)}...` : value);

const mapAnalyticsMethod = analytics => ({
  [LOG_EVENT]: params => {
    const event = {};
    Object.keys(params).forEach(param => {
      if (param.length > 40) throw new Error(`Param key '${param}' can't have more than 40 characters!`);
      event[param] = sanitizeValue(params[param], 100);
    });
    analytics.logEvent(params.category, { ...event });
  },
  [SET_USER_ID]: ({ userId }) => analytics.setUserId(userId),
  [SET_CURRENT_SCREEN]: ({ screenName }) => analytics.setCurrentScreen(screenName)
});

const tracker = (analytics, eventDefinition, action, prevState, nextState) => {
  const { type, ...event } = eventDefinition(action, prevState, nextState);
  const analyticsMethod = mapAnalyticsMethod(analytics)[type];
  analyticsMethod(event);
};

const createAnalyticsMiddleware = (eventsMapper, analytics) => store => next => action => {
  const analyticsEventCreator = eventsMapper(action);
  const prevState = store.getState();
  const result = next(action);
  const nextState = store.getState();

  if (analyticsEventCreator) {
    if (Array.isArray(analyticsEventCreator))
      analyticsEventCreator.forEach(eventDefinition =>
        tracker(analytics, eventDefinition, action, prevState, nextState)
      );
    else tracker(analytics, analyticsEventCreator, action, prevState, nextState);
  }

  return result;
};

export default {
  createAnalyticsMiddleware,
  tracker,
  trackEvent,
  trackScreen,
  setUserTracking
};

// Here this is how we track an event using a function from above 

const event = trackEvent(action => ({
  category: categories(action),
  action: actions(action),
  label: labels(action)
}));

onst eventsMapper = action => {
  switch (action.type) {
    case `${NavigationActions.NAVIGATE}`:
    case `${NavigationActions.BACK}`:
    case `${NavigationActions.RESET}`:
      return pageView;
    case `${authActions.CHANGE_PASSWORD_SUCCESS}`:
    case `${authActions.EDIT_ADDITIONAL_INFORMATION_FAILURE}`:
    case `${authActions.EDIT_ADDITIONAL_INFORMATION_SUCCESS}`:
    case `${authActions.GET_IDENTITY_QUESTIONS_SUCCESS}`:
  return event;
    default:
      return [];
  }
};

// here we create the middleware with firebase analytics

export default createAnalyticsMiddleware(eventsMapper, analytics());

Project Files

Javascript

Click To Expand

#### `package.json`: ```json { "version": "1.6.3", "private": true, "rnpm": { "assets": [ "./src/fonts/" ] }, "scripts": { "release": "bash newRelease.sh", "postinstall": "patch-package", "postversion": "react-native-version", "start": "react-native start", "test": "jest", "coverage": "jest --coverage", "clean": "rm -rf $TMPDIR/react-* && watchman watch-del-all && npm cache clean", "force-clean": "npm run android:clean && rm -rf $TMPDIR/react-* && watchman watch-del-all && rm -rf ios/build && rm -rf node_modules/ && npm cache clean && npm i", "android:clean": "cd android/ && ./gradlew clean", "lint": "eslint src", "lint-fix": "eslint src --fix", "lint-diff": "git diff --name-only --cached --relative --diff-filter=ACMR | grep \\.js$ | xargs eslint", "open-debug-menu": "adb shell input keyevent KEYCODE_MENU", "android": "node scripts/android", "ios": "node scripts/ios", "deploy": "node scripts/deploy", "apk": "node scripts/apk" }, "dependencies": { "@react-native-community/async-storage": "^1.4.2", "@react-native-community/google-signin": "^4.0.0", "@react-native-community/push-notification-ios": "^1.1.1", "@react-native-community/viewpager": "^3.3.0", "@react-native-firebase/analytics": "^6.4.0", "@react-native-firebase/app": "^6.4.0", "@react-native-firebase/crashlytics": "^6.4.0", "apisauce": "^0.15.2", "buffer": "^5.0.6", "change-case": "^3.0.2", "d3-interpolate": "^1.2.0", "d3-scale": "^2.1.0", "expr-eval": "^1.2.2", "i18next": "^17.0.10", "jail-monkey": "^1.0.0", "lodash": "^4.17.10", "moment": "^2.19.2", "numeral": "^2.0.6", "patch-package": "^6.2.2", "postinstall-postinstall": "^2.1.0", "prop-types": "^15.6.1", "react": "16.11.0", "react-i18next": "^10.12.0", "react-native": "0.62.2", "react-native-azure-auth": "^1.5.1", "react-native-calendars": "^1.265.0", "react-native-collapsible": "^0.12.0", "react-native-config": "^1.0.0", "react-native-dash": "^0.0.11", "react-native-device-info": "^5.5.7", "react-native-document-picker": "^3.4.0", "react-native-drawer": "^2.5.0", "react-native-elements": "^1.0.0", "react-native-fbsdk": "^1.1.2", "react-native-file-viewer": "^2.1.1", "react-native-flipper": "^0.39.0", "react-native-gesture-handler": "^1.6.1", "react-native-get-real-path": "git://github.com/Wraptime/react-native-get-real-path.git", "react-native-google-analytics-bridge": "^5.6.3", "react-native-huawei-protected-apps": "^0.1.7", "react-native-icon-badge": "^1.1.3", "react-native-image-picker": "^0.28.0", "react-native-image-resizer": "^1.2.3", "react-native-keyboard-aware-scroll-view": "^0.9.1", "react-native-linear-gradient": "^2.5.2", "react-native-login-twitter": "https://github.com/clockin/react-native-login-twitter.git", "react-native-maps": "^0.27.1", "react-native-material-bottom-navigation": "^1.0.1", "react-native-material-buttons": "^0.5.0", "react-native-material-kit": "^0.6.0-beta.1", "react-native-paper": "^2.15.1", "react-native-push-notification": "https://github.com/zo0r/react-native-push-notification.git", "react-native-reanimated": "^1.8.0", "react-native-screens": "^2.4.0", "react-native-sensitive-info": "^5.5.5", "react-native-share": "1.2.1", "react-native-shimmer-placeholder": "^1.0.29", "react-native-simple-markdown": "^1.1.0", "react-native-splash-screen": "^3.1.1", "react-native-svg": "^9.4.0", "react-native-vector-icons": "^6.6.0", "react-native-walkthrough-tooltip": "^0.6.1", "react-navigation": "^4.0.0", "react-navigation-redux-helpers": "^2.0.5", "react-navigation-stack": "^1.7.3", "react-navigation-tabs": "^2.8.11", "react-redux": "^5.0.7", "reactotron-apisauce": "^2.1.3", "reactotron-react-native": "^2.1.5", "reactotron-redux": "^2.1.3", "recompose": "^0.27.1", "redux": "^4.0.0", "redux-form": "6.8.0", "redux-recompose": "^1.0.11", "redux-thunk": "^2.3.0", "reinput": "^3.7.1", "reselect": "^4.0.0", "rn-fetch-blob": "^0.12.0", "rn-redux-middleware-flipper": "^0.1.0", "rollbar-react-native": "^0.7.1", "seamless-immutable": "^7.1.3", "url": "^0.11.0", "victory-native": "^32.0.2" }, "devDependencies": { "@babel/core": "^7.6.2", "@babel/runtime": "^7.6.2", "@react-native-community/eslint-config": "^0.0.5", "babel-eslint": "^10.0.1", "babel-jest": "^24.9.0", "babel-plugin-import-glob": "^2.0.0", "babel-plugin-module-resolver": "^4.0.0", "babel-preset-es2015": "^6.24.1", "babel-preset-react-native": "^4.0.1", "babel-register": "^6.26.0", "eslint": "^6.5.1", "eslint-config-airbnb": "^17.1.0", "eslint-config-prettier": "^3.3.0", "eslint-import-resolver-babel-module": "^5.1.2", "eslint-plugin-import": "^2.14.0", "eslint-plugin-jsx-a11y": "^6.1.2", "eslint-plugin-prettier": "^3.0.0", "eslint-plugin-react": "^7.11.1", "husky": "^1.1.4", "jest": "^24.9.0", "jest-react-native": "^18.0.0", "metro-react-native-babel-preset": "^0.58.0", "prettier": "^1.15.2", "prettier-eslint": "^8.8.2", "react-native-schemes-manager": "^1.0.5", "react-native-version": "^3.1.0", "react-test-renderer": "16.11.0" }, "jest": { "preset": "react-native", "transform": { "^.+\\.js$": "/node_modules/react-native/jest/preprocessor.js" } }, "husky": { "hooks": { "pre-commit": "yarn run lint-diff" } } } ``` #### `firebase.json` for react-native-firebase v6: ```json # N/A ```

iOS

Click To Expand

#### `ios/Podfile`: - [ ] I'm not using Pods - [x] I'm using Pods and my Podfile looks like: ```ruby platform :ios, '9.0' require_relative '../node_modules/@react-native-community/cli-platform-ios/native_modules' project 'utility', 'Debug' => :debug, 'Release' => :release, 'Debug.development.appname' => :debug, 'Debug.production.appname' => :debug, 'Debug.staging.appname' => :debug, 'Release.development.appname' => :release, 'Release.staging.appname' => :release, 'Release.production.appname' => :release def add_flipper_pods!(versions = {}) configurations = ['Debug', 'Release', 'Debug.development.appname', 'Debug.production.appname', 'Debug.staging.appname', 'Release.development.appname', 'Release.staging.appname', 'Release.production.appname'] versions['Flipper'] ||= '~> 0.33.1' versions['DoubleConversion'] ||= '1.1.7' versions['Flipper-Folly'] ||= '~> 2.1' versions['Flipper-Glog'] ||= '0.3.6' versions['Flipper-PeerTalk'] ||= '~> 0.0.4' versions['Flipper-RSocket'] ||= '~> 1.0' pod 'FlipperKit', versions['Flipper'], :configuration => configurations pod 'FlipperKit/FlipperKitLayoutPlugin', versions['Flipper'], :configuration => configurations pod 'FlipperKit/SKIOSNetworkPlugin', versions['Flipper'], :configuration => configurations pod 'FlipperKit/FlipperKitUserDefaultsPlugin', versions['Flipper'], :configuration => configurations pod 'FlipperKit/FlipperKitReactPlugin', versions['Flipper'], :configuration => configurations # List all transitive dependencies for FlipperKit pods # to avoid them being linked in Release builds pod 'Flipper', versions['Flipper'], :configuration => configurations pod 'Flipper-DoubleConversion', versions['DoubleConversion'], :configuration => configurations pod 'Flipper-Folly', versions['Flipper-Folly'], :configuration => configurations pod 'Flipper-Glog', versions['Flipper-Glog'], :configuration => configurations pod 'Flipper-PeerTalk', versions['Flipper-PeerTalk'], :configuration => configurations pod 'Flipper-RSocket', versions['Flipper-RSocket'], :configuration => configurations pod 'FlipperKit/Core', versions['Flipper'], :configuration => configurations pod 'FlipperKit/CppBridge', versions['Flipper'], :configuration => configurations pod 'FlipperKit/FBCxxFollyDynamicConvert', versions['Flipper'], :configuration => configurations pod 'FlipperKit/FBDefines', versions['Flipper'], :configuration => configurations pod 'FlipperKit/FKPortForwarding', versions['Flipper'], :configuration => configurations pod 'FlipperKit/FlipperKitHighlightOverlay', versions['Flipper'], :configuration => configurations pod 'FlipperKit/FlipperKitLayoutTextSearchable', versions['Flipper'], :configuration => configurations pod 'FlipperKit/FlipperKitNetworkPlugin', versions['Flipper'], :configuration => configurations end # Post Install processing for Flipper def flipper_post_install(installer) installer.pods_project.targets.each do |target| if target.name == 'YogaKit' target.build_configurations.each do |config| config.build_settings['SWIFT_VERSION'] = '4.1' end end # Enable Flipper for simulators only if target.name.start_with?('Flipper') || target.name == 'react-native-flipper' target.build_configurations.each do |config| config.build_settings['VALID_ARCHS'] = 'arm64 arm64e' end end end end # Post Install fix for wrong Flipper linking def fix_pods_post_install(installer, pods) installer.pods_project.targets.each do |target| if pods.include?(target.name) target.build_configurations.each do |config| config.build_settings['USE_HEADERMAP'] = 'NO' end end end end target 'utility' do pod 'FBLazyVector', :path => "../node_modules/react-native/Libraries/FBLazyVector" pod 'FBReactNativeSpec', :path => "../node_modules/react-native/Libraries/FBReactNativeSpec" pod 'RCTRequired', :path => "../node_modules/react-native/Libraries/RCTRequired" pod 'RCTTypeSafety', :path => "../node_modules/react-native/Libraries/TypeSafety" pod 'React', :path => '../node_modules/react-native/' pod 'React-Core', :path => '../node_modules/react-native/' pod 'React-CoreModules', :path => '../node_modules/react-native/React/CoreModules' pod 'React-Core/DevSupport', :path => '../node_modules/react-native/' pod 'React-RCTActionSheet', :path => '../node_modules/react-native/Libraries/ActionSheetIOS' pod 'React-RCTAnimation', :path => '../node_modules/react-native/Libraries/NativeAnimation' pod 'React-RCTBlob', :path => '../node_modules/react-native/Libraries/Blob' pod 'React-RCTImage', :path => '../node_modules/react-native/Libraries/Image' pod 'React-RCTLinking', :path => '../node_modules/react-native/Libraries/LinkingIOS' pod 'React-RCTNetwork', :path => '../node_modules/react-native/Libraries/Network' pod 'React-RCTSettings', :path => '../node_modules/react-native/Libraries/Settings' pod 'React-RCTText', :path => '../node_modules/react-native/Libraries/Text' pod 'React-RCTVibration', :path => '../node_modules/react-native/Libraries/Vibration' pod 'React-Core/RCTWebSocket', :path => '../node_modules/react-native/' pod 'React-cxxreact', :path => '../node_modules/react-native/ReactCommon/cxxreact' pod 'React-jsi', :path => '../node_modules/react-native/ReactCommon/jsi' pod 'React-jsiexecutor', :path => '../node_modules/react-native/ReactCommon/jsiexecutor' pod 'React-jsinspector', :path => '../node_modules/react-native/ReactCommon/jsinspector' pod 'ReactCommon/callinvoker', :path => "../node_modules/react-native/ReactCommon" pod 'ReactCommon/turbomodule/core', :path => "../node_modules/react-native/ReactCommon" pod 'Yoga', :path => '../node_modules/react-native/ReactCommon/yoga', :modular_headers => true 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' # Utility Pods (Manual Linking) # GoogleSignIn: Check https://github.com/react-native-community/google-signin/blob/e29ec23fd818c8edee962262474c34ddb7c830f1/docs/ios-guide.md pod 'GoogleSignIn', '~> 5.0.2' target 'utilityTests' do inherit! :complete # Pods for testing end use_native_modules! # Enables Flipper. # # Note that if you have use_frameworks! enabled, Flipper will not work and # you should disable these next few lines. add_flipper_pods! post_install do |installer| flipper_post_install(installer) fix_pods_post_install(installer, ["Rollbar", "react-native-material-kit"]) end end target 'utility-tvOS' do # Pods for utility-tvOS target 'utility-tvOSTests' do inherit! :search_paths # Pods for testing end end ``` #### `AppDelegate.m`: ```objc #import "AppDelegate.h" #import #import #import #import #if DEBUG && TARGET_OS_SIMULATOR #import #import #import #import #import #import static void InitializeFlipper(UIApplication *application) { FlipperClient *client = [FlipperClient sharedClient]; SKDescriptorMapper *layoutDescriptorMapper = [[SKDescriptorMapper alloc] initWithDefaults]; [client addPlugin:[[FlipperKitLayoutPlugin alloc] initWithRootNode:application withDescriptorMapper:layoutDescriptorMapper]]; [client addPlugin:[[FKUserDefaultsPlugin alloc] initWithSuiteName:nil]]; [client addPlugin:[FlipperKitReactPlugin new]]; [client addPlugin:[[FlipperKitNetworkPlugin alloc] initWithNetworkAdapter:[SKIOSNetworkAdapter new]]]; [client start]; } #endif #import #import #import "RNGoogleSignin.h" #import #import #import #import #import // react-native-firebase #import // react-native-rollbar #import "ReactNativeConfig.h" @implementation AppDelegate - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { #if DEBUG && TARGET_OS_SIMULATOR InitializeFlipper(application); #endif [[FBSDKApplicationDelegate sharedInstance] application:application didFinishLaunchingWithOptions:launchOptions]; [FIRApp configure]; // react-native-firebase RCTSetLogThreshold(RCTLogLevelInfo); RCTSetLogFunction(CrashlyticsReactLogFunction); // START Rollbar NSString *rollbarPostClientAccessKey = [ReactNativeConfig envFor:@"ROLLBAR_POST_CLIENT_ACCESS_KEY"]; RollbarConfiguration *config = [RollbarConfiguration configuration]; config.environment = @"appname"; [RollbarReactNative initWithAccessToken:rollbarPostClientAccessKey configuration:config]; //END Rollbar RCTBridge *bridge = [[RCTBridge alloc] initWithDelegate:self launchOptions:launchOptions]; RCTRootView *rootView = [[RCTRootView alloc] initWithBridge:bridge moduleName:@"utility" initialProperties:nil]; UNUserNotificationCenter *center = [UNUserNotificationCenter currentNotificationCenter]; center.delegate = self; rootView.backgroundColor = [[UIColor alloc] initWithRed:1.0f green:1.0f blue:1.0f alpha:1]; self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds]; UIViewController *rootViewController = [UIViewController new]; rootViewController.view = rootView; self.window.rootViewController = rootViewController; [self.window makeKeyAndVisible]; return YES; } - (NSURL *)sourceURLForBridge:(RCTBridge *)bridge { #if DEBUG return [[RCTBundleURLProvider sharedSettings] jsBundleURLForBundleRoot:@"index" fallbackResource:nil]; #else return [[NSBundle mainBundle] URLForResource:@"main" withExtension:@"jsbundle"]; #endif } // - (BOOL)application:(UIApplication *)application // openURL:(NSURL *)url // sourceApplication:(NSString *)sourceApplication annotation:(id)annotation // { // return [RCTLinkingManager application:application openURL:url // sourceApplication:sourceApplication annotation:annotation]; // } - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url options:(NSDictionary *)options { BOOL FBhandled = [[FBSDKApplicationDelegate sharedInstance] application:application openURL:url sourceApplication:options[UIApplicationOpenURLOptionsSourceApplicationKey] annotation:options[UIApplicationOpenURLOptionsAnnotationKey] ]; BOOL Ghandled = [RNGoogleSignin application:application openURL:url options:options[UIApplicationOpenURLOptionsAnnotationKey] ]; BOOL TWhandled = [[Twitter sharedInstance] application:application openURL:url options:options]; BOOL OutlookHandled = [RCTLinkingManager application:application openURL:url sourceApplication:options[UIApplicationOpenURLOptionsSourceApplicationKey] annotation:options[UIApplicationOpenURLOptionsAnnotationKey] ]; return FBhandled || Ghandled || TWhandled || OutlookHandled; } // Support Universal Linking // https://facebook.github.io/react-native/docs/linking // https://developer.apple.com/documentation/uikit/uiapplicationdelegate/1623072-application?language=objc - (BOOL)application:(UIApplication *)application continueUserActivity:(NSUserActivity *)userActivity restorationHandler:(void (^)(NSArray> *restorableObjects))restorationHandler { return [RCTLinkingManager application:application continueUserActivity:userActivity restorationHandler:restorationHandler]; } // Required to register for notifications - (void)application:(UIApplication *)application didRegisterUserNotificationSettings:(UIUserNotificationSettings *)notificationSettings { [RNCPushNotificationIOS didRegisterUserNotificationSettings:notificationSettings]; } // Required for the register event. - (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken { [RNCPushNotificationIOS didRegisterForRemoteNotificationsWithDeviceToken:deviceToken]; } // Required for the notification event. You must call the completion handler after handling the remote notification. - (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler { [RNCPushNotificationIOS didReceiveRemoteNotification:userInfo fetchCompletionHandler:completionHandler]; } // Required for the registrationError event. - (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error { [RNCPushNotificationIOS didFailToRegisterForRemoteNotificationsWithError:error]; } // Required for the localNotification event. - (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification { [RNCPushNotificationIOS didReceiveLocalNotification:notification]; } RCTLogFunction CrashlyticsReactLogFunction = ^( RCTLogLevel level, __unused RCTLogSource source, NSString *fileName, NSNumber *lineNumber, NSString *message ) { NSString *log = RCTFormatLog([NSDate date], level, fileName, lineNumber, message); #ifdef DEBUG fprintf(stderr, "%s\n", log.UTF8String); fflush(stderr); #else CLS_LOG(@"REACT LOG: %s", log.UTF8String); #endif int aslLevel; switch(level) { case RCTLogLevelTrace: aslLevel = ASL_LEVEL_DEBUG; break; case RCTLogLevelInfo: aslLevel = ASL_LEVEL_NOTICE; break; case RCTLogLevelWarning: aslLevel = ASL_LEVEL_WARNING; break; case RCTLogLevelError: aslLevel = ASL_LEVEL_ERR; break; case RCTLogLevelFatal: aslLevel = ASL_LEVEL_CRIT; break; } asl_log(NULL, NULL, aslLevel, "%s", message.UTF8String); }; @end ```


Android

Click To Expand

#### Have you converted to AndroidX? - [x] my application is an AndroidX application? - [x] I am using `android/gradle.settings` `jetifier=true` for Android compatibility? - [x] I am using the NPM package `jetifier` for react-native compatibility? #### `android/build.gradle`: ```groovy buildscript { ext { buildToolsVersion = "28.0.3" minSdkVersion = 19 compileSdkVersion = 28 targetSdkVersion = 28 supportLibVersion = "27.1.1" // This was added to solve a transitive dependency issue with rn-device-info googlePlayServicesIidVersion = "16.0.1" } repositories { google() jcenter() // Needed by Crashlytics, check: https://rnfirebase.io/crashlytics/android-setup maven { url 'https://maven.fabric.io/public' } } dependencies { classpath 'com.android.tools.build:gradle:3.5.2' classpath 'com.google.gms:google-services:4.2.0' classpath 'com.google.firebase:firebase-appdistribution-gradle:1.3.1' // Needed by Crashlytics, check: https://rnfirebase.io/crashlytics/android-setup classpath 'io.fabric.tools:gradle:1.28.1' } } allprojects { repositories { mavenLocal() maven { // All of React Native (JS, Obj-C sources, Android binaries) is installed from npm url("$rootDir/../node_modules/react-native/android") } maven { // Android JSC is installed from npm url("$rootDir/../node_modules/jsc-android/dist") } google() jcenter() maven { url 'https://www.jitpack.io' } } } ``` #### `android/app/build.gradle`: ```groovy apply plugin: "com.android.application" import com.android.build.OutputFile /** * The react.gradle file registers a task for each build variant (e.g. bundleDebugJsAndAssets * and bundleReleaseJsAndAssets). * These basically call `react-native bundle` with the correct arguments during the Android build * cycle. By default, bundleDebugJsAndAssets is skipped, as in debug/dev mode we prefer to load the * bundle directly from the development server. Below you can see all the possible configurations * and their defaults. If you decide to add a configuration block, make sure to add it before the * `apply from: "../../node_modules/react-native/react.gradle"` line. * * project.ext.react = [ * // the name of the generated asset file containing your JS bundle * bundleAssetName: "index.android.bundle", * * // the entry file for bundle generation. If none specified and * // "index.android.js" exists, it will be used. Otherwise "index.js" is * // default. Can be overridden with ENTRY_FILE environment variable. * entryFile: "index.android.js", * * // https://facebook.github.io/react-native/docs/performance#enable-the-ram-format * bundleCommand: "ram-bundle", * * // whether to bundle JS and assets in debug mode * bundleInDebug: false, * * // whether to bundle JS and assets in release mode * bundleInRelease: true, * * // whether to bundle JS and assets in another build variant (if configured). * // See http://tools.android.com/tech-docs/new-build-system/user-guide#TOC-Build-Variants * // The configuration property can be in the following formats * // 'bundleIn${productFlavor}${buildType}' * // 'bundleIn${buildType}' * // bundleInFreeDebug: true, * // bundleInPaidRelease: true, * // bundleInBeta: true, * * // whether to disable dev mode in custom build variants (by default only disabled in release) * // for example: to disable dev mode in the staging build type (if configured) * devDisabledInStaging: true, * // The configuration property can be in the following formats * // 'devDisabledIn${productFlavor}${buildType}' * // 'devDisabledIn${buildType}' * * // the root of your project, i.e. where "package.json" lives * root: "../../", * * // where to put the JS bundle asset in debug mode * jsBundleDirDebug: "$buildDir/intermediates/assets/debug", * * // where to put the JS bundle asset in release mode * jsBundleDirRelease: "$buildDir/intermediates/assets/release", * * // where to put drawable resources / React Native assets, e.g. the ones you use via * // require('./image.png')), in debug mode * resourcesDirDebug: "$buildDir/intermediates/res/merged/debug", * * // where to put drawable resources / React Native assets, e.g. the ones you use via * // require('./image.png')), in release mode * resourcesDirRelease: "$buildDir/intermediates/res/merged/release", * * // by default the gradle tasks are skipped if none of the JS files or assets change; this means * // that we don't look at files in android/ or ios/ to determine whether the tasks are up to * // date; if you have any other folders that you want to ignore for performance reasons (gradle * // indexes the entire tree), add them here. Alternatively, if you have JS files in android/ * // for example, you might want to remove it from here. * inputExcludes: ["android/**", "ios/**"], * * // override which node gets called and with what additional arguments * nodeExecutableAndArgs: ["node"], * * // supply additional arguments to the packager * extraPackagerArgs: [] * ] */ project.ext.react = [ enableHermes: false, // clean and rebuild if changing bundleInDevelopmentUtilityRelease: true, bundleInStagingUtilityRelease: true, bundleInProductionUtilityRelease: true, extraPackagerArgs: ["--sourcemap-output", "sourcemap.android.js", "--sourcemap-sources-root", "./"] ] apply from: "../../node_modules/react-native/react.gradle" apply from: "../../node_modules/react-native-vector-icons/fonts.gradle" /** * Set this to true to create two separate APKs instead of one: * - An APK that only works on ARM devices * - An APK that only works on x86 devices * The advantage is the size of the APK is reduced by about 4MB. * Upload all the APKs to the Play Store and people will download * the correct one based on the CPU architecture of their device. */ def enableSeparateBuildPerCPUArchitecture = false /** * Run Proguard to shrink the Java bytecode in release builds. */ def enableProguardInReleaseBuilds = false /** * The preferred build flavor of JavaScriptCore. * * For example, to use the international variant, you can use: * `def jscFlavor = 'org.webkit:android-jsc-intl:+'` * * The international variant includes ICU i18n library and necessary data * allowing to use e.g. `Date.toLocaleString` and `String.localeCompare` that * give correct results when using with locales other than en-US. Note that * this variant is about 6MiB larger per architecture than default. */ def jscFlavor = 'org.webkit:android-jsc:+' /** * Whether to enable the Hermes VM. * * This should be set on project.ext.react and mirrored here. If it is not set * on project.ext.react, JavaScript will not be compiled to Hermes Bytecode * and the benefits of using Hermes will therefore be sharply reduced. */ def enableHermes = project.ext.react.get("enableHermes", false); apply from: project(':react-native-config').projectDir.getPath() + "/dotenv.gradle" apply plugin: 'com.google.firebase.appdistribution' // Needed by Crashlytics, check: https://rnfirebase.io/crashlytics/android-setup apply plugin: 'io.fabric' task downloadResources(type: Exec) { def utilityId = project.env.get("UTILITY_ID") workingDir "$rootDir/.." commandLine "sh", "-c", "./downloadResources.sh $utilityId" } android { flavorDimensions "default" compileSdkVersion rootProject.ext.compileSdkVersion compileOptions { sourceCompatibility JavaVersion.VERSION_1_8 targetCompatibility JavaVersion.VERSION_1_8 } defaultConfig { applicationId "com.utility" minSdkVersion rootProject.ext.minSdkVersion targetSdkVersion rootProject.ext.targetSdkVersion versionCode 14 versionName "1.6.3" multiDexEnabled true resValue "string", "build_config_package", "com.utility" firebaseAppDistribution { groups="Companytesters, Utility-testers" } } signingConfigs { debug { storeFile file('keystore/debug.keystore') keyAlias 'androiddebugkey' keyPassword 'android' storePassword 'android' } release { if (project.hasProperty('UTILITYGO_RELEASE_STORE_FILE')) { storeFile file(UTILITYGO_RELEASE_STORE_FILE) storePassword UTILITYGO_RELEASE_STORE_PASSWORD keyAlias UTILITYGO_RELEASE_KEY_ALIAS keyPassword UTILITYGO_RELEASE_KEY_PASSWORD } } } packagingOptions { pickFirst "lib/armeabi-v7a/libc++_shared.so" pickFirst "lib/arm64-v8a/libc++_shared.so" pickFirst "lib/x86/libc++_shared.so" pickFirst "lib/x86_64/libc++_shared.so" } splits { abi { reset() enable enableSeparateBuildPerCPUArchitecture universalApk false // If true, also generate a universal APK include "armeabi-v7a", "x86", "arm64-v8a", "x86_64" } } buildTypes { release { minifyEnabled enableProguardInReleaseBuilds proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro" signingConfig signingConfigs.release } } productFlavors { developmentUtility { applicationIdSuffix "Utility.dev" resValue "string", "app_name", "Utility-dev" resValue "string", "build_config_package", "com.utility" firebaseAppDistribution { serviceCredentialsFile="$rootDir/google-credentials/Utilitydev-64217b1c9caf.json" } } stagingUtility { applicationIdSuffix "Utility.stage" resValue "string", "app_name", "Utility-stage" resValue "string", "build_config_package", "com.utility" firebaseAppDistribution { groups="Company-testers, Utility-testers" serviceCredentialsFile="$rootDir/google-credentials/Utility-stage-865f0dff4b4c.json" } } productionUtility { applicationIdSuffix "Utility" resValue "string", "app_name", "Utility" resValue "string", "build_config_package", "com.utility" firebaseAppDistribution { groups="Company-testers, Utility-testers" serviceCredentialsFile="$rootDir/google-credentials/Utility-prod-fd7362c23485.json" } } } // applicationVariants are e.g. debug, release applicationVariants.all { variant -> variant.javaCompiler.dependsOn(downloadResources) variant.outputs.each { output -> // For each separate APK per architecture, set a unique version code as described here: // http://tools.android.com/tech-docs/new-build-system/user-guide/apk-splits def versionCodes = ["armeabi-v7a":1, "x86":2, "arm64-v8a": 3, "x86_64": 4] def abi = output.getFilter(OutputFile.ABI) if (abi != null) { // null for the universal-debug, universal-release variants output.versionCodeOverride = versionCodes.get(abi) * 1048576 + defaultConfig.versionCode } } } } dependencies { implementation fileTree(dir: "libs", include: ["*.jar"]) implementation "com.facebook.react:react-native:+" // From node_modules implementation "androidx.swiperefreshlayout:swiperefreshlayout:1.0.0" // BEGIN FLIPPER debugImplementation("com.facebook.flipper:flipper:${FLIPPER_VERSION}") { exclude group:'com.facebook.fbjni' } debugImplementation("com.facebook.flipper:flipper-network-plugin:${FLIPPER_VERSION}") { exclude group:'com.facebook.flipper' } debugImplementation("com.facebook.flipper:flipper-fresco-plugin:${FLIPPER_VERSION}") { exclude group:'com.facebook.flipper' } //END FLIPPER if (enableHermes) { def hermesPath = "../../node_modules/hermes-engine/android/"; debugImplementation files(hermesPath + "hermes-debug.aar") releaseImplementation files(hermesPath + "hermes-release.aar") } else { implementation jscFlavor } implementation ('com.google.android.gms:play-services-gcm:16.1.0') implementation 'com.facebook.android:facebook-android-sdk:[5,6)' implementation(project(':react-native-maps')){ exclude group: 'com.google.android.gms', module: 'play-services-base' exclude group: 'com.google.android.gms', module: 'play-services-maps' } implementation(project(':react-native-community_google-signin')){ exclude group: "com.google.android.gms" } implementation 'com.google.android.gms:play-services-maps:16.1.0' implementation 'com.google.android.gms:play-services-basement:16.2.0' implementation 'com.google.android.gms:play-services-stats:16.0.1' implementation 'com.google.android.gms:play-services-auth:16.0.1' } // Run this once to be able to run the application with BUCK // puts all compile dependencies into folder libs for BUCK to use task copyDownloadableDepsToLibs(type: Copy) { from configurations.compile into 'libs' } apply plugin: 'com.google.gms.google-services' apply from: file("../../node_modules/@react-native-community/cli-platform-android/native_modules.gradle"); applyNativeModulesAppBuildGradle(project)``` #### `android/settings.gradle`: ```groovy rootProject.name = 'utility' include ':react-native-push-notification' project(':react-native-push-notification').projectDir = new File(rootProject.projectDir, '../node_modules/react-native-push-notification/android') include ':react-native-twitter-signin' project(':react-native-twitter-signin').projectDir = new File(rootProject.projectDir, '../node_modules/react-native-twitter-signin/android') apply from: file("../node_modules/@react-native-community/cli-platform-android/native_modules.gradle"); applyNativeModulesSettingsGradle(settings) include ':react-native-fbsdk' project(':react-native-fbsdk').projectDir = new File(rootProject.projectDir, '../node_modules/react-native-fbsdk/android') include ':app' ``` #### `MainApplication.java`: ```java package com.utility; import android.app.Application; import android.content.Context; import com.facebook.react.PackageList; import com.facebook.react.ReactApplication; import com.facebook.react.ReactNativeHost; import com.facebook.react.ReactPackage; import com.facebook.react.ReactInstanceManager; import com.facebook.react.PackageList; import com.facebook.soloader.SoLoader; import java.lang.reflect.InvocationTargetException; import java.util.List; import com.facebook.reactnative.androidsdk.FBSDKPackage; import com.rollbar.RollbarReactNative; import com.goldenowl.twittersignin.TwitterSigninPackage; import com.dieam.reactnativepushnotification.ReactNativePushNotificationPackage; import com.utility.widget.ClaimWidgetPackage; public class MainApplication extends Application implements ReactApplication { private final ReactNativeHost mReactNativeHost = new ReactNativeHost(this) { @Override public boolean getUseDeveloperSupport() { return BuildConfig.DEBUG; } @Override protected List getPackages() { @SuppressWarnings("UnnecessaryLocalVariable") final List packages = new PackageList(this).getPackages(); packages.add(new ClaimWidgetPackage()); return packages; } @Override protected String getJSMainModuleName() { return "index"; } }; @Override public ReactNativeHost getReactNativeHost() { return mReactNativeHost; } @Override public void onCreate() { super.onCreate(); SoLoader.init(this, /* native exopackage */ false); initializeFlipper(this, getReactNativeHost().getReactInstanceManager()); RollbarReactNative.init(this, BuildConfig.ROLLBAR_POST_CLIENT_ACCESS_KEY, "appname"); } /** * Loads Flipper in React Native templates. Call this in the onCreate method with something like * initializeFlipper(this, getReactNativeHost().getReactInstanceManager()); * * @param context * @param reactInstanceManager */ private static void initializeFlipper( Context context, ReactInstanceManager reactInstanceManager) { if (BuildConfig.DEBUG) { try { /* We use reflection here to pick up the class that initializes Flipper, since Flipper library is not available in release mode */ Class aClass = Class.forName("com.utility.ReactNativeFlipper"); aClass .getMethod("initializeFlipper", Context.class, ReactInstanceManager.class) .invoke(null, context, reactInstanceManager); } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (NoSuchMethodException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } } } } ``` #### `AndroidManifest.xml`: ```xml ```


Environment

Click To Expand

**`react-native info` output:** ``` System: OS: Linux 5.4 Ubuntu 20.04.1 LTS (Focal Fossa) CPU: (8) x64 Intel(R) Core(TM) i7-8550U CPU @ 1.80GHz Memory: 514.11 MB / 15.53 GB Shell: 5.8 - /usr/bin/zsh Binaries: Node: 12.18.2 - /tmp/yarn--1595945817567-0.2254903664387895/node Yarn: 1.22.4 - /tmp/yarn--1595945817567-0.2254903664387895/yarn npm: 6.14.5 - ~/.nvm/versions/node/v12.18.2/bin/npm Watchman: 4.9.0 - /usr/local/bin/watchman SDKs: Android SDK: API Levels: 23, 25, 28, 30 Build Tools: 28.0.3, 30.0.1 System Images: android-28 | Intel x86 Atom_64, android-28 | Google Play Intel x86 Atom, android-30 | Google APIs Intel x86 Atom Android NDK: Not Found IDEs: Android Studio: Not Found Languages: Java: 1.8.0_252 - /usr/bin/javac Python: 2.7.18 - /usr/bin/python npmPackages: @react-native-community/cli: Not Found react: 16.11.0 => 16.11.0 react-native: 0.62.2 => 0.62.2 npmGlobalPackages: *react-native*: Not Found ``` - **Platform that you're experiencing the issue on**: - [ ] iOS - [ ] Android - [ ] **iOS** but have not tested behavior on Android - [ ] **Android** but have not tested behavior on iOS - [x] Both - **`react-native-firebase` version you're using that has this issue:** - `6.4.0` - **`Firebase` module(s) you're using that has the issue:** - `@react-native-firebase/analytics@^6.4.0": ` - **Are you using `TypeScript`?** - `no`


taltrui commented 3 years ago

Hi! This is happening to me too, and my case is similar to OP, anyone any ideas what could be?

ricechenghk commented 3 years ago

Yes, I am having same problem too

ubbe-xyz commented 3 years ago

I'm experiencing the same issue, during development it seems some logged events appear there and some not. I'm trying on Android and with Analytics Debug Mode on.

mikehardy commented 3 years ago

All I can recommend is checking adb logcat carefully, and if there is a problem despite using an override to get the most recent firebase-android-sdk (so you know you are on the most recent release for a reproduction) then investigate if the same still happens without the react-native-firebase layer on top - you can do a pure firebase-android-sdk reproduction based off their quickstarts here: https://github.com/firebase/quickstart-android/tree/master/analytics

ubbe-xyz commented 3 years ago

From our side it's now working, the events we were sending were malformed and hence didn't show-up properly πŸ˜…

I think as an improvement the debug view could be more explicit when receiving malformed events, at the moment it just shows them in a different color and can be hard to notice πŸ€”

mikehardy commented 3 years ago

@lluia to confirm I understand correctly: you had "inconsistent" tracking (which to me means some events showed and some did not) but it seems now that in debug view they all show but some are there but a different color, this could be used to identify they are malformed? I ask because this could form a good addition to the docs (edit button on top right of every page!) if it could be reduced to a single authoritative statement: "if you send malformed events they will [maybe not appear / maybe be this color]" ?

ubbe-xyz commented 3 years ago

@mikehardy you're correct πŸ‘πŸ½ , the malformed events some appeared and some not, and appeared in a red color with a diferent icon than the rest of events ( even though is not that obvious they were wrong πŸ˜… ).

Yup, super happy to add this to the docs πŸ’―

stale[bot] commented 3 years ago

Hello πŸ‘‹, to help manage issues we automatically close stale issues. This issue has been automatically marked as stale because it has not had activity for quite some time. Has this issue been fixed, or does it still require the community's attention?

This issue will be closed in 15 days if no further activity occurs. Thank you for your contributions.

stale[bot] commented 3 years ago

Closing this issue after a prolonged period of inactivity. If this is still present in the latest release, please feel free to create a new issue with up-to-date information.