Replugin源码阅读-replugin-host-library

1.replugin-host-gralde:   宿主脚本
2.replugin-host-library:  宿主库
3.replugin-plugin-gradle: 插件脚本
4.replugin-plugin-library:插件库

打算从以上四个模块对replugin进行拆解阅读

RePlugin的思想是Hook ClassLoader, 因此接下来就围绕ClassLoader进行分析

一、流程总结

1.1 插件的安装
1.2 内置插件
1.3 外置插件
1.4 replugin-host-gradle
1.5 replugin-host-library
1.6 replugin-plugin-gradle
1.7 replugin-plugin-library
1.1 插件的安装

  RePlugin插件的安装并不会真正的处理插件APK中的dex、so库、资源等, 只是将插件移动到需要的位置, 这个位置默认是宿主的context.getFilesDir(), 然后将插件信息包装成Plugin对象并绑定宿主的Context、宿主ClassLoader、负责和宿主通信的PluginContextImpl类, 最后将Plugin对象存入插件管理进程统一管理.

1.2 内置插件

  内置插件的安装是在初始化时就自动安装和加载了

1.3 外置插件

  外置插件的安装需要调用Replugin.install()方法来安装插件, 这个过程和内置插件类似, 区别就是内置插件是通过assets目录下的json文件来生成插件对象, 外置插件则是通过获取插件apk的PackageInfo来生成插件对象, 但是并不会处理apk中的dex、so库、资源等, 只有当真正使用这个插件中的类时才会去真正的解析加载这个插件

1.4 replugin-host-gradle

  主程序使用的Gradle插件, 主要职责是在我们的主程序打包的过程中(编译的过程中)动态的修改AndroidManifest.xml的信息, 动态的生成占位各种Activity、provider和service的声明.
  其次还会动态生成一个HostBuildConfig的java类, 这个类是根据app下的build.gradle中配置的参数信息(repluginHostConfig)产生的, 这个类的路径在BuildConfig的同级目录的gen包下.
  然后会扫描内置插件目录assets/plugins目录, 解析插件文件生成包含文件名、包名、版本、路径的plugins-build.json文件, 这个文件的路径在assets目录.

1.5 replugin-host-library

  这个库是由主程序依赖的, 也是RePlugin的核心, 它的主要职责是初始化Replugin的整体框架, 整体框架使用了Binder机制来实现多进程直接的沟通和数据共享, 或者说是插件之间和宿主之间沟通和数据共享, hook住ClassLoader, 加载插件、启动插件、多插件的管理全部都由这个库辅助

1.6 replugin-plugin-gradle

  这个是插件工程使用的gradle插件, 这个库使用了Transform API和Javassist实现了编译期间动态修改字节码文件, 主要是替换插件工程中的Activity的继承全部替换成Replugin库中定义的XXXActivity, 动态的将插件APK中调用LocalBroadcastManager的地方修改为Replugin中的PluginLocalBroadcastManager调用, 动态修改ContentResolver和ContentProvider的调用修改成Replugin调用, 动态的修改插件工程中所有调用Resource.getIdentifier方法的地方, 将第三个参数修改为插件工程的包名

1.7 replugin-plugin-library

  这个库是由插件工程依赖的, 这个库的主要目的是通过反射的方式来使主程序中接口和功能, 这个库在主程序加载插件apk后悔进行初始化.

二、RePluginClassLoader初始化

2.1 ClassLoader
2.2 RePluginApplication初始化
2.3 PatchClassLoaderUtils.patch
2.4 创建RePluginClassLoader
2.1 ClassLoader

Android系统默认三个ClassLoader

1. BootClassLoader: 系统启动时创建, 一般不需要用到. 加载系统类
2. PathClassLoader: 应用启动时创建, 只能加载内部dex
3. DexClassLoader: 可以加载外部的dex

Replugin提供了两个ClassLoader
RePluginClassLoader: 宿主APP中的ClassLoader, 加载应用内部的Activity
PluginDexClassLoader: 加载插件的Loader.

