RQHTTPService.m 48 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983
  1. //
  2. // RQHTTPService.m
  3. // RQCommon
  4. //
  5. // Created by 张嵘 on 2018/11/16.
  6. // Copyright © 2018 张嵘. All rights reserved.
  7. //
  8. #import "RQHTTPService.h"
  9. #import <AFNetworking/AFNetworkActivityIndicatorManager.h>
  10. #import <CocoaSecurity/CocoaSecurity.h>
  11. #import <MJExtension/MJExtension.h>
  12. /**
  13. * 知识点
  14. //- (RACSignal *)replayLast 就是用Capacity为1的RACReplaySubject来替换- (RACSignal *)replay的`subject。
  15. 其作用是使后来订阅者只收到最后的历史值。
  16. //- (RACSignal *)replayLazily和- (RACSignal *)replay的区别就是replayLazily会在第一次订阅的时候才订阅sourceSignal。
  17. // replay、replayLast、replayLazily的区别 ReactiveCocoa提供了这三个简便的方法允许多个订阅者订阅一个信号,却不会重复执行订阅代码,并且能给新加的订阅者提供订阅前的值。
  18. replay和replayLast使信号变成热信号,且会提供所有值(-replay) 或者最新的值(-replayLast) 给订阅者。
  19. replayLazily 会提供所有的值给订阅者 replayLazily还是冷信号 避免了冷信号的副作用
  20. *
  21. */
  22. /// The Http request error domain
  23. NSString *const RQHTTPServiceErrorDomain = @"RQHTTPServiceErrorDomain";
  24. /// 请求成功,但statusCode != 0
  25. NSString *const RQHTTPServiceErrorResponseCodeKey = @"RQHTTPServiceErrorResponseCodeKey";
  26. NSString * const RQHTTPServiceErrorRequestURLKey = @"RQHTTPServiceErrorRequestURLKey";
  27. NSString * const RQHTTPServiceErrorHTTPStatusCodeKey = @"RQHTTPServiceErrorHTTPStatusCodeKey";
  28. NSString * const RQHTTPServiceErrorDescriptionKey = @"RQHTTPServiceErrorDescriptionKey";
  29. NSString * const RQHTTPServiceErrorMessagesKey = @"RQHTTPServiceErrorMessagesKey";
  30. NSString * const RQNetworkingReachabilityDidChangeNotification = @"RQNetworkingReachabilityDidChangeNotification";
  31. /// 连接服务器失败 default
  32. NSInteger const RQHTTPServiceErrorConnectionFailed = 668;
  33. NSInteger const RQHTTPServiceErrorJSONParsingFailed = 669;
  34. NSInteger const RQHTTPServiceErrorBadRequest = 670;
  35. NSInteger const RQHTTPServiceErrorRequestForbidden = 671;
  36. /// 服务器请求失败
  37. NSInteger const RQHTTPServiceErrorServiceRequestFailed = 672;
  38. NSInteger const RQHTTPServiceErrorSecureConnectionFailed = 673;
  39. /// 登录相关
  40. NSString * const RQWeChatOnRespNotification = @"RQWeChatOnRespNotification";
  41. NSString * const RQWeBanindOnRespNotification = @"RQWeBanindOnRespNotification";
  42. @interface RQHTTPService ()
  43. @property (nonatomic, assign, readwrite) AFNetworkReachabilityStatus networkReachabilityStatus;
  44. @end
  45. @implementation RQHTTPService
  46. static id service_ = nil;
  47. #pragma mark - HTTPService
  48. +(instancetype) sharedInstance {
  49. static dispatch_once_t onceToken;
  50. dispatch_once(&onceToken, ^{
  51. service_ = [[self alloc] initWithBaseURL:[NSURL URLWithString:[RQConfigureManager requestBaseUrl]] sessionConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]];
  52. });
  53. return service_;
  54. }
  55. + (id)allocWithZone:(struct _NSZone *)zone{
  56. static dispatch_once_t onceToken;
  57. dispatch_once(&onceToken, ^{
  58. service_ = [super allocWithZone:zone];
  59. });
  60. return service_;
  61. }
  62. - (id)copyWithZone:(NSZone *)zone {
  63. return service_;
  64. }
  65. - (instancetype)initWithBaseURL:(NSURL *)url sessionConfiguration:(nullable NSURLSessionConfiguration *)configuration{
  66. if (self = [super initWithBaseURL:url sessionConfiguration:configuration]) {
  67. /// 配置
  68. [self _configHTTPService];
  69. }
  70. return self;
  71. }
  72. /// config service
  73. - (void)_configHTTPService{
  74. AFJSONResponseSerializer *responseSerializer = [AFJSONResponseSerializer serializer];
  75. #if DEBUG
  76. responseSerializer.removesKeysWithNullValues = NO;
  77. #else
  78. responseSerializer.removesKeysWithNullValues = YES;
  79. #endif
  80. responseSerializer.readingOptions = NSJSONReadingAllowFragments;
  81. /// config
  82. self.responseSerializer = responseSerializer;
  83. self.requestSerializer = [AFJSONRequestSerializer serializer];
  84. /// 安全策略
  85. AFSecurityPolicy *securityPolicy = [AFSecurityPolicy defaultPolicy];
  86. //allowInvalidCertificates 是否允许无效证书(也就是自建的证书),默认为NO
  87. //如果是需要验证自建证书,需要设置为YES
  88. securityPolicy.allowInvalidCertificates = YES;
  89. //validatesDomainName 是否需要验证域名,默认为YES;
  90. //假如证书的域名与你请求的域名不一致,需把该项设置为NO
  91. //主要用于这种情况:客户端请求的是子域名,而证书上的是另外一个域名。因为SSL证书上的域名是独立的,假如证书上注册的域名是www.google.com,那么mail.google.com是无法验证通过的;当然,有钱可以注册通配符的域名*.google.com,但这个还是比较贵的。
  92. securityPolicy.validatesDomainName = NO;
  93. self.securityPolicy = securityPolicy;
  94. /// 支持解析
  95. self.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"application/json",
  96. @"text/json",
  97. @"text/javascript",
  98. @"text/html",
  99. @"text/plain",
  100. @"text/html; charset=UTF-8",
  101. nil];
  102. /// 开启网络监测
  103. [[AFNetworkActivityIndicatorManager sharedManager] setEnabled:YES];
  104. [self.reachabilityManager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {
  105. switch (status) {
  106. case AFNetworkReachabilityStatusUnknown:
  107. // 未知网络
  108. NSLog(@"当前网络:未知网络");
  109. break;
  110. case AFNetworkReachabilityStatusNotReachable:
  111. // 无网络
  112. NSLog(@"当前网络:无网络");
  113. break;
  114. case AFNetworkReachabilityStatusReachableViaWWAN:
  115. // 蜂窝数据
  116. NSLog(@"当前网络:蜂窝数据");
  117. break;
  118. case AFNetworkReachabilityStatusReachableViaWiFi:
  119. // 无线网络
  120. NSLog(@"当前网络:无线网络");
  121. break;
  122. default:
  123. break;
  124. }
  125. if (_networkReachabilityStatus != status) {
  126. _networkReachabilityStatus = status;
  127. // 网络改变通知
  128. [[NSNotificationCenter defaultCenter] postNotificationName:RQNetworkingReachabilityDidChangeNotification object:[NSNumber numberWithInteger:status]];
  129. }
  130. }];
  131. [self.reachabilityManager startMonitoring];
  132. }
  133. #pragma mark - Request
  134. - (RACSignal *)enqueueRequest:(RQHTTPRequest *) request resultClass:(Class /*subclass of RQObject*/) resultClass{
  135. /// request 必须的有值
  136. if (!request) return [RACSignal error:[NSError errorWithDomain:RQHTTPServiceErrorDomain code:-1 userInfo:nil]];
  137. @weakify(self);
  138. // if ([request.urlParameters.method isEqualToString:RQ_HTTTP_METHOD_GET] || [request.urlParameters.path isEqualToString:RQ_POST_Code]) {
  139. // self.requestSerializer = [self _requestSerializerWithRequest:request];
  140. // } else if ([request.urlParameters.method isEqualToString:RQ_HTTTP_METHOD_POST] && ![request.urlParameters.path isEqualToString:RQ_POST_Code]) {
  141. // self.requestSerializer = [self _requestJSONSerializerWithRequest:request];
  142. // } else if ([request.urlParameters.method isEqualToString:RQ_HTTTP_METHOD_DELETE]) {
  143. // NSString *deletePath = [NSString stringWithFormat:@"%@/%@",request.urlParameters.path,request.urlParameters.parameters.allValues.firstObject];
  144. // request.urlParameters.path = deletePath;
  145. // request.urlParameters.parameters = @{};
  146. // self.requestSerializer = [self _requestSerializerWithRequest:request];
  147. // } else {
  148. // self.requestSerializer = [self _requestJSONSerializerWithRequest:request];
  149. // }
  150. // self.requestSerializer.HTTPMethodsEncodingParametersInURI = [NSSet setWithObjects:@"GET", @"HEAD", nil];
  151. // if (RQ_USER_MANAGER.isLogin) {
  152. // [self.requestSerializer setValue:RQ_USER_MANAGER.currentUser.token forHTTPHeaderField:@"Authorization"];
  153. // }
  154. if ([request.urlParameters.method isEqualToString:RQ_HTTTP_METHOD_GET] && ([request.urlParameters.path containsString:@"/userInfo/vip/info2"])) {
  155. NSString *newPath = [NSString stringWithFormat:@"%@/%@",request.urlParameters.path,request.urlParameters.parameters.allValues.firstObject];
  156. request.urlParameters.path = newPath;
  157. request.urlParameters.parameters = @{};
  158. self.requestSerializer = [self _requestSerializerWithRequest:request];
  159. } else {
  160. /// 覆盖manager请求序列化
  161. self.requestSerializer = [self _requestJSONSerializerWithRequest:request];
  162. /// @RQ-MARK:请求参数
  163. request.urlParameters.path = [NSString stringWithFormat:@"%@?%@",request.urlParameters.path, [self _signWithParameters:request.urlParameters.parameters]];
  164. }
  165. // request.urlParameters.path = [NSString stringWithFormat:@"%@",request.urlParameters.path];
  166. /// 发起请求
  167. /// concat:按一定顺序拼接信号,当多个信号发出的时候,有顺序的接收信号。 这里传进去的参数,不是parameters而是之前通过
  168. /// urlParametersWithMethod:(NSString *)method path:(NSString *)path parameters:(NSDictionary *)parameters;穿进去的参数
  169. return [[[self enqueueRequestWithPath:request.urlParameters.path parameters:request.urlParameters.parameters method:request.urlParameters.method]
  170. reduceEach:^RACStream *(NSURLResponse *response, NSDictionary * responseObject){
  171. @strongify(self);
  172. /// 请求成功 这里解析数据
  173. return [[self parsedResponseOfClass:resultClass fromJSON:responseObject]
  174. map:^(id parsedResult) {
  175. RQHTTPResponse *parsedResponse = [[RQHTTPResponse alloc] initWithResponseObject:responseObject parsedResult:parsedResult];
  176. NSAssert(parsedResponse != nil, @"Could not create RQHTTPResponse with response %@ and parsedResult %@", response, parsedResult);
  177. return parsedResponse;
  178. }];
  179. }]
  180. concat];
  181. }
  182. /// 请求数据
  183. - (RACSignal *)enqueueRequestWithPath:(NSString *)path parameters:(id)parameters method:(NSString *)method{
  184. @weakify(self);
  185. /// 创建信号
  186. RACSignal *signal = [RACSignal createSignal:^(id<RACSubscriber> subscriber) {
  187. @strongify(self);
  188. /// 获取request
  189. NSError *serializationError = nil;
  190. NSMutableURLRequest *request;
  191. // NSDictionary *dic = parameters;
  192. // YYCache *cache = [[YYCache alloc] initWithName:@"getTeachingVideoByTypeId"];
  193. // NSString *cacheResponseKeyName = [NSString stringWithFormat:@"Response-getTeachingVideoByTypeId=%@",dic[@"videoTypeId"]];
  194. // NSString *cacheResponseObjectKeyName = [NSString stringWithFormat:@"ResponseObject-getTeachingVideoByTypeId=%@",dic[@"videoTypeId"]];
  195. // if ([path containsString:@"getTeachingVideoByTypeId"]) {
  196. // if ([cache objectForKey:cacheResponseObjectKeyName] && [cache objectForKey:cacheResponseKeyName]) {
  197. // /// 打包成元祖 回调数据
  198. // NSURLResponse *response = (NSURLResponse *)[cache objectForKey:cacheResponseKeyName];
  199. // NSDictionary *responseObject = (NSDictionary *)[cache objectForKey:cacheResponseObjectKeyName];
  200. // dispatch_async(self.completionQueue ?: dispatch_get_main_queue(), ^{
  201. // [subscriber sendNext:RACTuplePack(response , responseObject)];
  202. // [subscriber sendCompleted];
  203. // });
  204. // return [RACDisposable disposableWithBlock:^{
  205. //
  206. // }];
  207. // }
  208. // }
  209. if ([RQ_CACHE_MANAGER isNeedCacheWithPath:path]) {
  210. NSDictionary *dic = [RQ_CACHE_MANAGER getCacheWithPath:path parameters:parameters];
  211. if (!RQObjectIsNil(dic)) {
  212. NSURLResponse *response = (NSURLResponse *)dic[@"response"];
  213. NSDictionary *responseObject = (NSDictionary *)dic[@"responseObject"];
  214. dispatch_async(self.completionQueue ?: dispatch_get_main_queue(), ^{
  215. [subscriber sendNext:RACTuplePack(response , responseObject)];
  216. [subscriber sendCompleted];
  217. });
  218. return [RACDisposable disposableWithBlock:^{
  219. }];
  220. }
  221. }
  222. if ([path containsString:@"zzjs.zzxcx.net"]) {
  223. request = [self.requestSerializer requestWithMethod:method URLString:[[NSURL URLWithString:path relativeToURL:[NSURL URLWithString:@""]] absoluteString] parameters:parameters error:&serializationError];
  224. } else if ([path containsString:@"jsjp-admin.zzxcx.net"] || [path containsString:@"jsjp-admin1.zzxcx.net"] || [path containsString:@"192.168.8.63:8080"]) {
  225. request = [self.requestSerializer requestWithMethod:method URLString:[[NSURL URLWithString:path relativeToURL:[NSURL URLWithString:@""]] absoluteString] parameters:parameters error:&serializationError];
  226. } else {
  227. if ([path isEqualToString:RQ_POST_AddCollections]) {
  228. NSDictionary *dic = parameters;
  229. request = [self.requestSerializer requestWithMethod:method URLString:[[NSURL URLWithString:path relativeToURL:self.baseURL] absoluteString] parameters:dic[@"list"] error:&serializationError];
  230. } else {
  231. request = [self.requestSerializer requestWithMethod:method URLString:[[NSURL URLWithString:path relativeToURL:self.baseURL] absoluteString] parameters:parameters error:&serializationError];
  232. }
  233. }
  234. if (serializationError) {
  235. #pragma clang diagnostic push
  236. #pragma clang diagnostic ignored "-Wgnu"
  237. dispatch_async(self.completionQueue ?: dispatch_get_main_queue(), ^{
  238. [subscriber sendError:serializationError];
  239. });
  240. #pragma clang diagnostic pop
  241. return [RACDisposable disposableWithBlock:^{
  242. }];
  243. }
  244. /// 获取请求任务
  245. __block NSURLSessionDataTask *task = nil;
  246. task = [self dataTaskWithRequest:request uploadProgress:nil downloadProgress:nil completionHandler:^(NSURLResponse *response, NSDictionary * responseObject, NSError *error) {
  247. if (error) {
  248. NSError *parseError = [self _errorFromRequestWithTask:task httpResponse:(NSHTTPURLResponse *)response responseObject:responseObject error:error];
  249. [self HTTPRequestLog:task body:parameters error:parseError];
  250. [subscriber sendError:parseError];
  251. } else {
  252. /// 断言
  253. NSAssert([responseObject isKindOfClass:NSDictionary.class], @"responseObject is not an NSDictionary: %@", responseObject);
  254. /// 在这里判断数据是否正确
  255. /// 判断
  256. NSInteger statusCode = [responseObject[RQHTTPServiceResponseCodeKey] integerValue];
  257. if (statusCode == RQHTTPResponseCodeSuccess || statusCode == RQHTTPResponseCodeJSJPSuccess) {
  258. // NSHTTPURLResponse *httpUrlResponse = (NSHTTPURLResponse *)response;
  259. // /// 存储token
  260. // NSString *token = [[[httpUrlResponse allHeaderFields] valueForKey:RQHTTPRequestTokenKey] rq_stringValueExtension];
  261. // [self saveToken:token];
  262. [self HTTPRequestLog:task body:parameters error:nil];
  263. /// 打包成元祖 回调数据
  264. [subscriber sendNext:RACTuplePack(response , responseObject)];
  265. // if ([path containsString:@"getTeachingVideoByTypeId"]) {
  266. // [cache setObject:response forKey:cacheResponseKeyName];
  267. // [cache setObject:responseObject forKey:cacheResponseObjectKeyName];
  268. // }
  269. if ([RQ_CACHE_MANAGER isNeedCacheWithPath:path]) {
  270. [RQ_CACHE_MANAGER saveCacheWithPath:path parameters:parameters response:response responseObject:responseObject];
  271. }
  272. [subscriber sendCompleted];
  273. }else{
  274. if (([path isEqualToString:RQ_GET_Userinfo] || [path isEqualToString:RQ_GET_Refresh_token]) && statusCode == 0) {
  275. /// 打包成元祖 回调数据
  276. NSMutableDictionary *dic = [NSMutableDictionary dictionaryWithDictionary:responseObject];
  277. [dic setValue:path forKey:@"path"];
  278. [subscriber sendNext:RACTuplePack(response , dic.copy)];
  279. [subscriber sendCompleted];
  280. }
  281. if (statusCode == RQHTTPResponseCodeNotLogin) {
  282. // /// 删除账号
  283. // [SAMKeychain deleteRawLogin];
  284. // /// 先退出用户
  285. // [RQ_USER_MANAGER logoutUser];
  286. //
  287. // /// 延迟一段时间
  288. // dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  289. // /// 这里切换 到账号登录的界面
  290. // [RQNotificationCenter postNotificationName:RQSwitchRootViewControllerNotification object:nil userInfo:@{RQSwitchRootViewControllerUserInfoKey:@(RQSwitchRootViewControllerFromTypeLogout)}];
  291. // RQ_USER_MANAGER.isShouldLogin;
  292. // });
  293. RACSignal *signalA = [[[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
  294. /// 删除账号
  295. [SAMKeychain deleteRawLogin];
  296. /// 先退出用户
  297. [RQ_USER_MANAGER logoutUser];
  298. [subscriber sendNext:@1];
  299. [subscriber sendCompleted];
  300. return nil;
  301. }] delay:1] deliverOnMainThread];
  302. RACSignal *signalB = [[[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
  303. /// 这里切换 主页面
  304. [RQNotificationCenter postNotificationName:RQSwitchRootViewControllerNotification object:nil userInfo:@{RQSwitchRootViewControllerUserInfoKey:@(RQSwitchRootViewControllerFromTypeLogout)}];
  305. [subscriber sendNext:@2];
  306. return nil;
  307. }] delay:1] deliverOnMainThread];
  308. // 把signalA拼接到signalB后,signalA发送完成,signalB才会被激活。
  309. RACSignal *concatSignal = [signalA concat:signalB];
  310. // 以后只需要面对拼接信号开发。
  311. // 订阅拼接的信号,不需要单独订阅signalA,signalB
  312. // 内部会自动订阅。
  313. // 注意:第一个信号必须发送完成,第二个信号才会被激活
  314. [concatSignal subscribeNext:^(id x) {
  315. NSLog(@"%@",x);
  316. if ([x integerValue] == 2) {
  317. RQ_USER_MANAGER.isShouldLogin;
  318. }
  319. }];
  320. // [[[[[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
  321. //
  322. // /// 延迟一段时间
  323. // [subscriber sendNext:@1];
  324. // return nil;
  325. // }] delay:1] deliverOnMainThread] then:^RACSignal * _Nonnull{
  326. // return [[[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
  327. // /// 这里切换 主页面
  328. // [RQNotificationCenter postNotificationName:RQSwitchRootViewControllerNotification object:nil userInfo:@{RQSwitchRootViewControllerUserInfoKey:@(RQSwitchRootViewControllerFromTypeLogout)}];
  329. // /// 延迟一段时间
  330. // [subscriber sendNext:@2];
  331. // return nil;
  332. // }] delay:1] deliverOnMainThread];
  333. // }] subscribeNext:^(id x) {
  334. // /// 这里切换 到账号登录的界面
  335. // NSLog(@"%@",x);
  336. // RQ_USER_MANAGER.isShouldLogin;
  337. // }];
  338. // [self login:^{
  339. // /// 这里需要重新配置序列化
  340. // self.requestSerializer = [self _requestSerializerWithRequest:[RQHTTPRequest requestWithParameters:[RQURLParameters urlParametersWithMethod:method path:path parameters:parameters]]];
  341. // /// 重新发起请求
  342. // [[self enqueueRequestWithPath:path parameters:parameters method:method] subscribe:subscriber];
  343. // } cancel:^{
  344. // NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  345. // userInfo[RQHTTPServiceErrorResponseCodeKey] = @(statusCode);
  346. // NSString *msgTips = responseObject[RQHTTPServiceResponseMsgKey];
  347. //#if defined(DEBUG)||defined(_DEBUG)
  348. // msgTips = RQStringIsNotEmpty(msgTips)?[NSString stringWithFormat:@"%@(%zd)",msgTips,statusCode]:[NSString stringWithFormat:@"服务器出错了,请稍后重试(%zd)~",statusCode]; /// 调试模式
  349. //#else
  350. // msgTips = RQStringIsNotEmpty(msgTips)?msgTips:@"服务器出错了,请稍后重试~"; /// 发布模式
  351. //#endif
  352. // userInfo[RQHTTPServiceErrorMessagesKey] = msgTips;
  353. // if (task.currentRequest.URL != nil) userInfo[RQHTTPServiceErrorRequestURLKey] = task.currentRequest.URL.absoluteString;
  354. // if (task.error != nil) userInfo[NSUnderlyingErrorKey] = task.error;
  355. // NSError *requestError = [NSError errorWithDomain:RQHTTPServiceErrorDomain code:statusCode userInfo:userInfo];
  356. // [self HTTPRequestLog:task body:parameters error:requestError];
  357. // [subscriber sendError:requestError];
  358. // }];
  359. } else {
  360. NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  361. userInfo[RQHTTPServiceErrorResponseCodeKey] = @(statusCode);
  362. NSString *msgTips = responseObject[RQHTTPServiceResponseMsgKey];
  363. if ([responseObject[RQHTTPServiceResponseDataKey] isKindOfClass:[NSString class]] && RQStringIsEmpty(msgTips)) {
  364. msgTips = responseObject[RQHTTPServiceResponseDataKey];
  365. #if defined(DEBUG)||defined(_DEBUG)
  366. msgTips = RQStringIsNotEmpty(msgTips)?[NSString stringWithFormat:@"%@(%zd)",msgTips,statusCode]:[NSString stringWithFormat:@"服务器出错了,请稍后重试(%zd)~",statusCode]; /// 调试模式
  367. #else
  368. msgTips = RQStringIsNotEmpty(msgTips)?msgTips:@"服务器出错了,请稍后重试~";/// 发布模式
  369. #endif
  370. } else if ([responseObject[RQHTTPServiceResponseBodyKey] isKindOfClass:[NSString class]] && RQStringIsEmpty(msgTips)) {
  371. msgTips = responseObject[RQHTTPServiceResponseBodyKey];
  372. #if defined(DEBUG)||defined(_DEBUG)
  373. msgTips = RQStringIsNotEmpty(msgTips)?[NSString stringWithFormat:@"%@(%zd)",msgTips,statusCode]:[NSString stringWithFormat:@"服务器出错了,请稍后重试(%zd)~",statusCode]; /// 调试模式
  374. #else
  375. msgTips = RQStringIsNotEmpty(msgTips)?msgTips:@"服务器出错了,请稍后重试~";/// 发布模式
  376. #endif
  377. } else {
  378. #if defined(DEBUG)||defined(_DEBUG)
  379. msgTips = RQObjectIsNil(msgTips)? [NSString stringWithFormat:@"服务器出错了,请稍后重试(%zd)~",statusCode] : (RQStringIsNotEmpty(msgTips)?[NSString stringWithFormat:@"%@(%zd)",msgTips,statusCode]:[NSString stringWithFormat:@"服务器出错了,请稍后重试(%zd)~",statusCode]); /// 调试模式
  380. #else
  381. msgTips = RQStringIsNotEmpty(msgTips)?msgTips:@"服务器出错了,请稍后重试~";/// 发布模式
  382. #endif
  383. }
  384. userInfo[RQHTTPServiceErrorMessagesKey] = msgTips;
  385. if (task.currentRequest.URL != nil) userInfo[RQHTTPServiceErrorRequestURLKey] = task.currentRequest.URL.absoluteString;
  386. if (task.error != nil) userInfo[NSUnderlyingErrorKey] = task.error;
  387. NSError *requestError = [NSError errorWithDomain:RQHTTPServiceErrorDomain code:statusCode userInfo:userInfo];
  388. [self HTTPRequestLog:task body:parameters error:requestError];
  389. [subscriber sendError:requestError];
  390. }
  391. }
  392. }
  393. }];
  394. /// 开启请求任务
  395. [task resume];
  396. return [RACDisposable disposableWithBlock:^{
  397. [task cancel];
  398. }];
  399. }];
  400. /// replayLazily:replayLazily会在第一次订阅的时候才订阅sourceSignal
  401. /// 会提供所有的值给订阅者 replayLazily还是冷信号 避免了冷信号的副作用
  402. return [[signal
  403. replayLazily]
  404. setNameWithFormat:@"-enqueueRequestWithPath: %@ parameters: %@ method: %@", path, parameters , method];
  405. }
  406. #pragma mark - Upload
  407. - (RACSignal *)enqueueUploadRequest:(RQHTTPRequest *)request resultClass:(Class)resultClass fileDatas:(NSArray<NSData *> *)fileDatas name:(NSString *)name mimeType:(NSString *)mimeType{
  408. /// request 必须的有值
  409. if (!request) return [RACSignal error:[NSError errorWithDomain:RQHTTPServiceErrorDomain code:-1 userInfo:nil]];
  410. /// 断言
  411. NSAssert(RQStringIsNotEmpty(name), @"name is empty: %@", name);
  412. @weakify(self);
  413. /// 覆盖manager 请求序列化
  414. self.requestSerializer = [self _requestSerializerWithRequest:request];
  415. /// 发起请求
  416. /// concat:按一定顺序拼接信号,当多个信号发出的时候,有顺序的接收信号。 这里传进去的参数,不是parameters而是之前通过
  417. /// urlParametersWithMethod:(NSString *)method path:(NSString *)path parameters:(NSDictionary *)parameters;穿进去的参数
  418. return [[[self enqueueUploadRequestWithPath:request.urlParameters.path parameters:request.urlParameters.parameters constructingBodyWithBlock:^(id<AFMultipartFormData> formData) {
  419. NSInteger count = fileDatas.count;
  420. for (int i = 0; i< count; i++) {
  421. /// 取出fileData
  422. NSData *fileData = fileDatas[i];
  423. /// 断言
  424. NSAssert([fileData isKindOfClass:NSData.class], @"fileData is not an NSData class: %@", fileData);
  425. // 在网络开发中,上传文件时,是文件不允许被覆盖,文件重名
  426. // 要解决此问题,
  427. // 可以在上传时使用当前的系统事件作为文件名
  428. static NSDateFormatter *formatter = nil;
  429. static dispatch_once_t onceToken;
  430. dispatch_once(&onceToken, ^{
  431. formatter = [[NSDateFormatter alloc] init];
  432. formatter.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
  433. });
  434. // 设置时间格式
  435. [formatter setDateFormat:@"yyyyMMddHHmmss"];
  436. NSString *dateString = [formatter stringFromDate:[NSDate date]];
  437. NSString *fileName = [NSString stringWithFormat:@"senba_empty_%@_%d.jpg", dateString , i];
  438. [formData appendPartWithFileData:fileData name:name fileName:fileName mimeType:RQStringIsNotEmpty(mimeType)?mimeType:@"application/octet-stream"];
  439. }
  440. }]
  441. reduceEach:^RACStream *(NSURLResponse *response, NSDictionary * responseObject){
  442. @strongify(self);
  443. /// 请求成功 这里解析数据
  444. return [[self parsedResponseOfClass:resultClass fromJSON:responseObject]
  445. map:^(id parsedResult) {
  446. RQHTTPResponse *parsedResponse = [[RQHTTPResponse alloc] initWithResponseObject:responseObject parsedResult:parsedResult];
  447. NSAssert(parsedResponse != nil, @"Could not create RQHTTPResponse with response %@ and parsedResult %@", response, parsedResult);
  448. return parsedResponse;
  449. }];
  450. }]
  451. concat];;
  452. }
  453. - (RACSignal *)enqueueUploadRequestWithPath:(NSString *)path parameters:(id)parameters constructingBodyWithBlock:(void (^)(id <AFMultipartFormData> formData))block{
  454. @weakify(self);
  455. /// 创建信号
  456. RACSignal *signal = [RACSignal createSignal:^(id<RACSubscriber> subscriber) {
  457. @strongify(self);
  458. /// 获取request
  459. NSError *serializationError = nil;
  460. NSMutableURLRequest *request = [self.requestSerializer multipartFormRequestWithMethod:@"POST" URLString:[[NSURL URLWithString:path relativeToURL:self.baseURL] absoluteString] parameters:parameters constructingBodyWithBlock:block error:&serializationError];
  461. if (serializationError) {
  462. #pragma clang diagnostic push
  463. #pragma clang diagnostic ignored "-Wgnu"
  464. dispatch_async(self.completionQueue ?: dispatch_get_main_queue(), ^{
  465. [subscriber sendError:serializationError];
  466. });
  467. #pragma clang diagnostic pop
  468. return [RACDisposable disposableWithBlock:^{
  469. }];
  470. }
  471. __block NSURLSessionDataTask *task = [self uploadTaskWithStreamedRequest:request progress:nil completionHandler:^(NSURLResponse * __unused response, id responseObject, NSError *error) {
  472. if (error) {
  473. NSError *parseError = [self _errorFromRequestWithTask:task httpResponse:(NSHTTPURLResponse *)response responseObject:responseObject error:error];
  474. [self HTTPRequestLog:task body:parameters error:parseError];
  475. [subscriber sendError:parseError];
  476. } else {
  477. /// 断言
  478. NSAssert([responseObject isKindOfClass:NSDictionary.class], @"responseObject is not an NSDictionary: %@", responseObject);
  479. /// 在这里判断数据是否正确
  480. /// 判断
  481. NSInteger statusCode = [responseObject[RQHTTPServiceResponseCodeKey] integerValue];
  482. if (statusCode == RQHTTPResponseCodeSuccess || statusCode == RQHTTPResponseCodeJSJPSuccess) {
  483. /// 打包成元祖 回调数据
  484. [subscriber sendNext:RACTuplePack(response , responseObject)];
  485. [subscriber sendCompleted];
  486. }else{
  487. if (statusCode == RQHTTPResponseCodeNotLogin) {
  488. /// 删除账号
  489. [SAMKeychain deleteRawLogin];
  490. /// 先退出用户
  491. [RQ_USER_MANAGER logoutUser];
  492. /// 延迟一段时间
  493. [[[[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
  494. [subscriber sendNext:@1];
  495. return nil;
  496. }] delay:1] deliverOnMainThread] subscribeNext:^(id x) {
  497. /// 这里切换 到账号登录的界面
  498. NSLog(@"%@",x);
  499. // RQ_USER_MANAGER.isShouldLogin;
  500. }];
  501. /// 需要登录
  502. // [self login:^{
  503. // /// 这里需要重新配置序列化
  504. // self.requestSerializer = [self _requestSerializerWithRequest:[RQHTTPRequest requestWithParameters:[RQURLParameters urlParametersWithMethod:@"POST" path:path parameters:parameters]]];
  505. // /// 重新发起请求
  506. // [self enqueueUploadRequestWithPath:path parameters:parameters constructingBodyWithBlock:block];
  507. // } cancel:^{
  508. // NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  509. // userInfo[RQHTTPServiceErrorResponseCodeKey] = @(statusCode);
  510. // NSString *msgTips = responseObject[RQHTTPServiceResponseMsgKey];
  511. //#if defined(DEBUG)||defined(_DEBUG)
  512. // msgTips = RQStringIsNotEmpty(msgTips)?[NSString stringWithFormat:@"%@(%zd)",msgTips,statusCode]:[NSString stringWithFormat:@"服务器出错了,请稍后重试(%zd)~",statusCode]; /// 调试模式
  513. //#else
  514. // msgTips = RQStringIsNotEmpty(msgTips)?msgTips:@"服务器出错了,请稍后重试~"; /// 发布模式
  515. //#endif
  516. // userInfo[RQHTTPServiceErrorMessagesKey] = msgTips;
  517. // if (task.currentRequest.URL != nil) userInfo[RQHTTPServiceErrorRequestURLKey] = task.currentRequest.URL.absoluteString;
  518. // if (task.error != nil) userInfo[NSUnderlyingErrorKey] = task.error;
  519. // [subscriber sendError:[NSError errorWithDomain:RQHTTPServiceErrorDomain code:statusCode userInfo:userInfo]];
  520. // }];
  521. }else{
  522. NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  523. userInfo[RQHTTPServiceErrorResponseCodeKey] = @(statusCode);
  524. NSString *msgTips = responseObject[RQHTTPServiceResponseMsgKey];
  525. #if defined(DEBUG)||defined(_DEBUG)
  526. msgTips = RQStringIsNotEmpty(msgTips)?[NSString stringWithFormat:@"%@(%zd)",msgTips,statusCode]:[NSString stringWithFormat:@"服务器出错了,请稍后重试(%zd)~",statusCode]; /// 调试模式
  527. #else
  528. msgTips = RQStringIsNotEmpty(msgTips)?msgTips:@"服务器出错了,请稍后重试~"; /// 发布模式
  529. #endif
  530. userInfo[RQHTTPServiceErrorMessagesKey] = msgTips;
  531. if (task.currentRequest.URL != nil) userInfo[RQHTTPServiceErrorRequestURLKey] = task.currentRequest.URL.absoluteString;
  532. if (task.error != nil) userInfo[NSUnderlyingErrorKey] = task.error;
  533. [subscriber sendError:[NSError errorWithDomain:RQHTTPServiceErrorDomain code:statusCode userInfo:userInfo]];
  534. }
  535. }
  536. }
  537. }];
  538. [task resume];
  539. return [RACDisposable disposableWithBlock:^{
  540. [task cancel];
  541. }];
  542. }];
  543. /// replayLazily:replayLazily会在第一次订阅的时候才订阅sourceSignal
  544. /// 会提供所有的值给订阅者 replayLazily还是冷信号 避免了冷信号的副作用
  545. return [[signal
  546. replayLazily]
  547. setNameWithFormat:@"-enqueueUploadRequestWithPath: %@ parameters: %@", path, parameters];
  548. }
  549. #pragma mark Parsing (数据解析)
  550. - (NSError *)parsingErrorWithFailureReason:(NSString *)localizedFailureReason {
  551. NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  552. userInfo[NSLocalizedDescriptionKey] = NSLocalizedString(@"Could not parse the service response.", @"");
  553. if (localizedFailureReason != nil) userInfo[NSLocalizedFailureReasonErrorKey] = localizedFailureReason;
  554. return [NSError errorWithDomain:RQHTTPServiceErrorDomain code:RQHTTPServiceErrorJSONParsingFailed userInfo:userInfo];
  555. }
  556. /// 解析数据
  557. - (RACSignal *)parsedResponseOfClass:(Class)resultClass fromJSON:(NSDictionary *)responseObject {
  558. /// 必须是RQBaseModel的子类 且 最外层responseObject必须是字典
  559. NSParameterAssert((resultClass == nil || [resultClass isSubclassOfClass:RQBaseModel.class]));
  560. /// 这里主要解析的是 data:对应的数据
  561. if ([responseObject.allKeys containsObject:RQHTTPServiceResponseDataKey] && !RQObjectIsNil(responseObject[RQHTTPServiceResponseDataKey])) {
  562. responseObject = responseObject[RQHTTPServiceResponseDataKey];
  563. } else if ([responseObject.allKeys containsObject:RQHTTPServiceResponseBodyKey] && !RQObjectIsNil(responseObject[RQHTTPServiceResponseBodyKey])) {
  564. responseObject = responseObject[RQHTTPServiceResponseBodyKey];
  565. }
  566. /// 解析
  567. return [RACSignal createSignal:^ id (id<RACSubscriber> subscriber) {
  568. /// 解析字典
  569. void (^parseJSONDictionary)(NSDictionary *) = ^(NSDictionary *JSONDictionary) {
  570. if (resultClass == nil) {
  571. [subscriber sendNext:JSONDictionary];
  572. return;
  573. }
  574. /// 这里继续取出数据 data{"list":[]}
  575. NSArray * JSONArray = JSONDictionary[RQHTTPServiceResponseDataListKey];
  576. if ([JSONArray isKindOfClass:[NSArray class]]) {
  577. /// 字典数组 转对应的模型
  578. NSArray *parsedObjects = [NSArray yy_modelArrayWithClass:resultClass.class json:JSONArray];
  579. /// 这里还需要解析是否是RQObject的子类
  580. for (id parsedObject in parsedObjects) {
  581. /// 确保解析出来的类 也是 RQObject
  582. ((RQBaseModel *)parsedObject).total = [JSONDictionary[RQHTTPServiceResponseDataTotalKey] integerValue];
  583. NSAssert([parsedObject isKindOfClass:RQBaseModel.class], @"Parsed model object is not an RQObject: %@", parsedObject);
  584. }
  585. [subscriber sendNext:parsedObjects];
  586. }else{
  587. /// 字典转模型
  588. RQBaseModel *parsedObject = [resultClass yy_modelWithDictionary:JSONDictionary];
  589. if (parsedObject == nil) {
  590. // Don't treat "no class found" errors as real parsing failures.
  591. // In theory, this makes parsing code forward-compatible with
  592. // API additions.
  593. // 模型解析失败
  594. NSError *error = [NSError errorWithDomain:@"" code:2222 userInfo:@{}];
  595. [subscriber sendError:error];
  596. return;
  597. }
  598. /// 确保解析出来的类 也是 BaseModel
  599. NSAssert([parsedObject isKindOfClass:RQBaseModel.class], @"Parsed model object is not an BaseModel: %@", parsedObject);
  600. /// 发送数据
  601. [subscriber sendNext:parsedObject];
  602. }
  603. };
  604. if ([responseObject isKindOfClass:NSArray.class]) {
  605. if (resultClass == nil) {
  606. [subscriber sendNext:responseObject];
  607. }else{
  608. /// 数组 保证数组里面装的是同一种 NSDcitionary
  609. for (NSDictionary *JSONDictionary in responseObject) {
  610. if (![JSONDictionary isKindOfClass:NSDictionary.class]) {
  611. NSString *failureReason = [NSString stringWithFormat:NSLocalizedString(@"Invalid JSON array element: %@", @""), JSONDictionary];
  612. [subscriber sendError:[self parsingErrorWithFailureReason:failureReason]];
  613. return nil;
  614. }
  615. }
  616. /// 字典数组 转对应的模型
  617. NSArray *parsedObjects = [NSArray yy_modelArrayWithClass:resultClass.class json:responseObject];
  618. /// 这里还需要解析是否是RQObject的子类
  619. for (id parsedObject in parsedObjects) {
  620. /// 确保解析出来的类 也是 BaseModel
  621. NSAssert([parsedObject isKindOfClass:RQBaseModel.class], @"Parsed model object is not an BaseModel: %@", parsedObject);
  622. }
  623. [subscriber sendNext:parsedObjects];
  624. }
  625. [subscriber sendCompleted];
  626. } else if ([responseObject isKindOfClass:NSDictionary.class]) {
  627. /// 解析字典
  628. parseJSONDictionary(responseObject);
  629. [subscriber sendCompleted];
  630. } else if (responseObject == nil || [responseObject isKindOfClass:[NSNull class]]) {
  631. [subscriber sendNext:nil];
  632. [subscriber sendCompleted];
  633. } else if ([responseObject isKindOfClass:[NSString class]]) {
  634. /// RQ-MARK: 如果responseObject是NSString类型进入该条件
  635. [subscriber sendNext:responseObject];
  636. [subscriber sendCompleted];
  637. } else if ([responseObject isKindOfClass:[NSNumber class]]) {
  638. /// RQ-MARK: 如果responseObject是NSNumber类型进入该条件
  639. [subscriber sendNext:responseObject];
  640. [subscriber sendCompleted];
  641. } else {
  642. NSString *failureReason = [NSString stringWithFormat:NSLocalizedString(@"Response wasn't an array or dictionary (%@): %@", @""), [responseObject class], responseObject];
  643. [subscriber sendError:[self parsingErrorWithFailureReason:failureReason]];
  644. }
  645. return nil;
  646. }];
  647. }
  648. #pragma mark - Error Handling
  649. /// 请求错误解析
  650. - (NSError *)_errorFromRequestWithTask:(NSURLSessionTask *)task httpResponse:(NSHTTPURLResponse *)httpResponse responseObject:(NSDictionary *)responseObject error:(NSError *)error {
  651. /// 不一定有值,则HttpCode = 0;
  652. NSInteger HTTPCode = httpResponse.statusCode;
  653. NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  654. /// default errorCode is RQHTTPServiceErrorConnectionFailed,意味着连接不上服务器
  655. NSInteger errorCode = RQHTTPServiceErrorConnectionFailed;
  656. NSString *errorDesc = @"服务器出错了,请稍后重试~";
  657. /// 其实这里需要处理后台数据错误,一般包在 responseObject
  658. /// HttpCode错误码解析 https://www.guhei.net/post/jb1153
  659. /// 1xx : 请求消息 [100 102]
  660. /// 2xx : 请求成功 [200 206]
  661. /// 3xx : 请求重定向[300 307]
  662. /// 4xx : 请求错误 [400 417] 、[422 426] 、449、451
  663. /// 5xx 、600: 服务器错误 [500 510] 、600
  664. NSInteger httpFirstCode = HTTPCode/100;
  665. if (httpFirstCode>0) {
  666. if (httpFirstCode==4) {
  667. /// 请求出错了,请稍后重试
  668. if (HTTPCode == 408) {
  669. #if defined(DEBUG)||defined(_DEBUG)
  670. errorDesc = @"请求超时,请稍后再试(408)~"; /// 调试模式
  671. #else
  672. errorDesc = @"请求超时,请稍后再试~"; /// 发布模式
  673. #endif
  674. }else{
  675. #if defined(DEBUG)||defined(_DEBUG)
  676. errorDesc = [NSString stringWithFormat:@"请求出错了,请稍后重试(%zd)~",HTTPCode]; /// 调试模式
  677. #else
  678. errorDesc = @"请求出错了,请稍后重试~"; /// 发布模式
  679. #endif
  680. }
  681. }else if (httpFirstCode == 5 || httpFirstCode == 6){
  682. /// 服务器出错了,请稍后重试
  683. #if defined(DEBUG)||defined(_DEBUG)
  684. errorDesc = [NSString stringWithFormat:@"服务器出错了,请稍后重试(%zd)~",HTTPCode]; /// 调试模式
  685. #else
  686. errorDesc = @"服务器出错了,请稍后重试~"; /// 发布模式
  687. #endif
  688. }else if (!self.reachabilityManager.isReachable){
  689. /// 网络不给力,请检查网络
  690. errorDesc = @"网络开小差了,请稍后重试~";
  691. }
  692. }else{
  693. if (!self.reachabilityManager.isReachable){
  694. /// 网络不给力,请检查网络
  695. errorDesc = @"网络开小差了,请稍后重试~";
  696. }
  697. }
  698. switch (HTTPCode) {
  699. case 400:{
  700. errorCode = RQHTTPServiceErrorBadRequest; /// 请求失败
  701. break;
  702. }
  703. case 403:{
  704. errorCode = RQHTTPServiceErrorRequestForbidden; /// 服务器拒绝请求
  705. break;
  706. }
  707. case 422:{
  708. errorCode = RQHTTPServiceErrorServiceRequestFailed; /// 请求出错
  709. break;
  710. }
  711. default:
  712. /// 从error中解析
  713. if ([error.domain isEqual:NSURLErrorDomain]) {
  714. #if defined(DEBUG)||defined(_DEBUG)
  715. errorDesc = [NSString stringWithFormat:@"请求出错了,请稍后重试(%zd)~",error.code]; /// 调试模式
  716. #else
  717. errorDesc = @"请求出错了,请稍后重试~"; /// 发布模式
  718. #endif
  719. switch (error.code) {
  720. case NSURLErrorSecureConnectionFailed:
  721. case NSURLErrorServerCertificateHasBadDate:
  722. case NSURLErrorServerCertificateHasUnknownRoot:
  723. case NSURLErrorServerCertificateUntrusted:
  724. case NSURLErrorServerCertificateNotYetValid:
  725. case NSURLErrorClientCertificateRejected:
  726. case NSURLErrorClientCertificateRequired:
  727. errorCode = RQHTTPServiceErrorSecureConnectionFailed; /// 建立安全连接出错了
  728. break;
  729. case NSURLErrorTimedOut:{
  730. #if defined(DEBUG)||defined(_DEBUG)
  731. errorDesc = @"请求超时,请稍后再试(-1001)~"; /// 调试模式
  732. #else
  733. errorDesc = @"请求超时,请稍后再试~"; /// 发布模式
  734. #endif
  735. break;
  736. }
  737. case NSURLErrorNotConnectedToInternet:{
  738. #if defined(DEBUG)||defined(_DEBUG)
  739. errorDesc = @"网络开小差了,请稍后重试(-1009)~"; /// 调试模式
  740. #else
  741. errorDesc = @"网络开小差了,请稍后重试~"; /// 发布模式
  742. #endif
  743. break;
  744. }
  745. }
  746. }
  747. }
  748. userInfo[RQHTTPServiceErrorHTTPStatusCodeKey] = @(HTTPCode);
  749. userInfo[RQHTTPServiceErrorDescriptionKey] = errorDesc;
  750. if (task.currentRequest.URL != nil) userInfo[RQHTTPServiceErrorRequestURLKey] = task.currentRequest.URL.absoluteString;
  751. if (task.error != nil) userInfo[NSUnderlyingErrorKey] = task.error;
  752. return [NSError errorWithDomain:RQHTTPServiceErrorDomain code:errorCode userInfo:userInfo];
  753. }
  754. #pragma mark - 打印请求日志
  755. - (void)HTTPRequestLog:(NSURLSessionTask *)task body:params error:(NSError *)error {
  756. NSLog(@">>>>>>>>>>>>>>>>>>>>>👇 REQUEST FINISH 👇>>>>>>>>>>>>>>>>>>>>>>>>>>");
  757. NSLog(@"Request%@=======>:%@", error?@"失败":@"成功", task.currentRequest.URL.absoluteString);
  758. NSLog(@"requestBody======>:%@", params);
  759. NSLog(@"requstHeader=====>:%@", task.currentRequest.allHTTPHeaderFields);
  760. NSLog(@"response=========>:%@", task.response);
  761. NSLog(@"error============>:%@", error);
  762. NSLog(@"<<<<<<<<<<<<<<<<<<<<<👆 REQUEST FINISH 👆<<<<<<<<<<<<<<<<<<<<<<<<<<");
  763. #if defined(DEBUG)||defined(_DEBUG)
  764. if([QMUIConsole sharedInstance].canShow){
  765. NSMutableString *log = [NSMutableString stringWithFormat:@">>>>>>>>>>>>>>>>>>>>>👇 REQUEST FINISH 👇>>>>>>>>>>>>>>>>>>>>>>>>>>"];
  766. [log appendFormat:@"Request%@=======>:%@", error?@"失败":@"成功", task.currentRequest.URL.absoluteString];
  767. [log appendFormat:@"requestBody======>:%@", params];
  768. [log appendFormat:@"requstHeader=====>:%@", task.currentRequest.allHTTPHeaderFields];
  769. [log appendFormat:@"response=========>:%@", task.response];
  770. [log appendFormat:@"error============>:%@", error];
  771. [log appendString:@"<<<<<<<<<<<<<<<<<<<<<👆 REQUEST FINISH 👆<<<<<<<<<<<<<<<<<<<<<<<<<<"];
  772. [QMUIConsole log:log];
  773. }
  774. #endif
  775. }
  776. #pragma mark - Parameter 签名 MD5 生成一个 sign ,这里请根据实际项目来定
  777. /// 基础的请求参数
  778. - (NSMutableDictionary *)_parametersWithRequest:(RQHTTPRequest *)request{
  779. NSMutableDictionary *parameters = [NSMutableDictionary dictionary];
  780. /// 模型转字典
  781. NSDictionary *extendsUrlParams = [request.urlParameters.extendsParameters mj_keyValues].copy;
  782. if ([extendsUrlParams count]) {
  783. [parameters addEntriesFromDictionary:extendsUrlParams];
  784. }
  785. if ([request.urlParameters.parameters count]) {
  786. [parameters addEntriesFromDictionary:request.urlParameters.parameters];
  787. }
  788. return parameters;
  789. }
  790. /// 带签名的请求参数
  791. - (NSString *)_signWithParameters:(NSDictionary *) parameters {
  792. /// 按照ASCII码排序
  793. NSArray *sortedKeys = [[parameters allKeys] sortedArrayUsingSelector:@selector(compare:)];
  794. NSMutableArray *kvs = [NSMutableArray array];
  795. for (id key in sortedKeys) {
  796. /// value 为 empty 跳过
  797. if(RQObjectIsNil(parameters[key])) continue;
  798. NSString * value = [parameters[key] rq_stringValueExtension];
  799. // if (RQObjectIsNil(value)||!RQStringIsNotEmpty(value)) continue;
  800. value = [value rq_removeBothEndsWhitespaceAndNewline];
  801. // value = [value rq_URLEncoding];
  802. [kvs addObject:[NSString stringWithFormat:@"%@=%@",key,value]];
  803. }
  804. NSString *paramString = [kvs componentsJoinedByString:@"&"];
  805. /// 拼接私钥
  806. // NSString *keyValue = RQHTTPServiceKeyValue;
  807. // NSString *signedString = [NSString stringWithFormat:@"%@&%@=%@",paramString,RQHTTPServiceKey,keyValue];
  808. /// 拼接时间戳
  809. NSString *ts = [NSString stringWithFormat:@"%.f", [NSDate date].timeIntervalSince1970 * 1000];
  810. NSString *sign = [NSString stringWithFormat:@"%@&key=%@",paramString ,ts];
  811. /// md5
  812. sign = [CocoaSecurity md5:sign].hex;
  813. NSString *signedStr = [NSString stringWithFormat:@"ts=%@&sign=%@&user=ios&v=jsjp",ts ,sign];
  814. return signedStr;
  815. }
  816. /// 序列化
  817. - (AFHTTPRequestSerializer *)_requestSerializerWithRequest:(RQHTTPRequest *) request {
  818. /// 获取基础参数(参数+拓展参数)
  819. NSMutableDictionary *parameters = [self _parametersWithRequest:request];
  820. /// 获取带签名的参数
  821. NSString *sign = [self _signWithParameters:parameters];
  822. /// 赋值
  823. parameters[RQHTTPServiceSignKey] = [sign length]?sign:@"";
  824. /// 请求序列化
  825. AFHTTPRequestSerializer *requestSerializer = [AFHTTPRequestSerializer serializer];
  826. /// 配置请求头
  827. for (NSString *key in parameters) {
  828. NSString *value = [[parameters[key] rq_stringValueExtension] copy];
  829. if (value.length==0) continue;
  830. /// value只能是字符串,否则崩溃
  831. [requestSerializer setValue:value forHTTPHeaderField:key];
  832. }
  833. return requestSerializer;
  834. }
  835. - (AFJSONRequestSerializer *)_requestJSONSerializerWithRequest:(RQHTTPRequest *) request {
  836. /// 获取基础参数(参数+拓展参数)
  837. NSMutableDictionary *parameters = [self _parametersWithRequest:request];
  838. /// 获取带签名的参数
  839. NSString *sign = [self _signWithParameters:parameters];
  840. /// 赋值
  841. parameters[RQHTTPServiceSignKey] = [sign length]?sign:@"";
  842. /// 请求序列化
  843. AFJSONRequestSerializer *requestSerializer = [AFJSONRequestSerializer serializer];
  844. /// 配置请求头
  845. if ([request.urlParameters.path containsString:@"jsjp-admin.zzxcx.net"] || [request.urlParameters.path containsString:@"jsjp-admin1.zzxcx.net"] || [request.urlParameters.path containsString:@"192.168.8.63:8080"]) {
  846. } else {
  847. for (NSString *key in parameters) {
  848. NSString *value = [[parameters[key] rq_stringValueExtension] copy];
  849. if (value.length==0) continue;
  850. /// value只能是字符串,否则崩溃
  851. [requestSerializer setValue:value forHTTPHeaderField:key];
  852. }
  853. }
  854. return requestSerializer;
  855. }
  856. @end