Files
RestKit/Code/RKModelLoader.m

147 lines
5.1 KiB
Objective-C

//
// RKModelLoader.m
// RestKit
//
// Created by Blake Watters on 8/8/09.
// Copyright 2009 Two Toasters. All rights reserved.
//
#import <CoreData/CoreData.h>
#import "RKModelLoader.h"
#import "RKResponse.h"
#import "RKModelManager.h"
@implementation RKModelLoader
@synthesize mapper = _mapper, delegate = _delegate, callback = _callback;
+ (id)loaderWithMapper:(RKModelMapper*)mapper {
return [[[self alloc] initWithMapper:mapper] autorelease];
}
- (id)initWithMapper:(RKModelMapper*)mapper {
if (self = [self init]) {
_mapper = [mapper retain];
}
return self;
}
- (void)dealloc {
[_mapper release];
[super dealloc];
}
- (SEL)callback {
return @selector(loadModelsFromResponse:);
}
- (BOOL)encounteredErrorWhileProcessingRequest:(RKResponse*)response {
NSString* errorMessage = nil;
RKRequest* request = response.request;
if ([response isFailure]) {
[_delegate modelLoaderRequest:response.request didFailWithError:response.failureError response:response model:(id<RKModelMappable>)request.userData];
return YES;
} else if ([response isClientError]) {
// TODO: These assumptions are around how Rails serializes error into a payload. Factor out into an adapter...
if ([response isXML]) {
errorMessage = [[(Element*)[response payloadXMLDocument] selectElement:@"error"] contentsText];
} else if ([response isJSON]) {
errorMessage = [[[response payloadJSONDictionary] valueForKey:@"errors"] componentsJoinedByString:@", "];
}
if (nil == errorMessage) {
errorMessage = [response payloadString];
}
// TODO: Eliminate this method in favor of a single error dispatch
[_delegate modelLoaderRequest:response.request didReturnErrorMessage:errorMessage response:response model:(id<RKModelMappable>)request.userData];
return YES;
} else if ([response isServerError]) {
errorMessage = [response payloadString];
[_delegate modelLoaderRequest:response.request didReturnErrorMessage:errorMessage response:response model:(id<RKModelMappable>)request.userData];
return YES;
}
return NO;
}
// TODO: This locking implementation needs to be further cleaned up...
- (void)lockManagedObjectContext {
NSManagedObjectContext* managedObjectContext = [[[RKModelManager manager] objectStore] managedObjectContext];
[managedObjectContext retain];
[managedObjectContext lock];
}
- (void)unlockManagedObjectContext {
NSManagedObjectContext* managedObjectContext = [[[RKModelManager manager] objectStore] managedObjectContext];
[managedObjectContext unlock];
[managedObjectContext release];
}
- (void)informDelegateOfModelLoadWithInfoDictionary:(NSDictionary*)dictionary {
RKResponse* response = [dictionary objectForKey:@"response"];
NSArray* models = [dictionary objectForKey:@"models"];
[dictionary release];
RKRequest* request = response.request;
[_delegate modelLoaderRequest:request didLoadModels:models response:response model:(id<RKModelMappable>)request.userData];
// Release the response now that we have finished all our processing
[response release];
}
- (void)processLoadModelsInBackground:(RKResponse *)response {
NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init];
[self lockManagedObjectContext];
id mapperResult = [_mapper mapFromString:[response payloadString]];
NSArray* models = nil;
if ([mapperResult isKindOfClass:[NSArray class]]) {
models = mapperResult;
} else if ([mapperResult conformsToProtocol:@protocol(RKModelMappable)]) {
models = [NSArray arrayWithObject:mapperResult];
}
[self unlockManagedObjectContext];
NSDictionary* infoDictionary = [[NSDictionary dictionaryWithObjectsAndKeys:response, @"response", models, @"models", nil] retain];
[self performSelectorOnMainThread:@selector(informDelegateOfModelLoadWithInfoDictionary:) withObject:infoDictionary waitUntilDone:NO];
[pool release];
}
- (void)loadModelsFromResponse:(RKResponse*)response {
if (NO == [self encounteredErrorWhileProcessingRequest:response] && [response isSuccessful]) {
// Retain the response to prevent this thread from dealloc'ing before we have finished processing
[response retain];
[self performSelectorInBackground:@selector(processLoadModelsInBackground:) withObject:response];
}
}
///////////////////////////////////////////////////////////////////////////////////////////////////
// RKRequestDelegate
//
// If our delegate responds to the messages, forward them back...
- (void)requestDidStartLoad:(RKRequest*)request {
if ([_delegate respondsToSelector:@selector(requestDidStartLoad:)]) {
[_delegate requestDidStartLoad:request];
}
}
- (void)requestDidFinishLoad:(RKRequest*)request {
if ([_delegate respondsToSelector:@selector(requestDidFinishLoad:)]) {
[(NSObject<RKRequestDelegate>*)_delegate requestDidFinishLoad:request];
}
}
- (void)request:(RKRequest*)request didFailLoadWithError:(NSError*)error {
if ([_delegate respondsToSelector:@selector(request:didFailLoadWithError:)]) {
[(NSObject<RKRequestDelegate>*)_delegate request:request didFailLoadWithError:error];
}
}
- (void)requestDidCancelLoad:(RKRequest*)request {
if ([_delegate respondsToSelector:@selector(requestDidCancelLoad:)]) {
[(NSObject<RKRequestDelegate>*)_delegate requestDidCancelLoad:request];
}
}
@end