RQHTTPService.m 48 KB

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