123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505 |
- //
- // QNUploader.h
- // QiniuSDK
- //
- // Created by bailong on 14-9-28.
- // Copyright (c) 2014年 Qiniu. All rights reserved.
- //
- #import <Foundation/Foundation.h>
- #if __IPHONE_OS_VERSION_MIN_REQUIRED
- #import <MobileCoreServices/MobileCoreServices.h>
- #import <UIKit/UIKit.h>
- #if !TARGET_OS_MACCATALYST
- #import <AssetsLibrary/AssetsLibrary.h>
- #import "QNALAssetFile.h"
- #endif
- #if __IPHONE_OS_VERSION_MAX_ALLOWED >= 80000
- #import "QNPHAssetFile.h"
- #import <Photos/Photos.h>
- #endif
- #if __IPHONE_OS_VERSION_MAX_ALLOWED >= 90000
- #import "QNPHAssetResource.h"
- #endif
- #else
- #import <CoreServices/CoreServices.h>
- #endif
- #import "QNUploadManager.h"
- #import "QNAsyncRun.h"
- #import "QNConfiguration.h"
- #import "QNCrc32.h"
- #import "QNFile.h"
- #import "QNUtils.h"
- #import "QNResponseInfo.h"
- #import "QNFormUpload.h"
- #import "QNPartsUpload.h"
- #import "QNConcurrentResumeUpload.h"
- #import "QNUpToken.h"
- #import "QNUploadOption.h"
- #import "QNReportItem.h"
- #import "QNServerConfigMonitor.h"
- #import "QNDnsPrefetch.h"
- #import "QNZone.h"
- #import "QNUploadSourceFile.h"
- #import "QNUploadSourceStream.h"
- @interface QNUploadManager ()
- @property (nonatomic) QNConfiguration *config;
- @end
- @implementation QNUploadManager
- - (instancetype)init {
- return [self initWithConfiguration:nil];
- }
- - (instancetype)initWithRecorder:(id<QNRecorderDelegate>)recorder {
- return [self initWithRecorder:recorder recorderKeyGenerator:nil];
- }
- - (instancetype)initWithRecorder:(id<QNRecorderDelegate>)recorder
- recorderKeyGenerator:(QNRecorderKeyGenerator)recorderKeyGenerator {
- QNConfiguration *config = [QNConfiguration build:^(QNConfigurationBuilder *builder) {
- builder.recorder = recorder;
- builder.recorderKeyGen = recorderKeyGenerator;
- }];
- return [self initWithConfiguration:config];
- }
- - (instancetype)initWithConfiguration:(QNConfiguration *)config {
- if (self = [super init]) {
- if (config == nil) {
- config = [QNConfiguration build:^(QNConfigurationBuilder *builder){
- }];
- }
- _config = config;
- [[QNTransactionManager shared] addDnsLocalLoadTransaction];
- [QNServerConfigMonitor startMonitor];
- }
- return self;
- }
- + (instancetype)sharedInstanceWithConfiguration:(QNConfiguration *)config {
- static QNUploadManager *sharedInstance = nil;
- static dispatch_once_t onceToken;
- dispatch_once(&onceToken, ^{
- sharedInstance = [[self alloc] initWithConfiguration:config];
- });
- return sharedInstance;
- }
- - (void)putData:(NSData *)data
- key:(NSString *)key
- token:(NSString *)token
- complete:(QNUpCompletionHandler)completionHandler
- option:(QNUploadOption *)option {
-
- [self putData:data fileName:nil key:key token:token complete:completionHandler option:option];
- }
- - (void)putData:(NSData *)data
- fileName:(NSString *)fileName
- key:(NSString *)key
- token:(NSString *)token
- complete:(QNUpCompletionHandler)completionHandler
- option:(QNUploadOption *)option {
- if ([QNUploadManager checkAndNotifyError:key token:token input:data complete:completionHandler]) {
- return;
- }
- QNUpToken *t = [QNUpToken parse:token];
- if (t == nil || ![t isValid]) {
- QNResponseInfo *info = [QNResponseInfo responseInfoWithInvalidToken:@"invalid token"];
- [QNUploadManager complete:token
- key:key
- source:data
- responseInfo:info
- response:nil
- taskMetrics:nil
- complete:completionHandler];
- return;
- }
-
- QNServerConfigMonitor.token = token;
- [[QNTransactionManager shared] addDnsCheckAndPrefetchTransaction:self.config.zone token:t];
-
- QNUpTaskCompletionHandler complete = ^(QNResponseInfo *info, NSString *key, QNUploadTaskMetrics *metrics, NSDictionary *resp) {
- [QNUploadManager complete:token
- key:key
- source:data
- responseInfo:info
- response:resp
- taskMetrics:metrics
- complete:completionHandler];
- };
- QNFormUpload *up = [[QNFormUpload alloc] initWithData:data
- key:key
- fileName:fileName
- token:t
- option:option
- configuration:self.config
- completionHandler:complete];
- QNAsyncRun(^{
- [up run];
- });
- }
- - (void)putInputStream:(NSInputStream *)inputStream
- sourceId:(NSString *)sourceId
- size:(long long)size
- fileName:(NSString *)fileName
- key:(NSString *)key
- token:(NSString *)token
- complete:(QNUpCompletionHandler)completionHandler
- option:(QNUploadOption *)option {
-
- if ([QNUploadManager checkAndNotifyError:key token:token input:inputStream complete:completionHandler]) {
- return;
- }
- @autoreleasepool {
- QNUploadSourceStream *source = [QNUploadSourceStream stream:inputStream sourceId:sourceId size:size fileName:fileName];
- [self putInternal:source key:key token:token complete:completionHandler option:option];
- }
- }
- - (void)putFile:(NSString *)filePath
- key:(NSString *)key
- token:(NSString *)token
- complete:(QNUpCompletionHandler)completionHandler
- option:(QNUploadOption *)option {
-
- if ([QNUploadManager checkAndNotifyError:key token:token input:filePath complete:completionHandler]) {
- return;
- }
- @autoreleasepool {
- NSError *error = nil;
- __block QNFile *file = [[QNFile alloc] init:filePath error:&error];
- if (error) {
- QNResponseInfo *info = [QNResponseInfo responseInfoWithFileError:error];
- [QNUploadManager complete:token
- key:key
- source:nil
- responseInfo:info
- response:nil
- taskMetrics:nil
- complete:completionHandler];
- return;
- }
- [self putFileInternal:file key:key token:token complete:completionHandler option:option];
- }
- }
- #if !TARGET_OS_MACCATALYST
- - (void)putALAsset:(ALAsset *)asset
- key:(NSString *)key
- token:(NSString *)token
- complete:(QNUpCompletionHandler)completionHandler
- option:(QNUploadOption *)option {
- #if __IPHONE_OS_VERSION_MIN_REQUIRED
-
- if ([QNUploadManager checkAndNotifyError:key token:token input:asset complete:completionHandler]) {
- return;
- }
- @autoreleasepool {
- NSError *error = nil;
- __block QNALAssetFile *file = [[QNALAssetFile alloc] init:asset error:&error];
- if (error) {
- QNResponseInfo *info = [QNResponseInfo responseInfoWithFileError:error];
- [QNUploadManager complete:token
- key:key
- source:nil
- responseInfo:info
- response:nil
- taskMetrics:nil
- complete:completionHandler];
- return;
- }
- [self putFileInternal:file key:key token:token complete:completionHandler option:option];
- }
- #endif
- }
- #endif
- - (void)putPHAsset:(PHAsset *)asset
- key:(NSString *)key
- token:(NSString *)token
- complete:(QNUpCompletionHandler)completionHandler
- option:(QNUploadOption *)option {
- #if (defined(__IPHONE_OS_VERSION_MAX_ALLOWED) && __IPHONE_OS_VERSION_MAX_ALLOWED >= 90100)
-
- if ([QNUploadManager checkAndNotifyError:key token:token input:asset complete:completionHandler]) {
- return;
- }
- @autoreleasepool {
- NSError *error = nil;
- __block QNPHAssetFile *file = [[QNPHAssetFile alloc] init:asset error:&error];
- if (error) {
- QNResponseInfo *info = [QNResponseInfo responseInfoWithFileError:error];
- [QNUploadManager complete:token
- key:key
- source:nil
- responseInfo:info
- response:nil
- taskMetrics:nil
- complete:completionHandler];
- return;
- }
- [self putFileInternal:file key:key token:token complete:completionHandler option:option];
- }
- #endif
- }
- - (void)putPHAssetResource:(PHAssetResource *)assetResource
- key:(NSString *)key
- token:(NSString *)token
- complete:(QNUpCompletionHandler)completionHandler
- option:(QNUploadOption *)option {
- #if (defined(__IPHONE_OS_VERSION_MAX_ALLOWED) && __IPHONE_OS_VERSION_MAX_ALLOWED >= 90000)
-
- if ([QNUploadManager checkAndNotifyError:key token:token input:assetResource complete:completionHandler]) {
- return;
- }
- @autoreleasepool {
- NSError *error = nil;
- __block QNPHAssetResource *file = [[QNPHAssetResource alloc] init:assetResource error:&error];
- if (error) {
- QNResponseInfo *info = [QNResponseInfo responseInfoWithFileError:error];
- [QNUploadManager complete:token
- key:key
- source:nil
- responseInfo:info
- response:nil
- taskMetrics:nil
- complete:completionHandler];
- return;
- }
- [self putFileInternal:file key:key token:token complete:completionHandler option:option];
- }
- #endif
- }
- - (void)putFileInternal:(id<QNFileDelegate>)file
- key:(NSString *)key
- token:(NSString *)token
- complete:(QNUpCompletionHandler)completionHandler
- option:(QNUploadOption *)option {
- [self putInternal:[QNUploadSourceFile file:file]
- key:key token:token
- complete:completionHandler
- option:option];
- }
- - (void)putInternal:(id<QNUploadSource>)source
- key:(NSString *)key
- token:(NSString *)token
- complete:(QNUpCompletionHandler)completionHandler
- option:(QNUploadOption *)option {
-
- @autoreleasepool {
- QNUpToken *t = [QNUpToken parse:token];
- if (t == nil || ![t isValid]) {
- QNResponseInfo *info = [QNResponseInfo responseInfoWithInvalidToken:@"invalid token"];
- [QNUploadManager complete:token
- key:key
- source:source
- responseInfo:info
- response:nil
- taskMetrics:nil
- complete:completionHandler];
- return;
- }
- QNUpTaskCompletionHandler complete = ^(QNResponseInfo *info, NSString *key, QNUploadTaskMetrics *metrics, NSDictionary *resp) {
- [QNUploadManager complete:token
- key:key
- source:source
- responseInfo:info
- response:resp
- taskMetrics:metrics
- complete:completionHandler];
- };
- QNServerConfigMonitor.token = token;
- [[QNTransactionManager shared] addDnsCheckAndPrefetchTransaction:self.config.zone token:t];
- long long sourceSize = [source getSize];
- if (sourceSize > 0 && sourceSize <= self.config.putThreshold) {
- NSError *error;
- NSData *data = [source readData:(NSInteger)sourceSize dataOffset:0 error:&error];
- [source close];
- if (error) {
- QNResponseInfo *info = [QNResponseInfo responseInfoWithFileError:error];
- [QNUploadManager complete:token
- key:key
- source:source
- responseInfo:info
- response:nil
- taskMetrics:nil
- complete:completionHandler];
- return;
- }
-
- [self putData:data
- fileName:[source getFileName]
- key:key
- token:token
- complete:completionHandler
- option:option];
- return;
- }
- NSString *recorderKey = key;
- if (self.config.recorder != nil && self.config.recorderKeyGen != nil) {
- recorderKey = self.config.recorderKeyGen(key, [source getId]);
- }
-
- if (self.config.useConcurrentResumeUpload) {
- QNConcurrentResumeUpload *up = [[QNConcurrentResumeUpload alloc]
- initWithSource:source
- key:key
- token:t
- option:option
- configuration:self.config
- recorder:self.config.recorder
- recorderKey:recorderKey
- completionHandler:complete];
- QNAsyncRun(^{
- [up run];
- });
- } else {
- QNPartsUpload *up = [[QNPartsUpload alloc]
- initWithSource:source
- key:key
- token:t
- option:option
- configuration:self.config
- recorder:self.config.recorder
- recorderKey:recorderKey
- completionHandler:complete];
- QNAsyncRun(^{
- [up run];
- });
- }
- }
- }
- + (BOOL)checkAndNotifyError:(NSString *)key
- token:(NSString *)token
- input:(NSObject *)input
- complete:(QNUpCompletionHandler)completionHandler {
- if (completionHandler == nil) {
- @throw [NSException exceptionWithName:NSInvalidArgumentException
- reason:@"no completionHandler"
- userInfo:nil];
- return YES;
- }
-
- QNResponseInfo *info = nil;
- if (input == nil) {
- info = [QNResponseInfo responseInfoOfZeroData:@"no input data"];
- } else if ([input isKindOfClass:[NSData class]] && [(NSData *)input length] == 0) {
- info = [QNResponseInfo responseInfoOfZeroData:@"no input data"];
- } else if (token == nil || [token isEqual:[NSNull null]] || [token isEqualToString:@""]) {
- info = [QNResponseInfo responseInfoWithInvalidToken:@"no token"];
- }
- if (info != nil) {
- [QNUploadManager complete:token
- key:key
- source:nil
- responseInfo:info
- response:nil
- taskMetrics:nil
- complete:completionHandler];
- return YES;
- } else {
- return NO;
- }
- }
- + (void)complete:(NSString *)token
- key:(NSString *)key
- source:(NSObject *)source
- responseInfo:(QNResponseInfo *)responseInfo
- response:(NSDictionary *)response
- taskMetrics:(QNUploadTaskMetrics *)taskMetrics
- complete:(QNUpCompletionHandler)completionHandler {
-
- [QNUploadManager reportQuality:key source:source responseInfo:responseInfo taskMetrics:taskMetrics token:token];
-
- QNAsyncRunInMain(^{
- if (completionHandler) {
- completionHandler(responseInfo, key, response);
- }
- });
- }
- //MARK:-- 统计quality日志
- + (void)reportQuality:(NSString *)key
- source:(NSObject *)source
- responseInfo:(QNResponseInfo *)responseInfo
- taskMetrics:(QNUploadTaskMetrics *)taskMetrics
- token:(NSString *)token{
-
- QNUpToken *upToken = [QNUpToken parse:token];
- QNUploadTaskMetrics *taskMetricsP = taskMetrics ?: [QNUploadTaskMetrics emptyMetrics];
-
- QNReportItem *item = [QNReportItem item];
- [item setReportValue:QNReportLogTypeQuality forKey:QNReportQualityKeyLogType];
- [item setReportValue:taskMetricsP.upType forKey:QNReportQualityKeyUpType];
- [item setReportValue:@([[NSDate date] timeIntervalSince1970]) forKey:QNReportQualityKeyUpTime];
- [item setReportValue:responseInfo.qualityResult forKey:QNReportQualityKeyResult];
- [item setReportValue:upToken.bucket forKey:QNReportQualityKeyTargetBucket];
- [item setReportValue:key forKey:QNReportQualityKeyTargetKey];
- [item setReportValue:taskMetricsP.totalElapsedTime forKey:QNReportQualityKeyTotalElapsedTime];
- [item setReportValue:taskMetricsP.ucQueryMetrics.totalElapsedTime forKey:QNReportQualityKeyUcQueryElapsedTime];
- [item setReportValue:taskMetricsP.requestCount forKey:QNReportQualityKeyRequestsCount];
- [item setReportValue:taskMetricsP.regionCount forKey:QNReportQualityKeyRegionsCount];
- [item setReportValue:taskMetricsP.bytesSend forKey:QNReportQualityKeyBytesSent];
-
- [item setReportValue:[QNUtils systemName] forKey:QNReportQualityKeyOsName];
- [item setReportValue:[QNUtils systemVersion] forKey:QNReportQualityKeyOsVersion];
- [item setReportValue:[QNUtils sdkLanguage] forKey:QNReportQualityKeySDKName];
- [item setReportValue:[QNUtils sdkVersion] forKey:QNReportQualityKeySDKVersion];
-
- [item setReportValue:responseInfo.requestReportErrorType forKey:QNReportQualityKeyErrorType];
- NSString *errorDesc = responseInfo.requestReportErrorType ? responseInfo.message : nil;
- [item setReportValue:errorDesc forKey:QNReportQualityKeyErrorDescription];
-
- [item setReportValue:taskMetricsP.lastMetrics.lastMetrics.hijacked forKey:QNReportBlockKeyHijacking];
-
- long long fileSize = -1;
- if ([source conformsToProtocol:@protocol(QNUploadSource)]) {
- fileSize = [(id <QNUploadSource>)source getSize];
- } else if ([source isKindOfClass:[NSData class]]) {
- fileSize = [(NSData *)source length];
- }
- [item setReportValue:@(fileSize) forKey:QNReportQualityKeyFileSize];
- if (responseInfo.isOK && fileSize > 0 && taskMetrics.totalElapsedTime) {
- NSNumber *speed = [QNUtils calculateSpeed:fileSize totalTime:taskMetrics.totalElapsedTime.longLongValue];
- [item setReportValue:speed forKey:QNReportQualityKeyPerceptiveSpeed];
- }
-
- [kQNReporter reportItem:item token:token];
- }
- @end
|