首页
社区
课程
招聘
[原创]APK中获取鸿蒙应用Ability信息
2021-12-24 15:52 12436

[原创]APK中获取鸿蒙应用Ability信息

2021-12-24 15:52
12436

搬过来一篇我在2021-06-29号写的文章,这篇文章提到的方法可以在APK中获取到鸿蒙的几个“核心”类,如果想使用源码,大家可以直接撸到文章最后,我把这部分代码贡献给了开源项目LibChecker,大家复制即可。


我知道文章几乎就没啥排版,大家将就一下吧,感谢~~~



Android中,我们获取应用列表之后,再获取应用的PackageInfo实例即可获得四大组件列表,在鸿蒙中,我们获取到通过IBundleManager获取到Ability列表即可。IBundleManager类似Android中的PackageManager,在Android中,我们通过Context即可获取PackageManager,在鸿蒙中,通过ohos.app.Context也同样可以得到IBundleManager。

鸿蒙中的ohos.app.Context从何处来,根据我的Android的开发经验,我首选Application。

鸿蒙系统中也有一个Application,此Application并非Android中的android.app.Application,而是ohos.app.Application


整理一下步骤,我将尝试依次获取如下类型的实例:

  1. ohos.app.Application

  2. ohos.app.Context

  3. ohos.bundle.IBundleManager


我们知道在鸿蒙中有ohos.abilityshell.HarmonyApplication,它是继承android.app.Application,所以它只是一个Android中的普通的Application。HarmonyApplication源码是公开的,我在里面发现了这个:

private ohos.app.Application application = new ohos.app.Application();



但是,在Android系统中是没有这个东西的,所以我们无法直接的在Android系统get一个ohos.app.Application对象。不过,我们可以尝试构造一个ohos.app.Application对象。这里要注意的是,我们也没有办法直接引用这个类,因为鸿蒙并没有公开相应的代码,所以只有反射咯。


实例化Application很简单,就是调用一个无参的构造函数即可

val applicationContext = context.applicationContext
val classLoader = applicationContext.classLoader
val clazzOhosApplication = Class.forName("ohos.app.Application")
sOhosApplication = clazzOhosApplication.newInstance()


此时我们可以通过sOhosApplication利用反射拿到ohos.app.Context

可惜,事情到这里并没结束,我们通过ohos.app.Context获取IBundleManager依然返回null。我猜测是初始化ohos.app.Application的时候出了问题

接下来继续在HarmonyApplication里面找线索,由于里面用了很多闭源的类,所以在阅读的时候造成了很大障碍,最终,我在下面的函数中找到了线索

private void attachHapModuleContext(AbilityContext harmonyAbilityPackage, HapModuleInfo hapModuleInfo) {
    if (harmonyAbilityPackage != null) {
        ContextDeal deal = new ContextDeal(getApplicationContext(), getClassLoader());
        ...
        deal.setApplication(this.application);
        ...
    }
    harmonyAbilityPackage.attachBaseContext(deal);
}


为了方便阅读,我把无关的代码删除了。上面if中的逻辑和初始化ohos.app.Application是相关的,所以利用反射复制一遍上述代码的逻辑

private fun initApplication(context: Context) {
    val applicationContext = context.applicationContext
    val classLoader = applicationContext.classLoader
    try {
        val clazzOhosApplication = Class.forName("ohos.app.Application")
        sOhosApplication = clazzOhosApplication.newInstance()
        val clazzContextDeal = Class.forName("ohos.app.ContextDeal")
        val contextDealConstructor = clazzContextDeal.getConstructor(
            Context::class.java, ClassLoader::class.java
        )
        val contextDeal = contextDealConstructor.newInstance(applicationContext, classLoader)
        val setApplicationMethod =
            clazzContextDeal.getDeclaredMethod("setApplication", clazzOhosApplication)
        setApplicationMethod.invoke(contextDeal, sOhosApplication)
    } catch (e: Throwable) {
        Log.w(TAG, e)
    }
}


然后再次运行,遗憾的是,此时我们通过ohos.app.Context获取IBundleManager依然返回null。

