从APKiD看APK壳检测
枫糖甜酒 发表于 北京 移动安全 3969浏览 · 2023-11-02 10:14

之前朋友检测APK壳是用的APKiD - https://github.com/rednaga/APKiD ,看了一下项目简介发现参加过两次BlackHat orz

一个是工具介绍,一个是快速识别RASP SDK,不过感觉中文网络上对于这个工具还停留在使用阶段,简单看一下代码实现,本篇文章使用的是 APKiD 2.1.5

简介

APKiD gives you information about how an APK was made. It identifies many compilers, packers, obfuscators, and other weird stuff. It's PEiD for Android.

APKiD是一个获取APK信息(编译器、壳、混淆...)的工具,项目地址 https://github.com/rednaga/APKiD
PS:这里提到的PEiD是一个查壳工具,所以APKiD即APK查壳的工具

环境搭建

pip install apkid

这种方式的安装可以直接通过apkid命令来对目标进行探测

可以看到这里使用了腾讯的加壳保护

为了方便断点调试和看数据,我这里以本地源代码的方式进行运行

git clone https://github.com/rednaga/APKiD
cd APKiD
./prep-release.py

运行 ./prep-release.py 之后会在本地生成 rules.yarc 规则文件
编写一个python加载代码调用APKiD

from apkid.apkid import Options, Scanner

def loader_apkid(apk_path):
    options = Options(json=True)
    rules = options.rules_manager.load()

    scanner = Scanner(rules, options)
    scanner.scan(apk_path)


if __name__ == '__main__':
    loader_apkid("./tmp/litiaotiao.apk")

指定APK,运行该脚本就可以得到结果了

一些前置知识

APK文件结构

APK其实就是一个ZIP文件,APK文件的基本结构如下

├── META-INF/
   ├── MANIFEST.MF 包含了存储在APK中的数字签名信息和其他与应用程序完整性验证相关的文件
   └── CERT.SF 
   └── CERT.RSA | DSA 
├── res/ 存放应用程序的资源文件
├── assets/ 存放应用程序需要在运行时读取的原始资源文件
├── lib/ 一些库文件,*.so,也可能会按照不同CPU架构(如armeabi-v7ax86arm64-v8a等)存放
├── AndroidManifest.xml APK的清单文件
├── classes.dex 可执行的DEX字节码文件,其中包含应用程序的Java类和方法
├── resources.arsc Android资源表文件,包含了应用程序使用的所有资源的编译后的二进制表示
└── other files...

APKID作者们在blackhat 2018上的PPT

我们解压一个APK也可以看到相关文件

Yara文件

Yara简介

Yara是一个旨在(但不限于)帮助恶意软件研究人员识别和分类恶意软件样本的开源工具

除了分析恶意软件样本之外,其实对于某些特征值的识别也可以用Yara来做,比如这里的APKiD就用到了Yara,你说这个是恶意文件吗,其实不然,里面更多的是特征值

Yara项目地址 https://github.com/VirusTotal/yara (是不是觉得VirusTotal似曾相识,没错就是https://www.virustotal.com/ 团队)
Yara的每一条描述、规则都由一系列字符串和一个布尔型表达式构成,并阐述其逻辑。

Yara 规则demo

打开apkid 的 rules 文件夹,里面的都是yara规则

我们查看这一条规则,这里的描述是腾讯乐固的壳,不过感觉特征跟腾讯御安全相似

rule tencent_legu : packer
{
  meta:
    description = "Tencent's Legu"
    url         = "https://blog.quarkslab.com/a-glimpse-into-tencents-legu-packer.html"
    sample      = "9ff3a53f76c7a6d7e3de3b8567c9606f2cc08ec4aaaae596a27361018d839c58"
    author      = "Mert Arıkan"

  strings:
    $a = "assets/tosversion"
    $b = "assets/0OO00l111l1l"
    $c = "assets/0OO00oo01l1l"
    $d = "assets/o0oooOO0ooOo.dat"

  condition:
    is_apk
    and $b
    and ($a or $c or $d)
}

