iOS逆向篇之微信搶紅包(下)

上文已經(jīng)知道收到紅包消息與開紅包方法,那么只需在收到紅包消息時带膀,調(diào)用開紅包方法,就可以完成自動搶紅包
完整的工程地址

一 紅包消息時調(diào)用開紅包方法

上篇hook到了收到紅包消息的方法就是CMessageMgr類中的onNewSyncAddMessage:方法,于是把開紅包的代碼放到此方法中。
onNewSyncAddMessage:方法中叹侄,當(dāng)m_uiMessageType=49的時候是紅包消息,此時判斷是否開啟了自動搶紅包功能

- (void)onNewSyncAddMessage:(CMessageWrap *)m_Wrap{
if(MSHookIvar<unsigned int>(m_Wrap,"m_uiMessageType") != 49){//不是紅包消息
        %orig;
        return;
    }
if([WCDefaults boolForKey:WCSWITCHKEY] == NO){//是否開啟自動搶紅包
        %orig;
        return;
    }
    WCPayInfoItem *item =[m_Wrap m_oWCPayInfoItem];
    NSString *url =[item  m_c2cNativeUrl];
    NSInteger length = [@"wxpay://c2cbizmessagehandler/hongbao/receivehongbao?" length];
    id componets = [url substringFromIndex:length];
    NSDictionary *dictionary=[%c(WCBizUtil) dictionaryWithDecodedComponets:componets separator:@"&"];
    NSMutableDictionary *mul_dict =[%c(NSMutableDictionary) dictionary];
    [mul_dict setObject:@"1" forKey:@"msgType"];
    [mul_dict setObject:dictionary[@"sendid"] forKey:@"sendId"];
    [mul_dict setObject:dictionary[@"channelid"] forKey:@"channelId"];
    CContactMgr *server =[[%c(MMServiceCenter) defaultCenter] getService:[%c(CContactMgr) class]];
    CContact *contact = [server getSelfContact];
    id  displayName = [contact getContactDisplayName];
    [mul_dict setObject:displayName forKey:@"nickName"];
    id m_nsHeadImgUrl = [contact m_nsHeadImgUrl];
    [mul_dict setObject:m_nsHeadImgUrl forKey:@"headImg"];
    if ( url )
    {
        [mul_dict setObject:url forKey:@"nativeUrl"];        
    }
     NSString *m_nsUsrName = MSHookIvar<NSString *>(m_Wrap,"m_nsFromUsr");
    [mul_dict setObject:m_nsUsrName forKey:@"sessionUserName"];
    NSLog(@"---%@",m_nsUsrName);
//因?yàn)閙_data獲取不到昨登,所以編譯不過
    NSDictionary  *m_struct = [m_data m_structDicRedEnvelopesBaseInfo];
    NSString *timingIdentifier= [m_struct objectForKey:@"timingIdentifier"];
    if ( [timingIdentifier length])
    { [mul_dict setObject:timingIdentifier forKey:@"timingIdentifier"];}
    
    WCPayLogicMgr *payLogicMgr =[[%c(MMServiceCenter) defaultCenter] getService:[%c(WCPayLogicMgr) class]];
    [payLogicMgr setRealnameReportScene:1003];
    id subScript = [m_struct objectForKeyedSubscript:@"agree_duty"];
    [payLogicMgr checkHongbaoOpenLicense:subScript acceptCallback:^(){
        WCRedEnvelopesLogicMgr *envelopesLogicMgr = [[%c(MMServiceCenter) defaultCenter] getService:[%c(WCRedEnvelopesLogicMgr) class]];
        [envelopesLogicMgr OpenRedEnvelopesRequest:mul_dict];
    }denyCallback:^(){
        
    }];
}
%end

這個方法主要就是拼接mul_dict然后作為參數(shù)趾代,調(diào)用WCRedEnvelopesLogicMgr類的OpenRedEnvelopesRequest:函數(shù)。mul_dict需要的參數(shù)丰辣,能從CMessageWrap中獲取的直接獲取撒强,但是timingIdentifier獲取不到禽捆,所以要另作分析,要想辦法獲取到timingIdentifier飘哨。

