美文网首页
dart脚本实战 -- apk对齐、签名、注入渠道

dart脚本实战 -- apk对齐、签名、注入渠道

作者: 旺仔_100 | 来源:发表于2024-01-14 19:19 被阅读0次

    一、背景

    看了张风捷特烈大佬的《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传入;也可以通过命令行执行:参数直接放到命令行后面。

    相关文章

      网友评论

          本文标题:dart脚本实战 -- apk对齐、签名、注入渠道

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