81

我正在尝试使用 TDD 和新的 XCTest 框架编写一个 iOS 应用程序。我的一种方法从 Internet 检索文件(给定一个 NSURL 对象)并将其存储在用户的文档中。该方法的签名类似于:

- (void) fetchAndStoreImage:(NSURL *)imageUrl

我正在尝试以一种在没有连接到 Internet 时不会失败的方式编写此方法的测试。我的方法(取自上一个问题)是使用 NSURL 调用本地文件系统中的图像的方法。

创建启用了单元测试的新项目时,Tests 目录有一个名为“Supporting Files”的子目录。我想那是我的测试图像应该去的地方。我的问题是如何获取指向此目录中图像的 NSURL 对象,因为我不希望将测试图像与应用程序捆绑在一起。任何帮助表示赞赏。

4

8 回答 8

133

实际上,[NSBundle mainBundle]运行单元测试时的路径不是你的应用程序的路径,而是/Developer/usr/bin,所以这行不通。

在单元测试中获取资源的方法在这里:OCUnit & NSBundle

简而言之,使用:

[[NSBundle bundleForClass:[self class]] resourcePath]

或者在你的情况下:

[[NSBundle bundleForClass:[self class]] resourceURL]
于 2013-10-28T13:44:08.807 回答
60

斯威夫特 5.3

注意:Swift 5.3 包括包管理器资源 SE-0271功能,可用于应用程序包和测试包资源。

资源并不总是供包的客户使用;资源的一种用途可能包括仅单元测试需要的测试夹具。此类资源不会与库代码一起合并到包的客户端中,而只会在运行包的测试时使用。

斯威夫特 4、5:

let testBundle = Bundle(for: type(of: self))
guard var fileUrl = testBundle.url(forResource: "imageName", withExtension: "png") 
  else { fatalError() }

// path approach
guard let filePath = testBundle.path(forResource: "dataName", ofType: "csv")
  else { fatalError() }
let fileUrl = URL(fileURLWithPath: filePath)

Bundle 提供了发现配置的主要路径和测试路径的方法:

@testable 
import Example

class ExampleTests: XCTestCase {
    
