Oracle® Fusion Middleware Oracle WebLogic Serverログ・ファイルの構成とログ・メッセージのフィルタ処理 11g リリース1(10.3.5) B61637-03 |
|
前 |
WebLogic Serverメッセージ・カタログおよびNonCatalogLogger
でメッセージが生成されるとき、それらのメッセージはjava.util.logging.Logger
オブジェクトに配信されます。Logger
オブジェクトは、メッセージを説明するWLLogRecord
オブジェクトを割り当て、Logger
をサブスクライブしているすべてのメッセージ・ハンドラにそのWLLogRecord
をパブリッシュします。
以下の節では、メッセージ・ハンドラの作成とサブスクライブについて説明します。
WebLogic Serverのロガーおよびハンドラの詳細は、「ロガーとハンドラの役割」を参照してください。
WebLogic Serverは、ログ・メッセージを受信して出力するメッセージ・ハンドラをインスタンス化し、サブスクライブさせます。独自のメッセージ・ハンドラを作成して、WebLogic ServerのLogger
オブジェクトをサブスクライブするようにもできます(図5-1を参照)。
たとえば、アプリケーションがクライアントJVMで動作しており、そのアプリケーションがアプリケーションの生成するメッセージをリスニングするようにする場合は、ハンドラを作成し、そのハンドラをクライアントJVMのLogger
オブジェクトにサブスクライブすることができます。アプリケーションが特定サブシステムの障害を知らせるログ・メッセージを受信した場合、そのアプリケーションは以下のようなアクションを実行できます。
WebLogic Serverの管理者にログ・メッセージを電子メールで送信します。
自身またはそのサブシステムを終了または再起動します。
注意: ユーザー独自のメッセージ・ハンドラを作成する場合は、サーバーの初期化が完了して実行状態になる前に、WebLogic Serverプロセスで実行されるカスタム・コードを実行しないように注意します。場合によっては、初期化中のサーバー・サービスにカスタム・コードが干渉する場合があります。たとえば、IIOPサーバー・サービスの初期化の前にPortableRemoteObject を使用する発信RMI呼出しを行うカスタム・ログ・ハンドラは、サーバーの起動が失敗する原因になる場合があります。 |
作成し、Logger
オブジェクトをサブスクライブさせたハンドラでは、そのLoggerの重大度とフィルタ基準を満たすすべてのメッセージが受信されます。ハンドラでは、Loggerがパブリッシュする特定のメッセージだけに応答するように重大度とフィルタ基準を追加で指定することもできます。
ハンドラを作成してサブスクライブさせるには:
以下の最小限のインポート文が含まれるハンドラ・クラスを作成します。
import java.util.logging.Handler; import java.util.logging.LogRecord; import java.util.logging.ErrorManager; import weblogic.logging.WLLogRecord; import weblogic.logging.WLLevel; import weblogic.logging.WLErrorManager; import weblogic.logging.LoggingHelper;
ハンドラ・クラスで、java.util.logging.Handler
を拡張します。
ハンドラ・クラスで、Handler.publish(LogRecord record)
メソッドを実装します。
このメソッドは次のように機能します。
受信したLogRecord
オブジェクトをWLLogRecord
オブジェクトとしてキャストします。
ハンドラで設定されているフィルタを適用します。
WLLogRecord
オブジェクトがフィルタの基準を満たしている場合は、WLLogRecord
のメソッドを使用してメッセージからデータを取得します。
必要に応じて、メッセージのデータを1つまたは複数のリソースに書き込みます。
ハンドラ・クラスで、Handler.flush
メソッドとHandler.close
メソッドを実装します。
リソースと連係して機能するハンドラはすべて、flush
メソッド(バッファされた出力をフラッシュするため)およびclose
メソッド(開いているリソースを閉じるため)を実装する必要があります。
親Logger
オブジェクトが閉じると、すべてのハンドラでHandler.close
メソッドが呼び出されます。closeメソッドは、flush
メソッドを呼び出してからそれ独自のロジックを実行します。
Handler
オブジェクトが受信するメッセージのタイプを指定するフィルタ・クラスを作成します。「ロガーおよびハンドラのフィルタの設定」を参照してください。
次のいずれかのLoggingHelper
メソッドを呼び出すクラスを作成します。
getClientLogger
(現在のコンテキストがクライアントJVMである場合)。
getServerLogger
(現在のコンテキストがサーバーJVMであり、サーバーのLogger
オブジェクトにハンドラをアタッチする必要がある場合)。
getDomainLogger
(現在のコンテキストが管理サーバーであり、ドメインのLogger
オブジェクトにハンドラをアタッチする必要がある場合)。
LoggingHelper.getDomainLogger()
は、ドメイン・ログを管理するLogger
オブジェクトを取得します。カスタム・ハンドラにこのロガーをサブスクライブさせて、単一の場所にある全サーバーからのログ・メッセージを処理できます。
このクラスで、Logger.addHandler(Handler myHandler)
メソッドを呼び出します。
省略可能です。Logger.setFilter(Filter myFilter)
メソッドを呼び出してフィルタを設定します。
この例では、JDBCデータ・ソースに接続し、メッセージをデータベース表に挿入するSQL文を発行するハンドラを作成します。この例では、以下のクラスを実装します。
Handler
クラス。「例: ハンドラ・クラスの実装」を参照してください。
Filter
クラス。「ロガーおよびハンドラのフィルタの設定」を参照してください。
ハンドラおよびフィルタにサーバーのLogger
クラスをサブスクライブさせるクラス。「例: ロガー・クラスのサブスクライブ」を参照してください。
例5-1のHandler
クラスの例では、次のようにしてデータベースにメッセージを書き込みます。
java.util.logging.Handler
を拡張します。
javax.naming.InitialContext
オブジェクトを作成し、Context.lookup
メソッドを呼び出してmyPoolDataSource
というデータ・ソースをルックアップします。
javax.sql.DataSource.getConnection
メソッドを呼び出して、データ・ソースとの接続を確立します。
setErrorManager
メソッドを実装します。このメソッドは、このハンドラのjava.util.logging.ErrorManager
オブジェクトを作成します。
このハンドラでエラーが生じると、エラー・マネージャのerror
メソッドが呼び出されます。この例のerror
メソッドは次のように機能します。
エラー・メッセージを標準エラーに出力します。
LoggingHelper.getServerLogger().removeHandler(MyJDBCHandler.this)
を呼び出してハンドラを無効にします。
注意: 独立したクラス・ファイルでErrorManager クラスを定義するかわりに、この例ではErrorManager が無名内部クラスとして内包されています。 |
エラー・マネージャの詳細は、Sun APIのドキュメントのjava.util.logging.ErrorManager
クラス(http://download.oracle.com/javase/6/docs/api/java/util/logging/ErrorManager.html
)を参照してください。
Handler.publish(LogRecord record)
メソッドを実装します。このメソッドは、次のことを実行します。
受信した各LogRecord
オブジェクトをWLLogRecord
オブジェクトとしてキャストします。
isLoggable
メソッドを呼び出して、ハンドラに設定されているフィルタを適用します。isLoggable
メソッドは、このハンドラ・クラスの最後に定義されています。
WLLogRecord
のメソッドを使用して、メッセージからデータを取得します。
WLLogRecord
メソッドの詳細は、Oracle WebLogic Server APIリファレンスのweblogic.logging.WLLogRecord
クラスに関する説明を参照してください。
メッセージのデータをSQL prepareStatement
としてフォーマットし、データベースの更新を実行します。
この例で使用される表のスキーマは次のとおりです。
flush
メソッドを呼び出して、接続をフラッシュします。
Handler.close
メソッドを実装して、データ・ソースとの接続を閉じます。
親Logger
オブジェクトが閉じると、Handler.close
メソッドが呼び出されます。このメソッドは、Handler.flush
メソッドを呼び出してからそれ独自のロジックを実行します。
例5-1に、この節で説明した手順を示します。
例5-1 例: ハンドラ・クラスの実装
import java.util.logging.Handler; import java.util.logging.LogRecord; import java.util.logging.Filter; import java.util.logging.ErrorManager; import weblogic.logging.WLLogRecord; import weblogic.logging.WLLevel; import weblogic.logging.WLErrorManager; import javax.naming.InitialContext; import javax.naming.NamingException; import javax.sql.DataSource; import java.sql.Connection; import java.sql.SQLException; import java.sql.Statement; import java.sql.PreparedStatement; import weblogic.logging.LoggingHelper; public class MyJDBCHandler extends Handler { private Connection con = null; private PreparedStatement stmt = null; public MyJDBCHandler() throws NamingException, SQLException { InitialContext ctx = new InitialContext(); DataSource ds = (DataSource)ctx.lookup("myPoolDataSource"); con = ds.getConnection(); PreparedStatement stmt = con.prepareStatement setErrorManager(new ErrorManager() { public void error(String msg, Exception ex, int code) { System.err.println("Error reported by MyJDBCHandler " + msg + ex.getMessage()); //Removing any prior istantiation of this handler LoggingHelper.getServerLogger().removeHandler( MyJDBCHandler.this); } }); } public void publish(LogRecord record) { WLLogRecord rec = (WLLogRecord)record; if (!isLoggable(rec)) return; try { ("INSERT INTO myserverLog VALUES (?, ?, ? ,?)"); stmt.setEscapeProcessing(true); stmt.setString(1, rec.getId()); stmt.setString(2, rec.getLevel().getLocalizedName()); stmt.setString(3, rec.getLoggerName()); stmt.setString(4, rec.getMessage()); stmt.executeUpdate(); flush(); } catch(SQLException sqex) { reportError("Error publihsing to SQL", sqex, ErrorManager.WRITE_FAILURE); } } public void flush() { try { con.commit(); } catch(SQLException sqex) { reportError("Error flushing connection of MyJDBCHandler", sqex, ErrorManager.FLUSH_FAILURE); } } public boolean isLoggable(LogRecord record) { Filter filter = getFilter(); if (filter != null) { return filter.isLoggable(record); } else { return true; } } public void close() { try { con.close(); } catch(SQLException sqex) { reportError("Error closing connection of MyJDBCHandler", sqex, ErrorManager.CLOSE_FAILURE); } } }
例5-2のLogger
クラスの例では、以下のことを行います。
LoggingHelper.getServerLogger
メソッドを呼び出して、Logger
オブジェクトを取得します。
Logger.addHandler(Handler myHandler)
メソッドを呼び出します。
Logger.getHandlers
メソッドを呼び出して、Logger
オブジェクトのすべてのハンドラを取得します。
myHandler
が見つかるまで配列を検索します。
Handler.setFilter(Filter myFilter)
メソッドを呼び出します。
サーバーが起動するたびにハンドラとフィルタがサーバーのLogger
オブジェクトをサブスクライブするようにする場合は、このクラスをWebLogic Server起動クラスとしてデプロイします。起動クラスの詳細は、Oracle WebLogic Server管理コンソール・オンライン・ヘルプのカスタム・クラスを使用したサーバーの構成に関する項を参照してください。
例5-2 例: ロガー・クラスのサブスクライブ
import java.util.logging.Logger; import java.util.logging.Handler; import java.util.logging.Filter; import java.util.logging.LogRecord; import weblogic.logging.LoggingHelper; import weblogic.logging.FileStreamHandler; import weblogic.logging.WLLogRecord; import weblogic.logging.WLLevel; import java.rmi.RemoteException; import weblogic.jndi.Environment; import javax.naming.Context; public class LogConfigImpl { public void configureLogger() throws RemoteException { Logger logger = LoggingHelper.getServerLogger(); try { Handler h = null; h = new MyJDBCHandler(); logger.addHandler(h); h.setFilter(new MyFilter()); } catch(Exception nmex) { System.err.println("Error adding MyJDBCHandler to logger " + nmex.getMessage()); logger.removeHandler(h); } } public static void main(String[] argv) throws Exception { LogConfigImpl impl = new LogConfigImpl(); impl.configureLogger(); } }
例5-3のAppender
クラスの例では、JDBCデータ・ソースに接続し、メッセージをデータベース表に挿入するSQL文を発行します。
AppenderSkelton
を拡張します。
javax.naming.InitialContext
オブジェクトを作成し、Context.lookup
メソッドを呼び出してMyDataSource
というデータ・ソースをルックアップします。
javax.sql.DataSource.getConnection
メソッドを呼び出して、データ・ソースとの接続を確立します。
append(LoggingEvent event)
メソッドを実装します。このメソッドは、次のことを実行します。
close
メソッドを実装して、データ・ソースとの接続を閉じます。
例5-3に、この節で説明した手順を示します。
例5-3 例: サンプルLog4jアペンダの起動
package weblogic.logging.examples; import java.util.Enumeration; import org.apache.log4j.AppenderSkeleton; import org.apache.log4j.PropertyConfigurator; import org.apache.log4j.Logger; import org.apache.log4j.spi.Filter; import org.apache.log4j.spi.LoggingEvent; import weblogic.logging.LoggerNotAvailableException; import weblogic.logging.NonCatalogLogger; import weblogic.logging.Severities; import weblogic.logging.log4j.AppenderNames; import weblogic.logging.log4j.Log4jLoggingHelper; import weblogic.logging.log4j.WLLog4jLevel; import weblogic.logging.log4j.WLLog4jLogEvent; import org.apache.log4j.jdbc.JDBCAppender; import java.sql.Connection; import java.sql.SQLException; import javax.naming.InitialContext; import weblogic.logging.log4j.WLLog4jLogEvent; import weblogic.logging.Severities; /** * This class sets up a Log4j Appender as a listener to the * Server Logger for log events. */ public class Log4jAppenderExampleStartup { public static void main(String[] args) { try { System.out.println("Invoked the appender example startup class"); Logger serverLogger = Log4jLoggingHelper.getLog4jServerLogger(); // Configure the JDBC appender MyJDBCAppender jdbcAppender = new MyJDBCAppender(); // Now add the JDBC appender to the server logger serverLogger.addAppender(jdbcAppender); // Now test the filter NonCatalogLogger nc = new NonCatalogLogger("MyAppenderTest"); nc.info("Test INFO message"); nc.warning("Test WARNING message"); } catch(Exception ex) { System.err.println("Init failure " + ex.getMessage()); ex.printStackTrace(); } } private static class MyJDBCAppender extends AppenderSkeleton { private Connection connection; private java.sql.PreparedStatement stmt; public MyJDBCAppender() throws javax.naming.NamingException, SQLException { InitialContext ctx = new InitialContext(); javax.sql.DataSource ds = (javax.sql.DataSource) ctx.lookup ("MyDataSource"); connection = ds.getConnection(); // Table schema creation SQL command // Create table SERVER_LOG (server_name char(30),msg_id char(20), // severity_level char(20),logger_name char(100),message varchar(2048), // timestamp long); stmt = connection.prepareStatement("INSERT INTO SERVER_LOG VALUES (?, ?, ?, ?, ?, ?)"); stmt.setEscapeProcessing(true); connection.setAutoCommit(true); } // Override execute method public void append(LoggingEvent event) { WLLog4jLogEvent wlsEvent = (WLLog4jLogEvent) event; try { stmt.setString(1, wlsEvent.getServerName()); stmt.setString(2, wlsEvent.getId()); stmt.setString(3, Severities.severityNumToString(wlsEvent.getSeverity())); stmt.setString(4, wlsEvent.getSubsystem()); stmt.setString(5, wlsEvent.getMessage().toString()); stmt.setLong(6, wlsEvent.getTimestamp()); stmt.executeUpdate(); } catch (SQLException e) { System.err.println(e.toString()); } } public boolean requiresLayout() { return false; } public void close() { try { stmt.close(); connection.close(); } catch(SQLException sqlex) { System.err.println("Error closing JDBC appender"); sqlex.printStackTrace(); } } } }
WebLogic Server 8.1より前のリリースで、WebLogicロギング・サービスからメッセージを受信するには、Java Management Extensions (JMX)リスナーを作成して、それをLogBroadcasterRuntimeMBean
に登録する方法しかありませんでした。WebLogic Server 8.1では、Javaロギング・ハンドラを使用してログ・メッセージを受信(サブスクライブ)することもできます。
Javaロギング・ハンドラでもJMXリスナーでも同様の結果が得られますが、JavaロギングAPIにはFormatter
クラスがあり、Handler
オブジェクトはこのクラスを使用して受信したメッセージをフォーマットできます。JMXでは、メッセージをフォーマットするこのようなAPIは提供されません。フォーマッタの詳細は、Sun APIのドキュメントのFormatter
クラスについては、(http://download.oracle.com/javase/6/docs/api/java/util/logging/Formatter.html
)を参照してください。
さらに、JavaロギングHandler
APIはJMX APIよりも使いやすく、過程が直接的です。たとえば、次のコードはJavaロギングLogger
オブジェクトを取得し、ハンドラにそのオブジェクトをサブスクライブさせます。
Logger logger = LoggingHelper.getServerLogger(); Handler h = null; h = new MyJDBCHandler(); logger.addHandler(h)
JMXリスナーを登録して同様の結果を得るには、例5-4のようなコードを使用する必要があります。コードでMBeanHome
インタフェース、RemoteMBeanServer
インタフェース、およびLogBroadcasterRuntimeMBean
をルックアップし、それからリスナーを登録します。
ローカル・マシンのログ・メッセージのサブスクライブにはJavaロギング・ハンドラ、リモート・マシンからのログ・メッセージの受信にはJMXリスナーが適しています。すでにモニターにJMXを使用しており、ログ・メッセージのフォーマットを変更したりログ・メッセージを別の出力に転送したりせずに単純にリスニングする場合は、JMXリスナーを使用します。それ以外の場合は、Javaロギング・ハンドラを使用します。
例5-4 JMXリスナーの登録
MBeanHome home = null; RemoteMBeanServer rmbs = null; //domain variables String url = "t3://localhost:7001"; String serverName = "Server1"; String username = "weblogic"; String password = "weblogic"; //Using MBeanHome to get MBeanServer. try { Environment env = new Environment(); env.setProviderUrl(url); env.setSecurityPrincipal(username); env.setSecurityCredentials(password); Context ctx = env.getInitialContext(); //Getting the Administration MBeanHome. home = (MBeanHome) ctx.lookup(MBeanHome.ADMIN_JNDI_NAME); System.out.println("Got the Admin MBeanHome: " + home ); rmbs = home.getMBeanServer(); } catch (Exception e) { System.out.println("Caught exception: " + e); } try { //Instantiating your listener class. MyListener listener = new MyListener(); MyFilter filter = new MyFilter(); //Construct the WebLogicObjectName of the server's //log broadcaster. WebLogicObjectName logBCOname = new WebLogicObjectName("TheLogBroadcaster", "LogBroadcasterRuntime", domainName, serverName); //Passing the name of the MBean and your listener class to the //addNotificationListener method of MBeanServer. rmbs.addNotificationListener(logBCOname, listener, filter, null); } catch(Exception e) { System.out.println("Exception: " + e); } }