以下是使用梆梆收费后的命令包,同时目前梆梆为了安全,并不支持放线上签名,所以我们要自己多写一个签名的命令,那么我们直接开干:
首先确定相关配置,新建global.gradle
// 用来存放应用中的所有配置变量,统一管理,而不再是每个moudle里都自己写一份,修改起来更加的方便
ext {
// Android API 信息
android = [compileSdkVersion: 31, // 配置编译SDK版本
buildToolsVersion: '31.0.0', // 配置编辑工具版本
minSdkVersion : 21, // 兼容的最低SDK版本
targetSdkVersion : 31, // 用于提高指定版本的设备上程序运行体验
]
// 依赖包管理
dependencies = ['androidx_appcompat': 'androidx.appcompat:appcompat:1.3.0',
'constraintlayout' : 'androidx.constraintlayout:constraintlayout:2.0.4',
'annotation' : 'androidx.annotation:annotation:1.0.0',
'material' : 'com.google.android.material:material:1.2.1']
// 梆梆加固配置
jiagu = [APIKey : 'XXX',
APISecretKey : 'XXX',
zipPath : "jiagu/secapi-4.1.5-SNAPSHOT.jar",
apksigner : "jiagu/33.02/apksigner.jar",
keystoreFile : "xxxx.keystore"
]
}
创建加固命令,新建jiagu.gradle
import org.apache.tools.ant.taskdefs.condition.Os
// 加固后所有apk的保存路径
def PROD_OUTPUT_PATH = "${projectDir.absolutePath}/jiagu/apk/"
// 加固前的api文件夹路径
def PROD_APK_PATH = "${projectDir.absolutePath}/build/outputs/apk/prod/release/"
/**
* 加固
* @param config 配置加固可选项
* @param apkPath 要加固的文件路径
* @param outputPath 输出路径
* @param automulpkg 是否自动生成多渠道包
*/
def jiaGu(String config, String apkPath, String outputPath, boolean automulpkg) {
println "File path before hardening:${apkPath}"
println "Hardened file path:${outputPath}"
// 直接梆梆命令加固,得出未签名的加固apk
exec {
executable = 'java'
args = ['-jar', rootProject.ext.jiagu["zipPath"], '-i', '梆梆网站',
'-u', '梆梆账号', '--password', '梆梆密码',
'-a', 'APIKey', '-c', 'APISecretKey',
'-f', '0', '-t', '策略ID',
'-p', apkPath, '-d', outputPath,
'-m', '1', '--action', 'ud']
}
// 获取加固后的apk文件路径
File apkFile = null
File apkFolder = file(outputPath)
// 获取该文件夹下的所有文件
File[] apkFiles = apkFolder.listFiles()
int i = 0
for (File file : apkFiles) {
if (file.isFile() && file.getName().endsWith(".apk")) {
// 获取后缀名为.apk的文件
apkFile = file
i++
}
}
if (i > 1) {
println("There are multiple folders under this folder apk:" + apkFile.absolutePath)
return
}
if (apkFile == null || !apkFile.exists()) {
println("Can't find the apk:" + apkFile.absolutePath)
return
}
// 获取文件名
String newFile = ""
int position = apkFile.absolutePath.lastIndexOf(".")
if (position > 0) {
newFile = apkFile.absolutePath.substring(0, position)
}
println ("apksigner:" + rootProject.ext.jiagu["apksigner"])
println ("keystoreFile:" + rootProject.ext.jiagu["keystoreFile"])
println ("_sign.apk:" + newFile + '_sign.apk')
println ("absolutePath:" + apkFile.absolutePath)
exec {
executable = 'java'
args = ['-jar', rootProject.ext.jiagu["apksigner"],
'sign', '--verbose',
'--v1-signing-enabled', 'true', '--v2-signing-enabled', 'true',
'--ks', rootProject.ext.jiagu["keystoreFile"],
'--ks-key-alias', '别名', '--ks-pass', 'pass:密码',
'--out',
newFile + '_sign.apk',
apkFile.absolutePath]
}
}
/**
* 检查file并且进行删除
*/
private static void checkOutputDir(File apkOutputFile) {
if (apkOutputFile.exists()) {
File[] files = apkOutputFile.listFiles()
if (files != null) {
for (File file : files) {
file.delete()
}
}
} else {
apkOutputFile.mkdirs()
}
}
/**
* @return 当前时间,用于给加固后的包名
*/
static def getCurTime() {
return new Date().format("yyyy-MM-dd-HH-mm-ss")
}
/**
* prod加固
* 执行命令:./gradlew releaseJiaGuProd
*/
task releaseJiaGuProd(dependsOn: 'assembleProdRelease') {
group = "publish"
doLast {
File apkOutputFile = new File(PROD_OUTPUT_PATH, getCurTime())
// 检查file并且进行删除
checkOutputDir(apkOutputFile)
File apkFile = null
// 加固前的文件夹
File apkFolder = file(PROD_APK_PATH)
// 获取该文件夹下的所有文件
File[] apkFiles = apkFolder.listFiles()
int i = 0
for (File file : apkFiles) {
if (file.isFile() && file.getName().endsWith(".apk")) {
// 获取后缀名为.apk的文件
apkFile = file
i++
}
}
if (i > 1) {
println("There are multiple folders under this folder apk:" + apkFile.absolutePath)
return
}
if (apkFile == null || !apkFile.exists()) {
println("Can't find the apk:" + apkFile.absolutePath)
return
}
// 进行加固
jiaGu("-", apkFile.absolutePath, apkOutputFile.absolutePath, true)
}
}
最后记得在app的bulid依赖进gradle文件
apply from: '../jiagu.gradle'
完工