美文网首页
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传入;也可以通过命令行执行:参数直接放到命令行后面。

相关文章

  • 动态库注入

    一、前言二、注入思路三、动态库注入实现四、分析实现按钮监听五、实战修改微信步数 一、前言 在文章《应用签名-脚本签...

  • Shell脚本查看apk签名信息

    Shell脚本查看apk签名信息 用shell写了一个查看apk签名的脚本。代码很少也很简单 支持递归目录查询 使...

  • 2016/12/13 周二

    GET 修改成语达人签名为readboy.key 使用脚本查看apk签名信息 修改apk版本号为正确的,重新编译运...

  • 校验apk渠道、签名

    突然接到要求:根据现有apk获取apk里面的渠道信息、版本信息、签名信息 虽然不知道为什么这么麻烦,按道理,这些信...

  • 多渠道打包

    # android多渠道打包 # ## 1.如何查看apk的签名信息## 1将apk解压; 2.找到META-IN...

  • apk签名bash脚本

    一般apk在打包之后还会进行加固、重新签名,公司用的乐固不支持对每个项目自动化签名,但一个个签名实在麻烦,所以写个...

  • Apk重新签名的工具

    该工具可以完成以下几个工作 删除老的签名 对apk进行重新签名 对齐操作 缺点 只支持mac和linux,不支持w...

  • android 多渠道打包

    android多渠道打包 1.如何查看apk的签名信息 1将apk解压; 2.找到META-INF 下的.RSA文...

  • Python 脚本构建Android APK 自动加固、打渠道包

    Python 脚本构建Android APK 自动加固、打渠道包并上传服务器 常规流程 打出原始apk 使用乐固工...

  • 美团多渠道快速打包

    说明 首先使用shell脚本打出源apk,根据channel中声明的渠道信息复制相应apk文件,调用美团写入方法写...

网友评论

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

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