Джерси 3 с CDI Weld 4 вместо HK2

У меня есть небольшой API JAX-RS (Jersey 3.0.1), работающий на автономном Jetty 11, но я хочу использовать Weld 4.0.0 вместо HK2. Jetty правильно загружает контекст Weld (модуль Jetty ci-decorate был установлен, как описано в Документация по сварке)

WELD-ENV-001212: Jetty CdiDecoratingListener support detected, CDI injection will be available in Listeners, Servlets and Filters.

Когда дело доходит до вызова конечной точки API, возникает исключение:

WARNING: The following warnings have been detected: WARNING: Unknown HK2 failure detected:
MultiException stack 1 of 1
org.jboss.weld.exceptions.IllegalStateException: WELD-001304: More than one context active for scope type jakarta.enterprise.context.RequestScoped
    at org.jboss.weld.manager.BeanManagerImpl.internalGetContext(BeanManagerImpl.java:672)
    at org.jboss.weld.manager.BeanManagerImpl.isContextActive(BeanManagerImpl.java:658)
    at org.jboss.weld.bean.ManagedBean.create(ManagedBean.java:163)
    at org.jboss.weld.contexts.unbound.DependentContextImpl.get(DependentContextImpl.java:64)
    at org.jboss.weld.bean.ContextualInstanceStrategy$DefaultContextualInstanceStrategy.get(ContextualInstanceStrategy.java:100)
    at org.jboss.weld.bean.ContextualInstance.get(ContextualInstance.java:50)
    at org.jboss.weld.manager.BeanManagerImpl.getReference(BeanManagerImpl.java:694)
    at org.jboss.weld.manager.BeanManagerImpl.getReference(BeanManagerImpl.java:717)
    at org.jboss.weld.util.ForwardingBeanManager.getReference(ForwardingBeanManager.java:64)
    at org.jboss.weld.bean.builtin.BeanManagerProxy.getReference(BeanManagerProxy.java:87)
    at org.glassfish.jersey.ext.cdi1x.internal.CdiUtil.getBeanReference(CdiUtil.java:127)
    at org.glassfish.jersey.ext.cdi1x.internal.AbstractCdiBeanSupplier$1.getInstance(AbstractCdiBeanSupplier.java:69)
    at org.glassfish.jersey.ext.cdi1x.internal.AbstractCdiBeanSupplier._provide(AbstractCdiBeanSupplier.java:103)
    at org.glassfish.jersey.ext.cdi1x.internal.RequestScopedCdiBeanSupplier.get(RequestScopedCdiBeanSupplier.java:46)
    at org.glassfish.jersey.inject.hk2.InstanceSupplierFactoryBridge.provide(InstanceSupplierFactoryBridge.java:53)
    at org.jvnet.hk2.internal.FactoryCreator.create(FactoryCreator.java:129)

Мои зависимости в pom.xml выглядят следующим образом:

        <dependency>
            <groupId>org.jboss.weld.servlet</groupId>
            <artifactId>weld-servlet-core</artifactId>
            <version>4.0.0.Final</version>
        </dependency>
        <dependency>
            <groupId>jakarta.servlet</groupId>
            <artifactId>jakarta.servlet-api</artifactId>
            <version>5.0.0</version>
            <scope>provided</scope>
        </dependency>
        <dependency>
            <groupId>jakarta.ws.rs</groupId>
            <artifactId>jakarta.ws.rs-api</artifactId>
            <version>3.0.0</version>
        </dependency>
        <dependency>
            <groupId>org.glassfish.jersey.containers</groupId>
            <artifactId>jersey-container-servlet</artifactId>
            <version>3.0.1</version>
        </dependency>
        <dependency>
            <groupId>org.glassfish.jersey.inject</groupId>
            <artifactId>jersey-hk2</artifactId>
            <version>3.0.1</version>
        </dependency>
        <dependency>
            <groupId>org.glassfish.jersey.ext.cdi</groupId>
            <artifactId>jersey-weld2-se</artifactId>
            <version>3.0.1</version>
        </dependency>

Для контекста Weld существует пустой файл beans.xml в папке webapp/WEB-INF/beans.xml, объявленной с помощью bean-discovery-mode="all". Далее показан класс конечной точки JAX-RS:

import de.test.weld.service.AddressRepository;
import jakarta.inject.Inject;
import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.MediaType;

@Path("address")
public class AddressApi {

    @Inject
    private AddressRepository addressRepository;

    @GET
    @Produces(MediaType.APPLICATION_JSON)
    public String getAddress() {
        return "{\"street\": \"" + addressRepository.getStreet(0) + "\", \"postcode\":  \"" +
                addressRepository.getPostcode(0) + "\", \"city\": \"" + addressRepository.getCity(0) + "\"}";
    }
}

Компонент AddressRepositoryImpl, который я хочу внедрить, имеет аннотацию @jakarta.inject.Singleton.

Буду признателен за любой намек, чтобы он работал с CDI Weld.


person kladderradatsch    schedule 15.03.2021    source источник


Ответы (1)


Проблема была в неправильной зависимости

<dependency>
   <groupId>org.glassfish.jersey.ext.cdi</groupId>
   <artifactId>jersey-weld2-se</artifactId>
   <version>3.0.1</version>
</dependency>

который существует только для сред, отличных от JEE, то есть для Java Standard Edition (SE) или не-веб-приложений.

Тогда следующая зависимость все еще отсутствовала:

 <dependency>
     <groupId>org.glassfish.jersey.ext.cdi</groupId>
     <artifactId>jersey-cdi1x</artifactId>
     <version>3.0.1</version>
 </dependency>

Решение относится к: проблемам Jersey Github

person kladderradatsch    schedule 15.03.2021