二 其他方法獲取timingIdentifier

當(dāng)紅包消息收到后胚想,點(diǎn)擊紅包消息時有菊花在轉(zhuǎn),說明在請求網(wǎng)絡(luò)杖玲,請求過后才是紅包彈出的界面顿仇,所以現(xiàn)在可以進(jìn)行hook點(diǎn)擊紅包消息時的方法

  1. 通過分析知道WCRedEnvelopesLogicMgr類就是紅包管理類淘正,所以直接hook這個類中的所有方法:


然后進(jìn)行分析摆马,發(fā)現(xiàn)以下三個方法是點(diǎn)擊紅包消息時調(diào)用的方法:

[<WCRedEnvelopesLogicMgr: 0x10ef9aff0> ReceiverQueryRedEnvelopesRequest:{
 agreeDuty = 0;
 channelId = 1;
 inWay = 1;
 msgType = 1;
 nativeUrl = "wxpay://c2cbizmessagehandler/hongbao/receivehongbao?msgtype=1&channelid=1&sendid=1000039501201805287004922762149&sendusername=wxid_hy6hye79l4q241&ver=6&sign=9995652ef3b076bbc7b626c0272f7cbbb2168732bc1a9af34777acbf4206d97062af9a8e3c84ad8812d45b48039d9c74e2f6fadb01223e74a9361a61947d4a19789abfde732b339ffb17c05c7bd7a2a0";
 sendId = 1000039501201805287004922762149;
 } ]
 2018-05-28 10:21:42.258137+0800 WeChat[28565:2640576] -[<WCRedEnvelopesLogicMgr: 0x10ef9aff0> GetHongbaoBusinessRequest:{
 agreeDuty = 0;
 channelId = 1;
 inWay = 1;
 msgType = 1;
 nativeUrl = "wxpay://c2cbizmessagehandler/hongbao/receivehongbao?msgtype=1&channelid=1&sendid=1000039501201805287004922762149&sendusername=wxid_hy6hye79l4q241&ver=6&sign=9995652ef3b076bbc7b626c0272f7cbbb2168732bc1a9af34777acbf4206d97062af9a8e3c84ad8812d45b48039d9c74e2f6fadb01223e74a9361a61947d4a19789abfde732b339ffb17c05c7bd7a2a0";
 sendId = 1000039501201805287004922762149;
 } CMDID:3 OutputType:1 ]

 2018-05-28 10:21:42.490993+0800 WeChat[28565:2640576] [<WCRedEnvelopesLogicMgr: 0x10ef9aff0> OnWCToHongbaoCommonResponse:<HongBaoRes: 0x10efd1550> Request:<HongBaoReq: 0x10ed82840> ]

ReceiverQueryRedEnvelopesRequest:方法和GetHongbaoBusinessRequest:方法都是紅包請求方法,參數(shù)也差不多鸿吆,通過ida分析ReceiverQueryRedEnvelopesRequest:發(fā)現(xiàn)這個方法就是對GetHongbaoBusinessRequest:方法的封裝囤采。

  • 那么可以得到ReceiverQueryRedEnvelopesRequest:和OnWCToHongbaoCommonResponse:Request:這兩個就是關(guān)鍵方法
  1. 在onNewSyncAddMessage:收到紅包消息的時候,主動調(diào)用GetHongbaoBusinessRequest:方法
  WCRedEnvelopesLogicMgr *envelopesLogicMgr = [[%c(MMServiceCenter) defaultCenter] getService:[%c(WCRedEnvelopesLogicMgr) class]];
    NSMutableDictionary *param =[%c(NSMutableDictionary) dictionary];
    [param setObject:@"1" forKey:@"msgType"];
    [param setObject:dictionary[@"channelid"] forKey:@"channelId"];
    [param setObject:@"0" forKey:@"agreeDuty"];
    [param setObject:@"1" forKey:@"inWay"];
