如果你想了解更多,欢迎收看

 

项目背景与目标


作为B站的重度用户,Feed流中越来越多的竖屏视频与广告让人非常厌烦,于是决定逆向一把,过滤掉这些玩意。本文最后你将可以实现:

  • 完全可自定义的过滤规则
  • 自定显示或隐藏某些UI界面
  • 更改播放器的默认行为
  • 控制面板
     

 

逆向基础知识


本文只着重分享逆向的实战经验与过程。但在开始之前,笔者还是要先简单讲一下几个比较重要的知识点。

  • 函数调用机制
    在Objective-C中,简单来说,一个函数调用例如 [obj foo: arg]会在底层转变为obj_msgSend(obj, @selector(foo:), arg),接着在运行时按照 method cache → 本类方法列表 → 父类链 查这个 SEL 对应的 IMP(实现函数) 并调用。
    Method Swizzle(方法交换)
    这是OC中提供的一种可以在运行时修改“SEL → IMP”映射的机制,能够在查找IMP的过程中使后续消息派发跳转到开发者自定义的新实现。
    如果需要更改App的行为,绝大多数情况下,都是通过此机制Hook原函数并加入自定义逻辑。所以我们的目标就是找到某些功能对应的函数实现,并且加入我们的自定义逻辑,下文我们就将通过多种不同途径来寻找对应的函数。
  • 反射机制
    反射能够在编译时不感知目标类型与方法签名的情况下,仅靠 String(类名/方法名/SEL)完成查找、实例化与方法调用。我们在写逻辑时,就要靠他来对原App的具体变量进行访问。
  • 框架注入
    在重新打包App时,可以注入我们自己写的framework,再在OC中使用+load函数即可让我们的代码在App启动后自动运行,实现上面的方法交换逻辑。

前期准备


砸壳
App Store 下载的安装包,都是通过加密的,我们无法直接逆向,所以需要通过砸壳来获取未加密的IPA文件。这里有两种途径:

  1. 准备一台越狱的iPhone,通过frida插件在App运行时提取未加密的IPA,https://github.com/AloneMonkey/frida-ios-dump
     
  2. 对于一些主流App,会有第三方网站提供IPA文件下载,例如:https://decrypt.day,不过安全性不受保证。
    安装与调试
  3. 在安装IPA之前需要对App进行重签名,这里笔者习惯使用Sideloadly,注意非Apple Developer开发者账号单次签名有效期只有7天,反之是一年。在选择设备后就可以安装到自己的手机上了。
     

调试可直接在Xcode中打开空白工程,并连接iPhone,在Debug->Attach to Process->xxxxxx,例如这里B站的process name就是 bili-universal

在成功Attach上之后,就可以开始我们的逆向之旅了。


更改 UI 行为


首先暂停App之后,一个最简单的入手方法就是通过UI,我们在LLDB中输入
po [[UIWindow keyWindow] recursiveDescription]
即可打印出目前屏幕上的UI树。

这里我们可以看到所有UIView的类名,属性,地址,甚至直接可以搜索屏幕上的元素。
例如此处搜索搜索框的文字,我们就能知道搜索框是由BBList.SearchControl实现。


如果我们仅仅需要Hook UI的一些行为,有了类名之后,到这里已经成功一大半了,下一步我们应该使用method swizzle找到一个合适的时机,将我们想要执行的代码塞进去就行了。那么我们一般寻找这个界面的ViewController就可以了,在VC中有各种生命周期可供Hook。
那么我们应该如何查找UIView对应的生命周期呢?简单来说就是递归调用NextResponder,由于篇幅篇幅原因,这里省略原理。总之nextResponder会按照 UIView → superview → ViewController 的顺序,所以例如这里的SearchControl,只需要在lldb中调用nextResponder,直到找到VC。
 


可以利用反射机制Hook该VC,并通过swizzle其生命周期函数,执行自己的逻辑,比如显示隐藏一个UIView,或者插入额外的view。
例如本项目的一个功能,禁止显示竖屏视频的按钮,就是通过Hook UIStackView的addArrangedSubView,再判断这些按钮是不是命中了类名,如果命中就拒绝执行。

@interface UIStackView (GSVStackBlock)
- (void)gsv_addArrangedSubview:(UIView *)view;
- (void)gsv_insertArrangedSubview:(UIView *)view atIndex:(NSUInteger)stackIndex;
@end

@implementation UIStackView (GSVStackBlock)

- (void)gsv_addArrangedSubview:(UIView *)view {
    if (gsv_shouldBlockView(view)) {
        return;
    }
    [self gsv_addArrangedSubview:view];
}

- (void)gsv_insertArrangedSubview:(UIView *)view atIndex:(NSUInteger)stackIndex {
    if (gsv_shouldBlockView(view)) {
        return;
    }
    [self gsv_insertArrangedSubview:view atIndex:stackIndex];
}

