Dar*_*ski 8 unit-testing nsfetchedresultscontroller xctest swift
我有一个Swift应用程序,用于从持久存储中NSFetchedResultsController获取List对象:
let fetchedResultsController: NSFetchedResultsController = ...
var error : NSError?
fetchedResultsController.performFetch(&error)
if let error = error {
NSLog("Error: \(error)")
}
let lists: [List] = fetchedResultsController.fetchedObjects! as [List]
NSLog("lists count = \(lists.count)")
for list: List in lists {
NSLog("List: \(list.description)")
}
Run Code Online (Sandbox Code Playgroud)
它的工作方式与预期的一样,我将List对象描述打印到控制台.我想为我的应用程序编写一些单元测试,所以我创建了扩展的类XCTestCase.代码编译没有问题,测试运行,但不幸的是我无法List在该上下文中获取对象.
我在控制台中得到的只是List对象的数量和致命的错误:
lists count = 59
fatal error: NSArray element failed to match the Swift Array Element type
Run Code Online (Sandbox Code Playgroud)
线路上升:
for list: List in lists {
Run Code Online (Sandbox Code Playgroud)
我很确定我已经正确配置了目标,因为我可以创建List对象并将其插入到托管对象上下文中,而不会出现我的应用程序源代码以及单元测试源代码的问题.我遇到的唯一问题是从测试单元获取.我想知道为什么在模拟器中运行应用程序时提取工作正常,并且在单元测试期间执行时失败.
任何可能出错的想法都将受到赞赏.
更新:
更具体地说,我的实现如何,这里是我正在玩的完整代码示例:
var error: NSError? = nil
let urls = NSFileManager.defaultManager().URLsForDirectory(.DocumentDirectory, inDomains: .UserDomainMask)
let applicationDocumentsDirectory = urls[urls.count-1] as NSURL
let modelURL = NSBundle.mainBundle().URLForResource("CheckLists", withExtension: "momd")!
let managedObjectModel = NSManagedObjectModel(contentsOfURL: modelURL)
var coordinator: NSPersistentStoreCoordinator? = NSPersistentStoreCoordinator(managedObjectModel: managedObjectModel)
let url = applicationDocumentsDirectory.URLByAppendingPathComponent("CheckLists.sqlite")
if coordinator!.addPersistentStoreWithType(NSSQLiteStoreType, configuration: nil, URL: url, options: nil, error: &error) == nil {
NSLog("Error1: \(error)")
abort()
}
var managedObjectContext = NSManagedObjectContext()
managedObjectContext.persistentStoreCoordinator = coordinator
let fetchRequest = NSFetchRequest()
fetchRequest.entity = NSEntityDescription.entityForName("List", inManagedObjectContext: managedObjectContext)
fetchRequest.sortDescriptors = [ NSSortDescriptor(key: "name", ascending: true) ]
let fetchedResultsController = NSFetchedResultsController(
fetchRequest: fetchRequest,
managedObjectContext: managedObjectContext,
sectionNameKeyPath: nil,
cacheName: "ListFetchedResultsControllerCache"
)
fetchedResultsController.performFetch(&error)
if let error = error {
NSLog("Error2: \(error)")
abort()
}
let fetchedObjects: [AnyObject]? = fetchedResultsController.fetchedObjects
if let fetchedObjects = fetchedObjects {
NSLog("Fetched objects count: \(fetchedObjects.count)")
for fetchedObject in fetchedObjects {
NSLog("Fetched object: \(fetchedObject.description)")
}
}
else {
NSLog("Fetched objects array is nil")
}
let fetchedLists: [List]? = fetchedResultsController.fetchedObjects as? [List]
if let fetchedLists = fetchedLists {
NSLog("Fetched lists count: \(fetchedLists.count)")
for fetchedList in fetchedLists {
NSLog("Fetched list: \(fetchedList.description)")
}
}
else {
NSLog("Fetched lists array is nil")
}
Run Code Online (Sandbox Code Playgroud)
当我从应用程序的源代码执行它,在模拟器中运行应用程序时,控制台输出如下所示:
Fetched objects count: 3
Fetched object: <CheckLists.List: 0x7a6866f0> (entity: List; id: 0x7a686020 <x-coredata://7A87B5BE-C2FA-4150-B9E3-879FDE07F0B9/List/p2> ; data: {
name = "List 1";
})
Fetched object: <CheckLists.List: 0x7a686930> (entity: List; id: 0x7a686030 <x-coredata://7A87B5BE-C2FA-4150-B9E3-879FDE07F0B9/List/p1> ; data: {
name = "List 2";
})
Fetched object: <CheckLists.List: 0x7a686970> (entity: List; id: 0x7a686040 <x-coredata://7A87B5BE-C2FA-4150-B9E3-879FDE07F0B9/List/p3> ; data: {
name = "List 3";
})
Fetched lists count: 3
Fetched list: <CheckLists.List: 0x7a6866f0> (entity: List; id: 0x7a686020 <x-coredata://7A87B5BE-C2FA-4150-B9E3-879FDE07F0B9/List/p2> ; data: {
name = "List 1";
})
Fetched list: <CheckLists.List: 0x7a686930> (entity: List; id: 0x7a686030 <x-coredata://7A87B5BE-C2FA-4150-B9E3-879FDE07F0B9/List/p1> ; data: {
name = "List 2";
})
Fetched list: <CheckLists.List: 0x7a686970> (entity: List; id: 0x7a686040 <x-coredata://7A87B5BE-C2FA-4150-B9E3-879FDE07F0B9/List/p3> ; data: {
name = "List 3";
})
Run Code Online (Sandbox Code Playgroud)
但是,当我从单元测试中执行此代码时,我得到此输出:
Fetched objects count: 3
Fetched object: <CheckLists.List: 0x7a07df50> (entity: List; id: 0x7a07d7e0 <x-coredata://7A87B5BE-C2FA-4150-B9E3-879FDE07F0B9/List/p2> ; data: {
name = "List 1";
})
Fetched object: <CheckLists.List: 0x7a07e190> (entity: List; id: 0x7a07d7f0 <x-coredata://7A87B5BE-C2FA-4150-B9E3-879FDE07F0B9/List/p1> ; data: {
name = "List 2";
})
Fetched object: <CheckLists.List: 0x7a07e1d0> (entity: List; id: 0x7a07d800 <x-coredata://7A87B5BE-C2FA-4150-B9E3-879FDE07F0B9/List/p3> ; data: {
name = "List 3";
})
Fetched lists array is nil
Run Code Online (Sandbox Code Playgroud)
我希望它更容易理解问题所在.不知何故,这句话:
let fetchedLists: [List]? = fetchedResultsController.fetchedObjects as? [List]
Run Code Online (Sandbox Code Playgroud)
List当app在模拟器中运行时会生成一个对象数组,但是nil从单元测试执行时它会生成失败.
该问题与目标配置有关。我已经通过一些解决方法解决了这个问题。
以前,为了使List实体类在我的单元测试目标中可访问,我已将其添加到此目标中。所以,List班级有两个目标。事实上,ListSwift 已知有两个类,每个目标对应一个类:MyAppTarget.List和MyUnitTestTarget.List。Retched 结果控制器返回对象数组MyAppTarget.List,但在单元测试目标中,List被假定为MyUnitTestTarget.List类。这行代码就是这样:
let fetchedLists: [List]? = fetchedResultsController.fetchedObjects as? [List]
Run Code Online (Sandbox Code Playgroud)
从单元测试目标执行时生成nil,而不是从主目标执行时生成的正确数组。为了解决这个问题,我将其更改为:
let fetchedLists: [MyAppTarget.List]? = fetchedResultsController.fetchedObjects as? [MyAppTarget.List]
Run Code Online (Sandbox Code Playgroud)
并将List课程公开。更改后,它按预期工作。
但仍然对我来说有点令人困惑,MyAppTarget.List无法转换为MyUnitTestTarget.List. 此外,这意味着我需要公开每个实体NSManagedObject子类才能在单元测试中使用它。到目前为止我还没有找到更好的解决方案。
也许有更好的方法来解决这个问题。我没有看到一个选项来告诉NSFetchedResultsController它应该返回MyAppTarget.List主目标和MyUnitTestTarget.List单元测试目标。它将始终使用.xcdatamodeld给定实体的文件中的配置。另外,即使有一种方法可以强制转换MyAppTarget.List到MyUnitTestTarget.List单元测试内部,它仍然要求该类List是公共的。
更新:
NSFetchedResultsController我找到了一种更改运行时返回的实体类的方法。这是一个更清晰和简单的解决方案: https: //stackoverflow.com/a/25858758/514181
它允许在单元测试中无缝使用 CoreData 实体,而无需强制转换或公开实体类。