AFNetworking上传图片间歇工作

时间:2012-08-26 06:09:05

标签: php objective-c ios cocoa-touch afnetworking

我正在使用AFNetworking将图像和一些参数上传到PHP脚本(使用CodeIgniter构建),该脚本将接收图像,将文件名和参数放在数据库中并将图像移动到永久位置。

这是Obj-C:

NSURL *url = [NSURL URLWithString:@"http://my_api_endpoint"];
NSData *imageToUpload = UIImageJPEGRepresentation(_mainMedia, .25f);
NSDictionary *params = [[NSDictionary alloc]initWithObjectsAndKeys:
                        _topicText.text,@"Topic",
                        @"1",@"Category",
                        @"1",@"Creator",
                        nil];
AFHTTPClient *client= [AFHTTPClient clientWithBaseURL:url];

NSString *timeStamp = [NSString stringWithFormat:@"%0.0f.jpg", [[NSDate date] timeIntervalSince1970]];

NSMutableURLRequest *request = [client multipartFormRequestWithMethod:@"POST" path:@"apidebate/debates" parameters:params constructingBodyWithBlock: ^(id <AFMultipartFormData>formData) {
    [formData appendPartWithFileData: imageToUpload name:@"MainMedia" fileName:timeStamp mimeType:@"image/jpeg"];
}];

AFHTTPRequestOperation *operation = [[AFHTTPRequestOperation alloc] initWithRequest:request];

[operation setCompletionBlockWithSuccess:^(AFHTTPRequestOperation *operation, id responseObject) {
    NSString *response = [operation responseString];
    NSLog(@"response: [%@]",response);

    [self dismissViewControllerAnimated:YES completion:nil];
    [self.delegate createTopicViewControllerDidCreate:self];

    [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:NO];
} failure:^(AFHTTPRequestOperation *operation, NSError *error) {
    [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:NO];
    if([operation.response statusCode] == 403){
        NSLog(@"Upload Failed");
        return;
    }
    NSLog(@"error: %@", [operation error]);

    [self dismissViewControllerAnimated:YES completion:nil];
    [self.delegate createTopicViewControllerDidCreate:self];
}];

[operation setUploadProgressBlock:^(NSInteger bytesWritten, long long totalBytesWritten, long long totalBytesExpectedToWrite) {
    NSLog(@"Sent %lld of %lld bytes", totalBytesWritten, totalBytesExpectedToWrite);
    float width = totalBytesWritten / totalBytesExpectedToWrite;

}];

[operation start];

这是PHP:

//CONTROLLER FROM API
function debates_post()
{
mail('myemailaddress@gmail.com', 'Test', 'Posted');
$tmp_dir = "images/posted/";

if(isset($_FILES['MainMedia'])){
    $SaniFileName = preg_replace('/[^a-zA-Z0-9-_\.]/','', basename($_FILES['MainMedia']['name']));

    $file = $tmp_dir . $SaniFileName;
    move_uploaded_file($_FILES['MainMedia']['tmp_name'], $file);
}
else
    $SaniFileName = NULL;

$data = array('Topic'=>$this->post('Topic'), 'MainMedia'=>$this->post('MainMedia'), 'Category'=>$this->post('Category'), 'Creator'=>$this->post('Creator'));
$insert = $this->debate->post_debate($this->post('Topic'), $SaniFileName, $this->post('Category'), $this->post('Creator'));
if($insert){
    $message = $this->db->insert_id();
}
else{
    $message = 'Insert failed';
}

$this->response($message, 200);
}

//MODEL
function post_debate($Topic=NULL, $MainMedia='', $Category=NULL, $Creator=NULL){
$MainMedia = ($MainMedia)?$MainMedia:'';
$data = array(
                'Topic' => $Topic,
                'MainMedia' => $MainMedia,
                'Category' => $Category,
                'Creator' => $Creator,
                'Created' => date('Y-m-d h:i:s')
            );
return $this->db->insert('debate_table', $data);
}

我目前的问题是从iOS上传很少完成,并且没有模式。我可以添加大或小的照片或根本没有照片只有参数,它可以在20%的时间内工作。当它失败时,这是我在X-Code中得到的错误信息:

2012-08-26 01:52:10.698 DebateIt[24215:907] error: Error Domain=NSURLErrorDomain 
Code=-1021 "request body stream exhausted" UserInfo=0x1dd7a0d0 
{NSErrorFailingURLStringKey=http://my_api_url, 
NSErrorFailingURLKey=http://my_api_url,
NSLocalizedDescription=request body stream exhausted, 
NSUnderlyingError=0x1e8535a0 "request body stream exhausted"}

这是世界上的什么?

我有一个基本的HTML表单,我将相同的图像发布到同一个端点,每次都可以使用任何大小的图像。 iOS似乎允许不同尺寸的图像很好......但无论大小,它可能只会在第二次尝试时使用。

思想?

3 个答案:

答案 0 :(得分:3)

我也正在使用AFNetworking上传图片,我也没有问题,可能是因为我用base64编码我的图像数据?

NSMutableDictionary *params = [NSMutableDictionary dictionaryWithObjectsAndKeys:
                               token, @"token",
                               [UIImageJPEGRepresentation(image, 0.8) base64EncodedString],@"photo",
                               nil];

NSMutableURLRequest *request = [self.httpClient requestWithMethod:@"POST"
                                                             path:@"/user/upload/photo.json"
                                                       parameters:params];

答案 1 :(得分:0)

前段时间我有类似的错误。它是由“Content-Length”http标头的值与实际的主体大小不匹配引起的。错误是微妙的。之所以发生这种情况,是因为我使用字符串长度作为内容大小,并且当字符串在UTF-8编码中具有双字节字符时,这些字符串长度为1,而帖子正文大小为2。检查您的POST正文大小v / s“Content-Length”。

我通过在模拟器中运行应用程序的网络嗅探来调试这种奇怪的网络错误。我喜欢HTTPScoop。

答案 2 :(得分:0)

首先,请确保已下载最新版本的AFNetworking。

你可以[[AFHTTPRequestOperation alloc] initWithRequest:...]然后使用直接属性访问器(operation.completionBlock = ^ {...})或者使用-setCompletionBlockWithSuccess:failure:来设置completionBlock。请记住,在请求下载完成后执行完成块。

对于multipart表单块,-appendWithFileData:mimeType:name也被删除了一段时间。你想要的方法是-appendPartWithFileData:name:fileName:mimeType:。

进行这两项更改,一切都应该有效。