Jersey工厂尝试两次创建@Context变量

时间:2018-03-23 16:16:12

标签: java spring spring-boot jersey spring-jersey

我遇到 Jersey 2.26 Spring Boot 2 的问题。

我添加了一个工厂来将@Context变量注入到方法中,但它会在方法之前执行两次,然后再在方法之后执行:

@GET
@Path("/user-entitlements")
public Set<String> getUserEntitlements(@Context User user) {
    return service.getUserEntitlements(user);
}

我有一个创建这些用户的工厂:

public class UserFactory implements Factory<User> {

    private final User user;

    @Inject
    public UserFactory(HttpServletRequest request, RequestIdentityService requestIdentityService) {
        String userName = requestIdentityService.getCurrentUser(request);
        user = new User(userName);
    }

    @Override
    public User provide() {
        return user;
    }

    @Override
    public void dispose(User instance) {
    }
}

在球衣配置中我注册了这个工厂:

    register(
        new AbstractBinder() {
            @Override
            public void configure() {
                bindFactory(UserFactory.class)
                    .to(User.class)
                    .in(RequestScoped.class);
            }
        }
    );

之前它运行良好,但在我将应用程序升级到jersey 2.26和spring boot 2之后,它停止了工作并抛出异常:

java.lang.IllegalStateException: Proxiable context org.glassfish.jersey.inject.hk2.RequestContext@3a94c454 findOrCreate returned a null for descriptor SystemDescriptor(
    implementation=org.glassfish.jersey.inject.hk2.SupplierFactoryBridge
    contracts={javax.servlet.http.HttpServletRequest}
    scope=org.glassfish.jersey.process.internal.RequestScoped
    qualifiers={}
    descriptorType=PROVIDE_METHOD
    descriptorVisibility=NORMAL
    metadata=
    rank=0
    loader=org.glassfish.hk2.utilities.binding.AbstractBinder$2@464366a8
    proxiable=true
    proxyForSameScope=false
    analysisName=null
    id=20
    locatorId=0
    identityHashCode=1360647282
    reified=true) and handle ServiceHandle(SystemDescriptor(
    implementation=com.rest.UserFactory
    contracts={org.glassfish.hk2.api.Factory}
    scope=org.glassfish.hk2.api.PerLookup
    qualifiers={}
    descriptorType=CLASS
    descriptorVisibility=NORMAL
    metadata=
    rank=0
    loader=org.glassfish.hk2.utilities.binding.AbstractBinder$2@33335025
    proxiable=null
    proxyForSameScope=null
    analysisName=null
    id=172
    locatorId=0
    identityHashCode=790201459
    reified=true),1915928862)

异常显示,因为UserFactory被调用两次。首先在调用getUserEntitlements之前调用它,以注入User。 但是在该方法返回之后,它再次被调用(这次它获得了一个不允许某些操作的HttpRequest实现)。

为什么可以召唤两次? 好像我会要求它提高请求和响应。

1 个答案:

答案 0 :(得分:1)

问题在于,在新泽西州,他们从hk2转移到其上方的抽象层,并使代码可以识别Java 8(因为它是Java EE 8的实现 - https://jersey.github.io/release-notes/2.26.html)。

所以为了解决这个问题我必须:

  1. 导入正确的AbstractBinder - org.glassfish.jersey.internal.inject.AbstractBinder上一个来自HK2 - org.glassfish.hk2.utilities.binding.AbstractBinder)。

  2. 更新工厂以实施Supplier<User>而不是Factory<User>

  3. 在此之后,一切都像以前一样工作,工厂只被调用一次。