Skip to content

Commit

Permalink
RUMM-2651 SR - skip new lines and spaces when obfuscating texts
Browse files Browse the repository at this point in the history
  • Loading branch information
mariusc83 committed Oct 13, 2022
1 parent e772754 commit 483486f
Show file tree
Hide file tree
Showing 6 changed files with 408 additions and 20 deletions.
5 changes: 5 additions & 0 deletions detekt.yml
Original file line number Diff line number Diff line change
Expand Up @@ -663,6 +663,7 @@ datadog:
# region Java misc
- "java.text.SimpleDateFormat.format(java.util.Date):java.lang.NullPointerException"
- "java.util.TimeZone.getTimeZone(kotlin.String):java.lang.NullPointerException"
- "java.lang.Character.toChars(kotlin.Int):java.lang.IllegalArgumentException"
- "java.lang.Runtime.addShutdownHook(java.lang.Thread):java.lang.IllegalArgumentException,java.lang.IllegalStateException,java.lang.SecurityException"
- "java.lang.System.arraycopy(kotlin.Any, kotlin.Int, kotlin.Any, kotlin.Int, kotlin.Int):java.lang.IndexOutOfBoundsException,java.lang.ArrayStoreException,java.lang.NullPointerException"
- "java.lang.System.loadLibrary(kotlin.String):java.lang.SecurityException,java.lang.UnsatisfiedLinkError,java.lang.NullPointerException"
Expand Down Expand Up @@ -914,6 +915,7 @@ datadog:
- "java.util.Stack.isNotEmpty()"
- "java.util.Stack.pop()"
- "java.util.Stack.push(com.datadog.android.sessionreplay.recorder.Node)"
- "java.util.stream.IntStream.forEach(java.util.function.IntConsumer)"
# endregion
# region Java Concurrency
- "java.lang.Thread.UncaughtExceptionHandler.uncaughtException(java.lang.Thread, kotlin.Throwable)"
Expand Down Expand Up @@ -976,6 +978,7 @@ datadog:
- "java.io.StringWriter.constructor()"
# endregion
# region Java misc
- "java.lang.Character.isWhitespace(kotlin.Int)"
- "java.lang.Class.hashCode()"
- "java.lang.Class.isAssignableFrom(java.lang.Class)"
- "java.lang.IllegalArgumentException.constructor(kotlin.String)"
Expand All @@ -996,6 +999,7 @@ datadog:
- "java.lang.ref.WeakReference.constructor(kotlin.Any)"
- "java.lang.ref.WeakReference.get()"
- "java.lang.StringBuilder.append(kotlin.Char)"
- "java.lang.StringBuilder.append(kotlin.CharArray)"
- "java.lang.StringBuilder.append(kotlin.String)"
- "java.math.BigInteger.toLong()"
- "java.nio.charset.Charset.defaultCharset()"
Expand Down Expand Up @@ -1199,6 +1203,7 @@ datadog:
# endregion
# region Kotlin String
- "kotlin.String.all(kotlin.Function1)"
- "kotlin.String.codePoints()"
- "kotlin.String.contains(kotlin.Char, kotlin.Boolean)"
- "kotlin.String.contains(kotlin.CharSequence, kotlin.Boolean)"
- "kotlin.String.count(kotlin.Function1)"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,11 @@ package com.datadog.android.sessionreplay.recorder.mapper
import android.widget.TextView

