Как провести модульное тестирование моих моделей сейчас, когда я использую Core Data? - PullRequest
56 голосов
/ 05 декабря 2009

Я разрабатывал приложение для iphone, используя модель домена, и отложил аспект сохранения приложения до сих пор. Базовые данные выглядят как действительно хорошее решение, так как у меня уже есть четко определенная модель, но я сталкиваюсь с трудностями из-за существующих модульных тестов.

Вот простой пример того, что у меня сейчас:

- (void)test_full_name_returns_correct_string {
    Patient *patient = [[Patient alloc] init];  
    patient.firstName = @"charlie";
    patient.lastName = @"chaplin";
    STAssertTrue([[patient fullName] isEqualToString:@"charlie chaplin"], @"should have matched full name");
}  

Как я могу заставить эту работу работать, когда мой объект Patient выходит из NSManagedObject и использует @dynamic для свойств firstName и lastName?

Кто-нибудь еще сталкивался с этим типом с Core Data? Спасибо.

Ответы [ 2 ]

85 голосов
/ 05 декабря 2009

Вам необходимо создать базовый стек данных либо внутри каждого метода, либо в -setUp, а затем разобрать его. Использование NSInMemoryPersistentStore обеспечит быструю работу и оперативную память для ваших модульных тестов. Добавьте @property (nonatomic,retain) NSManagedObjectContext *moc к вашему подклассу TestCase. Тогда:

- (void)setUp {
  NSManagedObjectModel *mom = [NSManagedObjectModel mergedModelFromBundles:[NSArray arrayWithObject:bundleContainingXCDataModel]];
  NSPersistentStoreCoordinator *psc = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:mom];
  STAssertTrue([psc addPersistentStoreWithType:NSInMemoryStoreType configuration:nil URL:nil options:nil error:NULL] ? YES : NO, @"Should be able to add in-memory store");    
  self.moc = [[NSManagedObjectContext alloc] init];
  self.moc.persistentStoreCoordinator = psc;

  [mom release];
  [psc release];

}

- (void)tearDown {
  self.moc = nil;
}

Тогда ваш метод теста будет выглядеть так:

- (void)test_full_name_returns_correct_string {
    Patient *patient = [NSEntityDescription insertNewObjectForEntityForName:@"Person" inManagedObjectContext:self.moc];

    patient.firstName = @"charlie";
    patient.lastName = @"chaplin";
    STAssertTrue([[patient fullName] isEqualToString:@"charlie chaplin"], @"should have matched full name");
}

при условии, что ваша сущность называется Person. Кстати, в вашей версии метода произошла утечка памяти; пациент должен быть -release 'd в неосновной версии данных (insertNewObjectForEntityForName:managedObjectContext: возвращает автоматически выпущенный экземпляр).

22 голосов
/ 02 июня 2014

Я использовал ответ выше Барри Уорка, но мне пришлось внести некоторые изменения, чтобы он работал с текущими проектами Xcode 5, iOS 7.

Недвижимость осталась прежней:

@interface SIDataTest : XCTestCase
    @property (nonatomic, retain) NSManagedObjectContext *moc;
@end

На самом деле настройку пришлось изменить, во-первых, чтобы не выпускать, а во-вторых, предоставить URL модели.

- (void)setUp
{
    [super setUp];
    NSURL *modelURL = [[NSBundle mainBundle] URLForResource:@"SimpleInvoice" withExtension:@"momd"];
    NSManagedObjectModel *mom = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
    NSPersistentStoreCoordinator *psc = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:mom];
    XCTAssertTrue([psc addPersistentStoreWithType:NSInMemoryStoreType configuration:nil URL:nil options:nil error:NULL] ? YES : NO, @"Should be able to add in-memory store");
    self.moc = [[NSManagedObjectContext alloc] init];
    self.moc.persistentStoreCoordinator = psc;
}

Вот пример теста:

- (void)testCreateNew
{
    Invoice *newInvoice = [NSEntityDescription insertNewObjectForEntityForName:@"Invoice" inManagedObjectContext:self.moc];
    newInvoice.dueDate = [NSDate date];
    NSString* title = [[NSString alloc] initWithFormat:@"Invoice %@", @112];
    newInvoice.title = title;

    // Save the context.
    NSError *error = nil;
    if (![self.moc save:&error]) {
        // Replace this implementation with code to handle the error appropriately.
        // abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
        XCTFail(@"Error saving in \"%s\" : %@, %@", __PRETTY_FUNCTION__, error, [error userInfo]);
    }
    XCTAssertFalse(self.moc.hasChanges,"All the changes should be saved");
}
...