13

在 JSF 2.0.9、Weblogic 10.3.4 上运行。我们现在在生产环境中运行 JSF,但遇到了会话复制和故障转移的一些问题。我们正在为我们的 bean 使用 viewscope,并且我确保它们是可序列化/瞬态的,并且瞬态变量实际上是无状态的。但是会话故障转移不起作用。我已经进行了广泛的测试并通过在 web.xml 中设置以下参数来设法使其工作

    <context-param>
        <param-name>javax.faces.STATE_SAVING_METHOD</param-name>
        <param-value>client</param-value>
    </context-param>


    <context-param>
        <param-name>javax.faces.PARTIAL_STATE_SAVING</param-name>
        <param-value>false</param-value>
    </context-param>

如果我设置STATE_SAVING_METHODserver我在故障转移时得到一个视图过期异常。如果我设置为clientwith PARTIAL_STATE_SAVINGtotrue我会收到以下错误:

java.lang.IndexOutOfBoundsException: Index: 0, Size: 0
    at java.util.ArrayList.RangeCheck(ArrayList.java:547)
    at java.util.ArrayList.get(ArrayList.java:322)
    at javax.faces.component.AttachedObjectListHolder.restoreState(AttachedObjectListHolder.java:165)
    at javax.faces.component.UIComponentBase.restoreState(UIComponentBase.java:1433)
    at com.sun.faces.application.view.StateManagementStrategyImpl$1.visit(StateManagementStrategyImpl.java:265)
    at com.sun.faces.component.visit.FullVisitContext.invokeVisitCallback(FullVisitContext.java:151)
    at javax.faces.component.UIComponent.visitTree(UIComponent.java:1507)
    at javax.faces.component.UIComponent.visitTree(UIComponent.java:1521)
    at com.sun.faces.component.visit.VisitUtils.doFullNonIteratingVisit(VisitUtils.java:75)
    at com.sun.faces.application.view.StateManagementStrategyImpl.restoreView(StateManagementStrategyImpl.java:282)
    at com.sun.faces.application.StateManagerImpl.restoreView(StateManagerImpl.java:181)
    at com.sun.faces.application.view.ViewHandlingStrategy.restoreView(ViewHandlingStrategy.java:123)
    at com.sun.faces.application.view.FaceletViewHandlingStrategy.restoreView(FaceletViewHandlingStrategy.java:448)
    at com.sun.faces.application.view.MultiViewHandler.restoreView(MultiViewHandler.java:148)
    at com.sun.faces.lifecycle.RestoreViewPhase.execute(RestoreViewPhase.java:187)
    at com.sun.faces.lifecycle.Phase.doPhase(Phase.java:101)
    at com.sun.faces.lifecycle.RestoreViewPhase.doPhase(RestoreViewPhase.java:111)
    at com.sun.faces.lifecycle.LifecycleImpl.execute(LifecycleImpl.java:118)
    at javax.faces.webapp.FacesServlet.service(FacesServlet.java:508)
    at weblogic.servlet.internal.StubSecurityHelper$ServletServiceAction.run(StubSecurityHelper.java:227)
    at weblogic.servlet.internal.StubSecurityHelper.invokeServlet(StubSecurityHelper.java:125)
    at weblogic.servlet.internal.ServletStubImpl.execute(ServletStubImpl.java:301)
    at weblogic.servlet.internal.TailFilter.doFilter(TailFilter.java:27)
    at weblogic.servlet.internal.FilterChainImpl.doFilter(FilterChainImpl.java:57)
    at weblogic.servlet.internal.RequestEventsFilter.doFilter(RequestEventsFilter.java:27)
    at weblogic.servlet.internal.FilterChainImpl.doFilter(FilterChainImpl.java:57)
    at weblogic.servlet.internal.WebAppServletContext$ServletInvocationAction.wrapRun(WebAppServletContext.java:3730)
    at weblogic.servlet.internal.WebAppServletContext$ServletInvocationAction.run(WebAppServletContext.java:3696)
    at weblogic.security.acl.internal.AuthenticatedSubject.doAs(AuthenticatedSubject.java:321)
    at weblogic.security.service.SecurityManager.runAs(SecurityManager.java:120)
    at weblogic.servlet.internal.WebAppServletContext.securedExecute(WebAppServletContext.java:2273)
    at weblogic.servlet.internal.WebAppServletContext.execute(WebAppServletContext.java:2179)
    at weblogic.servlet.internal.ServletRequestImpl.run(ServletRequestImpl.java:1490)
    at weblogic.work.ExecuteThread.execute(ExecuteThread.java:256)
    at weblogic.work.ExecuteThread.run(ExecuteThread.java:221)

