前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >自定义 Gradle Plugin

自定义 Gradle Plugin

作者头像
艳龙
发布2021-12-16 18:04:01
3400
发布2021-12-16 18:04:01
举报
文章被收录于专栏:yanlongli_艳龙yanlongli_艳龙

Plugin 的写法

1. 写在 build.gradle

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

代码语言:javascript
复制
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的名称,使用时如下引用:
代码语言:javascript
复制
apply plugin: 'mdemo'

*.properties 内容如下:

代码语言:javascript
复制
implementation-class=com.tencent.demo.PluginDemo

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

  • groovy目录

PluginDemo.groovy

代码语言:javascript
复制
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

代码语言:javascript
复制
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

代码语言:javascript
复制
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 配置如下
代码语言:javascript
复制
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 配置
代码语言:javascript
复制
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 )

代码语言:javascript
复制
....
....

apply plugin: 'anti-debug-plugin'

MDebugInfo {
    debug {
        checkJavaDebuggableInterval 60
        checkJavaDebuggable true
    }
    release {
        checkJavaDebuggableInterval 60
        checkJavaDebuggable true
    }
}

dependencies {
   ...
   ...
}

groovy 中 plugin 的实现

代码语言:javascript
复制
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文件

代码语言:javascript
复制
// 调用函数
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!

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2021/10/7 下,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Plugin 的写法
    • 1. 写在 build.gradle 中
      • 2. 写在 buildSrc ⽬录下
        • 3. 独立第三方组件
        • 关于Plugin 的一个Demo
          • 关键实现:
          领券
          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档