if([m_nsUsrName cons containsString:@"@chatroom"]){//群紅包
      [param setObject:@"0" forKey:@"inWay"];
    }
     [param setObject:dictionary[@"sendid"] forKey:@"sendId"];
     [param setObject:url forKey:@"nativeUrl"];
    [envelopesLogicMgr ReceiverQueryRedEnvelopesRequest:param];
  • 拼接參數(shù)惩淳,m_nsFromUsr字段包含@chatroom就是群紅包

可以得到蕉毯,主動調(diào)用ReceiverQueryRedEnvelopesRequest:方法和手動點(diǎn)擊紅包消息是一樣的效果。那么我們再來分析OnWCToHongbaoCommonResponse:Request:方法思犁,通過打印參數(shù)類型得到此方法的發(fā)第一個參數(shù)是HongBaoRes代虾,然后通過頭文件得到HongBaoRes如下:

@interface SKBuiltinBuffer_t : NSObject
@property(retain, nonatomic) NSData *buffer; 
@property(nonatomic) unsigned int iLen;

@end
@interface SKBuiltinString_t : NSObject
@property(retain, nonatomic) NSString *string;
@end

@interface BaseResponse : NSObject
@property(retain, nonatomic) SKBuiltinString_t *errMsg;
@property(nonatomic) int ret;

@end
@interface HongBaoRes : NSObject
@property(retain, nonatomic) BaseResponse *baseResponse;
@property(nonatomic) int cgiCmdid; // @dynamic cgiCmdid;
@property(retain, nonatomic) NSString *errorMsg; // @dynamic errorMsg;
@property(nonatomic) int errorType; // @dynamic errorType;
@property(retain, nonatomic) NSString *platMsg; // @dynamic platMsg;
@property(nonatomic) int platRet; // @dynamic platRet;
@property(retain, nonatomic) SKBuiltinBuffer_t *retText; // @dynamic retText;
@end

然后我們在OnWCToHongbaoCommonResponse:Request:中打印第一個參數(shù)的相關(guān)信息:

- (void)OnWCToHongbaoCommonResponse:(HongBaoRes *)arg1 Request:(id)arg2
{
    NSLog(@"platMsg:%@",arg1.platMsg);
    NSLog(@"cgiCmdid:%d",arg1.cgiCmdid);
    NSLog(@"platRet:%d",arg1.platRet);
    NSLog(@"errorType:%d",arg1.errorType);
    NSLog(@"errorMsg:%@",arg1.errorMsg);
    NSData *buffer = arg1.retText.buffer;
    NSString *aString = [[NSString alloc] initWithData:buffer encoding:NSUTF8StringEncoding];
     NSLog(@"retText:%@",aString);
    %orig;
}

打印輸出:

2018-05-28 11:31:44.197315+0800 WeChat[28877:2671064] platMsg:
2018-05-28 11:31:44.197483+0800 WeChat[28877:2671064] cgiCmdid:3
2018-05-28 11:31:44.197575+0800 WeChat[28877:2671064] platRet:0
2018-05-28 11:31:44.197654+0800 WeChat[28877:2671064] errorType:0
2018-05-28 11:31:44.197852+0800 WeChat[28877:2671064] errorMsg:ok
2018-05-28 11:31:44.198045+0800 WeChat[28877:2671064] retText:{"retcode":0,"retmsg":"ok","sendId":"1000039501201805287004922762149","wishing":"恭喜發(fā)財,大吉大利","isSender":0,"receiveStatus":0,"hbStatus":2,"statusMess":"給你發(fā)了一個紅包","hbType":0,"watermark":"","sendUserName":"wxid_hy6hye79l4q241","timingIdentifier":"35971B49DE4AE3E604E17E3EAB463272"}

