UnsatisfiedDependencyException: не было объекта, доступного для инъекции в SystemInjecteeImpl
есть ошибки при использовании DI в приложении Джерси Rest:
org.glassfish.hk2.api.UnsatisfiedDependencyException: There was no object available for injection at SystemInjecteeImpl(requiredType=PricingService,parent=PricingResource,qualifiers={},position=0,optional=false,self=false,unqualified=null,1633188703)
Я совершенно новичок в этой концепции, и это кажется довольно сложным, так как некоторые примеры, кажется, устарели. Как я понимаю, есть несколько способов заставить DI работать: родной HK2, Spring/HK2 Bridge. Что проще и проще настроить? Как настроить программно (не поклонник XML) для Джерси 2.x?
ResourceConfig
import org.glassfish.jersey.server.ResourceConfig;
public class ApplicationConfig extends ResourceConfig {
public ApplicationConfig() {
register(new ApplicationBinder());
packages(true, "api");
}
}
AbstractBinder
public class ApplicationBinder extends AbstractBinder {
@Override
protected void configure() {
bind(PricingService.class).to(PricingService.class).in(Singleton.class);
}
}
PricingResource
@Path("/prices")
public class PricingResource {
private final PricingService pricingService;
@Inject
public PricingResource(PricingService pricingService) {
this.pricingService = pricingService;
}
@GET
@Produces(MediaType.APPLICATION_JSON)
public Collection<Price> findPrices() {
return pricingService.findPrices();
}
}
PricingService
@Singleton
public class PricingService {
// no constructors...
// findPrices() ...
}
обновление
public class Main {
public static final String BASE_URI = "http://localhost:8080/api/";
public static HttpServer startServer() {
return createHttpServerWith(new ResourceConfig().packages("api").register(JacksonFeature.class));
}
private static HttpServer createHttpServerWith(ResourceConfig rc) {
HttpServer httpServer = GrizzlyHttpServerFactory.createHttpServer(URI.create(BASE_URI), rc);
StaticHttpHandler staticHttpHandler = new StaticHttpHandler("src/main/webapp");
staticHttpHandler.setFileCacheEnabled(false);
staticHttpHandler.start();
httpServer.getServerConfiguration().addHttpHandler(staticHttpHandler);
return httpServer;
}
public static void main(String[] args) throws IOException {
System.setProperty("java.util.logging.config.file", "src/main/resources/logging.properties");
final HttpServer server = startServer();
System.out.println(String.format("Jersey app started with WADL available at "
+ "%sapplication.wadlnHit enter to stop it...", BASE_URI));
server.start();
System.in.read();
server.stop();
}
}
UPDATE3:
public class PricingResourceTest extends JerseyTest {
@Mock
private PricingService pricingServiceMock;
@Override
protected Application configure() {
MockitoAnnotations.initMocks(this);
enable(TestProperties.LOG_TRAFFIC);
enable(TestProperties.DUMP_ENTITY);
ResourceConfig config = new ResourceConfig(PricingResource.class);
config.register(new AbstractBinder() {
@Override
protected void configure() {
bind(pricingServiceMock).to(PricingService.class);
}
});
return config;
}
@Test
public void testFindPrices(){
when(pricingServiceMock.findPrices()).thenReturn(getMockedPrices());
Response response = target("/prices")
.request()
.get();
verify(pricingServiceMock).findPrices();
List<Price> prices = response.readEntity(new GenericType<List<Price>>(){});
// assertEquals("Should return status 200", 200, response.getStatus());
assertTrue(prices.get(0).getId() == getMockedPrices().get(0).getId());
}
private List<Price> getMockedPrices(){
List<Price> mockedPrices = Arrays.asList(new Price(1L, 12.0, 50.12, 12L));
return mockedPrices;
}
}
вывод JUnit:
INFO: 1 * Client response received on thread main
1 < 200
1 < Content-Length: 4
1 < Content-Type: application/json
[{}]
java.lang.AssertionError
во время отладки:
prices.get(0)
is Price
объект, который имеет null
присвоено всем полям.
UPDATE4:
добавил configure()
:
config.register(JacksonFeature.class);
config.register(JacksonJsonProvider.class);
теперь вывод Junit немного лучше:
INFO: 1 * Client response received on thread main
1 < 200
1 < Content-Length: 149
1 < Content-Type: application/json
[{"id":2,"recurringPrice":122.0,"oneTimePrice":6550.12,"recurringCount":2},{"id":2,"recurringPrice":122.0,"oneTimePrice":6550.12,"recurringCount":2}]
действительно имеет правильное число prices
но поля всех цен - это null. Это приводит к предположению, что проблема может заключаться в чтении сущности:
List<Price> prices = response.readEntity(new GenericType<List<Price>>(){});
вот как это исправить
изменить зависимость Moxy кому:
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-json-jackson</artifactId>
</dependency>
и добавьте аннотации к объекту "цена".
@XmlRootElement
@JsonIgnoreProperties(ignoreUnknown = true)
2 ответов
забыть InjectableProvider
. Тебе это не нужно. Проблема в том, что mock сервис не впрыскивается. Это тот, который создан платформой DI. Таким образом, вы проверяете изменения в макете сервиса, который никогда не был затронут.
Итак, что вам нужно сделать, это связать макет с платформой DI. Вы можете просто создать другой AbstractBinder
для тестирования. Это может быть простой анонимный, где вы свяжете макет
ResourceConfig config = new ResourceConfig(PricingResource.class);
config.register(new AbstractBinder() {
@Override
protected void configure() {
bind(pricingServiceMock).to(PricingService.class);
}
});
вот вы где просто привязка издевались службы. Таким образом, фреймворк введет макет в ресурс. Теперь, когда вы изменяете его в запросе, изменения будут видны в утверждении
О, и вам все равно нужно сделать свой when(..).then(..)
для инициализации данных в модели сервиса. Это также то, что вам не хватает
@Test
public void testFindPrices(){
Mockito.when(pricingServiceMock.findSomething()).thenReturn(list);
я исправил эту проблему, добавив следующую зависимость к моему приложению. группа компиляции: 'org.в GlassFish.Джерси.стеклотара.glassfish", название: 'jersey-gf-cdi', версия:'2.14'
тогда не нужно было никаких "AbstractBinder" кода.