rule分为三部分

  • meta 元数据,提供描述信息
  • strings 字符串部分,列举了需要匹配的特定字符串
  • condition 指定了规则的匹配条件

可以看到这里的规则就是需要满足是APK、存在 assets/0OO00l111l1l,以及 a,c,d对应字符串中的一个
打开我们通过腾讯加壳的APK软件,可以看到对应的特征值确实存在

而 is_apk 这个判定条件实际上在 common.yara里面

rule is_apk : file_type
{
  meta:
    description = "APK"

  strings:
    $zip_head = "PK"
    $manifest = "AndroidManifest.xml"

  condition:
    $zip_head at 0 and $manifest and #manifest >= 2
}

这里的规则就是检查目标文件开头是否匹配字符串PK,对于压缩包我们知道头文件是50 4B 也就是PK

同是目标文件包含字符串"AndroidManifest.xml",且数量>=2

这就是这条tencent_legu规则的详情

深入学习yara

APKiD是如何工作的

工作流程

主要实现逻辑的方法在

def scan_file_obj(self, file: IO, file_path: str = '$FILE$'):
    if file_path == '$FILE$':
        file_name = file_path
    else:
        file_name = os.path.basename(file_path)

    results: Dict[str, List[yara.Match]] = {}
    if not self._should_scan(file, file_name):
        return results

    matches: List[yara.Matches] = self.rules.match(data=file.read(), timeout=self.options.timeout)
    if len(matches) > 0:
        results[file_path] = matches
    if self._is_zipfile(file, file_name):
        with zipfile.ZipFile(file) as zf:
            zip_results = self._scan_zip(zf)
        for entry_name, entry_matches in zip_results.items():
            results[f'{file_path}!{entry_name}'] = entry_matches
    return results

这里会先对全文件进行一次rules.match匹配,然后对于压缩包类型的文件,解压之后再遍历扫描一次
遍历扫描的时候,针对的文件也就是上面截图中的那些,判断逻辑是

def _should_scan(self, file: IO, name: str) -> bool:
    if self.options.typing == 'magic':
        file_type = Scanner._type_file(file)
        return file_type is not None
    elif self.options.typing == 'filename':
        name = name.lower()
        return name.startswith('classes') \
               or name.startswith('AndroidManifest.xml') \
               or name.startswith('lib/') \
               or name.endswith('.so') \
               or name.endswith('.dex') \
               or name.endswith('.apk')
    return True

扫描结束之后把结果经过处理存储在result里面就OK了,整体逻辑也很清晰
相对于其他的APK扫描工具,它的特点在于使用了yara文件来作为规则,可以在 "打标" 的时候更灵活

其他的一些开源工具

https://github.com/Andy10101/ApkDetecter/ 为例
规则是通过字典的方式放置的,如果想要扩充的话在该部分添加即可

self.protectflag_dict = {"libsecexe.so": u"该APK已加固=>梆梆加固", "libAPKProtect.so": u"该APK已加固=>APKProtect加固",
                           "libprotectClass.so": u"该APK已加固=>360加固", "libNSaferOnly.so": u"该APK已加固=>通付盾加固",
                           "libnqshield.so": u"该APK已加固=>网秦加固", "libshell.so": u"该APK已加固=>腾讯加固",
                           "ijiami.dat": u"该APK已加固=>爱加密加固", "libddog.so": u"该APK已加固=>娜迦加固",
                           "libmobisec.so": u"该APK已加固=>阿里加固", "libbaiduprotect.so": u"该APK已加固=>百度加固"}

后续对比是通过有文件名包含对应关键字来判断

for key in self.protectflag_dict.keys():
    if file_name.has_key(key):
        self.protectflag = self.protectflag + self.protectflag_dict[key]

if file_name.has_key("key.dat") and all_dir_name.has_key("apkprotect.com"):
    if self.protectflag == "" or (u"APKProtect加固" not in self.protectflag):
        self.protectflag = self.protectflag + u"APKProtect加固"

师傅们可以选择自己喜欢的选择使用

参考链接

0 条评论
某人
表情
可输入 255