可以得到retText:中有timingIdentifier

  • 所以獲取timingIdentifier
    在收到紅包消息時主動調(diào)用ReceiverQueryRedEnvelopesRequest:方法激蹲,然后在- (void)OnWCToHongbaoCommonResponse:(HongBaoRes *)arg1 Request:(id)arg2方法中的第一個參數(shù)HongBaoRes的retText屬性里面包含timingIdentifier
  • 分析HongBaoRes參數(shù)
    cgiCmdid==3棉磨,點(diǎn)擊紅包消息,沒有被搶過的紅包
    HongBaoRes的retText中:
    receiveStatus == 0沒有搶過的紅包 receiveStatus == 2搶過的紅包;isSender==0自己不是發(fā)紅包消息的人;isSender==1自己發(fā)的紅包
  1. 獲取到timingIdentifier
    要在紅包還沒有搶過的情況下才搶紅包
NSError *error;
NSDictionary *responseDict = [NSJSONSerialization JSONObjectWithData:arg1.retText.buffer options:NSJSONReadingMutableContainers error:&error];
NSString *receiveStatus =[NSString stringWithFormat:@"%@",responseDict[@"receiveStatus"]];
if(arg1 !=nil && arg2 !=nil && arg1.cgiCmdid==3 && [receiveStatus isEqualToString:@"0"])//沒有被搶過的紅包
 {   
      NSString *timingIdentifier = responseDict[@"timingIdentifier"];
}

那么獲取到timingIdentifier后怎么通知收到紅包消息的方法->CMessageMgr類中的onNewSyncAddMessage:方法呢学辱?而且紅包消息方法一下來很多條呢乘瓤?
我們可以用單例隊(duì)列來保存紅包消息,在獲取到timingIdentifier后取出隊(duì)列中的參數(shù)策泣,然后調(diào)用開紅包方法

三 開始搶紅包

  1. 單例隊(duì)列來保存紅包消息
    WeChatMessageWarp.h
@interface WeChatMessageWarp : NSObject

+(instancetype)shareInstanceQueue;

- (void)inputQueue:(NSMutableDictionary *)param;
- (NSMutableDictionary *)getParamQueue;
@end

WeChatMessageWarp.m

@interface WeChatMessageWarp()
@property (nonatomic,strong)NSMutableArray  *queue;
@end

@implementation WeChatMessageWarp
+(instancetype)shareInstanceQueue
{
    static WeChatMessageWarp *queue = nil;
    static dispatch_once_t  once;
    dispatch_once(&once,^{
        queue = [[WeChatMessageWarp alloc]init];
    });
    return queue;
}
- (instancetype)init
{
    self = [super init];
    if (self) {
        _queue = [NSMutableArray array];
    }
    return self;
}

- (void)inputQueue:(NSMutableDictionary *)param{
    [self.queue addObject:param];
}
- (NSMutableDictionary *)getParamQueue{
    if (self.queue.count == 0 && !self.queue.firstObject) {
        return nil;
    } 
    NSMutableDictionary *first = self.queue.firstObject;
    [self.queue removeObjectAtIndex:0];
    return first;
    
}
@end
  • 在收到紅包消息的時候保存參數(shù)mul_dict衙傀,參數(shù)重可以標(biāo)記是不是自己寫的紅包插件
 WeChatMessageWarp *share = [WeChatMessageWarp shareInstanceQueue];
    //標(biāo)記是插件搶紅包
    [mul_dict setObject:@"YES" forKey:@"isMySelfMeryin"];
    [share inputQueue:mul_dict];
  • 然后在獲取到timingIdentifier后把參數(shù)取出來,并判斷是不是自己寫的插件
 WeChatMessageWarp *share = [WeChatMessageWarp shareInstanceQueue];
 NSMutableDictionary *warp = [share getParamQueue];
 NSString *flag = [warp objectForKey:@"isMySelfMeryin"];
 if(![flag isEqualToString:@"YES"]){//自己寫的插件搶紅包
        return;
  }
 [warp removeObjectForKey:@"isMySelfMeryin"];
 if(timingIdentifier.length >0 && warp.count >0){
        [warp setObject:timingIdentifier forKey:@"timingIdentifier"];
  }
  1. 開始搶紅包
    搶紅包的時候萨咕,要判斷插件是否設(shè)置了延遲時間统抬,然后延遲幾秒開始搶紅包
