IOS代码注入APP&重签实战

代码地址: github地址

如何预防此类代码注入[IOS端监测APP被代码注入点击查看

]

本文介绍的是ios的简单代码注入实现,这次的APP用大千影视,来实现代码注入去广告功能。这次代码注入主要分为几个部分:

  1. 使用一台越狱手机,通过frida-ios-dump砸壳大千影视.app获得解密后的ipa包。

  2. 通过mach-o view,reveal,hopper-disassembler等软件对该软件进行分析,找到相关方法,进行hook。

  3. 编写sh脚本,重新签名app。

如何通过rida-ios-dump砸壳,网上有许多教程,十分简单,就不演示了。我们先编写下sh脚本,来将导出的mach-o重新签名变成自己的app。代码如下

# ${SRCROOT} 它是工程文件所在的目录
TEMP_PATH="${SRCROOT}/Temp"
#资源文件夹,我们提前在工程目录下新建一个APP文件夹,里面放ipa包
ASSETS_PATH="${SRCROOT}/APP"
#目标ipa包路径
TARGET_IPA_PATH="${ASSETS_PATH}/*.ipa"
#清空Temp文件夹
rm -rf "${SRCROOT}/Temp"
mkdir -p "${SRCROOT}/Temp"



#----------------------------------------
# 1. 解压IPA到Temp下
unzip -oqq "$TARGET_IPA_PATH" -d "$TEMP_PATH"
# 拿到解压的临时的APP的路径
TEMP_APP_PATH=$(set -- "$TEMP_PATH/Payload/"*.app;echo "$1")
# echo "路径是:$TEMP_APP_PATH"


#----------------------------------------
# 2. 将解压出来的.app拷贝进入工程下
# BUILT_PRODUCTS_DIR 工程生成的APP包的路径
# TARGET_NAME target名称
TARGET_APP_PATH="$BUILT_PRODUCTS_DIR/$TARGET_NAME.app"
echo "app路径:$TARGET_APP_PATH"

rm -rf "$TARGET_APP_PATH"
mkdir -p "$TARGET_APP_PATH"
cp -rf "$TEMP_APP_PATH/" "$TARGET_APP_PATH"



#----------------------------------------
# 3. 删除extension和WatchAPP.个人证书没法签名Extention
rm -rf "$TARGET_APP_PATH/PlugIns"
rm -rf "$TARGET_APP_PATH/Watch"



#----------------------------------------
# 4. 更新info.plist文件 CFBundleIdentifier
#  设置:"Set : KEY Value" "目标文件路径"
/usr/libexec/PlistBuddy -c "Set :CFBundleIdentifier $PRODUCT_BUNDLE_IDENTIFIER" "$TARGET_APP_PATH/Info.plist"


#----------------------------------------
# 5. 给MachO文件上执行权限
# 拿到MachO文件的路径
APP_BINARY=`plutil -convert xml1 -o - $TARGET_APP_PATH/Info.plist|grep -A1 Exec|tail -n1|cut -f2 -d\>|cut -f1 -d\<`
#上可执行权限
chmod +x "$TARGET_APP_PATH/$APP_BINARY"



#----------------------------------------
# 6. 重签名第三方 FrameWorks
TARGET_APP_FRAMEWORKS_PATH="$TARGET_APP_PATH/Frameworks"
if [ -d "$TARGET_APP_FRAMEWORKS_PATH" ];
then
for FRAMEWORK in "$TARGET_APP_FRAMEWORKS_PATH/"*
do

#签名
/usr/bin/codesign --force --sign "$EXPANDED_CODE_SIGN_IDENTITY" "$FRAMEWORK"
done
fi


将sh文件放到代码根目录下,再创建一个APP文件夹,然后将导出的app包放入APP文件夹下。如图1:
\color{HotPink}{注意:在脚本添加之前,请先将工程安装在真机设备上。}

图一

接下来我们在BuildPhases内添加脚本,

image.png

然后添加脚本路径,${SRCROOT}是本身代码路径的环境变量。
image.png

最后选择真机运行,代码就会替换之前的app,将大千影视的app包重新签名替换原来的mach-o文件。这样我们的机子上就安装了一个重签了我们自己的证书的应用
image.png

接下来我们通过xcode来调试下这个应用
image.png

在这个播放页面我们通过xcode 的view debug来查看广告的窗口
image.png

通过class-dump导出头文件
class-dump路径 -H 需要导出的框架路径 -o 导出的头文件存放路
我们将这个导出的文件放到Sublime Text中
image.png