  func testExample() {
    let bundleMain = Bundle.main
    let bundleDoingTest = Bundle(for: type(of: self ))
    let bundleBeingTested = Bundle(identifier: "com.example.Example")!
    
    print("bundleMain.bundlePath : \(bundleMain.bundlePath)")
    // …/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/Library/Xcode/Agents
    print("bundleDoingTest.bundlePath : \(bundleDoingTest.bundlePath)")
    // …/PATH/TO/Debug/ExampleTests.xctest
    print("bundleBeingTested.bundlePath : \(bundleBeingTested.bundlePath)")
    // …/PATH/TO/Debug/Example.app
    
    print("bundleMain = " + bundleMain.description) // Xcode Test Agent
    print("bundleDoingTest = " + bundleDoingTest.description) // Test Case Bundle
    print("bundleUnderTest = " + bundleBeingTested.description) // App Bundle

Xcode URL 将Developer/Xcode/DerivedData类似于...

file:///Users/
  UserName/
    Library/
      Developer/
        Xcode/
          DerivedData/
            App-qwertyuiop.../
              Build/
                Products/
                  Debug-iphonesimulator/
                    AppTests.xctest/
                      imageName.png

...与Developer/CoreSimulator/DevicesURL不同

file:///Users/
  UserName/
    Library/
    Developer/
      CoreSimulator/
        Devices/
          _UUID_/
            data/
              Containers/
                Bundle/
                  Application/
                    _UUID_/
                      App.app/

另请注意,默认情况下,单元测试可执行文件与应用程序代码链接。但是,单元测试代码应该只在测试包中具有 Target Membership。应用程序代码应仅在应用程序包中具有目标成员资格。在运行时,单元测试目标包被注入到应用程序包中执行

Swift 包管理器 (SPM) 4:

let testBundle = Bundle(for: type(of: self)) 
print("testBundle.bundlePath = \(testBundle.bundlePath) ")

注意:默认情况下,命令行将swift test创建一个MyProjectPackageTests.xctest测试包。并且,swift package generate-xcodeproj将创建一个MyProjectTests.xctest测试包。这些不同的测试包有不同的路径此外,不同的测试包可能有一些内部目录结构和内容差异

在任何一种情况下,.bundlePathand.bundleURL都会返回当前在 macOS 上运行的测试包的路径。但是,Bundle目前尚未为 Ubuntu for Swift 4 实现。

另外,Swift 4 命令行swift buildswift test没有提供复制资源的机制。

但是,通过一些努力,可以在 macOS Xcode、macOS 命令行和 Ubuntu 命令行环境中设置使用 Swift Package Manger 的进程。一个例子可以在这里找到:004.4'2 SW Dev Swift Package Manager (SPM) With Resources Qref

于 2015-06-05T19:04:45.873 回答
15

只是为了附加到正确的答案,这是如何获取 UnitTests/Supporting Files 中文件的 filePath 的示例:

NSString *filePath = [[[NSBundle bundleForClass:[self class]] resourcePath] stringByAppendingPathComponent:@"YourFileName.json"];
XCTAssertNotNil(filePath);

这可能对某人有用。

于 2014-11-13T18:49:10.700 回答
3
  1. 您可以像在任何目标中一样引用捆绑文件。
  2. 检查文件是否在 Copy Bundle Resources 构建阶段(您的测试目标)中被复制
  3. 要访问本地文件:

    NSURL*imageUrl=[[NSBundle mainBundle]URLForResource:@"imageName" withExtension:@"png"];
    

您可以使用以下方法进行异步访问并等待响应:https ://github.com/travisjeffery/TRVSMonitor

如果您dataset1.json在测试目标 (2) 中添加了:

NSString *p=[[NSBundle mainBundle] pathForResource:@"dataset1" ofType:@"json"];
NSLog(@"%@",p);

2013-10-29 15:49:30.547 PlayerSample[13771:70b] WT(0):/Users/bpds/Library/Application Support/iPhone Simulator/7.0/Applications/7F78780B-684A-40E0-AA35-A3B5D8AA9DBD/PlayerSample。 app.app/dataset1.json

于 2013-10-28T10:04:12.160 回答
2

Swift基于接受的答案的版本:

let url = URL(fileURLWithPath: Bundle(for: type(of: self)).path(forResource: "my", ofType: "json") ?? "TODO: Proper checking for file")
于 2019-06-20T15:38:13.717 回答
1

这是它的 Swift 版本、Xcode 7、iOS 9 等。

let testBundle = NSBundle(forClass: self.dynamicType)
let path = testBundle.pathForResource("someImage", ofType: "jpg")
XCTAssertNotNil(path)

注意: someImage.jpg 必须包含在您的测试目标中。


编辑:斯威夫特 5

let testBundle = Bundle(for: type(of: self))
let path = testBundle.path(forResource: "someImage", ofType: "jpg")
XCTAssertNotNil(path)
于 2015-12-30T00:06:45.590 回答
1

我面临的问题是应用程序代码试图访问主包以获取诸如 bundleIdentifier 之类的东西,并且由于主包不是我的单元测试项目,它会返回 nil。

一个适用于 Swift 3.0.1 和 Objective-C 的解决方法是在 NSBundle 上创建一个 Objective-C 类别并将其包含在您的单元测试项目中。您不需要桥接头或任何东西。该类别将被加载,现在当您的应用程序代码请求主包时,您的类别将返回单元测试包。

@interface NSBundle (MainBundle)

+(NSBundle *)mainBundle;

@end

@implementation NSBundle (Main)

#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wobjc-protocol-method-implementation"
+(NSBundle *)mainBundle
{
    return [NSBundle bundleForClass:[SomeUnitTest class]];
}
#pragma clang diagnostic pop

@end
于 2016-12-18T22:46:37.473 回答
0

斯威夫特 5

let testBundle = Bundle(for: self)

在上面的一些答案中发现,使用let testBundle = Bundle(for: type(of: self))不会编译,而是始终产生Segmentation fault: 11的错误。

于 2020-08-14T22:36:27.163 回答