aboutsummaryrefslogtreecommitdiff
path: root/app/src/main/java/moe/yuuta/workmode/access/AccessLayer.kt
blob: 78ef5dd286aae9f69533ff5d1c9d3e0eeff7fc43 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
package moe.yuuta.workmode.access

import android.content.Context
import android.content.pm.LauncherApps
import android.content.pm.PackageManager
import android.os.Bundle
import android.os.PersistableBundle
import android.os.Process
import android.os.UserHandle
import android.system.Os
import moe.yuuta.workmode.BuildConfig
import java.lang.reflect.Field
import java.lang.reflect.Method

/**
 * An layer to access package suspending related APIs, it is a low-level layer which is used to call System APIs directly.
 *
 * TODO: Multi-user support
 */
internal class AccessLayer(private val mContext: Context) {
    private val mPM: PackageManager = mContext.packageManager

    fun setPackagesSuspended(packageNames: Array<String>, suspended: Boolean,
                             appExtras: PersistableBundle, launcherExtras: PersistableBundle,
                             dialogMessage: String): Array<String> {
        // ApplicationPackageManager ALWAYS uses context.getOpPackageName() as the argument "callingPackage"
        // My callingPackage MUSTN'T equals to 'android'
        // If we are using packageName of 'android', system will show disabled
        // by admin dialog instead of suspended dialog
        // F**k Google
        val func: Method = Class.forName("android.content.pm.IPackageManager")
                .getDeclaredMethod("setPackagesSuspendedAsUser",
                        Array<String>::class.java,
                        Boolean::class.java,
                        PersistableBundle::class.java,
                        PersistableBundle::class.java,
                        String::class.java,
                        String::class.java,
                        Int::class.java)

        // It's an unstable design
        val iPM: Field = mPM::class.java.getDeclaredField("mPM")
        iPM.isAccessible = true

        return func.invoke(iPM.get(mPM),
                packageNames,
                suspended,
                appExtras,
                launcherExtras,
                dialogMessage,
                BuildConfig.APPLICATION_ID,
                UserHandle.getUserHandleForUid(mPM.getPackageUid(mContext.packageName, 0)).hashCode()) as Array<String>
    }

    /**
     * This method will SET your UID and you WON'T BE ABLE TO GO BACK.
     * Create a new process and access it.
     */
    fun getSuspendedPackageAppExtras(packageName: String): PersistableBundle? {
        Os.setuid(mPM.getPackageUid(packageName, 0))
        // ApplicationPackageManager ALWAYS uses context.getOpPackageName() as the package name
        // F**k Google
        val func: Method = Class.forName("android.content.pm.IPackageManager")
                .getDeclaredMethod("getSuspendedPackageAppExtras",
                        String::class.java,
                        Int::class.java)

        // It's an unstable design
        val iPM: Field = mPM::class.java.getDeclaredField("mPM")
        iPM.isAccessible = true

        return func.invoke(iPM.get(mPM),
                packageName,
                UserHandle.getUserHandleForUid(mPM.getPackageUid(packageName, 0)).hashCode()) as PersistableBundle?
    }

    @Throws(PackageManager.NameNotFoundException::class)
    fun isPackageSuspended(packageName: String): Boolean {
        val func: Method = PackageManager::class.java.getDeclaredMethod("isPackageSuspended",
                String::class.java)
        return func.invoke(mPM, packageName) as Boolean
    }

    fun getSuspendedPackageLauncherExtras(packageName: String): Bundle? =
            mContext.getSystemService(LauncherApps::class.java).getSuspendedPackageLauncherExtras(packageName, Process.myUserHandle())
}