阅读此文档的过程中遇到任何问题,请关注公众号【移动端Android和iOS开发技术分享】或加QQ群【309580013

1.目标

在学习的过程中,会遇到有些算法比较麻烦,没有办法直接还原。那我们就另辟蹊径,不去分析具体的算法实现。直接使用rpc的方式调用算法函数,本文章以某安App的X-App-Token签名函数为例。

2.操作环境

  • mac系统

  • frida-dexdump:导出加固后dex文件

  • Charles:抓取http接口

  • 已Root安卓机:脱壳

  • Python3.8:实现rpc功能

  • Jadx:导出dex文件为源码

  • Android Studio:静态分析

3.流程

寻找切入点

通过Charles抓包获取到关键词为X-App-Token,这也就是我们的切入点:

静态分析

使用查壳工具发现该apk使用的是360加固,启动App后,使用frida-dexdump的frida-dexdump -FU命令导出dex文件:

由于dex文件较多,不方便查询,使用jadx把多个dex文件导出为源码:

import os

for file in os.listdir(os.curdir):
   if file.find(".dex") > 0:
       sh = 'jadx -j 1 -r -d ./ ./' + file
       print(sh)
       os.system(sh)

将以上的python脚本放到dex同级目录,切换到dex目录,并执行以上脚本,执行完成后会生成sources文件夹,使用Android Studio打开该文件夹,全局搜索X-App-Token:

找到关键函数:

private final String[] m13135() {
       String str;
       Locale locale = Locale.getDefault();
       String valueOf = String.valueOf(Build.VERSION.SDK_INT);
       String str2 = locale.getLanguage() + '-' + ((Object) locale.getCountry());
       byte[] bytes = (this.f16167.m13205() + "; ; ; " + this.f16167.m13207() + "; " + ((Object) Build.MANUFACTURER) + "; " + ((Object) Build.BRAND) + "; " + ((Object) Build.MODEL) + "; " + ((Object) Build.DISPLAY) + "; " + ((Object) C4765.m13174().m12851())).getBytes(Charsets.UTF_8);
       Intrinsics.checkNotNullExpressionValue(bytes, "this as java.lang.String).getBytes(charset)");
       String encodeToString = Base64.encodeToString(bytes, 0);
       Intrinsics.checkNotNullExpressionValue(encodeToString, "encodeToString(device.to…eArray(), Base64.DEFAULT)");
       String sb = new StringBuilder(encodeToString).reverse().toString();
       Intrinsics.checkNotNullExpressionValue(sb, "StringBuilder(device).reverse().toString()");
       String replace = new Regex("\\r\\n|\\r|\\n|=").replace(sb, BuildConfig.FLAVOR);
       String as = AuthUtils.getAS(this.f16166, replace);
       if (C4765.m13166().m13307()) {
           str = "1";
      } else {
           str = C4765.m13166().m13300() ? "2" : "0";
      }
       Intrinsics.checkNotNullExpressionValue(as, "appToken");
       String r1 = this.f16167.m13203();
       Intrinsics.checkNotNullExpressionValue(r1, "appMetadata.channel");
       return new String[]{"User-Agent", this.f16170, "X-Requested-With", "XMLHttpRequest", "X-Sdk-Int", valueOf, "X-Sdk-Locale", str2, "X-App-Id", "com.coolapk.market", "X-App-Token", as, "X-App-Version", this.f16168, "X-App-Code", String.valueOf(this.f16169), "X-Api-Version", "12", "X-App-Device", replace, "X-Dark-Mode", str, "X-App-Channel", r1, "X-App-Mode", this.f16167.m13197().toString(), "X-App-Supported", String.valueOf(this.f16167.m13199())};
  }

删除无关代码后:

private final String[] m13135() {
       byte[] bytes = (this.f16167.m13205() + "; ; ; " + this.f16167.m13207() + "; " + ((Object) Build.MANUFACTURER) + "; " + ((Object) Build.BRAND) + "; " + ((Object) Build.MODEL) + "; " + ((Object) Build.DISPLAY) + "; " + ((Object) C4765.m13174().m12851())).getBytes(Charsets.UTF_8);
       String encodeToString = Base64.encodeToString(bytes, 0);
       String sb = new StringBuilder(encodeToString).reverse().toString();
       String replace = new Regex("\\r\\n|\\r|\\n|=").replace(sb, BuildConfig.FLAVOR);
       String as = AuthUtils.getAS(this.f16166, replace);
  }

