1

我正在尝试将项目从 struts 2.2.3 升级到 struts 2.3.15.1。在升级版本后,应用程序正在运行,但每次我提交一个页面时,它都会在日志中抛出一个非常长的警告。似乎 ParametersInterceptor 试图在我的类中设置一个名为“action:PersonalInfo_next”的属性,它是提交按钮表单元素的名称。这个名称是由 struts 创建的,但是当表单返回时,它会将其视为标准表单字段并尝试为我保存它。

我尝试将 action:.* 添加到 struts.xml 中的 excludeParams 选项,但似乎没有效果:

            <interceptor-ref name="params">
                <param name="excludeParams">dojo\..*,action:.*</param>
            </interceptor-ref>

警告如下:

10/09/13 15:16:18,939 DEBUG [ com.opensymphony.xwork2.interceptor.ParametersInterceptor ]: Parameter [action:PersonalInfo_next] didn't match acceptedPattern pattern!
10/09/13 15:16:18,939 WARN [ com.opensymphony.xwork2.ognl.OgnlValueStack ]: Error setting expression 'action:PersonalInfo_next' with value '[Ljava.lang.String;@47773'
ognl.ExpressionSyntaxException: Malformed OGNL expression: action:PersonalInfo_next [ognl.ParseException: Encountered " ":" ": "" at line 1, column 7.
Was expecting one of:
    <EOF> 
    "," ...
    "=" ...
    "?" ...
    "||" ...
    "or" ...
    "&&" ...
    "and" ...
[snip]
at ognl.Ognl.parseExpression(Ognl.java:112)
at com.opensymphony.xwork2.ognl.OgnlUtil.compile(OgnlUtil.java:268)
at com.opensymphony.xwork2.ognl.OgnlUtil.setValue(OgnlUtil.java:230)
at com.opensymphony.xwork2.ognl.OgnlValueStack.trySetValue(OgnlValueStack.java:183)
at com.opensymphony.xwork2.ognl.OgnlValueStack.setValue(OgnlValueStack.java:170)
at com.opensymphony.xwork2.ognl.OgnlValueStack.setParameter(OgnlValueStack.java:148)
at com.opensymphony.xwork2.interceptor.ParametersInterceptor.setParameters(ParametersInterceptor.java:318)
at com.opensymphony.xwork2.interceptor.ParametersInterceptor.doIntercept(ParametersInterceptor.java:231)
at com.opensymphony.xwork2.interceptor.MethodFilterInterceptor.intercept(MethodFilterInterceptor.java:98)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:246)
at com.opensymphony.xwork2.interceptor.StaticParametersInterceptor.intercept(StaticParametersInterceptor.java:191)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:246)
at org.apache.struts2.interceptor.CheckboxInterceptor.intercept(CheckboxInterceptor.java:91)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:246)
[snip]
Caused by: ognl.ParseException: Encountered " ":" ": "" at line 1, column 7.
Was expecting one of:
<EOF> 
"," ...
"=" ...
"?" ...
[...]

每次提交都会有超过 300 行这样的内容。

乍一看,我认为这可能与这个问题有关,但是当 devMode 被明确禁用时会发生这种情况。由于它是一个警告并且应用程序正在运行,我还玩弄了通过 log4j 配置忽略它的想法,但看起来大部分都直接打印到 stdout 或 stderr 所以这也不是一个选项。

作为参考,这里是我的 struts.xml 中的相关操作元素:

    <action name="PersonalInfo_*" class="project.web.actions.EditPersonalInfo" method="{1}">
        <interceptor-ref name="defaultStack">
            <param name="store.operationMode">AUTOMATIC</param>
        </interceptor-ref>
        <result name="input">/jsp/personalInfo.jsp</result>
        <result name="next" type="redirectAction">
            <param name="parse">true</param>
            <param name="actionName">Address_open</param>
            <param name="namespace">/</param>
            <param name="id">${id}</param>
        </result>
    </action>

还有来自 JSP 的表单和提交元素:(JSP 相当大)

<s:form action="PersonalInfo_personalInfo" namespace="/" id="appForm" cssClass="form personal_info" autocomplete="off">
[lots of fields]
<s:submit value="Next" cssClass="ksu-button" action="PersonalInfo_next"/>
4

1 回答 1

2

事实证明,我们的堆栈中正在进行一些定制,这些定制正在向齿轮扔扳手。这是我们组织中的某个人多年前编写的一些较旧的通用代码。

我的所有操作都扩展的基本操作类是扩展 com.opensymphony.xwork2.ActionSupport 并实现了 com.opensymphony.xwork2.interceptor.ParameterNameAware 接口的类,这意味着它有一个可接受的ParameterName 方法,如下所示:

public boolean acceptableParameterName(String parameterName) {
    return prepareInvoked || getPreprepareParams().contains(parameterName);
}

该 getPrepareaParams 方法返回一个列表,该列表具有唯一的条目“id”,并且当第一次调用 prepare() 时,prepareInvoked 设置为 true。

我们的堆栈调用 params 拦截器两次。第一次通过它只处理“id”参数。第二次通过,acceptableParameterName 方法总是返回 true。我不确定它的预期功能是什么。我最好的猜测是,这是一种优化某些东西的尝试,可能是针对使用此通用代码的其他应用程序。

但是升级后此方法的意外副作用是我的 struts.xml 中的默认 excludeParams 和显式 excludeParams 设置都被忽略了。

我已经从基类中删除了这个方法,一切似乎都在起作用。params 拦截器现在有一个 DEBUG 行:

DEBUG [ com.opensymphony.xwork2.interceptor.ParametersInterceptor ]: Parameter [action:PersonalInfo_next] didn't match acceptedPattern pattern!

与我之前收到的 300 多行警告相比,这是完全可以接受的!

于 2013-11-14T23:06:12.127 回答