//開始開紅包
             WCRedEnvelopesLogicMgr *envelopesLogicMgr = [[%c(MMServiceCenter) defaultCenter] getService:[%c(WCRedEnvelopesLogicMgr) class]];
            if(envelopesLogicMgr){
                //延遲搶紅包時間
                NSString *time = [WCDefaults valueForKey:WCTIMEKEY];
                float second = 1;
                if(time){//默認(rèn)1秒
                    second =[time floatValue];
                }
                
                dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(second * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                     [envelopesLogicMgr OpenRedEnvelopesRequest:warp];
                });
                
            }

四 優(yōu)化工程結(jié)構(gòu)

完成了自動搶紅包代碼后,一個類中的代碼太多危队,我們可以按功能拆分代碼

  • 把設(shè)置界面的代碼出來
    新建->iOS -Other -Empty->WeChatSettingViewControllerCell.xm 注意targets勾選dylib
    把設(shè)置界面的代碼拷貝到WeChatSettingViewControllerCell.xm 然后build生成WeChatSettingViewControllerCell.mm聪建,然后把.mm拖到工程中,注意targets勾選dylib
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末交掏,一起剝皮案震驚了整個濱河市妆偏,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌盅弛,老刑警劉巖钱骂,帶你破解...
    沈念sama閱讀 222,104評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件叔锐,死亡現(xiàn)場離奇詭異,居然都是意外死亡见秽,警方通過查閱死者的電腦和手機(jī)愉烙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,816評論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來解取,“玉大人步责,你說我怎么就攤上這事≠骺啵” “怎么了蔓肯?”我有些...
    開封第一講書人閱讀 168,697評論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長振乏。 經(jīng)常有香客問我蔗包,道長,這世上最難降的妖魔是什么慧邮? 我笑而不...
    開封第一講書人閱讀 59,836評論 1 298
  • 正文 為了忘掉前任调限,我火速辦了婚禮,結(jié)果婚禮上误澳,老公的妹妹穿的比我還像新娘耻矮。我一直安慰自己,他們只是感情好忆谓,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,851評論 6 397
  • 文/花漫 我一把揭開白布裆装。 她就那樣靜靜地躺著,像睡著了一般陪毡。 火紅的嫁衣襯著肌膚如雪米母。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,441評論 1 310
  • 那天毡琉,我揣著相機(jī)與錄音铁瞒,去河邊找鬼。 笑死桅滋,一個胖子當(dāng)著我的面吹牛慧耍,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播丐谋,決...
    沈念sama閱讀 40,992評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼芍碧,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了号俐?” 一聲冷哼從身側(cè)響起泌豆,我...
    開封第一講書人閱讀 39,899評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎吏饿,沒想到半個月后踪危,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蔬浙,經(jīng)...
    沈念sama閱讀 46,457評論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,529評論 3 341
  • 正文 我和宋清朗相戀三年贞远,在試婚紗的時候發(fā)現(xiàn)自己被綠了畴博。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,664評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡蓝仲,死狀恐怖俱病,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情袱结,我是刑警寧澤亮隙,帶...
    沈念sama閱讀 36,346評論 5 350
  • 正文 年R本政府宣布,位于F島的核電站擎勘,受9級特大地震影響咱揍,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜棚饵,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,025評論 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望掩完。 院中可真熱鬧噪漾,春花似錦、人聲如沸且蓬。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,511評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽恶阴。三九已至诈胜,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間冯事,已是汗流浹背焦匈。 一陣腳步聲響...
    開封第一講書人閱讀 33,611評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留昵仅,地道東北人缓熟。 一個月前我還...
    沈念sama閱讀 49,081評論 3 377
  • 正文 我出身青樓,卻偏偏與公主長得像摔笤,于是被迫代替她去往敵國和親够滑。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,675評論 2 359

推薦閱讀更多精彩內(nèi)容