From 350f96f1f4f9a170b3569eb9b967b528801ff2ed Mon Sep 17 00:00:00 2001 From: Nikita Ogorodnikov Date: Tue, 12 Jul 2022 11:53:28 +0200 Subject: [PATCH] RUMM-2322: Report back interaction as tap, prevent reporting empty target name --- .../gestures/GesturesListener.kt | 18 +- .../gestures/WindowCallbackWrapper.kt | 2 +- .../GesturesListenerScrollSwipeTest.kt | 173 ++++++++---------- .../gestures/WindowCallbackWrapperTest.kt | 6 +- 4 files changed, 96 insertions(+), 103 deletions(-) diff --git a/dd-sdk-android/src/main/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/GesturesListener.kt b/dd-sdk-android/src/main/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/GesturesListener.kt index 54d7fb1aba..61316c8737 100644 --- a/dd-sdk-android/src/main/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/GesturesListener.kt +++ b/dd-sdk-android/src/main/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/GesturesListener.kt @@ -87,10 +87,14 @@ internal class GesturesListener( val scrollTarget = findTargetForScroll(decorView, startDownEvent.x, startDownEvent.y) if (scrollTarget != null) { scrollTargetReference = WeakReference(scrollTarget) + val targetId: String = resourceIdName(scrollTarget.id) + val attributes = resolveAttributes(scrollTarget, targetId, null) + // although we report scroll here, while it can be swipe in the end, it is fine, + // because the final type is taken from stopUserAction call anyway rumMonitor.startUserAction( - RumActionType.CUSTOM, - "", - emptyMap() + RumActionType.SCROLL, + resolveTargetName(interactionPredicate, scrollTarget), + attributes ) } else { return false @@ -132,14 +136,16 @@ internal class GesturesListener( private fun resolveAttributes( scrollTarget: View, targetId: String, - onUpEvent: MotionEvent + onUpEvent: MotionEvent? ): MutableMap { val attributes = mutableMapOf( RumAttributes.ACTION_TARGET_CLASS_NAME to scrollTarget.targetClassName(), RumAttributes.ACTION_TARGET_RESOURCE_ID to targetId ) - gestureDirection = resolveGestureDirection(onUpEvent) - attributes.put(RumAttributes.ACTION_GESTURE_DIRECTION, gestureDirection) + if (onUpEvent != null) { + gestureDirection = resolveGestureDirection(onUpEvent) + attributes.put(RumAttributes.ACTION_GESTURE_DIRECTION, gestureDirection) + } attributesProviders.forEach { it.extractAttributes(scrollTarget, attributes) diff --git a/dd-sdk-android/src/main/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/WindowCallbackWrapper.kt b/dd-sdk-android/src/main/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/WindowCallbackWrapper.kt index 8ab0ab2acf..fa946104ca 100644 --- a/dd-sdk-android/src/main/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/WindowCallbackWrapper.kt +++ b/dd-sdk-android/src/main/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/WindowCallbackWrapper.kt @@ -130,7 +130,7 @@ internal class WindowCallbackWrapper( } else { customTargetName } - GlobalRum.get().addUserAction(RumActionType.CUSTOM, targetName, emptyMap()) + GlobalRum.get().addUserAction(RumActionType.TAP, targetName, emptyMap()) } // endregion diff --git a/dd-sdk-android/src/test/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/GesturesListenerScrollSwipeTest.kt b/dd-sdk-android/src/test/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/GesturesListenerScrollSwipeTest.kt index 284abb2228..b22a21f61c 100644 --- a/dd-sdk-android/src/test/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/GesturesListenerScrollSwipeTest.kt +++ b/dd-sdk-android/src/test/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/GesturesListenerScrollSwipeTest.kt @@ -20,7 +20,6 @@ import com.datadog.android.rum.tracking.InteractionPredicate import com.datadog.android.utils.forge.Configurator import com.datadog.tools.unit.extensions.TestConfigurationExtension import com.nhaarman.mockitokotlin2.any -import com.nhaarman.mockitokotlin2.argumentCaptor import com.nhaarman.mockitokotlin2.eq import com.nhaarman.mockitokotlin2.inOrder import com.nhaarman.mockitokotlin2.mock @@ -34,7 +33,6 @@ import fr.xgouchet.elmyr.junit5.ForgeConfiguration import fr.xgouchet.elmyr.junit5.ForgeExtension import java.lang.ref.WeakReference import java.util.stream.Stream -import org.assertj.core.api.Assertions.assertThat import org.junit.jupiter.api.Test import org.junit.jupiter.api.extension.ExtendWith import org.junit.jupiter.api.extension.Extensions @@ -97,11 +95,12 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() } val expectedResourceName = forge.anAlphabeticalString() mockResourcesForTarget(scrollingTarget, expectedResourceName) - val expectedAttributes: MutableMap = mutableMapOf( + val expectedStartAttributes = mutableMapOf( RumAttributes.ACTION_TARGET_CLASS_NAME to scrollingTarget.javaClass.canonicalName, - RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName, - RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection + RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName ) + val expectedStopAttributes = expectedStartAttributes + + (RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection) testedListener = GesturesListener( WeakReference(mockWindow) ) @@ -116,14 +115,10 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() // Then inOrder(rumMonitor.mockInstance) { - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) - val argumentCaptor = argumentCaptor>() - verify(rumMonitor.mockInstance).stopUserAction( - eq(RumActionType.SCROLL), - eq(""), - argumentCaptor.capture() - ) - assertThat(argumentCaptor.firstValue).isEqualTo(expectedAttributes) + verify(rumMonitor.mockInstance) + .startUserAction(RumActionType.SCROLL, "", expectedStartAttributes) + verify(rumMonitor.mockInstance) + .stopUserAction(RumActionType.SCROLL, "", expectedStopAttributes) } verifyNoMoreInteractions(rumMonitor.mockInstance) } @@ -168,11 +163,12 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() } val expectedResourceName = forge.anAlphabeticalString() mockResourcesForTarget(scrollingTarget, expectedResourceName) - val expectedAttributes: MutableMap = mutableMapOf( + val expectedStartAttributes = mutableMapOf( RumAttributes.ACTION_TARGET_CLASS_NAME to scrollingTarget.javaClass.canonicalName, - RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName, - RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection + RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName ) + val expectedStopAttributes = expectedStartAttributes + + (RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection) testedListener = GesturesListener( WeakReference(mockWindow) ) @@ -187,14 +183,10 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() // Then inOrder(rumMonitor.mockInstance) { - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) - val argumentCaptor = argumentCaptor>() - verify(rumMonitor.mockInstance).stopUserAction( - eq(RumActionType.SCROLL), - eq(""), - argumentCaptor.capture() - ) - assertThat(argumentCaptor.firstValue).isEqualTo(expectedAttributes) + verify(rumMonitor.mockInstance) + .startUserAction(RumActionType.SCROLL, "", expectedStartAttributes) + verify(rumMonitor.mockInstance) + .stopUserAction(RumActionType.SCROLL, "", expectedStopAttributes) } verifyNoMoreInteractions(rumMonitor.mockInstance) } @@ -232,16 +224,19 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() } val expectedResourceName = forge.anAlphabeticalString() mockResourcesForTarget(scrollingTarget, expectedResourceName) - val expectedAttributes1: MutableMap = mutableMapOf( + val expectedStartAttributes1 = mutableMapOf( RumAttributes.ACTION_TARGET_CLASS_NAME to scrollingTarget.javaClass.canonicalName, - RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName, - RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection1 + RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName ) - val expectedAttributes2: MutableMap = mutableMapOf( + val expectedStartAttributes2 = mutableMapOf( RumAttributes.ACTION_TARGET_CLASS_NAME to scrollingTarget.javaClass.canonicalName, - RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName, - RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection2 + RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName ) + val expectedStopAttributes1 = expectedStartAttributes1 + + (RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection1) + + val expectedStopAttributes2 = expectedStartAttributes2 + + (RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection2) testedListener = GesturesListener( WeakReference(mockWindow) ) @@ -264,22 +259,14 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() // Then inOrder(rumMonitor.mockInstance) { - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) - val argumentCaptor1 = argumentCaptor>() - verify(rumMonitor.mockInstance).stopUserAction( - eq(RumActionType.SCROLL), - eq(""), - argumentCaptor1.capture() - ) - assertThat(argumentCaptor1.firstValue).isEqualTo(expectedAttributes1) - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) - val argumentCaptor2 = argumentCaptor>() - verify(rumMonitor.mockInstance).stopUserAction( - eq(RumActionType.SCROLL), - eq(""), - argumentCaptor2.capture() - ) - assertThat(argumentCaptor2.firstValue).isEqualTo(expectedAttributes2) + verify(rumMonitor.mockInstance) + .startUserAction(RumActionType.SCROLL, "", expectedStartAttributes1) + verify(rumMonitor.mockInstance) + .stopUserAction(RumActionType.SCROLL, "", expectedStopAttributes1) + verify(rumMonitor.mockInstance) + .startUserAction(RumActionType.SCROLL, "", expectedStartAttributes2) + verify(rumMonitor.mockInstance) + .stopUserAction(RumActionType.SCROLL, "", expectedStopAttributes2) } verifyNoMoreInteractions(rumMonitor.mockInstance) } @@ -410,11 +397,12 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() } val expectedResourceName = forge.anAlphabeticalString() mockResourcesForTarget(scrollingTarget, expectedResourceName) - val expectedAttributes: MutableMap = mutableMapOf( + val expectedStartAttributes = mutableMapOf( RumAttributes.ACTION_TARGET_CLASS_NAME to scrollingTarget.javaClass.canonicalName, - RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName, - RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection + RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName ) + val expectedStopAttributes = expectedStartAttributes + + (RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection) testedListener = GesturesListener( WeakReference(mockWindow) ) @@ -430,14 +418,10 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() // Then inOrder(rumMonitor.mockInstance) { - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) - val argumentCaptor = argumentCaptor>() - verify(rumMonitor.mockInstance).stopUserAction( - eq(RumActionType.SWIPE), - eq(""), - argumentCaptor.capture() - ) - assertThat(argumentCaptor.firstValue).isEqualTo(expectedAttributes) + verify(rumMonitor.mockInstance) + .startUserAction(RumActionType.SCROLL, "", expectedStartAttributes) + verify(rumMonitor.mockInstance) + .stopUserAction(RumActionType.SWIPE, "", expectedStopAttributes) } verifyNoMoreInteractions(rumMonitor.mockInstance) } @@ -488,11 +472,12 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() } val expectedResourceName = forge.anAlphabeticalString() mockResourcesForTarget(scrollingTarget, expectedResourceName) - val expectedAttributes: MutableMap = mutableMapOf( + val expectedStartAttributes = mutableMapOf( RumAttributes.ACTION_TARGET_CLASS_NAME to scrollingTarget.javaClass.simpleName, - RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName, - RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection + RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName ) + val expectedStopAttributes = expectedStartAttributes + + (RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection) testedListener = GesturesListener( WeakReference(mockWindow) ) @@ -508,14 +493,10 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() // Then inOrder(rumMonitor.mockInstance) { - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) - val argumentCaptor = argumentCaptor>() - verify(rumMonitor.mockInstance).stopUserAction( - eq(RumActionType.SWIPE), - eq(""), - argumentCaptor.capture() - ) - assertThat(argumentCaptor.firstValue).isEqualTo(expectedAttributes) + verify(rumMonitor.mockInstance) + .startUserAction(RumActionType.SCROLL, "", expectedStartAttributes) + verify(rumMonitor.mockInstance) + .stopUserAction(RumActionType.SWIPE, "", expectedStopAttributes) } verifyNoMoreInteractions(rumMonitor.mockInstance) } @@ -567,7 +548,11 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() // Then inOrder(rumMonitor.mockInstance) { - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) + verify(rumMonitor.mockInstance).startUserAction( + eq(RumActionType.SCROLL), + eq(fakeCustomTargetName), + any() + ) verify(rumMonitor.mockInstance).stopUserAction( eq(RumActionType.SCROLL), eq(fakeCustomTargetName), @@ -623,7 +608,11 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() // Then inOrder(rumMonitor.mockInstance) { - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) + verify(rumMonitor.mockInstance).startUserAction( + eq(RumActionType.SCROLL), + eq(""), + any() + ) verify(rumMonitor.mockInstance).stopUserAction( eq(RumActionType.SCROLL), eq(""), @@ -679,7 +668,11 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() // Then inOrder(rumMonitor.mockInstance) { - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) + verify(rumMonitor.mockInstance).startUserAction( + eq(RumActionType.SCROLL), + eq(""), + any() + ) verify(rumMonitor.mockInstance).stopUserAction( eq(RumActionType.SCROLL), eq(""), @@ -724,16 +717,18 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() } val expectedResourceName = forge.anAlphabeticalString() mockResourcesForTarget(scrollingTarget, expectedResourceName) - val expectedAttributes1: MutableMap = mutableMapOf( + val expectedStartAttributes1 = mutableMapOf( RumAttributes.ACTION_TARGET_CLASS_NAME to scrollingTarget.javaClass.canonicalName, - RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName, - RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection1 + RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName ) - val expectedAttributes2: MutableMap = mutableMapOf( + val expectedStartAttributes2 = mutableMapOf( RumAttributes.ACTION_TARGET_CLASS_NAME to scrollingTarget.javaClass.canonicalName, - RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName, - RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection2 + RumAttributes.ACTION_TARGET_RESOURCE_ID to expectedResourceName ) + val expectedStopAttributes1 = expectedStartAttributes1 + + (RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection1) + val expectedStopAttributes2 = expectedStartAttributes2 + + (RumAttributes.ACTION_GESTURE_DIRECTION to expectedDirection2) testedListener = GesturesListener( WeakReference(mockWindow) ) @@ -758,22 +753,14 @@ internal class GesturesListenerScrollSwipeTest : AbstractGesturesListenerTest() // Then inOrder(rumMonitor.mockInstance) { - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) - val argumentCaptor1 = argumentCaptor>() - verify(rumMonitor.mockInstance).stopUserAction( - eq(RumActionType.SWIPE), - eq(""), - argumentCaptor1.capture() - ) - assertThat(argumentCaptor1.firstValue).isEqualTo(expectedAttributes1) - verify(rumMonitor.mockInstance).startUserAction(RumActionType.CUSTOM, "", emptyMap()) - val argumentCaptor2 = argumentCaptor>() - verify(rumMonitor.mockInstance).stopUserAction( - eq(RumActionType.SWIPE), - eq(""), - argumentCaptor2.capture() - ) - assertThat(argumentCaptor2.firstValue).isEqualTo(expectedAttributes2) + verify(rumMonitor.mockInstance) + .startUserAction(RumActionType.SCROLL, "", expectedStartAttributes1) + verify(rumMonitor.mockInstance) + .stopUserAction(RumActionType.SWIPE, "", expectedStopAttributes1) + verify(rumMonitor.mockInstance) + .startUserAction(RumActionType.SCROLL, "", expectedStartAttributes2) + verify(rumMonitor.mockInstance) + .stopUserAction(RumActionType.SWIPE, "", expectedStopAttributes2) } verifyNoMoreInteractions(rumMonitor.mockInstance) } diff --git a/dd-sdk-android/src/test/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/WindowCallbackWrapperTest.kt b/dd-sdk-android/src/test/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/WindowCallbackWrapperTest.kt index 86a4f69be3..a81983b37f 100644 --- a/dd-sdk-android/src/test/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/WindowCallbackWrapperTest.kt +++ b/dd-sdk-android/src/test/kotlin/com/datadog/android/rum/internal/instrumentation/gestures/WindowCallbackWrapperTest.kt @@ -403,7 +403,7 @@ internal class WindowCallbackWrapperTest { // Then verify(rumMonitor.mockInstance).addUserAction( - RumActionType.CUSTOM, + RumActionType.TAP, WindowCallbackWrapper.BACK_DEFAULT_TARGET_NAME, emptyMap() ) @@ -429,7 +429,7 @@ internal class WindowCallbackWrapperTest { // Then verify(rumMonitor.mockInstance).addUserAction( - RumActionType.CUSTOM, + RumActionType.TAP, WindowCallbackWrapper.BACK_DEFAULT_TARGET_NAME, emptyMap() ) @@ -456,7 +456,7 @@ internal class WindowCallbackWrapperTest { // Then verify(rumMonitor.mockInstance).addUserAction( - RumActionType.CUSTOM, + RumActionType.TAP, customTargetName, emptyMap() )