我们已经在 Struts 2.1.8 上运行了一段时间,所有 Struts 操作都按预期工作,即 Struts 操作的 href 使用不带扩展名的操作名称呈现。
以下是设置链接的 JSP 代码:
<ul id="top_menu">
<li id="itemHome" class="active"><s:a action="viewHome">Home</s:a></li>
<li><s:a action="viewSearch">Search</s:a></li>
<li><s:a action="viewBookMarks">My Bookmarks</s:a></li>
<li><s:a action="viewSupport">Support</s:a></li>
</ul>
链接正确呈现到http://localhost/viewHome
, http://localhost/viewSearch
等2.1.8下
我们刚刚升级到 Struts 2.2.1(已检查从该版本到 v2.3.4.1 的所有版本),现在看到 Struts 操作链接呈现为http://localhost/viewHome.action
, http://localhost/viewSearch.action
, etc.
我的研究表明,一般建议的解决方案是使用
<constant name="struts.action.extension" value=""/>
在struts.xml中删除.action后缀。虽然这可以使 URL 正确呈现,但会导致意外的副作用。 Struts 现在认为每个 URL 都是一个操作,包括对 .css、.png 等的请求。
我在 web.xml 中的过滤器映射没有改变。尽管它发送 /* 到 Struts,但我们在 2.1.8 下没有看到这种行为
<filter>
<filter-name>struts2</filter-name>
<filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>struts2</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
为了解决这个问题,我不得不使用排除模式阻止 Struts 尝试将这些请求视为操作。
<constant name="struts.action.excludePattern" value="/index.html,/images/.*,/js/.*,/css/.*"/>
虽然这可行,但最后一个障碍是我的日志充满了来自作为 Struts 操作处理的 Struts 标记请求的错误。当我将这些 URI 添加到排除模式时,Struts Dojo 标记似乎在某些页面中不起作用。
struts.xml 受影响的部分是:
<constant name="struts.devMode" value="true" />
<!-- Set URL's to have no .action extension -->
<constant name="struts.action.extension" value=""/>
<constant name="struts.action.excludePattern" value="/index.html,/images/.*,/js/.*,/css/.*"/>
日志中的 Struts 错误示例如下:
2012-09-26 17:12:57,984 [http-bio-8080-exec-9] ERROR org.apache.struts2.dispatcher.Dispatcher- Could not find action or result
/struts/dojo/struts_dojo.js
There is no Action mapped for namespace [/] and action name [struts_dojo.js] associated with context path []. - [unknown location]
at com.opensymphony.xwork2.DefaultActionProxy.prepare(DefaultActionProxy.java:185)
at org.apache.struts2.impl.StrutsActionProxy.prepare(StrutsActionProxy.java:63)
at org.apache.struts2.impl.StrutsActionProxyFactory.createActionProxy(StrutsActionProxyFactory.java:39)
at com.opensymphony.xwork2.DefaultActionProxyFactory.createActionProxy(DefaultActionProxyFactory.java:58)
at org.apache.struts2.dispatcher.Dispatcher.serviceAction(Dispatcher.java:501)
at org.apache.struts2.dispatcher.ng.ExecuteOperations.executeAction(ExecuteOperations.java:77)
at org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter.doFilter(StrutsPrepareAndExecuteFilter.java:91)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:243)
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:210)
at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:225)
at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:123)
at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:472)
at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:168)
at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:98)
at org.apache.catalina.valves.AccessLogValve.invoke(AccessLogValve.java:927)
at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:118)
at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:407)
at org.apache.coyote.http11.AbstractHttp11Processor.process(AbstractHttp11Processor.java:1001)
at org.apache.coyote.AbstractProtocol$AbstractConnectionHandler.process(AbstractProtocol.java:579)
at org.apache.tomcat.util.net.JIoEndpoint$SocketProcessor.run(JIoEndpoint.java:310)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
at java.lang.Thread.run(Thread.java:680)
我不确定 struts Dojo 请求是真实的 URI 还是虚拟的或其他什么。
在这一点上,我感觉我已经克服了很多困难来解决本质上是从 URL 中删除 .action 扩展名的简单问题,因为回到 2.1.8 JAR 可以解决所有问题,但我决心找到如果可能的话,前进的道路。
很感谢任何形式的帮助。