在attachHapModuleContext方法中的最后执行了AbilityContext.attachBaseContext(ContextDeal),可惜AbilityContext也没有开源,我们无法知道AbilityContext.attachBaseContext(ContextDeal)里面做了什么。

不过,我发现ohos.app.Application是继承AbilityContext的(反射打印一下便知),将上面的代码修改一下,调用一下attachBaseContext

private fun initApplication(context: Context) {
    val applicationContext = context.applicationContext
    val classLoader = applicationContext.classLoader
    try {
        val clazzOhosApplication = Class.forName("ohos.app.Application")
        sOhosApplication = clazzOhosApplication.newInstance()
        LogUtil.logParents(clazzOhosApplication)
        LogUtil.logClassMethods(clazzOhosApplication)
        val clazzContextDeal = Class.forName("ohos.app.ContextDeal")
        val contextDealConstructor = clazzContextDeal.getConstructor(
            Context::class.java, ClassLoader::class.java
        )
        val contextDeal = contextDealConstructor.newInstance(applicationContext, classLoader)
        val setApplicationMethod =
            clazzContextDeal.getDeclaredMethod("setApplication", clazzOhosApplication)
        setApplicationMethod.invoke(contextDeal, sOhosApplication)
        val attachBaseContextMethod =
            clazzOhosApplication.getMethod("attachBaseContext", ohos.app.Context::class.java)
        attachBaseContextMethod.invoke(sOhosApplication, contextDeal)
    } catch (e: Throwable) {
        Log.w(TAG, e)
    }
}


再次尝试获取IBundleManager,这次终于成功了,我们有了IBundleManager实例。

在Android中,我们获取四大组件信息就要先获得PackageInfo,然后再获取组件信息

packageInfo.activities
packageInfo.receivers
packageInfo.services
packageInfo.providers


在鸿蒙中,我们获取BundleInfo,从BundleInfo中可直接获得AbilityInfo,但是要注意的是,只有鸿蒙应用才有bundleInfo,鸿蒙系统安装apk文件是没有bundleInfo的,返回结果为null

fun getBundleInfo(bundleName: String, flags: Int): BundleInfo? {
    try {
        return mIBundleManager!!.getBundleInfo(bundleName, flags)
    } catch (e: RemoteException) {
        Log.w("IBundleManagerDelegate", "bundleName: $bundleName", e)
    }
    return null
}


为了方便大家测试,这里告诉大家华为远程模拟器P40中

  1. com.huawei.email是鸿蒙应用

  2. com.sina.weibo是Android应用


然后,程序又挂了...我们还需要一个权限:

<uses-permission android:name="ohos.permission.GET_BUNDLE_INFO" />

接下来再获取BundleInfo就可正常获取了

ohosContext!!.bundleManager!!.getBundleInfo(bundleName, flags)


想要查看鸿蒙系统中哪些是鸿蒙应用,可以使用Android开发工具箱的应用统计功能(PRO),其中一项为应用类型统计。


鸿蒙应用中所有组件信息都为AbilityInfo类型,通过type(AbilityInfo.AbilityType)来区分组件类别。至此,我们拿到了AbilityInfo实例,也就是拿到了组件所有信息了。


获取系统信息就简单了,下面是获取版本信息

SystemVersion.getVersion()

SystemVersion还可以获取其他信息,这里就不再列举了


最后,我将此部分代码贡献给了开源应用LibChecker

https://github.com/zhaobozhen/LibChecker


LibChecker是一个查看并分析App使用的第三方库的应用。提供了一些基本功能:App的ABI架构查看和统计、原生库的查看、四大组件查看,除此之外还对知名库进行标记、统计等。也欢迎大家试用



原文链接

https://mp.weixin.qq.com/s/_uLvujT9OouBEDNCMQeKcQ


[培训]内核驱动高级班,冲击BAT一流互联网大厂工作,每周日13:00-18:00直播授课

最后于 2021-12-24 16:04 被suandroid编辑 ,原因:
收藏
点赞4
打赏
分享
最新回复 (0)
游客
登录 | 注册 方可回帖
返回