iOS網(wǎng)絡(luò)請(qǐng)求模擬庫(kù)OHHTTPStubs的介紹和使用

你能使用OHHTTPStubs做什么

OHHTTPStubs的主要功能有兩點(diǎn):

  1. 偽造網(wǎng)絡(luò)請(qǐng)求返回的數(shù)據(jù)
  2. 模擬網(wǎng)絡(luò)請(qǐng)求時(shí)的慢網(wǎng)環(huán)境

我們通常會(huì)在以下情況下使用:

  1. 偽造數(shù)據(jù)、模擬慢網(wǎng)環(huán)境剧罩,檢測(cè)APP在網(wǎng)絡(luò)環(huán)境不好的情況下的行為。
  2. 單元測(cè)試時(shí)姓建,使用OHHTTPStubs提供模擬數(shù)據(jù)。
  3. 在開(kāi)發(fā)階段缤苫,模擬網(wǎng)絡(luò)請(qǐng)求數(shù)據(jù)速兔,不依賴服務(wù)器而進(jìn)行本地開(kāi)發(fā),通過(guò)這樣的方式來(lái)加速開(kāi)發(fā)活玲。

基本用法

  1. 攔截域名為mywebservice.com的http請(qǐng)求涣狗,并返回一個(gè)數(shù)組對(duì)象。
[OHHTTPStubs stubRequestsPassingTest:^BOOL(NSURLRequest * _Nonnull request) {
    return [request.URL.host isEqualToString:@"mywebservice.com"];
} withStubResponse:^OHHTTPStubsResponse * _Nonnull(NSURLRequest * _Nonnull request) {
    NSArray *array = @[@"Hello", @"world"];
    return [OHHTTPStubsResponse responseWithJSONObject:array statusCode:200 headers:nil];
}];
  1. mywebservice.com發(fā)送GET請(qǐng)求舒憾,這個(gè)時(shí)候會(huì)收到成功的返回镀钓,并且responseObject為數(shù)組[@"Hello", @"world"]
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
[manager GET:@"http://mywebservice.com" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable
responseObject) {
    NSArray *data = responseObject;
    NSLog(@"%@", data);
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
    NSLog(@"%@", error);
}];

可以看到镀迂,OHHTTPStubs的使用非常簡(jiǎn)單方便掸宛,設(shè)置需要攔截的URL,提供攔截到請(qǐng)求之后招拙,提供需要返回的OHHTTPStubsResponse對(duì)象唧瘾。

OHHTTPStubs主要接口

  1. 添加stub
+(id<OHHTTPStubsDescriptor>)stubRequestsPassingTest:(OHHTTPStubsTestBlock)testBlock
                                   withStubResponse:(OHHTTPStubsResponseBlock)responseBlock;

這是OHHTTPStubs的類方法,我們通過(guò)調(diào)用這個(gè)方法來(lái)對(duì)URL進(jìn)行攔截别凤。
在參數(shù)testBlock中進(jìn)行URL的匹配饰序,如果返回YES,則該請(qǐng)求將會(huì)被攔截规哪。
在參數(shù)responseBlock中求豫,返回?cái)r截了該請(qǐng)求之后的響應(yīng)數(shù)據(jù)OHHTTPStubsResponse對(duì)象。

  1. OHHTTPStubsResponse對(duì)象

OHHTTPStubsResponse對(duì)象是作為被攔截的請(qǐng)求的響應(yīng)诉稍,它包含HTTP響應(yīng)頭蝠嘉、響應(yīng)正文、狀態(tài)碼杯巨、響應(yīng)時(shí)間等信息蚤告。

OHHTTPStubsResponse這個(gè)類提供了快捷方法來(lái)創(chuàng)建該對(duì)象。

  • 使用NSData作為響應(yīng)數(shù)據(jù):
+(instancetype)responseWithData:(NSData*)data
                     statusCode:(int)statusCode
                        headers:(nullable NSDictionary*)httpHeaders;
  • 使用文件的內(nèi)容來(lái)作為響應(yīng)數(shù)據(jù):
+(instancetype)responseWithFileAtPath:(NSString *)filePath
                           statusCode:(int)statusCode
                              headers:(nullable NSDictionary*)httpHeaders;

+(instancetype)responseWithFileURL:(NSURL *)fileURL
                        statusCode:(int)statusCode
                           headers:(nullable NSDictionary *)httpHeaders;
  • 使用JSON對(duì)象來(lái)作為響應(yīng)數(shù)據(jù):
