美文网首页
自定义 Gradle Plugin

自定义 Gradle Plugin

作者: yanlong107 | 来源:发表于2021-10-07 22:30 被阅读0次

    Plugin 的写法

    1. 写在 build.gradle

    build.gradle
    tips:单个项目使用, 进行一些简单任务, 不方便进行复用

    class PluginDemo implements Plugin<Project> {
    @Override
        void apply(Project target) {
            println 'Hello word!'
    } }
    apply plugin: PluginDemo
    
    
    or
    
     
    class ExtensionDemo {
        def name = 'dalong'
    }
    class PluginDemo implements Plugin<Project> {
    @Override
    void apply(Project target) {
            def extension = target.extensions.create('mplugin', ExtensionDemo)
            target.afterEvaluate {
                println "Hello ${extension.name}!"
            }
    } }
    apply plugin: PluginDemo
    mplugin {
        name 'dalong1111'
    }
    
    

    2. 写在 buildSrc ⽬录下

    tips:单个项目使用, 进行了代码分离,可以进行一定程度的复用

    • 目录结构:


    • *.properties
      resources 目录是固定写法,可以包含多个 *.properties文件。 其中 *plugin的名称。
      例如: mdemo.properties , mdemopluge的名称,使用时如下引用:

    apply plugin: 'mdemo'
    

    *.properties 内容如下:

    implementation-class=com.tencent.demo.PluginDemo
    

    com.tencent.demo.PluginDemo 是实现类Plugin接口的具体类

    • groovy目录

    PluginDemo.groovy

    class PluginDemo implements Plugin<Project> {
        @Override
        public void apply(Project project) {
            def extension = project.extensions.create('mplugin', ExtensionDemo)
            project.afterEvaluate {
                println "Hello ${extension.name}!"
            }
        }
    }
    

    ExtensionDemo.groovy

    class ExtensionDemo {
        def name = "dalong"
    }
    
    • 关于buildSrc

      1. 这是 gradle 的⼀个特殊目录,这个目录的 build.gradle 会自动被执⾏,不需再配置到settings.gradle
      2. buildSrc 的执⾏早于任何⼀个 project,也早于 settings.gradle
      3. buildSrc 中配置的plugin,会被添加到settings.gradle 中的所有子projectclasspath中, 因此所有的project 可以使用 apply plugin: '***' 来使用自定义的plugin
    • build.gradle
      官网参考:
      https://docs.gradle.org/current/userguide/organizing_gradle_projects.html#sec:build_sources

    buildSrc/build.gradle

    repositories {
        mavenCentral()
    }
    
    dependencies {
        testImplementation 'junit:junit:4.12'
    }
    
    or
    
    apply plugin: 'groovy'
    
    dependencies {
        implementation gradleApi() //gradle sdk
        implementation localGroovy() //groovy sdk
    }  
    
    

    3. 独立第三方组件

    和创建model工程一样创建一个plugin的model工程。

    • build.gradle 配置如下
    apply plugin: 'groovy'
    apply plugin: 'maven'
    
    allprojects {
        repositories {
            maven { url 'http://maven.oa.com/nexus/content/repositories/android' }
            maven { url 'http://maven.oa.com/nexus/content/repositories/thirdparty/' }
            mavenLocal()
        }
    }
    
    dependencies {
        implementation gradleApi()
        implementation localGroovy()
    }
    
    // plugin 发布到本地仓库
    group = 'mdemo'
    version="1.0.0"
    uploadArchives {
        repositories {
           mavenDeployer {
                repository(url: uri('../repo'))
            }
        }
    }
    
    
    • setting.gradle 中需要 include , 新创建的plugin的model工程
    • 工程根目录下build.gradle 配置
    project.version="1.0.0"
    
    buildscript {
    
        repositories {
            maven {
                url './repo'
            }
            maven { url "******" }
            mavenLocal()
        }
    
    
        dependencies {
            classpath 'com.android.tools.build:gradle:3.3.2'
            classpath 'mdemo:1.0.0'
    
            // NOTE: Do not place your application dependencies here; they belong
            // in the individual module build.gradle files
        }
    }
    
    allprojects {
        repositories {
    
            maven { url "******" }
            mavenLocal()
        }
    }
    

    关于Plugin 的一个Demo

    目标: 通过Plugin 的方式,配置应用程序的反调试检测

    关键实现:

    • 1、通过第三种方式添加plugin插件工程(plugin开发完后发布在本地,在app工程中引入)
    • 2、在plugin中,通过config配置控制行为,添加task的方式来

    添加自定义config配置( app model 中 build.gradle )

    ....
    ....
    
    apply plugin: 'anti-debug-plugin'
    
    MDebugInfo {
        debug {
            checkJavaDebuggableInterval 60
            checkJavaDebuggable true
        }
        release {
            checkJavaDebuggableInterval 60
            checkJavaDebuggable true
        }
    }
    
    dependencies {
       ...
       ...
    }
    

    groovy 中 plugin 的实现

    public class MDebugPlugin implements Plugin<Project> {
    
    
        @Override
        void apply(Project project) {
            project.extensions.create("MDebugInfo", MDebugInfoExtension);
            project.MDebugInfo.extensions.create("debug", DebugBuildConfig);
            project.MDebugInfo.extensions.create("release", ReleaseBuildConfig);
    
            // 创建 task 任务
            project.tasks.create(name: "genMDebugSoDebug", type: GenMDebugSoDebug, dependsOn: ["preBuild"]);
            project.tasks.create(name: "genMDebugSoRelease", type: GenMDebugSoRelease, dependsOn: ["preBuild"]);
    
            // 设置task执行时机
            project.tasks.whenTaskAdded { theTask ->
                if (theTask.name.equals("transformNativeLibsWithStripDebugSymbolForRelease")) {
                    theTask.dependsOn "genMDebugSoRelease"
                }
    
                if (theTask.name.equals("transformNativeLibsWithStripDebugSymbolForDebug")) {
                    theTask.dependsOn "genMDebugSoDebug"
                }
            }
        }
    }
    

    groovy 中 plugin 的实现
    GenMDebugSoDebug、GenMDebugSoRelease task 任务中通过NDK来编译so文件

    // 调用函数
    String jniDir = project.android.sourceSets.main.jniLibs.srcDirs[0];
    genMDebugSo(MConfig, "${project.buildDir}", "${project.rootProject.projectDir}", jniDir, true);
    
    task 实现函数
    public void genMDebugSo(BuildConfig buildConfig, String buildPath, String rootPath, String jniDir, boolean isRelease) {
            //1 创建jni临时目录
            String jniTmpPath = buildPath + "/jniTmp/jni";
            File jniTmpFile = new File(jniTmpPath);
            if (!jniTmpFile.exists()) {
                jniTmpFile.mkdirs();
            }
            //2 拷贝jni相关文件
            copyFile(jniTmpPath, "/jni", "Android.mk");
            copyFile(jniTmpPath, "/jni", "Application.mk");
            copyFile(jniTmpPath, "/jni", "MDebug.cpp");
            copyFile(jniTmpPath, "/jni", "JNIModel.h");
    
            String ndkPath = System.getenv("ANDROID_NDK_HOME");
            if (ndkPath == null) {
                Properties properties = new Properties();
                File localFile = new File(rootPath + "/" + "local.properties");
                properties.load(new InputStreamReader(new FileInputStream(localFile)));
                ndkPath = properties.getProperty("ndk.dir")
            }
            println "ndk.dir: " + ndkPath + "\n";
            println "jniDir: " + jniDir + "\n";
            File jniSoDir = new File(jniDir + File.separator + "armeabi");
            jniSoDir.mkdirs();
            if (!ndkPath.endsWith(File.separator)){
                ndkPath += File.separator;
            }
    
            int checkJavaDebuggerInterval = buildConfig.checkJavaDebuggableInterval;
            boolean checkJavaDebugger = buildConfig.checkJavaDebuggable;
    
            String argvs =  " .......“;
    
            print "argvs " + argvs + "\n";
            //编译so
            String cmdStr= ndkPath+"ndk-build " + argvs + " NDK_PROJECT_PATH=" + buildPath + "/jniTmp APP_BUILD_SCRIPT="+jniTmpPath+"/Android.mk";
            println cmdStr + "\n";
            Process process = Runtime.getRuntime().exec(cmdStr);
            BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream()));
            String line;
            while((line = reader.readLine())!= null){
                println "ndk-build info: " +line;
            }
            reader = new BufferedReader(new InputStreamReader(process.getErrorStream()));
            while((line = reader.readLine())!= null){
                println "ndk-build err: " + line;
            }
            process.waitFor();
    
            //5 拷贝so
            String outPath = jniSoDir.getAbsoluteFile();
            (new AntBuilder()).copy(file: buildPath + "/jniTmp/libs/armeabi/libad.so", tofile: outPath+"/libad.so");
            println "copy "+ buildPath + "/jniTmp/libs/armeabi/libad.so to "+outPath+"/libad.so";
        }
    
    
    • 3、通过 plugin的方式,添加对应的功能so会编译早app中的jni目录。 这样在app build 或者 run 的时候就可以使用这个功能了。

    End!

    相关文章

      网友评论

          本文标题:自定义 Gradle Plugin

          本文链接:https://www.haomeiwen.com/subject/fwjnxhtx.html