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.taglib;
017
018import static org.opengion.fukurou.util.StringUtil.nval;
019
020import java.io.File;
021import java.util.Locale;
022import java.util.concurrent.ConcurrentMap;                                              // 6.4.3.3 (2016/03/04)
023import java.util.Arrays;
024import java.util.Map;                                                                                   // 6.9.9.0 (2018/08/20)
025
026import org.opengion.fukurou.business.ArrayTableModel;
027import org.opengion.fukurou.model.DataModel;                                    // 6.7.9.1 (2017/05/19) ArrayTableModel を、DataModel I/F に変更
028import org.opengion.fukurou.business.BizLogicHelper;
029import org.opengion.fukurou.db.Transaction;
030import org.opengion.fukurou.util.ErrMsg;
031import org.opengion.fukurou.util.ErrorMessage;
032import org.opengion.fukurou.util.HybsLoader;
033import org.opengion.fukurou.util.HybsLoaderConfig;
034import org.opengion.fukurou.util.HybsLoaderFactory;
035import org.opengion.fukurou.util.StringUtil;
036import org.opengion.fukurou.util.ToString;                                              // 6.1.1.0 (2015/01/17)
037import org.opengion.hayabusa.common.HybsSystem;
038import org.opengion.hayabusa.common.HybsSystemException;
039import org.opengion.hayabusa.db.DBTableModel;
040
041/**
042 * 業務ロジックを呼び出すためのタグです。
043 *
044 * logics属性に呼び出す業務ロジックのクラス名を記述します。
045 * このタグでは、複数の業務ロジックを1度に呼び出すことができ、
046 * DB接続のcommit,rollbackは一番最後に、1度のみ実行されます。
047 * 各業務ロジックは、記述した順番に呼び出されます。
048 *
049 * 業務ロジックは、{@link org.opengion.fukurou.business.AbstractBizLogic AbstractBizLogic}を
050 * 継承したサブクラスである必要があります。
051 * 標準サブクラスとして、BizLogic_ENTRYとBizLogic_TABLE を用意しています。
052 * BizLogic_ENTRY は、パラメーターのみを使用する業務ロジックです。
053 * BizLogic_TABLE は、配列型テーブルモデルをメインカーソルとした業務ロジックです。
054 *   配列型テーブルモデルでは、初期処理、ループ処理、後処理とメソッドがコールされます。
055 *
056 *  fstchk()          変更区分に関わらず   処理を始める前に呼び出し
057 *  first()           変更区分に関わらず   最初の行でのみ呼び出し
058 *    useLoop == true
059 *      befall( int row ) 変更区分に関わらず   各行について呼び出し(insert,modify,deleteの前に呼び出し)
060 *      insert( int row ) 変更区分が"A"の場合 各行について呼び出し
061 *      modify( int row ) 変更区分が"C"の場合 各行について呼び出し
062 *      delete( int row ) 変更区分が"D"の場合 各行について呼び出し
063 *      allrow( int row ) 変更区分に関わらず   各行について呼び出し(insert,modify,deleteの後に呼び出し)
064 *  last()            変更区分に関わらず   最後の行でのみ呼び出し
065 *
066 * 業務ロジッククラスについては、ホットデプロイ機能により、動的コンパイル、クラスロードが
067 * 行われます。
068 *
069 * 業務ロジックのソースディレクトリは、システムリソースの BIZLOGIC_SRC_PATH で定義されます。
070 * また、同様にクラスディレクトリは、システムリソースの BIZLOGIC_CLASS_PATH で定義されます。
071 * さらに、システムリソースの BIZLOGIC_HOTDEPLOY を false に設定することで、動的コンパイル
072 * 、クラスロードを行わなくすることもできます。
073 * この場合、予めコンパイルされたクラスを、初回呼び出し時に1回のみロードされます。
074 *
075 * SystemData の USE_SQL_INJECTION_CHECK が true か、quotCheck 属性が true の場合は、
076 * SQLインジェクション対策用のシングルクォートチェックを行います。リクエスト引数に
077 * シングルクォート(')が含まれると、エラーになります。
078 * 同様にUSE_XSS_CHECKがtrueか、xssCheck属性がtrueの場合は、
079 * クロスサイトススクリプティング(XSS)対策のためless/greater than signのチェックを行います。
080 *
081 * ※ このタグは、Transaction タグの対象です。
082 *
083 * @og.formSample
084 * ●形式:
085 *       ・<og:bizLog
086 *             logics       = "業務ロジックのクラス名"
087 *             command      = "ENTRY"
088 *             scope        = "session"
089 *             dbid         = "DEFAULT"
090 *             tableId      = "DEFAULT"
091 *             selectedAll  = "false"
092 *             modifyType   = "A"
093 *             keys         = "SYSTEM_ID"
094 *             vals         = "{@SYSTEM_ID}"
095 *        />
096 * ●body:なし
097 *
098 * ●Tag定義:
099 *   <og:bizLogic
100 *       logics           ○【TAG】実行する業務ロジック名を指定します(必須)。
101 *       command            【TAG】コマンドをセットします(初期値:ENTRY)
102 *       scope              【TAG】キャッシュする場合のスコープ[request/page/session/application]を指定します(初期値:session)
103 *       dbid               【TAG】(通常は使いません)Queryオブジェクトを作成する時のDB接続IDを指定します(初期値:null)
104 *       tableId            【TAG】(通常は使いません)結果のDBTableModelを、sessionに登録するときのキーを指定します
105 *       selectedAll        【TAG】データを全件選択済みとして処理するかどうか[true/false]を指定します(初期値:false)
106 *       modifyType         【TAG】DB検索時の モディファイタイプを指定します[A:追加/C:更新/D:削除]
107 *       keys               【TAG】リンク先に渡すキーをCSV形式で複数指定します
108 *       vals               【TAG】リンク先に渡す値をCSV形式で複数指定します
109 *       stopError          【TAG】処理エラーの時に処理を中止するかどうか[true/false]を設定します(初期値:true)
110 *       dispError          【TAG】エラー時にメッセージを表示するか[true/false]を設定します。通常はstopErrorと併用(初期値:true)
111 *       quotCheck          【TAG】リクエスト情報の シングルクォート(') 存在チェックを実施するかどうか[true/false]を設定します (初期値:USE_SQL_INJECTION_CHECK[=true])
112 *       xssCheck           【TAG】リクエスト情報の HTMLTag開始/終了文字(><) 存在チェックを実施するかどうか[true/false]を設定します (初期値:USE_XSS_CHECK[=true])
113 *       multi              【TAG】vals属性でパラメーターを取得する際、複数件存在する場合に、値を連結するかどうかを指定します(初期値:false)
114 *       debug              【TAG】デバッグ情報を出力するかどうか[true/false]を指定します(初期値:false)
115 *   />
116 *
117 * ●使用例
118 *     <!-- 業務ロジックの呼び出しを行います -->
119 *     <og:bizLogic logics="org.opengion.logic.gf9110.BizLogic_0001" keys="SYSTEM_ID" vals="{@MEM.SYSTEM_ID}" />
120 *
121 * @og.rev 5.1.1.0 (2009/12/01) 新規作成
122 * @og.group 業務ロジック
123 *
124 * @version 5.0
125 * @author Hiroki Nakamura
126 * @since JDK1.6,
127 */
128public class BizLogicTag extends CommonTagSupport {
129        /** このプログラムのVERSION文字列を設定します。   {@value} */
130        private static final String VERSION = "8.0.2.0 (2021/11/30)" ;
131        private static final long serialVersionUID = 802020211130L ;
132
133        /** command 引数に渡す事の出来る コマンド  エントリー {@value} */
134        public static final String CMD_ENTRY            = "ENTRY" ;             // 5.1.9.0 (2010/08/01)
135        private static final String ERR_MSG_ID          = HybsSystem.ERR_MSG_KEY;
136
137        private                         String          command         = CMD_ENTRY;
138        private                         String[]        logics          ;
139        private                         String          dbid            ;
140        private transient       DBTableModel table              ;
141        private                         String          tableId         = HybsSystem.TBL_MDL_KEY;
142        private                         boolean         selectedAll     ;
143        private                         String          modifyType      ;
144        private                         String[]        keys            ;
145        private                         String[]        vals            ;
146
147        private                         boolean         stopError       = true;
148        private                         boolean         quotCheck       = HybsSystem.sysBool( "USE_SQL_INJECTION_CHECK" );      // 4.0.0 (2005/08/31)
149        private                         boolean         xssCheck        = HybsSystem.sysBool( "USE_XSS_CHECK" );                        // 5.0.0.2 (2009/09/15)
150
151        private transient       ErrorMessage errMessage ;
152        private                         int             errCode                 = ErrorMessage.OK;
153        private                         int             executeCount    = -1;
154        private transient       DataModel<String>       arrTable;                       // 6.7.9.1 (2017/05/19) ArrayTableModel を、DataModel I/F に変更
155        private transient       HybsLoader                      loader  ;                       // 6.3.9.0 (2015/11/06) transient 追加
156
157        // 8.0.2.0 (2021/11/30) static定義するので、Tomcat再起動が必要
158        private static final String SRC_PATH            = HybsSystem.sys( "REAL_PATH" ) + HybsSystem.sys( "BIZLOGIC_SRC_PATH" );
159        private static final String CLS_PATH            = HybsSystem.sys( "REAL_PATH" ) + HybsSystem.sys( "BIZLOGIC_CLASS_PATH" );
160        private static final String WEB_LIB                     = HybsSystem.sys( "REAL_PATH" ) + "WEB-INF" + File.separator + "lib";
161        private static final String WEB_CLS                     = HybsSystem.sys( "REAL_PATH" ) + "WEB-INF" + File.separator + "classes";
162
163//      private                         String  srcDir                  = HybsSystem.sys( "REAL_PATH" ) + HybsSystem.sys( "BIZLOGIC_SRC_PATH" );
164        private                         String  srcDir                  = SRC_PATH;             // 8.0.2.0 (2021/11/30)
165//      private                         String  classDir                = HybsSystem.sys( "REAL_PATH" ) + HybsSystem.sys( "BIZLOGIC_CLASS_PATH" );
166        private                         String  classDir                = CLS_PATH;             // 8.0.2.0 (2021/11/30)
167        private                         boolean isAutoCompile   = HybsSystem.sysBool( "BIZLOGIC_AUTO_COMPILE" );
168        private                         boolean isHotDeploy             = HybsSystem.sysBool( "BIZLOGIC_HOT_DEPLOY" );
169        private                         boolean isMulti                 ;                               // 5.1.8.0 (2010/07/01) 追加
170
171        // 5.9.26.1 (2017/11/10) 実行エラーの際に、エラーを画面に出力するかどうか。
172        private                         boolean dispError               = true;
173
174        private static final String     CLASS_PATH;
175
176        // HotDeploy機能を使用する場合に、Javaクラスをコンパイルするためのクラスパスを設定します。
177        // 対象となるクラスパスは、WEB-INF/classes 及び WEB-INF/lib/*.jar です。
178
179        static {
180                final StringBuilder sb = new StringBuilder( BUFFER_MIDDLE )
181                        .append( '.' ).append( File.pathSeparatorChar );
182//              final File lib = new File( HybsSystem.sys( "REAL_PATH" ) + "WEB-INF" + File.separator + "lib" );
183                final File lib = new File( WEB_LIB );                   // 8.0.2.0 (2021/11/30)
184                final File[] libFiles = lib.listFiles();
185                // 6.3.9.0 (2015/11/06) null になっている可能性がある(findbugs)
186                if( libFiles != null ) {
187                        for( final File file : libFiles ) {
188                                // 5.1.1.2 (2009/12/10) File.pathSeparatorCharを使用
189                                // 5.1.8.0 (2010/07/01) libの検索パスの不具合対応
190                                sb.append( file.getAbsolutePath() ).append( File.pathSeparatorChar );
191                        }
192                }
193//              sb.append( HybsSystem.sys( "REAL_PATH" ) + "WEB-INF" + File.separator + "classes" )
194                sb.append( WEB_CLS )                                                    // 8.0.2.0 (2021/11/30)
195                        .append( File.pathSeparatorChar )
196                        // 5.1.8.0 (2010/07/01) bizの下も検索パスに追加
197//                      .append( HybsSystem.sys( "REAL_PATH" ) + HybsSystem.sys( "BIZLOGIC_CLASS_PATH" ) ).append( File.pathSeparatorChar );
198                        .append( CLS_PATH ).append( File.pathSeparatorChar );   // 8.0.2.0 (2021/11/30)
199
200                CLASS_PATH = sb.toString();
201        }
202
203        /**
204         * デフォルトコンストラクター
205         *
206         * @og.rev 6.4.2.0 (2016/01/29) PMD refactoring. Each class should declare at least one constructor.
207         */
208        public BizLogicTag() { super(); }               // これも、自動的に呼ばれるが、空のメソッドを作成すると警告されるので、明示的にしておきます。
209
210        /**
211         * Taglibの開始タグが見つかったときに処理する doStartTag() を オーバーライドします。
212         *
213         * @og.rev 5.3.4.0 (2011/04/01) command=ENTRY以外ではDBTableModelの処理を行わない。
214         * @og.rev 6.4.8.1 (2016/07/02) xssCheckを、doStartTag に移動
215         *
216         * @return      後続処理の指示(SKIP_BODY)
217         */
218        @Override
219        public int doStartTag() {
220                useXssCheck( xssCheck );                        // 6.4.8.1 (2016/07/02)
221
222                // 5.3.4.0 (2011/04/01)
223                if( CMD_ENTRY.equals( command ) ) {
224                        startQueryTransaction( tableId );
225                }
226
227                return SKIP_BODY ;
228        }
229
230        /**
231         * Taglibの終了タグが見つかったときに処理する doEndTag() を オーバーライドします。
232         *
233         * @og.rev 5.1.8.0 (2010/07/01) isMulti対応
234         * @og.rev 5.3.4.0 (2011/04/01) command=ENTRY以外ではDBTableModelの処理を行わない。
235         * @og.rev 6.4.8.1 (2016/07/02) xssCheckを、doStartTag に移動
236         * @og.rev 5.9.26.1 (2017/11/10) dispErrorの処理追加
237         *
238         * @return      後続処理の指示
239         */
240        @Override
241        public int doEndTag() {
242                debugPrint();
243
244                // 5.3.4.0 (2011/04/01)
245                useQuotCheck( quotCheck );
246
247                makeVals();
248                execute();
249
250                final String err = TaglibUtil.makeHTMLErrorTable( errMessage,getResource() );
251                if( err != null && err.length() > 0 ) {
252                        setSessionAttribute( ERR_MSG_ID,errMessage );
253                }
254
255                if( table != null && ! commitTableObject( tableId, table ) ) {
256                        jspPrint( "BizLoicTag 処理が割り込まれました。DBTableModel は登録しません。" );
257                        return SKIP_PAGE;
258                }
259
260                // 5.9.26.1 (2017/11/10) エラーメッセージをリクエスト変数で持つようにしておく
261                setRequestAttribute( "DB.ERR_MSG", err );
262                // 5.9.26.1 (2017/11/10) dispErrorで表示をコントロール
263                if( dispError ) {
264                        jspPrint( err );
265                }
266
267                if( errCode >= ErrorMessage.NG && stopError )  {
268                        return SKIP_PAGE;
269                }
270                else {
271                        return EVAL_PAGE;
272                }
273        }
274
275        /**
276         * タグリブオブジェクトをリリースします。
277         * キャッシュされて再利用されるので、フィールドの初期設定を行います。
278         *
279         * @og.rev 5.1.8.0 (2010/07/01) isMultiを追加
280         * @og.rev 5.1.9.0 (2010/08/01) Transaction 対応
281         * @og.rev 5.9.26.1 (2017/11/10) dispError追加
282         * @og.rev 8.0.2.0 (2021/11/30) srcDir,classDir static定義定数を使用
283         */
284        @Override
285        protected void release2() {
286                super.release2();
287                command                 = CMD_ENTRY;
288                logics                  = null;
289                dbid                    = null;
290                table                   = null;
291                tableId                 = HybsSystem.TBL_MDL_KEY;
292                selectedAll             = false;
293                modifyType      = null;
294                keys                    = null;
295                vals                    = null;
296                stopError               = true;
297                quotCheck               = HybsSystem.sysBool( "USE_SQL_INJECTION_CHECK" );
298                xssCheck                = HybsSystem.sysBool( "USE_XSS_CHECK" );
299                errMessage              = null;
300                errCode                 = ErrorMessage.OK;
301                executeCount    = -1;
302                arrTable                = null;
303                loader                  = null;
304//              srcDir                  = HybsSystem.sys( "REAL_PATH" ) + HybsSystem.sys( "BIZLOGIC_SRC_PATH" );
305                srcDir                  = SRC_PATH;             // 8.0.2.0 (2021/11/30)
306//              classDir                = HybsSystem.sys( "REAL_PATH" ) + HybsSystem.sys( "BIZLOGIC_CLASS_PATH" );
307                classDir                = CLS_PATH;             // 8.0.2.0 (2021/11/30)
308                isAutoCompile   = HybsSystem.sysBool( "BIZLOGIC_AUTO_COMPILE" );
309                isHotDeploy             = HybsSystem.sysBool( "BIZLOGIC_HOT_DEPLOY" );
310                isMulti                 = false;                // 5.1.8.0 (2010/07/01) 追加
311                dispError               = true;                 // 5.9.26.1 (2017/11/10) 追加
312        }
313
314        /**
315         * 業務ロジックを実行します。
316         *
317         * @og.rev 5.1.9.0 (2010/08/01) Transaction 対応
318         * @og.rev 5.3.4.0 (2011/04/01) command=ENTRY以外ではDBTableModelの処理を行わない。
319         * @og.rev 5.3.7.0 (2011/07/01) TransactionReal の引数変更 、Transaction対応で、close処理を入れる。
320         * @og.rev 5.6.0.3 (2012/01/24) arrTable に変更された値を、table に書き戻す処理を追加
321         * @og.rev 6.3.6.1 (2015/08/28) Transaction でAutoCloseableを使用したtry-with-resources構築に対応。
322         * @og.rev 6.3.6.1 (2015/08/28) Transaction でAutoCloseableを使用したtry-with-resources構築に対応。
323         * @og.rev 6.4.3.3 (2016/03/04) Map#forEach で対応する。
324         * @og.rev 6.7.9.1 (2017/05/19) ArrayTableModel を、DataModel I/F に変更
325         */
326        private void execute() {
327
328                // 5.3.4.0 (2011/04/01)
329                if( CMD_ENTRY.equals( command ) ) {
330                        table = (DBTableModel)getObject( tableId );
331                }
332
333                final int[] rowNos;
334                if( table != null ) {
335                        rowNos = getParameterRows();
336                        String[][] tblVals = new String[rowNos.length][table.getColumnCount()];
337                        String[] modTypes = new String[rowNos.length];
338                        for( int i=0; i<rowNos.length; i++ ) {
339                                tblVals[i]  = table.getValues( rowNos[i] );
340                                modTypes[i] = table.getModifyType( rowNos[i] );
341                        }
342                        arrTable = new ArrayTableModel( table.getNames(), tblVals, modTypes );
343                }
344                else {
345                        return;
346                }
347
348                // 5.1.9.0 (2010/08/01) Transaction 対応
349                // 5.3.7.0 (2011/07/01) Transaction対応で、close処理を入れる。
350        //              conn = ConnectionFactory.connection( dbid, null );
351                // 6.3.6.1 (2015/08/28) Transaction でAutoCloseableを使用したtry-with-resources構築に対応。
352                try( Transaction tran = getTransaction() ) {
353                        errMessage = new ErrorMessage();
354                        loader = HybsLoaderFactory.getLoader(
355                                                new HybsLoaderConfig( srcDir, classDir, isAutoCompile, isHotDeploy, CLASS_PATH )
356                                        );
357
358                        boolean rtn = false;
359                        for( int i=0; i<logics.length; i++ ) {
360        //                      BizLogic logic = (BizLogic)loader.newInstance( logics[i] );
361        //                      rtn = call( logic );
362        //                      rtn = call( logics[i] );
363                                rtn = call( logics[i] , tran );         // 5.1.9.0 (2010/08/01) Transaction 対応
364                                if( !rtn ) { break; }
365                        }
366
367                        // 5.6.0.3 (2012/01/24) arrTable に変更された値を、table に書き戻す処理
368                        // 6.9.7.0 (2018/05/14) PMD These nested if statements could be combined
369//                      if( arrTable != null ) {
370//                              // 6.4.3.3 (2016/03/04) 変更が無い場合は、nulllではなく、空のConcurrentMapを返しましす。
371//                              // 6.7.9.1 (2017/05/19) ArrayTableModel を、DataModel I/F に変更
372//                              if( arrTable instanceof ArrayTableModel ) {
373                        // 6.9.8.0 (2018/05/28) FindBugs:常に true を返す instanceof
374//                      if( arrTable != null && arrTable instanceof ArrayTableModel ) {
375                        if( arrTable != null ) {
376                                // 6.4.3.3 (2016/03/04) 変更が無い場合は、nulllではなく、空のConcurrentMapを返しましす。
377                                // 6.7.9.1 (2017/05/19) ArrayTableModel を、DataModel I/F に変更
378                                        final ConcurrentMap<Integer,String[]> valMap = ((ArrayTableModel)arrTable).getModifyVals();
379                                        valMap.forEach( (seq,vals) -> table.setValues( vals , rowNos[seq] ) );
380//                              }
381
382                                // 8.2.1.1 (2022/07/19) DELETE だけ特別に処置する(論理削除で)
383                                for( int i=0; i<rowNos.length; i++ ) {
384                                        if( DataModel.DELETE_TYPE.equals( arrTable.getModifyType(i) ) ) {
385                                                table.rowDelete( rowNos[i] );           // 論理削除…後で物理削除している。
386                                        }
387                                }
388                        }
389
390                        executeCount = rowNos.length;
391                        errCode = errMessage.getKekka();
392                        setRequestAttribute( "DB.COUNT"   , String.valueOf( executeCount ) );
393                        setRequestAttribute( "DB.ERR_CODE", String.valueOf( errCode ) );
394
395                        if( errCode < ErrorMessage.NG ) {
396        //                      Closer.commit( conn );
397                                tran.commit();                          // 5.1.9.0 (2010/08/01) Transaction 対応
398
399        //                      if( table != null && rowNos.length > 0 ) {
400                                        for( int j=rowNos.length-1; j>=0; j-- ) {
401                                                final int row = rowNos[j];
402                                                if( DBTableModel.DELETE_TYPE.equals( table.getModifyType( row ) ) ) {
403                                                        table.removeValue( row );
404                                                }
405                                                else {
406                                                        table.resetModify( row );
407                                                }
408                                        }
409        //                      }
410                        }
411                        else {
412        //                      Closer.rollback( conn );
413                                tran.rollback();                                // 5.1.9.0 (2010/08/01) Transaction 対応
414                        }
415        //              ConnectionFactory.close( conn, dbid );
416                }
417
418                // エラーメッセージの行番号を元の選択行に戻します。
419                final ErrMsg[] errs = errMessage.toArray();
420                final ErrorMessage errMsgTmp = new ErrorMessage();
421                for( int i=0; i<errs.length; i++ ) {
422                        if( table != null && rowNos.length > 0 ) {
423                                errMsgTmp.addMessage( errs[i].copy( rowNos[errs[i].getNo()] + 1 ) );
424                        }
425                        else {
426                                errMsgTmp.addMessage( errs[i].copy( errs[i].getNo() + 1 ) );
427                        }
428                }
429                errMessage = errMsgTmp;
430        }
431
432        /**
433         * 業務ロジックをCallします。
434         *
435         * @og.rev 5.1.9.0 (2010/08/01) Transaction 対応
436         * @og.rev 5.1.9.0 (2010/08/01) DBIDをセット、ConnectonではなくTransactionを渡すように変更
437         * @og.rev 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
438         * @og.rev 6.7.9.1 (2017/05/19) RETURN を返す変数が、logicName + .RETURN にします。
439         * @og.rev 6.9.9.0 (2018/08/20) 戻り値を返せるようにします。
440         *
441         * @param       logicName       業務ロジック名
442         * @param       tran            ランザクションオブジェクト
443         *
444         * @return 業務ロジックの呼び出しが成功したかどうか
445         */
446        private boolean call( final String logicName , final Transaction tran ) {
447                final BizLogicHelper logicHp = new BizLogicHelper( logicName, loader );
448
449                if( logicHp.isRequireTable() ) {
450                        if( arrTable == null ) {
451                                // 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
452                                final String errMsg = "TableModelが存在しません。logic=[" + logicName + "]" ;
453                                throw new HybsSystemException( errMsg );
454                        }
455                        else if( arrTable.getRowCount() == 0 ) { return true; } // 0件のときは呼び出ししない
456                        else { logicHp.setTable( arrTable ); }
457                }
458                logicHp.setTransaction( tran );
459                logicHp.setDbid( dbid );                // 5.1.9.0 (2010/08/01) DBIDをセット
460                logicHp.setKeys( keys );
461                logicHp.setVals( vals );
462                logicHp.setUserId( getUser().getUserID() );
463                logicHp.setParentPgId( getGUIInfoAttri( "KEY" ) );
464                logicHp.setLoader( loader );
465                if( isDebug() ) { logicHp.setDebug(); }
466
467                boolean rtn = false;
468                try {
469                        rtn = logicHp.exec();
470                }
471                catch( final Throwable th ) {
472                        tran.rollback();                // 5.1.9.0 (2010/08/01) Transaction 対応
473                        // 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
474                        final String errMsg = "業務ロジックの処理中にエラーが発生しました。" + th.getMessage() ;
475                        throw new HybsSystemException( errMsg,th );
476                }
477                errMessage.append( logicHp.getErrMsg() );
478
479                // 6.7.9.1 (2017/05/19) RETURN を返す変数が、logicName + .RETURN にします。
480                // 現状は、BizLogicHelper.RETURN なので、本来は、分けて使いたかったはずなので。
481                setRequestAttribute( logicName + ".RETURN"   , logicHp.getReturn() );
482                setRequestAttribute( "RETURN", logicHp.getReturn() );
483
484                // 6.9.9.0 (2018/08/20) 戻り値を返せるようにします。
485                final Map<String,String> rtnMap = logicHp.getReturnMap();
486                if( rtnMap != null ) {
487                        rtnMap.forEach( (k,v) -> setRequestAttribute( k,v ) );
488                }
489
490                if( isDebug() ) { jspPrint( logicHp.getDebugMsg() ); }
491
492                return rtn;
493        }
494
495        /**
496         * Valsの配列を生成します。
497         *
498         * @og.rev 5.1.8.0 (2010/07/01) 新規作成-
499         * @og.rev 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
500         */
501        private void makeVals() {
502                if( keys != null && keys.length > 0 ) {
503                        final boolean isSetVal = vals != null && vals.length > 0 ;
504                        if( isSetVal ) {
505                                if( keys.length != vals.length ) {
506                                        // 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
507                                        final String errMsg = "keysとValsの個数が異なります。"  + CR
508                                                                + "  keys=" + Arrays.toString( keys ) + CR
509                                                                + "  vals=" + Arrays.toString( vals ) ;
510                                        throw new HybsSystemException( errMsg );
511                                }
512                        }
513                        else {
514                                vals = new String[keys.length];
515                        }
516                        for( int i=0; i<keys.length; i++ ) {
517                                if( isSetVal ) {
518                                        if( isMulti )   { vals[i] = StringUtil.array2csv( getRequestParameterValues( vals[i] ) );}
519                                        else                    { vals[i] = getRequestParameter( vals[i] ); }
520                                }
521                                else {
522                                        if( isMulti )   { vals[i] = StringUtil.array2csv( getRequestValues( keys[i] ) );}
523                                        else                    { vals[i] = getRequestValue( keys[i] ); }
524                                }
525                        }
526                }
527        }
528
529        /**
530         * 【TAG】コマンドをセットします(初期値:ENTRY)。
531         *
532         * @og.tag
533         * command=ENTRY以外ではDBTableModelの処理を行いません。
534         * コマンドは,HTMLから[get/post]指定されますので,CMD_xxx で設定される
535         * フィールド定数値のいづれかを、指定できます。
536         *
537         * @param       cmd コマンド (public static final 宣言されている文字列)
538         * @see         <a href="../../../../constant-values.html#org.opengion.hayabusa.taglib.BizLogicTag.CMD_ENTRY">コマンド定数</a>
539         */
540        public void setCommand( final String cmd ) {
541                final String cmd2 = getRequestParameter( cmd );
542                if( cmd2 != null && cmd2.length() >= 0 ) { command = cmd2.toUpperCase(Locale.JAPAN); }
543        }
544
545        /**
546         * 【TAG】実行する業務ロジック名を指定します。
547         *
548         * @og.tag
549         * 実行する業務ロジック名を指定します。業務ロジック名は、クラス名を指定します。
550         * クラス名については、クラス自身の名称のみを指定することができます。
551         * (パッケージ名を含めた完全な形のクラス名を指定することもできます)
552         * また、CSV形式で、複数指定することもできます。
553         * この場合、指定した順番に処理されます。
554         *
555         * @param       lgs 業務ロジック名
556         */
557        public void setLogics( final String lgs ) {
558                logics = getCSVParameter( lgs );
559        }
560
561        /**
562         * 【TAG】(通常は使いません)Queryオブジェクトを作成する時のDB接続IDを指定します(初期値:null)。
563         *
564         * @og.tag
565         * Queryオブジェクトを作成する時のDB接続IDを指定します。
566         * これは、システムリソースで、DEFAULT_DB_URL 等で指定している データベース接続先
567         * 情報に、XX_DB_URL を定義することで、 dbid="XX" とすると、この 接続先を使用して
568         * データベースにアクセスできます。
569         *
570         * @param       id データベース接続ID
571         */
572        public void setDbid( final String id ) {
573                dbid = nval( getRequestParameter( id ),dbid );
574        }
575
576        /**
577         * 【TAG】(通常は使いません)結果のDBTableModelを、sessionに登録するときのキーを指定します
578         *              (初期値:HybsSystem#TBL_MDL_KEY[={@og.value HybsSystem#TBL_MDL_KEY}])。
579         *
580         * @og.tag
581         * 検索結果より、DBTableModelオブジェクトを作成します。これを、下流のviewタグ等に
582         * 渡す場合に、通常は、session を利用します。その場合の登録キーです。
583         * query タグを同時に実行して、結果を求める場合、同一メモリに配置される為、
584         * この tableId 属性を利用して、メモリ空間を分けます。
585         * (初期値:HybsSystem#TBL_MDL_KEY[={@og.value HybsSystem#TBL_MDL_KEY}])。
586         *
587         * @param       id テーブルID (sessionに登録する時のID)
588         */
589        public void setTableId( final String id ) {
590                tableId   = nval( getRequestParameter( id ),tableId );
591        }
592
593        /**
594         * 【TAG】データを全件選択済みとして処理するかどうか[true/false]を指定します(初期値:false)。
595         *
596         * @og.tag
597         * 全てのデータを選択済みデータとして扱って処理します。
598         * 全件処理する場合に、(true/false)を指定します。
599         * 初期値は false です。
600         *
601         * @param  all データを全件選択済み [true:全件選択済み/false:通常]
602         */
603        public void setSelectedAll( final String all ) {
604                selectedAll = nval( getRequestParameter( all ),selectedAll );
605        }
606
607        /**
608         * 【TAG】DB検索時の モディファイタイプを指定します[A:追加/C:更新/D:削除]。
609         *
610         * @og.tag
611         * DB検索時に、そのデータをA(追加)、C(更新)、D(削除)のモディファイタイプを
612         * つけた状態にします。
613         * その状態で、そのまま、update する事が可能になります。
614         *
615         * @param   type モディファイタイプ [A:追加/C:更新/D:削除]
616         */
617        public void setModifyType( final String type ) {
618                modifyType = nval( getRequestParameter( type ),modifyType );
619        }
620
621        /**
622         * 【TAG】リンク先に渡すキーをCSV形式で複数指定します。
623         *
624         * @og.tag リンク先に渡すキーを指定します。
625         * Keysだけを指定して、Valsを指定しない場合、Keysで指定された項目名に対応するパラメーターを取得し、
626         * Valsとして使用します。
627         *
628         * @og.rev 3.5.6.2 (2004/07/05) CommonTagSupport#getCSVParameter を使用
629         *
630         * @param       key リンク先に渡すキー(CSV形式)
631         */
632        public void setKeys( final String key ) {
633                keys = getCSVParameter( getRequestParameter( key ) );
634        }
635
636        /**
637         * 【TAG】リンク先に渡す値をCSV形式で複数指定します。
638         *
639         * @og.tag リンク先に渡す値を指定します。
640         * Keysだけを指定して、Valsを指定しない場合、Keysで指定された項目名に対応するパラメーターを取得し、
641         * Valsとして使用します。
642         * 分解方法は、CSV変数を先に分解してから、getRequestParameter で値を取得します。
643         * こうしないとデータ自身にカンマを持っている場合に分解をミスる為です。
644         *
645         * @og.rev 3.5.6.2 (2004/07/05) CommonTagSupport#getCSVParameter を使用
646         * @og.rev 5.1.8.0 (2010/07/01) isMuitl対応
647         *
648         * @param       val リンク先に渡す値(CSV形式)
649         */
650        public void setVals( final String val ) {
651                vals = StringUtil.csv2Array( val );
652        }
653
654        /**
655         * 【TAG】処理エラーの時に処理を中止するかどうか[true/false]を設定します(初期値:true)。
656         *
657         * @og.tag
658         * false(中止しない)に設定する場合、後続処理では、{&#064;DB.ERR_CODE}の値により、
659         * PLSQL/SQLの異常/正常終了によって分岐処理は可能となります。
660         * 初期値は、true(中止する)です。
661         *
662         * @param   flag  処理の中止 [true:中止する/false:中止しない]
663         */
664        public void setStopError( final String flag ) {
665                stopError = nval( getRequestParameter( flag ),stopError );
666        }
667
668        /**
669         * 【TAG】PLSQL/SQL処理エラーの時にエラーを画面表示するか[true/false]を設定します(初期値:true)。
670         *
671         * @og.tag
672         * false(表示しない)に設定する場合、後続処理では、{&#064;DB.ERR_MSG}の値により、
673         * 本来表示されるはずだったメッセージを取得可能です。
674         * stopErrorと併用して、JSON形式でエラーを返す場合等に利用します。
675         * 初期値は、true(表示する)です。
676         *
677         * @og.rev 5.9.26.1 (2017/11/10) 新規追加
678         *
679         * @param   flag  [true:表示する/false:表示しない]
680         */
681        public void setDispError( final String flag ) {
682                dispError = nval( getRequestParameter( flag ),dispError );
683        }
684
685        /**
686         * 【TAG】リクエスト情報の シングルクォート(') 存在チェックを実施するかどうか[true/false]を設定します
687         *              (初期値:USE_SQL_INJECTION_CHECK[={@og.value SystemData#USE_SQL_INJECTION_CHECK}])。
688         *
689         * @og.tag
690         * SQLインジェクション対策の一つとして、暫定的ではありますが、SQLのパラメータに
691         * 渡す文字列にシングルクォート(') を許さない設定にすれば、ある程度は防止できます。
692         * 数字タイプの引数には、 or 5=5 などのシングルクォートを使用しないコードを埋めても、
693         * 数字チェックで検出可能です。文字タイプの場合は、必ず (')をはずして、
694         * ' or 'A' like 'A のような形式になる為、(')チェックだけでも有効です。
695         * (') が含まれていたエラーにする(true)/かノーチェックか(false)を指定します。
696         * (初期値:システム定数のUSE_SQL_INJECTION_CHECK[={@og.value SystemData#USE_SQL_INJECTION_CHECK}])。
697         *
698         * @param   flag クォートチェック [true:する/それ以外:しない]
699         * @see         org.opengion.hayabusa.common.SystemData#USE_SQL_INJECTION_CHECK
700         */
701        public void setQuotCheck( final String flag ) {
702                quotCheck = nval( getRequestParameter( flag ),quotCheck );
703        }
704
705        /**
706         * 【TAG】リクエスト情報の HTMLTag開始/終了文字(&gt;&lt;) 存在チェックを実施するかどうか[true/false]を設定します
707         *              (初期値:USE_XSS_CHECK[={@og.value SystemData#USE_XSS_CHECK}])。
708         *
709         * @og.tag
710         * クロスサイトスクリプティング(XSS)対策の一環としてless/greater than signについてのチェックを行います。
711         * (&gt;&lt;) が含まれていたエラーにする(true)/かノーチェックか(false)を指定します。
712         * (初期値:システム定数のUSE_XSS_CHECK[={@og.value SystemData#USE_XSS_CHECK}])。
713         *
714         * @param       flag    XSSチェックする [true:チェックする/false:しない]
715         * @see         org.opengion.hayabusa.common.SystemData#USE_XSS_CHECK
716         */
717        public void setXssCheck( final String flag ) {
718                xssCheck = nval( getRequestParameter( flag ),xssCheck );
719        }
720
721        /**
722         * 【TAG】vals属性でパラメーターを取得する際、複数件存在する場合に、値を連結するかどうかを指定します(初期値:false)。
723         *
724         * @og.tag
725         * この属性がtrueに指定された場合、パラメーターが複数存在する場合に、カンマで連結します。
726         * 初期値は、false(連結しない)です。
727         *
728         * @og.rev 5.1.8.0 (2010/07/01) 新規作成
729         *
730         * @param       flag    値連結 [true:する/false:しない]
731         */
732        public void setMulti( final String flag ) {
733                 isMulti = nval( getRequestParameter( flag ),isMulti );
734        }
735
736        /**
737         * 表示データの HybsSystem.ROW_SEL_KEY を元に、選ばれた 行を処理の対象とします。
738         *
739         * @return      選択行の配列
740         */
741        @Override
742        protected int[] getParameterRows() {
743                final int[] rowNo ;
744                if( selectedAll ) {
745                        final int rowCnt = table.getRowCount();
746                        rowNo = new int[ rowCnt ];
747                        for( int i=0; i<rowCnt; i++ ) {
748                                rowNo[i] = i;
749                        }
750                } else {
751                        rowNo = super.getParameterRows();
752                }
753                return rowNo ;
754        }
755
756        /**
757         * このオブジェクトの文字列表現を返します。
758         * 基本的にデバッグ目的に使用します。
759         *
760         * @return このクラスの文字列表現
761         * @og.rtnNotNull
762         */
763        @Override
764        public String toString() {
765                return ToString.title( this.getClass().getName() )
766                                .println( "VERSION"                     ,VERSION                )
767                                .println( "command"                     ,command                )
768                                .println( "logics"                      ,logics                 )
769                                .println( "dbid"                        ,dbid                   )
770                                .println( "tableId"                     ,tableId                )
771                                .println( "selectedAll"         ,selectedAll    )
772                                .println( "modifyType"          ,modifyType             )
773                                .println( "keys"                        ,keys                   )
774                                .println( "vals"                        ,vals                   )
775                                .println( "stopError"           ,stopError              )
776                                .println( "quotCheck"           ,quotCheck              )
777                                .println( "xssCheck"            ,xssCheck               )
778                                .println( "executeCount"        ,executeCount   )
779                                .println( "errCode"                     ,errCode                )
780                                .println( "Other..."            ,getAttributes().getAttribute() )
781                                .fixForm().toString() ;
782        }
783}