+ (instancetype)responseWithJSONObject:(id)jsonObject
                            statusCode:(int)statusCode
                               headers:(nullable NSDictionary *)httpHeaders;
  • 直接返回一個(gè)網(wǎng)絡(luò)錯(cuò)誤:
+(instancetype)responseWithError:(NSError*)error;
  1. 網(wǎng)絡(luò)狀況的模擬

網(wǎng)絡(luò)狀況的模擬主要是通過(guò)OHHTTPStubsResponse對(duì)象的requestTimeresponseTime兩個(gè)屬性來(lái)控制服爷。

  • requestTime
    請(qǐng)求在發(fā)送前必須等待的時(shí)間
  • responseTime
    正數(shù)時(shí)杜恰,responseTime就是從發(fā)送請(qǐng)求到接收到完整響應(yīng)數(shù)據(jù)的時(shí)間。
    負(fù)數(shù)時(shí)仍源,responseTime就是下載速度心褐,會(huì)根據(jù)需要下載的數(shù)據(jù)的大小,動(dòng)態(tài)的計(jì)算下載完成所需要的時(shí)間笼踩。

對(duì)于responseTime逗爹,OHHTTPStubs已經(jīng)定義了一些常量值供我們使用

const double OHHTTPStubsDownloadSpeed1KBPS  =-     8 / 8; // kbps -> KB/s
const double OHHTTPStubsDownloadSpeedSLOW   =-    12 / 8; // kbps -> KB/s
const double OHHTTPStubsDownloadSpeedGPRS   =-    56 / 8; // kbps -> KB/s
const double OHHTTPStubsDownloadSpeedEDGE   =-   128 / 8; // kbps -> KB/s
const double OHHTTPStubsDownloadSpeed3G     =-  3200 / 8; // kbps -> KB/s
const double OHHTTPStubsDownloadSpeed3GPlus =-  7200 / 8; // kbps -> KB/s
const double OHHTTPStubsDownloadSpeedWifi   =- 12000 / 8; // kbps -> KB/s

可以通過(guò)如下代碼設(shè)置requestTimeresponseTime

[[OHHTTPStubsResponse responseWithData:data statusCode:200 headers:@{@"Content-Type":@"application/json"}]
                requestTime:1.0f responseTime:OHHTTPStubsDownloadSpeed3G];
  1. 移除注冊(cè)的stub

查看stubRequestsPassingTest:withStubResponse:的源碼

+(id<OHHTTPStubsDescriptor>)stubRequestsPassingTest:(OHHTTPStubsTestBlock)testBlock
                                   withStubResponse:(OHHTTPStubsResponseBlock)responseBlock
{
    OHHTTPStubsDescriptor* stub = [OHHTTPStubsDescriptor stubDescriptorWithTestBlock:testBlock
                                                                       responseBlock:responseBlock];
    [OHHTTPStubs.sharedInstance addStub:stub];
    return stub;
}

+ (instancetype)sharedInstance
{
    static OHHTTPStubs *sharedInstance = nil;

    static dispatch_once_t predicate;
    dispatch_once(&predicate, ^{
        sharedInstance = [[self alloc] init];
    });

    return sharedInstance;
}

我們可以看到,每次添加一個(gè)stub之后嚎于,會(huì)創(chuàng)建一個(gè)OHHTTPStubsDescriptor實(shí)例由單例對(duì)象OHHTTPStubs持有掘而。如果我們不手動(dòng)將添加的stub移除挟冠,勢(shì)必會(huì)造成內(nèi)存常駐。我們可以調(diào)用以下方法移除添加的stub镣屹。

+(BOOL)removeStub:(id<OHHTTPStubsDescriptor>)stubDesc;
+(void)removeAllStubs;
  1. 啟用和禁用stubs

所有的stubs默認(rèn)是啟用的圃郊,即默認(rèn)會(huì)對(duì)需要攔截的請(qǐng)求進(jìn)行攔截价涝∨冢可以通過(guò)以下方法啟用或禁用攔截:

+(void)setEnabled:(BOOL)enabled;
+ (void)setEnabled:(BOOL)enabled forSessionConfiguration:(NSURLSessionConfiguration *)sessionConfig;
  1. 添加監(jiān)聽(tīng)事件
/**
 添加一個(gè)在每個(gè)stub被觸發(fā)都執(zhí)行一次的block

 @param block 每個(gè)stub被觸發(fā)都執(zhí)行一次的block
 */
