Fix ANR in TelephonyStatusControlSession

Feature.get() blocks on the main thread, which cause the ANR.

Cancel the job instead to fix.

Fix: 287702163
Test: Manually with MobileNetworkSettings
Test: atest TelephonyStatusControlSessionTest
Change-Id: Id873e56359dbf198c31686c2280c979294c95c3d
This commit is contained in:
Chaohui Wang
2023-06-21 16:34:07 +08:00
parent de5c8613ba
commit 88fd45b1e6
4 changed files with 168 additions and 120 deletions

View File

@@ -0,0 +1,81 @@
/*
* Copyright (C) 2023 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.network.telephony
import android.content.Context
import androidx.lifecycle.testing.TestLifecycleOwner
import androidx.test.core.app.ApplicationProvider
import androidx.test.ext.junit.runners.AndroidJUnit4
import com.android.settings.core.BasePreferenceController
import com.android.settingslib.spa.testutils.waitUntil
import com.google.common.truth.Truth.assertThat
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.test.runTest
import org.junit.Test
import org.junit.runner.RunWith
@OptIn(ExperimentalCoroutinesApi::class)
@RunWith(AndroidJUnit4::class)
class TelephonyStatusControlSessionTest {
private val context: Context = ApplicationProvider.getApplicationContext()
@Test
fun init() = runTest {
val controller = TestController(context)
val session = TelephonyStatusControlSession(
controllers = listOf(controller),
lifecycle = TestLifecycleOwner().lifecycle,
)
waitUntil { controller.availabilityStatus == STATUS }
session.close()
}
@Test
fun close() = runTest {
val controller = TestController(context)
val session = TelephonyStatusControlSession(
controllers = listOf(controller),
lifecycle = TestLifecycleOwner().lifecycle,
)
session.close()
assertThat(controller.availabilityStatus).isNull()
}
private companion object {
const val KEY = "key"
const val STATUS = BasePreferenceController.AVAILABLE
}
private class TestController(context: Context) : BasePreferenceController(context, KEY),
TelephonyAvailabilityHandler {
var availabilityStatus: Int? = null
override fun getAvailabilityStatus(): Int = STATUS
override fun setAvailabilityStatus(status: Int) {
availabilityStatus = status
}
override fun unsetAvailabilityStatus() {
availabilityStatus = null
}
}
}