/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ package org.mozilla.fenix.components import android.app.Application import android.app.PendingIntent import android.content.Context import android.content.Intent import android.net.Uri import android.os.StrictMode import mozilla.components.lib.crash.CrashReporter import mozilla.components.lib.crash.service.CrashReporterService import mozilla.components.lib.crash.service.GleanCrashReporterService import mozilla.components.lib.crash.service.MozillaSocorroService import mozilla.components.lib.crash.service.SentryService import mozilla.components.service.nimbus.NimbusApi import mozilla.components.service.nimbus.Nimbus as NimbusEnabled import mozilla.components.service.nimbus.NimbusDisabled import mozilla.components.service.nimbus.NimbusServerSettings import org.mozilla.fenix.BuildConfig import org.mozilla.fenix.Config import org.mozilla.fenix.FeatureFlags import org.mozilla.fenix.HomeActivity import org.mozilla.fenix.R import org.mozilla.fenix.ReleaseChannel import org.mozilla.fenix.components.metrics.AdjustMetricsService import org.mozilla.fenix.components.metrics.GleanMetricsService import org.mozilla.fenix.components.metrics.LeanplumMetricsService import org.mozilla.fenix.components.metrics.MetricController import org.mozilla.fenix.ext.components import org.mozilla.fenix.ext.settings import org.mozilla.fenix.perf.lazyMonitored import org.mozilla.fenix.utils.Mockable import org.mozilla.geckoview.BuildConfig.MOZ_APP_BUILDID import org.mozilla.geckoview.BuildConfig.MOZ_APP_VENDOR import org.mozilla.geckoview.BuildConfig.MOZ_APP_VERSION import org.mozilla.geckoview.BuildConfig.MOZ_UPDATE_CHANNEL /** * Component group for all functionality related to analytics e.g. crash reporting and telemetry. */ @Mockable class Analytics( private val context: Context ) { val crashReporter: CrashReporter by lazyMonitored { val services = mutableListOf() if (isSentryEnabled()) { val sentryService = SentryService( context, BuildConfig.SENTRY_TOKEN, tags = mapOf("geckoview" to "$MOZ_APP_VERSION-$MOZ_APP_BUILDID"), environment = BuildConfig.BUILD_TYPE, sendEventForNativeCrashes = false, // Do not send native crashes to Sentry sentryProjectUrl = getSentryProjectUrl() ) services.add(sentryService) } // The name "Fenix" here matches the product name on Socorro and is unrelated to the actual app name: // https://bugzilla.mozilla.org/show_bug.cgi?id=1523284 val socorroService = MozillaSocorroService(context, appName = "Fenix", version = MOZ_APP_VERSION, buildId = MOZ_APP_BUILDID, vendor = MOZ_APP_VENDOR, releaseChannel = MOZ_UPDATE_CHANNEL) services.add(socorroService) val intent = Intent(context, HomeActivity::class.java).apply { flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TOP } val pendingIntent = PendingIntent.getActivity( context, 0, intent, 0 ) CrashReporter( context = context, services = services, telemetryServices = listOf(GleanCrashReporterService(context)), shouldPrompt = CrashReporter.Prompt.ALWAYS, promptConfiguration = CrashReporter.PromptConfiguration( appName = context.getString(R.string.app_name), organizationName = "Mozilla" ), enabled = true, nonFatalCrashIntent = pendingIntent ) } val leanplumMetricsService by lazyMonitored { LeanplumMetricsService(context as Application) } val metrics: MetricController by lazyMonitored { MetricController.create( listOf( GleanMetricsService(context, lazy { context.components.core.store }), leanplumMetricsService, AdjustMetricsService(context as Application) ), isDataTelemetryEnabled = { context.settings().isTelemetryEnabled }, isMarketingDataTelemetryEnabled = { context.settings().isMarketingTelemetryEnabled } ) } val experiments: NimbusApi by lazyMonitored { if (FeatureFlags.nimbusExperiments) { // Eventually we'll want to use `NimbusDisabled` when we have no NIMBUS_ENDPOINT. // but we keep this here to not mix feature flags and how we configure Nimbus. val url: String? = BuildConfig.NIMBUS_ENDPOINT val serverSettings = if (!url.isNullOrBlank()) { NimbusServerSettings(url = Uri.parse(url)) } else { null } NimbusEnabled(context, serverSettings).apply { // Global opt out state is stored in Nimbus, and shouldn't be toggled to `true` // from the app unless the user does so from a UI control. // However, the user may have opt-ed out of mako experiments already, so // we should respect that setting here. val enabled = context.components.strictMode.resetAfter(StrictMode.allowThreadDiskReads()) { context.settings().isExperimentationEnabled } if (!enabled) { globalUserParticipation = enabled } // Nimbus should look after downloading experiment definitions from remote settings // on another thread, and making sure we don't hit the server each time we start. updateExperiments() } } else { NimbusDisabled() } } } fun isSentryEnabled() = !BuildConfig.SENTRY_TOKEN.isNullOrEmpty() private fun getSentryProjectUrl(): String? { val baseUrl = "https://sentry.prod.mozaws.net/operations" return when (Config.channel) { ReleaseChannel.Nightly -> "$baseUrl/fenix" ReleaseChannel.Release -> "$baseUrl/fenix-fennec" ReleaseChannel.Beta -> "$baseUrl/fenix-fennec-beta" else -> null } }