2

Guice/Gin を使用して新しいプロジェクトを設計したので、特にテスト時にコードをよりモジュール化してスワップ可能にすることができました。

ただし、実際にこれを機能させる方法を見つけることができません。テストで新しい Gin/Guice モジュールを作成し、「ベース」モジュールをインストールして、特定のテスト実装に置き換えたいバインディングをオーバーロードするだけでよいという印象を受けました。

GWTTestCase を使用してモジュール全体をロードする必要はありません。実行する必要がある詳細なテストの種類が非常に遅く、不必要だからです。

Jukito ( http://code.google.com/p/jukito/ )、gwt-test-utils ( http://code.google.com/p/gwt-test-utils/wiki/HowToUseWithGIN )を使用してみましたまた、guice を使用してこれを行うためのリソースもいくつかあります ( http://fabiostrozzi.eu/2011/03/27/junit-tests-easy-guice/ )。

これらの方法はどれも結果をもたらしていません。

Ginモジュールのミラーguiceモジュールを定義した場合、Guiceアプローチが機能する可能性があると思います。ただし、これらの両方を管理する必要はありません。人々がGuiceでテストすると仮定するように、GINモジュールをテストしたいだけです。

これは本当にシンプルであるべきだと思いますが、うまくいく例を誰か教えてもらえますか?

アップデート

この質問の別の見方は次のとおりです。

注入しているクラスが外部 Gin モジュールにある場合、Jukito サイト ( http://code.google.com/p/jukito/ )の例を取得するにはどうすればよいですか?

**更新 - Thomas Boyer の回答を参照して**

トムのヒントをありがとう、私はアダプターを使用する例を見つけることができませんでしたが、とにかく GinModuleAdapter を使用するために Jukito の例を拡張しようとしました:

@RunWith(JukitoRunner.class)
public class MyGinTest {

  public static class Module extends JukitoModule {
    protected void configureTest() {
        install(new GinModuleAdapter(new ClientModule()));
    }
  }

  @Test
  @Inject
  public void testAdd(SyncedDOMModel mod){
      assertNotNull(mod);
  }
}

このテストを実行しようとしたときに、次の例外を受け取りました。

java.lang.AssertionError: should never be actually called
    at com.google.gwt.inject.rebind.adapter.GwtDotCreateProvider.get(GwtDotCreateProvider.java:43)
    at com.google.inject.internal.InternalFactoryToProviderAdapter.get(InternalFactoryToProviderAdapter.java:40)
    at com.google.inject.internal.ProviderToInternalFactoryAdapter$1.call(ProviderToInternalFactoryAdapter.java:46)
    at com.google.inject.internal.InjectorImpl.callInContext(InjectorImpl.java:1031)
    at com.google.inject.internal.ProviderToInternalFactoryAdapter.get(ProviderToInternalFactoryAdapter.java:40)
    at com.google.inject.Scopes$1$1.get(Scopes.java:65)
    at com.google.inject.internal.InternalFactoryToProviderAdapter.get(InternalFactoryToProviderAdapter.java:40)
    at com.google.inject.internal.InternalInjectorCreator$1.call(InternalInjectorCreator.java:204)
    at com.google.inject.internal.InternalInjectorCreator$1.call(InternalInjectorCreator.java:198)
    at com.google.inject.internal.InjectorImpl.callInContext(InjectorImpl.java:1024)
    at com.google.inject.internal.InternalInjectorCreator.loadEagerSingletons(InternalInjectorCreator.java:198)
    at com.google.inject.internal.InternalInjectorCreator.injectDynamically(InternalInjectorCreator.java:179)
    at com.google.inject.internal.InternalInjectorCreator.build(InternalInjectorCreator.java:109)
    at com.google.inject.Guice.createInjector(Guice.java:95)
    at com.google.inject.Guice.createInjector(Guice.java:72)
    at com.google.inject.Guice.createInjector(Guice.java:62)
    at org.jukito.JukitoRunner.ensureInjector(JukitoRunner.java:118)
    at org.jukito.JukitoRunner.computeTestMethods(JukitoRunner.java:177)
    at org.jukito.JukitoRunner.validateInstanceMethods(JukitoRunner.java:276)
    at org.junit.runners.BlockJUnit4ClassRunner.collectInitializationErrors(BlockJUnit4ClassRunner.java:102)
    at org.junit.runners.ParentRunner.validate(ParentRunner.java:344)
    at org.junit.runners.ParentRunner.<init>(ParentRunner.java:74)
    at org.junit.runners.BlockJUnit4ClassRunner.<init>(BlockJUnit4ClassRunner.java:55)
    at org.jukito.JukitoRunner.<init>(JukitoRunner.java:72)

私の gin モジュールは GWTP プロジェクトの一部であり、次のようになります。

public class ClientModule extends AbstractPresenterModule {

    @Override
    protected void configure() {
        install(new DefaultModule(ClientPlaceManager.class));

        bindPresenter(MainPagePresenter.class, MainPagePresenter.MyView.class,
                MainPageView.class, MainPagePresenter.MyProxy.class);

        bindConstant().annotatedWith(DefaultPlace.class).to(NameTokens.main);

        bindPresenterWidget(MapTreePresenter.class,
                MapTreePresenter.MyView.class, MapTreeView.class);

        bindPresenterWidget(MapTreeItemPresenter.class,
                MapTreeItemPresenter.MyView.class, MapTreeItemView.class);

        bind(ResourcePool.class).to(DefferredResourcePool.class);

        bind(WebSocket.class).to(WebSocketImpl.class);

    }

}

ご覧のとおり、 test に挿入するクラスは、モジュールでバインドするSyncedDOMModela を使用します。WebSocketテスト中は、実際の WebSocket とサーバーを使用したくありません。したがって、基本的にすべてをエミュレートするクラスを使用して、テストでそのバインディングをオーバーロードしたいと考えています。この場合、モックを使用するよりも、WebSocket の別の実装を挿入する方が簡単です。

参考になれば、これは SyncedDOMMOdel クラスの基本的な概要です。

public class SyncedDOMMOdel {
 ....
    @Inject
    public SyncedDOMModel(WebSocket socket){
        this.socket = socket;
    }

 ....
}
4

1 に答える 1

2

を使用しGinModuleAdapterて、任意GinModuleのものをGuiceとして使用できますModule

明らかに、GINの特定の機能の恩恵を受けることはありません。デフォルトでGWT.create()は、特定のバインディングがない場合(これには、Guiceでスローされるインターフェイスと抽象クラスが含まれます)、名前が終了するインターフェイスに特定の機能がないRemoteService場合は、インターフェイスが自動的に検索されます。Asyncバインディング。また、単体テスト以外の場合と同様に、JSNIまたは遅延バインディング()に
依存するものを使用することはできません。GWT.create()GWTTestCase

于 2013-01-26T22:00:59.610 に答える