Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix app start span end-time is wrong if SDK init is deferred #2519

Merged
merged 10 commits into from
Feb 8, 2023
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
- Fix transaction performance collector oom ([#2505](https://github.com/getsentry/sentry-java/pull/2505))
- Remove authority from URLs sent to Sentry ([#2366](https://github.com/getsentry/sentry-java/pull/2366))
- Fix `sentry-bom` containing incorrect artifacts ([#2504](https://github.com/getsentry/sentry-java/pull/2504))
- Fix app start span end-time is wrong if SDK init is deferred ([#2519](https://github.com/getsentry/sentry-java/pull/2519))

### Dependencies

Expand Down
1 change: 1 addition & 0 deletions sentry-android-core/api/sentry-android-core.api
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,7 @@ public final class io/sentry/android/core/AppLifecycleIntegration : io/sentry/In
}

public final class io/sentry/android/core/AppStartState {
public fun getAppStartEndTime ()Lio/sentry/SentryDate;
public fun getAppStartInterval ()Ljava/lang/Long;
public fun getAppStartMillis ()Ljava/lang/Long;
public fun getAppStartTime ()Lio/sentry/SentryDate;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -56,8 +56,7 @@ public final class ActivityLifecycleIntegration
private boolean isAllActivityCallbacksAvailable;

private boolean firstActivityCreated = false;
private boolean firstActivityResumed = false;
private boolean foregroundImportance = false;
private final boolean foregroundImportance;

private @Nullable ISpan appStartSpan;
private final @NotNull WeakHashMap<Activity, ISpan> ttidSpanMap = new WeakHashMap<>();
Expand Down Expand Up @@ -210,6 +209,11 @@ private void startTracing(final @NotNull Activity activity) {
getAppStartDesc(coldStart),
appStartTime,
Instrumenter.SENTRY);

// in case there's already an end time (e.g. due to deferred SDK init)
// we can finish the app-start span
finishAppStartSpan();

// The first activity ttidSpan should start at the same time as the app start time
ttidSpanMap.put(
activity,
Expand Down Expand Up @@ -328,28 +332,17 @@ public synchronized void onActivityStarted(final @NotNull Activity activity) {
@SuppressLint("NewApi")
@Override
public synchronized void onActivityResumed(final @NotNull Activity activity) {
if (!firstActivityResumed) {

// we only finish the app start if the process is of foregroundImportance
if (foregroundImportance) {
// sets App start as finished when the very first activity calls onResume
AppStartState.getInstance().setAppStartEnd();
} else {
if (options != null) {
options
.getLogger()
.log(
SentryLevel.DEBUG,
"App Start won't be reported because Process wasn't of foregroundImportance.");
}
}

// finishes app start span
if (performanceEnabled && appStartSpan != null) {
appStartSpan.finish();
}
firstActivityResumed = true;
// app start span
@Nullable final SentryDate appStartStartTime = AppStartState.getInstance().getAppStartTime();
@Nullable final SentryDate appStartEndTime = AppStartState.getInstance().getAppStartEndTime();
// in case the SentryPerformanceProvider is disabled it does not set the app start times,
// and we need to set the end time manually here,
// the start time gets set manually in SentryAndroid.init()
if (appStartStartTime != null && appStartEndTime == null) {
AppStartState.getInstance().setAppStartEnd();
}
finishAppStartSpan();
markushi marked this conversation as resolved.
Show resolved Hide resolved

final ISpan ttidSpan = ttidSpanMap.get(activity);
final View rootView = activity.findViewById(android.R.id.content);
Expand Down Expand Up @@ -507,4 +500,17 @@ private void setColdStart(final @Nullable Bundle savedInstanceState) {
return APP_START_WARM;
}
}

private void finishAppStartSpan() {
final @Nullable SentryDate appStartEndTime = AppStartState.getInstance().getAppStartEndTime();
if (appStartSpan != null
&& !appStartSpan.isFinished()
&& performanceEnabled
&& appStartEndTime != null) {

final SpanStatus status =
appStartSpan.getStatus() != null ? appStartSpan.getStatus() : SpanStatus.OK;
appStartSpan.finish(status, appStartEndTime);
}
}
}
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
package io.sentry.android.core;

import android.os.SystemClock;
import io.sentry.DateUtils;
import io.sentry.SentryDate;
import io.sentry.SentryLongDate;
import org.jetbrains.annotations.ApiStatus;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
Expand Down Expand Up @@ -84,6 +86,20 @@ public SentryDate getAppStartTime() {
return appStartTime;
}

@Nullable
public SentryDate getAppStartEndTime() {
@Nullable final SentryDate start = getAppStartTime();
if (start != null) {
@Nullable final Long durationMillis = getAppStartInterval();
if (durationMillis != null) {
final long startNanos = start.nanoTimestamp();
final long endNanos = startNanos + DateUtils.millisToNanos(durationMillis);
return new SentryLongDate(endNanos);
}
}
return null;
}

@Nullable
public Long getAppStartMillis() {
return appStartMillis;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,8 @@ public final class SentryPerformanceProvider extends EmptySecureContentProvider
private static long appStartMillis = SystemClock.uptimeMillis();

private boolean firstActivityCreated = false;
private boolean firstActivityResumed = false;

private @Nullable Application application;

public SentryPerformanceProvider() {
Expand Down Expand Up @@ -91,9 +93,6 @@ public void onActivityCreated(@NotNull Activity activity, @Nullable Bundle saved
final boolean coldStart = savedInstanceState == null;
AppStartState.getInstance().setColdStart(coldStart);

if (application != null) {
application.unregisterActivityLifecycleCallbacks(this);
}
firstActivityCreated = true;
}
}
Expand All @@ -102,7 +101,16 @@ public void onActivityCreated(@NotNull Activity activity, @Nullable Bundle saved
public void onActivityStarted(@NotNull Activity activity) {}

@Override
public void onActivityResumed(@NotNull Activity activity) {}
public void onActivityResumed(@NotNull Activity activity) {
if (!firstActivityResumed) {
markushi marked this conversation as resolved.
Show resolved Hide resolved
// sets App start as finished when the very first activity calls onResume
firstActivityResumed = true;
AppStartState.getInstance().setAppStartEnd();
}
if (application != null) {
application.unregisterActivityLifecycleCallbacks(this);
}
}

@Override
public void onActivityPaused(@NotNull Activity activity) {}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import android.app.ActivityManager.RunningAppProcessInfo
import android.app.Application
import android.os.Bundle
import io.sentry.Breadcrumb
import io.sentry.DateUtils
import io.sentry.Hub
import io.sentry.Scope
import io.sentry.SentryDate
Expand Down Expand Up @@ -650,66 +651,110 @@ class ActivityLifecycleIntegrationTest {
}

@Test
fun `App start end time is set`() {
val sut = fixture.getSut(14)
fun `When firstActivityCreated is true, start transaction with given appStartTime`() {
val sut = fixture.getSut()
fixture.options.tracesSampleRate = 1.0
sut.register(fixture.hub, fixture.options)

setAppStartTime()
val date = SentryNanotimeDate(Date(0), 0)
setAppStartTime(date)

val activity = mock<Activity>()
sut.onActivityCreated(activity, null)
sut.onActivityResumed(activity)
sut.onActivityCreated(activity, fixture.bundle)

// SystemClock.uptimeMillis() always returns 0, can't assert real values
assertNotNull(AppStartState.getInstance().appStartInterval)
// call only once
verify(fixture.hub).startTransaction(any(), check<TransactionOptions> { assertEquals(date, it.startTimestamp) })
}

@Test
fun `App start end time isnt set if not foregroundImportance`() {
val sut = fixture.getSut(14, importance = RunningAppProcessInfo.IMPORTANCE_BACKGROUND)
fun `When firstActivityCreated is true, do not create app start span if not foregroundImportance`() {
val sut = fixture.getSut(importance = RunningAppProcessInfo.IMPORTANCE_BACKGROUND)
fixture.options.tracesSampleRate = 1.0
sut.register(fixture.hub, fixture.options)

setAppStartTime()
// usually set by SentryPerformanceProvider
val date = SentryNanotimeDate(Date(0), 0)
setAppStartTime(date)
AppStartState.getInstance().setAppStartEnd(1)

val activity = mock<Activity>()
sut.onActivityCreated(activity, null)
sut.onActivityResumed(activity)
sut.onActivityCreated(activity, fixture.bundle)

assertNull(AppStartState.getInstance().appStartInterval)
// call only once
verify(fixture.hub).startTransaction(any(), check<TransactionOptions> { assertNull(it.startTimestamp) })
}

@Test
fun `When firstActivityCreated is true, start transaction with given appStartTime`() {
val sut = fixture.getSut()
fun `Create and finish app start span immediately in case SDK init is deferred`() {
val sut = fixture.getSut(importance = RunningAppProcessInfo.IMPORTANCE_FOREGROUND)
fixture.options.tracesSampleRate = 1.0
sut.register(fixture.hub, fixture.options)

val date = SentryNanotimeDate(Date(0), 0)
setAppStartTime(date)
// usually set by SentryPerformanceProvider
val startDate = SentryNanotimeDate(Date(0), 0)
setAppStartTime(startDate)
AppStartState.getInstance().setColdStart(false)
AppStartState.getInstance().setAppStartEnd(1)

val endDate = AppStartState.getInstance().appStartEndTime!!

val activity = mock<Activity>()
sut.onActivityCreated(activity, fixture.bundle)

// call only once
verify(fixture.hub).startTransaction(any(), check<TransactionOptions> { assertEquals(date, it.startTimestamp) })
val appStartSpanCount = fixture.transaction.children.count {
it.spanContext.operation.startsWith("app.start.warm") &&
it.startDate.nanoTimestamp() == startDate.nanoTimestamp() &&
it.finishDate!!.nanoTimestamp() == endDate.nanoTimestamp()
}
assertEquals(1, appStartSpanCount)
}

@Test
fun `When firstActivityCreated is true, do not use appStartTime if not foregroundImportance`() {
val sut = fixture.getSut(importance = RunningAppProcessInfo.IMPORTANCE_BACKGROUND)
fun `When SentryPerformanceProvider is disabled, app start time span is still created`() {
val sut = fixture.getSut(importance = RunningAppProcessInfo.IMPORTANCE_FOREGROUND)
fixture.options.tracesSampleRate = 1.0
sut.register(fixture.hub, fixture.options)

val date = SentryNanotimeDate(Date(0), 0)
setAppStartTime(date)
// usually done by SentryPerformanceProvider, if disabled it's done by
// SentryAndroid.init
val startDate = SentryNanotimeDate(Date(0), 0)
setAppStartTime(startDate)
AppStartState.getInstance().setColdStart(false)

// when activity is created
val activity = mock<Activity>()
sut.onActivityCreated(activity, fixture.bundle)
// then app-start end time should still be null
assertNull(AppStartState.getInstance().appStartEndTime)

// call only once
verify(fixture.hub).startTransaction(any(), check<TransactionOptions> { assertNull(it.startTimestamp) })
// when activity is resumed
sut.onActivityResumed(activity)
// end-time should be set
assertNotNull(AppStartState.getInstance().appStartEndTime)
}

@Test
fun `When app-start end time is already set, it should not be overwritten`() {
val sut = fixture.getSut(importance = RunningAppProcessInfo.IMPORTANCE_FOREGROUND)
fixture.options.tracesSampleRate = 1.0
sut.register(fixture.hub, fixture.options)

// usually done by SentryPerformanceProvider
val startDate = SentryNanotimeDate(Date(0), 0)
setAppStartTime(startDate)
AppStartState.getInstance().setColdStart(false)
AppStartState.getInstance().setAppStartEnd(1234)

// when activity is created and resumed
val activity = mock<Activity>()
sut.onActivityCreated(activity, fixture.bundle)
sut.onActivityResumed(activity)

// then the end time should not be overwritten
assertEquals(
DateUtils.millisToNanos(1234),
AppStartState.getInstance().appStartEndTime!!.nanoTimestamp()
)
}

@Test
Expand Down
Original file line number Diff line number Diff line change
@@ -1,16 +1,20 @@
package io.sentry.android.core

import android.app.Application
import android.content.pm.ProviderInfo
import android.os.Bundle
import androidx.test.ext.junit.runners.AndroidJUnit4
import io.sentry.SentryNanotimeDate
import org.junit.runner.RunWith
import org.mockito.kotlin.any
import org.mockito.kotlin.mock
import org.mockito.kotlin.verify
import java.util.Date
import kotlin.test.BeforeTest
import kotlin.test.Test
import kotlin.test.assertEquals
import kotlin.test.assertFalse
import kotlin.test.assertNotNull
import kotlin.test.assertTrue

@RunWith(AndroidJUnit4::class)
Expand Down Expand Up @@ -73,6 +77,26 @@ class SentryPerformanceProviderTest {
assertFalse(AppStartState.getInstance().isColdStart!!)
}

@Test
fun `provider sets app start end on first activity resume, and unregisters afterwards`() {
val providerInfo = ProviderInfo()

val mockContext = ContextUtilsTest.createMockContext(true)
providerInfo.authority = AUTHORITY

val provider = SentryPerformanceProvider()
provider.attachInfo(mockContext, providerInfo)

provider.onActivityCreated(mock(), Bundle())
provider.onActivityResumed(mock())

assertNotNull(AppStartState.getInstance().appStartInterval)
assertNotNull(AppStartState.getInstance().appStartEndTime)

verify((mockContext.applicationContext as Application))
.unregisterActivityLifecycleCallbacks(any())
}

companion object {
private const val AUTHORITY = "io.sentry.sample.SentryPerformanceProvider"
}
Expand Down