internal class MaskAllTextWireframeMapper(
viewWireframeMapper: ViewWireframeMapper = ViewWireframeMapper()
viewWireframeMapper: ViewWireframeMapper = ViewWireframeMapper(),
private val stringObfuscator: StringObfuscator = StringObfuscator()
) : TextWireframeMapper(viewWireframeMapper) {

override fun resolveTextValue(textView: TextView): String {
return String(CharArray(textView.text.length) { CHARACTER_MASK })
}

companion object {
private const val CHARACTER_MASK = 'x'
return stringObfuscator.obfuscate(textView.text.toString())
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
/*
* Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0.
* This product includes software developed at Datadog (https://www.datadoghq.com/).
* Copyright 2016-Present Datadog, Inc.
*/

package com.datadog.android.sessionreplay.recorder.mapper

import android.os.Build
import androidx.annotation.RequiresApi

internal class StringObfuscator {

fun obfuscate(stringValue: String): String {
return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
obfuscateUsingCodeStream(stringValue)
} else {
obfuscateUsingCharacterCode(stringValue)
}
}

private fun obfuscateUsingCharacterCode(stringValue: String): String {
return String(
CharArray(stringValue.length) {
val character = stringValue[it]
// Given that we replace each printable character with `x` for 2 chars expressions
// as emojis we will have 2 `x` instead of 1 but this will not be a problem as the
// obfuscation will still be applied.
if (Character.isWhitespace(character.code)) {
character
} else {
CHARACTER_MASK
}
}
)
}

@RequiresApi(Build.VERSION_CODES.N)
private fun obfuscateUsingCodeStream(stringValue: String): String {
// Because we are using the CharSequence.codePoints() stream we are going to correctly
// handle the cases where the text contains 2 chars expression. In this case one single
// codePoint will be returned for those 2 chars and the obfuscation char will be `x`.
val stringBuilder = StringBuilder()
stringValue.codePoints().forEach {
if (Character.isWhitespace(it)) {
// I don't think we should log this case here. I could not even reproduce it
// in my tests. As long as there is a valid string there there should not be
// any problem.
@Suppress("SwallowedException")
try {
stringBuilder.append(Character.toChars(it))
} catch (e: IllegalArgumentException) {
stringBuilder.append(CHARACTER_MASK)
}
} else {
stringBuilder.append(CHARACTER_MASK)
}
}
return stringBuilder.toString()
}

companion object {
private const val CHARACTER_MASK = 'x'
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ import com.datadog.android.sessionreplay.recorder.densityNormalized
import com.nhaarman.mockitokotlin2.mock
import com.nhaarman.mockitokotlin2.whenever
import fr.xgouchet.elmyr.Forge
import fr.xgouchet.elmyr.annotation.StringForgery
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.BeforeEach
import org.junit.jupiter.api.Test
Expand All @@ -25,6 +26,9 @@ internal abstract class BaseTextViewWireframeMapperTest : BaseWireframeMapperTes

lateinit var testedTextWireframeMapper: TextWireframeMapper

@StringForgery
lateinit var fakeText: String

@BeforeEach
fun `set up`() {
testedTextWireframeMapper = initTestedMapper()
Expand All @@ -44,7 +48,6 @@ internal abstract class BaseTextViewWireframeMapperTest : BaseWireframeMapperTes
// Given
val fakeFontSize = forge.aFloat(min = 0f)
val fakeStyleColor = forge.aStringMatching("#[0-9a-f]{6}ff")
val fakeText = forge.aString()
val fakeFontColor = fakeStyleColor
.substring(1)
.toLong(16)
Expand Down Expand Up @@ -81,7 +84,7 @@ internal abstract class BaseTextViewWireframeMapperTest : BaseWireframeMapperTes
) {
// Given
val mockTextView: TextView = forge.aMockView<TextView>().apply {
whenever(this.text).thenReturn(forge.aString())
whenever(this.text).thenReturn(fakeText)
whenever(this.typeface).thenReturn(mock())
whenever(this.textAlignment).thenReturn(fakeTextAlignment)
}
Expand All @@ -108,7 +111,6 @@ internal abstract class BaseTextViewWireframeMapperTest : BaseWireframeMapperTes
forge: Forge
) {
// Given
val fakeText = forge.aString()
val mockTextView: TextView = forge.aMockView<TextView>().apply {
whenever(this.text).thenReturn(fakeText)
whenever(this.typeface).thenReturn(mock())
Expand All @@ -133,7 +135,6 @@ internal abstract class BaseTextViewWireframeMapperTest : BaseWireframeMapperTes
@Test
fun `M resolve a TextWireframe W map() { TextView with textPadding }`(forge: Forge) {
// Given
val fakeText = forge.aString()
val fakeTextPaddingTop = forge.anInt()
val fakeTextPaddingBottom = forge.anInt()
val fakeTextPaddingStart = forge.anInt()
Expand Down Expand Up @@ -192,7 +193,7 @@ internal abstract class BaseTextViewWireframeMapperTest : BaseWireframeMapperTes
}
val mockTextView = forge.aMockView<TextView>().apply {
whenever(this.background).thenReturn(mockDrawable)
whenever(this.text).thenReturn(forge.aString())
whenever(this.text).thenReturn(fakeText)
whenever(this.typeface).thenReturn(mock())
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,20 +6,21 @@

package com.datadog.android.sessionreplay.recorder.mapper

import android.widget.Button
import android.widget.TextView
import com.datadog.android.sessionreplay.recorder.aMockView
import com.datadog.android.sessionreplay.utils.ForgeConfigurator
import com.datadog.tools.unit.extensions.ApiLevelExtension
import com.nhaarman.mockitokotlin2.mock
import com.nhaarman.mockitokotlin2.whenever
import fr.xgouchet.elmyr.Forge
import fr.xgouchet.elmyr.annotation.StringForgery
import fr.xgouchet.elmyr.junit5.ForgeConfiguration
import fr.xgouchet.elmyr.junit5.ForgeExtension
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
import org.mockito.Mock
import org.mockito.junit.jupiter.MockitoExtension
import org.mockito.junit.jupiter.MockitoSettings
import org.mockito.quality.Strictness
Expand All @@ -33,34 +34,43 @@ import org.mockito.quality.Strictness
@ForgeConfiguration(ForgeConfigurator::class)
internal class MaskAllTextViewWireframeMapperTest : BaseTextViewWireframeMapperTest() {

@Mock
lateinit var mockStringObfuscator: StringObfuscator

@StringForgery
lateinit var fakeMaskedStringValue: String

// region super

override fun initTestedMapper(): TextWireframeMapper {
return MaskAllTextWireframeMapper()
whenever(mockStringObfuscator.obfuscate(fakeText)).thenReturn(fakeMaskedStringValue)
return MaskAllTextWireframeMapper(stringObfuscator = mockStringObfuscator)
}

override fun resolveTextValue(textView: TextView): String {
return String(CharArray(textView.text.length) { 'x' })
return fakeMaskedStringValue
}

// endregion

// region Unit tests

@Test
fun `M resolve a TextWireframe with masked text W map() { TextView with text }`(forge: Forge) {
fun `M resolve a TextWireframe with masked text W map(){TextView}`(
forge: Forge
) {
// Given
val fakeText = forge.aString { 'x' }
val mockButton: TextView = forge.aMockView<Button>().apply {
whenever(this.text).thenReturn(forge.aString(fakeText.length))
whenever(mockStringObfuscator.obfuscate(fakeText)).thenReturn(fakeMaskedStringValue)
val mockTextView: TextView = forge.aMockView<TextView>().apply {
whenever(this.text).thenReturn(fakeText)
whenever(this.typeface).thenReturn(mock())
}

// When
val textWireframe = testedTextWireframeMapper.map(mockButton, fakePixelDensity)
val textWireframe = testedTextWireframeMapper.map(mockTextView, fakePixelDensity)

// Then
val expectedWireframe = mockButton.toTextWireframe().copy(text = fakeText)
val expectedWireframe = mockTextView.toTextWireframe().copy(text = fakeMaskedStringValue)
assertThat(textWireframe).isEqualTo(expectedWireframe)
}

Expand Down
Loading

0 comments on commit 483486f

Please sign in to comment.