1

我正在使用 jetty-9.0.5.v20130815、Struts 2.3.15.1,当我尝试使用 Chrome 下载文件时,就在 Chrome 中会发生此错误:

ERROR 17/09/2013 10:26:11
(br.com.arquimagem.support.web.struts2.interceptor.ExceptionHandlerInterceptor):Unexpected error on action: class br.com.arquimagem.support.web.action.CommonFilesAction
    org.eclipse.jetty.io.EofException
    at org.eclipse.jetty.io.ChannelEndPoint.flush(ChannelEndPoint.java:186)
    at org.eclipse.jetty.io.WriteFlusher.completeWrite(WriteFlusher.java:400)
    at org.eclipse.jetty.io.SelectChannelEndPoint.onSelected(SelectChannelEndPoint.java:111)
    at org.eclipse.jetty.io.SelectorManager$ManagedSelector.processKey(SelectorManager.java:498)
    at org.eclipse.jetty.io.SelectorManager$ManagedSelector.select(SelectorManager.java:455)
    at org.eclipse.jetty.io.SelectorManager$ManagedSelector.run(SelectorManager.java:420)
    at org.eclipse.jetty.util.thread.QueuedThreadPool.runJob(QueuedThreadPool.java:601)
    at org.eclipse.jetty.util.thread.QueuedThreadPool$3.run(QueuedThreadPool.java:532)
    at java.lang.Thread.run(Thread.java:724)
    Caused by: java.io.IOException: An established connection was aborted by the software in your host machine
    at sun.nio.ch.SocketDispatcher.write0(Native Method)
    at sun.nio.ch.SocketDispatcher.write(SocketDispatcher.java:51)
    at sun.nio.ch.IOUtil.writeFromNativeBuffer(IOUtil.java:93)
    at sun.nio.ch.IOUtil.write(IOUtil.java:65)
    at sun.nio.ch.SocketChannelImpl.write(SocketChannelImpl.java:487)
    at org.eclipse.jetty.io.ChannelEndPoint.flush(ChannelEndPoint.java:165)
    ... 8 more


2013-09-17 10:26:11.441:WARN:oejs.Response:qtp776813247-5034: Committed before 500 STREAM
2013-09-17 10:26:11.456:WARN:oejs.ServletHandler:qtp776813247-5034: /GedocLog/CommonFile
    java.lang.IllegalStateException: Committed
    at org.eclipse.jetty.server.Response.resetBuffer(Response.java:999)
    at org.eclipse.jetty.server.Response.sendError(Response.java:351)
    at org.apache.struts2.dispatcher.Dispatcher.sendError(Dispatcher.java:906)
    at org.apache.struts2.dispatcher.Dispatcher.serviceAction(Dispatcher.java:586)
    at org.apache.struts2.dispatcher.ng.ExecuteOperations.executeAction(ExecuteOperations.java:77)
    at org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter.doFilter(StrutsPrepareAndExecuteFilter.java:99)
    at org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1486)
    at org.eclipse.jetty.servlet.ServletHandler.doHandle(ServletHandler.java:503)
    at org.eclipse.jetty.server.handler.ScopedHandler.handle(ScopedHandler.java:138)
    at org.eclipse.jetty.security.SecurityHandler.handle(SecurityHandler.java:564)
    at org.eclipse.jetty.server.session.SessionHandler.doHandle(SessionHandler.java:213)
    at org.eclipse.jetty.server.handler.ContextHandler.doHandle(ContextHandler.java:1096)
    at org.eclipse.jetty.servlet.ServletHandler.doScope(ServletHandler.java:432)
    at org.eclipse.jetty.server.session.SessionHandler.doScope(SessionHandler.java:175)
    at org.eclipse.jetty.server.handler.ContextHandler.doScope(ContextHandler.java:1030)
    at org.eclipse.jetty.server.handler.ScopedHandler.handle(ScopedHandler.java:136)
    at org.eclipse.jetty.server.handler.ContextHandlerCollection.handle(ContextHandlerCollection.java:201)
    at org.eclipse.jetty.server.handler.HandlerCollection.handle(HandlerCollection.java:109)
    at org.eclipse.jetty.server.handler.HandlerWrapper.handle(HandlerWrapper.java:97)
    at org.eclipse.jetty.server.Server.handle(Server.java:445)
    at org.eclipse.jetty.server.HttpChannel.handle(HttpChannel.java:268)
    at org.eclipse.jetty.server.HttpConnection.onFillable(HttpConnection.java:229)
    at org.eclipse.jetty.io.AbstractConnection$ReadCallback.run(AbstractConnection.java:358)
    at org.eclipse.jetty.util.thread.QueuedThreadPool.runJob(QueuedThreadPool.java:601)
    at org.eclipse.jetty.util.thread.QueuedThreadPool$3.run(QueuedThreadPool.java:532)
    at java.lang.Thread.run(Thread.java:724)

我的 struts.xml:

<package name="default" namespace="/" extends="struts-default">
    <interceptors>
        <interceptor name="customException"
            class="br.com.arquimagem.support.web.struts2.interceptor.ExceptionHandlerInterceptor">
            <param name="result">globalError</param>
        </interceptor>
        <interceptor name="customFileUpload"
            class="br.com.arquimagem.support.web.struts2.interceptor.CustomFileUploadInterceptor">
            <param name="result">globalError</param>
        </interceptor>

        <!-- Default Stack -->
        <interceptor-stack name="defaultStack">
            <interceptor-ref name="customException" />
            <!-- <interceptor-ref name="customFileUpload" /-->
            <!-- <interceptor-ref name="alias" /-->
            <!-- <interceptor-ref name="prepare" /-->
            <!-- <interceptor-ref name="i18n" /-->
            <!-- <interceptor-ref name="chain" /-->
            <interceptor-ref name="params" />
            <!-- <interceptor-ref name="conversionError" /-->
            <interceptor-ref name="validation" />
            <!-- <interceptor-ref name="workflow"<param name="inputResultName">generalError</param
                </interceptor-ref-->
        </interceptor-stack>

        <!-- Upload Stack -->
        <interceptor-stack name="uploadStack">
            <interceptor-ref name="customException" />
            <interceptor-ref name="customFileUpload" />
            <interceptor-ref name="params" />
            <interceptor-ref name="validation" />
        </interceptor-stack>
    </interceptors>

    <default-interceptor-ref name="defaultStack" />

    <default-action-ref name="index" />

    <global-results>
        <result name="globalError">/pages/error/general.jsp
        </result>
        <result name="generalError">/pages/error/general.jsp
        </result>
    </global-results>

    <action name="CommonFile"
        class="br.com.arquimagem.support.web.action.CommonFilesAction">
        <result name="sendFile" type="stream">
            <param name="bufferSize">1024</param>
            <param name="allowCaching">false</param>
        </result>
        <result>/pages/resultCommand.jsp</result>
    </action>
</package> </struts>

为什么这仅在 Chrome 中发生,大约一个月?

自定义拦截器:

package br.com.arquimagem.support.web.struts2.interceptor;

import java.lang.reflect.InvocationTargetException;
import java.util.Collections;

import org.apache.log4j.Logger;

import br.com.arquimagem.support.exception.InfraStructureException;
import br.com.arquimagem.support.exception.InvalidParameterException;
import br.com.arquimagem.support.log4j.InfraStructureLoggerFactory;
import br.com.arquimagem.support.web.action.AbstractBaseAction;
import br.com.arquimagem.support.web.util.MessageConstants;
import br.com.arquimagem.support.web.util.Struts2Utils;

import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.ActionSupport;
import com.opensymphony.xwork2.interceptor.ExceptionHolder;
import com.opensymphony.xwork2.interceptor.ExceptionMappingInterceptor;

public class ExceptionHandlerInterceptor extends ExceptionMappingInterceptor
{
    private static final long serialVersionUID = -2677219300348931782L;

    private String            result           = AbstractBaseAction.ERROR;

    protected static Logger   logger           = InfraStructureLoggerFactory
                                                       .getLogger(ExceptionHandlerInterceptor.class);

    @Override
    public String intercept(ActionInvocation invocation) throws Exception {

        try {
            return super.intercept(invocation);
        }
        catch (InvocationTargetException e) {

            if (e.getTargetException() instanceof OutOfMemoryError) {

                publishOutOfMemoryException(invocation, new ExceptionHolder(e));
            }
            else {
                publishException(invocation, new ExceptionHolder(e));

                throw e;
            }
        }
        catch (InvalidParameterException ipe) {

            publishInvalidParameterException(invocation, new ExceptionHolder(ipe));
        }
        catch (InfraStructureException ise) {

            publishInfraException(invocation, new ExceptionHolder(ise));
        }
        catch (Exception ex) {

            publishException(invocation, new ExceptionHolder(ex));
        }

        return result;
    }

    protected void publishAuthorizationException(ActionInvocation invocation, ExceptionHolder exceptionHolder) {

        super.publishException(invocation, exceptionHolder);

        logger.error("AuthorizationException exception on action: " + invocation.getAction().getClass().toString(),
                exceptionHolder.getException());

        setErrorMessage(MessageConstants.COMMON_0003, invocation);
    }

    protected void publishInvalidParameterException(ActionInvocation invocation, ExceptionHolder exceptionHolder) {

        super.publishException(invocation, exceptionHolder);

        logger.error("InvalidParameter exception on action: " + invocation.getAction().getClass().toString(),
                exceptionHolder.getException());

        setErrorMessage(MessageConstants.COMMON_0005, invocation);
    }

    protected void publishInfraException(ActionInvocation invocation, ExceptionHolder exceptionHolder) {

        super.publishException(invocation, exceptionHolder);

        logger.error("Infra structure exception on action: " + invocation.getAction().getClass().toString(),
                exceptionHolder.getException());

        setErrorMessage(MessageConstants.COMMON_0004, invocation);
    }

    @Override
    protected void publishException(ActionInvocation invocation, ExceptionHolder exceptionHolder) {

        super.publishException(invocation, exceptionHolder);

        logger.error("Unexpected error on action: " + invocation.getAction().getClass().toString(),
                exceptionHolder.getException());

        setErrorMessage(MessageConstants.COMMON_0006, invocation);
    }

    protected void publishOutOfMemoryException(ActionInvocation invocation, ExceptionHolder exceptionHolder) {

        super.publishException(invocation, exceptionHolder);

        logger.error("OutOfMemory Exception: " + invocation.getAction().getClass().toString(),
                exceptionHolder.getException());

        setErrorMessage(MessageConstants.COMMON_0014, invocation);
    }

    public String getResult() {

        return result;
    }

    public void setResult(String result) {

        this.result = result;
    }

    private void setErrorMessage(String key, ActionInvocation invocation) {

        try {
            ((AbstractBaseAction) invocation.getAction()).addError(key, Collections.emptyList());
        }
        catch (Exception e) {
            ((ActionSupport) invocation.getAction()).addActionError(Struts2Utils.getTextFromInvocation(invocation, key,
                    Collections.EMPTY_LIST));
        }
    }
}

而我的 CustomFileUploadInterceptor 只是扩展 FileUploadInterceptor。

4

0 回答 0