nasMixUploadManager.m 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963
  1. //
  2. // nasMixUploadManager.m
  3. //
  4. //
  5. // Created by David on 2024/8/27.
  6. //
  7. #import "nasMixUploadManager.h"
  8. #import "customUploadOperation.h"
  9. #import "nasUploadFileManager.h"
  10. #import "frpFileExistModel.h"
  11. #import "frpUploadModel.h"
  12. #define Kboundary2 @"Boundaryhxd"
  13. @interface nasMixUploadManager ()<NSURLSessionDataDelegate>
  14. //排队等候下载的上传地址数组
  15. @property(nonatomic,strong) NSMutableArray *uploadWaitingUrlArr;
  16. //正在下载的上传地址数组
  17. @property(nonatomic,strong) NSMutableArray *uploadingOperationArr;
  18. @property(nonatomic,strong)NSLock *arrayLock;//数组遍历是修改会挂掉
  19. @end
  20. @implementation nasMixUploadManager
  21. + (instancetype)shareManager {
  22. static nasMixUploadManager *_instance;
  23. static dispatch_once_t onceToken;
  24. dispatch_once(&onceToken, ^{
  25. _instance = [[self alloc] init];
  26. });
  27. return _instance;
  28. }
  29. - (instancetype)init {
  30. if (self = [super init]) {
  31. _maxUploadLoadCount = 2;
  32. _arrayLock = [NSLock new];
  33. //[self registeNotification];
  34. }
  35. return self;
  36. }
  37. /** 添加要上传的 模型 */
  38. - (void)addUploadWithModels:(NSArray *)fileModels{
  39. HLog(@"添加任务-- %ld",fileModels.count)
  40. for (uploadFileDataModel *model in fileModels) {
  41. BOOL needAddType = YES;
  42. //1. 排查上传中
  43. for (customUploadOperation *operationDoing in self.uploadingOperationArr) {
  44. if([operationDoing.fileModel.filename isEqualToString:model.filename]){
  45. needAddType = NO;
  46. break;
  47. }
  48. }
  49. //1. 排查等待下载
  50. for (uploadFileDataModel *waitModel in self.uploadWaitingUrlArr) {
  51. if([waitModel.filename isEqualToString:model.filename]){
  52. needAddType = NO;
  53. break;
  54. }
  55. }
  56. if(needAddType){
  57. [self.uploadWaitingUrlArr addObject:model];
  58. }
  59. }
  60. //启动上传
  61. [self beginUploadAfterDeleteOperationBy:nil];
  62. }
  63. //在添加XX后 启动下载
  64. - (void)beginUploadAfterDeleteOperationBy:(customUploadOperation*)operation
  65. {
  66. HLog(@"beginUploadAfterDeleteOperationBy")
  67. BOOL isCanUseCellular = [HWDataManager getBoolWithKey:stringKeyAddSn(Const_file_Transfe_canUse_Cellular_all)];
  68. if(!isCanUseCellular){//不允许流量上传
  69. //
  70. if([AFNetworkReachabilityManager sharedManager].networkReachabilityStatus == AFNetworkReachabilityStatusReachableViaWWAN){
  71. mainBlock(^{
  72. [[NSNotificationCenter defaultCenter] postNotificationName:uploadFileSuspendAllNotification object:nil];
  73. [[iToast makeText:NSLocalizedString(@"File_Transfer_By_Cellular_tip",nil)] show];
  74. });
  75. return;
  76. }
  77. }
  78. // if(ksharedAppDelegate.DisabledFileTransferType){
  79. // if(ksharedAppDelegate.isImageNewFor130){
  80. // [[iToast makeText:NSLocalizedString(@"File_Transfer_Disable_tip",nil)] show];
  81. // }
  82. // else{
  83. // [[iToast makeText:NSLocalizedString(@"File_Transfer_Disable_tip2",nil)] show];
  84. // }
  85. // return;
  86. // }
  87. if(operation){
  88. HLog(@"删除完成的上传任务")
  89. [_arrayLock lock];
  90. if(operation.fileHandle){
  91. [operation closeFileHandleFun];
  92. }
  93. [self.uploadingOperationArr removeObject:operation];
  94. [_arrayLock unlock];
  95. }
  96. if(self.uploadingOperationArr.count == _maxUploadLoadCount){
  97. HLog(@"正在上传的数量达到了最大值 %ld",_maxUploadLoadCount)
  98. return;
  99. }
  100. if(self.uploadWaitingUrlArr.count == 0){
  101. HLog(@"没有等待中的上传任务")
  102. return;
  103. }
  104. NSInteger canAddTaskNumber = _maxUploadLoadCount - self.uploadingOperationArr.count;
  105. //新添加的任务
  106. NSMutableArray *newTaskArr = [NSMutableArray new];
  107. [_arrayLock lock];
  108. for (int i=0; i<canAddTaskNumber; i++) {
  109. if(self.uploadWaitingUrlArr.count >= 1){
  110. //创建上传任务
  111. uploadFileDataModel *WaitingModel = self.uploadWaitingUrlArr.firstObject;
  112. customUploadOperation * operation = [customUploadOperation new];
  113. operation.fileModel = WaitingModel;
  114. //等待下载中的任务
  115. [self.uploadWaitingUrlArr removeObjectAtIndex:0];
  116. HLog(@"添加一个新的上传任务 %@",self.uploadingOperationArr)
  117. //添加到下载中数组
  118. [self.uploadingOperationArr addObject:operation];
  119. //[self checkFileUploadStateWithOperation:operation];
  120. [newTaskArr addObject:operation];
  121. //[weakSelf handleTaskDidUploadWith:operation withState:state];
  122. }
  123. }
  124. //开始检测新的上传任务
  125. for (customUploadOperation * operation in newTaskArr) {
  126. [self checkFileUploadStateWithOperation:operation];
  127. }
  128. [newTaskArr removeAllObjects];
  129. [_arrayLock unlock];
  130. }
  131. #pragma mark 检测文件是否上传过了
  132. - (void)checkFileUploadStateWithOperation:(customUploadOperation*)operation
  133. {
  134. NSMutableDictionary*paraDict = [NSMutableDictionary new];
  135. //operation.fileModel.filename = @"Img_4107.png";// test same name file
  136. if(operation.fileModel.savePath){
  137. NSString *absPath = [[NSString alloc] initWithFormat:@"%@%@",operation.fileModel.savePath,operation.fileModel.filename];
  138. [paraDict setValue:absPath forKey:@"absPath"];
  139. NSNumber *totalBytesNumber = [NSNumber numberWithLong:operation.fileModel.totalBytes];
  140. [paraDict setValue:totalBytesNumber forKey:@"fileSize"];
  141. }
  142. KWeakSelf
  143. [[netWorkManager shareInstance] cloudPhoneGETCallBackCode:@"isFileExist" Parameters:paraDict success:^(id _Nonnull responseObject) {
  144. frpFileExistModel *model = [[frpFileExistModel alloc] initWithDictionary:responseObject error:nil];
  145. if(model && model.status == 0){
  146. [weakSelf checkFileUploadStateFunAfterNetWith:model WithOperation:operation];
  147. }
  148. else if(model && model.status == 5){//云机空间不足
  149. //上传路径
  150. NSString * backupsDefaultPath = [HWDataManager getStringWithKey:stringKeyAddSn(Const_photo_upload_default_path)];
  151. NSArray * stringArr = [backupsDefaultPath componentsSeparatedByString:@"/"];
  152. NSString *lastStr = stringArr.lastObject;
  153. NSString *repStr = [[NSString alloc] initWithFormat:@"/%@",lastStr];
  154. NSString *name = [backupsDefaultPath stringByReplacingOccurrencesOfString:repStr withString:@""];
  155. name = [iTools changePathToShowPathBy:name];
  156. NSString* showTip = [[NSString alloc] initWithFormat:@"【%@】%@",name,NSLocalizedString(@"disk_space_not_tip_for_95",nil)];
  157. [[iToast makeText:showTip] show];
  158. [weakSelf handleUploadFailOneFileBy:operation];
  159. }
  160. else{
  161. HLog(@"isFileExist接口异常:%@",responseObject)
  162. [weakSelf handleUploadFailOneFileBy:operation];
  163. }
  164. } failure:^(NSError * _Nonnull error) {
  165. HLog(@"%@",error)
  166. [weakSelf handleUploadFailOneFileBy:operation];
  167. }];
  168. }
  169. #pragma mark 检测分家是否存在后处理 是否要上传
  170. - (void)checkFileUploadStateFunAfterNetWith:(frpFileExistModel*)model WithOperation:(customUploadOperation*)operation
  171. {
  172. if(!model){
  173. [self handleUploadFailOneFileBy:operation];
  174. return;
  175. }
  176. operation.fileModel.didUploadBytes = 0;
  177. operation.fileModel.taskId = model.data.fileTaskId;
  178. if(!model.data.exist){//未上传过
  179. [self prepareToUploadFileWithOperation:operation];
  180. }
  181. else if(model.data.isComplete){//上传过了 并且文件上传完了
  182. //判断下文件创建长度是否一致 一致则是上传完了 不一致 重新上传一个 可能是同名的文件而已
  183. // if(model.data.size >= operation.fileModel.totalBytes){//上传完了
  184. // [self handleFileDidUploadFunWithOperation:operation];
  185. // }
  186. // else{//未上传完 isComplete 这个字段有问题
  187. // operation.fileModel.didUploadBytes = model.data.size;
  188. // [self prepareToUploadFileWithOperation:operation];
  189. // }
  190. //修复同名文件丢失问题
  191. if(model.data.size == operation.fileModel.totalBytes){//上传完了
  192. [self handleFileDidUploadFunWithOperation:operation];
  193. }
  194. else{//同名文件 处理
  195. [self prepareToUploadFileWithOperation:operation];
  196. }
  197. }
  198. else{//上传过了 未上传完成
  199. operation.fileModel.didUploadBytes = model.data.size;
  200. [self prepareToUploadFileWithOperation:operation];
  201. }
  202. }
  203. #pragma mark 任务上传过了了
  204. - (void)handleFileDidUploadFunWithOperation:(customUploadOperation*)operation
  205. {
  206. mainBlock(^{
  207. [[iToast makeText:NSLocalizedString(@"File_upload_file_already_exists",nil)] show];
  208. });
  209. NSMutableArray *delArr = [NSMutableArray new];
  210. [delArr addObject:operation.fileModel];
  211. [[nasUploadFileManager shareInstance] deleteUploadFileRecordBy:delArr withDelCache:YES complete:^(BOOL isSuccess) {
  212. if (isSuccess) {
  213. }
  214. }];
  215. }
  216. #pragma mark 准备上传文件
  217. - (void)prepareToUploadFileWithOperation:(customUploadOperation*)operation
  218. {
  219. NSMutableDictionary *paraDict = [NSMutableDictionary new];
  220. NSString* taskUid = operation.fileModel.taskId;
  221. if(!taskUid || taskUid.length == 0){
  222. taskUid = [iTools getTaskUidStr];
  223. operation.fileModel.taskId = taskUid;
  224. }
  225. [paraDict setObject:taskUid forKey:@"taskId"];
  226. [paraDict setObject:@0 forKey:@"position"];
  227. [paraDict setObject:@"true" forKey:@"isLast"];
  228. if(operation.fileModel.savePath){
  229. [paraDict setObject:operation.fileModel.savePath forKey:@"savePath"];
  230. }
  231. else{
  232. HLog(@"获取保存路径失败")
  233. [self handleUploadFailOneFileBy:operation];
  234. return;
  235. }
  236. if(operation.fileModel.filename){
  237. [paraDict setObject:operation.fileModel.filename forKey:@"filename"];
  238. }
  239. else{
  240. HLog(@"获取用户名失败")
  241. return;
  242. }
  243. KWeakSelf
  244. if(operation.fileModel.curUploadFileType == uploadFileTypeImage){
  245. [paraDict setObject:@1 forKey:@"imageType"];
  246. if(!operation.fileModel.imageData){
  247. NSString*pathStr = [cachesFileManager getFilePathWithName:operation.fileModel.filename type:operation.fileModel.curUploadFileType];
  248. NSData *imageData = [NSData dataWithContentsOfFile:pathStr];
  249. //修复出现过 图片数据小于totalBytes
  250. if(imageData){
  251. long curDataLengt = [imageData length];
  252. if(curDataLengt == operation.fileModel.didUploadBytes && curDataLengt < operation.fileModel.totalBytes){
  253. HLog(@"totalBytes 获取出错?")
  254. [self handleFileDidUploadFunWithOperation:operation];
  255. return;
  256. }
  257. }
  258. if (!imageData) {
  259. if(!operation.fileModel.asset){
  260. NSString *curLocalIdentifier = operation.fileModel.localIdentifier;
  261. PHFetchResult *fetchResult = [PHAsset fetchAssetsWithLocalIdentifiers:@[curLocalIdentifier] options:nil];
  262. PHAsset *asset = fetchResult.firstObject;
  263. operation.fileModel.asset = asset;
  264. }
  265. [[PHImageManager defaultManager] requestImageDataForAsset:operation.fileModel.asset options:nil resultHandler:^(NSData * _Nullable imageData, NSString * _Nullable dataUTI, UIImageOrientation orientation, NSDictionary * _Nullable info) {
  266. // 直接得到最终的 NSData 数据
  267. if (imageData) {
  268. operation.fileModel.imageData = imageData;
  269. if(operation.fileModel.totalBytes == 0){
  270. operation.fileModel.totalBytes = [imageData length];
  271. }
  272. [weakSelf afterGetImageDataFunWithOperation:operation];
  273. }
  274. else{
  275. HLog(@"\n\n\n\nrequestImageDataForAsset fail imageData nil\n\n\n")
  276. [weakSelf handleUploadFailOneFileBy:operation];
  277. }
  278. }];
  279. return;
  280. }
  281. operation.fileModel.imageData = imageData;
  282. }
  283. NSData *curData = operation.fileModel.imageData;
  284. operation.onceDataLengt = [curData length];
  285. if(operation.fileModel.totalBytes == 0){//异常处理
  286. operation.fileModel.totalBytes = operation.onceDataLengt;
  287. }
  288. [self startUpload:paraDict operation:operation data:curData success:^(id _Nonnull responseObject) {
  289. frpUploadModel *model = [[frpUploadModel alloc] initWithDictionary:responseObject error:nil];
  290. if(model && model.position == operation.fileModel.totalBytes && model.position != 0){
  291. HLog(@"%@上传完成 000",operation.fileModel.filename)
  292. [weakSelf handleUploadDoneOneFileBy:operation];
  293. }
  294. else{
  295. HLog(@"%@上传完成异常 %ld---%ld",operation.fileModel.filename,model.position,operation.fileModel.totalBytes)
  296. [weakSelf handleUploadDoneOneFileBy:operation];
  297. }
  298. } faild:^(NSError * _Nonnull error) {
  299. HLog(@"%@上传失败",operation.fileModel.filename)
  300. [weakSelf handleUploadFailOneFileBy:operation];
  301. }];
  302. }
  303. else{
  304. [paraDict setObject:@1 forKey:@"videoType"];
  305. [paraDict setObject:@"false" forKey:@"isLast"];
  306. if(![cachesFileManager checkFileIsSaveState:operation.fileModel.filename withType:uploadFileTypeVideo]){
  307. if(!operation.fileModel.asset){
  308. NSString *curLocalIdentifier = operation.fileModel.localIdentifier;
  309. PHFetchResult *fetchResult = [PHAsset fetchAssetsWithLocalIdentifiers:@[curLocalIdentifier] options:nil];
  310. PHAsset *asset = fetchResult.firstObject;
  311. operation.fileModel.asset = asset;
  312. }
  313. //真正的视频数据
  314. PHVideoRequestOptions *options = [[PHVideoRequestOptions alloc] init];
  315. options.version = PHVideoRequestOptionsVersionOriginal;
  316. [[PHImageManager defaultManager] requestAVAssetForVideo:operation.fileModel.asset options:options resultHandler:^(AVAsset *asset, AVAudioMix *audioMix, NSDictionary *info) {
  317. if ([asset isKindOfClass:[AVURLAsset class]]) {
  318. AVURLAsset* urlAsset = (AVURLAsset*)asset;
  319. BOOL isSuc = [cachesFileManager copyVideoItemAtPath:[urlAsset.URL path] fileName:operation.fileModel.filename error:nil];
  320. if (isSuc) {
  321. [weakSelf afterGetVideoDataFunWithOperation:operation];
  322. }
  323. else{
  324. [weakSelf handleUploadFailOneFileBy:operation];
  325. }
  326. }
  327. else{
  328. [weakSelf handleUploadFailOneFileBy:operation];
  329. }
  330. }];
  331. return;
  332. }
  333. long curPosition = operation.fileModel.didUploadBytes;
  334. [self beginUploadVideoDataFunBy:operation with:curPosition withPara:paraDict success:^(id _Nonnull responseObject) {
  335. } faild:^(NSError * _Nonnull error) {
  336. }];
  337. }
  338. }
  339. #pragma mark 视频上传
  340. - (void)beginUploadVideoDataFunBy:(customUploadOperation*)operation with:(NSInteger)position withPara:(NSMutableDictionary*)paraDict success:(netWork_Success)success faild:(netWork_Faild)faildStr
  341. {
  342. if(operation.isCancelType){
  343. HLog(@"防止取消不了任务生效")
  344. return;
  345. }
  346. BOOL isLastPicece = NO;
  347. if((operation.fileModel.totalBytes - position) <= MaxNasUploadPieceSzie){
  348. [paraDict setObject:@"true" forKey:@"isLast"];
  349. isLastPicece = YES;
  350. }
  351. else{
  352. [paraDict setObject:@"false" forKey:@"isLast"];
  353. }
  354. [paraDict setObject:[NSNumber numberWithLong:position] forKey:@"position"];
  355. //视频数据切片
  356. __block NSData *videoData = [self cutVideoFileFunAtIndex:position withMaxLenght:MaxNasUploadPieceSzie withModel:operation.fileModel withOperation:operation];
  357. if(!videoData ||videoData.length ==0){
  358. HLog(@"视频没获取到")
  359. [self handleUploadFailOneFileBy:operation];
  360. return;
  361. }
  362. operation.onceDataLengt = [videoData length];
  363. KWeakSelf
  364. [self startUpload:paraDict operation:operation data:videoData success:^(id _Nonnull responseObject) {
  365. frpUploadModel *model = [[frpUploadModel alloc] initWithDictionary:responseObject error:nil];
  366. if(model && model.position >= operation.fileModel.totalBytes){
  367. HLog(@"%@上传完成 001",operation.fileModel.filename)
  368. [weakSelf handleUploadDoneOneFileBy:operation];
  369. }
  370. else{
  371. HLog(@"%@上传完成一片 %ld",operation.fileModel.filename,model.position)
  372. [weakSelf beginUploadVideoDataFunBy:operation with:model.position withPara:paraDict success:^(id _Nonnull responseObject) {
  373. success(responseObject);
  374. } faild:^(NSError * _Nonnull error) {
  375. NSError *err = error;
  376. if(error.code != -999){
  377. faildStr(err);
  378. }
  379. }];
  380. }
  381. } faild:^(NSError * _Nonnull error) {
  382. HLog(@"%@上传失败",operation.fileModel.filename)
  383. [weakSelf handleUploadFailOneFileBy:operation];
  384. }];
  385. }
  386. #pragma mark 分段读视频文件
  387. -(NSData*)cutVideoFileFunAtIndex:(NSUInteger)dataIndex withMaxLenght:(NSInteger)maxLengt withModel:(uploadFileDataModel*)dataModel withOperation:(customUploadOperation*)operation{
  388. //return [operation cutVideoFileFunAtIndex:dataIndex withMaxLenght:maxLengt];
  389. HLog(@"视频切片开始 线程:%@",[NSThread currentThread]);
  390. NSString *filePath = [cachesFileManager getFilePathWithName:dataModel.filename type:uploadFileTypeVideo]; // 文件路径
  391. NSFileManager *manager0 = [NSFileManager defaultManager];
  392. if(![manager0 fileExistsAtPath:filePath]) {
  393. return [NSData new];
  394. }
  395. NSFileHandle *fileHandle = [NSFileHandle fileHandleForReadingAtPath:filePath]; // 创建文件句柄
  396. // 设置分段读取的大小,这里以每次读取1KB为例
  397. const NSUInteger chunkSize = maxLengt;//cutVideoPieceSzie;//5 * 1024 *1024;
  398. NSMutableData *data = [NSMutableData data];
  399. if (fileHandle) {
  400. long long endOfFile = [fileHandle seekToEndOfFile];
  401. if(dataModel.totalBytes == 0
  402. || dataModel.totalBytes < endOfFile){//异常处理
  403. dataModel.totalBytes = endOfFile;
  404. }
  405. //异常处理
  406. if(endOfFile == dataIndex){
  407. dataModel.totalBytes = endOfFile;
  408. dataModel.didUploadBytes = endOfFile;
  409. dataModel.curUploadStateType = uploadStateDone;
  410. [fileHandle closeFile];
  411. return data;
  412. }
  413. if (endOfFile >= chunkSize) {
  414. // 读取文件的分段数据到某个位置
  415. [fileHandle seekToFileOffset:dataIndex];
  416. // 读取文件的分段数据
  417. NSData* chunk = [fileHandle readDataOfLength:chunkSize];
  418. if (chunk) {
  419. [data appendData:chunk];
  420. }
  421. }
  422. else{
  423. // 读取文件的分段数据到某个位置
  424. [fileHandle seekToFileOffset:dataIndex];
  425. [data appendData:[fileHandle readDataToEndOfFile]];
  426. }
  427. // 在这里可以对文件内容进行处理
  428. // ...
  429. // 关闭文件句柄
  430. [fileHandle closeFile];
  431. }
  432. HLog(@"视频切片完成 dataIndex:%ld --长度:%ld",dataIndex,[data length])
  433. return data;
  434. }
  435. #pragma mark 处理上传完成
  436. - (void)handleUploadDoneOneFileBy:(customUploadOperation*)operation
  437. {
  438. HLog(@"handleUploadDoneOneFileBy")
  439. [[NSNotificationCenter defaultCenter] postNotificationName:nasUploadTaskExeEnd object:operation.fileModel];
  440. [self beginUploadAfterDeleteOperationBy:operation];
  441. }
  442. #pragma mark 处理删除失败
  443. - (void)handleUploadFailOneFileBy:(customUploadOperation*)operation
  444. {
  445. operation.fileModel.curUploadStateType = uploadStateFail;
  446. [[NSNotificationCenter defaultCenter] postNotificationName:nasUploadTaskExeError object:operation.fileModel];
  447. [self beginUploadAfterDeleteOperationBy:operation];
  448. }
  449. #pragma mark 根据 asset 获取到图片数据
  450. - (void)afterGetImageDataFunWithOperation:(customUploadOperation*)operation
  451. {
  452. [cachesFileManager getFileNameWithContent:operation.fileModel.imageData fileName:operation.fileModel.filename type:uploadFileTypeImage];
  453. [self prepareToUploadFileWithOperation:operation];
  454. }
  455. #pragma mark 根据 asset 获取到视频数据
  456. - (void)afterGetVideoDataFunWithOperation:(customUploadOperation*)operation
  457. {
  458. [self prepareToUploadFileWithOperation:operation];
  459. }
  460. #pragma mark 开始上传
  461. - (void)startUpload:(NSMutableDictionary *)params operation:(customUploadOperation*)operation data:(NSData *)data success:(netWork_Success)success faild:(netWork_Faild)faildStr {
  462. NSString *urlString = ksharedAppDelegate.NASFileByBoxService;
  463. urlString = [[NSString alloc] initWithFormat:@"%@uploadFile",urlString];
  464. NSURL *URL = [NSURL URLWithString:urlString];
  465. NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:URL];
  466. [request setHTTPMethod:@"POST"];
  467. //请求体
  468. NSMutableData *bodyData = [self getBodyDataWithRequest:request withModel:operation.fileModel withData:data withPara:params];
  469. //设置请求体
  470. [request setHTTPMethod:@"POST"];
  471. [request setHTTPBody:bodyData];
  472. //设置请求体长度
  473. NSInteger length = [bodyData length];
  474. [request setValue:[NSString stringWithFormat:@"%ld",length] forHTTPHeaderField:@"Content-Length"];
  475. //设置 POST请求文件上传
  476. [request setValue:[NSString stringWithFormat:@"multipart/form-data; boundary=%@",Kboundary2] forHTTPHeaderField:@"Content-Type"];
  477. KWeakSelf
  478. //回话对象
  479. NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:nil];
  480. operation.session = session;
  481. //请求task
  482. /*
  483. 第一个参数:请求对象
  484. 第二个参数:传递是要上传的数据(请求体)
  485. 第三个参数:
  486. */
  487. NSURLSessionUploadTask *uploadTask = [session uploadTaskWithRequest:request fromData:nil completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
  488. //解析
  489. [weakSelf handleCustomUploadResultBy:data withResponse:response withError:error success:success faild:faildStr];
  490. }];
  491. // NSString *filePath = [cachesFileManager getFilePathWithName:dataModel.filename type:uploadFileTypeVideo]; // 文件路径
  492. // NSURL *filePathUrl = [NSURL URLWithString:filePath];
  493. //
  494. // NSURLSessionUploadTask *uploadTask = [session uploadTaskWithRequest:request fromFile:filePathUrl completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
  495. // HLog(@"data string:%@",[[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding]);
  496. //[weakSelf handleCustomUploadResultBy:data withResponse:response withError:error];
  497. // }];
  498. // NSURLSessionDataTask *uploadTask = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
  499. //
  500. // HLog(@"data string:%@",[[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding]);
  501. //[weakSelf handleCustomUploadResultBy:data withResponse:response withError:error];
  502. //
  503. //// NSJSONSerialization *object = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:nil];
  504. //// NSDictionary *dict = (NSDictionary *)object;
  505. //// NSLog(@"=====%@",[dict objectForKey:@"success"]);
  506. // }];
  507. //执行Task
  508. [uploadTask resume];
  509. operation.dataTask = uploadTask;
  510. }
  511. #pragma mark 用系统方法写的上传处理
  512. - (void)handleCustomUploadResultBy:(NSData*)data withResponse:(NSURLResponse*)response withError:(NSError*)error success:(netWork_Success)success faild:(netWork_Faild)faildStr{
  513. if(error){
  514. HLog(@"上传错误:%@",error)
  515. // -1005 网络中断 1009 网络似乎中断
  516. if(error.code == -1005
  517. ||error.code == -1009){//网络中断
  518. [[nasUploadFileManager shareInstance] saveUploadingTaskByNetWorkErrorFun];
  519. return;
  520. }
  521. if(error.code != -999){
  522. faildStr(error);
  523. }
  524. return;
  525. }
  526. NSJSONSerialization *object = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:nil];
  527. NSDictionary *dict = (NSDictionary *)object;
  528. frpUploadModel *model = [[frpUploadModel alloc] initWithDictionary:dict error:nil];
  529. if(model && [model.msg isEqualToString:@"success"]){
  530. success(dict);
  531. }
  532. else{
  533. NSError *err = [NSError new];
  534. faildStr(err);
  535. }
  536. }
  537. - (NSMutableData *)getBodyDataWithRequest:(NSMutableURLRequest *)request withModel:(uploadFileDataModel*)dataModel withData:(NSData*)data withPara:(NSMutableDictionary*)params{
  538. //1 边界符号要配置请求头里面去
  539. /*
  540. multipart/form-data 是表单格式
  541. charset=utf-8 是utf-8编码
  542. bounary 是表单开头
  543. */
  544. [request setValue:[NSString stringWithFormat:@"multipart/form-data; charset=utf-8; boundary=%@", Kboundary2] forHTTPHeaderField:@"Content-Type"];
  545. /// body
  546. NSMutableData *boydData = [NSMutableData data];
  547. // 2.1 边界符号(开始边界)
  548. //2.1.1 其它参数
  549. NSMutableString *paraString = [NSMutableString new];
  550. //[paraString appendFormat:@"--%@\r\n",Kboundary];//\n:换行 \n:切换到行首
  551. for (NSString *key in params) {
  552. NSString *value = params[key];
  553. [paraString appendFormat:@"--%@\r\n",Kboundary2];//\n:换行 \n:切换到行首
  554. [paraString appendFormat:@"Content-Disposition: form-data; name=\"%@\"",key];
  555. [paraString appendFormat:@"\r\n"];
  556. [paraString appendFormat:@"\r\n"];
  557. [paraString appendFormat:@"%@\r\n",value];
  558. }
  559. //[boydData appendData:[paraString dataUsingEncoding:NSUTF8StringEncoding]];
  560. // body每一个段内容以换行符作为结束标示
  561. NSString *fileBeginBoundary = [NSString stringWithFormat:@"--%@\r\n", Kboundary2];
  562. //[boydData appendData:[fileBeginBoundary dataUsingEncoding:NSUTF8StringEncoding]];
  563. [paraString appendString:fileBeginBoundary];
  564. // 2.2 属性配置 名字;key;类型
  565. NSString *serverFileKey = @"image"; //key
  566. //NSString *serverFileKey = @"file";
  567. NSString *serverContentTypes = @"image/png"; //类型
  568. if (dataModel.curUploadFileType == uploadFileTypeVideo) {
  569. serverFileKey = @"video";
  570. serverContentTypes = @"video/mp4";
  571. }
  572. NSString *serverFileName = dataModel.filename; //name
  573. // filename已命名文件; name相当于一个key, 这个名字和服务器保持一致
  574. /*
  575. 理解key,表单发送给服务端,服务端拿到数据之后,可以将任务解析成一个字典了imageDict;图片数据会通过这个字典里面的name来获取图片(伪代码 image = imageDict[serverFileKey])
  576. */
  577. //2.3 拼接数据(创建一个字符串来拼装)
  578. NSMutableString *string = [NSMutableString new];
  579. [string appendFormat:@"Content-Disposition:form-data; name=\"%@\"; filename=\"%@\" ", @"file", serverFileName];
  580. //[string appendFormat:@"%@", KNewLine];
  581. [string appendFormat:@"\r\n"];
  582. [string appendFormat:@"Content-Type:%@", serverContentTypes];
  583. // [string appendFormat:@"%@", KNewLine];
  584. // [string appendFormat:@"%@", KNewLine];
  585. [string appendFormat:@"\r\n"];
  586. [string appendFormat:@"\r\n"];
  587. // [boydData appendData:[string dataUsingEncoding:NSUTF8StringEncoding]];
  588. [paraString appendString:string];
  589. [boydData appendData:[paraString dataUsingEncoding:NSUTF8StringEncoding]];
  590. // 2.3 拼接数据(拼接文件数据)
  591. [boydData appendData:data];
  592. // 2.4 边界符号 (结束边界)
  593. NSString *fileEndBoundary = [NSString stringWithFormat:@"\r\n--%@--", Kboundary2];
  594. [boydData appendData:[fileEndBoundary dataUsingEncoding:NSUTF8StringEncoding]];
  595. return boydData;
  596. }
  597. #pragma mark 上传的文件已存在
  598. - (void)handleTaskDidUploadWith:(customUploadOperation*)operation withState:(NSInteger)state
  599. {
  600. if(state == 1){
  601. [[iToast makeText:NSLocalizedString(@"File_upload_file_already_exists",nil)] show];
  602. }
  603. HLog(@"上传的文件已存在 删除任务:_uploadingOperationArr:%@",_uploadingOperationArr);
  604. [self.uploadingOperationArr removeObject:operation];
  605. NSMutableArray *delArr = [NSMutableArray new];
  606. [delArr addObject:operation.fileModel];
  607. [[nasUploadFileManager shareInstance] deleteUploadFileRecordBy:delArr withDelCache:NO complete:^(BOOL isSuccess) {
  608. if (isSuccess) {
  609. }
  610. }];
  611. [self beginUploadAfterDeleteOperationBy:nil];
  612. }
  613. #pragma mark - <NSURLSessionDataDelegate>
  614. // ssl 服务 证书信任
  615. - (void)URLSession:(NSURLSession *)session
  616. didReceiveChallenge:(NSURLAuthenticationChallenge *)challenge
  617. completionHandler:(void (^)(NSURLSessionAuthChallengeDisposition disposition, NSURLCredential * _Nullable credential))completionHandler{
  618. if(![challenge.protectionSpace.authenticationMethod isEqualToString:@"NSURLAuthenticationMethodServerTrust"]) {
  619. return;
  620. }
  621. // 信任该插件
  622. NSURLCredential *credential = [[NSURLCredential alloc] initWithTrust:challenge.protectionSpace.serverTrust];
  623. // 第一个参数 告诉系统如何处置
  624. completionHandler(NSURLSessionAuthChallengeUseCredential,credential);
  625. }
  626. //当请求协议是https的时候回调用该方法
  627. //Challenge 挑战 质询(受保护空间)
  628. //NSURLAuthenticationMethodServerTrust 服务器信任证书
  629. - (void)URLSession:(NSURLSession *)session
  630. task:(NSURLSessionTask *)task
  631. didReceiveChallenge:(NSURLAuthenticationChallenge *)challenge
  632. completionHandler:(void (^)(NSURLSessionAuthChallengeDisposition disposition, NSURLCredential * __nullable credential))completionHandler {
  633. if(![challenge.protectionSpace.authenticationMethod isEqualToString:@"NSURLAuthenticationMethodServerTrust"]) {
  634. return;
  635. }
  636. // 信任该插件
  637. NSURLCredential *credential = [[NSURLCredential alloc] initWithTrust:challenge.protectionSpace.serverTrust];
  638. // 第一个参数 告诉系统如何处置
  639. completionHandler(NSURLSessionAuthChallengeUseCredential,credential);
  640. }
  641. // 接受到响应调用
  642. - (void)URLSession:(NSURLSession *)session
  643. dataTask:(NSURLSessionDataTask *)dataTask
  644. didReceiveResponse:(NSURLResponse *)response
  645. completionHandler:(void (^)(NSURLSessionResponseDisposition disposition))completionHandler {
  646. completionHandler(NSURLSessionResponseAllow);
  647. }
  648. // 上传进度
  649. - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didSendBodyData:(int64_t)bytesSent totalBytesSent:(int64_t)totalBytesSent totalBytesExpectedToSend:(int64_t)totalBytesExpectedToSend{
  650. //每包发送的大小bytesSent,totalBytesSent已经上传了多少;totalBytesExpectedToSend总共要发送多少
  651. // 32768 = 32KB
  652. HLog(@"didSendBodyData: %lld--%lld-%lld", bytesSent, totalBytesSent, totalBytesExpectedToSend);
  653. [_arrayLock lock];
  654. for (customUploadOperation*operation in _uploadingOperationArr) {
  655. HLog(@"正在遍历数组 _uploadingOperationArr:%@",_uploadingOperationArr);
  656. if(operation.dataTask == task){
  657. operation.fileModel.didUploadBytes += bytesSent;
  658. //调整下大小 因为上传的数据 除了data 长度 还包含了 参数大小
  659. if(totalBytesSent == totalBytesExpectedToSend){
  660. operation.fileModel.didUploadBytes -= (totalBytesExpectedToSend - operation.onceDataLengt);
  661. }
  662. NSTimeInterval curTime = [[NSDate date] timeIntervalSince1970];
  663. NSTimeInterval timeDiff = curTime - operation.preNotTimeInterval;
  664. //HLog(@"控制刷新时间为1秒:%f",timeDiff);
  665. CGFloat RefreshTimer = 0.8;
  666. if(operation.fileModel.totalBytes <= 5*1024*1024){//小于5M 事实
  667. RefreshTimer = 0.2;
  668. }
  669. if (operation.preNotTimeInterval <= 0
  670. || timeDiff > RefreshTimer ) {
  671. operation.preNotTimeInterval = curTime;
  672. [[NSNotificationCenter defaultCenter] postNotificationName:uploadFileUploadingNotification object:operation.fileModel];
  673. }
  674. break;
  675. }
  676. }
  677. [_arrayLock unlock];
  678. }
  679. #pragma mark 取消单个任务
  680. - (void)cancelUploadTaskFunWith:(uploadFileDataModel*)fileModel
  681. {
  682. HLog(@"取消任务-- %@--%@",fileModel.filename,fileModel.taskId)
  683. BOOL didUnLockType = NO;
  684. [_arrayLock lock];
  685. for (customUploadOperation*operation in _uploadingOperationArr) {
  686. HLog(@"正在遍历数组 _uploadingOperationArr:%@",_uploadingOperationArr);
  687. if([operation.fileModel.filename isEqualToString:fileModel.filename]){
  688. operation.isCancelType = YES;
  689. [operation.dataTask cancel];
  690. if(operation.fileHandle){
  691. [operation closeFileHandleFun];
  692. }
  693. didUnLockType = YES;
  694. [_arrayLock unlock];
  695. [self beginUploadAfterDeleteOperationBy:operation];
  696. return;
  697. }
  698. }
  699. for (uploadFileDataModel*waitModel in _uploadWaitingUrlArr) {
  700. if([waitModel.filename isEqualToString:fileModel.filename]){
  701. [_uploadWaitingUrlArr removeObject:waitModel];
  702. didUnLockType = YES;
  703. [_arrayLock unlock];
  704. [self beginUploadAfterDeleteOperationBy:nil];
  705. return;
  706. }
  707. }
  708. if (!didUnLockType) {
  709. [_arrayLock unlock];
  710. }
  711. }
  712. #pragma mark 取消所有任务
  713. - (void)cancelUploadAllTaskFun
  714. {
  715. HLog(@"取消所有任务")
  716. [_arrayLock lock];
  717. [_uploadWaitingUrlArr removeAllObjects];
  718. for (customUploadOperation*operation in _uploadingOperationArr) {
  719. HLog(@"正在遍历数组 _uploadingOperationArr:%@",_uploadingOperationArr);
  720. operation.isCancelType = YES;
  721. [operation.dataTask cancel];
  722. if(operation.fileHandle){
  723. [operation closeFileHandleFun];
  724. }
  725. }
  726. [_uploadingOperationArr removeAllObjects];
  727. [_arrayLock unlock];
  728. }
  729. #pragma mark 判断是否在上传中
  730. - (BOOL)checkUploadTaskDoingFun
  731. {
  732. if(_uploadWaitingUrlArr.count >0 || _uploadingOperationArr.count>0){
  733. return YES;
  734. }
  735. return NO;
  736. }
  737. - (BOOL)isUploadIngType
  738. {
  739. if(self.uploadWaitingUrlArr.count >0){
  740. return YES;
  741. }
  742. if(self.uploadingOperationArr.count >0){
  743. return YES;
  744. }
  745. return NO;
  746. }
  747. #pragma mark - lazy load
  748. - (NSMutableArray *)uploadWaitingUrlArr {
  749. if (!_uploadWaitingUrlArr) {
  750. _uploadWaitingUrlArr = [NSMutableArray array];
  751. }
  752. return _uploadWaitingUrlArr;
  753. }
  754. - (NSMutableArray *)uploadingOperationArr {
  755. if (!_uploadingOperationArr) {
  756. _uploadingOperationArr = [NSMutableArray array];
  757. }
  758. return _uploadingOperationArr;
  759. }
  760. @end