dart脚本实战 -- apk对齐、签名、注入渠道
2024-01-14 本文已影响0人
旺仔_100
一、背景
看了张风捷特烈大佬的《Dart 开发命令行工具》
觉得很有意思,然后想到我们项目中的apk,需要通过腾讯进行加固、4字节对齐、签名、注入渠道等操作。
虽然只有三大步操作,但是每个操作都需要生产apk,然后去对应的文件中查找,加固命令复杂,容易出错,还要执行脚本,复制备份,以及不需要自己找文件夹,直接在程序中打开文件夹等。
于是就写了下面的脚本,解放双手并且不会出错。
完整代码如下:
import 'dart:io';
Future<void> main(List<String> arguments) async {
// 调用 zipalign 命令
if (arguments.isEmpty || arguments.length != 2) {
print('请传入apk完整路径和版本');
return;
}
String path = arguments[0];
String version = arguments[1];
///切割,拼接路径
String path4kb = generate4Kb(path);
await runZipAlign(path, path4kb);
String directoryPath = '/Users/xxx/Documents/packer-tool-v2.2/parent';
String directoryOutPath = '/Users/xxx/Documents/packer-tool-v2.2/output';
// 删除其内容
await deleteDirectory(Directory(directoryPath));
await deleteDirectory(Directory(directoryOutPath));
print('Directory deleted successfully.');
await runApkSigner(
path4kb,
'/Users/x x x/Documents/packer-tool-v2.2/parent/app-release-$version.apk');
// 进入目录
Directory.current = '/Users/xxx/Documents/packer-tool-v2.2';
// 执行 shell 脚本
await runShellScript('sh', ['packer.sh']);
// String sourcePath = '/Users/xxx/Documents/packer-tool-v2.2/parent';
String destinationPath = '/Users/xxx/Desktop/keep_release';
// 复制目录内容
await copyDirectory(Directory(directoryOutPath), Directory(destinationPath));
print('Directory copied successfully.');
openDirectory(directoryOutPath);
}
Future<void> runZipAlign(String inputApk, String outputApk) async {
// 请确保 zipalign 工具在系统路径中可用
ProcessResult result =
await Process.run('zipalign', ['-v', '4', inputApk, outputApk]);
// 处理标准输出
print('stdout: ${result.stdout}');
// 处理标准错误输出
print('stderr: ${result.stderr}');
// 输出命令的退出码
print('Exit code: ${result.exitCode}');
}
Future<void> runApkSigner(String inputApk, String outputApk) async {
ProcessResult result = await Process.run(
'apksigner',
[
'sign',
'--verbose',
'--v1-signing-enabled',
'true',
'--v2-signing-enabled',
'true',
'--v3-signing-enabled',
'false',
'--ks',
'/Users/xxx/Desktop/writerassistant/writer_assistant/android/xx',
'--ks-key-alias',
'xxx',
'--ks-pass',
'pass:xxx',
'--key-pass',
'pass:xxx',
'--out',
outputApk,
inputApk,
],
);
print('apksigner stdout: ${result.stdout}');
print('apksigner stderr: ${result.stderr}');
print('apksigner exit code: ${result.exitCode}');
}
Future<void> runShellScript(String command, List<String> arguments) async {
ProcessResult result = await Process.run(command, arguments);
// 处理标准输出
print('stdout: ${result.stdout}');
// 处理标准错误输出
print('stderr: ${result.stderr}');
// 输出命令的退出码
print('Exit code: ${result.exitCode}');
}
String generate4Kb(String path) {
if (path.endsWith(".apk")) {
var paths = path.split(".apk");
return '${paths[0]}_4kb.apk';
} else {
throw Exception("路径不对,需要 .apk结尾");
}
}
Future<void> deleteDirectory(Directory directory) async {
if (await directory.exists()) {
await for (var entity in directory.list(recursive: true)) {
if (entity is File) {
await entity.delete();
} else if (entity is Directory) {
await entity.delete(recursive: true);
}
}
// await directory.delete();
}
}
Future<void> copyDirectory(Directory source, Directory destination) async {
if (await source.exists()) {
await destination.create(recursive: true);
await for (var entity in source.list(recursive: true)) {
if (entity is File) {
String newPath = destination.path + entity.path.substring(source.path.length);
await entity.copy(newPath);
} else if (entity is Directory) {
String newPath = destination.path + entity.path.substring(source.path.length);
await Directory(newPath).create(recursive: true);
}
}
}
}
Future<void> openDirectory(String folderPath) async{
// 创建一个 ProcessResult 对象
ProcessResult result = await Process.run('open', [folderPath]);
// 检查命令是否成功执行
if (result.exitCode == 0) {
print('Folder opened successfully: $folderPath');
} else {
print('Failed to open folder: $folderPath');
print('Error: ${result.stderr}');
}
}
这个代码的main函数是需要两个参数的,一个apk的完整路径,一个是产物apk的版本号。
参数可以从运行设置中的Program arguments传入;也可以通过命令行执行:参数直接放到命令行后面。