Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

iPhone - Corrupt JPEG data for image received over HTTP

I'm getting an image over HTTP, using NSURLConnection, as follows -

NSMutableData *receivedData;

- (void)getImage {
    self.receivedData = [[NSMutableData alloc] init];
    NSURLConnection *theConnection = // create connection
}

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {    
   [receivedData appendData:data];
}

-(void)connectionDidFinishLoading:(NSURLConnection *)connection {
   [connection release];

   UIImage *theImage = [UIImage imageWithData:receivedData];
}

Usually it works just fine, but sometimes I'm seeing this get logged - : Corrupt JPEG data: premature end of data segment

At this point, the image does not completely render. I'll see maybe 75% of it, and then the lower right hand corner is a grey box.

Any ideas on how to approach fixing this? Am I constructing my image improperly?

like image 773
bpapa Avatar asked Jun 30 '09 17:06

bpapa


4 Answers

Your HTTP code looks correct. You might want to log the size of the receivedData once it's done loading and compare it to the expected size of the image on the server. If it's the expected size, then maybe the image itself is corrupted on the server.

like image 96
Marc Novakowski Avatar answered Nov 11 '22 19:11

Marc Novakowski


ASI-HTTP can fix this problem.

NSURL *coverRequestUrl = [NSURL URLWithString:imageStringURL];
ASIHTTPRequest *coverRequest = [[ASIHTTPRequest alloc] initWithURL:coverRequestUrl];
[coverRequest setDelegate:self];
[coverRequest setDidFinishSelector:@selector(imageRecieved:)];

[appDelegate.queue addOperation:coverRequest];
[appDelegate.queue go];

My queue variable in appDelegate is ASINetwork queue object. Because I send asynchronous request, so I use it.

- (void)imageRecieved:(ASIHTTPRequest *)response
{
    UIImage *myImage = [UIImage imageWithData:[response responseData]];
}
like image 28
Jerapong Nampetch Avatar answered Nov 11 '22 19:11

Jerapong Nampetch


I fixed this problem by using an NSMutableDictionary.

NSMutableDictionary *dataDictionary;

In my loadData function, I define my data:

NSMutableData *receivedData = receivedData = [[NSMutableData alloc] init];

Then I load the data into my dictionary where the key is [theConnection description] and the object is my data.

[dataDictionary setObject:receivedData forKey:[theConnection description]];

That way in the delegates, I can look up the correct data object for the connection that is passed to the delegate and save to the right data instance otherwise I end up with the JPEG munging/corruption problem.

In didReceiveData, I put:

//get the object for the connection that has been passed to connectionDidRecieveData and that object will be the data variable for that instance of the connection.
NSMutableData *theReceivedData = [dataDictionary objectForKey:[connection description]];

//then act on that data
[theReceivedData appendData:data];

Similarly, in didReceiveResponse, I put:

NSMutableData *theReceivedData = [dataDictionary objectForKey:[connection description]];
[theReceivedData setLength:0];

And in connectionDidFinishLoading: NSMutableData *theReceivedData = [dataDictionary objectForKey:[connection description]]; img = [[UIImage alloc] initWithData:theReceivedData];

And this seems to work very well. By the way, my code is based on Apple's tutorial for NSUrlConnection with the addition of an NSMutableDictionary to keep track of individual connections. I hope this helps. Let me know if you want me to post my full image handling code.

like image 39
Jackson Avatar answered Nov 11 '22 19:11

Jackson


I have seen this also. If you save the data to a file and then read the data back into an image, it works perfectly. I suspect there is http header information in the jpeg image data.

Hope someone finds a solution to this because the save to file workaround sucks.

// problem

UIImage *newImage = [UIImage imageWithData:receivedData];

// crappy workaround

[receivedData writeToFile:[NSString stringWithFormat:@"a.jpg"] atomically:NO];
UIImage *newImage = [UIImage imageWithContentsOfFile:@"a.jpg"];
like image 27
2 revs, 2 users 90%Will Avatar answered Nov 11 '22 20:11

2 revs, 2 users 90%Will