+(void)onStubActivation:( nullable void(^)(NSURLRequest* request, id<OHHTTPStubsDescriptor> stub, OHHTTPStubsResponse* responseStub) )block;

/**
 添加一個(gè)在OHHTTPStubs發(fā)生請(qǐng)求重定向的時(shí)候被執(zhí)行的block

 @param block 在OHHTTPStubs發(fā)生請(qǐng)求重定向的時(shí)候被執(zhí)行的block
 */
+(void)onStubRedirectResponse:( nullable void(^)(NSURLRequest* request, NSURLRequest* redirectRequest, id<OHHTTPStubsDescriptor> stub, OHHTTPStubsResponse* responseStub) )block;

/**
 添加一個(gè)在stub完成之后會(huì)被執(zhí)行的block

 @param block 在stub完成之后會(huì)被執(zhí)行的block
 */
+(void)afterStubFinish:( nullable void(^)(NSURLRequest* request, id<OHHTTPStubsDescriptor> stub, OHHTTPStubsResponse* responseStub, NSError *error) )block;

/**
 添加一個(gè)在OHHTTPStubs遇到無(wú)法攔截的請(qǐng)求時(shí)調(diào)用的block

 @param block 在OHHTTPStubs遇到無(wú)法攔截的請(qǐng)求時(shí)調(diào)用的block
 */
+(void)onStubMissing:( nullable void(^)(NSURLRequest* request) )block;

  1. POST請(qǐng)求時(shí)的HTTPBody

當(dāng)使用NSURLSession發(fā)送POST請(qǐng)求時(shí),請(qǐng)求的body會(huì)在到達(dá)OHHTTPStubs時(shí)被置為nil色瘩。也就是說(shuō)在testBlockresponseBlock中伪窖,直接獲取NSURLRequestHTTPBody會(huì)得到nil

OHHTTPStubs通過(guò)方法置換在調(diào)用NSURLRequestsetHTTPBody:時(shí)居兆,將HTTPBody的內(nèi)容做了一個(gè)備份覆山。提供了方法OHHTTPStubs_HTTPBody來(lái)獲取備份的HTTPBody

具體代碼如下:

NSString * const OHHTTPStubs_HTTPBodyKey = @"HTTPBody";

@implementation NSURLRequest (HTTPBodyTesting)

- (NSData*)OHHTTPStubs_HTTPBody
{
    return [NSURLProtocol propertyForKey:OHHTTPStubs_HTTPBodyKey inRequest:self];
}

@end

#pragma mark - NSMutableURLRequest+HTTPBodyTesting

typedef void(*OHHHTTPStubsSetterIMP)(id, SEL, id);
static OHHHTTPStubsSetterIMP orig_setHTTPBody;

static void OHHTTPStubs_setHTTPBody(id self, SEL _cmd, NSData* HTTPBody)
{
    // store the http body via NSURLProtocol
    if (HTTPBody) {
        [NSURLProtocol setProperty:HTTPBody forKey:OHHTTPStubs_HTTPBodyKey inRequest:self];
    } else {
        // unfortunately resetting does not work properly as the NSURLSession also uses this to reset the property
    }

    orig_setHTTPBody(self, _cmd, HTTPBody);
}

/**
 *   Swizzles setHTTPBody: in order to maintain a copy of the http body for later
 *   reference and calls the original implementation.
 *
 *   @warning Should not be used in production, testing only.
 */
@interface NSMutableURLRequest (HTTPBodyTesting) @end

@implementation NSMutableURLRequest (HTTPBodyTesting)

+ (void)load
{
    orig_setHTTPBody = (OHHHTTPStubsSetterIMP)OHHTTPStubsReplaceMethod(@selector(setHTTPBody:),
                                                                       (IMP)OHHTTPStubs_setHTTPBody,
                                                                       [NSMutableURLRequest class],
                                                                       NO);
}

@end