由此可看出,X-App-Token参数由AuthUtils.getAS方法生成,本篇文章的目的是通过rpc直接调用getAS函数,所以不会去具体分析getAS方法的实现。

调用getAS方法的入参有两个,第一个是context,第二个参数即为我们需要拼接的参数。看源码可知,该参数由多个参数拼成,然后再base64。具体查看每一个函数,整理后的结果如下:

this.f16167.m13205():android_id

this.f16167.m13207():wifi的mac地址

((Object) Build.MANUFACTURER):硬件制造商

((Object) Build.MODEL):系统定制商

((Object) Build.DISPLAY):显示屏参数

((Object) C4765.m13174().m12851()):这个参数为空,我们就暂且不管

你也可以直接使用命令frida-trace -UF -j '*!*getBytes*':

{
 onEnter(log, args, state) {
 log(`String.getBytes=${this.toString()}=`)
   log(`String.getBytes(${args.map(JSON.stringify).join(', ')})`);
},

 onLeave(log, retval, state) {
   if (retval !== undefined) {
     log(`<= ${JSON.stringify(retval)}`);
  }
}
}

获取到结果如下,根据结果反推参数值

e8e69e7384cb09c0; ; ; F4:F5:DB:24:A6:E1; Xiaomi; xiaomi; MI 5X; QL1515-tiffany-build-20171026203938; null

结果

至此,我们的getAS函数的入参已经确定,接下来就是实现RPC

python源码如下:

import frida
from flask import Flask, request
import base64
result = {}


def on_message(message, data):
   if message['type'] == 'send':
       payload = message['payload']
       if "###" in payload:
           global result
           array = payload.split("###")
           result[array[0]] = array[1]
       print(message['payload'])
   elif message['type'] == 'error':
       print(message['stack'])


js_code = '''
rpc.exports = {
  // 函数名getAS
  sign: function(params){
      Java.perform(function(){
          //拿到context上下文
          var currentApplication = Java.use('android.app.ActivityThread').currentApplication();
          var context = currentApplication.getApplicationContext();

          // use 加载的类路径
          var AuthUtils = Java.use('com.coolapk.market.util.AuthUtils');
          var sign = AuthUtils.getAS(context, params); // context,params
          send(params+"###"+sign);
      }
  )
  }
};
'''

process = frida.get_usb_device().attach('酷安')
script = process.create_script(js_code)
script.on('message', on_message)
script.load()

app = Flask(__name__)


@app.route('/get_sign')
def get_sign():
   device_id = request.args.get('device_id', '')
   mac_address = request.args.get('mac_address', '').upper()
   manufacturer = request.args.get('manufacturer', '')
   model = request.args.get('model', '')
   display = request.args.get('display', '')
   params = f'{device_id}; ; ; {mac_address}; {manufacturer}; {model}; {display}; null'
   base_en = base64.encodebytes(params.encode('utf-8')).decode('utf-8')
   base_en = base_en[::-1]
   base_en = base_en.replace('\n', '')
   base_en = base_en.replace('\r', '')
   base_en = base_en.replace('=', '')
   script.exports.sign(base_en)
   sign = result[base_en]
   return sign


if __name__ == '__main__':
   app.run()

运行python脚本后,浏览器调用get_sign方法即可获取到X-App-Token

End

阅读此文档的过程中遇到任何问题,请关注公众号【移动端Android和iOS开发技术分享】或加QQ群【309580013

 