@end

@interface GSVStackHooker : NSObject
@end

@implementation GSVStackHooker

+ (void)load {
    Class cls = [UIStackView class];
    if (!cls) { return; }
    gsv_swizzleInstance(cls, @selector(addArrangedSubview:), @selector(gsv_addArrangedSubview:));
    gsv_swizzleInstance(cls, @selector(insertArrangedSubview:atIndex:), @selector(gsv_insertArrangedSubview:atIndex:));
}

@end

 

 

更改 App 逻辑


在打包后的App中,虽然绝大多数函数名都被strip掉,但是对于OC类,仍然保留了methodDescption, 可以由一个实例的地址,拿到OC的类中方法与属性签名
 


例如在本项目中禁止短视频播放器的视频播放,就是通过播放器的UIView,找到他的ViewController,接着尝试打印他的方法,发现里面有setLooping方法,所以按照推测我们将viewDidAppear的时候强制setLooping为false,即可实现。

@implementation UIViewController (GSVHook)

- (void)gsv_viewDidAppear:(BOOL)animated {
    [self gsv_viewDidAppear:animated];
    
    NSString *className = NSStringFromClass(self.class);
    GSVSettingsManager *settings = [GSVSettingsManager sharedManager];
    
    // 功能1:强制设置循环为false(针对BBStoryPlayerContainer)
    if ([className isEqualToString:@"BBStoryPlayerContainer"] && settings.forceSetLoopingFalse) {
        SEL loopingSel = NSSelectorFromString(@"setLooping:");
        if ([self respondsToSelector:loopingSel]) {
            ((void (*)(id, SEL, BOOL))objc_msgSend)(self, loopingSel, NO);
        } else {
            NSLog(@"[GSV] %@ 不响应 setLooping:,跳过", className);
        }
    }
}

@end


目前看起来逆向也太简单了,但并不是所有情况下,都能这么简单直接的找到所需要的类或函数,例如过滤Feed流中的短视频。


进阶


Feed流的显示,无非就是一个UICollectionView,然后有若干个Cell不断的进行替换,所以我有下面几个思路:

  • 暴力隐藏UIView
    最简单的方案就像是上文中一样,直接暴力判断UIView的className是否等于xxxxx,然后再递归遍历其子View,如果发现有text为竖屏的UILabel,就强制设置isHidden = true,然后再该Cell移除屏幕 prepareForReuse的时候恢复其透明度。防止下次用的时候默认被隐藏了。
    这个方法极其的简单粗暴,但也非常不完美,因为并不能改变layout,所以feed流上会出现一块一块的白补丁,Pass。
static NSString *gsv_firstPortraitLabelPath(UIView *root, BOOL isContentViewRoot) {
    if (!root) { return nil; }
    NSString *rootSeg = isContentViewRoot ? @"contentView" : NSStringFromClass(root.class);
    return gsv_findPortraitLabelPathRecursive(root, rootSeg);
 }
  • CollectionView的DataSource
    如果你了解CollectionView的工作方式,就是不停的到DataSource delegate中去询问需要展示几个Cell,每个Cell的内容等参数。所以在DataSource中,一定会有真正数据源的访问。接着我们可以顺着这条线寻找到真正的数据源。
    这里我们就要通过上面method Description中打印出的方法地址,设置断点。切记给你的断点-N 加个名字,因为全是地址,断点多了不然会忘了是哪个。
     

接着操作一下app,就能命中这个断点。
在命中之后,使用di命令即可显示该函数下的汇编代码,这里我们可以借助AI进行分析,并在可疑的地址前下断点,再去访问加载到寄存器中的实例类名,并且通过偏移再去寻找其属性,这里略去具体过程。


分析得出该 CollectionView的dataSource就是VC,同时在运行 cellForItemAtIndexPath时,访问了viewController->viewModel->dataFatcher->cardData Array。
但不幸的是,从viewModel开始,就是Swift类了,Swift静态派发的方法,并不能像OC一样使用Swizzle进行替换,并且可用的信息也非常之少。所以最后这条路也行不通了。

  • 网络库
    既然从UI下手行不通,那我们就尝试两头堵,从网络库入手试试看。首先我们使用Charles抓包抓到负责Feed流的网络请求,得到他的URL。


接着我们Hook NSURLSession的delegate,打印出看否有符合我们要求的域名,结果果然是没有。看起来他也没有走自带的网络库,后来我想会不会是从前端将数据传递进来的,我还Hook了JSRuntime相关的函数,结果依然令人失望。

  • IDA
    似乎我们两头堵也不行,那咋办呢,这个时候就需要碰碰运气了。比如我又想到,获取数据一定跟刷新有关系,那么在CollectionView中如果想刷新,一定要调用 reload方法,所以我们在reload方法下断点。


