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.resource;
017
018import org.opengion.fukurou.db.DBUtil;
019import org.opengion.fukurou.db.ApplicationInfo;
020import org.opengion.fukurou.util.StringUtil;
021import org.opengion.hayabusa.common.HybsSystem;
022import static org.opengion.fukurou.system.HybsConst.BUFFER_MIDDLE;                              // 8.0.0.0 (2021/10/01)
023
024import java.util.Collections;
025import java.util.Map;                                                                                                                   // 7.2.6.0 (2020/06/30)
026import java.util.LinkedHashMap;                                                                                                 // 7.2.6.0 (2020/06/30)
027import java.util.StringJoiner;                                                                                                  // 8.3.0.1 (2022/08/12)
028
029/**
030 * systemId と lang に対応した画面データを作成するデータロードクラスです。
031 *
032 * 画面データは、画面ID(GUIKEY)に対して、各種画面情報を持っています。
033 * 従来と異なるのは、同一画面IDに対して、アドレスやロールズを変えた情報を持てると言う
034 * 事です。これは、カスタマイズ時に、画面IDは変えずに、実際のアクセスされるアドレスを
035 * 変える事で、他のアプリケーションへの影響を最小限にして開発できます。
036 * linkタグや、submit などの gamenID を指定するカスタムタグでは、実際のアクセス先は、
037 * ログインユーザーのロールズでアクセス可能な画面のアドレスに転送されます。
038 * 作番毎のカスタマイズや、ユーザーロールに応じた飛び先変更などにも使用できます。
039 *
040 * 画面データでは、複数階層持てるように、画面階層(GUILVL)を持っています。このレベルに
041 * 応じて、分類(CLASSIFY)の表示方法が変わります。(擬似階層構造)
042 *
043 * 画面データでは、言語(LANG)は、条件から消えました。実際に名称を表示させる時は、
044 * 画面カラムID(LABEL_CLM)に対応する ラベル定義より、言語に応じたラベルを取得します。
045 * エンジン内部で使用している GUIInfo オブジェクト構築時に割り当てます。
046 * 分類(CLASSIFY)は、コードリソースに登録します。
047 *
048 * 画面データを作成する場合は、同一画面IDで、作成区分(KBSAKU)違いの場合は、
049 * 最も大きな作成区分を持つ画面情報を使用します。
050 * 作成区分(KBSAKU)='0' のデータは、マスタリソースとして、エンジンとともに
051 * 配布されるリソースになります。
052 *
053 * 画面データは、カラム定義のような、読込フラグ(FGLOAD)はありません。
054 * 画面情報(GUIInfo)は、ユーザーログイン毎に作成されます。(キャッシュは
055 * セッション情報に登録されます。)
056 * これは、画面アクセス条件を、ログイン時に済ますことで、高速化を図っています。
057 * 画面IDの件数が少ないことと、画面IDを自動作成した場合でも、
058 * ほとんどのケースで、すべて使用される可能性が非常に高い為です。
059 *
060 * SYSTEM_ID='**' は、共通リソースです。
061 * これは、システム間で共通に使用されるリソース情報を登録しておきます。
062 *
063 * @og.rev 4.0.0.0 (2004/12/31) 新規作成
064 * @og.group リソース管理
065 *
066 * @version     4.0
067 * @author      Kazuhiko Hasegawa
068 * @since       JDK5.0,
069 */
070final class GUIDataLoader {
071        // リソースの接続先を、取得します。
072        private final String DBID = HybsSystem.sys( "RESOURCE_DBID" );
073
074        // DBリソースの初期一括読み込みのクエリー
075        // ソート順は、画面IDオブジェクトの優先順(後優先)で、画面表示順ではありません。
076        // 5.6.4.3 (2013/05/24) FAQ追加 現段階ではシステムコードは考慮しない
077        // 6.3.8.4 (2015/10/09) GE80(FAQテーブル)の取得は廃止。(helpタグで行う)
078        // 6.3.9.0 (2015/11/06) コンパイル時に静的な値に初期化されるフィールドは static フィールドにしてください(findbugs)。
079
080        // 7.3.1.3 (2021/03/09)
081//      private static final String SEL_CLM = "select GUIKEY,GUILVL,LABEL_CLM,ADDRESS,SEQNO,GROUPS"
082//                                                                      + ",'' as CLASSIFY,ROLES,RWMODE,TARGET,PARAM,KBLINK,DYUPD,SYSTEM_ID"
083//                                                                      + ",KBSAKU" ;
084
085        // 7.3.1.3 (2021/03/09)
086        // 7.4.5.0 (2021/08/31) Firebird 対応
087//      private static final String QUERY = "select a.* from ("
088//                                                                      +       SEL_CLM + ",0 as SNO"
089//                                                                      + " from GEA11 where SYSTEM_ID='**' and FGJ='1'"                // エンジン共通
090//                                                                      + " union all "
091//                                                                      +  SEL_CLM + ",1 as SNO"
092//                                                                      + " from GEA11 where SYSTEM_ID IN (?,?) and FGJ='1'"    // RESOURCE_BASE_SYSTEM_ID , 最上位ののSYSTEM_ID
093//                                                                      + " ) a "               // 8.0.0.0 (2021/08/31)
094//                                                                      + " order by a.SNO,a.KBSAKU,a.SEQNO,a.GUIKEY" ;
095
096        // 8.0.0.0 (2021/10/01) RESOURCE_BASE_SYSTEM_ID は、SYSTEM_IDの配列で複数指定できる。
097//      private static final String QUERY = "select GUIKEY,GUILVL,LABEL_CLM,ADDRESS,SEQNO,GROUPS"
098//                                                                      + ",'' as CLASSIFY,ROLES,RWMODE,TARGET,PARAM,KBLINK,DYUPD,SYSTEM_ID,KBSAKU"
099//                                                                      + " from GEA11 where SYSTEM_ID = ? and FGJ='1'"                 // 8.0.0.0 注意 IN (?,?) → = ? に変更
100//                                                                      + " order by KBSAKU,SEQNO,GUIKEY" ;
101
102        // 8.3.0.1 (2022/08/12) CLASSIFYが正しく表示されない不具合対応
103        private static final String QUERY1 = "select GUIKEY,GUILVL,LABEL_CLM,ADDRESS,SEQNO,GROUPS"
104                                                                        + ",'' as CLASSIFY,ROLES,RWMODE,TARGET,PARAM,KBLINK,DYUPD,SYSTEM_ID,KBSAKU";
105        private static final String QUERY2 = "from GEA11 where SYSTEM_ID in (" ;
106        private static final String QUERY3 = ") and FGJ='1'"
107                                                                        + " order by KBSAKU,SEQNO,GUIKEY,SYSID_SEQ DESC" ;
108
109        // 7.2.6.0 (2020/06/30) "**"以外にベースとなるSYSTEM_ID(RESOURCE_BASE_SYSTEM_ID)設定の対応
110        /** 7.2.9.1 (2020/10/23) Collections.synchronizedMap で同期処理を行います。  */
111        private final Map<String,GUIData> guiMap = Collections.synchronizedMap( new LinkedHashMap<>() );        // 集約するキーは、GUIKEY+ROLES         // 7.2.9.4 (2020/11/20) private 追加
112
113        // 8.0.0.0 (2021/10/01) RESOURCE_BASE_SYSTEM_ID は、SYSTEM_IDの配列で複数指定できる。
114//      private final String SYSTEM_ID ;                                                                                        // システムID
115//      private final String BASE_SYS_ID ;                                                                                      // 7.2.9.2 (2020/10/30) ベースシステムID
116        private final String[] SYS_ARRAY;                                                                                       // 8.0.0.0 (2021/10/01)
117
118        /** コネクションにアプリケーション情報を追記するかどうか指定 */
119        public static final boolean USE_DB_APPLICATION_INFO = HybsSystem.sysBool( "USE_DB_APPLICATION_INFO" ) ;
120
121        // 3.8.7.0 (2006/12/15) アクセスログ取得の為,ApplicationInfoオブジェクトを設定
122        private final ApplicationInfo appInfo;
123
124        /**
125         *  SystemId 毎に ファクトリオブジェクトを作成します。
126         *
127         * @og.rev 7.2.9.2 (2020/10/30) ベースとなるSYSTEM_ID(RESOURCE_BASE_SYSTEM_ID)の取得
128         * @og.rev 8.0.0.0 (2021/10/01) RESOURCE_BASE_SYSTEM_ID は、SYSTEM_IDの配列で複数指定できる。
129         *
130//       * @param       systemId        システムID
131//       * @param       baseSys         ベースとなるSYSTEM_ID
132         * @param       sysAry          階層リソースの元となるSYSTEM_IDの配列(前方優先)
133         */
134//      GUIDataLoader( final String systemId,final String baseSys ) {
135        GUIDataLoader( final String[] sysAry ) {
136//              SYSTEM_ID   = systemId;
137//              BASE_SYS_ID = baseSys ;                 // 7.2.9.2 (2020/10/30)
138                SYS_ARRAY       = sysAry ;                      // 8.0.0.0 (2021/10/01)
139
140                // 3.8.7.0 (2006/12/15) アクセスログ取得の為、ApplicationInfoオブジェクトを設定
141                if( USE_DB_APPLICATION_INFO ) {
142                        appInfo = new ApplicationInfo();
143                        // ユーザーID、IPアドレス、ホスト名
144//                      appInfo.setClientInfo( SYSTEM_ID,HybsSystem.HOST_ADRS,HybsSystem.HOST_NAME );
145                        appInfo.setClientInfo( SYS_ARRAY[0],HybsSystem.HOST_ADRS,HybsSystem.HOST_NAME );
146                        // 画面ID、操作、プログラムID
147                        appInfo.setModuleInfo( "GUIDataLoader",null,null );
148                }
149                else {
150                        appInfo = null;
151                }
152
153                // ApplicationInfo の設定が終わってから実行します。
154                loadDBResource();
155        }
156
157        /**
158         * DBリソースより 画面データを取得、設定します。
159         * DBリソースは、GUIKEY,GUILVL,LABEL_CLM,ADDRESS,SEQNO,GROUPS,
160         * CLASSIFY,ROLES,RWMODE,TARGET,PARAM,KBLINK,DYUPD の順番で、GUIKEY の重複を許します。
161         * 重複している場合(ロール違い等)は、一つのオブジェクトとして作成され、
162         * 個々のログインユーザー毎にユニークになるように、設定する必要があります。
163         *
164         * ※ 以下のロジックは、後方優先であり、SYSTEM_IDの配列は前方優先なので逆順で回します。
165         *
166         * @og.rev 3.8.7.0 (2006/12/15) アクセスログ取得の為,ApplicationInfoオブジェクトを設定
167         * @og.rev 4.0.0.0 (2007/10/31) ロールの継承機能の追加・分類の取得を追加(暫定対応)
168         * @og.rev 5.3.1.0 (2011/01/01) 通常画面に対してアドレスを設定しない場合にロールが効かないバグを修正します。
169         * @og.rev 5.3.1.0 (2011/01/01) ロール継承機能廃止
170         * @og.rev 7.2.6.1 (2020/07/17) "**"以外にベースとなるSYSTEM_ID(RESOURCE_BASE_SYSTEM_ID)設定の対応
171         * @og.rev 8.3.0.1 (2022/08/12) CLASSIFYが正しく表示されない不具合対応 (8.0.0.0対応改善)
172         */
173        private void loadDBResource() {
174                final int size = SYS_ARRAY.length;
175
176                final int[] cnt = new int[size];        // 各SYSTEM_ID の個数
177                int selCnt = 0;
178
179                // 8.3.0.1 (2022/08/12) CLASSIFYが正しく表示されない不具合対応
180                final StringBuilder buf1 = new StringBuilder(BUFFER_MIDDLE);
181//              final StringBuilder buf2 = new StringBuilder(BUFFER_MIDDLE);
182                final StringJoiner sj = new StringJoiner(",");
183                buf1.append( QUERY1 )                                                                                                   // select句
184                        .append( ",case" );
185                for( int j=0; j<size; j++ ) {
186                        buf1.append( " when '" ).append( SYS_ARRAY[j] ).append( "' = SYSTEM_ID THEN " ).append( j );
187//                      buf2.append( ",?");
188                        sj.add("?");
189                }
190                buf1.append( " else 99 end as SYSID_SEQ " )
191                        .append( QUERY2 )                                                                                                       // from句
192                        .append( sj.toString() )                                                                                        // バインド変数
193                        .append( QUERY3 );                                                                                                      // order by句
194
195
196//              for( int j=size-1; j>=0; j-- ) {        // SYSTEM_IDの配列は、前方優先なので、逆順で回す必要がある。8.3.0.1 (2022/08/12) Delete
197//                      final String sysId = SYS_ARRAY[j];                                                                      // 8.3.0.1 (2022/08/12) Delete
198//                      final String[] args = new String[] { BASE_SYS_ID,SYSTEM_ID };           // 7.2.6.1 (2020/07/17)
199//                      final String[] args = new String[] { sysId } ;                                          // 8.3.0.1 (2022/08/12) Delete
200
201                        // 8.3.0.1 (2022/08/12) CLASSIFYが正しく表示されない不具合対応
202//                      final String[][] gea11 = DBUtil.dbExecute( QUERY,args,appInfo,DBID );
203                        final String[][] gea11 = DBUtil.dbExecute( buf1.toString(),SYS_ARRAY,appInfo,DBID );
204//                      final int[] cnt = new int[3];   // **,BASE_SYS_ID,SYSTEM_ID の個数
205
206                        // 7.2.6.0 (2020/06/30) "**"以外にベースとなるSYSTEM_ID(RESOURCE_BASE_SYSTEM_ID)設定の対応
207                        final int len = gea11.length;
208                        selCnt += len;
209                        String classify = "";
210                        for( int i=0; i<len; i++ ) {
211                                final String[] vals = gea11[i];
212//                              final int idx = Integer.parseInt( vals[GUIData.SNO] );
213
214                                // ロールの継承対応
215                                final int level = Integer.parseInt( vals[GUIData.GUILVL] );
216                                // 小分類
217                                if( level == 2 ) {
218                                        classify = vals[GUIData.GUIKEY];                // 暫定対応
219                                }
220                                // 通常
221                                else if( level >= 3 ) {
222                                        vals[GUIData.CLASSIFY] = classify;              // 暫定対応
223                                }
224
225                                // 5.3.1.0 (2011/01/01) 通常画面に対してアドレスを設定しない場合にロールが効かないバグを修正します。
226                                if( ( level == 1 || level == 2 ) && StringUtil.isEmpty( vals[GUIData.ADDRESS] ) ) {
227                                        vals[GUIData.ROLES] = null;
228                                }
229
230//                              final String key = vals[GUIData.GUIKEY] + "_" + vals[GUIData.ROLES] ;
231//                              guiMap.put( key,new GUIData( vals ) );          // GUIKEY+ROLES が同一の画面リソースは、後設定が有効となる。
232                                guiMap.put( vals[GUIData.GUIKEY], new GUIData( vals ) );                // 8.3.0.1 (2022/08/12) Modify
233
234//                              cnt[idx]++ ;
235//                              cnt[j]++ ;                                                                                                              // 8.3.0.1 (2022/08/12) Delete
236                        }
237//              }                                                                                                                                               // 8.3.0.1 (2022/08/12) Delete
238//              final int guiSize = guiMap.size();
239
240//              System.out.println( "  GUIDataLoader [" + guiSize + "] "
241//                      +       " ** [" + cnt[0] + "] " + BASE_SYS_ID + " [" + cnt[1] + "] " + SYSTEM_ID + " [" + cnt[2] + "] loaded" );
242
243                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE );
244                buf.append( "  " ).append( SYS_ARRAY[0] ).append( "  GUIDataLoader [" ).append( selCnt )
245                        .append( "] Map=[" ).append( guiMap.size() ).append( "] " );
246                for( int j=0; j<size; j++ ) {
247                        buf.append( SYS_ARRAY[j] ).append( "=[" ).append( cnt[j] ).append( "] " );
248                }
249                buf.append( "loaded." );
250                System.out.println( buf );
251        }
252
253        /**
254         * すべてのGUIData オブジェクト配列を取得します。
255         * プールに持っているすべてのキャッシュを、GUIData オブジェクト配列
256         * にして返します。
257         * このリソースは、List で管理しており、読み込み時にすべてキャッシュされます。
258         *
259         * @og.rev 7.2.6.0 (2020/06/30) "**"以外にベースとなるSYSTEM_ID(RESOURCE_BASE_SYSTEM_ID)設定の対応
260         *
261         * @return      すべてのGUIDataオブジェクト配列
262         */
263        public GUIData[] getAllData() {
264                synchronized( guiMap ) {                                                                                                // 7.2.6.0 (2020/06/30)
265                        if( guiMap.isEmpty() ) { loadDBResource(); }
266                        return guiMap.values().toArray( new GUIData[guiMap.size()] );
267                }
268        }
269
270        /**
271         * GUIData オブジェクトのキャッシュをクリアします。
272         *
273         * @og.rev 7.2.6.0 (2020/06/30) "**"以外にベースとなるSYSTEM_ID(RESOURCE_BASE_SYSTEM_ID)設定の対応
274         */
275        public void clear() {
276                synchronized( guiMap ) {                                                                                                // 7.2.6.0 (2020/06/30)
277                        guiMap.clear();
278                }
279        }
280}