Encode SpaSearchLandingKey with Base64

To ensure the landing key is the same as the indexing key.

Bug: 358238959
Flag: EXEMPT bug fix
Test: manual - search roaming
Test: unit test
Change-Id: I26ce712feaa83e9e70a3eb39039e5c3527f24cca
This commit is contained in:
Chaohui Wang
2024-08-16 15:12:40 +08:00
parent eb1fc8415b
commit 6afa25df8b
7 changed files with 100 additions and 18 deletions

View File

@@ -27,9 +27,6 @@ import com.android.settings.core.SubSettingLauncher
import com.android.settings.overlay.FeatureFactory.Companion.featureFactory import com.android.settings.overlay.FeatureFactory.Companion.featureFactory
import com.android.settings.password.PasswordUtils import com.android.settings.password.PasswordUtils
import com.android.settings.spa.SpaDestination import com.android.settings.spa.SpaDestination
import com.android.settings.spa.SpaSearchLanding.SpaSearchLandingKey
import com.google.protobuf.ByteString
import com.google.protobuf.InvalidProtocolBufferException
class SpaSearchLandingActivity : Activity() { class SpaSearchLandingActivity : Activity() {
override fun onCreate(savedInstanceState: Bundle?) { override fun onCreate(savedInstanceState: Bundle?) {
@@ -48,22 +45,17 @@ class SpaSearchLandingActivity : Activity() {
companion object { companion object {
@VisibleForTesting @VisibleForTesting
fun tryLaunch(context: Context, keyString: String) { fun tryLaunch(context: Context, keyString: String) {
val key = val key = decodeToSpaSearchLandingKey(keyString) ?: return
try {
SpaSearchLandingKey.parseFrom(ByteString.copyFromUtf8(keyString))
} catch (e: InvalidProtocolBufferException) {
Log.w(TAG, "arg key ($keyString) invalid", e)
return
}
if (key.hasSpaPage()) { if (key.hasSpaPage()) {
val destination = key.spaPage.destination val destination = key.spaPage.destination
if (destination.isNotEmpty()) { if (destination.isNotEmpty()) {
Log.d(TAG, "Launch SPA search result: ${key.spaPage}")
SpaDestination(destination = destination, highlightMenuKey = null) SpaDestination(destination = destination, highlightMenuKey = null)
.startFromExportedActivity(context) .startFromExportedActivity(context)
} }
} }
if (key.hasFragment()) { if (key.hasFragment()) {
Log.d(TAG, "Launch fragment search result: ${key.fragment}")
val arguments = val arguments =
Bundle().apply { Bundle().apply {
key.fragment.argumentsMap.forEach { (k, v) -> key.fragment.argumentsMap.forEach { (k, v) ->

View File

@@ -0,0 +1,34 @@
/*
* Copyright (C) 2024 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.settings.spa.search
import android.util.Base64
import android.util.Log
import com.android.settings.spa.SpaSearchLanding.SpaSearchLandingKey
private const val TAG = "SpaSearchLandingKeyExt"
fun SpaSearchLandingKey.encodeToString(): String =
Base64.encodeToString(toByteArray(), Base64.DEFAULT)
fun decodeToSpaSearchLandingKey(input: String): SpaSearchLandingKey? =
try {
SpaSearchLandingKey.parseFrom(Base64.decode(input, Base64.DEFAULT))
} catch (e: Exception) {
Log.w(TAG, "SpaSearchLandingKey ($input) invalid", e)
null
}

View File

@@ -96,7 +96,7 @@ class SpaSearchRepository(
keywords: String? = null, keywords: String? = null,
) = ) =
SearchIndexableRaw(context).apply { SearchIndexableRaw(context).apply {
key = spaSearchLandingKey.toByteString().toStringUtf8() key = spaSearchLandingKey.encodeToString()
title = itemTitle title = itemTitle
this.keywords = keywords this.keywords = keywords
intentAction = SEARCH_LANDING_ACTION intentAction = SEARCH_LANDING_ACTION

View File

@@ -30,8 +30,8 @@ import com.android.settings.spa.SpaSearchLanding.BundleValue
import com.android.settings.spa.SpaSearchLanding.SpaSearchLandingFragment import com.android.settings.spa.SpaSearchLanding.SpaSearchLandingFragment
import com.android.settings.spa.SpaSearchLanding.SpaSearchLandingKey import com.android.settings.spa.SpaSearchLanding.SpaSearchLandingKey
import com.android.settings.spa.search.SpaSearchLandingActivity import com.android.settings.spa.search.SpaSearchLandingActivity
import com.android.settings.spa.search.decodeToSpaSearchLandingKey
import com.google.common.truth.Truth.assertThat import com.google.common.truth.Truth.assertThat
import com.google.protobuf.ByteString
import org.junit.Before import org.junit.Before
import org.junit.Test import org.junit.Test
import org.junit.runner.RunWith import org.junit.runner.RunWith
@@ -104,7 +104,7 @@ class MobileNetworkSettingsSearchIndexTest {
searchIndexableData.searchIndexProvider.getDynamicRawDataToIndex(context, true) searchIndexableData.searchIndexProvider.getDynamicRawDataToIndex(context, true)
assertThat(dynamicRawDataToIndex).hasSize(1) assertThat(dynamicRawDataToIndex).hasSize(1)
val rawData = dynamicRawDataToIndex[0] val rawData = dynamicRawDataToIndex[0]
val key = SpaSearchLandingKey.parseFrom(ByteString.copyFromUtf8(rawData.key)) val key = decodeToSpaSearchLandingKey(rawData.key)
assertThat(key) assertThat(key)
.isEqualTo( .isEqualTo(
SpaSearchLandingKey.newBuilder() SpaSearchLandingKey.newBuilder()

View File

@@ -52,7 +52,7 @@ class SpaSearchLandingActivityTest {
.setSpaPage(SpaSearchLandingSpaPage.newBuilder().setDestination(DESTINATION)) .setSpaPage(SpaSearchLandingSpaPage.newBuilder().setDestination(DESTINATION))
.build() .build()
SpaSearchLandingActivity.tryLaunch(context, key.toByteString().toStringUtf8()) SpaSearchLandingActivity.tryLaunch(context, key.encodeToString())
verify(context).startActivity(argThat { getStringExtra(KEY_DESTINATION) == DESTINATION }) verify(context).startActivity(argThat { getStringExtra(KEY_DESTINATION) == DESTINATION })
} }
@@ -70,7 +70,7 @@ class SpaSearchLandingActivityTest {
BundleValue.newBuilder().setIntValue(ARGUMENT_VALUE).build())) BundleValue.newBuilder().setIntValue(ARGUMENT_VALUE).build()))
.build() .build()
SpaSearchLandingActivity.tryLaunch(context, key.toByteString().toStringUtf8()) SpaSearchLandingActivity.tryLaunch(context, key.encodeToString())
val intent = argumentCaptor<Intent> { verify(context).startActivity(capture()) }.firstValue val intent = argumentCaptor<Intent> { verify(context).startActivity(capture()) }.firstValue
assertThat(intent.getStringExtra(SettingsActivity.EXTRA_SHOW_FRAGMENT)) assertThat(intent.getStringExtra(SettingsActivity.EXTRA_SHOW_FRAGMENT))

View File

@@ -0,0 +1,57 @@
/*
* Copyright (C) 2024 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.settings.spa.search
import androidx.test.ext.junit.runners.AndroidJUnit4
import com.android.settings.spa.SpaSearchLanding.BundleValue
import com.android.settings.spa.SpaSearchLanding.SpaSearchLandingFragment
import com.android.settings.spa.SpaSearchLanding.SpaSearchLandingKey
import com.google.common.truth.Truth.assertThat
import org.junit.Test
import org.junit.runner.RunWith
@RunWith(AndroidJUnit4::class)
class SpaSearchLandingKeyExtTest {
@Test
fun encodeToString_thenDecode_shouldDecodeCorrectly() {
val encoded = KEY.encodeToString()
val decoded = decodeToSpaSearchLandingKey(encoded)
assertThat(decoded).isEqualTo(KEY)
}
@Test
fun decodeToSpaSearchLandingKey_badString_shouldReturnNull() {
val decoded = decodeToSpaSearchLandingKey("bad")
assertThat(decoded).isNull()
}
private companion object {
val KEY: SpaSearchLandingKey =
SpaSearchLandingKey.newBuilder()
.setFragment(
SpaSearchLandingFragment.newBuilder()
.setFragmentName("Destination")
.setPreferenceKey("preference_key")
.putArguments(
"argument_key", BundleValue.newBuilder().setIntValue(123).build()))
.build()
}
}

View File

@@ -23,7 +23,6 @@ import com.android.settings.spa.SpaSearchLanding.SpaSearchLandingSpaPage
import com.android.settings.spa.search.SpaSearchRepository.Companion.createSearchIndexableData import com.android.settings.spa.search.SpaSearchRepository.Companion.createSearchIndexableData
import com.android.settingslib.spa.framework.common.SettingsPageProvider import com.android.settingslib.spa.framework.common.SettingsPageProvider
import com.google.common.truth.Truth.assertThat import com.google.common.truth.Truth.assertThat
import com.google.protobuf.ByteString
import org.junit.Test import org.junit.Test
import org.junit.runner.RunWith import org.junit.runner.RunWith
import org.mockito.kotlin.mock import org.mockito.kotlin.mock
@@ -46,7 +45,7 @@ class SpaSearchRepositoryTest {
assertThat(searchIndexableData.targetClass).isEqualTo(pageProvider::class.java) assertThat(searchIndexableData.targetClass).isEqualTo(pageProvider::class.java)
assertThat(dynamicRawDataToIndex).hasSize(1) assertThat(dynamicRawDataToIndex).hasSize(1)
val rawData = dynamicRawDataToIndex[0] val rawData = dynamicRawDataToIndex[0]
val key = SpaSearchLandingKey.parseFrom(ByteString.copyFromUtf8(rawData.key)) val key = decodeToSpaSearchLandingKey(rawData.key)
assertThat(key) assertThat(key)
.isEqualTo( .isEqualTo(
SpaSearchLandingKey.newBuilder() SpaSearchLandingKey.newBuilder()