Как отобразить RK-ответ RKClient с помощью RKObjectMapping - PullRequest
1 голос
/ 10 декабря 2011

Я делаю многоэтапную публикацию данных изображения и некоторых значений, используя RestKit's RKClient, вот так:

RKParams* params = [RKParams params];
[params setValue:foo.accountId forParam:@"accountId"];
[params setValue:foo.identifier forParam:@"fooId"];
[params setValue:_photoId forParam:@"photoId"];
[params setData:data MIMEType:@"image/png" forParam:@"image"];
[[RKClient sharedClient] post:@"/foo/uploadPhoto" params:params delegate:self];

Это прекрасно работает, и мой внутренний сервер отвечает JSON-представлением объекта модели на стороне сервера, это выглядит так:

{"id":"4ee2b4670364720c089e75b9","accountId":"4ebee3469ae2d8adf983c561","fooId":"4ec0983d036463d900841f0b","photoId":"E5B20AF1-9F10-4175-8262-852BDA3DEDE9","filename":"4ebee3469ae2d8adf983c561_4ec0983d036463d900841f0b_E5B20AF1-9F10-4175-8262-852BDA3DEDE9","contentType":"image/png"}

Теперь мне нужно сопоставить это с моим объектом на стороне клиента (iOS). Объект модели на стороне клиента почти такой же, но не идентичный (поэтому использование метода objectFromString в RKJSONParser не является опцией), поэтому у меня есть определенный RKObjectMapping, определяющий отображение. Делегат RKClient получает только RKResponse, так как я могу использовать ответ вместе с картографом для получения экземпляра моего объекта модели на стороне клиента?

Примечание. Для ясности я очень хорошо знаю, как это работает при использовании RKObjectManager для публикации объекта и сопоставления ответа. Уникальная часть моей ситуации заключается в том, что я использую RKClient для получения должности, состоящей из нескольких частей. К сожалению, RKClient, похоже, не имеет простых методов для обработки отображения ответов, как это делает RKObjectManager ... если только я что-то упустил (что я надеюсь, и вы все укажете на меня;)

1 Ответ

1 голос
/ 12 декабря 2011

Ну, этот пост был похож (но не функционален), и он дал мне некоторые идеи о новой технике использования этого метода в RKObjectLoader

- (RKObjectLoader *)postObject:(id<NSObject>)object delegate:(id<RKObjectLoaderDelegate>)delegate block:(void ( ^ ) ( RKObjectLoader *))block

Так что теперь я мог получить преимущество картографирования, которое не было очевидным, как использовать RKClient.

Настройка маршрутизатора:

RKObjectManager *objectManager = [RKObjectManager objectManagerWithBaseURL:kApiUrlBase];
[objectManager.router routeClass:[PAPetPhoto class] toResourcePath:@"/pet/uploadPhoto" forMethod:RKRequestMethodPOST];

Настройка отображения:

RKObjectMapping *papetPhotoMapping = [RKObjectMapping mappingForClass:[PAPetPhoto class]];
[papetPhotoMapping mapKeyPath:@"id" toAttribute:@"identifier"];
[papetPhotoMapping mapAttributes:@"accountId", @"petId", @"photoId", @"filename", @"contentType", nil];
[objectManager.mappingProvider addObjectMapping:papetPhotoMapping];
[objectManager.mappingProvider setSerializationMapping:[papetPhotoMapping inverseMapping] forClass:[PAPetPhoto class]];
[objectManager.mappingProvider setMapping:papetPhotoMapping forKeyPath:@"petPhoto"];

Пост: (обратите внимание, поскольку я собрал все свои параметры в блоке, мой объект - просто фиктивный экземпляр для запуска правильной маршрутизации и отображения).

    PAPetPhoto *photo = [[PAPetPhoto alloc] init];
    [[RKObjectManager sharedManager] postObject:photo delegate:self block:^(RKObjectLoader *loader){

        RKParams* params = [RKParams params];
        [params setValue:pet.accountId forParam:@"accountId"];
        [params setValue:pet.identifier forParam:@"petId"];
        [params setValue:_photoId forParam:@"photoId"];
        [params setValue:_isThumb ? @"THUMB" : @"FULL" forParam:@"photoSize"];
        [params setData:data MIMEType:@"image/png" forParam:@"image"];

        loader.params = params;
    }];

Конечная точка сервера (Java, Spring MVC)

    @RequestMapping(value = "/uploadPhoto", method = RequestMethod.POST)
    @ResponseBody
    public Map<String, Object> handleFormUpload(@RequestParam("accountId") String accountId,
                                    @RequestParam("petId") String petId,
                                    @RequestParam("photoId") String photoId,
                                    @RequestParam("photoSize") PhotoSizeEnum photoSize,
                                    @RequestParam("image") Part image) throws IOException {

        if (log.isTraceEnabled()) 
            log.trace("uploadPhoto. accountId=" + accountId + " petId=" + petId + " photoId=" + photoId + " photoSize=" + photoSize);

        PetPhoto petPhoto = petDao.savePetPhoto(accountId, petId, photoId, photoSize, image);

        Map<String, Object> map = GsonUtils.wrapWithKeypath(petPhoto, "petPhoto");
        return map;
    }

Ответ сервера JSON (обратите внимание на keyPath «petPhoto», который соответствует настройке сопоставления):

{
    petPhoto =     {
        accountId = 4ebee3469ae2d8adf983c561;
        contentType = "image/png";
        filename = "4ebee3469ae2d8adf983c561_4ec0983d036463d900841f09_3FED4959-1042-4D8B-91A8-76AA873851A3";
        id = 4ee2e80203646ecd096d5201;
        petId = 4ec0983d036463d900841f09;
        photoId = "3FED4959-1042-4D8B-91A8-76AA873851A3";
    };
}

Делегат:

- (void) objectLoader:(RKObjectLoader*)objectLoader didLoadObject:(id)object {

    if ([objectLoader wasSentToResourcePath:@"/pet/uploadPhoto"]) {
       PAPetPhoto *photo = (PAPetPhoto*)object;
    }
}
...