jersey テスト フレームワークを使用して、JAX-RS レストフル API をテストすることができました。しかし、何を試しても、EJB を適切にモックすることはできません。
@peeskiletによって提供された例を使用すると、私の EJB がリソースで参照されるポイントは null のままです。
だから私はそれが私の依存関係ではないことを知っています。問題は、EJB がネストされていることだと思います。このコードの動作に影響を与えている可能性があると確信している要因がいくつかありますが、対処方法がわかりません。 1.モックしようとしている ApplicationService はインターフェイスであり、 ApplicationServiceImpl クラスがあります。その中に実際のロジックがあります。私のテストはそのロジックに焦点を当てていないので、私の思考プロセスはインターフェイスをモックすることで十分でした。2. ただし、インターフェースの実装には、MongoDB シングルトンに接続する EJB も含まれています。
私の質問: これらの要因のいずれかが、リソース クラスのインターフェイス参照が常に null であり、明らかにモックとインジェクションに失敗する理由として、ここで役割を果たしている可能性がありますか? もしそうなら、どうすれば修正できますか?
インターフェイスではなく実装を直接モックするように切り替える必要がある場合、DAO ejb もモックする必要がありますか?
明確にするために、EJB を介して注入されたフィールド/オブジェクトに依存するメソッドを呼び出さない限り、テスト フレームの作業はまったく問題なく動作します。
マイ リソース クラス
@Path("/stats")
@Api(value = "stats", authorizations = {
@Authorization(value = "apiKey", scopes = {})
})
@Stateless
@Produces({MediaType.APPLICATION_JSON})
public class StatsResource {
private final static Logger LOG = Logger.getLogger(StatsResource.class.getName());
@EJB
ApplicationService applicationService;
public void setApplicationService(ApplicationService applicationService) {
this.applicationService = applicationService;
}
@GET
public String getHere(){
return "Got Here!";
}
@GET
@Path("test")
@ApiOperation(value = "Finds all applications",
notes = "Returns brief display which uses a limited subset of Application.class fields",
response = Application.class,
responseContainer = "List<Application>"
)
@JsonView(Views.Brief.class)
@Asynchronous
public void getApplications(@Suspended
final AsyncResponse asyncResponse) throws Exception {
asyncResponse.resume(doGetApplications());
}
private ApplicationList doGetApplications() throws Exception {
return new ApplicationList(applicationService.getAll());
}
}
私のテストケース
public class StatsResourceNGTest extends JerseyTestNg.ContainerPerMethodTest {
@Mock
private ApplicationService applicationService;
@Override
protected Application configure() {
enable(TestProperties.LOG_TRAFFIC);
enable(TestProperties.DUMP_ENTITY);
MockitoAnnotations.initMocks(this);
Logger.getLogger(StatsResourceNGTest.class.getName()).log(Level.INFO, "Mock App Service: {0}", applicationService.toString());
AbstractBinder binder = new AbstractBinder() {
@Override
protected void configure() {
bindFactory(MockApplicationServiceFactory.class)
.to(ApplicationResource.class);
}
};
ResourceConfig config = new IdentityRestApplication(true);
config.register(binder);
config.register(StatsResource.class);
return config;
}
@Test
public void assertMockInjectionWorked() throws Exception {
Response response = target("/stats/test").request(MediaType.APPLICATION_JSON).get();
verify(applicationService, times(1)).getAll();
Logger.getLogger(ApplicationResourceNGTest.class.getName()).log(Level.INFO, "Status: {0}", response.getStatus());
String msg = response.readEntity(String.class);
Logger.getLogger(ApplicationResourceNGTest.class.getName()).log(Level.INFO, "Response: {0}", msg);
Assert.assertNotNull(msg);
response.close();
}
}
私のHK2 MockFactory
public class MockApplicationServiceFactory implements Factory<ApplicationService> {
private List<Application> appList;
@Override
public ApplicationService provide() {
appList = new ArrayList<>();
for (int i = 1; i < 6; i++) {
String appname = "App " + i;
Application app = new ApplicationBuilder()
.setName(appname)
.setDescription(appname + " Description")
.setTenantId(new ObjectId())
.createApplication();
appList.add(app);
}
try {
final ApplicationService mockedService = Mockito.mock(ApplicationService.class);
Mockito.when(mockedService.getAll()).thenReturn(appList);
return mockedService;
} catch (ApplicationException ex) {
Logger.getLogger(ApplicationResourceNGTest.class.getName()).log(Level.SEVERE, null, ex);
} catch (Exception ex) {
Logger.getLogger(ApplicationResourceNGTest.class.getName()).log(Level.SEVERE, null, ex);
}
return null;
}
@Override
public void dispose(ApplicationService t) {
}
}