我遇到 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实现)。
为什么可以召唤两次? 好像我会要求它提高请求和响应。
答案 0 :(得分:1)
问题在于,在新泽西州,他们从hk2转移到其上方的抽象层,并使代码可以识别Java 8(因为它是Java EE 8的实现 - https://jersey.github.io/release-notes/2.26.html)。
所以为了解决这个问题我必须:
导入正确的AbstractBinder
- org.glassfish.jersey.internal.inject.AbstractBinder
(不上一个来自HK2 - org.glassfish.hk2.utilities.binding.AbstractBinder
)。
更新工厂以实施Supplier<User>
而不是Factory<User>
。
在此之后,一切都像以前一样工作,工厂只被调用一次。