OSDN Git Service

Ver8.5.2.0
[opengion/opengionV8.git] / uap / webapps / gf / src / org / opengion / hayabusa / taglib / BizLogicTag.java
1 /*
2  * Copyright (c) 2009 The openGion Project.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *     http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
13  * either express or implied. See the License for the specific language
14  * governing permissions and limitations under the License.
15  */
16 package org.opengion.hayabusa.taglib;
17
18 import static org.opengion.fukurou.util.StringUtil.nval;
19
20 import java.io.File;
21 import java.util.Locale;
22 import java.util.concurrent.ConcurrentMap;                                              // 6.4.3.3 (2016/03/04)
23 import java.util.Arrays;
24 import java.util.Map;                                                                                   // 6.9.9.0 (2018/08/20)
25
26 import org.opengion.fukurou.business.ArrayTableModel;
27 import org.opengion.fukurou.model.DataModel;                                    // 6.7.9.1 (2017/05/19) ArrayTableModel を、DataModel I/F に変更
28 import org.opengion.fukurou.business.BizLogicHelper;
29 import org.opengion.fukurou.db.Transaction;
30 import org.opengion.fukurou.util.ErrMsg;
31 import org.opengion.fukurou.util.ErrorMessage;
32 import org.opengion.fukurou.util.HybsLoader;
33 import org.opengion.fukurou.util.HybsLoaderConfig;
34 import org.opengion.fukurou.util.HybsLoaderFactory;
35 import org.opengion.fukurou.util.StringUtil;
36 import org.opengion.fukurou.util.ToString;                                              // 6.1.1.0 (2015/01/17)
37 import org.opengion.hayabusa.common.HybsSystem;
38 import org.opengion.hayabusa.common.HybsSystemException;
39 import org.opengion.hayabusa.db.DBTableModel;
40
41 /**
42  * 業務ロジックを呼び出すためのタグです。
43  *
44  * logics属性に呼び出す業務ロジックのクラス名を記述します。
45  * このタグでは、複数の業務ロジックを1度に呼び出すことができ、
46  * DB接続のcommit,rollbackは一番最後に、1度のみ実行されます。
47  * 各業務ロジックは、記述した順番に呼び出されます。
48  *
49  * 業務ロジックは、{@link org.opengion.fukurou.business.AbstractBizLogic AbstractBizLogic}を
50  * 継承したサブクラスである必要があります。
51  * 標準サブクラスとして、BizLogic_ENTRYとBizLogic_TABLE を用意しています。
52  * BizLogic_ENTRY は、パラメーターのみを使用する業務ロジックです。
53  * BizLogic_TABLE は、配列型テーブルモデルをメインカーソルとした業務ロジックです。
54  *   配列型テーブルモデルでは、初期処理、ループ処理、後処理とメソッドがコールされます。
55  *
56  *  fstchk()          変更区分に関わらず   処理を始める前に呼び出し
57  *  first()           変更区分に関わらず   最初の行でのみ呼び出し
58  *    useLoop == true
59  *      befall( int row ) 変更区分に関わらず  各行について呼び出し(insert,modify,deleteの前に呼び出し)
60  *      insert( int row ) 変更区分が"A"の場合 各行について呼び出し
61  *      modify( int row ) 変更区分が"C"の場合 各行について呼び出し
62  *      delete( int row ) 変更区分が"D"の場合 各行について呼び出し
63  *      allrow( int row ) 変更区分に関わらず  各行について呼び出し(insert,modify,deleteの後に呼び出し)
64  *  last()            変更区分に関わらず   最後の行でのみ呼び出し
65  *
66  * 業務ロジッククラスについては、ホットデプロイ機能により、動的コンパイル、クラスロードが
67  * 行われます。
68  *
69  * 業務ロジックのソースディレクトリは、システムリソースの BIZLOGIC_SRC_PATH で定義されます。
70  * また、同様にクラスディレクトリは、システムリソースの BIZLOGIC_CLASS_PATH で定義されます。
71  * さらに、システムリソースの BIZLOGIC_HOTDEPLOY を false に設定することで、動的コンパイル
72  * 、クラスロードを行わなくすることもできます。
73  * この場合、予めコンパイルされたクラスを、初回呼び出し時に1回のみロードされます。
74  *
75  * SystemData の USE_SQL_INJECTION_CHECK が true か、quotCheck 属性が true の場合は、
76  * SQLインジェクション対策用のシングルクォートチェックを行います。リクエスト引数に
77  * シングルクォート(')が含まれると、エラーになります。
78  * 同様にUSE_XSS_CHECKがtrueか、xssCheck属性がtrueの場合は、
79  * クロスサイトススクリプティング(XSS)対策のためless/greater than signのチェックを行います。
80  *
81  * ※ このタグは、Transaction タグの対象です。
82  *
83  * @og.formSample
84  * ●形式:
85  *       ・<og:bizLog
86  *             logics       = "業務ロジックのクラス名"
87  *             command      = "ENTRY"
88  *             scope        = "session"
89  *             dbid         = "DEFAULT"
90  *             tableId      = "DEFAULT"
91  *             selectedAll  = "false"
92  *             modifyType   = "A"
93  *             keys         = "SYSTEM_ID"
94  *             vals         = "{@SYSTEM_ID}"
95  *        />
96  * ●body:なし
97  *
98  * ●Tag定義:
99  *   <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  */
128 public class BizLogicTag extends CommonTagSupport {
129         /** このプログラムのVERSION文字列を設定します。       {@value} */
130         private static final String VERSION = "8.3.1.0 (2022/10/14)" ;
131         private static final long serialVersionUID = 831020221014L ;
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          * @og.rev 8.3.1.0 (2022/10/14) 正常終了時、過去のエラーメッセージが消えない不具合対応
238          *
239          * @return      後続処理の指示
240          */
241         @Override
242         public int doEndTag() {
243                 debugPrint();
244
245                 // 5.3.4.0 (2011/04/01)
246                 useQuotCheck( quotCheck );
247
248                 makeVals();
249                 execute();
250
251                 final String err = TaglibUtil.makeHTMLErrorTable( errMessage,getResource() );
252                 if( err != null && err.length() > 0 ) {
253                         setSessionAttribute( ERR_MSG_ID,errMessage );
254                 }
255                 // 8.3.1.0 (2022/10/14) Add
256                 else {
257                         removeSessionAttribute( ERR_MSG_ID );
258                 }
259
260                 if( table != null && ! commitTableObject( tableId, table ) ) {
261                         jspPrint( "BizLoicTag 処理が割り込まれました。DBTableModel は登録しません。" );
262                         return SKIP_PAGE;
263                 }
264
265                 // 5.9.26.1 (2017/11/10) エラーメッセージをリクエスト変数で持つようにしておく
266                 setRequestAttribute( "DB.ERR_MSG", err );
267                 // 5.9.26.1 (2017/11/10) dispErrorで表示をコントロール
268                 if( dispError ) {
269                         jspPrint( err );
270                 }
271
272                 if( errCode >= ErrorMessage.NG && stopError ) {
273                         return SKIP_PAGE;
274                 }
275                 else {
276                         return EVAL_PAGE;
277                 }
278         }
279
280         /**
281          * タグリブオブジェクトをリリースします。
282          * キャッシュされて再利用されるので、フィールドの初期設定を行います。
283          *
284          * @og.rev 5.1.8.0 (2010/07/01) isMultiを追加
285          * @og.rev 5.1.9.0 (2010/08/01) Transaction 対応
286          * @og.rev 5.9.26.1 (2017/11/10) dispError追加
287          * @og.rev 8.0.2.0 (2021/11/30) srcDir,classDir static定義定数を使用
288          */
289         @Override
290         protected void release2() {
291                 super.release2();
292                 command                 = CMD_ENTRY;
293                 logics                  = null;
294                 dbid                    = null;
295                 table                   = null;
296                 tableId                 = HybsSystem.TBL_MDL_KEY;
297                 selectedAll             = false;
298                 modifyType              = null;
299                 keys                    = null;
300                 vals                    = null;
301                 stopError               = true;
302                 quotCheck               = HybsSystem.sysBool( "USE_SQL_INJECTION_CHECK" );
303                 xssCheck                = HybsSystem.sysBool( "USE_XSS_CHECK" );
304                 errMessage              = null;
305                 errCode                 = ErrorMessage.OK;
306                 executeCount    = -1;
307                 arrTable                = null;
308                 loader                  = null;
309 //              srcDir                  = HybsSystem.sys( "REAL_PATH" ) + HybsSystem.sys( "BIZLOGIC_SRC_PATH" );
310                 srcDir                  = SRC_PATH;                                                                                             // 8.0.2.0 (2021/11/30)
311 //              classDir                = HybsSystem.sys( "REAL_PATH" ) + HybsSystem.sys( "BIZLOGIC_CLASS_PATH" );
312                 classDir                = CLS_PATH;                                                                                             // 8.0.2.0 (2021/11/30)
313                 isAutoCompile   = HybsSystem.sysBool( "BIZLOGIC_AUTO_COMPILE" );
314                 isHotDeploy             = HybsSystem.sysBool( "BIZLOGIC_HOT_DEPLOY" );
315                 isMulti                 = false;                                                                                                // 5.1.8.0 (2010/07/01) 追加
316                 dispError               = true;                                                                                                 // 5.9.26.1 (2017/11/10) 追加
317         }
318
319         /**
320          * 業務ロジックを実行します。
321          *
322          * @og.rev 5.1.9.0 (2010/08/01) Transaction 対応
323          * @og.rev 5.3.4.0 (2011/04/01) command=ENTRY以外ではDBTableModelの処理を行わない。
324          * @og.rev 5.3.7.0 (2011/07/01) TransactionReal の引数変更 、Transaction対応で、close処理を入れる。
325          * @og.rev 5.6.0.3 (2012/01/24) arrTable に変更された値を、table に書き戻す処理を追加
326          * @og.rev 6.3.6.1 (2015/08/28) Transaction でAutoCloseableを使用したtry-with-resources構築に対応。
327          * @og.rev 6.3.6.1 (2015/08/28) Transaction でAutoCloseableを使用したtry-with-resources構築に対応。
328          * @og.rev 6.4.3.3 (2016/03/04) Map#forEach で対応する。
329          * @og.rev 6.7.9.1 (2017/05/19) ArrayTableModel を、DataModel I/F に変更
330          */
331         private void execute() {
332
333                 // 5.3.4.0 (2011/04/01)
334                 if( CMD_ENTRY.equals( command ) ) {
335                         table = (DBTableModel)getObject( tableId );
336                 }
337
338                 final int[] rowNos;
339                 if( table != null ) {
340                         rowNos = getParameterRows();
341                         String[][] tblVals = new String[rowNos.length][table.getColumnCount()];
342                         String[] modTypes = new String[rowNos.length];
343                         for( int i=0; i<rowNos.length; i++ ) {
344                                 tblVals[i]  = table.getValues( rowNos[i] );
345                                 modTypes[i] = table.getModifyType( rowNos[i] );
346                         }
347                         arrTable = new ArrayTableModel( table.getNames(), tblVals, modTypes );
348                 }
349                 else {
350                         return;
351                 }
352
353                 // 5.1.9.0 (2010/08/01) Transaction 対応
354                 // 5.3.7.0 (2011/07/01) Transaction対応で、close処理を入れる。
355         //              conn = ConnectionFactory.connection( dbid, null );
356                 // 6.3.6.1 (2015/08/28) Transaction でAutoCloseableを使用したtry-with-resources構築に対応。
357                 try( Transaction tran = getTransaction() ) {
358                         errMessage = new ErrorMessage();
359                         loader = HybsLoaderFactory.getLoader(
360                                                 new HybsLoaderConfig( srcDir, classDir, isAutoCompile, isHotDeploy, CLASS_PATH )
361                                         );
362
363                         boolean rtn = false;
364                         for( int i=0; i<logics.length; i++ ) {
365         //                      BizLogic logic = (BizLogic)loader.newInstance( logics[i] );
366         //                      rtn = call( logic );
367         //                      rtn = call( logics[i] );
368                                 rtn = call( logics[i] , tran );                                                                 // 5.1.9.0 (2010/08/01) Transaction 対応
369                                 if( !rtn ) { break; }
370                         }
371
372                         // 5.6.0.3 (2012/01/24) arrTable に変更された値を、table に書き戻す処理
373                         // 6.9.7.0 (2018/05/14) PMD These nested if statements could be combined
374 //                      if( arrTable != null ) {
375 //                              // 6.4.3.3 (2016/03/04) 変更が無い場合は、nulllではなく、空のConcurrentMapを返しましす。
376 //                              // 6.7.9.1 (2017/05/19) ArrayTableModel を、DataModel I/F に変更
377 //                              if( arrTable instanceof ArrayTableModel ) {
378                         // 6.9.8.0 (2018/05/28) FindBugs:常に true を返す instanceof
379 //                      if( arrTable != null && arrTable instanceof ArrayTableModel ) {
380                         if( arrTable != null ) {
381                                 // 6.4.3.3 (2016/03/04) 変更が無い場合は、nulllではなく、空のConcurrentMapを返しましす。
382                                 // 6.7.9.1 (2017/05/19) ArrayTableModel を、DataModel I/F に変更
383                                         final ConcurrentMap<Integer,String[]> valMap = ((ArrayTableModel)arrTable).getModifyVals();
384                                         valMap.forEach( (seq,vals) -> table.setValues( vals , rowNos[seq] ) );
385 //                              }
386
387                                 // 8.2.1.1 (2022/07/19) DELETE だけ特別に処置する(論理削除で)
388                                 for( int i=0; i<rowNos.length; i++ ) {
389                                         if( DataModel.DELETE_TYPE.equals( arrTable.getModifyType(i) ) ) {
390                                                 // 論理削除…後で物理削除している。
391                                                 table.rowDelete( rowNos[i] );
392                                         }
393                                 }
394                         }
395
396                         executeCount = rowNos.length;
397                         errCode = errMessage.getKekka();
398                         setRequestAttribute( "DB.COUNT"   , String.valueOf( executeCount ) );
399                         setRequestAttribute( "DB.ERR_CODE", String.valueOf( errCode ) );
400
401                         if( errCode < ErrorMessage.NG ) {
402         //                      Closer.commit( conn );
403                                 tran.commit();                                                                                                  // 5.1.9.0 (2010/08/01) Transaction 対応
404
405         //                      if( table != null && rowNos.length > 0 ) {
406                                         for( int j=rowNos.length-1; j>=0; j-- ) {
407                                                 final int row = rowNos[j];
408                                                 if( DBTableModel.DELETE_TYPE.equals( table.getModifyType( row ) ) ) {
409                                                         table.removeValue( row );
410                                                 }
411                                                 else {
412                                                         table.resetModify( row );
413                                                 }
414                                         }
415         //                      }
416                         }
417                         else {
418         //                      Closer.rollback( conn );
419                                 tran.rollback();                                                                                                // 5.1.9.0 (2010/08/01) Transaction 対応
420                         }
421         //              ConnectionFactory.close( conn, dbid );
422                 }
423
424                 // エラーメッセージの行番号を元の選択行に戻します。
425                 final ErrMsg[] errs = errMessage.toArray();
426                 final ErrorMessage errMsgTmp = new ErrorMessage();
427                 for( int i=0; i<errs.length; i++ ) {
428                         if( table != null && rowNos.length > 0 ) {
429                                 errMsgTmp.addMessage( errs[i].copy( rowNos[errs[i].getNo()] + 1 ) );
430                         }
431                         else {
432                                 errMsgTmp.addMessage( errs[i].copy( errs[i].getNo() + 1 ) );
433                         }
434                 }
435                 errMessage = errMsgTmp;
436         }
437
438         /**
439          * 業務ロジックをCallします。
440          *
441          * @og.rev 5.1.9.0 (2010/08/01) Transaction 対応
442          * @og.rev 5.1.9.0 (2010/08/01) DBIDをセット、ConnectonではなくTransactionを渡すように変更
443          * @og.rev 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
444          * @og.rev 6.7.9.1 (2017/05/19) RETURN を返す変数が、logicName + .RETURN にします。
445          * @og.rev 6.9.9.0 (2018/08/20) 戻り値を返せるようにします。
446          *
447          * @param       logicName       業務ロジック名
448          * @param       tran            ランザクションオブジェクト
449          *
450          * @return      業務ロジックの呼び出しが成功したかどうか
451          */
452         private boolean call( final String logicName , final Transaction tran ) {
453                 final BizLogicHelper logicHp = new BizLogicHelper( logicName, loader );
454
455                 if( logicHp.isRequireTable() ) {
456                         if( arrTable == null ) {
457                                 // 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
458                                 final String errMsg = "TableModelが存在しません。logic=[" + logicName + "]" ;
459                                 throw new HybsSystemException( errMsg );
460                         }
461                         else if( arrTable.getRowCount() == 0 ) { return true; } // 0件のときは呼び出ししない
462                         else { logicHp.setTable( arrTable ); }
463                 }
464                 logicHp.setTransaction( tran );
465                 logicHp.setDbid( dbid );                                                                                                // 5.1.9.0 (2010/08/01) DBIDをセット
466                 logicHp.setKeys( keys );
467                 logicHp.setVals( vals );
468                 logicHp.setUserId( getUser().getUserID() );
469                 logicHp.setParentPgId( getGUIInfoAttri( "KEY" ) );
470                 logicHp.setLoader( loader );
471                 if( isDebug() ) { logicHp.setDebug(); }
472
473                 boolean rtn = false;
474                 try {
475                         rtn = logicHp.exec();
476                 }
477                 catch( final Throwable th ) {
478                         tran.rollback();                // 5.1.9.0 (2010/08/01) Transaction 対応
479                         // 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
480                         final String errMsg = "業務ロジックの処理中にエラーが発生しました。" + th.getMessage() ;
481                         throw new HybsSystemException( errMsg,th );
482                 }
483                 errMessage.append( logicHp.getErrMsg() );
484
485                 // 6.7.9.1 (2017/05/19) RETURN を返す変数が、logicName + .RETURN にします。
486                 // 現状は、BizLogicHelper.RETURN なので、本来は、分けて使いたかったはずなので。
487                 setRequestAttribute( logicName + ".RETURN"   , logicHp.getReturn() );
488                 setRequestAttribute( "RETURN", logicHp.getReturn() );
489
490                 // 6.9.9.0 (2018/08/20) 戻り値を返せるようにします。
491                 final Map<String,String> rtnMap = logicHp.getReturnMap();
492                 if( rtnMap != null ) {
493                         rtnMap.forEach( (k,v) -> setRequestAttribute( k,v ) );
494                 }
495
496                 if( isDebug() ) { jspPrint( logicHp.getDebugMsg() ); }
497
498                 return rtn;
499         }
500
501         /**
502          * Valsの配列を生成します。
503          *
504          * @og.rev 5.1.8.0 (2010/07/01) 新規作成-
505          * @og.rev 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
506          */
507         private void makeVals() {
508                 if( keys != null && keys.length > 0 ) {
509                         final boolean isSetVal = vals != null && vals.length > 0 ;
510                         if( isSetVal ) {
511                                 if( keys.length != vals.length ) {
512                                         // 5.6.7.0 (2013/07/27) Exception を throw するとき、一旦、errMsg 変数にセットします。
513                                         final String errMsg = "keysとValsの個数が異なります。"  + CR
514                                                                 + "  keys=" + Arrays.toString( keys ) + CR
515                                                                 + "  vals=" + Arrays.toString( vals ) ;
516                                         throw new HybsSystemException( errMsg );
517                                 }
518                         }
519                         else {
520                                 vals = new String[keys.length];
521                         }
522                         for( int i=0; i<keys.length; i++ ) {
523                                 if( isSetVal ) {
524                                         if( isMulti )   { vals[i] = StringUtil.array2csv( getRequestParameterValues( vals[i] ) );}
525                                         else                    { vals[i] = getRequestParameter( vals[i] ); }
526                                 }
527                                 else {
528                                         if( isMulti )   { vals[i] = StringUtil.array2csv( getRequestValues( keys[i] ) );}
529                                         else                    { vals[i] = getRequestValue( keys[i] ); }
530                                 }
531                         }
532                 }
533         }
534
535         /**
536          * 【TAG】コマンドをセットします(初期値:ENTRY)。
537          *
538          * @og.tag
539          * command=ENTRY以外ではDBTableModelの処理を行いません。
540          * コマンドは、HTMLから[get/post]指定されますので、CMD_xxx で設定される
541          * フィールド定数値のいづれかを、指定できます。
542          *
543          * @param       cmd     コマンド (public static final 宣言されている文字列)
544          * @see         <a href="../../../../constant-values.html#org.opengion.hayabusa.taglib.BizLogicTag.CMD_ENTRY">コマンド定数</a>
545          */
546         public void setCommand( final String cmd ) {
547                 final String cmd2 = getRequestParameter( cmd );
548                 if( cmd2 != null && cmd2.length() >= 0 ) { command = cmd2.toUpperCase(Locale.JAPAN); }
549         }
550
551         /**
552          * 【TAG】実行する業務ロジック名を指定します。
553          *
554          * @og.tag
555          * 実行する業務ロジック名を指定します。業務ロジック名は、クラス名を指定します。
556          * クラス名については、クラス自身の名称のみを指定することができます。
557          * (パッケージ名を含めた完全な形のクラス名を指定することもできます)
558          * また、CSV形式で、複数指定することもできます。
559          * この場合、指定した順番に処理されます。
560          *
561          * @param       lgs     業務ロジック名
562          */
563         public void setLogics( final String lgs ) {
564                 logics = getCSVParameter( lgs );
565         }
566
567         /**
568          * 【TAG】(通常は使いません)Queryオブジェクトを作成する時のDB接続IDを指定します(初期値:null)。
569          *
570          * @og.tag
571          * Queryオブジェクトを作成する時のDB接続IDを指定します。
572          * これは、システムリソースで、DEFAULT_DB_URL 等で指定している データベース接続先
573          * 情報に、XX_DB_URL を定義することで、 dbid="XX" とすると、この 接続先を使用して
574          * データベースにアクセスできます。
575          *
576          * @param       id      データベース接続ID
577          */
578         public void setDbid( final String id ) {
579                 dbid = nval( getRequestParameter( id ),dbid );
580         }
581
582         /**
583          * 【TAG】(通常は使いません)結果のDBTableModelを、sessionに登録するときのキーを指定します
584          *              (初期値:HybsSystem#TBL_MDL_KEY[={@og.value HybsSystem#TBL_MDL_KEY}])。
585          *
586          * @og.tag
587          * 検索結果より、DBTableModelオブジェクトを作成します。これを、下流のviewタグ等に
588          * 渡す場合に、通常は、session を利用します。その場合の登録キーです。
589          * query タグを同時に実行して、結果を求める場合、同一メモリに配置される為、
590          * この tableId 属性を利用して、メモリ空間を分けます。
591          * (初期値:HybsSystem#TBL_MDL_KEY[={@og.value HybsSystem#TBL_MDL_KEY}])。
592          *
593          * @param       id      テーブルID (sessionに登録する時のID)
594          */
595         public void setTableId( final String id ) {
596                 tableId = nval( getRequestParameter( id ),tableId );
597         }
598
599         /**
600          * 【TAG】データを全件選択済みとして処理するかどうか[true/false]を指定します(初期値:false)。
601          *
602          * @og.tag
603          * 全てのデータを選択済みデータとして扱って処理します。
604          * 全件処理する場合に、(true/false)を指定します。
605          * 初期値は false です。
606          *
607          * @param       all     データを全件選択済み [true:全件選択済み/false:通常]
608          */
609         public void setSelectedAll( final String all ) {
610                 selectedAll = nval( getRequestParameter( all ),selectedAll );
611         }
612
613         /**
614          * 【TAG】DB検索時の モディファイタイプを指定します[A:追加/C:更新/D:削除]。
615          *
616          * @og.tag
617          * DB検索時に、そのデータをA(追加)、C(更新)、D(削除)のモディファイタイプを
618          * つけた状態にします。
619          * その状態で、そのまま、update する事が可能になります。
620          *
621          * @param       type    モディファイタイプ [A:追加/C:更新/D:削除]
622          */
623         public void setModifyType( final String type ) {
624                 modifyType = nval( getRequestParameter( type ),modifyType );
625         }
626
627         /**
628          * 【TAG】リンク先に渡すキーをCSV形式で複数指定します。
629          *
630          * @og.tag
631          * リンク先に渡すキーを指定します。
632          * Keysだけを指定して、Valsを指定しない場合、Keysで指定された項目名に対応するパラメーターを取得し、
633          * Valsとして使用します。
634          *
635          * @og.rev 3.5.6.2 (2004/07/05) CommonTagSupport#getCSVParameter を使用
636          *
637          * @param       key     リンク先に渡すキー(CSV形式)
638          */
639         public void setKeys( final String key ) {
640                 keys = getCSVParameter( getRequestParameter( key ) );
641         }
642
643         /**
644          * 【TAG】リンク先に渡す値をCSV形式で複数指定します。
645          *
646          * @og.tag
647          * リンク先に渡す値を指定します。
648          * Keysだけを指定して、Valsを指定しない場合、Keysで指定された項目名に対応するパラメーターを取得し、
649          * Valsとして使用します。
650          * 分解方法は、CSV変数を先に分解してから、getRequestParameter で値を取得します。
651          * こうしないとデータ自身にカンマを持っている場合に分解をミスる為です。
652          *
653          * @og.rev 3.5.6.2 (2004/07/05) CommonTagSupport#getCSVParameter を使用
654          * @og.rev 5.1.8.0 (2010/07/01) isMuitl対応
655          *
656          * @param       val     リンク先に渡す値(CSV形式)
657          */
658         public void setVals( final String val ) {
659                 vals = StringUtil.csv2Array( val );
660         }
661
662         /**
663          * 【TAG】処理エラーの時に処理を中止するかどうか[true/false]を設定します(初期値:true)。
664          *
665          * @og.tag
666          * false(中止しない)に設定する場合、後続処理では、{&#064;DB.ERR_CODE}の値により、
667          * PLSQL/SQLの異常/正常終了によって分岐処理は可能となります。
668          * 初期値は、true(中止する)です。
669          *
670          * @param       flag    処理の中止 [true:中止する/false:中止しない]
671          */
672         public void setStopError( final String flag ) {
673                 stopError = nval( getRequestParameter( flag ),stopError );
674         }
675
676         /**
677          * 【TAG】PLSQL/SQL処理エラーの時にエラーを画面表示するか[true/false]を設定します(初期値:true)。
678          *
679          * @og.tag
680          * false(表示しない)に設定する場合、後続処理では、{&#064;DB.ERR_MSG}の値により、
681          * 本来表示されるはずだったメッセージを取得可能です。
682          * stopErrorと併用して、JSON形式でエラーを返す場合等に利用します。
683          * 初期値は、true(表示する)です。
684          *
685          * @og.rev 5.9.26.1 (2017/11/10) 新規追加
686          *
687          * @param       flag    [true:表示する/false:表示しない]
688          */
689         public void setDispError( final String flag ) {
690                 dispError = nval( getRequestParameter( flag ),dispError );
691         }
692
693         /**
694          * 【TAG】リクエスト情報の シングルクォート(') 存在チェックを実施するかどうか[true/false]を設定します
695          *              (初期値:USE_SQL_INJECTION_CHECK[={@og.value SystemData#USE_SQL_INJECTION_CHECK}])。
696          *
697          * @og.tag
698          * SQLインジェクション対策の一つとして、暫定的ではありますが、SQLのパラメータに
699          * 渡す文字列にシングルクォート(') を許さない設定にすれば、ある程度は防止できます。
700          * 数字タイプの引数には、 or 5=5 などのシングルクォートを使用しないコードを埋めても、
701          * 数字チェックで検出可能です。文字タイプの場合は、必ず (')をはずして、
702          * ' or 'A' like 'A のような形式になる為、(')チェックだけでも有効です。
703          * (') が含まれていたエラーにする(true)/かノーチェックか(false)を指定します。
704          * (初期値:システム定数のUSE_SQL_INJECTION_CHECK[={@og.value SystemData#USE_SQL_INJECTION_CHECK}])。
705          *
706          * @param       flag    クォートチェック [true:する/それ以外:しない]
707          * @see         org.opengion.hayabusa.common.SystemData#USE_SQL_INJECTION_CHECK
708          */
709         public void setQuotCheck( final String flag ) {
710                 quotCheck = nval( getRequestParameter( flag ),quotCheck );
711         }
712
713         /**
714          * 【TAG】リクエスト情報の HTMLTag開始/終了文字(&gt;&lt;) 存在チェックを実施するかどうか[true/false]を設定します
715          *              (初期値:USE_XSS_CHECK[={@og.value SystemData#USE_XSS_CHECK}])。
716          *
717          * @og.tag
718          * クロスサイトスクリプティング(XSS)対策の一環としてless/greater than signについてのチェックを行います。
719          * (&gt;&lt;) が含まれていたエラーにする(true)/かノーチェックか(false)を指定します。
720          * (初期値:システム定数のUSE_XSS_CHECK[={@og.value SystemData#USE_XSS_CHECK}])。
721          *
722          * @param       flag    XSSチェックする [true:チェックする/false:しない]
723          * @see         org.opengion.hayabusa.common.SystemData#USE_XSS_CHECK
724          */
725         public void setXssCheck( final String flag ) {
726                 xssCheck = nval( getRequestParameter( flag ),xssCheck );
727         }
728
729         /**
730          * 【TAG】vals属性でパラメーターを取得する際、複数件存在する場合に、値を連結するかどうかを指定します(初期値:false)。
731          *
732          * @og.tag
733          * この属性がtrueに指定された場合、パラメーターが複数存在する場合に、カンマで連結します。
734          * 初期値は、false(連結しない)です。
735          *
736          * @og.rev 5.1.8.0 (2010/07/01) 新規作成
737          *
738          * @param       flag    値連結 [true:する/false:しない]
739          */
740         public void setMulti( final String flag ) {
741                 isMulti = nval( getRequestParameter( flag ),isMulti );
742         }
743
744         /**
745          * 表示データの HybsSystem.ROW_SEL_KEY を元に、選ばれた 行を処理の対象とします。
746          *
747          * @return      選択行の配列
748          */
749         @Override
750         protected int[] getParameterRows() {
751                 final int[] rowNo ;
752                 if( selectedAll ) {
753                         final int rowCnt = table.getRowCount();
754                         rowNo = new int[ rowCnt ];
755                         for( int i=0; i<rowCnt; i++ ) {
756                                 rowNo[i] = i;
757                         }
758                 } else {
759                         rowNo = super.getParameterRows();
760                 }
761                 return rowNo ;
762         }
763
764         /**
765          * このオブジェクトの文字列表現を返します。
766          * 基本的にデバッグ目的に使用します。
767          *
768          * @return このクラスの文字列表現
769          * @og.rtnNotNull
770          */
771         @Override
772         public String toString() {
773                 return ToString.title( this.getClass().getName() )
774                                 .println( "VERSION"                     ,VERSION                )
775                                 .println( "command"                     ,command                )
776                                 .println( "logics"                      ,logics                 )
777                                 .println( "dbid"                        ,dbid                   )
778                                 .println( "tableId"                     ,tableId                )
779                                 .println( "selectedAll"         ,selectedAll    )
780                                 .println( "modifyType"          ,modifyType             )
781                                 .println( "keys"                        ,keys                   )
782                                 .println( "vals"                        ,vals                   )
783                                 .println( "stopError"           ,stopError              )
784                                 .println( "quotCheck"           ,quotCheck              )
785                                 .println( "xssCheck"            ,xssCheck               )
786                                 .println( "executeCount"        ,executeCount   )
787                                 .println( "errCode"                     ,errCode                )
788                                 .println( "Other..."            ,getAttributes().getAttribute() )
789                                 .fixForm().toString() ;
790         }
791 }