RQHTTPService.m 48 KB

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