接下来我们需要新建一个动态库framework和这个app包相关联。


image.png

我们创建一个SQHook的framework。


image.png

添加一个类InjectDelAD
#import "InjectDelAD.h"

@implementation InjectDelAD
+(void)load{
    NSLog(@"你好!!~!!!!!!!!~~~~~~~");
}
@end

image.png

然后在脚本的最后添加

#注入
yololib "$TARGET_APP_PATH/$APP_BINARY" "Frameworks/SQHook.framework/SQHook"

yololib会将动态库添的链接添加到mach-o的动态库申明中。
现在load commands中没有我们的动态库声明


image.png

保存好脚本重新编译,这时候可以打印出来了。(注:要下载yololib源码编译出可执行文件后放到/usr/local/bin下)


image.png

这时候再看mach-o文件,动态库已经挂载上去了。
image.png

接下来我们开始调试,在广告页面我们点view debug
image.png

我们看到广告页面使用的是SaveAllSentientBeingsABeforeOrEndPlayADView,我们去刚刚导出的.h文件中查找这个类

//
//     Generated by class-dump 3.5 (64 bit) (Debug version compiled Sep 17 2017 16:24:48).
//
//     class-dump is Copyright (C) 1997-1998, 2000-2001, 2004-2015 by Steve Nygard.
//

#import <UIKit/UIImageView.h>

@class GCDTimerTool, NewADModel, UIButton, UILabel;

@interface SaveAllSentientBeingsABeforeOrEndPlayADView : UIImageView
{
    UILabel *_timerLabel;
    UILabel *_noticeLabel;
    NewADModel *_model;
    GCDTimerTool *_timer;
    UIButton *_fullScreenBtn;
    UIButton *_backBtn;
    UIImageView *_shadow;
}

+ (void)showWithPlayEndFlag:(_Bool)arg1 willPlayNext:(_Bool)arg2 PlayerView:(id)arg3 withModel:(id)arg4 closeHandler:(CDUnknownBlockType)arg5;
@property(retain, nonatomic) UIImageView *shadow; // @synthesize shadow=_shadow;
@property(retain, nonatomic) UIButton *backBtn; // @synthesize backBtn=_backBtn;
@property(retain, nonatomic) UIButton *fullScreenBtn; // @synthesize fullScreenBtn=_fullScreenBtn;
@property(retain, nonatomic) GCDTimerTool *timer; // @synthesize timer=_timer;
@property(retain, nonatomic) NewADModel *model; // @synthesize model=_model;
@property(retain, nonatomic) UILabel *noticeLabel; // @synthesize noticeLabel=_noticeLabel;
@property(retain, nonatomic) UILabel *timerLabel; // @synthesize timerLabel=_timerLabel;
- (void).cxx_destruct;
- (void)dealloc;
- (void)setBtnHidden;
- (void)layoutSubviews;
- (id)initWithCloseHandler:(CDUnknownBlockType)arg1;

@end


我们看到这个初始化方法- (id)initWithCloseHandler:(CDUnknownBlockType)arg1;这时候我们设置返回空,是不是就没广告呢。这个方法是oc方法可以用runtime方法交换进行hook。

//
//  InjectDelAD.m
//  SQHook
//
//  Created by Sem on 2020/8/26.
//  Copyright © 2020 SEM. All rights reserved.
//

#import "InjectDelAD.h"
#import <UIKit/UIKit.h>
#import <objc/runtime.h>
@implementation InjectDelAD
+(void)load{
    NSLog(@"你好!!~!!!!!!!!~~~~~~~");
    Method initADView = class_getInstanceMethod(objc_getClass("SaveAllSentientBeingsABeforeOrEndPlayADView"), @selector(initWithCloseHandler:));
    method_exchangeImplementations(initADView, class_getInstanceMethod(self, @selector(initWithCloseHandlerHook:)));
    
}
-(instancetype)initWithCloseHandlerHook:(id)arg{
     NSLog(@"交换方法执行~~~~~~~~~!!!");
    return nil;
}
@end


再编译下看下效果发现广告没了。。。


image.png

结束语:通过上述方法也可以去掉开机广告等,这部分的代码注入只能hook掉oc方法,对于一些系统的函数什么的要使用其他的方式,如fishhook什么的,一些静态函数需要更改mach-o文件等。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,053评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,527评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,779评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,685评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,699评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,609评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,989评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,654评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,890评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,634评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,716评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,394评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,976评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,950评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,191评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,849评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,458评论 2 342