1

編集:これはすべて、他のリンクフラグ設定のタイプミスが原因でした。詳細については、以下の私の回答を参照してください。


iOS View Controllerのテスト中にUIWebViewのメソッドが呼び出されていることを確認できるように、UIWebViewをモックしようとしています。私は、SVNリビジョン70(この質問の時点で最新)から構築されたOCMock静的ライブラリと、SVNのGoogle Toolbox for Mac(GTM)ユニットテストフレームワーク、リビジョン410を使用しています。ビューコントローラが期待されるメソッドを呼び出そうとすると、次のエラーが発生します。

Test Case '-[FirstLookViewControllerTests testViewDidLoad]' started.
2010-11-11 07:32:02.272 Unit Test[38367:903] -[NSInvocation getArgumentAtIndexAsObject:]: unrecognized selector sent to instance 0x6869ea0
2010-11-11 07:32:02.277 Unit Test[38367:903] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[NSInvocation getArgumentAtIndexAsObject:]: unrecognized selector sent to instance 0x6869ea0'
*** Call stack at first throw:
(
    0   CoreFoundation                      0x010cebe9 __exceptionPreprocess + 185
    1   libobjc.A.dylib                     0x012235c2 objc_exception_throw + 47
    2   CoreFoundation                      0x010d06fb -[NSObject(NSObject) doesNotRecognizeSelector:] + 187
    3   CoreFoundation                      0x01040366 ___forwarding___ + 966
    4   CoreFoundation                      0x0103ff22 _CF_forwarding_prep_0 + 50
    5   Unit Test                           0x0000b29f -[OCMockRecorder matchesInvocation:] + 216
    6   Unit Test                           0x0000c1c1 -[OCMockObject handleInvocation:] + 111
    7   Unit Test                           0x0000c12a -[OCMockObject forwardInvocation:] + 43
    8   CoreFoundation                      0x01040404 ___forwarding___ + 1124
    9   CoreFoundation                      0x0103ff22 _CF_forwarding_prep_0 + 50
    10  Unit Test                           0x0000272a -[MyViewController viewDidLoad] + 100
    11  Unit Test                           0x0000926c -[MyViewControllerTests testViewDidLoad] + 243
    12  Unit Test                           0x0000537f -[SenTestCase invokeTest] + 163
    13  Unit Test                           0x000058a4 -[GTMTestCase invokeTest] + 146
    14  Unit Test                           0x0000501c -[SenTestCase performTest] + 37
    15  Unit Test                           0x000040c9 -[GTMIPhoneUnitTestDelegate runTests] + 1413
    16  Unit Test                           0x00003a87 -[GTMIPhoneUnitTestDelegate applicationDidFinishLaunching:] + 197
    17  UIKit                               0x00309253 -[UIApplication _callInitializationDelegatesForURL:payload:suspended:] + 1252
    18  UIKit                               0x0030b55e -[UIApplication _runWithURL:payload:launchOrientation:statusBarStyle:statusBarHidden:] + 439
    19  UIKit                               0x0030aef0 -[UIApplication _run] + 452
    20  UIKit                               0x0031742e UIApplicationMain + 1160
    21  Unit Test                           0x0000468c main + 104
    22  Unit Test                           0x000026bd start + 53
    23  ???                                 0x00000002 0x0 + 2
)
terminate called after throwing an instance of 'NSException'
/Users/gjritter/src/google-toolbox-for-mac-read-only/UnitTesting/RunIPhoneUnitTest.sh: line 151: 38367 Abort trap              "$TARGET_BUILD_DIR/$EXECUTABLE_PATH" -RegisterForSystemEvents

私のテストコードは次のとおりです。

- (void)testViewDidLoad {
    MyViewController *viewController = [[MyViewController alloc] init];

    id mockWebView = [OCMockObject mockForClass:[UIWebView class]];
    [[mockWebView expect] setDelegate:viewController];

    viewController.webView = mockWebView;

    [viewController viewDidLoad];
    [mockWebView verify];
    [mockWebView release];
}

私のビューコントローラコードは次のとおりです。

- (void)viewDidLoad {
    [super viewDidLoad];
    webView.delegate = self;
}

代わりに次を使用すると、テストが正常に実行されることがわかりました。

- (void)testViewDidLoad {
    MyViewController *viewController = [[MyViewController alloc] init];

    id mockWebView = [OCMockObject partialMockForObject:[[UIWebView alloc] init]];
    //[[mockWebView expect] setDelegate:viewController];

    viewController.webView = mockWebView;

    [viewController viewDidLoad];
    [mockWebView verify];
    [mockWebView release];
}

ただし、コメントアウトされた期待値を追加するとすぐに、部分モックを使用するとエラーが返されました。

同じプロジェクトでモックを正常に使用している他のテストがあります。

何か案は?UIKitオブジェクトのモックはOCMockでサポートされていますか?

編集:以下の回答のアドバイスに基づいて、次のテストを試しましたが、同じエラーが発生します:

- (void)testViewDidLoadLoadsWebView {
    MyViewController *viewController = [[MyViewController alloc] init];
    UIWebView *webView = [[UIWebView alloc] init];

    // This test fails in the same fashion with or without the next line commented
    //viewController.view;

    id mockWebView = [OCMockObject partialMockForObject:webView];
    // When I comment out the following line, the test passes
    [[mockWebView expect] loadRequest:[OCMArg any]];

    viewController.webView = mockWebView;

    [viewController viewDidLoad];
    [mockWebView verify];
    [mockWebView release];
}
4

2 に答える 2

4

UIKit クラスは謎めいた野獣であり、モックをいじくり回していると、何時間ものデバッグの楽しみにつながる可能性があることがわかりました。とはいえ、少しの忍耐でうまくいくことがわかりました。

コードで最初に気付くのは、コントローラーがテストでビューをロードしていないことです。私は通常、テストを実行する前に常にビューを強制的にロードするようにしています。もちろん、これは、Web ビューの初期化に対する期待値を記述できないことを意味しますが、この場合は実際に記述する必要はありません。あなたはこれを行うことができます:

- (void)testViewDidLoadSetsWebViewDelegateToSelf {
    MyViewController *viewController = [[MyViewController alloc] init];
    // Force the view to load
    viewController.view;

    assertThat(controller.webView.delegate, equalTo(controller));
}

とはいえ、後で Web ビューをモックしたい場合は、既存の Web ビューに部分モックを使用することをお勧めします。

- (void)testWebViewDoesSomething {
    MyViewController *viewController = [[MyViewController alloc] init];
    // Force the view to load
    viewController.view;

    id mockWebView = [OCMockObject partialMockForObject:controller.webView];
    [[mockWebView expect] someMethod];

    [controller doWhatever];

    [mockWebView verify];
}

実際、UIView サブクラスには常に部分モックを使用するのが最善であることがわかりました。UIView の完全なモックを作成すると、スーパービューに追加するなど、ビューに関連する何かをしようとすると、ほとんどの場合、めちゃくちゃに爆発します。

于 2010-11-13T21:15:55.520 に答える
3

これは、数十回見るまで気付かない、文字が 1 つずれている問題の 1 つであることが判明しました。

OCMockフォーラムのこの投稿によると、ユニット テスト ターゲットの Other Linker Flags を に設定しました-ObjC -forceload $(PROJECT_DIR)/Libraries/libOCMock.a。これは間違っています; -forceloadだったはず-force_loadです。このタイプミスを修正すると、テストが機能しました。

于 2010-11-16T03:11:21.707 に答える