web-dev-qa-db-ja.com

AFNetworking-タイムアウトが発生した場合に再試行されるようにリクエストを設定するにはどうすればよいですか?

最近、ASIHTTPRequestからAFNetworkingに移行しました。これはすばらしいことです。ただし、接続しているサーバーにはいくつかの問題があり、リクエストがタイムアウトすることがあります。 ASIHTTPRequestを使用する場合、次のセレクターを使用して、タイムアウトが発生した場合に要求の再試行回数を設定することができました。

-setNumberOfTimesToRetryOnTimeout:

これは、この投稿でさらに参照できます ASIHTTPRequestを再試行できますか?

慣れていない場合、これはAFNetworkingです https://github.com/AFNetworking/AFNetworking#readme

AFNetworkingで同等のAPIを見つけることができませんでした。AFNetworkingを使用してタイムアウトが発生した場合にネットワーク要求を再試行するための解決策を見つけた人はいますか?

23
MTurner

AFNetworkingの開発者であるMattThompsonは、親切にも私にこれに答えてくれました。以下は、解決策を説明するgithubリンクです。

https://github.com/AFNetworking/AFNetworking/issues/39

基本的に、AFNetworkingはこの機能をサポートしていません。以下に示すように、ケースバイケースで実装するのは開発者に任されています(githubでのMatt Thompsonの回答から引用)

- (void)downloadFileRetryingNumberOfTimes:(NSUInteger)ntimes 
                              success:(void (^)(id responseObject))success 
                              failure:(void (^)(NSError *error))failure
{
    if (ntimes <= 0) {
        if (failure) {
            NSError *error = ...;
            failure(error);
        }
    } else {
        [self getPath:@"/path/to/file" parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) {
            if (success) {
                success(...);
            }
        } failure:^(AFHTTPRequestOperation *operation, NSError *error) {
            [self downloadFileRetryingNumberOfTimes:ntimes - 1 success:success failure:failure];
        }];
    }
}
58
MTurner

私の場合、再試行機能が頻繁に必要だったので、この再試行ポリシーカテゴリを思いついたのでAFNetworking + RetryPolicy

AFNetworking 3.0に関しては、うまく機能する可能性があります。

2
Jakub Truhlář

ApiClientクラスにプライベートメソッドを実装しました。

- (void)sendRequest:(NSURLRequest *)request successBlock:(void (^)(AFHTTPRequestOperation *operation, id responseObject))successBlock failureBlock:(void (^)(AFHTTPRequestOperation *operation, NSError *error))failureBlock
{
    __block NSUInteger numberOfRetries = 3;
    __block __weak void (^weakSendRequestBlock)(void);
    void (^sendRequestBlock)(void);
    weakSendRequestBlock = sendRequestBlock = ^{
        __strong typeof (weakSendRequestBlock)strongSendRequestBlock = weakSendRequestBlock;
        numberOfRetries--;

        AFHTTPRequestOperation *operation = [self.httpManager HTTPRequestOperationWithRequest:request success:successBlock failure:^(AFHTTPRequestOperation *operation, NSError *error) {
            NSInteger statusCode = [[[error userInfo] objectForKey:AFNetworkingOperationFailingURLResponseErrorKey] statusCode];

            if (numberOfRetries > 0 && (statusCode == 500 || statusCode == 502 || statusCode == 503 || statusCode == 0)) {
                dispatch_after(dispatch_time(DISPATCH_TIME_NOW, 2 * NSEC_PER_SEC), dispatch_get_main_queue(), ^{
                    strongSendRequestBlock();
                });
            } else {
                if (failureBlock) {
                    failureBlock(operation, error);
                }
            }
        }];

        [self.httpManager.operationQueue addOperation:operation];
    };

    sendRequestBlock();
}

使用例:

- (void)getSomeDetails:(DictionaryResultBlock)block
    {
        if (!block) {
            return;
        }

        NSString *urlString = @"your url string";
        NSMutableURLRequest *request = [self.httpManager.requestSerializer requestWithMethod:@"POST" URLString:[[NSURL URLWithString:urlString relativeToURL:self.defaultUrl] absoluteString] parameters:nil error:nil];

        // Configure you request here
       [request setValue:version forHTTPHeaderField:@"client-version"];

        NSMutableDictionary *bodyParams = @{};   
        [request setHTTPBody:[NSJSONSerialization dataWithJSONObject:bodyParams options:0 error:nil]];

        [self sendRequest:request successBlock:^(AFHTTPRequestOperation *operation, id responseObject) {
            id response = [NSJSONSerialization JSONObjectWithData:responseObject options:0 error:nil];
            block(response, nil);
        } failureBlock:^(AFHTTPRequestOperation *operation, NSError *error) {
            block(nil, error);
        }];
    }
2

あなたの答えに基づいて、パラメータとしてブロックをとるブロックを使用することで、さらに一般的な(そしてトリッキーな)何かを行うことができます:

typedef void (^CallbackBlock)(NSError* error, NSObject* response);
- (void) performBlock:(void (^)(CallbackBlock callback)) blockToExecute retryingNumberOfTimes:(NSUInteger)ntimes onCompletion:(void (^)(NSError* error, NSObject* response)) onCompletion {
    blockToExecute(^(NSError* error, NSObject* response){
        if (error == nil) {
            onCompletion(nil, response);
        } else {
            if (ntimes <= 0) {
                if (onCompletion) {
                    onCompletion(error, nil);
                }
            } else {
                [self performBlock:blockToExecute retryingNumberOfTimes:(ntimes - 1) onCompletion:onCompletion];
            }
        };
    });
}

次に、次のように非同期HTTPリクエストを囲みます。

[self performBlock:^(CallbackBlock callback) {

        [...]
        AFHTTPRequestOperationManager *manager = [WSManager getHTTPRequestOperationManager];
        [manager POST:base parameters:parameters success:^(AFHTTPRequestOperation *operation, id responseObject) {

            dispatch_async(dispatch_get_main_queue(), ^(void){
                    if (callback) {
                        callback(nil, responseObject);
                    }
                });

            } failure:^(AFHTTPRequestOperation *operation, NSError *error) {
                if (callback) {
                    NSError* errorCode = [[NSError alloc] initWithDomain:AppErrorDomain code:[operation.response statusCode] userInfo:@{ NSLocalizedDescriptionKey :error.localizedDescription}];
                    callback(errorCode, nil);
                }
            }];

    } retryingNumberOfTimes:5 onCompletion:^(NSError *error,  NSObject* response) {
        //everything done
    }];

このように、再試行はHTTPリクエストが終了するのを待ち、各リクエストメソッドに再試行ループを実装する必要はありません。

1
Damien Praca