所以我的问题是:

  • STATE_SAVING_METHOD- client& PARTIAL_STATE_SAVING-false我要让故障转移工作的唯一方法吗?
  • 客户端/错误组合的成本是多少。这个内存/cpu广泛吗?
  • 这是一个错误吗?如果是,它会在 2.1 或 2.2 中解决吗?

提前致谢。

4

2 回答 2

14

我终于完成了这项工作,但并非没有一些额外的东西。首先,我在 web.xml 中添加了以下内容(是的,agressive 拼写错误):

    <context-param>
        <param-name>com.sun.faces.enableAgressiveSessionDirtying</param-name>
        <param-value>true</param-value>
    </context-param>

客户端保存现在是服务器,部分状态保存仍然是错误的(真的根本不起作用)

其次,在实现了一个 HttpSessionAttributeListener 之后,我发现com.sun.faces.renderkit.ServerSideStateHelper.LogicalViewMap在会话中保持状态的那个只被添加了一次,并且再也没有被删除/添加/替换。因此,尽管它在本地会话中被更新,但这些更改从未复制到第二个 jvm。Weblogic 文档声明必须在会话属性上调用 setAttribute 才能使复制工作。为了解决这个问题,我创建了一个阶段监听器,如下所示:

public class ViewPhaseListener implements PhaseListener {

    public void afterPhase(PhaseEvent phaseEvent)
    {

    }

    public void beforePhase(PhaseEvent phaseEvent)
    {
        HttpServletRequest request = ((HttpServletRequest) phaseEvent.getFacesContext().getExternalContext().getRequest());
        HttpSession session = request.getSession();

        session.setAttribute("com.sun.faces.renderkit.ServerSideStateHelper.LogicalViewMap", session.getAttribute("com.sun.faces.renderkit.ServerSideStateHelper.LogicalViewMap"));

    }

    public PhaseId getPhaseId()
    {
        return PhaseId.RENDER_RESPONSE;
          //To change body of implemented methods use File | Settings | File Templates.
    }
}

这只是在每次请求后替换属性并确保它复制。另外一点,我将使用以下内容限制存储在视图中的数据:

<context-param>
        <param-name>com.sun.faces.numberOfViewsInSession</param-name>
        <param-value>3</param-value>
    </context-param>

    <context-param>
        <param-name>com.sun.faces.numberOfLogicalViews</param-name>
        <param-value>1</param-value>
    </context-param>

希望这可以帮助任何有同样问题的人。

于 2013-04-12T09:40:44.783 回答
0

会话复制应该由负载平衡器处理,因为 JSF 应用程序只知道它部署到的节点上的上下文。因此,将 STATE_SAVING_METHOD 设置为clientserver无关紧要。

在我看来,您的负载平衡器配置不正确。如果您使用 Apache HTTP Server 作为代理,请参阅以下链接以获取有关会话粘性的更多信息:

http://httpd.apache.org/docs/2.2/mod/mod_proxy_balancer.html

我建议设置ProxySet stickysession=ROUTEID看看是否能解决问题。

于 2012-09-06T17:25:39.710 回答