6

すべてのサーバーログにTomcat7とLo4jを使用し、クライアントにGWTを使用しています(AJAX呼び出しのみ)。未処理の例外はすべて、catalina.logに記録されます。

ここで、すべての例外をキャッチし、ユーザー固有のTomcatSessionDataの一部を追加します。

いくつかの方法があります。

  • すべてのサーブレットをキャッチしてみてください(より良い解決策があるはずです)。
  • http://tomcat.apache.org/tomcat-7.0-doc/aio.html:コネクタを変更する必要があり、イベントハンドラ(EventType.ERROR)内でTomcatセッションを使用できるかどうかわかりません。 。
  • もっといい方法?

これを達成するための最良の方法は何でしょうか?

4

3 に答える 3

7

あなたの質問から私が理解したことから、あなたは2つの方法のうちの少なくとも1つを使用することを試みることができます:

基本的なロギングサーブレット

すべてのサーブレットのソースコードにアクセスできる場合は、AJAXで透過的に動作するすべてのリクエストのロギング/すべてを担当する基本的なスーパーサーブレットを使用して、少しリファクタリングを行うことができます。エラー転送ディレクティブはなく、グローバルはありません。例外ハンドラ。サーブレットのエントリポイントとして使用するとservice(ServletRequest,ServletResponse)します(ただし、すべてのdo*()メソッドに対して次のことを実行できます)。次に、抽象スーパーサーブレットを作成し、そこからサーブレットを継承するだけです。

<servlet>
    <servlet-name>servlet1</servlet-name>
    <servlet-class>stackoverflow.Servlet1</servlet-class>
</servlet>

<servlet>
    <servlet-name>servlet2</servlet-name>
    <servlet-class>stackoverflow.Servlet2</servlet-class>
</servlet>

<servlet-mapping>
    <servlet-name>servlet1</servlet-name>
    <url-pattern>servlet1</url-pattern>
</servlet-mapping>

<servlet-mapping>
    <servlet-name>servlet2</servlet-name>
    <url-pattern>servlet2</url-pattern>
</servlet-mapping>
public abstract class BasicServlet extends HttpServlet {

    /**
     * Won't let it be {@code abstract} - we don't want to force any sub-servlet to implement this method.
     */
    protected void doService(ServletRequest request, ServletResponse response) {
    }

    @Override
    public final void service(ServletRequest request, ServletResponse response) {
        try {
            doService(request, response);
        } catch ( Throwable ex ) {
            err.println(ex.getMessage());
        }
    }

}
public final class Servlet1 extends BasicServlet {

    @Override
    protected void doService(ServletRequest request, ServletResponse response) {
        out.println("I'm servlet #1");
    }

}
public final class Servlet2 extends BasicServlet {

    @Override
    protected void doService(ServletRequest request, ServletResponse response) {
        out.println("I'm servlet #2");
    }

}

この方法の利点は、サーブレットクラスを変更する以外に何も構成する必要がなく、外部構成やコンテキストに依存しないことです。欠点は、常にを拡張する必要があることですBasicServlet

フィルター

現在、実際にはテストしていません。詳細については、 http://docs.oracle.com/javaee/6/api/javax/servlet/Filter.htmlを参照してください。フィルタを使用すると、各リクエストをインターセプトできます(JSPには、このようなフィルタ実装を使用して、例外をデバッグし、共通ログファイルに書き込みます)。欠点は、たとえば、フィルターが独自のフィルターの前にある場合など、フィルターがすべての例外/ケースをカバーできることが保証されていないことです。

<filter>
    <filter-name>exceptionLoggingFilter</filter-name>
    <filter-class>stackoverflow.ExceptionLoggingFilter</filter-class>
</filter>

<filter-mapping>
    <filter-name>exceptionLoggingFilter</filter-name>
    <url-pattern>*</url-pattern> <!-- we will process every request -->
</filter-mapping>
public final class ExceptionLoggingFilter implements Filter {

    @Override
    public void init(FilterConfig filterConfig) {
    }

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) {
        try {
            filterChain.doFilter(request, response);
        } catch ( Throwable ex ) {
            err.println(ex);
        }
    }

    @Override
    public void destroy() {
    }

}

お役に立てれば。

于 2013-01-12T11:44:59.937 に答える
3

GWT関数doUnexpectedFailureをオーバーライドするだけで機能しました。

@Override
protected void doUnexpectedFailure(Throwable t) {
  ServerLog.error(t.getMessage(), t);
  super.doUnexpectedFailure(t);
}
于 2013-01-13T18:42:10.613 に答える
2

1)次のように、Webアプリのエラーページを定義できます。

<error-page>
        <exception-type>java.lang.Throwable</exception-type>
        <location>/error</location>
</error-page>

次に、で別のサーブレットをバインドし/error、そこで例外を処理できます。

2)setUncaughtExceptionHandlerすべてのHTTPコネクタスレッドに対して可能です。この手法は、現在の参照を含むサーブレットフィルターで使用できますHttpRequest(たとえば、ローカルスレッドを介して)。ちなみに、これは非同期I/Oでは機能しません。

于 2012-12-27T13:59:19.640 に答える