"Error creating bean with name 'scopedTarget.get'" 收到 RabbitMQ 消息时
"Error creating bean with name 'scopedTarget.get'" when receiving a RabbitMQ message
当我没有请求范围时,我在忽略 @Scope
时遇到了一些问题。我可以解释。
我正在使用 Hibernate Envers 在审计中保存用户的一些信息 table。这在存在请求(来自我的 Rest Controller)时效果很好,因为用户需要登录,所以在这种情况下总能找到用户。
但是,我也在使用 RabbitMQ。当我收到一些需要对数据库执行某些操作的消息时(例如,插入一个实体),我的代码尝试从应用程序上下文加载 User
bean。 User
Bean 与 @Scope
请求绑定,而 Spring 找不到线程绑定请求。
因此,我收到错误消息:
Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'scopedTarget.get': Scope 'request' is not active for the current thread; consider defining a scoped proxy for this bean if you intend to refer to it from a singleton; nested exception is java.lang.IllegalStateException: No thread-bound request found: Are you referring to request attributes outside of an actual web request, or processing a request outside of the originally receiving thread? If you are actually operating within a web request and still receive this message, your code is probably running outside of DispatcherServlet/DispatcherPortlet: In this case, use RequestContextListener or RequestContextFilter to expose the current request.
at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:355)
at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:197)
at org.springframework.aop.target.SimpleBeanTargetSource.getTarget(SimpleBeanTargetSource.java:35)
at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.getTarget(CglibAopProxy.java:705)
at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:655)
at com.domain.config.UserConfiguration$User$$EnhancerBySpringCGLIB$$c7e6e64e.getUserId(<generated>)
at com.domain.audit.RevisionEntityListener.newRevision(RevisionEntityListener.java:28)
at org.hibernate.envers.internal.revisioninfo.DefaultRevisionInfoGenerator.generate(DefaultRevisionInfoGenerator.java:93)
at org.hibernate.envers.internal.synchronization.AuditProcess.getCurrentRevisionData(AuditProcess.java:114)
at org.hibernate.envers.internal.synchronization.AuditProcess.executeInSession(AuditProcess.java:96)
at org.hibernate.envers.internal.synchronization.AuditProcess.doBeforeTransactionCompletion(AuditProcess.java:153)
at org.hibernate.envers.internal.synchronization.AuditProcessManager.doBeforeTransactionCompletion(AuditProcessManager.java:46)
at org.hibernate.engine.spi.ActionQueue$BeforeTransactionCompletionProcessQueue.beforeTransactionCompletion(ActionQueue.java:928)
... 38 common frames omitted
Caused by: java.lang.IllegalStateException: No thread-bound request found: Are you referring to request attributes outside of an actual web request, or processing a request outside of the originally receiving thread? If you are actually operating within a web request and still receive this message, your code is probably running outside of DispatcherServlet/DispatcherPortlet: In this case, use RequestContextListener or RequestContextFilter to expose the current request.
at org.springframework.web.context.request.RequestContextHolder.currentRequestAttributes(RequestContextHolder.java:131)
at org.springframework.web.context.request.AbstractRequestAttributesScope.get(AbstractRequestAttributesScope.java:41)
at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:340)
... 50 common frames omitted
我想忽略这种情况,因为在这种情况下我的意图是用 "system" 之类的东西填充 "user" 信息。表明此更改不是由用户直接进行的(因为没有涉及用户)。
我的代码如下。
RevisionEntityListener
(Hibernate Envers):
@Component
public class RevisionEntityListener implements RevisionListener, ApplicationContextAware {
private static ApplicationContext applicationContext;
@Override
public void newRevision(Object revisionEntity) {
RevEntity revEntity = (RevEntity) revisionEntity;
User user = applicationContext.getBean(User .class);
revEntity.setUserId(user.getUserId());
// code
}
private static void setStaticApplicationContext(ApplicationContext context) {
applicationContext = context;
}
@Override
public void setApplicationContext(ApplicationContext applicationContext)
throws BeansException {
setStaticApplicationContext(applicationContext);
}
}
我的UserConfiguration
:
@Configuration
public class UserConfiguration {
@Autowired
private OAuth2ClientContext context;
@Bean
@Scope(proxyMode = ScopedProxyMode.TARGET_CLASS, scopeName = "request")
public User get() {
User user = null;
try {
String token = context.getAccessToken().getValue();
// code
return user;
}
@Getter
@Setter
@ToString
public static class User {
private Long userId;
// all user information
}
}
我不知道这是否是最佳选择,但我创建了一个 if
来查看 SecurityContextHolder.getContext().getAuthentication()
是否存在:
User user;
if (SecurityContextHolder.getContext().getAuthentication() != null) {
user = applicationContext.getBean(User .class);
} else {
user = createSystemUser();
}
revEntity.setUserId(user.getUserId());
当我没有请求范围时,我在忽略 @Scope
时遇到了一些问题。我可以解释。
我正在使用 Hibernate Envers 在审计中保存用户的一些信息 table。这在存在请求(来自我的 Rest Controller)时效果很好,因为用户需要登录,所以在这种情况下总能找到用户。
但是,我也在使用 RabbitMQ。当我收到一些需要对数据库执行某些操作的消息时(例如,插入一个实体),我的代码尝试从应用程序上下文加载 User
bean。 User
Bean 与 @Scope
请求绑定,而 Spring 找不到线程绑定请求。
因此,我收到错误消息:
Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'scopedTarget.get': Scope 'request' is not active for the current thread; consider defining a scoped proxy for this bean if you intend to refer to it from a singleton; nested exception is java.lang.IllegalStateException: No thread-bound request found: Are you referring to request attributes outside of an actual web request, or processing a request outside of the originally receiving thread? If you are actually operating within a web request and still receive this message, your code is probably running outside of DispatcherServlet/DispatcherPortlet: In this case, use RequestContextListener or RequestContextFilter to expose the current request.
at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:355)
at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:197)
at org.springframework.aop.target.SimpleBeanTargetSource.getTarget(SimpleBeanTargetSource.java:35)
at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.getTarget(CglibAopProxy.java:705)
at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:655)
at com.domain.config.UserConfiguration$User$$EnhancerBySpringCGLIB$$c7e6e64e.getUserId(<generated>)
at com.domain.audit.RevisionEntityListener.newRevision(RevisionEntityListener.java:28)
at org.hibernate.envers.internal.revisioninfo.DefaultRevisionInfoGenerator.generate(DefaultRevisionInfoGenerator.java:93)
at org.hibernate.envers.internal.synchronization.AuditProcess.getCurrentRevisionData(AuditProcess.java:114)
at org.hibernate.envers.internal.synchronization.AuditProcess.executeInSession(AuditProcess.java:96)
at org.hibernate.envers.internal.synchronization.AuditProcess.doBeforeTransactionCompletion(AuditProcess.java:153)
at org.hibernate.envers.internal.synchronization.AuditProcessManager.doBeforeTransactionCompletion(AuditProcessManager.java:46)
at org.hibernate.engine.spi.ActionQueue$BeforeTransactionCompletionProcessQueue.beforeTransactionCompletion(ActionQueue.java:928)
... 38 common frames omitted
Caused by: java.lang.IllegalStateException: No thread-bound request found: Are you referring to request attributes outside of an actual web request, or processing a request outside of the originally receiving thread? If you are actually operating within a web request and still receive this message, your code is probably running outside of DispatcherServlet/DispatcherPortlet: In this case, use RequestContextListener or RequestContextFilter to expose the current request.
at org.springframework.web.context.request.RequestContextHolder.currentRequestAttributes(RequestContextHolder.java:131)
at org.springframework.web.context.request.AbstractRequestAttributesScope.get(AbstractRequestAttributesScope.java:41)
at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:340)
... 50 common frames omitted
我想忽略这种情况,因为在这种情况下我的意图是用 "system" 之类的东西填充 "user" 信息。表明此更改不是由用户直接进行的(因为没有涉及用户)。
我的代码如下。
RevisionEntityListener
(Hibernate Envers):
@Component
public class RevisionEntityListener implements RevisionListener, ApplicationContextAware {
private static ApplicationContext applicationContext;
@Override
public void newRevision(Object revisionEntity) {
RevEntity revEntity = (RevEntity) revisionEntity;
User user = applicationContext.getBean(User .class);
revEntity.setUserId(user.getUserId());
// code
}
private static void setStaticApplicationContext(ApplicationContext context) {
applicationContext = context;
}
@Override
public void setApplicationContext(ApplicationContext applicationContext)
throws BeansException {
setStaticApplicationContext(applicationContext);
}
}
我的UserConfiguration
:
@Configuration
public class UserConfiguration {
@Autowired
private OAuth2ClientContext context;
@Bean
@Scope(proxyMode = ScopedProxyMode.TARGET_CLASS, scopeName = "request")
public User get() {
User user = null;
try {
String token = context.getAccessToken().getValue();
// code
return user;
}
@Getter
@Setter
@ToString
public static class User {
private Long userId;
// all user information
}
}
我不知道这是否是最佳选择,但我创建了一个 if
来查看 SecurityContextHolder.getContext().getAuthentication()
是否存在:
User user;
if (SecurityContextHolder.getContext().getAuthentication() != null) {
user = applicationContext.getBean(User .class);
} else {
user = createSystemUser();
}
revEntity.setUserId(user.getUserId());