Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

RestKit Relationship Mapping with Primary Key

I am having trouble mapping relationships when the JSON response only contains the primary key and not a fully nested array to create the new object.

I have 2 classes - Shop and Item, and as you would expect the Shop->Item has a one-to-many relationship.

I have a local core-data store of shops (and items), each with a primary key. I then wish to download a list of Items as JSON and map to core-data entities, but only include the primary key of the shop, and not all the shop details as a nested array - this would be huge waste of network traffic as I am downloading details of 500+ Items.

Here is the JSON from the two requests:

/shops

{
    "id" : 1,
    "shop" : "Shop A",
    "city" : "New York"
},
{
    "id" : 2,
    "shop" : "Shop B",
    "city" : "London"
},
...

/items

{
    "id" : 1,
    "name" : "Shoes",
    "manufacturer" : "Vans",
    "shopId" : 1
},
{
    "id" : 2,
    "name" : "T-shirt",
    "manufacturer" : "Animal",
    "shopId" : 2
},
{
    "id" : 3,
    "name" : "Scarf",
    "manufacturer" : "Ted Baker",
    "shopId" : 1
},
{
    "id" : 4,
    "name" : "Sunglasses",
    "manufacturer" : "Ray-Ban",
    "shopId" : 3
},
...

Here is my code at the moment.

AppDelegate.m

...

NSURL *baseURL = [NSURL URLWithString:@"http://localhost/company/API"];
RKObjectManager *objectManager = [RKObjectManager managerWithBaseURL:baseURL];

[AFNetworkActivityIndicatorManager sharedManager].enabled = YES;

[objectManager.HTTPClient setDefaultHeader:@"Accept" value:@"application/json"];

NSManagedObjectModel *managedObjectModel = [NSManagedObjectModel mergedModelFromBundles:nil];
RKManagedObjectStore *managedObjectStore = [[RKManagedObjectStore alloc] initWithManagedObjectModel:managedObjectModel];
objectManager.managedObjectStore = managedObjectStore;

// Shop Mapping

RKEntityMapping *shopMapping = [RKEntityMapping mappingForEntityForName:NSStringFromClass([Shop class])
                                                       inManagedObjectStore:objectManager.managedObjectStore];
NSDictionary *shopMappingAttributes = [NSDictionary dictionaryWithObjectsAndKeys:@"objectId",@"id",@"name",@"shop",@"city",@"city",nil];
shopMapping.identificationAttributes = @[@"objectId"];
[shopMapping addAttributeMappingsFromDictionary:shopMappingAttributes];
[objectManager addResponseDescriptor:[RKResponseDescriptor responseDescriptorWithMapping:shopMapping
                                                                             pathPattern:@"/shops"
                                                                                 keyPath:nil
                                                                             statusCodes:RKStatusCodeIndexSetForClass(RKStatusCodeClassSuccessful)]];


// Item Mapping

RKEntityMapping *itemMapping = [RKEntityMapping mappingForEntityForName:NSStringFromClass([Item class])
                                                       inManagedObjectStore:objectManager.managedObjectStore];
NSDictionary *itemMappingAttributes = [NSDictionary dictionaryWithObjectsAndKeys:@"objectId",@"id",@"name", @"name",@"manufacturer",@"manufacturer",nil];
itemMapping.identificationAttributes = @[@"objectId"];
[itemMapping addAttributeMappingsFromDictionary:itemMappingAttributes];

// Define the relationship mapping

[objectManager addResponseDescriptor:[RKResponseDescriptor responseDescriptorWithMapping:itemMapping
                                                                             pathPattern:@"/items"
                                                                                 keyPath:nil
                                                                             statusCodes:RKStatusCodeIndexSetForClass(RKStatusCodeClassSuccessful)]];

...

ItemsTableViewController.m

- (void)viewDidLoad {
    [super viewDidLoad];

    // Update Shops
    [[RKObjectManager sharedManager] getObjectsAtPath:@"/shops"
                                       parameters:nil
                                          success:nil
                                          failure:^(RKObjectRequestOperation *operation, NSError *error) {
                                              NSLog(@"Error: %@",error);
                                          }];

    // Update/Get Items
    NSDictionary *parameters = @{
                             @"username": self.username,
                             @"password": self.password,
                             @"API_key": @"abc123",
                             };

    NSMutableURLRequest *request = [[RKObjectManager sharedManager] requestWithObject:nil
                                                                           method:RKRequestMethodPOST
                                                                             path:@"/items"
                                                                       parameters:parameters];

    RKManagedObjectRequestOperation *operation = [[RKObjectManager sharedManager] managedObjectRequestOperationWithRequest:request managedObjectContext:[RKManagedObjectStore defaultStore].mainQueueManagedObjectContext
                                                                  success:^(RKObjectRequestOperation *operation, RKMappingResult *mappingResult) {
                                                                      Item *item = [mappingResult firstObject];
                                                                      NSLog(@"Mapped the Item: %@", item);
                                                                  } failure:^(RKObjectRequestOperation *operation, NSError *error) {
                                                                      NSLog(@"Error: %@",error);
                                                                  }];
    NSOperationQueue *operationQueue = [NSOperationQueue new];
    [operationQueue addOperation:operation];
}

EDIT: Wain, I have this in the relevant place in the app delegate but get an NSException

NSEntityDescription *itemEntity = [NSEntityDescription entityForName:@"Item" inManagedObjectContext:managedObjectStore.mainQueueManagedObjectContext];
NSRelationshipDescription *shopRelationship = [itemEntity relationshipsByName][@"shop"];
RKConnectionDescription *connection = [[RKConnectionDescription alloc] initWithRelationship:shopRelationship attributes:@{ @"shopId": @"objectId" }];
[itemMapping addConnection:connection];

NSException

*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '+entityForName: nil is not a legal NSManagedObjectContext parameter searching for entity name 'Item''

What have I missed?

like image 742
Andy Avatar asked Sep 21 '13 16:09

Andy


1 Answers

You need to add a transient attribute to the item (called shopId) and an associated mapping.

Configure the relationship using foreign key mapping as:

NSEntityDescription *itemEntity = [NSEntityDescription entityForName:@"Item" inManagedObjectContext:managedObjectContext];
NSRelationshipDescription *shopRelationship = [itemEntity relationshipsByName][@"shop"];
RKConnectionDescription *connection = [[RKConnectionDescription alloc] initWithRelationship:shopRelationship attributes:@{ @"shopId": @"id" }];

Then use addConnection: to add it to your mapping.

like image 96
Wain Avatar answered Oct 05 '22 10:10

Wain