这里我们使用backtrace展开调用栈。我们可以看到reload是由app中一些堆栈调用上来的,我们想知道下面的那些frame到底是什么。这里我们就搬出IDA了,IDA 可以将所有汇编代码映射到静态地址中。并且解析出所有符号。
所以我们只需要通过目前App运行的动态地址,计算出静态地址,再在IDA中通过地址查询,就可以找到地址对应的函数。
如何计算IDA中的静态地址呢?

  1. 先在lldb中获取我们App,运行的基地址,例如这次是 0x104e14000:

2. 使用frame中的地址减去基地址得到偏移量,在使用偏移量加上0x100000000

3. 在IDA中使用静态地址中搜索
经过一番尝试,还真找到了底层的Request
接着查看这个函数的伪代码,好家伙,连NSJSONSerialization都出来了,那基本上证明了这就是我们需要找的函数。
 


接着直接Hook这个函数,如果判断是这个接口的话,先手动把Data序列化一遍,然后递归遍历是否有广告或者竖屏视频,如果有的话直接移除,最后再给他反序列回Data,透传给原函数的实现,这样就完美了。

 "config": {
     "auto_refresh_by_behavior": 1,
     "auto_refresh_time": 1200,
     "auto_refresh_time_by_active": 1800,
     "auto_refresh_time_by_appear": 1800,
     "auto_refresh_time_by_behavior": 5,
     "autoplay_card": 11,
     "card_density_exp": 1,
     "close_small_window": 1,
     "column": 2,
     "enable_rcmd_guide": true,
     "exposure_duration_end_ratio": 0.800000011920929,
     "exposure_duration_min_ms": 1,
     "exposure_duration_start_ratio": 0.800000011920929,
     "feed_clean_abtest": 0,
     "history_cache_size": 10,
     "home_transfer_test": 0,
     "inline_sound_cold_state": 2,
     "is_back_to_homepage": true,
     "rcmd_label_mng_entrance": 1,
     "show_inline_danmaku": 1,
     "single_autoplay_flag": 1,
     "small_cover_wh_ratio": 1.333333,
     "space_enlarge_exp": 1,
     "story_mode_v2_guide_exp": 6,
     "three_point_style": 1,
     "toast": {},
     "trigger_loadmore_left_line_num": 1,
     "video_mode": 1,
     "visible_area": 80
 },
 "interest_choose": null,
 "items": [
     {
         "args": {
             "aid": 114902270222302,
             "tid": 76859374,
             "tname": "鬼畜星探企划第二十二期",
             "up_id": 1722956360,
             "up_name": "会跳舞的杰哥"
         },
         "can_play": 1,
         "card_goto": "inline_av_v2",
         "card_type": "large_cover_v9",
         "cover": "http://i0.hdslb.com/bfs/archive/2a45b6c09a3f51e9cc5f0ae237a2a1cc28e2be1c.jpg",
         "cover_left_1_content_description": "119.3K观看",
         "cover_left_2_content_description": "4694弹幕",
         "cover_left_icon_1": 1,
         "cover_left_icon_2": 3,
         "cover_left_text_1": "119.3K",
         "cover_left_text_2": "4694",
         "cover_right_content_description": "9分钟34秒",
         "cover_right_text": "9:34",
         "goto": "av",
         "idx": 1758725170,
         "inline_progress_bar": {
             "icon_drag": "https://i0.hdslb.com/bfs/archive/c1461e2c6ca97783ac0298b6ebb2d85d94b8f37c.json",
             "icon_drag_hash": "31df8ce99de871afaa66a7a78f44deec",
             "icon_stop": "https://i0.hdslb.com/bfs/archive/6ee2f9b016f20714705cb5b8f15da1446587d172.json",
             "icon_stop_hash": "5648c2926c1c93eb2d30748994ba7b96"
         },
         "like_button": {
             "aid": 114902270222302,
             "count": 4291,
             "dislike_night_resource": {
                 "content_hash": "c370e8d031381f4716d7564956a8b182",
                 "url": "https://i0.hdslb.com/bfs/archive/c9a20055b712068bfe293878639dc9066ba2690b.json"
             },

收尾工作
最后我们用AI写个开关面板,然后再Hook一下UITableView,把B站讨厌的会员购icon与界面换成弹出我们的设置面板。
 


因为我们的代码都写在一个Framework里,我们只需要构建出产物,接着在打包的时候,选择Inject frameworks,接着把产物添加进去就OK了。
 


好了,现在终于可以在不看广告的情况下愉快的刷B站了。

关于我:

前不知名数码博主/独立开发者/客户端程序员 

全网同名,欢迎关注

11
3