【Android逆向】rpc调用某安App的X-App-Token签名函数的更多相关文章

  1. Ratel:一直站在Android逆向巅峰的平头哥

    本文来源:带动行业内卷,渣总义不容辞 字越少事儿越大,请关注github(可以点击阅读原文): https://github.com/virjarRatel 平头哥(ratel)是一个Android逆 ...

  2. Android逆向分析(2) APK的打包与安装背后的故事

    前言 上一次我们反编译了手Q,并遇到了Apktool反编译直接crash的问题,虽然笔者很想在这次解决这个问题,但在解决途中,发现该保护依赖于很多知识,所以本次先插入一下,正所谓知其然知其所以然,授之 ...

  3. Android逆向分析(2) APK的打包与安装

    http://blog.zhaiyifan.cn/2016/02/13/android-reverse-2/ 2/18日增加对aidl和java编译的描述. 前言 上一次我们反编译了手Q,并遇到了Ap ...

  4. Android逆向之so的半自动化逆向

    因为工作需要,转型干android逆向,有几个月了.不过对于so的逆向,任然停留在,难难难的阶段,虽然上次自己还是逆向了一个15k左右的小so文件,但是,那个基本是靠,一步一步跟代码,查看堆栈信息来自 ...

  5. Android逆向之静态分析

    想必打过CTF的小伙伴多多少少都触过Android逆向,所以斗哥将给大家整一期关于Android逆向的静态分析与动态分析.本期先带来Android逆向的静态分析,包括逆向工具使用.文件说明.例题解析等 ...

  6. Android 逆向实战篇(加密数据包破解)

    1. 实战背景由于工作需要,要爬取某款App的数据,App的具体名称此处不便透露,避免他们发现并修改加密逻辑我就得重新破解了. 爬取这款App时发现,抓包抓到的数据是加密过的,如图1所示(原数据较长, ...

  7. 【转】Android逆向入门流程

    原文:https://www.jianshu.com/p/71fb7ccc05ff 0.写在前面 本文是笔者自学笔记,以破解某目标apk的方式进行学习,中间辅以原理性知识,方便面试需求. 参考文章的原 ...

  8. 在Android Studio中调用so中的方法

    本节用的so是上节用Android Studio创建的so.想在Android Studio中调用so中的方法,需要先引用so.Android Studio中引用so的方法有二种,下面开始介绍. 一 ...

  9. Android逆向之旅---Native层的Hook神器Cydia Substrate使用详解

    一.前言 在之前已经介绍过了Android中一款hook神器Xposed,那个框架使用非常简单,方法也就那几个,其实最主要的是我们如何找到一个想要hook的应用的那个突破点.需要逆向分析app即可.不 ...

  10. Android逆向系列文章— Android基础逆向(6)

    本文作者:HAI_ 0×00 前言 不知所以然,请看 Android逆向-Android基础逆向(1) Android逆向-Android基础逆向(2) Android逆向-Android基础逆向(2 ...

随机推荐

  1. HCNP Routing&Switching之IP安全

    前文我们了解了DHCP安全相关话题,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/16637627.html:今天我们来聊一聊IP安全相关话题: 技术背景 随着 ...

  2. 食之无味?App Startup 可能比你想象中要简单

    请点赞关注,你的支持对我意义重大. Hi,我是小彭.本文已收录到 GitHub · AndroidFamily 中.这里有 Android 进阶成长知识体系,有志同道合的朋友,关注公众号 [彭旭锐] ...

  3. day35-IO流02

    JavaOI流02 4.常用的类 4.1文件字节流输入流-FileInputStream InputStream抽象类是所有类字节输入流的超类 InputStream常用的子类: FileInputS ...

  4. 【Azure 环境】Azure Resource Graph Explorer 中实现动态数组数据转换成多行记录模式 - mv-expand

    问题描述 想对Azure中全部VM的NSG资源进行收集,如果只是查看一个VM的NSG设定,可以在门户页面中查看表格模式,但是如果想把导出成表格,可以在Azure Resource Graph Expl ...

  5. 在Windows Server 2019中配置多元密码策略

    长久以来,微软活动目录中的账户只能配置同一个密码策略.上到管理员账户,下到普通用户的密码策略都是一样的.而且密码策略只能在域级别配置生效.OU级别的密码策略只会对该OU中计算机的本地账户生效.通常认为 ...

  6. 1 Java内存区域管理

    目录 1 关于自动内存管理 2 运行时数据区域 2.1 程序计数器 2.2 虚拟机栈 2.2.1 局部变量表 2.2.2 操作数栈 2.3 本地方法栈 2.4 堆 2.5 方法区 2.5.1 运行时常 ...

  7. 在logstash中启动X-Pack Management功能后配置logstash的情况说明

    开启X-Pack Management功能后,启动logstsh的时候就不用再配置logstash.conf文件了,启动的时候也不用再使用-f指定这个文件进行启动了 一旦启动了logstash的集中管 ...

  8. 使用Prometheus和Grafana监控nacos集群

    官方文档:https://nacos.io/zh-cn/docs/monitor-guide.html 按照部署文档搭建好Nacos集群 配置application.properties文件,暴露me ...

  9. Module加载的详细说明-保证你有所收获

    模块 HTML 网页中,浏览器通过<script>标签加载 JavaScript 脚本. <!-- 页面内嵌的脚本 --> <script type="appl ...

  10. python中类与对象的命名空间(静态属性的陷阱)、__dict__ 和 dir() 在继承中使用说明

    1. 面向对象的概念 1)类是一类抽象的事物,对象是一个具体的事物:用类创建对象的过程,称为实例化. 2)类就是一个模子,只知道在这个模子里有什么属性.什么方法,但是不知道这些属性.方法具体是什么: ...