Repo created
This commit is contained in:
parent
d327c31227
commit
0b2aca0925
638 changed files with 76461 additions and 0 deletions
1
app/test/.gitignore
vendored
Normal file
1
app/test/.gitignore
vendored
Normal file
|
|
@ -0,0 +1 @@
|
|||
/build
|
||||
30
app/test/build.gradle.kts
Normal file
30
app/test/build.gradle.kts
Normal file
|
|
@ -0,0 +1,30 @@
|
|||
plugins {
|
||||
id("com.android.application")
|
||||
kotlin("android")
|
||||
}
|
||||
|
||||
android {
|
||||
namespace = "com.topjohnwu.magisk.test"
|
||||
|
||||
defaultConfig {
|
||||
applicationId = "com.topjohnwu.magisk.test"
|
||||
versionCode = 1
|
||||
versionName = "1.0"
|
||||
proguardFile("proguard-rules.pro")
|
||||
}
|
||||
|
||||
buildTypes {
|
||||
release {
|
||||
isMinifyEnabled = true
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
setupTestApk()
|
||||
|
||||
dependencies {
|
||||
implementation(libs.test.runner)
|
||||
implementation(libs.test.rules)
|
||||
implementation(libs.test.junit)
|
||||
implementation(libs.test.uiautomator)
|
||||
}
|
||||
13
app/test/proguard-rules.pro
vendored
Normal file
13
app/test/proguard-rules.pro
vendored
Normal file
|
|
@ -0,0 +1,13 @@
|
|||
# Keep all test dependencies
|
||||
-keep class org.junit.** { *; }
|
||||
-keep class androidx.test.** { *; }
|
||||
|
||||
# Make sure the classloader constructor is kept
|
||||
-keepclassmembers class com.topjohnwu.magisk.test.TestClassLoader { <init>(); }
|
||||
|
||||
# Repackage dependencies
|
||||
-repackageclasses 'deps'
|
||||
-allowaccessmodification
|
||||
|
||||
# Keep attributes for stacktrace
|
||||
-keepattributes *
|
||||
23
app/test/src/main/AndroidManifest.xml
Normal file
23
app/test/src/main/AndroidManifest.xml
Normal file
|
|
@ -0,0 +1,23 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:tools="http://schemas.android.com/tools">
|
||||
|
||||
<uses-permission
|
||||
android:name="android.permission.QUERY_ALL_PACKAGES"
|
||||
tools:ignore="QueryAllPackagesPermission" />
|
||||
|
||||
<queries tools:node="removeAll" />
|
||||
|
||||
<application tools:node="replace">
|
||||
<uses-library android:name="android.test.runner" />
|
||||
</application>
|
||||
|
||||
<instrumentation
|
||||
android:name="com.topjohnwu.magisk.test.AppTestRunner"
|
||||
android:targetPackage="com.topjohnwu.magisk" />
|
||||
|
||||
<instrumentation
|
||||
android:name="com.topjohnwu.magisk.test.TestRunner"
|
||||
android:targetPackage="com.topjohnwu.magisk.test" />
|
||||
|
||||
</manifest>
|
||||
|
|
@ -0,0 +1,88 @@
|
|||
package com.topjohnwu.magisk.test
|
||||
|
||||
import android.content.BroadcastReceiver
|
||||
import android.content.Context
|
||||
import android.content.Intent
|
||||
import android.content.IntentFilter
|
||||
import android.os.ParcelFileDescriptor.AutoCloseInputStream
|
||||
import androidx.annotation.Keep
|
||||
import androidx.test.ext.junit.runners.AndroidJUnit4
|
||||
import androidx.test.platform.app.InstrumentationRegistry
|
||||
import org.junit.After
|
||||
import org.junit.Assert.assertTrue
|
||||
import org.junit.Test
|
||||
import org.junit.runner.RunWith
|
||||
import java.util.concurrent.CountDownLatch
|
||||
import java.util.concurrent.TimeUnit
|
||||
|
||||
@Keep
|
||||
@RunWith(AndroidJUnit4::class)
|
||||
class AppMigrationTest {
|
||||
|
||||
companion object {
|
||||
private const val APP_PKG = "com.topjohnwu.magisk"
|
||||
private const val STUB_PKG = "repackaged.$APP_PKG"
|
||||
private const val RECEIVER_TIMEOUT = 20L
|
||||
}
|
||||
|
||||
private val instrumentation get() = InstrumentationRegistry.getInstrumentation()
|
||||
private val context get() = instrumentation.context
|
||||
private val uiAutomation get() = instrumentation.uiAutomation
|
||||
private val registeredReceivers = mutableListOf<BroadcastReceiver>()
|
||||
|
||||
class PackageRemoveMonitor(
|
||||
context: Context,
|
||||
private val packageName: String
|
||||
) : BroadcastReceiver() {
|
||||
|
||||
val latch = CountDownLatch(1)
|
||||
|
||||
init {
|
||||
val filter = IntentFilter(Intent.ACTION_PACKAGE_REMOVED)
|
||||
filter.addDataScheme("package")
|
||||
context.registerReceiver(this, filter)
|
||||
}
|
||||
|
||||
override fun onReceive(context: Context, intent: Intent) {
|
||||
if (intent.action != Intent.ACTION_PACKAGE_REMOVED)
|
||||
return
|
||||
val data = intent.data ?: return
|
||||
val pkg = data.schemeSpecificPart
|
||||
if (pkg == packageName) latch.countDown()
|
||||
}
|
||||
}
|
||||
|
||||
@After
|
||||
fun tearDown() {
|
||||
registeredReceivers.forEach(context::unregisterReceiver)
|
||||
}
|
||||
|
||||
private fun testAppMigration(pkg: String, method: String) {
|
||||
val receiver = PackageRemoveMonitor(context, pkg)
|
||||
registeredReceivers.add(receiver)
|
||||
|
||||
// Trigger the test to run migration
|
||||
val pfd = uiAutomation.executeShellCommand(
|
||||
"am instrument -w --user 0 -e class .Environment#$method " +
|
||||
"$pkg.test/${AppTestRunner::class.java.name}"
|
||||
)
|
||||
val output = AutoCloseInputStream(pfd).reader().use { it.readText() }
|
||||
assertTrue("$method failed, inst out: $output", output.contains("OK ("))
|
||||
|
||||
// Wait for migration to complete
|
||||
assertTrue(
|
||||
"$pkg uninstallation failed",
|
||||
receiver.latch.await(RECEIVER_TIMEOUT, TimeUnit.SECONDS)
|
||||
)
|
||||
}
|
||||
|
||||
@Test
|
||||
fun testAppHide() {
|
||||
testAppMigration(APP_PKG, "setupAppHide")
|
||||
}
|
||||
|
||||
@Test
|
||||
fun testAppRestore() {
|
||||
testAppMigration(STUB_PKG, "setupAppRestore")
|
||||
}
|
||||
}
|
||||
35
app/test/src/main/java/com/topjohnwu/magisk/test/Runners.kt
Normal file
35
app/test/src/main/java/com/topjohnwu/magisk/test/Runners.kt
Normal file
|
|
@ -0,0 +1,35 @@
|
|||
package com.topjohnwu.magisk.test
|
||||
|
||||
import android.os.Bundle
|
||||
import androidx.test.platform.app.InstrumentationRegistry
|
||||
import androidx.test.runner.AndroidJUnitRunner
|
||||
|
||||
open class TestRunner : AndroidJUnitRunner() {
|
||||
override fun onCreate(arguments: Bundle) {
|
||||
// Support short-hand ".ClassName"
|
||||
arguments.getString("class")?.let {
|
||||
val classArg = it.split(",").joinToString(separator = ",") { clz ->
|
||||
if (clz.startsWith(".")) {
|
||||
"com.topjohnwu.magisk.test$clz"
|
||||
} else {
|
||||
clz
|
||||
}
|
||||
}
|
||||
arguments.putString("class", classArg)
|
||||
}
|
||||
super.onCreate(arguments)
|
||||
}
|
||||
}
|
||||
|
||||
class AppTestRunner : TestRunner() {
|
||||
override fun onCreate(arguments: Bundle) {
|
||||
// Force using the target context's classloader to run tests
|
||||
arguments.putString("classLoader", TestClassLoader::class.java.name)
|
||||
super.onCreate(arguments)
|
||||
}
|
||||
}
|
||||
|
||||
private val targetClassLoader inline get() =
|
||||
InstrumentationRegistry.getInstrumentation().targetContext.classLoader
|
||||
|
||||
class TestClassLoader : ClassLoader(targetClassLoader)
|
||||
Loading…
Add table
Add a link
Reference in a new issue