Wildfly / Infinispan HTTP 会话复制在解组 CGLIB 会话 Bean 时命中 ClassNotFoundException

Wildfly / Infinispan HTTP session replication hits ClassNotFoundException when unmarshalling CGLIB Session Bean

我是 运行 Wildfly 20.0.1.Final standalone,双节点集群。我正在尝试在节点之间实现 HTTP 会话共享。

在我的 Spring 网络应用程序中,我的 web.xml 中有 <distributable/>

我的会话对象是这样的:

package my.package;

@Component
@Scope(value = WebApplicationContext.SCOPE_SESSION, proxyMode = ScopedProxyMode.INTERFACES)
public class MySessionBean implements Serializable {
  // omitted for brevity
}

如你所见,我有 ScopedProxyMode.TARGET_CLASS

当我在 Wildfly 中执行故障转移时,我的 HTTP 会话无法恢复,因为我遇到了这个警告:

2021-02-22 13:24:18,651 WARN  [org.wildfly.clustering.web.infinispan] (default task-1) WFLYCLWEBINF0007: 
Failed to activate attributes of session Pd9oI0OBiZSC9we0uXsZdBwkLnadO1l4TUfvoJZf: 
org.wildfly.clustering.marshalling.spi.InvalidSerializedFormException: 
java.lang.ClassNotFoundException: my.package.MySessionBean$$EnhancerBySpringCGLIB$c0fa1df 
from [Module "deployment.myDeployment.war" from Service Module Loader]
...
Caused by: java.lang.ClassNotFoundException: my.package.MySessionBean$$EnhancerBySpringCGLIB$c0fa1df from [Module "deployment.myDeployment.war" from Service Module Loader]
    at org.jboss.modules.ModuleClassLoader.findClass(ModuleClassLoader.java:255)
    at org.jboss.modules.ConcurrentClassLoader.performLoadClassUnchecked(ConcurrentClassLoader.java:410)
    at org.jboss.modules.ConcurrentClassLoader.performLoadClass(ConcurrentClassLoader.java:398)
    at org.jboss.modules.ConcurrentClassLoader.loadClass(ConcurrentClassLoader.java:116)
    at java.base/java.lang.Class.forName0(Native Method)
    at java.base/java.lang.Class.forName(Class.java:398)
    at org.jboss.marshalling@2.0.9.Final//org.jboss.marshalling.ModularClassResolver.resolveClass(ModularClassResolver.java:133)
    at org.jboss.marshalling.river@2.0.9.Final//org.jboss.marshalling.river.RiverUnmarshaller.doReadClassDescriptor(RiverUnmarshaller.java:1033)
    at org.jboss.marshalling.river@2.0.9.Final//org.jboss.marshalling.river.RiverUnmarshaller.doReadNewObject(RiverUnmarshaller.java:1366)
    at org.jboss.marshalling.river@2.0.9.Final//org.jboss.marshalling.river.RiverUnmarshaller.doReadObject(RiverUnmarshaller.java:283)
    at org.jboss.marshalling.river@2.0.9.Final//org.jboss.marshalling.river.RiverUnmarshaller.doReadObject(RiverUnmarshaller.java:216)
    at org.jboss.marshalling@2.0.9.Final//org.jboss.marshalling.AbstractObjectInput.readObject(AbstractObjectInput.java:41)
    at org.wildfly.clustering.marshalling.spi@20.0.1.Final//org.wildfly.clustering.marshalling.spi.util.MapExternalizer.readObject(MapExternalizer.java:65)
...

请注意,ClassNotFoundException 正在抱怨,因为缺少 my.package.MySessionBean$$EnhancerBySpringCGLIB$c0fa1df,这是我的 MySessionBean bean 的 Spring-增强 bean。

更改为 ScopedProxyMode.INTERFACES 不是一个选项。

你能用这个给我指明正确的方向吗?

我设法通过创建一个名为 MySessionDTO 的简单 POJO 并在我的会话中使用它来解决此问题。

所以最初我有这个(在问题中抛出异常):

request.getSession().setAttribute("mySession", mySessionBean);

...在我创建 MySessionDTO 之后(见下文),我将其重构为:

request.getSession().setAttribute("mySession", mySessionBean.getMySessionDTO());

MySessionDTO 是一个简单的 POJO:

package my.package;

import java.io.Serializable;

public class MySessionDTO extends MySessionBean implements Serializable {
  public MySessionDTO (MySessionBean mySessionBean) {
    this.setAttributeX(mySessionBean.getAttributeX());
    this.setAttributeY(mySessionBean.getAttributeY());
  }
}