注意事項(xiàng)

  • OHHTTPStubs不能用于后臺(tái)會(huì)話(由[NSURLSessionConfiguration backgroundSessionConfiguration]創(chuàng)建的會(huì)話)泥栖,因?yàn)楹笈_(tái)會(huì)話由iOS系統(tǒng)自己維護(hù)簇宽,并且不允許使用NSURLProtocols
  • OHHTTPStubs不能模擬數(shù)據(jù)上傳吧享。NSURLProtocolClient協(xié)議并沒(méi)有提供任何方式通知delegate數(shù)據(jù)已經(jīng)發(fā)送魏割,所以一個(gè)NSURLRequestHTTPBodyHTTPBodyStream或是由-[NSURLSession uploadTaskWithRequest:fromData:]提供的數(shù)據(jù)都會(huì)被忽略钢颂。最重要的是使用OHHTTPStubs來(lái)stub一個(gè)請(qǐng)求后钞它,代理方法-URLSession:task:didSendBodyData:totalBytesSent:totalBytesExpectedToSend:永遠(yuǎn)也不會(huì)被調(diào)用。
  • OHHTTPStubs還有一個(gè)重定向的問(wèn)題殊鞭,具有零延遲的重定向會(huì)以一個(gè)空響應(yīng)結(jié)束遭垛。

提交到App Store

OHHTTPStubs中沒(méi)有使用任何私有的API,它是可以被提交到App Store的操灿。但是我們基本上只會(huì)在開(kāi)發(fā)階段使用stubs锯仪,所以是沒(méi)有必要把stubs提交到App Store的。

我們可以通過(guò)#if DEBUG塊的方式來(lái)避免提交相關(guān)代碼到App Store趾盐。

為了避免在發(fā)布前忘記移除對(duì)OHHTTPStubs的導(dǎo)入卵酪,可以使用下面的方式對(duì)pod進(jìn)行配置:

pod 'OHHTTPStubs', :configurations => ['Debug', 'Development']

或以下代碼:

pod 'OHHTTPStubs', :configurations => 'Debug'

如果配置為僅在Debug模式下導(dǎo)入OHHTTPStubs,那么在其它模式下使用OHHTTPStubs就會(huì)報(bào)錯(cuò)谤碳。所以一定要記得使用#if DEBUG塊或者刪除相關(guān)的代碼溃卡。

最后附上github地址:https://github.com/AliSoftware/OHHTTPStubs

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市蜒简,隨后出現(xiàn)的幾起案子瘸羡,更是在濱河造成了極大的恐慌,老刑警劉巖搓茬,帶你破解...
    沈念sama閱讀 216,591評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件犹赖,死亡現(xiàn)場(chǎng)離奇詭異队他,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)峻村,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén)麸折,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人粘昨,你說(shuō)我怎么就攤上這事垢啼。” “怎么了张肾?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,823評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵芭析,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我吞瞪,道長(zhǎng)馁启,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,204評(píng)論 1 292
  • 正文 為了忘掉前任芍秆,我火速辦了婚禮惯疙,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘妖啥。我一直安慰自己霉颠,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,228評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布迹栓。 她就那樣靜靜地躺著掉分,像睡著了一般。 火紅的嫁衣襯著肌膚如雪克伊。 梳的紋絲不亂的頭發(fā)上酥郭,一...
    開(kāi)封第一講書(shū)人閱讀 51,190評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音愿吹,去河邊找鬼不从。 笑死,一個(gè)胖子當(dāng)著我的面吹牛犁跪,可吹牛的內(nèi)容都是我干的椿息。 我是一名探鬼主播,決...
    沈念sama閱讀 40,078評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼坷衍,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼寝优!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起枫耳,我...
    開(kāi)封第一講書(shū)人閱讀 38,923評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤乏矾,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體钻心,經(jīng)...
    沈念sama閱讀 45,334評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡凄硼,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,550評(píng)論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了捷沸。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片摊沉。...
    茶點(diǎn)故事閱讀 39,727評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖痒给,靈堂內(nèi)的尸體忽然破棺而出说墨,到底是詐尸還是另有隱情,我是刑警寧澤侈玄,帶...
    沈念sama閱讀 35,428評(píng)論 5 343
  • 正文 年R本政府宣布婉刀,位于F島的核電站吟温,受9級(jí)特大地震影響序仙,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜鲁豪,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,022評(píng)論 3 326
  • 文/蒙蒙 一潘悼、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧爬橡,春花似錦治唤、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,672評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至柜裸,卻和暖如春缕陕,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背疙挺。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,826評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工扛邑, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人铐然。 一個(gè)月前我還...
    沈念sama閱讀 47,734評(píng)論 2 368
  • 正文 我出身青樓蔬崩,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親搀暑。 傳聞我的和親對(duì)象是個(gè)殘疾皇子沥阳,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,619評(píng)論 2 354