【问题标题】:Put Spring EntityManager in thread for async Servlet 3.0?将 Spring EntityManager 放入异步 Servlet 3.0 的线程中?
【发布时间】:2012-10-16 17:06:35
【问题描述】:

我们正在尝试在我们的 Spring 应用程序中启用 Servlet 的 3.0“异步支持”:

  • 为所有 servlet 和过滤器添加了 web.xml (async-supported)true(/async-supported)

  • 将请求处理代码重写为

**

request.getAsyncContext().start(new Runnable() {
  @Override
  public void run() {
    handleRequest(servlet, request, response);
  }
});

**

这会导致两个问题:

  1. 运行异步代码时Spring Security身份验证丢失。

  2. 也没有 EntityManager/session/...。

当然,这两个问题都是由于处理程序代码没有在与创建 Runnable 的线程相同的线程中执行。理想情况下,工作线程应该从调用线程“继承” Spring 上下文。

我可以通过在构建时保存身份验证并在执行时设置它来解决 Spring Security 问题。

private final Authentication authentication = SecurityContextHolder.getContext().getAuthentication();

try {
  SecurityContextHolder.clearContext();
  SecurityContextHolder.getContext().setAuthentication(authentication);

  ..run code here..

} finally {
  SecurityContextHolder.clearContext();
} 

但是,对于第二个问题,我不知道如何将会话“转移”到执行线程。对于 servlet 2.x,我们使用 OpenEntityManagerInViewFilter 将会话粘贴到线程。

<filter>
  <filter-name>openEntityManagerInViewFilter</filter-name>
  <filter-class>org.springframework.orm.jpa.support.OpenEntityManagerInViewFilter</filter-class>
  <async-supported>true</async-supported>
  <init-param>
    <param-name>entityManagerFactoryBeanName</param-name>
    <param-value>entityManagerFactory</param-value>
  </init-param>
</filter>

但是现在这当然不再有意义了,因为它所附着的线程与执行请求的线程不同。

我尝试过各种版本的 Spring 3.1 和 3.2。但是没有效果。

有谁知道如何解决这个问题?最好有黑客攻击。 :-)

我的web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://java.sun.com/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
         version="3.0">

  <servlet>
    <servlet-name>Foo</servlet-name>
    <servlet-class>com.foo.Main</servlet-class>
    <async-supported>true</async-supported>
  </servlet>

  <servlet-mapping>
    <servlet-name>Foo</servlet-name>
    <url-pattern>/restricted/*</url-pattern>
  </servlet-mapping>

  <filter>
    <filter-name>openEntityManagerInViewFilter</filter-name>
    <filter-class>org.springframework.orm.jpa.support.OpenEntityManagerInViewFilter</filter-class>
    <async-supported>true</async-supported>
    <init-param>
      <param-name>entityManagerFactoryBeanName</param-name>
      <param-value>entityManagerFactory</param-value>
    </init-param>
  </filter>

  <filter>
    <filter-name>springSecurityFilterChain</filter-name>
    <filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
    <async-supported>true</async-supported>
  </filter>

  <filter-mapping>
    <filter-name>springSecurityFilterChain</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>

  <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>

  <listener>
    <listener-class>org.bar.ServletInit</listener-class>
  </listener>

  <context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>/WEB-INF/applicationContext.xml</param-value>
  </context-param>

  <context-param>
    <param-name>contextInitializerClasses</param-name>
    <param-value>com.foo.SpringCtxInitializer</param-value>
  </context-param>

  <filter-mapping>
    <filter-name>openEntityManagerInViewFilter</filter-name>
    <url-pattern>/restricted/*</url-pattern>
  </filter-mapping>

</web-app>

【问题讨论】:

    标签: spring hibernate jpa spring-security jpa-2.0


    【解决方案1】:

    您需要为该异步线程打开一个新的会话/实体管理器(会话不能在线程之间共享)。为此,您可能需要使用OpenSessionInViewInterceptor(或者我猜对于JPA,它称为OpenEntityManagerInViewInterceptor),但您需要手动调用其方法或为此目的编写自己的方面并使用代理。

    【讨论】:

    • 我通过复制/粘贴过滤器的代码取得了某种成功。但是我不知道下游的实体管理器会发生什么——如果另一个异步处理也发生了。当下游发生故障时如何保证事务的完整性?
    • 事务完整性在这里不是问题,因为一切都将在 Spring 启动的事务中完成,无论是在服务层还是在拦截器中,即使这发生在不同的线程中。在共享EntityManager 时,除非您想遍历所有线程并检查它们的ThreadLocals,否则您无法强制不在线程之间共享它,它是纯Java,您只需要小心。
    • 我不知道我是否真的想这样做。 :-) 我希望 Spring 3.2 能够为此提供支持。
    猜你喜欢
    • 2023-03-27
    • 2011-06-15
    • 2017-02-09
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-09-10
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多