public class RePluginClassLoader extends PathClassLoader;
public class PluginDexClassLoader extends DexClassLoader;
2.2 RePluginApplication初始化
调用链:
RepluginApplication.attachBaseContext -> RepluginApplication.App.attachBaseContext
                                      -> PMF.init()
                                      -> PatchClassLoaderUtils.patch()                                   
2.3 PatchClassLoaderUtils.patch
public static boolean patch(Application application) {
    //1.oBase指向ContextImpl: 具体结合LoadedApk.makeApplication()
    Context oBase = application.getBaseContext();
    // 2.反射获取ComTextImpl.mPackageInfo变量 -> LoadedApk
    Object oPackageInfo = ReflectUtils.readField(oBase, "mPackageInfo");
    // 3.反射获取LoadedApk.mClassLoader -> ClassLoader = PathClassLoader
    ClassLoader oClassLoader = (ClassLoader) ReflectUtils.readField(oPackageInfo, "mClassLoader");
    // 外界可自定义ClassLoader的实现,但一定要基于RePluginClassLoader类
    // 4.创建RePluginClassLoader
    ClassLoader cl = RePlugin.getConfig().getCallbacks().createClassLoader(oClassLoader.getParent(), oClassLoader);
    // 将新的ClassLoader写入mPackageInfo.mClassLoader
    // 5.将新创建的RePluginClassLoader替换掉LoadedApk中的PathClassLoader
    ReflectUtils.writeField(oPackageInfo, "mClassLoader", cl);
    // 设置线程上下文中的ClassLoader为RePluginClassLoader
    // 防止在个别Java库用到了Thread.currentThread().getContextClassLoader()时,“用了原来的PathClassLoader”,或为空指针
    Thread.currentThread().setContextClassLoader(cl);
    return true;
}
2.4 创建RePluginClassLoader
RePlugin.getConfig().getCallbacks().createClassLoader:
/**
 * @param parent   该ClassLoader的父亲,通常为BootClassLoader
 * @param original 宿主的原ClassLoader,通常为PathClassLoader
 */
public RePluginClassLoader(ClassLoader parent, ClassLoader orig) {
    // 由于PathClassLoader在初始化时会做一些Dir的处理,所以这里必须要传一些内容进来
    // 但我们最终不用它,而是拷贝所有的Fields
    super("", "", parent);
    mOrig = orig;
    // 将原来宿主里的关键字段,拷贝到这个对象上,这样骗系统以为用的还是以前的东西(尤其是DexPathList)
    // 注意,这里用的是“浅拷贝”
    copyFromOriginal(orig);
    initMethods(orig);
}

三、加载插件

3.1 Replugin.startActivity
3.2 PluginLibraryInternalProxy.startActivity
3.1 Replugin.startActivity
// 调用链:
Replugin.startActivity() -> Factory.startActivityWithNoInjectCN()
                         -> PluginCommImpl.startActivity()
                         -> PluginLibraryInternalProxy.startActivity()
                                     
public static boolean startActivity(Context context, Intent intent) {
    // TODO 先用旧的开启Activity方案,以后再优化
    ComponentName cn = intent.getComponent();
    if (cn == null) {
        // TODO 需要支持Action方案
        return false;
    }
    String plugin = cn.getPackageName();
    String cls = cn.getClassName();
    return Factory.startActivityWithNoInjectCN(context, intent, plugin, cls, IPluginManager.PROCESS_AUTO);
}
3.2 PluginLibraryInternalProxy.startActivity
/**
 * 启动一个插件中的activity, 如果插件不存在会触发下载界面
 * @param context 应用上下文或Activity上下文
 * @param intent 
 * @param plugin 插件名
 * @param activity 待启动的activity类名
 * @param process 是否在指定进程中启动
 * @param download 下载
 * @return 插件机制是否成功, 例如没有插件存在, 没有合适的Activity坑
 */
