001/*
002 * Copyright (c) 2009 The openGion Project.
003 *
004 * Licensed under the Apache License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 *
008 *     http://www.apache.org/licenses/LICENSE-2.0
009 *
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
013 * either express or implied. See the License for the specific language
014 * governing permissions and limitations under the License.
015 */
016package org.opengion.hayabusa.common;
017
018import java.io.PrintWriter;
019import java.io.IOException;
020import java.io.UnsupportedEncodingException;
021import java.lang.reflect.Field;
022import java.lang.reflect.Modifier;
023import java.net.MalformedURLException;
024import java.net.URL;
025import java.sql.Connection;
026import java.sql.PreparedStatement;
027import java.sql.ResultSet;
028import java.sql.SQLException;
029import java.util.ArrayList;
030import java.util.Date;
031import java.util.Enumeration;
032import java.util.HashMap;
033import java.util.LinkedHashMap;
034import java.util.TreeMap;
035import java.util.List;
036import java.util.Locale;
037import java.util.Map;
038import java.util.Set;
039import java.util.Collections;                                                                                   // 6.4.3.2 (2016/02/19)
040import java.util.Calendar;                                                                                              // 6.9.7.0 (2018/05/14)
041
042import javax.servlet.ServletContext;
043import javax.servlet.FilterRegistration;                                                                // 6.9.3.1 (2018/04/02)
044import javax.servlet.ServletRegistration;                                                               // 6.9.3.1 (2018/04/02)
045
046import org.opengion.fukurou.db.ConnectionFactory;
047import org.opengion.fukurou.system.BuildNumber;                                                 // 6.4.2.0 (2016/01/29) hayabusa.common.BuildNumber → fukurou.system.BuildNumber に移動
048import org.opengion.fukurou.system.Closer;
049import org.opengion.fukurou.util.FindClassFiles;
050import org.opengion.fukurou.system.LogWriter;
051import org.opengion.fukurou.system.ThrowUtil ;                                                  // 6.4.2.0 (2016/01/29)
052import org.opengion.fukurou.util.StringUtil;
053import org.opengion.fukurou.xml.XMLFileLoader;                                                  // 6.0.0.0 (2014/04/11)
054import org.opengion.fukurou.system.DateSet;                                                             // 6.9.7.0 (2018/05/14)
055import static org.opengion.fukurou.system.HybsConst.CR ;                                // 6.1.0.0 (2014/12/26)
056import static org.opengion.fukurou.system.HybsConst.BUFFER_MIDDLE;              // 6.1.0.0 (2014/12/26) refactoring
057import static org.opengion.fukurou.system.HybsConst.DB_FETCH_SIZE;              // 6.9.4.1 (2018/04/09)
058
059/**
060 * ログインしたサーブレットコンテキストに対応したシステムパラメータデータを取得するクラスです。
061 *
062 * システムパラメータデータ(GE12)は、パラメータ(PARAM_ID)に対して、各種設定値(PARAM)を
063 * 持っています。
064 * 従来は、resource.システムパラメータ の一般設定用の システムパラメータ ファイルと
065 * エンジン内部で持っている org/hsgw/hayabusa/resource/properties の
066 * システムパラメータ ファイルにより維持されていましたが、システムパラメータ
067 * 定義テーブル(GE12)と、commom/SystemData.java クラスによる管理に変更されました。
068 * 
069 * システムパラメータは、DBへのアクセスを決定するため、初期設定値を定義する必要があります。
070 * これは、、アプリケーション・ディスクリプタ(WEB-INF/web.xml)に、context-param として、
071 * キーと値のセットで、初期アクセス用の情報を渡します。
072 * システムパラメータ定義テーブル(GE12)には、SYSTEM_IDとして、通常のシステムIDと、
073 * エンジンパラメータがあります。エンジンパラメータは、SYSTEM_ID='**'として、登録
074 * されています。
075 *
076 * <table class="plain">
077 *   <caption>システムパラメータの説明</caption>
078 *   <tr><th>種類        </th><th>SYSTEM_ID</th><th>作成区分</th><th>説明</th></tr>
079 *   <tr><td>エンジン共通</td><td>**  </td><td>0:エンジン</td><td>エンジン共通で定義しているパラメータ                     </td></tr>
080 *   <tr><td>エンジン個別</td><td>個別  </td><td>0:エンジン</td><td>システム毎にエンジンが登録しているパラメータ </td></tr>
081 *   <tr><td>システム共通</td><td>**  </td><td>1:システム </td><td>システム毎にエンジンが登録しているパラメータ </td></tr>
082 *   <tr><td>システム個別</td><td>個別  </td><td>1:システム </td><td>システム毎に各自が独自に登録しているパラメータ        </td></tr>
083 * </table>
084 *
085 * <table class="plain">
086 *   <caption>アプリケーション・ディスクリプタ(WEB-INF/web.xml)設定情報</caption>
087 *   <tr><th>パラメータ      </th><th>設定値例                           </th><th>解説                                     </th></tr>
088 *   <tr><td>SYSTEM_ID  </td><td>GE                                             </td><td>このアプリケーションのシステムID     </td></tr>
089 *   <tr><td>TOMCAT_PORT</td><td>8823                                   </td><td>Tomcat起動時ポート番号        </td></tr>
090 *   <tr><td>LOG_FILE   </td><td>log/log_$(yyyyMMdd).txt</td><td>JSPアクセスログ出力先          </td></tr>
091 * </table>
092 *
093 * ※ 5.6.7.0 (2013/07/27)
094 *    InitFileLoader により、特定のクラスパス内の 拡張XDK 形式の xml ファイル を処理します。
095 *    クラスパスが、"/resource" 以下のすべての xml ファイルは、DBIDが RESOURCE の接続先に対して処理されます。
096 *    クラスパスが、"/xml" 以下のすべての xml ファイルは、DBIDが DEFAULT の接続先に対して処理されます。
097 *    各クラスパス以下のファイルは、実フォルダでも、jar形式に圧縮された形式でも処理されます。
098 *
099 * ※ 6.0.0.0 (2014/04/11)
100 *    InitFileLoader が廃止になり、代わりに、XMLFileLoader を使用します。処理自体は、ほぼ同様です。
101 *
102 * @og.rev 4.0.0.0 (2005/01/31) 新規作成
103 * @og.rev 4.0.0.0 (2007/10/26) loadDBResourceのコネクションをFactoryから取るように変更
104 * @og.group 初期化
105 *
106 * @version  4.0
107 * @author   Kazuhiko Hasegawa
108 * @since    JDK5.0,
109 */
110public final class SystemParameter {
111
112        /** plugin クラスの取得用クラスパス指定       {@value}        */
113        public static final String PLUGIN = "org/opengion/plugin";
114
115        // 4.0.0.0 (2007/10/05) CONTXT_PATH に、DEFAULT '**' NOT NULL 制約を付ける。
116        /** システム共通/個別パラメータ(SYSTEM_ID in ('**',?) and KBSAKU &gt; '0')の一括読込のクエリー {@value}        */
117        public static final String QUERY = "SELECT PARAM_ID,PARAM,FGCRYPT"
118                                                                        + " FROM GE12 WHERE SYSTEM_ID IN (?,'**')"
119                                                                        + " AND CONTXT_PATH IN (?,'**')"
120                                                                        + " AND FGJ='1' AND KBSAKU > '0'"
121                                                                        + " ORDER BY SYSTEM_ID,CONTXT_PATH,FGJ,KBSAKU,SEQNO" ;
122
123        /** システム共通/個別パラメータ(SYSTEM_ID in ('**',?) and KBSAKU &gt; '0')の一括登録のクエリー {@value}        */
124        public static final String INS_SYS = "INSERT INTO GE12"
125//                                                                      + " ( SYSTEM_ID,CONTXT_PATH,PARAM_ID,SEQNO,PARAM,TITLE,CONTENTS,PARAM_LVL,FGJ,KBSAKU )"
126                                                                        + " ( SYSTEM_ID,CONTXT_PATH,PARAM_ID,SEQNO,PARAM,TITLE,CONTENTS,PARAM_LVL,FGJ,KBSAKU,DYSET )"   // 6.9.7.0 (2018/05/14)
127                                                                        + " SELECT"
128//                                                                      + " ?,?,?,?,?,TITLE,CONTENTS,PARAM_LVL,'1','0'"
129                                                                        + " ?,?,?,?,?,TITLE,CONTENTS,PARAM_LVL,'1','0',?"                                                                                               // 6.9.7.0 (2018/05/14)
130                                                                        + " FROM GE12 WHERE SYSTEM_ID='**' AND FGJ='1' AND KBSAKU='0' AND PARAM_ID=?" ;
131
132//      /** エンジン個別(SYSTEM_ID='個別' KBSAKU='0'  CONTXT_PATH='自身')パラメータの一括削除のクエリー  {@value}        */
133//      public static final String DEL_SYS = "DELETE FROM GE12 WHERE SYSTEM_ID=? AND KBSAKU='0' AND CONTXT_PATH=?";
134
135        /** エンジン個別(SYSTEM_ID='個別' KBSAKU='0'  CONTXT_PATH='自身')パラメータの一括削除と、過去情報の削除クエリー  {@value}        */
136        public static final String DEL_SYS = "DELETE FROM GE12 WHERE ( SYSTEM_ID=? AND KBSAKU='0' AND CONTXT_PATH=? )"                                  // 6.9.7.0 (2018/05/14)
137                                                                        + " OR ( SYSTEM_ID != '**' AND PARAM_LVL='9' AND ( DYSET < ? OR DYSET IS NULL ) )" ;                    // 過去のデータを考慮(DYSET is null)
138
139        /** 6.9.7.0 (2018/05/14) DYSET フォーマット   {@value}        */
140        public static final String DYSET_FORMAT = "yyyyMMddHHmmss" ;                    // 6.9.7.0 (2018/05/14)
141
142//      /** 6.9.3.0 (2018/03/26) データ検索時のフェッチサイズ  {@value} */
143//      private static final int DB_FETCH_SIZE = 1001 ;                                                 // HybsSystem に設定する前なので、HybsSystem.sysInt では取得できない。
144
145        /** システムパラメータ構築中のエラーをセットしていきます。 */
146        private static final List<String> ERR_MSG_LIST = new ArrayList<>();             // 6.4.1.1 (2016/01/16) errMsgList → ERR_MSG_LIST refactoring
147
148        /**
149         *      デフォルトコンストラクターをprivateにして、
150         *      オブジェクトの生成をさせないようにする。
151         *
152         */
153        private SystemParameter() {}
154
155        /**
156         * コンテキスト毎に システムパラメータオブジェクトを作成します。
157         * このクラスのスタートポイントメソッドになります。
158         *
159         * @og.rev 4.1.0.1 (2008/01/23) ログ出力先の変更(全てLogWriter経由で出力)
160         * @og.rev 5.5.4.4 (2012/07/20) LogWriter.log 追加
161         * @og.rev 5.5.4.4 (2012/07/20) SQLException は、catch しているので、loadDBResource からは、throws されない。
162         * @og.rev 5.7.2.0 (2014/01/10) Throwable の情報をもう少し詳細に出します。
163         * @og.rev 6.4.3.2 (2016/02/19) engParamとsysParamを初期設定して、同期化しておく。
164         * @og.rev 6.4.7.1 (2016/06/17) セキュリティの関連で、ログの詳細を出していませんでしたが、出すように変更します。
165         * @og.rev 6.8.6.0 (2018/01/19) ImageIO.write で、Java9で、NumberFormatException が出てしまう(暫定)対策。
166         *
167         * @param context Servletコンテキスト
168         *
169         * @return      システムパラメータのマップ
170         */
171        public static Map<String,String> makeSystemParameter( final ServletContext context ) {
172                ERR_MSG_LIST.clear() ;  // 呼び出し都度、エラーリストをクリアします。
173
174                // 6.4.3.2 (2016/02/19)
175                final Map<String,String> engParam = new LinkedHashMap<>();      // 最終的にマージして返す。
176                final Map<String,String> sysParam = new LinkedHashMap<>();      // DB書き戻し等あるので、設定中は分けておく。
177
178                try {
179                        // 6.8.6.0 (2018/01/19) ImageIO.write (暫定)対策
180                        // ※ ImageIO.getReaderMIMETypes(); での Java 9 has java.specification.version == 9 で、
181                        // Caused by: java.lang.NumberFormatException: For input string: "" が発生する対応
182                        // https://github.com/jai-imageio/jai-imageio-core/issues/24
183                        final String javaSpecVer = System.getProperty( "java.specification.version" );
184                        if( !javaSpecVer.startsWith( "1." ) ) {
185                                System.setProperty( "java.specification.version" , "1." + javaSpecVer );        // "java.specification.version" , "9" の場合、"1.9" に無理やり変更します。
186                        }
187
188                        final String contextName = getContextName( context );
189
190                        System.out.println( "Context Initialized [" + contextName + "]  " + new Date() );
191                        System.out.print( "  Version [" + BuildNumber.VERSION_NO + " " );
192                        System.out.print( BuildNumber.BUILD_TYPE );
193                        System.out.println( "]" );
194
195                        // システムデータクラスより、エンジンパラメータ情報を初期設定します。
196                        // エンジンパラメータは、SystemData クラスの public static メンバーです。
197                        loadParameter( engParam , SystemData.class );                           // 6.4.3.2 (2016/02/19)
198
199                        // コンテキストより取得できるシステムパラメータ情報を取得します。
200                        // web.xml で設定したパラメータを取得します。
201                        // SYSTEM_ID,DB_DRIVER,DB_URL などは、loadDBResource で使用します。
202                        loadInitialParameter( sysParam , context , contextName );       // 6.4.3.2 (2016/02/19)
203
204                        // システム個別に設定される、エンジン起動時情報を初期設定します。
205                        // エンジン起動時情報は、BuildNumber クラスの public static メンバーです。
206                        loadParameter( sysParam , BuildNumber.class );                          // 6.4.3.2 (2016/02/19)
207                        loadParameter( sysParam , PLUGIN );
208
209                        // GE12 データベースより読み取ります。
210                        // 引数のMapに読み込んだ値を追加した Map を返します。
211                        // つまり、システムパラメータ情報の上書きを行います。
212                        // 5.5.4.4 (2012/07/20) SQLException は、catch しているので、loadDBResource からは、throws されない。
213        //              try {
214                                loadDBResource( sysParam );                                                             // 6.4.3.2 (2016/02/19)
215        //              }
216        //              catch( final SQLException ex ) {
217        //              final String errMsg = "DB終了(close)処理を実行できませんでした。" + CR
218        //                                              + ex.getMessage() + ":" + ex.getSQLState() ;
219        //                      LogWriter.log( ex );                                    // 5.5.4.4 (2012/07/20) LogWriter.log 追加
220        //                      ERR_MSG_LIST.add( errMsg );
221        //              }
222                }
223                catch( final Throwable th ) {           // 3.6.1.0 (2005/01/05)
224                        LogWriter.log( th );                                            // 5.5.4.4 (2012/07/20) LogWriter.log 追加
225                        // 5.7.2.0 (2014/01/10) Throwable の情報をもう少し詳細に出します。
226                        final String errMsg = "処理を実行できませんでした。"  + CR ;
227        //                                      + th.getMessage()                                               + CR
228        //                                      + StringUtil.ogStackTrace( th ) ;
229        //              ERR_MSG_LIST.add( errMsg );
230        //              final String errMsg = "処理を実行できませんでした。"  + CR ;                  // 6.4.7.1 (2016/06/17)
231                        ERR_MSG_LIST.add( ThrowUtil.ogStackTrace( errMsg,th ) );                                // 6.4.2.0 (2016/01/29)
232                }
233                finally {
234                        // 初期値のエンジンパラメータに個別のシステムパラメータを追加設定します。
235                        // つまり、エンジンパラメータ情報に上書きを行います。
236                        // 6.3.9.1 (2015/11/27) engParam が null になることは無い(findbugs)。
237                                engParam.putAll( sysParam );
238
239                        final int errCnt = ERR_MSG_LIST.size();
240                        if( errCnt > 0 ) {
241                                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE )
242                                        .append( "【システムパラメータの初期化に失敗しました。】" )
243                                        .append( CR )
244                                        .append( "Tomcat の設定状況をご確認ください。" )
245                                        .append( CR )
246                                        .append( "========================================" )
247                                        .append( CR );
248
249                                for( int i=0; i<errCnt; i++ ) {
250                                        buf.append( ERR_MSG_LIST.get(i) )
251                                                .append( CR )
252                                                .append( "----------------------------------------" )
253                                                .append( CR );
254                                }
255                                System.out.println( buf );
256                                engParam.put( HybsSystem.LOCAL_CONTX_ERR_KEY,buf.toString() );
257                        }
258                }
259                return Collections.synchronizedMap( engParam ) ;                // 6.4.3.2 (2016/02/19) 同期化しておく。
260        }
261
262        /**
263         * コンテキストより、アプリケーション設定情報を取得します。
264         * 初期値は、アプリケーション・ディスクリプタ(WEB-INF/web.xml)に
265         * context-param として、キーと値のセットで、初期アクセス用の情報を渡します。
266         * データベースへの接続は、WEB-INF の DBConfig.xml で設定された情報を使用します。
267         *
268         * ここでは、各コンテキスト毎の内部情報を取得します。その中には、
269         * BuildNumber クラスで定義されている各種フィールド属性も含まれます。
270         *
271         * REAL_PATH       : アドレス(/)に対する、実ディレクトリパス
272         * CONTEXT_NAME    : アクセス先の仮想フォルダ名(URLのコンテキスト名)
273         * JSP             : アクセス先のJSPフォルダ名(/URLのコンテキスト名/jsp)
274         * SYS_LOG_URL     : jsp以下のログファイル名                                                                             // 7.1.0.0 (2020/01/20) jsp以下のログファイル名
275         * SYSTEM_ID       : web.xml で指定する、SYSTEM_ID       の設定値
276         * TOMCAT_PORT     : web.xml で指定する、Tomcat起動時ポート番号(8823)
277         * LOG_FILE        : web.xml で指定する、JSPアクセスログ出力先(log/log_$(yyyyMMdd).txt)
278         * SERVLET_INFO    : サーブレット情報 [例: Apache Tomcat/5.5.9     ]
279         * TOMCAT_WORK     : Tomcatワークの位置 [例: H:\java\tomcat5.5.17\work\Catalina\localhost\ver4  ]
280         * TOMCAT_HOME     : Tomcat環境の位置   [例: H:\java\tomcat5.5.17  ]
281         * LOGS_DIR        : Tomcatのログフォルダ                                                                    // 7.1.0.0 (2020/01/20) Tomcatのログフォルダ
282         * JAVA_HOME       : Java実行環境の位置 [例: H:\java\jdk150\jre ]
283         * HOST_NAME       : ホストの名前
284         * HOST_ADRS       : ホストのアドレス
285         * SERVER_INFO     : サーバー情報     [例: HN50G5 ( 200.1.50.165 ) ]
286         * ENGINE_INFO     : バージョン番号 [例: 4.3.6.6 ]
287         * RESOURCE_DBID   : "RESOURCE" 固定値を設定。WEB-INF/web.xml で指定しても無効です。
288         * OFFICE_HOME     : オフィースのHOME                                                                                 // 7.1.0.0 (2020/01/20) オフィースのHOME
289         * EXT_DIR         : 拡張jarフォルダ                                                                                 // 7.1.0.0 (2020/01/20) 拡張jarフォルダ
290         * IS_SERVICE      : サービス起動かどうか [サービス起動時はtrue]                                        // 7.1.0.0 (2020/01/20) サービス起動
291         *
292         * @og.rev 4.1.0.0 (2007/12/27) web.xmlからTOMCAT_PORTを読む処理を追加
293         * @og.rev 4.2.0.0 (2008/02/18) TOMCAT_PORTを環境変数から取得するよう変更
294         * @og.rev 4.2.0.0 (2008/02/20) web.xmlでSYSTEM_IDが空白の場合に大文字コンテキスト名が設定されるよう変更
295         * @og.rev 4.3.6.6 (2009/05/15) コンテキスト単位にエンジンバージョン情報を持つ(バージョンアップ判定用)
296         * @og.rev 5.6.7.1 (2013/08/09) RESOURCE_DBID の値に、"RESOURCE" を設定しておきます。
297         * @og.rev 5.6.7.3 (2013/08/23) TOMCAT_HOME を追加
298         * @og.rev 5.7.3.2 (2014/02/28) Tomcat8 対応。getRealPath( "/" ) の互換性のための修正。
299         * @og.rev 6.2.4.1 (2015/05/22) REAL_PATH 対応。realPath は、HybsSystem経由で、取得する。
300         * @og.rev 6.4.3.2 (2016/02/19) returnをやめて、引数のMapに直接値を設定します。
301         * @og.rev 6.9.0.1 (2018/02/05) RequestParam で設定される値を、できるだけ構築しておきます。
302         * @og.rev 6.9.3.1 (2018/04/02) web.xml で定義されている filter、servlet のキー名称を取得し、あれば、true を設定します。
303         * @og.rev 7.1.0.0 (2020/01/20) OFFICE_HOME,EXT_DIR,IS_SERVICE 環境変数の追加
304         *
305         * @param map 読み書きするMapオブジェクト
306         * @param context Servletコンテキスト
307         * @param contextName コンテキスト名
308         *
309         */
310        private static void loadInitialParameter( final Map<String,String> map,final ServletContext context,final String contextName ) {
311                // コンテキストの初期化パラメータ他の情報を登録しておきます。
312                final Enumeration<?> enume = context.getInitParameterNames() ;          // 4.3.3.6 (2008/11/15) Generics警告対応
313                while( enume.hasMoreElements() ) {
314                        final String key = (String)enume.nextElement();
315                        String val = context.getInitParameter( key );
316                        if( val != null && val.isEmpty() ) { val = null; }
317                        map.put( key,val );
318                }
319
320                // SYSTEM_IDがnullの場合は大文字のコンテキスト名を設定
321                if( map.get( "SYSTEM_ID" ) == null ){ // 4.2.0.0 (2008/02/20)
322                        map.put( "SYSTEM_ID", contextName.toUpperCase( Locale.JAPAN ) );
323                }
324
325                // 6.2.4.1 (2015/05/22) REAL_PATH 対応。realPath は、HybsSystem経由で、取得する。
326                final String REAL_PATH = context.getRealPath( "" ) ;                                    // 7.1.0.0 (2020/01/20)
327//              HybsSystem.setRealPath( context.getRealPath( "" ) );
328                HybsSystem.setRealPath( REAL_PATH );                                                                    // 7.1.0.0 (2020/01/20)
329
330                // 各種システム情報を登録しておきます。
331//              map.put( "REAL_PATH"    ,HybsSystem.getRealPath() );                                    // 6.2.4.1 (2015/05/22)
332                map.put( "REAL_PATH"    ,REAL_PATH );                                                                   // 6.2.4.1 (2015/05/22) 7.1.0.0 (2020/01/20)
333                map.put( "CONTEXT_NAME" ,contextName );
334                map.put( "JSP"                  ,"/" + contextName + "/jsp" );
335//              map.put( "SYS_LOG_URL"  ,REAL_PATH + System.getenv( "SYS_LOG_URL" ) );  // 7.1.0.0 (2020/01/20) jsp以下のログファイル名
336
337                map.put( "SERVLET_INFO" ,context.getServerInfo() );
338                map.put( "TOMCAT_WORK"  ,String.valueOf( context.getAttribute( "javax.servlet.context.tempdir" ) ) );
339
340                // 7.1.0.0 (2020/01/20) TOMCAT_HOME を先に取得しておく。
341                final String TOMCAT_HOME = System.getProperty( "catalina.home" );
342//              map.put( "TOMCAT_HOME"  ,System.getProperty( "catalina.home" ) );                                       // 5.6.7.3 (2013/08/23)
343                map.put( "TOMCAT_HOME"  ,TOMCAT_HOME );                                                                                         // 7.1.0.0 (2020/01/20) TomcatのHOMEフォルダ
344                map.put( "TOMCAT_LOGS"  ,TOMCAT_HOME + "/logs" );                                                                       // 7.1.0.0 (2020/01/20) Tomcatのログフォルダ
345//              map.put( "LOGS_DIR"             ,System.getProperty( "logsDir" ) );                                                     // 7.1.0.0 (2020/01/20) Tomcatのログフォルダ(コンテキスト単位)
346                map.put( "SERVICE_NAME" ,StringUtil.nval( System.getenv( "SERVICE_NAME" ),"" ) );       // 7.1.0.0 (2020/01/20) サービス起動時のサービス名
347
348                map.put( "JAVA_HOME"    ,System.getProperty( "java.home" ) );
349
350                map.put( "HOST_NAME"    ,HybsSystem.HOST_NAME  );
351                map.put( "HOST_ADRS"    ,HybsSystem.HOST_ADRS  );
352                map.put( "SERVER_INFO"  ,HybsSystem.HOST_NAME  + " ( " + HybsSystem.HOST_ADRS + " )" );
353                map.put( "ENGINE_INFO"  ,BuildNumber.ENGINE_INFO );
354                // 5.6.7.1 (2013/08/09) RESOURCE_DBID の値に、"RESOURCE" を設定しておきます。
355                map.put( "RESOURCE_DBID", "RESOURCE" );
356
357                // 7.1.0.0 (2020/01/20) OFFICE_HOME,EXT_DIR 環境変数の追加
358                map.put( "OFFICE_HOME"  ,System.getenv( "OFFICE_HOME" ) );                              // 7.1.0.0 (2020/01/20) オフィースのHOME
359                map.put( "EXT_DIR"              ,System.getenv( "EXT_DIR" ) );                                  // 7.1.0.0 (2020/01/20) 拡張jarフォルダ
360
361                // 7.1.0.0 (2020/01/20) バッチ起動の場合は、sun.java.command = "org.apache.catalina.startup.Bootstrap start" , サービス起動時は、未設定
362                final boolean IS_SERVICE = null == System.getProperty( "sun.java.command" );
363                map.put( "IS_SERVICE"   , String.valueOf( IS_SERVICE ) );                               // 7.1.0.0 (2020/01/20) サービスなら、true
364
365                // 6.3.9.1 (2015/11/27) final化にともない、ロジック整理
366                final String TOMCAT_PORT = StringUtil.nval( System.getenv( "CONNECTOR_PORT" ) , map.get( "TOMCAT_PORT" ) );
367
368                // 6.3.9.1 (2015/11/27) final化にともない、ロジック整理
369                final String HOST_URL = TOMCAT_PORT == null || TOMCAT_PORT.isEmpty()
370                                                                        ? "**" : HybsSystem.HOST_NAME + ":" + TOMCAT_PORT + "/" + contextName + "/";
371                map.put( "HOST_URL", HOST_URL );
372
373                // 6.9.0.1 (2018/02/05) RequestParam で設定される値を、できるだけ構築しておきます。
374                map.put( "SERVER_NAME"  , "localhost" );                                                // HybsSystem.HOST_NAME でも良いかも。
375                map.put( "SERVER_URL"   , "http://localhost:" + TOMCAT_PORT + "/" );
376                map.put( "CONTEXT_URL"  , "http://localhost:" + TOMCAT_PORT + "/" + contextName + "/" );
377
378                // 6.9.3.1 (2018/04/02) web.xml で定義されている filter、servlet のキー名称を取得し、あれば、true を設定します。
379                // キーワードは、USE_(大文字のフィルター名) です。
380                final Map<String,? extends FilterRegistration> filterMap = context.getFilterRegistrations();
381                for( final String fkey : filterMap.keySet() ) {
382                        map.put( "USE_" + fkey.toUpperCase( Locale.JAPAN ) , "true" );
383                }
384
385                // 6.9.3.1 (2018/04/02) web.xml で定義されている filter、servlet のキー名称を取得し、あれば、true を設定します。
386                // キーワードは、USE_(大文字のフィルター名) です。
387                final Map<String,? extends ServletRegistration> servletMap = context.getServletRegistrations();
388                for( final String skey : servletMap.keySet() ) {
389                        map.put( "USE_" + skey.toUpperCase( Locale.JAPAN ) , "true" );
390                }
391
392                System.out.println( "    Load Initial Parameter [" + map.size() + "] finished." );
393        }
394
395        /**
396         * アプリケーション個別に設定しているリソースDB(GE12)を取得します。
397         *
398         * データベースへの接続は、WEB-INF の DBConfig.xml で設定された情報を元に、
399         * org.opengion.fukurou.db.ConnectionFactory で接続先を取得します。
400         * ここでは、web.xml で定義された各アプリケーション個別のパラメ―タを取得します。
401         * SYSTEM_ID(必須) です。
402         *
403         * @og.rev 4.0.0.0 (2007/10/10) 接続先情報の管理見直し(コンテキスト初期設定)
404         * @og.rev 4.0.0.0 (2007/10/26) コネクションをファクトリーから取ってくるように変更
405         * @og.rev 4.3.6.5 (2009/05/08) dataパス内のXMLファイルも読み取るようにする
406         * @og.rev 4.3.6.6 (2009/05/15) ↑を廃止。自動インストール対応。
407         * @og.rev 5.1.2.0 (2010/01/01) connection.setAutoCommit は、ConnectionFactory で設定済みなので、コメントアウト
408         * @og.rev 5.1.9.0 (2010/08/01) 自動インストールの設定見直し(画面からのインストール対応)
409         * @og.rev 5.5.4.4 (2012/07/20) SQLException は、catch しているので、このメソッドからは、throws されない。
410         * @og.rev 5.5.4.5 (2012/07/27) 初期起動時のDB接続先は、RESOURCE_DBID とする。
411         * @og.rev 5.6.6.1 (2013/07/12) xml パス内のXMLファイルがあれば、DB登録します。
412         * @og.rev 5.6.7.0 (2013/07/27) InitFileLoader で、resource以下は、DBID=RESOURCE xml以下は、DBID=DEFAULT に登録します。
413         * @og.rev 5.6.7.3 (2013/08/23) DBID=RESOURCE 漏れ
414         * @og.rev 5.7.2.0 (2014/01/10) RuntimeException は、catch しないようにします。
415         * @og.rev 6.4.2.1 (2016/02/05) try-with-resources 文で記述。
416         * @og.rev 6.4.3.2 (2016/02/19) returnをやめて、引数のMapに直接値を設定します。
417         * @og.rev 6.4.7.1 (2016/06/17) セキュリティの関連で、ログの詳細を出していませんでしたが、出すように変更します。
418         * @og.rev 6.9.3.0 (2018/03/26) データ検索時のフェッチサイズを設定。
419         *
420         * @param       sysParam        入力システムパラメータマップ
421         */
422        private static void loadDBResource( final Map<String,String> sysParam ) {
423                final String SYSTEM_ID          = sysParam.get( "SYSTEM_ID" );
424
425                // 必須項目チェックを行います。SYSTEM_IDは必須です。
426                // これは、web.xml で定義が必要です。
427                // 4.0.0.0 (2007/10/23)接続情報XML化につきDB_URLチェックを削除
428                if( SYSTEM_ID == null || SYSTEM_ID.isEmpty() ) {
429                        final String errMsg = "システムパラメータの必須項目(SYSTEM_ID,DB_URL)が null です。" + CR
430                                                        + "SYSTEM_ID=[" + SYSTEM_ID + "] " + CR
431                                                        + "Versino=[" + BuildNumber.VERSION_NO + "] " + CR ;
432                        ERR_MSG_LIST.add( errMsg );
433                        return ;                        // 6.4.3.2 (2016/02/19)
434                }
435
436                Connection                      defConn         = null;                 // 5.6.7.0 (2013/07/27) DBID=DEFAULT  のコネクション
437                Connection                      rscConn         = null;                 // 5.6.7.0 (2013/07/27) DBID=RESOURCE のコネクション
438                boolean                         errFlag         = true;
439                // 6.4.1.1 (2016/01/16) PMD refactoring. Avoid declaring a variable if it is unreferenced before a possible exit point.
440                final String CONTEXT_NAME       = sysParam.get( "CONTEXT_NAME" );       // コンテキスト別システムリソース
441                final String HOST_URL           = sysParam.get( "HOST_URL" );           // 4.1.0.0 (2007/12/21) TOMCATへのアクセス用
442                final String RESOURCE_DBID      = sysParam.get( "RESOURCE_DBID" );      // 5.5.4.5 (2012/07/27) 初期起動時のDB接続先
443                try {
444
445                        // 4.0.0.0(2007/10/25)ConnectionをConnectionFactory経由で取得するように変更する。
446                        // コンテキスト名で接続しにいく。ApplicationInfoは使わないのでnull
447                        ConnectionFactory.init( CONTEXT_NAME, null ); // ConnectionFactoryの初期化
448
449                        defConn = ConnectionFactory.connection( null, null );                   // 5.6.7.0 (2013/07/27) DBID=DEFAULT のコネクション
450                        rscConn = ConnectionFactory.connection( RESOURCE_DBID, null );  // 5.6.7.0 (2013/07/27) DBID=RESOURCE のコネクション
451
452                        // 4.3.6.6 (2009/05/15)
453                        // 5.1.9.0 (2010/08/01) 自動インストールの設定見直し(画面からのインストール対応)
454                        final SystemInstaller installer = new SystemInstaller( defConn , rscConn , new PrintWriter( System.out, true ) );
455                        installer.autoInsUpd( SYSTEM_ID, CONTEXT_NAME, HOST_URL );
456
457                        // resource パス内のXMLファイルがあれば、先にDB登録します。(DBID=RESOURCE)
458                        // 6.0.0.0 (2014/04/11) XMLFileLoader に変更。
459                        // 7.0.1.4 (2018/11/26) XMLFileLoader をfinal宣言(どうせ、new してるので使い回しを止める。)
460//                      XMLFileLoader loader = new XMLFileLoader( rscConn,true );       // リソースコネクションとuseTimeStamp=true 指定
461//                      loader.loadClassPathFiles( "resource" ) ;
462                        final XMLFileLoader rscLoad = new XMLFileLoader( rscConn,true );        // リソースコネクションとuseTimeStamp=true 指定
463                        rscLoad.loadClassPathFiles( "resource" ) ;
464
465                        // 5.6.6.1 (2013/07/12) xml パス内のXMLファイルがあれば、DB登録します。
466                        // 6.0.0.0 (2014/04/11) XMLFileLoader に変更。
467                        // 7.0.1.4 (2018/11/26) XMLFileLoader をfinal宣言(どうせ、new してるので使い回しを止める。)
468//                      loader = new XMLFileLoader( defConn,true );                                     // デフォルトコネクションとuseTimeStamp=true 指定
469//                      loader.loadClassPathFiles( "xml" ) ;
470                        final XMLFileLoader xmlLoad = new XMLFileLoader( defConn,true );                                        // デフォルトコネクションとuseTimeStamp=true 指定
471                        xmlLoad.loadClassPathFiles( "xml" ) ;
472
473                        // コンテキスト単位のシステムパラメータを GE12 に設定します。
474                        // dbXMLResourceInsert の後に登録する必要があります。
475                        dbResourceUpdate( rscConn,sysParam );                   // 5.6.7.0 (2013/07/27) DBID=RESOURCE のコネクション
476                        rscConn.commit();
477        //              Closer.commit( rscConn );
478
479                        final Map<String,String> userMap = new HashMap<>(100);
480                        // 6.4.2.1 (2016/02/05) try-with-resources 文
481                        try( final PreparedStatement pstmt = rscConn.prepareStatement( QUERY ) ) {
482                                pstmt.setFetchSize( DB_FETCH_SIZE );                            // 6.9.3.0 (2018/03/26) データ検索時のフェッチサイズ
483                                pstmt.setString( 1,SYSTEM_ID );
484                                pstmt.setString( 2,HOST_URL ); // 4.1.0.0 (2007/12/21)
485                                try( final ResultSet resultSet = pstmt.executeQuery() ) {
486
487                                        while( resultSet.next() ) {
488                                                final String key = resultSet.getString(1);
489                                                final String val = resultSet.getString(2);
490                                                // 6.4.3.2 (2016/02/19) null値の登録はやめます。
491                                                userMap.put( key,val == null ? "" : val );              // 6.4.3.2 (2016/02/19)
492                                        }
493                                }
494                        }
495
496                        // DBを検索して、ユーザー設定リソース情報を取得します。
497
498                        System.out.println( "    Load DB Resource [" + userMap.size() + "] finished." );
499                        // リソースをマージします。
500                        sysParam.putAll( userMap );
501                        errFlag = false;        // エラーでない
502                }
503                catch( final SQLException ex ) {
504                        Closer.rollback( defConn );                     // 5.6.7.0 (2013/07/27) DBID=DEFAULT のコネクション
505                        Closer.rollback( rscConn );                     // 5.6.7.0 (2013/07/27) DBID=RESOURCE のコネクション
506                        LogWriter.log( ex );
507                        final String errMsg = ex.getMessage() + ":" + ex.getSQLState() ;
508                        ERR_MSG_LIST.add( errMsg );
509                }
510                catch( final UnsupportedEncodingException ex ) {
511                        Closer.rollback( defConn );                     // 5.6.7.0 (2013/07/27) DBID=DEFAULT のコネクション
512                        Closer.rollback( rscConn );                     // 5.6.7.0 (2013/07/27) DBID=RESOURCE のコネクション
513                        LogWriter.log( ex );
514                        final String errMsg = "UTF-8 がサポートされていない Java VM は、正規VMではありません。"
515                                                        + ex.getMessage();
516                        ERR_MSG_LIST.add( errMsg );
517                }
518                catch( final RuntimeException ex ) {
519                        Closer.rollback( defConn );                     // 5.6.7.0 (2013/07/27) DBID=DEFAULT のコネクション
520                        Closer.rollback( rscConn );                     // 5.6.7.0 (2013/07/27) DBID=RESOURCE のコネクション
521                        // 5.7.2.0 (2014/01/10) RuntimeException は、catch しないようにします。
522                        // 6.4.7.1 (2016/06/17)
523                        final String errMsg = "システムパラメータの設定で、エラーが発生しました。"
524                                                        + ex.getMessage();
525                        ERR_MSG_LIST.add( errMsg );
526
527                        throw new HybsSystemException( ex );            // 6.4.3.2 (2016/02/19) 生のRuntimeExceptionをラップしてthrowする。
528                }
529                finally {
530                        if( errFlag ) {
531                                ConnectionFactory.remove( defConn, null );                              // 5.6.7.0 (2013/07/27) DBID=DEFAULT のコネクション
532                                ConnectionFactory.remove( rscConn, RESOURCE_DBID );             // 5.6.7.3 (2013/08/23) DBID=RESOURCE 漏れ
533                        }
534                        else {
535                                ConnectionFactory.close( defConn, null );                               // 5.6.7.0 (2013/07/27) DBID=DEFAULT のコネクション
536                                ConnectionFactory.close( rscConn, RESOURCE_DBID );              // 5.6.7.3 (2013/08/23) DBID=RESOURCE 漏れ
537                        }
538                }
539        }
540
541        /**
542         * エンジン内部定義の初期リソース情報をDB(GE12)に登録します。
543         *
544         * 初期リソース情報は、KBSAKU='0' で登録されている情報で、一旦すべて削除
545         * してから、全てのリソース情報を追加するという形をとります。
546         * ただし、属性情報(名称や概要など)を別途登録する場合は、全てを
547         * 削除せずに、UPDATE する方向で検討したいと思います。
548         * なお、この情報をDB登録する理由は、リソースの設定値を変えたい場合に、
549         * キーが判らない(JavaDOCからしか読み取れない)のでは不便な為に
550         * 用意しておくだけで、内部では SystemData オブジェクトとして定義
551         * されている値を使用するため、このデータベース値は、使用していません。
552         *
553         * ※ 6.9.7.0 (2018/05/14)
554         *    自身のサーバーで追記した情報以外を消すため、登録日付が古いデータは削除します。
555         *    現状は、1ヶ月にしています。
556         *
557         * @og.rev 6.9.7.0 (2018/05/14) DYSETの登録と、旧データ(1ヶ月前)の削除を行います。
558         *
559         * @param       conn    登録用コネクション(リソース用)
560         * @param       map             入力システムパラメータマップ
561         * @throws      SQLException データベースアクセスエラー
562         */
563        private static void dbResourceUpdate( final Connection conn,final Map<String,String> map ) throws SQLException {
564
565                final String SYSTEM_ID = map.get( "SYSTEM_ID" );
566                final String HOST_URL  = map.get( "HOST_URL" );         // 4.1.0.0 (2007/12/21)
567
568                // 6.9.7.0 (2018/05/14) 現在時刻から、1ヶ月を引いた日付と比較して、削除データを決めます。
569                final Calendar now = Calendar.getInstance();
570                now.add( Calendar.MONTH , -1 );
571                final String OLD_DATE  = DateSet.getDate( now.getTimeInMillis() , DYSET_FORMAT );
572
573                // 既存の設定値を全件DELETEします。
574                int delCnt;
575                PreparedStatement pstmt = null;
576                try {
577                        pstmt = conn.prepareStatement( DEL_SYS );
578                        pstmt.setString( 1, SYSTEM_ID );
579                        pstmt.setString( 2, HOST_URL );                                         // 4.1.0.0 (2007/12/21)
580                        pstmt.setString( 3, OLD_DATE );                                         // 6.9.7.0 (2018/05/14) 古いデータを消します。
581                        delCnt = pstmt.executeUpdate();
582                }
583                finally {
584                        Closer.stmtClose( pstmt );
585                }
586
587                // 新設定値を全件INSERTします。
588                final Set<String> keyset = map.keySet();
589                final String[] keys = keyset.toArray( new String[keyset.size()] );
590
591                final String DYSET = DateSet.getDate( DYSET_FORMAT );   // 6.9.7.0 (2018/05/14)
592                int insCnt = 0;
593                try {
594                        pstmt = conn.prepareStatement( INS_SYS );
595                        for( int i=0; i<keys.length; i++ ) {
596                                pstmt.setString( 1,SYSTEM_ID );                                 // SYSTEM_ID
597                                pstmt.setString( 2,HOST_URL);                                   // CONTXT_PATH
598                                pstmt.setString( 3,keys[i] );                                   // PARAM_ID
599                                pstmt.setInt(    4,( i + 1 ) * 10 );                    // SEQNO
600                                pstmt.setString( 5,map.get( keys[i] ) );                // PARAM
601                                pstmt.setString( 6,DYSET );                                             // DYSET 6.9.7.0 (2018/05/14)
602                                pstmt.setString( 7,keys[i] );                                   // PARAM_ID
603                                insCnt += pstmt.executeUpdate();
604                        }
605                }
606                finally {
607                        Closer.stmtClose( pstmt );
608                }
609
610                System.out.print( "    DB Context Resource Reconfiguration " );
611                System.out.println( "DELETE=[" + delCnt + "],INSERT=[" + insCnt + "] finished." );
612        }
613
614        /**
615         *  ServletContext の名称を取得します。
616         *
617         * コンテキストのアクセスされたパス( /training など )の名称を、
618         * 取得します。(アクセス先の仮想フォルダ名)
619         * 以前は、配備記述子(WEB-INF/web.xml)の display-name 要素を見て、
620         * 無ければ、実フォルダ名を返していました。
621         * ※ warファイルのバージョン管理(セッション管理)対応として、## 以下の文字列を削除します。
622         *    ## は、war化の際に、"%23%23" に変換されるため、それも取り除き処理の対象とします。
623         *
624         * @og.rev 6.6.0.0 (2016/12/01) コンテキストパスから、##バージョン番号を取り去った値を返すようにします。
625         *
626         * @param  context Servletコンテキスト
627         *
628         * @return コンテキストのコンテキスト名
629         */
630        private static String getContextName( final ServletContext context ) {
631                String name = null;
632                try {
633                        final String path = context.getResource( "/" ).getFile();
634                        final int st = path.lastIndexOf( '/',path.length()-2 );         // 後ろから '/' を検索した位置(一番後ろは、'/' が付いているので注意)
635                        int ed = path.indexOf( "##",st+1 );                                                     // '/' 以降で、"##" を検索した位置
636                        if( ed < 0 ) {
637                                ed = path.indexOf( "%23%23",st+1 );                                             // '/' 以降で、"%23%23" を検索した位置
638                        }
639                        final int ed2 = ed > 0 ? ed : path.length()-1 ;                         // ## が無ければ、一番最後の文字を含まない、一つ前まで。あれば、そこまで
640                        name = path.substring( st+1,ed2 );
641                }
642                catch( final MalformedURLException ex ) {
643                        LogWriter.log( ex );
644                        final String errMsg = "このパス名は、正しいフォームではありません。 "
645                                                + ex.getMessage();
646                        ERR_MSG_LIST.add( errMsg );
647                }
648                return name ;
649        }
650
651        /**
652         * 指定のクラスの public static なフィールドキーと値のマップを作成します。
653         * 主に、エンジン関連のクラスにパラメータファイルをクラスとして定義し、
654         * エンジンとともに配布します。配布されたクラスを元に、パラメータを
655         * 読み取ります。
656         * この処理は リフレクションを使用してクラスの public static フィールドを
657         * 取得し、LinkedHashMap により、取得順をキープしたまま、Mapを返します。
658         *
659         * @og.rev 5.7.2.0 (2014/01/10) ERR_MSG_LIST は、一旦文字列に変換してから追加します。
660         * @og.rev 6.4.3.2 (2016/02/19) returnをやめて、引数のMapに直接値を設定します。
661         *
662         * @param  map   読み書きするMapオブジェクト
663         * @param  cls   クラスオブジェクト
664         */
665        private static void loadParameter( final Map<String,String> map , final Class<?> cls ) {
666
667                int cnt = 0;
668                try {
669                //      final Field[] field = cls.getFields(); の取得。public フィールドのみ
670                        for( final Field fld : cls.getFields() ) {
671                                if( Modifier.isStatic( fld.getModifiers() ) ) {
672                                        map.put( fld.getName() , (String)fld.get( null ) );
673                                        cnt++ ;
674                                }
675                        }
676                }
677                catch( final IllegalAccessException ex ) {
678                        LogWriter.log( ex );
679                        // 5.7.2.0 (2014/01/10) ERR_MSG_LIST は、一旦文字列に変換してから追加します。
680                        final String errMsg = "クラスから、パラメータを取得できませんでした。" + CR
681                                                                + "  クラス名=[" + cls.getName() + "]"                              + CR
682                                                                + ex.getMessage();
683                        ERR_MSG_LIST.add( errMsg );
684                }
685
686                System.out.println( "    ClassLoad " + cls.getName() + " Parameter [" + cnt + "] finished." );
687        }
688
689        /**
690         * 指定のキーワードのファイルをクラスパスより取得し、キーと値のマップを作成します。
691         * 主に、エンジン関連のクラスにパラメータファイルをPlugInクラスとして定義し、配布します。
692         * この処理の取得に、クラスパスの順序が関係します。最初に取得されたキーは、あとから
693         * 読み取られたクラスパスは、再セットしません。
694         *
695         * @og.rev 5.3.6.0 (2011/06/01) 並び順を、キーの名称順とする。
696         * @og.rev 5.7.2.0 (2014/01/10) ERR_MSG_LIST は、一旦文字列に変換してから追加します。
697         * @og.rev 5.7.2.0 (2014/01/10) RuntimeException は、catch しないようにします。
698         * @og.rev 6.4.3.2 (2016/02/19) returnをやめて、引数のMapに直接値を設定します。
699         * @og.rev 6.6.0.0 (2016/12/01) コンテキストパスから、##バージョン番号を取り去った値を返すようにします。
700         *
701         * @param       ioMap   読み書きするMapオブジェクト
702         * @param       keyword クラスオブジェクトを検索する元
703         */
704        private static void loadParameter( final Map<String,String> ioMap , final String keyword ) {
705                final Map<String,String> map = new TreeMap<>();         // 5.3.6.0 (2011/06/01) 並び順を、キーの名称順とする。
706                try {
707                        final ClassLoader loader = Thread.currentThread().getContextClassLoader();
708                        final Enumeration<URL> enume = loader.getResources( keyword );                  // 4.3.3.6 (2008/11/15) Generics警告対応
709                        while( enume != null && enume.hasMoreElements() ) {
710                                final URL url = enume.nextElement();            // 4.3.3.6 (2008/11/15) Generics警告対応
711                                // jar:file:/実ディレクトリ または、file:/実ディレクトリ
712                                final String dir = url.getFile().replaceAll( "%23%23","##" );           // 6.6.0.0 (2016/12/01)
713
714                                final FindClassFiles filenames = new FindClassFiles( dir,keyword );
715                                final String[] names = filenames.getFilenames();
716                                for( int i=0; i<names.length; i++ ) {
717                                        final String val = names[i];
718                                        final String key = val.substring( val.lastIndexOf( '.' )+1 );
719                                        if( key.indexOf( '_' ) >= 0 && !map.containsKey( key ) && key.indexOf( '$' ) < 0 ) {
720                                                map.put( key , val );
721                                        }
722                                }
723                                System.out.println( "    FileCheck " + dir + " [" + names.length + "] find." );
724                        }
725                }
726                catch( final IOException ex ) {
727                        LogWriter.log( ex );
728                        // 5.7.2.0 (2014/01/10) ERR_MSG_LIST は、一旦文字列に変換してから追加します。
729                        final String errMsg = "キーワードから、パラメータを取得できませんでした。"       + CR
730                                                + "  キーワード名=[" + keyword + "]"                                          + CR
731                                                + ex.getMessage();
732                        ERR_MSG_LIST.add( errMsg );
733                }
734                // 5.7.2.0 (2014/01/10) RuntimeException は、catch しないようにします。
735                System.out.println( "    FileLoad " + keyword + " Parameter [" + map.size() + "] finished." );
736                ioMap.putAll( map );            // TreeMap で、キーの名称順に並んだMapを、LinkedHashMapに登録順に設定する。
737        }
738}