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.hayabusa.common.HybsSystem;
019import org.opengion.hayabusa.common.HybsSystemException;
020
021import org.opengion.fukurou.system.DateSet;                                             // 6.4.2.0 (2016/01/29)
022import org.opengion.fukurou.util.HybsDateUtil;                                  // 6.4.2.0 (2016/01/29)
023import org.opengion.fukurou.db.ApplicationInfo;
024import org.opengion.fukurou.db.DBUtil;
025
026import java.util.Map;
027import java.util.HashMap;
028import java.util.LinkedHashMap;
029import java.util.Collections;                                                                   // 6.4.3.1 (2016/02/12)
030
031/**
032 * ユーザーアクセス画面管理テーブルを維持する為のクラスです。
033 * @og.group リソース管理
034 *
035 * @version  4.1.1.0
036 * @author   Sen.Li
037 * @since    JDK5.0,
038 */
039public final class UserAccessTable {
040        // 5.3.0.0 (2010/12/01) 画面アクセスの履歴(順番)を管理する機能を追加
041        private static final String GEA09_QUERY  = "SELECT GUIKEY,SEQNO,NAME_JA,LAST_ACCESS,FGFAVORITE,CLASSIFY,NEXTGUI FROM GEA09"
042                                                                        + " WHERE SYSTEM_ID=? AND USERID=?"
043                                                                        + " ORDER BY SEQNO,CLASSIFY,GUIKEY";
044        private static final String GEA09_DELETE = "DELETE FROM GEA09"
045                                                                        + " WHERE SYSTEM_ID=? AND USERID=? AND GUIKEY=?";
046        private static final String GEA09_INSERT = "INSERT INTO GEA09(SYSTEM_ID,USERID,GUIKEY,SEQNO,NAME_JA,CLASSIFY,FGJ"
047                                                                        + ",DYSET,USRSET,PGUPD) VALUES(?,?,?,?,?,?,?,?,?,?)";
048        // 5.3.0.0 (2010/12/01) 画面アクセスの履歴(順番)を管理する機能を追加
049        private static final String GEA09_UPDATE = "UPDATE GEA09 SET LAST_ACCESS=?,NEXTGUI=?"
050                                                                        + " WHERE SYSTEM_ID=? AND USERID=? AND GUIKEY=?";
051
052        private static final String DBID = HybsSystem.sys( "RESOURCE_DBID" );
053        private static final int GEA09_GUIKEY           = 0;
054        private static final int GEA09_SEQNO            = 1;
055        private static final int GEA09_NAME_JA          = 2;
056        private static final int GEA09_LAST_ACCESS      = 3;
057        private static final int GEA09_FGFAVORITE       = 4;
058        private static final int GEA09_CLASSIFY         = 5;
059        private static final int GEA09_NEXTGUI          = 6;    // 5.3.0.0 (2010/12/01) 追加
060        private static final String  FGFAVORITE_ON      = "1";
061
062        /** フラグキーの enum */
063        private enum FgKeys { GUIMAP_ONLY,GEA09_ONLY };
064//      private static enum FgKeys { GUIMAP_ONLY,GEA09_ONLY };
065
066        /**
067         * コンストラクター
068         * オブジェクトを作成できないように、privateにします。
069         *
070         * @og.rev 4.3.1.1 (2008/08/23) コンストラクタは、void 宣言を付けません。
071         */
072        private UserAccessTable() { }
073
074        /**
075         * ユーザーアクセス画面管理テーブルを整理します。
076         * このメソッドでは guiMap へのセットをしています(副作用の扱い)。
077         *
078         * @og.rev 4.1.1.0 (2008/01/30) 新規追加
079         * @og.rev 5.3.0.0 (2010/12/01) 画面アクセスの履歴(順番)を管理する機能を追加
080         * @og.rev 6.3.9.0 (2015/11/06) Map.keySet() ではなく、Map.entrySet() を使う様に変更。
081         * @og.rev 6.4.2.0 (2016/01/29) DateSet.getDate( String ) を利用するように修正します。
082         * @og.rev 6.4.2.0 (2016/01/29) HybsDateUtil.getDatePlus( String,int ) を直接利用するように修正します。
083         * @og.rev 6.4.3.1 (2016/02/12) Collections.unmodifiableMap( Map<? extends K,? extends V> ) で作成された変更不可のMapを返します。
084         * @og.rev 6.4.3.4 (2016/03/11) forループを、forEach メソッドに置き換えます。
085         *
086         * @param  guiMap               画面オブジェクトマップへの参照
087         * @param  systemId             システムID
088         * @param  userId               ユーザーID
089         * @param  lang                 言語
090         *
091         * @return      お気に入りマップ(Collections.unmodifiableMapされた書き換え不可のMap)
092         */
093        public static Map<String,FavoriteGUIData> makeAccessDB(
094                                                        final Map<String,GUIInfo> guiMap,final String systemId,final String userId,final String lang ) {
095                final String today                      = DateSet.getDate( "yyyyMMddHHmmss" );                          // 6.4.2.0 (2016/01/29)
096                final Map<String,FgKeys>        fgKeyMap = new HashMap<>();
097                /** コネクションにアプリケーション情報を追記するかどうか指定 */
098                final boolean USE_DB_APPLICATION_INFO  = HybsSystem.sysBool( "USE_DB_APPLICATION_INFO" ) ;
099                ApplicationInfo appInfo = null;
100                if( USE_DB_APPLICATION_INFO ) {
101                        appInfo = new ApplicationInfo();
102                        // ユーザーID,IPアドレス,ホスト名
103                        appInfo.setClientInfo( systemId,HybsSystem.HOST_ADRS,HybsSystem.HOST_NAME );
104                        // 画面ID,操作,プログラムID
105                        appInfo.setModuleInfo( "UserAccessTable",userId,"makeAccessDB" );
106                }
107
108                // guiMapのキーで、新マップを作成、
109                // 6.4.3.4 (2016/03/11) forループを、forEach メソッドに置き換えます。
110                guiMap.keySet().forEach( k -> fgKeyMap.put( k,FgKeys.GUIMAP_ONLY ) );
111
112                final Map<String,FavoriteGUIData> favoriteGuiMap = new LinkedHashMap<>();
113                // 4.1.1.0(2008/01/22)ユーザーアクセスの日付を取得し、accessPastDays 前の日付を計算する。
114                final String accessPastDays     = HybsSystem.sys( "ACCESS_TOKEI_PAST_DAYS" );
115                final int    diffDate                   = ( accessPastDays == null ) ? 0 : -Integer.parseInt( accessPastDays );
116                final String judgeTime          = HybsDateUtil.getDatePlus( today.substring( 0,8 ),diffDate );                  // 6.4.2.0 (2016/01/29)
117                String lastAccessTime   = null;
118                String key                              = null;
119                String[]   args                 = new String[] { systemId,userId };
120                final String[][] vals                   = DBUtil.dbExecute( GEA09_QUERY,args,appInfo,DBID );
121                final int len                                   = vals.length;
122
123                for( int i=0; i<len; i++ ) {
124                        key = vals[i][GEA09_GUIKEY];
125                        final GUIInfo gui = guiMap.get( key );
126                        if( gui == null ) {
127                                fgKeyMap.put( key,FgKeys.GEA09_ONLY );
128                        }
129                        else {
130                                // 5.3.0.0 (2010/12/01) 画面アクセスの履歴(順番)を管理する機能を追加
131                                final String nxtgui = vals[i][GEA09_NEXTGUI];
132                                if( nxtgui != null ) {
133                                        final String[] keys = nxtgui.split( "," );
134                                        for( final String nextKey : keys ) {
135                                                gui.setNextGuiKey( nextKey );           // DB から復活
136                                        }
137                                }
138
139                                fgKeyMap.remove( key );
140                                // 4.1.1.0(2008/01/22)お気に入りマップの作成
141                                if( FGFAVORITE_ON.equals( vals[i][GEA09_FGFAVORITE] ) && gui.isRead() ) {
142                                        favoriteGuiMap.put( key,new FavoriteGUIData( vals[i][GEA09_GUIKEY],vals[i][GEA09_SEQNO]
143                                                ,vals[i][GEA09_NAME_JA],vals[i][GEA09_CLASSIFY] ) );
144                                }
145                                // 4.1.1.0(2008/01/22)ACCESS_TOKEI_PAST_DAYSに定義された期間以内の画面を格上する。
146                                lastAccessTime = vals[i][GEA09_LAST_ACCESS];
147                                // 6.0.0.1 (2014/04/25) These nested if statements could be combined
148                                if( lastAccessTime.length() >= 8 && lastAccessTime.compareTo( judgeTime ) >=0 ) {
149                                        gui.setLevelUp();
150                                }
151                        }
152                }
153
154                final ResourceManager resource = ResourceFactory.newInstance( lang );
155                final String undefined = resource.getLabel( "UNDEFINED" );
156                // 6.3.9.0 (2015/11/06) Map.keySet() ではなく、Map.entrySet() を使う様に変更。
157                for( final Map.Entry<String,FgKeys> entry : fgKeyMap.entrySet() ) {
158                        final String conKey = entry.getKey();
159                        final FgKeys fgKey  = entry.getValue();
160                        switch ( fgKey ) {
161                                case GUIMAP_ONLY:
162                                        args = new String[] { systemId,userId,conKey,"9999999",guiMap.get(conKey).getLabel()
163                                                                                  ,undefined,"1",today,userId,"UsrAccsTbl" };
164                        DBUtil.dbExecute( GEA09_INSERT,args,appInfo,DBID );
165                                        break;
166                                case GEA09_ONLY:
167                                        args = new String[] { systemId,userId,conKey };
168                                        DBUtil.dbExecute( GEA09_DELETE,args,appInfo,DBID );
169                                        break;
170                                default:
171                                        final String errMsg = "guiMapとGEA09の突合せフラグが GUIMAP_ONLY と GEA09_ONLY 以外の値がセットされています。"
172                                                                  + "画面ID:" + conKey + " フラグ:" + fgKey;
173                                        throw new HybsSystemException( errMsg );
174                        }
175                }
176
177                return Collections.unmodifiableMap( favoriteGuiMap );           // 6.4.3.1 (2016/02/12)
178        }
179
180        /**
181         * ユーザーアクセス画面管理テーブルの最終アクセス時間を更新します。
182         *
183         * @og.rev 4.1.1.0 (2008/01/30) 新規追加
184         * @og.rev 5.3.0.0 (2010/12/01) 画面アクセスの履歴(順番)を管理する機能を追加
185         * @og.rev 6.8.4.2 (2017/12/25) nextGuiKeys(NEXTGUI)が、1000Byteのため、文字数制限しておきます。
186         *
187         * @param systemId                      システムID
188         * @param userId                        ユーザーID
189         * @param guiKey                        画面ID
190         * @param lastAccessTime        画面の最終アクセス時間
191         * @param nextGuiKeys           次にアクセスしている画面IDのCSV文字列
192         */
193        public static void updateLastAccessTime( final String systemId,final String userId,final String guiKey,final String lastAccessTime,final String nextGuiKeys ) {
194                /** コネクションにアプリケーション情報を追記するかどうか指定 */
195                final boolean USE_DB_APPLICATION_INFO  = HybsSystem.sysBool( "USE_DB_APPLICATION_INFO" ) ;
196                ApplicationInfo appInfo = null;
197                if( USE_DB_APPLICATION_INFO ) {
198                        appInfo = new ApplicationInfo();
199                        // ユーザーID,IPアドレス,ホスト名
200                        appInfo.setClientInfo( systemId,HybsSystem.HOST_ADRS,HybsSystem.HOST_NAME );
201                        // 画面ID,操作,プログラムID
202                        appInfo.setModuleInfo( "UserAccessTable",userId,"updateLastAccessTime" );
203                }
204
205                // 6.8.4.2 (2017/12/25) nextGuiKeys(NEXTGUI)が、1000Byteのため、文字数制限しておきます。
206                final String nextGui = nextGuiKeys != null && nextGuiKeys.length() > 1000 ? nextGuiKeys.substring( 0,1000 ) : nextGuiKeys ;
207
208                // 5.3.0.0 (2010/12/01) 画面アクセスの履歴(順番)を管理する機能を追加
209                final String[] args = new String[] { lastAccessTime,nextGui,systemId,userId,guiKey };   // 6.8.4.2 (2017/12/25)
210                DBUtil.dbExecute( GEA09_UPDATE,args,appInfo,DBID );
211        }
212}