public boolean startActivity(Context context, Intent intent, String plugin, String activity, int process, boolean download) {
    // 是否启动下载
    // 若插件不可用(不存在或版本不匹配),则直接弹出“下载插件”对话框
    // 因为已经打开UpdateActivity,故在这里返回True,告诉外界已经打开,无需处理
    if (download) {
        if (PluginTable.getPluginInfo(plugin) == null) {
            // 如果用户在下载即将完成时突然点按“取消”,则有可能出现插件已下载成功,但没有及时加载进来的情况
            // 因此我们会判断这种情况,如果是,则重新加载一次即可,反之则提示用户下载
            // 原因:“取消”会触发Task.release方法,最终调用mDownloadTask.destroy,导致“下载服务”的Receiver被注销,即使文件下载了也没有回调回来
            // NOTE isNeedToDownload方法会调用pluginDownloaded再次尝试加载
            if (isNeedToDownload(context, plugin)) {
                return RePlugin.getConfig().getCallbacks().onPluginNotExistsForActivity(context, plugin, intent, process);
            }
        }
    }
    /* 检查是否是动态注册的类 */
    // 如果要启动的 Activity 是动态注册的类,则不使用坑位机制,而是直接动态类。
    // 原因:宿主的某些动态注册的类不能运行在坑位中(如'桌面'插件的入口Activity)
    if (Factory2.isDynamicClass(plugin, activity)) {
        intent.putExtra(IPluginManager.KEY_COMPATIBLE, true);
        intent.setComponent(new ComponentName(IPC.getPackageName(), activity));
        context.startActivity(intent);
        return true;
    }
    // 如果插件状态出现问题,则每次弹此插件的Activity都应提示无法使用,或提示升级(如有新版)
    if (PluginStatusController.getStatus(plugin) < PluginStatusController.STATUS_OK) {
        return RePlugin.getConfig().getCallbacks().onPluginNotExistsForActivity(context, plugin, intent, process);
    }
    // 若为首次加载插件,且是“大插件”,则应异步加载,同时弹窗提示“加载中”
        // Added by Jiongxuan Zhang
        if (!RePlugin.isPluginDexExtracted(plugin)) {
            PluginDesc pd = PluginDesc.get(plugin);
            if (pd != null && pd.isLarge()) {
                if (LOG) {
                    LogDebug.d(PLUGIN_TAG, "PM.startActivity(): Large Plugin! p=" + plugin);
                }
                return RePlugin.getConfig().getCallbacks().onLoadLargePluginForActivity(context, plugin, intent, process);
            }
        }

        // WARNING:千万不要修改intent内容,尤其不要修改其ComponentName
        // 因为一旦分配坑位有误(或压根不是插件Activity),则外界还需要原封不动的startActivity到系统中
        // 可防止出现“本来要打开宿主,结果被改成插件”,进而无法打开宿主Activity的问题

        // 缓存打开前的Intent对象,里面将包括Action等内容
        Intent from = new Intent(intent);

        // 帮助填写打开前的Intent的ComponentName信息(如有。没有的情况如直接通过Action打开等)
        if (!TextUtils.isEmpty(plugin) && !TextUtils.isEmpty(activity)) {
            from.setComponent(new ComponentName(plugin, activity));
        }

        ComponentName cn = mPluginMgr.mLocal.loadPluginActivity(intent, plugin, activity, process);
        if (cn == null) {
            if (LOG) {
                LogDebug.d(PLUGIN_TAG, "plugin cn not found: intent=" + intent + " plugin=" + plugin + " activity=" + activity + " process=" + process);
            }
            return false;
        }

        // 将Intent指向到“坑位”。这样:
        // from:插件原Intent
        // to:坑位Intent
        intent.setComponent(cn);

        if (LOG) {
            LogDebug.d(PLUGIN_TAG, "start activity: real intent=" + intent);
        }

//        if (RePluginInternal.FOR_DEV) {
//            try {
//                String str = cn.getPackageName() + "/" + cn.getClassName();
//                if (LOG) {
//                    LogDebug.d(PLUGIN_TAG, "str=" + str);
//                }
//                new ProcessBuilder().command("am", "start", "-D", "--user", "0", "-n", str).start();
//            } catch (IOException e) {
//                e.printStackTrace();
//            }
//        } else {

        context.startActivity(intent);

        // 通知外界,已准备好要打开Activity了
        // 其中:from为要打开的插件的Intent,to为坑位Intent
        RePlugin.getConfig().getEventCallbacks().onPrepareStartPitActivity(context, from, intent);

        return true;
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351